test_graphviews.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. import pytest
  2. import networkx as nx
  3. from networkx.utils import edges_equal, nodes_equal
  4. # Note: SubGraph views are not tested here. They have their own testing file
  5. class TestReverseView:
  6. def setup_method(self):
  7. self.G = nx.path_graph(9, create_using=nx.DiGraph())
  8. self.rv = nx.reverse_view(self.G)
  9. def test_pickle(self):
  10. import pickle
  11. rv = self.rv
  12. prv = pickle.loads(pickle.dumps(rv, -1))
  13. assert rv._node == prv._node
  14. assert rv._adj == prv._adj
  15. assert rv.graph == prv.graph
  16. def test_contains(self):
  17. assert (2, 3) in self.G.edges
  18. assert (3, 2) not in self.G.edges
  19. assert (2, 3) not in self.rv.edges
  20. assert (3, 2) in self.rv.edges
  21. def test_iter(self):
  22. expected = sorted(tuple(reversed(e)) for e in self.G.edges)
  23. assert sorted(self.rv.edges) == expected
  24. def test_exceptions(self):
  25. nxg = nx.graphviews
  26. G = nx.Graph()
  27. pytest.raises(nx.NetworkXNotImplemented, nxg.reverse_view, G)
  28. def test_subclass(self):
  29. class MyGraph(nx.DiGraph):
  30. def my_method(self):
  31. return "me"
  32. def to_directed_class(self):
  33. return MyGraph()
  34. M = MyGraph()
  35. M.add_edge(1, 2)
  36. RM = nx.reverse_view(M)
  37. print("RM class", RM.__class__)
  38. RMC = RM.copy()
  39. print("RMC class", RMC.__class__)
  40. print(RMC.edges)
  41. assert RMC.has_edge(2, 1)
  42. assert RMC.my_method() == "me"
  43. class TestMultiReverseView:
  44. def setup_method(self):
  45. self.G = nx.path_graph(9, create_using=nx.MultiDiGraph())
  46. self.G.add_edge(4, 5)
  47. self.rv = nx.reverse_view(self.G)
  48. def test_pickle(self):
  49. import pickle
  50. rv = self.rv
  51. prv = pickle.loads(pickle.dumps(rv, -1))
  52. assert rv._node == prv._node
  53. assert rv._adj == prv._adj
  54. assert rv.graph == prv.graph
  55. def test_contains(self):
  56. assert (2, 3, 0) in self.G.edges
  57. assert (3, 2, 0) not in self.G.edges
  58. assert (2, 3, 0) not in self.rv.edges
  59. assert (3, 2, 0) in self.rv.edges
  60. assert (5, 4, 1) in self.rv.edges
  61. assert (4, 5, 1) not in self.rv.edges
  62. def test_iter(self):
  63. expected = sorted((v, u, k) for u, v, k in self.G.edges)
  64. assert sorted(self.rv.edges) == expected
  65. def test_exceptions(self):
  66. nxg = nx.graphviews
  67. MG = nx.MultiGraph(self.G)
  68. pytest.raises(nx.NetworkXNotImplemented, nxg.reverse_view, MG)
  69. def test_generic_multitype():
  70. nxg = nx.graphviews
  71. G = nx.DiGraph([(1, 2)])
  72. with pytest.raises(nx.NetworkXError):
  73. nxg.generic_graph_view(G, create_using=nx.MultiGraph)
  74. G = nx.MultiDiGraph([(1, 2)])
  75. with pytest.raises(nx.NetworkXError):
  76. nxg.generic_graph_view(G, create_using=nx.DiGraph)
  77. class TestToDirected:
  78. def setup_method(self):
  79. self.G = nx.path_graph(9)
  80. self.dv = nx.to_directed(self.G)
  81. self.MG = nx.path_graph(9, create_using=nx.MultiGraph())
  82. self.Mdv = nx.to_directed(self.MG)
  83. def test_directed(self):
  84. assert not self.G.is_directed()
  85. assert self.dv.is_directed()
  86. def test_already_directed(self):
  87. dd = nx.to_directed(self.dv)
  88. Mdd = nx.to_directed(self.Mdv)
  89. assert edges_equal(dd.edges, self.dv.edges)
  90. assert edges_equal(Mdd.edges, self.Mdv.edges)
  91. def test_pickle(self):
  92. import pickle
  93. dv = self.dv
  94. pdv = pickle.loads(pickle.dumps(dv, -1))
  95. assert dv._node == pdv._node
  96. assert dv._succ == pdv._succ
  97. assert dv._pred == pdv._pred
  98. assert dv.graph == pdv.graph
  99. def test_contains(self):
  100. assert (2, 3) in self.G.edges
  101. assert (3, 2) in self.G.edges
  102. assert (2, 3) in self.dv.edges
  103. assert (3, 2) in self.dv.edges
  104. def test_iter(self):
  105. revd = [tuple(reversed(e)) for e in self.G.edges]
  106. expected = sorted(list(self.G.edges) + revd)
  107. assert sorted(self.dv.edges) == expected
  108. class TestToUndirected:
  109. def setup_method(self):
  110. self.DG = nx.path_graph(9, create_using=nx.DiGraph())
  111. self.uv = nx.to_undirected(self.DG)
  112. self.MDG = nx.path_graph(9, create_using=nx.MultiDiGraph())
  113. self.Muv = nx.to_undirected(self.MDG)
  114. def test_directed(self):
  115. assert self.DG.is_directed()
  116. assert not self.uv.is_directed()
  117. def test_already_directed(self):
  118. uu = nx.to_undirected(self.uv)
  119. Muu = nx.to_undirected(self.Muv)
  120. assert edges_equal(uu.edges, self.uv.edges)
  121. assert edges_equal(Muu.edges, self.Muv.edges)
  122. def test_pickle(self):
  123. import pickle
  124. uv = self.uv
  125. puv = pickle.loads(pickle.dumps(uv, -1))
  126. assert uv._node == puv._node
  127. assert uv._adj == puv._adj
  128. assert uv.graph == puv.graph
  129. assert hasattr(uv, "_graph")
  130. def test_contains(self):
  131. assert (2, 3) in self.DG.edges
  132. assert (3, 2) not in self.DG.edges
  133. assert (2, 3) in self.uv.edges
  134. assert (3, 2) in self.uv.edges
  135. def test_iter(self):
  136. expected = sorted(self.DG.edges)
  137. assert sorted(self.uv.edges) == expected
  138. class TestChainsOfViews:
  139. @classmethod
  140. def setup_class(cls):
  141. cls.G = nx.path_graph(9)
  142. cls.DG = nx.path_graph(9, create_using=nx.DiGraph())
  143. cls.MG = nx.path_graph(9, create_using=nx.MultiGraph())
  144. cls.MDG = nx.path_graph(9, create_using=nx.MultiDiGraph())
  145. cls.Gv = nx.to_undirected(cls.DG)
  146. cls.DGv = nx.to_directed(cls.G)
  147. cls.MGv = nx.to_undirected(cls.MDG)
  148. cls.MDGv = nx.to_directed(cls.MG)
  149. cls.Rv = cls.DG.reverse()
  150. cls.MRv = cls.MDG.reverse()
  151. cls.graphs = [
  152. cls.G,
  153. cls.DG,
  154. cls.MG,
  155. cls.MDG,
  156. cls.Gv,
  157. cls.DGv,
  158. cls.MGv,
  159. cls.MDGv,
  160. cls.Rv,
  161. cls.MRv,
  162. ]
  163. for G in cls.graphs:
  164. G.edges, G.nodes, G.degree
  165. def test_pickle(self):
  166. import pickle
  167. for G in self.graphs:
  168. H = pickle.loads(pickle.dumps(G, -1))
  169. assert edges_equal(H.edges, G.edges)
  170. assert nodes_equal(H.nodes, G.nodes)
  171. def test_subgraph_of_subgraph(self):
  172. SGv = nx.subgraph(self.G, range(3, 7))
  173. SDGv = nx.subgraph(self.DG, range(3, 7))
  174. SMGv = nx.subgraph(self.MG, range(3, 7))
  175. SMDGv = nx.subgraph(self.MDG, range(3, 7))
  176. for G in self.graphs + [SGv, SDGv, SMGv, SMDGv]:
  177. SG = nx.induced_subgraph(G, [4, 5, 6])
  178. assert list(SG) == [4, 5, 6]
  179. SSG = SG.subgraph([6, 7])
  180. assert list(SSG) == [6]
  181. # subgraph-subgraph chain is short-cut in base class method
  182. assert SSG._graph is G
  183. def test_restricted_induced_subgraph_chains(self):
  184. """Test subgraph chains that both restrict and show nodes/edges.
  185. A restricted_view subgraph should allow induced subgraphs using
  186. G.subgraph that automagically without a chain (meaning the result
  187. is a subgraph view of the original graph not a subgraph-of-subgraph.
  188. """
  189. hide_nodes = [3, 4, 5]
  190. hide_edges = [(6, 7)]
  191. RG = nx.restricted_view(self.G, hide_nodes, hide_edges)
  192. nodes = [4, 5, 6, 7, 8]
  193. SG = nx.induced_subgraph(RG, nodes)
  194. SSG = RG.subgraph(nodes)
  195. assert RG._graph is self.G
  196. assert SSG._graph is self.G
  197. assert SG._graph is RG
  198. assert edges_equal(SG.edges, SSG.edges)
  199. # should be same as morphing the graph
  200. CG = self.G.copy()
  201. CG.remove_nodes_from(hide_nodes)
  202. CG.remove_edges_from(hide_edges)
  203. assert edges_equal(CG.edges(nodes), SSG.edges)
  204. CG.remove_nodes_from([0, 1, 2, 3])
  205. assert edges_equal(CG.edges, SSG.edges)
  206. # switch order: subgraph first, then restricted view
  207. SSSG = self.G.subgraph(nodes)
  208. RSG = nx.restricted_view(SSSG, hide_nodes, hide_edges)
  209. assert RSG._graph is not self.G
  210. assert edges_equal(RSG.edges, CG.edges)
  211. def test_subgraph_copy(self):
  212. for origG in self.graphs:
  213. G = nx.Graph(origG)
  214. SG = G.subgraph([4, 5, 6])
  215. H = SG.copy()
  216. assert type(G) == type(H)
  217. def test_subgraph_todirected(self):
  218. SG = nx.induced_subgraph(self.G, [4, 5, 6])
  219. SSG = SG.to_directed()
  220. assert sorted(SSG) == [4, 5, 6]
  221. assert sorted(SSG.edges) == [(4, 5), (5, 4), (5, 6), (6, 5)]
  222. def test_subgraph_toundirected(self):
  223. SG = nx.induced_subgraph(self.G, [4, 5, 6])
  224. SSG = SG.to_undirected()
  225. assert list(SSG) == [4, 5, 6]
  226. assert sorted(SSG.edges) == [(4, 5), (5, 6)]
  227. def test_reverse_subgraph_toundirected(self):
  228. G = self.DG.reverse(copy=False)
  229. SG = G.subgraph([4, 5, 6])
  230. SSG = SG.to_undirected()
  231. assert list(SSG) == [4, 5, 6]
  232. assert sorted(SSG.edges) == [(4, 5), (5, 6)]
  233. def test_reverse_reverse_copy(self):
  234. G = self.DG.reverse(copy=False)
  235. H = G.reverse(copy=True)
  236. assert H.nodes == self.DG.nodes
  237. assert H.edges == self.DG.edges
  238. G = self.MDG.reverse(copy=False)
  239. H = G.reverse(copy=True)
  240. assert H.nodes == self.MDG.nodes
  241. assert H.edges == self.MDG.edges
  242. def test_subgraph_edgesubgraph_toundirected(self):
  243. G = self.G.copy()
  244. SG = G.subgraph([4, 5, 6])
  245. SSG = SG.edge_subgraph([(4, 5), (5, 4)])
  246. USSG = SSG.to_undirected()
  247. assert list(USSG) == [4, 5]
  248. assert sorted(USSG.edges) == [(4, 5)]
  249. def test_copy_subgraph(self):
  250. G = self.G.copy()
  251. SG = G.subgraph([4, 5, 6])
  252. CSG = SG.copy(as_view=True)
  253. DCSG = SG.copy(as_view=False)
  254. assert hasattr(CSG, "_graph") # is a view
  255. assert not hasattr(DCSG, "_graph") # not a view
  256. def test_copy_disubgraph(self):
  257. G = self.DG.copy()
  258. SG = G.subgraph([4, 5, 6])
  259. CSG = SG.copy(as_view=True)
  260. DCSG = SG.copy(as_view=False)
  261. assert hasattr(CSG, "_graph") # is a view
  262. assert not hasattr(DCSG, "_graph") # not a view
  263. def test_copy_multidisubgraph(self):
  264. G = self.MDG.copy()
  265. SG = G.subgraph([4, 5, 6])
  266. CSG = SG.copy(as_view=True)
  267. DCSG = SG.copy(as_view=False)
  268. assert hasattr(CSG, "_graph") # is a view
  269. assert not hasattr(DCSG, "_graph") # not a view
  270. def test_copy_multisubgraph(self):
  271. G = self.MG.copy()
  272. SG = G.subgraph([4, 5, 6])
  273. CSG = SG.copy(as_view=True)
  274. DCSG = SG.copy(as_view=False)
  275. assert hasattr(CSG, "_graph") # is a view
  276. assert not hasattr(DCSG, "_graph") # not a view
  277. def test_copy_of_view(self):
  278. G = nx.MultiGraph(self.MGv)
  279. assert G.__class__.__name__ == "MultiGraph"
  280. G = G.copy(as_view=True)
  281. assert G.__class__.__name__ == "MultiGraph"
  282. def test_subclass(self):
  283. class MyGraph(nx.DiGraph):
  284. def my_method(self):
  285. return "me"
  286. def to_directed_class(self):
  287. return MyGraph()
  288. for origG in self.graphs:
  289. G = MyGraph(origG)
  290. SG = G.subgraph([4, 5, 6])
  291. H = SG.copy()
  292. assert SG.my_method() == "me"
  293. assert H.my_method() == "me"
  294. assert 3 not in H or 3 in SG