RCData.java
/* Copyright (c) 2001-2024, 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 6721 2024-04-03 17:05:14Z fredt $ */
/**
 * Manages all the details we need to connect up to JDBC database(s),
 * in a declarative way.
 * <P>
 * The file {@code src/org/hsqldb/sample/SqlFileEmbedder.java}
 * in the HSQLDB distribution provides an example of how to use RCData for your
 * own programs.
 *
 * @see <A href="../../../../util-guide/sqltool-chapt.html#sqltool_auth-sect"
 *      target="guide">
 *     The RC File section of the HyperSQL Utilities Guide</A>
 * @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
               ? "<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<String> 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     = null;
        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 {@code RCData} object.
     *
     * <P>
     * The parameters driver, charset, truststore, and libpath are optional.
     * Setting these parameters to {@code 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 a non-optional parameter is set to {@code 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 SQLException on database access error
     * @throws MalformedURLException on malformed URL
     */
    public Connection getConnection()
            throws 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 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 {@code ${system.property}} replaced by the corresponding Java
     * System Properties.
     *
     * @param inString String
     * @return formatted string
     */
    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.isEmpty()) {
                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.
     * <P>
     * Database implementations are free to provide their own transaction
     * isolation levels, so you can't depend upon this method too 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;
    }
}