2010-10-28 18:57:58 +02:00
|
|
|
// Protocol Buffers - Google's data interchange format
|
|
|
|
// Copyright 2008 Google Inc. All rights reserved.
|
|
|
|
// http://code.google.com/p/protobuf/
|
|
|
|
//
|
|
|
|
// Redistribution and use in source and binary forms, with or without
|
|
|
|
// modification, are permitted provided that the following conditions are
|
|
|
|
// met:
|
|
|
|
//
|
|
|
|
// * Redistributions of source code must retain the above copyright
|
|
|
|
// notice, this list of conditions and the following disclaimer.
|
|
|
|
// * Redistributions in binary form must reproduce the above
|
|
|
|
// copyright notice, this list of conditions and the following disclaimer
|
|
|
|
// in the documentation and/or other materials provided with the
|
|
|
|
// distribution.
|
|
|
|
// * Neither the name of Google Inc. nor the names of its
|
|
|
|
// contributors may be used to endorse or promote products derived from
|
|
|
|
// this software without specific prior written permission.
|
|
|
|
//
|
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
package com.google.protobuf;
|
|
|
|
|
|
|
|
import java.io.FilterInputStream;
|
|
|
|
import java.io.InputStream;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.OutputStream;
|
|
|
|
import java.util.Collection;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A partial implementation of the {@link MessageLite} interface which
|
|
|
|
* implements as many methods of that interface as possible in terms of other
|
|
|
|
* methods.
|
|
|
|
*
|
|
|
|
* @author kenton@google.com Kenton Varda
|
|
|
|
*/
|
|
|
|
public abstract class AbstractMessageLite implements MessageLite {
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public ByteString toByteString() {
|
2010-10-28 18:57:58 +02:00
|
|
|
try {
|
|
|
|
final ByteString.CodedBuilder out =
|
|
|
|
ByteString.newCodedBuilder(getSerializedSize());
|
|
|
|
writeTo(out.getCodedOutput());
|
|
|
|
return out.build();
|
|
|
|
} catch (IOException e) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Serializing to a ByteString threw an IOException (should " +
|
|
|
|
"never happen).", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public byte[] toByteArray() {
|
2010-10-28 18:57:58 +02:00
|
|
|
try {
|
|
|
|
final byte[] result = new byte[getSerializedSize()];
|
|
|
|
final CodedOutputStream output = CodedOutputStream.newInstance(result);
|
|
|
|
writeTo(output);
|
|
|
|
output.checkNoSpaceLeft();
|
|
|
|
return result;
|
|
|
|
} catch (IOException e) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Serializing to a byte array threw an IOException " +
|
|
|
|
"(should never happen).", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public void writeTo(final OutputStream output) throws IOException {
|
2010-10-28 18:57:58 +02:00
|
|
|
final int bufferSize =
|
|
|
|
CodedOutputStream.computePreferredBufferSize(getSerializedSize());
|
|
|
|
final CodedOutputStream codedOutput =
|
|
|
|
CodedOutputStream.newInstance(output, bufferSize);
|
|
|
|
writeTo(codedOutput);
|
|
|
|
codedOutput.flush();
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public void writeDelimitedTo(final OutputStream output) throws IOException {
|
2010-10-28 18:57:58 +02:00
|
|
|
final int serialized = getSerializedSize();
|
|
|
|
final int bufferSize = CodedOutputStream.computePreferredBufferSize(
|
|
|
|
CodedOutputStream.computeRawVarint32Size(serialized) + serialized);
|
|
|
|
final CodedOutputStream codedOutput =
|
|
|
|
CodedOutputStream.newInstance(output, bufferSize);
|
|
|
|
codedOutput.writeRawVarint32(serialized);
|
|
|
|
writeTo(codedOutput);
|
|
|
|
codedOutput.flush();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A partial implementation of the {@link Message.Builder} interface which
|
|
|
|
* implements as many methods of that interface as possible in terms of
|
|
|
|
* other methods.
|
|
|
|
*/
|
|
|
|
@SuppressWarnings("unchecked")
|
|
|
|
public static abstract class Builder<BuilderType extends Builder>
|
|
|
|
implements MessageLite.Builder {
|
|
|
|
// The compiler produces an error if this is not declared explicitly.
|
|
|
|
@Override
|
|
|
|
public abstract BuilderType clone();
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public BuilderType mergeFrom(final CodedInputStream input)
|
2010-10-28 18:57:58 +02:00
|
|
|
throws IOException {
|
|
|
|
return mergeFrom(input, ExtensionRegistryLite.getEmptyRegistry());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Re-defined here for return type covariance.
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public abstract BuilderType mergeFrom(
|
2010-10-28 18:57:58 +02:00
|
|
|
final CodedInputStream input,
|
|
|
|
final ExtensionRegistryLite extensionRegistry)
|
|
|
|
throws IOException;
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public BuilderType mergeFrom(final ByteString data)
|
2010-10-28 18:57:58 +02:00
|
|
|
throws InvalidProtocolBufferException {
|
|
|
|
try {
|
|
|
|
final CodedInputStream input = data.newCodedInput();
|
|
|
|
mergeFrom(input);
|
|
|
|
input.checkLastTagWas(0);
|
|
|
|
return (BuilderType) this;
|
|
|
|
} catch (InvalidProtocolBufferException e) {
|
|
|
|
throw e;
|
|
|
|
} catch (IOException e) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Reading from a ByteString threw an IOException (should " +
|
|
|
|
"never happen).", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public BuilderType mergeFrom(
|
2010-10-28 18:57:58 +02:00
|
|
|
final ByteString data,
|
|
|
|
final ExtensionRegistryLite extensionRegistry)
|
|
|
|
throws InvalidProtocolBufferException {
|
|
|
|
try {
|
|
|
|
final CodedInputStream input = data.newCodedInput();
|
|
|
|
mergeFrom(input, extensionRegistry);
|
|
|
|
input.checkLastTagWas(0);
|
|
|
|
return (BuilderType) this;
|
|
|
|
} catch (InvalidProtocolBufferException e) {
|
|
|
|
throw e;
|
|
|
|
} catch (IOException e) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Reading from a ByteString threw an IOException (should " +
|
|
|
|
"never happen).", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public BuilderType mergeFrom(final byte[] data)
|
2010-10-28 18:57:58 +02:00
|
|
|
throws InvalidProtocolBufferException {
|
|
|
|
return mergeFrom(data, 0, data.length);
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public BuilderType mergeFrom(final byte[] data, final int off,
|
2010-10-28 18:57:58 +02:00
|
|
|
final int len)
|
|
|
|
throws InvalidProtocolBufferException {
|
|
|
|
try {
|
|
|
|
final CodedInputStream input =
|
|
|
|
CodedInputStream.newInstance(data, off, len);
|
|
|
|
mergeFrom(input);
|
|
|
|
input.checkLastTagWas(0);
|
|
|
|
return (BuilderType) this;
|
|
|
|
} catch (InvalidProtocolBufferException e) {
|
|
|
|
throw e;
|
|
|
|
} catch (IOException e) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Reading from a byte array threw an IOException (should " +
|
|
|
|
"never happen).", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public BuilderType mergeFrom(
|
2010-10-28 18:57:58 +02:00
|
|
|
final byte[] data,
|
|
|
|
final ExtensionRegistryLite extensionRegistry)
|
|
|
|
throws InvalidProtocolBufferException {
|
|
|
|
return mergeFrom(data, 0, data.length, extensionRegistry);
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public BuilderType mergeFrom(
|
2010-10-28 18:57:58 +02:00
|
|
|
final byte[] data, final int off, final int len,
|
|
|
|
final ExtensionRegistryLite extensionRegistry)
|
|
|
|
throws InvalidProtocolBufferException {
|
|
|
|
try {
|
|
|
|
final CodedInputStream input =
|
|
|
|
CodedInputStream.newInstance(data, off, len);
|
|
|
|
mergeFrom(input, extensionRegistry);
|
|
|
|
input.checkLastTagWas(0);
|
|
|
|
return (BuilderType) this;
|
|
|
|
} catch (InvalidProtocolBufferException e) {
|
|
|
|
throw e;
|
|
|
|
} catch (IOException e) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Reading from a byte array threw an IOException (should " +
|
|
|
|
"never happen).", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public BuilderType mergeFrom(final InputStream input) throws IOException {
|
2010-10-28 18:57:58 +02:00
|
|
|
final CodedInputStream codedInput = CodedInputStream.newInstance(input);
|
|
|
|
mergeFrom(codedInput);
|
|
|
|
codedInput.checkLastTagWas(0);
|
|
|
|
return (BuilderType) this;
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public BuilderType mergeFrom(
|
2010-10-28 18:57:58 +02:00
|
|
|
final InputStream input,
|
|
|
|
final ExtensionRegistryLite extensionRegistry)
|
|
|
|
throws IOException {
|
|
|
|
final CodedInputStream codedInput = CodedInputStream.newInstance(input);
|
|
|
|
mergeFrom(codedInput, extensionRegistry);
|
|
|
|
codedInput.checkLastTagWas(0);
|
|
|
|
return (BuilderType) this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* An InputStream implementations which reads from some other InputStream
|
|
|
|
* but is limited to a particular number of bytes. Used by
|
|
|
|
* mergeDelimitedFrom(). This is intentionally package-private so that
|
|
|
|
* UnknownFieldSet can share it.
|
|
|
|
*/
|
|
|
|
static final class LimitedInputStream extends FilterInputStream {
|
|
|
|
private int limit;
|
|
|
|
|
|
|
|
LimitedInputStream(InputStream in, int limit) {
|
|
|
|
super(in);
|
|
|
|
this.limit = limit;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public int available() throws IOException {
|
|
|
|
return Math.min(super.available(), limit);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public int read() throws IOException {
|
|
|
|
if (limit <= 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
final int result = super.read();
|
|
|
|
if (result >= 0) {
|
|
|
|
--limit;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public int read(final byte[] b, final int off, int len)
|
|
|
|
throws IOException {
|
|
|
|
if (limit <= 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
len = Math.min(len, limit);
|
|
|
|
final int result = super.read(b, off, len);
|
|
|
|
if (result >= 0) {
|
|
|
|
limit -= result;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public long skip(final long n) throws IOException {
|
|
|
|
final long result = super.skip(Math.min(n, limit));
|
|
|
|
if (result >= 0) {
|
|
|
|
limit -= result;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public boolean mergeDelimitedFrom(
|
2010-10-28 18:57:58 +02:00
|
|
|
final InputStream input,
|
|
|
|
final ExtensionRegistryLite extensionRegistry)
|
|
|
|
throws IOException {
|
|
|
|
final int firstByte = input.read();
|
|
|
|
if (firstByte == -1) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
final int size = CodedInputStream.readRawVarint32(firstByte, input);
|
|
|
|
final InputStream limitedInput = new LimitedInputStream(input, size);
|
|
|
|
mergeFrom(limitedInput, extensionRegistry);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-10-10 16:10:34 +02:00
|
|
|
@Override
|
|
|
|
public boolean mergeDelimitedFrom(final InputStream input)
|
2010-10-28 18:57:58 +02:00
|
|
|
throws IOException {
|
|
|
|
return mergeDelimitedFrom(input,
|
|
|
|
ExtensionRegistryLite.getEmptyRegistry());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Construct an UninitializedMessageException reporting missing fields in
|
|
|
|
* the given message.
|
|
|
|
*/
|
|
|
|
protected static UninitializedMessageException
|
|
|
|
newUninitializedMessageException(MessageLite message) {
|
|
|
|
return new UninitializedMessageException(message);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds the {@code values} to the {@code list}. This is a helper method
|
|
|
|
* used by generated code. Users should ignore it.
|
|
|
|
*
|
|
|
|
* @throws NullPointerException if any of the elements of {@code values} is
|
|
|
|
* null.
|
|
|
|
*/
|
|
|
|
protected static <T> void addAll(final Iterable<T> values,
|
|
|
|
final Collection<? super T> list) {
|
|
|
|
for (final T value : values) {
|
|
|
|
if (value == null) {
|
|
|
|
throw new NullPointerException();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (values instanceof Collection) {
|
|
|
|
final
|
|
|
|
Collection<T> collection = (Collection<T>) values;
|
|
|
|
list.addAll(collection);
|
|
|
|
} else {
|
|
|
|
for (final T value : values) {
|
|
|
|
list.add(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|