Protocol Buffers - Google's data interchange format (grpc依赖)
https://developers.google.com/protocol-buffers/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
320 lines
12 KiB
320 lines
12 KiB
// Protocol Buffers - Google's data interchange format |
|
// Copyright 2008 Google Inc. All rights reserved. |
|
// |
|
// Use of this source code is governed by a BSD-style |
|
// license that can be found in the LICENSE file or at |
|
// https://developers.google.com/open-source/licenses/bsd |
|
|
|
#import "GPBCodedInputStream.h" |
|
#import "GPBMessage_PackagePrivate.h" |
|
#import "GPBTestUtilities.h" |
|
#import "GPBUnknownField.h" |
|
#import "GPBUnknownField_PackagePrivate.h" |
|
#import "GPBUnknownFields.h" |
|
#import "GPBWireFormat.h" |
|
#import "objectivec/Tests/Unittest.pbobjc.h" |
|
#import "objectivec/Tests/UnittestMset.pbobjc.h" |
|
|
|
@interface WireFormatTests : GPBTestCase |
|
@end |
|
|
|
@implementation WireFormatTests |
|
|
|
- (void)testSerialization { |
|
TestAllTypes* message = [self allSetRepeatedCount:kGPBDefaultRepeatCount]; |
|
|
|
NSData* rawBytes = message.data; |
|
[self assertFieldsInOrder:rawBytes]; |
|
XCTAssertEqual(message.serializedSize, (size_t)rawBytes.length); |
|
|
|
TestAllTypes* message2 = [TestAllTypes parseFromData:rawBytes error:NULL]; |
|
|
|
[self assertAllFieldsSet:message2 repeatedCount:kGPBDefaultRepeatCount]; |
|
} |
|
|
|
- (void)testSerializationPacked { |
|
TestPackedTypes* message = [self packedSetRepeatedCount:kGPBDefaultRepeatCount]; |
|
|
|
NSData* rawBytes = message.data; |
|
[self assertFieldsInOrder:rawBytes]; |
|
XCTAssertEqual(message.serializedSize, (size_t)rawBytes.length); |
|
|
|
TestPackedTypes* message2 = [TestPackedTypes parseFromData:rawBytes error:NULL]; |
|
|
|
[self assertPackedFieldsSet:message2 repeatedCount:kGPBDefaultRepeatCount]; |
|
} |
|
|
|
- (void)testSerializeExtensions { |
|
// TestAllTypes and TestAllExtensions should have compatible wire formats, |
|
// so if we serealize a TestAllExtensions then parse it as TestAllTypes |
|
// it should work. |
|
|
|
TestAllExtensions* message = [self allExtensionsSetRepeatedCount:kGPBDefaultRepeatCount]; |
|
NSData* rawBytes = message.data; |
|
[self assertFieldsInOrder:rawBytes]; |
|
XCTAssertEqual(message.serializedSize, (size_t)rawBytes.length); |
|
|
|
TestAllTypes* message2 = [TestAllTypes parseFromData:rawBytes error:NULL]; |
|
|
|
[self assertAllFieldsSet:message2 repeatedCount:kGPBDefaultRepeatCount]; |
|
} |
|
|
|
- (void)testSerializePackedExtensions { |
|
// TestPackedTypes and TestPackedExtensions should have compatible wire |
|
// formats; check that they serialize to the same string. |
|
TestPackedExtensions* message = [self packedExtensionsSetRepeatedCount:kGPBDefaultRepeatCount]; |
|
NSData* rawBytes = message.data; |
|
[self assertFieldsInOrder:rawBytes]; |
|
|
|
TestPackedTypes* message2 = [self packedSetRepeatedCount:kGPBDefaultRepeatCount]; |
|
NSData* rawBytes2 = message2.data; |
|
|
|
XCTAssertEqualObjects(rawBytes, rawBytes2); |
|
} |
|
|
|
- (void)testParseExtensions { |
|
// TestAllTypes and TestAllExtensions should have compatible wire formats, |
|
// so if we serialize a TestAllTypes then parse it as TestAllExtensions |
|
// it should work. |
|
|
|
TestAllTypes* message = [self allSetRepeatedCount:kGPBDefaultRepeatCount]; |
|
NSData* rawBytes = message.data; |
|
[self assertFieldsInOrder:rawBytes]; |
|
|
|
GPBExtensionRegistry* registry = [self extensionRegistry]; |
|
|
|
TestAllExtensions* message2 = [TestAllExtensions parseFromData:rawBytes |
|
extensionRegistry:registry |
|
error:NULL]; |
|
|
|
[self assertAllExtensionsSet:message2 repeatedCount:kGPBDefaultRepeatCount]; |
|
} |
|
|
|
- (void)testExtensionsSerializedSize { |
|
size_t allSet = [self allSetRepeatedCount:kGPBDefaultRepeatCount].serializedSize; |
|
size_t extensionSet = [self allExtensionsSetRepeatedCount:kGPBDefaultRepeatCount].serializedSize; |
|
XCTAssertEqual(allSet, extensionSet); |
|
} |
|
|
|
- (void)testParsePackedExtensions { |
|
// Ensure that packed extensions can be properly parsed. |
|
TestPackedExtensions* message = [self packedExtensionsSetRepeatedCount:kGPBDefaultRepeatCount]; |
|
NSData* rawBytes = message.data; |
|
[self assertFieldsInOrder:rawBytes]; |
|
|
|
GPBExtensionRegistry* registry = [self extensionRegistry]; |
|
|
|
TestPackedExtensions* message2 = [TestPackedExtensions parseFromData:rawBytes |
|
extensionRegistry:registry |
|
error:NULL]; |
|
|
|
[self assertPackedExtensionsSet:message2 repeatedCount:kGPBDefaultRepeatCount]; |
|
} |
|
|
|
const int kUnknownTypeId = 1550055; |
|
const int kUnknownTypeId2 = 1550056; |
|
|
|
- (void)testSerializeMessageSet { |
|
// Set up a MSetMessage with two known messages and an unknown one. |
|
MSetMessage* message_set = [MSetMessage message]; |
|
[[message_set getExtension:[MSetMessageExtension1 messageSetExtension]] setI:123]; |
|
[[message_set getExtension:[MSetMessageExtension2 messageSetExtension]] setStr:@"foo"]; |
|
|
|
GPBUnknownField* unknownField = |
|
[[[GPBUnknownField alloc] initWithNumber:kUnknownTypeId] autorelease]; |
|
[unknownField addLengthDelimited:DataFromCStr("bar")]; |
|
GPBUnknownFieldSet* unknownFieldSet = [[[GPBUnknownFieldSet alloc] init] autorelease]; |
|
[unknownFieldSet addField:unknownField]; |
|
[message_set setUnknownFields:unknownFieldSet]; |
|
|
|
GPBUnknownFields* ufs = [[[GPBUnknownFields alloc] init] autorelease]; |
|
GPBUnknownFields* group = [ufs addGroupWithFieldNumber:GPBWireFormatMessageSetItem]; |
|
[group addFieldNumber:GPBWireFormatMessageSetTypeId varint:kUnknownTypeId2]; |
|
[group addFieldNumber:GPBWireFormatMessageSetMessage lengthDelimited:DataFromCStr("baz")]; |
|
XCTAssertTrue([message_set mergeUnknownFields:ufs |
|
extensionRegistry:[MSetUnittestMsetRoot extensionRegistry] |
|
error:NULL]); |
|
|
|
NSData* data = [message_set data]; |
|
|
|
// Parse back using MSetRawMessageSet and check the contents. |
|
MSetRawMessageSet* raw = [MSetRawMessageSet parseFromData:data error:NULL]; |
|
|
|
GPBUnknownFields* ufs2 = [[[GPBUnknownFields alloc] initFromMessage:raw] autorelease]; |
|
XCTAssertTrue(ufs2.empty); |
|
XCTAssertEqual([raw.unknownFields countOfFields], (NSUInteger)0); |
|
|
|
XCTAssertEqual(raw.itemArray.count, (NSUInteger)4); |
|
XCTAssertEqual((uint32_t)[raw.itemArray[0] typeId], |
|
[MSetMessageExtension1 messageSetExtension].fieldNumber); |
|
XCTAssertEqual((uint32_t)[raw.itemArray[1] typeId], |
|
[MSetMessageExtension2 messageSetExtension].fieldNumber); |
|
XCTAssertEqual([raw.itemArray[2] typeId], kUnknownTypeId); |
|
XCTAssertEqual([raw.itemArray[3] typeId], kUnknownTypeId2); |
|
|
|
MSetMessageExtension1* message1 = |
|
[MSetMessageExtension1 parseFromData:[((MSetRawMessageSet_Item*)raw.itemArray[0]) message] |
|
error:NULL]; |
|
XCTAssertEqual(message1.i, 123); |
|
|
|
MSetMessageExtension2* message2 = |
|
[MSetMessageExtension2 parseFromData:[((MSetRawMessageSet_Item*)raw.itemArray[1]) message] |
|
error:NULL]; |
|
XCTAssertEqualObjects(message2.str, @"foo"); |
|
|
|
XCTAssertEqualObjects([raw.itemArray[2] message], DataFromCStr("bar")); |
|
XCTAssertEqualObjects([raw.itemArray[3] message], DataFromCStr("baz")); |
|
} |
|
|
|
- (void)testParseMessageSet { |
|
// Set up a MSetRawMessageSet with two known messages and an unknown one. |
|
MSetRawMessageSet* raw = [MSetRawMessageSet message]; |
|
|
|
{ |
|
MSetRawMessageSet_Item* item = [MSetRawMessageSet_Item message]; |
|
item.typeId = [MSetMessageExtension1 messageSetExtension].fieldNumber; |
|
MSetMessageExtension1* message = [MSetMessageExtension1 message]; |
|
message.i = 123; |
|
item.message = [message data]; |
|
[raw.itemArray addObject:item]; |
|
} |
|
|
|
{ |
|
MSetRawMessageSet_Item* item = [MSetRawMessageSet_Item message]; |
|
item.typeId = [MSetMessageExtension2 messageSetExtension].fieldNumber; |
|
MSetMessageExtension2* message = [MSetMessageExtension2 message]; |
|
message.str = @"foo"; |
|
item.message = [message data]; |
|
[raw.itemArray addObject:item]; |
|
} |
|
|
|
{ |
|
MSetRawMessageSet_Item* item = [MSetRawMessageSet_Item message]; |
|
item.typeId = kUnknownTypeId; |
|
item.message = DataFromCStr("bar"); |
|
[raw.itemArray addObject:item]; |
|
} |
|
|
|
NSData* data = [raw data]; |
|
|
|
// Parse as a MSetMessage and check the contents. |
|
MSetMessage* messageSet = [MSetMessage parseFromData:data |
|
extensionRegistry:[MSetUnittestMsetRoot extensionRegistry] |
|
error:NULL]; |
|
|
|
XCTAssertEqual([[messageSet getExtension:[MSetMessageExtension1 messageSetExtension]] i], 123); |
|
XCTAssertEqualObjects([[messageSet getExtension:[MSetMessageExtension2 messageSetExtension]] str], |
|
@"foo"); |
|
|
|
GPBUnknownFields* ufs = [[[GPBUnknownFields alloc] initFromMessage:messageSet] autorelease]; |
|
XCTAssertEqual(ufs.count, (NSUInteger)1); |
|
GPBUnknownFields* group = [ufs firstGroup:GPBWireFormatMessageSetItem]; |
|
XCTAssertNotNil(group); |
|
XCTAssertEqual(group.count, (NSUInteger)2); |
|
uint64_t varint = 0; |
|
XCTAssertTrue([group getFirst:GPBWireFormatMessageSetTypeId varint:&varint]); |
|
XCTAssertEqual(varint, kUnknownTypeId); |
|
XCTAssertEqualObjects([group firstLengthDelimited:GPBWireFormatMessageSetMessage], |
|
DataFromCStr("bar")); |
|
|
|
XCTAssertEqual([messageSet.unknownFields countOfFields], (NSUInteger)1); |
|
GPBUnknownField* unknownField = [messageSet.unknownFields getField:kUnknownTypeId]; |
|
XCTAssertNotNil(unknownField); |
|
XCTAssertEqual(unknownField.lengthDelimitedList.count, (NSUInteger)1); |
|
XCTAssertEqualObjects(unknownField.lengthDelimitedList[0], DataFromCStr("bar")); |
|
} |
|
|
|
- (void)testParseMessageSet_FirstValueSticks { |
|
MSetRawBreakableMessageSet* raw = [MSetRawBreakableMessageSet message]; |
|
|
|
{ |
|
MSetRawBreakableMessageSet_Item* item = [MSetRawBreakableMessageSet_Item message]; |
|
|
|
[item.typeIdArray addValue:[MSetMessageExtension1 messageSetExtension].fieldNumber]; |
|
MSetMessageExtension1* message1 = [MSetMessageExtension1 message]; |
|
message1.i = 123; |
|
NSData* itemData = [message1 data]; |
|
[item.messageArray addObject:itemData]; |
|
|
|
[item.typeIdArray addValue:[MSetMessageExtension2 messageSetExtension].fieldNumber]; |
|
MSetMessageExtension2* message2 = [MSetMessageExtension2 message]; |
|
message2.str = @"foo"; |
|
itemData = [message2 data]; |
|
[item.messageArray addObject:itemData]; |
|
|
|
[raw.itemArray addObject:item]; |
|
} |
|
|
|
NSData* data = [raw data]; |
|
|
|
// Parse as a MSetMessage and check the contents. |
|
NSError* err = nil; |
|
MSetMessage* messageSet = [MSetMessage parseFromData:data |
|
extensionRegistry:[MSetUnittestMsetRoot extensionRegistry] |
|
error:&err]; |
|
XCTAssertNotNil(messageSet); |
|
XCTAssertNil(err); |
|
XCTAssertTrue([messageSet hasExtension:[MSetMessageExtension1 messageSetExtension]]); |
|
XCTAssertEqual([[messageSet getExtension:[MSetMessageExtension1 messageSetExtension]] i], 123); |
|
XCTAssertFalse([messageSet hasExtension:[MSetMessageExtension2 messageSetExtension]]); |
|
GPBUnknownFields* ufs = [[[GPBUnknownFields alloc] initFromMessage:messageSet] autorelease]; |
|
XCTAssertTrue(ufs.empty); |
|
} |
|
|
|
- (void)testParseMessageSet_PartialValuesDropped { |
|
MSetRawBreakableMessageSet* raw = [MSetRawBreakableMessageSet message]; |
|
|
|
{ |
|
MSetRawBreakableMessageSet_Item* item = [MSetRawBreakableMessageSet_Item message]; |
|
[item.typeIdArray addValue:[MSetMessageExtension1 messageSetExtension].fieldNumber]; |
|
// No payload. |
|
[raw.itemArray addObject:item]; |
|
} |
|
|
|
{ |
|
MSetRawBreakableMessageSet_Item* item = [MSetRawBreakableMessageSet_Item message]; |
|
// No type ID. |
|
MSetMessageExtension2* message = [MSetMessageExtension2 message]; |
|
message.str = @"foo"; |
|
NSData* itemData = [message data]; |
|
[item.messageArray addObject:itemData]; |
|
[raw.itemArray addObject:item]; |
|
} |
|
|
|
{ |
|
MSetRawBreakableMessageSet_Item* item = [MSetRawBreakableMessageSet_Item message]; |
|
// Neither type ID nor payload. |
|
[raw.itemArray addObject:item]; |
|
} |
|
|
|
NSData* data = [raw data]; |
|
|
|
// Parse as a MSetMessage and check the contents. |
|
NSError* err = nil; |
|
MSetMessage* messageSet = [MSetMessage parseFromData:data |
|
extensionRegistry:[MSetUnittestMsetRoot extensionRegistry] |
|
error:&err]; |
|
XCTAssertNotNil(messageSet); |
|
XCTAssertNil(err); |
|
XCTAssertEqual([messageSet extensionsCurrentlySet].count, |
|
(NSUInteger)0); // None because they were all partial and dropped. |
|
GPBUnknownFields* ufs = [[[GPBUnknownFields alloc] initFromMessage:messageSet] autorelease]; |
|
XCTAssertTrue(ufs.empty); |
|
} |
|
|
|
- (void)assertFieldsInOrder:(NSData*)data { |
|
GPBCodedInputStream* input = [GPBCodedInputStream streamWithData:data]; |
|
int32_t previousTag = 0; |
|
|
|
while (YES) { |
|
int32_t tag = [input readTag]; |
|
if (tag == 0) { |
|
break; |
|
} |
|
|
|
XCTAssertGreaterThan(tag, previousTag); |
|
[input skipField:tag]; |
|
} |
|
} |
|
|
|
@end
|
|
|