/*! ****************************************************************************** * * Pentaho Data Integration * * Copyright (C) 2002-2016 by Pentaho : http://www.pentaho.com * ******************************************************************************* * * 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.pentaho.di.trans.steps.update; import java.sql.SQLException; import java.util.ArrayList; import org.apache.commons.lang.ArrayUtils; import org.pentaho.di.core.Const; import org.pentaho.di.core.util.Utils; import org.pentaho.di.core.database.Database; import org.pentaho.di.core.database.DatabaseMeta; import org.pentaho.di.core.exception.KettleDatabaseException; import org.pentaho.di.core.exception.KettleException; import org.pentaho.di.core.exception.KettleStepException; import org.pentaho.di.core.row.RowMeta; import org.pentaho.di.core.row.RowMetaInterface; import org.pentaho.di.core.row.ValueMetaInterface; import org.pentaho.di.i18n.BaseMessages; import org.pentaho.di.trans.Trans; import org.pentaho.di.trans.TransMeta; import org.pentaho.di.trans.step.BaseStep; import org.pentaho.di.trans.step.StepDataInterface; import org.pentaho.di.trans.step.StepInterface; import org.pentaho.di.trans.step.StepMeta; import org.pentaho.di.trans.step.StepMetaInterface; /** * Update data in a database table, does NOT ever perform an insert. * * @author Matt * @since 26-apr-2003 */ public class Update extends BaseStep implements StepInterface { private static Class<?> PKG = UpdateMeta.class; // for i18n purposes, needed by Translator2!! private UpdateMeta meta; private UpdateData data; public Update( StepMeta stepMeta, StepDataInterface stepDataInterface, int copyNr, TransMeta transMeta, Trans trans ) { super( stepMeta, stepDataInterface, copyNr, transMeta, trans ); } private synchronized Object[] lookupValues( RowMetaInterface rowMeta, Object[] row ) throws KettleException { Object[] outputRow = row; Object[] add; // Create the output row and copy the input values if ( !Utils.isEmpty( meta.getIgnoreFlagField() ) ) { // add flag field! outputRow = new Object[data.outputRowMeta.size()]; for ( int i = 0; i < rowMeta.size(); i++ ) { outputRow[i] = row[i]; } } // OK, now do the lookup. // We need the lookupvalues for that. Object[] lookupRow = new Object[data.lookupParameterRowMeta.size()]; int lookupIndex = 0; for ( int i = 0; i < data.keynrs.length; i++ ) { if ( data.keynrs[i] >= 0 ) { lookupRow[lookupIndex] = row[data.keynrs[i]]; lookupIndex++; } if ( data.keynrs2[i] >= 0 ) { lookupRow[lookupIndex] = row[data.keynrs2[i]]; lookupIndex++; } } RowMetaInterface returnRowMeta = null; if ( !meta.isSkipLookup() ) { data.db.setValues( data.lookupParameterRowMeta, lookupRow, data.prepStatementLookup ); if ( log.isDebug() ) { logDebug( BaseMessages.getString( PKG, "Update.Log.ValuesSetForLookup", data.lookupParameterRowMeta .getString( lookupRow ), rowMeta.getString( row ) ) ); } add = data.db.getLookup( data.prepStatementLookup ); returnRowMeta = data.db.getReturnRowMeta(); } else { add = null; } incrementLinesInput(); if ( add == null && !meta.isSkipLookup() ) { /* * nothing was found: throw error! */ if ( !meta.isErrorIgnored() ) { if ( getStepMeta().isDoingErrorHandling() ) { outputRow = null; if ( data.stringErrorKeyNotFound == null ) { data.stringErrorKeyNotFound = BaseMessages.getString( PKG, "Update.Exception.KeyCouldNotFound" ) + data.lookupParameterRowMeta.getString( lookupRow ); data.stringFieldnames = ""; for ( int i = 0; i < data.lookupParameterRowMeta.size(); i++ ) { if ( i > 0 ) { data.stringFieldnames += ", "; } data.stringFieldnames += data.lookupParameterRowMeta.getValueMeta( i ).getName(); } } putError( rowMeta, row, 1L, data.stringErrorKeyNotFound, data.stringFieldnames, "UPD001" ); } else { throw new KettleDatabaseException( BaseMessages.getString( PKG, "Update.Exception.KeyCouldNotFound" ) + data.lookupParameterRowMeta.getString( lookupRow ) ); } } else { if ( log.isDetailed() ) { logDetailed( BaseMessages.getString( PKG, "Update.Log.KeyCouldNotFound" ) + data.lookupParameterRowMeta.getString( lookupRow ) ); } if ( !Utils.isEmpty( meta.getIgnoreFlagField() ) ) { // set flag field! outputRow[rowMeta.size()] = Boolean.FALSE; } } } else { if ( !meta.isSkipLookup() ) { if ( log.isRowLevel() ) { logRowlevel( BaseMessages.getString( PKG, "Update.Log.FoundRow" ) + data.lookupReturnRowMeta.getString( add ) ); } } /* * Row was found: * * UPDATE row or do nothing? */ boolean update = false; if ( meta.isSkipLookup() ) { // Update fields directly update = true; } else { for ( int i = 0; i < data.valuenrs.length; i++ ) { ValueMetaInterface valueMeta = rowMeta.getValueMeta( data.valuenrs[i] ); Object rowvalue = row[data.valuenrs[i]]; ValueMetaInterface returnValueMeta = returnRowMeta.getValueMeta( i ); Object retvalue = add[i]; if ( returnValueMeta.compare( retvalue, valueMeta, rowvalue ) != 0 ) { update = true; } } } if ( update ) { // Create the update row... Object[] updateRow = new Object[data.updateParameterRowMeta.size()]; for ( int i = 0; i < data.valuenrs.length; i++ ) { updateRow[i] = row[data.valuenrs[i]]; // the setters } // add the where clause parameters, they are exactly the same for lookup and update for ( int i = 0; i < lookupRow.length; i++ ) { updateRow[data.valuenrs.length + i] = lookupRow[i]; } if ( log.isRowLevel() ) { logRowlevel( BaseMessages.getString( PKG, "Update.Log.UpdateRow" ) + data.lookupParameterRowMeta.getString( lookupRow ) ); } data.db.setValues( data.updateParameterRowMeta, updateRow, data.prepStatementUpdate ); data.db.insertRow( data.prepStatementUpdate, meta.useBatchUpdate(), true ); incrementLinesUpdated(); } else { incrementLinesSkipped(); } if ( !Utils.isEmpty( meta.getIgnoreFlagField() ) ) { // add flag field! outputRow[rowMeta.size()] = Boolean.TRUE; } } return outputRow; } public boolean processRow( StepMetaInterface smi, StepDataInterface sdi ) throws KettleException { meta = (UpdateMeta) smi; data = (UpdateData) sdi; boolean sendToErrorRow = false; String errorMessage = null; Object[] r = getRow(); // Get row from input rowset & set row busy! if ( r == null ) { // no more input to be expected... setOutputDone(); return false; } if ( first ) { first = false; // What's the output Row format? data.outputRowMeta = getInputRowMeta().clone(); meta.getFields( data.outputRowMeta, getStepname(), null, null, this, repository, metaStore ); data.schemaTable = meta.getDatabaseMeta().getQuotedSchemaTableCombination( environmentSubstitute( meta.getSchemaName() ), environmentSubstitute( meta.getTableName() ) ); // lookup the values! if ( log.isDetailed() ) { logDetailed( BaseMessages.getString( PKG, "Update.Log.CheckingRow" ) + getInputRowMeta().getString( r ) ); } ArrayList<Integer> keynrs = new ArrayList<Integer>( meta.getKeyStream().length ); ArrayList<Integer> keynrs2 = new ArrayList<Integer>( meta.getKeyStream().length ); for ( int i = 0; i < meta.getKeyStream().length; i++ ) { int keynr = getInputRowMeta().indexOfValue( meta.getKeyStream()[i] ); if ( keynr < 0 && // couldn't find field! !"IS NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) && // No field needed! !"IS NOT NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) // No field needed! ) { throw new KettleStepException( BaseMessages.getString( PKG, "Update.Exception.FieldRequired", meta .getKeyStream()[i] ) ); } keynrs.add( keynr ); // this operator needs two bindings if ( "= ~NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) ) { keynrs.add( keynr ); keynrs2.add( -1 ); } int keynr2 = getInputRowMeta().indexOfValue( meta.getKeyStream2()[i] ); if ( keynr2 < 0 && // couldn't find field! "BETWEEN".equalsIgnoreCase( meta.getKeyCondition()[i] ) // 2 fields needed! ) { throw new KettleStepException( BaseMessages.getString( PKG, "Update.Exception.FieldRequired", meta .getKeyStream2()[i] ) ); } keynrs2.add( keynr2 ); if ( log.isDebug() ) { logDebug( BaseMessages.getString( PKG, "Update.Log.FieldHasDataNumbers", meta.getKeyStream()[i] ) + "" + keynrs.get( keynrs.size() - 1 ) ); } } data.keynrs = ArrayUtils.toPrimitive( keynrs.toArray( new Integer[0] ) ); data.keynrs2 = ArrayUtils.toPrimitive( keynrs2.toArray( new Integer[0] ) ); // Cache the position of the compare fields in Row row // data.valuenrs = new int[meta.getUpdateLookup().length]; for ( int i = 0; i < meta.getUpdateLookup().length; i++ ) { data.valuenrs[i] = getInputRowMeta().indexOfValue( meta.getUpdateStream()[i] ); if ( data.valuenrs[i] < 0 ) { // couldn't find field! throw new KettleStepException( BaseMessages.getString( PKG, "Update.Exception.FieldRequired", meta .getUpdateStream()[i] ) ); } if ( log.isDebug() ) { logDebug( BaseMessages.getString( PKG, "Update.Log.FieldHasDataNumbers", meta.getUpdateStream()[i] ) + "" + data.valuenrs[i] ); } } if ( meta.isSkipLookup() ) { // We skip lookup // but we need fields for update data.lookupParameterRowMeta = new RowMeta(); for ( int i = 0; i < meta.getKeyLookup().length; i++ ) { if ( "BETWEEN".equalsIgnoreCase( meta.getKeyCondition()[i] ) ) { data.lookupParameterRowMeta.addValueMeta( getInputRowMeta().searchValueMeta( meta.getKeyStream()[i] ) ); data.lookupParameterRowMeta .addValueMeta( getInputRowMeta().searchValueMeta( meta.getKeyStream2()[i] ) ); } else { if ( "= ~NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) ) { data.lookupParameterRowMeta .addValueMeta( getInputRowMeta().searchValueMeta( meta.getKeyStream()[i] ) ); data.lookupParameterRowMeta.addValueMeta( getInputRowMeta() .searchValueMeta( meta.getKeyStream()[i] ).clone() ); } else if ( !"IS NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) && !"IS NOT NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) ) { data.lookupParameterRowMeta .addValueMeta( getInputRowMeta().searchValueMeta( meta.getKeyStream()[i] ) ); } } } } else { setLookup( getInputRowMeta() ); } prepareUpdate( getInputRowMeta() ); } try { Object[] outputRow = lookupValues( getInputRowMeta(), r ); // add new values to the row in rowset[0]. if ( outputRow != null ) { putRow( data.outputRowMeta, outputRow ); // copy non-ignored rows to output rowset(s); } if ( checkFeedback( getLinesRead() ) ) { if ( log.isBasic() ) { logBasic( BaseMessages.getString( PKG, "Update.Log.LineNumber" ) + getLinesRead() ); } } } catch ( KettleException e ) { if ( getStepMeta().isDoingErrorHandling() ) { sendToErrorRow = true; errorMessage = e.toString(); } else { logError( BaseMessages.getString( PKG, "Update.Log.ErrorInStep" ), e ); setErrors( 1 ); stopAll(); setOutputDone(); // signal end to receiver(s) return false; } if ( sendToErrorRow ) { // Simply add this row to the error row putError( getInputRowMeta(), r, 1, errorMessage, null, "UPD001" ); } } return true; } public void setLookup( RowMetaInterface rowMeta ) throws KettleDatabaseException { data.lookupParameterRowMeta = new RowMeta(); data.lookupReturnRowMeta = new RowMeta(); DatabaseMeta databaseMeta = meta.getDatabaseMeta(); String sql = "SELECT "; for ( int i = 0; i < meta.getUpdateLookup().length; i++ ) { if ( i != 0 ) { sql += ", "; } sql += databaseMeta.quoteField( meta.getUpdateLookup()[i] ); data.lookupReturnRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getUpdateStream()[i] ) ); } sql += " FROM " + data.schemaTable + " WHERE "; for ( int i = 0; i < meta.getKeyLookup().length; i++ ) { if ( i != 0 ) { sql += " AND "; } sql += " ( ( "; sql += databaseMeta.quoteField( meta.getKeyLookup()[i] ); if ( "BETWEEN".equalsIgnoreCase( meta.getKeyCondition()[i] ) ) { sql += " BETWEEN ? AND ? "; data.lookupParameterRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getKeyStream()[i] ) ); data.lookupParameterRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getKeyStream2()[i] ) ); } else { if ( "IS NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) || "IS NOT NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) ) { sql += " " + meta.getKeyCondition()[i] + " "; } else if ( "= ~NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) ) { sql += " IS NULL AND "; if ( databaseMeta.requiresCastToVariousForIsNull() ) { sql += "CAST(? AS VARCHAR(256)) IS NULL"; } else { sql += "? IS NULL"; } // null check data.lookupParameterRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getKeyStream()[i] ) ); sql += " ) OR ( " + databaseMeta.quoteField( meta.getKeyLookup()[i] ) + " = ?"; // equality check, cloning so auto-rename because of adding same fieldname does not cause problems data.lookupParameterRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getKeyStream()[i] ).clone() ); } else { sql += " " + meta.getKeyCondition()[i] + " ? "; data.lookupParameterRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getKeyStream()[i] ) ); } } sql += " ) ) "; } try { if ( log.isDetailed() ) { logDetailed( "Setting preparedStatement to [" + sql + "]" ); } data.prepStatementLookup = data.db.getConnection().prepareStatement( databaseMeta.stripCR( sql ) ); } catch ( SQLException ex ) { throw new KettleDatabaseException( "Unable to prepare statement for SQL statement [" + sql + "]", ex ); } } // Lookup certain fields in a table public void prepareUpdate( RowMetaInterface rowMeta ) throws KettleDatabaseException { DatabaseMeta databaseMeta = meta.getDatabaseMeta(); data.updateParameterRowMeta = new RowMeta(); String sql = "UPDATE " + data.schemaTable + Const.CR; sql += "SET "; for ( int i = 0; i < meta.getUpdateLookup().length; i++ ) { if ( i != 0 ) { sql += ", "; } sql += databaseMeta.quoteField( meta.getUpdateLookup()[i] ); sql += " = ?" + Const.CR; data.updateParameterRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getUpdateStream()[i] ) ); } sql += "WHERE "; for ( int i = 0; i < meta.getKeyLookup().length; i++ ) { if ( i != 0 ) { sql += "AND "; } sql += " ( ( "; sql += databaseMeta.quoteField( meta.getKeyLookup()[i] ); if ( "BETWEEN".equalsIgnoreCase( meta.getKeyCondition()[i] ) ) { sql += " BETWEEN ? AND ? "; data.updateParameterRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getKeyStream()[i] ) ); data.updateParameterRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getKeyStream2()[i] ) ); } else if ( "IS NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) || "IS NOT NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) ) { sql += " " + meta.getKeyCondition()[i] + " "; } else if ( "= ~NULL".equalsIgnoreCase( meta.getKeyCondition()[i] ) ) { sql += " IS NULL AND "; if ( databaseMeta.requiresCastToVariousForIsNull() ) { sql += "CAST(? AS VARCHAR(256)) IS NULL"; } else { sql += "? IS NULL"; } // null check data.updateParameterRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getKeyStream()[i] ) ); sql += " ) OR ( " + databaseMeta.quoteField( meta.getKeyLookup()[i] ) + " = ?"; // equality check, cloning so auto-rename because of adding same fieldname does not cause problems data.updateParameterRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getKeyStream()[i] ).clone() ); } else { sql += " " + meta.getKeyCondition()[i] + " ? "; data.updateParameterRowMeta.addValueMeta( rowMeta.searchValueMeta( meta.getKeyStream()[i] ) ); } sql += " ) ) "; } try { if ( log.isDetailed() ) { logDetailed( "Setting update preparedStatement to [" + sql + "]" ); } data.prepStatementUpdate = data.db.getConnection().prepareStatement( databaseMeta.stripCR( sql ) ); } catch ( SQLException ex ) { throw new KettleDatabaseException( "Unable to prepare statement for SQL statement [" + sql + "]", ex ); } } public boolean init( StepMetaInterface smi, StepDataInterface sdi ) { meta = (UpdateMeta) smi; data = (UpdateData) sdi; if ( super.init( smi, sdi ) ) { if ( meta.getDatabaseMeta() == null ) { logError( BaseMessages.getString( PKG, "Update.Init.ConnectionMissing", getStepname() ) ); return false; } data.db = new Database( this, meta.getDatabaseMeta() ); data.db.shareVariablesWith( this ); try { if ( getTransMeta().isUsingUniqueConnections() ) { synchronized ( getTrans() ) { data.db.connect( getTrans().getTransactionId(), getPartitionID() ); } } else { data.db.connect( getPartitionID() ); } if ( log.isDetailed() ) { logDetailed( BaseMessages.getString( PKG, "Update.Log.ConnectedToDB" ) ); } data.db.setCommit( meta.getCommitSize( this ) ); return true; } catch ( KettleException ke ) { logError( BaseMessages.getString( PKG, "Update.Log.ErrorOccurred" ) + ke.getMessage() ); setErrors( 1 ); stopAll(); } } return false; } public void dispose( StepMetaInterface smi, StepDataInterface sdi ) { meta = (UpdateMeta) smi; data = (UpdateData) sdi; if ( data.db != null ) { try { if ( !data.db.isAutoCommit() ) { if ( getErrors() == 0 ) { data.db.emptyAndCommit( data.prepStatementUpdate, meta.useBatchUpdate() ); } else { data.db.rollback(); } } data.db.closePreparedStatement( data.prepStatementUpdate ); data.db.closePreparedStatement( data.prepStatementLookup ); } catch ( KettleDatabaseException e ) { logError( BaseMessages.getString( PKG, "Update.Log.UnableToCommitUpdateConnection" ) + data.db + "] :" + e.toString() ); setErrors( 1 ); } finally { data.db.disconnect(); } } super.dispose( smi, sdi ); } }