GPBDictionaryTests.pddm 50 KB

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