| // 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 "GPBTestUtilities.h" |
| |
| #import <objc/runtime.h> |
| |
| #import "GPBArray_PackagePrivate.h" |
| #import "GPBDescriptor.h" |
| #import "GPBDictionary_PackagePrivate.h" |
| #import "GPBMessage_PackagePrivate.h" |
| #import "GPBUnknownField_PackagePrivate.h" |
| #import "GPBUnknownFieldSet_PackagePrivate.h" |
| #import "google/protobuf/Unittest.pbobjc.h" |
| #import "google/protobuf/UnittestObjc.pbobjc.h" |
| |
| @interface MessageTests : GPBTestCase |
| @end |
| |
| @implementation MessageTests |
| |
| // TODO(thomasvl): this should get split into a few files of logic junks, it is |
| // a jumble of things at the moment (and the testutils have a bunch of the real |
| // assertions). |
| |
| - (TestAllTypes *)mergeSource { |
| TestAllTypes *message = [TestAllTypes message]; |
| [message setOptionalInt32:1]; |
| [message setOptionalString:@"foo"]; |
| [message setOptionalForeignMessage:[ForeignMessage message]]; |
| [message.repeatedStringArray addObject:@"bar"]; |
| return message; |
| } |
| |
| - (TestAllTypes *)mergeDestination { |
| TestAllTypes *message = [TestAllTypes message]; |
| [message setOptionalInt64:2]; |
| [message setOptionalString:@"baz"]; |
| ForeignMessage *foreignMessage = [ForeignMessage message]; |
| [foreignMessage setC:3]; |
| [message setOptionalForeignMessage:foreignMessage]; |
| [message.repeatedStringArray addObject:@"qux"]; |
| return message; |
| } |
| |
| - (TestAllTypes *)mergeDestinationWithoutForeignMessageIvar { |
| TestAllTypes *message = [TestAllTypes message]; |
| [message setOptionalInt64:2]; |
| [message setOptionalString:@"baz"]; |
| [message.repeatedStringArray addObject:@"qux"]; |
| return message; |
| } |
| |
| - (TestAllTypes *)mergeResult { |
| TestAllTypes *message = [TestAllTypes message]; |
| [message setOptionalInt32:1]; |
| [message setOptionalInt64:2]; |
| [message setOptionalString:@"foo"]; |
| ForeignMessage *foreignMessage = [ForeignMessage message]; |
| [foreignMessage setC:3]; |
| [message setOptionalForeignMessage:foreignMessage]; |
| [message.repeatedStringArray addObject:@"qux"]; |
| [message.repeatedStringArray addObject:@"bar"]; |
| return message; |
| } |
| |
| - (TestAllTypes *)mergeResultForDestinationWithoutForeignMessageIvar { |
| TestAllTypes *message = [TestAllTypes message]; |
| [message setOptionalInt32:1]; |
| [message setOptionalInt64:2]; |
| [message setOptionalString:@"foo"]; |
| ForeignMessage *foreignMessage = [ForeignMessage message]; |
| [message setOptionalForeignMessage:foreignMessage]; |
| [message.repeatedStringArray addObject:@"qux"]; |
| [message.repeatedStringArray addObject:@"bar"]; |
| return message; |
| } |
| |
| - (TestAllExtensions *)mergeExtensionsDestination { |
| TestAllExtensions *message = [TestAllExtensions message]; |
| [message setExtension:[UnittestRoot optionalInt32Extension] value:@5]; |
| [message setExtension:[UnittestRoot optionalStringExtension] value:@"foo"]; |
| ForeignMessage *foreignMessage = [ForeignMessage message]; |
| foreignMessage.c = 4; |
| [message setExtension:[UnittestRoot optionalForeignMessageExtension] |
| value:foreignMessage]; |
| TestAllTypes_NestedMessage *nestedMessage = |
| [TestAllTypes_NestedMessage message]; |
| [message setExtension:[UnittestRoot optionalNestedMessageExtension] |
| value:nestedMessage]; |
| return message; |
| } |
| |
| - (TestAllExtensions *)mergeExtensionsSource { |
| TestAllExtensions *message = [TestAllExtensions message]; |
| [message setExtension:[UnittestRoot optionalInt64Extension] value:@6]; |
| [message setExtension:[UnittestRoot optionalStringExtension] value:@"bar"]; |
| ForeignMessage *foreignMessage = [ForeignMessage message]; |
| [message setExtension:[UnittestRoot optionalForeignMessageExtension] |
| value:foreignMessage]; |
| TestAllTypes_NestedMessage *nestedMessage = |
| [TestAllTypes_NestedMessage message]; |
| nestedMessage.bb = 7; |
| [message setExtension:[UnittestRoot optionalNestedMessageExtension] |
| value:nestedMessage]; |
| return message; |
| } |
| |
| - (TestAllExtensions *)mergeExtensionsResult { |
| TestAllExtensions *message = [TestAllExtensions message]; |
| [message setExtension:[UnittestRoot optionalInt32Extension] value:@5]; |
| [message setExtension:[UnittestRoot optionalInt64Extension] value:@6]; |
| [message setExtension:[UnittestRoot optionalStringExtension] value:@"bar"]; |
| ForeignMessage *foreignMessage = [ForeignMessage message]; |
| foreignMessage.c = 4; |
| [message setExtension:[UnittestRoot optionalForeignMessageExtension] |
| value:foreignMessage]; |
| TestAllTypes_NestedMessage *nestedMessage = |
| [TestAllTypes_NestedMessage message]; |
| nestedMessage.bb = 7; |
| [message setExtension:[UnittestRoot optionalNestedMessageExtension] |
| value:nestedMessage]; |
| return message; |
| } |
| |
| - (void)testMergeFrom { |
| TestAllTypes *result = [[self.mergeDestination copy] autorelease]; |
| [result mergeFrom:self.mergeSource]; |
| NSData *resultData = [result data]; |
| NSData *mergeResultData = [self.mergeResult data]; |
| XCTAssertEqualObjects(resultData, mergeResultData); |
| XCTAssertEqualObjects(result, self.mergeResult); |
| |
| // Test when destination does not have an Ivar (type is an object) but source |
| // has such Ivar. |
| // The result must has the Ivar which is same as the one in source. |
| result = [[self.mergeDestinationWithoutForeignMessageIvar copy] autorelease]; |
| [result mergeFrom:self.mergeSource]; |
| resultData = [result data]; |
| mergeResultData = |
| [self.mergeResultForDestinationWithoutForeignMessageIvar data]; |
| XCTAssertEqualObjects(resultData, mergeResultData); |
| XCTAssertEqualObjects( |
| result, self.mergeResultForDestinationWithoutForeignMessageIvar); |
| |
| // Test when destination is empty. |
| // The result must is same as the source. |
| result = [TestAllTypes message]; |
| [result mergeFrom:self.mergeSource]; |
| resultData = [result data]; |
| mergeResultData = [self.mergeSource data]; |
| XCTAssertEqualObjects(resultData, mergeResultData); |
| XCTAssertEqualObjects(result, self.mergeSource); |
| } |
| |
| - (void)testMergeFromWithExtensions { |
| TestAllExtensions *result = [self mergeExtensionsDestination]; |
| [result mergeFrom:[self mergeExtensionsSource]]; |
| NSData *resultData = [result data]; |
| NSData *mergeResultData = [[self mergeExtensionsResult] data]; |
| XCTAssertEqualObjects(resultData, mergeResultData); |
| XCTAssertEqualObjects(result, [self mergeExtensionsResult]); |
| |
| // Test merging from data. |
| result = [self mergeExtensionsDestination]; |
| NSData *data = [[self mergeExtensionsSource] data]; |
| XCTAssertNotNil(data); |
| [result mergeFromData:data |
| extensionRegistry:[UnittestRoot extensionRegistry]]; |
| resultData = [result data]; |
| XCTAssertEqualObjects(resultData, mergeResultData); |
| XCTAssertEqualObjects(result, [self mergeExtensionsResult]); |
| } |
| |
| - (void)testIsEquals { |
| TestAllTypes *result = [[self.mergeDestination copy] autorelease]; |
| [result mergeFrom:self.mergeSource]; |
| XCTAssertEqualObjects(result.data, self.mergeResult.data); |
| XCTAssertEqualObjects(result, self.mergeResult); |
| TestAllTypes *result2 = [[self.mergeDestination copy] autorelease]; |
| XCTAssertNotEqualObjects(result2.data, self.mergeResult.data); |
| XCTAssertNotEqualObjects(result2, self.mergeResult); |
| } |
| |
| // ================================================================= |
| // Required-field-related tests. |
| |
| - (TestRequired *)testRequiredInitialized { |
| TestRequired *message = [TestRequired message]; |
| [message setA:1]; |
| [message setB:2]; |
| [message setC:3]; |
| return message; |
| } |
| |
| - (void)testRequired { |
| TestRequired *message = [TestRequired message]; |
| |
| XCTAssertFalse(message.initialized); |
| [message setA:1]; |
| XCTAssertFalse(message.initialized); |
| [message setB:1]; |
| XCTAssertFalse(message.initialized); |
| [message setC:1]; |
| XCTAssertTrue(message.initialized); |
| } |
| |
| - (void)testRequiredForeign { |
| TestRequiredForeign *message = [TestRequiredForeign message]; |
| |
| XCTAssertTrue(message.initialized); |
| |
| [message setOptionalMessage:[TestRequired message]]; |
| XCTAssertFalse(message.initialized); |
| |
| [message setOptionalMessage:self.testRequiredInitialized]; |
| XCTAssertTrue(message.initialized); |
| |
| [message.repeatedMessageArray addObject:[TestRequired message]]; |
| XCTAssertFalse(message.initialized); |
| |
| [message.repeatedMessageArray removeAllObjects]; |
| [message.repeatedMessageArray addObject:self.testRequiredInitialized]; |
| XCTAssertTrue(message.initialized); |
| } |
| |
| - (void)testRequiredExtension { |
| TestAllExtensions *message = [TestAllExtensions message]; |
| |
| XCTAssertTrue(message.initialized); |
| |
| [message setExtension:[TestRequired single] value:[TestRequired message]]; |
| XCTAssertFalse(message.initialized); |
| |
| [message setExtension:[TestRequired single] |
| value:self.testRequiredInitialized]; |
| XCTAssertTrue(message.initialized); |
| |
| [message addExtension:[TestRequired multi] value:[TestRequired message]]; |
| XCTAssertFalse(message.initialized); |
| |
| [message setExtension:[TestRequired multi] |
| index:0 |
| value:self.testRequiredInitialized]; |
| XCTAssertTrue(message.initialized); |
| } |
| |
| - (void)testDataFromUninitialized { |
| TestRequired *message = [TestRequired message]; |
| NSData *data = [message data]; |
| // In DEBUG, the data generation will fail, but in non DEBUG, it passes |
| // because the check isn't done (for speed). |
| #ifdef DEBUG |
| XCTAssertNil(data); |
| #else |
| XCTAssertNotNil(data); |
| XCTAssertFalse(message.initialized); |
| #endif // DEBUG |
| } |
| |
| - (void)testInitialized { |
| // We're mostly testing that no exception is thrown. |
| TestRequired *message = [TestRequired message]; |
| XCTAssertFalse(message.initialized); |
| } |
| |
| - (void)testDataFromNestedUninitialized { |
| TestRequiredForeign *message = [TestRequiredForeign message]; |
| [message setOptionalMessage:[TestRequired message]]; |
| [message.repeatedMessageArray addObject:[TestRequired message]]; |
| [message.repeatedMessageArray addObject:[TestRequired message]]; |
| NSData *data = [message data]; |
| // In DEBUG, the data generation will fail, but in non DEBUG, it passes |
| // because the check isn't done (for speed). |
| #ifdef DEBUG |
| XCTAssertNil(data); |
| #else |
| XCTAssertNotNil(data); |
| XCTAssertFalse(message.initialized); |
| #endif // DEBUG |
| } |
| |
| - (void)testNestedInitialized { |
| // We're mostly testing that no exception is thrown. |
| |
| TestRequiredForeign *message = [TestRequiredForeign message]; |
| [message setOptionalMessage:[TestRequired message]]; |
| [message.repeatedMessageArray addObject:[TestRequired message]]; |
| [message.repeatedMessageArray addObject:[TestRequired message]]; |
| |
| XCTAssertFalse(message.initialized); |
| } |
| |
| - (void)testParseUninitialized { |
| NSError *error = nil; |
| TestRequired *msg = |
| [TestRequired parseFromData:GPBEmptyNSData() error:&error]; |
| // In DEBUG, the parse will fail, but in non DEBUG, it passes because |
| // the check isn't done (for speed). |
| #ifdef DEBUG |
| XCTAssertNil(msg); |
| XCTAssertNotNil(error); |
| XCTAssertEqualObjects(error.domain, GPBMessageErrorDomain); |
| XCTAssertEqual(error.code, GPBMessageErrorCodeMissingRequiredField); |
| #else |
| XCTAssertNotNil(msg); |
| XCTAssertNil(error); |
| XCTAssertFalse(msg.initialized); |
| #endif // DEBUG |
| } |
| |
| - (void)testCoding { |
| NSData *data = |
| [NSKeyedArchiver archivedDataWithRootObject:[self mergeResult]]; |
| id unarchivedObject = [NSKeyedUnarchiver unarchiveObjectWithData:data]; |
| |
| XCTAssertEqualObjects(unarchivedObject, [self mergeResult]); |
| |
| // Intentionally doing a pointer comparison. |
| XCTAssertNotEqual(unarchivedObject, [self mergeResult]); |
| } |
| |
| - (void)testObjectReset { |
| // Tests a failure where clearing out defaults values caused an over release. |
| TestAllTypes *message = [TestAllTypes message]; |
| message.hasOptionalNestedMessage = NO; |
| [message setOptionalNestedMessage:[TestAllTypes_NestedMessage message]]; |
| message.hasOptionalNestedMessage = NO; |
| [message setOptionalNestedMessage:[TestAllTypes_NestedMessage message]]; |
| [message setOptionalNestedMessage:nil]; |
| message.hasOptionalNestedMessage = NO; |
| } |
| |
| - (void)testSettingHasToYes { |
| TestAllTypes *message = [TestAllTypes message]; |
| XCTAssertThrows([message setHasOptionalNestedMessage:YES]); |
| } |
| |
| - (void)testRoot { |
| XCTAssertNotNil([UnittestRoot extensionRegistry]); |
| } |
| |
| - (void)testGPBMessageSize { |
| // See the note in GPBMessage_PackagePrivate.h about why we want to keep the |
| // base instance size pointer size aligned. |
| size_t messageSize = class_getInstanceSize([GPBMessage class]); |
| XCTAssertEqual((messageSize % sizeof(void *)), (size_t)0, |
| @"Base size isn't pointer size aligned"); |
| |
| // Since we add storage ourselves (see +allocWithZone: in GPBMessage), confirm |
| // that the size of some generated classes is still the same as the base for |
| // that logic to work as desired. |
| size_t testMessageSize = class_getInstanceSize([TestAllTypes class]); |
| XCTAssertEqual(testMessageSize, messageSize); |
| } |
| |
| - (void)testInit { |
| TestAllTypes *message = [TestAllTypes message]; |
| [self assertClear:message]; |
| } |
| |
| - (void)testAccessors { |
| TestAllTypes *message = [TestAllTypes message]; |
| [self setAllFields:message repeatedCount:kGPBDefaultRepeatCount]; |
| [self assertAllFieldsSet:message repeatedCount:kGPBDefaultRepeatCount]; |
| } |
| |
| - (void)testKVC_ValueForKey { |
| TestAllTypes *message = [TestAllTypes message]; |
| [self setAllFields:message repeatedCount:kGPBDefaultRepeatCount]; |
| [self assertAllFieldsKVCMatch:message]; |
| } |
| |
| - (void)testKVC_SetValue_ForKey { |
| TestAllTypes *message = [TestAllTypes message]; |
| [self setAllFieldsViaKVC:message repeatedCount:kGPBDefaultRepeatCount]; |
| [self assertAllFieldsKVCMatch:message]; |
| [self assertAllFieldsSet:message repeatedCount:kGPBDefaultRepeatCount]; |
| [self assertAllFieldsKVCMatch:message]; |
| } |
| |
| - (void)testDescription { |
| // No real test, just exercise code |
| TestAllTypes *message = [TestAllTypes message]; |
| [self setAllFields:message repeatedCount:kGPBDefaultRepeatCount]; |
| |
| GPBUnknownFieldSet *unknownFields = |
| [[[GPBUnknownFieldSet alloc] init] autorelease]; |
| GPBUnknownField *field = |
| [[[GPBUnknownField alloc] initWithNumber:2] autorelease]; |
| [field addVarint:2]; |
| [unknownFields addField:field]; |
| field = [[[GPBUnknownField alloc] initWithNumber:3] autorelease]; |
| [field addVarint:4]; |
| [unknownFields addField:field]; |
| |
| [message setUnknownFields:unknownFields]; |
| |
| NSString *description = [message description]; |
| XCTAssertGreaterThan([description length], 0U); |
| |
| GPBMessage *message2 = [TestAllExtensions message]; |
| [message2 setExtension:[UnittestRoot optionalInt32Extension] value:@1]; |
| |
| [message2 addExtension:[UnittestRoot repeatedInt32Extension] value:@2]; |
| |
| description = [message2 description]; |
| XCTAssertGreaterThan([description length], 0U); |
| } |
| |
| - (void)testSetter { |
| // Test to make sure that if we set a value that has a default value |
| // with the default, that the has is set, and the value gets put into the |
| // message correctly. |
| TestAllTypes *message = [TestAllTypes message]; |
| GPBDescriptor *descriptor = [[message class] descriptor]; |
| XCTAssertNotNil(descriptor); |
| GPBFieldDescriptor *fieldDescriptor = |
| [descriptor fieldWithName:@"defaultInt32"]; |
| XCTAssertNotNil(fieldDescriptor); |
| GPBGenericValue defaultValue = [fieldDescriptor defaultValue]; |
| [message setDefaultInt32:defaultValue.valueInt32]; |
| XCTAssertTrue(message.hasDefaultInt32); |
| XCTAssertEqual(message.defaultInt32, defaultValue.valueInt32); |
| |
| // Do the same thing with an object type. |
| message = [TestAllTypes message]; |
| fieldDescriptor = [descriptor fieldWithName:@"defaultString"]; |
| XCTAssertNotNil(fieldDescriptor); |
| defaultValue = [fieldDescriptor defaultValue]; |
| [message setDefaultString:defaultValue.valueString]; |
| XCTAssertTrue(message.hasDefaultString); |
| XCTAssertEqualObjects(message.defaultString, defaultValue.valueString); |
| |
| // Test default string type. |
| message = [TestAllTypes message]; |
| XCTAssertEqualObjects(message.defaultString, @"hello"); |
| XCTAssertFalse(message.hasDefaultString); |
| fieldDescriptor = [descriptor fieldWithName:@"defaultString"]; |
| XCTAssertNotNil(fieldDescriptor); |
| defaultValue = [fieldDescriptor defaultValue]; |
| [message setDefaultString:defaultValue.valueString]; |
| XCTAssertEqualObjects(message.defaultString, @"hello"); |
| XCTAssertTrue(message.hasDefaultString); |
| [message setDefaultString:nil]; |
| XCTAssertEqualObjects(message.defaultString, @"hello"); |
| XCTAssertFalse(message.hasDefaultString); |
| message.hasDefaultString = NO; |
| XCTAssertFalse(message.hasDefaultString); |
| XCTAssertEqualObjects(message.defaultString, @"hello"); |
| |
| // Test default bytes type. |
| NSData *defaultBytes = [@"world" dataUsingEncoding:NSUTF8StringEncoding]; |
| XCTAssertEqualObjects(message.defaultBytes, defaultBytes); |
| XCTAssertFalse(message.hasDefaultString); |
| fieldDescriptor = [descriptor fieldWithName:@"defaultBytes"]; |
| XCTAssertNotNil(fieldDescriptor); |
| defaultValue = [fieldDescriptor defaultValue]; |
| [message setDefaultBytes:defaultValue.valueData]; |
| XCTAssertEqualObjects(message.defaultBytes, defaultBytes); |
| XCTAssertTrue(message.hasDefaultBytes); |
| [message setDefaultBytes:nil]; |
| XCTAssertEqualObjects(message.defaultBytes, defaultBytes); |
| XCTAssertFalse(message.hasDefaultBytes); |
| message.hasDefaultBytes = NO; |
| XCTAssertFalse(message.hasDefaultBytes); |
| XCTAssertEqualObjects(message.defaultBytes, defaultBytes); |
| |
| // Test optional string. |
| XCTAssertFalse(message.hasOptionalString); |
| XCTAssertEqualObjects(message.optionalString, @""); |
| XCTAssertFalse(message.hasOptionalString); |
| message.optionalString = nil; |
| XCTAssertFalse(message.hasOptionalString); |
| XCTAssertEqualObjects(message.optionalString, @""); |
| NSString *string = @"string"; |
| message.optionalString = string; |
| XCTAssertEqualObjects(message.optionalString, string); |
| XCTAssertTrue(message.hasOptionalString); |
| message.optionalString = nil; |
| XCTAssertFalse(message.hasOptionalString); |
| XCTAssertEqualObjects(message.optionalString, @""); |
| |
| // Test optional data. |
| XCTAssertFalse(message.hasOptionalBytes); |
| XCTAssertEqualObjects(message.optionalBytes, GPBEmptyNSData()); |
| XCTAssertFalse(message.hasOptionalBytes); |
| message.optionalBytes = nil; |
| XCTAssertFalse(message.hasOptionalBytes); |
| XCTAssertEqualObjects(message.optionalBytes, GPBEmptyNSData()); |
| NSData *data = [@"bytes" dataUsingEncoding:NSUTF8StringEncoding]; |
| message.optionalBytes = data; |
| XCTAssertEqualObjects(message.optionalBytes, data); |
| XCTAssertTrue(message.hasOptionalBytes); |
| message.optionalBytes = nil; |
| XCTAssertFalse(message.hasOptionalBytes); |
| XCTAssertEqualObjects(message.optionalBytes, GPBEmptyNSData()); |
| |
| // Test lazy message setting |
| XCTAssertFalse(message.hasOptionalLazyMessage); |
| XCTAssertNotNil(message.optionalLazyMessage); |
| XCTAssertFalse(message.hasOptionalLazyMessage); |
| message.hasOptionalLazyMessage = NO; |
| XCTAssertFalse(message.hasOptionalLazyMessage); |
| XCTAssertNotNil(message.optionalLazyMessage); |
| XCTAssertFalse(message.hasOptionalLazyMessage); |
| message.optionalLazyMessage = nil; |
| XCTAssertFalse(message.hasOptionalLazyMessage); |
| |
| // Test nested messages |
| XCTAssertFalse(message.hasOptionalLazyMessage); |
| message.optionalLazyMessage.bb = 1; |
| XCTAssertTrue(message.hasOptionalLazyMessage); |
| XCTAssertEqual(message.optionalLazyMessage.bb, 1); |
| XCTAssertNotNil(message.optionalLazyMessage); |
| message.optionalLazyMessage = nil; |
| XCTAssertFalse(message.hasOptionalLazyMessage); |
| XCTAssertEqual(message.optionalLazyMessage.bb, 0); |
| XCTAssertFalse(message.hasOptionalLazyMessage); |
| XCTAssertNotNil(message.optionalLazyMessage); |
| |
| // -testDefaultSubMessages tests the "defaulting" handling of fields |
| // containing messages. |
| } |
| |
| - (void)testRepeatedSetters { |
| TestAllTypes *message = [TestAllTypes message]; |
| [self setAllFields:message repeatedCount:kGPBDefaultRepeatCount]; |
| [self modifyRepeatedFields:message]; |
| [self assertRepeatedFieldsModified:message |
| repeatedCount:kGPBDefaultRepeatCount]; |
| } |
| |
| - (void)testClear { |
| TestAllTypes *message = [TestAllTypes message]; |
| [self setAllFields:message repeatedCount:kGPBDefaultRepeatCount]; |
| [self clearAllFields:message]; |
| [self assertClear:message]; |
| TestAllTypes *message2 = [TestAllTypes message]; |
| XCTAssertEqualObjects(message, message2); |
| } |
| |
| - (void)testClearKVC { |
| TestAllTypes *message = [TestAllTypes message]; |
| [self setAllFields:message repeatedCount:kGPBDefaultRepeatCount]; |
| [self clearAllFields:message]; |
| [self assertClear:message]; |
| [self assertClearKVC:message]; |
| } |
| |
| - (void)testClearExtension { |
| // clearExtension() is not actually used in TestUtil, so try it manually. |
| GPBMessage *message1 = [TestAllExtensions message]; |
| [message1 setExtension:[UnittestRoot optionalInt32Extension] value:@1]; |
| |
| XCTAssertTrue([message1 hasExtension:[UnittestRoot optionalInt32Extension]]); |
| [message1 clearExtension:[UnittestRoot optionalInt32Extension]]; |
| XCTAssertFalse([message1 hasExtension:[UnittestRoot optionalInt32Extension]]); |
| |
| GPBMessage *message2 = [TestAllExtensions message]; |
| [message2 addExtension:[UnittestRoot repeatedInt32Extension] value:@1]; |
| |
| XCTAssertEqual( |
| [[message2 getExtension:[UnittestRoot repeatedInt32Extension]] count], |
| (NSUInteger)1); |
| [message2 clearExtension:[UnittestRoot repeatedInt32Extension]]; |
| XCTAssertEqual( |
| [[message2 getExtension:[UnittestRoot repeatedInt32Extension]] count], |
| (NSUInteger)0); |
| |
| // Clearing an unset extension field shouldn't make the target message |
| // visible. |
| GPBMessage *message3 = [TestAllExtensions message]; |
| GPBMessage *extension_msg = |
| [message3 getExtension:[UnittestObjcRoot recursiveExtension]]; |
| XCTAssertFalse([message3 hasExtension:[UnittestObjcRoot recursiveExtension]]); |
| [extension_msg clearExtension:[UnittestRoot optionalInt32Extension]]; |
| XCTAssertFalse([message3 hasExtension:[UnittestObjcRoot recursiveExtension]]); |
| } |
| |
| - (void)testDefaultingSubMessages { |
| TestAllTypes *message = [TestAllTypes message]; |
| |
| // Initially they should all not have values. |
| |
| XCTAssertFalse(message.hasOptionalGroup); |
| XCTAssertFalse(message.hasOptionalNestedMessage); |
| XCTAssertFalse(message.hasOptionalForeignMessage); |
| XCTAssertFalse(message.hasOptionalImportMessage); |
| XCTAssertFalse(message.hasOptionalPublicImportMessage); |
| XCTAssertFalse(message.hasOptionalLazyMessage); |
| |
| // They should auto create something when fetched. |
| |
| TestAllTypes_OptionalGroup *optionalGroup = [message.optionalGroup retain]; |
| TestAllTypes_NestedMessage *optionalNestedMessage = |
| [message.optionalNestedMessage retain]; |
| ForeignMessage *optionalForeignMessage = |
| [message.optionalForeignMessage retain]; |
| ImportMessage *optionalImportMessage = [message.optionalImportMessage retain]; |
| PublicImportMessage *optionalPublicImportMessage = |
| [message.optionalPublicImportMessage retain]; |
| TestAllTypes_NestedMessage *optionalLazyMessage = |
| [message.optionalLazyMessage retain]; |
| |
| XCTAssertNotNil(optionalGroup); |
| XCTAssertNotNil(optionalNestedMessage); |
| XCTAssertNotNil(optionalForeignMessage); |
| XCTAssertNotNil(optionalImportMessage); |
| XCTAssertNotNil(optionalPublicImportMessage); |
| XCTAssertNotNil(optionalLazyMessage); |
| |
| // Although they were created, they should not respond to hasValue until that |
| // submessage is mutated. |
| |
| XCTAssertFalse(message.hasOptionalGroup); |
| XCTAssertFalse(message.hasOptionalNestedMessage); |
| XCTAssertFalse(message.hasOptionalForeignMessage); |
| XCTAssertFalse(message.hasOptionalImportMessage); |
| XCTAssertFalse(message.hasOptionalPublicImportMessage); |
| XCTAssertFalse(message.hasOptionalLazyMessage); |
| |
| // And they set that value back in to the message since the value created was |
| // mutable (so a second fetch should give the same object). |
| |
| XCTAssertEqual(message.optionalGroup, optionalGroup); |
| XCTAssertEqual(message.optionalNestedMessage, optionalNestedMessage); |
| XCTAssertEqual(message.optionalForeignMessage, optionalForeignMessage); |
| XCTAssertEqual(message.optionalImportMessage, optionalImportMessage); |
| XCTAssertEqual(message.optionalPublicImportMessage, |
| optionalPublicImportMessage); |
| XCTAssertEqual(message.optionalLazyMessage, optionalLazyMessage); |
| |
| // And the default objects for a second message should be distinct (again, |
| // since they are mutable, each needs their own copy). |
| |
| TestAllTypes *message2 = [TestAllTypes message]; |
| |
| // Intentionally doing a pointer comparison. |
| XCTAssertNotEqual(message2.optionalGroup, optionalGroup); |
| XCTAssertNotEqual(message2.optionalNestedMessage, optionalNestedMessage); |
| XCTAssertNotEqual(message2.optionalForeignMessage, optionalForeignMessage); |
| XCTAssertNotEqual(message2.optionalImportMessage, optionalImportMessage); |
| XCTAssertNotEqual(message2.optionalPublicImportMessage, |
| optionalPublicImportMessage); |
| XCTAssertNotEqual(message2.optionalLazyMessage, optionalLazyMessage); |
| |
| // Setting the values to nil will clear the has flag, and on next access you |
| // get back new submessages. |
| |
| message.optionalGroup = nil; |
| message.optionalNestedMessage = nil; |
| message.optionalForeignMessage = nil; |
| message.optionalImportMessage = nil; |
| message.optionalPublicImportMessage = nil; |
| message.optionalLazyMessage = nil; |
| |
| XCTAssertFalse(message.hasOptionalGroup); |
| XCTAssertFalse(message.hasOptionalNestedMessage); |
| XCTAssertFalse(message.hasOptionalForeignMessage); |
| XCTAssertFalse(message.hasOptionalImportMessage); |
| XCTAssertFalse(message.hasOptionalPublicImportMessage); |
| XCTAssertFalse(message.hasOptionalLazyMessage); |
| |
| // Intentionally doing a pointer comparison. |
| XCTAssertNotEqual(message.optionalGroup, optionalGroup); |
| XCTAssertNotEqual(message.optionalNestedMessage, optionalNestedMessage); |
| XCTAssertNotEqual(message.optionalForeignMessage, optionalForeignMessage); |
| XCTAssertNotEqual(message.optionalImportMessage, optionalImportMessage); |
| XCTAssertNotEqual(message.optionalPublicImportMessage, |
| optionalPublicImportMessage); |
| XCTAssertNotEqual(message.optionalLazyMessage, optionalLazyMessage); |
| |
| [optionalGroup release]; |
| [optionalNestedMessage release]; |
| [optionalForeignMessage release]; |
| [optionalImportMessage release]; |
| [optionalPublicImportMessage release]; |
| [optionalLazyMessage release]; |
| } |
| |
| - (void)testMultiplePointersToAutocreatedMessage { |
| // Multiple objects pointing to the same autocreated message. |
| TestAllTypes *message = [TestAllTypes message]; |
| TestAllTypes *message2 = [TestAllTypes message]; |
| message2.optionalGroup = message.optionalGroup; |
| XCTAssertTrue([message2 hasOptionalGroup]); |
| XCTAssertFalse([message hasOptionalGroup]); |
| message2.optionalGroup.a = 42; |
| XCTAssertTrue([message hasOptionalGroup]); |
| XCTAssertTrue([message2 hasOptionalGroup]); |
| } |
| |
| - (void)testCopyWithAutocreatedMessage { |
| // Mutable copy should not copy autocreated messages. |
| TestAllTypes *message = [TestAllTypes message]; |
| message.optionalGroup.a = 42; |
| XCTAssertNotNil(message.optionalNestedMessage); |
| TestAllTypes *message2 = [[message copy] autorelease]; |
| XCTAssertTrue([message2 hasOptionalGroup]); |
| XCTAssertFalse([message2 hasOptionalNestedMessage]); |
| |
| // Intentionally doing a pointer comparison. |
| XCTAssertNotEqual(message.optionalNestedMessage, |
| message2.optionalNestedMessage); |
| } |
| |
| - (void)testClearAutocreatedSubmessage { |
| // Call clear on an intermediate submessage should cause it to get recreated |
| // on the next call. |
| TestRecursiveMessage *message = [TestRecursiveMessage message]; |
| TestRecursiveMessage *message_inner = [message.a.a.a retain]; |
| XCTAssertNotNil(message_inner); |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(message_inner, message.a.a)); |
| [message.a.a clear]; |
| XCTAssertFalse(GPBWasMessageAutocreatedBy(message_inner, message.a.a)); |
| |
| // Intentionally doing a pointer comparison. |
| XCTAssertNotEqual(message.a.a.a, message_inner); |
| [message_inner release]; |
| } |
| |
| - (void)testRetainAutocreatedSubmessage { |
| // Should be able to retain autocreated submessage while the creator is |
| // dealloced. |
| TestAllTypes *message = [TestAllTypes message]; |
| |
| ForeignMessage *subMessage; |
| @autoreleasepool { |
| TestAllTypes *message2 = [TestAllTypes message]; |
| subMessage = message2.optionalForeignMessage; // Autocreated |
| message.optionalForeignMessage = subMessage; |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(message.optionalForeignMessage, |
| message2)); |
| } |
| |
| // Should be the same object, and should still be live. |
| XCTAssertEqual(message.optionalForeignMessage, subMessage); |
| XCTAssertNotNil([subMessage description]); |
| } |
| |
| - (void)testSetNilAutocreatedSubmessage { |
| TestRecursiveMessage *message = [TestRecursiveMessage message]; |
| TestRecursiveMessage *message_inner = [message.a.a retain]; |
| XCTAssertFalse([message hasA]); |
| XCTAssertFalse([message.a hasA]); |
| message.a.a = nil; |
| |
| // |message.a| has to be made visible, but |message.a.a| was set to nil so |
| // shouldn't be. |
| XCTAssertTrue([message hasA]); |
| XCTAssertFalse([message.a hasA]); |
| |
| // Setting submessage to nil should cause it to lose its creator. |
| XCTAssertFalse(GPBWasMessageAutocreatedBy(message_inner, message.a)); |
| |
| // After setting to nil, getting it again should create a new autocreated |
| // message. |
| // Intentionally doing a pointer comparison. |
| XCTAssertNotEqual(message.a.a, message_inner); |
| |
| [message_inner release]; |
| } |
| |
| - (void)testSetDoesntHaveAutocreatedSubmessage { |
| // Clearing submessage (set has == NO) should NOT cause it to lose its |
| // creator. |
| TestAllTypes *message = [TestAllTypes message]; |
| TestAllTypes_NestedMessage *nestedMessage = message.optionalNestedMessage; |
| XCTAssertFalse([message hasOptionalNestedMessage]); |
| [message setHasOptionalNestedMessage:NO]; |
| XCTAssertFalse([message hasOptionalNestedMessage]); |
| XCTAssertEqual(message.optionalNestedMessage, nestedMessage); |
| } |
| |
| - (void)testSetAutocreatedMessageBecomesVisible { |
| // Setting a value should cause the submessage to appear to its creator. |
| // Test this several levels deep. |
| TestRecursiveMessage *message = [TestRecursiveMessage message]; |
| message.a.a.a.a.i = 42; |
| XCTAssertTrue([message hasA]); |
| XCTAssertTrue([message.a hasA]); |
| XCTAssertTrue([message.a.a hasA]); |
| XCTAssertTrue([message.a.a.a hasA]); |
| XCTAssertFalse([message.a.a.a.a hasA]); |
| XCTAssertEqual(message.a.a.a.a.i, 42); |
| } |
| |
| - (void)testClearUnsetFieldOfAutocreatedMessage { |
| // Clearing an unset field should not cause the submessage to appear to its |
| // creator. |
| TestRecursiveMessage *message = [TestRecursiveMessage message]; |
| message.a.a.a.a.hasI = NO; |
| XCTAssertFalse([message hasA]); |
| XCTAssertFalse([message.a hasA]); |
| XCTAssertFalse([message.a.a hasA]); |
| XCTAssertFalse([message.a.a.a hasA]); |
| } |
| |
| - (void)testAutocreatedSubmessageAssignSkip { |
| TestRecursiveMessage *message = [TestRecursiveMessage message]; |
| TestRecursiveMessage *messageLevel1 = [message.a retain]; |
| TestRecursiveMessage *messageLevel2 = [message.a.a retain]; |
| TestRecursiveMessage *messageLevel3 = [message.a.a.a retain]; |
| TestRecursiveMessage *messageLevel4 = [message.a.a.a.a retain]; |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(messageLevel1, message)); |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(messageLevel2, messageLevel1)); |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(messageLevel3, messageLevel2)); |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(messageLevel4, messageLevel3)); |
| |
| // Test skipping over an autocreated submessage and ensure it gets unset. |
| message.a = message.a.a; |
| XCTAssertEqual(message.a, messageLevel2); |
| XCTAssertTrue([message hasA]); |
| XCTAssertEqual(message.a.a, messageLevel3); |
| XCTAssertFalse([message.a hasA]); |
| XCTAssertEqual(message.a.a.a, messageLevel4); |
| XCTAssertFalse(GPBWasMessageAutocreatedBy(messageLevel1, |
| message)); // Because it was orphaned. |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(messageLevel2, messageLevel1)); |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(messageLevel3, messageLevel2)); |
| |
| [messageLevel1 release]; |
| [messageLevel2 release]; |
| [messageLevel3 release]; |
| [messageLevel4 release]; |
| } |
| |
| - (void)testAutocreatedSubmessageAssignLoop { |
| TestRecursiveMessage *message = [TestRecursiveMessage message]; |
| TestRecursiveMessage *messageLevel1 = [message.a retain]; |
| TestRecursiveMessage *messageLevel2 = [message.a.a retain]; |
| TestRecursiveMessage *messageLevel3 = [message.a.a.a retain]; |
| TestRecursiveMessage *messageLevel4 = [message.a.a.a.a retain]; |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(messageLevel1, message)); |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(messageLevel2, messageLevel1)); |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(messageLevel3, messageLevel2)); |
| XCTAssertTrue(GPBWasMessageAutocreatedBy(messageLevel4, messageLevel3)); |
| |
| // Test a property with a loop. You'd never do this but at least ensure the |
| // autocreated submessages behave sanely. |
| message.a.a = message.a; |
| XCTAssertTrue([message hasA]); |
| XCTAssertEqual(message.a, messageLevel1); |
| XCTAssertTrue([message.a hasA]); |
| XCTAssertEqual(message.a.a, messageLevel1); |
| XCTAssertTrue([message.a.a hasA]); |
| XCTAssertEqual(message.a.a.a, messageLevel1); |
| XCTAssertFalse(GPBWasMessageAutocreatedBy(messageLevel1, |
| message)); // Because it was assigned. |
| XCTAssertFalse(GPBWasMessageAutocreatedBy(messageLevel2, |
| messageLevel1)); // Because it was orphaned. |
| XCTAssertFalse([messageLevel2 hasA]); |
| |
| // Break the retain loop. |
| message.a.a = nil; |
| XCTAssertTrue([message hasA]); |
| XCTAssertFalse([message.a hasA]); |
| |
| [messageLevel1 release]; |
| [messageLevel2 release]; |
| [messageLevel3 release]; |
| [messageLevel4 release]; |
| } |
| |
| - (void)testSetAutocreatedSubmessage { |
| // Setting autocreated submessage to another value should cause the old one to |
| // lose its creator. |
| TestAllTypes *message = [TestAllTypes message]; |
| TestAllTypes_NestedMessage *nestedMessage = |
| [message.optionalNestedMessage retain]; |
| |
| message.optionalNestedMessage = [TestAllTypes_NestedMessage message]; |
| XCTAssertTrue([message hasOptionalNestedMessage]); |
| XCTAssertTrue(message.optionalNestedMessage != nestedMessage); |
| XCTAssertFalse(GPBWasMessageAutocreatedBy(nestedMessage, message)); |
| |
| [nestedMessage release]; |
| } |
| |
| - (void)testAutocreatedUnknownFields { |
| // Doing anything with (except reading) unknown fields should cause the |
| // submessage to become visible. |
| TestAllTypes *message = [TestAllTypes message]; |
| XCTAssertNotNil(message.optionalNestedMessage); |
| XCTAssertFalse([message hasOptionalNestedMessage]); |
| XCTAssertNil(message.optionalNestedMessage.unknownFields); |
| XCTAssertFalse([message hasOptionalNestedMessage]); |
| |
| GPBUnknownFieldSet *unknownFields = |
| [[[GPBUnknownFieldSet alloc] init] autorelease]; |
| message.optionalNestedMessage.unknownFields = unknownFields; |
| XCTAssertTrue([message hasOptionalNestedMessage]); |
| |
| message.optionalNestedMessage = nil; |
| XCTAssertFalse([message hasOptionalNestedMessage]); |
| [message.optionalNestedMessage setUnknownFields:unknownFields]; |
| XCTAssertTrue([message hasOptionalNestedMessage]); |
| } |
| |
| - (void)testSetAutocreatedSubmessageToSelf { |
| // Setting submessage to itself should cause it to become visible. |
| TestAllTypes *message = [TestAllTypes message]; |
| XCTAssertNotNil(message.optionalNestedMessage); |
| XCTAssertFalse([message hasOptionalNestedMessage]); |
| message.optionalNestedMessage = message.optionalNestedMessage; |
| XCTAssertTrue([message hasOptionalNestedMessage]); |
| } |
| |
| - (void)testAutocreatedSubmessageMemoryLeaks { |
| // Test for memory leaks with autocreated submessages. |
| TestRecursiveMessage *message; |
| TestRecursiveMessage *messageLevel1; |
| TestRecursiveMessage *messageLevel2; |
| TestRecursiveMessage *messageLevel3; |
| TestRecursiveMessage *messageLevel4; |
| @autoreleasepool { |
| message = [[TestRecursiveMessage alloc] init]; |
| messageLevel1 = [message.a retain]; |
| messageLevel2 = [message.a.a retain]; |
| messageLevel3 = [message.a.a.a retain]; |
| messageLevel4 = [message.a.a.a.a retain]; |
| message.a.i = 1; |
| } |
| |
| XCTAssertEqual(message.retainCount, (NSUInteger)1); |
| [message release]; |
| XCTAssertEqual(messageLevel1.retainCount, (NSUInteger)1); |
| [messageLevel1 release]; |
| XCTAssertEqual(messageLevel2.retainCount, (NSUInteger)1); |
| [messageLevel2 release]; |
| XCTAssertEqual(messageLevel3.retainCount, (NSUInteger)1); |
| [messageLevel3 release]; |
| XCTAssertEqual(messageLevel4.retainCount, (NSUInteger)1); |
| [messageLevel4 release]; |
| } |
| |
| - (void)testDefaultingArrays { |
| // Basic tests for default creation of arrays in a message. |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| TestRecursiveMessageWithRepeatedField *message2 = |
| [TestRecursiveMessageWithRepeatedField message]; |
| |
| // Simply accessing the array should not make any fields visible. |
| XCTAssertNotNil(message.a.a.iArray); |
| XCTAssertFalse([message hasA]); |
| XCTAssertFalse([message.a hasA]); |
| XCTAssertNotNil(message2.a.a.strArray); |
| XCTAssertFalse([message2 hasA]); |
| XCTAssertFalse([message2.a hasA]); |
| |
| // But adding an element to the array should. |
| [message.a.a.iArray addValue:42]; |
| XCTAssertTrue([message hasA]); |
| XCTAssertTrue([message.a hasA]); |
| XCTAssertEqual([message.a.a.iArray count], (NSUInteger)1); |
| [message2.a.a.strArray addObject:@"foo"]; |
| XCTAssertTrue([message2 hasA]); |
| XCTAssertTrue([message2.a hasA]); |
| XCTAssertEqual([message2.a.a.strArray count], (NSUInteger)1); |
| } |
| |
| - (void)testAutocreatedArrayShared { |
| // Multiple objects pointing to the same array. |
| TestRecursiveMessageWithRepeatedField *message1a = |
| [TestRecursiveMessageWithRepeatedField message]; |
| TestRecursiveMessageWithRepeatedField *message1b = |
| [TestRecursiveMessageWithRepeatedField message]; |
| message1a.a.iArray = message1b.a.iArray; |
| XCTAssertTrue([message1a hasA]); |
| XCTAssertFalse([message1b hasA]); |
| [message1a.a.iArray addValue:1]; |
| XCTAssertTrue([message1a hasA]); |
| XCTAssertTrue([message1b hasA]); |
| XCTAssertEqual(message1a.a.iArray, message1b.a.iArray); |
| |
| TestRecursiveMessageWithRepeatedField *message2a = |
| [TestRecursiveMessageWithRepeatedField message]; |
| TestRecursiveMessageWithRepeatedField *message2b = |
| [TestRecursiveMessageWithRepeatedField message]; |
| message2a.a.strArray = message2b.a.strArray; |
| XCTAssertTrue([message2a hasA]); |
| XCTAssertFalse([message2b hasA]); |
| [message2a.a.strArray addObject:@"bar"]; |
| XCTAssertTrue([message2a hasA]); |
| XCTAssertTrue([message2b hasA]); |
| XCTAssertEqual(message2a.a.strArray, message2b.a.strArray); |
| } |
| |
| - (void)testAutocreatedArrayCopy { |
| // Copy should not copy autocreated arrays. |
| TestAllTypes *message = [TestAllTypes message]; |
| XCTAssertNotNil(message.repeatedStringArray); |
| XCTAssertNotNil(message.repeatedInt32Array); |
| TestAllTypes *message2 = [[message copy] autorelease]; |
| // Pointer conparisions. |
| XCTAssertNotEqual(message.repeatedStringArray, message2.repeatedStringArray); |
| XCTAssertNotEqual(message.repeatedInt32Array, message2.repeatedInt32Array); |
| |
| // Mutable copy should copy empty arrays that were explicitly set (end up |
| // with different objects that are equal). |
| TestAllTypes *message3 = [TestAllTypes message]; |
| message3.repeatedInt32Array = [GPBInt32Array arrayWithValue:42]; |
| message3.repeatedStringArray = [NSMutableArray arrayWithObject:@"wee"]; |
| XCTAssertNotNil(message.repeatedInt32Array); |
| XCTAssertNotNil(message.repeatedStringArray); |
| TestAllTypes *message4 = [[message3 copy] autorelease]; |
| XCTAssertNotEqual(message3.repeatedInt32Array, message4.repeatedInt32Array); |
| XCTAssertEqualObjects(message3.repeatedInt32Array, |
| message4.repeatedInt32Array); |
| XCTAssertNotEqual(message3.repeatedStringArray, message4.repeatedStringArray); |
| XCTAssertEqualObjects(message3.repeatedStringArray, |
| message4.repeatedStringArray); |
| } |
| |
| - (void)testAutocreatedArrayRetain { |
| // Should be able to retain autocreated array while the creator is dealloced. |
| TestAllTypes *message = [TestAllTypes message]; |
| |
| @autoreleasepool { |
| TestAllTypes *message2 = [TestAllTypes message]; |
| message.repeatedInt32Array = message2.repeatedInt32Array; |
| message.repeatedStringArray = message2.repeatedStringArray; |
| // Pointer conparision |
| XCTAssertEqual(message.repeatedInt32Array->_autocreator, message2); |
| XCTAssertTrue([message.repeatedStringArray |
| isKindOfClass:[GPBAutocreatedArray class]]); |
| XCTAssertEqual( |
| ((GPBAutocreatedArray *)message.repeatedStringArray)->_autocreator, |
| message2); |
| } |
| |
| XCTAssertNil(message.repeatedInt32Array->_autocreator); |
| XCTAssertTrue( |
| [message.repeatedStringArray isKindOfClass:[GPBAutocreatedArray class]]); |
| XCTAssertNil( |
| ((GPBAutocreatedArray *)message.repeatedStringArray)->_autocreator); |
| } |
| |
| - (void)testSetNilAutocreatedArray { |
| // Setting array to nil should cause it to lose its delegate. |
| TestAllTypes *message = [TestAllTypes message]; |
| GPBInt32Array *repeatedInt32Array = [message.repeatedInt32Array retain]; |
| GPBAutocreatedArray *repeatedStringArray = |
| (GPBAutocreatedArray *)[message.repeatedStringArray retain]; |
| XCTAssertTrue([repeatedStringArray isKindOfClass:[GPBAutocreatedArray class]]); |
| XCTAssertEqual(repeatedInt32Array->_autocreator, message); |
| XCTAssertEqual(repeatedStringArray->_autocreator, message); |
| message.repeatedInt32Array = nil; |
| message.repeatedStringArray = nil; |
| XCTAssertNil(repeatedInt32Array->_autocreator); |
| XCTAssertNil(repeatedStringArray->_autocreator); |
| [repeatedInt32Array release]; |
| [repeatedStringArray release]; |
| } |
| |
| - (void)testSetOverAutocreatedArrayAndSetAgain { |
| // Ensure when dealing with replacing an array it is handled being either |
| // an autocreated one or a straight NSArray. |
| |
| // The real test here is that nothing crashes while doing the work. |
| TestAllTypes *message = [TestAllTypes message]; |
| [message.repeatedStringArray addObject:@"foo"]; |
| XCTAssertEqual(message.repeatedStringArray_Count, (NSUInteger)1); |
| message.repeatedStringArray = [NSMutableArray arrayWithObjects:@"bar", @"bar2", nil]; |
| XCTAssertEqual(message.repeatedStringArray_Count, (NSUInteger)2); |
| message.repeatedStringArray = [NSMutableArray arrayWithObject:@"baz"]; |
| XCTAssertEqual(message.repeatedStringArray_Count, (NSUInteger)1); |
| } |
| |
| - (void)testReplaceAutocreatedArray { |
| // Replacing array should orphan the old one and cause its creator to become |
| // visible. |
| { |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| XCTAssertNotNil(message.a); |
| XCTAssertNotNil(message.a.iArray); |
| XCTAssertFalse([message hasA]); |
| GPBInt32Array *iArray = [message.a.iArray retain]; |
| XCTAssertEqual(iArray->_autocreator, message.a); // Pointer comparision |
| message.a.iArray = [GPBInt32Array arrayWithValue:1]; |
| XCTAssertTrue([message hasA]); |
| XCTAssertNotEqual(message.a.iArray, iArray); // Pointer comparision |
| XCTAssertNil(iArray->_autocreator); |
| [iArray release]; |
| } |
| |
| { |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| XCTAssertNotNil(message.a); |
| XCTAssertNotNil(message.a.strArray); |
| XCTAssertFalse([message hasA]); |
| GPBAutocreatedArray *strArray = |
| (GPBAutocreatedArray *)[message.a.strArray retain]; |
| XCTAssertTrue([strArray isKindOfClass:[GPBAutocreatedArray class]]); |
| XCTAssertEqual(strArray->_autocreator, message.a); // Pointer comparision |
| message.a.strArray = [NSMutableArray arrayWithObject:@"foo"]; |
| XCTAssertTrue([message hasA]); |
| XCTAssertNotEqual(message.a.strArray, strArray); // Pointer comparision |
| XCTAssertNil(strArray->_autocreator); |
| [strArray release]; |
| } |
| } |
| |
| - (void)testSetAutocreatedArrayToSelf { |
| // Setting array to itself should cause it to become visible. |
| { |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| XCTAssertNotNil(message.a); |
| XCTAssertNotNil(message.a.iArray); |
| XCTAssertFalse([message hasA]); |
| message.a.iArray = message.a.iArray; |
| XCTAssertTrue([message hasA]); |
| XCTAssertNil(message.a.iArray->_autocreator); |
| } |
| |
| { |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| XCTAssertNotNil(message.a); |
| XCTAssertNotNil(message.a.strArray); |
| XCTAssertFalse([message hasA]); |
| message.a.strArray = message.a.strArray; |
| XCTAssertTrue([message hasA]); |
| XCTAssertTrue([message.a.strArray isKindOfClass:[GPBAutocreatedArray class]]); |
| XCTAssertNil(((GPBAutocreatedArray *)message.a.strArray)->_autocreator); |
| } |
| } |
| |
| - (void)testAutocreatedArrayRemoveAllValues { |
| // Calling removeAllValues on autocreated array should not cause it to be |
| // visible. |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| [message.a.iArray removeAll]; |
| XCTAssertFalse([message hasA]); |
| [message.a.strArray removeAllObjects]; |
| XCTAssertFalse([message hasA]); |
| } |
| |
| - (void)testDefaultingMaps { |
| // Basic tests for default creation of maps in a message. |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| TestRecursiveMessageWithRepeatedField *message2 = |
| [TestRecursiveMessageWithRepeatedField message]; |
| |
| // Simply accessing the map should not make any fields visible. |
| XCTAssertNotNil(message.a.a.iToI); |
| XCTAssertFalse([message hasA]); |
| XCTAssertFalse([message.a hasA]); |
| XCTAssertNotNil(message2.a.a.strToStr); |
| XCTAssertFalse([message2 hasA]); |
| XCTAssertFalse([message2.a hasA]); |
| |
| // But adding an element to the map should. |
| [message.a.a.iToI setInt32:100 forKey:200]; |
| XCTAssertTrue([message hasA]); |
| XCTAssertTrue([message.a hasA]); |
| XCTAssertEqual([message.a.a.iToI count], (NSUInteger)1); |
| [message2.a.a.strToStr setObject:@"foo" forKey:@"bar"]; |
| XCTAssertTrue([message2 hasA]); |
| XCTAssertTrue([message2.a hasA]); |
| XCTAssertEqual([message2.a.a.strToStr count], (NSUInteger)1); |
| } |
| |
| - (void)testAutocreatedMapShared { |
| // Multiple objects pointing to the same map. |
| TestRecursiveMessageWithRepeatedField *message1a = |
| [TestRecursiveMessageWithRepeatedField message]; |
| TestRecursiveMessageWithRepeatedField *message1b = |
| [TestRecursiveMessageWithRepeatedField message]; |
| message1a.a.iToI = message1b.a.iToI; |
| XCTAssertTrue([message1a hasA]); |
| XCTAssertFalse([message1b hasA]); |
| [message1a.a.iToI setInt32:1 forKey:2]; |
| XCTAssertTrue([message1a hasA]); |
| XCTAssertTrue([message1b hasA]); |
| XCTAssertEqual(message1a.a.iToI, message1b.a.iToI); |
| |
| TestRecursiveMessageWithRepeatedField *message2a = |
| [TestRecursiveMessageWithRepeatedField message]; |
| TestRecursiveMessageWithRepeatedField *message2b = |
| [TestRecursiveMessageWithRepeatedField message]; |
| message2a.a.strToStr = message2b.a.strToStr; |
| XCTAssertTrue([message2a hasA]); |
| XCTAssertFalse([message2b hasA]); |
| [message2a.a.strToStr setObject:@"bar" forKey:@"foo"]; |
| XCTAssertTrue([message2a hasA]); |
| XCTAssertTrue([message2b hasA]); |
| XCTAssertEqual(message2a.a.strToStr, message2b.a.strToStr); |
| } |
| |
| - (void)testAutocreatedMapCopy { |
| // Copy should not copy autocreated maps. |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| XCTAssertNotNil(message.strToStr); |
| XCTAssertNotNil(message.iToI); |
| TestRecursiveMessageWithRepeatedField *message2 = |
| [[message copy] autorelease]; |
| // Pointer conparisions. |
| XCTAssertNotEqual(message.strToStr, message2.strToStr); |
| XCTAssertNotEqual(message.iToI, message2.iToI); |
| |
| // Mutable copy should copy empty arrays that were explicitly set (end up |
| // with different objects that are equal). |
| TestRecursiveMessageWithRepeatedField *message3 = |
| [TestRecursiveMessageWithRepeatedField message]; |
| message3.iToI = [GPBInt32Int32Dictionary dictionaryWithInt32:10 forKey:20]; |
| message3.strToStr = |
| [NSMutableDictionary dictionaryWithObject:@"abc" forKey:@"123"]; |
| XCTAssertNotNil(message.iToI); |
| XCTAssertNotNil(message.iToI); |
| TestRecursiveMessageWithRepeatedField *message4 = |
| [[message3 copy] autorelease]; |
| XCTAssertNotEqual(message3.iToI, message4.iToI); |
| XCTAssertEqualObjects(message3.iToI, message4.iToI); |
| XCTAssertNotEqual(message3.strToStr, message4.strToStr); |
| XCTAssertEqualObjects(message3.strToStr, message4.strToStr); |
| } |
| |
| - (void)testAutocreatedMapRetain { |
| // Should be able to retain autocreated map while the creator is dealloced. |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| |
| @autoreleasepool { |
| TestRecursiveMessageWithRepeatedField *message2 = |
| [TestRecursiveMessageWithRepeatedField message]; |
| message.iToI = message2.iToI; |
| message.strToStr = message2.strToStr; |
| // Pointer conparision |
| XCTAssertEqual(message.iToI->_autocreator, message2); |
| XCTAssertTrue([message.strToStr |
| isKindOfClass:[GPBAutocreatedDictionary class]]); |
| XCTAssertEqual( |
| ((GPBAutocreatedDictionary *)message.strToStr)->_autocreator, |
| message2); |
| } |
| |
| XCTAssertNil(message.iToI->_autocreator); |
| XCTAssertTrue( |
| [message.strToStr isKindOfClass:[GPBAutocreatedDictionary class]]); |
| XCTAssertNil( |
| ((GPBAutocreatedDictionary *)message.strToStr)->_autocreator); |
| } |
| |
| - (void)testSetNilAutocreatedMap { |
| // Setting map to nil should cause it to lose its delegate. |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| GPBInt32Int32Dictionary *iToI = [message.iToI retain]; |
| GPBAutocreatedDictionary *strToStr = |
| (GPBAutocreatedDictionary *)[message.strToStr retain]; |
| XCTAssertTrue([strToStr isKindOfClass:[GPBAutocreatedDictionary class]]); |
| XCTAssertEqual(iToI->_autocreator, message); |
| XCTAssertEqual(strToStr->_autocreator, message); |
| message.iToI = nil; |
| message.strToStr = nil; |
| XCTAssertNil(iToI->_autocreator); |
| XCTAssertNil(strToStr->_autocreator); |
| [iToI release]; |
| [strToStr release]; |
| } |
| |
| - (void)testSetOverAutocreatedMapAndSetAgain { |
| // Ensure when dealing with replacing a map it is handled being either |
| // an autocreated one or a straight NSDictionary. |
| |
| // The real test here is that nothing crashes while doing the work. |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| message.strToStr[@"foo"] = @"bar"; |
| XCTAssertEqual(message.strToStr_Count, (NSUInteger)1); |
| message.strToStr = |
| [NSMutableDictionary dictionaryWithObjectsAndKeys:@"bar", @"key1", @"baz", @"key2", nil]; |
| XCTAssertEqual(message.strToStr_Count, (NSUInteger)2); |
| message.strToStr = |
| [NSMutableDictionary dictionaryWithObject:@"baz" forKey:@"mumble"]; |
| XCTAssertEqual(message.strToStr_Count, (NSUInteger)1); |
| } |
| |
| - (void)testReplaceAutocreatedMap { |
| // Replacing map should orphan the old one and cause its creator to become |
| // visible. |
| { |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| XCTAssertNotNil(message.a); |
| XCTAssertNotNil(message.a.iToI); |
| XCTAssertFalse([message hasA]); |
| GPBInt32Int32Dictionary *iToI = [message.a.iToI retain]; |
| XCTAssertEqual(iToI->_autocreator, message.a); // Pointer comparision |
| message.a.iToI = [GPBInt32Int32Dictionary dictionaryWithInt32:6 forKey:7]; |
| XCTAssertTrue([message hasA]); |
| XCTAssertNotEqual(message.a.iToI, iToI); // Pointer comparision |
| XCTAssertNil(iToI->_autocreator); |
| [iToI release]; |
| } |
| |
| { |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| XCTAssertNotNil(message.a); |
| XCTAssertNotNil(message.a.strToStr); |
| XCTAssertFalse([message hasA]); |
| GPBAutocreatedDictionary *strToStr = |
| (GPBAutocreatedDictionary *)[message.a.strToStr retain]; |
| XCTAssertTrue([strToStr isKindOfClass:[GPBAutocreatedDictionary class]]); |
| XCTAssertEqual(strToStr->_autocreator, message.a); // Pointer comparision |
| message.a.strToStr = |
| [NSMutableDictionary dictionaryWithObject:@"abc" forKey:@"def"]; |
| XCTAssertTrue([message hasA]); |
| XCTAssertNotEqual(message.a.strToStr, strToStr); // Pointer comparision |
| XCTAssertNil(strToStr->_autocreator); |
| [strToStr release]; |
| } |
| } |
| |
| - (void)testSetAutocreatedMapToSelf { |
| // Setting map to itself should cause it to become visible. |
| { |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| XCTAssertNotNil(message.a); |
| XCTAssertNotNil(message.a.iToI); |
| XCTAssertFalse([message hasA]); |
| message.a.iToI = message.a.iToI; |
| XCTAssertTrue([message hasA]); |
| XCTAssertNil(message.a.iToI->_autocreator); |
| } |
| |
| { |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| XCTAssertNotNil(message.a); |
| XCTAssertNotNil(message.a.strToStr); |
| XCTAssertFalse([message hasA]); |
| message.a.strToStr = message.a.strToStr; |
| XCTAssertTrue([message hasA]); |
| XCTAssertTrue([message.a.strToStr isKindOfClass:[GPBAutocreatedDictionary class]]); |
| XCTAssertNil(((GPBAutocreatedDictionary *)message.a.strToStr)->_autocreator); |
| } |
| } |
| |
| - (void)testAutocreatedMapRemoveAllValues { |
| // Calling removeAll on autocreated map should not cause it to be visible. |
| TestRecursiveMessageWithRepeatedField *message = |
| [TestRecursiveMessageWithRepeatedField message]; |
| [message.a.iToI removeAll]; |
| XCTAssertFalse([message hasA]); |
| [message.a.strToStr removeAllObjects]; |
| XCTAssertFalse([message hasA]); |
| } |
| |
| - (void)testExtensionAccessors { |
| TestAllExtensions *message = [TestAllExtensions message]; |
| [self setAllExtensions:message repeatedCount:kGPBDefaultRepeatCount]; |
| [self assertAllExtensionsSet:message repeatedCount:kGPBDefaultRepeatCount]; |
| } |
| |
| - (void)testExtensionRepeatedSetters { |
| TestAllExtensions *message = [TestAllExtensions message]; |
| [self setAllExtensions:message repeatedCount:kGPBDefaultRepeatCount]; |
| [self modifyRepeatedExtensions:message]; |
| [self assertRepeatedExtensionsModified:message |
| repeatedCount:kGPBDefaultRepeatCount]; |
| } |
| |
| - (void)testExtensionDefaults { |
| [self assertExtensionsClear:[TestAllExtensions message]]; |
| } |
| |
| - (void)testExtensionIsEquals { |
| TestAllExtensions *message = [TestAllExtensions message]; |
| [self setAllExtensions:message repeatedCount:kGPBDefaultRepeatCount]; |
| [self modifyRepeatedExtensions:message]; |
| TestAllExtensions *message2 = [TestAllExtensions message]; |
| [self setAllExtensions:message2 repeatedCount:kGPBDefaultRepeatCount]; |
| XCTAssertFalse([message isEqual:message2]); |
| message2 = [TestAllExtensions message]; |
| [self setAllExtensions:message2 repeatedCount:kGPBDefaultRepeatCount]; |
| [self modifyRepeatedExtensions:message2]; |
| XCTAssertEqualObjects(message, message2); |
| } |
| |
| - (void)testExtensionsMergeFrom { |
| TestAllExtensions *message = [TestAllExtensions message]; |
| [self setAllExtensions:message repeatedCount:kGPBDefaultRepeatCount]; |
| [self modifyRepeatedExtensions:message]; |
| |
| message = [TestAllExtensions message]; |
| [self setAllExtensions:message repeatedCount:kGPBDefaultRepeatCount]; |
| TestAllExtensions *message2 = [TestAllExtensions message]; |
| [self modifyRepeatedExtensions:message2]; |
| [message2 mergeFrom:message]; |
| |
| XCTAssertEqualObjects(message, message2); |
| } |
| |
| - (void)testDefaultingExtensionMessages { |
| TestAllExtensions *message = [TestAllExtensions message]; |
| |
| // Initially they should all not have values. |
| |
| XCTAssertFalse([message hasExtension:[UnittestRoot optionalGroupExtension]]); |
| XCTAssertFalse([message hasExtension:[UnittestRoot optionalGroupExtension]]); |
| XCTAssertFalse( |
| [message hasExtension:[UnittestRoot optionalNestedMessageExtension]]); |
| XCTAssertFalse( |
| [message hasExtension:[UnittestRoot optionalForeignMessageExtension]]); |
| XCTAssertFalse( |
| [message hasExtension:[UnittestRoot optionalImportMessageExtension]]); |
| XCTAssertFalse([message |
| hasExtension:[UnittestRoot optionalPublicImportMessageExtension]]); |
| XCTAssertFalse( |
| [message hasExtension:[UnittestRoot optionalLazyMessageExtension]]); |
| |
| // They should auto create something when fetched. |
| |
| TestAllTypes_OptionalGroup *optionalGroup = |
| [message getExtension:[UnittestRoot optionalGroupExtension]]; |
| TestAllTypes_NestedMessage *optionalNestedMessage = |
| [message getExtension:[UnittestRoot optionalNestedMessageExtension]]; |
| ForeignMessage *optionalForeignMessage = |
| [message getExtension:[UnittestRoot optionalForeignMessageExtension]]; |
| ImportMessage *optionalImportMessage = |
| [message getExtension:[UnittestRoot optionalImportMessageExtension]]; |
| PublicImportMessage *optionalPublicImportMessage = [message |
| getExtension:[UnittestRoot optionalPublicImportMessageExtension]]; |
| TestAllTypes_NestedMessage *optionalLazyMessage = |
| [message getExtension:[UnittestRoot optionalLazyMessageExtension]]; |
| |
| XCTAssertNotNil(optionalGroup); |
| XCTAssertNotNil(optionalNestedMessage); |
| XCTAssertNotNil(optionalForeignMessage); |
| XCTAssertNotNil(optionalImportMessage); |
| XCTAssertNotNil(optionalPublicImportMessage); |
| XCTAssertNotNil(optionalLazyMessage); |
| |
| // Although it auto-created empty messages, it should not show that it has |
| // them. |
| |
| XCTAssertFalse([message hasExtension:[UnittestRoot optionalGroupExtension]]); |
| XCTAssertFalse([message hasExtension:[UnittestRoot optionalGroupExtension]]); |
| XCTAssertFalse([message hasExtension:[UnittestRoot optionalNestedMessageExtension]]); |
| XCTAssertFalse([message hasExtension:[UnittestRoot optionalForeignMessageExtension]]); |
| XCTAssertFalse([message hasExtension:[UnittestRoot optionalImportMessageExtension]]); |
| XCTAssertFalse([message hasExtension:[UnittestRoot optionalPublicImportMessageExtension]]); |
| XCTAssertFalse([message hasExtension:[UnittestRoot optionalLazyMessageExtension]]); |
| |
| // And they set that value back in to the message since the value created was |
| // mutable (so a second fetch should give the same object). |
| |
| XCTAssertEqual([message getExtension:[UnittestRoot optionalGroupExtension]], |
| optionalGroup); |
| XCTAssertEqual( |
| [message getExtension:[UnittestRoot optionalNestedMessageExtension]], |
| optionalNestedMessage); |
| XCTAssertEqual( |
| [message getExtension:[UnittestRoot optionalForeignMessageExtension]], |
| optionalForeignMessage); |
| XCTAssertEqual( |
| [message getExtension:[UnittestRoot optionalImportMessageExtension]], |
| optionalImportMessage); |
| XCTAssertEqual( |
| [message getExtension:[UnittestRoot optionalPublicImportMessageExtension]], |
| optionalPublicImportMessage); |
| XCTAssertEqual( |
| [message getExtension:[UnittestRoot optionalLazyMessageExtension]], |
| optionalLazyMessage); |
| |
| // And the default objects for a second message should be distinct (again, |
| // since they are mutable, each needs their own copy). |
| |
| TestAllExtensions *message2 = [TestAllExtensions message]; |
| |
| // Intentionally doing a pointer comparison. |
| XCTAssertNotEqual( |
| [message2 getExtension:[UnittestRoot optionalGroupExtension]], |
| optionalGroup); |
| XCTAssertNotEqual( |
| [message2 getExtension:[UnittestRoot optionalNestedMessageExtension]], |
| optionalNestedMessage); |
| XCTAssertNotEqual( |
| [message2 getExtension:[UnittestRoot optionalForeignMessageExtension]], |
| optionalForeignMessage); |
| XCTAssertNotEqual( |
| [message2 getExtension:[UnittestRoot optionalImportMessageExtension]], |
| optionalImportMessage); |
| XCTAssertNotEqual( |
| [message2 getExtension:[UnittestRoot optionalPublicImportMessageExtension]], |
| optionalPublicImportMessage); |
| XCTAssertNotEqual( |
| [message2 getExtension:[UnittestRoot optionalLazyMessageExtension]], |
| optionalLazyMessage); |
| |
| // Clear values, and on next access you get back new submessages. |
| |
| [message setExtension:[UnittestRoot optionalGroupExtension] value:nil]; |
| [message setExtension:[UnittestRoot optionalGroupExtension] value:nil]; |
| [message setExtension:[UnittestRoot optionalNestedMessageExtension] |
| value:nil]; |
| [message setExtension:[UnittestRoot optionalForeignMessageExtension] |
| value:nil]; |
| [message setExtension:[UnittestRoot optionalImportMessageExtension] |
| value:nil]; |
| [message setExtension:[UnittestRoot optionalPublicImportMessageExtension] |
| value:nil]; |
| [message setExtension:[UnittestRoot optionalLazyMessageExtension] value:nil]; |
| |
| XCTAssertFalse([message hasExtension:[UnittestRoot optionalGroupExtension]]); |
| XCTAssertFalse([message hasExtension:[UnittestRoot optionalGroupExtension]]); |
| XCTAssertFalse( |
| [message hasExtension:[UnittestRoot optionalNestedMessageExtension]]); |
| XCTAssertFalse( |
| [message hasExtension:[UnittestRoot optionalForeignMessageExtension]]); |
| XCTAssertFalse( |
| [message hasExtension:[UnittestRoot optionalImportMessageExtension]]); |
| XCTAssertFalse([message |
| hasExtension:[UnittestRoot optionalPublicImportMessageExtension]]); |
| XCTAssertFalse( |
| [message hasExtension:[UnittestRoot optionalLazyMessageExtension]]); |
| |
| XCTAssertEqual([message getExtension:[UnittestRoot optionalGroupExtension]], |
| optionalGroup); |
| XCTAssertEqual( |
| [message getExtension:[UnittestRoot optionalNestedMessageExtension]], |
| optionalNestedMessage); |
| XCTAssertEqual( |
| [message getExtension:[UnittestRoot optionalForeignMessageExtension]], |
| optionalForeignMessage); |
| XCTAssertEqual( |
| [message getExtension:[UnittestRoot optionalImportMessageExtension]], |
| optionalImportMessage); |
| XCTAssertEqual( |
| [message |
| getExtension:[UnittestRoot optionalPublicImportMessageExtension]], |
| optionalPublicImportMessage); |
| XCTAssertEqual( |
| [message getExtension:[UnittestRoot optionalLazyMessageExtension]], |
| optionalLazyMessage); |
| } |
| |
| - (void)testMultiplePointersToAutocreatedExtension { |
| // 2 objects point to the same auto-created extension. One should "has" it. |
| // The other should not. |
| TestAllExtensions *message = [TestAllExtensions message]; |
| TestAllExtensions *message2 = [TestAllExtensions message]; |
| GPBExtensionDescriptor *extension = [UnittestRoot optionalGroupExtension]; |
| [message setExtension:extension value:[message2 getExtension:extension]]; |
| XCTAssertEqual([message getExtension:extension], |
| [message2 getExtension:extension]); |
| XCTAssertFalse([message2 hasExtension:extension]); |
| XCTAssertTrue([message hasExtension:extension]); |
| |
| TestAllTypes_OptionalGroup *extensionValue = |
| [message2 getExtension:extension]; |
| extensionValue.a = 1; |
| XCTAssertTrue([message2 hasExtension:extension]); |
| XCTAssertTrue([message hasExtension:extension]); |
| } |
| |
| - (void)testCopyWithAutocreatedExtension { |
| // Mutable copy shouldn't copy autocreated extensions. |
| TestAllExtensions *message = [TestAllExtensions message]; |
| GPBExtensionDescriptor *optionalGroupExtension = |
| [UnittestRoot optionalGroupExtension]; |
| GPBExtensionDescriptor *optionalNestedMessageExtesion = |
| [UnittestRoot optionalNestedMessageExtension]; |
| TestAllTypes_OptionalGroup *optionalGroup = |
| [message getExtension:optionalGroupExtension]; |
| optionalGroup.a = 42; |
| XCTAssertNotNil(optionalGroup); |
| XCTAssertNotNil([message getExtension:optionalNestedMessageExtesion]); |
| XCTAssertTrue([message hasExtension:optionalGroupExtension]); |
| XCTAssertFalse([message hasExtension:optionalNestedMessageExtesion]); |
| |
| TestAllExtensions *message2 = [[message copy] autorelease]; |
| |
| // message2 should end up with its own copy of the optional group. |
| XCTAssertTrue([message2 hasExtension:optionalGroupExtension]); |
| XCTAssertEqualObjects([message getExtension:optionalGroupExtension], |
| [message2 getExtension:optionalGroupExtension]); |
| // Intentionally doing a pointer comparison. |
| XCTAssertNotEqual([message getExtension:optionalGroupExtension], |
| [message2 getExtension:optionalGroupExtension]); |
| |
| XCTAssertFalse([message2 hasExtension:optionalNestedMessageExtesion]); |
| // Intentionally doing a pointer comparison (auto creation should be |
| // different) |
| XCTAssertNotEqual([message getExtension:optionalNestedMessageExtesion], |
| [message2 getExtension:optionalNestedMessageExtesion]); |
| } |
| |
| - (void)testClearMessageAutocreatedExtension { |
| // Call clear should cause it to recreate its autocreated extensions. |
| TestAllExtensions *message = [TestAllExtensions message]; |
| GPBExtensionDescriptor *optionalGroupExtension = |
| [UnittestRoot optionalGroupExtension]; |
| TestAllTypes_OptionalGroup *optionalGroup = |
| [[message getExtension:optionalGroupExtension] retain]; |
| [message clear]; |
| TestAllTypes_OptionalGroup *optionalGroupNew = |
| [message getExtension:optionalGroupExtension]; |
| |
| // Intentionally doing a pointer comparison. |
| XCTAssertNotEqual(optionalGroup, optionalGroupNew); |
| [optionalGroup release]; |
| } |
| |
| - (void)testRetainAutocreatedExtension { |
| // Should be able to retain autocreated extension while the creator is |
| // dealloced. |
| TestAllExtensions *message = [TestAllExtensions message]; |
| GPBExtensionDescriptor *optionalGroupExtension = |
| [UnittestRoot optionalGroupExtension]; |
| |
| @autoreleasepool { |
| TestAllExtensions *message2 = [TestAllExtensions message]; |
| [message setExtension:optionalGroupExtension |
| value:[message2 getExtension:optionalGroupExtension]]; |
| XCTAssertTrue(GPBWasMessageAutocreatedBy( |
| [message getExtension:optionalGroupExtension], message2)); |
| } |
| |
| XCTAssertFalse(GPBWasMessageAutocreatedBy( |
| [message getExtension:optionalGroupExtension], message)); |
| } |
| |
| - (void)testClearAutocreatedExtension { |
| // Clearing autocreated extension should NOT cause it to lose its creator. |
| TestAllExtensions *message = [TestAllExtensions message]; |
| GPBExtensionDescriptor *optionalGroupExtension = |
| [UnittestRoot optionalGroupExtension]; |
| TestAllTypes_OptionalGroup *optionalGroup = |
| [[message getExtension:optionalGroupExtension] retain]; |
| [message clearExtension:optionalGroupExtension]; |
| TestAllTypes_OptionalGroup *optionalGroupNew = |
| [message getExtension:optionalGroupExtension]; |
| XCTAssertEqual(optionalGroup, optionalGroupNew); |
| XCTAssertFalse([message hasExtension:optionalGroupExtension]); |
| [optionalGroup release]; |
| |
| // Clearing autocreated extension should not cause its creator to become |
| // visible |
| GPBExtensionDescriptor *recursiveExtension = |
| [UnittestObjcRoot recursiveExtension]; |
| TestAllExtensions *message_lvl2 = [message getExtension:recursiveExtension]; |
| TestAllExtensions *message_lvl3 = |
| [message_lvl2 getExtension:recursiveExtension]; |
| [message_lvl3 clearExtension:recursiveExtension]; |
| XCTAssertFalse([message hasExtension:recursiveExtension]); |
| } |
| |
| - (void)testSetAutocreatedExtensionBecomesVisible { |
| // Setting an extension should cause the extension to appear to its creator. |
| // Test this several levels deep. |
| TestAllExtensions *message = [TestAllExtensions message]; |
| GPBExtensionDescriptor *recursiveExtension = |
| [UnittestObjcRoot recursiveExtension]; |
| TestAllExtensions *message_lvl2 = [message getExtension:recursiveExtension]; |
| TestAllExtensions *message_lvl3 = |
| [message_lvl2 getExtension:recursiveExtension]; |
| TestAllExtensions *message_lvl4 = |
| [message_lvl3 getExtension:recursiveExtension]; |
| XCTAssertFalse([message hasExtension:recursiveExtension]); |
| XCTAssertFalse([message_lvl2 hasExtension:recursiveExtension]); |
| XCTAssertFalse([message_lvl3 hasExtension:recursiveExtension]); |
| XCTAssertFalse([message_lvl4 hasExtension:recursiveExtension]); |
| [message_lvl4 setExtension:[UnittestRoot optionalInt32Extension] value:@(1)]; |
| XCTAssertTrue([message hasExtension:recursiveExtension]); |
| XCTAssertTrue([message_lvl2 hasExtension:recursiveExtension]); |
| XCTAssertTrue([message_lvl3 hasExtension:recursiveExtension]); |
| XCTAssertFalse([message_lvl4 hasExtension:recursiveExtension]); |
| XCTAssertFalse(GPBWasMessageAutocreatedBy(message_lvl4, message_lvl3)); |
| XCTAssertFalse(GPBWasMessageAutocreatedBy(message_lvl3, message_lvl2)); |
| XCTAssertFalse(GPBWasMessageAutocreatedBy(message_lvl2, message)); |
| } |
| |
| - (void)testSetAutocreatedExtensionToSelf { |
| // Setting extension to itself should cause it to become visible. |
| TestAllExtensions *message = [TestAllExtensions message]; |
| GPBExtensionDescriptor *optionalGroupExtension = |
| [UnittestRoot optionalGroupExtension]; |
| XCTAssertNotNil([message getExtension:optionalGroupExtension]); |
| XCTAssertFalse([message hasExtension:optionalGroupExtension]); |
| [message setExtension:optionalGroupExtension |
| value:[message getExtension:optionalGroupExtension]]; |
| XCTAssertTrue([message hasExtension:optionalGroupExtension]); |
| } |
| |
| - (void)testAutocreatedExtensionMemoryLeaks { |
| GPBExtensionDescriptor *recursiveExtension = |
| [UnittestObjcRoot recursiveExtension]; |
| |
| // Test for memory leaks with autocreated extensions. |
| TestAllExtensions *message; |
| TestAllExtensions *message_lvl2; |
| TestAllExtensions *message_lvl3; |
| TestAllExtensions *message_lvl4; |
| @autoreleasepool { |
| message = [[TestAllExtensions alloc] init]; |
| message_lvl2 = [[message getExtension:recursiveExtension] retain]; |
| message_lvl3 = [[message_lvl2 getExtension:recursiveExtension] retain]; |
| message_lvl4 = [[message_lvl3 getExtension:recursiveExtension] retain]; |
| [message_lvl2 setExtension:[UnittestRoot optionalInt32Extension] |
| value:@(1)]; |
| } |
| |
| XCTAssertEqual(message.retainCount, (NSUInteger)1); |
| @autoreleasepool { |
| [message release]; |
| } |
| XCTAssertEqual(message_lvl2.retainCount, (NSUInteger)1); |
| @autoreleasepool { |
| [message_lvl2 release]; |
| } |
| XCTAssertEqual(message_lvl3.retainCount, (NSUInteger)1); |
| @autoreleasepool { |
| [message_lvl3 release]; |
| } |
| XCTAssertEqual(message_lvl4.retainCount, (NSUInteger)1); |
| [message_lvl4 release]; |
| } |
| |
| - (void)testSetExtensionWithAutocreatedValue { |
| GPBExtensionDescriptor *recursiveExtension = |
| [UnittestObjcRoot recursiveExtension]; |
| |
| TestAllExtensions *message; |
| @autoreleasepool { |
| message = [[TestAllExtensions alloc] init]; |
| [message getExtension:recursiveExtension]; |
| } |
| |
| // This statements checks that the extension value isn't accidentally |
| // dealloced when removing it from the autocreated map. |
| [message setExtension:recursiveExtension |
| value:[message getExtension:recursiveExtension]]; |
| XCTAssertTrue([message hasExtension:recursiveExtension]); |
| [message release]; |
| } |
| |
| - (void)testRecursion { |
| TestRecursiveMessage *message = [TestRecursiveMessage message]; |
| XCTAssertNotNil(message.a); |
| XCTAssertNotNil(message.a.a); |
| XCTAssertEqual(message.a.a.i, 0); |
| } |
| |
| - (void)testGenerateAndParseUnknownMessage { |
| GPBUnknownFieldSet *unknowns = |
| [[[GPBUnknownFieldSet alloc] init] autorelease]; |
| [unknowns mergeVarintField:123 value:456]; |
| GPBMessage *message = [GPBMessage message]; |
| [message setUnknownFields:unknowns]; |
| NSData *data = [message data]; |
| GPBMessage *message2 = |
| [GPBMessage parseFromData:data extensionRegistry:nil error:NULL]; |
| XCTAssertEqualObjects(message, message2); |
| } |
| |
| - (void)testDelimitedWriteAndParseMultipleMessages { |
| GPBUnknownFieldSet *unknowns1 = |
| [[[GPBUnknownFieldSet alloc] init] autorelease]; |
| [unknowns1 mergeVarintField:123 value:456]; |
| GPBMessage *message1 = [GPBMessage message]; |
| [message1 setUnknownFields:unknowns1]; |
| |
| GPBUnknownFieldSet *unknowns2 = |
| [[[GPBUnknownFieldSet alloc] init] autorelease]; |
| [unknowns2 mergeVarintField:789 value:987]; |
| [unknowns2 mergeVarintField:654 value:321]; |
| GPBMessage *message2 = [GPBMessage message]; |
| [message2 setUnknownFields:unknowns2]; |
| |
| NSMutableData *delimitedData = [NSMutableData data]; |
| [delimitedData appendData:[message1 delimitedData]]; |
| [delimitedData appendData:[message2 delimitedData]]; |
| GPBCodedInputStream *input = |
| [GPBCodedInputStream streamWithData:delimitedData]; |
| GPBMessage *message3 = [GPBMessage parseDelimitedFromCodedInputStream:input |
| extensionRegistry:nil |
| error:NULL]; |
| GPBMessage *message4 = [GPBMessage parseDelimitedFromCodedInputStream:input |
| extensionRegistry:nil |
| error:NULL]; |
| XCTAssertEqualObjects(message1, message3); |
| XCTAssertEqualObjects(message2, message4); |
| } |
| |
| - (void)testDuplicateEnums { |
| XCTAssertEqual(TestEnumWithDupValue_Foo1, TestEnumWithDupValue_Foo2); |
| } |
| |
| - (void)testWeirdDefaults { |
| ObjcWeirdDefaults *message = [ObjcWeirdDefaults message]; |
| GPBDescriptor *descriptor = [[message class] descriptor]; |
| GPBFieldDescriptor *fieldDesc = [descriptor fieldWithName:@"foo"]; |
| XCTAssertNotNil(fieldDesc); |
| XCTAssertTrue(fieldDesc.hasDefaultValue); |
| XCTAssertFalse(message.hasFoo); |
| XCTAssertEqualObjects(message.foo, @""); |
| |
| fieldDesc = [descriptor fieldWithName:@"bar"]; |
| XCTAssertNotNil(fieldDesc); |
| XCTAssertTrue(fieldDesc.hasDefaultValue); |
| XCTAssertFalse(message.hasBar); |
| XCTAssertEqualObjects(message.bar, GPBEmptyNSData()); |
| } |
| |
| - (void)testEnumDescriptorFromExtensionDescriptor { |
| GPBExtensionDescriptor *extDescriptor = |
| [UnittestRoot optionalForeignEnumExtension]; |
| XCTAssertEqual(extDescriptor.dataType, GPBDataTypeEnum); |
| GPBEnumDescriptor *enumDescriptor = extDescriptor.enumDescriptor; |
| GPBEnumDescriptor *expectedDescriptor = ForeignEnum_EnumDescriptor(); |
| XCTAssertEqualObjects(enumDescriptor, expectedDescriptor); |
| } |
| |
| - (void)testPropertyNaming { |
| // objectivec_helpers.cc has some special handing to get proper all caps |
| // for a few cases to meet objc developer expectations. |
| // |
| // This "test" confirms that the expected names are generated, otherwise the |
| // test itself will fail to compile. |
| ObjCPropertyNaming *msg = [ObjCPropertyNaming message]; |
| // On their own, at the end, in the middle. |
| msg.URL = @"good"; |
| msg.thumbnailURL = @"good"; |
| msg.URLFoo = @"good"; |
| msg.someURLBlah = @"good"; |
| msg.HTTP = @"good"; |
| msg.HTTPS = @"good"; |
| // No caps since it was "urls". |
| [msg.urlsArray addObject:@"good"]; |
| } |
| |
| - (void)testEnumNaming { |
| // objectivec_helpers.cc has some interesting cases to deal with in |
| // EnumValueName/EnumValueShortName. Confirm that things generated as |
| // expected. |
| |
| // This block just has to compile to confirm we got the expected types/names. |
| // The *_IsValidValue() calls are just there to keep the projects warnings |
| // flags happy by providing use of the variables/values. |
| |
| Foo aFoo = Foo_SerializedSize; |
| Foo_IsValidValue(aFoo); |
| aFoo = Foo_Size; |
| Foo_IsValidValue(aFoo); |
| |
| Category_Enum aCat = Category_Enum_Red; |
| Category_Enum_IsValidValue(aCat); |
| |
| Time aTime = Time_Base; |
| Time_IsValidValue(aTime); |
| aTime = Time_SomethingElse; |
| Time_IsValidValue(aTime); |
| |
| // This block confirms the names in the decriptors is what we wanted. |
| |
| GPBEnumDescriptor *descriptor; |
| NSString *valueName; |
| |
| descriptor = Foo_EnumDescriptor(); |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqualObjects(@"Foo", descriptor.name); |
| valueName = [descriptor enumNameForValue:Foo_SerializedSize]; |
| XCTAssertEqualObjects(@"Foo_SerializedSize", valueName); |
| valueName = [descriptor enumNameForValue:Foo_Size]; |
| XCTAssertEqualObjects(@"Foo_Size", valueName); |
| |
| descriptor = Category_Enum_EnumDescriptor(); |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqualObjects(@"Category_Enum", descriptor.name); |
| valueName = [descriptor enumNameForValue:Category_Enum_Red]; |
| XCTAssertEqualObjects(@"Category_Enum_Red", valueName); |
| |
| descriptor = Time_EnumDescriptor(); |
| XCTAssertNotNil(descriptor); |
| XCTAssertEqualObjects(@"Time", descriptor.name); |
| valueName = [descriptor enumNameForValue:Time_Base]; |
| XCTAssertEqualObjects(@"Time_Base", valueName); |
| valueName = [descriptor enumNameForValue:Time_SomethingElse]; |
| XCTAssertEqualObjects(@"Time_SomethingElse", valueName); |
| } |
| |
| - (void)testNegativeEnums { |
| EnumTestMsg *msg = [EnumTestMsg message]; |
| |
| // Defaults |
| XCTAssertEqual(msg.foo, EnumTestMsg_MyEnum_Zero); |
| XCTAssertEqual(msg.bar, EnumTestMsg_MyEnum_One); |
| XCTAssertEqual(msg.baz, EnumTestMsg_MyEnum_NegOne); |
| // Bounce to wire and back. |
| NSData *data = [msg data]; |
| XCTAssertNotNil(data); |
| EnumTestMsg *msgPrime = [EnumTestMsg parseFromData:data error:NULL]; |
| XCTAssertEqualObjects(msgPrime, msg); |
| XCTAssertEqual(msgPrime.foo, EnumTestMsg_MyEnum_Zero); |
| XCTAssertEqual(msgPrime.bar, EnumTestMsg_MyEnum_One); |
| XCTAssertEqual(msgPrime.baz, EnumTestMsg_MyEnum_NegOne); |
| |
| // Other values |
| msg.bar = EnumTestMsg_MyEnum_Two; |
| msg.baz = EnumTestMsg_MyEnum_NegTwo; |
| XCTAssertEqual(msg.bar, EnumTestMsg_MyEnum_Two); |
| XCTAssertEqual(msg.baz, EnumTestMsg_MyEnum_NegTwo); |
| // Bounce to wire and back. |
| data = [msg data]; |
| XCTAssertNotNil(data); |
| msgPrime = [EnumTestMsg parseFromData:data error:NULL]; |
| XCTAssertEqualObjects(msgPrime, msg); |
| XCTAssertEqual(msgPrime.foo, EnumTestMsg_MyEnum_Zero); |
| XCTAssertEqual(msgPrime.bar, EnumTestMsg_MyEnum_Two); |
| XCTAssertEqual(msgPrime.baz, EnumTestMsg_MyEnum_NegTwo); |
| |
| // Repeated field (shouldn't ever be an issue since developer has to use the |
| // right GPBArray methods themselves). |
| msg.mumbleArray = [GPBEnumArray |
| arrayWithValidationFunction:EnumTestMsg_MyEnum_IsValidValue]; |
| [msg.mumbleArray addValue:EnumTestMsg_MyEnum_Zero]; |
| [msg.mumbleArray addValue:EnumTestMsg_MyEnum_One]; |
| [msg.mumbleArray addValue:EnumTestMsg_MyEnum_Two]; |
| [msg.mumbleArray addValue:EnumTestMsg_MyEnum_NegOne]; |
| [msg.mumbleArray addValue:EnumTestMsg_MyEnum_NegTwo]; |
| XCTAssertEqual([msg.mumbleArray valueAtIndex:0], EnumTestMsg_MyEnum_Zero); |
| XCTAssertEqual([msg.mumbleArray valueAtIndex:1], EnumTestMsg_MyEnum_One); |
| XCTAssertEqual([msg.mumbleArray valueAtIndex:2], EnumTestMsg_MyEnum_Two); |
| XCTAssertEqual([msg.mumbleArray valueAtIndex:3], EnumTestMsg_MyEnum_NegOne); |
| XCTAssertEqual([msg.mumbleArray valueAtIndex:4], EnumTestMsg_MyEnum_NegTwo); |
| // Bounce to wire and back. |
| data = [msg data]; |
| XCTAssertNotNil(data); |
| msgPrime = [EnumTestMsg parseFromData:data error:NULL]; |
| XCTAssertEqualObjects(msgPrime, msg); |
| XCTAssertEqual([msgPrime.mumbleArray valueAtIndex:0], |
| EnumTestMsg_MyEnum_Zero); |
| XCTAssertEqual([msgPrime.mumbleArray valueAtIndex:1], EnumTestMsg_MyEnum_One); |
| XCTAssertEqual([msgPrime.mumbleArray valueAtIndex:2], EnumTestMsg_MyEnum_Two); |
| XCTAssertEqual([msgPrime.mumbleArray valueAtIndex:3], |
| EnumTestMsg_MyEnum_NegOne); |
| XCTAssertEqual([msgPrime.mumbleArray valueAtIndex:4], |
| EnumTestMsg_MyEnum_NegTwo); |
| } |
| |
| - (void)testOneBasedEnumHolder { |
| // Test case for https://github.com/google/protobuf/issues/1453 |
| // Message with no explicit defaults, but a non zero default for an enum. |
| MessageWithOneBasedEnum *enumMsg = [MessageWithOneBasedEnum message]; |
| XCTAssertEqual(enumMsg.enumField, MessageWithOneBasedEnum_OneBasedEnum_One); |
| } |
| |
| - (void)testBoolOffsetUsage { |
| // Bools use storage within has_bits; this test ensures that this is honored |
| // in all places where things should crash or fail based on reading out of |
| // field storage instead. |
| BoolOnlyMessage *msg1 = [BoolOnlyMessage message]; |
| BoolOnlyMessage *msg2 = [BoolOnlyMessage message]; |
| |
| msg1.boolField1 = YES; |
| msg2.boolField1 = YES; |
| msg1.boolField3 = YES; |
| msg2.boolField3 = YES; |
| msg1.boolField5 = YES; |
| msg2.boolField5 = YES; |
| msg1.boolField7 = YES; |
| msg2.boolField7 = YES; |
| msg1.boolField9 = YES; |
| msg2.boolField9 = YES; |
| msg1.boolField11 = YES; |
| msg2.boolField11 = YES; |
| msg1.boolField13 = YES; |
| msg2.boolField13 = YES; |
| msg1.boolField15 = YES; |
| msg2.boolField15 = YES; |
| msg1.boolField17 = YES; |
| msg2.boolField17 = YES; |
| msg1.boolField19 = YES; |
| msg2.boolField19 = YES; |
| msg1.boolField21 = YES; |
| msg2.boolField21 = YES; |
| msg1.boolField23 = YES; |
| msg2.boolField23 = YES; |
| msg1.boolField25 = YES; |
| msg2.boolField25 = YES; |
| msg1.boolField27 = YES; |
| msg2.boolField27 = YES; |
| msg1.boolField29 = YES; |
| msg2.boolField29 = YES; |
| msg1.boolField31 = YES; |
| msg2.boolField31 = YES; |
| |
| msg1.boolField32 = YES; |
| msg2.boolField32 = YES; |
| |
| XCTAssertTrue(msg1 != msg2); // Different pointers. |
| XCTAssertEqual([msg1 hash], [msg2 hash]); |
| XCTAssertEqualObjects(msg1, msg2); |
| |
| BoolOnlyMessage *msg1Prime = [[msg1 copy] autorelease]; |
| XCTAssertTrue(msg1Prime != msg1); // Different pointers. |
| XCTAssertEqual([msg1 hash], [msg1Prime hash]); |
| XCTAssertEqualObjects(msg1, msg1Prime); |
| |
| // Field set in one, but not the other means they don't match (even if |
| // set to default value). |
| msg1Prime.boolField2 = NO; |
| XCTAssertNotEqualObjects(msg1Prime, msg1); |
| // And when set to different values. |
| msg1.boolField2 = YES; |
| XCTAssertNotEqualObjects(msg1Prime, msg1); |
| // And then they match again. |
| msg1.boolField2 = NO; |
| XCTAssertEqualObjects(msg1Prime, msg1); |
| XCTAssertEqual([msg1 hash], [msg1Prime hash]); |
| } |
| |
| @end |