Spaces:
Running
Running
Merge pull request #73 from biggraph/darabos-rdkit
Browse files- .github/workflows/test.yaml +1 -1
- README.md +2 -1
- examples/AIMO +1 -1
- examples/Bio demo +857 -0
- examples/Graph RAG +1 -1
- examples/Image processing +1 -1
- examples/LynxScribe demo +1 -1
- examples/NetworkX demo +0 -0
- examples/PyTorch demo +1 -1
- examples/RAG chatbot app +1 -1
- examples/drug_target_data_sample.csv +6 -0
- lynxkite-app/src/lynxkite_app/crdt.py +0 -1
- lynxkite-app/web/package-lock.json +21 -4
- lynxkite-app/web/package.json +1 -1
- lynxkite-app/web/src/index.css +18 -16
- lynxkite-app/web/src/workspace/Workspace.tsx +16 -2
- lynxkite-app/web/src/workspace/nodes/LynxKiteNode.tsx +1 -1
- lynxkite-app/web/src/workspace/nodes/NodeParameter.tsx +3 -3
- lynxkite-app/web/src/workspace/nodes/NodeWithTableView.tsx +36 -36
- lynxkite-app/web/tests/examples.spec.ts +5 -0
- lynxkite-app/web/tests/lynxkite.ts +1 -1
- lynxkite-bio/README.md +3 -0
- lynxkite-bio/pyproject.toml +24 -0
- lynxkite-bio/src/lynxkite_bio/__init__.py +68 -0
- lynxkite-core/pyproject.toml +1 -1
- lynxkite-core/src/lynxkite/core/workspace.py +3 -1
- lynxkite-graph-analytics/src/lynxkite_graph_analytics/__init__.py +1 -1
- lynxkite-graph-analytics/src/lynxkite_graph_analytics/lynxkite_ops.py +60 -14
- lynxkite-graph-analytics/src/lynxkite_graph_analytics/pytorch_model_ops.py +1 -1
.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.
|
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.
|
636 |
-
"resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.
|
637 |
-
"integrity": "sha512-
|
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.
|
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 |
-
|
294 |
-
|
295 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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,
|
358 |
-
|
359 |
-
|
360 |
-
|
|
|
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;
|
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={{
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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
|
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
|
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
|
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
|
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 |
-
|
26 |
-
|
27 |
-
|
28 |
-
|
29 |
-
|
30 |
-
|
31 |
-
|
32 |
-
|
33 |
-
|
34 |
-
|
35 |
-
|
36 |
-
|
37 |
-
|
38 |
-
|
39 |
-
|
40 |
-
|
41 |
-
|
42 |
-
|
43 |
-
|
44 |
-
|
45 |
-
|
46 |
-
{
|
47 |
-
<
|
48 |
-
|
49 |
-
<
|
50 |
-
|
51 |
-
|
52 |
-
|
53 |
-
|
54 |
-
|
55 |
-
)
|
56 |
-
|
57 |
-
|
58 |
-
|
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.
|
|
|
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
|
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 |
-
|
87 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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":
|
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(
|
340 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
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 =
|
|
|
|
|
|
|
|
|
371 |
edges = edges.to_records()
|
372 |
v = {
|
373 |
"animationDuration": 500,
|
374 |
"animationEasingUpdate": "quinticInOut",
|
|
|
375 |
"series": [
|
376 |
{
|
377 |
"type": "graph",
|
378 |
-
|
|
|
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 |
-
{
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
410 |
if isinstance(df, pl.LazyFrame):
|
411 |
df = df.collect()
|
412 |
if isinstance(df, pl.DataFrame):
|
413 |
-
|
414 |
-
|
|
|
|
|
|
|
|
|
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 |
+
)
|