org.hpccsystems.spark.HpccFileWriter Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of spark-hpcc Show documentation
Show all versions of spark-hpcc Show documentation
Spark connector for reading files residing in an HPCC cluster environment
/*******************************************************************************
* HPCC SYSTEMS software Copyright (C) 2018 HPCC Systems®.
*
* 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.hpccsystems.spark;
import java.io.Serializable;
import java.util.List;
import java.util.Arrays;
import java.util.Iterator;
import java.util.regex.Pattern;
import java.math.BigDecimal;
import java.util.regex.Matcher;
import scala.reflect.ClassTag$;
import org.hpccsystems.commons.errors.HpccFileException;
import org.hpccsystems.dfs.client.CompressionAlgorithm;
import org.hpccsystems.dfs.client.DataPartition;
import org.hpccsystems.dfs.client.HPCCRemoteFileWriter;
import org.hpccsystems.dfs.cluster.RemapInfo;
import org.hpccsystems.dfs.cluster.NullRemapper;
import org.hpccsystems.commons.ecl.FieldDef;
import org.hpccsystems.commons.ecl.RecordDefinitionTranslator;
import org.hpccsystems.ws.client.HPCCWsDFUClient;
import org.hpccsystems.ws.client.utils.Connection;
import org.hpccsystems.ws.client.wrappers.ArrayOfEspExceptionWrapper;
import org.hpccsystems.ws.client.wrappers.wsdfu.DFUCreateFileWrapper;
import org.hpccsystems.ws.client.wrappers.wsdfu.DFUFilePartWrapper;
import org.hpccsystems.ws.client.wrappers.wsdfu.DFUFileTypeWrapper;
import io.opentelemetry.api.trace.Span;
import io.opentelemetry.api.trace.StatusCode;
import org.hpccsystems.spark.SparkSchemaTranslator;
import org.apache.logging.log4j.Logger;
import org.apache.logging.log4j.LogManager;
import org.apache.spark.api.java.function.Function2;
import org.apache.spark.SparkContext;
import org.apache.spark.api.java.JavaRDD;
import org.apache.spark.rdd.RDD;
import org.apache.spark.sql.Row;
import org.apache.spark.sql.types.*;
import org.apache.spark.sql.execution.python.EvaluatePython;
import net.razorvine.pickle.Unpickler;
/**
* A helper class that creates a job in Spark that writes a given RDD to HPCC Systems.
*/
public class HpccFileWriter implements Serializable
{
static private final long serialVersionUID = 1L;
static private final int DefaultExpiryTimeSecs = 300;
static private final Logger log = LogManager.getLogger(HpccFileWriter.class);
// Transient so Java serialization does not try to serialize this
private transient HPCCWsDFUClient dfuClient = null;
private transient Connection connectionInfo = null;
private String parentTraceID = "";
private String parentSpanID = "";
private static void registerPicklingFunctions()
{
EvaluatePython.registerPicklers();
Unpickler.registerConstructor("pyspark.sql.types", "Row", new RowConstructor());
Unpickler.registerConstructor("pyspark.sql.types", "_create_row", new RowConstructor());
Unpickler.registerConstructor("org.hpccsystems", "PySparkField", new PySparkFieldConstructor());
}
static
{
registerPicklingFunctions();
}
public HpccFileWriter(Connection espconninfo) throws HpccFileException
{
this.connectionInfo = espconninfo;
}
/**
* HpccFileWriter Constructor
* Attempts to open a connection to the specified HPCC cluster and validates the user.
* @param connectionString of format {http|https}://{HOST}:{PORT}. Host and port are the same as the ecl watch host and port.
* @param user a valid ecl watch account
* @param pass the password for the provided user
* @throws Exception general exception
*/
public HpccFileWriter(String connectionString, String user, String pass) throws Exception
{
// Make sure we setup opentelemetry before HPCC4j
Utils.getOpenTelemetry();
// Verify connection & password
final Pattern connectionRegex = Pattern.compile("(http|https)://([^:]+):([0-9]+)", Pattern.CASE_INSENSITIVE);
Matcher matches = connectionRegex.matcher(connectionString);
if (matches.find() == false)
{
throw new Exception("Invalid connection string. Expected format: {http|https}://{HOST}:{PORT}");
}
this.connectionInfo = new Connection(matches.group(1), matches.group(2), matches.group(3));
this.connectionInfo.setUserName(user);
this.connectionInfo.setPassword(pass);
}
/**
* Set the trace context for the current job
* @param parentTraceID
* @param parentSpanID
*/
public void setTraceContext(String parentTraceID, String parentSpanID)
{
this.parentTraceID = parentTraceID;
this.parentSpanID = parentSpanID;
}
private void abortFileCreation()
{
log.error("Abort file creation was called. This is currently a stub.");
}
private class FilePartWriteResults implements Serializable
{
static private final long serialVersionUID = 1L;
public long numRecords = 0;
public long dataLength = 0;
public boolean successful = true; // Default to true for empty partitions
public String errorMessage = null;
}
/**
* Saves the provided RDD to the specified file within the specified cluster. Will use HPCC default file compression.
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param scalaRDD The RDD to save to HPCC
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception wrapper
*/
public long saveToHPCC(RDD scalaRDD, String clusterName, String fileName) throws Exception, ArrayOfEspExceptionWrapper
{
JavaRDD rdd = JavaRDD.fromRDD(scalaRDD, ClassTag$.MODULE$.apply(Row.class));
return this.saveToHPCC(rdd, clusterName, fileName);
}
/**
* Saves the provided RDD to the specified file within the specified cluster. Will use HPCC default file compression.
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param schema The Schema of the provided RDD
* @param scalaRDD The RDD to save to HPCC
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception wrapper
*/
public long saveToHPCC(StructType schema, RDD scalaRDD, String clusterName, String fileName) throws Exception, ArrayOfEspExceptionWrapper
{
JavaRDD rdd = JavaRDD.fromRDD(scalaRDD, ClassTag$.MODULE$.apply(Row.class));
return this.saveToHPCC(schema, rdd, clusterName, fileName);
}
/**
* Saves the provided RDD to the specified file within the specified cluster. Will use HPCC default file compression.
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param javaRDD The RDD to save to HPCC
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception wrapper
*/
public long saveToHPCC(JavaRDD javaRDD, String clusterName, String fileName) throws Exception, ArrayOfEspExceptionWrapper
{
return this.saveToHPCC(SparkContext.getOrCreate(), null, javaRDD, clusterName, fileName, CompressionAlgorithm.DEFAULT, false);
}
/**
* Saves the provided RDD to the specified file within the specified cluster. Will use HPCC default file compression.
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param schema The Schema of the provided RDD
* @param javaRDD The RDD to save to HPCC
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception
*/
public long saveToHPCC(StructType schema, JavaRDD javaRDD, String clusterName, String fileName) throws Exception, ArrayOfEspExceptionWrapper
{
return this.saveToHPCC(SparkContext.getOrCreate(), schema, javaRDD, clusterName, fileName, CompressionAlgorithm.DEFAULT, false);
}
/**
* Saves the provided RDD to the specified file within the specified cluster
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param scalaRDD The RDD to save to HPCC
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @param fileCompression compression algorithm to use on files
* @param overwrite overwrite flag
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception wrapper
*/
public long saveToHPCC(RDD scalaRDD, String clusterName, String fileName, CompressionAlgorithm fileCompression, boolean overwrite)
throws Exception, ArrayOfEspExceptionWrapper
{
JavaRDD rdd = JavaRDD.fromRDD(scalaRDD, ClassTag$.MODULE$.apply(Row.class));
return this.saveToHPCC(SparkContext.getOrCreate(), null, rdd, clusterName, fileName, fileCompression, overwrite);
}
/**
* Saves the provided RDD to the specified file within the specified cluster
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param schema The Schema of the provided RDD
* @param scalaRDD The RDD to save to HPCC
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @param fileCompression compression algorithm to use on files
* @param overwrite overwrite flag
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception wrapper
*/
public long saveToHPCC(StructType schema, RDD scalaRDD, String clusterName, String fileName, CompressionAlgorithm fileCompression,
boolean overwrite) throws Exception, ArrayOfEspExceptionWrapper
{
JavaRDD rdd = JavaRDD.fromRDD(scalaRDD, ClassTag$.MODULE$.apply(Row.class));
return this.saveToHPCC(SparkContext.getOrCreate(), schema, rdd, clusterName, fileName, fileCompression, overwrite);
}
/**
* Saves the provided RDD to the specified file within the specified cluster
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param javaRDD The RDD to save to HPCC
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @param fileCompression compression algorithm to use on files
* @param overwrite overwrite flag
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception wrapper
*/
public long saveToHPCC(JavaRDD javaRDD, String clusterName, String fileName, CompressionAlgorithm fileCompression, boolean overwrite)
throws Exception, ArrayOfEspExceptionWrapper
{
return this.saveToHPCC(SparkContext.getOrCreate(), null, javaRDD, clusterName, fileName, fileCompression, overwrite);
}
/**
* Saves the provided RDD to the specified file within the specified cluster
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param schema The Schema of the provided RDD
* @param javaRDD The RDD to save to HPCC
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @param fileCompression compression algorithm to use on files
* @param overwrite overwrite flag
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception wrapper
*/
public long saveToHPCC(StructType schema, JavaRDD javaRDD, String clusterName, String fileName, CompressionAlgorithm fileCompression,
boolean overwrite) throws Exception, ArrayOfEspExceptionWrapper
{
return this.saveToHPCC(SparkContext.getOrCreate(), schema, javaRDD, clusterName, fileName, fileCompression, overwrite);
}
/**
* Saves the provided RDD to the specified file within the specified cluster. Will use HPCC default file compression.
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param sc The current SparkContext
* @param scalaRDD The RDD to save to HPCC
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception wrapper
*/
public long saveToHPCC(SparkContext sc, RDD scalaRDD, String clusterName, String fileName) throws Exception, ArrayOfEspExceptionWrapper
{
JavaRDD rdd = JavaRDD.fromRDD(scalaRDD, ClassTag$.MODULE$.apply(Row.class));
return saveToHPCC(sc, null, rdd, clusterName, fileName, CompressionAlgorithm.DEFAULT, false);
}
/**
* Saves the provided RDD to the specified file within the specified cluster. Will use HPCC default file compression.
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param sc The current SparkContext
* @param javaRDD The RDD to save to HPCC
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception wrapper
*/
public long saveToHPCC(SparkContext sc, JavaRDD javaRDD, String clusterName, String fileName) throws Exception, ArrayOfEspExceptionWrapper
{
return saveToHPCC(sc, null, javaRDD, clusterName, fileName, CompressionAlgorithm.DEFAULT, false);
}
/**
* Saves the provided RDD to the specified file within the specified cluster
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param sc The current SparkContext
* @param scalaRDD The RDD to save to HPCC
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @param fileCompression compression algorithm to use on files
* @param overwrite overwrite flag
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception wrapper
*/
public long saveToHPCC(SparkContext sc, RDD scalaRDD, String clusterName, String fileName, CompressionAlgorithm fileCompression,
boolean overwrite) throws Exception, ArrayOfEspExceptionWrapper
{
JavaRDD rdd = JavaRDD.fromRDD(scalaRDD, ClassTag$.MODULE$.apply(Row.class));
return this.saveToHPCC(sc, null, rdd, clusterName, fileName, fileCompression, overwrite);
}
/**
* Saves the provided RDD to the specified file within the specified cluster
* Note: PySpark datasets can be written to HPCC by first calling inferSchema to generate a valid Java Schema
* and converting the PySpark RDD to a JavaRDD via the _py2java() helper
* @param sc The current SparkContext
* @param rddSchema rdd schema
* @param rdd java rdd row
* @param clusterName The name of the cluster to save to.
* @param fileName The name of the logical file in HPCC to create. Follows HPCC file name conventions.
* @param fileCompression compression algorithm to use on files
* @param overwrite ovewrite flag
* @return Returns the number of records written
* @throws Exception general exception
* @throws ArrayOfEspExceptionWrapper array of esp exception wrapper
*/
public long saveToHPCC(SparkContext sc, StructType rddSchema, JavaRDD rdd, String clusterName, String fileName,
CompressionAlgorithm fileCompression, boolean overwrite) throws Exception, ArrayOfEspExceptionWrapper
{
this.dfuClient = HPCCWsDFUClient.get(this.connectionInfo);
if (sc == null || rdd == null)
{
throw new Exception("Aborting write. A valid non-null SparkContext and RDD must be provided.");
}
StructType schema = rddSchema;
if (schema == null)
{
Row firstRow = rdd.first();
schema = firstRow.schema();
}
FieldDef recordDef = SparkSchemaTranslator.toHPCCRecordDef(schema);
String eclRecordDefn = RecordDefinitionTranslator.toECLRecord(recordDef);
boolean isCompressed = fileCompression != CompressionAlgorithm.NONE;
Span createFileSpan = Utils.createChildSpan(parentTraceID, parentSpanID, "HpccFileWriter/CreateFile_" + fileName);
createFileSpan.setAttribute("cluster_name", clusterName);
createFileSpan.setAttribute("compressed", isCompressed);
createFileSpan.setAttribute("record_definition", eclRecordDefn);
createFileSpan.setStatus(StatusCode.OK);
DFUCreateFileWrapper createResult = null;
try
{
createResult = dfuClient.createFile(fileName, clusterName, eclRecordDefn, DefaultExpiryTimeSecs, isCompressed, DFUFileTypeWrapper.Flat, "");
}
catch (Exception e)
{
createFileSpan.recordException(e);
createFileSpan.setStatus(StatusCode.ERROR);
throw e;
}
finally
{
createFileSpan.end();
}
Span repartSpan = Utils.createChildSpan(parentTraceID, parentSpanID, "HpccFileWriter/Repartition_" + fileName);
repartSpan.setStatus(StatusCode.OK);
DFUFilePartWrapper[] dfuFileParts = createResult.getFileParts();
DataPartition[] hpccPartitions = DataPartition.createPartitions(dfuFileParts,
new NullRemapper(new RemapInfo(), createResult.getFileAccessInfo()), dfuFileParts.length, createResult.getFileAccessInfoBlob());
if (hpccPartitions.length != rdd.getNumPartitions())
{
rdd = rdd.repartition(hpccPartitions.length);
if (rdd.getNumPartitions() != hpccPartitions.length)
{
Exception wrappedException = new Exception("Repartitioning RDD failed. Aborting write.");
repartSpan.recordException(wrappedException);
repartSpan.setStatus(StatusCode.ERROR);
repartSpan.end();
throw wrappedException;
}
}
repartSpan.end();
//------------------------------------------------------------------------------
// Write partitions to file parts
//------------------------------------------------------------------------------
Function2, Iterator> writeFunc = (Integer partitionIndex, Iterator it) ->
{
HpccFileWriter.registerPicklingFunctions();
DataPartition thisPart = hpccPartitions[partitionIndex];
Span filePartWriteSpan = Utils.createChildSpan(parentTraceID, parentSpanID, "HpccFileWriter/WritePart_" + fileName + "_" + partitionIndex);
filePartWriteSpan.setStatus(StatusCode.OK);
HPCCRemoteFileWriter.FileWriteContext writeContext = new HPCCRemoteFileWriter.FileWriteContext();
writeContext.recordDef = recordDef;
writeContext.fileCompression = fileCompression;
writeContext.parentSpan = filePartWriteSpan;
GenericRowRecordAccessor recordAccessor = new GenericRowRecordAccessor(recordDef);
HPCCRemoteFileWriter fileWriter = new HPCCRemoteFileWriter(writeContext, thisPart, recordAccessor);
FilePartWriteResults result = new FilePartWriteResults();
try
{
fileWriter.writeRecords(it);
fileWriter.close();
result.dataLength = fileWriter.getBytesWritten();
result.numRecords = fileWriter.getRecordsWritten();
result.successful = true;
}
catch (Exception e)
{
result.successful = false;
result.errorMessage = e.getMessage();
filePartWriteSpan.recordException(e);
filePartWriteSpan.setStatus(StatusCode.ERROR);
}
finally
{
filePartWriteSpan.end();
}
List resultList = Arrays.asList(result);
return resultList.iterator();
};
// Create Write Job
Span issueWriteSpan = Utils.createChildSpan(parentTraceID, parentSpanID, "HpccFileWriter/WriteDataset_" + fileName);
issueWriteSpan.setStatus(StatusCode.OK);
JavaRDD writeResultsRDD = rdd.mapPartitionsWithIndex(writeFunc, true);
List writeResultsList = writeResultsRDD.collect();
long recordsWritten = 0;
long dataWritten = 0;
for (int i = 0; i < writeResultsList.size(); i++)
{
FilePartWriteResults result = writeResultsList.get(i);
recordsWritten += result.numRecords;
dataWritten += result.dataLength;
if (result.successful == false)
{
abortFileCreation();
Exception wrappedException = new Exception("Writing failed with error: " + result.errorMessage);
issueWriteSpan.recordException(wrappedException);
issueWriteSpan.setStatus(StatusCode.ERROR);
issueWriteSpan.end();
throw wrappedException;
}
}
issueWriteSpan.end();
//------------------------------------------------------------------------------
// Publish and finalize the temp file
//------------------------------------------------------------------------------
Span publishFileSpan = Utils.createChildSpan(parentTraceID, parentSpanID, "HpccFileWriter/PublishFile_" + fileName);
publishFileSpan.setAttribute("records_written", recordsWritten);
publishFileSpan.setAttribute("data_written", dataWritten);
publishFileSpan.setAttribute("overwrite", overwrite);
publishFileSpan.setStatus(StatusCode.OK);
try
{
dfuClient.publishFile(createResult.getFileID(), eclRecordDefn, recordsWritten, dataWritten, overwrite);
}
catch (Exception e)
{
Exception wrappedException = new Exception("Failed to publish file with error: " + e.getMessage());
publishFileSpan.recordException(wrappedException);
publishFileSpan.setStatus(StatusCode.ERROR);
throw wrappedException;
}
finally
{
publishFileSpan.end();
}
return recordsWritten;
}
/**
* Generates an inferred schema based on an example Map of FieldNames to Example Field Objects.
* This function is targeted primary at helping PySpark users write datasets back to HPCC.
* @param exampleFields list of python spark fields
* @return Returns a valid Spark schema based on the example rowDictionary
* @throws Exception general exception
*/
public StructType inferSchema(List exampleFields) throws Exception
{
return generateRowSchema(exampleFields);
}
private StructType generateRowSchema(List exampleFields) throws Exception
{
StructField[] fields = new StructField[exampleFields.size()];
int index = 0;
for (PySparkField fieldInfo : exampleFields)
{
fields[index] = generateSchemaField(fieldInfo.getName(), fieldInfo.getValue());
index++;
}
return new StructType(fields);
}
private StructField generateSchemaField(String name, Object obj) throws Exception
{
Metadata empty = Metadata.empty();
boolean nullable = false;
DataType type = DataTypes.NullType;
if (obj instanceof String)
{
type = DataTypes.StringType;
}
else if (obj instanceof Byte)
{
type = DataTypes.ByteType;
}
else if (obj instanceof Short)
{
type = DataTypes.ShortType;
}
else if (obj instanceof Integer)
{
type = DataTypes.IntegerType;
}
else if (obj instanceof Long)
{
type = DataTypes.LongType;
}
else if (obj instanceof byte[])
{
type = DataTypes.BinaryType;
}
else if (obj instanceof Boolean)
{
type = DataTypes.BooleanType;
}
else if (obj instanceof Float)
{
type = DataTypes.FloatType;
}
else if (obj instanceof Double)
{
type = DataTypes.DoubleType;
}
else if (obj instanceof BigDecimal)
{
BigDecimal decimal = (BigDecimal) obj;
int precision = decimal.precision();
int scale = decimal.scale();
// Spark SQL only supports 38 digits in decimal values
if (precision > DecimalType.MAX_PRECISION())
{
scale -= (precision - DecimalType.MAX_PRECISION());
if (scale < 0)
{
scale = 0;
}
precision = DecimalType.MAX_PRECISION();
}
type = DataTypes.createDecimalType(precision, scale);
}
else if (obj instanceof List)
{
List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy