blob: b2c75ba0c25315869113999496254ce605ffc001 [file] [log] [blame]
// 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"
#import "google/protobuf/UnittestObjcOptions.pbobjc.h"
#import "google/protobuf/UnittestImport.pbobjc.h"
// Helper class to test KVO.
@interface GPBKVOTestObserver : NSObject {
id observee_;
NSString *keyPath_;
}
@property (nonatomic) BOOL didObserve;
- (id)initWithObservee:(id)observee keyPath:(NSString *)keyPath;
@end
@implementation GPBKVOTestObserver
@synthesize didObserve;
- (id)initWithObservee:(id)observee keyPath:(NSString *)keyPath {
if (self = [super init]) {
observee_ = [observee retain];
keyPath_ = [keyPath copy];
[observee_ addObserver:self forKeyPath:keyPath_ options:0 context:NULL];
}
return self;
}
- (void)dealloc {
[observee_ removeObserver:self forKeyPath:keyPath_];
[observee_ release];
[keyPath_ release];
[super dealloc];
}
- (void)observeValueForKeyPath:(NSString *)keyPath
ofObject:(id)object
change:(NSDictionary *)change
context:(void *)context
{
#pragma unused(object)
#pragma unused(change)
#pragma unused(context)
if ([keyPath isEqualToString:keyPath_]) {
self.didObserve = YES;
}
}
@end
@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
}
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
- (void)testCoding {
GPBMessage *original = [self mergeResult];
NSData *data =
[NSKeyedArchiver archivedDataWithRootObject:original];
id unarchivedObject = [NSKeyedUnarchiver unarchiveObjectWithData:data];
XCTAssertEqualObjects(unarchivedObject, original);
// Intentionally doing a pointer comparison.
XCTAssertNotEqual(unarchivedObject, original);
}
- (void)testSecureCoding {
GPBMessage *original = [self mergeResult];
NSString *key = @"testing123";
NSMutableData *data = [NSMutableData data];
NSKeyedArchiver *archiver = [[NSKeyedArchiver alloc] initForWritingWithMutableData:data];
[archiver setRequiresSecureCoding:YES];
[archiver encodeObject:original forKey:key];
[archiver finishEncoding];
NSKeyedUnarchiver *unarchiver = [[NSKeyedUnarchiver alloc] initForReadingWithData:data];
[unarchiver setRequiresSecureCoding:YES];
id unarchivedObject = [unarchiver decodeObjectOfClass:[GPBMessage class]
forKey:key];
[unarchiver finishDecoding];
XCTAssertEqualObjects(unarchivedObject, original);
// Intentionally doing a pointer comparison.
XCTAssertNotEqual(unarchivedObject, original);
}
#pragma clang diagnostic pop
- (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)testKVOBasic {
TestAllTypes *message = [TestAllTypes message];
GPBKVOTestObserver *observer =
[[[GPBKVOTestObserver alloc] initWithObservee:message
keyPath:@"optionalString"]
autorelease];
XCTAssertFalse(observer.didObserve);
message.defaultString = @"Hello";
XCTAssertFalse(observer.didObserve);
message.optionalString = @"Hello";
XCTAssertTrue(observer.didObserve);
}
- (void)testKVOAutocreate {
TestAllTypes *message = [TestAllTypes message];
GPBKVOTestObserver *autocreateObserver =
[[[GPBKVOTestObserver alloc] initWithObservee:message
keyPath:@"optionalImportMessage"]
autorelease];
GPBKVOTestObserver *innerFieldObserver =
[[[GPBKVOTestObserver alloc] initWithObservee:message
keyPath:@"optionalImportMessage.d"]
autorelease];
XCTAssertFalse(autocreateObserver.didObserve);
XCTAssertFalse(innerFieldObserver.didObserve);
int a = message.optionalImportMessage.d;
XCTAssertEqual(a, 0);
// Autocreation of fields is not observed by KVO when getting values.
XCTAssertFalse(autocreateObserver.didObserve);
XCTAssertFalse(innerFieldObserver.didObserve);
message.optionalImportMessage.d = 2;
// Autocreation of fields is not observed by KVO.
// This is undefined behavior. The library makes no guarantees with regards
// to KVO firing if an autocreation occurs as part of a setter.
// This test exists just to be aware if the behavior changes.
XCTAssertFalse(autocreateObserver.didObserve);
// Values set inside of an autocreated field are observed.
XCTAssertTrue(innerFieldObserver.didObserve);
// Explicit setting of a message field is observed.
message.optionalImportMessage = [ImportMessage message];
XCTAssertTrue(autocreateObserver.didObserve);
}
- (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 comparison
message.a.iArray = [GPBInt32Array arrayWithValue:1];
XCTAssertTrue([message hasA]);
XCTAssertNotEqual(message.a.iArray, iArray); // Pointer comparison
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 comparison
message.a.strArray = [NSMutableArray arrayWithObject:@"foo"];
XCTAssertTrue([message hasA]);
XCTAssertNotEqual(message.a.strArray, strArray); // Pointer comparison
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 alloc] init] autorelease];
[message3.iToI setInt32: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 comparison
message.a.iToI = [[[GPBInt32Int32Dictionary alloc] init] autorelease];
[message.a.iToI setInt32:6 forKey:7];
XCTAssertTrue([message hasA]);
XCTAssertNotEqual(message.a.iToI, iToI); // Pointer comparison
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 comparison
message.a.strToStr =
[NSMutableDictionary dictionaryWithObject:@"abc" forKey:@"def"];
XCTAssertTrue([message hasA]);
XCTAssertNotEqual(message.a.strToStr, strToStr); // Pointer comparison
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 descriptors 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)testReservedWordNaming {
// objectivec_helpers.cc has some special handing to make sure that
// some "reserved" objc names get renamed in a way so they
// don't conflict.
//
// This "test" confirms that the expected names are generated,
// otherwise the test itself will fail to compile.
self_Class *msg = [self_Class message];
// Some ObjC/C/C++ keywords.
msg.className_p = msg.hasClassName_p;
msg.cmd = msg.hasCmd;
msg.nullable_p = msg.hasNullable_p;
msg.typeof_p = msg.hasTypeof_p;
msg.instancetype_p = msg.hasInstancetype_p;
msg.nil_p = msg.hasNil_p;
msg.instancetype_p = msg.hasInstancetype_p;
msg.public_p = msg.hasPublic_p;
// Some that would override NSObject methods
msg.camltype = msg.hasCamltype;
msg.isNsdictionary = msg.hasIsNsdictionary;
msg.dealloc_p = msg.hasDealloc_p;
msg.zone_p = msg.hasZone_p;
msg.accessibilityLabel_p = msg.hasAccessibilityLabel_p;
// Some that we shouldn't need to handle.
msg.atomic = msg.hasAtomic;
msg.nonatomic = msg.hasNonatomic;
msg.strong = msg.hasStrong;
msg.nullResettable = msg.hasNullResettable;
// Some that would override GPBMessage methods
msg.clear_p = msg.hasClear_p;
msg.data_p = msg.hasData_p;
// Some MacTypes
msg.fixed = msg.hasFixed;
msg.style = msg.hasStyle;
// Some C Identifiers
msg.generic = msg.hasGeneric;
msg.block = msg.hasBlock;
}
- (void)testOneBasedEnumHolder {
// Test case for https://github.com/protocolbuffers/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]);
}
- (void)testCopyingMapFields {
TestMessageOfMaps *msg = [TestMessageOfMaps message];
msg.strToStr[@"foo"] = @"bar";
[msg.strToInt setInt32:1 forKey:@"mumble"];
[msg.intToStr setObject:@"wee" forKey:42];
[msg.intToInt setInt32:123 forKey:321];
[msg.strToBool setBool:YES forKey:@"one"];
[msg.boolToStr setObject:@"something" forKey:YES];
[msg.boolToBool setBool:YES forKey:NO];
[msg.intToBool setBool:YES forKey:13];
[msg.boolToInt setInt32:111 forKey:NO];
TestAllTypes *subMsg1 = [TestAllTypes message];
subMsg1.optionalInt32 = 1;
TestAllTypes *subMsg2 = [TestAllTypes message];
subMsg1.optionalInt32 = 2;
TestAllTypes *subMsg3 = [TestAllTypes message];
subMsg1.optionalInt32 = 3;
msg.strToMsg[@"baz"] = subMsg1;
[msg.intToMsg setObject:subMsg2 forKey:222];
[msg.boolToMsg setObject:subMsg3 forKey:YES];
TestMessageOfMaps *msg2 = [[msg copy] autorelease];
XCTAssertNotNil(msg2);
XCTAssertEqualObjects(msg2, msg);
XCTAssertTrue(msg2 != msg); // ptr compare
XCTAssertTrue(msg.strToStr != msg2.strToStr); // ptr compare
XCTAssertTrue(msg.intToStr != msg2.intToStr); // ptr compare
XCTAssertTrue(msg.intToInt != msg2.intToInt); // ptr compare
XCTAssertTrue(msg.strToBool != msg2.strToBool); // ptr compare
XCTAssertTrue(msg.boolToStr != msg2.boolToStr); // ptr compare
XCTAssertTrue(msg.boolToBool != msg2.boolToBool); // ptr compare
XCTAssertTrue(msg.intToBool != msg2.intToBool); // ptr compare
XCTAssertTrue(msg.boolToInt != msg2.boolToInt); // ptr compare
XCTAssertTrue(msg.strToMsg != msg2.strToMsg); // ptr compare
XCTAssertTrue(msg.intToMsg != msg2.intToMsg); // ptr compare
XCTAssertTrue(msg.boolToMsg != msg2.boolToMsg); // ptr compare
XCTAssertTrue(msg.strToMsg[@"baz"] != msg2.strToMsg[@"baz"]); // ptr compare
XCTAssertEqualObjects(msg.strToMsg[@"baz"], msg2.strToMsg[@"baz"]);
XCTAssertTrue([msg.intToMsg objectForKey:222] != [msg2.intToMsg objectForKey:222]); // ptr compare
XCTAssertEqualObjects([msg.intToMsg objectForKey:222], [msg2.intToMsg objectForKey:222]);
XCTAssertTrue([msg.boolToMsg objectForKey:YES] != [msg2.boolToMsg objectForKey:YES]); // ptr compare
XCTAssertEqualObjects([msg.boolToMsg objectForKey:YES], [msg2.boolToMsg objectForKey:YES]);
}
- (void)testPrefixedNames {
// The fact that this compiles is sufficient as a test.
// The assertions are just there to avoid "not-used" warnings.
// Verify that enum types and values get the prefix.
GPBTESTTestObjcProtoPrefixEnum value = GPBTESTTestObjcProtoPrefixEnum_Value;
XCTAssertNotEqual(value, 0);
// Verify that roots get the prefix.
GPBTESTUnittestObjcOptionsRoot *root = nil;
XCTAssertNil(root);
// Verify that messages that don't already have the prefix get a prefix.
GPBTESTTestObjcProtoPrefixMessage *prefixedMessage = nil;
XCTAssertNil(prefixedMessage);
// Verify that messages that already have a prefix aren't prefixed twice.
GPBTESTTestHasAPrefixMessage *alreadyPrefixedMessage = nil;
XCTAssertNil(alreadyPrefixedMessage);
// Verify that enums that already have a prefix aren't prefixed twice.
GPBTESTTestHasAPrefixEnum prefixedValue = GPBTESTTestHasAPrefixEnum_ValueB;
XCTAssertNotEqual(prefixedValue, 0);
// Verify that classes named the same as prefixes are prefixed.
GPBTESTGPBTEST *prefixMessage = nil;
XCTAssertNil(prefixMessage);
// Verify that classes that have the prefix followed by a lowercase
// letter DO get the prefix.
GPBTESTGPBTESTshouldGetAPrefixMessage *shouldGetAPrefixMessage = nil;
XCTAssertNil(shouldGetAPrefixMessage);
}
@end