// Protocol Buffers - Google's data interchange format // Copyright 2008 Google Inc. All rights reserved. // https://developers.google.com/protocol-buffers/ // // 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. #import #import "GPBUtilities_PackagePrivate.h" #import #import "GPBTestUtilities.h" #import "GPBDescriptor.h" #import "GPBDescriptor_PackagePrivate.h" #import "GPBMessage.h" #import "google/protobuf/MapUnittest.pbobjc.h" #import "google/protobuf/Unittest.pbobjc.h" #import "google/protobuf/UnittestObjc.pbobjc.h" @interface UtilitiesTests : GPBTestCase @end // Support code for testing typedef struct { uint32_t _has_storage_[1]; BOOL aBool; int32_t aInt32; uint32_t aUInt32; int64_t aInt64; uint64_t aUInt64; float aFloat; double aDouble; id aObject; BOOL _hasTest; BOOL stopper; BOOL shouldNotBeCounted; GPBInt32Array *anArray; } ApplyFunctionsTest_Storage; @interface ApplyFunctionsTest : GPBMessage @property(nonatomic, readwrite) BOOL aBool; @property(nonatomic, readwrite) int32_t aInt32; @property(nonatomic, readwrite) uint32_t aUInt32; @property(nonatomic, readwrite) int64_t aInt64; @property(nonatomic, readwrite) uint64_t aUInt64; @property(nonatomic, readwrite) float aFloat; @property(nonatomic, readwrite) double aDouble; @property(nonatomic, readwrite, retain) id aObject; @property(nonatomic, readwrite) BOOL _hasTest; @property(nonatomic, readwrite) BOOL stopper; @property(nonatomic, readwrite) BOOL shouldNotBeCounted; @property(nonatomic, readwrite, retain) GPBInt32Array *anArray; @end @implementation ApplyFunctionsTest @dynamic aBool, aInt32, aUInt32, aInt64, aUInt64, aFloat, aDouble, aObject; @dynamic _hasTest, stopper, shouldNotBeCounted, anArray; + (GPBDescriptor *)descriptor { static GPBDescriptor *descriptor = NULL; if (!descriptor) { static GPBMessageFieldDescription fields[] = { #define FIELD_ENTRY(NAME, INDEX) \ { \ .name = "a" #NAME, .hasIndex = INDEX, .type = GPBType##NAME, \ .offset = offsetof(ApplyFunctionsTest_Storage, a##NAME), \ } FIELD_ENTRY(Bool, 1), FIELD_ENTRY(Int32, 2), FIELD_ENTRY(UInt32, 3), FIELD_ENTRY(Int64, 4), FIELD_ENTRY(UInt64, 5), FIELD_ENTRY(Float, 6), FIELD_ENTRY(Double, 7), #undef FIELD_ENTRY { .name = "aObject", .type = GPBTypeString, .hasIndex = 8, .offset = offsetof(ApplyFunctionsTest_Storage, aObject), }, { .name = "stopper", .type = GPBTypeBool, .hasIndex = 9, .offset = offsetof(ApplyFunctionsTest_Storage, stopper), }, { .name = "shouldNotBeCounted", .type = GPBTypeBool, .hasIndex = 10, .offset = offsetof(ApplyFunctionsTest_Storage, shouldNotBeCounted), }, { .name = "anArray", .type = GPBTypeInt32, .hasIndex = 11, .flags = GPBFieldRepeated, .offset = offsetof(ApplyFunctionsTest_Storage, anArray), }, }; descriptor = [GPBDescriptor allocDescriptorForClass:[self class] rootClass:Nil file:nil fields:fields fieldCount:sizeof(fields) / sizeof(GPBMessageFieldDescription) oneofs:NULL oneofCount:0 enums:NULL enumCount:0 ranges:NULL rangeCount:0 storageSize:sizeof(ApplyFunctionsTest_Storage) wireFormat:NO]; } return descriptor; } @end typedef struct { int calledBool; int calledInt32; int calledUInt32; int calledInt64; int calledUInt64; int calledFloat; int calledDouble; int calledObject; int hitCount; } TestApplyFunctionsContext; // Really, who needs templates? // Macro for testing apply functions. Declares a variety of different functions // base on |NAME|. #define TEST_APPLY_FUNCTIONS_FUNC(NAME) \ static BOOL TestApplyFunction##NAME(GPBFieldDescriptor *field, \ void *voidContext) { \ TestApplyFunctionsContext *context = voidContext; \ if (field->getSel_ == sel_getUid("stopper")) return NO; \ context->called##NAME += 1; \ context->hitCount += 1; \ return YES; \ } TEST_APPLY_FUNCTIONS_FUNC(Bool) TEST_APPLY_FUNCTIONS_FUNC(Int32) TEST_APPLY_FUNCTIONS_FUNC(UInt32) TEST_APPLY_FUNCTIONS_FUNC(Int64) TEST_APPLY_FUNCTIONS_FUNC(UInt64) TEST_APPLY_FUNCTIONS_FUNC(Float) TEST_APPLY_FUNCTIONS_FUNC(Double) TEST_APPLY_FUNCTIONS_FUNC(Object) @implementation UtilitiesTests - (void)testRightShiftFunctions { XCTAssertEqual((1UL << 31) >> 31, 1UL); XCTAssertEqual((1 << 31) >> 31, -1); XCTAssertEqual((1ULL << 63) >> 63, 1ULL); XCTAssertEqual((1LL << 63) >> 63, -1LL); XCTAssertEqual(GPBLogicalRightShift32((1 << 31), 31), 1); XCTAssertEqual(GPBLogicalRightShift64((1LL << 63), 63), 1LL); } - (void)testMutability { ApplyFunctionsTest *foo_message = [ApplyFunctionsTest message]; XCTAssertEqual(0, [foo_message aInt32]); [foo_message setAInt32:100]; XCTAssertEqual(100, [foo_message aInt32]); } - (void)testSerializedSize { ApplyFunctionsTest *foo_message = [ApplyFunctionsTest message]; [foo_message setAInt32:100]; size_t size1 = [foo_message serializedSize]; [foo_message setAInt64:100]; size_t size2 = [foo_message serializedSize]; // Intentionally doing a pointer comparison. XCTAssertNotEqual(size1, size2); } - (void)testCopying { ApplyFunctionsTest *foo_message = [ApplyFunctionsTest message]; [foo_message setAInt32:100]; [foo_message setAObject:@"Happy"]; ApplyFunctionsTest *foo = [[foo_message copy] autorelease]; XCTAssertNotEqual(foo, foo_message); // Pointer comparision XCTAssertEqualObjects(foo, foo_message); } - (void)testApplyFunctions { // Covers ApplyFunctionsToProtoVariables and // ApplyFunctionsBasedOnEncodingType. // This test depends on the layout of the ivars to be in the order // declared in the interface. If this is not true, it will fail and will // need to be rewritten to accomodate. TestApplyFunctionsContext context; memset(&context, 0, sizeof(context)); GPBApplyFunctions foo = GPBAPPLY_FUNCTIONS_INIT(TestApplyFunction); ApplyFunctionsTest *msg = [ApplyFunctionsTest message]; GPBApplyFunctionsToMessageFields(&foo, msg, &context); // Only eight vars should be set. // "stopper" should cause the loop to quit so it and shouldNotBeCounted should // not be counted. // "_hasTest" should be skipped over. // Each of the vars should only be set once. XCTAssertEqual(context.hitCount, 8); XCTAssertEqual(context.calledBool, 1); XCTAssertEqual(context.calledInt32, 1); XCTAssertEqual(context.calledUInt32, 1); XCTAssertEqual(context.calledInt64, 1); XCTAssertEqual(context.calledUInt64, 1); XCTAssertEqual(context.calledFloat, 1); XCTAssertEqual(context.calledDouble, 1); XCTAssertEqual(context.calledObject, 1); } - (void)testGPBDecodeTextFormatName { uint8_t decodeData[] = { 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, }; 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"); // Long name so multiple decode ops are needed. inputStr = @"longFieldNameIsLooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong1000"; XCTAssertEqualObjects(GPBDecodeTextFormatName(decodeData, 1000, inputStr), @"long_field_name_is_looooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooooong_1000"); } - (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"; NSString *expected = @"_cmd: true\n" @"isProxy: true\n" @"SubEnum: retainCount\n" @"New {\n" @" copy: \"foo\"\n" @"}\n"; 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]; } // TODO(thomasvl): add test with extensions once those format with correct names. @end