GPBDictionaryTests.pddm 46 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040
  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. //%// To let the testing macros work, add some extra methods to simplify things.
  712. //%@interface GPB##KEY_NAME##EnumDictionary (TestingTweak)
  713. //%+ (instancetype)dictionaryWithValue:(int32_t)value forKey:(KEY_TYPE##KisP$S##KisP)key;
  714. //%- (instancetype)initWithValues:(const int32_t [])values
  715. //% forKeys:(const KEY_TYPE##KisP$S##KisP [])keys
  716. //% count:(NSUInteger)count;
  717. //%@end
  718. //%
  719. //%static BOOL TestingEnum_IsValidValue(int32_t value) {
  720. //% switch (value) {
  721. //% case 700:
  722. //% case 701:
  723. //% case 702:
  724. //% case 703:
  725. //% return YES;
  726. //% default:
  727. //% return NO;
  728. //% }
  729. //%}
  730. //%
  731. //%@implementation GPB##KEY_NAME##EnumDictionary (TestingTweak)
  732. //%+ (instancetype)dictionaryWithValue:(int32_t)value forKey:(KEY_TYPE##KisP$S##KisP)key {
  733. //% // Cast is needed to compiler knows what class we are invoking initWithValues: on to get the
  734. //% // type correct.
  735. //% return [[(GPB##KEY_NAME##EnumDictionary*)[self alloc] initWithValidationFunction:TestingEnum_IsValidValue
  736. //% KEY_NAME$S rawValues:&value
  737. //% KEY_NAME$S forKeys:&key
  738. //% KEY_NAME$S count:1] autorelease];
  739. //%}
  740. //%- (instancetype)initWithValues:(const int32_t [])values
  741. //% forKeys:(const KEY_TYPE##KisP$S##KisP [])keys
  742. //% count:(NSUInteger)count {
  743. //% return [self initWithValidationFunction:TestingEnum_IsValidValue
  744. //% rawValues:values
  745. //% forKeys:keys
  746. //% count:count];
  747. //%}
  748. //%@end
  749. //%
  750. //%
  751. //
  752. // BOOL test macros
  753. //
  754. //TODO(thomasvl): enum tests
  755. //%PDDM-DEFINE BOOL_TESTS_FOR_POD_VALUE(VALUE_NAME, VALUE_TYPE, VAL1, VAL2)
  756. //%BOOL_TESTS_COMMON(Bool, BOOL, , , YES, NO, VALUE_NAME, VALUE_TYPE, , POD, VAL1, VAL2)
  757. //%PDDM-DEFINE TESTS_FOR_BOOL_KEY_OBJECT_VALUE(VALUE_NAME, VALUE_TYPE, VAL1, VAL2)
  758. //%BOOL_TESTS_COMMON(Bool, BOOL, , , YES, NO, VALUE_NAME, VALUE_TYPE, Objects, OBJECT, VAL1, VAL2)
  759. //%PDDM-DEFINE BOOL_TESTS_COMMON(KEY_NAME, KEY_TYPE, KisP, KSUFFIX, KEY1, KEY2, VALUE_NAME, VALUE_TYPE, VSUFFIX, VHELPER, VAL1, VAL2)
  760. //%#pragma mark - KEY_NAME -> VALUE_NAME
  761. //%
  762. //%@interface GPB##KEY_NAME##VALUE_NAME##DictionaryTests : XCTestCase
  763. //%@end
  764. //%
  765. //%@implementation GPB##KEY_NAME##VALUE_NAME##DictionaryTests
  766. //%
  767. //%- (void)testEmpty {
  768. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict = [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] init];
  769. //% XCTAssertNotNil(dict);
  770. //% XCTAssertEqual(dict.count, 0U);
  771. //%VALUE_NOT_FOUND##VHELPER(dict, KEY1)
  772. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  773. //% #pragma unused(aKey, aValue, stop)
  774. //% XCTFail(@"Shouldn't get here!");
  775. //% }];
  776. //% [dict release];
  777. //%}
  778. //%
  779. //%- (void)testOne {
  780. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict = [GPB##KEY_NAME##VALUE_NAME##Dictionary dictionaryWithValue:VAL1 forKey:KEY1];
  781. //% XCTAssertNotNil(dict);
  782. //% XCTAssertEqual(dict.count, 1U);
  783. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  784. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  785. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  786. //% XCTAssertEqual##KSUFFIX(aKey, KEY1);
  787. //% XCTAssertEqual##VSUFFIX(aValue, VAL1);
  788. //% XCTAssertNotEqual(stop, NULL);
  789. //% }];
  790. //%}
  791. //%
  792. //%- (void)testBasics {
  793. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2 };
  794. //% const VALUE_TYPE kValues[] = { VAL1, VAL2 };
  795. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  796. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  797. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  798. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  799. //% XCTAssertNotNil(dict);
  800. //% XCTAssertEqual(dict.count, 2U);
  801. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  802. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  803. //%
  804. //% __block NSUInteger idx = 0;
  805. //% KEY_TYPE KisP##*seenKeys = malloc(2 * sizeof(KEY_TYPE##KisP));
  806. //% VALUE_TYPE *seenValues = malloc(2 * sizeof(VALUE_TYPE));
  807. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  808. //% XCTAssertLessThan(idx, 2U);
  809. //% seenKeys[idx] = aKey;
  810. //% seenValues[idx] = aValue;
  811. //% XCTAssertNotEqual(stop, NULL);
  812. //% ++idx;
  813. //% }];
  814. //% for (int i = 0; i < 2; ++i) {
  815. //% BOOL foundKey = NO;
  816. //% for (int j = 0; (j < 2) && !foundKey; ++j) {
  817. //% if (COMPARE_KEYS##KSUFFIX(kKeys[i], seenKeys[j])) {
  818. //% foundKey = YES;
  819. //% XCTAssertEqual##VSUFFIX(kValues[i], seenValues[j], @"i = %d, j = %d", i, j);
  820. //% }
  821. //% }
  822. //% XCTAssertTrue(foundKey, @"i = %d", i);
  823. //% }
  824. //% free(seenKeys);
  825. //% free(seenValues);
  826. //%
  827. //% // Stopping the enumeration.
  828. //% idx = 0;
  829. //% [dict enumerateKeysAndValuesUsingBlock:^(KEY_TYPE KisP##aKey, VALUE_TYPE aValue, BOOL *stop) {
  830. //% #pragma unused(aKey, aValue)
  831. //% if (idx == 0) *stop = YES;
  832. //% XCTAssertNotEqual(idx, 2U);
  833. //% ++idx;
  834. //% }];
  835. //% [dict release];
  836. //%}
  837. //%
  838. //%- (void)testEquality {
  839. //% const KEY_TYPE KisP##kKeys1[] = { KEY1, KEY2 };
  840. //% const KEY_TYPE KisP##kKeys2[] = { KEY2, KEY1 };
  841. //% const VALUE_TYPE kValues1[] = { VAL1, VAL2 };
  842. //% const VALUE_TYPE kValues2[] = { VAL2, VAL1 };
  843. //% const VALUE_TYPE kValues3[] = { VAL2 };
  844. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict1 =
  845. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues1
  846. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  847. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  848. //% XCTAssertNotNil(dict1);
  849. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict1prime =
  850. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues1
  851. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  852. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  853. //% XCTAssertNotNil(dict1prime);
  854. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  855. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues2
  856. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  857. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues2)];
  858. //% XCTAssertNotNil(dict2);
  859. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict3 =
  860. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues1
  861. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys2
  862. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues1)];
  863. //% XCTAssertNotNil(dict3);
  864. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict4 =
  865. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues3
  866. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys1
  867. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues3)];
  868. //% XCTAssertNotNil(dict4);
  869. //%
  870. //% // 1/1Prime should be different objects, but equal.
  871. //% XCTAssertNotEqual(dict1, dict1prime);
  872. //% XCTAssertEqualObjects(dict1, dict1prime);
  873. //% // Equal, so they must have same hash.
  874. //% XCTAssertEqual([dict1 hash], [dict1prime hash]);
  875. //%
  876. //% // 2 is save keys, different values; not equal.
  877. //% XCTAssertNotEqualObjects(dict1, dict2);
  878. //%
  879. //% // 3 is different keys, samae values; not equal.
  880. //% XCTAssertNotEqualObjects(dict1, dict3);
  881. //%
  882. //% // 4 Fewer pairs; not equal
  883. //% XCTAssertNotEqualObjects(dict1, dict4);
  884. //%
  885. //% [dict1 release];
  886. //% [dict1prime release];
  887. //% [dict2 release];
  888. //% [dict3 release];
  889. //% [dict4 release];
  890. //%}
  891. //%
  892. //%- (void)testCopy {
  893. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2 };
  894. //% const VALUE_TYPE kValues[] = { VAL1, VAL2 };
  895. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  896. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  897. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  898. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  899. //% XCTAssertNotNil(dict);
  900. //%
  901. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 = [dict copy];
  902. //% XCTAssertNotNil(dict2);
  903. //%
  904. //% // Should be new object but equal.
  905. //% XCTAssertNotEqual(dict, dict2);
  906. //% XCTAssertEqualObjects(dict, dict2);
  907. //% XCTAssertTrue([dict2 isKindOfClass:[GPB##KEY_NAME##VALUE_NAME##Dictionary class]]);
  908. //%
  909. //% [dict2 release];
  910. //% [dict release];
  911. //%}
  912. //%
  913. //%- (void)testDictionaryFromDictionary {
  914. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2 };
  915. //% const VALUE_TYPE kValues[] = { VAL1, VAL2 };
  916. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  917. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  918. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  919. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  920. //% XCTAssertNotNil(dict);
  921. //%
  922. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  923. //% [GPB##KEY_NAME##VALUE_NAME##Dictionary dictionaryWithDictionary:dict];
  924. //% XCTAssertNotNil(dict2);
  925. //%
  926. //% // Should be new pointer, but equal objects.
  927. //% XCTAssertNotEqual(dict, dict2);
  928. //% XCTAssertEqualObjects(dict, dict2);
  929. //% [dict release];
  930. //%}
  931. //%
  932. //%- (void)testAdds {
  933. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict = [GPB##KEY_NAME##VALUE_NAME##Dictionary dictionary];
  934. //% XCTAssertNotNil(dict);
  935. //%
  936. //% XCTAssertEqual(dict.count, 0U);
  937. //% [dict setValue:VAL1 forKey:KEY1];
  938. //% XCTAssertEqual(dict.count, 1U);
  939. //%
  940. //% const KEY_TYPE KisP##kKeys[] = { KEY2 };
  941. //% const VALUE_TYPE kValues[] = { VAL2 };
  942. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  943. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  944. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  945. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  946. //% XCTAssertNotNil(dict2);
  947. //% [dict addEntriesFromDictionary:dict2];
  948. //% XCTAssertEqual(dict.count, 2U);
  949. //%
  950. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  951. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  952. //% [dict2 release];
  953. //%}
  954. //%
  955. //%- (void)testRemove {
  956. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2};
  957. //% const VALUE_TYPE kValues[] = { VAL1, VAL2 };
  958. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  959. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  960. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  961. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  962. //% XCTAssertNotNil(dict);
  963. //% XCTAssertEqual(dict.count, 2U);
  964. //%
  965. //% [dict removeValueForKey:KEY2];
  966. //% XCTAssertEqual(dict.count, 1U);
  967. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  968. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  969. //%
  970. //% // Remove again does nothing.
  971. //% [dict removeValueForKey:KEY2];
  972. //% XCTAssertEqual(dict.count, 1U);
  973. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  974. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  975. //%
  976. //% [dict removeAll];
  977. //% XCTAssertEqual(dict.count, 0U);
  978. //%VALUE_NOT_FOUND##VHELPER(dict, KEY1)
  979. //%VALUE_NOT_FOUND##VHELPER(dict, KEY2)
  980. //% [dict release];
  981. //%}
  982. //%
  983. //%- (void)testInplaceMutation {
  984. //% const KEY_TYPE KisP##kKeys[] = { KEY1, KEY2 };
  985. //% const VALUE_TYPE kValues[] = { VAL1, VAL2 };
  986. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict =
  987. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues
  988. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys
  989. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues)];
  990. //% XCTAssertNotNil(dict);
  991. //% XCTAssertEqual(dict.count, 2U);
  992. //%DECLARE_VALUE_STORAGE##VHELPER(VALUE_TYPE, value)TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  993. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  994. //%
  995. //% [dict setValue:VAL2 forKey:KEY1];
  996. //% XCTAssertEqual(dict.count, 2U);
  997. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL2)
  998. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  999. //%
  1000. //% [dict setValue:VAL1 forKey:KEY2];
  1001. //% XCTAssertEqual(dict.count, 2U);
  1002. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL2)
  1003. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL1)
  1004. //%
  1005. //% const KEY_TYPE KisP##kKeys2[] = { KEY2, KEY1 };
  1006. //% const VALUE_TYPE kValues2[] = { VAL2, VAL1 };
  1007. //% GPB##KEY_NAME##VALUE_NAME##Dictionary *dict2 =
  1008. //% [[GPB##KEY_NAME##VALUE_NAME##Dictionary alloc] initWithValues:kValues2
  1009. //% KEY_NAME$S VALUE_NAME$S forKeys:kKeys2
  1010. //% KEY_NAME$S VALUE_NAME$S count:GPBARRAYSIZE(kValues2)];
  1011. //% XCTAssertNotNil(dict2);
  1012. //% [dict addEntriesFromDictionary:dict2];
  1013. //% XCTAssertEqual(dict.count, 2U);
  1014. //%TEST_VALUE##VHELPER(dict, value, KEY1, VAL1)
  1015. //%TEST_VALUE##VHELPER(dict, value, KEY2, VAL2)
  1016. //%
  1017. //% [dict2 release];
  1018. //% [dict release];
  1019. //%}
  1020. //%
  1021. //%@end
  1022. //%