/*! ******************************************************************************
*
* 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.rowgenerator;
import java.text.DecimalFormat;
import java.util.ArrayList;
import java.util.List;
import org.pentaho.di.core.CheckResult;
import org.pentaho.di.core.CheckResultInterface;
import org.pentaho.di.core.Const;
import org.pentaho.di.core.util.Utils;
import org.pentaho.di.core.RowMetaAndData;
import org.pentaho.di.core.database.DatabaseMeta;
import org.pentaho.di.core.exception.KettleException;
import org.pentaho.di.core.exception.KettleStepException;
import org.pentaho.di.core.exception.KettleXMLException;
import org.pentaho.di.core.row.RowMetaInterface;
import org.pentaho.di.core.row.ValueMetaInterface;
import org.pentaho.di.core.variables.VariableSpace;
import org.pentaho.di.core.xml.XMLHandler;
import org.pentaho.di.i18n.BaseMessages;
import org.pentaho.di.repository.ObjectId;
import org.pentaho.di.repository.Repository;
import org.pentaho.di.trans.Trans;
import org.pentaho.di.trans.TransMeta;
import org.pentaho.di.trans.step.BaseStepMeta;
import org.pentaho.di.trans.step.StepDataInterface;
import org.pentaho.di.trans.step.StepIOMeta;
import org.pentaho.di.trans.step.StepIOMetaInterface;
import org.pentaho.di.trans.step.StepInterface;
import org.pentaho.di.trans.step.StepMeta;
import org.pentaho.di.trans.step.StepMetaInterface;
import org.pentaho.metastore.api.IMetaStore;
import org.w3c.dom.Node;
/*
* Created on 4-apr-2003
*/
public class RowGeneratorMeta extends BaseStepMeta implements StepMetaInterface {
private static Class<?> PKG = RowGeneratorMeta.class; // for i18n purposes, needed by Translator2!!
private boolean neverEnding;
private String intervalInMs;
private String rowTimeField;
private String lastTimeField;
private String rowLimit;
private String[] currency;
private String[] decimal;
private String[] group;
private String[] value;
private String[] fieldName;
private String[] fieldType;
private String[] fieldFormat;
private int[] fieldLength;
private int[] fieldPrecision;
/** Flag : set empty string **/
private boolean[] setEmptyString;
public RowGeneratorMeta() {
super(); // allocate BaseStepMeta
}
public void loadXML( Node stepnode, List<DatabaseMeta> databases, IMetaStore metaStore ) throws KettleXMLException {
readData( stepnode );
}
public void allocate( int nrfields ) {
fieldName = new String[nrfields];
fieldType = new String[nrfields];
fieldFormat = new String[nrfields];
fieldLength = new int[nrfields];
fieldPrecision = new int[nrfields];
currency = new String[nrfields];
decimal = new String[nrfields];
group = new String[nrfields];
value = new String[nrfields];
setEmptyString = new boolean[nrfields];
}
public Object clone() {
RowGeneratorMeta retval = (RowGeneratorMeta) super.clone();
int nrfields = fieldName.length;
retval.allocate( nrfields );
System.arraycopy( fieldName, 0, retval.fieldName, 0, nrfields );
System.arraycopy( fieldType, 0, retval.fieldType, 0, nrfields );
System.arraycopy( fieldFormat, 0, retval.fieldFormat, 0, nrfields );
System.arraycopy( fieldLength, 0, retval.fieldLength, 0, nrfields );
System.arraycopy( fieldPrecision, 0, retval.fieldPrecision, 0, nrfields );
System.arraycopy( currency, 0, retval.currency, 0, nrfields );
System.arraycopy( decimal, 0, retval.decimal, 0, nrfields );
System.arraycopy( group, 0, retval.group, 0, nrfields );
System.arraycopy( value, 0, retval.value, 0, nrfields );
System.arraycopy( setEmptyString, 0, retval.setEmptyString, 0, nrfields );
return retval;
}
private void readData( Node stepnode ) throws KettleXMLException {
try {
Node fields = XMLHandler.getSubNode( stepnode, "fields" );
int nrfields = XMLHandler.countNodes( fields, "field" );
allocate( nrfields );
String slength, sprecision;
for ( int i = 0; i < nrfields; i++ ) {
Node fnode = XMLHandler.getSubNodeByNr( fields, "field", i );
fieldName[i] = XMLHandler.getTagValue( fnode, "name" );
fieldType[i] = XMLHandler.getTagValue( fnode, "type" );
fieldFormat[i] = XMLHandler.getTagValue( fnode, "format" );
currency[i] = XMLHandler.getTagValue( fnode, "currency" );
decimal[i] = XMLHandler.getTagValue( fnode, "decimal" );
group[i] = XMLHandler.getTagValue( fnode, "group" );
value[i] = XMLHandler.getTagValue( fnode, "nullif" );
slength = XMLHandler.getTagValue( fnode, "length" );
sprecision = XMLHandler.getTagValue( fnode, "precision" );
fieldLength[i] = Const.toInt( slength, -1 );
fieldPrecision[i] = Const.toInt( sprecision, -1 );
String emptyString = XMLHandler.getTagValue( fnode, "set_empty_string" );
setEmptyString[i] = !Utils.isEmpty( emptyString ) && "Y".equalsIgnoreCase( emptyString );
}
// Is there a limit on the number of rows we process?
rowLimit = XMLHandler.getTagValue( stepnode, "limit" );
neverEnding = "Y".equalsIgnoreCase( XMLHandler.getTagValue( stepnode, "never_ending" ) );
intervalInMs = XMLHandler.getTagValue( stepnode, "interval_in_ms" );
rowTimeField = XMLHandler.getTagValue( stepnode, "row_time_field" );
lastTimeField = XMLHandler.getTagValue( stepnode, "last_time_field" );
} catch ( Exception e ) {
throw new KettleXMLException( "Unable to load step info from XML", e );
}
}
public void setDefault() {
int i, nrfields = 0;
allocate( nrfields );
DecimalFormat decimalFormat = new DecimalFormat();
for ( i = 0; i < nrfields; i++ ) {
fieldName[i] = "field" + i;
fieldType[i] = "Number";
fieldFormat[i] = "\u00A40,000,000.00;\u00A4-0,000,000.00";
fieldLength[i] = 9;
fieldPrecision[i] = 2;
currency[i] = decimalFormat.getDecimalFormatSymbols().getCurrencySymbol();
decimal[i] = new String( new char[] { decimalFormat.getDecimalFormatSymbols().getDecimalSeparator() } );
group[i] = new String( new char[] { decimalFormat.getDecimalFormatSymbols().getGroupingSeparator() } );
value[i] = "-";
setEmptyString[i] = false;
}
rowLimit = "10";
neverEnding = false;
intervalInMs = "5000";
rowTimeField = "now";
lastTimeField = "FiveSecondsAgo";
}
public void getFields( RowMetaInterface row, String origin, RowMetaInterface[] info, StepMeta nextStep,
VariableSpace space, Repository repository, IMetaStore metaStore ) throws KettleStepException {
try {
List<CheckResultInterface> remarks = new ArrayList<CheckResultInterface>();
RowMetaAndData rowMetaAndData = RowGenerator.buildRow( this, remarks, origin );
if ( !remarks.isEmpty() ) {
StringBuilder stringRemarks = new StringBuilder();
for ( CheckResultInterface remark : remarks ) {
stringRemarks.append( remark.toString() ).append( Const.CR );
}
throw new KettleStepException( stringRemarks.toString() );
}
for ( ValueMetaInterface valueMeta : rowMetaAndData.getRowMeta().getValueMetaList() ) {
valueMeta.setOrigin( origin );
}
row.mergeRowMeta( rowMetaAndData.getRowMeta() );
} catch ( Exception e ) {
throw new KettleStepException( e );
}
}
public String getXML() {
StringBuilder retval = new StringBuilder( 300 );
retval.append( " <fields>" ).append( Const.CR );
for ( int i = 0; i < fieldName.length; i++ ) {
if ( fieldName[i] != null && fieldName[i].length() != 0 ) {
retval.append( " <field>" ).append( Const.CR );
retval.append( " " ).append( XMLHandler.addTagValue( "name", fieldName[i] ) );
retval.append( " " ).append( XMLHandler.addTagValue( "type", fieldType[i] ) );
retval.append( " " ).append( XMLHandler.addTagValue( "format", fieldFormat[i] ) );
retval.append( " " ).append( XMLHandler.addTagValue( "currency", currency[i] ) );
retval.append( " " ).append( XMLHandler.addTagValue( "decimal", decimal[i] ) );
retval.append( " " ).append( XMLHandler.addTagValue( "group", group[i] ) );
retval.append( " " ).append( XMLHandler.addTagValue( "nullif", value[i] ) );
retval.append( " " ).append( XMLHandler.addTagValue( "length", fieldLength[i] ) );
retval.append( " " ).append( XMLHandler.addTagValue( "precision", fieldPrecision[i] ) );
retval.append( " " ).append( XMLHandler.addTagValue( "set_empty_string", setEmptyString[i] ) );
retval.append( " </field>" ).append( Const.CR );
}
}
retval.append( " </fields>" ).append( Const.CR );
retval.append( " " ).append( XMLHandler.addTagValue( "limit", rowLimit ) );
retval.append( " " ).append( XMLHandler.addTagValue( "never_ending", neverEnding ? "Y" : "N" ) );
retval.append( " " ).append( XMLHandler.addTagValue( "interval_in_ms", intervalInMs ) );
retval.append( " " ).append( XMLHandler.addTagValue( "row_time_field", rowTimeField ) );
retval.append( " " ).append( XMLHandler.addTagValue( "last_time_field", lastTimeField ) );
return retval.toString();
}
public void readRep( Repository rep, IMetaStore metaStore, ObjectId id_step, List<DatabaseMeta> databases ) throws KettleException {
try {
int nrfields = rep.countNrStepAttributes( id_step, "field_name" );
allocate( nrfields );
for ( int i = 0; i < nrfields; i++ ) {
fieldName[i] = rep.getStepAttributeString( id_step, i, "field_name" );
fieldType[i] = rep.getStepAttributeString( id_step, i, "field_type" );
fieldFormat[i] = rep.getStepAttributeString( id_step, i, "field_format" );
currency[i] = rep.getStepAttributeString( id_step, i, "field_currency" );
decimal[i] = rep.getStepAttributeString( id_step, i, "field_decimal" );
group[i] = rep.getStepAttributeString( id_step, i, "field_group" );
value[i] = rep.getStepAttributeString( id_step, i, "field_nullif" );
fieldLength[i] = (int) rep.getStepAttributeInteger( id_step, i, "field_length" );
fieldPrecision[i] = (int) rep.getStepAttributeInteger( id_step, i, "field_precision" );
setEmptyString[i] = rep.getStepAttributeBoolean( id_step, i, "set_empty_string", false );
}
rowLimit = rep.getStepAttributeString( id_step, "limit" );
neverEnding = rep.getStepAttributeBoolean( id_step, "never_ending" );
intervalInMs = rep.getStepAttributeString( id_step, "interval_in_ms" );
rowTimeField = rep.getStepAttributeString( id_step, "row_time_field" );
lastTimeField = rep.getStepAttributeString( id_step, "last_time_field" );
} catch ( Exception e ) {
throw new KettleException( "Unexpected error reading step information from the repository", e );
}
}
public void saveRep( Repository rep, IMetaStore metaStore, ObjectId id_transformation, ObjectId id_step ) throws KettleException {
try {
for ( int i = 0; i < fieldName.length; i++ ) {
if ( fieldName[i] != null && fieldName[i].length() != 0 ) {
rep.saveStepAttribute( id_transformation, id_step, i, "field_name", fieldName[i] );
rep.saveStepAttribute( id_transformation, id_step, i, "field_type", fieldType[i] );
rep.saveStepAttribute( id_transformation, id_step, i, "field_format", fieldFormat[i] );
rep.saveStepAttribute( id_transformation, id_step, i, "field_currency", currency[i] );
rep.saveStepAttribute( id_transformation, id_step, i, "field_decimal", decimal[i] );
rep.saveStepAttribute( id_transformation, id_step, i, "field_group", group[i] );
rep.saveStepAttribute( id_transformation, id_step, i, "field_nullif", value[i] );
rep.saveStepAttribute( id_transformation, id_step, i, "field_length", fieldLength[i] );
rep.saveStepAttribute( id_transformation, id_step, i, "field_precision", fieldPrecision[i] );
rep.saveStepAttribute( id_transformation, id_step, i, "set_empty_string", setEmptyString[i] );
}
}
rep.saveStepAttribute( id_transformation, id_step, "limit", rowLimit );
rep.saveStepAttribute( id_transformation, id_step, "never_ending", neverEnding );
rep.saveStepAttribute( id_transformation, id_step, "interval_in_ms", intervalInMs );
rep.saveStepAttribute( id_transformation, id_step, "row_time_field", rowTimeField );
rep.saveStepAttribute( id_transformation, id_step, "last_time_field", lastTimeField );
} catch ( Exception e ) {
throw new KettleException( "Unable to save step information to the repository for id_step=" + id_step, e );
}
}
public void check( List<CheckResultInterface> remarks, TransMeta transMeta, StepMeta stepMeta,
RowMetaInterface prev, String[] input, String[] output, RowMetaInterface info, VariableSpace space,
Repository repository, IMetaStore metaStore ) {
CheckResult cr;
if ( prev != null && prev.size() > 0 ) {
cr =
new CheckResult( CheckResultInterface.TYPE_RESULT_ERROR, BaseMessages.getString(
PKG, "RowGeneratorMeta.CheckResult.NoInputStreamsError" ), stepMeta );
remarks.add( cr );
} else {
cr =
new CheckResult( CheckResultInterface.TYPE_RESULT_OK, BaseMessages.getString(
PKG, "RowGeneratorMeta.CheckResult.NoInputStreamOk" ), stepMeta );
remarks.add( cr );
String strLimit = transMeta.environmentSubstitute( rowLimit );
if ( Const.toLong( strLimit, -1L ) <= 0 ) {
cr =
new CheckResult( CheckResultInterface.TYPE_RESULT_WARNING, BaseMessages.getString(
PKG, "RowGeneratorMeta.CheckResult.WarnNoRows" ), stepMeta );
remarks.add( cr );
} else {
cr =
new CheckResult( CheckResultInterface.TYPE_RESULT_OK, BaseMessages.getString(
PKG, "RowGeneratorMeta.CheckResult.WillReturnRows", strLimit ), stepMeta );
remarks.add( cr );
}
}
// See if we have input streams leading to this step!
if ( input.length > 0 ) {
cr =
new CheckResult( CheckResultInterface.TYPE_RESULT_ERROR, BaseMessages.getString(
PKG, "RowGeneratorMeta.CheckResult.NoInputError" ), stepMeta );
remarks.add( cr );
} else {
cr =
new CheckResult( CheckResultInterface.TYPE_RESULT_OK, BaseMessages.getString(
PKG, "RowGeneratorMeta.CheckResult.NoInputOk" ), stepMeta );
remarks.add( cr );
}
}
public StepInterface getStep( StepMeta stepMeta, StepDataInterface stepDataInterface, int cnr,
TransMeta transMeta, Trans trans ) {
return new RowGenerator( stepMeta, stepDataInterface, cnr, transMeta, trans );
}
public StepDataInterface getStepData() {
return new RowGeneratorData();
}
/**
* Returns the Input/Output metadata for this step. The generator step only produces output, does not accept input!
*/
public StepIOMetaInterface getStepIOMeta() {
return new StepIOMeta( false, true, false, false, false, false );
}
public boolean isNeverEnding() {
return neverEnding;
}
public void setNeverEnding( boolean neverEnding ) {
this.neverEnding = neverEnding;
}
public String getIntervalInMs() {
return intervalInMs;
}
public void setIntervalInMs( String intervalInMs ) {
this.intervalInMs = intervalInMs;
}
public String getRowTimeField() {
return rowTimeField;
}
public void setRowTimeField( String rowTimeField ) {
this.rowTimeField = rowTimeField;
}
public String getLastTimeField() {
return lastTimeField;
}
public void setLastTimeField( String lastTimeField ) {
this.lastTimeField = lastTimeField;
}
public boolean[] getSetEmptyString() {
return setEmptyString;
}
public void setSetEmptyString( boolean[] setEmptyString ) {
this.setEmptyString = setEmptyString;
}
/**
* @return Returns the currency.
*/
public String[] getCurrency() {
return currency;
}
/**
* @param currency
* The currency to set.
*/
public void setCurrency( String[] currency ) {
this.currency = currency;
}
/**
* @return Returns the decimal.
*/
public String[] getDecimal() {
return decimal;
}
/**
* @param decimal
* The decimal to set.
*/
public void setDecimal( String[] decimal ) {
this.decimal = decimal;
}
/**
* @return Returns the fieldFormat.
*/
public String[] getFieldFormat() {
return fieldFormat;
}
/**
* @param fieldFormat
* The fieldFormat to set.
*/
public void setFieldFormat( String[] fieldFormat ) {
this.fieldFormat = fieldFormat;
}
/**
* @return Returns the fieldLength.
*/
public int[] getFieldLength() {
return fieldLength;
}
/**
* @param fieldLength
* The fieldLength to set.
*/
public void setFieldLength( int[] fieldLength ) {
this.fieldLength = fieldLength;
}
/**
* @return Returns the fieldName.
*/
public String[] getFieldName() {
return fieldName;
}
/**
* @param fieldName
* The fieldName to set.
*/
public void setFieldName( String[] fieldName ) {
this.fieldName = fieldName;
}
/**
* @return Returns the fieldPrecision.
*/
public int[] getFieldPrecision() {
return fieldPrecision;
}
/**
* @param fieldPrecision
* The fieldPrecision to set.
*/
public void setFieldPrecision( int[] fieldPrecision ) {
this.fieldPrecision = fieldPrecision;
}
/**
* @return Returns the fieldType.
*/
public String[] getFieldType() {
return fieldType;
}
/**
* @param fieldType
* The fieldType to set.
*/
public void setFieldType( String[] fieldType ) {
this.fieldType = fieldType;
}
/**
* @return Returns the group.
*/
public String[] getGroup() {
return group;
}
/**
* @param group
* The group to set.
*/
public void setGroup( String[] group ) {
this.group = group;
}
/**
* @return the setEmptyString
*/
public boolean[] isSetEmptyString() {
return setEmptyString;
}
/**
* @param setEmptyString
* the setEmptyString to set
*/
public void setEmptyString( boolean[] setEmptyString ) {
this.setEmptyString = setEmptyString;
}
/**
* @return Returns the rowLimit.
*/
public String getRowLimit() {
return rowLimit;
}
/**
* @param rowLimit
* The rowLimit to set.
*/
public void setRowLimit( String rowLimit ) {
this.rowLimit = rowLimit;
}
/**
* @return Returns the value.
*/
public String[] getValue() {
return value;
}
/**
* @param value
* The value to set.
*/
public void setValue( String[] value ) {
this.value = value;
}
}