liquibase.snapshot.ResultSetCache Maven / Gradle / Ivy
package liquibase.snapshot;
import liquibase.database.Database;
import liquibase.database.core.InformixDatabase;
import liquibase.database.jvm.JdbcConnection;
import liquibase.exception.DatabaseException;
import liquibase.executor.jvm.ColumnMapRowMapper;
import liquibase.executor.jvm.RowMapperResultSetExtractor;
import liquibase.util.JdbcUtils;
import liquibase.util.StringUtils;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.*;
class ResultSetCache {
private Map timesSingleQueried = new HashMap();
private Map didBulkQuery = new HashMap();
private Map>> cacheBySchema = new HashMap>>();
private Map info = new HashMap();
public List get(ResultSetExtractor resultSetExtractor) throws DatabaseException {
try {
String wantedKey = resultSetExtractor.wantedKeyParameters().createParamsKey(resultSetExtractor.database);
String schemaKey = resultSetExtractor.wantedKeyParameters().createSchemaKey(resultSetExtractor.database);
Map> cache = cacheBySchema.get(schemaKey);
if (cache == null ) {
cache = new HashMap>();
cacheBySchema.put(schemaKey, cache);
}
if (cache.containsKey(wantedKey)) {
return cache.get(wantedKey);
}
if (didBulkQuery.containsKey(schemaKey) && didBulkQuery.get(schemaKey)) {
return new ArrayList();
}
List results;
if (resultSetExtractor.shouldBulkSelect(schemaKey, this)) {
cache.clear(); //remove any existing single fetches that may be duplicated
results = resultSetExtractor.bulkFetch();
didBulkQuery.put(schemaKey, true);
} else {
Integer previousCount = timesSingleQueried.get(schemaKey);
if (previousCount == null) {
previousCount = 0;
}
timesSingleQueried.put(schemaKey, previousCount+1);
results = resultSetExtractor.fastFetch();
}
for (CachedRow row : results) {
for (String rowKey : resultSetExtractor.rowKeyParameters(row).getKeyPermutations()) {
if (!cache.containsKey(rowKey)) {
cache.put(rowKey, new ArrayList());
}
cache.get(rowKey).add(row);
}
}
List returnList = cache.get(wantedKey);
if (returnList == null) {
returnList = new ArrayList();
}
return returnList;
} catch (SQLException e) {
throw new DatabaseException(e);
}
}
public T getInfo(String key, Class type) {
return (T) info.get(key);
}
public void putInfo(String key, Object value) {
info.put(key, value);
}
public static class RowData {
private Database database;
private String[] parameters;
private String catalog;
private String schema;
private String[] keyPermutations;
protected RowData(String catalog, String schema, Database database, String... parameters) {
this.database = database;
this.catalog = catalog;
this.schema = schema;
this.parameters = parameters;
}
public String[] getKeyPermutations() {
if (keyPermutations == null) {
this.keyPermutations = permutations(parameters);
}
return keyPermutations;
}
protected String[] permutations(String[] params) {
return permute(params, 0);
}
private String[] permute(String[] params, int fromIndex) {
String[] nullVersion = Arrays.copyOf(params, params.length);
nullVersion[fromIndex] = null;
if (params.length == fromIndex + 1) {
return new String[] {
createKey(database, params),
createKey(database, nullVersion)
};
} else {
List permutations = new ArrayList();
Collections.addAll(permutations, permute(params, fromIndex + 1));
Collections.addAll(permutations, permute(nullVersion, fromIndex + 1));
return permutations.toArray(new String[permutations.size()]);
}
}
public String createSchemaKey(Database database) {
if (!database.supportsCatalogs() && ! database.supportsSchemas()) {
return "all";
} else if (database.supportsCatalogs() && database.supportsSchemas()) {
return (catalog+"."+schema).toLowerCase();
} else {
if (catalog == null && schema != null) {
return schema.toLowerCase();
} else {
if (catalog == null) {
return "all";
}
return catalog.toLowerCase();
}
}
}
public String createKey(Database database, String... params) {
String key = StringUtils.join(params, ":");
if (!database.isCaseSensitive()) {
return key.toLowerCase();
}
return key;
}
public String createParamsKey(Database database) {
return createKey(database, parameters);
}
}
public abstract static class ResultSetExtractor {
private final Database database;
public ResultSetExtractor(Database database) {
this.database = database;
}
boolean shouldBulkSelect(String schemaKey, ResultSetCache resultSetCache) {
return resultSetCache.getTimesSingleQueried(schemaKey) >= 3;
}
List executeAndExtract(String sql, Database database) throws DatabaseException, SQLException {
if (sql == null) {
return new ArrayList();
}
Statement statement = null;
ResultSet resultSet = null;
try {
statement = ((JdbcConnection) database.getConnection()).createStatement();
resultSet = statement.executeQuery(sql);
return extract(resultSet);
} finally {
JdbcUtils.close(resultSet, statement);
}
}
public boolean equals(Object expectedValue, Object foundValue) {
return equals(expectedValue, foundValue, true);
}
public boolean equals(Object expectedValue, Object foundValue, boolean equalIfEitherNull) {
if (expectedValue == null && foundValue == null) {
return true;
}
if (expectedValue == null || foundValue == null) {
return equalIfEitherNull;
}
return expectedValue.equals(foundValue);
}
public abstract RowData rowKeyParameters(CachedRow row);
public abstract RowData wantedKeyParameters();
public abstract List fastFetch() throws SQLException, DatabaseException;
public abstract List bulkFetch() throws SQLException, DatabaseException;
protected List extract(ResultSet resultSet) throws SQLException {
return extract(resultSet, false);
}
protected List extract(ResultSet resultSet, final boolean informixIndexTrimHint) throws SQLException {
List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy