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.io.*;
import java.lang.ref.SoftReference;
import java.util.concurrent.atomic.AtomicReference;

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

import org.apache.avro.Schema;
import org.apache.avro.io.BinaryDecoder;
import org.apache.avro.io.BinaryEncoder;
import org.apache.avro.io.DecoderFactory;
import org.apache.avro.io.EncoderFactory;

/**
 * Wrapper for Schema information needed to encode and decode Avro-format
 * data.
 */
public class AvroSchema implements FormatSchema
{
    public final static String TYPE_ID = "avro";

    protected final static DecoderFactory DECODER_FACTORY = DecoderFactory.get();

    protected final static EncoderFactory ENCODER_FACTORY = EncoderFactory.get();

    protected final static ThreadLocal> decoderRecycler
        = new ThreadLocal>();
    
    protected final static ThreadLocal> encoderRecycler
        = new ThreadLocal>();
    
    protected final Schema _avroSchema;

    protected final AtomicReference _reader = new AtomicReference();
    
    public AvroSchema(Schema asch)
    {
        _avroSchema = asch;
    }

    @Override
    public String getSchemaType() {
        return TYPE_ID;
    }

    public Schema getAvroSchema() { return _avroSchema; }

    public static BinaryDecoder decoder(InputStream in, boolean buffering)
    {
        SoftReference ref = decoderRecycler.get();
        BinaryDecoder prev = (ref == null) ? null : ref.get();
        /* Factory will check if the decoder has a matching type for reuse.
         * If not, it will drop the instance being reused and will return
         * a new, proper one.
         */
        BinaryDecoder next = buffering
            ? DECODER_FACTORY.binaryDecoder(in, prev)
            : DECODER_FACTORY.directBinaryDecoder(in, prev);

        decoderRecycler.set(new SoftReference(next));
        return next;
    }

    public static BinaryDecoder decoder(byte[] buffer, int offset, int len)
    {
        SoftReference ref = decoderRecycler.get();
        BinaryDecoder prev = (ref == null) ? null : ref.get();
        
        if (prev != null) {
            return DECODER_FACTORY.binaryDecoder(buffer, offset, len, prev);
        }
        prev = DECODER_FACTORY.binaryDecoder(buffer, offset, len, null);
        decoderRecycler.set(new SoftReference(prev));
        return prev;
    }
    
    
    public static BinaryEncoder encoder(OutputStream out, boolean buffering)
    {
        SoftReference ref = encoderRecycler.get();
        BinaryEncoder prev = (ref == null) ? null : ref.get();
        /* Factory will check if the encoder has a matching type for reuse.
         * If not, it will drop the instance being reused and will return
         * a new, proper one.
         */
        BinaryEncoder next =
            buffering
            ? ENCODER_FACTORY.binaryEncoder(out, prev)
            : ENCODER_FACTORY.directBinaryEncoder(out, prev);
        
        encoderRecycler.set(new SoftReference(next));

        return next;
    }

    public AvroStructureReader getReader()
    {
        AvroStructureReader r = _reader.get();
        if (r == null) {
            AvroReaderFactory f = new AvroReaderFactory();
            r = f.createReader(_avroSchema);
            _reader.set(r);
        }
        return r;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy