All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.hsqldb.util.RCData Maven / Gradle / Ivy

There is a newer version: 2.7.3
Show newest version
/* Copyright (c) 2001-2021, The HSQL Development Group
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * Redistributions of source code must retain the above copyright notice, this
 * list of conditions and the following disclaimer.
 *
 * Redistributions in binary form must reproduce the above copyright notice,
 * this list of conditions and the following disclaimer in the documentation
 * and/or other materials provided with the distribution.
 *
 * Neither the name of the HSQL Development Group nor the names of its
 * contributors may be used to endorse or promote products derived from this
 * software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL HSQL DEVELOPMENT GROUP, HSQLDB.ORG,
 * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */


package org.hsqldb.util;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.io.IOException;
import java.net.MalformedURLException;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;
import java.util.Properties;
import java.util.Set;
import java.util.HashSet;
import java.util.StringTokenizer;
import java.util.regex.Pattern;

/* $Id: RCData.java 6299 2021-02-09 17:10:48Z fredt $ */

/**
 * Manages all the details we need to connect up to JDBC database(s),
 * in a declarative way.
 * 

* The file src/org/hsqldb/sample/SqlFileEmbedder.java * in the HSQLDB distribution provides an example of how to use RCData for your * own programs. * * @see * The RC File section of the HyperSQL Utilities Guide * @see org.hsqldb.sample.SqlFileEmbedder * @author Blaine Simpson (blaine dot simpson at admc dot com) */ public class RCData { public static final String DEFAULT_JDBC_DRIVER = "org.hsqldb.jdbc.JDBCDriver"; private String defaultJdbcDriverName = DEFAULT_JDBC_DRIVER; public void setDefaultJdbcDriver(String defaultJdbcDriverName) { this.defaultJdbcDriverName = defaultJdbcDriverName; } public String getDefaultJdbcDriverName() { return defaultJdbcDriverName; } /** * DISABLED DUE TO SECURITY CONCERNS. * Just for testing and debugging. * * N.b. this echoes passwords! public void report() { System.err.println("urlid: " + id + ", url: " + url + ", username: " + username + ", password: " + password); } */ public String toString() { return "id: " + angleBracketNull(id) + ", url: " + angleBracketNull(url) + ", username: " + angleBracketNull(username) + ", password: <" + (password == null ? "NULL" : "PRESENT") + ">" + ", ti: " + angleBracketNull(ti) + ", driver: " + angleBracketNull(driver) + ", truststore: " + angleBracketNull(truststore) + ", libpath: " + angleBracketNull(libpath); } private static String angleBracketNull(final String s) { return s == null ? "" : s; } /** * Creates a RCDataObject by looking up the given key in the * given authentication file. * * @param file File containing the authentication information. * @param dbKey Key to look up in the file. * If null, then will echo all urlids in the file to stdout. * (A rather ill-conceived design). * @throws Exception any exception */ public RCData(File file, String dbKey) throws Exception { // This set is so we can catch duplicates. Set idPatterns = new HashSet(); if (file == null) { throw new IllegalArgumentException("RC file name not specified"); } if (!file.canRead()) { throw new IOException("Please set up authentication file '" + file + "'"); } // System.err.println("Using RC file '" + file + "'"); StringTokenizer tokenizer; boolean loadingStanza = false; String s; String[] tokens; String keyword, value; int linenum = 0; BufferedReader br = new BufferedReader(new FileReader(file)); try { while ((s = br.readLine()) != null) { ++linenum; s = s.trim(); if (s.isEmpty()) { continue; } if (s.charAt(0) == '#') { continue; } tokenizer = new StringTokenizer(s); if (tokenizer.countTokens() == 1) { keyword = tokenizer.nextToken(); value = ""; } else if (tokenizer.countTokens() > 1) { keyword = tokenizer.nextToken(); value = tokenizer.nextToken("").trim(); } else { throw new Exception("Corrupt line " + linenum + " in '" + file + "': " + s); } if (keyword.equals("urlid")) { tokens = value.split("\\s*,\\s*", -1); for (int i = 0; i < tokens.length; i++) { if (idPatterns.contains(tokens[i])) throw new Exception("ID Pattern '" + tokens[i] + "' repeated at line " + linenum + " in '" + file + "'"); idPatterns.add(tokens[i]); if (dbKey == null) { System.out.println(tokens[i]); continue; } loadingStanza = Pattern.compile(tokens[i]).matcher(dbKey).matches(); if (id == null && loadingStanza) id = dbKey; } continue; } if (dbKey == null) continue; if (loadingStanza) { if (keyword.equals("url")) { url = value; } else if (keyword.equals("username")) { username = value; } else if (keyword.equals("driver")) { driver = value; } else if (keyword.equals("charset")) { charset = value; } else if (keyword.equals("truststore")) { truststore = value; } else if (keyword.equals("password")) { password = value; } else if (keyword.equals("transiso")) { ti = value; } else if (keyword.equals("libpath")) { libpath = value; } else { throw new Exception("Bad line " + linenum + " in '" + file + "': " + s); } } } } finally { try { br.close(); } catch (IOException ioe) { // Can only report on so many errors at one time } br = null; // Encourage GC } //System.err.println(idPatterns.size() + " patterns: " + idPatterns); if (dbKey == null) { return; } if (libpath != null) { throw new IllegalArgumentException( "Sorry, 'libpath' not supported yet"); } if (id == null) throw new IllegalArgumentException( "No match for '" + dbKey + "' in file '" + file + "'"); } /** * Convenience constructor for backward compatibility. * * @see #RCData(String,String,String,String,String,String,String,String) */ public RCData(String id, String url, String username, String password, String driver, String charset, String truststore) throws Exception { this(id, url, username, password, driver, charset, truststore, null); } /** * Wrapper for unset Transaction Isolation. */ public RCData(String id, String url, String username, String password, String driver, String charset, String truststore, String libpath) throws Exception { this(id, url, username, password, driver, charset, truststore, libpath, null); } /** * Creates a new RCData object. * *

* The parameters driver, charset, truststore, and libpath are optional. * Setting these parameters to NULL will set them to their * default values. * * @param id The identifier for these connection settings * @param url The URL of the database to connect to * @param username The username to log in as * @param password The password of the username * @param driver The JDBC driver to use * @param charset The character set to use * @param truststore The trust store to use * @param libpath The JDBC library to add to CLASSPATH * @param ti The transaction level * @throws Exception if the a non-optional parameter is set to NULL */ public RCData(String id, String url, String username, String password, String driver, String charset, String truststore, String libpath, String ti) throws Exception { this.id = id; this.url = url; this.username = username; this.password = password; this.ti = ti; this.driver = driver; this.charset = charset; this.truststore = truststore; this.libpath = libpath; if (libpath != null) { throw new IllegalArgumentException( "Sorry, 'libpath' not supported yet"); } // We now require only id to be set by this constructor. // This allows using programs to add settings to an RC object partially // populated by RC file. // Will not find out about missing 'url' until try to actually connect. if (id == null) { throw new Exception("id was not set"); } } /* Purposefully not using JavaBean paradigm so that these fields can * be used as a traditional, public DO */ public String id; public String url; public String username; public String password; public String ti; public String driver; public String charset; public String truststore; public String libpath; /** * Gets a JDBC Connection using the data of this RCData object. * * @return New JDBC Connection * @throws ClassNotFoundException on class not found * @throws SQLException on database access error * @throws MalformedURLException on malformed URL */ public Connection getConnection() throws ClassNotFoundException, SQLException, MalformedURLException { return getConnection(null, null); } /** * Gets a JDBC Connection using the data of this RCData object with * specified override elements * * @param curDriverIn driver * @param curTrustStoreIn trusted store * @return New JDBC Connection * @throws ClassNotFoundException on class not found * @throws MalformedURLException on malformed URL * @throws SQLException on database access error */ public Connection getConnection(String curDriverIn, String curTrustStoreIn) throws MalformedURLException, SQLException { // Local vars to satisfy compiler warnings String curDriver = null; String curTrustStore = null; Properties sysProps = System.getProperties(); if (curDriverIn == null) { // If explicit driver not specified curDriver = ((driver == null) ? DEFAULT_JDBC_DRIVER : driver); } else { curDriver = expandSysPropVars(curDriverIn); } if (curTrustStoreIn == null) { if (truststore != null) { curTrustStore = expandSysPropVars(truststore); } } else { curTrustStore = expandSysPropVars(curTrustStoreIn); } if (curTrustStore == null) { sysProps.remove("javax.net.ssl.trustStore"); } else { sysProps.put("javax.net.ssl.trustStore", curTrustStore); } String urlString; if (url == null) { throw new MalformedURLException( "url string is required to establish a connection, but is null" ); } try { urlString = expandSysPropVars(url); } catch (IllegalArgumentException iae) { throw new MalformedURLException(iae.toString() + " for URL '" + url + "'"); } String userString = null; if (username != null) try { userString = expandSysPropVars(username); } catch (IllegalArgumentException iae) { throw new MalformedURLException(iae.toString() + " for user name '" + username + "'"); } String passwordString = null; if (password != null) try { passwordString = expandSysPropVars(password); } catch (IllegalArgumentException iae) { throw new MalformedURLException(iae.toString() + " for password"); } // Every modern JDBC driver will register the driver as SP service // or a module service, so this should never be needed: //Class.forName(curDriver); Connection c = (userString == null) ? DriverManager.getConnection(urlString) : DriverManager.getConnection(urlString, userString, passwordString); if (ti != null) RCData.setTI(c, ti); // Would like to verify the setting made by checking // c.getTransactionIsolation(). Unfortunately, the spec allows for // databases to substitute levels according to some rules, and it's // impossible to know what to expect since custom levels are permitted. // Debug: // System.err.println("TI set to " + ti + "\nPOST: " // + SqlTool.tiToString(c.getTransactionIsolation())); return c; } /** * Returns a copy of the given String with System property names in the * format ${system.property} replaced by the corresponding Java * System Properties. */ public static String expandSysPropVars(String inString) { String outString = inString; int varOffset, varEnd; String varVal, varName; while (true) { // Recursive substitution for ${x} variables. varOffset = outString.indexOf("${"); if (varOffset < 0) { break; } varEnd = outString.indexOf('}', varOffset + 2); if (varEnd < 0) { break; } varName = outString.substring(varOffset + 2, varEnd); if (varName.length() < 1) { throw new IllegalArgumentException("Bad variable setting"); } varVal = System.getProperty(varName); if (varVal == null) { throw new IllegalArgumentException( "No Java system property with name '" + varName + "'"); } outString = outString.substring(0, varOffset) + varVal + outString.substring(varEnd + 1); } return outString; } /** * Set Transaction Isolation level on the specified JDBC Connection */ public static void setTI(Connection c, String tiString) throws SQLException { int i = -1; if (tiString.equals("TRANSACTION_READ_UNCOMMITTED")) i = Connection.TRANSACTION_READ_UNCOMMITTED; if (tiString.equals("TRANSACTION_READ_COMMITTED")) i = Connection.TRANSACTION_READ_COMMITTED; if (tiString.equals("TRANSACTION_REPEATABLE_READ")) i = Connection.TRANSACTION_REPEATABLE_READ; if (tiString.equals("TRANSACTION_SERIALIZABLE")) i = Connection.TRANSACTION_SERIALIZABLE; if (tiString.equals("TRANSACTION_NONE")) i = Connection.TRANSACTION_NONE; if (i < 0) { throw new SQLException( "Trans. isol. value not supported by " + RCData.class.getName() + ": " + tiString); } c.setTransactionIsolation(i); } /** * Return a String representation for the given numerical * java.sql.Connection Transaction level. *

* Database implementations are free to provide their own transaction * isolation levels, so you can't depend upon this method to much. * * @param ti Transaction levle * @return The string representation */ public static String tiToString(int ti) { switch (ti) { case Connection.TRANSACTION_READ_UNCOMMITTED: return "TRANSACTION_READ_UNCOMMITTED"; case Connection.TRANSACTION_READ_COMMITTED: return "TRANSACTION_READ_COMMITTED"; case Connection.TRANSACTION_REPEATABLE_READ: return "TRANSACTION_REPEATABLE_READ"; case Connection.TRANSACTION_SERIALIZABLE: return "TRANSACTION_SERIALIZABLE"; case Connection.TRANSACTION_NONE: return "TRANSACTION_NONE"; } return "Custom Transaction Isolation numerical value: " + ti; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy