org.etlunit.feature.database.DatabaseFeatureModule Maven / Gradle / Ivy
package org.etlunit.feature.database;
import com.google.inject.Binder;
import com.google.inject.Injector;
import com.google.inject.Module;
import org.codehaus.jackson.JsonNode;
import org.etlunit.*;
import org.etlunit.feature.AbstractFeature;
import org.etlunit.feature.FeatureModule;
import org.etlunit.feature.RuntimeOption;
import org.etlunit.feature.assertion.AssertionFeatureModule;
import org.etlunit.feature.database.util.DimensionalHashMap;
import org.etlunit.feature.database.util.DimensionalMap;
import org.etlunit.feature.execute.ExecuteFeatureModule;
import org.etlunit.io.FileBuilder;
import org.etlunit.json.validator.JsonUtils;
import org.etlunit.parser.*;
import org.etlunit.util.IOUtils;
import javax.inject.Inject;
import javax.inject.Named;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.util.*;
@FeatureModule
public class DatabaseFeatureModule extends AbstractFeature
{
private RuntimeSupport runtimeSupport;
private DatabaseRuntimeSupport databaseRuntimeSupport;
private static final List prerequisites = Arrays.asList("logging", "execute", "assertion", "file");
private ExecuteFeatureModule executeFeatureModule;
private AssertionFeatureModule assertionFeatureModule;
private final Map implementationMap = new HashMap();
protected final DimensionalMap initializedDbs = new DimensionalHashMap();
private DatabaseConfiguration databaseConfiguration;
private DatabaseClassListener classListener;
private JDBCClientImpl jdbcClient;
public void addDatabaseImplementation(DatabaseImplementation impl)
{
if (implementationMap.containsKey(impl.getImplementationId()))
{
throw new IllegalArgumentException("Database Implementation " + impl.getImplementationId() + " already installed");
}
implementationMap.put(impl.getImplementationId(), impl);
}
@Inject
public void setExecuteFeature(ExecuteFeatureModule f)
{
executeFeatureModule = f;
}
@Inject
public void setAssertionFeature(AssertionFeatureModule f)
{
assertionFeatureModule = f;
}
@Inject
public void setRuntimeSupport(RuntimeSupport runtimeSupport)
{
this.runtimeSupport = runtimeSupport;
}
@Override
protected Injector preCreateSub(Injector inj)
{
this.databaseRuntimeSupport = postCreate(new DatabaseRuntimeSupportImpl());
jdbcClient = postCreate(new JDBCClientImpl());
Injector childInjector = inj.createChildInjector(new Module()
{
public void configure(Binder binder)
{
binder.bind(DatabaseRuntimeSupport.class).toInstance(databaseRuntimeSupport);
binder.bind(JDBCClient.class).toInstance(jdbcClient);
}
});
return childInjector;
}
@Override
public void initialize(Injector inj)
{
if (featureConfiguration != null)
{
databaseConfiguration = new DatabaseConfiguration(
featureConfiguration,
runtimeSupport.getProjectName(),
runtimeSupport.getProjectVersion(),
runtimeSupport.getProjectUser(),
runtimeSupport.getProjectUID()
);
classListener = new DatabaseClassListener(this, databaseConfiguration);
inj.injectMembers(classListener);
}
DbOperationsExecutor object = new DbOperationsExecutor(this);
inj.injectMembers(object);
executeFeatureModule.extend(object);
DbAssertExtender object1 = new DbAssertExtender(this);
inj.injectMembers(object1);
assertionFeatureModule.extend(object1);
}
@Override
public List getPrerequisites()
{
return prerequisites;
}
public String getFeatureName()
{
return "database";
}
@Override
public ClassListener getListener()
{
return classListener;
}
public SQLAggregator resolveSQLRef(String ddlRef, DatabaseConnection databaseConnection, SQLAggregator.SQLLocator locator) throws IOException, TestExecutionError
{
String text = locator.locate(ddlRef, databaseConnection);
SQLAggregatorImpl ddlAgg = new SQLAggregatorImpl(ddlRef, locator, databaseConnection);
return ddlAgg;
}
public SQLAggregator resolveSQLRef(String text) throws IOException, TestExecutionError
{
SQLAggregatorImpl ddlAgg = new SQLAggregatorImpl(text);
return ddlAgg;
}
public SQLAggregator resolveDDLRef(final String ddlRef, DatabaseConnection databaseConnection) throws IOException, TestExecutionError
{
SQLAggregatorImpl ddlAgg = new SQLAggregatorImpl(ddlRef, new SQLAggregator.SQLLocator()
{
public String locate(String ref, DatabaseConnection connection) throws IOException, TestExecutionError
{
return getDDLText(ref, connection);
}
}, databaseConnection);
File output = new FileBuilder(databaseRuntimeSupport.getGeneratedSourceDirectory(databaseConnection, null)).name("GENERATED_" + ddlRef.replace('/', '_')).file();
IOUtils.writeBufferToFile(output, new StringBuffer(ddlAgg.getText()));
return ddlAgg;
}
protected String getDDLText(final String ddlRef, DatabaseConnection databaseConnection) throws IOException, TestExecutionError
{
// Enhance this to look at the classpath if the file isn't found
File ddl = new FileBuilder(runtimeSupport.getFeatureSourceDirectory(databaseConnection.getImplementationId())).subdir(databaseConnection.getId()).mkdirs().name(ddlRef).file();
String text = null;
if (ddl.exists())
{
text = IOUtils.readFileToString(ddl);
}
else
{
// check for a classpath reference
String name = databaseConnection.getImplementationId() + "/" + databaseConnection.getId() + "/" + ddlRef;
URL ref = getClass().getClassLoader().getResource(name);
if (ref == null)
{
applicationLog.info("Unable to find ddl for reference: " + name + ", trying without the connection ID");
// try one more time, this time without the connection id
name = databaseConnection.getImplementationId() + "/" + ddlRef;
ref = getClass().getClassLoader().getResource(name);
// foul
if (ref == null)
{
throw new TestExecutionError("Error resolving ddl ref: " + name, DatabaseConstants.ERR_MISSSING_DDL);
}
}
text = IOUtils.readURLToString(ref);
}
return text;
}
protected void updateDatabaseFlag(String mode, DatabaseConnection databaseConnection)
{
JsonNode configNode = databaseConnection.getDatabaseConfiguration().getJsonNode();
File flagFile = getFlagFile(mode, databaseConnection);
try
{
IOUtils.writeBufferToFile(flagFile, new StringBuffer(configNode.toString()));
for (String script : databaseConnection.getSchemaScripts())
{
File ref = getDDLFlagFile(script, mode, databaseConnection);
StringBuilder bui = new StringBuilder();
SQLAggregator SQLAggregator = resolveDDLRef(script, databaseConnection);
bui.append(SQLAggregator.getText());
long crc = IOUtils.checkSumStream(new ByteArrayInputStream(bui.toString().getBytes())).getValue();
IOUtils.writeBufferToFile(ref, new StringBuffer("{\"checksum\": " + crc + "}"));
}
}
catch (Exception e)
{
throw new IllegalArgumentException("", e);
}
}
private File getDDLFlagFile(String ref, String mode, DatabaseConnection databaseConnection)
{
return runtimeSupport.createGeneratedSourceFile(databaseConnection.getImplementationId(), "ddl." + ref + "." + databaseConnection.getId() + "." + mode + "." + runtimeSupport.getProjectUID() + ".chk");
}
private File getFlagFile(String mode, DatabaseConnection databaseConnection)
{
return runtimeSupport.createGeneratedSourceFile(databaseConnection.getImplementationId(), "database." + databaseConnection.getId() + "." + mode + "." + runtimeSupport.getProjectUID() + ".tag");
}
protected boolean databaseOutOfDate(String mode, DatabaseConnection databaseConnection)
{
File flagFile = getFlagFile(mode, databaseConnection);
JsonNode configNode = databaseConnection.getDatabaseConfiguration().getJsonNode();
try
{
if (flagFile.exists())
{
JsonNode flgNode = JsonUtils.loadJson(flagFile);
if (flgNode.equals(configNode))
{
for (String script : databaseConnection.getSchemaScripts())
{
File ref = getDDLFlagFile(script, mode, databaseConnection);
if (!ref.exists())
{
return true;
}
StringBuilder bui = new StringBuilder();
SQLAggregator SQLAggregator = resolveDDLRef(script, databaseConnection);
bui.append(SQLAggregator.getText());
long crc = IOUtils.checkSumStream(new ByteArrayInputStream(bui.toString().getBytes())).getValue();
JsonNode node = JsonUtils.loadJson(ref);
if (node.get("checksum").asLong() != crc)
{
return true;
}
}
return false;
}
}
}
catch (Exception e)
{
throw new IllegalArgumentException("", e);
}
return true;
}
protected void propogateProperties(ETLTestValueObjectBuilder builder, DatabaseConnection databaseConnection, String mode)
{
builder.key("database-name").value(databaseConnection.getDatabaseName(mode));
builder.key("login-name").value(databaseConnection.getLoginName(mode));
builder.key("password").value(databaseConnection.getPassword(mode));
builder.key("server-name").value(databaseConnection.getServerName());
}
DatabaseImplementation getImplementation(String connectionId)
{
DatabaseConnection conn = databaseConfiguration.getDatabaseConnection(connectionId);
return implementationMap.get(conn.getImplementationId());
}
public DatabaseConfiguration getDatabaseConfiguration()
{
return databaseConfiguration;
}
@Override
public void dispose()
{
for (Map.Entry entry : implementationMap.entrySet())
{
entry.getValue().dispose();
}
}
@Override
protected List getSupportedFolderNamesSub()
{
return Arrays.asList("data", "sql");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy