You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

GraphTest.cs 18 kB

6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
6 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437
  1. using Microsoft.VisualStudio.TestTools.UnitTesting;
  2. using System;
  3. using static Tensorflow.Binding;
  4. namespace Tensorflow.Native.UnitTest
  5. {
  6. [TestClass]
  7. public class GraphTest : CApiTest
  8. {
  9. /// <summary>
  10. /// Port from c_api_test.cc
  11. /// `TEST(CAPI, Graph)`
  12. /// </summary>
  13. [TestMethod]
  14. public void Graph()
  15. {
  16. var s = new Status();
  17. var graph = new Graph();
  18. // Make a placeholder operation.
  19. var feed = c_test_util.Placeholder(graph, s);
  20. EXPECT_EQ("feed", feed.name);
  21. EXPECT_EQ("Placeholder", feed.OpType);
  22. EXPECT_EQ("", feed.Device);
  23. EXPECT_EQ(1, feed.NumOutputs);
  24. EXPECT_EQ(TF_DataType.TF_INT32, feed.OutputType(0));
  25. EXPECT_EQ(1, feed.OutputListLength("output"));
  26. EXPECT_EQ(0, feed.NumInputs);
  27. EXPECT_EQ(0, feed.OutputNumConsumers(0));
  28. EXPECT_EQ(0, feed.NumControlInputs);
  29. EXPECT_EQ(0, feed.NumControlOutputs);
  30. AttrValue attr_value = null;
  31. ASSERT_TRUE(c_test_util.GetAttrValue(feed, "dtype", ref attr_value, s));
  32. EXPECT_EQ(attr_value.Type, DataType.DtInt32);
  33. // Test not found errors in TF_Operation*() query functions.
  34. EXPECT_EQ(-1, c_api.TF_OperationOutputListLength(feed, "bogus", s.Handle));
  35. EXPECT_EQ(TF_Code.TF_INVALID_ARGUMENT, s.Code);
  36. Assert.IsFalse(c_test_util.GetAttrValue(feed, "missing", ref attr_value, s));
  37. EXPECT_EQ("Operation 'feed' has no attr named 'missing'.", s.Message);
  38. // Make a constant oper with the scalar "3".
  39. var three = c_test_util.ScalarConst(3, graph, s);
  40. EXPECT_EQ(TF_Code.TF_OK, s.Code);
  41. // Add oper.
  42. var add = c_test_util.Add(feed, three, graph, s);
  43. EXPECT_EQ(TF_Code.TF_OK, s.Code);
  44. // Test TF_Operation*() query functions.
  45. EXPECT_EQ("add", add.name);
  46. EXPECT_EQ("AddN", add.OpType);
  47. EXPECT_EQ("", add.Device);
  48. EXPECT_EQ(1, add.NumOutputs);
  49. EXPECT_EQ(TF_DataType.TF_INT32, add.OutputType(0));
  50. EXPECT_EQ(1, add.OutputListLength("sum"));
  51. EXPECT_EQ(TF_Code.TF_OK, s.Code);
  52. EXPECT_EQ(2, add.InputListLength("inputs"));
  53. EXPECT_EQ(TF_Code.TF_OK, s.Code);
  54. EXPECT_EQ(TF_DataType.TF_INT32, add.InputType(0));
  55. EXPECT_EQ(TF_DataType.TF_INT32, add.InputType(1));
  56. var add_in_0 = add.Input(0);
  57. EXPECT_EQ(feed, add_in_0.oper);
  58. EXPECT_EQ(0, add_in_0.index);
  59. var add_in_1 = add.Input(1);
  60. EXPECT_EQ(three, add_in_1.oper);
  61. EXPECT_EQ(0, add_in_1.index);
  62. EXPECT_EQ(0, add.OutputNumConsumers(0));
  63. EXPECT_EQ(0, add.NumControlInputs);
  64. EXPECT_EQ(0, add.NumControlOutputs);
  65. ASSERT_TRUE(c_test_util.GetAttrValue(add, "T", ref attr_value, s));
  66. EXPECT_EQ(DataType.DtInt32, attr_value.Type);
  67. ASSERT_TRUE(c_test_util.GetAttrValue(add, "N", ref attr_value, s));
  68. EXPECT_EQ(2, (int)attr_value.I);
  69. // Placeholder oper now has a consumer.
  70. EXPECT_EQ(1, feed.OutputNumConsumers(0));
  71. TF_Input[] feed_port = feed.OutputConsumers(0, 1);
  72. EXPECT_EQ(1, feed_port.Length);
  73. EXPECT_EQ(add, feed_port[0].oper);
  74. EXPECT_EQ(0, feed_port[0].index);
  75. // The scalar const oper also has a consumer.
  76. EXPECT_EQ(1, three.OutputNumConsumers(0));
  77. TF_Input[] three_port = three.OutputConsumers(0, 1);
  78. EXPECT_EQ(add, three_port[0].oper);
  79. EXPECT_EQ(1, three_port[0].index);
  80. // Serialize to GraphDef.
  81. var graph_def = c_test_util.GetGraphDef(graph);
  82. // Validate GraphDef is what we expect.
  83. bool found_placeholder = false;
  84. bool found_scalar_const = false;
  85. bool found_add = false;
  86. foreach (var n in graph_def.Node)
  87. {
  88. if (c_test_util.IsPlaceholder(n))
  89. {
  90. Assert.IsFalse(found_placeholder);
  91. found_placeholder = true;
  92. }
  93. else if (c_test_util.IsScalarConst(n, 3))
  94. {
  95. Assert.IsFalse(found_scalar_const);
  96. found_scalar_const = true;
  97. }
  98. else if (c_test_util.IsAddN(n, 2))
  99. {
  100. Assert.IsFalse(found_add);
  101. found_add = true;
  102. }
  103. else
  104. {
  105. Assert.Fail($"Unexpected NodeDef: {n}");
  106. }
  107. }
  108. ASSERT_TRUE(found_placeholder);
  109. ASSERT_TRUE(found_scalar_const);
  110. ASSERT_TRUE(found_add);
  111. // Add another oper to the graph.
  112. var neg = c_test_util.Neg(add, graph, s);
  113. EXPECT_EQ(TF_Code.TF_OK, s.Code);
  114. // Serialize to NodeDef.
  115. var node_def = neg.node_def;
  116. // Validate NodeDef is what we expect.
  117. ASSERT_TRUE(c_test_util.IsNeg(node_def, "add"));
  118. // Serialize to GraphDef.
  119. var graph_def2 = c_test_util.GetGraphDef(graph);
  120. // Compare with first GraphDef + added NodeDef.
  121. graph_def.Node.Add(node_def);
  122. EXPECT_EQ(graph_def, graph_def2);
  123. // Look up some nodes by name.
  124. Operation neg2 = c_api.TF_GraphOperationByName(graph, "neg");
  125. EXPECT_EQ(neg, neg2);
  126. var node_def2 = neg2.node_def;
  127. EXPECT_EQ(node_def, node_def2);
  128. Operation feed2 = c_api.TF_GraphOperationByName(graph, "feed");
  129. EXPECT_EQ(feed, feed2);
  130. node_def = feed.node_def;
  131. node_def2 = feed2.node_def;
  132. EXPECT_EQ(node_def, node_def2);
  133. // Test iterating through the nodes of a graph.
  134. found_placeholder = false;
  135. found_scalar_const = false;
  136. found_add = false;
  137. bool found_neg = false;
  138. uint pos = 0;
  139. Operation oper;
  140. while ((oper = c_api.TF_GraphNextOperation(graph, ref pos)) != IntPtr.Zero)
  141. {
  142. if (oper.Equals(feed))
  143. {
  144. Assert.IsFalse(found_placeholder);
  145. found_placeholder = true;
  146. }
  147. else if (oper.Equals(three))
  148. {
  149. Assert.IsFalse(found_scalar_const);
  150. found_scalar_const = true;
  151. }
  152. else if (oper.Equals(add))
  153. {
  154. Assert.IsFalse(found_add);
  155. found_add = true;
  156. }
  157. else if (oper.Equals(neg))
  158. {
  159. Assert.IsFalse(found_neg);
  160. found_neg = true;
  161. }
  162. else
  163. {
  164. node_def = oper.node_def;
  165. Assert.Fail($"Unexpected Node: {node_def.ToString()}");
  166. }
  167. }
  168. ASSERT_TRUE(found_placeholder);
  169. ASSERT_TRUE(found_scalar_const);
  170. ASSERT_TRUE(found_add);
  171. ASSERT_TRUE(found_neg);
  172. graph.Dispose();
  173. s.Dispose();
  174. }
  175. /// <summary>
  176. /// Port from c_api_test.cc
  177. /// `TEST(CAPI, ImportGraphDef)`
  178. /// </summary>
  179. [TestMethod]
  180. public void ImportGraphDef()
  181. {
  182. var s = new Status();
  183. var graph = new Graph().as_default();
  184. // Create a simple graph.
  185. c_test_util.Placeholder(graph, s);
  186. var oper = c_test_util.ScalarConst(3, graph, s);
  187. c_test_util.Neg(oper, graph, s);
  188. // Export to a GraphDef.
  189. var graph_def = new Buffer();
  190. c_api.TF_GraphToGraphDef(graph, graph_def.Handle, s.Handle);
  191. EXPECT_EQ(TF_Code.TF_OK, s.Code);
  192. // Import it, with a prefix, in a fresh graph.
  193. graph.Dispose();
  194. graph = new Graph().as_default();
  195. using (var opts = c_api.TF_NewImportGraphDefOptions())
  196. {
  197. c_api.TF_ImportGraphDefOptionsSetPrefix(opts, "imported");
  198. c_api.TF_GraphImportGraphDef(graph, graph_def.Handle, opts, s.Handle);
  199. EXPECT_EQ(TF_Code.TF_OK, s.Code);
  200. }
  201. Operation scalar = graph.OperationByName("imported/scalar");
  202. Operation feed = graph.OperationByName("imported/feed");
  203. Operation neg = graph.OperationByName("imported/neg");
  204. // Test basic structure of the imported graph.
  205. EXPECT_EQ(0, scalar.NumInputs);
  206. EXPECT_EQ(0, feed.NumInputs);
  207. EXPECT_EQ(1, neg.NumInputs);
  208. var neg_input = neg.Input(0);
  209. EXPECT_EQ(scalar, neg_input.oper);
  210. EXPECT_EQ(0, neg_input.index);
  211. // Test that we can't see control edges involving the source and sink nodes.
  212. EXPECT_EQ(0, scalar.NumControlInputs);
  213. EXPECT_EQ(0, scalar.GetControlInputs().Length);
  214. EXPECT_EQ(0, scalar.NumControlOutputs);
  215. EXPECT_EQ(0, scalar.GetControlOutputs().Length);
  216. EXPECT_EQ(0, feed.NumControlInputs);
  217. EXPECT_EQ(0, feed.GetControlInputs().Length);
  218. EXPECT_EQ(0, feed.NumControlOutputs);
  219. EXPECT_EQ(0, feed.GetControlOutputs().Length);
  220. EXPECT_EQ(0, neg.NumControlInputs);
  221. EXPECT_EQ(0, neg.GetControlInputs().Length);
  222. EXPECT_EQ(0, neg.NumControlOutputs);
  223. EXPECT_EQ(0, neg.GetControlOutputs().Length);
  224. static SafeImportGraphDefResultsHandle ImportGraph(Status s, Graph graph, Buffer graph_def, Operation scalar)
  225. {
  226. using var opts = c_api.TF_NewImportGraphDefOptions();
  227. c_api.TF_ImportGraphDefOptionsSetPrefix(opts, "imported2");
  228. c_api.TF_ImportGraphDefOptionsAddInputMapping(opts, "scalar", 0, new TF_Output(scalar, 0));
  229. c_api.TF_ImportGraphDefOptionsAddReturnOutput(opts, "feed", 0);
  230. c_api.TF_ImportGraphDefOptionsAddReturnOutput(opts, "scalar", 0);
  231. EXPECT_EQ(2, c_api.TF_ImportGraphDefOptionsNumReturnOutputs(opts));
  232. c_api.TF_ImportGraphDefOptionsAddReturnOperation(opts, "scalar");
  233. EXPECT_EQ(1, c_api.TF_ImportGraphDefOptionsNumReturnOperations(opts));
  234. var results = c_api.TF_GraphImportGraphDefWithResults(graph, graph_def.Handle, opts, s.Handle);
  235. EXPECT_EQ(TF_Code.TF_OK, s.Code);
  236. return results;
  237. }
  238. // Import it again, with an input mapping, return outputs, and a return
  239. // operation, into the same graph.
  240. Operation feed2;
  241. using (SafeImportGraphDefResultsHandle results = ImportGraph(s, graph, graph_def, scalar))
  242. {
  243. Operation scalar2 = graph.OperationByName("imported2/scalar");
  244. feed2 = graph.OperationByName("imported2/feed");
  245. Operation neg2 = graph.OperationByName("imported2/neg");
  246. // Check input mapping
  247. neg_input = neg.Input(0);
  248. EXPECT_EQ(scalar, neg_input.oper);
  249. EXPECT_EQ(0, neg_input.index);
  250. // Check return outputs
  251. var return_outputs = graph.ReturnOutputs(results);
  252. ASSERT_EQ(2, return_outputs.Length);
  253. EXPECT_EQ(feed2, return_outputs[0].oper);
  254. EXPECT_EQ(0, return_outputs[0].index);
  255. EXPECT_EQ(scalar, return_outputs[1].oper); // remapped
  256. EXPECT_EQ(0, return_outputs[1].index);
  257. // Check return operation
  258. var return_opers = graph.ReturnOperations(results);
  259. ASSERT_EQ(1, return_opers.Length);
  260. EXPECT_EQ(scalar2, return_opers[0]); // not remapped
  261. }
  262. // Import again, with control dependencies, into the same graph.
  263. using (var opts = c_api.TF_NewImportGraphDefOptions())
  264. {
  265. c_api.TF_ImportGraphDefOptionsSetPrefix(opts, "imported3");
  266. c_api.TF_ImportGraphDefOptionsAddControlDependency(opts, feed);
  267. c_api.TF_ImportGraphDefOptionsAddControlDependency(opts, feed2);
  268. c_api.TF_GraphImportGraphDef(graph, graph_def.Handle, opts, s.Handle);
  269. EXPECT_EQ(TF_Code.TF_OK, s.Code);
  270. }
  271. var scalar3 = graph.OperationByName("imported3/scalar");
  272. var feed3 = graph.OperationByName("imported3/feed");
  273. var neg3 = graph.OperationByName("imported3/neg");
  274. ASSERT_TRUE(scalar3 != IntPtr.Zero);
  275. ASSERT_TRUE(feed3 != IntPtr.Zero);
  276. ASSERT_TRUE(neg3 != IntPtr.Zero);
  277. // Check that newly-imported scalar and feed have control deps (neg3 will
  278. // inherit them from input)
  279. var control_inputs = scalar3.GetControlInputs();
  280. ASSERT_EQ(2, scalar3.NumControlInputs);
  281. EXPECT_EQ(feed, control_inputs[0]);
  282. EXPECT_EQ(feed2, control_inputs[1]);
  283. control_inputs = feed3.GetControlInputs();
  284. ASSERT_EQ(2, feed3.NumControlInputs);
  285. EXPECT_EQ(feed, control_inputs[0]);
  286. EXPECT_EQ(feed2, control_inputs[1]);
  287. // Export to a graph def so we can import a graph with control dependencies
  288. graph_def = new Buffer();
  289. c_api.TF_GraphToGraphDef(graph, graph_def.Handle, s.Handle);
  290. EXPECT_EQ(TF_Code.TF_OK, s.Code);
  291. // Import again, with remapped control dependency, into the same graph
  292. using (var opts = c_api.TF_NewImportGraphDefOptions())
  293. {
  294. c_api.TF_ImportGraphDefOptionsSetPrefix(opts, "imported4");
  295. c_api.TF_ImportGraphDefOptionsRemapControlDependency(opts, "imported/feed", feed);
  296. c_api.TF_GraphImportGraphDef(graph, graph_def.Handle, opts, s.Handle);
  297. ASSERT_EQ(TF_Code.TF_OK, s.Code);
  298. }
  299. var scalar4 = graph.OperationByName("imported4/imported3/scalar");
  300. var feed4 = graph.OperationByName("imported4/imported2/feed");
  301. // Check that imported `imported3/scalar` has remapped control dep from
  302. // original graph and imported control dep
  303. control_inputs = scalar4.GetControlInputs();
  304. ASSERT_EQ(2, scalar4.NumControlInputs);
  305. EXPECT_EQ(feed, control_inputs[0]);
  306. EXPECT_EQ(feed4, control_inputs[1]);
  307. // Can add nodes to the imported graph without trouble.
  308. c_test_util.Add(feed, scalar, graph, s);
  309. ASSERT_EQ(TF_Code.TF_OK, s.Code);
  310. }
  311. /// <summary>
  312. /// Port from c_api_test.cc
  313. /// `TEST(CAPI, ImportGraphDef_WithReturnOutputs)`
  314. /// </summary>
  315. [TestMethod]
  316. public void ImportGraphDef_WithReturnOutputs()
  317. {
  318. var s = new Status();
  319. var graph = new Graph().as_default();
  320. // Create a graph with two nodes: x and 3
  321. c_test_util.Placeholder(graph, s);
  322. ASSERT_TRUE(graph.OperationByName("feed") != null);
  323. var oper = c_test_util.ScalarConst(3, graph, s);
  324. ASSERT_TRUE(graph.OperationByName("scalar") != null);
  325. c_test_util.Neg(oper, graph, s);
  326. ASSERT_TRUE(graph.OperationByName("neg") != null);
  327. // Export to a GraphDef.
  328. var graph_def = graph.ToGraphDef(s);
  329. ASSERT_EQ(TF_Code.TF_OK, s.Code);
  330. // Import it in a fresh graph with return outputs.
  331. graph.Dispose();
  332. graph = new Graph().as_default();
  333. var opts = new ImportGraphDefOptions();
  334. opts.AddReturnOutput("feed", 0);
  335. opts.AddReturnOutput("scalar", 0);
  336. EXPECT_EQ(2, opts.NumReturnOutputs);
  337. var return_outputs = graph.ImportGraphDefWithReturnOutputs(graph_def, opts, s);
  338. ASSERT_EQ(TF_Code.TF_OK, s.Code);
  339. var scalar = graph.OperationByName("scalar");
  340. var feed = graph.OperationByName("feed");
  341. var neg = graph.OperationByName("neg");
  342. ASSERT_TRUE(scalar != IntPtr.Zero);
  343. ASSERT_TRUE(feed != IntPtr.Zero);
  344. ASSERT_TRUE(neg != IntPtr.Zero);
  345. // Check return outputs
  346. EXPECT_EQ(feed, return_outputs[0].oper);
  347. EXPECT_EQ(0, return_outputs[0].index);
  348. EXPECT_EQ(scalar, return_outputs[1].oper);
  349. EXPECT_EQ(0, return_outputs[1].index);
  350. opts.Dispose();
  351. graph_def.Dispose();
  352. graph.Dispose();
  353. s.Dispose();
  354. }
  355. /// <summary>
  356. /// `TEST(CAPI, ImportGraphDef_MissingUnusedInputMappings)`
  357. /// </summary>
  358. [TestMethod]
  359. public void ImportGraphDef_MissingUnusedInputMappings()
  360. {
  361. }
  362. [Ignore]
  363. [TestMethod]
  364. public void ImportGraphMeta()
  365. {
  366. var dir = "my-save-dir/";
  367. using (var sess = tf.Session())
  368. {
  369. var new_saver = tf.train.import_meta_graph(dir + "my-model-10000.meta");
  370. new_saver.restore(sess, dir + "my-model-10000");
  371. var labels = tf.constant(0, dtype: tf.int32, shape: new int[] { 100 }, name: "labels");
  372. var batch_size = tf.size(labels);
  373. var logits = tf.get_collection<ITensorOrOperation>("logits")[0] as Tensor;
  374. var loss = tf.losses.sparse_softmax_cross_entropy(labels: labels,
  375. logits: logits);
  376. }
  377. }
  378. }
  379. }