blob: d4cfaa6cac8d03c8d1119753ce57067fb46a0931 [file] [log] [blame]
// Copyright (C) 2009 The Android Open Source Project
//
// Licensed under the Apache 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://www.apache.org/licenses/LICENSE-2.0
//
// 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 com.google.gerrit.server.schema;
import static java.util.concurrent.TimeUnit.MILLISECONDS;
import static java.util.concurrent.TimeUnit.SECONDS;
import com.google.common.base.Strings;
import com.google.gerrit.extensions.events.LifecycleListener;
import com.google.gerrit.extensions.persistence.DataSourceInterceptor;
import com.google.gerrit.metrics.CallbackMetric1;
import com.google.gerrit.metrics.Description;
import com.google.gerrit.metrics.Field;
import com.google.gerrit.metrics.MetricMaker;
import com.google.gerrit.server.config.ConfigSection;
import com.google.gerrit.server.config.ConfigUtil;
import com.google.gerrit.server.config.GerritServerConfig;
import com.google.gerrit.server.config.ThreadSettingsConfig;
import com.google.gwtorm.jdbc.SimpleDataSource;
import com.google.inject.Inject;
import com.google.inject.Provider;
import com.google.inject.ProvisionException;
import com.google.inject.Singleton;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.sql.SQLException;
import java.util.Properties;
import javax.sql.DataSource;
import org.apache.commons.dbcp.BasicDataSource;
import org.eclipse.jgit.lib.Config;
/** Provides access to the DataSource. */
@Singleton
public class DataSourceProvider implements Provider<DataSource>, LifecycleListener {
private static final String DATABASE_KEY = "database";
private final Config cfg;
private final MetricMaker metrics;
private final Context ctx;
private final DataSourceType dst;
private final ThreadSettingsConfig threadSettingsConfig;
private DataSource ds;
@Inject
protected DataSourceProvider(
@GerritServerConfig Config cfg,
MetricMaker metrics,
ThreadSettingsConfig threadSettingsConfig,
Context ctx,
DataSourceType dst) {
this.cfg = cfg;
this.metrics = metrics;
this.threadSettingsConfig = threadSettingsConfig;
this.ctx = ctx;
this.dst = dst;
}
@Override
public synchronized DataSource get() {
if (ds == null) {
ds = open(cfg, ctx, dst);
}
return ds;
}
@Override
public void start() {}
@Override
public synchronized void stop() {
if (ds instanceof BasicDataSource) {
try {
((BasicDataSource) ds).close();
} catch (SQLException e) {
// Ignore the close failure.
}
}
}
public enum Context {
SINGLE_USER,
MULTI_USER
}
private DataSource open(Config cfg, Context context, DataSourceType dst) {
ConfigSection dbs = new ConfigSection(cfg, DATABASE_KEY);
String driver = dbs.optional("driver");
if (Strings.isNullOrEmpty(driver)) {
driver = dst.getDriver();
}
String url = dbs.optional("url");
if (Strings.isNullOrEmpty(url)) {
url = dst.getUrl();
}
String username = dbs.optional("username");
String password = dbs.optional("password");
String interceptor = dbs.optional("dataSourceInterceptorClass");
boolean usePool;
if (context == Context.SINGLE_USER) {
usePool = false;
} else {
usePool = cfg.getBoolean(DATABASE_KEY, "connectionpool", dst.usePool());
}
if (usePool) {
final BasicDataSource lds = new BasicDataSource();
lds.setDriverClassName(driver);
lds.setUrl(url);
if (username != null && !username.isEmpty()) {
lds.setUsername(username);
}
if (password != null && !password.isEmpty()) {
lds.setPassword(password);
}
int poolLimit = threadSettingsConfig.getDatabasePoolLimit();
lds.setMaxActive(poolLimit);
lds.setMinIdle(cfg.getInt(DATABASE_KEY, "poolminidle", 4));
lds.setMaxIdle(cfg.getInt(DATABASE_KEY, "poolmaxidle", Math.min(poolLimit, 16)));
lds.setMaxWait(
ConfigUtil.getTimeUnit(
cfg,
DATABASE_KEY,
null,
"poolmaxwait",
MILLISECONDS.convert(30, SECONDS),
MILLISECONDS));
lds.setInitialSize(lds.getMinIdle());
long evictIdleTimeMs = 1000L * 60;
lds.setMinEvictableIdleTimeMillis(evictIdleTimeMs);
lds.setTimeBetweenEvictionRunsMillis(evictIdleTimeMs / 2);
lds.setTestOnBorrow(true);
lds.setTestOnReturn(true);
lds.setValidationQuery(dst.getValidationQuery());
lds.setValidationQueryTimeout(5);
exportPoolMetrics(lds);
return intercept(interceptor, lds);
}
// Don't use the connection pool.
//
try {
final Properties p = new Properties();
p.setProperty("driver", driver);
p.setProperty("url", url);
if (username != null) {
p.setProperty("user", username);
}
if (password != null) {
p.setProperty("password", password);
}
return intercept(interceptor, new SimpleDataSource(p));
} catch (SQLException se) {
throw new ProvisionException("Database unavailable", se);
}
}
private void exportPoolMetrics(BasicDataSource pool) {
CallbackMetric1<Boolean, Integer> cnt =
metrics.newCallbackMetric(
"sql/connection_pool/connections",
Integer.class,
new Description("SQL database connections").setGauge().setUnit("connections"),
Field.ofBoolean("active"));
metrics.newTrigger(
cnt,
() -> {
synchronized (pool) {
cnt.set(true, pool.getNumActive());
cnt.set(false, pool.getNumIdle());
}
});
}
private DataSource intercept(String interceptor, DataSource ds) {
if (interceptor == null) {
return ds;
}
try {
Constructor<?> c = Class.forName(interceptor).getConstructor();
DataSourceInterceptor datasourceInterceptor = (DataSourceInterceptor) c.newInstance();
return datasourceInterceptor.intercept("reviewDb", ds);
} catch (ClassNotFoundException
| SecurityException
| NoSuchMethodException
| IllegalArgumentException
| InstantiationException
| IllegalAccessException
| InvocationTargetException e) {
throw new ProvisionException("Cannot intercept datasource", e);
}
}
}