@marimo-team/islands 0.23.7-dev2 → 0.23.7-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.
- 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-Den2QMz-.js} +621 -235
- 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-BZocwq8O.js → code-visibility-geywCqhR.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 +1181 -1144
- 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-DJ1NyBGw.js → mermaid-DO-Daq7u.js} +72 -69
- 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-CP9dWJEm.js → reveal-component-BYYXwJoo.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 +4 -4
- 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/useNotebookActions.tsx +3 -1
- package/src/components/editor/app-container.tsx +7 -1
- package/src/components/editor/controls/Controls.tsx +3 -1
- 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/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/core/__test__/sanitize.test.ts +8 -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/plugins/layout/__test__/MermaidPlugin.test.ts +50 -0
- package/src/plugins/layout/mermaid/MermaidPlugin.tsx +11 -1
- package/src/plugins/layout/mermaid/mermaid.tsx +7 -3
- 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
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
/* Copyright 2026 Marimo. All rights reserved. */
|
|
2
|
+
|
|
3
|
+
import { isEmpty } from "lodash-es";
|
|
4
|
+
import { InfoIcon } from "lucide-react";
|
|
5
|
+
import React from "react";
|
|
6
|
+
import { z } from "zod";
|
|
7
|
+
|
|
8
|
+
// A value worth rendering: drop null/undefined and empty containers
|
|
9
|
+
// (`{}`, `[]`), but keep meaningful primitives (`0`, `false`, `""`).
|
|
10
|
+
function isUninformative(value: unknown): boolean {
|
|
11
|
+
if (value == null) {
|
|
12
|
+
return true;
|
|
13
|
+
}
|
|
14
|
+
if (typeof value === "object") {
|
|
15
|
+
return isEmpty(value);
|
|
16
|
+
}
|
|
17
|
+
return false;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
// Zod schema matching the Python SuccessResult dataclass
|
|
21
|
+
const SuccessResultSchema = z.looseObject({
|
|
22
|
+
status: z.string().default("success"),
|
|
23
|
+
auth_required: z.boolean().default(false),
|
|
24
|
+
action_url: z.any(),
|
|
25
|
+
next_steps: z.any(),
|
|
26
|
+
meta: z.any(),
|
|
27
|
+
message: z.string().nullish(),
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
type SuccessResult = z.infer<typeof SuccessResultSchema>;
|
|
31
|
+
|
|
32
|
+
const PrettySuccessResult: React.FC<{ data: SuccessResult }> = ({ data }) => {
|
|
33
|
+
const {
|
|
34
|
+
status,
|
|
35
|
+
auth_required,
|
|
36
|
+
action_url: _action_url,
|
|
37
|
+
meta: _meta,
|
|
38
|
+
next_steps: _next_steps,
|
|
39
|
+
message,
|
|
40
|
+
...rest
|
|
41
|
+
} = data;
|
|
42
|
+
|
|
43
|
+
return (
|
|
44
|
+
<div className="flex flex-col gap-1.5">
|
|
45
|
+
<div className="flex items-center justify-between">
|
|
46
|
+
<h3 className="text-xs font-semibold text-muted-foreground">
|
|
47
|
+
Tool Result
|
|
48
|
+
</h3>
|
|
49
|
+
<div className="flex items-center gap-2">
|
|
50
|
+
<span className="text-xs px-2 py-0.5 bg-(--grass-2) text-(--grass-11) rounded-full font-medium capitalize">
|
|
51
|
+
{status}
|
|
52
|
+
</span>
|
|
53
|
+
{auth_required && (
|
|
54
|
+
<span className="text-xs px-2 py-0.5 bg-(--amber-2) text-(--amber-11) rounded-full">
|
|
55
|
+
Auth Required
|
|
56
|
+
</span>
|
|
57
|
+
)}
|
|
58
|
+
</div>
|
|
59
|
+
</div>
|
|
60
|
+
|
|
61
|
+
{message && (
|
|
62
|
+
<div className="flex items-start gap-2">
|
|
63
|
+
<InfoIcon className="h-3 w-3 text-(--blue-11) mt-0.5 shrink-0" />
|
|
64
|
+
<div className="text-xs text-foreground">{message}</div>
|
|
65
|
+
</div>
|
|
66
|
+
)}
|
|
67
|
+
|
|
68
|
+
{rest && (
|
|
69
|
+
<div className="space-y-3">
|
|
70
|
+
{Object.entries(rest).map(([key, value]) => {
|
|
71
|
+
if (isUninformative(value)) {
|
|
72
|
+
return null;
|
|
73
|
+
}
|
|
74
|
+
return (
|
|
75
|
+
<div key={key} className="space-y-1.5">
|
|
76
|
+
<span className="text-xs text-muted-foreground">{key}</span>
|
|
77
|
+
<pre className="bg-(--slate-2) p-2 text-muted-foreground border border-(--slate-4) rounded text-xs overflow-auto scrollbar-thin max-h-64">
|
|
78
|
+
{JSON.stringify(value, null, 2)}
|
|
79
|
+
</pre>
|
|
80
|
+
</div>
|
|
81
|
+
);
|
|
82
|
+
})}
|
|
83
|
+
</div>
|
|
84
|
+
)}
|
|
85
|
+
</div>
|
|
86
|
+
);
|
|
87
|
+
};
|
|
88
|
+
|
|
89
|
+
export const ResultRenderer: React.FC<{ result: unknown }> = ({ result }) => {
|
|
90
|
+
const parseResult = SuccessResultSchema.safeParse(result);
|
|
91
|
+
|
|
92
|
+
if (parseResult.success) {
|
|
93
|
+
return <PrettySuccessResult data={parseResult.data} />;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
return (
|
|
97
|
+
<div className="text-xs font-medium text-muted-foreground mb-1 max-h-64 overflow-y-auto scrollbar-thin">
|
|
98
|
+
{typeof result === "string" ? result : JSON.stringify(result, null, 2)}
|
|
99
|
+
</div>
|
|
100
|
+
);
|
|
101
|
+
};
|
|
@@ -34,6 +34,7 @@ interface TableTopBarProps extends Partial<ExportActionProps> {
|
|
|
34
34
|
showTableExplorer?: boolean;
|
|
35
35
|
togglePanel?: (panelType: PanelType) => void;
|
|
36
36
|
isAnyPanelOpen?: boolean;
|
|
37
|
+
sizeBytes?: number | null;
|
|
37
38
|
}
|
|
38
39
|
|
|
39
40
|
export const TableTopBar: React.FC<TableTopBarProps> = ({
|
|
@@ -48,6 +49,7 @@ export const TableTopBar: React.FC<TableTopBarProps> = ({
|
|
|
48
49
|
togglePanel,
|
|
49
50
|
isAnyPanelOpen,
|
|
50
51
|
downloadAs,
|
|
52
|
+
sizeBytes,
|
|
51
53
|
}) => {
|
|
52
54
|
const [internalValue, setInternalValue] = useState(searchQuery || "");
|
|
53
55
|
const debouncedSearch = useDebounce(internalValue, 500);
|
|
@@ -130,7 +132,9 @@ export const TableTopBar: React.FC<TableTopBarProps> = ({
|
|
|
130
132
|
Explore
|
|
131
133
|
</Button>
|
|
132
134
|
)}
|
|
133
|
-
{downloadAs &&
|
|
135
|
+
{downloadAs && (
|
|
136
|
+
<ExportMenu downloadAs={downloadAs} sizeBytes={sizeBytes} />
|
|
137
|
+
)}
|
|
134
138
|
</div>
|
|
135
139
|
</div>
|
|
136
140
|
);
|
|
@@ -70,6 +70,9 @@ interface DataTableProps<TData> extends Partial<ExportActionProps> {
|
|
|
70
70
|
setSorting?: OnChangeFn<SortingState>; // controlled sorting
|
|
71
71
|
// Pagination
|
|
72
72
|
totalRows: number | TooManyRows;
|
|
73
|
+
// JSON-serialized size of the currently-rendered data. Forwarded to
|
|
74
|
+
// ExportMenu so hosts can size-gate the Export button via downloadSizeLimitAtom.
|
|
75
|
+
sizeBytes?: number | null;
|
|
73
76
|
totalColumns: number;
|
|
74
77
|
pagination?: boolean;
|
|
75
78
|
manualPagination?: boolean; // server-side pagination
|
|
@@ -121,6 +124,7 @@ const DataTableInternal = <TData,>({
|
|
|
121
124
|
selection,
|
|
122
125
|
totalColumns,
|
|
123
126
|
totalRows,
|
|
127
|
+
sizeBytes,
|
|
124
128
|
manualSorting = false,
|
|
125
129
|
sorting,
|
|
126
130
|
setSorting,
|
|
@@ -309,6 +313,7 @@ const DataTableInternal = <TData,>({
|
|
|
309
313
|
togglePanel={togglePanel}
|
|
310
314
|
isAnyPanelOpen={isAnyPanelOpen}
|
|
311
315
|
downloadAs={downloadAs}
|
|
316
|
+
sizeBytes={sizeBytes}
|
|
312
317
|
/>
|
|
313
318
|
<Table
|
|
314
319
|
className={cn(
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
/* Copyright 2026 Marimo. All rights reserved. */
|
|
2
2
|
|
|
3
|
+
import { useAtomValue } from "jotai";
|
|
3
4
|
import {
|
|
4
5
|
BracesIcon,
|
|
5
6
|
BrickWallIcon,
|
|
@@ -9,6 +10,7 @@ import {
|
|
|
9
10
|
} from "lucide-react";
|
|
10
11
|
import React from "react";
|
|
11
12
|
import { useLocale } from "react-aria";
|
|
13
|
+
import { downloadSizeLimitAtom } from "./download-policy/atoms";
|
|
12
14
|
import { logNever } from "@/utils/assertNever";
|
|
13
15
|
import { cn } from "@/utils/cn";
|
|
14
16
|
import { copyToClipboard } from "@/utils/copy";
|
|
@@ -84,6 +86,11 @@ export interface ExportActionProps {
|
|
|
84
86
|
error?: string | null;
|
|
85
87
|
missing_packages?: string[] | null;
|
|
86
88
|
}>;
|
|
89
|
+
// JSON-serialized size of the currently-rendered data. Used together with
|
|
90
|
+
// downloadSizeLimitAtom to disable the Export button when a host (e.g.,
|
|
91
|
+
// marimo-lsp inside VS Code) declares a download size cap. Null/undefined
|
|
92
|
+
// means "no info" and the gate stays disabled (fail-open).
|
|
93
|
+
sizeBytes?: number | null;
|
|
87
94
|
}
|
|
88
95
|
|
|
89
96
|
const labelForDownloadFormat = (format: DownloadFormat): string =>
|
|
@@ -94,12 +101,19 @@ const labelForCopyFormat = (format: CopyFormat): string =>
|
|
|
94
101
|
export const ExportMenu: React.FC<ExportActionProps> = (props) => {
|
|
95
102
|
const { locale } = useLocale();
|
|
96
103
|
const [open, setOpen] = React.useState(false);
|
|
104
|
+
const policy = useAtomValue(downloadSizeLimitAtom);
|
|
105
|
+
const disabled = !!(
|
|
106
|
+
policy &&
|
|
107
|
+
props.sizeBytes != null &&
|
|
108
|
+
props.sizeBytes > policy.limitBytes
|
|
109
|
+
);
|
|
97
110
|
|
|
98
111
|
const button = (
|
|
99
112
|
<Button
|
|
100
113
|
data-testid="export-button"
|
|
101
114
|
size="xs"
|
|
102
115
|
variant="text"
|
|
116
|
+
disabled={disabled}
|
|
103
117
|
className={cn(
|
|
104
118
|
"print:hidden text-xs gap-1",
|
|
105
119
|
open ? "text-primary" : "text-muted-foreground",
|
|
@@ -113,7 +127,10 @@ export const ExportMenu: React.FC<ExportActionProps> = (props) => {
|
|
|
113
127
|
const resolveDownloadUrl = async (
|
|
114
128
|
format: DownloadFormat,
|
|
115
129
|
onRetry: () => void,
|
|
116
|
-
): Promise<{
|
|
130
|
+
): Promise<{
|
|
131
|
+
url: string;
|
|
132
|
+
filename: string;
|
|
133
|
+
} | null> => {
|
|
117
134
|
let response: Awaited<ReturnType<typeof props.downloadAs>>;
|
|
118
135
|
try {
|
|
119
136
|
response = await props.downloadAs({ format });
|
|
@@ -143,7 +160,10 @@ export const ExportMenu: React.FC<ExportActionProps> = (props) => {
|
|
|
143
160
|
return null;
|
|
144
161
|
}
|
|
145
162
|
|
|
146
|
-
return {
|
|
163
|
+
return {
|
|
164
|
+
url: response.url,
|
|
165
|
+
filename: response.filename,
|
|
166
|
+
};
|
|
147
167
|
};
|
|
148
168
|
|
|
149
169
|
const handleDownload = async (format: DownloadFormat) => {
|
|
@@ -229,8 +249,15 @@ export const ExportMenu: React.FC<ExportActionProps> = (props) => {
|
|
|
229
249
|
|
|
230
250
|
return (
|
|
231
251
|
<DropdownMenu modal={false} open={open} onOpenChange={setOpen}>
|
|
232
|
-
<Tooltip
|
|
233
|
-
|
|
252
|
+
<Tooltip
|
|
253
|
+
content={disabled ? policy?.unavailableMessage : "Export"}
|
|
254
|
+
open={open ? false : undefined}
|
|
255
|
+
>
|
|
256
|
+
<DropdownMenuTrigger asChild={true} disabled={disabled}>
|
|
257
|
+
<span tabIndex={disabled ? 0 : -1} className="inline-flex">
|
|
258
|
+
{button}
|
|
259
|
+
</span>
|
|
260
|
+
</DropdownMenuTrigger>
|
|
234
261
|
</Tooltip>
|
|
235
262
|
<DropdownMenuContent side="bottom" className="print:hidden">
|
|
236
263
|
<DropdownMenuLabel className="text-xs text-muted-foreground">
|
|
@@ -53,6 +53,7 @@ import {
|
|
|
53
53
|
canUndoDeletesAtom,
|
|
54
54
|
getNotebook,
|
|
55
55
|
hasDisabledCellsAtom,
|
|
56
|
+
undoLabelAtom,
|
|
56
57
|
useCellActions,
|
|
57
58
|
} from "@/core/cells/cells";
|
|
58
59
|
import { disabledCellIds } from "@/core/cells/utils";
|
|
@@ -137,6 +138,7 @@ export function useNotebookActions() {
|
|
|
137
138
|
|
|
138
139
|
const hasDisabledCells = useAtomValue(hasDisabledCellsAtom);
|
|
139
140
|
const canUndoDeletes = useAtomValue(canUndoDeletesAtom);
|
|
141
|
+
const undoLabel = useAtomValue(undoLabelAtom);
|
|
140
142
|
const { selectedLayout } = useLayoutState();
|
|
141
143
|
const { setLayoutView } = useLayoutActions();
|
|
142
144
|
const togglePresenting = useTogglePresenting();
|
|
@@ -525,7 +527,7 @@ export function useNotebookActions() {
|
|
|
525
527
|
},
|
|
526
528
|
{
|
|
527
529
|
icon: <Undo2Icon size={14} strokeWidth={1.5} />,
|
|
528
|
-
label:
|
|
530
|
+
label: undoLabel,
|
|
529
531
|
hidden: !canUndoDeletes || kioskMode,
|
|
530
532
|
handle: () => {
|
|
531
533
|
undoDeleteCell();
|
|
@@ -15,6 +15,7 @@ interface Props {
|
|
|
15
15
|
connection: ConnectionStatus;
|
|
16
16
|
isRunning: boolean;
|
|
17
17
|
width: AppConfig["width"];
|
|
18
|
+
onReconnect?: () => void;
|
|
18
19
|
}
|
|
19
20
|
|
|
20
21
|
export const AppContainer: React.FC<PropsWithChildren<Props>> = ({
|
|
@@ -22,13 +23,18 @@ export const AppContainer: React.FC<PropsWithChildren<Props>> = ({
|
|
|
22
23
|
connection,
|
|
23
24
|
isRunning,
|
|
24
25
|
children,
|
|
26
|
+
onReconnect,
|
|
25
27
|
}) => {
|
|
26
28
|
const connectionState = connection.state;
|
|
27
29
|
|
|
28
30
|
return (
|
|
29
31
|
<>
|
|
30
32
|
<DynamicFavicon isRunning={isRunning} />
|
|
31
|
-
<StatusOverlay
|
|
33
|
+
<StatusOverlay
|
|
34
|
+
connection={connection}
|
|
35
|
+
isRunning={isRunning}
|
|
36
|
+
onReconnect={onReconnect}
|
|
37
|
+
/>
|
|
32
38
|
<PyodideLoader>
|
|
33
39
|
<WrappedWithSidebar>
|
|
34
40
|
{/** oxlint-ignore-next-line -- ID is used by other components to grab the DOM element */}
|
|
@@ -27,6 +27,7 @@ import { Functions } from "@/utils/functions";
|
|
|
27
27
|
import {
|
|
28
28
|
canUndoDeletesAtom,
|
|
29
29
|
needsRunAtom,
|
|
30
|
+
undoLabelAtom,
|
|
30
31
|
useCellActions,
|
|
31
32
|
} from "../../../core/cells/cells";
|
|
32
33
|
import { ConfigButton } from "../../app-config/app-config-button";
|
|
@@ -56,6 +57,7 @@ export const Controls = ({
|
|
|
56
57
|
running,
|
|
57
58
|
}: ControlsProps): JSX.Element => {
|
|
58
59
|
const undoAvailable = useAtomValue(canUndoDeletesAtom);
|
|
60
|
+
const undoLabel = useAtomValue(undoLabelAtom);
|
|
59
61
|
const needsRun = useAtomValue(needsRunAtom);
|
|
60
62
|
const { undoDeleteCell } = useCellActions();
|
|
61
63
|
const closed = connectionState === WebSocketState.CLOSED;
|
|
@@ -63,7 +65,7 @@ export const Controls = ({
|
|
|
63
65
|
let undoControl: JSX.Element | null = null;
|
|
64
66
|
if (!closed && undoAvailable) {
|
|
65
67
|
undoControl = (
|
|
66
|
-
<Tooltip content=
|
|
68
|
+
<Tooltip content={undoLabel}>
|
|
67
69
|
<Button
|
|
68
70
|
data-testid="undo-delete-cell"
|
|
69
71
|
size="medium"
|
|
@@ -11,6 +11,9 @@ import { prettyError } from "@/utils/errors";
|
|
|
11
11
|
import { Functions } from "@/utils/functions";
|
|
12
12
|
import { type FilePath, PathBuilder } from "@/utils/paths";
|
|
13
13
|
import { resolvePaths } from "@/utils/pathUtils";
|
|
14
|
+
import { mapWithConcurrency } from "@/utils/semaphore";
|
|
15
|
+
|
|
16
|
+
const FILE_OP_CONCURRENCY = 5;
|
|
14
17
|
|
|
15
18
|
/**
|
|
16
19
|
* Normalized result of a file mutation: the server response when successful,
|
|
@@ -159,28 +162,26 @@ export class RequestingTree {
|
|
|
159
162
|
? (this.delegate.find(parentId)?.data.path ?? parentId)
|
|
160
163
|
: this.rootPath;
|
|
161
164
|
|
|
162
|
-
await
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
}
|
|
165
|
+
await mapWithConcurrency(fromIds, FILE_OP_CONCURRENCY, async (id) => {
|
|
166
|
+
const node = this.delegate.find(id);
|
|
167
|
+
if (!node) {
|
|
168
|
+
return;
|
|
169
|
+
}
|
|
170
|
+
const originalPath = node.data.path;
|
|
171
|
+
const newPath = this.path.join(
|
|
172
|
+
parentPath,
|
|
173
|
+
this.path.basename(originalPath as FilePath),
|
|
174
|
+
);
|
|
175
|
+
const result = await this.callbacks
|
|
176
|
+
.renameFileOrFolder({ path: originalPath, newPath })
|
|
177
|
+
.then(handleFileResponse);
|
|
178
|
+
if (!result) {
|
|
179
|
+
return;
|
|
180
|
+
}
|
|
179
181
|
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
);
|
|
182
|
+
this.delegate.move({ id, parentId, index: 0 });
|
|
183
|
+
this.delegate.update({ id, changes: { path: newPath } });
|
|
184
|
+
});
|
|
184
185
|
|
|
185
186
|
this.onChange(this.delegate.data);
|
|
186
187
|
|
|
@@ -262,11 +263,12 @@ export class RequestingTree {
|
|
|
262
263
|
this.rootPath,
|
|
263
264
|
...ids.map((id) => this.delegate.find(id)?.data.path),
|
|
264
265
|
].filter(Boolean);
|
|
265
|
-
// Request
|
|
266
|
-
const data = await
|
|
267
|
-
openFolders
|
|
266
|
+
// Request open folders with bounded concurrency; swallow per-folder errors.
|
|
267
|
+
const data = await mapWithConcurrency(
|
|
268
|
+
openFolders,
|
|
269
|
+
FILE_OP_CONCURRENCY,
|
|
270
|
+
(path) =>
|
|
268
271
|
this.callbacks.listFiles({ path: path }).catch(() => ({ files: [] })),
|
|
269
|
-
),
|
|
270
272
|
);
|
|
271
273
|
|
|
272
274
|
for (const [idx, openFolder] of openFolders.entries()) {
|
|
@@ -3,13 +3,14 @@
|
|
|
3
3
|
import { type DropzoneOptions, useDropzone } from "react-dropzone";
|
|
4
4
|
import { toast } from "@/components/ui/use-toast";
|
|
5
5
|
import { useRequestClient } from "@/core/network/requests";
|
|
6
|
-
import { serializeBlob } from "@/utils/blob";
|
|
7
6
|
import { withLoadingToast } from "@/utils/download";
|
|
8
7
|
import { Logger } from "@/utils/Logger";
|
|
9
8
|
import { type FilePath, PathBuilder } from "@/utils/paths";
|
|
9
|
+
import { mapWithConcurrency } from "@/utils/semaphore";
|
|
10
10
|
import { refreshRoot } from "./state";
|
|
11
11
|
|
|
12
12
|
const MAX_SIZE = 1024 * 1024 * 100; // 100MB
|
|
13
|
+
const UPLOAD_CONCURRENCY = 5;
|
|
13
14
|
|
|
14
15
|
export function useFileExplorerUpload(options: DropzoneOptions = {}) {
|
|
15
16
|
const { sendCreateFileOrFolder } = useRequestClient();
|
|
@@ -59,30 +60,28 @@ export function useFileExplorerUpload(options: DropzoneOptions = {}) {
|
|
|
59
60
|
loadingTitle,
|
|
60
61
|
async (progress) => {
|
|
61
62
|
progress.addTotal(acceptedFiles.length);
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
63
|
+
await mapWithConcurrency(
|
|
64
|
+
acceptedFiles,
|
|
65
|
+
UPLOAD_CONCURRENCY,
|
|
66
|
+
async (file) => {
|
|
67
|
+
// We strip the leading slash since File.path can return
|
|
68
|
+
// `/path/to/file`.
|
|
69
|
+
const filePath = stripLeadingSlash(getPath(file));
|
|
70
|
+
let directoryPath = "" as FilePath;
|
|
71
|
+
if (filePath) {
|
|
72
|
+
directoryPath =
|
|
73
|
+
PathBuilder.guessDeliminator(filePath).dirname(filePath);
|
|
74
|
+
}
|
|
71
75
|
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
name: file.name,
|
|
82
|
-
contents: base64,
|
|
83
|
-
});
|
|
84
|
-
progress.increment(1);
|
|
85
|
-
}
|
|
76
|
+
await sendCreateFileOrFolder({
|
|
77
|
+
path: directoryPath,
|
|
78
|
+
type: "file",
|
|
79
|
+
name: file.name,
|
|
80
|
+
file,
|
|
81
|
+
});
|
|
82
|
+
progress.increment(1);
|
|
83
|
+
},
|
|
84
|
+
);
|
|
86
85
|
await refreshRoot();
|
|
87
86
|
},
|
|
88
87
|
onFinish,
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/* Copyright 2026 Marimo. All rights reserved. */
|
|
2
|
+
// @vitest-environment jsdom
|
|
3
|
+
|
|
4
|
+
import { fireEvent, render } from "@testing-library/react";
|
|
5
|
+
import { createStore, Provider as JotaiProvider } from "jotai";
|
|
6
|
+
import type React from "react";
|
|
7
|
+
import { describe, expect, it, vi } from "vitest";
|
|
8
|
+
import { TooltipProvider } from "@/components/ui/tooltip";
|
|
9
|
+
import { viewStateAtom } from "@/core/mode";
|
|
10
|
+
import {
|
|
11
|
+
type ConnectionStatus,
|
|
12
|
+
WebSocketClosedReason,
|
|
13
|
+
WebSocketState,
|
|
14
|
+
} from "@/core/websocket/types";
|
|
15
|
+
import { StatusOverlay } from "../status";
|
|
16
|
+
|
|
17
|
+
function renderOverlay(
|
|
18
|
+
connection: ConnectionStatus,
|
|
19
|
+
onReconnect?: () => void,
|
|
20
|
+
): ReturnType<typeof render> {
|
|
21
|
+
const store = createStore();
|
|
22
|
+
store.set(viewStateAtom, { mode: "edit", cellAnchor: null });
|
|
23
|
+
const wrapper: React.FC<React.PropsWithChildren> = ({ children }) => (
|
|
24
|
+
<JotaiProvider store={store}>
|
|
25
|
+
<TooltipProvider>{children}</TooltipProvider>
|
|
26
|
+
</JotaiProvider>
|
|
27
|
+
);
|
|
28
|
+
return render(
|
|
29
|
+
<StatusOverlay
|
|
30
|
+
connection={connection}
|
|
31
|
+
isRunning={false}
|
|
32
|
+
onReconnect={onReconnect}
|
|
33
|
+
/>,
|
|
34
|
+
{ wrapper },
|
|
35
|
+
);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
describe("StatusOverlay disconnect indicator", () => {
|
|
39
|
+
it("invokes onReconnect when the disconnect icon is clicked", () => {
|
|
40
|
+
const onReconnect = vi.fn();
|
|
41
|
+
const { getByTestId } = renderOverlay(
|
|
42
|
+
{
|
|
43
|
+
state: WebSocketState.CLOSED,
|
|
44
|
+
code: WebSocketClosedReason.KERNEL_DISCONNECTED,
|
|
45
|
+
reason: "kernel not found",
|
|
46
|
+
},
|
|
47
|
+
onReconnect,
|
|
48
|
+
);
|
|
49
|
+
|
|
50
|
+
const icon = getByTestId("disconnected-indicator") as HTMLButtonElement;
|
|
51
|
+
expect(icon.tagName).toBe("BUTTON");
|
|
52
|
+
expect(icon.disabled).toBe(false);
|
|
53
|
+
expect(icon.getAttribute("aria-label")).toBe("Reconnect to app");
|
|
54
|
+
fireEvent.click(icon);
|
|
55
|
+
expect(onReconnect).toHaveBeenCalledTimes(1);
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
it("renders a disabled button when no onReconnect is provided", () => {
|
|
59
|
+
const { getByTestId } = renderOverlay({
|
|
60
|
+
state: WebSocketState.CLOSED,
|
|
61
|
+
code: WebSocketClosedReason.KERNEL_DISCONNECTED,
|
|
62
|
+
reason: "kernel not found",
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
const button = getByTestId("disconnected-indicator");
|
|
66
|
+
expect((button as HTMLButtonElement).disabled).toBe(true);
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
it.each([
|
|
70
|
+
[
|
|
71
|
+
WebSocketClosedReason.MALFORMED_QUERY,
|
|
72
|
+
"the kernel did not recognize a request; please file a bug with marimo",
|
|
73
|
+
],
|
|
74
|
+
[
|
|
75
|
+
WebSocketClosedReason.KERNEL_STARTUP_ERROR,
|
|
76
|
+
"Failed to start kernel sandbox",
|
|
77
|
+
],
|
|
78
|
+
])(
|
|
79
|
+
"renders a disabled button for non-recoverable close reason %s",
|
|
80
|
+
(code, reason) => {
|
|
81
|
+
const onReconnect = vi.fn();
|
|
82
|
+
const { getByTestId } = renderOverlay(
|
|
83
|
+
{ state: WebSocketState.CLOSED, code, reason },
|
|
84
|
+
onReconnect,
|
|
85
|
+
);
|
|
86
|
+
|
|
87
|
+
const button = getByTestId("disconnected-indicator") as HTMLButtonElement;
|
|
88
|
+
expect(button.disabled).toBe(true);
|
|
89
|
+
fireEvent.click(button);
|
|
90
|
+
expect(onReconnect).not.toHaveBeenCalled();
|
|
91
|
+
},
|
|
92
|
+
);
|
|
93
|
+
|
|
94
|
+
it("does not render the disconnect icon when another tab has taken over", () => {
|
|
95
|
+
const onReconnect = vi.fn();
|
|
96
|
+
const { queryByTestId } = renderOverlay(
|
|
97
|
+
{
|
|
98
|
+
state: WebSocketState.CLOSED,
|
|
99
|
+
code: WebSocketClosedReason.ALREADY_RUNNING,
|
|
100
|
+
reason: "another browser tab is already connected to the kernel",
|
|
101
|
+
canTakeover: true,
|
|
102
|
+
},
|
|
103
|
+
onReconnect,
|
|
104
|
+
);
|
|
105
|
+
|
|
106
|
+
expect(queryByTestId("disconnected-indicator")).toBeNull();
|
|
107
|
+
});
|
|
108
|
+
});
|
|
@@ -7,16 +7,26 @@ import { Tooltip } from "@/components/ui/tooltip";
|
|
|
7
7
|
import { notebookScrollToRunning } from "@/core/cells/actions";
|
|
8
8
|
import { onlyScratchpadIsRunningAtom } from "@/core/cells/cells";
|
|
9
9
|
import { viewStateAtom } from "@/core/mode";
|
|
10
|
-
import {
|
|
10
|
+
import {
|
|
11
|
+
type ConnectionStatus,
|
|
12
|
+
WebSocketClosedReason,
|
|
13
|
+
WebSocketState,
|
|
14
|
+
} from "@/core/websocket/types";
|
|
11
15
|
import { cn } from "@/utils/cn";
|
|
12
16
|
|
|
13
17
|
export const StatusOverlay: React.FC<{
|
|
14
18
|
connection: ConnectionStatus;
|
|
15
19
|
isRunning: boolean;
|
|
16
|
-
|
|
20
|
+
onReconnect?: () => void;
|
|
21
|
+
}> = ({ connection, isRunning, onReconnect }) => {
|
|
17
22
|
const { mode } = useAtomValue(viewStateAtom);
|
|
18
23
|
const isClosed = connection.state === WebSocketState.CLOSED;
|
|
19
24
|
const isOpen = connection.state === WebSocketState.OPEN;
|
|
25
|
+
// Only KERNEL_DISCONNECTED is recoverable by a retry. Other terminal
|
|
26
|
+
// reasons (MALFORMED_QUERY, KERNEL_STARTUP_ERROR) would deterministically
|
|
27
|
+
// fail the same way; ALREADY_RUNNING is handled by `LockedIcon` below.
|
|
28
|
+
const canReconnect =
|
|
29
|
+
isClosed && connection.code === WebSocketClosedReason.KERNEL_DISCONNECTED;
|
|
20
30
|
|
|
21
31
|
return (
|
|
22
32
|
<>
|
|
@@ -28,7 +38,11 @@ export const StatusOverlay: React.FC<{
|
|
|
28
38
|
)}
|
|
29
39
|
>
|
|
30
40
|
{isOpen && isRunning && <RunningIcon />}
|
|
31
|
-
{isClosed && !connection.canTakeover &&
|
|
41
|
+
{isClosed && !connection.canTakeover && (
|
|
42
|
+
<DisconnectedIcon
|
|
43
|
+
onReconnect={canReconnect ? onReconnect : undefined}
|
|
44
|
+
/>
|
|
45
|
+
)}
|
|
32
46
|
{isClosed && connection.canTakeover && <LockedIcon />}
|
|
33
47
|
</div>
|
|
34
48
|
</>
|
|
@@ -37,13 +51,33 @@ export const StatusOverlay: React.FC<{
|
|
|
37
51
|
|
|
38
52
|
const topLeftStatus = "print:hidden pointer-events-auto hover:cursor-pointer";
|
|
39
53
|
|
|
40
|
-
const DisconnectedIcon
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
54
|
+
const DisconnectedIcon: React.FC<{ onReconnect?: () => void }> = ({
|
|
55
|
+
onReconnect,
|
|
56
|
+
}) => {
|
|
57
|
+
const disabled = !onReconnect;
|
|
58
|
+
return (
|
|
59
|
+
<Tooltip
|
|
60
|
+
content={
|
|
61
|
+
disabled ? "App disconnected" : "App disconnected — click to reconnect"
|
|
62
|
+
}
|
|
63
|
+
>
|
|
64
|
+
{/* Wrapper span keeps the tooltip reachable when the button is
|
|
65
|
+
disabled — a disabled <button> swallows pointer events. */}
|
|
66
|
+
<span tabIndex={disabled ? 0 : -1}>
|
|
67
|
+
<button
|
|
68
|
+
type="button"
|
|
69
|
+
className={cn(topLeftStatus, "bg-transparent border-0 p-0")}
|
|
70
|
+
aria-label={disabled ? "App disconnected" : "Reconnect to app"}
|
|
71
|
+
data-testid="disconnected-indicator"
|
|
72
|
+
onClick={onReconnect}
|
|
73
|
+
disabled={disabled}
|
|
74
|
+
>
|
|
75
|
+
<UnlinkIcon className="w-[25px] h-[25px] text-(--red-11)" />
|
|
76
|
+
</button>
|
|
77
|
+
</span>
|
|
78
|
+
</Tooltip>
|
|
79
|
+
);
|
|
80
|
+
};
|
|
47
81
|
|
|
48
82
|
const LockedIcon = () => (
|
|
49
83
|
<Tooltip content="Notebook locked">
|