org.verdictdb.VerdictContext Maven / Gradle / Ivy
/*
* Copyright 2018 University of Michigan
*
* 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 org.verdictdb;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.Collection;
import java.util.LinkedList;
import java.util.List;
import java.util.Properties;
import org.apache.commons.lang3.RandomStringUtils;
import org.verdictdb.commons.VerdictOption;
import org.verdictdb.connection.CachedDbmsConnection;
import org.verdictdb.connection.ConcurrentJdbcConnection;
import org.verdictdb.connection.DbmsConnection;
import org.verdictdb.connection.JdbcConnection;
import org.verdictdb.coordinator.ExecutionContext;
import org.verdictdb.coordinator.VerdictResultStream;
import org.verdictdb.coordinator.VerdictSingleResult;
import org.verdictdb.core.scrambling.ScrambleMetaSet;
import org.verdictdb.core.sqlobject.CreateSchemaQuery;
import org.verdictdb.exception.VerdictDBDbmsException;
import org.verdictdb.exception.VerdictDBException;
import org.verdictdb.metastore.CachedScrambleMetaStore;
import org.verdictdb.metastore.ScrambleMetaStore;
import org.verdictdb.metastore.VerdictMetaStore;
import org.verdictdb.sqlsyntax.SqlSyntax;
import org.verdictdb.sqlsyntax.SqlSyntaxList;
public class VerdictContext {
private DbmsConnection conn;
private boolean isClosed = false;
private VerdictMetaStore metaStore;
private final String contextId;
private long executionSerialNumber = 0;
private VerdictOption options;
/**
* Maintains the list of open executions. Each query is processed on a separate execution context.
*/
private List executionContexts = new LinkedList<>();
public VerdictContext(DbmsConnection conn) {
this.conn = new CachedDbmsConnection(conn);
this.contextId = RandomStringUtils.randomAlphanumeric(5);
this.options = new VerdictOption();
this.metaStore = getCachedMetaStore(conn, options);
}
public VerdictContext(DbmsConnection conn, VerdictOption options) throws VerdictDBException {
this.conn = new CachedDbmsConnection(conn);
this.contextId = RandomStringUtils.randomAlphanumeric(5);
this.options = options;
this.metaStore = getCachedMetaStore(conn, options);
initialize(options);
}
private VerdictMetaStore getCachedMetaStore(DbmsConnection conn, VerdictOption option) {
CachedScrambleMetaStore metaStore = new CachedScrambleMetaStore(new ScrambleMetaStore(conn, options));
metaStore.refreshCache();
return metaStore;
}
/**
* Creates the schema for temp tables.
* @throws VerdictDBException
*/
private void initialize(VerdictOption option) throws VerdictDBException {
String schema = option.getVerdictTempSchemaName();
CreateSchemaQuery query = new CreateSchemaQuery(schema);
query.setIfNotExists(true);
conn.execute(query);
}
/**
* This method does not support concurrent execution of queries; thus, should not be used in
* production.
*
* @param jdbcConn
* @return
* @throws VerdictDBDbmsException
*/
public static VerdictContext fromJdbcConnection(Connection jdbcConn)
throws VerdictDBDbmsException {
DbmsConnection conn = JdbcConnection.create(jdbcConn);
return new VerdictContext(conn);
}
/**
* Uses a connection pool.
*
* @param jdbcConnectionString
* @return
* @throws SQLException
* @throws VerdictDBException
*/
public static VerdictContext fromConnectionString(String jdbcConnectionString)
throws VerdictDBException {
attemptLoadDriverClass(jdbcConnectionString);
VerdictOption options = new VerdictOption();
options.parseConnectionString(jdbcConnectionString);
return new VerdictContext(ConcurrentJdbcConnection.create(jdbcConnectionString), options);
}
/**
* Uses a connection pool.
*
* @param jdbcConnectionString
* @param info
* @return
* @throws SQLException
* @throws VerdictDBException
*/
public static VerdictContext fromConnectionString(String jdbcConnectionString, Properties info)
throws VerdictDBException {
attemptLoadDriverClass(jdbcConnectionString);
VerdictOption options = new VerdictOption();
options.parseConnectionString(jdbcConnectionString);
options.parseProperties(info);
options.parseConnectionString(jdbcConnectionString);
return new VerdictContext(ConcurrentJdbcConnection.create(jdbcConnectionString, info), options);
// Connection jdbcConn = DriverManager.getConnection(jdbcConnectionString, info);
// return fromJdbcConnection(jdbcConn);
}
/**
* Uses a connection pool.
*
* @param jdbcConnectionString
* @param user
* @param password
* @return
* @throws SQLException
* @throws VerdictDBException
*/
public static VerdictContext fromConnectionString(
String jdbcConnectionString, String user, String password)
throws VerdictDBException {
attemptLoadDriverClass(jdbcConnectionString);
Properties info = new Properties();
info.setProperty("user", user);
info.setProperty("password", password);
VerdictOption options = new VerdictOption();
options.parseConnectionString(jdbcConnectionString);
return new VerdictContext(ConcurrentJdbcConnection.create(jdbcConnectionString, info), options);
}
public static VerdictContext fromConnectionString(
String jdbcConnectionString, VerdictOption options)
throws VerdictDBException {
attemptLoadDriverClass(jdbcConnectionString);
options.parseConnectionString(jdbcConnectionString);
return new VerdictContext(ConcurrentJdbcConnection.create(jdbcConnectionString), options);
}
public static VerdictContext fromConnectionString(
String jdbcConnectionString, String user, String password, VerdictOption options)
throws VerdictDBException {
attemptLoadDriverClass(jdbcConnectionString);
Properties info = new Properties();
info.setProperty("user", user);
info.setProperty("password", password);
options.parseConnectionString(jdbcConnectionString);
return new VerdictContext(ConcurrentJdbcConnection.create(jdbcConnectionString, info), options);
}
private static void attemptLoadDriverClass(String jdbcConnectionString) {
SqlSyntax syntax = SqlSyntaxList.getSyntaxFromConnectionString(jdbcConnectionString);
Collection driverClassNames = syntax.getCandidateJDBCDriverClassNames();
for (String className : driverClassNames) {
try {
Class.forName(className);
// System.out.println(className + " has been loaded into the classpath.");
} catch (ClassNotFoundException e) {
}
}
}
public DbmsConnection getConnection() {
return conn;
}
public void setDefaultSchema(String schema) {
conn.setDefaultSchema(schema);
}
public void close() {
this.abort(); // terminates all ExecutionContexts first.
conn.close();
isClosed = true;
}
public boolean isClosed() {
return isClosed;
}
@Deprecated
public JdbcConnection getJdbcConnection() {
DbmsConnection testConn = conn;
if (testConn instanceof CachedDbmsConnection) {
testConn = ((CachedDbmsConnection) conn).getOriginalConnection();
}
return (testConn instanceof JdbcConnection) ? (JdbcConnection) testConn : null;
}
public DbmsConnection getCopiedConnection() {
return conn.copy();
}
public String getContextId() {
return contextId;
}
public VerdictOption getOptions() {
return options;
}
public ExecutionContext createNewExecutionContext() {
long execSerialNumber = getNextExecutionSerialNumber();
ExecutionContext exec =
new ExecutionContext(conn.copy(), metaStore, contextId, execSerialNumber, options.copy());
executionContexts.add(exec);
return exec;
}
private synchronized long getNextExecutionSerialNumber() {
executionSerialNumber++;
return executionSerialNumber;
}
public ScrambleMetaSet getScrambleMetaSet() {
return metaStore.retrieve();
}
public VerdictMetaStore getMetaStore() {
return metaStore;
}
private void removeExecutionContext(ExecutionContext exec) {
exec.terminate();
executionContexts.remove(exec);
}
/** terminates all open execution context. */
public void abort() {
for (ExecutionContext context : executionContexts) {
context.terminate();
}
}
public void scramble(String originalSchema, String originalTable) {}
public void scramble(
String originalSchema, String originalTable, String newSchema, String newTable) {}
/**
* Returns a reliable result set as an answer. Right now, simply returns the first batch of
* Continuous results.
*
* Automatically spawns an independent execution context, then runs a query using it.
*
* @param query Either a select query or a create-scramble query
* @return A single query result is returned. If the query is a create-scramble query, the number
* of inserted rows are returned.
* @throws VerdictDBException
*/
public VerdictSingleResult sql(String query) throws VerdictDBException {
ExecutionContext exec = createNewExecutionContext();
VerdictSingleResult result = exec.sql(query);
removeExecutionContext(exec);
return result;
}
/**
* @param query Either a select query or a create-scramble query.
* @return Reader enables progressive query result consumption. If this is a create-scramble
* query, the number of inserted rows are returned.
* @throws VerdictDBException
*/
public VerdictResultStream streamsql(String query) throws VerdictDBException {
ExecutionContext exec = createNewExecutionContext();
VerdictResultStream stream = exec.streamsql(query);
return stream;
}
}