darabos commited on
Commit
7535b07
·
unverified ·
2 Parent(s): 145bf26 1a42cda

Merge pull request #75 from biggraph/darabos-ga

Browse files
.github/workflows/test.yaml CHANGED
@@ -24,7 +24,20 @@ jobs:
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
 
@@ -43,6 +56,13 @@ jobs:
43
  cd lynxkite-graph-analytics
44
  pytest
45
 
 
 
 
 
 
 
 
46
  - uses: actions/setup-node@v4
47
  with:
48
  node-version: lts/*
 
24
  run: |
25
  eval `ssh-agent -s`
26
  ssh-add - <<< '${{ secrets.LYNXSCRIBE_DEPLOY_KEY }}'
27
+ uv pip install \
28
+ -e lynxkite-core/[dev] \
29
+ -e lynxkite-app/[dev] \
30
+ -e lynxkite-graph-analytics/[dev] \
31
+ -e lynxkite-bio \
32
+ -e lynxkite-lynxscribe/ \
33
+ -e lynxkite-pillow-example/
34
+ env:
35
+ UV_SYSTEM_PYTHON: 1
36
+
37
+ - name: Run pre-commits
38
+ run: |
39
+ uv pip install pre-commit
40
+ pre-commit run --all-files
41
  env:
42
  UV_SYSTEM_PYTHON: 1
43
 
 
56
  cd lynxkite-graph-analytics
57
  pytest
58
 
59
+ - name: Try building the documentation
60
+ run: |
61
+ uv pip install mkdocs-material mkdocstrings[python]
62
+ mkdocs build
63
+ env:
64
+ UV_SYSTEM_PYTHON: 1
65
+
66
  - uses: actions/setup-node@v4
67
  with:
68
  node-version: lts/*
.gitignore CHANGED
@@ -16,3 +16,9 @@ joblib-cache
16
  *.egg-info
17
 
18
  lynxkite_crdt_data
 
 
 
 
 
 
 
16
  *.egg-info
17
 
18
  lynxkite_crdt_data
19
+
20
+ # Playwright
21
+ /test-results/
22
+ /playwright-report/
23
+ /blob-report/
24
+ /playwright/.cache/
docs/lynxkite-graph-analytics.md CHANGED
@@ -3,4 +3,4 @@
3
  This is the classical LynxKite experience!
4
  The graph analytics plugin is a collection of graph algorithms that can be run on a LynxKite graph.
5
 
6
- ::: lynxkite_plugins.graph_analytics.lynxkite_ops
 
3
  This is the classical LynxKite experience!
4
  The graph analytics plugin is a collection of graph algorithms that can be run on a LynxKite graph.
5
 
6
+ ::: lynxkite_graph_analytics.lynxkite_ops
lynxkite-app/web/src/workspace/Workspace.tsx CHANGED
@@ -180,6 +180,40 @@ function LynxKiteFlow() {
180
  }),
181
  [],
182
  );
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
183
  const closeNodeSearch = useCallback(() => {
184
  setNodeSearchSettings(undefined);
185
  setSuppressSearchUntil(Date.now() + 200);
 
180
  }),
181
  [],
182
  );
183
+
184
+ // Global keyboard shortcuts.
185
+ useEffect(() => {
186
+ const handleKeyDown = (event: KeyboardEvent) => {
187
+ // Show the node search dialog on "/".
188
+ if (
189
+ event.key === "/" &&
190
+ !nodeSearchSettings &&
191
+ !isTypingInFormElement()
192
+ ) {
193
+ event.preventDefault();
194
+ setNodeSearchSettings({
195
+ pos: { x: 100, y: 100 },
196
+ boxes: catalog.data![state.workspace.env!],
197
+ });
198
+ }
199
+ };
200
+ // TODO: Switch to keydown once https://github.com/xyflow/xyflow/pull/5055 is merged.
201
+ document.addEventListener("keyup", handleKeyDown);
202
+ return () => {
203
+ document.removeEventListener("keyup", handleKeyDown);
204
+ };
205
+ }, [catalog.data, nodeSearchSettings, state.workspace.env]);
206
+
207
+ function isTypingInFormElement() {
208
+ const activeElement = document.activeElement;
209
+ return (
210
+ activeElement &&
211
+ (activeElement.tagName === "INPUT" ||
212
+ activeElement.tagName === "TEXTAREA" ||
213
+ (activeElement as HTMLElement).isContentEditable)
214
+ );
215
+ }
216
+
217
  const closeNodeSearch = useCallback(() => {
218
  setNodeSearchSettings(undefined);
219
  setSuppressSearchUntil(Date.now() + 200);
lynxkite-app/web/tests/basic.spec.ts CHANGED
@@ -21,6 +21,9 @@ test("Box creation & deletion per env", async () => {
21
  const envs = await workspace.getEnvs();
22
  for (const env of envs) {
23
  await workspace.setEnv(env);
 
 
 
24
  const catalog = await workspace.getCatalog();
25
  expect(catalog).not.toHaveLength(0);
26
  const op = catalog[0];
 
21
  const envs = await workspace.getEnvs();
22
  for (const env of envs) {
23
  await workspace.setEnv(env);
24
+ // TODO: Opening the catalog immediately after setting the env can fail.
25
+ // Let's fix this!
26
+ await new Promise((resolve) => setTimeout(resolve, 500));
27
  const catalog = await workspace.getCatalog();
28
  expect(catalog).not.toHaveLength(0);
29
  const op = catalog[0];
lynxkite-app/web/tests/examples.spec.ts CHANGED
@@ -4,49 +4,49 @@ import { Workspace } from "./lynxkite";
4
 
5
  test("LynxKite Graph Analytics example", async ({ page }) => {
6
  const ws = await Workspace.open(page, "NetworkX demo");
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();
18
  });
19
 
20
  test.fail("AIMO example", async ({ page }) => {
21
  // Fails because of missing OPENAI_API_KEY
22
  const ws = await Workspace.open(page, "AIMO");
23
- expect(await ws.isErrorFree()).toBeTruthy();
24
  });
25
 
26
  test.fail("LynxScribe example", async ({ page }) => {
27
  // Fails because of missing OPENAI_API_KEY
28
  const ws = await Workspace.open(page, "LynxScribe demo");
29
- expect(await ws.isErrorFree()).toBeTruthy();
30
  });
31
 
32
  test.fail("Graph RAG", async ({ page }) => {
33
  // Fails due to some issue with ChromaDB
34
  const ws = await Workspace.open(page, "Graph RAG");
35
- expect(await ws.isErrorFree(process.env.CI ? 2000 : 500)).toBeTruthy();
36
  });
37
 
38
  test.fail("RAG chatbot app", async ({ page }) => {
39
  // Fail due to all operation being unknown
40
  const ws = await Workspace.open(page, "RAG chatbot app");
41
- expect(await ws.isErrorFree()).toBeTruthy();
42
  });
43
 
44
  test("Airlines demo", async ({ page }) => {
45
  const ws = await Workspace.open(page, "Airlines demo");
46
- expect(await ws.isErrorFree(process.env.CI ? 10000 : 500)).toBeTruthy();
47
  });
48
 
49
  test("Pillow example", async ({ page }) => {
50
  const ws = await Workspace.open(page, "Image processing");
51
- expect(await ws.isErrorFree()).toBeTruthy();
52
  });
 
4
 
5
  test("LynxKite Graph Analytics example", async ({ page }) => {
6
  const ws = await Workspace.open(page, "NetworkX demo");
7
+ await ws.expectErrorFree(process.env.CI ? 2000 : 1000);
8
  });
9
 
10
  test("Bio example", async ({ page }) => {
11
  const ws = await Workspace.open(page, "Bio demo");
12
+ await ws.expectErrorFree();
13
  });
14
 
15
  test("Pytorch example", async ({ page }) => {
16
  const ws = await Workspace.open(page, "PyTorch demo");
17
+ await ws.expectErrorFree();
18
  });
19
 
20
  test.fail("AIMO example", async ({ page }) => {
21
  // Fails because of missing OPENAI_API_KEY
22
  const ws = await Workspace.open(page, "AIMO");
23
+ await ws.expectErrorFree();
24
  });
25
 
26
  test.fail("LynxScribe example", async ({ page }) => {
27
  // Fails because of missing OPENAI_API_KEY
28
  const ws = await Workspace.open(page, "LynxScribe demo");
29
+ await ws.expectErrorFree();
30
  });
31
 
32
  test.fail("Graph RAG", async ({ page }) => {
33
  // Fails due to some issue with ChromaDB
34
  const ws = await Workspace.open(page, "Graph RAG");
35
+ await ws.expectErrorFree(process.env.CI ? 2000 : 500);
36
  });
37
 
38
  test.fail("RAG chatbot app", async ({ page }) => {
39
  // Fail due to all operation being unknown
40
  const ws = await Workspace.open(page, "RAG chatbot app");
41
+ await ws.expectErrorFree();
42
  });
43
 
44
  test("Airlines demo", async ({ page }) => {
45
  const ws = await Workspace.open(page, "Airlines demo");
46
+ await ws.expectErrorFree(process.env.CI ? 10000 : 500);
47
  });
48
 
49
  test("Pillow example", async ({ page }) => {
50
  const ws = await Workspace.open(page, "Image processing");
51
+ await ws.expectErrorFree();
52
  });
lynxkite-app/web/tests/lynxkite.ts CHANGED
@@ -54,7 +54,7 @@ export class Workspace {
54
 
55
  async addBox(boxName) {
56
  //TODO: Support passing box parameters (id, position, etc.)
57
- const allBoxes = await this.getBoxes();
58
  if (allBoxes) {
59
  // Avoid overlapping with existing nodes
60
  const numNodes = allBoxes.length || 1;
@@ -63,13 +63,10 @@ export class Workspace {
63
  }
64
 
65
  // Some x,y offset, otherwise the box handle may fall outside the viewport.
66
- await this.page
67
- .locator(".react-flow__pane")
68
- .click({ position: { x: 20, y: 20 } });
69
  await this.page.locator(".node-search").getByText(boxName).click();
70
- await this.page.keyboard.press("Escape");
71
- // Workaround to wait for the deselection animation after choosing a box. Otherwise, the next box will not be added.
72
- await new Promise((resolve) => setTimeout(resolve, 200));
73
  }
74
 
75
  async getCatalog() {
@@ -79,14 +76,22 @@ export class Workspace {
79
  .allInnerTexts();
80
  // Dismiss the catalog menu
81
  await this.page.keyboard.press("Escape");
82
- await new Promise((resolve) => setTimeout(resolve, 200));
83
  return catalog;
84
  }
85
 
 
 
 
 
 
 
 
86
  async deleteBoxes(boxIds: string[]) {
87
  for (const boxId of boxIds) {
88
- await this.getBoxHandle(boxId).first().click();
89
  await this.page.keyboard.press("Backspace");
 
90
  }
91
  }
92
 
@@ -95,7 +100,7 @@ export class Workspace {
95
  }
96
 
97
  getBoxes() {
98
- return this.page.locator(".react-flow__node").all();
99
  }
100
 
101
  getBoxHandle(boxId: string, pos?: string) {
@@ -144,19 +149,13 @@ export class Workspace {
144
  await this.page.mouse.up();
145
  }
146
 
147
- async isErrorFree(executionWaitTime?): Promise<boolean> {
148
  // TODO: Workaround, to account for workspace execution. Once
149
  // we have a load indicator we can use that instead.
150
  await new Promise((resolve) =>
151
  setTimeout(resolve, executionWaitTime ? executionWaitTime : 500),
152
  );
153
- const boxes = await this.getBoxes();
154
- for (const box of boxes) {
155
- if (await box.locator(".error").isVisible()) {
156
- return false;
157
- }
158
- }
159
- return true;
160
  }
161
 
162
  async close() {
 
54
 
55
  async addBox(boxName) {
56
  //TODO: Support passing box parameters (id, position, etc.)
57
+ const allBoxes = await this.getBoxes().all();
58
  if (allBoxes) {
59
  // Avoid overlapping with existing nodes
60
  const numNodes = allBoxes.length || 1;
 
63
  }
64
 
65
  // Some x,y offset, otherwise the box handle may fall outside the viewport.
66
+ await this.page.locator(".ws-name").click();
67
+ await this.page.keyboard.press("/");
 
68
  await this.page.locator(".node-search").getByText(boxName).click();
69
+ await expect(this.getBoxes()).toHaveCount(allBoxes.length + 1);
 
 
70
  }
71
 
72
  async getCatalog() {
 
76
  .allInnerTexts();
77
  // Dismiss the catalog menu
78
  await this.page.keyboard.press("Escape");
79
+ await expect(this.page.locator(".node-search")).not.toBeVisible();
80
  return catalog;
81
  }
82
 
83
+ async selectBox(boxId: string) {
84
+ const box = this.getBox(boxId);
85
+ // Click on the resizer, so we don't click on any parameters by accident.
86
+ await box.locator(".react-flow__resize-control").click();
87
+ await expect(box).toHaveClass(/selected/);
88
+ }
89
+
90
  async deleteBoxes(boxIds: string[]) {
91
  for (const boxId of boxIds) {
92
+ await this.selectBox(boxId);
93
  await this.page.keyboard.press("Backspace");
94
+ await expect(this.getBox(boxId)).not.toBeVisible();
95
  }
96
  }
97
 
 
100
  }
101
 
102
  getBoxes() {
103
+ return this.page.locator(".react-flow__node");
104
  }
105
 
106
  getBoxHandle(boxId: string, pos?: string) {
 
149
  await this.page.mouse.up();
150
  }
151
 
152
+ async expectErrorFree(executionWaitTime?) {
153
  // TODO: Workaround, to account for workspace execution. Once
154
  // we have a load indicator we can use that instead.
155
  await new Promise((resolve) =>
156
  setTimeout(resolve, executionWaitTime ? executionWaitTime : 500),
157
  );
158
+ await expect(this.getBoxes().locator(".error")).not.toBeVisible();
 
 
 
 
 
 
159
  }
160
 
161
  async close() {