stream.io.SQLWriter Maven / Gradle / Ivy
/**
*
*/
package stream.io;
import java.io.Serializable;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.Statement;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import stream.Data;
import stream.ProcessContext;
import stream.ProcessorException;
import stream.annotations.Description;
import stream.annotations.Parameter;
import stream.data.DataFactory;
import stream.io.sql.HsqlDialect;
import stream.io.sql.MysqlDialect;
/**
* @author chris
*
*/
@Description(group = "Data Stream.Output")
public class SQLWriter extends AbstractSQLProcessor {
static Logger log = LoggerFactory.getLogger(SQLWriter.class);
boolean dropTable = false;
String table;
String[] keys;
final LinkedHashSet keysToStore = new LinkedHashSet();
Map> tableSchema = null;
transient boolean tableExists = false;
transient long count = 0L;
transient Connection connection = null;
transient List columns = new ArrayList();
/**
* @return the table
*/
public String getTable() {
return table;
}
/**
* @param table
* the table to set
*/
@Parameter(required = true, description = "The database table to insert items into.")
public void setTable(String table) {
this.table = table;
}
/**
* @return the keys
*/
public String[] getKeys() {
return keys;
}
/**
* @param keys
* the keys to set
*/
@Parameter(required = false, description = "A list of attributes to insert (columns), empty string for all attributes.")
public void setKeys(String[] keys) {
this.keys = keys;
}
/**
* @return the dropTable
*/
public boolean isDropTable() {
return dropTable;
}
/**
* @param dropTable
* the dropTable to set
*/
@Parameter(required = false, defaultValue = "false")
public void setDropTable(boolean dropTable) {
this.dropTable = dropTable;
}
/**
* @see stream.AbstractProcessor#init(stream.ProcessContext)
*/
@Override
public void init(ProcessContext ctx) throws Exception {
super.init(ctx);
if (table == null || table.trim().equals(""))
throw new Exception("No 'table' attribute provided!");
init();
log.debug("init(ProcessContext) done.");
}
private void init() throws Exception {
connection = openConnection();
log.debug("Opened connection to {} = {}", getUrl(), connection);
log.debug("Dialect = {} ", dialect);
if (url.toLowerCase().startsWith("jdbc:mysql")) {
dialect = new MysqlDialect();
}
if (url.toLowerCase().startsWith("jdbc:hsqldb"))
dialect = new HsqlDialect();
log.debug("Using dialect {}", dialect);
if (dropTable) {
log.debug("Dropping existing table '{}'", getTable());
try {
Statement stmt = connection.createStatement();
int ret = stmt.executeUpdate("DROP TABLE " + getTable());
log.debug("Return of DROP TABLE: {}", ret);
} catch (Exception e) {
log.error("Failed to drop table: {}", e.getMessage());
}
} else {
Map> schema = dialect.getTableSchema(connection,
getTable());
if (schema != null) {
log.debug("Using existing table schema: {}", schema);
log.info("Existing schema is: {}", schema);
if (tableSchema == null)
tableSchema = new LinkedHashMap>(schema);
else
tableSchema.putAll(schema);
if (keys != null) {
for (String key : keys) {
if (tableSchema.containsKey(key)) {
} else {
log.info("Removing non-selected key '{}'", key);
tableSchema.remove(key);
}
}
}
log.debug("Types:\n{}", tableSchema);
} else {
// throw new Exception("Cannot determine table-schema!");
}
}
}
public boolean hasTable(String name) {
if (tableExists)
return true;
tableExists = super.hasTable(name);
return tableExists;
}
/**
* @see stream.Processor#process(stream.Data)
*/
@Override
public Data process(Data input) {
if (connection == null) {
try {
init();
} catch (Exception e) {
throw new ProcessorException(this,
"Failed to initialize database connection: "
+ e.getMessage());
}
}
if (tableSchema == null) {
log.debug("No table-schema found, does table exist? {}",
this.hasTable(getTable()));
tableSchema = dialect.getTableSchema(connection, getTable());
log.debug("Tried to read schema from database: {}", tableSchema);
if (tableSchema == null) {
log.debug("Creating new table {} from first item {}",
getTable(), input);
Data sample = DataFactory.create();
if (keys != null) {
for (String key : keys)
sample.put(key, input.get(key));
} else {
sample.putAll(input);
}
Map> schema = dialect.getColumnTypes(sample);
if (createTable(getTable(), schema)) {
tableSchema = schema;
} else {
throw new ProcessorException(this,
"Failed to create table " + getTable()
+ " for item: " + input);
}
}
}
if (!hasTable(getTable())) {
if (keys != null) {
for (String key : keys) {
Serializable value = input.get(key);
if (value == null)
throw new ProcessorException(
this,
"Cannot determine type of key '"
+ key
+ "' for table creation! First item does not provide a value for '"
+ key + "'!");
tableSchema.put(key, value.getClass());
}
} else {
for (String key : input.keySet()) {
tableSchema.put(key, input.get(key).getClass());
}
}
if (!this.createTable(getTable(), tableSchema)) {
throw new ProcessorException(this, "Failed to create table '"
+ getTable() + "'!");
} else {
this.tableExists = true;
}
}
try {
StringBuffer insert = new StringBuffer("INSERT INTO ");
insert.append(getTable());
insert.append(" ( ");
StringBuffer values = new StringBuffer(" VALUES ( ");
List valueObject = new ArrayList();
Iterator it = tableSchema.keySet().iterator();
while (it.hasNext()) {
String key = it.next();
Serializable value = input.get(key);
if (value != null) {
valueObject.add(value);
insert.append(dialect.mapColumnName(key));
values.append("?");
if (it.hasNext()) {
insert.append(", ");
values.append(", ");
}
}
}
insert.append(" ) ");
values.append(" ) ");
insert.append(values.toString());
log.debug("INSERT statement is: {}", insert);
PreparedStatement ps = connection.prepareStatement(insert
.toString());
for (int i = 0; i < valueObject.size(); i++) {
ps.setObject(i + 1, valueObject.get(i));
}
int ret = ps.executeUpdate();
if (ret == 1)
count++;
log.debug("INSERT retured {}", ret);
ps.close();
} catch (Exception e) {
log.error("Failed to insert data item: {}", e.getMessage());
}
return input;
}
/**
* @see stream.AbstractProcessor#finish()
*/
@Override
public void finish() throws Exception {
super.finish();
log.debug("Closing SQL writer, {} items written.", count);
log.debug("Closing SQL connection...");
connection.close();
this.tableSchema = null;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy