| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363 | // 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 <XCTest/XCTest.h>#import "GPBUtilities_PackagePrivate.h"#import <objc/runtime.h>#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 testingtypedef 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;}@endtypedef 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
 |