@marimo-team/islands 0.23.7-dev4 → 0.23.7-dev43
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/{ConnectedDataExplorerComponent-PmilQqXR.js → ConnectedDataExplorerComponent-2lBNiUv6.js} +14 -14
- package/dist/{ErrorBoundary-Da4UeYxT.js → ErrorBoundary-D3wrPNma.js} +1 -1
- package/dist/{any-language-editor-BVR0l12r.js → any-language-editor-VWs_7v27.js} +15 -15
- package/dist/apl-Bdc61P1y.js +4 -0
- package/dist/{arc-CHF8PiiF.js → arc-DfkSnvZm.js} +2 -2
- package/dist/{architecture-7HQA4BMR-D0JB_3hE.js → architecture-7HQA4BMR-CS9jOrqM.js} +1 -1
- package/dist/{architectureDiagram-VXUJARFQ-BXQEUDtK.js → architectureDiagram-VXUJARFQ-CXVJxFhH.js} +16 -16
- package/dist/asciiarmor-DVRHDGzT.js +4 -0
- package/dist/asn1-BmuKfkfu.js +4 -0
- package/dist/assets/__vite-browser-external-CAdMKBac.js +1 -0
- package/dist/assets/worker-CpBbwbQo.js +73 -0
- package/dist/{blockDiagram-VD42YOAC-DhJe-Y9i.js → blockDiagram-VD42YOAC-DGDaxR8I.js} +11 -11
- package/dist/brainfuck-DPrRpyTV.js +4 -0
- package/dist/{button-CA5pI2YF.js → button-Dj4BTre0.js} +5 -0
- package/dist/{c4Diagram-YG6GDRKO-9dSfzOFR.js → c4Diagram-YG6GDRKO-C2hc6ne8.js} +5 -5
- package/dist/{capabilities-6laDasij.js → capabilities-C9rrYCzf.js} +1 -1
- package/dist/{channel-MqYIiKgS.js → channel-BBoIVUrJ.js} +1 -1
- package/dist/{chat-ui-B-gbqk_F.js → chat-ui-CufH8sfF.js} +626 -234
- package/dist/{check-CFM2mVDr.js → check-BcUIXnUT.js} +1 -1
- package/dist/{chunk-4BX2VUAB-BwfrWBqN.js → chunk-4BX2VUAB-CzXltWHN.js} +1 -1
- package/dist/{chunk-55IACEB6-D8THf2mi.js → chunk-55IACEB6-B-1mjMMC.js} +1 -1
- package/dist/{chunk-5FQGJX7Z-CO1e63h_.js → chunk-5FQGJX7Z-BOg95xG5.js} +2 -2
- package/dist/{chunk-ABZYJK2D-BrBb_0yY.js → chunk-ABZYJK2D-D0cLy8Bb.js} +2 -2
- package/dist/{chunk-ATLVNIR6-D-0XqNah.js → chunk-ATLVNIR6-BXsEjlHF.js} +2 -2
- package/dist/{chunk-B4BG7PRW-8iRKvugR.js → chunk-B4BG7PRW-Q1usn6T3.js} +7 -7
- package/dist/{chunk-CVBHYZKI-B6xhgaBd.js → chunk-CVBHYZKI-B_c5YBcW.js} +1 -1
- package/dist/{chunk-DI55MBZ5-C0_2D4m4.js → chunk-DI55MBZ5-D1qLYNrb.js} +6 -6
- package/dist/{chunk-EXTU4WIE-Jiw9ca1u.js → chunk-EXTU4WIE-BKNXdLmD.js} +2 -2
- package/dist/{chunk-FMBD7UC4-CHdus51S.js → chunk-FMBD7UC4-Ie8M9q0W.js} +1 -1
- package/dist/{chunk-HN2XXSSU-2Vfbq-kU.js → chunk-HN2XXSSU-E3n-Ys7Z.js} +1 -1
- package/dist/{chunk-JA3XYJ7Z-6wbaigKe.js → chunk-JA3XYJ7Z-D6c6cOBG.js} +3 -3
- package/dist/{chunk-JZLCHNYA-_rfptlUP.js → chunk-JZLCHNYA-BvsPHJmL.js} +6 -6
- package/dist/{chunk-MI3HLSF2-Do0-KRc0.js → chunk-MI3HLSF2-CUYEasXO.js} +1 -1
- package/dist/{chunk-N4CR4FBY-DIZG9dVD.js → chunk-N4CR4FBY-8ycT-O9a.js} +7 -7
- package/dist/{chunk-QN33PNHL-Cc64y40m.js → chunk-QN33PNHL-Bb-eUBW3.js} +2 -2
- package/dist/{chunk-QXUST7PY-BDG0-0Or.js → chunk-QXUST7PY-DV8yRwBd.js} +10 -10
- package/dist/{chunk-QZHKN3VN-B_Mdb8GC.js → chunk-QZHKN3VN-DRjXVwuJ.js} +1 -1
- package/dist/{chunk-S3R3BYOJ-DphMP0FA.js → chunk-S3R3BYOJ-mQeCz5CE.js} +4 -4
- package/dist/{chunk-TZMSLE5B-C9LUoYkc.js → chunk-TZMSLE5B-BqW10dHe.js} +3 -3
- package/dist/classDiagram-2ON5EDUG--Yh__LHb.js +30 -0
- package/dist/classDiagram-v2-WZHVMYZB-BC7X7Xtc.js +30 -0
- package/dist/{clike-sBZrGeF8.js → clike-DTxNUn7l.js} +1 -1
- package/dist/clojure-DEttQW5T.js +4 -0
- package/dist/cmake-jNlx_DaM.js +4 -0
- package/dist/cobol-BvvIm5MJ.js +4 -0
- package/dist/{code-block-37QAKDTI-0JNwiPGv.js → code-block-37QAKDTI-BsGy1AOJ.js} +1 -1
- package/dist/{code-visibility-zYxaYsuj.js → code-visibility-Cu7CFm0r.js} +573 -585
- package/dist/coffeescript-DTpBMyFU.js +4 -0
- package/dist/commonlisp-BmmUG8jb.js +4 -0
- package/dist/{copy-TGGAUEWp.js → copy-DLf4aN7I.js} +2 -2
- package/dist/{cose-bilkent-S5V4N54A-DXHZkJKX.js → cose-bilkent-S5V4N54A-kjoZoid4.js} +2 -2
- package/dist/crystal-BxyqmEWC.js +4 -0
- package/dist/css-BGoCtuG3.js +4 -0
- package/dist/cypher-BmCbdl3u.js +4 -0
- package/dist/d-CFrlbjZt.js +4 -0
- package/dist/{dagre-6UL2VRFP-tH87fkPA.js → dagre-6UL2VRFP-DRBWoQUw.js} +10 -10
- package/dist/{data-grid-overlay-editor-CWUN78-s.js → data-grid-overlay-editor-efe5ZagF.js} +2 -2
- package/dist/{diagram-PSM6KHXK-2VjPSCDn.js → diagram-PSM6KHXK-H66ATWP2.js} +18 -18
- package/dist/{diagram-QEK2KX5R-CiLmNyta.js → diagram-QEK2KX5R-DItl5Wns.js} +14 -14
- package/dist/{diagram-S2PKOQOG-Zha_1CLx.js → diagram-S2PKOQOG-CtuW_ZuL.js} +14 -14
- package/dist/diff-D6XwL6P8.js +4 -0
- package/dist/{dist-Brkazupz.js → dist--sWVZwjW.js} +1 -1
- package/dist/{dist-FN0ZA_8F.js → dist-21ButRCu.js} +1 -1
- package/dist/{dist-BetEKbPG.js → dist-B8RaFTRF.js} +1 -1
- package/dist/dist-BoHGySTM.js +5 -0
- package/dist/dist-ByAz19Qc.js +5 -0
- package/dist/dist-C93EysN4.js +5 -0
- package/dist/{dist-BHnX0ia_.js → dist-CY-lVor6.js} +1 -1
- package/dist/{dist-YP-G7W0f.js → dist-CYDuv4bR.js} +1 -1
- package/dist/{dist-CMjD5MQb.js → dist-Cfo5EE2t.js} +1 -1
- package/dist/dist-CjivSDvN.js +5 -0
- package/dist/dist-Cqwx-MH7.js +5 -0
- package/dist/{dist-ESg7xyoD.js → dist-D3ZI9nhS.js} +2 -2
- package/dist/{dist-DkC6YEo0.js → dist-DMZNjfX4.js} +1 -1
- package/dist/{dist-ChC1BhqM.js → dist-DbpcoFAV.js} +1 -1
- package/dist/dist-FUNenbiQ.js +5 -0
- package/dist/{dist-BEOU2g1b.js → dist-zhSud5X3.js} +1 -1
- package/dist/{dockerfile-COvlVLcE.js → dockerfile-twL37N91.js} +1 -1
- package/dist/dtd-Bw0lRN0-.js +4 -0
- package/dist/dylan-B55eBHTt.js +4 -0
- package/dist/ecl-C8G4p0wn.js +4 -0
- package/dist/eiffel-BmH46VJl.js +4 -0
- package/dist/elm-DhzeFqkl.js +4 -0
- package/dist/{erDiagram-Q2GNP2WA-biHZS05w.js → erDiagram-Q2GNP2WA--19X2kU5.js} +14 -14
- package/dist/erlang-CY-wdlsU.js +4 -0
- package/dist/{error-banner-DnBPzEWg.js → error-banner-CVkfBUT3.js} +2 -2
- package/dist/{esm-Dd1z1auZ.js → esm-CWp0KQeK.js} +1 -1
- package/dist/{esm-CYEyrE3Y.js → esm-DjNnlmpf.js} +96 -96
- package/dist/{extends-CzJgxo2J.js → extends-vAi97cpa.js} +4 -4
- package/dist/{factor-C2GT7jfQ.js → factor-CajWS6mS.js} +1 -1
- package/dist/factor-DWkgl0xw.js +4 -0
- package/dist/{flowDiagram-NV44I4VS-CWWlUpBR.js → flowDiagram-NV44I4VS-DQmWlo7f.js} +16 -16
- package/dist/{formats-CgaK7Gmx.js → formats-CpgZM9BM.js} +3 -3
- package/dist/forth-Cij_ie2t.js +4 -0
- package/dist/fortran-Br3X9cfm.js +4 -0
- package/dist/{ganttDiagram-JELNMOA3-D7B2c4Z9.js → ganttDiagram-JELNMOA3-BOGXJ8Lk.js} +7 -7
- package/dist/gas-DYsGcMN2.js +4 -0
- package/dist/gherkin-eVgXQ0fQ.js +4 -0
- package/dist/{gitGraph-G5XIXVHT-BdepdFa_.js → gitGraph-G5XIXVHT-DGlbae5m.js} +1 -1
- package/dist/{gitGraphDiagram-V2S2FVAM-CtLvNR1S.js → gitGraphDiagram-V2S2FVAM-DjzxfF0P.js} +14 -14
- package/dist/{glide-data-editor-CvlvtPWJ.js → glide-data-editor-BK9s_dqy.js} +13 -13
- package/dist/groovy-_NFHIXG7.js +4 -0
- package/dist/haskell-D6xNG_bH.js +4 -0
- package/dist/haxe-sU_rzAwn.js +5 -0
- package/dist/{html-to-image-hMMPiNe_.js → html-to-image-DxWM1HVj.js} +2395 -2301
- package/dist/idl-Ds_VbrUx.js +4 -0
- package/dist/{info-VBDWY6EO--JNA2rNu.js → info-VBDWY6EO-D2lvLLw5.js} +1 -1
- package/dist/{infoDiagram-HS3SLOUP-BbZyOxsP.js → infoDiagram-HS3SLOUP-ChNufFsP.js} +12 -12
- package/dist/{input-BAOe64zx.js → input-Cc1Vvw9A.js} +6 -6
- package/dist/javascript-CztfIl0i.js +4 -0
- package/dist/{journeyDiagram-XKPGCS4Q-BU2mjjzl.js → journeyDiagram-XKPGCS4Q-BO_O4Ij1.js} +6 -6
- package/dist/julia-DDv40QMV.js +4 -0
- package/dist/{kanban-definition-3W4ZIXB7-BlmczUuw.js → kanban-definition-3W4ZIXB7-CPpiiiWk.js} +11 -11
- package/dist/{katex-qPqrBHZ8.js → katex-9-9QRhxz.js} +1 -1
- package/dist/{label-BCWi-Oqu.js → label-BLqV33b1.js} +2 -2
- package/dist/{line-BWRi3U3S.js → line-C5s_12ee.js} +3 -3
- package/dist/{linear-DnHwODZa.js → linear-2NnK4cxi.js} +2 -2
- package/dist/livescript-BEOngLLc.js +4 -0
- package/dist/{loader-BvW0-YWZ.js → loader-Dr8Qem8p.js} +1 -1
- package/dist/lua-9-7BhQ4Y.js +4 -0
- package/dist/main.js +1175 -1142
- package/dist/mathematica-DDa0Pfxm.js +4 -0
- package/dist/mbox-iO03mmoE.js +4 -0
- package/dist/{mermaid-4DMBBIKO-CG1ECj5W.js → mermaid-4DMBBIKO-B7VQMwJx.js} +1 -1
- package/dist/{mermaid-CEbzCxCc.js → mermaid-DO-Daq7u.js} +46 -46
- package/dist/{mermaid-parser.core-CleJseNW.js → mermaid-parser.core-DreccfmS.js} +7 -7
- package/dist/{mhchem-BwoRNwg_.js → mhchem-yiCCuiEF.js} +1 -1
- package/dist/{mindmap-definition-VGOIOE7T-CcSYqYP9.js → mindmap-definition-VGOIOE7T-CC1_Vl0f.js} +13 -13
- package/dist/mirc-C9z5LT4X.js +4 -0
- package/dist/mllike-jGJbdm_C.js +6 -0
- package/dist/modelica-DzF7oIEL.js +4 -0
- package/dist/mscgen-DB-u125o.js +6 -0
- package/dist/mumps-CRTFHhzh.js +4 -0
- package/dist/nsis-C4NPTuox.js +4 -0
- package/dist/{nsis-B5K1qoyo.js → nsis-ClF3r5Tr.js} +1 -1
- package/dist/ntriples-BCOoGph1.js +4 -0
- package/dist/{number-overlay-editor-_GnlYFHC.js → number-overlay-editor-CpKi64Fy.js} +1 -1
- package/dist/octave-DTwNlazz.js +4 -0
- package/dist/{ordinal-2jIulmcR.js → ordinal-B43ZeR68.js} +1 -1
- package/dist/oz-DD38AzSz.js +4 -0
- package/dist/{packet-DYOGHKS2-CBxXGWNr.js → packet-DYOGHKS2-CmWtF3uO.js} +1 -1
- package/dist/pascal-BohSp9jV.js +4 -0
- package/dist/perl-f5OutoPM.js +4 -0
- package/dist/{pie-VRWISCQL-Bmdnqjip.js → pie-VRWISCQL-B6u8vus8.js} +1 -1
- package/dist/{pieDiagram-ADFJNKIX-DNyLF5H2.js → pieDiagram-ADFJNKIX-Di34MOFQ.js} +19 -19
- package/dist/pig-Dv7wSmHb.js +4 -0
- package/dist/powershell-rYgjKB39.js +4 -0
- package/dist/{process-output-Bza_GK7Q.js → process-output-DBYxXdrN.js} +30 -25
- package/dist/properties-BFUNLRDN.js +4 -0
- package/dist/protobuf-B9QJQPPv.js +4 -0
- package/dist/{pug-tjbzJCFk.js → pug-B_rby2yb.js} +1 -1
- package/dist/pug-DzvWpaMC.js +4 -0
- package/dist/puppet-B_K-n_xK.js +4 -0
- package/dist/python-CAiFcaA2.js +4 -0
- package/dist/q-cLeFIBLK.js +4 -0
- package/dist/{quadrantDiagram-AYHSOK5B-rXwjifrj.js → quadrantDiagram-AYHSOK5B-B9kVk1ny.js} +3 -3
- package/dist/r-04Y-Wco3.js +4 -0
- package/dist/{radar-ZZBFDIW7-BmCWDffL.js → radar-ZZBFDIW7-XAmXSa8s.js} +1 -1
- package/dist/{react-vega-B-rkEqtS.js → react-vega-Cavbrg4l.js} +1 -1
- package/dist/{react-vega-k9ODWPlI.js → react-vega-Dh6-UKKe.js} +13 -13
- package/dist/{requirementDiagram-UZGBJVZJ-DBdrMVbs.js → requirementDiagram-UZGBJVZJ-BxGfGYEx.js} +13 -13
- package/dist/{reveal-component-h0_DDpxE.js → reveal-component-CSA3g_zD.js} +10 -10
- package/dist/rpm-FUdrIia9.js +5 -0
- package/dist/ruby-DMjFXuEW.js +4 -0
- package/dist/{sankeyDiagram-TZEHDZUN-CxmzalGv.js → sankeyDiagram-TZEHDZUN-D09PBJ-n.js} +4 -4
- package/dist/sas-DzHZxjXK.js +4 -0
- package/dist/scheme-DxHd_Rb9.js +4 -0
- package/dist/semaphore-CNDGTzkX.js +46 -0
- package/dist/{sequenceDiagram-WL72ISMW-CVCDsJ9h.js → sequenceDiagram-WL72ISMW-t_Dpemj0.js} +7 -7
- package/dist/shell-C8Kwypgf.js +4 -0
- package/dist/sieve-DdyqOKXZ.js +4 -0
- package/dist/smalltalk-pB7X1D9y.js +4 -0
- package/dist/sparql-NhBO6oOa.js +4 -0
- package/dist/{spec-DSIuqd3f.js → spec-hVaaZsY5.js} +4 -4
- package/dist/{src-BY0BGg6V.js → src-Bf2iLOlr.js} +1 -1
- package/dist/{stateDiagram-FKZM4ZOC-D_2djEhW.js → stateDiagram-FKZM4ZOC-B18gTP_j.js} +16 -16
- package/dist/stateDiagram-v2-4FDKWEC3-B6e_t14A.js +29 -0
- package/dist/{step-DGAGWg3y.js → step-CWipAYTY.js} +1 -1
- package/dist/{strings-B_FOH6eV.js → strings-BiIhGaI8.js} +4 -4
- package/dist/style.css +1 -1
- package/dist/stylus-SfWSnzPv.js +4 -0
- package/dist/swift-jRPdq2zR.js +4 -0
- package/dist/{swiper-component-KkEVUDd3.js → swiper-component-DlD2GU2g.js} +2 -2
- package/dist/tcl-_hpTHGX3.js +4 -0
- package/dist/textile-C9h8slqH.js +4 -0
- package/dist/{time-CMdrp3hw.js → time-C1SGcFMH.js} +2 -2
- package/dist/{timeline-definition-IT6M3QCI-E4NzxCs3.js → timeline-definition-IT6M3QCI-DJnh1ks5.js} +3 -3
- package/dist/{toDate-CHtl9vts.js → toDate-CJWlVNGD.js} +3 -3
- package/dist/toml-DWvtinD4.js +4 -0
- package/dist/{tooltip-B0mtKTXm.js → tooltip-DRaMBu06.js} +3 -3
- package/dist/{treemap-GDKQZRPO-CoKHPxa7.js → treemap-GDKQZRPO-Du95DV6u.js} +1 -1
- package/dist/troff-Dwo_A0y7.js +4 -0
- package/dist/ttcn-V--CPFKq.js +4 -0
- package/dist/ttcn-cfg-CPSMchTG.js +4 -0
- package/dist/turtle-B4rPGBWu.js +4 -0
- package/dist/{types-DBtDeUKD.js → types-Dzuoc3LN.js} +1 -1
- package/dist/{useAsyncData-B6hCGywC.js → useAsyncData-C56Khv_R.js} +1 -1
- package/dist/{useDateFormatter-B3mCQMP3.js → useDateFormatter-B_9k85Ex.js} +2 -2
- package/dist/{useDeepCompareMemoize-CmwDuYUH.js → useDeepCompareMemoize-Dt98v2ua.js} +1 -1
- package/dist/{useIframeCapabilities-DbdLoEDm.js → useIframeCapabilities-BkYHTrss.js} +1 -1
- package/dist/{useLifecycle-CjMjllqy.js → useLifecycle-BF6-z62y.js} +3 -3
- package/dist/{useTheme-CByZUW0p.js → useTheme-DykuNHR2.js} +2 -2
- package/dist/vb-DaMBBd4j.js +4 -0
- package/dist/vbscript-BMJQqcE2.js +4 -0
- package/dist/{vega-component-CC8TqWWV.js → vega-component-BtvQ-Kc4.js} +26 -24
- package/dist/velocity-CGq2QRq2.js +4 -0
- package/dist/verilog-CUNo8F5u.js +4 -0
- package/dist/vhdl-CCzA0msW.js +4 -0
- package/dist/webidl-CqIMDIBL.js +4 -0
- package/dist/xquery-XC5Kbr-1.js +4 -0
- package/dist/{xychartDiagram-PRI3JC2R-CuxTvjw5.js → xychartDiagram-PRI3JC2R-Dk2d_bX0.js} +10 -10
- package/dist/yacas-CGOv7Dzy.js +4 -0
- package/dist/z80-CXhVmi-f.js +4 -0
- package/dist/{zod-BxdsqRPd.js → zod-BWkcDORu.js} +1 -1
- package/package.json +3 -3
- package/src/components/chat/chat-components.tsx +47 -0
- package/src/components/chat/chat-display.tsx +41 -7
- package/src/components/chat/chat-panel.tsx +37 -10
- package/src/components/chat/chat-utils.ts +42 -20
- package/src/components/chat/reasoning-accordion.tsx +14 -3
- package/src/components/chat/tool-call/shared.ts +13 -0
- package/src/components/chat/tool-call/tool-approval-card.tsx +62 -0
- package/src/components/chat/tool-call/tool-args.tsx +26 -0
- package/src/components/chat/tool-call/tool-call-view.tsx +99 -0
- package/src/components/chat/tool-call/tool-error-card.tsx +81 -0
- package/src/components/chat/tool-call/tool-history-row.tsx +153 -0
- package/src/components/chat/tool-call/tool-result.tsx +101 -0
- package/src/components/data-table/TableTopBar.tsx +5 -1
- package/src/components/data-table/data-table.tsx +5 -0
- package/src/components/data-table/download-policy/atoms.ts +10 -0
- package/src/components/data-table/export-actions.tsx +31 -4
- package/src/components/editor/actions/pair-with-agent-modal.tsx +140 -49
- package/src/components/editor/actions/useNotebookActions.tsx +3 -1
- package/src/components/editor/app-container.tsx +7 -1
- package/src/components/editor/chrome/panels/context-aware-panel/context-aware-panel.tsx +10 -2
- package/src/components/editor/chrome/wrapper/app-chrome.tsx +1 -0
- package/src/components/editor/chrome/wrapper/footer.tsx +4 -1
- package/src/components/editor/chrome/wrapper/panels.tsx +4 -1
- package/src/components/editor/chrome/wrapper/sidebar.tsx +4 -1
- package/src/components/editor/controls/Controls.tsx +11 -3
- package/src/components/editor/file-tree/requesting-tree.tsx +27 -25
- package/src/components/editor/file-tree/upload.tsx +23 -24
- package/src/components/editor/header/__tests__/status.test.tsx +108 -0
- package/src/components/editor/header/status.tsx +44 -10
- package/src/components/editor/navigation/__tests__/clipboard.test.ts +106 -0
- package/src/components/editor/navigation/__tests__/navigation.test.ts +70 -0
- package/src/components/editor/navigation/clipboard.ts +99 -25
- package/src/components/editor/navigation/navigation.ts +15 -1
- package/src/components/editor/notebook-cell.tsx +3 -0
- package/src/components/home/components.tsx +6 -0
- package/src/components/pages/run-page.tsx +4 -1
- package/src/core/ai/tools/__tests__/registry.test.ts +10 -12
- package/src/core/ai/tools/registry.ts +9 -5
- package/src/core/cells/__tests__/cells.test.ts +187 -0
- package/src/core/cells/__tests__/pending-cut-service.test.tsx +123 -0
- package/src/core/cells/cells.ts +102 -17
- package/src/core/cells/document-changes.ts +6 -1
- package/src/core/cells/pending-cut-service.ts +55 -0
- package/src/core/cells/utils.ts +11 -0
- package/src/core/codemirror/cells/extensions.ts +10 -0
- package/src/core/codemirror/markdown/__tests__/commands.test.ts +3 -3
- package/src/core/codemirror/markdown/commands.ts +1 -2
- package/src/core/edit-app.tsx +2 -1
- package/src/core/hotkeys/hotkeys.ts +5 -0
- package/src/core/islands/worker/worker.tsx +3 -2
- package/src/core/network/requests-network.ts +21 -3
- package/src/core/network/types.ts +12 -1
- package/src/core/run-app.tsx +2 -1
- package/src/core/wasm/__tests__/utils.test.ts +34 -0
- package/src/core/wasm/bridge.ts +14 -1
- package/src/core/wasm/utils.ts +14 -0
- package/src/core/wasm/worker/bootstrap.ts +3 -2
- package/src/core/wasm/worker/worker.ts +3 -2
- package/src/core/websocket/__tests__/useMarimoKernelConnection.hook.test.tsx +155 -0
- package/src/core/websocket/__tests__/useMarimoKernelConnection.test.ts +137 -0
- package/src/core/websocket/transports/basic.ts +2 -0
- package/src/core/websocket/transports/transport.ts +1 -0
- package/src/core/websocket/useMarimoKernelConnection.tsx +130 -55
- package/src/core/websocket/useWebSocket.tsx +5 -2
- package/src/css/app/Cell.css +10 -0
- package/src/plugins/impl/DataTablePlugin.tsx +12 -0
- package/src/plugins/impl/TabsPlugin.tsx +35 -7
- package/src/plugins/impl/__tests__/TabsPlugin.test.tsx +154 -0
- package/src/plugins/impl/data-frames/DataFramePlugin.tsx +6 -0
- package/src/plugins/impl/vega/resolve-data.ts +8 -1
- package/src/utils/__tests__/id-tree.test.ts +71 -0
- package/src/utils/__tests__/semaphore.test.ts +218 -0
- package/src/utils/fileToBase64.ts +8 -7
- package/src/utils/id-tree.tsx +89 -0
- package/src/utils/semaphore.ts +88 -0
- package/dist/apl-BKoVld9y.js +0 -4
- package/dist/asciiarmor-DQrKIjoo.js +0 -4
- package/dist/asn1-BZvnj0dq.js +0 -4
- package/dist/assets/__vite-browser-external-rrUYDKRl.js +0 -1
- package/dist/assets/worker-Bfy15ViQ.js +0 -73
- package/dist/brainfuck-D558nlUv.js +0 -4
- package/dist/classDiagram-2ON5EDUG-CBHMR6ZU.js +0 -30
- package/dist/classDiagram-v2-WZHVMYZB-BsUtUGM_.js +0 -30
- package/dist/clojure-Cq8mTSrE.js +0 -4
- package/dist/cmake-D8HCovWK.js +0 -4
- package/dist/cobol-UolN-9iU.js +0 -4
- package/dist/coffeescript-VdNuWrt5.js +0 -4
- package/dist/commonlisp-ALX7fpDc.js +0 -4
- package/dist/crystal-PbyO9Q_s.js +0 -4
- package/dist/css-DFklJkr_.js +0 -4
- package/dist/cypher-BifNeYlv.js +0 -4
- package/dist/d-BA-JP4PJ.js +0 -4
- package/dist/diff-CtkDpav4.js +0 -4
- package/dist/dist-BuBwsFva.js +0 -5
- package/dist/dist-BzmEQ9u7.js +0 -5
- package/dist/dist-Cih01ssx.js +0 -5
- package/dist/dist-CqfONiY9.js +0 -5
- package/dist/dist-D0iD0Fi9.js +0 -5
- package/dist/dist-DtNLXm8d.js +0 -5
- package/dist/dtd-DW3_UFEG.js +0 -4
- package/dist/dylan-pDhodO2N.js +0 -4
- package/dist/ecl-BJT8-YD7.js +0 -4
- package/dist/eiffel-Dmns-9vS.js +0 -4
- package/dist/elm-Da4sO4Bz.js +0 -4
- package/dist/erlang-C-zBsDi7.js +0 -4
- package/dist/factor-4xPWlWB5.js +0 -4
- package/dist/forth-l-c75zSd.js +0 -4
- package/dist/fortran-DIujSODW.js +0 -4
- package/dist/gas-CXnG5g_b.js +0 -4
- package/dist/gherkin-VPeqd4-X.js +0 -4
- package/dist/groovy-CphhZQgg.js +0 -4
- package/dist/haskell-CCvlS5Iq.js +0 -4
- package/dist/haxe-C_bi66fP.js +0 -5
- package/dist/idl-1DcP4Dm8.js +0 -4
- package/dist/javascript-DUIGhBvO.js +0 -4
- package/dist/julia-Cs2G4PQi.js +0 -4
- package/dist/livescript-DMtVFaAN.js +0 -4
- package/dist/lua-BAoLtdJg.js +0 -4
- package/dist/mathematica-C_NoFtbo.js +0 -4
- package/dist/mbox-DcFJFYrH.js +0 -4
- package/dist/mirc-71dccf_u.js +0 -4
- package/dist/mllike-CWcOFVDq.js +0 -6
- package/dist/modelica-Ape2VXxx.js +0 -4
- package/dist/mscgen-Cc6TwbSN.js +0 -6
- package/dist/mumps-h-ZbdkJ9.js +0 -4
- package/dist/nsis-C0p3m7JW.js +0 -4
- package/dist/ntriples-c9lEeT5w.js +0 -4
- package/dist/octave-DzEgB_74.js +0 -4
- package/dist/oz-CAxvHkyQ.js +0 -4
- package/dist/pascal-BJzu1sgP.js +0 -4
- package/dist/perl--IrOzZ2Z.js +0 -4
- package/dist/pig-CiBKKNhC.js +0 -4
- package/dist/powershell-KY0j6Qop.js +0 -4
- package/dist/properties-BW8q3ziV.js +0 -4
- package/dist/protobuf-BGaeuTGV.js +0 -4
- package/dist/pug-DjOKK-4J.js +0 -4
- package/dist/puppet-DWm2o6zX.js +0 -4
- package/dist/python-Bp2gezZy.js +0 -4
- package/dist/q-DljPshos.js +0 -4
- package/dist/r-BajPMnEu.js +0 -4
- package/dist/rpm-BKx-ZZ62.js +0 -5
- package/dist/ruby-DJq_HNKc.js +0 -4
- package/dist/sas-WANvpcOU.js +0 -4
- package/dist/scheme-CliBbhGF.js +0 -4
- package/dist/shell-BwhrNUvM.js +0 -4
- package/dist/sieve-BIVePvMp.js +0 -4
- package/dist/smalltalk-D6G48JmY.js +0 -4
- package/dist/sparql-jjc3BmEP.js +0 -4
- package/dist/stateDiagram-v2-4FDKWEC3-Cv9Av10H.js +0 -29
- package/dist/stylus-WPBPQ4PE.js +0 -4
- package/dist/swift-O1Qy6iCm.js +0 -4
- package/dist/tcl-BAFdhvsi.js +0 -4
- package/dist/textile-DFuzhNLG.js +0 -4
- package/dist/toml-DRSTeely.js +0 -4
- package/dist/troff-B_ZjwBW0.js +0 -4
- package/dist/ttcn-CAyiB3ic.js +0 -4
- package/dist/ttcn-cfg-BS5_BGBJ.js +0 -4
- package/dist/turtle-CUBEDy3E.js +0 -4
- package/dist/vb-DY9S6-U2.js +0 -4
- package/dist/vbscript-gaHC39Jq.js +0 -4
- package/dist/velocity-TfCOtJZ_.js +0 -4
- package/dist/verilog-c2JOX8mv.js +0 -4
- package/dist/vhdl-dHBirRiO.js +0 -4
- package/dist/webidl-Bauj-i07.js +0 -4
- package/dist/xquery-CtaEAOt8.js +0 -4
- package/dist/yacas-BZ85agQP.js +0 -4
- package/dist/z80-hCgR-L4U.js +0 -4
- package/src/components/chat/tool-call-accordion.tsx +0 -247
- /package/dist/{ImageComparisonComponent-DaocPIse.js → ImageComparisonComponent-CNHIsPDj.js} +0 -0
- /package/dist/{Plot-PIeIvFnD.js → Plot-4wn-lMVn.js} +0 -0
- /package/dist/{apl-Dt8GMXYg.js → apl-BCgCq9iM.js} +0 -0
- /package/dist/{array-B-MVxRIF.js → array-tvvEqPy7.js} +0 -0
- /package/dist/{asciiarmor-CitDQ85h.js → asciiarmor-BtqU-KJQ.js} +0 -0
- /package/dist/{asn1-abrf9SMK.js → asn1-Dmb-dTMx.js} +0 -0
- /package/dist/{asterisk-BUZwqih-.js → asterisk-DaVJJDnV.js} +0 -0
- /package/dist/{brainfuck-BL-Boof0.js → brainfuck-C1HoZKlE.js} +0 -0
- /package/dist/{chunk-4F5CHEZ2-C6tO9vjs.js → chunk-4F5CHEZ2-BZq7Kom7.js} +0 -0
- /package/dist/{chunk-B2363JML-Ds8wZXyP.js → chunk-B2363JML-D9-XOau1.js} +0 -0
- /package/dist/{chunk-DR5Q36YT-CP69aZS_.js → chunk-DR5Q36YT-BflwErH1.js} +0 -0
- /package/dist/{chunk-FRFDVMJY-BgQv1HBE.js → chunk-FRFDVMJY-BSBUAX7r.js} +0 -0
- /package/dist/{chunk-PL6DKKU2-DHfTUHy8.js → chunk-PL6DKKU2-B0MTXvyc.js} +0 -0
- /package/dist/{chunk-SJTYNZTY-Diciw4sx.js → chunk-SJTYNZTY-CEG4F0pB.js} +0 -0
- /package/dist/{chunk-TQ3KTPDO-CQfP9npd.js → chunk-TQ3KTPDO-DiCtqVSi.js} +0 -0
- /package/dist/{chunk-UMXZTB3W-MSKeGL7W.js → chunk-UMXZTB3W-97iS1iEl.js} +0 -0
- /package/dist/{click-outside-container-BZgN7xS_.js → click-outside-container-BDd67_1U.js} +0 -0
- /package/dist/{clike-RWg7anhx.js → clike-CdT0yHjt.js} +0 -0
- /package/dist/{clojure-DaojKHow.js → clojure-CdyrCpUv.js} +0 -0
- /package/dist/{cmake-DN-_v0XE.js → cmake-BFlPxym7.js} +0 -0
- /package/dist/{cobol-C3VpMyux.js → cobol-CcJXewp8.js} +0 -0
- /package/dist/{coffeescript-DIkz3Tbt.js → coffeescript-DnKuIKRo.js} +0 -0
- /package/dist/{colors-Cn2p_FA3.js → colors-CQAOa8cK.js} +0 -0
- /package/dist/{common-keywords-hbLeU7VU.js → common-keywords-FBrXPTcz.js} +0 -0
- /package/dist/{commonlisp-CB1boOiP.js → commonlisp-B-kok83Z.js} +0 -0
- /package/dist/{crystal-DI2oCml6.js → crystal-FYRYjI1I.js} +0 -0
- /package/dist/{css-BdEVwQDV.js → css-B45lc2V3.js} +0 -0
- /package/dist/{cypher-BNHToqxU.js → cypher-DZMLyVY_.js} +0 -0
- /package/dist/{cytoscape.esm-WbbDoCfu.js → cytoscape.esm-ayF70frT.js} +0 -0
- /package/dist/{d-D7we7I1b.js → d-x-VVT4o9.js} +0 -0
- /package/dist/{diff-Cia6fzjN.js → diff-Dxe2mpXk.js} +0 -0
- /package/dist/{dist-BK-3fF4P.js → dist-B4LJpMEg.js} +0 -0
- /package/dist/{dist-CxdUraQr.js → dist-B507mf_I.js} +0 -0
- /package/dist/{dist-C89sHDXk.js → dist-BGdYVvOu.js} +0 -0
- /package/dist/{dist-DquyVv5H.js → dist-BNyrZfqT.js} +0 -0
- /package/dist/{dist-Zn0KNbo9.js → dist-Bc5pmZIw.js} +0 -0
- /package/dist/{dist-C-J0pt5p.js → dist-BvCfQQQE.js} +0 -0
- /package/dist/{dist-D9r7Cmw7.js → dist-C2ej4eOH.js} +0 -0
- /package/dist/{dist-HVuryI1a.js → dist-C34oIrQ9.js} +0 -0
- /package/dist/{dist-CGLzXdrt.js → dist-CDFZi-QD.js} +0 -0
- /package/dist/{dist-C9fmTOin.js → dist-CYEylvZA.js} +0 -0
- /package/dist/{dist-DadjmS-4.js → dist-DJ6zJQZ4.js} +0 -0
- /package/dist/{dist-CtCY55Jf.js → dist-Dh3wkoyH.js} +0 -0
- /package/dist/{dist-C474qFoq.js → dist-Dhk6FMb0.js} +0 -0
- /package/dist/{dist-DZjQ_MBo.js → dist-KnujRhFL.js} +0 -0
- /package/dist/{dist-CinA9Enb.js → dist-WdPUFc56.js} +0 -0
- /package/dist/{dist-DBLeRrPp.js → dist-t_qL7eB8.js} +0 -0
- /package/dist/{dist-CyFFzJTb.js → dist-usPCDYx8.js} +0 -0
- /package/dist/{dtd-H4Hubdwp.js → dtd-C9VM_Wfu.js} +0 -0
- /package/dist/{duckdb-keywords-CZ_ZTscu.js → duckdb-keywords-CvJhR_Yd.js} +0 -0
- /package/dist/{dylan-fVO6rnq3.js → dylan-DTSnEIFO.js} +0 -0
- /package/dist/{ebnf-WEXPLEWb.js → ebnf-2D4Ctp3y.js} +0 -0
- /package/dist/{ecl-B94VPjNR.js → ecl-N04ptnRK.js} +0 -0
- /package/dist/{eiffel-C_R6TusS.js → eiffel-Dd8rpqr_.js} +0 -0
- /package/dist/{elm-DzCHbO2g.js → elm-GT2E866W.js} +0 -0
- /package/dist/{erlang-BGNkx6JU.js → erlang-Cf0Bp5pY.js} +0 -0
- /package/dist/{esm-Bb_hbWan.js → esm-BaaaPNGl.js} +0 -0
- /package/dist/{fcl-B_Gv5Jfx.js → fcl-Ccj8Z5Xd.js} +0 -0
- /package/dist/{forth-Bybw0cJ7.js → forth-wd_XzGTg.js} +0 -0
- /package/dist/{fortran-C6PoCLkI.js → fortran-DcwUTZFe.js} +0 -0
- /package/dist/{gas-BBlhenj4.js → gas-DeALIER3.js} +0 -0
- /package/dist/{gherkin-NXtNG85X.js → gherkin-CKTqaJNX.js} +0 -0
- /package/dist/{groovy-BoFYK9xM.js → groovy-Bwdp_d8D.js} +0 -0
- /package/dist/{haskell-BtBdvQ1n.js → haskell-DCdCcPLK.js} +0 -0
- /package/dist/{haxe-D--o6dr0.js → haxe-DAyktQWJ.js} +0 -0
- /package/dist/{http-Dc2fv19V.js → http-_DVAYWoR.js} +0 -0
- /package/dist/{idl-AqTq5l7e.js → idl-CBuZiRYu.js} +0 -0
- /package/dist/{init-D-g0ONX1.js → init-uv0kkh4g.js} +0 -0
- /package/dist/{javascript-DvwNVye9.js → javascript-DzigE11c.js} +0 -0
- /package/dist/{julia-DoKiagZC.js → julia-PwfB-0Cm.js} +0 -0
- /package/dist/{katex-B7pMJpE0.js → katex-C_XRmjAP.js} +0 -0
- /package/dist/{livescript-DxBZMiWB.js → livescript-BJLz1EbT.js} +0 -0
- /package/dist/{lua-DmS_0NTu.js → lua-ZC-XC2jf.js} +0 -0
- /package/dist/{math-BYK36kWZ.js → math-DFcdCCU8.js} +0 -0
- /package/dist/{mathematica-ChlDFeIC.js → mathematica-DCYMx6qB.js} +0 -0
- /package/dist/{mbox-CguZuODr.js → mbox-OxMK_9XI.js} +0 -0
- /package/dist/{mirc-CFtY8dqz.js → mirc-nJVyhA0H.js} +0 -0
- /package/dist/{mllike-C0EJrEOk.js → mllike-DRO89bsU.js} +0 -0
- /package/dist/{modelica-C1kO1nfS.js → modelica-Don3E6ZD.js} +0 -0
- /package/dist/{mscgen-DEYdr7AY.js → mscgen-DJfqD3bN.js} +0 -0
- /package/dist/{mumps-B3NVJs2V.js → mumps-SjGTvDYL.js} +0 -0
- /package/dist/{nginx-ComVAAGN.js → nginx-DasThI7R.js} +0 -0
- /package/dist/{node-sql-parser-DNGGJ-Rw.js → node-sql-parser-B8nBD36q.js} +0 -0
- /package/dist/{ntriples-DHol9X9H.js → ntriples-CNBKRl3I.js} +0 -0
- /package/dist/{octave-CYGz0bfo.js → octave-DdeVHNlx.js} +0 -0
- /package/dist/{oz-kPxb2ni5.js → oz-CcKSoNvN.js} +0 -0
- /package/dist/{pascal-bZ0yrJKy.js → pascal-6leftwNj.js} +0 -0
- /package/dist/{path-Du6n3sOU.js → path-BGaWgPKg.js} +0 -0
- /package/dist/{perl-z4hvqyqz.js → perl-BhJIwWzN.js} +0 -0
- /package/dist/{pig-DZO8QDF9.js → pig-r-xDHqRf.js} +0 -0
- /package/dist/{powershell-BSuaDQEC.js → powershell-D-BELeNi.js} +0 -0
- /package/dist/{properties-BXhGLlIx.js → properties-CnuDhbll.js} +0 -0
- /package/dist/{protobuf-DM6iybWV.js → protobuf-CO8RBhvX.js} +0 -0
- /package/dist/{puppet-Bn05sQT8.js → puppet-NmXHjLy8.js} +0 -0
- /package/dist/{python-Cvnhm0g7.js → python-DAQXi720.js} +0 -0
- /package/dist/{q-B9V8hzex.js → q-DlikXfV0.js} +0 -0
- /package/dist/{r-Cf0gFqmq.js → r-CuohilwT.js} +0 -0
- /package/dist/{rpm-D-LMkTV1.js → rpm-0Pjwp0Pb.js} +0 -0
- /package/dist/{ruby-DeuPikpK.js → ruby-Dq8NJTDG.js} +0 -0
- /package/dist/{sas-C9tjgAo9.js → sas-CuwonyVP.js} +0 -0
- /package/dist/{scheme-D1_bUF0G.js → scheme-CYU-RRIf.js} +0 -0
- /package/dist/{shell-CJBmnks3.js → shell-COPmX2qE.js} +0 -0
- /package/dist/{sieve-1fSV75CF.js → sieve-B_3zyLne.js} +0 -0
- /package/dist/{simple-mode-B90Wdavj.js → simple-mode-DSBniks8.js} +0 -0
- /package/dist/{smalltalk-sZNPD0HO.js → smalltalk-DRft7iPv.js} +0 -0
- /package/dist/{solr-DTkyqJ-Z.js → solr-RZ9uTl59.js} +0 -0
- /package/dist/{sparql-oHc1nm77.js → sparql-CN6qj55H.js} +0 -0
- /package/dist/{spreadsheet-CER0raqY.js → spreadsheet-BNNUNXA2.js} +0 -0
- /package/dist/{sql-ByOoEONQ.js → sql-B4x8IkwU.js} +0 -0
- /package/dist/{stylus-KzkX6zRB.js → stylus-Bn_ZjOQ3.js} +0 -0
- /package/dist/{swift-DqVxZvKo.js → swift-BLUJhMbz.js} +0 -0
- /package/dist/{tcl-BtWSwXfA.js → tcl-C86fxecl.js} +0 -0
- /package/dist/{textile-CWDbn9Ql.js → textile-DmHh2rsK.js} +0 -0
- /package/dist/{tiddlywiki-Cr9xyOY1.js → tiddlywiki-DI0mF2WJ.js} +0 -0
- /package/dist/{tiki-D5JONyfZ.js → tiki-2HU6XLLn.js} +0 -0
- /package/dist/{timer-D7JVdX9U.js → timer-YZl28NYN.js} +0 -0
- /package/dist/{toml-BfehlgmL.js → toml-GWANRNAD.js} +0 -0
- /package/dist/{treemap-qFGzn7xk.js → treemap-D-ka1hvx.js} +0 -0
- /package/dist/{troff-BZBk6AAu.js → troff-BHTsomIy.js} +0 -0
- /package/dist/{ttcn-DVwvXg0_.js → ttcn-DQuhn5Mn.js} +0 -0
- /package/dist/{ttcn-cfg-gjbVLf1L.js → ttcn-cfg-HjFYtdB-.js} +0 -0
- /package/dist/{turtle-CgxKXorV.js → turtle-nCay33Nv.js} +0 -0
- /package/dist/{vb-B9kSwTdM.js → vb-BG-XlqqJ.js} +0 -0
- /package/dist/{vbscript-DrUKSCdb.js → vbscript-B6vyW0-D.js} +0 -0
- /package/dist/{velocity-AlMYTnMy.js → velocity-CWegueqO.js} +0 -0
- /package/dist/{verilog-DLUaM05j.js → verilog-CzSQm4cG.js} +0 -0
- /package/dist/{vhdl-DUJOtSmO.js → vhdl-DqnNVL7r.js} +0 -0
- /package/dist/{webidl-CQp4aHk_.js → webidl-DXEUpDWH.js} +0 -0
- /package/dist/{xquery-IxkjlwOD.js → xquery-Ba_NB5bD.js} +0 -0
- /package/dist/{yacas-Bnctn5w8.js → yacas-HKQU6hyk.js} +0 -0
- /package/dist/{z80-DrFwhx53.js → z80-CXkHXLdj.js} +0 -0
package/src/core/cells/cells.ts
CHANGED
|
@@ -23,6 +23,7 @@ import {
|
|
|
23
23
|
splitEditor,
|
|
24
24
|
updateEditorCodeFromPython,
|
|
25
25
|
} from "../codemirror/language/utils";
|
|
26
|
+
import type { SerializedEditorState } from "../codemirror/types";
|
|
26
27
|
import { findCollapseRange, mergeOutlines } from "../dom/outline";
|
|
27
28
|
import type { CellMessage } from "../kernel/messages";
|
|
28
29
|
import { isErrorMime } from "../mime";
|
|
@@ -50,11 +51,36 @@ import {
|
|
|
50
51
|
canUndoDeletes,
|
|
51
52
|
disabledCellIds,
|
|
52
53
|
enabledCellIds,
|
|
54
|
+
getUndoLabel,
|
|
53
55
|
notebookIsRunning,
|
|
54
56
|
notebookNeedsRun,
|
|
55
57
|
notebookQueueOrRunningCount,
|
|
56
58
|
} from "./utils";
|
|
57
59
|
|
|
60
|
+
/**
|
|
61
|
+
* History entry for undoing a cell deletion.
|
|
62
|
+
*/
|
|
63
|
+
export interface UndoDeleteEntry {
|
|
64
|
+
type: "delete";
|
|
65
|
+
name: string;
|
|
66
|
+
serializedEditorState: SerializedEditorState;
|
|
67
|
+
column: CellColumnId;
|
|
68
|
+
index: CellIndex;
|
|
69
|
+
isSetupCell: boolean;
|
|
70
|
+
config: CellConfig;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
/**
|
|
74
|
+
* History entry for undoing a cut-paste (move).
|
|
75
|
+
*/
|
|
76
|
+
export interface UndoMoveEntry {
|
|
77
|
+
type: "move";
|
|
78
|
+
cellIds: CellId[];
|
|
79
|
+
placements: Array<{ columnId: CellColumnId; index: CellIndex }>;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
export type HistoryEntry = UndoDeleteEntry | UndoMoveEntry;
|
|
83
|
+
|
|
58
84
|
/**
|
|
59
85
|
* The state of the notebook.
|
|
60
86
|
*/
|
|
@@ -76,19 +102,9 @@ export interface NotebookState {
|
|
|
76
102
|
*/
|
|
77
103
|
cellHandles: Record<CellId, React.RefObject<CellHandle | null>>;
|
|
78
104
|
/**
|
|
79
|
-
*
|
|
80
|
-
*
|
|
81
|
-
* (CodeMirror types the serialized config as any.)
|
|
105
|
+
* Undo stack: deleted cells and cut-paste moves, in chronological order.
|
|
82
106
|
*/
|
|
83
|
-
history:
|
|
84
|
-
name: string;
|
|
85
|
-
// oxlint-disable-next-line typescript/no-explicit-any
|
|
86
|
-
serializedEditorState: any;
|
|
87
|
-
column: CellColumnId;
|
|
88
|
-
index: CellIndex;
|
|
89
|
-
isSetupCell: boolean;
|
|
90
|
-
config: CellConfig;
|
|
91
|
-
}[];
|
|
107
|
+
history: HistoryEntry[];
|
|
92
108
|
/**
|
|
93
109
|
* Key of cell to scroll to; typically set by actions that re-order the cell
|
|
94
110
|
* array. Call the SCROLL_TO_TARGET action to scroll to the specified cell
|
|
@@ -158,6 +174,10 @@ export interface CreateNewCellAction {
|
|
|
158
174
|
before: boolean;
|
|
159
175
|
/** Initial code content for the new cell */
|
|
160
176
|
code?: string;
|
|
177
|
+
/** Optional name for the new cell */
|
|
178
|
+
name?: string;
|
|
179
|
+
/** Optional cell configuration */
|
|
180
|
+
config?: CellConfig;
|
|
161
181
|
/** The last executed code for the new cell */
|
|
162
182
|
lastCodeRun?: string;
|
|
163
183
|
/** Timestamp of the last execution */
|
|
@@ -187,11 +207,13 @@ const {
|
|
|
187
207
|
cellId,
|
|
188
208
|
before,
|
|
189
209
|
code,
|
|
210
|
+
name,
|
|
211
|
+
config,
|
|
190
212
|
lastCodeRun = null,
|
|
191
213
|
lastExecutionTime = null,
|
|
192
214
|
autoFocus = true,
|
|
193
215
|
skipIfCodeExists = false,
|
|
194
|
-
hideCode =
|
|
216
|
+
hideCode = undefined,
|
|
195
217
|
} = action;
|
|
196
218
|
|
|
197
219
|
let columnId: CellColumnId;
|
|
@@ -234,8 +256,12 @@ const {
|
|
|
234
256
|
[newCellId]: createCell({
|
|
235
257
|
id: newCellId,
|
|
236
258
|
code,
|
|
259
|
+
name,
|
|
237
260
|
lastCodeRun,
|
|
238
|
-
config: createCellConfig({
|
|
261
|
+
config: createCellConfig({
|
|
262
|
+
...config,
|
|
263
|
+
...(hideCode != null && { hide_code: hideCode }),
|
|
264
|
+
}),
|
|
239
265
|
lastExecutionTime,
|
|
240
266
|
edited: Boolean(code) && code !== lastCodeRun,
|
|
241
267
|
}),
|
|
@@ -417,6 +443,40 @@ const {
|
|
|
417
443
|
scrollKey: null,
|
|
418
444
|
};
|
|
419
445
|
},
|
|
446
|
+
moveCellsRelativeTo: (
|
|
447
|
+
state,
|
|
448
|
+
action: {
|
|
449
|
+
cellIds: CellId[];
|
|
450
|
+
targetCellId: CellId;
|
|
451
|
+
position: "before" | "after";
|
|
452
|
+
previousPlacements?: Array<{ columnId: CellColumnId; index: CellIndex }>;
|
|
453
|
+
},
|
|
454
|
+
) => {
|
|
455
|
+
const { cellIds, targetCellId, position, previousPlacements } = action;
|
|
456
|
+
if (cellIds.length === 0) {
|
|
457
|
+
return state;
|
|
458
|
+
}
|
|
459
|
+
const newCellIds = state.cellIds.moveCellsRelativeTo(
|
|
460
|
+
cellIds,
|
|
461
|
+
targetCellId,
|
|
462
|
+
position,
|
|
463
|
+
);
|
|
464
|
+
// Only record undo when caller provided full before-state
|
|
465
|
+
const canUndoMove =
|
|
466
|
+
previousPlacements && previousPlacements.length === cellIds.length;
|
|
467
|
+
const history = canUndoMove
|
|
468
|
+
? [
|
|
469
|
+
...state.history,
|
|
470
|
+
{ type: "move" as const, cellIds, placements: previousPlacements },
|
|
471
|
+
]
|
|
472
|
+
: state.history;
|
|
473
|
+
return {
|
|
474
|
+
...state,
|
|
475
|
+
cellIds: newCellIds,
|
|
476
|
+
history,
|
|
477
|
+
scrollKey: null,
|
|
478
|
+
};
|
|
479
|
+
},
|
|
420
480
|
dropCellOverColumn: (
|
|
421
481
|
state,
|
|
422
482
|
action: { cellId: CellId; columnId: CellColumnId },
|
|
@@ -659,6 +719,7 @@ const {
|
|
|
659
719
|
history: [
|
|
660
720
|
...state.history,
|
|
661
721
|
{
|
|
722
|
+
type: "delete",
|
|
662
723
|
name: prevData.name,
|
|
663
724
|
serializedEditorState: serializedEditorState,
|
|
664
725
|
column: column.id,
|
|
@@ -675,7 +736,29 @@ const {
|
|
|
675
736
|
return state;
|
|
676
737
|
}
|
|
677
738
|
|
|
678
|
-
const
|
|
739
|
+
const last = state.history[state.history.length - 1];
|
|
740
|
+
|
|
741
|
+
if (last.type === "move") {
|
|
742
|
+
const { cellIds, placements } = last;
|
|
743
|
+
if (
|
|
744
|
+
cellIds.length === 0 ||
|
|
745
|
+
placements.length !== cellIds.length ||
|
|
746
|
+
cellIds.some((id) => !state.cellData[id])
|
|
747
|
+
) {
|
|
748
|
+
return { ...state, history: state.history.slice(0, -1) };
|
|
749
|
+
}
|
|
750
|
+
const toRestore = cellIds.map((id, i) => ({
|
|
751
|
+
id,
|
|
752
|
+
columnId: placements[i].columnId,
|
|
753
|
+
index: placements[i].index,
|
|
754
|
+
}));
|
|
755
|
+
return {
|
|
756
|
+
...state,
|
|
757
|
+
cellIds: state.cellIds.placeCells(toRestore),
|
|
758
|
+
history: state.history.slice(0, -1),
|
|
759
|
+
scrollKey: cellIds[0] ?? null,
|
|
760
|
+
};
|
|
761
|
+
}
|
|
679
762
|
|
|
680
763
|
const {
|
|
681
764
|
name,
|
|
@@ -684,7 +767,7 @@ const {
|
|
|
684
767
|
index,
|
|
685
768
|
isSetupCell,
|
|
686
769
|
config,
|
|
687
|
-
} =
|
|
770
|
+
} = last;
|
|
688
771
|
|
|
689
772
|
const cellId = isSetupCell ? SETUP_CELL_ID : CellId.create();
|
|
690
773
|
const undoCell = createCell({
|
|
@@ -790,7 +873,7 @@ const {
|
|
|
790
873
|
cellReducer: (cell) => {
|
|
791
874
|
return {
|
|
792
875
|
...cell,
|
|
793
|
-
config: { ...cell.config, ...config },
|
|
876
|
+
config: createCellConfig({ ...cell.config, ...config }),
|
|
794
877
|
};
|
|
795
878
|
},
|
|
796
879
|
});
|
|
@@ -1633,6 +1716,8 @@ export const canUndoDeletesAtom = atom((get) =>
|
|
|
1633
1716
|
canUndoDeletes(get(notebookAtom)),
|
|
1634
1717
|
);
|
|
1635
1718
|
|
|
1719
|
+
export const undoLabelAtom = atom((get) => getUndoLabel(get(notebookAtom)));
|
|
1720
|
+
|
|
1636
1721
|
export const needsRunAtom = atom((get) => notebookNeedsRun(get(notebookAtom)));
|
|
1637
1722
|
|
|
1638
1723
|
export const cellErrorsAtom = atom((get) => {
|
|
@@ -218,13 +218,14 @@ export function toDocumentChanges(
|
|
|
218
218
|
];
|
|
219
219
|
}
|
|
220
220
|
|
|
221
|
-
// dropCellOverCell/dropCellOverColumn/moveCellToIndex → set-config + reorder-cells
|
|
221
|
+
// dropCellOverCell/dropCellOverColumn/moveCellToIndex/moveCellsRelativeTo → set-config + reorder-cells
|
|
222
222
|
// Drag-and-drop reorders can move cells within or across columns.
|
|
223
223
|
// We emit config changes for cells whose column changed, then
|
|
224
224
|
// the full ordering.
|
|
225
225
|
case "dropCellOverCell":
|
|
226
226
|
case "dropCellOverColumn":
|
|
227
227
|
case "moveCellToIndex":
|
|
228
|
+
case "moveCellsRelativeTo":
|
|
228
229
|
return columnChanges(prevState, newState);
|
|
229
230
|
|
|
230
231
|
// updateCellCode → set-code
|
|
@@ -302,6 +303,10 @@ export function toDocumentChanges(
|
|
|
302
303
|
case "undoDeleteCell": {
|
|
303
304
|
const changes = newCellChanges(prevState, newState);
|
|
304
305
|
const colChanges = columnChanges(prevState, newState);
|
|
306
|
+
// Undo-cut has no new cells — always emit reorder to sync the move.
|
|
307
|
+
if (changes.length === 0) {
|
|
308
|
+
return colChanges;
|
|
309
|
+
}
|
|
305
310
|
// Only include column changes if layout actually changed
|
|
306
311
|
// (colChanges always has at least a reorder-cells change)
|
|
307
312
|
return colChanges.length > 1 ? [...changes, ...colChanges] : changes;
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
/* Copyright 2026 Marimo. All rights reserved. */
|
|
2
|
+
|
|
3
|
+
import { atom, useAtomValue } from "jotai";
|
|
4
|
+
import type { CellId } from "@/core/cells/ids";
|
|
5
|
+
import { createReducerAndAtoms } from "@/utils/createReducer";
|
|
6
|
+
|
|
7
|
+
interface PendingCutState {
|
|
8
|
+
cellIds: Set<CellId>;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
const initialState = (): PendingCutState => ({
|
|
12
|
+
cellIds: new Set(),
|
|
13
|
+
});
|
|
14
|
+
|
|
15
|
+
const {
|
|
16
|
+
valueAtom: pendingCutStateAtom,
|
|
17
|
+
useActions: usePendingCutActionsInternal,
|
|
18
|
+
} = createReducerAndAtoms(initialState, {
|
|
19
|
+
markForCut: (_state, action: { cellIds: CellId[] }) => {
|
|
20
|
+
return {
|
|
21
|
+
cellIds: new Set(action.cellIds),
|
|
22
|
+
};
|
|
23
|
+
},
|
|
24
|
+
clear: () => {
|
|
25
|
+
return initialState();
|
|
26
|
+
},
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
export { pendingCutStateAtom };
|
|
30
|
+
|
|
31
|
+
export const pendingCutCellIdsAtom = atom(
|
|
32
|
+
(get) => get(pendingCutStateAtom).cellIds,
|
|
33
|
+
);
|
|
34
|
+
|
|
35
|
+
export const clearPendingCutAtom = atom(null, (_get, set) => {
|
|
36
|
+
set(pendingCutStateAtom, initialState());
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
export function usePendingCutActions() {
|
|
40
|
+
return usePendingCutActionsInternal();
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
export function useIsPendingCut(cellId: CellId): boolean {
|
|
44
|
+
const cellIds = useAtomValue(pendingCutCellIdsAtom);
|
|
45
|
+
return cellIds.has(cellId);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
export function usePendingCutState() {
|
|
49
|
+
return useAtomValue(pendingCutStateAtom);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
export function useHasPendingCut(): boolean {
|
|
53
|
+
const state = useAtomValue(pendingCutStateAtom);
|
|
54
|
+
return state.cellIds.size > 0;
|
|
55
|
+
}
|
package/src/core/cells/utils.ts
CHANGED
|
@@ -65,6 +65,17 @@ export function canUndoDeletes(state: NotebookState) {
|
|
|
65
65
|
return state.history.length > 0;
|
|
66
66
|
}
|
|
67
67
|
|
|
68
|
+
/**
|
|
69
|
+
* Label for the undo action based on the last history entry type.
|
|
70
|
+
*/
|
|
71
|
+
export function getUndoLabel(state: NotebookState): string {
|
|
72
|
+
const last = state.history[state.history.length - 1];
|
|
73
|
+
if (!last) {
|
|
74
|
+
return "Undo cell deletion";
|
|
75
|
+
}
|
|
76
|
+
return last.type === "move" ? "Undo move" : "Undo cell deletion";
|
|
77
|
+
}
|
|
78
|
+
|
|
68
79
|
/**
|
|
69
80
|
* Get the status of the descendants of the given cell.
|
|
70
81
|
*/
|
|
@@ -10,6 +10,10 @@ import {
|
|
|
10
10
|
} from "@codemirror/view";
|
|
11
11
|
import { createTracebackInfoAtom } from "@/core/cells/cells";
|
|
12
12
|
import { type CellId, HTMLCellId, SCRATCH_CELL_ID } from "@/core/cells/ids";
|
|
13
|
+
import {
|
|
14
|
+
clearPendingCutAtom,
|
|
15
|
+
pendingCutCellIdsAtom,
|
|
16
|
+
} from "@/core/cells/pending-cut-service";
|
|
13
17
|
import { loroSyncAnnotation } from "@/core/codemirror/rtc/loro/sync";
|
|
14
18
|
import type { KeymapConfig } from "@/core/config/config-schema";
|
|
15
19
|
import type { HotkeyProvider } from "@/core/hotkeys/hotkeys";
|
|
@@ -326,6 +330,12 @@ function cellCodeEditing(hotkeys: HotkeyProvider): Extension[] {
|
|
|
326
330
|
code: nextCode,
|
|
327
331
|
formattingChange: isFormattingChange,
|
|
328
332
|
});
|
|
333
|
+
|
|
334
|
+
// Clear pending cut state if this cell was marked for cut
|
|
335
|
+
const pendingCutCellIds = store.get(pendingCutCellIdsAtom);
|
|
336
|
+
if (pendingCutCellIds.has(cellId)) {
|
|
337
|
+
store.set(clearPendingCutAtom);
|
|
338
|
+
}
|
|
329
339
|
}
|
|
330
340
|
});
|
|
331
341
|
|
|
@@ -246,7 +246,7 @@ describe("insertImage", () => {
|
|
|
246
246
|
path: "public",
|
|
247
247
|
type: "file",
|
|
248
248
|
name: "hello.png",
|
|
249
|
-
|
|
249
|
+
file: expect.any(File),
|
|
250
250
|
});
|
|
251
251
|
|
|
252
252
|
expect(view.state.doc.toString()).toMatchInlineSnapshot(
|
|
@@ -291,7 +291,7 @@ describe("insertImage", () => {
|
|
|
291
291
|
path: "nested/public", // store in public folder of notebook directory
|
|
292
292
|
type: "file",
|
|
293
293
|
name: "hello.png",
|
|
294
|
-
|
|
294
|
+
file: expect.any(File),
|
|
295
295
|
});
|
|
296
296
|
|
|
297
297
|
expect(view.state.doc.toString()).toMatchInlineSnapshot(
|
|
@@ -337,7 +337,7 @@ describe("insertImage", () => {
|
|
|
337
337
|
path: "/Users/user/Development/project/public",
|
|
338
338
|
type: "file",
|
|
339
339
|
name: "hello.png",
|
|
340
|
-
|
|
340
|
+
file: expect.any(File),
|
|
341
341
|
});
|
|
342
342
|
|
|
343
343
|
// Should convert absolute path to relative path
|
|
@@ -313,7 +313,6 @@ export async function insertImage(view: EditorView, file: File) {
|
|
|
313
313
|
// If the file is base64 encoded, we can save it locally to prevent large file strings
|
|
314
314
|
try {
|
|
315
315
|
if (dataUrl.startsWith("data:")) {
|
|
316
|
-
const base64 = dataUrl.split(",")[1];
|
|
317
316
|
let inputFilename = prompt(
|
|
318
317
|
"We can save your image as a file. Enter a filename.",
|
|
319
318
|
file.name,
|
|
@@ -348,7 +347,7 @@ export async function insertImage(view: EditorView, file: File) {
|
|
|
348
347
|
path: publicFolderPath as FilePath,
|
|
349
348
|
type: "file",
|
|
350
349
|
name: inputFilename,
|
|
351
|
-
|
|
350
|
+
file,
|
|
352
351
|
});
|
|
353
352
|
|
|
354
353
|
if (createFileRes.success) {
|
package/src/core/edit-app.tsx
CHANGED
|
@@ -79,7 +79,7 @@ export const EditApp: React.FC<AppProps> = ({
|
|
|
79
79
|
};
|
|
80
80
|
}, []);
|
|
81
81
|
|
|
82
|
-
const { connection } = useMarimoKernelConnection({
|
|
82
|
+
const { connection, reconnect } = useMarimoKernelConnection({
|
|
83
83
|
autoInstantiate: userConfig.runtime.auto_instantiate,
|
|
84
84
|
setCells: (cells, layout) => {
|
|
85
85
|
setCells(cells);
|
|
@@ -147,6 +147,7 @@ export const EditApp: React.FC<AppProps> = ({
|
|
|
147
147
|
connection={connection}
|
|
148
148
|
isRunning={isRunning}
|
|
149
149
|
width={appConfig.width}
|
|
150
|
+
onReconnect={reconnect}
|
|
150
151
|
>
|
|
151
152
|
<AppHeader
|
|
152
153
|
connection={connection}
|
|
@@ -9,6 +9,7 @@ import {
|
|
|
9
9
|
} from "rpc-anywhere";
|
|
10
10
|
import type { NotificationPayload } from "@/core/kernel/messages";
|
|
11
11
|
import type { ParentSchema } from "@/core/wasm/rpc";
|
|
12
|
+
import { shouldLoadDuckDBPackages } from "@/core/wasm/utils";
|
|
12
13
|
import { TRANSPORT_ID } from "@/core/wasm/worker/constants";
|
|
13
14
|
import { getPyodideVersion } from "@/core/wasm/worker/getPyodideVersion";
|
|
14
15
|
import { MessageBuffer } from "@/core/wasm/worker/message-buffer";
|
|
@@ -85,8 +86,8 @@ const requestHandler = createRPCRequestHandler({
|
|
|
85
86
|
loadPackages: async (code: string) => {
|
|
86
87
|
await pyodideReadyPromise; // Make sure loading is done
|
|
87
88
|
|
|
88
|
-
if (code
|
|
89
|
-
// Add pandas and duckdb to the code
|
|
89
|
+
if (shouldLoadDuckDBPackages(code)) {
|
|
90
|
+
// Add pandas and duckdb to the code for mo.sql and for remote duckdb sources
|
|
90
91
|
code = `import pandas\n${code}`;
|
|
91
92
|
code = `import duckdb\n${code}`;
|
|
92
93
|
code = `import sqlglot\n${code}`;
|
|
@@ -6,6 +6,19 @@ import { API, createClientWithRuntimeManager } from "./api";
|
|
|
6
6
|
import { waitForConnectionOpen } from "./connection";
|
|
7
7
|
import type { EditRequests, RunRequests } from "./types";
|
|
8
8
|
|
|
9
|
+
/**
|
|
10
|
+
* Options for POSTing FormData via openapi-fetch. openapi-fetch types
|
|
11
|
+
* request bodies from the JSON schema, so we bypass the body type and
|
|
12
|
+
* override the serializer to pass the FormData through unchanged; the
|
|
13
|
+
* browser then sets the multipart Content-Type with boundary.
|
|
14
|
+
*/
|
|
15
|
+
function multipartInit(formData: FormData) {
|
|
16
|
+
return {
|
|
17
|
+
body: formData as never,
|
|
18
|
+
bodySerializer: (body: unknown) => body as never,
|
|
19
|
+
};
|
|
20
|
+
}
|
|
21
|
+
|
|
9
22
|
const { handleResponse, handleResponseReturnNull } = API;
|
|
10
23
|
|
|
11
24
|
export function createNetworkRequests(): EditRequests & RunRequests {
|
|
@@ -298,10 +311,15 @@ export function createNetworkRequests(): EditRequests & RunRequests {
|
|
|
298
311
|
},
|
|
299
312
|
sendCreateFileOrFolder: async (request) => {
|
|
300
313
|
await waitForConnectionOpen();
|
|
314
|
+
const formData = new FormData();
|
|
315
|
+
formData.append("path", request.path);
|
|
316
|
+
formData.append("type", request.type);
|
|
317
|
+
formData.append("name", request.name);
|
|
318
|
+
if (request.file) {
|
|
319
|
+
formData.append("file", request.file, request.name);
|
|
320
|
+
}
|
|
301
321
|
return getClient()
|
|
302
|
-
.POST("/api/files/create",
|
|
303
|
-
body: request,
|
|
304
|
-
})
|
|
322
|
+
.POST("/api/files/create", multipartInit(formData))
|
|
305
323
|
.then(handleResponse);
|
|
306
324
|
},
|
|
307
325
|
sendDeleteFileOrFolder: async (request) => {
|
|
@@ -80,6 +80,17 @@ export type SaveUserConfigurationRequest =
|
|
|
80
80
|
export interface SetCellConfigRequest {
|
|
81
81
|
configs: Record<CellId, Partial<CellConfig>>;
|
|
82
82
|
}
|
|
83
|
+
/**
|
|
84
|
+
* Client-side shape for creating a file/directory/notebook. The HTTP
|
|
85
|
+
* transport sends this as multipart/form-data; the WASM bridge base64-encodes
|
|
86
|
+
* `file` internally and crosses the JS<->Py boundary as JSON.
|
|
87
|
+
*/
|
|
88
|
+
export interface FileCreateInput {
|
|
89
|
+
path: string;
|
|
90
|
+
type: "file" | "directory" | "notebook";
|
|
91
|
+
name: string;
|
|
92
|
+
file?: Blob;
|
|
93
|
+
}
|
|
83
94
|
export type UpdateUIElementRequest = schemas["UpdateUIElementRequest"];
|
|
84
95
|
export type ModelRequest = schemas["ModelRequest"];
|
|
85
96
|
export type NotebookDocumentTransactionRequest =
|
|
@@ -165,7 +176,7 @@ export interface EditRequests {
|
|
|
165
176
|
sendListFiles: (request: FileListRequest) => Promise<FileListResponse>;
|
|
166
177
|
sendSearchFiles: (request: FileSearchRequest) => Promise<FileSearchResponse>;
|
|
167
178
|
sendCreateFileOrFolder: (
|
|
168
|
-
request:
|
|
179
|
+
request: FileCreateInput,
|
|
169
180
|
) => Promise<FileCreateResponse>;
|
|
170
181
|
sendDeleteFileOrFolder: (
|
|
171
182
|
request: FileDeleteRequest,
|
package/src/core/run-app.tsx
CHANGED
|
@@ -38,7 +38,7 @@ export const RunApp: React.FC<AppProps> = ({ appConfig }) => {
|
|
|
38
38
|
};
|
|
39
39
|
}, []);
|
|
40
40
|
|
|
41
|
-
const { connection } = useMarimoKernelConnection({
|
|
41
|
+
const { connection, reconnect } = useMarimoKernelConnection({
|
|
42
42
|
autoInstantiate: true,
|
|
43
43
|
setCells: setCells,
|
|
44
44
|
sessionId: getSessionId(),
|
|
@@ -84,6 +84,7 @@ export const RunApp: React.FC<AppProps> = ({ appConfig }) => {
|
|
|
84
84
|
connection={connection}
|
|
85
85
|
isRunning={isRunning}
|
|
86
86
|
width={appConfig.width}
|
|
87
|
+
onReconnect={reconnect}
|
|
87
88
|
>
|
|
88
89
|
<AppHeader connection={connection} className="sm:pt-8">
|
|
89
90
|
{galleryHref && (
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
/* Copyright 2026 Marimo. All rights reserved. */
|
|
2
|
+
|
|
3
|
+
import { describe, expect, it } from "vitest";
|
|
4
|
+
import { shouldLoadDuckDBPackages } from "../utils";
|
|
5
|
+
|
|
6
|
+
describe("shouldLoadDuckDBPackages", () => {
|
|
7
|
+
it("loads for mo.sql", () => {
|
|
8
|
+
expect(shouldLoadDuckDBPackages('df = mo.sql("SELECT 1")')).toBe(true);
|
|
9
|
+
});
|
|
10
|
+
|
|
11
|
+
it("loads for duckdb imports and usage", () => {
|
|
12
|
+
expect(shouldLoadDuckDBPackages("import duckdb")).toBe(true);
|
|
13
|
+
expect(shouldLoadDuckDBPackages("from duckdb import sql")).toBe(true);
|
|
14
|
+
expect(shouldLoadDuckDBPackages("import pandas, duckdb")).toBe(true);
|
|
15
|
+
expect(shouldLoadDuckDBPackages("rows = duckdb.sql('SELECT 1')")).toBe(
|
|
16
|
+
true,
|
|
17
|
+
);
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
it("loads when package discovery found duckdb", () => {
|
|
21
|
+
expect(
|
|
22
|
+
shouldLoadDuckDBPackages("print('hello')", new Set(["duckdb"])),
|
|
23
|
+
).toBe(true);
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
it("does not load for incidental duckdb text", () => {
|
|
27
|
+
expect(shouldLoadDuckDBPackages("name = 'duckdb'")).toBe(false);
|
|
28
|
+
expect(shouldLoadDuckDBPackages("# import duckdb")).toBe(false);
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
it("does not load without mo.sql, duckdb usage, or discovery", () => {
|
|
32
|
+
expect(shouldLoadDuckDBPackages("print('hello')")).toBe(false);
|
|
33
|
+
});
|
|
34
|
+
});
|
package/src/core/wasm/bridge.ts
CHANGED
|
@@ -3,6 +3,7 @@
|
|
|
3
3
|
|
|
4
4
|
import { toast } from "@/components/ui/use-toast";
|
|
5
5
|
import { userConfigAtom } from "@/core/config/config";
|
|
6
|
+
import { serializeBlob } from "@/utils/blob";
|
|
6
7
|
import { Deferred } from "@/utils/Deferred";
|
|
7
8
|
import { throwNotImplemented } from "@/utils/functions";
|
|
8
9
|
import { Logger } from "@/utils/Logger";
|
|
@@ -431,9 +432,21 @@ export class PyodideBridge implements RunRequests, EditRequests {
|
|
|
431
432
|
sendCreateFileOrFolder: EditRequests["sendCreateFileOrFolder"] = async (
|
|
432
433
|
request,
|
|
433
434
|
) => {
|
|
435
|
+
// The WASM RPC boundary can only carry JSON, so we base64-encode the
|
|
436
|
+
// file bytes here. The HTTP transport uses multipart/form-data instead.
|
|
437
|
+
let contents: string | null = null;
|
|
438
|
+
if (request.file) {
|
|
439
|
+
const dataUrl = await serializeBlob(request.file);
|
|
440
|
+
contents = dataUrl.split(",")[1] ?? "";
|
|
441
|
+
}
|
|
434
442
|
const response = await this.rpc.proxy.request.bridge({
|
|
435
443
|
functionName: "create_file_or_directory",
|
|
436
|
-
payload:
|
|
444
|
+
payload: {
|
|
445
|
+
path: request.path,
|
|
446
|
+
type: request.type,
|
|
447
|
+
name: request.name,
|
|
448
|
+
contents,
|
|
449
|
+
},
|
|
437
450
|
});
|
|
438
451
|
return response as FileCreateResponse;
|
|
439
452
|
};
|
package/src/core/wasm/utils.ts
CHANGED
|
@@ -10,3 +10,17 @@ export function isWasm(): boolean {
|
|
|
10
10
|
document.querySelector("marimo-wasm") !== null
|
|
11
11
|
);
|
|
12
12
|
}
|
|
13
|
+
|
|
14
|
+
const DUCKDB_USAGE_PATTERN =
|
|
15
|
+
/(^|\n)\s*(?:import\s+[^\n#]*\bduckdb\b|from\s+duckdb\b|[^\n#]*\bduckdb\s*\.)/;
|
|
16
|
+
|
|
17
|
+
export function shouldLoadDuckDBPackages(
|
|
18
|
+
code: string,
|
|
19
|
+
foundPackages?: ReadonlySet<string>,
|
|
20
|
+
): boolean {
|
|
21
|
+
return (
|
|
22
|
+
code.includes("mo.sql") ||
|
|
23
|
+
DUCKDB_USAGE_PATTERN.test(code) ||
|
|
24
|
+
foundPackages?.has("duckdb") === true
|
|
25
|
+
);
|
|
26
|
+
}
|
|
@@ -9,6 +9,7 @@ import { WasmFileSystem } from "./fs";
|
|
|
9
9
|
import { getMarimoWheel } from "./getMarimoWheel";
|
|
10
10
|
import { t } from "./tracer";
|
|
11
11
|
import type { SerializedBridge, WasmController } from "./types";
|
|
12
|
+
import { shouldLoadDuckDBPackages } from "../utils";
|
|
12
13
|
|
|
13
14
|
const MAKE_SNAPSHOT = false;
|
|
14
15
|
|
|
@@ -163,8 +164,8 @@ export class DefaultWasmController implements WasmController {
|
|
|
163
164
|
private async loadNotebookDeps(code: string, foundPackages: Set<string>) {
|
|
164
165
|
const pyodide = this.requirePyodide;
|
|
165
166
|
|
|
166
|
-
if (code
|
|
167
|
-
// We need pandas and duckdb for mo.sql
|
|
167
|
+
if (shouldLoadDuckDBPackages(code, foundPackages)) {
|
|
168
|
+
// We need pandas and duckdb for mo.sql and for remote duckdb sources
|
|
168
169
|
code = `import pandas\n${code}`;
|
|
169
170
|
code = `import duckdb\n${code}`;
|
|
170
171
|
code = `import sqlglot\n${code}`;
|
|
@@ -34,6 +34,7 @@ import type {
|
|
|
34
34
|
SerializedBridge,
|
|
35
35
|
WasmController,
|
|
36
36
|
} from "./types";
|
|
37
|
+
import { shouldLoadDuckDBPackages } from "../utils";
|
|
37
38
|
|
|
38
39
|
/**
|
|
39
40
|
* Web worker responsible for running the notebook.
|
|
@@ -141,8 +142,8 @@ const requestHandler = createRPCRequestHandler({
|
|
|
141
142
|
const span = t.startSpan("loadPackages");
|
|
142
143
|
await pyodideReadyPromise; // Make sure loading is done
|
|
143
144
|
|
|
144
|
-
if (code
|
|
145
|
-
// Add pandas and duckdb to the code
|
|
145
|
+
if (shouldLoadDuckDBPackages(code)) {
|
|
146
|
+
// Add pandas and duckdb to the code for mo.sql and for remote duckdb sources
|
|
146
147
|
code = `import pandas\n${code}`;
|
|
147
148
|
code = `import duckdb\n${code}`;
|
|
148
149
|
code = `import sqlglot\n${code}`;
|