DescriptorsTest.cs 20 KB

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