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

org.wildfly.clustering.marshalling.protostream.DefaultProtoStreamReader Maven / Gradle / Ivy

/*
 * Copyright The WildFly Authors
 * SPDX-License-Identifier: Apache-2.0
 */

package org.wildfly.clustering.marshalling.protostream;

import java.io.IOException;
import java.io.InputStream;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.EnumSet;
import java.util.IdentityHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.infinispan.protostream.ProtobufTagMarshaller.ReadContext;
import org.infinispan.protostream.TagReader;
import org.infinispan.protostream.descriptors.WireType;

/**
 * {@link ProtoStreamWriter} implementation that reads from a {@link TagReader}.
 * @author Paul Ferraro
 */
public class DefaultProtoStreamReader extends AbstractProtoStreamOperation implements ProtoStreamReader, ReadContext {

    interface ProtoStreamReaderContext extends ProtoStreamOperation.Context {
        /**
         * Resolves an object from the specified reference.
         * @param reference an object reference
         * @return the resolved object
         */
        Object resolve(Reference reference);
    }

    private final TagReader reader;
    private final ProtoStreamReaderContext context;

    private int currentTag = 0;

    public DefaultProtoStreamReader(ReadContext context) {
        this(context, new DefaultProtoStreamReaderContext());
    }

    private DefaultProtoStreamReader(ReadContext context, ProtoStreamReaderContext readerContext) {
        super(context);
        this.reader = context.getReader();
        this.context = readerContext;
    }

    @Override
    public Context getContext() {
        return this.context;
    }

    @Override
    public TagReader getReader() {
        return this.reader;
    }

    @Override
    public  FieldSetReader createFieldSetReader(FieldReadable reader, int startIndex) {
        int endIndex = reader.nextIndex(startIndex);
        ProtoStreamReader offsetReader = new OffsetProtoStreamReader(this, startIndex);
        return new FieldSetReader<>() {
            @Override
            public T readField(T current) throws IOException {
                int tag = offsetReader.getCurrentTag();
                // Determine index relative to this field set
                int relativeIndex = WireType.getTagFieldNumber(tag) - startIndex;
                return reader.readFrom(offsetReader, relativeIndex, WireType.fromTag(tag), current);
            }

            @Override
            public boolean contains(int index) {
                return (index >= startIndex) && (index < endIndex);
            }
        };
    }

    @Override
    public int pushLimit(int limit) throws IOException {
        return this.reader.pushLimit(limit);
    }

    @Override
    public void popLimit(int oldLimit) {
        this.reader.popLimit(oldLimit);
    }

    @Override
    public int readTag() throws IOException {
        return this.currentTag = this.reader.readTag();
    }

    @Override
    public int getCurrentTag() {
        return this.currentTag;
    }

    @Override
    public void checkLastTagWas(int tag) throws IOException {
        this.reader.checkLastTagWas(tag);
    }

    @Override
    public boolean skipField(int tag) throws IOException {
        this.currentTag = 0;
        return this.reader.skipField(tag);
    }

    @Override
    public boolean isAtEnd() throws IOException {
        return this.reader.isAtEnd();
    }

    @Override
    public Object readAny() throws IOException {
        Object result = this.readObject(Any.class).get();
        if (result instanceof Reference) {
            Reference reference = (Reference) result;
            result = this.context.resolve(reference);
        } else {
            this.context.record(result);
        }
        return result;
    }

    @Override
    public  T readObject(Class targetClass) throws IOException {
        this.verifyWireType(WireType.LENGTH_DELIMITED);
        int limit = this.reader.readUInt32();
        int oldLimit = this.reader.pushLimit(limit);
        try {
            ProtoStreamMarshaller marshaller = this.findMarshaller(targetClass);
            T result = marshaller.readFrom(this);
            // Ensure marshaller reached limit
            this.reader.checkLastTagWas(0);
            return result;
        } finally {
            this.reader.popLimit(oldLimit);
        }
    }

    @Override
    public boolean readBool() throws IOException {
        this.verifyWireType(WireType.VARINT);
        this.currentTag = 0;
        return this.reader.readBool();
    }

    @Override
    public int readEnum() throws IOException {
        this.verifyWireType(WireType.VARINT);
        this.currentTag = 0;
        return this.reader.readEnum();
    }

    @Deprecated
    @Override
    public int readInt32() throws IOException {
        this.verifyWireType(WireType.VARINT);
        this.currentTag = 0;
        return this.reader.readInt32();
    }

    @Deprecated
    @Override
    public int readFixed32() throws IOException {
        this.verifyWireType(WireType.FIXED32);
        this.currentTag = 0;
        return this.reader.readFixed32();
    }

    @Override
    public int readUInt32() throws IOException {
        // Used with unsigned byte/short/int or length records
        this.verifyWireType(EnumSet.of(WireType.VARINT, WireType.LENGTH_DELIMITED));
        this.currentTag = 0;
        return this.reader.readUInt32();
    }

    @Override
    public int readSInt32() throws IOException {
        this.verifyWireType(WireType.VARINT);
        this.currentTag = 0;
        return this.reader.readSInt32();
    }

    @Override
    public int readSFixed32() throws IOException {
        this.verifyWireType(WireType.FIXED32);
        this.currentTag = 0;
        return this.reader.readSFixed32();
    }

    @Deprecated
    @Override
    public long readInt64() throws IOException {
        this.verifyWireType(WireType.VARINT);
        this.currentTag = 0;
        return this.reader.readInt64();
    }

    @Deprecated
    @Override
    public long readFixed64() throws IOException {
        this.verifyWireType(WireType.FIXED64);
        this.currentTag = 0;
        return this.reader.readFixed64();
    }

    @Override
    public long readUInt64() throws IOException {
        this.verifyWireType(WireType.VARINT);
        this.currentTag = 0;
        return this.reader.readUInt64();
    }

    @Override
    public long readSInt64() throws IOException {
        this.verifyWireType(WireType.VARINT);
        this.currentTag = 0;
        return this.reader.readSInt64();
    }

    @Override
    public long readSFixed64() throws IOException {
        this.verifyWireType(WireType.FIXED64);
        this.currentTag = 0;
        return this.reader.readSFixed64();
    }

    @Override
    public float readFloat() throws IOException {
        // Used with float and packed float arrays
        this.verifyWireType(EnumSet.of(WireType.FIXED32, WireType.VARINT));
        this.currentTag = 0;
        return this.reader.readFloat();
    }

    @Override
    public double readDouble() throws IOException {
        // Used with double and packed double arrays
        this.verifyWireType(EnumSet.of(WireType.FIXED64, WireType.VARINT));
        this.currentTag = 0;
        return this.reader.readDouble();
    }

    @Override
    public byte[] readByteArray() throws IOException {
        this.verifyWireType(WireType.LENGTH_DELIMITED);
        this.currentTag = 0;
        return this.reader.readByteArray();
    }

    @Override
    public ByteBuffer readByteBuffer() throws IOException {
        this.verifyWireType(WireType.LENGTH_DELIMITED);
        this.currentTag = 0;
        return this.reader.readByteBuffer();
    }

    @Override
    public String readString() throws IOException {
        this.verifyWireType(WireType.LENGTH_DELIMITED);
        this.currentTag = 0;
        return this.reader.readString();
    }

    @Override
    public byte[] fullBufferArray() throws IOException {
        this.verifyWireType(WireType.LENGTH_DELIMITED);
        return this.reader.fullBufferArray();
    }

    @Override
    public InputStream fullBufferInputStream() throws IOException {
        this.verifyWireType(WireType.LENGTH_DELIMITED);
        return this.reader.fullBufferInputStream();
    }

    @Override
    public boolean isInputStream() {
        return this.reader.isInputStream();
    }

    private void verifyWireType(WireType type) throws IOException {
        this.verifyWireType(EnumSet.of(type));
    }

    private void verifyWireType(Set types) throws IOException {
        WireType currentType = WireType.fromTag(this.currentTag);
        if (!types.contains(currentType)) {
            throw new IllegalStateException(currentType.name());
        }
    }

    private static class DefaultProtoStreamReaderContext implements ProtoStreamReaderContext {
        private final Map objects = new IdentityHashMap<>(128);
        private final List references = new ArrayList<>(128);

        @Override
        public void record(Object object) {
            if (object != null) {
                if (this.objects.putIfAbsent(object, Boolean.TRUE) == null) {
                    this.references.add(object);
                }
            }
        }

        @Override
        public Object resolve(Reference reference) {
            return this.references.get(reference.getAsInt());
        }
    }
}