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

io.vertx.oracleclient.impl.OracleDatabaseHelper Maven / Gradle / Ivy

There is a newer version: 4.5.10
Show newest version
/*
 * Copyright (c) 2011-2022 Contributors to the Eclipse Foundation
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License 2.0 which is available at
 * http://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
 * which is available at https://www.apache.org/licenses/LICENSE-2.0.
 *
 * SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
 */
package io.vertx.oracleclient.impl;

import io.vertx.oracleclient.OracleConnectOptions;
import io.vertx.oracleclient.ServerMode;
import oracle.jdbc.OracleConnection;
import oracle.jdbc.datasource.OracleDataSource;

import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.util.Map;
import java.util.Optional;

import static io.vertx.oracleclient.impl.Helper.getOrHandleSQLException;
import static io.vertx.oracleclient.impl.Helper.runOrHandleSQLException;
import static oracle.jdbc.OracleConnection.CONNECTION_PROPERTY_TNS_ADMIN;

public class OracleDatabaseHelper {

  public static OracleDataSource createDataSource(OracleConnectOptions options) {
    OracleDataSource oracleDataSource =
      getOrHandleSQLException(oracle.jdbc.pool.OracleDataSource::new);

    runOrHandleSQLException(() ->
      oracleDataSource.setURL(composeJdbcUrl(options)));

    configureStandardOptions(oracleDataSource, options);
    configureExtendedOptions(oracleDataSource, options);
    configureJdbcDefaults(oracleDataSource);

    return oracleDataSource;
  }

  /**
   * Composes an Oracle JDBC URL from {@code OracleConnectOptions}, as
   * specified in the javadoc of
   * {@link #createDataSource(OracleConnectOptions)}
   *
   * @param options Oracle Connection options. Must not be {@code null}.
   * @return An Oracle Connection JDBC URL
   */
  private static String composeJdbcUrl(OracleConnectOptions options) {
    StringBuilder url = new StringBuilder("jdbc:oracle:thin:@");
    String tnsAlias = options.getTnsAlias();
    if (tnsAlias != null) {
      return url.append(tnsAlias).toString();
    }
    if (options.isSsl()) {
      url.append("tcps://");
    }
    String host = options.getHost();
    if (host.indexOf(':') >= 0) { // IPv6 address
      url.append("[").append(host).append("]");
    } else {
      url.append(encodeUrl(host));
    }
    int port = options.getPort();
    if (port > 0) {
      url.append(":").append(port);
    }
    String serviceId = options.getServiceId();
    if (serviceId != null) {
      url.append(":").append(encodeUrl(serviceId));
    } else {
      String database = Optional.ofNullable(options.getServiceName()).orElse(options.getDatabase());
      if (database != null) {
        url.append("/").append(encodeUrl(database));
        if (options.getServerMode() == ServerMode.SHARED) {
          url.append(":").append(ServerMode.SHARED);
        }
      }
    }
    return url.toString();
  }

  /**
   * Configures an {@code OracleDataSource}.
   *
   * @param oracleDataSource An data source to configure
   * @param options OracleConnectOptions options. Not null.
   */
  private static void configureStandardOptions(OracleDataSource oracleDataSource, OracleConnectOptions options) {
    Map properties = options.getProperties();
    if (properties != null && !properties.isEmpty()) {
      for (Map.Entry entry : properties.entrySet()) {
        runOrHandleSQLException(() -> oracleDataSource.setConnectionProperty(entry.getKey(), entry.getValue()));
      }
    }

    String user = options.getUser();
    if (user != null) {
      runOrHandleSQLException(() -> oracleDataSource.setUser(user));
    }

    CharSequence password = options.getPassword();
    if (password != null) {
      runOrHandleSQLException(() -> oracleDataSource.setPassword(password.toString()));
    }

    int connectTimeout = options.getConnectTimeout();
    if (connectTimeout > 0) {
      runOrHandleSQLException(() -> oracleDataSource.setLoginTimeout(connectTimeout));
    }
  }

  private static void configureExtendedOptions(OracleDataSource oracleDataSource, OracleConnectOptions options) {
    String tnsAdmin = options.getTnsAdmin();
    if (tnsAdmin != null) {
      runOrHandleSQLException(() -> oracleDataSource.setConnectionProperty(CONNECTION_PROPERTY_TNS_ADMIN, tnsAdmin));
    }

    // TODO Iterate over the other properties.
  }

  /**
   * Configures an {@code oracleDataSource} with any connection properties that
   * this adapter requires by default.
   *
   * @param oracleDataSource An data source to configure
   */
  private static void configureJdbcDefaults(OracleDataSource oracleDataSource) {

    // Have the Oracle JDBC Driver implement behavior that the JDBC
    // Specification defines as correct. The javadoc for this property lists
    // all of it's effects. One effect is to have ResultSetMetaData describe
    // FLOAT columns as the FLOAT type, rather than the NUMBER type. This
    // effect allows the Oracle R2DBC Driver obtain correct metadata for
    // FLOAT type columns. The property is deprecated, but the deprecation note
    // explains that setting this to "false" is deprecated, and that it
    // should be set to true; If not set, the 21c driver uses a default value
    // of false.
    @SuppressWarnings("deprecation")
    String enableJdbcSpecCompliance =
      OracleConnection.CONNECTION_PROPERTY_J2EE13_COMPLIANT;
    runOrHandleSQLException(() ->
      oracleDataSource.setConnectionProperty(enableJdbcSpecCompliance, "true"));

    // Have the Oracle JDBC Driver cache PreparedStatements by default.
    runOrHandleSQLException(() -> {
      // Don't override a value set by user code
      String userValue = oracleDataSource.getConnectionProperty(
        OracleConnection.CONNECTION_PROPERTY_IMPLICIT_STATEMENT_CACHE_SIZE);

      if (userValue == null) {
        // The default value of the OPEN_CURSORS parameter in the 21c
        // and 19c databases is 50:
        // https://docs.oracle.com/en/database/oracle/oracle-database/21/refrn/OPEN_CURSORS.html#GUID-FAFD1247-06E5-4E64-917F-AEBD4703CF40
        // Assuming this default, then a default cache size of 25 will keep
        // each session at or below 50% of it's cursor capacity, which seems
        // reasonable.
        oracleDataSource.setConnectionProperty(
          OracleConnection.CONNECTION_PROPERTY_IMPLICIT_STATEMENT_CACHE_SIZE,
          "25");
      }
    });

    // TODO: Disable the result set cache? This is needed to support the
    //  SERIALIZABLE isolation level, which requires result set caching to be
    //  disabled.
  }

  private static String encodeUrl(String url) {
    return URLEncoder.encode(url, StandardCharsets.UTF_8);
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy