DescriptorsTest.cs 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457
  1. #region Copyright notice and license
  2. // Protocol Buffers - Google's data interchange format
  3. // Copyright 2008 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 Google.Protobuf.TestProtos;
  33. using NUnit.Framework;
  34. using ProtobufUnittest;
  35. using System;
  36. using System.Collections.Generic;
  37. using System.IO;
  38. using System.Linq;
  39. using UnitTest.Issues.TestProtos;
  40. namespace Google.Protobuf.Reflection
  41. {
  42. /// <summary>
  43. /// Tests for descriptors. (Not in its own namespace or broken up into individual classes as the
  44. /// size doesn't warrant it. On the other hand, this makes me feel a bit dirty...)
  45. /// </summary>
  46. public class DescriptorsTest
  47. {
  48. [Test]
  49. public void FileDescriptor_GeneratedCode()
  50. {
  51. TestFileDescriptor(
  52. UnittestProto3Reflection.Descriptor,
  53. UnittestImportProto3Reflection.Descriptor,
  54. UnittestImportPublicProto3Reflection.Descriptor);
  55. }
  56. [Test]
  57. public void FileDescriptor_BuildFromByteStrings()
  58. {
  59. // The descriptors have to be supplied in an order such that all the
  60. // dependencies come before the descriptors depending on them.
  61. var descriptorData = new List<ByteString>
  62. {
  63. UnittestImportPublicProto3Reflection.Descriptor.SerializedData,
  64. UnittestImportProto3Reflection.Descriptor.SerializedData,
  65. UnittestProto3Reflection.Descriptor.SerializedData
  66. };
  67. var converted = FileDescriptor.BuildFromByteStrings(descriptorData);
  68. Assert.AreEqual(3, converted.Count);
  69. TestFileDescriptor(converted[2], converted[1], converted[0]);
  70. }
  71. [Test]
  72. public void FileDescriptor_BuildFromByteStrings_WithExtensionRegistry()
  73. {
  74. var extension = UnittestCustomOptionsProto3Extensions.MessageOpt1;
  75. var byteStrings = new[]
  76. {
  77. DescriptorReflection.Descriptor.Proto.ToByteString(),
  78. UnittestCustomOptionsProto3Reflection.Descriptor.Proto.ToByteString()
  79. };
  80. var registry = new ExtensionRegistry { extension };
  81. var descriptor = FileDescriptor.BuildFromByteStrings(byteStrings, registry).Last();
  82. var message = descriptor.MessageTypes.Single(t => t.Name == nameof(TestMessageWithCustomOptions));
  83. var extensionValue = message.GetOptions().GetExtension(extension);
  84. Assert.AreEqual(-56, extensionValue);
  85. }
  86. private void TestFileDescriptor(FileDescriptor file, FileDescriptor importedFile, FileDescriptor importedPublicFile)
  87. {
  88. Assert.AreEqual("unittest_proto3.proto", file.Name);
  89. Assert.AreEqual("protobuf_unittest3", file.Package);
  90. Assert.AreEqual("UnittestProto", file.Proto.Options.JavaOuterClassname);
  91. Assert.AreEqual("unittest_proto3.proto", file.Proto.Name);
  92. // unittest_proto3.proto doesn't have any public imports, but unittest_import_proto3.proto does.
  93. Assert.AreEqual(0, file.PublicDependencies.Count);
  94. Assert.AreEqual(1, importedFile.PublicDependencies.Count);
  95. Assert.AreEqual(importedPublicFile, importedFile.PublicDependencies[0]);
  96. Assert.AreEqual(1, file.Dependencies.Count);
  97. Assert.AreEqual(importedFile, file.Dependencies[0]);
  98. Assert.Null(file.FindTypeByName<MessageDescriptor>("NoSuchType"));
  99. Assert.Null(file.FindTypeByName<MessageDescriptor>("protobuf_unittest3.TestAllTypes"));
  100. for (int i = 0; i < file.MessageTypes.Count; i++)
  101. {
  102. Assert.AreEqual(i, file.MessageTypes[i].Index);
  103. }
  104. Assert.AreEqual(file.EnumTypes[0], file.FindTypeByName<EnumDescriptor>("ForeignEnum"));
  105. Assert.Null(file.FindTypeByName<EnumDescriptor>("NoSuchType"));
  106. Assert.Null(file.FindTypeByName<EnumDescriptor>("protobuf_unittest3.ForeignEnum"));
  107. Assert.AreEqual(1, importedFile.EnumTypes.Count);
  108. Assert.AreEqual("ImportEnum", importedFile.EnumTypes[0].Name);
  109. for (int i = 0; i < file.EnumTypes.Count; i++)
  110. {
  111. Assert.AreEqual(i, file.EnumTypes[i].Index);
  112. }
  113. Assert.AreEqual(10, file.SerializedData[0]);
  114. }
  115. [Test]
  116. public void FileDescriptor_NonRootPath()
  117. {
  118. // unittest_proto3.proto used to be in google/protobuf. Now it's in the C#-specific location,
  119. // let's test something that's still in a directory.
  120. FileDescriptor file = UnittestWellKnownTypesReflection.Descriptor;
  121. Assert.AreEqual("google/protobuf/unittest_well_known_types.proto", file.Name);
  122. Assert.AreEqual("protobuf_unittest", file.Package);
  123. }
  124. [Test]
  125. public void FileDescriptor_BuildFromByteStrings_MissingDependency()
  126. {
  127. var descriptorData = new List<ByteString>
  128. {
  129. UnittestImportProto3Reflection.Descriptor.SerializedData,
  130. UnittestProto3Reflection.Descriptor.SerializedData,
  131. };
  132. // This will fail, because we're missing UnittestImportPublicProto3Reflection
  133. Assert.Throws<ArgumentException>(() => FileDescriptor.BuildFromByteStrings(descriptorData));
  134. }
  135. [Test]
  136. public void FileDescriptor_BuildFromByteStrings_DuplicateNames()
  137. {
  138. var descriptorData = new List<ByteString>
  139. {
  140. UnittestImportPublicProto3Reflection.Descriptor.SerializedData,
  141. UnittestImportPublicProto3Reflection.Descriptor.SerializedData,
  142. };
  143. // This will fail due to the same name being used twice
  144. Assert.Throws<ArgumentException>(() => FileDescriptor.BuildFromByteStrings(descriptorData));
  145. }
  146. [Test]
  147. public void FileDescriptor_BuildFromByteStrings_IncorrectOrder()
  148. {
  149. var descriptorData = new List<ByteString>
  150. {
  151. UnittestProto3Reflection.Descriptor.SerializedData,
  152. UnittestImportPublicProto3Reflection.Descriptor.SerializedData,
  153. UnittestImportProto3Reflection.Descriptor.SerializedData
  154. };
  155. // This will fail, because the dependencies should come first
  156. Assert.Throws<ArgumentException>(() => FileDescriptor.BuildFromByteStrings(descriptorData));
  157. }
  158. [Test]
  159. public void MessageDescriptorFromGeneratedCodeFileDescriptor()
  160. {
  161. var file = UnittestProto3Reflection.Descriptor;
  162. MessageDescriptor messageType = TestAllTypes.Descriptor;
  163. Assert.AreSame(typeof(TestAllTypes), messageType.ClrType);
  164. Assert.AreSame(TestAllTypes.Parser, messageType.Parser);
  165. Assert.AreEqual(messageType, file.MessageTypes[0]);
  166. Assert.AreEqual(messageType, file.FindTypeByName<MessageDescriptor>("TestAllTypes"));
  167. }
  168. [Test]
  169. public void MessageDescriptor()
  170. {
  171. MessageDescriptor messageType = TestAllTypes.Descriptor;
  172. MessageDescriptor nestedType = TestAllTypes.Types.NestedMessage.Descriptor;
  173. Assert.AreEqual("TestAllTypes", messageType.Name);
  174. Assert.AreEqual("protobuf_unittest3.TestAllTypes", messageType.FullName);
  175. Assert.AreEqual(UnittestProto3Reflection.Descriptor, messageType.File);
  176. Assert.IsNull(messageType.ContainingType);
  177. Assert.IsNull(messageType.Proto.Options);
  178. Assert.AreEqual("TestAllTypes", messageType.Name);
  179. Assert.AreEqual("NestedMessage", nestedType.Name);
  180. Assert.AreEqual("protobuf_unittest3.TestAllTypes.NestedMessage", nestedType.FullName);
  181. Assert.AreEqual(UnittestProto3Reflection.Descriptor, nestedType.File);
  182. Assert.AreEqual(messageType, nestedType.ContainingType);
  183. FieldDescriptor field = messageType.Fields.InDeclarationOrder()[0];
  184. Assert.AreEqual("single_int32", field.Name);
  185. Assert.AreEqual(field, messageType.FindDescriptor<FieldDescriptor>("single_int32"));
  186. Assert.Null(messageType.FindDescriptor<FieldDescriptor>("no_such_field"));
  187. Assert.AreEqual(field, messageType.FindFieldByNumber(1));
  188. Assert.Null(messageType.FindFieldByNumber(571283));
  189. var fieldsInDeclarationOrder = messageType.Fields.InDeclarationOrder();
  190. for (int i = 0; i < fieldsInDeclarationOrder.Count; i++)
  191. {
  192. Assert.AreEqual(i, fieldsInDeclarationOrder[i].Index);
  193. }
  194. Assert.AreEqual(nestedType, messageType.NestedTypes[0]);
  195. Assert.AreEqual(nestedType, messageType.FindDescriptor<MessageDescriptor>("NestedMessage"));
  196. Assert.Null(messageType.FindDescriptor<MessageDescriptor>("NoSuchType"));
  197. for (int i = 0; i < messageType.NestedTypes.Count; i++)
  198. {
  199. Assert.AreEqual(i, messageType.NestedTypes[i].Index);
  200. }
  201. Assert.AreEqual(messageType.EnumTypes[0], messageType.FindDescriptor<EnumDescriptor>("NestedEnum"));
  202. Assert.Null(messageType.FindDescriptor<EnumDescriptor>("NoSuchType"));
  203. for (int i = 0; i < messageType.EnumTypes.Count; i++)
  204. {
  205. Assert.AreEqual(i, messageType.EnumTypes[i].Index);
  206. }
  207. }
  208. [Test]
  209. public void FieldDescriptor_GeneratedCode()
  210. {
  211. TestFieldDescriptor(UnittestProto3Reflection.Descriptor, TestAllTypes.Descriptor, ForeignMessage.Descriptor, ImportMessage.Descriptor);
  212. }
  213. [Test]
  214. public void FieldDescriptor_BuildFromByteStrings()
  215. {
  216. // The descriptors have to be supplied in an order such that all the
  217. // dependencies come before the descriptors depending on them.
  218. var descriptorData = new List<ByteString>
  219. {
  220. UnittestImportPublicProto3Reflection.Descriptor.SerializedData,
  221. UnittestImportProto3Reflection.Descriptor.SerializedData,
  222. UnittestProto3Reflection.Descriptor.SerializedData
  223. };
  224. var converted = FileDescriptor.BuildFromByteStrings(descriptorData);
  225. TestFieldDescriptor(
  226. converted[2],
  227. converted[2].FindTypeByName<MessageDescriptor>("TestAllTypes"),
  228. converted[2].FindTypeByName<MessageDescriptor>("ForeignMessage"),
  229. converted[1].FindTypeByName<MessageDescriptor>("ImportMessage"));
  230. }
  231. public void TestFieldDescriptor(
  232. FileDescriptor unitTestProto3Descriptor,
  233. MessageDescriptor testAllTypesDescriptor,
  234. MessageDescriptor foreignMessageDescriptor,
  235. MessageDescriptor importMessageDescriptor)
  236. {
  237. FieldDescriptor primitiveField = testAllTypesDescriptor.FindDescriptor<FieldDescriptor>("single_int32");
  238. FieldDescriptor enumField = testAllTypesDescriptor.FindDescriptor<FieldDescriptor>("single_nested_enum");
  239. FieldDescriptor foreignMessageField = testAllTypesDescriptor.FindDescriptor<FieldDescriptor>("single_foreign_message");
  240. FieldDescriptor importMessageField = testAllTypesDescriptor.FindDescriptor<FieldDescriptor>("single_import_message");
  241. FieldDescriptor fieldInOneof = testAllTypesDescriptor.FindDescriptor<FieldDescriptor>("oneof_string");
  242. Assert.AreEqual("single_int32", primitiveField.Name);
  243. Assert.AreEqual("protobuf_unittest3.TestAllTypes.single_int32",
  244. primitiveField.FullName);
  245. Assert.AreEqual(1, primitiveField.FieldNumber);
  246. Assert.AreEqual(testAllTypesDescriptor, primitiveField.ContainingType);
  247. Assert.AreEqual(unitTestProto3Descriptor, primitiveField.File);
  248. Assert.AreEqual(FieldType.Int32, primitiveField.FieldType);
  249. Assert.IsNull(primitiveField.Proto.Options);
  250. Assert.AreEqual("single_nested_enum", enumField.Name);
  251. Assert.AreEqual(FieldType.Enum, enumField.FieldType);
  252. Assert.AreEqual(testAllTypesDescriptor.EnumTypes[0], enumField.EnumType);
  253. Assert.AreEqual("single_foreign_message", foreignMessageField.Name);
  254. Assert.AreEqual(FieldType.Message, foreignMessageField.FieldType);
  255. Assert.AreEqual(foreignMessageDescriptor, foreignMessageField.MessageType);
  256. Assert.AreEqual("single_import_message", importMessageField.Name);
  257. Assert.AreEqual(FieldType.Message, importMessageField.FieldType);
  258. Assert.AreEqual(importMessageDescriptor, importMessageField.MessageType);
  259. // For a field in a regular onoef, ContainingOneof and RealContainingOneof should be the same.
  260. Assert.AreEqual("oneof_field", fieldInOneof.ContainingOneof.Name);
  261. Assert.AreSame(fieldInOneof.ContainingOneof, fieldInOneof.RealContainingOneof);
  262. }
  263. [Test]
  264. public void FieldDescriptorLabel()
  265. {
  266. FieldDescriptor singleField =
  267. TestAllTypes.Descriptor.FindDescriptor<FieldDescriptor>("single_int32");
  268. FieldDescriptor repeatedField =
  269. TestAllTypes.Descriptor.FindDescriptor<FieldDescriptor>("repeated_int32");
  270. Assert.IsFalse(singleField.IsRepeated);
  271. Assert.IsTrue(repeatedField.IsRepeated);
  272. }
  273. [Test]
  274. public void EnumDescriptor()
  275. {
  276. // Note: this test is a bit different to the Java version because there's no static way of getting to the descriptor
  277. EnumDescriptor enumType = UnittestProto3Reflection.Descriptor.FindTypeByName<EnumDescriptor>("ForeignEnum");
  278. EnumDescriptor nestedType = TestAllTypes.Descriptor.FindDescriptor<EnumDescriptor>("NestedEnum");
  279. Assert.AreEqual("ForeignEnum", enumType.Name);
  280. Assert.AreEqual("protobuf_unittest3.ForeignEnum", enumType.FullName);
  281. Assert.AreEqual(UnittestProto3Reflection.Descriptor, enumType.File);
  282. Assert.Null(enumType.ContainingType);
  283. Assert.Null(enumType.Proto.Options);
  284. Assert.AreEqual("NestedEnum", nestedType.Name);
  285. Assert.AreEqual("protobuf_unittest3.TestAllTypes.NestedEnum",
  286. nestedType.FullName);
  287. Assert.AreEqual(UnittestProto3Reflection.Descriptor, nestedType.File);
  288. Assert.AreEqual(TestAllTypes.Descriptor, nestedType.ContainingType);
  289. EnumValueDescriptor value = enumType.FindValueByName("FOREIGN_FOO");
  290. Assert.AreEqual(value, enumType.Values[1]);
  291. Assert.AreEqual("FOREIGN_FOO", value.Name);
  292. Assert.AreEqual(4, value.Number);
  293. Assert.AreEqual((int) ForeignEnum.ForeignFoo, value.Number);
  294. Assert.AreEqual(value, enumType.FindValueByNumber(4));
  295. Assert.Null(enumType.FindValueByName("NO_SUCH_VALUE"));
  296. for (int i = 0; i < enumType.Values.Count; i++)
  297. {
  298. Assert.AreEqual(i, enumType.Values[i].Index);
  299. }
  300. }
  301. [Test]
  302. public void OneofDescriptor()
  303. {
  304. OneofDescriptor descriptor = TestAllTypes.Descriptor.FindDescriptor<OneofDescriptor>("oneof_field");
  305. Assert.IsFalse(descriptor.IsSynthetic);
  306. Assert.AreEqual("oneof_field", descriptor.Name);
  307. Assert.AreEqual("protobuf_unittest3.TestAllTypes.oneof_field", descriptor.FullName);
  308. var expectedFields = new[] {
  309. TestAllTypes.OneofBytesFieldNumber,
  310. TestAllTypes.OneofNestedMessageFieldNumber,
  311. TestAllTypes.OneofStringFieldNumber,
  312. TestAllTypes.OneofUint32FieldNumber }
  313. .Select(fieldNumber => TestAllTypes.Descriptor.FindFieldByNumber(fieldNumber))
  314. .ToList();
  315. foreach (var field in expectedFields)
  316. {
  317. Assert.AreSame(descriptor, field.ContainingOneof);
  318. }
  319. CollectionAssert.AreEquivalent(expectedFields, descriptor.Fields);
  320. }
  321. [Test]
  322. public void MapEntryMessageDescriptor()
  323. {
  324. var descriptor = MapWellKnownTypes.Descriptor.NestedTypes[0];
  325. Assert.IsNull(descriptor.Parser);
  326. Assert.IsNull(descriptor.ClrType);
  327. Assert.IsNull(descriptor.Fields[1].Accessor);
  328. }
  329. // From TestFieldOrdering:
  330. // string my_string = 11;
  331. // int64 my_int = 1;
  332. // float my_float = 101;
  333. // NestedMessage single_nested_message = 200;
  334. [Test]
  335. public void FieldListOrderings()
  336. {
  337. var fields = TestFieldOrderings.Descriptor.Fields;
  338. Assert.AreEqual(new[] { 11, 1, 101, 200 }, fields.InDeclarationOrder().Select(x => x.FieldNumber));
  339. Assert.AreEqual(new[] { 1, 11, 101, 200 }, fields.InFieldNumberOrder().Select(x => x.FieldNumber));
  340. }
  341. [Test]
  342. public void DescriptorProtoFileDescriptor()
  343. {
  344. var descriptor = Google.Protobuf.Reflection.FileDescriptor.DescriptorProtoFileDescriptor;
  345. Assert.AreEqual("google/protobuf/descriptor.proto", descriptor.Name);
  346. }
  347. [Test]
  348. public void DescriptorImportingExtensionsFromOldCodeGen()
  349. {
  350. // The extension collection includes a null extension. There's not a lot we can do about that
  351. // in itself, as the old generator didn't provide us the extension information.
  352. var extensions = TestProtos.OldGenerator.OldExtensions2Reflection.Descriptor.Extensions;
  353. Assert.AreEqual(1, extensions.UnorderedExtensions.Count);
  354. // Note: this assertion is present so that it will fail if OldExtensions2 is regenerated
  355. // with a new generator.
  356. Assert.Null(extensions.UnorderedExtensions[0].Extension);
  357. // ... but we can make sure we at least don't cause a failure when retrieving descriptors.
  358. // In particular, old_extensions1.proto imports old_extensions2.proto, and this used to cause
  359. // an execution-time failure.
  360. var importingDescriptor = TestProtos.OldGenerator.OldExtensions1Reflection.Descriptor;
  361. Assert.NotNull(importingDescriptor);
  362. }
  363. [Test]
  364. public void Proto3OptionalDescriptors()
  365. {
  366. var descriptor = TestProto3Optional.Descriptor;
  367. var field = descriptor.Fields[TestProto3Optional.OptionalInt32FieldNumber];
  368. Assert.NotNull(field.ContainingOneof);
  369. Assert.IsTrue(field.ContainingOneof.IsSynthetic);
  370. Assert.Null(field.RealContainingOneof);
  371. }
  372. [Test]
  373. public void SyntheticOneofReflection()
  374. {
  375. // Expect every oneof in TestProto3Optional to be synthetic
  376. var proto3OptionalDescriptor = TestProto3Optional.Descriptor;
  377. Assert.AreEqual(0, proto3OptionalDescriptor.RealOneofCount);
  378. foreach (var oneof in proto3OptionalDescriptor.Oneofs)
  379. {
  380. Assert.True(oneof.IsSynthetic);
  381. }
  382. // Expect no oneof in the original proto3 unit test file to be synthetic.
  383. foreach (var descriptor in ProtobufTestMessages.Proto3.TestMessagesProto3Reflection.Descriptor.MessageTypes)
  384. {
  385. Assert.AreEqual(descriptor.Oneofs.Count, descriptor.RealOneofCount);
  386. foreach (var oneof in descriptor.Oneofs)
  387. {
  388. Assert.False(oneof.IsSynthetic);
  389. }
  390. }
  391. // Expect no oneof in the original proto2 unit test file to be synthetic.
  392. foreach (var descriptor in ProtobufTestMessages.Proto2.TestMessagesProto2Reflection.Descriptor.MessageTypes)
  393. {
  394. Assert.AreEqual(descriptor.Oneofs.Count, descriptor.RealOneofCount);
  395. foreach (var oneof in descriptor.Oneofs)
  396. {
  397. Assert.False(oneof.IsSynthetic);
  398. }
  399. }
  400. }
  401. }
  402. }