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

org.opencastproject.db.Activator Maven / Gradle / Ivy

/*
 * Licensed to The Apereo Foundation under one or more contributor license
 * agreements. See the NOTICE file distributed with this work for additional
 * information regarding copyright ownership.
 *
 *
 * The Apereo Foundation licenses this file to you under the Educational
 * Community License, Version 2.0 (the "License"); you may not use this file
 * except in compliance with the License. You may obtain a copy of the License
 * at:
 *
 *   http://opensource.org/licenses/ecl2.txt
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the
 * License for the specific language governing permissions and limitations under
 * the License.
 *
 */

package org.opencastproject.db;

import com.mchange.v2.c3p0.ComboPooledDataSource;
import com.mchange.v2.c3p0.DataSources;

import org.osgi.framework.BundleActivator;
import org.osgi.framework.BundleContext;
import org.osgi.framework.ServiceRegistration;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.File;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.Hashtable;

import javax.sql.DataSource;

/** Registers {@link DataSource}. */
public class Activator implements BundleActivator {

  /** The logging facility */
  private static final Logger logger = LoggerFactory.getLogger(Activator.class);

  /** The default max idle time for the connection pool */
  private static final int DEFAULT_MAX_IDLE_TIME = 3600;

  private String rootDir;
  private ServiceRegistration datasourceRegistration;
  private ComboPooledDataSource pooledDataSource;

  @Override
  public void start(BundleContext bundleContext) throws Exception {
    // Use the configured storage directory
    rootDir = bundleContext.getProperty("org.opencastproject.storage.dir") + File.separator + "db";

    // Register the data source, defaulting to an embedded H2 database if DB configurations are not specified
    String jdbcDriver = getConfigProperty(bundleContext.getProperty("org.opencastproject.db.jdbc.driver"),
        "org.h2.Driver");
    String jdbcUrl = getConfigProperty(bundleContext.getProperty("org.opencastproject.db.jdbc.url"),
        "jdbc:h2:" + rootDir);
    if ("org.h2.Driver".equals(jdbcDriver)) {
      logger.warn("\n"
          + "######################################################\n"
          + "#                                                    #\n"
          + "# WARNING: Opencast is using an H2 database.         #\n"
          + "#          Never do this in production.              #\n"
          + "#                                                    #\n"
          + "#          For more information about database       #\n"
          + "#          configuration, see:                       #\n"
          + "#                                                    #\n"
          + "#          https://docs.opencast.org                 #\n"
          + "#                                                    #\n"
          + "######################################################");
    }
    String jdbcUser = getConfigProperty(bundleContext.getProperty("org.opencastproject.db.jdbc.user"), "sa");
    String jdbcPass = getConfigProperty(bundleContext.getProperty("org.opencastproject.db.jdbc.pass"), "sa");

    Integer maxPoolSize = getConfigProperty(bundleContext.getProperty("org.opencastproject.db.jdbc.pool.max.size"));
    Integer minPoolSize = getConfigProperty(bundleContext.getProperty("org.opencastproject.db.jdbc.pool.min.size"));
    Integer acquireIncrement = getConfigProperty(
        bundleContext.getProperty("org.opencastproject.db.jdbc.pool.acquire.increment"));
    Integer maxStatements = getConfigProperty(
        bundleContext.getProperty("org.opencastproject.db.jdbc.pool.max.statements"));
    Integer loginTimeout = getConfigProperty(
        bundleContext.getProperty("org.opencastproject.db.jdbc.pool.login.timeout"));
    Integer maxIdleTime = getConfigProperty(
        bundleContext.getProperty("org.opencastproject.db.jdbc.pool.max.idle.time"));
    Integer maxConnectionAge = getConfigProperty(
        bundleContext.getProperty("org.opencastproject.db.jdbc.pool.max.connection.age"));
    Boolean testConnectionOnCheckin = getConfigBooleanProperty(
        bundleContext.getProperty("org.opencastproject.db.jdbc.pool.test.connection.on.checkin"));
    Boolean testConnectionOnCheckout = getConfigBooleanProperty(
        bundleContext.getProperty("org.opencastproject.db.jdbc.pool.test.connection.on.checkout"));
    Integer idleConnectionTestPeriod = getConfigProperty(
        bundleContext.getProperty("org.opencastproject.db.jdbc.pool.idle.connection.test.period"));

    pooledDataSource = new ComboPooledDataSource();
    pooledDataSource.setDriverClass(jdbcDriver);
    pooledDataSource.setJdbcUrl(jdbcUrl);
    pooledDataSource.setUser(jdbcUser);
    pooledDataSource.setPassword(jdbcPass);
    if (minPoolSize != null) {
      pooledDataSource.setMinPoolSize(minPoolSize);
      // initial pool size should be at least the min value
      if (pooledDataSource.getInitialPoolSize() < minPoolSize) {
        pooledDataSource.setInitialPoolSize(minPoolSize);
      }
    }
    if (maxPoolSize != null) {
      pooledDataSource.setMaxPoolSize(maxPoolSize);
      // initial pool size should be at most the max value
      if (pooledDataSource.getInitialPoolSize() > maxPoolSize) {
        pooledDataSource.setInitialPoolSize(maxPoolSize);
      }
    }
    if (acquireIncrement != null) {
      pooledDataSource.setAcquireIncrement(acquireIncrement);
    }
    if (maxStatements != null) {
      pooledDataSource.setMaxStatements(maxStatements);
    }
    if (loginTimeout != null) {
      pooledDataSource.setLoginTimeout(loginTimeout);
    }

    // maxIdleTime should not be zero, otherwise the connection pool will hold on to stale connections
    // that have been closed by the database.
    if (maxIdleTime != null) {
      pooledDataSource.setMaxIdleTime(maxIdleTime);
    } else if (pooledDataSource.getMaxIdleTime() == 0) {
      logger.debug("Setting database connection pool max.idle.time to default of {}", DEFAULT_MAX_IDLE_TIME);
      pooledDataSource.setMaxIdleTime(DEFAULT_MAX_IDLE_TIME);
    }

    if (maxConnectionAge != null) {
      pooledDataSource.setMaxConnectionAge(maxConnectionAge);
    }
    if (testConnectionOnCheckin != null) {
      pooledDataSource.setTestConnectionOnCheckin(testConnectionOnCheckin);
    }
    if (testConnectionOnCheckout != null) {
      pooledDataSource.setTestConnectionOnCheckout(testConnectionOnCheckout);
    }
    if (idleConnectionTestPeriod != null) {
      pooledDataSource.setIdleConnectionTestPeriod(idleConnectionTestPeriod);
    }

    Connection connection = null;
    try {
      logger.info("Testing connectivity to database at {}", jdbcUrl);
      connection = pooledDataSource.getConnection();
      Hashtable dsProps = new Hashtable<>();
      dsProps.put("osgi.jndi.service.name", "jdbc/opencast");
      datasourceRegistration = bundleContext.registerService(DataSource.class.getName(), pooledDataSource, dsProps);
    } catch (SQLException e) {
      logger.error("Connection attempt to {} failed", jdbcUrl, e);
      throw e;
    } finally {
      if (connection != null) {
        connection.close();
      }
    }

    logger.info("Database connection pool established at {}", jdbcUrl);
    logger.info("Database connection pool parameters: max.size={}, min.size={}, max.idle.time={}",
        pooledDataSource.getMaxPoolSize(), pooledDataSource.getMinPoolSize(), pooledDataSource.getMaxIdleTime());
    Statement statement = pooledDataSource.getConnection().createStatement();

    long random = Math.round(Math.random() * 1000000);
    String tableName = "oc_temp_" + random;
    try {
      statement.executeUpdate("CREATE TABLE " + tableName + " ( id BIGINT NOT NULL, test BIGINT, PRIMARY KEY (id) );");
      runUpdate(statement, "INSERT INTO " + tableName + " VALUES (" + random + ", 0);");
      runUpdate(statement, "UPDATE " + tableName + " SET test = " + random + ";");
      ResultSet rs = statement.executeQuery("SELECT * FROM " + tableName + ";");
      while (rs.next()) {
        long id = rs.getLong("id");
        long test = rs.getLong("test");
        if (id != random || test != random) {
          throw new RuntimeException("Unable to verify updating a table functions correctly");
        }
      }
      runUpdate(statement, "DELETE FROM " + tableName + " WHERE id = " + random + ";");
      logger.info("Database credentials passed basic tests!");
    } catch (Exception e) {
      throw new RuntimeException("Unable to verify SQL credentials have required permissions!", e);
    } finally {
      try {
        statement.executeUpdate("DROP TABLE " + tableName + ";");
      } catch (Exception e) {
        logger.warn("Unable to delete temp table {}, please remove this yourself!", tableName, e);
      }
    }

  }

  private void runUpdate(Statement statement, String sql) throws RuntimeException, SQLException {
    int affected = statement.executeUpdate(sql);
    if (affected != 1) {
      throw new RuntimeException(
          "Unable to update on a testing table, check that your database user has the right permissions!");
    }
  }

  @Override
  public void stop(BundleContext context) throws Exception {
    logger.info("Shutting down database");
    if (datasourceRegistration != null) {
      datasourceRegistration.unregister();
    }
    logger.info("Shutting down connection pool");
    DataSources.destroy(pooledDataSource);
  }

  private String getConfigProperty(String config, String defaultValue) {
    return config == null ? defaultValue : config;
  }

  private Integer getConfigProperty(String config) {
    return config == null ? null : Integer.parseInt(config);
  }

  private Boolean getConfigBooleanProperty(String config) {
    return config == null ? null : Boolean.parseBoolean(config);
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy