net.snowflake.client.loader.StreamLoader Maven / Gradle / Ivy
/*
* Copyright (c) 2012-2019 Snowflake Computing Inc. All rights reserved.
*/
package net.snowflake.client.loader;
import static java.nio.charset.StandardCharsets.UTF_8;
import java.io.IOException;
import java.sql.Connection;
import java.sql.DatabaseMetaData;
import java.sql.SQLException;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.GregorianCalendar;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.TimeZone;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.zip.Deflater;
import net.snowflake.client.jdbc.SnowflakeType;
import net.snowflake.client.jdbc.SnowflakeUtil;
import net.snowflake.client.log.SFLogger;
import net.snowflake.client.log.SFLoggerFactory;
/** Stream Loader */
public class StreamLoader implements Loader, Runnable {
private static final SFLogger logger = SFLoggerFactory.getLogger(StreamLoader.class);
private static final String SYSTEM_PARAMETER_PREFIX = "net.snowflake.client.loader.";
static final String FILE_PREFIX = "stream_";
static final String FILE_SUFFIX = ".gz";
/** Default batch row size */
private static final long DEFAULT_BATCH_ROW_SIZE = -1L;
public static DatabaseMetaData metadata;
private BufferStage _stage = null;
private Operation _op = null;
private boolean _startTransaction = false;
// force not to truncate in case where loader.start is called in
// multiple times in a single job.
private boolean _is_first_start_call = true;
// force not to commit or rollback in case where loader.finish is called
// in multiple times in a single job.
private boolean _is_last_finish_call = true;
private boolean _oneBatch = false;
private boolean _truncate = false;
private String _before = null;
private String _after = null;
private ArrayBlockingQueue _queueData;
private Thread _thread;
private ArrayBlockingQueue _queuePut;
private PutQueue _put;
private ArrayBlockingQueue _queueProcess;
private ProcessQueue _process;
private String _remoteStage = "~";
private String _table;
private String _schema;
private String _database;
private List _columns;
private List _keys;
private long _batchRowSize = DEFAULT_BATCH_ROW_SIZE;
private long _csvFileBucketSize = BufferStage.FILE_BUCKET_SIZE;
private long _csvFileSize = BufferStage.FILE_SIZE;
boolean _testRemoteBadCSV = false; // TEST: inject bad csv in remote stage
boolean _preserveStageFile = false; // reserve stage file
private boolean _useLocalTimezone = false; // use local timezone instead of UTC
private boolean _mapTimeToTimestamp =
false; // map TIME to TIMESTAMP. Informatica V1 connector behavior
boolean _compressDataBeforePut = true; // compress data before PUT
boolean _compressFileByPut = false; // compress file by PUT
long _compressLevel = Deflater.BEST_SPEED; // compression level used to compress data before PUT
String _onError = OnError.DEFAULT;
boolean _copyEmptyFieldAsEmpty = false; // COPY command option to set EMPTY_FIELD_AS_NULL = false
boolean _testMode = false;
private final Connection _putConn;
private final Connection _processConn;
// a per-instance bit of random noise to make filenames more unique
private final String _noise;
// Track fatal errors
private AtomicBoolean _active = new AtomicBoolean(false);
private AtomicBoolean _aborted = new AtomicBoolean(false);
private RuntimeException _abortCause = new ConnectionError("Unknown exception");
private AtomicInteger _throttleCounter = new AtomicInteger(0);
private final GregorianCalendar _calendarUTC = new GregorianCalendar(TimeZone.getTimeZone("UTC"));
private GregorianCalendar _calendarLocal;
/** Resets calendar when start the job. */
private void resetCalendar() {
_calendarUTC.clear();
_calendarLocal = new GregorianCalendar(TimeZone.getDefault());
_calendarLocal.clear();
}
private DateFormat _dateFormat;
private DateFormat _timeFormat;
private DateFormat _timestampFormat;
private DateFormat _timestampTzFormat;
public StreamLoader(
Map properties,
Connection putConnection,
Connection processConnection) {
_putConn = putConnection;
_processConn = processConnection;
for (Map.Entry e : properties.entrySet()) {
setProperty(e.getKey(), e.getValue());
}
_noise = SnowflakeUtil.randomAlphaNumeric(6);
}
@Override
public void setProperty(LoaderProperty property, Object value) {
switch (property) {
case tableName:
_table = (String) value;
break;
case schemaName:
_schema = (String) value;
break;
case databaseName:
_database = (String) value;
break;
case remoteStage:
_remoteStage = (String) value;
break;
case columns:
if (value == null) {
_columns = null;
} else {
final List typeCheckedColumns = new ArrayList<>();
for (Object e : (List>) value) {
typeCheckedColumns.add((String) e);
}
_columns = typeCheckedColumns;
}
break;
case keys:
if (value == null) {
_keys = null;
} else {
final List typeCheckedKeys = new ArrayList<>();
for (Object e : (List>) value) {
typeCheckedKeys.add((String) e);
}
_keys = typeCheckedKeys;
}
break;
case operation:
_op = (Operation) value;
break;
case startTransaction:
_startTransaction = Boolean.valueOf(String.valueOf(value));
break;
case oneBatch:
_oneBatch = Boolean.valueOf(String.valueOf(value));
break;
case truncateTable:
_truncate = Boolean.valueOf(String.valueOf(value));
break;
case executeBefore:
_before = String.valueOf(value);
break;
case executeAfter:
_after = String.valueOf(value);
break;
case isFirstStartCall:
_is_first_start_call = Boolean.valueOf(String.valueOf(value));
break;
case isLastFinishCall:
_is_last_finish_call = Boolean.valueOf(String.valueOf(value));
break;
case batchRowSize:
_batchRowSize = parseLongValue(LoaderProperty.batchRowSize, value);
break;
case csvFileBucketSize:
_csvFileBucketSize = parseLongValue(LoaderProperty.csvFileBucketSize, value);
break;
case csvFileSize:
_csvFileSize = parseLongValue(LoaderProperty.csvFileSize, value);
break;
case preserveStageFile:
_preserveStageFile = Boolean.valueOf(String.valueOf(value));
break;
case useLocalTimezone:
_useLocalTimezone = Boolean.valueOf(String.valueOf(value));
break;
case copyEmptyFieldAsEmpty:
_copyEmptyFieldAsEmpty = Boolean.valueOf(String.valueOf(value));
break;
case mapTimeToTimestamp:
// NOTE: this is a special flag to change mapping
// from TIME. Informatica connector v1 maps to TIMESTAMP
// but a legitimate behavior is supposed to be to TIME.
_mapTimeToTimestamp = Boolean.valueOf(String.valueOf(value));
break;
case compressDataBeforePut:
_compressDataBeforePut = Boolean.valueOf(String.valueOf(value));
break;
case compressFileByPut:
_compressFileByPut = Boolean.valueOf(String.valueOf(value));
break;
case compressLevel:
_compressLevel = parseLongValue(LoaderProperty.compressLevel, value);
if ((_compressLevel < Deflater.BEST_SPEED || _compressLevel > Deflater.BEST_COMPRESSION)
&& _compressLevel != Deflater.DEFAULT_COMPRESSION) {
throw new IllegalArgumentException("invalid compression level");
}
break;
case onError:
String v = String.valueOf(value);
_onError = OnError.validate(v) ? v : OnError.DEFAULT;
break;
case testRemoteBadCSV:
_testRemoteBadCSV = Boolean.valueOf(String.valueOf(value));
break;
default:
// nop, this should ever happens
}
}
private long parseLongValue(LoaderProperty name, Object value) {
long ret;
if (value instanceof String) {
ret = Long.valueOf((String) value);
} else if (value instanceof Long) {
ret = (Long) value;
} else if (value instanceof Integer) {
ret = Long.valueOf((Integer) value);
} else {
throw new IllegalArgumentException(
String.format("'%s' Must be a LONG value", name.toString()));
}
return ret;
}
private void setPropertyBySystemProperty() {
final String BATCH_ROW_SIZE_KEY = SYSTEM_PARAMETER_PREFIX + "batchRowSize";
final String CSV_FILE_BUCKET_SIZE = SYSTEM_PARAMETER_PREFIX + "csvFileBucketSize";
final String CSV_FILE_SIZE = SYSTEM_PARAMETER_PREFIX + "csvFileSize";
final String COMPRESS_DATA_BEFORE_PUT_KEY = SYSTEM_PARAMETER_PREFIX + "compressDataBeforePut";
final String COMPRESS_FILE_BY_PUT_KEY = SYSTEM_PARAMETER_PREFIX + "compressFileByPut";
final String COMPRESS_LEVEL = SYSTEM_PARAMETER_PREFIX + "compressLevel";
Properties props = System.getProperties();
for (String propKey : props.stringPropertyNames()) {
String value = props.getProperty(propKey);
if (BATCH_ROW_SIZE_KEY.equals(propKey)) {
_batchRowSize = parseLongValue(LoaderProperty.batchRowSize, value);
} else if (CSV_FILE_BUCKET_SIZE.equals(propKey)) {
_csvFileBucketSize = parseLongValue(LoaderProperty.csvFileBucketSize, value);
} else if (CSV_FILE_SIZE.equals(propKey)) {
_csvFileSize = parseLongValue(LoaderProperty.csvFileSize, value);
} else if (COMPRESS_DATA_BEFORE_PUT_KEY.equals(propKey)) {
_compressDataBeforePut = Boolean.valueOf(value);
} else if (COMPRESS_FILE_BY_PUT_KEY.equals(propKey)) {
_compressFileByPut = Boolean.valueOf(value);
} else if (COMPRESS_LEVEL.equals(propKey)) {
_compressLevel = Long.valueOf(value);
}
}
}
private void initDateFormats() {
resetCalendar();
_dateFormat = new SimpleDateFormat(SnowflakeType.DATE_OR_TIME_FORMAT_PATTERN);
if (_mapTimeToTimestamp) {
// same format for TIME to TIMESTAMP
_timeFormat = _dateFormat;
} else {
_timeFormat = new SimpleDateFormat(SnowflakeType.TIME_FORMAT_PATTERN);
}
_timestampFormat = new SimpleDateFormat(SnowflakeType.TIMESTAMP_FORMAT_PATTERN);
_timestampTzFormat = new SimpleDateFormat(SnowflakeType.TIMESTAMP_FORMAT_TZ_PATTERN);
Calendar cal = !_useLocalTimezone ? _calendarUTC : _calendarLocal;
_dateFormat.setCalendar(cal);
_timeFormat.setCalendar(cal);
_timestampFormat.setCalendar(cal);
_timestampTzFormat.setCalendar(cal);
}
/** Starts the loader */
@Override
public void start() {
logger.debug("Start Loading", false);
// validate parameters
validateParameters();
if (_op == null) {
this.abort(new ConnectionError("Loader started with no operation"));
return;
}
initDateFormats();
initQueues();
if (_is_first_start_call) {
// is this the first start call?
try {
if (_startTransaction) {
logger.debug("Begin Transaction", false);
_processConn.createStatement().execute("begin transaction");
} else {
logger.debug("No Transaction started", false);
}
} catch (SQLException ex) {
abort(new Loader.ConnectionError("Failed to start Transaction", Utils.getCause(ex)));
}
if (_truncate) {
truncateTargetTable();
}
try {
if (_before != null) {
logger.debug("Running Execute Before SQL", false);
_processConn.createStatement().execute(_before);
}
} catch (SQLException ex) {
abort(
new Loader.ConnectionError(
String.format("Execute Before SQL failed to run: %s", _before),
Utils.getCause(ex)));
}
}
}
private void validateParameters() {
logger.debug("Validate Parameters", false);
if (Operation.INSERT != this._op) {
if (this._keys == null || this._keys.isEmpty()) {
throw new ConnectionError("Updating operations require keys");
}
}
setPropertyBySystemProperty();
logger.debug(
"Database Name: {}, Schema Name: {}, Table Name: {}, "
+ "Remote Stage: {}, Columns: {}, Keys: {}, Operation: {}, "
+ "Start Transaction: {}, OneBatch: {}, Truncate Table: {}, "
+ "Execute Before: {}, Execute After: {}, Batch Row Size: {}, "
+ "CSV File Bucket Size: {}, CSV File Size: {}, Preserve Stage File: {}, "
+ "Use Local TimeZone: {}, Copy Empty Field As Empty: {}, "
+ "MapTimeToTimestamp: {}, Compress Data before PUT: {}, "
+ "Compress File By Put: {}, Compress Level: {}, OnError: {}",
_database,
_schema,
_table,
_remoteStage,
_columns,
_keys,
_op,
_startTransaction,
_oneBatch,
_truncate,
_before,
_after,
_batchRowSize,
_csvFileBucketSize,
_csvFileSize,
_preserveStageFile,
_useLocalTimezone,
_copyEmptyFieldAsEmpty,
_mapTimeToTimestamp,
_compressDataBeforePut,
_compressFileByPut,
_compressLevel,
_onError);
}
String getNoise() {
return _noise;
}
public void abort(RuntimeException t) {
synchronized (this) {
// Abort once, keep first error.
logger.debug("Exception received. Aborting...", t);
if (_aborted.getAndSet(true)) {
return;
}
if (t != null) {
_abortCause = t;
}
// Rollback and do not process anything else
rollback();
}
}
boolean isAborted() {
synchronized (this) {
// don't synchronize unless the caller does
return _aborted.get();
}
}
@Override
public void rollback() {
logger.debug("Rollback", false);
try {
terminate();
logger.debug("Rollback", false);
this._processConn.createStatement().execute("rollback");
} catch (SQLException ex) {
logger.error(ex.getMessage(), ex);
}
}
@Override
public void submitRow(final Object[] row) {
try {
if (_aborted.get()) {
if (_listener.throwOnError()) {
throw _abortCause;
}
return;
}
} catch (Exception ex) {
abort(new Loader.ConnectionError("Throwing Error", Utils.getCause(ex)));
}
byte[] data = null;
try {
if (!_active.get()) {
logger.debug("Inactive loader. Row ignored", false);
return;
}
data = createCSVRecord(row);
} catch (Exception ex) {
abort(new Loader.ConnectionError("Creating data set for CSV", Utils.getCause(ex)));
}
try {
writeBytes(data);
_listener.addSubmittedRowCount(1);
if (_listener.needSuccessRecords()) {
_listener.recordProvided(_op, row);
}
} catch (Exception ex) {
abort(new Loader.ConnectionError("Writing Bytes to CSV files", Utils.getCause(ex)));
}
if (_batchRowSize > 0
&& _listener.getSubmittedRowCount() > 0
&& (_listener.getSubmittedRowCount() % _batchRowSize) == 0) {
logger.debug(
"Flushing Queue: Submitted Row Count: {}, Batch Row Size: {}",
_listener.getSubmittedRowCount(),
_batchRowSize);
// flush data loading
try {
flushQueues();
} catch (Exception ex) {
abort(new Loader.ConnectionError("Flush Queues", Utils.getCause(ex)));
}
try {
initQueues();
} catch (Exception ex) {
abort(new Loader.ConnectionError("Init Queues", Utils.getCause(ex)));
}
}
}
/** Initializes queues */
private void initQueues() {
logger.debug("Init Queues", false);
if (_active.getAndSet(true)) {
// NOP if the loader is already active
return;
}
// start PUT and PROCESS queues
_queuePut = new ArrayBlockingQueue<>(48);
_queueProcess = new ArrayBlockingQueue<>(48);
_put = new PutQueue(this);
_process = new ProcessQueue(this);
// Start queue. NOTE: This is not actively used
_queueData = new ArrayBlockingQueue<>(1024);
_thread = new Thread(this);
_thread.setName("StreamLoaderThread");
_thread.start();
// Create stage
_stage = new BufferStage(this, _op, _csvFileBucketSize, _csvFileSize);
}
/** Flushes data by joining PUT and PROCESS queues */
private void flushQueues() {
// Terminate data loading thread.
logger.debug("Flush Queues", false);
try {
_queueData.put(new byte[0]);
_thread.join(10000);
if (_thread.isAlive()) {
_thread.interrupt();
}
} catch (Exception ex) {
String msg = "Failed to join StreamLoader queue: " + ex.getMessage();
logger.error(msg, ex);
throw new DataError(msg, Utils.getCause(ex));
}
// Put last stage on queue
terminate();
// wait for the processing to finish
_put.join();
_process.join();
if (_aborted.get()) {
// Loader was aborted due to an exception.
// It was rolled back at that time.
// LOGGER.log(Level.WARNING,
// "Loader had been previously aborted by error", _abortCause);
throw _abortCause;
}
}
private void writeBytes(final byte[] data) throws IOException, InterruptedException {
// this loader was aborted
if (_aborted.get()) {
return;
}
boolean full = _stage.stageData(data);
if (full && !_oneBatch) {
// if Buffer stage is full and NOT one batch mode,
// queue PUT request.
queuePut(_stage);
_stage = new BufferStage(this, _op, _csvFileBucketSize, _csvFileSize);
}
}
private void truncateTargetTable() {
try {
// TODO: could be replaced with TRUNCATE?
_processConn.createStatement().execute("DELETE FROM " + this.getFullTableName());
} catch (SQLException ex) {
logger.error(ex.getMessage(), ex);
abort(new Loader.ConnectionError(Utils.getCause(ex)));
}
}
@Override
public void run() {
try {
while (true) {
byte[] data = this._queueData.take();
if (data.length == 0) {
break;
}
this.writeBytes(data);
}
} catch (Exception ex) {
logger.error(ex.getMessage(), ex);
abort(new Loader.ConnectionError(Utils.getCause(ex)));
}
}
private byte[] createCSVRecord(final Object[] data) {
StringBuilder sb = new StringBuilder();
for (int i = 0; i < data.length; ++i) {
if (i > 0) {
sb.append(',');
}
sb.append(
SnowflakeType.escapeForCSV(
SnowflakeType.lexicalValue(
data[i], _dateFormat, _timeFormat, _timestampFormat, _timestampTzFormat)));
}
return sb.toString().getBytes(UTF_8);
}
/**
* Finishes loader
*
* @throws Exception an exception raised in finishing loader.
*/
@Override
public void finish() throws Exception {
logger.debug("Finish Loading", false);
flushQueues();
if (_is_last_finish_call) {
try {
if (_after != null) {
logger.debug("Running Execute After SQL", false);
_processConn.createStatement().execute(_after);
}
// Loader successfully completed. Commit and return.
_processConn.createStatement().execute("commit");
logger.debug("Committed", false);
} catch (SQLException ex) {
try {
_processConn.createStatement().execute("rollback");
} catch (SQLException ex0) {
logger.debug("Failed to rollback", false);
}
logger.debug(String.format("Execute After SQL failed to run: %s", _after), ex);
throw new Loader.ConnectionError(Utils.getCause(ex));
}
}
}
@Override
public void close() {
logger.debug("Close Loader", false);
try {
this._processConn.close();
this._putConn.close();
} catch (SQLException ex) {
logger.error(ex.getMessage(), ex);
throw new ConnectionError(Utils.getCause(ex));
}
}
/** Set active to false (no-op if not active), add a stage with terminate flag onto the queue */
private void terminate() {
logger.debug("Terminate Loader", false);
boolean active = _active.getAndSet(false);
if (!active) {
return; // No-op
}
if (_stage == null) {
_stage = new BufferStage(this, Operation.INSERT, _csvFileBucketSize, _csvFileSize);
}
_stage.setTerminate(true);
try {
queuePut(_stage);
} catch (InterruptedException ex) {
logger.error("Unknown Error", ex);
}
logger.debug("Snowflake loader terminating", false);
}
// If operation changes, existing stage needs to be scheduled for processing.
@Override
public void resetOperation(Operation op) {
if (op.equals(_op)) {
// no-op
return;
}
logger.debug("Operation is changing from {} to {}", _op, op);
_op = op;
if (_stage != null) {
try {
queuePut(_stage);
} catch (InterruptedException ex) {
logger.error(_stage.getId(), ex);
}
}
_stage = new BufferStage(this, _op, _csvFileBucketSize, _csvFileSize);
}
String getTable() {
return _table;
}
String getBase() {
return BASE;
}
Connection getPutConnection() {
return _putConn;
}
Connection getProcessConnection() {
return _processConn;
}
String getRemoteStage() {
return _remoteStage;
}
List getKeys() {
return this._keys;
}
List getColumns() {
return this._columns;
}
String getColumnsAsString() {
// comma separate list of column names
StringBuilder sb = new StringBuilder("\"");
for (int i = 0; i < _columns.size(); i++) {
if (i > 0) {
sb.append("\",\"");
}
sb.append(_columns.get(i));
}
sb.append("\"");
return sb.toString();
}
String getFullTableName() {
return (_database == null ? "" : ("\"" + _database + "\"."))
+ (_schema == null ? "" : ("\"" + _schema + "\"."))
+ "\""
+ _table
+ "\"";
}
public LoadResultListener getListener() {
return _listener;
}
@Override
public void setListener(LoadResultListener _listener) {
this._listener = _listener;
}
private void queuePut(BufferStage stage) throws InterruptedException {
_queuePut.put(stage);
}
BufferStage takePut() throws InterruptedException {
return _queuePut.take();
}
void queueProcess(BufferStage stage) throws InterruptedException {
_queueProcess.put(stage);
}
BufferStage takeProcess() throws InterruptedException {
return _queueProcess.take();
}
void throttleUp() {
int open = this._throttleCounter.incrementAndGet();
logger.debug("PUT Throttle Up: {}", open);
if (open > 8) {
logger.debug(
"Will retry scheduling file for upload after {} seconds", (Math.pow(2, open - 7)));
try {
Thread.sleep(1000 * ((int) Math.pow(2, open - 7)));
} catch (InterruptedException ex) {
logger.error("Exception occurs while waiting", ex);
}
}
}
void throttleDown() {
int throttleLevel = this._throttleCounter.decrementAndGet();
logger.debug("PUT Throttle Down: {}", throttleLevel);
if (throttleLevel < 0) {
logger.debug("Unbalanced throttle", false);
_throttleCounter.set(0);
}
logger.debug("Connector throttle {}", throttleLevel);
}
private LoadResultListener _listener =
new LoadResultListener() {
private final AtomicInteger errorCount = new AtomicInteger(0);
private final AtomicInteger errorRecordCount = new AtomicInteger(0);
private final AtomicInteger submittedRowCount = new AtomicInteger(0);
@Override
public boolean needErrors() {
return false;
}
@Override
public boolean needSuccessRecords() {
return false;
}
@Override
public void addError(LoadingError error) {}
@Override
public boolean throwOnError() {
return false;
}
@Override
public void recordProvided(Operation op, Object[] record) {}
@Override
public void addProcessedRecordCount(Operation op, int i) {}
@Override
public void addOperationRecordCount(Operation op, int i) {}
@Override
public int getErrorCount() {
return errorCount.get();
}
@Override
public int getErrorRecordCount() {
return errorRecordCount.get();
}
@Override
public void resetErrorCount() {
errorCount.set(0);
}
@Override
public void resetErrorRecordCount() {
errorRecordCount.set(0);
}
@Override
public void addErrorCount(int count) {
errorCount.addAndGet(count);
}
@Override
public void addErrorRecordCount(int count) {
errorRecordCount.addAndGet(count);
}
@Override
public void resetSubmittedRowCount() {
submittedRowCount.set(0);
}
@Override
public void addSubmittedRowCount(int count) {
submittedRowCount.addAndGet(count);
}
@Override
public int getSubmittedRowCount() {
return submittedRowCount.get();
}
};
void setTestMode(boolean mode) {
this._testMode = mode;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy