darabos commited on
Commit
145bf26
·
unverified ·
2 Parent(s): 45b3519 8886bbc

Merge pull request #73 from biggraph/darabos-rdkit

Browse files
.github/workflows/test.yaml CHANGED
@@ -24,7 +24,7 @@ jobs:
24
  run: |
25
  eval `ssh-agent -s`
26
  ssh-add - <<< '${{ secrets.LYNXSCRIBE_DEPLOY_KEY }}'
27
- uv pip install -e lynxkite-core/[dev] lynxkite-app/[dev] lynxkite-graph-analytics/[dev] lynxkite-lynxscribe/ lynxkite-pillow-example/
28
  env:
29
  UV_SYSTEM_PYTHON: 1
30
 
 
24
  run: |
25
  eval `ssh-agent -s`
26
  ssh-add - <<< '${{ secrets.LYNXSCRIBE_DEPLOY_KEY }}'
27
+ uv pip install -e lynxkite-core/[dev] -e lynxkite-app/[dev] -e lynxkite-graph-analytics/[dev] -e lynxkite-bio -e lynxkite-lynxscribe/ -e lynxkite-pillow-example/
28
  env:
29
  UV_SYSTEM_PYTHON: 1
30
 
README.md CHANGED
@@ -14,6 +14,7 @@ original LynxKite. The primary goals of this rewrite are:
14
  - `lynxkite-graph-analytics`: Graph analytics plugin. The classical LynxKite experience!
15
  - `lynxkite-pillow`: A simple example plugin.
16
  - `lynxkite-lynxscribe`: A plugin for building and running LynxScribe applications.
 
17
  - `docs`: User-facing documentation. It's shared between all packages.
18
 
19
  ## Development
@@ -25,7 +26,7 @@ uv venv
25
  source .venv/bin/activate
26
  uvx pre-commit install
27
  # The [dev] tag is only needed if you intend on running tests
28
- uv pip install -e lynxkite-core/[dev] -e lynxkite-app/[dev] -e lynxkite-graph-analytics/[dev] -e lynxkite-lynxscribe/ -e lynxkite-pillow-example/
29
  ```
30
 
31
  This also builds the frontend, hopefully very quickly. To run it:
 
14
  - `lynxkite-graph-analytics`: Graph analytics plugin. The classical LynxKite experience!
15
  - `lynxkite-pillow`: A simple example plugin.
16
  - `lynxkite-lynxscribe`: A plugin for building and running LynxScribe applications.
17
+ - `lynxkite-bio`: Bioinformatics additions for LynxKite Graph Analytics.
18
  - `docs`: User-facing documentation. It's shared between all packages.
19
 
20
  ## Development
 
26
  source .venv/bin/activate
27
  uvx pre-commit install
28
  # The [dev] tag is only needed if you intend on running tests
29
+ uv pip install -e lynxkite-core/[dev] -e lynxkite-app/[dev] -e lynxkite-graph-analytics/[dev] -e lynxkite-bio -e lynxkite-lynxscribe/ -e lynxkite-pillow-example/
30
  ```
31
 
32
  This also builds the frontend, hopefully very quickly. To run it:
examples/AIMO CHANGED
@@ -1036,4 +1036,4 @@
1036
  "targetHandle": "input"
1037
  }
1038
  ]
1039
- }
 
1036
  "targetHandle": "input"
1037
  }
1038
  ]
1039
+ }
examples/Bio demo ADDED
@@ -0,0 +1,857 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ {
2
+ "env": "LynxKite Graph Analytics",
3
+ "nodes": [
4
+ {
5
+ "id": "Import CSV 1",
6
+ "type": "basic",
7
+ "data": {
8
+ "title": "Import CSV",
9
+ "params": {
10
+ "filename": "examples/drug_target_data_sample.csv",
11
+ "separator": "<auto>",
12
+ "columns": "<from file>"
13
+ },
14
+ "display": null,
15
+ "error": null,
16
+ "__execution_delay": 0.0,
17
+ "collapsed": null,
18
+ "meta": {
19
+ "params": {
20
+ "separator": {
21
+ "default": "<auto>",
22
+ "type": {
23
+ "type": "<class 'str'>"
24
+ },
25
+ "name": "separator"
26
+ },
27
+ "filename": {
28
+ "default": null,
29
+ "type": {
30
+ "type": "<class 'str'>"
31
+ },
32
+ "name": "filename"
33
+ },
34
+ "columns": {
35
+ "name": "columns",
36
+ "default": "<from file>",
37
+ "type": {
38
+ "type": "<class 'str'>"
39
+ }
40
+ }
41
+ },
42
+ "outputs": {
43
+ "output": {
44
+ "position": "right",
45
+ "name": "output",
46
+ "type": {
47
+ "type": "None"
48
+ }
49
+ }
50
+ },
51
+ "type": "basic",
52
+ "inputs": {},
53
+ "name": "Import CSV"
54
+ }
55
+ },
56
+ "position": {
57
+ "x": 300.92214591096035,
58
+ "y": 1018.8292637889027
59
+ },
60
+ "width": 200.0,
61
+ "height": 200.0
62
+ },
63
+ {
64
+ "id": "Parse SMILES 1",
65
+ "type": "basic",
66
+ "data": {
67
+ "title": "Parse SMILES",
68
+ "params": {
69
+ "smiles_column": "SMILES",
70
+ "table": "df",
71
+ "save_as": "mols"
72
+ },
73
+ "display": null,
74
+ "error": null,
75
+ "meta": {
76
+ "name": "Parse SMILES",
77
+ "type": "basic",
78
+ "params": {
79
+ "table": {
80
+ "default": "df",
81
+ "name": "table",
82
+ "type": {
83
+ "type": "<class 'str'>"
84
+ }
85
+ },
86
+ "smiles_column": {
87
+ "name": "smiles_column",
88
+ "default": "SMILES",
89
+ "type": {
90
+ "type": "<class 'str'>"
91
+ }
92
+ },
93
+ "save_as": {
94
+ "name": "save_as",
95
+ "type": {
96
+ "type": "<class 'str'>"
97
+ },
98
+ "default": "mols"
99
+ }
100
+ },
101
+ "inputs": {
102
+ "bundle": {
103
+ "name": "bundle",
104
+ "type": {
105
+ "type": "<class 'lynxkite_graph_analytics.lynxkite_ops.Bundle'>"
106
+ },
107
+ "position": "left"
108
+ }
109
+ },
110
+ "outputs": {
111
+ "output": {
112
+ "position": "right",
113
+ "type": {
114
+ "type": "None"
115
+ },
116
+ "name": "output"
117
+ }
118
+ }
119
+ }
120
+ },
121
+ "position": {
122
+ "x": 580.5892989328847,
123
+ "y": 1012.8823965480503
124
+ },
125
+ "width": 200.0,
126
+ "height": 200.0
127
+ },
128
+ {
129
+ "id": "Graph from molecule similarity 1",
130
+ "type": "basic",
131
+ "data": {
132
+ "title": "Graph from molecule similarity",
133
+ "params": {
134
+ "table": "df",
135
+ "average_degree": "3",
136
+ "mols_column": "mols"
137
+ },
138
+ "display": null,
139
+ "error": null,
140
+ "meta": {
141
+ "inputs": {
142
+ "bundle": {
143
+ "position": "left",
144
+ "name": "bundle",
145
+ "type": {
146
+ "type": "<class 'lynxkite_graph_analytics.lynxkite_ops.Bundle'>"
147
+ }
148
+ }
149
+ },
150
+ "outputs": {
151
+ "output": {
152
+ "type": {
153
+ "type": "None"
154
+ },
155
+ "position": "right",
156
+ "name": "output"
157
+ }
158
+ },
159
+ "type": "basic",
160
+ "name": "Graph from molecule similarity",
161
+ "params": {
162
+ "average_degree": {
163
+ "default": 10.0,
164
+ "type": {
165
+ "type": "<class 'int'>"
166
+ },
167
+ "name": "average_degree"
168
+ },
169
+ "table": {
170
+ "default": "df",
171
+ "name": "table",
172
+ "type": {
173
+ "type": "<class 'str'>"
174
+ }
175
+ },
176
+ "mols_column": {
177
+ "default": "mols",
178
+ "name": "mols_column",
179
+ "type": {
180
+ "type": "<class 'str'>"
181
+ }
182
+ }
183
+ }
184
+ },
185
+ "collapsed": null,
186
+ "__execution_delay": 0.0
187
+ },
188
+ "position": {
189
+ "x": 926.4337192214458,
190
+ "y": 1014.4451876264845
191
+ },
192
+ "height": 200.0,
193
+ "width": 200.0
194
+ },
195
+ {
196
+ "id": "Visualize graph 1",
197
+ "type": "visualization",
198
+ "data": {
199
+ "title": "Visualize graph",
200
+ "params": {
201
+ "color_nodes_by": "ORGANISM",
202
+ "label_by": "DRUG_NAME",
203
+ "color_edges_by": "similarity"
204
+ },
205
+ "display": {
206
+ "animationDuration": 500,
207
+ "animationEasingUpdate": "quinticInOut",
208
+ "tooltip": {
209
+ "show": true
210
+ },
211
+ "series": [
212
+ {
213
+ "type": "graph",
214
+ "lineStyle": {
215
+ "color": "gray",
216
+ "curveness": 0.3
217
+ },
218
+ "emphasis": {
219
+ "focus": "adjacency",
220
+ "lineStyle": {
221
+ "width": 10
222
+ }
223
+ },
224
+ "label": {
225
+ "position": "top",
226
+ "formatter": "{b}"
227
+ },
228
+ "data": [
229
+ {
230
+ "id": "0",
231
+ "x": -0.13734854736037372,
232
+ "y": 0.0003840689616843064,
233
+ "symbolSize": 22.360679774997894,
234
+ "itemStyle": {
235
+ "color": "#a6cee3"
236
+ },
237
+ "label": {
238
+ "show": true
239
+ },
240
+ "name": "phencyclidine",
241
+ "value": "Torpedo californica"
242
+ },
243
+ {
244
+ "id": "1",
245
+ "x": 0.4859426303201164,
246
+ "y": -0.09785193844993345,
247
+ "symbolSize": 22.360679774997894,
248
+ "itemStyle": {
249
+ "color": "#1f78b4"
250
+ },
251
+ "label": {
252
+ "show": true
253
+ },
254
+ "name": "triazolam",
255
+ "value": "Homo sapiens"
256
+ },
257
+ {
258
+ "id": "2",
259
+ "x": -0.36490537596845657,
260
+ "y": 0.655747709585604,
261
+ "symbolSize": 22.360679774997894,
262
+ "itemStyle": {
263
+ "color": "#1f78b4"
264
+ },
265
+ "label": {
266
+ "show": true
267
+ },
268
+ "name": "gentian violet",
269
+ "value": "Homo sapiens"
270
+ },
271
+ {
272
+ "id": "3",
273
+ "x": 0.8039553347691868,
274
+ "y": 0.44172015990267444,
275
+ "symbolSize": 22.360679774997894,
276
+ "itemStyle": {
277
+ "color": "#1f78b4"
278
+ },
279
+ "label": {
280
+ "show": true
281
+ },
282
+ "name": "ipratropium",
283
+ "value": "Homo sapiens"
284
+ },
285
+ {
286
+ "id": "4",
287
+ "x": -0.7876440417604583,
288
+ "y": -1.0,
289
+ "symbolSize": 22.360679774997894,
290
+ "itemStyle": {
291
+ "color": "#1f78b4"
292
+ },
293
+ "label": {
294
+ "show": true
295
+ },
296
+ "name": "deoxycholic acid",
297
+ "value": "Homo sapiens"
298
+ }
299
+ ],
300
+ "links": [
301
+ {
302
+ "source": "3",
303
+ "target": "4",
304
+ "lineStyle": {
305
+ "color": "#440154"
306
+ },
307
+ "value": 0.8481012658227848
308
+ },
309
+ {
310
+ "source": "0",
311
+ "target": "1",
312
+ "lineStyle": {
313
+ "color": "#3a538b"
314
+ },
315
+ "value": 0.8813559322033898
316
+ },
317
+ {
318
+ "source": "0",
319
+ "target": "4",
320
+ "lineStyle": {
321
+ "color": "#3a538b"
322
+ },
323
+ "value": 0.8813559322033898
324
+ },
325
+ {
326
+ "source": "0",
327
+ "target": "3",
328
+ "lineStyle": {
329
+ "color": "#26818e"
330
+ },
331
+ "value": 0.9047619047619048
332
+ },
333
+ {
334
+ "source": "1",
335
+ "target": "2",
336
+ "lineStyle": {
337
+ "color": "#23898d"
338
+ },
339
+ "value": 0.9090909090909091
340
+ },
341
+ {
342
+ "source": "1",
343
+ "target": "3",
344
+ "lineStyle": {
345
+ "color": "#1ea087"
346
+ },
347
+ "value": 0.921875
348
+ },
349
+ {
350
+ "source": "0",
351
+ "target": "2",
352
+ "lineStyle": {
353
+ "color": "#3ebc73"
354
+ },
355
+ "value": 0.9375
356
+ },
357
+ {
358
+ "source": "1",
359
+ "target": "4",
360
+ "lineStyle": {
361
+ "color": "#4fc369"
362
+ },
363
+ "value": 0.9420289855072463
364
+ },
365
+ {
366
+ "source": "2",
367
+ "target": "4",
368
+ "lineStyle": {
369
+ "color": "#9fd938"
370
+ },
371
+ "value": 0.9589041095890412
372
+ },
373
+ {
374
+ "source": "2",
375
+ "target": "3",
376
+ "lineStyle": {
377
+ "color": "#fde724"
378
+ },
379
+ "value": 0.9775280898876404
380
+ }
381
+ ]
382
+ }
383
+ ]
384
+ },
385
+ "error": null,
386
+ "__execution_delay": 0.0,
387
+ "collapsed": null,
388
+ "meta": {
389
+ "params": {
390
+ "label_by": {
391
+ "type": {
392
+ "format": "node attribute"
393
+ },
394
+ "name": "label_by",
395
+ "default": null
396
+ },
397
+ "color_nodes_by": {
398
+ "type": {
399
+ "format": "node attribute"
400
+ },
401
+ "name": "color_nodes_by",
402
+ "default": null
403
+ },
404
+ "color_edges_by": {
405
+ "name": "color_edges_by",
406
+ "type": {
407
+ "format": "edge attribute"
408
+ },
409
+ "default": null
410
+ }
411
+ },
412
+ "inputs": {
413
+ "graph": {
414
+ "type": {
415
+ "type": "<class 'lynxkite_graph_analytics.lynxkite_ops.Bundle'>"
416
+ },
417
+ "name": "graph",
418
+ "position": "left"
419
+ }
420
+ },
421
+ "outputs": {},
422
+ "position": {
423
+ "y": 167.0,
424
+ "x": 883.0
425
+ },
426
+ "type": "visualization",
427
+ "name": "Visualize graph"
428
+ }
429
+ },
430
+ "position": {
431
+ "x": 1254.2277640235457,
432
+ "y": 931.9705991370125
433
+ },
434
+ "height": 314.0,
435
+ "width": 407.0
436
+ },
437
+ {
438
+ "id": "Cypher 1",
439
+ "type": "basic",
440
+ "data": {
441
+ "title": "Cypher",
442
+ "params": {
443
+ "save_as": "result",
444
+ "query": "MATCH\n (a {ORGANISM: \"Homo sapiens\"})\n -[r]->\n (b {ORGANISM: \"Homo sapiens\"})\nWHERE\n r.similarity > 0.9\nRETURN a.DRUG_NAME, b.DRUG_NAME"
445
+ },
446
+ "display": null,
447
+ "error": null,
448
+ "meta": {
449
+ "outputs": {
450
+ "output": {
451
+ "type": {
452
+ "type": "None"
453
+ },
454
+ "name": "output",
455
+ "position": "right"
456
+ }
457
+ },
458
+ "inputs": {
459
+ "bundle": {
460
+ "position": "left",
461
+ "name": "bundle",
462
+ "type": {
463
+ "type": "<class 'lynxkite_graph_analytics.lynxkite_ops.Bundle'>"
464
+ }
465
+ }
466
+ },
467
+ "name": "Cypher",
468
+ "params": {
469
+ "query": {
470
+ "default": null,
471
+ "name": "query",
472
+ "type": {
473
+ "format": "textarea"
474
+ }
475
+ },
476
+ "save_as": {
477
+ "type": {
478
+ "type": "<class 'str'>"
479
+ },
480
+ "default": "result",
481
+ "name": "save_as"
482
+ }
483
+ },
484
+ "type": "basic",
485
+ "position": {
486
+ "x": 638.0,
487
+ "y": 577.0
488
+ }
489
+ },
490
+ "collapsed": null,
491
+ "__execution_delay": 0.0
492
+ },
493
+ "position": {
494
+ "x": 1209.2024653849007,
495
+ "y": 1567.9825632648467
496
+ },
497
+ "height": 267.0,
498
+ "width": 434.0
499
+ },
500
+ {
501
+ "id": "View tables 1",
502
+ "type": "table_view",
503
+ "data": {
504
+ "title": "View tables",
505
+ "params": {
506
+ "limit": 100.0
507
+ },
508
+ "display": {
509
+ "dataframes": {
510
+ "edges": {
511
+ "columns": [
512
+ "source",
513
+ "target",
514
+ "similarity"
515
+ ],
516
+ "data": [
517
+ [
518
+ 3.0,
519
+ 4.0,
520
+ 0.8481012658227848
521
+ ],
522
+ [
523
+ 0.0,
524
+ 1.0,
525
+ 0.8813559322033898
526
+ ],
527
+ [
528
+ 0.0,
529
+ 4.0,
530
+ 0.8813559322033898
531
+ ],
532
+ [
533
+ 0.0,
534
+ 3.0,
535
+ 0.9047619047619048
536
+ ],
537
+ [
538
+ 1.0,
539
+ 2.0,
540
+ 0.9090909090909091
541
+ ],
542
+ [
543
+ 1.0,
544
+ 3.0,
545
+ 0.921875
546
+ ],
547
+ [
548
+ 0.0,
549
+ 2.0,
550
+ 0.9375
551
+ ],
552
+ [
553
+ 1.0,
554
+ 4.0,
555
+ 0.9420289855072463
556
+ ],
557
+ [
558
+ 2.0,
559
+ 4.0,
560
+ 0.9589041095890412
561
+ ],
562
+ [
563
+ 2.0,
564
+ 3.0,
565
+ 0.9775280898876404
566
+ ]
567
+ ]
568
+ },
569
+ "nodes": {
570
+ "columns": [
571
+ "Unnamed: 0",
572
+ "DRUG_NAME",
573
+ "STRUCT_ID",
574
+ "TARGET_NAME",
575
+ "TARGET_CLASS",
576
+ "ACCESSION",
577
+ "GENE",
578
+ "SWISSPROT",
579
+ "ACT_VALUE",
580
+ "ACT_UNIT",
581
+ "ACT_TYPE",
582
+ "ACT_COMMENT",
583
+ "ACT_SOURCE",
584
+ "RELATION",
585
+ "MOA",
586
+ "MOA_SOURCE",
587
+ "ACT_SOURCE_URL",
588
+ "MOA_SOURCE_URL",
589
+ "ACTION_TYPE",
590
+ "TDL",
591
+ "ORGANISM",
592
+ "SMILES",
593
+ "InChI",
594
+ "InChIKey",
595
+ "INN",
596
+ "mols"
597
+ ],
598
+ "data": [
599
+ [
600
+ 9737,
601
+ "phencyclidine",
602
+ 2121,
603
+ "Acetylcholine receptor subunit alpha",
604
+ "Ion channel",
605
+ "P02710",
606
+ "CHRNA1",
607
+ "ACHA_TORCA",
608
+ 6.66,
609
+ null,
610
+ "IC50",
611
+ "Displacement of [3H]PCP from nAChR in Torpedo nobiliana electric organs membranes in presence of 100 uM carbachol by scintillation counting method",
612
+ "CHEMBL",
613
+ "=",
614
+ null,
615
+ "nan",
616
+ null,
617
+ "nan",
618
+ "nan",
619
+ "nan",
620
+ "Torpedo californica",
621
+ "C1CCN(CC1)C1(CCCCC1)C1=CC=CC=C1",
622
+ "InChI=1S/C17H25N/c1-4-10-16(11-5-1)17(12-6-2-7-13-17)18-14-8-3-9-15-18/h1,4-5,10-11H,2-3,6-9,12-15H2",
623
+ "JTJMJGYZQZDUJJ-UHFFFAOYSA-N",
624
+ "phencyclidine",
625
+ "<rdkit.Chem.rdchem.Mol object at 0x74d424169e70>"
626
+ ],
627
+ [
628
+ 12934,
629
+ "triazolam",
630
+ 2729,
631
+ "GABA A receptor alpha-3/beta-2/gamma-2",
632
+ "Ion channel",
633
+ "P18507|P34903|P47870",
634
+ "GABRG2|GABRA3|GABRB2",
635
+ "GBRG2_HUMAN|GBRA3_HUMAN|GBRB2_HUMAN",
636
+ 8.876,
637
+ null,
638
+ "Ki",
639
+ "nan",
640
+ "WOMBAT-PK",
641
+ "=",
642
+ 1.0,
643
+ "CHEMBL",
644
+ null,
645
+ "https://www.ebi.ac.uk/chembl/compound/inspect/CHEMBL646",
646
+ "POSITIVE ALLOSTERIC MODULATOR",
647
+ "Tclin|Tclin|Tclin|Tclin|Tclin|Tclin|Tclin|Tclin|Tclin",
648
+ "Homo sapiens",
649
+ "CC1=NN=C2CN=C(C3=CC(Cl)=CC=C3N12)C1=C(Cl)C=CC=C1",
650
+ "InChI=1S/C17H12Cl2N4/c1-10-21-22-16-9-20-17(12-4-2-3-5-14(12)19)13-8-11(18)6-7-15(13)23(10)16/h2-8H,9H2,1H3",
651
+ "JOFWLTCLBGQGBO-UHFFFAOYSA-N",
652
+ "triazolam",
653
+ "<rdkit.Chem.rdchem.Mol object at 0x74d42416ab20>"
654
+ ],
655
+ [
656
+ 15266,
657
+ "gentian violet",
658
+ 4138,
659
+ "D(2) dopamine receptor",
660
+ "GPCR",
661
+ "P14416",
662
+ "DRD2",
663
+ "DRD2_HUMAN",
664
+ 5.975,
665
+ null,
666
+ "Ki",
667
+ "DRUGMATRIX: Dopamine D2L radioligand binding (ligand: [3H] Spiperone)",
668
+ "DRUG MATRIX",
669
+ "=",
670
+ null,
671
+ "nan",
672
+ null,
673
+ "nan",
674
+ "nan",
675
+ "Tclin",
676
+ "Homo sapiens",
677
+ "CN(C)C1=CC=C(C=C1)C(C1=CC=C(C=C1)N(C)C)=C1C=CC(C=C1)=[N+](C)C",
678
+ "InChI=1S/C25H30N3/c1-26(2)22-13-7-19(8-14-22)25(20-9-15-23(16-10-20)27(3)4)21-11-17-24(18-12-21)28(5)6/h7-18H,1-6H3/q+1",
679
+ "LGLFFNDHMLKUMI-UHFFFAOYSA-N",
680
+ "gentian violet",
681
+ "<rdkit.Chem.rdchem.Mol object at 0x74d424169070>"
682
+ ],
683
+ [
684
+ 6488,
685
+ "ipratropium",
686
+ 1475,
687
+ "Muscarinic acetylcholine receptor M1",
688
+ "GPCR",
689
+ "P11229",
690
+ "CHRM1",
691
+ "ACM1_HUMAN",
692
+ 9.31,
693
+ null,
694
+ "Ki",
695
+ "nan",
696
+ "WOMBAT-PK",
697
+ "=",
698
+ null,
699
+ "nan",
700
+ null,
701
+ "nan",
702
+ "nan",
703
+ "Tclin",
704
+ "Homo sapiens",
705
+ "CC(C)[N+]1(C)[C@H]2CC[C@@H]1C[C@@H](C2)OC(=O)C(CO)C1=CC=CC=C1",
706
+ "InChI=1S/C20H30NO3/c1-14(2)21(3)16-9-10-17(21)12-18(11-16)24-20(23)19(13-22)15-7-5-4-6-8-15/h4-8,14,16-19,22H,9-13H2,1-3H3/q+1/t16-,17+,18+,19?,21?",
707
+ "OEXHQOGQTVQTAT-BHIXFJMTSA-N",
708
+ "ipratropium",
709
+ "<rdkit.Chem.rdchem.Mol object at 0x74d42416a7a0>"
710
+ ],
711
+ [
712
+ 17453,
713
+ "deoxycholic acid",
714
+ 4988,
715
+ "G-protein coupled bile acid receptor 1",
716
+ "GPCR",
717
+ "Q8TDU6",
718
+ "GPBAR1",
719
+ "GPBAR_HUMAN",
720
+ 6.2,
721
+ null,
722
+ "EC50",
723
+ "nan",
724
+ "IUPHAR",
725
+ "=",
726
+ null,
727
+ "nan",
728
+ null,
729
+ "nan",
730
+ "AGONIST",
731
+ "Tchem",
732
+ "Homo sapiens",
733
+ "C[C@H](CCC(O)=O)[C@H]1CC[C@H]2[C@@H]3CC[C@@H]4C[C@H](O)CC[C@]4(C)[C@H]3C[C@H](O)[C@]12C",
734
+ "InChI=1S/C24H40O4/c1-14(4-9-22(27)28)18-7-8-19-17-6-5-15-12-16(25)10-11-23(15,2)20(17)13-21(26)24(18,19)3/h14-21,25-26H,4-13H2,1-3H3,(H,27,28)/t14-,15-,16-,17+,18-,19+,20+,21+,23+,24-/m1/s1",
735
+ "KXGVEGMKQFWNSR-LLQZFEROSA-N",
736
+ "deoxycholic acid",
737
+ "<rdkit.Chem.rdchem.Mol object at 0x74d424168890>"
738
+ ]
739
+ ]
740
+ },
741
+ "result": {
742
+ "columns": [
743
+ "a.DRUG_NAME",
744
+ "b.DRUG_NAME"
745
+ ],
746
+ "data": [
747
+ [
748
+ "triazolam",
749
+ "gentian violet"
750
+ ],
751
+ [
752
+ "triazolam",
753
+ "ipratropium"
754
+ ],
755
+ [
756
+ "triazolam",
757
+ "deoxycholic acid"
758
+ ],
759
+ [
760
+ "gentian violet",
761
+ "deoxycholic acid"
762
+ ],
763
+ [
764
+ "gentian violet",
765
+ "ipratropium"
766
+ ]
767
+ ]
768
+ }
769
+ },
770
+ "relations": [
771
+ {
772
+ "df": "edges",
773
+ "source_column": "source",
774
+ "target_column": "target",
775
+ "source_table": "nodes",
776
+ "target_table": "nodes",
777
+ "source_key": "id",
778
+ "target_key": "id"
779
+ }
780
+ ],
781
+ "other": null
782
+ },
783
+ "error": null,
784
+ "meta": {
785
+ "params": {
786
+ "limit": {
787
+ "default": 100.0,
788
+ "name": "limit",
789
+ "type": {
790
+ "type": "<class 'int'>"
791
+ }
792
+ }
793
+ },
794
+ "inputs": {
795
+ "bundle": {
796
+ "name": "bundle",
797
+ "position": "left",
798
+ "type": {
799
+ "type": "<class 'lynxkite_graph_analytics.lynxkite_ops.Bundle'>"
800
+ }
801
+ }
802
+ },
803
+ "position": {
804
+ "x": 1183.0,
805
+ "y": 534.0
806
+ },
807
+ "name": "View tables",
808
+ "outputs": {},
809
+ "type": "table_view"
810
+ }
811
+ },
812
+ "position": {
813
+ "x": 1817.1767094971015,
814
+ "y": 1459.025582190881
815
+ },
816
+ "width": 322.0,
817
+ "height": 295.0
818
+ }
819
+ ],
820
+ "edges": [
821
+ {
822
+ "id": "Import CSV 1 Parse SMILES 1",
823
+ "source": "Import CSV 1",
824
+ "target": "Parse SMILES 1",
825
+ "sourceHandle": "output",
826
+ "targetHandle": "bundle"
827
+ },
828
+ {
829
+ "id": "Parse SMILES 1 Graph from molecule similarity 1",
830
+ "source": "Parse SMILES 1",
831
+ "target": "Graph from molecule similarity 1",
832
+ "sourceHandle": "output",
833
+ "targetHandle": "bundle"
834
+ },
835
+ {
836
+ "id": "Graph from molecule similarity 1 Visualize graph 1",
837
+ "source": "Graph from molecule similarity 1",
838
+ "target": "Visualize graph 1",
839
+ "sourceHandle": "output",
840
+ "targetHandle": "graph"
841
+ },
842
+ {
843
+ "id": "Graph from molecule similarity 1 Cypher 1",
844
+ "source": "Graph from molecule similarity 1",
845
+ "target": "Cypher 1",
846
+ "sourceHandle": "output",
847
+ "targetHandle": "bundle"
848
+ },
849
+ {
850
+ "id": "Cypher 1 View tables 1",
851
+ "source": "Cypher 1",
852
+ "target": "View tables 1",
853
+ "sourceHandle": "output",
854
+ "targetHandle": "bundle"
855
+ }
856
+ ]
857
+ }
examples/Graph RAG CHANGED
@@ -872,4 +872,4 @@
872
  "targetHandle": "edges"
873
  }
874
  ]
875
- }
 
872
  "targetHandle": "edges"
873
  }
874
  ]
875
+ }
examples/Image processing CHANGED
@@ -281,4 +281,4 @@
281
  "targetHandle": "image"
282
  }
283
  ]
284
- }
 
281
  "targetHandle": "image"
282
  }
283
  ]
284
+ }
examples/LynxScribe demo CHANGED
@@ -966,4 +966,4 @@
966
  "targetHandle": "chat_api"
967
  }
968
  ]
969
- }
 
966
  "targetHandle": "chat_api"
967
  }
968
  ]
969
+ }
examples/NetworkX demo CHANGED
The diff for this file is too large to render. See raw diff
 
examples/PyTorch demo CHANGED
@@ -620,4 +620,4 @@
620
  "targetHandle": "x"
621
  }
622
  ]
623
- }
 
620
  "targetHandle": "x"
621
  }
622
  ]
623
+ }
examples/RAG chatbot app CHANGED
@@ -605,4 +605,4 @@
605
  "targetHandle": "rag_graph"
606
  }
607
  ]
608
- }
 
605
  "targetHandle": "rag_graph"
606
  }
607
  ]
608
+ }
examples/drug_target_data_sample.csv ADDED
@@ -0,0 +1,6 @@
 
 
 
 
 
 
 
1
+ Unnamed: 0,DRUG_NAME,STRUCT_ID,TARGET_NAME,TARGET_CLASS,ACCESSION,GENE,SWISSPROT,ACT_VALUE,ACT_UNIT,ACT_TYPE,ACT_COMMENT,ACT_SOURCE,RELATION,MOA,MOA_SOURCE,ACT_SOURCE_URL,MOA_SOURCE_URL,ACTION_TYPE,TDL,ORGANISM,SMILES,InChI,InChIKey,INN
2
+ 9737,phencyclidine,2121,Acetylcholine receptor subunit alpha,Ion channel,P02710,CHRNA1,ACHA_TORCA,6.66,,IC50,Displacement of [3H]PCP from nAChR in Torpedo nobiliana electric organs membranes in presence of 100 uM carbachol by scintillation counting method,CHEMBL,=,,,,,,,Torpedo californica,C1CCN(CC1)C1(CCCCC1)C1=CC=CC=C1,"InChI=1S/C17H25N/c1-4-10-16(11-5-1)17(12-6-2-7-13-17)18-14-8-3-9-15-18/h1,4-5,10-11H,2-3,6-9,12-15H2",JTJMJGYZQZDUJJ-UHFFFAOYSA-N,phencyclidine
3
+ 12934,triazolam,2729,GABA A receptor alpha-3/beta-2/gamma-2,Ion channel,P18507|P34903|P47870,GABRG2|GABRA3|GABRB2,GBRG2_HUMAN|GBRA3_HUMAN|GBRB2_HUMAN,8.876,,Ki,,WOMBAT-PK,=,1.0,CHEMBL,,https://www.ebi.ac.uk/chembl/compound/inspect/CHEMBL646,POSITIVE ALLOSTERIC MODULATOR,Tclin|Tclin|Tclin|Tclin|Tclin|Tclin|Tclin|Tclin|Tclin,Homo sapiens,CC1=NN=C2CN=C(C3=CC(Cl)=CC=C3N12)C1=C(Cl)C=CC=C1,"InChI=1S/C17H12Cl2N4/c1-10-21-22-16-9-20-17(12-4-2-3-5-14(12)19)13-8-11(18)6-7-15(13)23(10)16/h2-8H,9H2,1H3",JOFWLTCLBGQGBO-UHFFFAOYSA-N,triazolam
4
+ 15266,gentian violet,4138,D(2) dopamine receptor,GPCR,P14416,DRD2,DRD2_HUMAN,5.975,,Ki,DRUGMATRIX: Dopamine D2L radioligand binding (ligand: [3H] Spiperone),DRUG MATRIX,=,,,,,,Tclin,Homo sapiens,CN(C)C1=CC=C(C=C1)C(C1=CC=C(C=C1)N(C)C)=C1C=CC(C=C1)=[N+](C)C,"InChI=1S/C25H30N3/c1-26(2)22-13-7-19(8-14-22)25(20-9-15-23(16-10-20)27(3)4)21-11-17-24(18-12-21)28(5)6/h7-18H,1-6H3/q+1",LGLFFNDHMLKUMI-UHFFFAOYSA-N,gentian violet
5
+ 6488,ipratropium,1475,Muscarinic acetylcholine receptor M1,GPCR,P11229,CHRM1,ACM1_HUMAN,9.31,,Ki,,WOMBAT-PK,=,,,,,,Tclin,Homo sapiens,CC(C)[N+]1(C)[C@H]2CC[C@@H]1C[C@@H](C2)OC(=O)C(CO)C1=CC=CC=C1,"InChI=1S/C20H30NO3/c1-14(2)21(3)16-9-10-17(21)12-18(11-16)24-20(23)19(13-22)15-7-5-4-6-8-15/h4-8,14,16-19,22H,9-13H2,1-3H3/q+1/t16-,17+,18+,19?,21?",OEXHQOGQTVQTAT-BHIXFJMTSA-N,ipratropium
6
+ 17453,deoxycholic acid,4988,G-protein coupled bile acid receptor 1,GPCR,Q8TDU6,GPBAR1,GPBAR_HUMAN,6.2,,EC50,,IUPHAR,=,,,,,AGONIST,Tchem,Homo sapiens,C[C@H](CCC(O)=O)[C@H]1CC[C@H]2[C@@H]3CC[C@@H]4C[C@H](O)CC[C@]4(C)[C@H]3C[C@H](O)[C@]12C,"InChI=1S/C24H40O4/c1-14(4-9-22(27)28)18-7-8-19-17-6-5-15-12-16(25)10-11-23(15,2)20(17)13-21(26)24(18,19)3/h14-21,25-26H,4-13H2,1-3H3,(H,27,28)/t14-,15-,16-,17+,18-,19+,20+,21+,23+,24-/m1/s1",KXGVEGMKQFWNSR-LLQZFEROSA-N,deoxycholic acid
lynxkite-app/src/lynxkite_app/crdt.py CHANGED
@@ -3,7 +3,6 @@
3
  import asyncio
4
  import contextlib
5
  import enum
6
- import pathlib
7
  import fastapi
8
  import os.path
9
  import pycrdt
 
3
  import asyncio
4
  import contextlib
5
  import enum
 
6
  import fastapi
7
  import os.path
8
  import pycrdt
lynxkite-app/web/package-lock.json CHANGED
@@ -8,7 +8,7 @@
8
  "name": "lynxkite",
9
  "version": "0.0.0",
10
  "dependencies": {
11
- "@esbuild/linux-x64": "^0.24.0",
12
  "@iconify-json/tabler": "^1.2.10",
13
  "@svgr/core": "^8.1.0",
14
  "@svgr/plugin-jsx": "^8.1.0",
@@ -632,9 +632,9 @@
632
  }
633
  },
634
  "node_modules/@esbuild/linux-x64": {
635
- "version": "0.24.2",
636
- "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.24.2.tgz",
637
- "integrity": "sha512-8Qi4nQcCTbLnK9WoMjdC9NiTG6/E38RNICU6sUNqK0QFxCYgoARqVqxdFmWkdonVsvGqWhmm7MO0jyTqLqwj0Q==",
638
  "cpu": [
639
  "x64"
640
  ],
@@ -3055,6 +3055,23 @@
3055
  "@esbuild/win32-x64": "0.24.2"
3056
  }
3057
  },
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
3058
  "node_modules/escalade": {
3059
  "version": "3.2.0",
3060
  "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
 
8
  "name": "lynxkite",
9
  "version": "0.0.0",
10
  "dependencies": {
11
+ "@esbuild/linux-x64": "^0.25.0",
12
  "@iconify-json/tabler": "^1.2.10",
13
  "@svgr/core": "^8.1.0",
14
  "@svgr/plugin-jsx": "^8.1.0",
 
632
  }
633
  },
634
  "node_modules/@esbuild/linux-x64": {
635
+ "version": "0.25.0",
636
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.25.0.tgz",
637
+ "integrity": "sha512-9yl91rHw/cpwMCNytUDxwj2XjFpxML0y9HAOH9pNVQDpQrBxHy01Dx+vaMu0N1CKa/RzBD2hB4u//nfc+Sd3Cw==",
638
  "cpu": [
639
  "x64"
640
  ],
 
3055
  "@esbuild/win32-x64": "0.24.2"
3056
  }
3057
  },
3058
+ "node_modules/esbuild/node_modules/@esbuild/linux-x64": {
3059
+ "version": "0.24.2",
3060
+ "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.24.2.tgz",
3061
+ "integrity": "sha512-8Qi4nQcCTbLnK9WoMjdC9NiTG6/E38RNICU6sUNqK0QFxCYgoARqVqxdFmWkdonVsvGqWhmm7MO0jyTqLqwj0Q==",
3062
+ "cpu": [
3063
+ "x64"
3064
+ ],
3065
+ "dev": true,
3066
+ "license": "MIT",
3067
+ "optional": true,
3068
+ "os": [
3069
+ "linux"
3070
+ ],
3071
+ "engines": {
3072
+ "node": ">=18"
3073
+ }
3074
+ },
3075
  "node_modules/escalade": {
3076
  "version": "3.2.0",
3077
  "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
lynxkite-app/web/package.json CHANGED
@@ -11,7 +11,7 @@
11
  "preview": "npx vite preview"
12
  },
13
  "dependencies": {
14
- "@esbuild/linux-x64": "^0.24.0",
15
  "@iconify-json/tabler": "^1.2.10",
16
  "@svgr/core": "^8.1.0",
17
  "@svgr/plugin-jsx": "^8.1.0",
 
11
  "preview": "npx vite preview"
12
  },
13
  "dependencies": {
14
+ "@esbuild/linux-x64": "^0.25.0",
15
  "@iconify-json/tabler": "^1.2.10",
16
  "@svgr/core": "^8.1.0",
17
  "@svgr/plugin-jsx": "^8.1.0",
lynxkite-app/web/src/index.css CHANGED
@@ -286,13 +286,19 @@ body {
286
  .entry-list .entry {
287
  display: flex;
288
  border-bottom: 1px solid whitesmoke;
289
- padding-left: 10px;
290
  color: #004165;
291
  cursor: pointer;
292
  user-select: none;
293
- text-decoration: none;
294
- justify-content: space-between;
295
- padding-right: 10px;
 
 
 
 
 
 
 
296
  }
297
 
298
  .entry-list .open .entry,
@@ -316,11 +322,6 @@ body {
316
  }
317
  }
318
 
319
- path.react-flow__edge-path {
320
- stroke-width: 2;
321
- stroke: black;
322
- }
323
-
324
  .react-flow__edge.selected path.react-flow__edge-path {
325
  outline: var(--xy-selection-border, var(--xy-selection-border-default));
326
  outline-offset: 10px;
@@ -354,13 +355,14 @@ path.react-flow__edge-path {
354
  margin-top: 10px;
355
  }
356
 
357
- .graph-tables, .graph-relations {
358
- flex: 1;
359
- padding-left: 10px;
360
- padding-right: 10px;
 
361
  }
362
 
363
- .graph-table-header{
364
  display: flex;
365
  justify-content: space-between;
366
  font-weight: bold;
@@ -400,7 +402,7 @@ path.react-flow__edge-path {
400
  font-weight: bold;
401
  display: block;
402
  margin-bottom: 2px;
403
- color: #666; /* Lighter text for labels */
404
  }
405
 
406
  .graph-relation-attributes input {
@@ -428,4 +430,4 @@ path.react-flow__edge-path {
428
 
429
  .add-relationship-button:hover {
430
  background-color: #218838;
431
- }
 
286
  .entry-list .entry {
287
  display: flex;
288
  border-bottom: 1px solid whitesmoke;
 
289
  color: #004165;
290
  cursor: pointer;
291
  user-select: none;
292
+
293
+ a {
294
+ text-decoration: none;
295
+ flex: 1;
296
+ padding-left: 10px;
297
+ }
298
+
299
+ button {
300
+ padding-right: 10px;
301
+ }
302
  }
303
 
304
  .entry-list .open .entry,
 
322
  }
323
  }
324
 
 
 
 
 
 
325
  .react-flow__edge.selected path.react-flow__edge-path {
326
  outline: var(--xy-selection-border, var(--xy-selection-border-default));
327
  outline-offset: 10px;
 
355
  margin-top: 10px;
356
  }
357
 
358
+ .graph-tables,
359
+ .graph-relations {
360
+ flex: 1;
361
+ padding-left: 10px;
362
+ padding-right: 10px;
363
  }
364
 
365
+ .graph-table-header {
366
  display: flex;
367
  justify-content: space-between;
368
  font-weight: bold;
 
402
  font-weight: bold;
403
  display: block;
404
  margin-bottom: 2px;
405
+ color: #666; /* Lighter text for labels */
406
  }
407
 
408
  .graph-relation-attributes input {
 
430
 
431
  .add-relationship-button:hover {
432
  background-color: #218838;
433
+ }
lynxkite-app/web/src/workspace/Workspace.tsx CHANGED
@@ -41,11 +41,11 @@ import NodeSearch, {
41
  type Catalog,
42
  type Catalogs,
43
  } from "./NodeSearch.tsx";
 
44
  import NodeWithImage from "./nodes/NodeWithImage.tsx";
45
  import NodeWithParams from "./nodes/NodeWithParams";
46
  import NodeWithTableView from "./nodes/NodeWithTableView.tsx";
47
  import NodeWithVisualization from "./nodes/NodeWithVisualization.tsx";
48
- import NodeWithGraphCreationView from "./nodes/GraphCreationNode.tsx";
49
 
50
  export default function (props: any) {
51
  return (
@@ -78,6 +78,9 @@ function LynxKiteFlow() {
78
  if (!state.workspace) return;
79
  if (!state.workspace.nodes) return;
80
  if (!state.workspace.edges) return;
 
 
 
81
  setNodes([...state.workspace.nodes] as Node[]);
82
  setEdges([...state.workspace.edges] as Edge[]);
83
  for (const node of state.workspace.nodes) {
@@ -284,7 +287,18 @@ function LynxKiteFlow() {
284
  proOptions={{ hideAttribution: true }}
285
  maxZoom={3}
286
  minZoom={0.3}
287
- defaultEdgeOptions={{ markerEnd: { type: MarkerType.Arrow } }}
 
 
 
 
 
 
 
 
 
 
 
288
  >
289
  <Controls />
290
  <MiniMap />
 
41
  type Catalog,
42
  type Catalogs,
43
  } from "./NodeSearch.tsx";
44
+ import NodeWithGraphCreationView from "./nodes/GraphCreationNode.tsx";
45
  import NodeWithImage from "./nodes/NodeWithImage.tsx";
46
  import NodeWithParams from "./nodes/NodeWithParams";
47
  import NodeWithTableView from "./nodes/NodeWithTableView.tsx";
48
  import NodeWithVisualization from "./nodes/NodeWithVisualization.tsx";
 
49
 
50
  export default function (props: any) {
51
  return (
 
78
  if (!state.workspace) return;
79
  if (!state.workspace.nodes) return;
80
  if (!state.workspace.edges) return;
81
+ for (const n of state.workspace.nodes) {
82
+ n.dragHandle = ".bg-primary";
83
+ }
84
  setNodes([...state.workspace.nodes] as Node[]);
85
  setEdges([...state.workspace.edges] as Edge[]);
86
  for (const node of state.workspace.nodes) {
 
287
  proOptions={{ hideAttribution: true }}
288
  maxZoom={3}
289
  minZoom={0.3}
290
+ defaultEdgeOptions={{
291
+ markerEnd: {
292
+ type: MarkerType.ArrowClosed,
293
+ color: "black",
294
+ width: 15,
295
+ height: 15,
296
+ },
297
+ style: {
298
+ strokeWidth: 2,
299
+ stroke: "black",
300
+ },
301
+ }}
302
  >
303
  <Controls />
304
  <MiniMap />
lynxkite-app/web/src/workspace/nodes/LynxKiteNode.tsx CHANGED
@@ -64,7 +64,7 @@ export default function LynxKiteNode(props: LynxKiteNodeProps) {
64
 
65
  return (
66
  <div
67
- className={`node-container·${expanded ? "expanded" : "collapsed"} `}
68
  style={{
69
  width: props.width || 200,
70
  height: expanded ? props.height || 200 : undefined,
 
64
 
65
  return (
66
  <div
67
+ className={`node-container ${expanded ? "expanded" : "collapsed"} `}
68
  style={{
69
  width: props.width || 200,
70
  height: expanded ? props.height || 200 : undefined,
lynxkite-app/web/src/workspace/nodes/NodeParameter.tsx CHANGED
@@ -31,7 +31,7 @@ export default function NodeParameter({
31
  <>
32
  <ParamName name={name} />
33
  <textarea
34
- className="textarea textarea-bordered w-full max-w-xs"
35
  rows={6}
36
  value={value}
37
  onChange={(evt) => onChange(evt.currentTarget.value, { delay: 2 })}
@@ -42,7 +42,7 @@ export default function NodeParameter({
42
  <>
43
  <ParamName name={name} />
44
  <select
45
- className="select select-bordered w-full max-w-xs"
46
  value={value || meta.type.enum[0]}
47
  onChange={(evt) => onChange(evt.currentTarget.value)}
48
  >
@@ -69,7 +69,7 @@ export default function NodeParameter({
69
  <>
70
  <ParamName name={name} />
71
  <input
72
- className="input input-bordered w-full max-w-xs"
73
  value={value || ""}
74
  onChange={(evt) => onChange(evt.currentTarget.value, { delay: 2 })}
75
  onBlur={(evt) => onChange(evt.currentTarget.value, { delay: 0 })}
 
31
  <>
32
  <ParamName name={name} />
33
  <textarea
34
+ className="textarea textarea-bordered w-full"
35
  rows={6}
36
  value={value}
37
  onChange={(evt) => onChange(evt.currentTarget.value, { delay: 2 })}
 
42
  <>
43
  <ParamName name={name} />
44
  <select
45
+ className="select select-bordered w-full"
46
  value={value || meta.type.enum[0]}
47
  onChange={(evt) => onChange(evt.currentTarget.value)}
48
  >
 
69
  <>
70
  <ParamName name={name} />
71
  <input
72
+ className="input input-bordered w-full"
73
  value={value || ""}
74
  onChange={(evt) => onChange(evt.currentTarget.value, { delay: 2 })}
75
  onBlur={(evt) => onChange(evt.currentTarget.value, { delay: 0 })}
lynxkite-app/web/src/workspace/nodes/NodeWithTableView.tsx CHANGED
@@ -19,45 +19,45 @@ export default function NodeWithTableView(props: any) {
19
  const display = props.data.display?.value;
20
  const single =
21
  display?.dataframes && Object.keys(display?.dataframes).length === 1;
 
 
22
  return (
23
  <LynxKiteNode {...props}>
24
  {display && [
25
- Object.entries(display.dataframes || {}).map(
26
- ([name, df]: [string, any]) => (
27
- <React.Fragment key={name}>
28
- {!single && (
29
- <div
30
- key={`${name}-header`}
31
- className="df-head"
32
- onClick={() => setOpen({ ...open, [name]: !open[name] })}
33
- >
34
- {name}
35
- </div>
36
- )}
37
- {(single || open[name]) &&
38
- (df.data.length > 1 ? (
39
- <Table
40
- key={`${name}-table`}
41
- columns={df.columns}
42
- data={df.data}
43
- />
44
- ) : df.data.length ? (
45
- <dl key={`${name}-dl`}>
46
- {df.columns.map((c: string, i: number) => (
47
- <React.Fragment key={`${name}-${c}`}>
48
- <dt>{c}</dt>
49
- <dd>
50
- <Markdown>{toMD(df.data[0][i])}</Markdown>
51
- </dd>
52
- </React.Fragment>
53
- ))}
54
- </dl>
55
- ) : (
56
- JSON.stringify(df.data)
57
- ))}
58
- </React.Fragment>
59
- ),
60
- ),
61
  Object.entries(display.others || {}).map(([name, o]) => (
62
  <>
63
  <div
 
19
  const display = props.data.display?.value;
20
  const single =
21
  display?.dataframes && Object.keys(display?.dataframes).length === 1;
22
+ const dfs = Object.entries(display?.dataframes || {});
23
+ dfs.sort();
24
  return (
25
  <LynxKiteNode {...props}>
26
  {display && [
27
+ dfs.map(([name, df]: [string, any]) => (
28
+ <React.Fragment key={name}>
29
+ {!single && (
30
+ <div
31
+ key={`${name}-header`}
32
+ className="df-head"
33
+ onClick={() => setOpen({ ...open, [name]: !open[name] })}
34
+ >
35
+ {name}
36
+ </div>
37
+ )}
38
+ {(single || open[name]) &&
39
+ (df.data.length > 1 ? (
40
+ <Table
41
+ key={`${name}-table`}
42
+ columns={df.columns}
43
+ data={df.data}
44
+ />
45
+ ) : df.data.length ? (
46
+ <dl key={`${name}-dl`}>
47
+ {df.columns.map((c: string, i: number) => (
48
+ <React.Fragment key={`${name}-${c}`}>
49
+ <dt>{c}</dt>
50
+ <dd>
51
+ <Markdown>{toMD(df.data[0][i])}</Markdown>
52
+ </dd>
53
+ </React.Fragment>
54
+ ))}
55
+ </dl>
56
+ ) : (
57
+ JSON.stringify(df.data)
58
+ ))}
59
+ </React.Fragment>
60
+ )),
 
 
61
  Object.entries(display.others || {}).map(([name, o]) => (
62
  <>
63
  <div
lynxkite-app/web/tests/examples.spec.ts CHANGED
@@ -7,6 +7,11 @@ test("LynxKite Graph Analytics example", async ({ page }) => {
7
  expect(await ws.isErrorFree(process.env.CI ? 2000 : 1000)).toBeTruthy();
8
  });
9
 
 
 
 
 
 
10
  test("Pytorch example", async ({ page }) => {
11
  const ws = await Workspace.open(page, "PyTorch demo");
12
  expect(await ws.isErrorFree()).toBeTruthy();
 
7
  expect(await ws.isErrorFree(process.env.CI ? 2000 : 1000)).toBeTruthy();
8
  });
9
 
10
+ test("Bio example", async ({ page }) => {
11
+ const ws = await Workspace.open(page, "Bio demo");
12
+ expect(await ws.isErrorFree()).toBeTruthy();
13
+ });
14
+
15
  test("Pytorch example", async ({ page }) => {
16
  const ws = await Workspace.open(page, "PyTorch demo");
17
  expect(await ws.isErrorFree()).toBeTruthy();
lynxkite-app/web/tests/lynxkite.ts CHANGED
@@ -113,7 +113,7 @@ export class Workspace {
113
  targetPosition?: { x: number; y: number },
114
  ) {
115
  // Move a box around, it is a best effort operation, the exact target position may not be reached
116
- const box = await this.getBox(boxId).boundingBox();
117
  if (!box) {
118
  return;
119
  }
 
113
  targetPosition?: { x: number; y: number },
114
  ) {
115
  // Move a box around, it is a best effort operation, the exact target position may not be reached
116
+ const box = await this.getBox(boxId).locator(".title").boundingBox();
117
  if (!box) {
118
  return;
119
  }
lynxkite-bio/README.md ADDED
@@ -0,0 +1,3 @@
 
 
 
 
1
+ # LynxKite Bio
2
+
3
+ An expansion for `lynxkite-graph-analytics` that provides algorithms for biological applications.
lynxkite-bio/pyproject.toml ADDED
@@ -0,0 +1,24 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ [project]
2
+ name = "lynxkite-bio"
3
+ version = "0.1.0"
4
+ description = "Additional boxes for LynxKite Graph Analytics that add algorithms for biology."
5
+ readme = "README.md"
6
+ requires-python = ">=3.11"
7
+ dependencies = [
8
+ "fsspec>=2025.2.0",
9
+ "joblib>=1.4.2",
10
+ "lynxkite-core",
11
+ "lynxkite-graph-analytics",
12
+ "pandas>=2.2.3",
13
+ "rdkit>=2024.9.5",
14
+ "scipy>=1.15.2",
15
+ ]
16
+
17
+ [project.optional-dependencies]
18
+ dev = [
19
+ "pytest>=8.3.4",
20
+ ]
21
+
22
+ [tool.uv.sources]
23
+ lynxkite-core = { path = "../lynxkite-core" }
24
+ lynxkite-graph-analytics = { path = "../lynxkite-graph-analytics" }
lynxkite-bio/src/lynxkite_bio/__init__.py ADDED
@@ -0,0 +1,68 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ """An expansion for `lynxkite-graph-analytics` that provides algorithms for biological applications."""
2
+
3
+ from lynxkite_graph_analytics import Bundle, RelationDefinition
4
+ from lynxkite.core import ops
5
+ import joblib
6
+ import numpy as np
7
+ import pandas as pd
8
+ import rdkit.Chem
9
+ import rdkit.Chem.rdFingerprintGenerator
10
+ import rdkit.Chem.Fingerprints.ClusterMols
11
+ import scipy
12
+
13
+ mem = joblib.Memory("../joblib-cache")
14
+ ENV = "LynxKite Graph Analytics"
15
+ op = ops.op_registration(ENV)
16
+
17
+
18
+ @op("Parse SMILES")
19
+ def parse_smiles(bundle: Bundle, *, table="df", smiles_column="SMILES", save_as="mols"):
20
+ """Parse SMILES strings into RDKit molecules."""
21
+ df = bundle.dfs[table]
22
+ mols = [rdkit.Chem.MolFromSmiles(smiles) for smiles in df[smiles_column].dropna()]
23
+ mols = [mol for mol in mols if mol is not None]
24
+ bundle = bundle.copy()
25
+ bundle.dfs[table] = df.assign(**{save_as: mols})
26
+ return bundle
27
+
28
+
29
+ def _get_similarity_matrix(mols):
30
+ mfpgen = rdkit.Chem.rdFingerprintGenerator.GetMorganGenerator(radius=2, fpSize=2048)
31
+ fps = [(0, mfpgen.GetFingerprint(mol)) for mol in mols]
32
+ similarity_matrix = rdkit.Chem.Fingerprints.ClusterMols.GetDistanceMatrix(
33
+ fps, metric=rdkit.Chem.DataStructs.TanimotoSimilarity, isSimilarity=1
34
+ )
35
+ return scipy.spatial.distance.squareform(similarity_matrix)
36
+
37
+
38
+ @op("Graph from molecule similarity")
39
+ def graph_from_similarity(
40
+ bundle: Bundle, *, table="df", mols_column="mols", average_degree=10
41
+ ):
42
+ """Creates edges for pairs of molecules that are the most similar."""
43
+ df = bundle.dfs[table]
44
+ mols = df[mols_column]
45
+ similarity_matrix = _get_similarity_matrix(mols)
46
+ i_idx, j_idx = np.triu_indices_from(similarity_matrix, k=1)
47
+ sim_values = similarity_matrix[i_idx, j_idx]
48
+ N = int(average_degree * len(mols))
49
+ top_n_idx = np.argsort(sim_values)[-N:]
50
+ top_n_pairs = [(i_idx[k], j_idx[k], sim_values[k]) for k in top_n_idx]
51
+ edges = pd.DataFrame(top_n_pairs, columns=["source", "target", "similarity"])
52
+ nodes = df.copy()
53
+ nodes.index.name = "id"
54
+ bundle = Bundle(
55
+ dfs={"edges": edges, "nodes": nodes},
56
+ relations=[
57
+ RelationDefinition(
58
+ df="edges",
59
+ source_column="source",
60
+ target_column="target",
61
+ source_table="nodes",
62
+ target_table="nodes",
63
+ source_key="id",
64
+ target_key="id",
65
+ )
66
+ ],
67
+ )
68
+ return bundle
lynxkite-core/pyproject.toml CHANGED
@@ -10,4 +10,4 @@ dependencies = [
10
  [project.optional-dependencies]
11
  dev = [
12
  "pytest",
13
- ]
 
10
  [project.optional-dependencies]
11
  dev = [
12
  "pytest",
13
+ ]
lynxkite-core/src/lynxkite/core/workspace.py CHANGED
@@ -1,5 +1,6 @@
1
  """For working with LynxKite workspaces."""
2
 
 
3
  from typing import Optional
4
  import dataclasses
5
  import os
@@ -65,7 +66,8 @@ async def execute(ws: Workspace):
65
 
66
  def save(ws: Workspace, path: str):
67
  """Persist a workspace to a local file in JSON format."""
68
- j = ws.model_dump_json(indent=2) + "\n"
 
69
  dirname, basename = os.path.split(path)
70
  os.makedirs(dirname, exist_ok=True)
71
  # Create temp file in the same directory to make sure it's on the same filesystem.
 
1
  """For working with LynxKite workspaces."""
2
 
3
+ import json
4
  from typing import Optional
5
  import dataclasses
6
  import os
 
66
 
67
  def save(ws: Workspace, path: str):
68
  """Persist a workspace to a local file in JSON format."""
69
+ j = ws.model_dump()
70
+ j = json.dumps(j, indent=2, sort_keys=True) + "\n"
71
  dirname, basename = os.path.split(path)
72
  os.makedirs(dirname, exist_ok=True)
73
  # Create temp file in the same directory to make sure it's on the same filesystem.
lynxkite-graph-analytics/src/lynxkite_graph_analytics/__init__.py CHANGED
@@ -1,3 +1,3 @@
1
- from . import lynxkite_ops # noqa (imported to trigger registration)
2
  from . import networkx_ops # noqa (imported to trigger registration)
3
  from . import pytorch_model_ops # noqa (imported to trigger registration)
 
1
+ from .lynxkite_ops import * # noqa (imported to trigger registration)
2
  from . import networkx_ops # noqa (imported to trigger registration)
3
  from . import pytorch_model_ops # noqa (imported to trigger registration)
lynxkite-graph-analytics/src/lynxkite_graph_analytics/lynxkite_ops.py CHANGED
@@ -83,12 +83,26 @@ class Bundle:
83
  # TODO: Use relations.
84
  graph = nx.DiGraph()
85
  if "nodes" in self.dfs:
86
- graph.add_nodes_from(
87
- self.dfs["nodes"].set_index("id").to_dict("index").items()
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
88
  )
89
- graph.add_edges_from(
90
- self.dfs["edges"][["source", "target"]].itertuples(index=False, name=None)
91
- )
92
  return graph
93
 
94
  def copy(self):
@@ -104,7 +118,7 @@ class Bundle:
104
  "dataframes": {
105
  name: {
106
  "columns": [str(c) for c in df.columns],
107
- "data": collect(df)[:limit],
108
  }
109
  for name, df in self.dfs.items()
110
  },
@@ -336,8 +350,14 @@ def _map_color(value):
336
 
337
 
338
  @op("Visualize graph", view="visualization")
339
- def visualize_graph(graph: Bundle, *, color_nodes_by: ops.NodeAttribute = None):
340
- nodes = graph.dfs["nodes"].copy()
 
 
 
 
 
 
341
  if color_nodes_by:
342
  nodes["color"] = _map_color(nodes[color_nodes_by])
343
  for cols in ["x y", "long lat"]:
@@ -367,15 +387,21 @@ def visualize_graph(graph: Bundle, *, color_nodes_by: ops.NodeAttribute = None):
367
  )
368
  curveness = 0.3
369
  nodes = nodes.to_records()
370
- edges = graph.dfs["edges"].drop_duplicates(["source", "target"])
 
 
 
 
371
  edges = edges.to_records()
372
  v = {
373
  "animationDuration": 500,
374
  "animationEasingUpdate": "quinticInOut",
 
375
  "series": [
376
  {
377
  "type": "graph",
378
- "roam": True,
 
379
  "lineStyle": {
380
  "color": "gray",
381
  "curveness": curveness,
@@ -386,6 +412,7 @@ def visualize_graph(graph: Bundle, *, color_nodes_by: ops.NodeAttribute = None):
386
  "width": 10,
387
  },
388
  },
 
389
  "data": [
390
  {
391
  "id": str(n.id),
@@ -394,11 +421,24 @@ def visualize_graph(graph: Bundle, *, color_nodes_by: ops.NodeAttribute = None):
394
  # Adjust node size to cover the same area no matter how many nodes there are.
395
  "symbolSize": 50 / len(nodes) ** 0.5,
396
  "itemStyle": {"color": n.color} if color_nodes_by else {},
 
 
 
 
 
397
  }
398
  for n in nodes
399
  ],
400
  "links": [
401
- {"source": str(r.source), "target": str(r.target)} for r in edges
 
 
 
 
 
 
 
 
402
  ],
403
  },
404
  ],
@@ -406,12 +446,18 @@ def visualize_graph(graph: Bundle, *, color_nodes_by: ops.NodeAttribute = None):
406
  return v
407
 
408
 
409
- def collect(df: pd.DataFrame):
 
 
410
  if isinstance(df, pl.LazyFrame):
411
  df = df.collect()
412
  if isinstance(df, pl.DataFrame):
413
- return [[d[c] for c in df.columns] for d in df.to_dicts()]
414
- return df.values.tolist()
 
 
 
 
415
 
416
 
417
  @op("View tables", view="table_view")
 
83
  # TODO: Use relations.
84
  graph = nx.DiGraph()
85
  if "nodes" in self.dfs:
86
+ df = self.dfs["nodes"]
87
+ if df.index.name != "id":
88
+ df = df.set_index("id")
89
+ graph.add_nodes_from(df.to_dict("index").items())
90
+ if "edges" in self.dfs:
91
+ edges = self.dfs["edges"]
92
+ graph.add_edges_from(
93
+ [
94
+ (
95
+ e["source"],
96
+ e["target"],
97
+ {
98
+ k: e[k]
99
+ for k in edges.columns
100
+ if k not in ["source", "target"]
101
+ },
102
+ )
103
+ for e in edges.to_records()
104
+ ]
105
  )
 
 
 
106
  return graph
107
 
108
  def copy(self):
 
118
  "dataframes": {
119
  name: {
120
  "columns": [str(c) for c in df.columns],
121
+ "data": df_for_frontend(df, limit).values.tolist(),
122
  }
123
  for name, df in self.dfs.items()
124
  },
 
350
 
351
 
352
  @op("Visualize graph", view="visualization")
353
+ def visualize_graph(
354
+ graph: Bundle,
355
+ *,
356
+ color_nodes_by: ops.NodeAttribute = None,
357
+ label_by: ops.NodeAttribute = None,
358
+ color_edges_by: ops.EdgeAttribute = None,
359
+ ):
360
+ nodes = df_for_frontend(graph.dfs["nodes"], 10_000)
361
  if color_nodes_by:
362
  nodes["color"] = _map_color(nodes[color_nodes_by])
363
  for cols in ["x y", "long lat"]:
 
387
  )
388
  curveness = 0.3
389
  nodes = nodes.to_records()
390
+ edges = df_for_frontend(
391
+ graph.dfs["edges"].drop_duplicates(["source", "target"]), 10_000
392
+ )
393
+ if color_edges_by:
394
+ edges["color"] = _map_color(edges[color_edges_by])
395
  edges = edges.to_records()
396
  v = {
397
  "animationDuration": 500,
398
  "animationEasingUpdate": "quinticInOut",
399
+ "tooltip": {"show": True},
400
  "series": [
401
  {
402
  "type": "graph",
403
+ # Mouse zoom/panning is disabled for now. It interacts badly with ReactFlow.
404
+ # "roam": True,
405
  "lineStyle": {
406
  "color": "gray",
407
  "curveness": curveness,
 
412
  "width": 10,
413
  },
414
  },
415
+ "label": {"position": "top", "formatter": "{b}"},
416
  "data": [
417
  {
418
  "id": str(n.id),
 
421
  # Adjust node size to cover the same area no matter how many nodes there are.
422
  "symbolSize": 50 / len(nodes) ** 0.5,
423
  "itemStyle": {"color": n.color} if color_nodes_by else {},
424
+ "label": {"show": label_by is not None},
425
+ "name": str(getattr(n, label_by, "")) if label_by else None,
426
+ "value": str(getattr(n, color_nodes_by, ""))
427
+ if color_nodes_by
428
+ else None,
429
  }
430
  for n in nodes
431
  ],
432
  "links": [
433
+ {
434
+ "source": str(r.source),
435
+ "target": str(r.target),
436
+ "lineStyle": {"color": r.color} if color_edges_by else {},
437
+ "value": str(getattr(r, color_edges_by, ""))
438
+ if color_edges_by
439
+ else None,
440
+ }
441
+ for r in edges
442
  ],
443
  },
444
  ],
 
446
  return v
447
 
448
 
449
+ def df_for_frontend(df: pd.DataFrame, limit: int) -> pd.DataFrame:
450
+ """Returns a DataFrame with values that are safe to send to the frontend."""
451
+ df = df[:limit]
452
  if isinstance(df, pl.LazyFrame):
453
  df = df.collect()
454
  if isinstance(df, pl.DataFrame):
455
+ df = df.to_pandas()
456
+ # Convert non-numeric columns to strings.
457
+ for c in df.columns:
458
+ if not pd.api.types.is_numeric_dtype(df[c]):
459
+ df[c] = df[c].astype(str)
460
+ return df
461
 
462
 
463
  @op("View tables", view="table_view")
lynxkite-graph-analytics/src/lynxkite_graph_analytics/pytorch_model_ops.py CHANGED
@@ -72,4 +72,4 @@ ops.register_passive_op(
72
  inputs=[ops.Input(name="input", position="top", type="tensor")],
73
  outputs=[ops.Output(name="output", position="bottom", type="tensor")],
74
  params=[ops.Parameter.basic("times", 1, int)],
75
- )
 
72
  inputs=[ops.Input(name="input", position="top", type="tensor")],
73
  outputs=[ops.Output(name="output", position="bottom", type="tensor")],
74
  params=[ops.Parameter.basic("times", 1, int)],
75
+ )