All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.hudi.avro.AvroSchemaUtils Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.apache.hudi.avro;

import org.apache.hudi.common.util.Option;
import org.apache.hudi.common.util.StringUtils;
import org.apache.hudi.exception.HoodieAvroSchemaException;
import org.apache.hudi.exception.InvalidUnionTypeException;
import org.apache.hudi.exception.MissingSchemaFieldException;
import org.apache.hudi.exception.SchemaBackwardsCompatibilityException;
import org.apache.hudi.exception.SchemaCompatibilityException;

import org.apache.avro.Schema;
import org.apache.avro.SchemaCompatibility;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Deque;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.function.BiFunction;
import java.util.stream.Collectors;

import static org.apache.hudi.common.util.ValidationUtils.checkState;

/**
 * Utils for Avro Schema.
 */
public class AvroSchemaUtils {

  private static final Logger LOG = LoggerFactory.getLogger(AvroSchemaUtils.class);

  private AvroSchemaUtils() {
  }

  /**
   * See {@link #isSchemaCompatible(Schema, Schema, boolean, boolean)} doc for more details
   */
  public static boolean isSchemaCompatible(Schema prevSchema, Schema newSchema) {
    return isSchemaCompatible(prevSchema, newSchema, true);
  }

  /**
   * See {@link #isSchemaCompatible(Schema, Schema, boolean, boolean)} doc for more details
   */
  public static boolean isSchemaCompatible(Schema prevSchema, Schema newSchema, boolean allowProjection) {
    return isSchemaCompatible(prevSchema, newSchema, true, allowProjection);
  }

  /**
   * Establishes whether {@code newSchema} is compatible w/ {@code prevSchema}, as
   * defined by Avro's {@link AvroSchemaCompatibility}.
   * From avro's compatibility standpoint, prevSchema is writer schema and new schema is reader schema.
   * {@code newSchema} is considered compatible to {@code prevSchema}, iff data written using {@code prevSchema}
   * could be read by {@code newSchema}
   *
   * @param prevSchema previous instance of the schema
   * @param newSchema new instance of the schema
   * @param checkNaming controls whether schemas fully-qualified names should be checked
   */
  public static boolean isSchemaCompatible(Schema prevSchema, Schema newSchema, boolean checkNaming, boolean allowProjection) {
    // NOTE: We're establishing compatibility of the {@code prevSchema} and {@code newSchema}
    //       as following: {@code newSchema} is considered compatible to {@code prevSchema},
    //       iff data written using {@code prevSchema} could be read by {@code newSchema}

    // In case schema projection is not allowed, new schema has to have all the same fields as the
    // old schema
    if (!allowProjection) {
      if (!canProject(prevSchema, newSchema)) {
        return false;
      }
    }

    AvroSchemaCompatibility.SchemaPairCompatibility result =
        AvroSchemaCompatibility.checkReaderWriterCompatibility(newSchema, prevSchema, checkNaming);
    return result.getType() == AvroSchemaCompatibility.SchemaCompatibilityType.COMPATIBLE;
  }

  /**
   * Check that each field in the prevSchema can be populated in the newSchema
   * @param prevSchema prev schema.
   * @param newSchema new schema
   * @return true if prev schema is a projection of new schema.
   */
  public static boolean canProject(Schema prevSchema, Schema newSchema) {
    return findMissingFields(prevSchema, newSchema, Collections.emptySet()).isEmpty();
  }

  /**
   * Check that each top level field in the prevSchema can be populated in the newSchema except specified columns
   * @param prevSchema prev schema.
   * @param newSchema new schema
   * @return List of fields that should be in the new schema
   */
  private static List findMissingFields(Schema prevSchema, Schema newSchema, Set exceptCols) {
    return prevSchema.getFields().stream()
        .filter(f -> !exceptCols.contains(f.name()))
        .filter(oldSchemaField -> SchemaCompatibility.lookupWriterField(newSchema, oldSchemaField) == null)
        .collect(Collectors.toList());
  }

  /**
   * Generates fully-qualified name for the Avro's schema based on the Table's name
   *
   * NOTE: PLEASE READ CAREFULLY BEFORE CHANGING
   *       This method should not change for compatibility reasons as older versions
   *       of Avro might be comparing fully-qualified names rather than just the record
   *       names
   */
  public static String getAvroRecordQualifiedName(String tableName) {
    String sanitizedTableName = HoodieAvroUtils.sanitizeName(tableName);
    return "hoodie." + sanitizedTableName + "." + sanitizedTableName + "_record";
  }

  /**
   * Validate whether the {@code targetSchema} is a "compatible" projection of {@code sourceSchema}.
   * Only difference of this method from {@link #isStrictProjectionOf(Schema, Schema)} is
   * the fact that it allows some legitimate type promotions (like {@code int -> long},
   * {@code decimal(3, 2) -> decimal(5, 2)}, etc.) that allows projection to have a "wider"
   * atomic type (whereas strict projection requires atomic type to be identical)
   */
  public static boolean isCompatibleProjectionOf(Schema sourceSchema, Schema targetSchema) {
    return isProjectionOfInternal(sourceSchema, targetSchema,
        AvroSchemaUtils::isAtomicSchemasCompatible);
  }

  private static boolean isAtomicSchemasCompatible(Schema oneAtomicType, Schema anotherAtomicType) {
    // NOTE: Checking for compatibility of atomic types, we should ignore their
    //       corresponding fully-qualified names (as irrelevant)
    return isSchemaCompatible(oneAtomicType, anotherAtomicType, false, true);
  }

  /**
   * Validate whether the {@code targetSchema} is a strict projection of {@code sourceSchema}.
   *
   * Schema B is considered a strict projection of schema A iff
   * 
    *
  1. Schemas A and B are equal, or
  2. *
  3. Schemas A and B are array schemas and element-type of B is a strict projection * of the element-type of A, or
  4. *
  5. Schemas A and B are map schemas and value-type of B is a strict projection * of the value-type of A, or
  6. *
  7. Schemas A and B are union schemas (of the same size) and every element-type of B * is a strict projection of the corresponding element-type of A, or
  8. *
  9. Schemas A and B are record schemas and every field of the record B has corresponding * counterpart (w/ the same name) in the schema A, such that the schema of the field of the schema * B is also a strict projection of the A field's schema
  10. *
*/ public static boolean isStrictProjectionOf(Schema sourceSchema, Schema targetSchema) { return isProjectionOfInternal(sourceSchema, targetSchema, Objects::equals); } private static boolean isProjectionOfInternal(Schema sourceSchema, Schema targetSchema, BiFunction atomicTypeEqualityPredicate) { if (sourceSchema.getType() == targetSchema.getType()) { if (sourceSchema.getType() == Schema.Type.RECORD) { for (Schema.Field targetField : targetSchema.getFields()) { Schema.Field sourceField = sourceSchema.getField(targetField.name()); if (sourceField == null || !isProjectionOfInternal(sourceField.schema(), targetField.schema(), atomicTypeEqualityPredicate)) { return false; } } return true; } else if (sourceSchema.getType() == Schema.Type.ARRAY) { return isProjectionOfInternal(sourceSchema.getElementType(), targetSchema.getElementType(), atomicTypeEqualityPredicate); } else if (sourceSchema.getType() == Schema.Type.MAP) { return isProjectionOfInternal(sourceSchema.getValueType(), targetSchema.getValueType(), atomicTypeEqualityPredicate); } else if (sourceSchema.getType() == Schema.Type.UNION) { List sourceNestedSchemas = sourceSchema.getTypes(); List targetNestedSchemas = targetSchema.getTypes(); if (sourceNestedSchemas.size() != targetNestedSchemas.size()) { return false; } for (int i = 0; i < sourceNestedSchemas.size(); ++i) { if (!isProjectionOfInternal(sourceNestedSchemas.get(i), targetNestedSchemas.get(i), atomicTypeEqualityPredicate)) { return false; } } return true; } } return atomicTypeEqualityPredicate.apply(sourceSchema, targetSchema); } public static Option findNestedFieldType(Schema schema, String fieldName) { if (StringUtils.isNullOrEmpty(fieldName)) { return Option.empty(); } String[] parts = fieldName.split("\\."); for (String part : parts) { Schema.Field foundField = resolveNullableSchema(schema).getField(part); if (foundField == null) { throw new HoodieAvroSchemaException(fieldName + " not a field in " + schema); } schema = foundField.schema(); } return Option.of(resolveNullableSchema(schema).getType()); } /** * Get gets a field from a record, works on nested fields as well (if you provide the whole name, eg: toplevel.nextlevel.child) * @return the field, including its lineage. * For example, if you have a schema: record(a:int, b:record(x:int, y:long, z:record(z1: int, z2: float, z3: double), c:bool) * "fieldName" | output * --------------------------------- * "a" | a:int * "b" | b:record(x:int, y:long, z:record(z1: int, z2: int, z3: int) * "c" | c:bool * "b.x" | b:record(x:int) * "b.z.z2" | b:record(z:record(z2:float)) * * this is intended to be used with appendFieldsToSchemaDedupNested */ public static Option findNestedField(Schema schema, String fieldName) { return findNestedField(schema, fieldName.split("\\."), 0); } private static Option findNestedField(Schema schema, String[] fieldParts, int index) { if (schema.getType().equals(Schema.Type.UNION)) { Option notUnion = findNestedField(resolveNullableSchema(schema), fieldParts, index); if (!notUnion.isPresent()) { return Option.empty(); } Schema.Field nu = notUnion.get(); return Option.of(new Schema.Field(nu.name(), nu.schema(), nu.doc(), nu.defaultVal())); } if (fieldParts.length <= index) { return Option.empty(); } Schema.Field foundField = schema.getField(fieldParts[index]); if (foundField == null) { return Option.empty(); } if (index == fieldParts.length - 1) { return Option.of(new Schema.Field(foundField.name(), foundField.schema(), foundField.doc(), foundField.defaultVal())); } Schema foundSchema = foundField.schema(); Option nestedPart = findNestedField(foundSchema, fieldParts, index + 1); if (!nestedPart.isPresent()) { return Option.empty(); } boolean isUnion = false; if (foundSchema.getType().equals(Schema.Type.UNION)) { isUnion = true; foundSchema = resolveNullableSchema(foundSchema); } Schema newSchema = createNewSchemaFromFieldsWithReference(foundSchema, Collections.singletonList(nestedPart.get())); return Option.of(new Schema.Field(foundField.name(), isUnion ? createNullableSchema(newSchema) : newSchema, foundField.doc(), foundField.defaultVal())); } /** * Adds newFields to the schema. Will add nested fields without duplicating the field * For example if your schema is "a.b.{c,e}" and newfields contains "a.{b.{d,e},x.y}", * It will stitch them together to be "a.{b.{c,d,e},x.y} */ public static Schema appendFieldsToSchemaDedupNested(Schema schema, List newFields) { return appendFieldsToSchemaBase(schema, newFields, true); } public static Schema mergeSchemas(Schema a, Schema b) { if (!a.getType().equals(Schema.Type.RECORD)) { return a; } List fields = new ArrayList<>(); for (Schema.Field f : a.getFields()) { Schema.Field foundField = b.getField(f.name()); fields.add(new Schema.Field(f.name(), foundField == null ? f.schema() : mergeSchemas(f.schema(), foundField.schema()), f.doc(), f.defaultVal())); } for (Schema.Field f : b.getFields()) { if (a.getField(f.name()) == null) { fields.add(new Schema.Field(f.name(), f.schema(), f.doc(), f.defaultVal())); } } return createNewSchemaFromFieldsWithReference(a, fields); } /** * Appends provided new fields at the end of the given schema * * NOTE: No deduplication is made, this method simply appends fields at the end of the list * of the source schema as is */ public static Schema appendFieldsToSchema(Schema schema, List newFields) { return appendFieldsToSchemaBase(schema, newFields, false); } private static Schema appendFieldsToSchemaBase(Schema schema, List newFields, boolean dedupNested) { List fields = schema.getFields().stream() .map(field -> new Schema.Field(field.name(), field.schema(), field.doc(), field.defaultVal())) .collect(Collectors.toList()); if (dedupNested) { for (Schema.Field f : newFields) { Schema.Field foundField = schema.getField(f.name()); if (foundField != null) { fields.set(foundField.pos(), new Schema.Field(foundField.name(), mergeSchemas(foundField.schema(), f.schema()), foundField.doc(), foundField.defaultVal())); } else { fields.add(f); } } } else { fields.addAll(newFields); } return createNewSchemaFromFieldsWithReference(schema, fields); } /** * Create a new schema but maintain all meta info from the old schema * * @param schema schema to get the meta info from * @param fields list of fields in order that will be in the new schema * * @return schema with fields from fields, and metadata from schema */ public static Schema createNewSchemaFromFieldsWithReference(Schema schema, List fields) { if (schema == null) { throw new IllegalArgumentException("Schema must not be null"); } Schema newSchema = Schema.createRecord(schema.getName(), schema.getDoc(), schema.getNamespace(), schema.isError()); Map schemaProps = Collections.emptyMap(); try { schemaProps = schema.getObjectProps(); } catch (Exception e) { LOG.warn("Error while getting object properties from schema: {}", schema, e); } for (Map.Entry prop : schemaProps.entrySet()) { newSchema.addProp(prop.getKey(), prop.getValue()); } newSchema.setFields(fields); return newSchema; } /** * Passed in {@code Union} schema and will try to resolve the field with the {@code fieldSchemaFullName} * w/in the union returning its corresponding schema * * @param schema target schema to be inspected * @param fieldSchemaFullName target field-name to be looked up w/in the union * @return schema of the field w/in the union identified by the {@code fieldSchemaFullName} */ public static Schema resolveUnionSchema(Schema schema, String fieldSchemaFullName) { if (schema.getType() != Schema.Type.UNION) { return schema; } List innerTypes = schema.getTypes(); if (innerTypes.size() == 2 && isNullable(schema)) { // this is a basic nullable field so handle it more efficiently return resolveNullableSchema(schema); } Schema nonNullType = innerTypes.stream() .filter(it -> it.getType() != Schema.Type.NULL && Objects.equals(it.getFullName(), fieldSchemaFullName)) .findFirst() .orElse(null); if (nonNullType == null) { throw new HoodieAvroSchemaException( String.format("Unsupported Avro UNION type %s: Only UNION of a null type and a non-null type is supported", schema)); } return nonNullType; } /** * Returns true in case provided {@link Schema} is nullable (ie accepting null values), * returns false otherwise */ public static boolean isNullable(Schema schema) { if (schema.getType() != Schema.Type.UNION) { return false; } List innerTypes = schema.getTypes(); return innerTypes.size() > 1 && innerTypes.stream().anyMatch(it -> it.getType() == Schema.Type.NULL); } /** * Resolves typical Avro's nullable schema definition: {@code Union(Schema.Type.NULL, )}, * decomposing union and returning the target non-null type */ public static Schema resolveNullableSchema(Schema schema) { if (schema.getType() != Schema.Type.UNION) { return schema; } List innerTypes = schema.getTypes(); if (innerTypes.size() != 2) { throw new HoodieAvroSchemaException( String.format("Unsupported Avro UNION type %s: Only UNION of a null type and a non-null type is supported", schema)); } Schema firstInnerType = innerTypes.get(0); Schema secondInnerType = innerTypes.get(1); if ((firstInnerType.getType() != Schema.Type.NULL && secondInnerType.getType() != Schema.Type.NULL) || (firstInnerType.getType() == Schema.Type.NULL && secondInnerType.getType() == Schema.Type.NULL)) { throw new HoodieAvroSchemaException( String.format("Unsupported Avro UNION type %s: Only UNION of a null type and a non-null type is supported", schema)); } return firstInnerType.getType() == Schema.Type.NULL ? secondInnerType : firstInnerType; } /** * Creates schema following Avro's typical nullable schema definition: {@code Union(Schema.Type.NULL, )}, * wrapping around provided target non-null type */ public static Schema createNullableSchema(Schema.Type avroType) { return createNullableSchema(Schema.create(avroType)); } public static Schema createNullableSchema(Schema schema) { checkState(schema.getType() != Schema.Type.NULL); return Schema.createUnion(Schema.create(Schema.Type.NULL), schema); } /** * Returns true in case when schema contains the field w/ provided name */ public static boolean containsFieldInSchema(Schema schema, String fieldName) { try { Schema.Field field = schema.getField(fieldName); return field != null; } catch (Exception e) { return false; } } /** * Checks whether writer schema is compatible with table schema considering {@code AVRO_SCHEMA_VALIDATE_ENABLE} * and {@code SCHEMA_ALLOW_AUTO_EVOLUTION_COLUMN_DROP} options. * To avoid collision of {@code SCHEMA_ALLOW_AUTO_EVOLUTION_COLUMN_DROP} and {@code DROP_PARTITION_COLUMNS} * partition column names should be passed as {@code dropPartitionColNames}. * Passed empty set means {@code DROP_PARTITION_COLUMNS} is disabled. * * @param tableSchema the latest dataset schema * @param writerSchema writer schema * @param shouldValidate whether {@link AvroSchemaCompatibility} check being performed * @param allowProjection whether column dropping check being performed * @param dropPartitionColNames partition column names to being excluded from column dropping check * @throws SchemaCompatibilityException if writer schema is not compatible */ public static void checkSchemaCompatible( Schema tableSchema, Schema writerSchema, boolean shouldValidate, boolean allowProjection, Set dropPartitionColNames) throws SchemaCompatibilityException { if (!allowProjection) { List missingFields = findMissingFields(tableSchema, writerSchema, dropPartitionColNames); if (!missingFields.isEmpty()) { throw new MissingSchemaFieldException(missingFields.stream().map(Schema.Field::name).collect(Collectors.toList()), writerSchema, tableSchema); } } // TODO(HUDI-4772) re-enable validations in case partition columns // being dropped from the data-file after fixing the write schema if (dropPartitionColNames.isEmpty() && shouldValidate) { AvroSchemaCompatibility.SchemaPairCompatibility result = AvroSchemaCompatibility.checkReaderWriterCompatibility(writerSchema, tableSchema, true); if (result.getType() != AvroSchemaCompatibility.SchemaCompatibilityType.COMPATIBLE) { throw new SchemaBackwardsCompatibilityException(result, writerSchema, tableSchema); } } } /** * Validate whether the {@code incomingSchema} is a valid evolution of {@code tableSchema}. * * @param incomingSchema schema of the incoming dataset * @param tableSchema latest table schema */ public static void checkValidEvolution(Schema incomingSchema, Schema tableSchema) { if (incomingSchema.getType() == Schema.Type.NULL) { return; } //not really needed for `hoodie.write.set.null.for.missing.columns` but good to check anyway List missingFields = new ArrayList<>(); findAnyMissingFields(incomingSchema, tableSchema, new ArrayDeque<>(), missingFields); if (!missingFields.isEmpty()) { throw new MissingSchemaFieldException(missingFields, incomingSchema, tableSchema); } //make sure that the table schema can be read using the incoming schema AvroSchemaCompatibility.SchemaPairCompatibility result = AvroSchemaCompatibility.checkReaderWriterCompatibility(incomingSchema, tableSchema, false); if (result.getType() != AvroSchemaCompatibility.SchemaCompatibilityType.COMPATIBLE) { throw new SchemaBackwardsCompatibilityException(result, incomingSchema, tableSchema); } } /** * Find all fields in the latest table schema that are not in * the incoming schema. */ private static void findAnyMissingFields(Schema incomingSchema, Schema latestTableSchema, Deque visited, List missingFields) { findAnyMissingFieldsRec(incomingSchema, latestTableSchema, visited, missingFields, incomingSchema, latestTableSchema); } /** * We want to pass the full schemas so that the error message has the entire schema to print from */ private static void findAnyMissingFieldsRec(Schema incomingSchema, Schema latestTableSchema, Deque visited, List missingFields, Schema fullIncomingSchema, Schema fullTableSchema) { if (incomingSchema.getType() == latestTableSchema.getType()) { if (incomingSchema.getType() == Schema.Type.RECORD) { visited.addLast(latestTableSchema.getName()); for (Schema.Field targetField : latestTableSchema.getFields()) { visited.addLast(targetField.name()); Schema.Field sourceField = incomingSchema.getField(targetField.name()); if (sourceField == null) { missingFields.add(String.join(".", visited)); } else { findAnyMissingFieldsRec(sourceField.schema(), targetField.schema(), visited, missingFields, fullIncomingSchema, fullTableSchema); } visited.removeLast(); } visited.removeLast(); } else if (incomingSchema.getType() == Schema.Type.ARRAY) { visited.addLast("element"); findAnyMissingFieldsRec(incomingSchema.getElementType(), latestTableSchema.getElementType(), visited, missingFields, fullIncomingSchema, fullTableSchema); visited.removeLast(); } else if (incomingSchema.getType() == Schema.Type.MAP) { visited.addLast("value"); findAnyMissingFieldsRec(incomingSchema.getValueType(), latestTableSchema.getValueType(), visited, missingFields, fullIncomingSchema, fullTableSchema); visited.removeLast(); } else if (incomingSchema.getType() == Schema.Type.UNION) { List incomingNestedSchemas = incomingSchema.getTypes(); List latestTableNestedSchemas = latestTableSchema.getTypes(); if (incomingNestedSchemas.size() != latestTableNestedSchemas.size()) { throw new InvalidUnionTypeException(createSchemaErrorString( String.format("Incoming batch field '%s' has union with %d types, while the table schema has %d types", String.join(".", visited), incomingNestedSchemas.size(), latestTableNestedSchemas.size()), fullIncomingSchema, fullTableSchema)); } if (incomingNestedSchemas.size() > 2) { throw new InvalidUnionTypeException(createSchemaErrorString( String.format("Union for incoming batch field '%s' should not have more than 2 types but has %d", String.join(".", visited), incomingNestedSchemas.size()), fullIncomingSchema, fullTableSchema)); } for (int i = 0; i < incomingNestedSchemas.size(); ++i) { findAnyMissingFieldsRec(incomingNestedSchemas.get(i), latestTableNestedSchemas.get(i), visited, missingFields, fullIncomingSchema, fullTableSchema); } } } } public static String createSchemaErrorString(String errorMessage, Schema writerSchema, Schema tableSchema) { return String.format("%s\nwriterSchema: %s\ntableSchema: %s", errorMessage, writerSchema, tableSchema); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy