Commit d7af73cc authored by Jean-Baptiste Queru's avatar Jean-Baptiste Queru

Merge commit 'remotes/korg/cupcake' into merge

parents 35be73bf 82155ace
// Copyright 2007 The Android Open Source Project // Copyright 2007 Google Inc.
// All Rights Reserved. // All Rights Reserved.
package com.google.common.io.protocol; package com.google.common.io.protocol;
...@@ -9,7 +9,6 @@ import java.util.*; ...@@ -9,7 +9,6 @@ import java.util.*;
* This class can be used to create a memory model of a .proto file. Currently, * This class can be used to create a memory model of a .proto file. Currently,
* it is assumed that tags ids are not large. This could be improved by storing * it is assumed that tags ids are not large. This could be improved by storing
* a start offset, relaxing the assumption to a dense number space. * a start offset, relaxing the assumption to a dense number space.
*
*/ */
public class ProtoBufType { public class ProtoBufType {
// Note: Values 0..15 are reserved for wire types! // Note: Values 0..15 are reserved for wire types!
...@@ -121,4 +120,51 @@ public class ProtoBufType { ...@@ -121,4 +120,51 @@ public class ProtoBufType {
public String toString() { public String toString() {
return typeName; return typeName;
} }
/**
* {@inheritDoc}
* <p>Two ProtoBufTypes are equals if the fields types are the same.
*/
public boolean equals(Object object) {
if (null == object) {
// trivial check
return false;
} else if (this == object) {
// trivial check
return true;
} else if (this.getClass() != object.getClass()) {
// different class
return false;
}
ProtoBufType other = (ProtoBufType) object;
return stringEquals(types, other.types);
}
/**
* {@inheritDoc}
*/
public int hashCode() {
if (types != null) {
return types.hashCode();
} else {
return super.hashCode();
}
}
public static boolean stringEquals(CharSequence a, CharSequence b) {
if (a == b) return true;
int length;
if (a != null && b != null && (length = a.length()) == b.length()) {
if (a instanceof String && b instanceof String) {
return a.equals(b);
} else {
for (int i = 0; i < length; i++) {
if (a.charAt(i) != b.charAt(i)) return false;
}
return true;
}
}
return false;
}
} }
// Copyright 2008 The Android Open Source Project // Copyright 2008 Google Inc. All Rights Reserved.
package com.google.common.io.protocol; package com.google.common.io.protocol;
import java.io.*;
/** /**
* Utility functions for dealing with ProtoBuf objects consolidated from * Utility functions for dealing with ProtoBuf objects consolidated from
* previous spot implementations across the codebase. * previous spot implementations across the codebase.
...@@ -24,14 +26,39 @@ public final class ProtoBufUtil { ...@@ -24,14 +26,39 @@ public final class ProtoBufUtil {
public static String getSubProtoValueOrEmpty( public static String getSubProtoValueOrEmpty(
ProtoBuf proto, int sub, int tag) { ProtoBuf proto, int sub, int tag) {
try { try {
ProtoBuf subProto = return getProtoValueOrEmpty(getSubProtoOrNull(proto, sub), tag);
(proto != null && proto.has(sub)) ? proto.getProtoBuf(sub) : null;
return getProtoValueOrEmpty(subProto, tag);
} catch (ClassCastException e) { } catch (ClassCastException e) {
return ""; return "";
} }
} }
/** Convenience method to get a subproto if the proto has it. */
public static ProtoBuf getSubProtoOrNull(ProtoBuf proto, int sub) {
return (proto != null && proto.has(sub)) ? proto.getProtoBuf(sub) : null;
}
/**
* Get an int with "tag" from the proto buffer. If the given field can't be
* retrieved, return the provided default value.
*
* @param proto The proto buffer.
* @param tag The tag value that identifies which protocol buffer field to
* retrieve.
* @param defaultValue The value to return if the field can't be retrieved.
* @return The result which should be an integer.
*/
public static int getProtoValueOrDefault(ProtoBuf proto, int tag,
int defaultValue) {
try {
return (proto != null && proto.has(tag))
? proto.getInt(tag) : defaultValue;
} catch (IllegalArgumentException e) {
return defaultValue;
} catch (ClassCastException e) {
return defaultValue;
}
}
/** /**
* Get an Int with "tag" from the proto buffer. * Get an Int with "tag" from the proto buffer.
* If the given field can't be retrieved, return 0. * If the given field can't be retrieved, return 0.
...@@ -42,12 +69,25 @@ public final class ProtoBufUtil { ...@@ -42,12 +69,25 @@ public final class ProtoBufUtil {
* @return The result which should be an integer. * @return The result which should be an integer.
*/ */
public static int getProtoValueOrZero(ProtoBuf proto, int tag) { public static int getProtoValueOrZero(ProtoBuf proto, int tag) {
return getProtoValueOrDefault(proto, tag, 0);
}
/**
* Get an Long with "tag" from the proto buffer.
* If the given field can't be retrieved, return 0.
*
* @param proto The proto buffer.
* @param tag The tag value that identifies which protocol buffer field to
* retrieve.
* @return The result which should be an integer.
*/
public static long getProtoLongValueOrZero(ProtoBuf proto, int tag) {
try { try {
return (proto != null && proto.has(tag)) ? proto.getInt(tag) : 0; return (proto != null && proto.has(tag)) ? proto.getLong(tag) : 0L;
} catch (IllegalArgumentException e) { } catch (IllegalArgumentException e) {
return 0; return 0L;
} catch (ClassCastException e) { } catch (ClassCastException e) {
return 0; return 0L;
} }
} }
...@@ -70,6 +110,39 @@ public final class ProtoBufUtil { ...@@ -70,6 +110,39 @@ public final class ProtoBufUtil {
} }
} }
/**
* Reads a single protocol buffer from the given input stream. This method is
* provided where the client needs incremental access to the contents of a
* protocol buffer which contains a sequence of protocol buffers.
* <p />
* Please use {@link #getInputStreamForProtoBufResponse} to obtain an input
* stream suitable for this method.
*
* @param umbrellaType the type of the "outer" protocol buffer containing
* the message to read
* @param is the stream to read the protocol buffer from
* @param result the result protocol buffer (must be empty, will be filled
* with the data read and the type will be set)
* @return the tag id of the message, -1 at the end of the stream
*/
public static int readNextProtoBuf(ProtoBufType umbrellaType,
InputStream is, ProtoBuf result) throws IOException {
long tagAndType = ProtoBuf.readVarInt(is, true /* permits EOF */);
if (tagAndType == -1) {
return -1;
}
if ((tagAndType & 7) != ProtoBuf.WIRETYPE_LENGTH_DELIMITED) {
throw new IOException("Message expected");
}
int tag = (int) (tagAndType >>> 3);
result.setType((ProtoBufType) umbrellaType.getData(tag));
int length = (int) ProtoBuf.readVarInt(is, false);
result.parse(is, length);
return tag;
}
/** /**
* A wrapper for <code> getProtoValueOrNegativeOne </code> that drills into * A wrapper for <code> getProtoValueOrNegativeOne </code> that drills into
* a sub message returning the long value if it exists, returning -1 if it * a sub message returning the long value if it exists, returning -1 if it
...@@ -85,13 +158,80 @@ public final class ProtoBufUtil { ...@@ -85,13 +158,80 @@ public final class ProtoBufUtil {
public static long getSubProtoValueOrNegativeOne( public static long getSubProtoValueOrNegativeOne(
ProtoBuf proto, int sub, int tag) { ProtoBuf proto, int sub, int tag) {
try { try {
ProtoBuf subProto = return getProtoValueOrNegativeOne(getSubProtoOrNull(proto, sub), tag);
(proto != null && proto.has(sub)) ? proto.getProtoBuf(sub) : null;
return getProtoValueOrNegativeOne(subProto, tag);
} catch (IllegalArgumentException e) { } catch (IllegalArgumentException e) {
return -1; return -1;
} catch (ClassCastException e) { } catch (ClassCastException e) {
return -1; return -1;
} }
} }
/**
* A wrapper for {@link #getProtoValueOrDefault(ProtoBuf, int, int)} that
* drills into a sub message returning the int value if it exists, returning
* the given default if it does not.
*
* @param proto The proto buffer.
* @param tag The tag value that identifies which protocol buffer field to
* retrieve.
* @param sub The sub tag value that identifies which protocol buffer
* sub-field to retrieve.
* @param defaultValue The value to return if the field is not present.
* @return The result which should be a long.
*/
public static int getSubProtoValueOrDefault(ProtoBuf proto, int sub, int tag,
int defaultValue) {
try {
return getProtoValueOrDefault(getSubProtoOrNull(proto, sub), tag,
defaultValue);
} catch (IllegalArgumentException e) {
return defaultValue;
} catch (ClassCastException e) {
return defaultValue;
}
}
/**
* Creates a sub ProtoBuf of the given Protobuf and sets it.
*
* @param proto The proto buffer.
* @param tag The tag value that identifies which protocol buffer field to
* create.
* @return the sub ProtoBuf generated.
*/
public static ProtoBuf createProtoBuf(ProtoBuf proto, int tag) {
ProtoBuf child = proto.createGroup(tag);
proto.setProtoBuf(tag, child);
return child;
}
/**
* Creates a sub ProtoBuf of the given Protobuf and adds it.
*
* @param proto The proto buffer.
* @param tag The tag value that identifies which protocol buffer field to
* add.
* @return the sub ProtoBuf generated.
*/
public static ProtoBuf addProtoBuf(ProtoBuf proto, int tag) {
ProtoBuf child = proto.createGroup(tag);
proto.addProtoBuf(tag, child);
return child;
}
/**
* Writes the ProtoBuf to the given DataOutput. This is useful for unit
* tests.
*
* @param output The data output to write to.
* @param protoBuf The proto buffer.
*/
public static void writeProtoBufToOutput(DataOutput output, ProtoBuf protoBuf)
throws IOException {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
protoBuf.outputTo(baos);
byte[] bytes = baos.toByteArray();
output.writeInt(bytes.length);
output.write(bytes);
}
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment