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.

common_walk.py 8.9 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293
  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. Created on Tue Aug 18 11:21:31 2020
  5. @author: ljia
  6. @references:
  7. [1] Thomas Gärtner, Peter Flach, and Stefan Wrobel. On graph kernels:
  8. Hardness results and efficient alternatives. Learning Theory and Kernel
  9. Machines, pages 129–143, 2003.
  10. """
  11. import sys
  12. from tqdm import tqdm
  13. import numpy as np
  14. import networkx as nx
  15. from gklearn.utils import SpecialLabel
  16. from gklearn.utils.parallel import parallel_gm, parallel_me
  17. from gklearn.utils.utils import direct_product_graph
  18. from gklearn.kernels import GraphKernel
  19. class CommonWalk(GraphKernel):
  20. def __init__(self, **kwargs):
  21. GraphKernel.__init__(self)
  22. self.__node_labels = kwargs.get('node_labels', [])
  23. self.__edge_labels = kwargs.get('edge_labels', [])
  24. self.__weight = kwargs.get('weight', 1)
  25. self.__compute_method = kwargs.get('compute_method', None)
  26. self.__ds_infos = kwargs.get('ds_infos', {})
  27. self.__compute_method = self.__compute_method.lower()
  28. def _compute_gm_series(self):
  29. self.__check_graphs(self._graphs)
  30. self.__add_dummy_labels(self._graphs)
  31. if not self.__ds_infos['directed']: # convert
  32. self._graphs = [G.to_directed() for G in self._graphs]
  33. # compute Gram matrix.
  34. gram_matrix = np.zeros((len(self._graphs), len(self._graphs)))
  35. from itertools import combinations_with_replacement
  36. itr = combinations_with_replacement(range(0, len(self._graphs)), 2)
  37. if self._verbose >= 2:
  38. iterator = tqdm(itr, desc='calculating kernels', file=sys.stdout)
  39. else:
  40. iterator = itr
  41. # direct product graph method - exponential
  42. if self.__compute_method == 'exp':
  43. for i, j in iterator:
  44. kernel = self.__kernel_do_exp(self._graphs[i], self._graphs[j], self.__weight)
  45. gram_matrix[i][j] = kernel
  46. gram_matrix[j][i] = kernel
  47. # direct product graph method - geometric
  48. elif self.__compute_method == 'geo':
  49. for i, j in iterator:
  50. kernel = self.__kernel_do_geo(self._graphs[i], self._graphs[j], self.__weight)
  51. gram_matrix[i][j] = kernel
  52. gram_matrix[j][i] = kernel
  53. return gram_matrix
  54. def _compute_gm_imap_unordered(self):
  55. self.__check_graphs(self._graphs)
  56. self.__add_dummy_labels(self._graphs)
  57. if not self.__ds_infos['directed']: # convert
  58. self._graphs = [G.to_directed() for G in self._graphs]
  59. # compute Gram matrix.
  60. gram_matrix = np.zeros((len(self._graphs), len(self._graphs)))
  61. # def init_worker(gn_toshare):
  62. # global G_gn
  63. # G_gn = gn_toshare
  64. # direct product graph method - exponential
  65. if self.__compute_method == 'exp':
  66. do_fun = self._wrapper_kernel_do_exp
  67. # direct product graph method - geometric
  68. elif self.__compute_method == 'geo':
  69. do_fun = self._wrapper_kernel_do_geo
  70. parallel_gm(do_fun, gram_matrix, self._graphs, init_worker=_init_worker_gm,
  71. glbv=(self._graphs,), n_jobs=self._n_jobs, verbose=self._verbose)
  72. return gram_matrix
  73. def _compute_kernel_list_series(self, g1, g_list):
  74. self.__check_graphs(g_list + [g1])
  75. self.__add_dummy_labels(g_list + [g1])
  76. if not self.__ds_infos['directed']: # convert
  77. g1 = g1.to_directed()
  78. g_list = [G.to_directed() for G in g_list]
  79. # compute kernel list.
  80. kernel_list = [None] * len(g_list)
  81. if self._verbose >= 2:
  82. iterator = tqdm(range(len(g_list)), desc='calculating kernels', file=sys.stdout)
  83. else:
  84. iterator = range(len(g_list))
  85. # direct product graph method - exponential
  86. if self.__compute_method == 'exp':
  87. for i in iterator:
  88. kernel = self.__kernel_do_exp(g1, g_list[i], self.__weight)
  89. kernel_list[i] = kernel
  90. # direct product graph method - geometric
  91. elif self.__compute_method == 'geo':
  92. for i in iterator:
  93. kernel = self.__kernel_do_geo(g1, g_list[i], self.__weight)
  94. kernel_list[i] = kernel
  95. return kernel_list
  96. def _compute_kernel_list_imap_unordered(self, g1, g_list):
  97. self.__check_graphs(g_list + [g1])
  98. self.__add_dummy_labels(g_list + [g1])
  99. if not self.__ds_infos['directed']: # convert
  100. g1 = g1.to_directed()
  101. g_list = [G.to_directed() for G in g_list]
  102. # compute kernel list.
  103. kernel_list = [None] * len(g_list)
  104. # def init_worker(g1_toshare, g_list_toshare):
  105. # global G_g1, G_g_list
  106. # G_g1 = g1_toshare
  107. # G_g_list = g_list_toshare
  108. # direct product graph method - exponential
  109. if self.__compute_method == 'exp':
  110. do_fun = self._wrapper_kernel_list_do_exp
  111. # direct product graph method - geometric
  112. elif self.__compute_method == 'geo':
  113. do_fun = self._wrapper_kernel_list_do_geo
  114. def func_assign(result, var_to_assign):
  115. var_to_assign[result[0]] = result[1]
  116. itr = range(len(g_list))
  117. len_itr = len(g_list)
  118. parallel_me(do_fun, func_assign, kernel_list, itr, len_itr=len_itr,
  119. init_worker=_init_worker_list, glbv=(g1, g_list), method='imap_unordered',
  120. n_jobs=self._n_jobs, itr_desc='calculating kernels', verbose=self._verbose)
  121. return kernel_list
  122. def _wrapper_kernel_list_do_exp(self, itr):
  123. return itr, self.__kernel_do_exp(G_g1, G_g_list[itr], self.__weight)
  124. def _wrapper_kernel_list_do_geo(self, itr):
  125. return itr, self.__kernel_do_geo(G_g1, G_g_list[itr], self.__weight)
  126. def _compute_single_kernel_series(self, g1, g2):
  127. self.__check_graphs([g1] + [g2])
  128. self.__add_dummy_labels([g1] + [g2])
  129. if not self.__ds_infos['directed']: # convert
  130. g1 = g1.to_directed()
  131. g2 = g2.to_directed()
  132. # direct product graph method - exponential
  133. if self.__compute_method == 'exp':
  134. kernel = self.__kernel_do_exp(g1, g2, self.__weight)
  135. # direct product graph method - geometric
  136. elif self.__compute_method == 'geo':
  137. kernel = self.__kernel_do_geo(g1, g2, self.__weight)
  138. return kernel
  139. def __kernel_do_exp(self, g1, g2, beta):
  140. """Calculate common walk graph kernel between 2 graphs using exponential
  141. series.
  142. Parameters
  143. ----------
  144. g1, g2 : NetworkX graphs
  145. Graphs between which the kernels are calculated.
  146. beta : integer
  147. Weight.
  148. Return
  149. ------
  150. kernel : float
  151. The common walk Kernel between 2 graphs.
  152. """
  153. # get tensor product / direct product
  154. gp = direct_product_graph(g1, g2, self.__node_labels, self.__edge_labels)
  155. # return 0 if the direct product graph have no more than 1 node.
  156. if nx.number_of_nodes(gp) < 2:
  157. return 0
  158. A = nx.adjacency_matrix(gp).todense()
  159. ew, ev = np.linalg.eig(A)
  160. # # remove imaginary part if possible.
  161. # # @todo: don't know if it is necessary.
  162. # for i in range(len(ew)):
  163. # if np.abs(ew[i].imag) < 1e-9:
  164. # ew[i] = ew[i].real
  165. # for i in range(ev.shape[0]):
  166. # for j in range(ev.shape[1]):
  167. # if np.abs(ev[i, j].imag) < 1e-9:
  168. # ev[i, j] = ev[i, j].real
  169. D = np.zeros((len(ew), len(ew)), dtype=complex) # @todo: use complex?
  170. for i in range(len(ew)):
  171. D[i][i] = np.exp(beta * ew[i])
  172. exp_D = ev * D * ev.T
  173. kernel = exp_D.sum()
  174. if (kernel.real == 0 and np.abs(kernel.imag) < 1e-9) or np.abs(kernel.imag / kernel.real) < 1e-9:
  175. kernel = kernel.real
  176. return kernel
  177. def _wrapper_kernel_do_exp(self, itr):
  178. i = itr[0]
  179. j = itr[1]
  180. return i, j, self.__kernel_do_exp(G_gn[i], G_gn[j], self.__weight)
  181. def __kernel_do_geo(self, g1, g2, gamma):
  182. """Calculate common walk graph kernel between 2 graphs using geometric
  183. series.
  184. Parameters
  185. ----------
  186. g1, g2 : NetworkX graphs
  187. Graphs between which the kernels are calculated.
  188. gamma : integer
  189. Weight.
  190. Return
  191. ------
  192. kernel : float
  193. The common walk Kernel between 2 graphs.
  194. """
  195. # get tensor product / direct product
  196. gp = direct_product_graph(g1, g2, self.__node_labels, self.__edge_labels)
  197. # return 0 if the direct product graph have no more than 1 node.
  198. if nx.number_of_nodes(gp) < 2:
  199. return 0
  200. A = nx.adjacency_matrix(gp).todense()
  201. mat = np.identity(len(A)) - gamma * A
  202. # try:
  203. return mat.I.sum()
  204. # except np.linalg.LinAlgError:
  205. # return np.nan
  206. def _wrapper_kernel_do_geo(self, itr):
  207. i = itr[0]
  208. j = itr[1]
  209. return i, j, self.__kernel_do_geo(G_gn[i], G_gn[j], self.__weight)
  210. def __check_graphs(self, Gn):
  211. for g in Gn:
  212. if nx.number_of_nodes(g) == 1:
  213. raise Exception('Graphs must contain more than 1 nodes to construct adjacency matrices.')
  214. def __add_dummy_labels(self, Gn):
  215. if len(self.__node_labels) == 0 or (len(self.__node_labels) == 1 and self.__node_labels[0] == SpecialLabel.DUMMY):
  216. for i in range(len(Gn)):
  217. nx.set_node_attributes(Gn[i], '0', SpecialLabel.DUMMY)
  218. self.__node_labels = [SpecialLabel.DUMMY]
  219. if len(self.__edge_labels) == 0 or (len(self.__edge_labels) == 1 and self.__edge_labels[0] == SpecialLabel.DUMMY):
  220. for i in range(len(Gn)):
  221. nx.set_edge_attributes(Gn[i], '0', SpecialLabel.DUMMY)
  222. self.__edge_labels = [SpecialLabel.DUMMY]
  223. def _init_worker_gm(gn_toshare):
  224. global G_gn
  225. G_gn = gn_toshare
  226. def _init_worker_list(g1_toshare, g_list_toshare):
  227. global G_g1, G_g_list
  228. G_g1 = g1_toshare
  229. G_g_list = g_list_toshare

A Python package for graph kernels, graph edit distances and graph pre-image problem.