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.
438 lines
16 KiB
438 lines
16 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 <XCTest/XCTest.h> |
|
#import <objc/runtime.h> |
|
|
|
#import "GPBDescriptor.h" |
|
#import "GPBDescriptor_PackagePrivate.h" |
|
#import "GPBMessage.h" |
|
#import "GPBTestUtilities.h" |
|
#import "GPBUnknownField.h" |
|
#import "GPBUnknownField_PackagePrivate.h" |
|
#import "GPBUtilities.h" |
|
#import "GPBUtilities_PackagePrivate.h" |
|
#import "objectivec/Tests/MapUnittest.pbobjc.h" |
|
#import "objectivec/Tests/Unittest.pbobjc.h" |
|
#import "objectivec/Tests/UnittestObjc.pbobjc.h" |
|
|
|
@interface UtilitiesTests : GPBTestCase |
|
@end |
|
|
|
@implementation UtilitiesTests |
|
|
|
- (void)testRightShiftFunctions { |
|
XCTAssertEqual((1UL << 31) >> 31, 1UL); |
|
XCTAssertEqual((int32_t)(1U << 31) >> 31, -1); |
|
XCTAssertEqual((1ULL << 63) >> 63, 1ULL); |
|
XCTAssertEqual((int64_t)(1ULL << 63) >> 63, -1LL); |
|
|
|
XCTAssertEqual(GPBLogicalRightShift32((1U << 31), 31), 1); |
|
XCTAssertEqual(GPBLogicalRightShift64((1ULL << 63), 63), 1LL); |
|
} |
|
|
|
- (void)testGPBDecodeTextFormatName { |
|
uint8_t decodeData[] = { |
|
// clang-format off |
|
0x6, |
|
// An inlined string (first to make sure the leading null is handled |
|
// correctly, and with a key of zero to check that). |
|
0x0, 0x0, 'z', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'I', 'J', 0x0, |
|
// All as is (00 op) |
|
0x1, 0x0A, 0x0, |
|
// Underscore, upper + 9 (10 op) |
|
0x3, 0xCA, 0x0, |
|
// Upper + 3 (10 op), underscore, upper + 5 (10 op) |
|
0x2, 0x44, 0xC6, 0x0, |
|
// All Upper for 4 (11 op), underscore, underscore, upper + 5 (10 op), |
|
// underscore, lower + 0 (01 op) |
|
0x4, 0x64, 0x80, 0xC5, 0xA1, 0x0, |
|
// 2 byte key: as is + 3 (00 op), underscore, lower + 4 (01 op), |
|
// underscore, lower + 3 (01 op), underscore, lower + 1 (01 op), |
|
// underscore, lower + 30 (01 op), as is + 30 (00 op), as is + 13 (00 op), |
|
// underscore, as is + 3 (00 op) |
|
0xE8, 0x07, 0x04, 0xA5, 0xA4, 0xA2, 0xBF, 0x1F, 0x0E, 0x84, 0x0, |
|
// clang-format on |
|
}; |
|
NSString *inputStr = @"abcdefghIJ"; |
|
|
|
// Empty inputs |
|
|
|
XCTAssertNil(GPBDecodeTextFormatName(nil, 1, NULL)); |
|
XCTAssertNil(GPBDecodeTextFormatName(decodeData, 1, NULL)); |
|
XCTAssertNil(GPBDecodeTextFormatName(nil, 1, inputStr)); |
|
|
|
// Keys not found. |
|
|
|
XCTAssertNil(GPBDecodeTextFormatName(decodeData, 5, inputStr)); |
|
XCTAssertNil(GPBDecodeTextFormatName(decodeData, -1, inputStr)); |
|
|
|
// Some name decodes. |
|
|
|
XCTAssertEqualObjects(GPBDecodeTextFormatName(decodeData, 1, inputStr), @"abcdefghIJ"); |
|
XCTAssertEqualObjects(GPBDecodeTextFormatName(decodeData, 2, inputStr), @"Abcd_EfghIJ"); |
|
XCTAssertEqualObjects(GPBDecodeTextFormatName(decodeData, 3, inputStr), @"_AbcdefghIJ"); |
|
XCTAssertEqualObjects(GPBDecodeTextFormatName(decodeData, 4, inputStr), @"ABCD__EfghI_j"); |
|
|
|
// An inlined string (and key of zero). |
|
XCTAssertEqualObjects(GPBDecodeTextFormatName(decodeData, 0, inputStr), @"zbcdefghIJ"); |
|
|
|
// clang-format off |
|
// Long name so multiple decode ops are needed. |
|
inputStr = @"longFieldNameIsLooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong1000"; |
|
XCTAssertEqualObjects(GPBDecodeTextFormatName(decodeData, 1000, inputStr), |
|
@"long_field_name_is_looooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong_1000"); |
|
// clang-format on |
|
} |
|
|
|
- (void)testTextFormat { |
|
TestAllTypes *message = [TestAllTypes message]; |
|
|
|
// Not kGPBDefaultRepeatCount because we are comparing to golden master file |
|
// which was generated with 2. |
|
[self setAllFields:message repeatedCount:2]; |
|
|
|
NSString *result = GPBTextFormatForMessage(message, nil); |
|
|
|
NSString *fileName = @"text_format_unittest_data.txt"; |
|
NSData *resultData = [result dataUsingEncoding:NSUTF8StringEncoding]; |
|
NSData *expectedData = [self getDataFileNamed:fileName dataToWrite:resultData]; |
|
NSString *expected = [[NSString alloc] initWithData:expectedData encoding:NSUTF8StringEncoding]; |
|
XCTAssertEqualObjects(expected, result); |
|
[expected release]; |
|
} |
|
|
|
- (void)testTextFormatExtra { |
|
// -testTextFormat uses all protos with fields that don't require special |
|
// handing for figuring out the names. The ObjC proto has a bunch of oddball |
|
// field and enum names that require the decode info to get right, so this |
|
// confirms they generated and decoded correctly. |
|
|
|
self_Class *message = [self_Class message]; |
|
message.cmd = YES; |
|
message.isProxy_p = YES; |
|
message.subEnum = self_autorelease_RetainCount; |
|
message.new_p.copy_p = @"foo"; |
|
|
|
// clang-format off |
|
NSString *expected = @"_cmd: true\n" |
|
@"isProxy: true\n" |
|
@"SubEnum: retainCount\n" |
|
@"New {\n" |
|
@" copy: \"foo\"\n" |
|
@"}\n"; |
|
// clang-format on |
|
NSString *result = GPBTextFormatForMessage(message, nil); |
|
XCTAssertEqualObjects(expected, result); |
|
} |
|
|
|
- (void)testTextFormatMaps { |
|
TestMap *message = [TestMap message]; |
|
|
|
// Map iteration order doesn't have to be stable, so use only one entry. |
|
[self setAllMapFields:message numEntries:1]; |
|
|
|
NSString *result = GPBTextFormatForMessage(message, nil); |
|
|
|
NSString *fileName = @"text_format_map_unittest_data.txt"; |
|
NSData *resultData = [result dataUsingEncoding:NSUTF8StringEncoding]; |
|
NSData *expectedData = [self getDataFileNamed:fileName dataToWrite:resultData]; |
|
NSString *expected = [[NSString alloc] initWithData:expectedData encoding:NSUTF8StringEncoding]; |
|
XCTAssertEqualObjects(expected, result); |
|
[expected release]; |
|
} |
|
|
|
- (void)testTextFormatExtensions { |
|
TestAllExtensions *message = [TestAllExtensions message]; |
|
|
|
// Not kGPBDefaultRepeatCount because we are comparing to golden master file |
|
// which was generated with 2. |
|
[self setAllExtensions:message repeatedCount:2]; |
|
|
|
NSString *result = GPBTextFormatForMessage(message, nil); |
|
|
|
// NOTE: ObjC TextFormat doesn't have the proper extension names so it |
|
// uses comments for the ObjC name and raw numbers for the fields instead |
|
// of the bracketed extension name. |
|
NSString *fileName = @"text_format_extensions_unittest_data.txt"; |
|
NSData *resultData = [result dataUsingEncoding:NSUTF8StringEncoding]; |
|
NSData *expectedData = [self getDataFileNamed:fileName dataToWrite:resultData]; |
|
NSString *expected = [[NSString alloc] initWithData:expectedData encoding:NSUTF8StringEncoding]; |
|
XCTAssertEqualObjects(expected, result); |
|
[expected release]; |
|
} |
|
|
|
- (void)testTextFormatUnknownFields { |
|
GPBUnknownFields *ufs = [[[GPBUnknownFields alloc] init] autorelease]; |
|
[ufs addFieldNumber:100 varint:5]; |
|
[ufs addFieldNumber:100 varint:4]; |
|
[ufs addFieldNumber:10 varint:1]; |
|
[ufs addFieldNumber:300 fixed32:0x50]; |
|
[ufs addFieldNumber:300 fixed32:0x40]; |
|
[ufs addFieldNumber:10 fixed32:0x10]; |
|
[ufs addFieldNumber:200 fixed64:0x5000]; |
|
[ufs addFieldNumber:200 fixed64:0x4000]; |
|
[ufs addFieldNumber:10 fixed64:0x1000]; |
|
[ufs addFieldNumber:10 lengthDelimited:DataFromCStr("foo")]; |
|
[ufs addFieldNumber:10 lengthDelimited:DataFromCStr("bar")]; |
|
GPBUnknownFields *group = [ufs addGroupWithFieldNumber:150]; |
|
[group addFieldNumber:2 varint:2]; |
|
[group addFieldNumber:1 varint:1]; |
|
group = [ufs addGroupWithFieldNumber:150]; |
|
[group addFieldNumber:1 varint:1]; |
|
[group addFieldNumber:3 fixed32:0x3]; |
|
[group addFieldNumber:2 fixed64:0x2]; |
|
TestEmptyMessage *message = [TestEmptyMessage message]; |
|
XCTAssertTrue([message mergeUnknownFields:ufs extensionRegistry:nil error:NULL]); |
|
|
|
NSString *expected = @"# --- Unknown fields ---\n" |
|
@"10: 1\n" |
|
@"10: 0x10\n" |
|
@"10: 0x1000\n" |
|
@"10: \"foo\"\n" |
|
@"10: \"bar\"\n" |
|
@"100: 5\n" |
|
@"100: 4\n" |
|
@"150: {\n" |
|
@" 1: 1\n" |
|
@" 2: 2\n" |
|
@"}\n" |
|
@"150: {\n" |
|
@" 1: 1\n" |
|
@" 2: 0x2\n" |
|
@" 3: 0x3\n" |
|
@"}\n" |
|
@"200: 0x5000\n" |
|
@"200: 0x4000\n" |
|
@"300: 0x50\n" |
|
@"300: 0x40\n"; |
|
NSString *result = GPBTextFormatForMessage(message, nil); |
|
XCTAssertEqualObjects(expected, result); |
|
} |
|
|
|
- (void)testSetRepeatedFields { |
|
TestAllTypes *message = [TestAllTypes message]; |
|
|
|
NSDictionary *repeatedFieldValues = @{ |
|
@"repeatedStringArray" : [@[ @"foo", @"bar" ] mutableCopy], |
|
@"repeatedBoolArray" : [GPBBoolArray arrayWithValue:YES], |
|
@"repeatedInt32Array" : [GPBInt32Array arrayWithValue:14], |
|
@"repeatedInt64Array" : [GPBInt64Array arrayWithValue:15], |
|
@"repeatedUint32Array" : [GPBUInt32Array arrayWithValue:16], |
|
@"repeatedUint64Array" : [GPBUInt64Array arrayWithValue:16], |
|
@"repeatedFloatArray" : [GPBFloatArray arrayWithValue:16], |
|
@"repeatedDoubleArray" : [GPBDoubleArray arrayWithValue:16], |
|
@"repeatedNestedEnumArray" : |
|
[GPBEnumArray arrayWithValidationFunction:TestAllTypes_NestedEnum_IsValidValue |
|
rawValue:TestAllTypes_NestedEnum_Foo], |
|
}; |
|
for (NSString *fieldName in repeatedFieldValues) { |
|
GPBFieldDescriptor *field = [message.descriptor fieldWithName:fieldName]; |
|
XCTAssertNotNil(field, @"No field with name: %@", fieldName); |
|
id expectedValues = repeatedFieldValues[fieldName]; |
|
GPBSetMessageRepeatedField(message, field, expectedValues); |
|
XCTAssertEqualObjects(expectedValues, [message valueForKeyPath:fieldName]); |
|
} |
|
} |
|
|
|
// Helper to add an unknown field data to messages. |
|
static void AddUnknownFields(GPBMessage *message, int num) { |
|
GPBUnknownFields *ufs = [[GPBUnknownFields alloc] init]; |
|
[ufs addFieldNumber:num varint:num]; |
|
// Can't fail since it is a varint. |
|
[message mergeUnknownFields:ufs extensionRegistry:nil error:NULL]; |
|
[ufs release]; |
|
} |
|
|
|
static BOOL HasUnknownFields(GPBMessage *message) { |
|
GPBUnknownFields *ufs = [[GPBUnknownFields alloc] initFromMessage:message]; |
|
BOOL result = !ufs.empty; |
|
[ufs release]; |
|
return result; |
|
} |
|
|
|
- (void)testDropMessageUnknownFieldsRecursively { |
|
TestAllExtensions *message = [TestAllExtensions message]; |
|
|
|
// Give it unknownFields. |
|
AddUnknownFields(message, 1777); |
|
|
|
// Given it extensions that include a message with unknown fields of its own. |
|
{ |
|
// Int |
|
[message setExtension:[UnittestRoot optionalInt32Extension] value:@5]; |
|
|
|
// Group |
|
OptionalGroup_extension *optionalGroup = [OptionalGroup_extension message]; |
|
optionalGroup.a = 123; |
|
AddUnknownFields(optionalGroup, 1779); |
|
[message setExtension:[UnittestRoot optionalGroupExtension] value:optionalGroup]; |
|
|
|
// Message |
|
TestAllTypes_NestedMessage *nestedMessage = [TestAllTypes_NestedMessage message]; |
|
nestedMessage.bb = 456; |
|
AddUnknownFields(nestedMessage, 1778); |
|
[message setExtension:[UnittestRoot optionalNestedMessageExtension] value:nestedMessage]; |
|
|
|
// Repeated Group |
|
RepeatedGroup_extension *repeatedGroup = [[RepeatedGroup_extension alloc] init]; |
|
repeatedGroup.a = 567; |
|
AddUnknownFields(repeatedGroup, 1780); |
|
[message addExtension:[UnittestRoot repeatedGroupExtension] value:repeatedGroup]; |
|
[repeatedGroup release]; |
|
|
|
// Repeated Message |
|
nestedMessage = [[TestAllTypes_NestedMessage alloc] init]; |
|
nestedMessage.bb = 678; |
|
AddUnknownFields(nestedMessage, 1781); |
|
[message addExtension:[UnittestRoot repeatedNestedMessageExtension] value:nestedMessage]; |
|
[nestedMessage release]; |
|
} |
|
|
|
// Confirm everything is there. |
|
|
|
XCTAssertNotNil(message); |
|
XCTAssertTrue(HasUnknownFields(message)); |
|
XCTAssertTrue([message hasExtension:[UnittestRoot optionalInt32Extension]]); |
|
|
|
{ |
|
XCTAssertTrue([message hasExtension:[UnittestRoot optionalGroupExtension]]); |
|
OptionalGroup_extension *optionalGroup = |
|
[message getExtension:[UnittestRoot optionalGroupExtension]]; |
|
XCTAssertNotNil(optionalGroup); |
|
XCTAssertEqual(optionalGroup.a, 123); |
|
XCTAssertTrue(HasUnknownFields(optionalGroup)); |
|
} |
|
|
|
{ |
|
XCTAssertTrue([message hasExtension:[UnittestRoot optionalNestedMessageExtension]]); |
|
TestAllTypes_NestedMessage *nestedMessage = |
|
[message getExtension:[UnittestRoot optionalNestedMessageExtension]]; |
|
XCTAssertNotNil(nestedMessage); |
|
XCTAssertEqual(nestedMessage.bb, 456); |
|
XCTAssertTrue(HasUnknownFields(nestedMessage)); |
|
} |
|
|
|
{ |
|
XCTAssertTrue([message hasExtension:[UnittestRoot repeatedGroupExtension]]); |
|
NSArray *repeatedGroups = [message getExtension:[UnittestRoot repeatedGroupExtension]]; |
|
XCTAssertEqual(repeatedGroups.count, (NSUInteger)1); |
|
RepeatedGroup_extension *repeatedGroup = repeatedGroups.firstObject; |
|
XCTAssertNotNil(repeatedGroup); |
|
XCTAssertEqual(repeatedGroup.a, 567); |
|
XCTAssertTrue(HasUnknownFields(repeatedGroup)); |
|
} |
|
|
|
{ |
|
XCTAssertTrue([message hasExtension:[UnittestRoot repeatedNestedMessageExtension]]); |
|
NSArray *repeatedNestedMessages = |
|
[message getExtension:[UnittestRoot repeatedNestedMessageExtension]]; |
|
XCTAssertEqual(repeatedNestedMessages.count, (NSUInteger)1); |
|
TestAllTypes_NestedMessage *repeatedNestedMessage = repeatedNestedMessages.firstObject; |
|
XCTAssertNotNil(repeatedNestedMessage); |
|
XCTAssertEqual(repeatedNestedMessage.bb, 678); |
|
XCTAssertTrue(HasUnknownFields(repeatedNestedMessage)); |
|
} |
|
|
|
// Drop them. |
|
GPBMessageDropUnknownFieldsRecursively(message); |
|
|
|
// Confirm unknowns are gone from within the messages. |
|
|
|
XCTAssertNotNil(message); |
|
XCTAssertFalse(HasUnknownFields(message)); |
|
XCTAssertTrue([message hasExtension:[UnittestRoot optionalInt32Extension]]); |
|
|
|
{ |
|
XCTAssertTrue([message hasExtension:[UnittestRoot optionalGroupExtension]]); |
|
OptionalGroup_extension *optionalGroup = |
|
[message getExtension:[UnittestRoot optionalGroupExtension]]; |
|
XCTAssertNotNil(optionalGroup); |
|
XCTAssertEqual(optionalGroup.a, 123); |
|
XCTAssertFalse(HasUnknownFields(optionalGroup)); |
|
} |
|
|
|
{ |
|
XCTAssertTrue([message hasExtension:[UnittestRoot optionalNestedMessageExtension]]); |
|
TestAllTypes_NestedMessage *nestedMessage = |
|
[message getExtension:[UnittestRoot optionalNestedMessageExtension]]; |
|
XCTAssertNotNil(nestedMessage); |
|
XCTAssertEqual(nestedMessage.bb, 456); |
|
XCTAssertFalse(HasUnknownFields(nestedMessage)); |
|
} |
|
|
|
{ |
|
XCTAssertTrue([message hasExtension:[UnittestRoot repeatedGroupExtension]]); |
|
NSArray *repeatedGroups = [message getExtension:[UnittestRoot repeatedGroupExtension]]; |
|
XCTAssertEqual(repeatedGroups.count, (NSUInteger)1); |
|
RepeatedGroup_extension *repeatedGroup = repeatedGroups.firstObject; |
|
XCTAssertNotNil(repeatedGroup); |
|
XCTAssertEqual(repeatedGroup.a, 567); |
|
XCTAssertFalse(HasUnknownFields(repeatedGroup)); |
|
} |
|
|
|
{ |
|
XCTAssertTrue([message hasExtension:[UnittestRoot repeatedNestedMessageExtension]]); |
|
NSArray *repeatedNestedMessages = |
|
[message getExtension:[UnittestRoot repeatedNestedMessageExtension]]; |
|
XCTAssertEqual(repeatedNestedMessages.count, (NSUInteger)1); |
|
TestAllTypes_NestedMessage *repeatedNestedMessage = repeatedNestedMessages.firstObject; |
|
XCTAssertNotNil(repeatedNestedMessage); |
|
XCTAssertEqual(repeatedNestedMessage.bb, 678); |
|
XCTAssertFalse(HasUnknownFields(repeatedNestedMessage)); |
|
} |
|
} |
|
|
|
- (void)testDropMessageUnknownFieldsRecursively_Maps { |
|
TestMap *message = [TestMap message]; |
|
|
|
{ |
|
ForeignMessage *foreignMessage = [ForeignMessage message]; |
|
AddUnknownFields(foreignMessage, 1000); |
|
[message.mapInt32ForeignMessage setObject:foreignMessage forKey:100]; |
|
|
|
foreignMessage = [ForeignMessage message]; |
|
AddUnknownFields(foreignMessage, 1001); |
|
[message.mapStringForeignMessage setObject:foreignMessage forKey:@"101"]; |
|
} |
|
|
|
// Confirm everything is there. |
|
|
|
XCTAssertNotNil(message); |
|
|
|
{ |
|
ForeignMessage *foreignMessage = [message.mapInt32ForeignMessage objectForKey:100]; |
|
XCTAssertNotNil(foreignMessage); |
|
XCTAssertTrue(HasUnknownFields(foreignMessage)); |
|
} |
|
|
|
{ |
|
ForeignMessage *foreignMessage = [message.mapStringForeignMessage objectForKey:@"101"]; |
|
XCTAssertNotNil(foreignMessage); |
|
XCTAssertTrue(HasUnknownFields(foreignMessage)); |
|
} |
|
|
|
GPBMessageDropUnknownFieldsRecursively(message); |
|
|
|
// Confirm unknowns are gone from within the messages. |
|
|
|
XCTAssertNotNil(message); |
|
|
|
{ |
|
ForeignMessage *foreignMessage = [message.mapInt32ForeignMessage objectForKey:100]; |
|
XCTAssertNotNil(foreignMessage); |
|
XCTAssertFalse(HasUnknownFields(foreignMessage)); |
|
} |
|
|
|
{ |
|
ForeignMessage *foreignMessage = [message.mapStringForeignMessage objectForKey:@"101"]; |
|
XCTAssertNotNil(foreignMessage); |
|
XCTAssertFalse(HasUnknownFields(foreignMessage)); |
|
} |
|
} |
|
|
|
@end
|
|
|