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

com.fasterxml.jackson.dataformat.avro.AvroSchema Maven / Gradle / Ivy

package com.fasterxml.jackson.dataformat.avro;

import java.util.Set;
import java.util.concurrent.atomic.AtomicReference;

import org.apache.avro.Schema;
import org.apache.avro.Schema.Type;
import org.apache.avro.SchemaCompatibility;
import org.apache.avro.SchemaCompatibility.SchemaCompatibilityType;
import org.apache.avro.SchemaCompatibility.SchemaPairCompatibility;

import com.fasterxml.jackson.core.FormatSchema;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.JsonMappingException;
import com.fasterxml.jackson.dataformat.avro.deser.AvroReaderFactory;
import com.fasterxml.jackson.dataformat.avro.deser.AvroStructureReader;

/**
 * Wrapper for Schema information needed to encode and decode Avro-format
 * data.
 */
public class AvroSchema implements FormatSchema
{
    /**
     * Format type id used by Jackson components to refer to Avro Format.
     */
    public final static String TYPE_ID = "avro";

    /**
     * Schema that was used for writing the data to decode; for simple instance
     * used for reading as well (reader schema).
     */
    protected final Schema _writerSchema;

    /**
     * Lazily instantiated value reader for this schema.
     */
    protected final AtomicReference _reader = new AtomicReference();

    public AvroSchema(Schema asch)
    {
        _writerSchema = asch;
    }

    /**
     * Method that will consider this schema instance (used as so-called "Writer Schema"),
     * and specified "Reader Schema" instance, and will either construct a new schema
     * with appropriate translations, to use for reading (if reader and writer schemas are
     * not same); or, if schemas are the same, return `this`.
     *

* Note that neither `this` instance nor `readerSchema` is ever modified: if an altered * version is needed, a new schema object will be constructed. *

* NOTE: this is a relatively expensive operation due to validation (although significant * part of cost is deferred until the first call to {@link #getReader}) so it is recommended * that these instances are reused whenever possible. * * @param readerSchema "Reader Schema" to use (in Avro terms): schema that specified how * reader wants to see the data; specifies part of translation needed along with this * schema (which would be "Writer Schema" in Avro terms). * * @throws JsonProcessingException If given reader schema is incompatible with (writer-) * schema this instance was constructed with, * * @since 2.9 */ public AvroSchema withReaderSchema(AvroSchema readerSchema) throws JsonProcessingException { Schema w = _writerSchema; Schema r = readerSchema.getAvroSchema(); if (r.equals(w)) { return this; } // First: apply simple renamings: w = Schema.applyAliases(w, r); // and then use Avro std lib to validate compatibility // 16-Jun-2017, tatu: First, a very common case is for Record names not // to match; so let's check that first if (r.getType() == w.getType()) { if (!_schemaNamesEqual(w, r)) { throw new JsonMappingException(null, String.format( "Incompatible writer/reader schemas: root %ss have different names (\"%s\" vs \"%s\"), no match via aliases", r.getType().getName(), w.getFullName(), r.getFullName())); } } SchemaPairCompatibility comp; try { comp = SchemaCompatibility.checkReaderWriterCompatibility(r, w); } catch (Exception e) { throw new JsonMappingException(null, String.format( "Failed to resolve given writer/reader schemas, problem: (%s) %s", e.getClass().getName(), e.getMessage())); } if (comp.getType() != SchemaCompatibilityType.COMPATIBLE) { throw new JsonMappingException(null, String.format("Incompatible writer/reader schemas: %s", comp.getDescription())); } return Resolving.create(w, r); } private boolean _schemaNamesEqual(Schema w, Schema r) { final String wname = w.getFullName(); final String rname = r.getFullName(); if ((wname == rname) || ((wname != null) && wname.equals(rname))) { return true; } // but may also have alias. NOTE! Avro lib itself does this, and we rely // on it, but basically only `NamedSchema` do NOT throw exception. But // we have no way of checking -- need to trust other cases bail out before // this (which they do). Unclean but... that's avrolib for you. Set aliases = r.getAliases(); return aliases.contains(wname); } /** * Similar to {@link #withReaderSchema} but will NOT verify compatibility of schemas: * this means that certain problems (such as missing default value for a newly added * field) that would be caught at construction may be surfaced later when using * schema. This is, however, sometimes necessary to work around potential BUT NOT ACTUAL * problems. * * @since 2.9 */ public AvroSchema withUnsafeReaderSchema(AvroSchema readerSchema) throws JsonProcessingException { Schema w = _writerSchema; Schema r = readerSchema.getAvroSchema(); if (r.equals(w)) { return this; } w = Schema.applyAliases(w, r); return Resolving.create(w, r); } @Override public String getSchemaType() { return TYPE_ID; } /** * Accessor for "writer schema" contained in this instance. */ public Schema getAvroSchema() { return _writerSchema; } /* /********************************************************************** /* Factory methods /********************************************************************** */ public AvroStructureReader getReader() { AvroStructureReader r = _reader.get(); if (r == null) { r = _constructReader(); _reader.set(r); } return r; } protected AvroStructureReader _constructReader() { return AvroReaderFactory.createFor(_writerSchema); } /* /********************************************************************** /* Standard method overrides /********************************************************************** */ @Override public String toString() { return String.format("{AvroSchema: name=%s}", _writerSchema.getFullName()); } @Override public int hashCode() { return _writerSchema.hashCode(); } @Override public boolean equals(Object o) { if (o == this) return true; if ((o == null) || o.getClass() != getClass()) return false; AvroSchema other = (AvroSchema) o; return _writerSchema.equals(other._writerSchema); } /* /********************************************************************** /* Specialized sub-class(es), helper classes /********************************************************************** */ /** * Sub-class that does writer-to-reader conversion by using "resolving decoder" * (by avro codec) on top of binary codec, exposing content using (reader) schema * this instance was configured with. */ private final static class Resolving extends AvroSchema { private final Schema _readerSchema; public Resolving(Schema writer, Schema reader) { super(writer); _readerSchema = reader; } public static Resolving create(Schema writer, Schema reader) { return new Resolving(writer, reader); } @Override protected AvroStructureReader _constructReader() { return AvroReaderFactory.createFor(_writerSchema, _readerSchema); } /* /********************************************************************** /* Standard method overrides /********************************************************************** */ @Override public String toString() { return String.format("{AvroSchema.Resolving: name=%s}", _writerSchema.getFullName()); } @Override public int hashCode() { return super.hashCode() ^ _readerSchema.hashCode(); } @Override public boolean equals(Object o) { if (o == this) return true; if ((o == null) || o.getClass() != getClass()) return false; Resolving other = (Resolving) o; return _writerSchema.equals(other._writerSchema) && _readerSchema.equals(other._readerSchema); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy