@marimo-team/frontend 0.23.9-dev20 → 0.23.9-dev22

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.
Files changed (76) hide show
  1. package/dist/assets/{CellStatus-cST_KROR.js → CellStatus-BqCsCfF0.js} +1 -1
  2. package/dist/assets/{JsonOutput-BwgMLuPn.js → JsonOutput-7sNcAFVa.js} +1 -1
  3. package/dist/assets/{MarimoErrorOutput-DIbgMg0T.js → MarimoErrorOutput-BOi3ico3.js} +1 -1
  4. package/dist/assets/{RenderHTML-CH8QrLVl.js → RenderHTML-CUf8D25v.js} +1 -1
  5. package/dist/assets/{add-cell-with-ai-DDmHtapY.js → add-cell-with-ai-XU2qpbrH.js} +1 -1
  6. package/dist/assets/{add-connection-dialog-CRbrpdQu.js → add-connection-dialog-C9ehEIhe.js} +1 -1
  7. package/dist/assets/{agent-panel-Ci6kmMWq.js → agent-panel-CteWNnTE.js} +1 -1
  8. package/dist/assets/ai-model-dropdown-tspiiyaT.js +5 -0
  9. package/dist/assets/{app-config-button-DrlCOjyd.js → app-config-button-Co0iUAmT.js} +1 -1
  10. package/dist/assets/{cell-editor-Bhvnqdc-.js → cell-editor-SWzZ01Xy.js} +1 -1
  11. package/dist/assets/{cell-link-rh-JpkCu.js → cell-link-DLb_k4Kf.js} +1 -1
  12. package/dist/assets/{cells-Cwex_Zk1.js → cells-yL2qMkYn.js} +41 -41
  13. package/dist/assets/{chat-display-DzBxD2g1.js → chat-display-CHd7vgqs.js} +1 -1
  14. package/dist/assets/{chat-panel-ipOkNc7y.js → chat-panel-DeHm9U27.js} +1 -1
  15. package/dist/assets/{chat-ui-DqHdyg0V.js → chat-ui-cSu2PBtp.js} +1 -1
  16. package/dist/assets/{column-preview-WtwlsutV.js → column-preview-D1138kdA.js} +1 -1
  17. package/dist/assets/{command-palette-C5YuwxOo.js → command-palette-DdhU05vN.js} +1 -1
  18. package/dist/assets/{common-B3E9c-gU.js → common-DqZ15mGE.js} +1 -1
  19. package/dist/assets/{components-DFkvRpxg.js → components-BK3xAU_5.js} +1 -1
  20. package/dist/assets/{components-wm4TXPI6.js → components-DXfCmhfc.js} +1 -1
  21. package/dist/assets/{datasource-rDrNZEmo.js → datasource-CJC51Mx7.js} +1 -1
  22. package/dist/assets/{dependency-graph-panel-BWe9yDlP.js → dependency-graph-panel-avUgiyMA.js} +1 -1
  23. package/dist/assets/{documentation-panel-BAxMFFrB.js → documentation-panel-BX7dDKRB.js} +1 -1
  24. package/dist/assets/{download-NRqwJhRT.js → download-DucsG72p.js} +1 -1
  25. package/dist/assets/{edit-page-CNuf6Bpw.js → edit-page-D-PmFfK7.js} +3 -3
  26. package/dist/assets/{error-panel-CByNONxp.js → error-panel-BqoOUjfw.js} +1 -1
  27. package/dist/assets/{file-explorer-panel-DcCFx-Mu.js → file-explorer-panel-8hhN0YIE.js} +1 -1
  28. package/dist/assets/{file-icons-CV0odlbf.js → file-icons-CNUmy2Dl.js} +1 -1
  29. package/dist/assets/{floating-outline-DmddQK6b.js → floating-outline-B_uWfcys.js} +1 -1
  30. package/dist/assets/{focus-DLnEejrn.js → focus-Cyz1EKwA.js} +1 -1
  31. package/dist/assets/{form-Dnx_5kyZ.js → form-Dyt6aN_7.js} +1 -1
  32. package/dist/assets/{home-page-ffvNN1h7.js → home-page-ahZaXEAO.js} +1 -1
  33. package/dist/assets/{hooks-DuHQuYPv.js → hooks-BmREojM6.js} +1 -1
  34. package/dist/assets/{html-to-image-CzpeYelq.js → html-to-image-CKzwKb6F.js} +1 -1
  35. package/dist/assets/{index-BN0lrf4C.js → index-CWMGYIgw.js} +3 -3
  36. package/dist/assets/{kiosk-mode-C6QWFFID.js → kiosk-mode-z2bbZ6_w.js} +1 -1
  37. package/dist/assets/{layout-dxaLq-eR.js → layout-i6tKZqCw.js} +3 -3
  38. package/dist/assets/{logs-panel-9W91k1zJ.js → logs-panel-CNV8j7by.js} +1 -1
  39. package/dist/assets/{markdown-renderer-B-iPeXPA.js → markdown-renderer-3mlKZ6Wa.js} +1 -1
  40. package/dist/assets/{name-cell-input-uFfmqWYT.js → name-cell-input-BnL86ka3.js} +1 -1
  41. package/dist/assets/{outline-panel-BAWIWb20.js → outline-panel-k7h8ZSSH.js} +1 -1
  42. package/dist/assets/{packages-panel-DncGPBWC.js → packages-panel-B60rK-z4.js} +1 -1
  43. package/dist/assets/{panels-ByGgkdwq.js → panels-DHCVX_8o.js} +1 -1
  44. package/dist/assets/{process-output-D06o1mmH.js → process-output-Bi7Z_Agd.js} +1 -1
  45. package/dist/assets/{radio-group-DUTOVun5.js → radio-group-CdBpUP4h.js} +1 -1
  46. package/dist/assets/{readonly-python-code-BnaYQo0K.js → readonly-python-code-beWODwjc.js} +1 -1
  47. package/dist/assets/{reveal-component-BXfhZjrl.js → reveal-component-fENoSU2O.js} +1 -1
  48. package/dist/assets/{run-page-Um3pVJx9.js → run-page-CxA8m2Pc.js} +1 -1
  49. package/dist/assets/{scratchpad-panel-D8kI0OKl.js → scratchpad-panel-Doe-hDIO.js} +1 -1
  50. package/dist/assets/{session-panel-07dCccS4.js → session-panel-C5xz5D-N.js} +1 -1
  51. package/dist/assets/{snippets-panel-s3hpmPTY.js → snippets-panel-rCxyqna5.js} +1 -1
  52. package/dist/assets/{state-DIFGjBXT.js → state-B6DTPcu0.js} +1 -1
  53. package/dist/assets/{state-B5L8iIIr.js → state-vUGPvNzI.js} +1 -1
  54. package/dist/assets/{textarea-D0VOSGcU.js → textarea-DRcRrFRn.js} +1 -1
  55. package/dist/assets/{tracing-CSRhEdtZ.js → tracing-DKMmxWD9.js} +1 -1
  56. package/dist/assets/{tracing-panel-DuaSjEhP.js → tracing-panel-KG023vH2.js} +2 -2
  57. package/dist/assets/{useCellActionButton-CtygwcqH.js → useCellActionButton-ZXUUkx8s.js} +1 -1
  58. package/dist/assets/{useDeleteCell-DeZ7T5A3.js → useDeleteCell-wNHy5lEM.js} +1 -1
  59. package/dist/assets/{useDependencyPanelTab-Bpa86UXQ.js → useDependencyPanelTab-B7Qj36AL.js} +1 -1
  60. package/dist/assets/{useNotebookActions-DuMI8tMy.js → useNotebookActions-67xj2Dr_.js} +1 -1
  61. package/dist/assets/{useRunCells-CScbBaxj.js → useRunCells-CJadPmdw.js} +1 -1
  62. package/dist/assets/{useSplitCell-CBlrUeHM.js → useSplitCell-C04l3e__.js} +1 -1
  63. package/dist/index.html +22 -22
  64. package/package.json +1 -1
  65. package/src/components/ai/__tests__/ai-utils.test.ts +43 -38
  66. package/src/components/ai/ai-model-dropdown.tsx +2 -2
  67. package/src/components/app-config/ai-config.tsx +1 -1
  68. package/src/components/databases/display.tsx +2 -0
  69. package/src/components/datasources/__tests__/utils.test.ts +82 -0
  70. package/src/components/datasources/utils.ts +16 -15
  71. package/src/core/ai/__tests__/model-registry.test.ts +72 -60
  72. package/src/core/ai/model-registry.ts +33 -28
  73. package/src/core/codemirror/format.ts +1 -0
  74. package/src/core/codemirror/language/languages/sql/sql.ts +1 -0
  75. package/src/core/codemirror/language/languages/sql/utils.ts +2 -0
  76. package/dist/assets/ai-model-dropdown-CNobct-x.js +0 -5
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-BN0lrf4C.js"></script>
69
+ <script type="module" crossorigin src="./assets/index-CWMGYIgw.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-Cwex_Zk1.js">
136
+ <link rel="modulepreload" crossorigin href="./assets/cells-yL2qMkYn.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-DUF4IRRI.js">
@@ -147,35 +147,35 @@
147
147
  <link rel="modulepreload" crossorigin href="./assets/usePress-DQ_tAz5W.js">
148
148
  <link rel="modulepreload" crossorigin href="./assets/input-C3Hrdlqq.js">
149
149
  <link rel="modulepreload" crossorigin href="./assets/ImperativeModal-B3Th7k4R.js">
150
- <link rel="modulepreload" crossorigin href="./assets/cell-link-rh-JpkCu.js">
150
+ <link rel="modulepreload" crossorigin href="./assets/cell-link-DLb_k4Kf.js">
151
151
  <link rel="modulepreload" crossorigin href="./assets/multi-map-CUuNtzHt.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-CR7cnzLX.js">
155
155
  <link rel="modulepreload" crossorigin href="./assets/links-D1JoyKTt.js">
156
- <link rel="modulepreload" crossorigin href="./assets/useRunCells-CScbBaxj.js">
156
+ <link rel="modulepreload" crossorigin href="./assets/useRunCells-CJadPmdw.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-OjtDb4gO.js">
160
- <link rel="modulepreload" crossorigin href="./assets/RenderHTML-CH8QrLVl.js">
161
- <link rel="modulepreload" crossorigin href="./assets/datasource-rDrNZEmo.js">
162
- <link rel="modulepreload" crossorigin href="./assets/state-DIFGjBXT.js">
160
+ <link rel="modulepreload" crossorigin href="./assets/RenderHTML-CUf8D25v.js">
161
+ <link rel="modulepreload" crossorigin href="./assets/datasource-CJC51Mx7.js">
162
+ <link rel="modulepreload" crossorigin href="./assets/state-B6DTPcu0.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-DIbgMg0T.js">
165
+ <link rel="modulepreload" crossorigin href="./assets/MarimoErrorOutput-BOi3ico3.js">
166
166
  <link rel="modulepreload" crossorigin href="./assets/spinner-Bhir8k53.js">
167
- <link rel="modulepreload" crossorigin href="./assets/html-to-image-CzpeYelq.js">
168
- <link rel="modulepreload" crossorigin href="./assets/focus-DLnEejrn.js">
167
+ <link rel="modulepreload" crossorigin href="./assets/html-to-image-CKzwKb6F.js">
168
+ <link rel="modulepreload" crossorigin href="./assets/focus-Cyz1EKwA.js">
169
169
  <link rel="modulepreload" crossorigin href="./assets/useAsyncData-bgszE9F0.js">
170
170
  <link rel="modulepreload" crossorigin href="./assets/LazyAnyLanguageCodeMirror-DFv3yT20.js">
171
171
  <link rel="modulepreload" crossorigin href="./assets/micromark-factory-space-BygYYKhs.js">
172
172
  <link rel="modulepreload" crossorigin href="./assets/chunk-5FQGJX7Z-BzXnQz5s.js">
173
- <link rel="modulepreload" crossorigin href="./assets/markdown-renderer-B-iPeXPA.js">
173
+ <link rel="modulepreload" crossorigin href="./assets/markdown-renderer-3mlKZ6Wa.js">
174
174
  <link rel="modulepreload" crossorigin href="./assets/command-KARR7KMq.js">
175
175
  <link rel="modulepreload" crossorigin href="./assets/field-zLmMOSA4.js">
176
176
  <link rel="modulepreload" crossorigin href="./assets/popover-Bz_0Vkyf.js">
177
177
  <link rel="modulepreload" crossorigin href="./assets/errors-vr57w7Ul.js">
178
- <link rel="modulepreload" crossorigin href="./assets/download-NRqwJhRT.js">
178
+ <link rel="modulepreload" crossorigin href="./assets/download-DucsG72p.js">
179
179
  <link rel="modulepreload" crossorigin href="./assets/table-BGPSHfig.js">
180
180
  <link rel="modulepreload" crossorigin href="./assets/useIframeCapabilities-CcI1zSdn.js">
181
181
  <link rel="modulepreload" crossorigin href="./assets/error-banner-LdWZDbqd.js">
@@ -196,25 +196,25 @@
196
196
  <link rel="modulepreload" crossorigin href="./assets/plus-BgB18UzY.js">
197
197
  <link rel="modulepreload" crossorigin href="./assets/trash-2-rVklqqFF.js">
198
198
  <link rel="modulepreload" crossorigin href="./assets/react-resizable-panels.browser.esm-CV8-hvjx.js">
199
- <link rel="modulepreload" crossorigin href="./assets/JsonOutput-BwgMLuPn.js">
199
+ <link rel="modulepreload" crossorigin href="./assets/JsonOutput-7sNcAFVa.js">
200
200
  <link rel="modulepreload" crossorigin href="./assets/chart-no-axes-column-nqk474t8.js">
201
201
  <link rel="modulepreload" crossorigin href="./assets/square-function-uY_yJr5g.js">
202
202
  <link rel="modulepreload" crossorigin href="./assets/spec-CPQR_o92.js">
203
203
  <link rel="modulepreload" crossorigin href="./assets/ellipsis-vertical-CkwWkOQL.js">
204
204
  <link rel="modulepreload" crossorigin href="./assets/refresh-cw-DHwG4Mac.js">
205
205
  <link rel="modulepreload" crossorigin href="./assets/tree-actions-BM_EJr3E.js">
206
- <link rel="modulepreload" crossorigin href="./assets/components-wm4TXPI6.js">
207
- <link rel="modulepreload" crossorigin href="./assets/column-preview-WtwlsutV.js">
206
+ <link rel="modulepreload" crossorigin href="./assets/components-DXfCmhfc.js">
207
+ <link rel="modulepreload" crossorigin href="./assets/column-preview-D1138kdA.js">
208
208
  <link rel="modulepreload" crossorigin href="./assets/icons-Ol38nIbL.js">
209
- <link rel="modulepreload" crossorigin href="./assets/radio-group-DUTOVun5.js">
210
- <link rel="modulepreload" crossorigin href="./assets/floating-outline-DmddQK6b.js">
209
+ <link rel="modulepreload" crossorigin href="./assets/radio-group-CdBpUP4h.js">
210
+ <link rel="modulepreload" crossorigin href="./assets/floating-outline-B_uWfcys.js">
211
211
  <link rel="modulepreload" crossorigin href="./assets/objectWithoutPropertiesLoose-DfWeGRFv.js">
212
212
  <link rel="modulepreload" crossorigin href="./assets/esm-CqWdmSnV.js">
213
- <link rel="modulepreload" crossorigin href="./assets/readonly-python-code-BnaYQo0K.js">
213
+ <link rel="modulepreload" crossorigin href="./assets/readonly-python-code-beWODwjc.js">
214
214
  <link rel="modulepreload" crossorigin href="./assets/file-headphone-BrQspHac.js">
215
215
  <link rel="modulepreload" crossorigin href="./assets/file-BrdxGLRX.js">
216
216
  <link rel="modulepreload" crossorigin href="./assets/image-DQHXdEQn.js">
217
- <link rel="modulepreload" crossorigin href="./assets/file-icons-CV0odlbf.js">
217
+ <link rel="modulepreload" crossorigin href="./assets/file-icons-CNUmy2Dl.js">
218
218
  <link rel="modulepreload" crossorigin href="./assets/switch-BrbMxTq7.js">
219
219
  <link rel="modulepreload" crossorigin href="./assets/events-CPoJAfgx.js">
220
220
  <link rel="modulepreload" crossorigin href="./assets/globals-CDDbRvHE.js">
@@ -223,11 +223,11 @@
223
223
  <link rel="modulepreload" crossorigin href="./assets/memoize-Tp7rARFe.js">
224
224
  <link rel="modulepreload" crossorigin href="./assets/get-C-qh_et5.js">
225
225
  <link rel="modulepreload" crossorigin href="./assets/_baseSet-CxV9N1bc.js">
226
- <link rel="modulepreload" crossorigin href="./assets/state-B5L8iIIr.js">
226
+ <link rel="modulepreload" crossorigin href="./assets/state-vUGPvNzI.js">
227
227
  <link rel="modulepreload" crossorigin href="./assets/label-xHqFtfdz.js">
228
- <link rel="modulepreload" crossorigin href="./assets/textarea-D0VOSGcU.js">
228
+ <link rel="modulepreload" crossorigin href="./assets/textarea-DRcRrFRn.js">
229
229
  <link rel="modulepreload" crossorigin href="./assets/refresh-ccw-C-n2VFP5.js">
230
- <link rel="modulepreload" crossorigin href="./assets/form-Dnx_5kyZ.js">
230
+ <link rel="modulepreload" crossorigin href="./assets/form-Dyt6aN_7.js">
231
231
  <link rel="modulepreload" crossorigin href="./assets/renderShortcut-BrHz5aiT.js">
232
232
  <link rel="modulepreload" crossorigin href="./assets/useBoolean-CtP63TnG.js">
233
233
  <link rel="modulepreload" crossorigin href="./assets/useDeepCompareMemoize-zUHU--0D.js">
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@marimo-team/frontend",
3
- "version": "0.23.9-dev20",
3
+ "version": "0.23.9-dev22",
4
4
  "main": "dist/main.js",
5
5
  "types": "dist/index.d.ts",
6
6
  "type": "module",
@@ -4,46 +4,51 @@ import type { AiModel } from "@marimo-team/llm-info";
4
4
  import { beforeEach, describe, expect, it, vi } from "vitest";
5
5
  import type { UserConfig } from "@/core/config/config-schema";
6
6
 
7
- // Mock the models.json import
8
7
  vi.mock("@marimo-team/llm-info/models.json", () => {
9
- const models: AiModel[] = [
10
- {
11
- name: "GPT-4",
12
- model: "gpt-4",
13
- description: "OpenAI GPT-4 model",
14
- providers: ["openai"],
15
- roles: ["chat", "edit"],
16
- thinking: false,
17
- },
18
- {
19
- name: "Claude 3",
20
- model: "claude-3-sonnet",
21
- description: "Anthropic Claude 3 Sonnet",
22
- providers: ["anthropic"],
23
- roles: ["chat", "edit"],
24
- thinking: false,
25
- },
26
- {
27
- name: "Gemini Pro",
28
- model: "gemini-pro",
29
- description: "Google Gemini Pro model",
30
- providers: ["google"],
31
- roles: ["chat", "edit"],
32
- thinking: false,
33
- },
34
- {
35
- name: "Ollama Model",
36
- model: "llama2",
37
- description: "Ollama Llama 2 model",
38
- providers: ["ollama"],
39
- roles: ["chat", "edit"],
40
- thinking: false,
41
- },
42
- ];
43
-
44
- return {
45
- models: models,
8
+ const make = (
9
+ overrides: Partial<AiModel> & Pick<AiModel, "name" | "model">,
10
+ ): AiModel => ({
11
+ description: "",
12
+ roles: ["chat", "edit"],
13
+ capabilities: [],
14
+ input_types: [],
15
+ output_types: [],
16
+ release_date: "1970-01-01",
17
+ ...overrides,
18
+ });
19
+
20
+ const models: Record<string, AiModel[]> = {
21
+ openai: [
22
+ make({
23
+ name: "GPT-4",
24
+ model: "gpt-4",
25
+ description: "OpenAI GPT-4 model",
26
+ }),
27
+ ],
28
+ anthropic: [
29
+ make({
30
+ name: "Claude 3",
31
+ model: "claude-3-sonnet",
32
+ description: "Anthropic Claude 3 Sonnet",
33
+ }),
34
+ ],
35
+ google: [
36
+ make({
37
+ name: "Gemini Pro",
38
+ model: "gemini-pro",
39
+ description: "Google Gemini Pro model",
40
+ }),
41
+ ],
42
+ ollama: [
43
+ make({
44
+ name: "Ollama Model",
45
+ model: "llama2",
46
+ description: "Ollama Llama 2 model",
47
+ }),
48
+ ],
46
49
  };
50
+
51
+ return { models };
47
52
  });
48
53
 
49
54
  // Must import after mock
@@ -305,7 +305,7 @@ const AiModelDropdownItem = ({
305
305
  <div className="flex flex-row w-full items-center">
306
306
  <span>{model.name}</span>
307
307
  <div className="ml-auto">
308
- {model.thinking && (
308
+ {model.capabilities.includes("thinking") && (
309
309
  <Tooltip content="Reasoning model">
310
310
  <BrainIcon
311
311
  className={`h-5 w-5 rounded-md p-1 ${getTagColour("thinking")}`}
@@ -362,7 +362,7 @@ export const AiModelInfoDisplay = ({
362
362
  </div>
363
363
  )}
364
364
 
365
- {model.thinking && (
365
+ {model.capabilities.includes("thinking") && (
366
366
  <div className="flex items-center gap-2">
367
367
  <div className="w-2 h-2 bg-purple-500 rounded-full animate-pulse" />
368
368
  <span className="text-xs text-muted-foreground">
@@ -509,7 +509,7 @@ const ModelInfoCard = ({ model }: { model: AiModel }) => {
509
509
  <Tooltip content="Custom model">
510
510
  {model.custom && <BotIcon className="h-4 w-4" />}
511
511
  </Tooltip>
512
- {model.thinking && (
512
+ {model.capabilities.includes("thinking") && (
513
513
  <div
514
514
  className={cn(
515
515
  "flex items-center gap-1 rounded px-1 py-0.5 w-fit",
@@ -53,6 +53,8 @@ export function dbDisplayName(name: string) {
53
53
  return "MongoDB";
54
54
  case "iceberg":
55
55
  return "Apache Iceberg";
56
+ case "dremio":
57
+ return "Dremio";
56
58
  default:
57
59
  return name;
58
60
  }
@@ -313,6 +313,26 @@ describe("sqlCode", () => {
313
313
  );
314
314
  });
315
315
 
316
+ it("should preserve dots inside quoted schema names", () => {
317
+ const sqlTableContext: SQLTableContext = {
318
+ engine: "postgres",
319
+ schema: "analytics.events",
320
+ defaultSchema: "public",
321
+ defaultDatabase: "mydb",
322
+ database: "remote",
323
+ dialect: "postgres",
324
+ };
325
+
326
+ const result = sqlCode({
327
+ table: mockTable,
328
+ columnName: mockColumn.name,
329
+ sqlTableContext,
330
+ });
331
+ expect(result).toBe(
332
+ '_df = mo.sql(f"""\nSELECT "email" FROM "remote"."analytics.events"."users" LIMIT 100\n""", engine=postgres)',
333
+ );
334
+ });
335
+
316
336
  it("should not quote * column name", () => {
317
337
  const sqlTableContext: SQLTableContext = {
318
338
  engine: "postgres",
@@ -334,6 +354,68 @@ describe("sqlCode", () => {
334
354
  });
335
355
  });
336
356
 
357
+ describe("Dremio dialect", () => {
358
+ it("should quote reserved column names and table path parts", () => {
359
+ const sqlTableContext: SQLTableContext = {
360
+ engine: "dremio_conn",
361
+ schema: "operations",
362
+ defaultSchema: "",
363
+ defaultDatabase: "",
364
+ database: "lakehouse",
365
+ dialect: "dremio",
366
+ };
367
+
368
+ const result = sqlCode({
369
+ table: { ...mockTable, name: "shipments" as const },
370
+ columnName: "order",
371
+ sqlTableContext,
372
+ });
373
+ expect(result).toBe(
374
+ '_df = mo.sql(f"""\nSELECT "order" FROM "lakehouse"."operations"."shipments" LIMIT 100\n""", engine=dremio_conn)',
375
+ );
376
+ });
377
+
378
+ it("should not quote * column name", () => {
379
+ const sqlTableContext: SQLTableContext = {
380
+ engine: "dremio_conn",
381
+ schema: "operations",
382
+ defaultSchema: "",
383
+ defaultDatabase: "",
384
+ database: "lakehouse",
385
+ dialect: "dremio",
386
+ };
387
+
388
+ const result = sqlCode({
389
+ table: { ...mockTable, name: "customers" as const },
390
+ columnName: "*",
391
+ sqlTableContext,
392
+ });
393
+ expect(result).toBe(
394
+ '_df = mo.sql(f"""\nSELECT * FROM "lakehouse"."operations"."customers" LIMIT 100\n""", engine=dremio_conn)',
395
+ );
396
+ });
397
+
398
+ it("should preserve dots inside quoted schema names", () => {
399
+ const sqlTableContext: SQLTableContext = {
400
+ engine: "dremio_conn",
401
+ schema: "samples.dremio.com",
402
+ defaultSchema: "",
403
+ defaultDatabase: "",
404
+ database: "Samples",
405
+ dialect: "dremio",
406
+ };
407
+
408
+ const result = sqlCode({
409
+ table: { ...mockTable, name: "airlines" as const },
410
+ columnName: "*",
411
+ sqlTableContext,
412
+ });
413
+ expect(result).toBe(
414
+ '_df = mo.sql(f"""\nSELECT * FROM "Samples"."samples.dremio.com"."airlines" LIMIT 100\n""", engine=dremio_conn)',
415
+ );
416
+ });
417
+ });
418
+
337
419
  describe("fallback behavior", () => {
338
420
  it("should use default formatter for unknown dialect", () => {
339
421
  const sqlTableContext: SQLTableContext = {
@@ -15,9 +15,9 @@ export function isSchemaless(schemaName: string) {
15
15
 
16
16
  interface SqlCodeFormatter {
17
17
  /**
18
- * Format the table name based on dialect-specific rules
18
+ * Format the table path based on dialect-specific rules
19
19
  */
20
- formatTableName: (tableName: string) => string;
20
+ formatTablePath: (tablePath: string[]) => string;
21
21
  /**
22
22
  * Format the SELECT clause
23
23
  */
@@ -25,7 +25,7 @@ interface SqlCodeFormatter {
25
25
  }
26
26
 
27
27
  const defaultFormatter: SqlCodeFormatter = {
28
- formatTableName: (tableName: string) => tableName,
28
+ formatTablePath: (tablePath: string[]) => tablePath.join("."),
29
29
  formatSelectClause: (columnName: string, tableName: string) =>
30
30
  `SELECT ${columnName} FROM ${tableName} LIMIT 100`,
31
31
  };
@@ -41,7 +41,8 @@ function getFormatter(dialect: string): SqlCodeFormatter {
41
41
  const quote = BigQueryDialect.spec.identifierQuotes;
42
42
  return {
43
43
  // BigQuery uses backticks for identifiers
44
- formatTableName: (tableName: string) => `${quote}${tableName}${quote}`,
44
+ formatTablePath: (tablePath: string[]) =>
45
+ `${quote}${tablePath.join(".")}${quote}`,
45
46
  formatSelectClause: defaultFormatter.formatSelectClause,
46
47
  };
47
48
  }
@@ -49,7 +50,7 @@ function getFormatter(dialect: string): SqlCodeFormatter {
49
50
  case "sqlserver":
50
51
  case "microsoft sql server":
51
52
  return {
52
- formatTableName: defaultFormatter.formatTableName,
53
+ formatTablePath: defaultFormatter.formatTablePath,
53
54
  formatSelectClause: (columnName: string, tableName: string) =>
54
55
  `SELECT TOP 100 ${columnName} FROM ${tableName}`,
55
56
  };
@@ -57,12 +58,11 @@ function getFormatter(dialect: string): SqlCodeFormatter {
57
58
  case "postgres":
58
59
  case "postgresql":
59
60
  case "duckdb":
61
+ case "dremio":
60
62
  // Quote column and table names to avoid raising errors on weird characters
61
63
  return {
62
- formatTableName: (tableName: string) => {
63
- const parts = tableName.split(".");
64
- return parts.map((part) => `"${part}"`).join(".");
65
- },
64
+ formatTablePath: (tablePath: string[]) =>
65
+ tablePath.map((part) => `"${part}"`).join("."),
66
66
  formatSelectClause: (columnName: string, tableName: string) =>
67
67
  `SELECT ${columnName === "*" ? "*" : `"${columnName}"`} FROM ${tableName} LIMIT 100`,
68
68
  };
@@ -114,26 +114,27 @@ export function sqlCode({
114
114
  database,
115
115
  dialect,
116
116
  } = sqlTableContext;
117
- let tableName = table.name;
117
+ const tablePath = [table.name];
118
118
 
119
119
  // Set the fully qualified table name based on schema and database
120
120
  if (isSchemaless(schema)) {
121
- tableName =
122
- database === defaultDatabase ? tableName : `${database}.${tableName}`;
121
+ if (database !== defaultDatabase) {
122
+ tablePath.unshift(database);
123
+ }
123
124
  } else {
124
125
  // Include schema if it's not the default schema
125
126
  if (schema !== defaultSchema) {
126
- tableName = `${schema}.${tableName}`;
127
+ tablePath.unshift(schema);
127
128
  }
128
129
 
129
130
  // Include database if it's not the default database
130
131
  if (database !== defaultDatabase) {
131
- tableName = `${database}.${tableName}`;
132
+ tablePath.unshift(database);
132
133
  }
133
134
  }
134
135
 
135
136
  const formatter = getFormatter(dialect);
136
- const formattedTableName = formatter.formatTableName(tableName);
137
+ const formattedTableName = formatter.formatTablePath(tablePath);
137
138
  const selectClause = formatter.formatSelectClause(
138
139
  columnName,
139
140
  formattedTableName,
@@ -1,46 +1,52 @@
1
1
  /* Copyright 2026 Marimo. All rights reserved. */
2
2
  import { beforeEach, describe, expect, it, vi } from "vitest";
3
3
 
4
- // Mock the models.json import
5
4
  vi.mock("@marimo-team/llm-info/models.json", () => {
6
- const models: AiModel[] = [
7
- {
8
- name: "GPT-4",
9
- model: "gpt-4",
10
- description: "OpenAI GPT-4 model",
11
- providers: ["openai"],
12
- roles: ["chat", "edit"],
13
- thinking: false,
14
- },
15
- {
16
- name: "Claude 3",
17
- model: "claude-3-sonnet",
18
- description: "Anthropic Claude 3 Sonnet",
19
- providers: ["anthropic"],
20
- roles: ["chat", "edit"],
21
- thinking: false,
22
- },
23
- {
24
- name: "Gemini Pro",
25
- model: "gemini-pro",
26
- description: "Google Gemini Pro model",
27
- providers: ["google"],
28
- roles: ["chat", "edit"],
29
- thinking: false,
30
- },
31
- {
32
- name: "Multi Provider Model",
33
- model: "multi-model",
34
- description: "Model available on multiple providers",
35
- providers: ["openai", "anthropic"],
36
- roles: ["chat", "edit"],
37
- thinking: false,
38
- },
39
- ];
40
-
41
- return {
42
- models: models,
5
+ const make = (
6
+ overrides: Partial<AiModel> & Pick<AiModel, "name" | "model">,
7
+ ): AiModel => ({
8
+ description: "",
9
+ roles: ["chat", "edit"],
10
+ capabilities: [],
11
+ input_types: [],
12
+ output_types: [],
13
+ release_date: "1970-01-01",
14
+ ...overrides,
15
+ });
16
+
17
+ const multiModel = make({
18
+ name: "Multi Provider Model",
19
+ model: "multi-model",
20
+ description: "Model available on multiple providers",
21
+ });
22
+
23
+ const models: Record<string, AiModel[]> = {
24
+ openai: [
25
+ make({
26
+ name: "GPT-4",
27
+ model: "gpt-4",
28
+ description: "OpenAI GPT-4 model",
29
+ }),
30
+ multiModel,
31
+ ],
32
+ anthropic: [
33
+ make({
34
+ name: "Claude 3",
35
+ model: "claude-3-sonnet",
36
+ description: "Anthropic Claude 3 Sonnet",
37
+ }),
38
+ multiModel,
39
+ ],
40
+ google: [
41
+ make({
42
+ name: "Gemini Pro",
43
+ model: "gemini-pro",
44
+ description: "Google Gemini Pro model",
45
+ }),
46
+ ],
43
47
  };
48
+
49
+ return { models };
44
50
  });
45
51
 
46
52
  import type { AiModel } from "@marimo-team/llm-info";
@@ -107,14 +113,15 @@ describe("AiModelRegistry", () => {
107
113
  });
108
114
 
109
115
  const ids = [...registry.getModelsMap().keys()];
110
- // Include custom and all default ones.
116
+ // Include custom and all default ones; iteration follows provider
117
+ // sections in the source data (openai → anthropic → google).
111
118
  expect(ids).toEqual([
112
119
  "openai/custom-gpt",
113
120
  "openai/gpt-4",
114
- "anthropic/claude-3-sonnet",
115
- "google/gemini-pro",
116
121
  "openai/multi-model",
122
+ "anthropic/claude-3-sonnet",
117
123
  "anthropic/multi-model",
124
+ "google/gemini-pro",
118
125
  ]);
119
126
  });
120
127
  });
@@ -125,9 +132,9 @@ describe("AiModelRegistry", () => {
125
132
  const openaiModels = registry.getModelsByProvider("openai");
126
133
 
127
134
  expect(openaiModels).toHaveLength(2); // gpt-4 and multi-model
128
- expect(
129
- openaiModels.every((model) => model.providers.includes("openai")),
130
- ).toBe(true);
135
+ expect(openaiModels.every((model) => model.provider === "openai")).toBe(
136
+ true,
137
+ );
131
138
  });
132
139
 
133
140
  it("should return empty array for provider with no models", () => {
@@ -147,9 +154,9 @@ describe("AiModelRegistry", () => {
147
154
  expect(customModel?.name).toBe("custom-gpt");
148
155
  expect(customModel?.model).toBe("custom-gpt");
149
156
  expect(customModel?.description).toBe("Custom model");
150
- expect(customModel?.providers).toEqual(["openai"]);
157
+ expect(customModel?.provider).toBe("openai");
151
158
  expect(customModel?.roles).toEqual([]);
152
- expect(customModel?.thinking).toBe(false);
159
+ expect(customModel?.capabilities).toEqual([]);
153
160
  });
154
161
 
155
162
  it("should filter models based on displayed models", () => {
@@ -309,7 +316,10 @@ describe("AiModelRegistry", () => {
309
316
 
310
317
  expect(multiModelInOpenai).toBeDefined();
311
318
  expect(multiModelInAnthropic).toBeDefined();
312
- expect(multiModelInOpenai).toEqual(multiModelInAnthropic);
319
+ // Same model id, but each entry belongs to its own provider.
320
+ expect(multiModelInOpenai?.provider).toBe("openai");
321
+ expect(multiModelInAnthropic?.provider).toBe("anthropic");
322
+ expect(multiModelInOpenai?.name).toBe(multiModelInAnthropic?.name);
313
323
  });
314
324
 
315
325
  it("should handle displayed models filter with non-existent models", () => {
@@ -339,20 +349,20 @@ describe("AiModelRegistry", () => {
339
349
  expect(model).toHaveProperty("name");
340
350
  expect(model).toHaveProperty("model");
341
351
  expect(model).toHaveProperty("description");
342
- expect(model).toHaveProperty("providers");
352
+ expect(model).toHaveProperty("provider");
343
353
  expect(model).toHaveProperty("roles");
344
- expect(model).toHaveProperty("thinking");
354
+ expect(model).toHaveProperty("capabilities");
345
355
  expect(model).toHaveProperty("custom");
346
356
 
347
357
  expect(typeof model.name).toBe("string");
348
358
  expect(typeof model.model).toBe("string");
349
359
  expect(typeof model.description).toBe("string");
350
- expect(Array.isArray(model.providers)).toBe(true);
360
+ expect(typeof model.provider).toBe("string");
351
361
  expect(Array.isArray(model.roles)).toBe(true);
352
- expect(typeof model.thinking).toBe("boolean");
362
+ expect(Array.isArray(model.capabilities)).toBe(true);
353
363
  expect(typeof model.custom).toBe("boolean");
354
364
 
355
- expect(model.providers).toContain(provider);
365
+ expect(model.provider).toBe(provider);
356
366
  }
357
367
  }
358
368
  });
@@ -367,31 +377,33 @@ describe("AiModelRegistry", () => {
367
377
 
368
378
  expect(customModel).toMatchInlineSnapshot(`
369
379
  {
380
+ "capabilities": [],
370
381
  "custom": true,
371
382
  "description": "Custom model",
383
+ "input_types": [],
372
384
  "model": "custom-gpt",
373
385
  "name": "custom-gpt",
374
- "providers": [
375
- "openai",
376
- ],
386
+ "output_types": [],
387
+ "provider": "openai",
388
+ "release_date": "1970-01-01",
377
389
  "roles": [],
378
- "thinking": false,
379
390
  }
380
391
  `);
381
392
  expect(defaultModel).toMatchInlineSnapshot(`
382
393
  {
394
+ "capabilities": [],
383
395
  "custom": false,
384
396
  "description": "OpenAI GPT-4 model",
397
+ "input_types": [],
385
398
  "model": "gpt-4",
386
399
  "name": "GPT-4",
387
- "providers": [
388
- "openai",
389
- ],
400
+ "output_types": [],
401
+ "provider": "openai",
402
+ "release_date": "1970-01-01",
390
403
  "roles": [
391
404
  "chat",
392
405
  "edit",
393
406
  ],
394
- "thinking": false,
395
407
  }
396
408
  `);
397
409
  });