@marimo-team/frontend 0.23.10 → 0.23.11-dev10
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/assets/{CellStatus-e0ex7Iei.js → CellStatus-CLGvVxcw.js} +1 -1
- package/dist/assets/{JsonOutput-DRNPZOvX.js → JsonOutput-uEJijGXp.js} +5 -5
- package/dist/assets/{MarimoErrorOutput-BH6hs0Ir.js → MarimoErrorOutput-DzoKyXWR.js} +2 -2
- package/dist/assets/{RenderHTML-BQ1PO4Wd.js → RenderHTML-DAt48X-F.js} +1 -1
- package/dist/assets/{RunButton-F8pLIvFp.js → RunButton-B7msyyYi.js} +1 -1
- package/dist/assets/{add-cell-with-ai-Bd_3tPEt.js → add-cell-with-ai-CAkcousR.js} +20 -20
- package/dist/assets/{add-connection-dialog-AhwxOztN.js → add-connection-dialog-ZyXXfAVG.js} +32 -32
- package/dist/assets/{agent-panel-Dm0KI1sF.js → agent-panel-BfIguQXh.js} +6 -6
- package/dist/assets/{ai-model-dropdown-CG4B4rqH.js → ai-model-dropdown-rE64ytSX.js} +3 -3
- package/dist/assets/{app-config-button-D_sFrSql.js → app-config-button-DLRZ5d9c.js} +1 -1
- package/dist/assets/{cell-editor-B3aYYyAI.js → cell-editor-2JIpvFn4.js} +15 -12
- package/dist/assets/{cell-link-Bj4-yOIx.js → cell-link-DY95GSb7.js} +1 -1
- package/dist/assets/{cells-DOA0Gew8.js → cells-D-v2lBet.js} +67 -67
- package/dist/assets/{chat-display-DI0jRLIv.js → chat-display-DO-tqsbY.js} +1 -1
- package/dist/assets/{chat-panel-BU4HHdf5.js → chat-panel-BT5gLfs6.js} +2 -2
- package/dist/assets/{chat-ui-C7igY2w5.js → chat-ui-Be4yDBZS.js} +4 -4
- package/dist/assets/column-preview-BzvFdLI5.js +1 -0
- package/dist/assets/{command-palette-Bjv1Z7v8.js → command-palette-CKKzfpGt.js} +1 -1
- package/dist/assets/{common-fDFYY_sv.js → common-Do4N1uVK.js} +1 -1
- package/dist/assets/{components--C6N-DXq.js → components-DUC0f1XD.js} +1 -1
- package/dist/assets/{datasource-CR6RRpTi.js → datasource-BQGI7c_u.js} +2 -2
- package/dist/assets/{dependency-graph-panel-BEgkvdX0.js → dependency-graph-panel-Czoya7c2.js} +1 -1
- package/dist/assets/{documentation-panel-HvbKykbI.js → documentation-panel-LokVYDIZ.js} +1 -1
- package/dist/assets/{download-DhxnAw14.js → download-WGU5w_3m.js} +3 -3
- package/dist/assets/{edit-page-BeWwLeT8.js → edit-page-BPQO1Uuz.js} +6 -6
- package/dist/assets/{error-panel-WLBQ3q9p.js → error-panel-BCm_e4eM.js} +1 -1
- package/dist/assets/{file-explorer-panel-DtzGlnzT.js → file-explorer-panel-D1NN-z8x.js} +3 -3
- package/dist/assets/{file-icons-D2f3nfbq.js → file-icons-CF-YT4hq.js} +1 -1
- package/dist/assets/{file-name-input-BNYf9WWM.js → file-name-input-BRVXQ3OU.js} +1 -1
- package/dist/assets/{floating-outline-BvHFWRoz.js → floating-outline-D33Zu-Ad.js} +1 -1
- package/dist/assets/{focus-Ldqh99xE.js → focus-BfFAxl9X.js} +1 -1
- package/dist/assets/{form-CxBAInfg.js → form-CVBFx2z3.js} +1 -1
- package/dist/assets/{home-page-DwFpLpXK.js → home-page-CL9Hv1Hg.js} +2 -2
- package/dist/assets/{hooks-DyMacA-R.js → hooks-83y0XCiC.js} +1 -1
- package/dist/assets/{html-to-image-CyAtzePO.js → html-to-image-IPNqWX7V.js} +2 -2
- package/dist/assets/index-BNWrEIlp.css +2 -0
- package/dist/assets/{index-fNBoXCyz.js → index-CuhY66ZR.js} +14 -14
- package/dist/assets/{kiosk-mode-CTWHjzXs.js → kiosk-mode-BSufcIZH.js} +1 -1
- package/dist/assets/{layout-dRNPwA-7.js → layout-ks0WJ2vR.js} +5 -5
- package/dist/assets/{logs-panel-BVLYycQb.js → logs-panel-DXN6sUaA.js} +1 -1
- package/dist/assets/{markdown-renderer-C9Ujvj0b.js → markdown-renderer-DWRPo52L.js} +1 -1
- package/dist/assets/{name-cell-input-DabvuruX.js → name-cell-input-BiDIx6YQ.js} +1 -1
- package/dist/assets/{outline-panel-B-ncbNWI.js → outline-panel-DVSGNRgP.js} +1 -1
- package/dist/assets/{packages-panel-CESeW_3T.js → packages-panel-DTX6CBnm.js} +1 -1
- package/dist/assets/{panels-BTEEcvYR.js → panels-NhGV0SBq.js} +1 -1
- package/dist/assets/{process-output-pPgH0ANl.js → process-output-BiWV34b7.js} +1 -1
- package/dist/assets/{radio-group-D7rh6Zek.js → radio-group-DV-lQvil.js} +1 -1
- package/dist/assets/{readonly-python-code-CvPx4CU_.js → readonly-python-code-B0glFTKW.js} +1 -1
- package/dist/assets/{reveal-component-DGQQBmed.js → reveal-component-C2M2FixA.js} +12 -12
- package/dist/assets/{run-page-yFxABzXq.js → run-page-B6I-Lshx.js} +1 -1
- package/dist/assets/{scratchpad-panel-B_nxW99u.js → scratchpad-panel-DU7perVk.js} +1 -1
- package/dist/assets/session-panel-B4UTmf-j.js +1 -0
- package/dist/assets/{snippets-panel-BSgcoo-M.js → snippets-panel-KRdl4bVk.js} +1 -1
- package/dist/assets/{state-BWM3qRkR.js → state-BSm0OM93.js} +1 -1
- package/dist/assets/{state-C8yHPSLN.js → state-CbkvENzk.js} +2 -2
- package/dist/assets/{textarea-bBuxbFm7.js → textarea-CBbixTxi.js} +1 -1
- package/dist/assets/{tracing-pt7eDHWc.js → tracing-D8LGpuXa.js} +1 -1
- package/dist/assets/{tracing-panel-BveZ5DZw.js → tracing-panel-TUP6kAd5.js} +2 -2
- package/dist/assets/{tree-actions-1KQDSu1H.js → tree-actions-BvrKFU0g.js} +1 -1
- package/dist/assets/{useCellActionButton-C4V7J6PS.js → useCellActionButton-CtOZIXD0.js} +1 -1
- package/dist/assets/{useDeleteCell-BvnurAZ9.js → useDeleteCell-BLvzLWAT.js} +1 -1
- package/dist/assets/{useDependencyPanelTab-DNlXaxbt.js → useDependencyPanelTab-BP0sNfUz.js} +1 -1
- package/dist/assets/{useNotebookActions-Dw6tY2qa.js → useNotebookActions-BI5dWutQ.js} +1 -1
- package/dist/assets/{useRunCells-cSZpNqnR.js → useRunCells-BClkx9Pq.js} +1 -1
- package/dist/assets/{useSplitCell-s7dSiBUa.js → useSplitCell-CpddCX_u.js} +1 -1
- package/dist/index.html +23 -23
- package/package.json +1 -1
- package/src/components/datasources/__tests__/column-preview.test.tsx +97 -0
- package/src/components/datasources/__tests__/filter-empty.test.ts +81 -0
- package/src/components/datasources/__tests__/utils.test.ts +62 -1
- package/src/components/datasources/column-preview.tsx +2 -4
- package/src/components/datasources/components.tsx +15 -7
- package/src/components/datasources/datasources.tsx +311 -178
- package/src/components/datasources/utils.ts +40 -1
- package/src/components/editor/ai/ai-completion-editor.tsx +6 -5
- package/src/components/editor/connections/components.tsx +13 -0
- package/src/components/editor/connections/storage/__tests__/__snapshots__/as-code.test.ts.snap +4 -4
- package/src/components/editor/connections/storage/as-code.ts +11 -4
- package/src/core/ai/__tests__/strip-wrapping-backticks.test.ts +133 -0
- package/src/core/ai/stream-completion-text.ts +48 -0
- package/src/core/ai/strip-wrapping-backticks.ts +88 -0
- package/src/core/cells/__tests__/cells.test.ts +33 -0
- package/src/core/cells/cells.ts +1 -1
- package/src/core/codemirror/ai/request.ts +2 -14
- package/src/core/datasets/__tests__/data-source.test.ts +226 -0
- package/src/core/datasets/data-source-connections.ts +88 -24
- package/dist/assets/column-preview-nu3Qo2OW.js +0 -1
- package/dist/assets/index-BAYF7dcV.css +0 -2
- package/dist/assets/session-panel-CTJPYbAa.js +0 -1
package/dist/index.html
CHANGED
|
@@ -66,7 +66,7 @@
|
|
|
66
66
|
<marimo-server-token data-token="{{ server_token }}" hidden></marimo-server-token>
|
|
67
67
|
<!-- /TODO -->
|
|
68
68
|
<title>{{ title }}</title>
|
|
69
|
-
<script type="module" crossorigin src="./assets/index-
|
|
69
|
+
<script type="module" crossorigin src="./assets/index-CuhY66ZR.js"></script>
|
|
70
70
|
<link rel="modulepreload" crossorigin href="./assets/preload-helper-BPPi7vOr.js">
|
|
71
71
|
<link rel="modulepreload" crossorigin href="./assets/chunk-LvLJmgfZ.js">
|
|
72
72
|
<link rel="modulepreload" crossorigin href="./assets/react-Bj1aDYRI.js">
|
|
@@ -133,7 +133,7 @@
|
|
|
133
133
|
<link rel="modulepreload" crossorigin href="./assets/debounce-DhnxH9Rh.js">
|
|
134
134
|
<link rel="modulepreload" crossorigin href="./assets/database-zap-kIkTfzTX.js">
|
|
135
135
|
<link rel="modulepreload" crossorigin href="./assets/main-B0OX4z33.js">
|
|
136
|
-
<link rel="modulepreload" crossorigin href="./assets/cells-
|
|
136
|
+
<link rel="modulepreload" crossorigin href="./assets/cells-D-v2lBet.js">
|
|
137
137
|
<link rel="modulepreload" crossorigin href="./assets/ErrorBoundary-DyYDV0HI.js">
|
|
138
138
|
<link rel="modulepreload" crossorigin href="./assets/kbd-CTUAEnEx.js">
|
|
139
139
|
<link rel="modulepreload" crossorigin href="./assets/useInstallPackage-GI6gu7_S.js">
|
|
@@ -147,38 +147,38 @@
|
|
|
147
147
|
<link rel="modulepreload" crossorigin href="./assets/usePress-DQ_tAz5W.js">
|
|
148
148
|
<link rel="modulepreload" crossorigin href="./assets/input-DA8xlX70.js">
|
|
149
149
|
<link rel="modulepreload" crossorigin href="./assets/ImperativeModal-BW1dmntO.js">
|
|
150
|
-
<link rel="modulepreload" crossorigin href="./assets/cell-link-
|
|
150
|
+
<link rel="modulepreload" crossorigin href="./assets/cell-link-DY95GSb7.js">
|
|
151
151
|
<link rel="modulepreload" crossorigin href="./assets/multi-map-C4OANz8X.js">
|
|
152
152
|
<link rel="modulepreload" crossorigin href="./assets/alert-yTS3WpF4.js">
|
|
153
153
|
<link rel="modulepreload" crossorigin href="./assets/chevron-right-CG5QYXYk.js">
|
|
154
154
|
<link rel="modulepreload" crossorigin href="./assets/dropdown-menu-CZ4XbE-c.js">
|
|
155
155
|
<link rel="modulepreload" crossorigin href="./assets/links-D1JoyKTt.js">
|
|
156
|
-
<link rel="modulepreload" crossorigin href="./assets/useRunCells-
|
|
156
|
+
<link rel="modulepreload" crossorigin href="./assets/useRunCells-BClkx9Pq.js">
|
|
157
157
|
<link rel="modulepreload" crossorigin href="./assets/copy-LK56fFow.js">
|
|
158
158
|
<link rel="modulepreload" crossorigin href="./assets/copy-BwrPA9zQ.js">
|
|
159
159
|
<link rel="modulepreload" crossorigin href="./assets/copy-icon-BuRdHNPA.js">
|
|
160
|
-
<link rel="modulepreload" crossorigin href="./assets/RenderHTML-
|
|
161
|
-
<link rel="modulepreload" crossorigin href="./assets/datasource-
|
|
162
|
-
<link rel="modulepreload" crossorigin href="./assets/state-
|
|
160
|
+
<link rel="modulepreload" crossorigin href="./assets/RenderHTML-DAt48X-F.js">
|
|
161
|
+
<link rel="modulepreload" crossorigin href="./assets/datasource-BQGI7c_u.js">
|
|
162
|
+
<link rel="modulepreload" crossorigin href="./assets/state-BSm0OM93.js">
|
|
163
163
|
<link rel="modulepreload" crossorigin href="./assets/package-Tv6ztuzw.js">
|
|
164
164
|
<link rel="modulepreload" crossorigin href="./assets/sparkles-lWUAsPhp.js">
|
|
165
|
-
<link rel="modulepreload" crossorigin href="./assets/MarimoErrorOutput-
|
|
165
|
+
<link rel="modulepreload" crossorigin href="./assets/MarimoErrorOutput-DzoKyXWR.js">
|
|
166
166
|
<link rel="modulepreload" crossorigin href="./assets/eye-off-Cm6DdkeR.js">
|
|
167
167
|
<link rel="modulepreload" crossorigin href="./assets/refresh-cw-DHwG4Mac.js">
|
|
168
|
-
<link rel="modulepreload" crossorigin href="./assets/tree-actions-
|
|
168
|
+
<link rel="modulepreload" crossorigin href="./assets/tree-actions-BvrKFU0g.js">
|
|
169
169
|
<link rel="modulepreload" crossorigin href="./assets/spinner-Bhir8k53.js">
|
|
170
|
-
<link rel="modulepreload" crossorigin href="./assets/html-to-image-
|
|
171
|
-
<link rel="modulepreload" crossorigin href="./assets/focus-
|
|
170
|
+
<link rel="modulepreload" crossorigin href="./assets/html-to-image-IPNqWX7V.js">
|
|
171
|
+
<link rel="modulepreload" crossorigin href="./assets/focus-BfFAxl9X.js">
|
|
172
172
|
<link rel="modulepreload" crossorigin href="./assets/useAsyncData-bgszE9F0.js">
|
|
173
173
|
<link rel="modulepreload" crossorigin href="./assets/LazyAnyLanguageCodeMirror-C7ALwTI_.js">
|
|
174
174
|
<link rel="modulepreload" crossorigin href="./assets/micromark-factory-space-bqhKsQDn.js">
|
|
175
175
|
<link rel="modulepreload" crossorigin href="./assets/chunk-5FQGJX7Z-D91x7QHS.js">
|
|
176
|
-
<link rel="modulepreload" crossorigin href="./assets/markdown-renderer-
|
|
176
|
+
<link rel="modulepreload" crossorigin href="./assets/markdown-renderer-DWRPo52L.js">
|
|
177
177
|
<link rel="modulepreload" crossorigin href="./assets/command-BmOGG6-2.js">
|
|
178
178
|
<link rel="modulepreload" crossorigin href="./assets/field-DCdxnSKm.js">
|
|
179
179
|
<link rel="modulepreload" crossorigin href="./assets/popover-CNCYAwbC.js">
|
|
180
180
|
<link rel="modulepreload" crossorigin href="./assets/errors-D3F5g1xA.js">
|
|
181
|
-
<link rel="modulepreload" crossorigin href="./assets/download-
|
|
181
|
+
<link rel="modulepreload" crossorigin href="./assets/download-WGU5w_3m.js">
|
|
182
182
|
<link rel="modulepreload" crossorigin href="./assets/table-BGPSHfig.js">
|
|
183
183
|
<link rel="modulepreload" crossorigin href="./assets/useIframeCapabilities-CcI1zSdn.js">
|
|
184
184
|
<link rel="modulepreload" crossorigin href="./assets/error-banner-D0xvrLog.js">
|
|
@@ -198,21 +198,21 @@
|
|
|
198
198
|
<link rel="modulepreload" crossorigin href="./assets/plus-BgB18UzY.js">
|
|
199
199
|
<link rel="modulepreload" crossorigin href="./assets/trash-2-rVklqqFF.js">
|
|
200
200
|
<link rel="modulepreload" crossorigin href="./assets/react-resizable-panels.browser.esm-CV8-hvjx.js">
|
|
201
|
-
<link rel="modulepreload" crossorigin href="./assets/JsonOutput-
|
|
201
|
+
<link rel="modulepreload" crossorigin href="./assets/JsonOutput-uEJijGXp.js">
|
|
202
202
|
<link rel="modulepreload" crossorigin href="./assets/chart-no-axes-column-nqk474t8.js">
|
|
203
203
|
<link rel="modulepreload" crossorigin href="./assets/square-function-uY_yJr5g.js">
|
|
204
204
|
<link rel="modulepreload" crossorigin href="./assets/spec-CPQR_o92.js">
|
|
205
|
-
<link rel="modulepreload" crossorigin href="./assets/column-preview-
|
|
205
|
+
<link rel="modulepreload" crossorigin href="./assets/column-preview-BzvFdLI5.js">
|
|
206
206
|
<link rel="modulepreload" crossorigin href="./assets/icons-Ol38nIbL.js">
|
|
207
|
-
<link rel="modulepreload" crossorigin href="./assets/radio-group-
|
|
208
|
-
<link rel="modulepreload" crossorigin href="./assets/floating-outline-
|
|
207
|
+
<link rel="modulepreload" crossorigin href="./assets/radio-group-DV-lQvil.js">
|
|
208
|
+
<link rel="modulepreload" crossorigin href="./assets/floating-outline-D33Zu-Ad.js">
|
|
209
209
|
<link rel="modulepreload" crossorigin href="./assets/objectWithoutPropertiesLoose-DfWeGRFv.js">
|
|
210
210
|
<link rel="modulepreload" crossorigin href="./assets/esm-xaw4Vy8N.js">
|
|
211
|
-
<link rel="modulepreload" crossorigin href="./assets/readonly-python-code-
|
|
211
|
+
<link rel="modulepreload" crossorigin href="./assets/readonly-python-code-B0glFTKW.js">
|
|
212
212
|
<link rel="modulepreload" crossorigin href="./assets/file-headphone-BrQspHac.js">
|
|
213
213
|
<link rel="modulepreload" crossorigin href="./assets/file-BrdxGLRX.js">
|
|
214
214
|
<link rel="modulepreload" crossorigin href="./assets/image-DQHXdEQn.js">
|
|
215
|
-
<link rel="modulepreload" crossorigin href="./assets/file-icons-
|
|
215
|
+
<link rel="modulepreload" crossorigin href="./assets/file-icons-CF-YT4hq.js">
|
|
216
216
|
<link rel="modulepreload" crossorigin href="./assets/switch-Ac7CgLgh.js">
|
|
217
217
|
<link rel="modulepreload" crossorigin href="./assets/events-CPoJAfgx.js">
|
|
218
218
|
<link rel="modulepreload" crossorigin href="./assets/globals-BivClSmn.js">
|
|
@@ -221,11 +221,11 @@
|
|
|
221
221
|
<link rel="modulepreload" crossorigin href="./assets/memoize-Tp7rARFe.js">
|
|
222
222
|
<link rel="modulepreload" crossorigin href="./assets/get-C-qh_et5.js">
|
|
223
223
|
<link rel="modulepreload" crossorigin href="./assets/_baseSet-CxV9N1bc.js">
|
|
224
|
-
<link rel="modulepreload" crossorigin href="./assets/state-
|
|
224
|
+
<link rel="modulepreload" crossorigin href="./assets/state-CbkvENzk.js">
|
|
225
225
|
<link rel="modulepreload" crossorigin href="./assets/label-C2_YQRio.js">
|
|
226
|
-
<link rel="modulepreload" crossorigin href="./assets/textarea-
|
|
226
|
+
<link rel="modulepreload" crossorigin href="./assets/textarea-CBbixTxi.js">
|
|
227
227
|
<link rel="modulepreload" crossorigin href="./assets/refresh-ccw-C-n2VFP5.js">
|
|
228
|
-
<link rel="modulepreload" crossorigin href="./assets/form-
|
|
228
|
+
<link rel="modulepreload" crossorigin href="./assets/form-CVBFx2z3.js">
|
|
229
229
|
<link rel="modulepreload" crossorigin href="./assets/renderShortcut--ku7VECI.js">
|
|
230
230
|
<link rel="modulepreload" crossorigin href="./assets/useBoolean-CE6PfxGj.js">
|
|
231
231
|
<link rel="modulepreload" crossorigin href="./assets/useDeepCompareMemoize-CVmg-GkP.js">
|
|
@@ -246,7 +246,7 @@
|
|
|
246
246
|
<link rel="stylesheet" crossorigin href="./assets/cells-pu5xzE_v.css">
|
|
247
247
|
<link rel="stylesheet" crossorigin href="./assets/markdown-renderer-DdDKmWlR.css">
|
|
248
248
|
<link rel="stylesheet" crossorigin href="./assets/JsonOutput-B7vuddcd.css">
|
|
249
|
-
<link rel="stylesheet" crossorigin href="./assets/index-
|
|
249
|
+
<link rel="stylesheet" crossorigin href="./assets/index-BNWrEIlp.css">
|
|
250
250
|
</head>
|
|
251
251
|
<body>
|
|
252
252
|
<div id="root"></div>
|
package/package.json
CHANGED
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
/* Copyright 2026 Marimo. All rights reserved. */
|
|
2
|
+
|
|
3
|
+
import { render } from "@testing-library/react";
|
|
4
|
+
import { Provider } from "jotai";
|
|
5
|
+
import type React from "react";
|
|
6
|
+
import { beforeEach, describe, expect, it, vi } from "vitest";
|
|
7
|
+
import { MockRequestClient } from "@/__mocks__/requests";
|
|
8
|
+
import type { SQLTableContext } from "@/core/datasets/data-source-connections";
|
|
9
|
+
import type { DataTable, DataTableColumn } from "@/core/kernel/messages";
|
|
10
|
+
import { requestClientAtom } from "@/core/network/requests";
|
|
11
|
+
import { store } from "@/core/state/jotai";
|
|
12
|
+
import { DatasetColumnPreview } from "../column-preview";
|
|
13
|
+
|
|
14
|
+
const wrapper = ({ children }: { children: React.ReactNode }) => (
|
|
15
|
+
<Provider store={store}>{children}</Provider>
|
|
16
|
+
);
|
|
17
|
+
|
|
18
|
+
const table: DataTable = {
|
|
19
|
+
name: "users",
|
|
20
|
+
columns: [],
|
|
21
|
+
source: "my_engine",
|
|
22
|
+
// Not "connection"/"catalog", so a preview is requested on mount.
|
|
23
|
+
source_type: "duckdb",
|
|
24
|
+
type: "table",
|
|
25
|
+
engine: "my_engine" as DataTable["engine"],
|
|
26
|
+
indexes: null,
|
|
27
|
+
num_columns: null,
|
|
28
|
+
num_rows: null,
|
|
29
|
+
variable_name: null,
|
|
30
|
+
primary_keys: null,
|
|
31
|
+
};
|
|
32
|
+
|
|
33
|
+
const column = { name: "email" } as DataTableColumn;
|
|
34
|
+
|
|
35
|
+
function renderPreview(sqlTableContext?: SQLTableContext) {
|
|
36
|
+
const client = MockRequestClient.create();
|
|
37
|
+
store.set(requestClientAtom, client);
|
|
38
|
+
render(
|
|
39
|
+
<DatasetColumnPreview
|
|
40
|
+
table={table}
|
|
41
|
+
column={column}
|
|
42
|
+
preview={undefined}
|
|
43
|
+
onAddColumnChart={vi.fn()}
|
|
44
|
+
sqlTableContext={sqlTableContext}
|
|
45
|
+
/>,
|
|
46
|
+
{ wrapper },
|
|
47
|
+
);
|
|
48
|
+
return client;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
const ctx = (
|
|
52
|
+
overrides: Partial<SQLTableContext> & { database: string; schema: string },
|
|
53
|
+
): SQLTableContext => ({
|
|
54
|
+
engine: "my_engine",
|
|
55
|
+
dialect: "duckdb",
|
|
56
|
+
...overrides,
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
describe("DatasetColumnPreview fullyQualifiedTableName", () => {
|
|
60
|
+
beforeEach(() => {
|
|
61
|
+
vi.clearAllMocks();
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
it("uses just the table name without a SQL context", () => {
|
|
65
|
+
const client = renderPreview(undefined);
|
|
66
|
+
expect(client.previewDatasetColumn).toHaveBeenCalledWith(
|
|
67
|
+
expect.objectContaining({ fullyQualifiedTableName: "users" }),
|
|
68
|
+
);
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
it("qualifies with database and schema for flat engines", () => {
|
|
72
|
+
const client = renderPreview(ctx({ database: "db", schema: "public" }));
|
|
73
|
+
expect(client.previewDatasetColumn).toHaveBeenCalledWith(
|
|
74
|
+
expect.objectContaining({ fullyQualifiedTableName: "db.public.users" }),
|
|
75
|
+
);
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
it("does not emit a double dot for schemaless tables", () => {
|
|
79
|
+
// Regression: previously produced "db..users".
|
|
80
|
+
const client = renderPreview(ctx({ database: "db", schema: "" }));
|
|
81
|
+
expect(client.previewDatasetColumn).toHaveBeenCalledWith(
|
|
82
|
+
expect.objectContaining({ fullyQualifiedTableName: "db.users" }),
|
|
83
|
+
);
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
it("includes the full schema path for nested namespaces", () => {
|
|
87
|
+
// Regression: previously dropped schemaPath and emitted "top.deep.users".
|
|
88
|
+
const client = renderPreview(
|
|
89
|
+
ctx({ database: "top", schema: "deep", schemaPath: ["nested", "deep"] }),
|
|
90
|
+
);
|
|
91
|
+
expect(client.previewDatasetColumn).toHaveBeenCalledWith(
|
|
92
|
+
expect.objectContaining({
|
|
93
|
+
fullyQualifiedTableName: "top.nested.deep.users",
|
|
94
|
+
}),
|
|
95
|
+
);
|
|
96
|
+
});
|
|
97
|
+
});
|
|
@@ -28,11 +28,15 @@ function makeSchema(opts: {
|
|
|
28
28
|
name: string;
|
|
29
29
|
tables: DataTable[];
|
|
30
30
|
tables_resolved?: boolean;
|
|
31
|
+
child_schemas?: DatabaseSchema[];
|
|
32
|
+
child_schemas_resolved?: boolean;
|
|
31
33
|
}): DatabaseSchema {
|
|
32
34
|
return {
|
|
33
35
|
name: opts.name,
|
|
34
36
|
tables: opts.tables,
|
|
35
37
|
tables_resolved: opts.tables_resolved ?? true,
|
|
38
|
+
child_schemas: opts.child_schemas ?? [],
|
|
39
|
+
child_schemas_resolved: opts.child_schemas_resolved ?? true,
|
|
36
40
|
};
|
|
37
41
|
}
|
|
38
42
|
|
|
@@ -180,4 +184,81 @@ describe("filterEmptyDatabases", () => {
|
|
|
180
184
|
|
|
181
185
|
expect(databases).toEqual(snapshot);
|
|
182
186
|
});
|
|
187
|
+
|
|
188
|
+
it("keeps a namespace that has only child namespaces (no own tables)", () => {
|
|
189
|
+
const databases = [
|
|
190
|
+
makeDatabase("iceberg", [
|
|
191
|
+
makeSchema({
|
|
192
|
+
name: "top",
|
|
193
|
+
tables: [],
|
|
194
|
+
child_schemas: [
|
|
195
|
+
makeSchema({ name: "nested", tables: [makeTable("t1")] }),
|
|
196
|
+
],
|
|
197
|
+
}),
|
|
198
|
+
]),
|
|
199
|
+
];
|
|
200
|
+
|
|
201
|
+
expect(filterEmptyDatabases(databases)).toBe(databases);
|
|
202
|
+
});
|
|
203
|
+
|
|
204
|
+
it("preserves a namespace whose child schemas are deferred", () => {
|
|
205
|
+
const databases = [
|
|
206
|
+
makeDatabase("iceberg", [
|
|
207
|
+
makeSchema({
|
|
208
|
+
name: "top",
|
|
209
|
+
tables: [],
|
|
210
|
+
child_schemas: [],
|
|
211
|
+
child_schemas_resolved: false,
|
|
212
|
+
}),
|
|
213
|
+
]),
|
|
214
|
+
];
|
|
215
|
+
|
|
216
|
+
expect(filterEmptyDatabases(databases)).toBe(databases);
|
|
217
|
+
});
|
|
218
|
+
|
|
219
|
+
it("hides a nested namespace that is resolved-empty", () => {
|
|
220
|
+
const databases = [
|
|
221
|
+
makeDatabase("iceberg", [
|
|
222
|
+
makeSchema({
|
|
223
|
+
name: "top",
|
|
224
|
+
tables: [makeTable("t1")],
|
|
225
|
+
child_schemas: [
|
|
226
|
+
makeSchema({ name: "empty_child", tables: [] }),
|
|
227
|
+
makeSchema({ name: "full_child", tables: [makeTable("t2")] }),
|
|
228
|
+
],
|
|
229
|
+
}),
|
|
230
|
+
]),
|
|
231
|
+
];
|
|
232
|
+
|
|
233
|
+
expect(filterEmptyDatabases(databases)).toEqual([
|
|
234
|
+
makeDatabase("iceberg", [
|
|
235
|
+
makeSchema({
|
|
236
|
+
name: "top",
|
|
237
|
+
tables: [makeTable("t1")],
|
|
238
|
+
child_schemas: [
|
|
239
|
+
makeSchema({ name: "full_child", tables: [makeTable("t2")] }),
|
|
240
|
+
],
|
|
241
|
+
}),
|
|
242
|
+
]),
|
|
243
|
+
]);
|
|
244
|
+
});
|
|
245
|
+
|
|
246
|
+
it("hides a parent namespace when all its descendants are empty", () => {
|
|
247
|
+
const databases = [
|
|
248
|
+
makeDatabase("iceberg", [
|
|
249
|
+
makeSchema({
|
|
250
|
+
name: "top",
|
|
251
|
+
tables: [],
|
|
252
|
+
child_schemas: [makeSchema({ name: "empty_child", tables: [] })],
|
|
253
|
+
}),
|
|
254
|
+
makeSchema({ name: "other", tables: [makeTable("t1")] }),
|
|
255
|
+
]),
|
|
256
|
+
];
|
|
257
|
+
|
|
258
|
+
expect(filterEmptyDatabases(databases)).toEqual([
|
|
259
|
+
makeDatabase("iceberg", [
|
|
260
|
+
makeSchema({ name: "other", tables: [makeTable("t1")] }),
|
|
261
|
+
]),
|
|
262
|
+
]);
|
|
263
|
+
});
|
|
183
264
|
});
|
|
@@ -4,7 +4,7 @@ import { describe, expect, it } from "vitest";
|
|
|
4
4
|
import type { SQLTableContext } from "@/core/datasets/data-source-connections";
|
|
5
5
|
import { DUCKDB_ENGINE } from "@/core/datasets/engines";
|
|
6
6
|
import type { DataTable, DataTableColumn } from "@/core/kernel/messages";
|
|
7
|
-
import { sqlCode } from "../utils";
|
|
7
|
+
import { sqlCode, tableUniqueId } from "../utils";
|
|
8
8
|
|
|
9
9
|
describe("sqlCode", () => {
|
|
10
10
|
const mockTable: DataTable = {
|
|
@@ -458,3 +458,64 @@ describe("sqlCode", () => {
|
|
|
458
458
|
});
|
|
459
459
|
});
|
|
460
460
|
});
|
|
461
|
+
|
|
462
|
+
describe("tableUniqueId", () => {
|
|
463
|
+
const ctx = (
|
|
464
|
+
overrides: Partial<SQLTableContext> & { database: string; schema: string },
|
|
465
|
+
): SQLTableContext => ({
|
|
466
|
+
engine: "e",
|
|
467
|
+
dialect: "duckdb",
|
|
468
|
+
...overrides,
|
|
469
|
+
});
|
|
470
|
+
|
|
471
|
+
it("returns just the table name without a context", () => {
|
|
472
|
+
expect(tableUniqueId(undefined, "t")).toBe("t");
|
|
473
|
+
});
|
|
474
|
+
|
|
475
|
+
it("uses database + schema for flat engines", () => {
|
|
476
|
+
expect(tableUniqueId(ctx({ database: "db", schema: "public" }), "t")).toBe(
|
|
477
|
+
"db.public.t",
|
|
478
|
+
);
|
|
479
|
+
});
|
|
480
|
+
|
|
481
|
+
it("does not duplicate the leaf schema for nested namespaces", () => {
|
|
482
|
+
// Regression: previously produced "top.nested.nested.t".
|
|
483
|
+
expect(
|
|
484
|
+
tableUniqueId(
|
|
485
|
+
ctx({ database: "top", schema: "nested", schemaPath: ["nested"] }),
|
|
486
|
+
"t",
|
|
487
|
+
),
|
|
488
|
+
).toBe("top.nested.t");
|
|
489
|
+
});
|
|
490
|
+
|
|
491
|
+
it("includes the full schema path for deeply nested namespaces", () => {
|
|
492
|
+
expect(
|
|
493
|
+
tableUniqueId(
|
|
494
|
+
ctx({
|
|
495
|
+
database: "top",
|
|
496
|
+
schema: "deep",
|
|
497
|
+
schemaPath: ["nested", "deep"],
|
|
498
|
+
}),
|
|
499
|
+
"t",
|
|
500
|
+
),
|
|
501
|
+
).toBe("top.nested.deep.t");
|
|
502
|
+
});
|
|
503
|
+
|
|
504
|
+
it("falls back to the flat schema when schemaPath is empty", () => {
|
|
505
|
+
expect(
|
|
506
|
+
tableUniqueId(
|
|
507
|
+
ctx({ database: "db", schema: "public", schemaPath: [] }),
|
|
508
|
+
"t",
|
|
509
|
+
),
|
|
510
|
+
).toBe("db.public.t");
|
|
511
|
+
});
|
|
512
|
+
|
|
513
|
+
it("does not emit a double dot for schemaless tables", () => {
|
|
514
|
+
expect(
|
|
515
|
+
tableUniqueId(ctx({ database: "db", schema: "", schemaPath: [] }), "t"),
|
|
516
|
+
).toBe("db.t");
|
|
517
|
+
expect(tableUniqueId(ctx({ database: "db", schema: "" }), "t")).toBe(
|
|
518
|
+
"db.t",
|
|
519
|
+
);
|
|
520
|
+
});
|
|
521
|
+
});
|
|
@@ -29,7 +29,7 @@ import { Button } from "../ui/button";
|
|
|
29
29
|
import { Tooltip } from "../ui/tooltip";
|
|
30
30
|
import { ColumnPreviewContainer } from "./components";
|
|
31
31
|
import { InstallPackageButton } from "./install-package-button";
|
|
32
|
-
import { convertStatsName, sqlCode } from "./utils";
|
|
32
|
+
import { convertStatsName, sqlCode, tableUniqueId } from "./utils";
|
|
33
33
|
|
|
34
34
|
export const DatasetColumnPreview: React.FC<{
|
|
35
35
|
table: DataTable;
|
|
@@ -48,9 +48,7 @@ export const DatasetColumnPreview: React.FC<{
|
|
|
48
48
|
tableName: table.name,
|
|
49
49
|
columnName: column.name,
|
|
50
50
|
sourceType: table.source_type,
|
|
51
|
-
fullyQualifiedTableName: sqlTableContext
|
|
52
|
-
? `${sqlTableContext.database}.${sqlTableContext.schema}.${table.name}`
|
|
53
|
-
: table.name,
|
|
51
|
+
fullyQualifiedTableName: tableUniqueId(sqlTableContext, table.name),
|
|
54
52
|
});
|
|
55
53
|
};
|
|
56
54
|
|
|
@@ -14,25 +14,31 @@ export const RotatingChevron: React.FC<{ isExpanded: boolean }> = ({
|
|
|
14
14
|
export const DatasourceLabel: React.FC<{
|
|
15
15
|
children: React.ReactNode;
|
|
16
16
|
className?: string;
|
|
17
|
-
|
|
17
|
+
style?: CSSProperties;
|
|
18
|
+
}> = ({ children, className, style }) => {
|
|
18
19
|
return (
|
|
19
20
|
<div
|
|
20
21
|
className={cn(
|
|
21
22
|
"flex gap-1.5 items-center font-bold py-1.5 text-muted-foreground bg-(--slate-2) text-sm",
|
|
22
23
|
className,
|
|
23
24
|
)}
|
|
25
|
+
style={style}
|
|
24
26
|
>
|
|
25
27
|
{children}
|
|
26
28
|
</div>
|
|
27
29
|
);
|
|
28
30
|
};
|
|
29
31
|
|
|
30
|
-
export const EmptyState: React.FC<{
|
|
31
|
-
content
|
|
32
|
-
className
|
|
33
|
-
|
|
32
|
+
export const EmptyState: React.FC<{
|
|
33
|
+
content: string;
|
|
34
|
+
className?: string;
|
|
35
|
+
style?: CSSProperties;
|
|
36
|
+
}> = ({ content, className, style }) => {
|
|
34
37
|
return (
|
|
35
|
-
<div
|
|
38
|
+
<div
|
|
39
|
+
className={cn("text-sm text-muted-foreground py-1", className)}
|
|
40
|
+
style={style}
|
|
41
|
+
>
|
|
36
42
|
{content}
|
|
37
43
|
</div>
|
|
38
44
|
);
|
|
@@ -61,13 +67,15 @@ export const ErrorState: React.FC<{
|
|
|
61
67
|
export const LoadingState: React.FC<{
|
|
62
68
|
message: string;
|
|
63
69
|
className?: string;
|
|
64
|
-
|
|
70
|
+
style?: CSSProperties;
|
|
71
|
+
}> = ({ message, className, style }) => {
|
|
65
72
|
return (
|
|
66
73
|
<div
|
|
67
74
|
className={cn(
|
|
68
75
|
"text-sm bg-blue-50 dark:bg-(--accent) text-blue-500 dark:text-blue-50 flex items-center gap-2 p-2 h-8",
|
|
69
76
|
className,
|
|
70
77
|
)}
|
|
78
|
+
style={style}
|
|
71
79
|
>
|
|
72
80
|
<LoaderCircle className="h-4 w-4 animate-spin" />
|
|
73
81
|
{message}
|