io.cdap.plugin.common.spark.SparkUtils Maven / Gradle / Ivy
/*
* Copyright © 2016-2019 Cask Data, Inc.
*
* 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 io.cdap.plugin.common.spark;
import com.google.common.base.Splitter;
import com.google.common.base.Strings;
import com.google.common.collect.Iterables;
import com.google.common.collect.Lists;
import io.cdap.cdap.api.data.format.StructuredRecord;
import io.cdap.cdap.api.data.schema.Schema;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.regex.Pattern;
import java.util.regex.PatternSyntaxException;
import javax.annotation.Nullable;
/**
* Spark plugin Utility class. Contains common code to be used in trainer and predictors.
*/
public final class SparkUtils {
private SparkUtils() {
}
/**
* Validate the config parameters for the spark sink and spark compute classes.
*
* @param inputSchema schema of the received record.
* @param featuresToInclude features to be used for training/prediction.
* @param featuresToExclude features to be excluded when training/predicting.
* @param predictionField field containing the prediction values.
*/
public static void validateConfigParameters(Schema inputSchema, @Nullable String featuresToInclude,
@Nullable String featuresToExclude, String predictionField,
@Nullable String cardinalityMapping) {
if (!Strings.isNullOrEmpty(featuresToExclude) && !Strings.isNullOrEmpty(featuresToInclude)) {
throw new IllegalArgumentException("Cannot specify values for both featuresToInclude and featuresToExclude. " +
"Please specify fields for one.");
}
Map fields = getFeatureList(inputSchema, featuresToInclude, featuresToExclude, predictionField);
for (String field : fields.keySet()) {
Schema.Field inputField = inputSchema.getField(field);
Schema schema = inputField.getSchema();
Schema.Type features = schema.isNullableSimple() ? schema.getNonNullable().getType() : schema.getType();
if (!(features.equals(Schema.Type.INT) || features.equals(Schema.Type.LONG) ||
features.equals(Schema.Type.FLOAT) || features.equals(Schema.Type.DOUBLE))) {
throw new IllegalArgumentException(String.format("Features must be of type : int, double, float, long but " +
"was of type %s for field %s.", features, field));
}
}
getCategoricalFeatureInfo(inputSchema, featuresToInclude, featuresToExclude, predictionField, cardinalityMapping);
}
/**
* Get the feature list of the features that have to be used for training/prediction depending on the
* featuresToInclude or featuresToInclude list.
*
* @param inputSchema schema of the received record.
* @param featuresToInclude features to be used for training/prediction.
* @param featuresToExclude features to be excluded when training/predicting.
* @param predictionField field containing the prediction values.
* @return feature list to be used for training/prediction.
*/
public static Map getFeatureList(Schema inputSchema, @Nullable String featuresToInclude,
@Nullable String featuresToExclude, String predictionField) {
if (!Strings.isNullOrEmpty(featuresToExclude) && !Strings.isNullOrEmpty(featuresToInclude)) {
throw new IllegalArgumentException("Cannot specify values for both featuresToInclude and featuresToExclude. " +
"Please specify fields for one.");
}
Map fields = new LinkedHashMap<>();
if (!Strings.isNullOrEmpty(featuresToInclude)) {
Iterable tokens = Splitter.on(',').trimResults().split(featuresToInclude);
String[] features = Iterables.toArray(tokens, String.class);
for (int i = 0; i < features.length; i++) {
String field = features[i];
Schema.Field inputField = inputSchema.getField(field);
if (!field.equals(predictionField) && inputField != null) {
fields.put(field, i);
}
}
return fields;
}
Set excludeFeatures = new HashSet<>();
if (!Strings.isNullOrEmpty(featuresToExclude)) {
excludeFeatures.addAll(Lists.newArrayList(Splitter.on(',').trimResults().split(featuresToExclude)));
}
Object[] inputSchemaFields = inputSchema.getFields().toArray();
for (int i = 0; i < inputSchemaFields.length; i++) {
String field = ((Schema.Field) inputSchemaFields[i]).getName();
if (!field.equals(predictionField) && !excludeFeatures.contains(field)) {
fields.put(field, i);
}
}
return fields;
}
/**
* Get the feature to cardinality mapping provided by the user.
*
* @param inputSchema schema of the received record.
* @param featuresToInclude features to be used for training/prediction.
* @param featuresToExclude features to be excluded when training/predicting.
* @param labelField field containing the prediction values.
* @param cardinalityMapping feature to cardinality mapping specified for categorical features.
* @return categoricalFeatureInfo for categorical features.
*/
public static Map getCategoricalFeatureInfo(Schema inputSchema,
@Nullable String featuresToInclude,
@Nullable String featuresToExclude,
String labelField,
@Nullable String cardinalityMapping) {
Map featureList = getFeatureList(inputSchema, featuresToInclude, featuresToExclude, labelField);
Map outputFieldMappings = new HashMap<>();
if (Strings.isNullOrEmpty(cardinalityMapping)) {
return outputFieldMappings;
}
try {
Map map = Splitter.on(',').trimResults().withKeyValueSeparator(":").split(cardinalityMapping);
for (Map.Entry field : map.entrySet()) {
String value = field.getValue();
try {
outputFieldMappings.put(featureList.get(field.getKey()), Integer.parseInt(value));
} catch (NumberFormatException e) {
throw new IllegalArgumentException(
String.format("Invalid cardinality %s. Please specify valid integer for cardinality.", value));
}
}
} catch (IllegalArgumentException e) {
throw new IllegalArgumentException(
String.format("Invalid categorical feature mapping. %s. Please make sure it is in the format " +
"'feature':'cardinality'.", e.getMessage()), e);
}
return outputFieldMappings;
}
/**
* Validate label field for trainer.
*
* @param inputSchema schema of the received record.
* @param labelField field from which to get the prediction.
*/
public static void validateLabelFieldForTrainer(Schema inputSchema, String labelField) {
Schema.Field prediction = inputSchema.getField(labelField);
if (prediction == null) {
throw new IllegalArgumentException(String.format("Label field %s does not exists in the input schema.",
labelField));
}
Schema predictionSchema = prediction.getSchema();
Schema.Type predictionFieldType = predictionSchema.isNullableSimple() ?
predictionSchema.getNonNullable().getType() : predictionSchema.getType();
if (predictionFieldType != Schema.Type.DOUBLE) {
throw new IllegalArgumentException(String.format("Label field must be of type Double, but was %s.",
predictionFieldType));
}
}
/**
* Creates a builder based off the given record. The record will be cloned without the prediction field.
*/
public static StructuredRecord.Builder cloneRecord(StructuredRecord record, Schema outputSchema,
String predictionField) {
List fields = new ArrayList<>(outputSchema.getFields());
fields.add(Schema.Field.of(predictionField, Schema.of(Schema.Type.DOUBLE)));
outputSchema = Schema.recordOf("records", fields);
StructuredRecord.Builder builder = StructuredRecord.builder(outputSchema);
for (Schema.Field field : outputSchema.getFields()) {
if (!predictionField.equals(field.getName())) {
builder.set(field.getName(), record.get(field.getName()));
}
}
return builder;
}
public static Schema getOutputSchema(Schema inputSchema, String predictionField) {
List fields = new ArrayList<>(inputSchema.getFields());
fields.add(Schema.Field.of(predictionField, Schema.of(Schema.Type.DOUBLE)));
return Schema.recordOf(inputSchema.getRecordName() + ".predicted", fields);
}
/**
* Validate config parameters for feature generator classes.
*
* @param inputSchema input Schema
* @param map Map of the input fields to map to the transformed output fields.
*/
public static void validateFeatureGeneratorConfig(Schema inputSchema, Map map, String pattern) {
try {
Pattern.compile(pattern);
} catch (PatternSyntaxException e) {
throw new IllegalArgumentException(String.format("Invalid expression - %s. Please provide a valid pattern for " +
"splitting the string. %s.", pattern, e.getMessage()), e);
}
for (Map.Entry entry : map.entrySet()) {
String key = entry.getKey();
String value = entry.getValue();
validateTextField(inputSchema, key);
if (value == null) {
throw new IllegalArgumentException(String.format("Output column name not specified for column : %s. " +
"Please make sure it is in the format 'input-column':" +
"'transformed-output-column'.", entry.getKey()));
} else if (inputSchema.getField(value) != null) {
throw new IllegalArgumentException(
String.format("Output column name %s for column %s is already present in the input schema. Please " +
"provide a different output field name.", value, key));
}
}
}
/**
* Validate the input field to be used for text based feature generation.
*
* @param inputSchema input schema coming in from the previous stage
* @param key text field on which to perform text based feature generation
*/
public static void validateTextField(Schema inputSchema, String key) {
if (inputSchema.getField(key) == null) {
throw new IllegalArgumentException(String.format("Input field %s does not exist in the input schema %s.",
key, inputSchema.toString()));
}
Schema schema = inputSchema.getField(key).getSchema();
Schema.Type type = schema.isNullable() ? schema.getNonNullable().getType() : schema.getType();
if (type == Schema.Type.ARRAY) {
Schema componentSchema = schema.getComponentSchema();
type = componentSchema.isNullable() ? componentSchema.getNonNullable().getType() : componentSchema.getType();
}
if (type != Schema.Type.STRING) {
throw new IllegalArgumentException(String.format("Field to be transformed should be of type String or " +
"Nullable String or Array of type String or Nullable " +
"String . But was %s for field %s.", type, key));
}
}
/**
* Gets the input field for feature generation. If the field is an array, returns the field. Otherwise, splits the
* field based on the given pattern.
* Returns an empty list if value is null.
* The method assumes the input field is an array of strings or a string.
* Validation for the field type should be performed at configure time. The method will throw an exception if the
* input field is not of required type.
*
* @param input input Structured Record
* @param inputField field to use for feature generation
* @param splitter Splitter object to be used for splitting the input string
* @return text to be used for feature generation
*/
public static List getInputFieldValue(StructuredRecord input, String inputField, Splitter splitter) {
List text = new ArrayList<>();
Schema schema = input.getSchema().getField(inputField).getSchema();
Schema.Type type = schema.isNullable() ? schema.getNonNullable().getType() : schema.getType();
try {
if (type == Schema.Type.ARRAY) {
Object value = input.get(inputField);
if (value instanceof List) {
text = input.get(inputField);
} else {
text = Lists.newArrayList((String[]) value);
}
} else {
String value = input.get(inputField);
if (value != null) {
text = Lists.newArrayList(splitter.split(value));
}
}
} catch (ClassCastException e) {
throw new IllegalArgumentException(String.format("Schema type mismatch for field %s. Please make sure the " +
"value to be used for feature generation is an array of " +
"string or a string.", inputField), e);
}
return text;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy