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;

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

	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());
		}
	}
}