nystrom-ncut 0.0.9__py3-none-any.whl → 0.0.10__py3-none-any.whl

Sign up to get free protection for your applications and to get access to all the features.
__init__.py ADDED
File without changes
nystrom_ncut/__init__.py CHANGED
@@ -1,12 +1,13 @@
1
- from .ncut_pytorch import (
2
- NCUT,
1
+ from .nystrom import (
2
+ DistanceRealization,
3
+ NCut,
3
4
  axis_align,
4
5
  )
5
6
  from .propagation_utils import (
7
+ distance_from_features,
6
8
  affinity_from_features,
7
9
  extrapolate_knn_with_subsampling,
8
10
  extrapolate_knn,
9
- quantile_normalize,
10
11
  )
11
12
  from .visualize_utils import (
12
13
  rgb_from_tsne_3d,
@@ -0,0 +1,7 @@
1
+ from .distance_realization import (
2
+ DistanceRealization,
3
+ )
4
+ from .normalized_cut import (
5
+ NCut,
6
+ axis_align,
7
+ )
@@ -0,0 +1,127 @@
1
+ import torch
2
+
3
+ from .nystrom import (
4
+ EigSolverOptions,
5
+ OnlineKernel,
6
+ OnlineNystromSubsampleFit,
7
+ solve_eig,
8
+ )
9
+ from ..common import (
10
+ DistanceOptions,
11
+ SampleOptions,
12
+ )
13
+ from ..propagation_utils import (
14
+ distance_from_features,
15
+ )
16
+
17
+
18
+ class GramKernel(OnlineKernel):
19
+ def __init__(
20
+ self,
21
+ distance: DistanceOptions,
22
+ eig_solver: EigSolverOptions,
23
+ ):
24
+ self.distance: DistanceOptions = distance
25
+ self.eig_solver: EigSolverOptions = eig_solver
26
+
27
+ # Anchor matrices
28
+ self.anchor_features: torch.Tensor = None # [n x d]
29
+ self.A: torch.Tensor = None # [n x n]
30
+ self.Ainv: torch.Tensor = None # [n x n]
31
+
32
+ # Updated matrices
33
+ self.a_r: torch.Tensor = None # [n]
34
+ self.b_r: torch.Tensor = None # [n]
35
+ self.matrix_sum: torch.Tensor = torch.zeros(()) # []
36
+ self.n_features: int = None # N
37
+
38
+ def fit(self, features: torch.Tensor) -> None:
39
+ self.anchor_features = features # [n x d]
40
+ self.A = -0.5 * distance_from_features(
41
+ self.anchor_features, # [n x d]
42
+ self.anchor_features,
43
+ distance=self.distance,
44
+ ) # [n x n]
45
+ d = features.shape[-1]
46
+ U, L = solve_eig(
47
+ self.A,
48
+ num_eig=d + 1, # d * (d + 3) // 2 + 1,
49
+ eig_solver=self.eig_solver,
50
+ ) # [n x (d + 1)], [d + 1]
51
+ self.Ainv = U @ torch.diag(1 / L) @ U.mT # [n x n]
52
+ self.a_r = torch.sum(self.A, dim=-1) # [n]
53
+ self.b_r = torch.zeros_like(self.a_r) # [n]
54
+ self.matrix_sum = torch.sum(self.a_r) # []
55
+ self.n_features = features.shape[0] # n
56
+
57
+ def update(self, features: torch.Tensor) -> torch.Tensor:
58
+ B = -0.5 * distance_from_features(
59
+ self.anchor_features, # [n x d]
60
+ features, # [m x d]
61
+ distance=self.distance,
62
+ ) # [n x m]
63
+ b_r = torch.sum(B, dim=-1) # [n]
64
+ b_c = torch.sum(B, dim=-2) # [m]
65
+ self.b_r = self.b_r + b_r # [n]
66
+ self.matrix_sum = (
67
+ torch.sum(self.a_r)
68
+ + 2 * torch.sum(self.b_r)
69
+ + self.Ainv @ self.b_r @ self.b_r
70
+ ) # []
71
+ self.n_features += features.shape[0] # N
72
+
73
+ row_sum = self.a_r + self.b_r # [n]
74
+ col_sum = b_c + B.mT @ self.Ainv @ self.b_r # [m]
75
+ shift = -(row_sum[:, None] + col_sum) / self.n_features + self.matrix_sum / (self.n_features ** 2) # [n x m]
76
+ return (B + shift).mT # [m x n]
77
+
78
+ def transform(self, features: torch.Tensor = None) -> torch.Tensor:
79
+ row_sum = self.a_r + self.b_r
80
+ if features is None:
81
+ B = self.A # [n x n]
82
+ col_sum = row_sum # [n]
83
+ else:
84
+ B = -0.5 * distance_from_features(
85
+ self.anchor_features,
86
+ features,
87
+ distance=self.distance,
88
+ )
89
+ b_c = torch.sum(B, dim=-2) # [m]
90
+ col_sum = b_c + B.mT @ self.Ainv @ self.b_r # [m]
91
+ shift = -(row_sum[:, None] + col_sum) / self.n_features + self.matrix_sum / (self.n_features ** 2) # [n x m]
92
+ return (B + shift).mT # [m x n]
93
+
94
+
95
+ class DistanceRealization(OnlineNystromSubsampleFit):
96
+ """Nystrom Distance Realization for large scale graph."""
97
+
98
+ def __init__(
99
+ self,
100
+ n_components: int = 100,
101
+ num_sample: int = 10000,
102
+ sample_method: SampleOptions = "farthest",
103
+ distance: DistanceOptions = "cosine",
104
+ eig_solver: EigSolverOptions = "svd_lowrank",
105
+ chunk_size: int = 8192,
106
+ ):
107
+ """
108
+ Args:
109
+ n_components (int): number of top eigenvectors to return
110
+ num_sample (int): number of samples for Nystrom-like approximation,
111
+ reduce only if memory is not enough, increase for better approximation
112
+ sample_method (str): subgraph sampling, ['farthest', 'random'].
113
+ farthest point sampling is recommended for better Nystrom-approximation accuracy
114
+ distance (str): distance metric for affinity matrix, ['cosine', 'euclidean', 'rbf'].
115
+ eig_solver (str): eigen decompose solver, ['svd_lowrank', 'lobpcg', 'svd', 'eigh'].
116
+ chunk_size (int): chunk size for large-scale matrix multiplication
117
+ """
118
+ OnlineNystromSubsampleFit.__init__(
119
+ self,
120
+ n_components=n_components,
121
+ kernel=GramKernel(distance, eig_solver),
122
+ num_sample=num_sample,
123
+ sample_method=sample_method,
124
+ eig_solver=eig_solver,
125
+ chunk_size=chunk_size,
126
+ )
127
+ self.distance: DistanceOptions = distance
@@ -1,22 +1,18 @@
1
- import logging
2
- from typing import Tuple
3
-
4
1
  import torch
5
2
  import torch.nn.functional as Fn
6
3
 
7
- from .common import (
8
- DistanceOptions,
9
- SampleOptions,
10
- )
11
4
  from .nystrom import (
12
5
  EigSolverOptions,
13
6
  OnlineKernel,
14
- OnlineNystrom,
7
+ OnlineNystromSubsampleFit,
15
8
  solve_eig,
16
9
  )
17
- from .propagation_utils import (
10
+ from ..common import (
11
+ DistanceOptions,
12
+ SampleOptions,
13
+ )
14
+ from ..propagation_utils import (
18
15
  affinity_from_features,
19
- run_subgraph_sampling,
20
16
  )
21
17
 
22
18
 
@@ -68,16 +64,16 @@ class LaplacianKernel(OnlineKernel):
68
64
  b_c = torch.sum(B, dim=-2) # [m]
69
65
  self.b_r = self.b_r + b_r # [n]
70
66
 
71
- rowscale = self.a_r + self.b_r # [n]
72
- colscale = b_c + B.mT @ self.Ainv @ self.b_r # [m]
73
- scale = (rowscale[:, None] * colscale) ** -0.5 # [n x m]
67
+ row_sum = self.a_r + self.b_r # [n]
68
+ col_sum = b_c + B.mT @ self.Ainv @ self.b_r # [m]
69
+ scale = (row_sum[:, None] * col_sum) ** -0.5 # [n x m]
74
70
  return (B * scale).mT # [m x n]
75
71
 
76
72
  def transform(self, features: torch.Tensor = None) -> torch.Tensor:
77
- rowscale = self.a_r + self.b_r # [n]
73
+ row_sum = self.a_r + self.b_r # [n]
78
74
  if features is None:
79
75
  B = self.A # [n x n]
80
- colscale = rowscale # [n]
76
+ col_sum = row_sum # [n]
81
77
  else:
82
78
  B = affinity_from_features(
83
79
  self.anchor_features, # [n x d]
@@ -86,12 +82,12 @@ class LaplacianKernel(OnlineKernel):
86
82
  distance=self.distance,
87
83
  ) # [n x m]
88
84
  b_c = torch.sum(B, dim=-2) # [m]
89
- colscale = b_c + B.mT @ self.Ainv @ self.b_r # [m]
90
- scale = (rowscale[:, None] * colscale) ** -0.5 # [n x m]
85
+ col_sum = b_c + B.mT @ self.Ainv @ self.b_r # [m]
86
+ scale = (row_sum[:, None] * col_sum) ** -0.5 # [n x m]
91
87
  return (B * scale).mT # [m x n]
92
88
 
93
89
 
94
- class NCUT(OnlineNystrom):
90
+ class NCut(OnlineNystromSubsampleFit):
95
91
  """Nystrom Normalized Cut for large scale graph."""
96
92
 
97
93
  def __init__(
@@ -102,7 +98,6 @@ class NCUT(OnlineNystrom):
102
98
  sample_method: SampleOptions = "farthest",
103
99
  distance: DistanceOptions = "cosine",
104
100
  eig_solver: EigSolverOptions = "svd_lowrank",
105
- normalize_features: bool = None,
106
101
  chunk_size: int = 8192,
107
102
  ):
108
103
  """
@@ -116,110 +111,18 @@ class NCUT(OnlineNystrom):
116
111
  farthest point sampling is recommended for better Nystrom-approximation accuracy
117
112
  distance (str): distance metric for affinity matrix, ['cosine', 'euclidean', 'rbf'].
118
113
  eig_solver (str): eigen decompose solver, ['svd_lowrank', 'lobpcg', 'svd', 'eigh'].
119
- normalize_features (bool): normalize input features before computing affinity matrix,
120
- default 'None' is True for cosine distance, False for euclidean distance and rbf
121
114
  chunk_size (int): chunk size for large-scale matrix multiplication
122
115
  """
123
- OnlineNystrom.__init__(
116
+ OnlineNystromSubsampleFit.__init__(
124
117
  self,
125
118
  n_components=n_components,
126
119
  kernel=LaplacianKernel(affinity_focal_gamma, distance, eig_solver),
120
+ num_sample=num_sample,
121
+ sample_method=sample_method,
127
122
  eig_solver=eig_solver,
128
123
  chunk_size=chunk_size,
129
124
  )
130
- self.num_sample: int = num_sample
131
- self.sample_method: SampleOptions = sample_method
132
- self.anchor_indices: torch.Tensor = None
133
125
  self.distance: DistanceOptions = distance
134
- self.normalize_features: bool = normalize_features
135
- if self.normalize_features is None:
136
- if distance in ["cosine"]:
137
- self.normalize_features = True
138
- if distance in ["euclidean", "rbf"]:
139
- self.normalize_features = False
140
-
141
- self.chunk_size: int = chunk_size
142
-
143
- def _fit_helper(
144
- self,
145
- features: torch.Tensor,
146
- precomputed_sampled_indices: torch.Tensor,
147
- ) -> Tuple[torch.Tensor, torch.Tensor]:
148
- _n = features.shape[0]
149
- if self.num_sample >= _n:
150
- logging.info(
151
- f"NCUT nystrom num_sample is larger than number of input samples, nyström approximation is not needed, setting num_sample={_n}"
152
- )
153
- self.num_sample = _n
154
-
155
- assert self.distance in ["cosine", "euclidean", "rbf"], "distance should be 'cosine', 'euclidean', 'rbf'"
156
-
157
- if self.normalize_features:
158
- # features need to be normalized for affinity matrix computation (cosine distance)
159
- features = torch.nn.functional.normalize(features, dim=-1)
160
-
161
- if precomputed_sampled_indices is not None:
162
- _sampled_indices = precomputed_sampled_indices
163
- else:
164
- _sampled_indices = run_subgraph_sampling(
165
- features,
166
- self.num_sample,
167
- sample_method=self.sample_method,
168
- )
169
- self.anchor_indices = torch.sort(_sampled_indices).values
170
- sampled_features = features[self.anchor_indices]
171
- OnlineNystrom.fit(self, sampled_features)
172
-
173
- _n_not_sampled = _n - len(sampled_features)
174
- if _n_not_sampled > 0:
175
- unsampled_indices = torch.full((_n,), True, device=features.device).scatter_(0, self.anchor_indices, False)
176
- unsampled_features = features[unsampled_indices]
177
- V_unsampled, _ = OnlineNystrom.update(self, unsampled_features)
178
- else:
179
- unsampled_indices = V_unsampled = None
180
- return unsampled_indices, V_unsampled
181
-
182
- def fit(
183
- self,
184
- features: torch.Tensor,
185
- precomputed_sampled_indices: torch.Tensor = None,
186
- ):
187
- """Fit Nystrom Normalized Cut on the input features.
188
- Args:
189
- features (torch.Tensor): input features, shape (n_samples, n_features)
190
- precomputed_sampled_indices (torch.Tensor): precomputed sampled indices, shape (num_sample,)
191
- override the sample_method, if not None
192
- Returns:
193
- (NCUT): self
194
- """
195
- NCUT._fit_helper(self, features, precomputed_sampled_indices)
196
- return self
197
-
198
- def fit_transform(
199
- self,
200
- features: torch.Tensor,
201
- precomputed_sampled_indices: torch.Tensor = None,
202
- ) -> Tuple[torch.Tensor, torch.Tensor]:
203
- """
204
- Args:
205
- features (torch.Tensor): input features, shape (n_samples, n_features)
206
- precomputed_sampled_indices (torch.Tensor): precomputed sampled indices, shape (num_sample,)
207
- override the sample_method, if not None
208
-
209
- Returns:
210
- (torch.Tensor): eigen_vectors, shape (n_samples, num_eig)
211
- (torch.Tensor): eigen_values, sorted in descending order, shape (num_eig,)
212
- """
213
- unsampled_indices, V_unsampled = NCUT._fit_helper(self, features, precomputed_sampled_indices)
214
- V_sampled, L = OnlineNystrom.transform(self)
215
-
216
- if unsampled_indices is not None:
217
- V = torch.zeros((len(unsampled_indices), self.n_components), device=features.device)
218
- V[~unsampled_indices] = V_sampled
219
- V[unsampled_indices] = V_unsampled
220
- else:
221
- V = V_sampled
222
- return V, L
223
126
 
224
127
 
225
128
  def axis_align(eigen_vectors: torch.Tensor, max_iter=300):
@@ -1,10 +1,15 @@
1
+ import logging
1
2
  from typing import Literal, Tuple
2
3
 
3
4
  import torch
4
5
 
5
- from .common import (
6
+ from ..common import (
7
+ SampleOptions,
6
8
  ceildiv,
7
9
  )
10
+ from ..propagation_utils import (
11
+ run_subgraph_sampling,
12
+ )
8
13
 
9
14
 
10
15
  EigSolverOptions = Literal["svd_lowrank", "lobpcg", "svd", "eigh"]
@@ -134,6 +139,102 @@ class OnlineNystrom:
134
139
  return VS, self.LS # [m x n_components], [n_components]
135
140
 
136
141
 
142
+ class OnlineNystromSubsampleFit(OnlineNystrom):
143
+ def __init__(
144
+ self,
145
+ n_components: int,
146
+ kernel: OnlineKernel,
147
+ num_sample: int,
148
+ sample_method: SampleOptions,
149
+ eig_solver: EigSolverOptions = "svd_lowrank",
150
+ chunk_size: int = 8192,
151
+ ):
152
+ OnlineNystrom.__init__(
153
+ self,
154
+ n_components=n_components,
155
+ kernel=kernel,
156
+ eig_solver=eig_solver,
157
+ chunk_size=chunk_size,
158
+ )
159
+ self.num_sample: int = num_sample
160
+ self.sample_method: SampleOptions = sample_method
161
+ self.anchor_indices: torch.Tensor = None
162
+
163
+ def _fit_helper(
164
+ self,
165
+ features: torch.Tensor,
166
+ precomputed_sampled_indices: torch.Tensor,
167
+ ) -> Tuple[torch.Tensor, torch.Tensor]:
168
+ _n = features.shape[0]
169
+ if self.num_sample >= _n:
170
+ logging.info(
171
+ f"NCUT nystrom num_sample is larger than number of input samples, nyström approximation is not needed, setting num_sample={_n}"
172
+ )
173
+ self.num_sample = _n
174
+
175
+ if precomputed_sampled_indices is not None:
176
+ self.anchor_indices = precomputed_sampled_indices
177
+ else:
178
+ self.anchor_indices = run_subgraph_sampling(
179
+ features,
180
+ self.num_sample,
181
+ sample_method=self.sample_method,
182
+ )
183
+ sampled_features = features[self.anchor_indices]
184
+ OnlineNystrom.fit(self, sampled_features)
185
+
186
+ _n_not_sampled = _n - len(sampled_features)
187
+ if _n_not_sampled > 0:
188
+ unsampled_indices = torch.full((_n,), True, device=features.device).scatter_(0, self.anchor_indices, False)
189
+ unsampled_features = features[unsampled_indices]
190
+ V_unsampled, _ = OnlineNystrom.update(self, unsampled_features)
191
+ else:
192
+ unsampled_indices = V_unsampled = None
193
+ return unsampled_indices, V_unsampled
194
+
195
+ def fit(
196
+ self,
197
+ features: torch.Tensor,
198
+ precomputed_sampled_indices: torch.Tensor = None,
199
+ ):
200
+ """Fit Nystrom Normalized Cut on the input features.
201
+ Args:
202
+ features (torch.Tensor): input features, shape (n_samples, n_features)
203
+ precomputed_sampled_indices (torch.Tensor): precomputed sampled indices, shape (num_sample,)
204
+ override the sample_method, if not None
205
+ Returns:
206
+ (NCut): self
207
+ """
208
+ OnlineNystromSubsampleFit._fit_helper(self, features, precomputed_sampled_indices)
209
+ return self
210
+
211
+ def fit_transform(
212
+ self,
213
+ features: torch.Tensor,
214
+ precomputed_sampled_indices: torch.Tensor = None,
215
+ ) -> Tuple[torch.Tensor, torch.Tensor]:
216
+ """
217
+ Args:
218
+ features (torch.Tensor): input features, shape (n_samples, n_features)
219
+ precomputed_sampled_indices (torch.Tensor): precomputed sampled indices, shape (num_sample,)
220
+ override the sample_method, if not None
221
+
222
+ Returns:
223
+ (torch.Tensor): eigen_vectors, shape (n_samples, num_eig)
224
+ (torch.Tensor): eigen_values, sorted in descending order, shape (num_eig,)
225
+ """
226
+ unsampled_indices, V_unsampled = OnlineNystromSubsampleFit._fit_helper(self, features, precomputed_sampled_indices)
227
+ V_sampled, L = OnlineNystrom.transform(self)
228
+
229
+ if unsampled_indices is not None:
230
+ V = torch.zeros((len(unsampled_indices), self.n_components), device=features.device)
231
+ V[~unsampled_indices] = V_sampled
232
+ V[unsampled_indices] = V_unsampled
233
+ else:
234
+ V = V_sampled
235
+ return V, L
236
+
237
+
137
238
  def solve_eig(
138
239
  A: torch.Tensor,
139
240
  num_eig: int,
@@ -47,6 +47,7 @@ def run_subgraph_sampling(
47
47
  sampled_indices = torch.randperm(features.shape[0])[:num_sample]
48
48
  else:
49
49
  raise ValueError("sample_method should be 'farthest' or 'random'")
50
+ sampled_indices = torch.sort(sampled_indices).values
50
51
  return sampled_indices.to(features.device)
51
52
 
52
53
 
@@ -36,12 +36,18 @@ def _rgb_with_dimensionality_reduction(
36
36
  device: str,
37
37
  ) -> Tuple[torch.Tensor, torch.Tensor]:
38
38
 
39
- features = extrapolate_knn(
40
- features,
41
- features,
42
- features,
43
- distance="cosine",
44
- )
39
+ if True:
40
+ _subgraph_indices = run_subgraph_sampling(
41
+ features,
42
+ num_sample=10000,
43
+ sample_method="farthest",
44
+ )
45
+ features = extrapolate_knn(
46
+ features[_subgraph_indices],
47
+ features[_subgraph_indices],
48
+ features,
49
+ distance="cosine",
50
+ )
45
51
 
46
52
  subgraph_indices = run_subgraph_sampling(
47
53
  features,
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.2
2
2
  Name: nystrom_ncut
3
- Version: 0.0.9
3
+ Version: 0.0.10
4
4
  Summary: Normalized Cut and Nyström Approximation
5
5
  Author-email: Huzheng Yang <huze.yann@gmail.com>, Wentinn Liao <wentinn.liao@gmail.com>
6
6
  Project-URL: Documentation, https://github.com/JophiArcana/Nystrom-NCUT/
@@ -0,0 +1,14 @@
1
+ __init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
2
+ nystrom_ncut/__init__.py,sha256=JKfF6atok5T9V692RhlhgeRO5a2cN-bfAVa9irmTLfs,463
3
+ nystrom_ncut/common.py,sha256=RMPQvg9R2s7V-q7zAStN9YCZt7gpc5Ut-KSKtvELBQ4,1934
4
+ nystrom_ncut/propagation_utils.py,sha256=WeWKxRBm01ITILMgjsit5_fCe9oW1kJOPmAjjcmliMo,10340
5
+ nystrom_ncut/visualize_utils.py,sha256=Z_bcoxwmWpTxhQ_yoAXqTnYDf269IuT0b0Sm2EVQpRw,17422
6
+ nystrom_ncut/nystrom/__init__.py,sha256=4EpxD3Cmc8Fif4vo8DG-6FpTfCnNanD5zCZxK3WrMwQ,121
7
+ nystrom_ncut/nystrom/distance_realization.py,sha256=8AWUlZKZEPfhQHxYTZt0uzKedVp8ZB1wb__7M2Fy-Eo,5529
8
+ nystrom_ncut/nystrom/normalized_cut.py,sha256=_U3zrbe6V-5TQ4uWmqckxs2JTIhygQlnRDTFBI1ghD4,7194
9
+ nystrom_ncut/nystrom/nystrom.py,sha256=VJPA17I8cVvjILUABJjkVA5kkXbTmHDyrtcWvu5xs-0,12571
10
+ nystrom_ncut-0.0.10.dist-info/LICENSE,sha256=2bm9uFabQZ3Ykb_SaSU_uUbAj2-htc6WJQmS_65qD00,1073
11
+ nystrom_ncut-0.0.10.dist-info/METADATA,sha256=sqs2WHdNbJeT5zvlq_WWHHRvHTz1mHVbDL3PsE1NMBI,6059
12
+ nystrom_ncut-0.0.10.dist-info/WHEEL,sha256=In9FTNxeP60KnTkGw7wk6mJPYd_dQSjEZmXdBdMCI-8,91
13
+ nystrom_ncut-0.0.10.dist-info/top_level.txt,sha256=gM8IWWHYysIRTCvCTcdS4RShOyl9pxpylgSwPUZR2XM,22
14
+ nystrom_ncut-0.0.10.dist-info/RECORD,,
@@ -1,11 +0,0 @@
1
- nystrom_ncut/__init__.py,sha256=Vlc_iAlfvTNUiJXpZLWUOaL2Q-YqZqgr7WoG6cVnD0g,439
2
- nystrom_ncut/common.py,sha256=RMPQvg9R2s7V-q7zAStN9YCZt7gpc5Ut-KSKtvELBQ4,1934
3
- nystrom_ncut/ncut_pytorch.py,sha256=-SKs9AdkafJSGkeYt4LwhbKZr8oq9JA5caAqjiVDAzU,11220
4
- nystrom_ncut/nystrom.py,sha256=1ngKzRrY9i2jEDo0EP8I9RsNTQzy4S7pmvkpayIkUOQ,8811
5
- nystrom_ncut/propagation_utils.py,sha256=AEKgWVw7x_podLEzELdCQeMmTJfAYfG4TiPuKbrN8Sw,10279
6
- nystrom_ncut/visualize_utils.py,sha256=p4wXXg47vqMn8c1N6NxGxtmPNwOzqquDvpMEjZVdq34,17196
7
- nystrom_ncut-0.0.9.dist-info/LICENSE,sha256=2bm9uFabQZ3Ykb_SaSU_uUbAj2-htc6WJQmS_65qD00,1073
8
- nystrom_ncut-0.0.9.dist-info/METADATA,sha256=umrvGtKsci280oi3D5I_BeLIt8ajBogC5veHYrRln48,6058
9
- nystrom_ncut-0.0.9.dist-info/WHEEL,sha256=In9FTNxeP60KnTkGw7wk6mJPYd_dQSjEZmXdBdMCI-8,91
10
- nystrom_ncut-0.0.9.dist-info/top_level.txt,sha256=j7g_j0S048EvguFFnGgD5Ewd3r2H6klsxd5A4dd-wHw,13
11
- nystrom_ncut-0.0.9.dist-info/RECORD,,