GPBDictionaryTests.pddm 46 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044
  1. // Protocol Buffers - Google's data interchange format
  2. // Copyright 2015 Google Inc. All rights reserved.
  3. // https://developers.google.com/protocol-buffers/
  4. //
  5. // Redistribution and use in source and binary forms, with or without
  6. // modification, are permitted provided that the following conditions are
  7. // met:
  8. //
  9. // * Redistributions of source code must retain the above copyright
  10. // notice, this list of conditions and the following disclaimer.
  11. // * Redistributions in binary form must reproduce the above
  12. // copyright notice, this list of conditions and the following disclaimer
  13. // in the documentation and/or other materials provided with the
  14. // distribution.
  15. // * Neither the name of Google Inc. nor the names of its
  16. // contributors may be used to endorse or promote products derived from
  17. // this software without specific prior written permission.
  18. //
  19. // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  20. // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  21. // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  22. // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  23. // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  24. // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  25. // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  26. // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  27. // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  28. // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  29. // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  30. //%PDDM-DEFINE TEST_FOR_POD_KEY(KEY_NAME, KEY_TYPE, KEY1, KEY2, KEY3, KEY4)
  31. //%TESTS_FOR_POD_VALUES(KEY_NAME, KEY_TYPE, , , KEY1, KEY2, KEY3, KEY4)
  32. //%TESTS_FOR_POD_KEY_OBJECT_VALUE(KEY_NAME, KEY_TYPE, KEY1, KEY2, KEY3, KEY4, Object, id, @"abc", @"def", @"ghi", @"jkl")
  33. //%PDDM-DEFINE TESTS_FOR_POD_VALUES(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4)
  34. //%TEST_HELPERS(KEY_NAME, KEY_TYPE, KisP)
  35. //%TESTS_FOR_POD_VALUE(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, UInt32, uint32_t, , 100U, 101U, 102U, 103U)
  36. //%TESTS_FOR_POD_VALUE(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, Int32, int32_t, , 200, 201, 202, 203)
  37. //%TESTS_FOR_POD_VALUE(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, UInt64, uint64_t, , 300U, 301U, 302U, 303U)
  38. //%TESTS_FOR_POD_VALUE(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, Int64, int64_t, , 400, 401, 402, 403)
  39. //%TESTS_FOR_POD_VALUE(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, Bool, BOOL, , YES, YES, NO, NO)
  40. //%TESTS_FOR_POD_VALUE(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, Float, float, , 500.f, 501.f, 502.f, 503.f)
  41. //%TESTS_FOR_POD_VALUE(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, Double, double, , 600., 601., 602., 603.)
  42. //%TESTS_FOR_POD_VALUE(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, Enum, int32_t, Raw, 700, 701, 702, 703)
  43. //%TESTS_FOR_ENUM_VALUE_RAW_ADDITIONS(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4)
  44. //%PDDM-DEFINE TESTS_FOR_POD_VALUE(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, VALUE_NAME, VALUE_TYPE, VACCESSOR, VAL1, VAL2, VAL3, VAL4)
  45. //%TESTS_COMMON(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, VALUE_NAME, VALUE_TYPE, , POD, VACCESSOR, VAL1, VAL2, VAL3, VAL4)
  46. //%PDDM-DEFINE TESTS_FOR_POD_KEY_OBJECT_VALUE(KEY_NAME, KEY_TYPE, KEY1, KEY2, KEY3, KEY4, VALUE_NAME, VALUE_TYPE, VAL1, VAL2, VAL3, VAL4)
  47. //%TESTS_COMMON(KEY_NAME, KEY_TYPE, , , KEY1, KEY2, KEY3, KEY4, VALUE_NAME, VALUE_TYPE, Objects, OBJECT, , VAL1, VAL2, VAL3, VAL4)
  48. //%PDDM-DEFINE TESTS_COMMON(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, VALUE_NAME, VALUE_TYPE, VSUFFIX, VHELPER, VACCESSOR, VAL1, VAL2, VAL3, VAL4)
  49. //%#pragma mark - KEY_NAME -> VALUE_NAME
  50. //%
  51. //%@interface GPB##KEY_NAME##VALUE_NAME##DictionaryTests : XCTestCase
  52. //%@end
  53. //%
  54. //%@implementation GPB##KEY_NAME##VALUE_NAME##DictionaryTests
  55. //%
  56. //%- (void)testEmpty {
  57. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict = [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] init];
  58. //% XCTAssertNotNil(dict);
  59. //% XCTAssertEqual(dict.count, 0U);
  60. //%VALUE_NOT_FOUND##VHELPER(dict, KEY1)
  61. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  62. //% #pragma unused(aKey, aValue, stop)
  63. //% XCTFail(@"Shouldn't get here!");
  64. //% }];
  65. //% [dict release];
  66. //%}
  67. //%
  68. //%- (void)testOne {
  69. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict = [GPB##KEY_NAME##VALUE_NAME##Dictionary dictionaryWithValue:VAL1 forKey:KEY1];
  70. //% XCTAssertNotNil(dict);
  71. //% XCTAssertEqual(dict.count, 1U);
  72. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  73. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  74. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  75. //% XCTAssertEqual##KSUFFIX(aKey, KEY1);
  76. //% XCTAssertEqual##VSUFFIX(aValue, VAL1);
  77. //% XCTAssertNotEqual(stop, NULL);
  78. //% }];
  79. //%}
  80. //%
  81. //%- (void)testBasics {
  82. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2, KEY3 };
  83. //% const VALUE_TYPE kValues[] = { VAL1, VAL2, VAL3 };
  84. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  85. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  86. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  87. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  88. //% XCTAssertNotNil(dict);
  89. //% XCTAssertEqual(dict.count, 3U);
  90. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  91. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  92. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  93. //%VALUE_NOT_FOUND##VHELPER(dict, KEY4)
  94. //%
  95. //% __block NSUInteger idx = 0;
  96. //% KEY_TYPE KisP##*seenKeys = malloc(3 * sizeof(KEY_TYPE##KisP));
  97. //% VALUE_TYPE *seenValues = malloc(3 * sizeof(VALUE_TYPE));
  98. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  99. //% XCTAssertLessThan(idx, 3U);
  100. //% seenKeys[idx] = aKey;
  101. //% seenValues[idx] = aValue;
  102. //% XCTAssertNotEqual(stop, NULL);
  103. //% ++idx;
  104. //% }];
  105. //% for (int i = 0; i < 3; ++i) {
  106. //% BOOL foundKey = NO;
  107. //% for (int j = 0; (j < 3) && !foundKey; ++j) {
  108. //% if (COMPARE_KEYS##KSUFFIX(kKeys[i], seenKeys[j])) {
  109. //% foundKey = YES;
  110. //% XCTAssertEqual##VSUFFIX(kValues[i], seenValues[j], @"i = %d, j = %d", i, j);
  111. //% }
  112. //% }
  113. //% XCTAssertTrue(foundKey, @"i = %d", i);
  114. //% }
  115. //% free(seenKeys);
  116. //% free(seenValues);
  117. //%
  118. //% // Stopping the enumeration.
  119. //% idx = 0;
  120. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  121. //% #pragma unused(aKey, aValue)
  122. //% if (idx == 1) *stop = YES;
  123. //% XCTAssertNotEqual(idx, 2U);
  124. //% ++idx;
  125. //% }];
  126. //% [dict release];
  127. //%}
  128. //%
  129. //%- (void)testEquality {
  130. //% const KEY_TYPE KisP##kKeys1[] = { KEY1, KEY2, KEY3, KEY4 };
  131. //% const KEY_TYPE KisP##kKeys2[] = { KEY2, KEY1, KEY4 };
  132. //% const VALUE_TYPE kValues1[] = { VAL1, VAL2, VAL3 };
  133. //% const VALUE_TYPE kValues2[] = { VAL1, VAL4, VAL3 };
  134. //% const VALUE_TYPE kValues3[] = { VAL1, VAL2, VAL3, VAL4 };
  135. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict1 =
  136. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues1
  137. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  138. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  139. //% XCTAssertNotNil(dict1);
  140. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict1prime =
  141. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues1
  142. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  143. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  144. //% XCTAssertNotNil(dict1prime);
  145. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  146. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues2
  147. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  148. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues2)];
  149. //% XCTAssertNotNil(dict2);
  150. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict3 =
  151. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues1
  152. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys2
  153. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  154. //% XCTAssertNotNil(dict3);
  155. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict4 =
  156. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues3
  157. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  158. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues3)];
  159. //% XCTAssertNotNil(dict4);
  160. //%
  161. //% // 1/1Prime should be different objects, but equal.
  162. //% XCTAssertNotEqual(dict1, dict1prime);
  163. //% XCTAssertEqualObjects(dict1, dict1prime);
  164. //% // Equal, so they must have same hash.
  165. //% XCTAssertEqual([dict1 hash], [dict1prime hash]);
  166. //%
  167. //% // 2 is save keys, different values; not equal.
  168. //% XCTAssertNotEqualObjects(dict1, dict2);
  169. //%
  170. //% // 3 is different keys, samae values; not equal.
  171. //% XCTAssertNotEqualObjects(dict1, dict3);
  172. //%
  173. //% // 4 extra pair; not equal
  174. //% XCTAssertNotEqualObjects(dict1, dict4);
  175. //%
  176. //% [dict1 release];
  177. //% [dict1prime release];
  178. //% [dict2 release];
  179. //% [dict3 release];
  180. //% [dict4 release];
  181. //%}
  182. //%
  183. //%- (void)testCopy {
  184. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2, KEY3, KEY4 };
  185. //% const VALUE_TYPE kValues[] = { VAL1, VAL2, VAL3, VAL4 };
  186. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  187. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  188. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  189. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  190. //% XCTAssertNotNil(dict);
  191. //%
  192. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 = [dict copy];
  193. //% XCTAssertNotNil(dict2);
  194. //%
  195. //% // Should be new object but equal.
  196. //% XCTAssertNotEqual(dict, dict2);
  197. //% XCTAssertEqualObjects(dict, dict2);
  198. //% XCTAssertTrue([dict2 isKindOfClass:[GPB##KEY_NAME##VALUE_NAME##Dictionary class]]);
  199. //%
  200. //% [dict2 release];
  201. //% [dict release];
  202. //%}
  203. //%
  204. //%- (void)testDictionaryFromDictionary {
  205. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2, KEY3, KEY4 };
  206. //% const VALUE_TYPE kValues[] = { VAL1, VAL2, VAL3, VAL4 };
  207. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  208. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  209. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  210. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  211. //% XCTAssertNotNil(dict);
  212. //%
  213. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  214. //% [GPB##KEY_NAME##VALUE_NAME##Dictionary dictionaryWithDictionary:dict];
  215. //% XCTAssertNotNil(dict2);
  216. //%
  217. //% // Should be new pointer, but equal objects.
  218. //% XCTAssertNotEqual(dict, dict2);
  219. //% XCTAssertEqualObjects(dict, dict2);
  220. //% [dict release];
  221. //%}
  222. //%
  223. //%- (void)testAdds {
  224. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict = [GPB##KEY_NAME##VALUE_NAME##Dictionary dictionary];
  225. //% XCTAssertNotNil(dict);
  226. //%
  227. //% XCTAssertEqual(dict.count, 0U);
  228. //% [dict setValue:VAL1 forKey:KEY1];
  229. //% XCTAssertEqual(dict.count, 1U);
  230. //%
  231. //% const KEY_TYPE KisP##kKeys[] = { KEY2, KEY3, KEY4 };
  232. //% const VALUE_TYPE kValues[] = { VAL2, VAL3, VAL4 };
  233. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  234. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  235. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  236. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  237. //% XCTAssertNotNil(dict2);
  238. //% [dict add##VACCESSOR##EntriesFromDictionary:dict2];
  239. //% XCTAssertEqual(dict.count, 4U);
  240. //%
  241. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  242. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  243. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  244. //%TEST_VALUE##VHELPER(dict, value, KEY4, VAL4)
  245. //% [dict2 release];
  246. //%}
  247. //%
  248. //%- (void)testRemove {
  249. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2, KEY3, KEY4 };
  250. //% const VALUE_TYPE kValues[] = { VAL1, VAL2, VAL3, VAL4 };
  251. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  252. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  253. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  254. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  255. //% XCTAssertNotNil(dict);
  256. //% XCTAssertEqual(dict.count, 4U);
  257. //%
  258. //% [dict removeValueForKey:KEY2];
  259. //% XCTAssertEqual(dict.count, 3U);
  260. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  261. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  262. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  263. //%TEST_VALUE##VHELPER(dict, value, KEY4, VAL4)
  264. //%
  265. //% // Remove again does nothing.
  266. //% [dict removeValueForKey:KEY2];
  267. //% XCTAssertEqual(dict.count, 3U);
  268. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  269. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  270. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  271. //%TEST_VALUE##VHELPER(dict, value, KEY4, VAL4)
  272. //%
  273. //% [dict removeValueForKey:KEY4];
  274. //% XCTAssertEqual(dict.count, 2U);
  275. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  276. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  277. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  278. //%VALUE_NOT_FOUND##VHELPER(dict, KEY4)
  279. //%
  280. //% [dict removeAll];
  281. //% XCTAssertEqual(dict.count, 0U);
  282. //%VALUE_NOT_FOUND##VHELPER(dict, KEY1)
  283. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  284. //%VALUE_NOT_FOUND##VHELPER(dict, KEY3)
  285. //%VALUE_NOT_FOUND##VHELPER(dict, KEY4)
  286. //% [dict release];
  287. //%}
  288. //%
  289. //%- (void)testInplaceMutation {
  290. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2, KEY3, KEY4 };
  291. //% const VALUE_TYPE kValues[] = { VAL1, VAL2, VAL3, VAL4 };
  292. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  293. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  294. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  295. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  296. //% XCTAssertNotNil(dict);
  297. //% XCTAssertEqual(dict.count, 4U);
  298. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  299. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  300. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  301. //%TEST_VALUE##VHELPER(dict, value, KEY4, VAL4)
  302. //%
  303. //% [dict setValue:VAL4 forKey:KEY1];
  304. //% XCTAssertEqual(dict.count, 4U);
  305. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL4)
  306. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  307. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  308. //%TEST_VALUE##VHELPER(dict, value, KEY4, VAL4)
  309. //%
  310. //% [dict setValue:VAL2 forKey:KEY4];
  311. //% XCTAssertEqual(dict.count, 4U);
  312. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL4)
  313. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  314. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  315. //%TEST_VALUE##VHELPER(dict, value, KEY4, VAL2)
  316. //%
  317. //% const KEY_TYPE KisP##kKeys2[] = { KEY2, KEY3 };
  318. //% const VALUE_TYPE kValues2[] = { VAL3, VAL1 };
  319. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  320. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues2
  321. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys2
  322. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues2)];
  323. //% XCTAssertNotNil(dict2);
  324. //% [dict add##VACCESSOR##EntriesFromDictionary:dict2];
  325. //% XCTAssertEqual(dict.count, 4U);
  326. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL4)
  327. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL3)
  328. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL1)
  329. //%TEST_VALUE##VHELPER(dict, value, KEY4, VAL2)
  330. //%
  331. //% [dict2 release];
  332. //% [dict release];
  333. //%}
  334. //%
  335. //%@end
  336. //%
  337. //%PDDM-DEFINE TESTS_FOR_ENUM_VALUE_RAW_ADDITIONS(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4)
  338. //%TESTS_FOR_ENUM_VALUE_RAW_ADDITIONS2(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, Enum, int32_t, , POD, 700, 801, 702, 803)
  339. //%PDDM-DEFINE TESTS_FOR_ENUM_VALUE_RAW_ADDITIONS2(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, KEY3, KEY4, VALUE_NAME, VALUE_TYPE, VSUFFIX, VHELPER, VAL1, VAL2, VAL3, VAL4)
  340. //%#pragma mark - KEY_NAME -> VALUE_NAME (Unknown Enums)
  341. //%
  342. //%@interface GPB##KEY_NAME##VALUE_NAME##DictionaryUnknownEnumTests : XCTestCase
  343. //%@end
  344. //%
  345. //%@implementation GPB##KEY_NAME##VALUE_NAME##DictionaryUnknownEnumTests
  346. //%
  347. //%- (void)testRawBasics {
  348. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2, KEY3 };
  349. //% const VALUE_TYPE kValues[] = { VAL1, VAL2, VAL3 };
  350. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  351. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  352. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues
  353. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  354. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  355. //% XCTAssertNotNil(dict);
  356. //% XCTAssertEqual(dict.count, 3U);
  357. //% XCTAssertTrue(dict.validationFunc == TestingEnum_IsValidValue); // Pointer comparison
  358. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_RAW_VALUE##VHELPER(dict, value, KEY1, VAL1)
  359. //%TEST_VALUE##VHELPER(dict, value, KEY2, kGPBUnrecognizedEnumeratorValue)
  360. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY2, VAL2)
  361. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY3, VAL3)
  362. //%RAW_VALUE_NOT_FOUND##VHELPER(dict, KEY4)
  363. //%
  364. //% __block NSUInteger idx = 0;
  365. //% KEY_TYPE KisP##*seenKeys = malloc(3 * sizeof(KEY_TYPE##KisP));
  366. //% VALUE_TYPE *seenValues = malloc(3 * sizeof(VALUE_TYPE));
  367. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  368. //% XCTAssertLessThan(idx, 3U);
  369. //% seenKeys[idx] = aKey;
  370. //% seenValues[idx] = aValue;
  371. //% XCTAssertNotEqual(stop, NULL);
  372. //% ++idx;
  373. //% }];
  374. //% for (int i = 0; i < 3; ++i) {
  375. //% BOOL foundKey = NO;
  376. //% for (int j = 0; (j < 3) && !foundKey; ++j) {
  377. //% if (COMPARE_KEYS##KSUFFIX(kKeys[i], seenKeys[j])) {
  378. //% foundKey = YES;
  379. //% if (i == 1) {
  380. //% XCTAssertEqual##VSUFFIX(kGPBUnrecognizedEnumeratorValue, seenValues[j], @"i = %d, j = %d", i, j);
  381. //% } else {
  382. //% XCTAssertEqual##VSUFFIX(kValues[i], seenValues[j], @"i = %d, j = %d", i, j);
  383. //% }
  384. //% }
  385. //% }
  386. //% XCTAssertTrue(foundKey, @"i = %d", i);
  387. //% }
  388. //% idx = 0;
  389. //% [dict enumerateKeysAndRawValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  390. //% XCTAssertLessThan(idx, 3U);
  391. //% seenKeys[idx] = aKey;
  392. //% seenValues[idx] = aValue;
  393. //% XCTAssertNotEqual(stop, NULL);
  394. //% ++idx;
  395. //% }];
  396. //% for (int i = 0; i < 3; ++i) {
  397. //% BOOL foundKey = NO;
  398. //% for (int j = 0; (j < 3) && !foundKey; ++j) {
  399. //% if (COMPARE_KEYS##KSUFFIX(kKeys[i], seenKeys[j])) {
  400. //% foundKey = YES;
  401. //% XCTAssertEqual##VSUFFIX(kValues[i], seenValues[j], @"i = %d, j = %d", i, j);
  402. //% }
  403. //% }
  404. //% XCTAssertTrue(foundKey, @"i = %d", i);
  405. //% }
  406. //% free(seenKeys);
  407. //% free(seenValues);
  408. //%
  409. //% // Stopping the enumeration.
  410. //% idx = 0;
  411. //% [dict enumerateKeysAndRawValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  412. //% #pragma unused(aKey, aValue)
  413. //% if (idx == 1) *stop = YES;
  414. //% XCTAssertNotEqual(idx, 2U);
  415. //% ++idx;
  416. //% }];
  417. //% [dict release];
  418. //%}
  419. //%
  420. //%- (void)testEqualityWithUnknowns {
  421. //% const KEY_TYPE KisP##kKeys1[] = { KEY1, KEY2, KEY3, KEY4 };
  422. //% const KEY_TYPE KisP##kKeys2[] = { KEY2, KEY1, KEY4 };
  423. //% const VALUE_TYPE kValues1[] = { VAL1, VAL2, VAL3 }; // Unknown
  424. //% const VALUE_TYPE kValues2[] = { VAL1, VAL4, VAL3 }; // Unknown
  425. //% const VALUE_TYPE kValues3[] = { VAL1, VAL2, VAL3, VAL4 }; // Unknowns
  426. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict1 =
  427. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  428. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues1
  429. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  430. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  431. //% XCTAssertNotNil(dict1);
  432. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict1prime =
  433. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  434. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues1
  435. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  436. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  437. //% XCTAssertNotNil(dict1prime);
  438. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  439. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  440. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues2
  441. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  442. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues2)];
  443. //% XCTAssertNotNil(dict2);
  444. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict3 =
  445. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  446. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues1
  447. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys2
  448. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  449. //% XCTAssertNotNil(dict3);
  450. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict4 =
  451. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  452. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues3
  453. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  454. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues3)];
  455. //% XCTAssertNotNil(dict4);
  456. //%
  457. //% // 1/1Prime should be different objects, but equal.
  458. //% XCTAssertNotEqual(dict1, dict1prime);
  459. //% XCTAssertEqualObjects(dict1, dict1prime);
  460. //% // Equal, so they must have same hash.
  461. //% XCTAssertEqual([dict1 hash], [dict1prime hash]);
  462. //%
  463. //% // 2 is save keys, different values; not equal.
  464. //% XCTAssertNotEqualObjects(dict1, dict2);
  465. //%
  466. //% // 3 is different keys, samae values; not equal.
  467. //% XCTAssertNotEqualObjects(dict1, dict3);
  468. //%
  469. //% // 4 extra pair; not equal
  470. //% XCTAssertNotEqualObjects(dict1, dict4);
  471. //%
  472. //% [dict1 release];
  473. //% [dict1prime release];
  474. //% [dict2 release];
  475. //% [dict3 release];
  476. //% [dict4 release];
  477. //%}
  478. //%
  479. //%- (void)testCopyWithUnknowns {
  480. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2, KEY3, KEY4 };
  481. //% const VALUE_TYPE kValues[] = { VAL1, VAL2, VAL3, VAL4 }; // Unknown
  482. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  483. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  484. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues
  485. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  486. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  487. //% XCTAssertNotNil(dict);
  488. //%
  489. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 = [dict copy];
  490. //% XCTAssertNotNil(dict2);
  491. //%
  492. //% // Should be new pointer, but equal objects.
  493. //% XCTAssertNotEqual(dict, dict2);
  494. //% XCTAssertEqual(dict.validationFunc, dict2.validationFunc); // Pointer comparison
  495. //% XCTAssertEqualObjects(dict, dict2);
  496. //%
  497. //% [dict2 release];
  498. //% [dict release];
  499. //%}
  500. //%
  501. //%- (void)testDictionaryFromDictionary {
  502. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2, KEY3, KEY4 };
  503. //% const VALUE_TYPE kValues[] = { VAL1, VAL2, VAL3, VAL4 }; // Unknowns
  504. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  505. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  506. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues
  507. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  508. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  509. //% XCTAssertNotNil(dict);
  510. //%
  511. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  512. //% [GPB##KEY_NAME##VALUE_NAME##Dictionary dictionaryWithDictionary:dict];
  513. //% XCTAssertNotNil(dict2);
  514. //%
  515. //% // Should be new pointer, but equal objects.
  516. //% XCTAssertNotEqual(dict, dict2);
  517. //% XCTAssertEqualObjects(dict, dict2);
  518. //% XCTAssertEqual(dict.validationFunc, dict2.validationFunc); // Pointer comparison
  519. //% [dict release];
  520. //%}
  521. //%
  522. //%- (void)testUnknownAdds {
  523. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  524. //% [GPB##KEY_NAME##VALUE_NAME##Dictionary dictionaryWithValidationFunction:TestingEnum_IsValidValue];
  525. //% XCTAssertNotNil(dict);
  526. //%
  527. //% XCTAssertEqual(dict.count, 0U);
  528. //% XCTAssertThrowsSpecificNamed([dict setValue:VAL2 forKey:KEY2], // Unknown
  529. //% NSException, NSInvalidArgumentException);
  530. //% XCTAssertEqual(dict.count, 0U);
  531. //% [dict setRawValue:VAL2 forKey:KEY2]; // Unknown
  532. //% XCTAssertEqual(dict.count, 1U);
  533. //%
  534. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY3, KEY4 };
  535. //% const VALUE_TYPE kValues[] = { VAL1, VAL3, VAL4 }; // Unknown
  536. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  537. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  538. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  539. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  540. //% XCTAssertNotNil(dict2);
  541. //% [dict addRawEntriesFromDictionary:dict2];
  542. //% XCTAssertEqual(dict.count, 4U);
  543. //%
  544. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  545. //%TEST_VALUE##VHELPER(dict, value, KEY2, kGPBUnrecognizedEnumeratorValue)
  546. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY2, VAL2)
  547. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  548. //%TEST_VALUE##VHELPER(dict, value, KEY4, kGPBUnrecognizedEnumeratorValue)
  549. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY4, VAL4)
  550. //% [dict2 release];
  551. //%}
  552. //%
  553. //%- (void)testUnknownRemove {
  554. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2, KEY3, KEY4 };
  555. //% const VALUE_TYPE kValues[] = { VAL1, VAL2, VAL3, VAL4 }; // Unknowns
  556. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  557. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  558. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues
  559. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  560. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  561. //% XCTAssertNotNil(dict);
  562. //% XCTAssertEqual(dict.count, 4U);
  563. //%
  564. //% [dict removeValueForKey:KEY2];
  565. //% XCTAssertEqual(dict.count, 3U);
  566. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  567. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  568. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  569. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY4, VAL4)
  570. //%
  571. //% // Remove again does nothing.
  572. //% [dict removeValueForKey:KEY2];
  573. //% XCTAssertEqual(dict.count, 3U);
  574. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  575. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  576. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  577. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY4, VAL4)
  578. //%
  579. //% [dict removeValueForKey:KEY4];
  580. //% XCTAssertEqual(dict.count, 2U);
  581. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  582. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  583. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  584. //%VALUE_NOT_FOUND##VHELPER(dict, KEY4)
  585. //%
  586. //% [dict removeAll];
  587. //% XCTAssertEqual(dict.count, 0U);
  588. //%VALUE_NOT_FOUND##VHELPER(dict, KEY1)
  589. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  590. //%VALUE_NOT_FOUND##VHELPER(dict, KEY3)
  591. //%VALUE_NOT_FOUND##VHELPER(dict, KEY4)
  592. //% [dict release];
  593. //%}
  594. //%
  595. //%- (void)testInplaceMutationUnknowns {
  596. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2, KEY3, KEY4 };
  597. //% const VALUE_TYPE kValues[] = { VAL1, VAL2, VAL3, VAL4 }; // Unknowns
  598. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  599. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  600. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues
  601. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  602. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  603. //% XCTAssertNotNil(dict);
  604. //% XCTAssertEqual(dict.count, 4U);
  605. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  606. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY2, VAL2)
  607. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  608. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY4, VAL4)
  609. //%
  610. //% XCTAssertThrowsSpecificNamed([dict setValue:VAL4 forKey:KEY1], // Unknown
  611. //% NSException, NSInvalidArgumentException);
  612. //% XCTAssertEqual(dict.count, 4U);
  613. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  614. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY2, VAL2)
  615. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  616. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY4, VAL4)
  617. //%
  618. //% [dict setRawValue:VAL4 forKey:KEY1]; // Unknown
  619. //% XCTAssertEqual(dict.count, 4U);
  620. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY1, VAL4)
  621. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY2, VAL2)
  622. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  623. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY4, VAL4)
  624. //%
  625. //% [dict setRawValue:VAL1 forKey:KEY4];
  626. //% XCTAssertEqual(dict.count, 4U);
  627. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY1, VAL4)
  628. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY2, VAL2)
  629. //%TEST_VALUE##VHELPER(dict, value, KEY3, VAL3)
  630. //%TEST_VALUE##VHELPER(dict, value, KEY4, VAL1)
  631. //%
  632. //% const KEY_TYPE KisP##kKeys2[] = { KEY2, KEY3 };
  633. //% const VALUE_TYPE kValues2[] = { VAL3, VAL2 }; // Unknown
  634. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  635. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  636. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues2
  637. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys2
  638. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues2)];
  639. //% XCTAssertNotNil(dict2);
  640. //% [dict addRawEntriesFromDictionary:dict2];
  641. //% XCTAssertEqual(dict.count, 4U);
  642. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY1, VAL4)
  643. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL3)
  644. //%TEST_RAW_VALUE##VHELPER(dict, value, KEY3, VAL2)
  645. //%TEST_VALUE##VHELPER(dict, value, KEY4, VAL1)
  646. //%
  647. //% [dict2 release];
  648. //% [dict release];
  649. //%}
  650. //%
  651. //%- (void)testCopyUnknowns {
  652. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2, KEY3, KEY4 };
  653. //% const VALUE_TYPE kValues[] = { VAL1, VAL2, VAL3, VAL4 };
  654. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  655. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValidationFunction:TestingEnum_IsValidValue
  656. //% KEY_NAME$S VALUE_NAME$S rawValues:kValues
  657. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  658. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  659. //% XCTAssertNotNil(dict);
  660. //%
  661. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 = [dict copy];
  662. //% XCTAssertNotNil(dict2);
  663. //%
  664. //% // Should be new pointer, but equal objects.
  665. //% XCTAssertNotEqual(dict, dict2);
  666. //% XCTAssertEqualObjects(dict, dict2);
  667. //% XCTAssertEqual(dict.validationFunc, dict2.validationFunc); // Pointer comparison
  668. //% XCTAssertTrue([dict2 isKindOfClass:[GPB##KEY_NAME##VALUE_NAME##Dictionary class]]);
  669. //%
  670. //% [dict2 release];
  671. //% [dict release];
  672. //%}
  673. //%
  674. //%@end
  675. //%
  676. //
  677. // Helpers for PODs
  678. //
  679. //%PDDM-DEFINE DECLARE_VALUE_STORAGEPOD(VALUE_TYPE, NAME)
  680. //% VALUE_TYPE NAME;
  681. //%
  682. //%PDDM-DEFINE VALUE_NOT_FOUNDPOD(DICT, KEY)
  683. //% XCTAssertFalse([DICT valueForKey:KEY value:NULL]);
  684. //%PDDM-DEFINE TEST_VALUEPOD(DICT, STORAGE, KEY, VALUE)
  685. //% XCTAssertTrue([DICT valueForKey:KEY value:NULL]);
  686. //% XCTAssertTrue([DICT valueForKey:KEY value:&STORAGE]);
  687. //% XCTAssertEqual(STORAGE, VALUE);
  688. //%PDDM-DEFINE COMPARE_KEYS(KEY1, KEY2)
  689. //%KEY1 == KEY2
  690. //%PDDM-DEFINE RAW_VALUE_NOT_FOUNDPOD(DICT, KEY)
  691. //% XCTAssertFalse([DICT valueForKey:KEY rawValue:NULL]);
  692. //%PDDM-DEFINE TEST_RAW_VALUEPOD(DICT, STORAGE, KEY, VALUE)
  693. //% XCTAssertTrue([DICT valueForKey:KEY rawValue:NULL]);
  694. //% XCTAssertTrue([DICT valueForKey:KEY rawValue:&STORAGE]);
  695. //% XCTAssertEqual(STORAGE, VALUE);
  696. //
  697. // Helpers for Objects
  698. //
  699. //%PDDM-DEFINE DECLARE_VALUE_STORAGEOBJECT(VALUE_TYPE, NAME)
  700. // Empty
  701. //%PDDM-DEFINE VALUE_NOT_FOUNDOBJECT(DICT, KEY)
  702. //% XCTAssertNil([DICT valueForKey:KEY]);
  703. //%PDDM-DEFINE TEST_VALUEOBJECT(DICT, STORAGE, KEY, VALUE)
  704. //% XCTAssertEqualObjects([DICT valueForKey:KEY], VALUE);
  705. //%PDDM-DEFINE COMPARE_KEYSObjects(KEY1, KEY2)
  706. //%[KEY1 isEqual:KEY2]
  707. //
  708. // Helpers for tests.
  709. //
  710. //%PDDM-DEFINE TEST_HELPERS(KEY_NAME, KEY_TYPE, KisP)
  711. //%#ifndef GPBARRAYSIZE
  712. //%#define GPBARRAYSIZE(a) ((sizeof(a) / sizeof((a[0]))))
  713. //%#endif // GPBARRAYSIZE
  714. //%
  715. //%// To let the testing macros work, add some extra methods to simplify things.
  716. //%@interface GPB##KEY_NAME##EnumDictionary (TestingTweak)
  717. //%+ (instancetype)dictionaryWithValue:(int32_t)value forKey:(KEY_TYPE##KisP$S##KisP)key;
  718. //%- (instancetype)initWithValues:(const int32_t [])values
  719. //% forKeys:(const KEY_TYPE##KisP$S##KisP [])keys
  720. //% count:(NSUInteger)count;
  721. //%@end
  722. //%
  723. //%static BOOL TestingEnum_IsValidValue(int32_t value) {
  724. //% switch (value) {
  725. //% case 700:
  726. //% case 701:
  727. //% case 702:
  728. //% case 703:
  729. //% return YES;
  730. //% default:
  731. //% return NO;
  732. //% }
  733. //%}
  734. //%
  735. //%@implementation GPB##KEY_NAME##EnumDictionary (TestingTweak)
  736. //%+ (instancetype)dictionaryWithValue:(int32_t)value forKey:(KEY_TYPE##KisP$S##KisP)key {
  737. //% // Cast is needed to compiler knows what class we are invoking initWithValues: on to get the
  738. //% // type correct.
  739. //% return [[(GPB##KEY_NAME##EnumDictionary*)[self alloc] initWithValidationFunction:TestingEnum_IsValidValue
  740. //% KEY_NAME$S rawValues:&value
  741. //% KEY_NAME$S forKeys:&key
  742. //% KEY_NAME$S count:1] autorelease];
  743. //%}
  744. //%- (instancetype)initWithValues:(const int32_t [])values
  745. //% forKeys:(const KEY_TYPE##KisP$S##KisP [])keys
  746. //% count:(NSUInteger)count {
  747. //% return [self initWithValidationFunction:TestingEnum_IsValidValue
  748. //% rawValues:values
  749. //% forKeys:keys
  750. //% count:count];
  751. //%}
  752. //%@end
  753. //%
  754. //%
  755. //
  756. // BOOL test macros
  757. //
  758. //TODO(thomasvl): enum tests
  759. //%PDDM-DEFINE BOOL_TESTS_FOR_POD_VALUE(VALUE_NAME, VALUE_TYPE, VAL1, VAL2)
  760. //%BOOL_TESTS_COMMON(Bool, BOOL, , , YES, NO, VALUE_NAME, VALUE_TYPE, , POD, VAL1, VAL2)
  761. //%PDDM-DEFINE TESTS_FOR_BOOL_KEY_OBJECT_VALUE(VALUE_NAME, VALUE_TYPE, VAL1, VAL2)
  762. //%BOOL_TESTS_COMMON(Bool, BOOL, , , YES, NO, VALUE_NAME, VALUE_TYPE, Objects, OBJECT, VAL1, VAL2)
  763. //%PDDM-DEFINE BOOL_TESTS_COMMON(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, VALUE_NAME, VALUE_TYPE, VSUFFIX, VHELPER, VAL1, VAL2)
  764. //%#pragma mark - KEY_NAME -> VALUE_NAME
  765. //%
  766. //%@interface GPB##KEY_NAME##VALUE_NAME##DictionaryTests : XCTestCase
  767. //%@end
  768. //%
  769. //%@implementation GPB##KEY_NAME##VALUE_NAME##DictionaryTests
  770. //%
  771. //%- (void)testEmpty {
  772. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict = [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] init];
  773. //% XCTAssertNotNil(dict);
  774. //% XCTAssertEqual(dict.count, 0U);
  775. //%VALUE_NOT_FOUND##VHELPER(dict, KEY1)
  776. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  777. //% #pragma unused(aKey, aValue, stop)
  778. //% XCTFail(@"Shouldn't get here!");
  779. //% }];
  780. //% [dict release];
  781. //%}
  782. //%
  783. //%- (void)testOne {
  784. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict = [GPB##KEY_NAME##VALUE_NAME##Dictionary dictionaryWithValue:VAL1 forKey:KEY1];
  785. //% XCTAssertNotNil(dict);
  786. //% XCTAssertEqual(dict.count, 1U);
  787. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  788. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  789. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  790. //% XCTAssertEqual##KSUFFIX(aKey, KEY1);
  791. //% XCTAssertEqual##VSUFFIX(aValue, VAL1);
  792. //% XCTAssertNotEqual(stop, NULL);
  793. //% }];
  794. //%}
  795. //%
  796. //%- (void)testBasics {
  797. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2 };
  798. //% const VALUE_TYPE kValues[] = { VAL1, VAL2 };
  799. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  800. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  801. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  802. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  803. //% XCTAssertNotNil(dict);
  804. //% XCTAssertEqual(dict.count, 2U);
  805. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  806. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  807. //%
  808. //% __block NSUInteger idx = 0;
  809. //% KEY_TYPE KisP##*seenKeys = malloc(2 * sizeof(KEY_TYPE##KisP));
  810. //% VALUE_TYPE *seenValues = malloc(2 * sizeof(VALUE_TYPE));
  811. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  812. //% XCTAssertLessThan(idx, 2U);
  813. //% seenKeys[idx] = aKey;
  814. //% seenValues[idx] = aValue;
  815. //% XCTAssertNotEqual(stop, NULL);
  816. //% ++idx;
  817. //% }];
  818. //% for (int i = 0; i < 2; ++i) {
  819. //% BOOL foundKey = NO;
  820. //% for (int j = 0; (j < 2) && !foundKey; ++j) {
  821. //% if (COMPARE_KEYS##KSUFFIX(kKeys[i], seenKeys[j])) {
  822. //% foundKey = YES;
  823. //% XCTAssertEqual##VSUFFIX(kValues[i], seenValues[j], @"i = %d, j = %d", i, j);
  824. //% }
  825. //% }
  826. //% XCTAssertTrue(foundKey, @"i = %d", i);
  827. //% }
  828. //% free(seenKeys);
  829. //% free(seenValues);
  830. //%
  831. //% // Stopping the enumeration.
  832. //% idx = 0;
  833. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  834. //% #pragma unused(aKey, aValue)
  835. //% if (idx == 0) *stop = YES;
  836. //% XCTAssertNotEqual(idx, 2U);
  837. //% ++idx;
  838. //% }];
  839. //% [dict release];
  840. //%}
  841. //%
  842. //%- (void)testEquality {
  843. //% const KEY_TYPE KisP##kKeys1[] = { KEY1, KEY2 };
  844. //% const KEY_TYPE KisP##kKeys2[] = { KEY2, KEY1 };
  845. //% const VALUE_TYPE kValues1[] = { VAL1, VAL2 };
  846. //% const VALUE_TYPE kValues2[] = { VAL2, VAL1 };
  847. //% const VALUE_TYPE kValues3[] = { VAL2 };
  848. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict1 =
  849. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues1
  850. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  851. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  852. //% XCTAssertNotNil(dict1);
  853. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict1prime =
  854. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues1
  855. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  856. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  857. //% XCTAssertNotNil(dict1prime);
  858. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  859. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues2
  860. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  861. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues2)];
  862. //% XCTAssertNotNil(dict2);
  863. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict3 =
  864. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues1
  865. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys2
  866. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  867. //% XCTAssertNotNil(dict3);
  868. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict4 =
  869. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues3
  870. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  871. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues3)];
  872. //% XCTAssertNotNil(dict4);
  873. //%
  874. //% // 1/1Prime should be different objects, but equal.
  875. //% XCTAssertNotEqual(dict1, dict1prime);
  876. //% XCTAssertEqualObjects(dict1, dict1prime);
  877. //% // Equal, so they must have same hash.
  878. //% XCTAssertEqual([dict1 hash], [dict1prime hash]);
  879. //%
  880. //% // 2 is save keys, different values; not equal.
  881. //% XCTAssertNotEqualObjects(dict1, dict2);
  882. //%
  883. //% // 3 is different keys, samae values; not equal.
  884. //% XCTAssertNotEqualObjects(dict1, dict3);
  885. //%
  886. //% // 4 Fewer pairs; not equal
  887. //% XCTAssertNotEqualObjects(dict1, dict4);
  888. //%
  889. //% [dict1 release];
  890. //% [dict1prime release];
  891. //% [dict2 release];
  892. //% [dict3 release];
  893. //% [dict4 release];
  894. //%}
  895. //%
  896. //%- (void)testCopy {
  897. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2 };
  898. //% const VALUE_TYPE kValues[] = { VAL1, VAL2 };
  899. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  900. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  901. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  902. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  903. //% XCTAssertNotNil(dict);
  904. //%
  905. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 = [dict copy];
  906. //% XCTAssertNotNil(dict2);
  907. //%
  908. //% // Should be new object but equal.
  909. //% XCTAssertNotEqual(dict, dict2);
  910. //% XCTAssertEqualObjects(dict, dict2);
  911. //% XCTAssertTrue([dict2 isKindOfClass:[GPB##KEY_NAME##VALUE_NAME##Dictionary class]]);
  912. //%
  913. //% [dict2 release];
  914. //% [dict release];
  915. //%}
  916. //%
  917. //%- (void)testDictionaryFromDictionary {
  918. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2 };
  919. //% const VALUE_TYPE kValues[] = { VAL1, VAL2 };
  920. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  921. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  922. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  923. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  924. //% XCTAssertNotNil(dict);
  925. //%
  926. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  927. //% [GPB##KEY_NAME##VALUE_NAME##Dictionary dictionaryWithDictionary:dict];
  928. //% XCTAssertNotNil(dict2);
  929. //%
  930. //% // Should be new pointer, but equal objects.
  931. //% XCTAssertNotEqual(dict, dict2);
  932. //% XCTAssertEqualObjects(dict, dict2);
  933. //% [dict release];
  934. //%}
  935. //%
  936. //%- (void)testAdds {
  937. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict = [GPB##KEY_NAME##VALUE_NAME##Dictionary dictionary];
  938. //% XCTAssertNotNil(dict);
  939. //%
  940. //% XCTAssertEqual(dict.count, 0U);
  941. //% [dict setValue:VAL1 forKey:KEY1];
  942. //% XCTAssertEqual(dict.count, 1U);
  943. //%
  944. //% const KEY_TYPE KisP##kKeys[] = { KEY2 };
  945. //% const VALUE_TYPE kValues[] = { VAL2 };
  946. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  947. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  948. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  949. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  950. //% XCTAssertNotNil(dict2);
  951. //% [dict addEntriesFromDictionary:dict2];
  952. //% XCTAssertEqual(dict.count, 2U);
  953. //%
  954. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  955. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  956. //% [dict2 release];
  957. //%}
  958. //%
  959. //%- (void)testRemove {
  960. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2};
  961. //% const VALUE_TYPE kValues[] = { VAL1, VAL2 };
  962. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  963. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  964. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  965. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  966. //% XCTAssertNotNil(dict);
  967. //% XCTAssertEqual(dict.count, 2U);
  968. //%
  969. //% [dict removeValueForKey:KEY2];
  970. //% XCTAssertEqual(dict.count, 1U);
  971. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  972. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  973. //%
  974. //% // Remove again does nothing.
  975. //% [dict removeValueForKey:KEY2];
  976. //% XCTAssertEqual(dict.count, 1U);
  977. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  978. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  979. //%
  980. //% [dict removeAll];
  981. //% XCTAssertEqual(dict.count, 0U);
  982. //%VALUE_NOT_FOUND##VHELPER(dict, KEY1)
  983. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  984. //% [dict release];
  985. //%}
  986. //%
  987. //%- (void)testInplaceMutation {
  988. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2 };
  989. //% const VALUE_TYPE kValues[] = { VAL1, VAL2 };
  990. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  991. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  992. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  993. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  994. //% XCTAssertNotNil(dict);
  995. //% XCTAssertEqual(dict.count, 2U);
  996. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  997. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  998. //%
  999. //% [dict setValue:VAL2 forKey:KEY1];
  1000. //% XCTAssertEqual(dict.count, 2U);
  1001. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL2)
  1002. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  1003. //%
  1004. //% [dict setValue:VAL1 forKey:KEY2];
  1005. //% XCTAssertEqual(dict.count, 2U);
  1006. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL2)
  1007. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL1)
  1008. //%
  1009. //% const KEY_TYPE KisP##kKeys2[] = { KEY2, KEY1 };
  1010. //% const VALUE_TYPE kValues2[] = { VAL2, VAL1 };
  1011. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  1012. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues2
  1013. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys2
  1014. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues2)];
  1015. //% XCTAssertNotNil(dict2);
  1016. //% [dict addEntriesFromDictionary:dict2];
  1017. //% XCTAssertEqual(dict.count, 2U);
  1018. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  1019. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  1020. //%
  1021. //% [dict2 release];
  1022. //% [dict release];
  1023. //%}
  1024. //%
  1025. //%@end
  1026. //%