GeneratedMessageTest.cs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633
  1. #region Copyright notice and license
  2. // Protocol Buffers - Google's data interchange format
  3. // Copyright 2015 Google Inc. All rights reserved.
  4. // https://developers.google.com/protocol-buffers/
  5. //
  6. // Redistribution and use in source and binary forms, with or without
  7. // modification, are permitted provided that the following conditions are
  8. // met:
  9. //
  10. // * Redistributions of source code must retain the above copyright
  11. // notice, this list of conditions and the following disclaimer.
  12. // * Redistributions in binary form must reproduce the above
  13. // copyright notice, this list of conditions and the following disclaimer
  14. // in the documentation and/or other materials provided with the
  15. // distribution.
  16. // * Neither the name of Google Inc. nor the names of its
  17. // contributors may be used to endorse or promote products derived from
  18. // this software without specific prior written permission.
  19. //
  20. // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  21. // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  22. // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  23. // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  24. // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  25. // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  26. // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  27. // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  28. // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  29. // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  30. // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  31. #endregion
  32. using System;
  33. using System.IO;
  34. using Google.Protobuf.TestProtos;
  35. using NUnit.Framework;
  36. using System.Collections;
  37. using System.Collections.Generic;
  38. using System.Linq;
  39. using Google.Protobuf.WellKnownTypes;
  40. namespace Google.Protobuf
  41. {
  42. /// <summary>
  43. /// Tests around the generated TestAllTypes message.
  44. /// </summary>
  45. public class GeneratedMessageTest
  46. {
  47. [Test]
  48. public void EmptyMessageFieldDistinctFromMissingMessageField()
  49. {
  50. // This demonstrates what we're really interested in...
  51. var message1 = new TestAllTypes { SingleForeignMessage = new ForeignMessage() };
  52. var message2 = new TestAllTypes(); // SingleForeignMessage is null
  53. EqualityTester.AssertInequality(message1, message2);
  54. }
  55. [Test]
  56. public void DefaultValues()
  57. {
  58. // Single fields
  59. var message = new TestAllTypes();
  60. Assert.AreEqual(false, message.SingleBool);
  61. Assert.AreEqual(ByteString.Empty, message.SingleBytes);
  62. Assert.AreEqual(0.0, message.SingleDouble);
  63. Assert.AreEqual(0, message.SingleFixed32);
  64. Assert.AreEqual(0L, message.SingleFixed64);
  65. Assert.AreEqual(0.0f, message.SingleFloat);
  66. Assert.AreEqual(ForeignEnum.FOREIGN_UNSPECIFIED, message.SingleForeignEnum);
  67. Assert.IsNull(message.SingleForeignMessage);
  68. Assert.AreEqual(ImportEnum.IMPORT_ENUM_UNSPECIFIED, message.SingleImportEnum);
  69. Assert.IsNull(message.SingleImportMessage);
  70. Assert.AreEqual(0, message.SingleInt32);
  71. Assert.AreEqual(0L, message.SingleInt64);
  72. Assert.AreEqual(TestAllTypes.Types.NestedEnum.NESTED_ENUM_UNSPECIFIED, message.SingleNestedEnum);
  73. Assert.IsNull(message.SingleNestedMessage);
  74. Assert.IsNull(message.SinglePublicImportMessage);
  75. Assert.AreEqual(0, message.SingleSfixed32);
  76. Assert.AreEqual(0L, message.SingleSfixed64);
  77. Assert.AreEqual(0, message.SingleSint32);
  78. Assert.AreEqual(0L, message.SingleSint64);
  79. Assert.AreEqual("", message.SingleString);
  80. Assert.AreEqual(0U, message.SingleUint32);
  81. Assert.AreEqual(0UL, message.SingleUint64);
  82. // Repeated fields
  83. Assert.AreEqual(0, message.RepeatedBool.Count);
  84. Assert.AreEqual(0, message.RepeatedBytes.Count);
  85. Assert.AreEqual(0, message.RepeatedDouble.Count);
  86. Assert.AreEqual(0, message.RepeatedFixed32.Count);
  87. Assert.AreEqual(0, message.RepeatedFixed64.Count);
  88. Assert.AreEqual(0, message.RepeatedFloat.Count);
  89. Assert.AreEqual(0, message.RepeatedForeignEnum.Count);
  90. Assert.AreEqual(0, message.RepeatedForeignMessage.Count);
  91. Assert.AreEqual(0, message.RepeatedImportEnum.Count);
  92. Assert.AreEqual(0, message.RepeatedImportMessage.Count);
  93. Assert.AreEqual(0, message.RepeatedNestedEnum.Count);
  94. Assert.AreEqual(0, message.RepeatedNestedMessage.Count);
  95. Assert.AreEqual(0, message.RepeatedPublicImportMessage.Count);
  96. Assert.AreEqual(0, message.RepeatedSfixed32.Count);
  97. Assert.AreEqual(0, message.RepeatedSfixed64.Count);
  98. Assert.AreEqual(0, message.RepeatedSint32.Count);
  99. Assert.AreEqual(0, message.RepeatedSint64.Count);
  100. Assert.AreEqual(0, message.RepeatedString.Count);
  101. Assert.AreEqual(0, message.RepeatedUint32.Count);
  102. Assert.AreEqual(0, message.RepeatedUint64.Count);
  103. // Oneof fields
  104. Assert.AreEqual(TestAllTypes.OneofFieldOneofCase.None, message.OneofFieldCase);
  105. Assert.AreEqual(0, message.OneofUint32);
  106. Assert.AreEqual("", message.OneofString);
  107. Assert.AreEqual(ByteString.Empty, message.OneofBytes);
  108. Assert.IsNull(message.OneofNestedMessage);
  109. }
  110. [Test]
  111. public void NullStringAndBytesRejected()
  112. {
  113. var message = new TestAllTypes();
  114. Assert.Throws<ArgumentNullException>(() => message.SingleString = null);
  115. Assert.Throws<ArgumentNullException>(() => message.OneofString = null);
  116. Assert.Throws<ArgumentNullException>(() => message.SingleBytes = null);
  117. Assert.Throws<ArgumentNullException>(() => message.OneofBytes = null);
  118. }
  119. [Test]
  120. public void RoundTrip_Empty()
  121. {
  122. var message = new TestAllTypes();
  123. // Without setting any values, there's nothing to write.
  124. byte[] bytes = message.ToByteArray();
  125. Assert.AreEqual(0, bytes.Length);
  126. TestAllTypes parsed = TestAllTypes.Parser.ParseFrom(bytes);
  127. Assert.AreEqual(message, parsed);
  128. }
  129. [Test]
  130. public void RoundTrip_SingleValues()
  131. {
  132. var message = new TestAllTypes
  133. {
  134. SingleBool = true,
  135. SingleBytes = ByteString.CopyFrom(1, 2, 3, 4),
  136. SingleDouble = 23.5,
  137. SingleFixed32 = 23,
  138. SingleFixed64 = 1234567890123,
  139. SingleFloat = 12.25f,
  140. SingleForeignEnum = ForeignEnum.FOREIGN_BAR,
  141. SingleForeignMessage = new ForeignMessage { C = 10 },
  142. SingleImportEnum = ImportEnum.IMPORT_BAZ,
  143. SingleImportMessage = new ImportMessage { D = 20 },
  144. SingleInt32 = 100,
  145. SingleInt64 = 3210987654321,
  146. SingleNestedEnum = TestAllTypes.Types.NestedEnum.FOO,
  147. SingleNestedMessage = new TestAllTypes.Types.NestedMessage { Bb = 35 },
  148. SinglePublicImportMessage = new PublicImportMessage { E = 54 },
  149. SingleSfixed32 = -123,
  150. SingleSfixed64 = -12345678901234,
  151. SingleSint32 = -456,
  152. SingleSint64 = -12345678901235,
  153. SingleString = "test",
  154. SingleUint32 = uint.MaxValue,
  155. SingleUint64 = ulong.MaxValue
  156. };
  157. byte[] bytes = message.ToByteArray();
  158. TestAllTypes parsed = TestAllTypes.Parser.ParseFrom(bytes);
  159. Assert.AreEqual(message, parsed);
  160. }
  161. [Test]
  162. public void RoundTrip_RepeatedValues()
  163. {
  164. var message = new TestAllTypes
  165. {
  166. RepeatedBool = { true, false },
  167. RepeatedBytes = { ByteString.CopyFrom(1, 2, 3, 4), ByteString.CopyFrom(5, 6) },
  168. RepeatedDouble = { -12.25, 23.5 },
  169. RepeatedFixed32 = { uint.MaxValue, 23 },
  170. RepeatedFixed64 = { ulong.MaxValue, 1234567890123 },
  171. RepeatedFloat = { 100f, 12.25f },
  172. RepeatedForeignEnum = { ForeignEnum.FOREIGN_FOO, ForeignEnum.FOREIGN_BAR },
  173. RepeatedForeignMessage = { new ForeignMessage(), new ForeignMessage { C = 10 } },
  174. RepeatedImportEnum = { ImportEnum.IMPORT_BAZ, ImportEnum.IMPORT_ENUM_UNSPECIFIED },
  175. RepeatedImportMessage = { new ImportMessage { D = 20 }, new ImportMessage { D = 25 } },
  176. RepeatedInt32 = { 100, 200 },
  177. RepeatedInt64 = { 3210987654321, long.MaxValue },
  178. RepeatedNestedEnum = { TestAllTypes.Types.NestedEnum.FOO, TestAllTypes.Types.NestedEnum.NEG },
  179. RepeatedNestedMessage = { new TestAllTypes.Types.NestedMessage { Bb = 35 }, new TestAllTypes.Types.NestedMessage { Bb = 10 } },
  180. RepeatedPublicImportMessage = { new PublicImportMessage { E = 54 }, new PublicImportMessage { E = -1 } },
  181. RepeatedSfixed32 = { -123, 123 },
  182. RepeatedSfixed64 = { -12345678901234, 12345678901234 },
  183. RepeatedSint32 = { -456, 100 },
  184. RepeatedSint64 = { -12345678901235, 123 },
  185. RepeatedString = { "foo", "bar" },
  186. RepeatedUint32 = { uint.MaxValue, uint.MinValue },
  187. RepeatedUint64 = { ulong.MaxValue, uint.MinValue }
  188. };
  189. byte[] bytes = message.ToByteArray();
  190. TestAllTypes parsed = TestAllTypes.Parser.ParseFrom(bytes);
  191. Assert.AreEqual(message, parsed);
  192. }
  193. // Note that not every map within map_unittest_proto3 is used. They all go through very
  194. // similar code paths. The fact that all maps are present is validation that we have codecs
  195. // for every type.
  196. [Test]
  197. public void RoundTrip_Maps()
  198. {
  199. var message = new TestMap
  200. {
  201. MapBoolBool = {
  202. { false, true },
  203. { true, false }
  204. },
  205. MapInt32Bytes = {
  206. { 5, ByteString.CopyFrom(6, 7, 8) },
  207. { 25, ByteString.CopyFrom(1, 2, 3, 4, 5) },
  208. { 10, ByteString.Empty }
  209. },
  210. MapInt32ForeignMessage = {
  211. { 0, new ForeignMessage { C = 10 } },
  212. { 5, null },
  213. },
  214. MapInt32Enum = {
  215. { 1, MapEnum.MAP_ENUM_BAR },
  216. { 2000, MapEnum.MAP_ENUM_FOO }
  217. }
  218. };
  219. byte[] bytes = message.ToByteArray();
  220. TestMap parsed = TestMap.Parser.ParseFrom(bytes);
  221. Assert.AreEqual(message, parsed);
  222. }
  223. [Test]
  224. public void MapWithEmptyEntry()
  225. {
  226. var message = new TestMap
  227. {
  228. MapInt32Bytes = { { 0, ByteString.Empty } }
  229. };
  230. byte[] bytes = message.ToByteArray();
  231. Assert.AreEqual(2, bytes.Length); // Tag for field entry (1 byte), length of entry (0; 1 byte)
  232. var parsed = TestMap.Parser.ParseFrom(bytes);
  233. Assert.AreEqual(1, parsed.MapInt32Bytes.Count);
  234. Assert.AreEqual(ByteString.Empty, parsed.MapInt32Bytes[0]);
  235. }
  236. [Test]
  237. public void MapWithOnlyValue()
  238. {
  239. // Hand-craft the stream to contain a single entry with just a value.
  240. var memoryStream = new MemoryStream();
  241. var output = new CodedOutputStream(memoryStream);
  242. output.WriteTag(TestMap.MapInt32ForeignMessageFieldNumber, WireFormat.WireType.LengthDelimited);
  243. var nestedMessage = new ForeignMessage { C = 20 };
  244. // Size of the entry (tag, size written by WriteMessage, data written by WriteMessage)
  245. output.WriteLength(2 + nestedMessage.CalculateSize());
  246. output.WriteTag(2, WireFormat.WireType.LengthDelimited);
  247. output.WriteMessage(nestedMessage);
  248. output.Flush();
  249. var parsed = TestMap.Parser.ParseFrom(memoryStream.ToArray());
  250. Assert.AreEqual(nestedMessage, parsed.MapInt32ForeignMessage[0]);
  251. }
  252. [Test]
  253. public void MapIgnoresExtraFieldsWithinEntryMessages()
  254. {
  255. // Hand-craft the stream to contain a single entry with three fields
  256. var memoryStream = new MemoryStream();
  257. var output = new CodedOutputStream(memoryStream);
  258. output.WriteTag(TestMap.MapInt32Int32FieldNumber, WireFormat.WireType.LengthDelimited);
  259. var key = 10; // Field 1
  260. var value = 20; // Field 2
  261. var extra = 30; // Field 3
  262. // Each field can be represented in a single byte, with a single byte tag.
  263. // Total message size: 6 bytes.
  264. output.WriteLength(6);
  265. output.WriteTag(1, WireFormat.WireType.Varint);
  266. output.WriteInt32(key);
  267. output.WriteTag(2, WireFormat.WireType.Varint);
  268. output.WriteInt32(value);
  269. output.WriteTag(3, WireFormat.WireType.Varint);
  270. output.WriteInt32(extra);
  271. output.Flush();
  272. var parsed = TestMap.Parser.ParseFrom(memoryStream.ToArray());
  273. Assert.AreEqual(value, parsed.MapInt32Int32[key]);
  274. }
  275. [Test]
  276. public void MapFieldOrderIsIrrelevant()
  277. {
  278. var memoryStream = new MemoryStream();
  279. var output = new CodedOutputStream(memoryStream);
  280. output.WriteTag(TestMap.MapInt32Int32FieldNumber, WireFormat.WireType.LengthDelimited);
  281. var key = 10;
  282. var value = 20;
  283. // Each field can be represented in a single byte, with a single byte tag.
  284. // Total message size: 4 bytes.
  285. output.WriteLength(4);
  286. output.WriteTag(2, WireFormat.WireType.Varint);
  287. output.WriteInt32(value);
  288. output.WriteTag(1, WireFormat.WireType.Varint);
  289. output.WriteInt32(key);
  290. output.Flush();
  291. var parsed = TestMap.Parser.ParseFrom(memoryStream.ToArray());
  292. Assert.AreEqual(value, parsed.MapInt32Int32[key]);
  293. }
  294. [Test]
  295. public void MapNonContiguousEntries()
  296. {
  297. var memoryStream = new MemoryStream();
  298. var output = new CodedOutputStream(memoryStream);
  299. // Message structure:
  300. // Entry for MapInt32Int32
  301. // Entry for MapStringString
  302. // Entry for MapInt32Int32
  303. // First entry
  304. var key1 = 10;
  305. var value1 = 20;
  306. output.WriteTag(TestMap.MapInt32Int32FieldNumber, WireFormat.WireType.LengthDelimited);
  307. output.WriteLength(4);
  308. output.WriteTag(1, WireFormat.WireType.Varint);
  309. output.WriteInt32(key1);
  310. output.WriteTag(2, WireFormat.WireType.Varint);
  311. output.WriteInt32(value1);
  312. // Second entry
  313. var key2 = "a";
  314. var value2 = "b";
  315. output.WriteTag(TestMap.MapStringStringFieldNumber, WireFormat.WireType.LengthDelimited);
  316. output.WriteLength(6); // 3 bytes per entry: tag, size, character
  317. output.WriteTag(1, WireFormat.WireType.LengthDelimited);
  318. output.WriteString(key2);
  319. output.WriteTag(2, WireFormat.WireType.LengthDelimited);
  320. output.WriteString(value2);
  321. // Third entry
  322. var key3 = 15;
  323. var value3 = 25;
  324. output.WriteTag(TestMap.MapInt32Int32FieldNumber, WireFormat.WireType.LengthDelimited);
  325. output.WriteLength(4);
  326. output.WriteTag(1, WireFormat.WireType.Varint);
  327. output.WriteInt32(key3);
  328. output.WriteTag(2, WireFormat.WireType.Varint);
  329. output.WriteInt32(value3);
  330. output.Flush();
  331. var parsed = TestMap.Parser.ParseFrom(memoryStream.ToArray());
  332. var expected = new TestMap
  333. {
  334. MapInt32Int32 = { { key1, value1 }, { key3, value3 } },
  335. MapStringString = { { key2, value2 } }
  336. };
  337. Assert.AreEqual(expected, parsed);
  338. }
  339. [Test]
  340. public void DuplicateKeys_LastEntryWins()
  341. {
  342. var memoryStream = new MemoryStream();
  343. var output = new CodedOutputStream(memoryStream);
  344. var key = 10;
  345. var value1 = 20;
  346. var value2 = 30;
  347. // First entry
  348. output.WriteTag(TestMap.MapInt32Int32FieldNumber, WireFormat.WireType.LengthDelimited);
  349. output.WriteLength(4);
  350. output.WriteTag(1, WireFormat.WireType.Varint);
  351. output.WriteInt32(key);
  352. output.WriteTag(2, WireFormat.WireType.Varint);
  353. output.WriteInt32(value1);
  354. // Second entry - same key, different value
  355. output.WriteTag(TestMap.MapInt32Int32FieldNumber, WireFormat.WireType.LengthDelimited);
  356. output.WriteLength(4);
  357. output.WriteTag(1, WireFormat.WireType.Varint);
  358. output.WriteInt32(key);
  359. output.WriteTag(2, WireFormat.WireType.Varint);
  360. output.WriteInt32(value2);
  361. output.Flush();
  362. var parsed = TestMap.Parser.ParseFrom(memoryStream.ToArray());
  363. Assert.AreEqual(value2, parsed.MapInt32Int32[key]);
  364. }
  365. [Test]
  366. public void CloneSingleNonMessageValues()
  367. {
  368. var original = new TestAllTypes
  369. {
  370. SingleBool = true,
  371. SingleBytes = ByteString.CopyFrom(1, 2, 3, 4),
  372. SingleDouble = 23.5,
  373. SingleFixed32 = 23,
  374. SingleFixed64 = 1234567890123,
  375. SingleFloat = 12.25f,
  376. SingleInt32 = 100,
  377. SingleInt64 = 3210987654321,
  378. SingleNestedEnum = TestAllTypes.Types.NestedEnum.FOO,
  379. SingleSfixed32 = -123,
  380. SingleSfixed64 = -12345678901234,
  381. SingleSint32 = -456,
  382. SingleSint64 = -12345678901235,
  383. SingleString = "test",
  384. SingleUint32 = uint.MaxValue,
  385. SingleUint64 = ulong.MaxValue
  386. };
  387. var clone = original.Clone();
  388. Assert.AreNotSame(original, clone);
  389. Assert.AreEqual(original, clone);
  390. // Just as a single example
  391. clone.SingleInt32 = 150;
  392. Assert.AreNotEqual(original, clone);
  393. }
  394. [Test]
  395. public void CloneRepeatedNonMessageValues()
  396. {
  397. var original = new TestAllTypes
  398. {
  399. RepeatedBool = { true, false },
  400. RepeatedBytes = { ByteString.CopyFrom(1, 2, 3, 4), ByteString.CopyFrom(5, 6) },
  401. RepeatedDouble = { -12.25, 23.5 },
  402. RepeatedFixed32 = { uint.MaxValue, 23 },
  403. RepeatedFixed64 = { ulong.MaxValue, 1234567890123 },
  404. RepeatedFloat = { 100f, 12.25f },
  405. RepeatedInt32 = { 100, 200 },
  406. RepeatedInt64 = { 3210987654321, long.MaxValue },
  407. RepeatedNestedEnum = { TestAllTypes.Types.NestedEnum.FOO, TestAllTypes.Types.NestedEnum.NEG },
  408. RepeatedSfixed32 = { -123, 123 },
  409. RepeatedSfixed64 = { -12345678901234, 12345678901234 },
  410. RepeatedSint32 = { -456, 100 },
  411. RepeatedSint64 = { -12345678901235, 123 },
  412. RepeatedString = { "foo", "bar" },
  413. RepeatedUint32 = { uint.MaxValue, uint.MinValue },
  414. RepeatedUint64 = { ulong.MaxValue, uint.MinValue }
  415. };
  416. var clone = original.Clone();
  417. Assert.AreNotSame(original, clone);
  418. Assert.AreEqual(original, clone);
  419. // Just as a single example
  420. clone.RepeatedDouble.Add(25.5);
  421. Assert.AreNotEqual(original, clone);
  422. }
  423. [Test]
  424. public void CloneSingleMessageField()
  425. {
  426. var original = new TestAllTypes
  427. {
  428. SingleNestedMessage = new TestAllTypes.Types.NestedMessage { Bb = 20 }
  429. };
  430. var clone = original.Clone();
  431. Assert.AreNotSame(original, clone);
  432. Assert.AreNotSame(original.SingleNestedMessage, clone.SingleNestedMessage);
  433. Assert.AreEqual(original, clone);
  434. clone.SingleNestedMessage.Bb = 30;
  435. Assert.AreNotEqual(original, clone);
  436. }
  437. [Test]
  438. public void CloneRepeatedMessageField()
  439. {
  440. var original = new TestAllTypes
  441. {
  442. RepeatedNestedMessage = { new TestAllTypes.Types.NestedMessage { Bb = 20 } }
  443. };
  444. var clone = original.Clone();
  445. Assert.AreNotSame(original, clone);
  446. Assert.AreNotSame(original.RepeatedNestedMessage, clone.RepeatedNestedMessage);
  447. Assert.AreNotSame(original.RepeatedNestedMessage[0], clone.RepeatedNestedMessage[0]);
  448. Assert.AreEqual(original, clone);
  449. clone.RepeatedNestedMessage[0].Bb = 30;
  450. Assert.AreNotEqual(original, clone);
  451. }
  452. [Test]
  453. public void CloneOneofField()
  454. {
  455. var original = new TestAllTypes
  456. {
  457. OneofNestedMessage = new TestAllTypes.Types.NestedMessage { Bb = 20 }
  458. };
  459. var clone = original.Clone();
  460. Assert.AreNotSame(original, clone);
  461. Assert.AreEqual(original, clone);
  462. // We should have cloned the message
  463. original.OneofNestedMessage.Bb = 30;
  464. Assert.AreNotEqual(original, clone);
  465. }
  466. [Test]
  467. public void OneofProperties()
  468. {
  469. // Switch the oneof case between each of the different options, and check everything behaves
  470. // as expected in each case.
  471. var message = new TestAllTypes();
  472. Assert.AreEqual("", message.OneofString);
  473. Assert.AreEqual(0, message.OneofUint32);
  474. Assert.AreEqual(ByteString.Empty, message.OneofBytes);
  475. Assert.IsNull(message.OneofNestedMessage);
  476. Assert.AreEqual(TestAllTypes.OneofFieldOneofCase.None, message.OneofFieldCase);
  477. message.OneofString = "sample";
  478. Assert.AreEqual("sample", message.OneofString);
  479. Assert.AreEqual(0, message.OneofUint32);
  480. Assert.AreEqual(ByteString.Empty, message.OneofBytes);
  481. Assert.IsNull(message.OneofNestedMessage);
  482. Assert.AreEqual(TestAllTypes.OneofFieldOneofCase.OneofString, message.OneofFieldCase);
  483. var bytes = ByteString.CopyFrom(1, 2, 3);
  484. message.OneofBytes = bytes;
  485. Assert.AreEqual("", message.OneofString);
  486. Assert.AreEqual(0, message.OneofUint32);
  487. Assert.AreEqual(bytes, message.OneofBytes);
  488. Assert.IsNull(message.OneofNestedMessage);
  489. Assert.AreEqual(TestAllTypes.OneofFieldOneofCase.OneofBytes, message.OneofFieldCase);
  490. message.OneofUint32 = 20;
  491. Assert.AreEqual("", message.OneofString);
  492. Assert.AreEqual(20, message.OneofUint32);
  493. Assert.AreEqual(ByteString.Empty, message.OneofBytes);
  494. Assert.IsNull(message.OneofNestedMessage);
  495. Assert.AreEqual(TestAllTypes.OneofFieldOneofCase.OneofUint32, message.OneofFieldCase);
  496. var nestedMessage = new TestAllTypes.Types.NestedMessage { Bb = 25 };
  497. message.OneofNestedMessage = nestedMessage;
  498. Assert.AreEqual("", message.OneofString);
  499. Assert.AreEqual(0, message.OneofUint32);
  500. Assert.AreEqual(ByteString.Empty, message.OneofBytes);
  501. Assert.AreEqual(nestedMessage, message.OneofNestedMessage);
  502. Assert.AreEqual(TestAllTypes.OneofFieldOneofCase.OneofNestedMessage, message.OneofFieldCase);
  503. message.ClearOneofField();
  504. Assert.AreEqual("", message.OneofString);
  505. Assert.AreEqual(0, message.OneofUint32);
  506. Assert.AreEqual(ByteString.Empty, message.OneofBytes);
  507. Assert.IsNull(message.OneofNestedMessage);
  508. Assert.AreEqual(TestAllTypes.OneofFieldOneofCase.None, message.OneofFieldCase);
  509. }
  510. [Test]
  511. public void OneofSerialization_NonDefaultValue()
  512. {
  513. var message = new TestAllTypes();
  514. message.OneofString = "this would take a bit of space";
  515. message.OneofUint32 = 10;
  516. var bytes = message.ToByteArray();
  517. Assert.AreEqual(3, bytes.Length); // 2 bytes for the tag + 1 for the value - no string!
  518. var message2 = TestAllTypes.Parser.ParseFrom(bytes);
  519. Assert.AreEqual(message, message2);
  520. Assert.AreEqual(TestAllTypes.OneofFieldOneofCase.OneofUint32, message2.OneofFieldCase);
  521. }
  522. [Test]
  523. public void OneofSerialization_DefaultValue()
  524. {
  525. var message = new TestAllTypes();
  526. message.OneofString = "this would take a bit of space";
  527. message.OneofUint32 = 0; // This is the default value for UInt32; normally wouldn't be serialized
  528. var bytes = message.ToByteArray();
  529. Assert.AreEqual(3, bytes.Length); // 2 bytes for the tag + 1 for the value - it's still serialized
  530. var message2 = TestAllTypes.Parser.ParseFrom(bytes);
  531. Assert.AreEqual(message, message2);
  532. Assert.AreEqual(TestAllTypes.OneofFieldOneofCase.OneofUint32, message2.OneofFieldCase);
  533. }
  534. [Test]
  535. public void IgnoreUnknownFields_RealDataStillRead()
  536. {
  537. var message = SampleMessages.CreateFullTestAllTypes();
  538. var stream = new MemoryStream();
  539. var output = new CodedOutputStream(stream);
  540. var unusedFieldNumber = 23456;
  541. Assert.IsFalse(TestAllTypes.Descriptor.Fields.InDeclarationOrder().Select(x => x.FieldNumber).Contains(unusedFieldNumber));
  542. output.WriteTag(unusedFieldNumber, WireFormat.WireType.LengthDelimited);
  543. output.WriteString("ignore me");
  544. message.WriteTo(output);
  545. output.Flush();
  546. stream.Position = 0;
  547. var parsed = TestAllTypes.Parser.ParseFrom(stream);
  548. Assert.AreEqual(message, parsed);
  549. }
  550. [Test]
  551. public void IgnoreUnknownFields_AllTypes()
  552. {
  553. // Simple way of ensuring we can skip all kinds of fields.
  554. var data = SampleMessages.CreateFullTestAllTypes().ToByteArray();
  555. var empty = Empty.Parser.ParseFrom(data);
  556. Assert.AreEqual(new Empty(), empty);
  557. }
  558. // This was originally seen as a conformance test failure.
  559. [Test]
  560. public void TruncatedMessageFieldThrows()
  561. {
  562. // 130, 3 is the message tag
  563. // 1 is the data length - but there's no data.
  564. var data = new byte[] { 130, 3, 1 };
  565. Assert.Throws<InvalidProtocolBufferException>(() => TestAllTypes.Parser.ParseFrom(data));
  566. }
  567. }
  568. }