@neuralnomads/codenomad-dev 0.10.3-dev-20260215-4f6c8523 → 0.11.1-dev-20260216-e16c5752

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/public/index.html CHANGED
@@ -24,9 +24,9 @@
24
24
  }
25
25
  })()
26
26
  </script>
27
- <script type="module" crossorigin src="/assets/main-EgCUCdV9.js"></script>
28
- <link rel="modulepreload" crossorigin href="/assets/index-D0PYs_Ly.js">
29
- <link rel="stylesheet" crossorigin href="/assets/index-YrKmkw3o.css">
27
+ <script type="module" crossorigin src="/assets/main-DN9FHv8o.js"></script>
28
+ <link rel="modulepreload" crossorigin href="/assets/index-Bi993fzd.js">
29
+ <link rel="stylesheet" crossorigin href="/assets/index-IUhCGbWv.css">
30
30
  <link rel="stylesheet" crossorigin href="/assets/main-HAZkIolJ.css">
31
31
  <link rel="manifest" href="/manifest.webmanifest"><script id="vite-plugin-pwa:register-sw" src="/registerSW.js"></script>
32
32
  <meta name="theme-color" content="#1a1a1a">
@@ -14,9 +14,9 @@
14
14
  }
15
15
  })()
16
16
  </script>
17
- <script type="module" crossorigin src="/assets/loading-CuOFA7ML.js"></script>
18
- <link rel="modulepreload" crossorigin href="/assets/index-D0PYs_Ly.js">
19
- <link rel="stylesheet" crossorigin href="/assets/index-YrKmkw3o.css">
17
+ <script type="module" crossorigin src="/assets/loading-Bu5vIfc2.js"></script>
18
+ <link rel="modulepreload" crossorigin href="/assets/index-Bi993fzd.js">
19
+ <link rel="stylesheet" crossorigin href="/assets/index-IUhCGbWv.css">
20
20
  <link rel="stylesheet" crossorigin href="/assets/loading-CmEVQgyj.css">
21
21
  <link rel="manifest" href="/manifest.webmanifest"><script id="vite-plugin-pwa:register-sw" src="/registerSW.js"></script>
22
22
  <meta name="theme-color" content="#1a1a1a">
package/public/sw.js CHANGED
@@ -1 +1 @@
1
- if(!self.define){let s,l={};const e=(e,r)=>(e=new URL(e+".js",r).href,l[e]||new Promise(l=>{if("document"in self){const s=document.createElement("script");s.src=e,s.onload=l,document.head.appendChild(s)}else s=e,importScripts(e),l()}).then(()=>{let s=l[e];if(!s)throw new Error(`Module ${e} didn’t register its module`);return s}));self.define=(r,i)=>{const n=s||("document"in self?document.currentScript.src:"")||location.href;if(l[n])return;let u={};const a=s=>e(s,n),t={module:{uri:n},exports:u,require:a};l[n]=Promise.all(r.map(s=>t[s]||a(s))).then(s=>(i(...s),u))}}define(["./workbox-60d14903"],function(s){"use strict";self.skipWaiting(),s.clientsClaim(),s.precacheAndRoute([{url:"ui-version.json",revision:"0e015acbf4ec89c7358974061d52f96b"},{url:"registerSW.js",revision:"1872c500de691dce40960bb85481de07"},{url:"pwa-64x64.png",revision:"55fc93ef82eddd32757c324c67f4d8f1"},{url:"pwa-512x512.png",revision:"d8a4783526b318a4a52001e9c9211042"},{url:"pwa-192x192.png",revision:"7d5867127ea8948a52f4eb886da6fca9"},{url:"monaco.worker.js",revision:"47d7f51fe40cc40367130f6c0e059886"},{url:"maskable-icon-512x512.png",revision:"976548c7ab2d6651e02451cffe3439ec"},{url:"manifest.webmanifest",revision:"3587c37712635a9f01cc4906c536d9a9"},{url:"logo.png",revision:"c88fc77a366f290821abc617e67e6674"},{url:"favicon.ico",revision:"62e8e3f28a97401e2932f62503cbdd6b"},{url:"apple-touch-icon-180x180.png",revision:"8252a341f68ec24937e6aaa5d422f87e"},{url:"assets/zig-VOosw3JB.js",revision:null},{url:"assets/zenscript-DVFEvuxE.js",revision:null},{url:"assets/yaml-Buea-lGh.js",revision:null},{url:"assets/xsl-CtQFsRM5.js",revision:null},{url:"assets/xml-sdJ4AIDG.js",revision:null},{url:"assets/wolfram-lXgVvXCa.js",revision:null},{url:"assets/wit-5i3qLPDT.js",revision:null},{url:"assets/wikitext-BhOHFoWU.js",revision:null},{url:"assets/wgsl-Dx-B1_4e.js",revision:null},{url:"assets/wenyan-BV7otONQ.js",revision:null},{url:"assets/wasm-MzD3tlZU.js",revision:null},{url:"assets/wasm-CG6Dc4jp.js",revision:null},{url:"assets/vyper-CDx5xZoG.js",revision:null},{url:"assets/vue-vine-_Ih-lPRR.js",revision:null},{url:"assets/vue-html-DAAvJJDi.js",revision:null},{url:"assets/vue-CCoi5OLL.js",revision:null},{url:"assets/vitesse-light-CVO1_9PV.js",revision:null},{url:"assets/vitesse-dark-D0r3Knsf.js",revision:null},{url:"assets/vitesse-black-Bkuqu6BP.js",revision:null},{url:"assets/viml-CJc9bBzg.js",revision:null},{url:"assets/vhdl-CeAyd5Ju.js",revision:null},{url:"assets/vesper-DU1UobuO.js",revision:null},{url:"assets/verilog-BQ8w6xss.js",revision:null},{url:"assets/vb-D17OF-Vu.js",revision:null},{url:"assets/vala-CsfeWuGM.js",revision:null},{url:"assets/v-BcVCzyr7.js",revision:null},{url:"assets/typst-DHCkPAjA.js",revision:null},{url:"assets/typespec-Df68jz8_.js",revision:null},{url:"assets/typescript-BPQ3VLAy.js",revision:null},{url:"assets/twig-CO9l9SDP.js",revision:null},{url:"assets/turtle-BsS91CYL.js",revision:null},{url:"assets/tsx-COt5Ahok.js",revision:null},{url:"assets/tsv-B_m7g4N7.js",revision:null},{url:"assets/ts-tags-zn1MmPIZ.js",revision:null},{url:"assets/toml-vGWfd6FD.js",revision:null},{url:"assets/tokyo-night-hegEt444.js",revision:null},{url:"assets/tex-Cppo0RY3.js",revision:null},{url:"assets/terraform-BETggiCN.js",revision:null},{url:"assets/templ-W15q3VgB.js",revision:null},{url:"assets/tcl-dwOrl1Do.js",revision:null},{url:"assets/tasl-QIJgUcNo.js",revision:null},{url:"assets/talonscript-CkByrt1z.js",revision:null},{url:"assets/systemd-4A_iFExJ.js",revision:null},{url:"assets/system-verilog-CnnmHF94.js",revision:null},{url:"assets/synthwave-84-CbfX1IO0.js",revision:null},{url:"assets/swift-Dg5xB15N.js",revision:null},{url:"assets/svelte-3Dk4HxPD.js",revision:null},{url:"assets/stylus-BEDo0Tqx.js",revision:null},{url:"assets/stata-BH5u7GGu.js",revision:null},{url:"assets/ssh-config-_ykCGR6B.js",revision:null},{url:"assets/sql-BLtJtn59.js",revision:null},{url:"assets/splunk-BtCnVYZw.js",revision:null},{url:"assets/sparql-rVzFXLq3.js",revision:null},{url:"assets/soy-Brmx7dQM.js",revision:null},{url:"assets/solidity-BbcW6ACK.js",revision:null},{url:"assets/solarized-light-L9t79GZl.js",revision:null},{url:"assets/solarized-dark-DXbdFlpD.js",revision:null},{url:"assets/snazzy-light-Bw305WKR.js",revision:null},{url:"assets/smalltalk-BERRCDM3.js",revision:null},{url:"assets/slack-ochin-DqwNpetd.js",revision:null},{url:"assets/slack-dark-BthQWCQV.js",revision:null},{url:"assets/shellsession-BADoaaVG.js",revision:null},{url:"assets/shellscript-Yzrsuije.js",revision:null},{url:"assets/shaderlab-Dg9Lc6iA.js",revision:null},{url:"assets/sdbl-DVxCFoDh.js",revision:null},{url:"assets/scss-OYdSNvt2.js",revision:null},{url:"assets/scheme-C98Dy4si.js",revision:null},{url:"assets/scala-C151Ov-r.js",revision:null},{url:"assets/sass-Cj5Yp3dK.js",revision:null},{url:"assets/sas-cz2c8ADy.js",revision:null},{url:"assets/rust-B1yitclQ.js",revision:null},{url:"assets/ruby-BvKwtOVI.js",revision:null},{url:"assets/rst-B0xPkSld.js",revision:null},{url:"assets/rosmsg-BJDFO7_C.js",revision:null},{url:"assets/rose-pine-moon-NleAzG8P.js",revision:null},{url:"assets/rose-pine-dawn-CnK8MTSM.js",revision:null},{url:"assets/rose-pine-BHrmToEH.js",revision:null},{url:"assets/riscv-BM1_JUlF.js",revision:null},{url:"assets/rel-C3B-1QV4.js",revision:null},{url:"assets/regexp-CDVJQ6XC.js",revision:null},{url:"assets/reg-C-SQnVFl.js",revision:null},{url:"assets/red-bN70gL4F.js",revision:null},{url:"assets/razor-WgofotgN.js",revision:null},{url:"assets/raku-DXvB9xmW.js",revision:null},{url:"assets/racket-BqYA7rlc.js",revision:null},{url:"assets/r-DiinP2Uv.js",revision:null},{url:"assets/qss-IeuSbFQv.js",revision:null},{url:"assets/qmldir-C8lEn-DE.js",revision:null},{url:"assets/qml-3beO22l8.js",revision:null},{url:"assets/python-B6aJPvgy.js",revision:null},{url:"assets/purescript-CklMAg4u.js",revision:null},{url:"assets/puppet-BMWR74SV.js",revision:null},{url:"assets/pug-CGlum2m_.js",revision:null},{url:"assets/proto-DyJlTyXw.js",revision:null},{url:"assets/prolog-CbFg5uaA.js",revision:null},{url:"assets/prisma-Dd19v3D-.js",revision:null},{url:"assets/powershell-Dpen1YoG.js",revision:null},{url:"assets/powerquery-CEu0bR-o.js",revision:null},{url:"assets/postcss-CXtECtnM.js",revision:null},{url:"assets/polar-C0HS_06l.js",revision:null},{url:"assets/poimandres-CS3Unz2-.js",revision:null},{url:"assets/po-BTJTHyun.js",revision:null},{url:"assets/plsql-ChMvpjG-.js",revision:null},{url:"assets/plastic-3e1v2bzS.js",revision:null},{url:"assets/pkl-u5AG7uiY.js",revision:null},{url:"assets/php-CDn_0X-4.js",revision:null},{url:"assets/perl-C0TMdlhV.js",revision:null},{url:"assets/pascal-D93ZcfNL.js",revision:null},{url:"assets/one-light-PoHY5YXO.js",revision:null},{url:"assets/one-dark-pro-DVMEJ2y_.js",revision:null},{url:"assets/ocaml-C0hk2d4L.js",revision:null},{url:"assets/objective-cpp-CLxacb5B.js",revision:null},{url:"assets/objective-c-DXmwc3jG.js",revision:null},{url:"assets/nushell-C-sUppwS.js",revision:null},{url:"assets/nord-Ddv68eIx.js",revision:null},{url:"assets/nix-BbRYJGeE.js",revision:null},{url:"assets/nim-CVrawwO9.js",revision:null},{url:"assets/night-owl-C39BiMTA.js",revision:null},{url:"assets/nginx-DknmC5AR.js",revision:null},{url:"assets/nextflow-CUEJCptM.js",revision:null},{url:"assets/narrat-DRg8JJMk.js",revision:null},{url:"assets/move-Bu9oaDYs.js",revision:null},{url:"assets/monokai-D4h5O-jR.js",revision:null},{url:"assets/mojo-1DNp92w6.js",revision:null},{url:"assets/mipsasm-CKIfxQSi.js",revision:null},{url:"assets/min-light-CTRr51gU.js",revision:null},{url:"assets/min-dark-CafNBF8u.js",revision:null},{url:"assets/mermaid-DKYwYmdq.js",revision:null},{url:"assets/mdx-Cmh6b_Ma.js",revision:null},{url:"assets/mdc-DUICxH0z.js",revision:null},{url:"assets/matlab-D7o27uSR.js",revision:null},{url:"assets/material-theme-palenight-Csfq5Kiy.js",revision:null},{url:"assets/material-theme-ocean-CyktbL80.js",revision:null},{url:"assets/material-theme-lighter-B0m2ddpp.js",revision:null},{url:"assets/material-theme-darker-BfHTSMKl.js",revision:null},{url:"assets/material-theme-D5KoaKCx.js",revision:null},{url:"assets/marko-CPi9NSCl.js",revision:null},{url:"assets/markdown-Cvjx9yec.js",revision:null},{url:"assets/make-CHLpvVh8.js",revision:null},{url:"assets/main-HAZkIolJ.css",revision:null},{url:"assets/main-EgCUCdV9.js",revision:null},{url:"assets/luau-CXu1NL6O.js",revision:null},{url:"assets/lua-BbnMAYS6.js",revision:null},{url:"assets/logo-BtOb2qkB.js",revision:null},{url:"assets/log-2UxHyX5q.js",revision:null},{url:"assets/loading-CuOFA7ML.js",revision:null},{url:"assets/loading-CmEVQgyj.css",revision:null},{url:"assets/llvm-BtvRca6l.js",revision:null},{url:"assets/liquid-DYVedYrR.js",revision:null},{url:"assets/light-plus-B7mTdjB0.js",revision:null},{url:"assets/less-B1dDrJ26.js",revision:null},{url:"assets/lean-DP1Csr6i.js",revision:null},{url:"assets/latex-BUKiar2Z.js",revision:null},{url:"assets/laserwave-DUszq2jm.js",revision:null},{url:"assets/kusto-BvAqAH-y.js",revision:null},{url:"assets/kotlin-BdnUsdx6.js",revision:null},{url:"assets/kdl-DV7GczEv.js",revision:null},{url:"assets/kanagawa-wave-DWedfzmr.js",revision:null},{url:"assets/kanagawa-lotus-CfQXZHmo.js",revision:null},{url:"assets/kanagawa-dragon-CkXjmgJE.js",revision:null},{url:"assets/julia-C8NyazO9.js",revision:null},{url:"assets/jsx-g9-lgVsj.js",revision:null},{url:"assets/jssm-C2t-YnRu.js",revision:null},{url:"assets/jsonnet-DFQXde-d.js",revision:null},{url:"assets/jsonl-DcaNXYhu.js",revision:null},{url:"assets/jsonc-Des-eS-w.js",revision:null},{url:"assets/json5-C9tS-k6U.js",revision:null},{url:"assets/json-Cp-IABpG.js",revision:null},{url:"assets/jison-wvAkD_A8.js",revision:null},{url:"assets/jinja-4LBKfQ-Z.js",revision:null},{url:"assets/javascript-wDzz0qaB.js",revision:null},{url:"assets/java-CylS5w8V.js",revision:null},{url:"assets/ini-BEwlwnbL.js",revision:null},{url:"assets/index-YrKmkw3o.css",revision:null},{url:"assets/index-DdQ7zIzB.js",revision:null},{url:"assets/index-DN20ggb1.js",revision:null},{url:"assets/index-D4PT0yE4.js",revision:null},{url:"assets/index-D0PYs_Ly.js",revision:null},{url:"assets/imba-DGztddWO.js",revision:null},{url:"assets/hy-DFXneXwc.js",revision:null},{url:"assets/hxml-Bvhsp5Yf.js",revision:null},{url:"assets/hurl-irOxFIW8.js",revision:null},{url:"assets/http-jrhK8wxY.js",revision:null},{url:"assets/html-derivative-BFtXZ54Q.js",revision:null},{url:"assets/html-GMplVEZG.js",revision:null},{url:"assets/houston-DnULxvSX.js",revision:null},{url:"assets/hlsl-D3lLCCz7.js",revision:null},{url:"assets/hjson-D5-asLiD.js",revision:null},{url:"assets/hcl-BWvSN4gD.js",revision:null},{url:"assets/haxe-CzTSHFRz.js",revision:null},{url:"assets/haskell-Df6bDoY_.js",revision:null},{url:"assets/handlebars-BL8al0AC.js",revision:null},{url:"assets/haml-B8DHNrY2.js",revision:null},{url:"assets/hack-CaT9iCJl.js",revision:null},{url:"assets/gruvbox-light-soft-hJgmCMqR.js",revision:null},{url:"assets/gruvbox-light-medium-DRw_LuNl.js",revision:null},{url:"assets/gruvbox-light-hard-CH1njM8p.js",revision:null},{url:"assets/gruvbox-dark-soft-CVdnzihN.js",revision:null},{url:"assets/gruvbox-dark-medium-GsRaNv29.js",revision:null},{url:"assets/gruvbox-dark-hard-CFHQjOhq.js",revision:null},{url:"assets/groovy-gcz8RCvz.js",revision:null},{url:"assets/graphql-ChdNCCLP.js",revision:null},{url:"assets/go-Dn2_MT6a.js",revision:null},{url:"assets/gnuplot-DdkO51Og.js",revision:null},{url:"assets/glsl-DplSGwfg.js",revision:null},{url:"assets/glimmer-ts-U6CK756n.js",revision:null},{url:"assets/glimmer-js-Rg0-pVw9.js",revision:null},{url:"assets/gleam-BspZqrRM.js",revision:null},{url:"assets/github-light-high-contrast-BfjtVDDH.js",revision:null},{url:"assets/github-light-default-D7oLnXFd.js",revision:null},{url:"assets/github-light-DAi9KRSo.js",revision:null},{url:"assets/github-dark-high-contrast-E3gJ1_iC.js",revision:null},{url:"assets/github-dark-dimmed-DH5Ifo-i.js",revision:null},{url:"assets/github-dark-default-Cuk6v7N8.js",revision:null},{url:"assets/github-dark-DHJKELXO.js",revision:null},{url:"assets/git-rebase-r7XF79zn.js",revision:null},{url:"assets/git-commit-F4YmCXRG.js",revision:null},{url:"assets/gherkin-DyxjwDmM.js",revision:null},{url:"assets/genie-D0YGMca9.js",revision:null},{url:"assets/gdshader-DkwncUOv.js",revision:null},{url:"assets/gdscript-DTMYz4Jt.js",revision:null},{url:"assets/gdresource-B7Tvp0Sc.js",revision:null},{url:"assets/fsharp-CXgrBDvD.js",revision:null},{url:"assets/fortran-free-form-D22FLkUw.js",revision:null},{url:"assets/fortran-fixed-form-BZjJHVRy.js",revision:null},{url:"assets/fluent-C4IJs8-o.js",revision:null},{url:"assets/fish-BvzEVeQv.js",revision:null},{url:"assets/fennel-BYunw83y.js",revision:null},{url:"assets/everforest-light-C8M2exoo.js",revision:null},{url:"assets/everforest-dark-BgDCqdQA.js",revision:null},{url:"assets/erlang-DsQrWhSR.js",revision:null},{url:"assets/erb-BOJIQeun.js",revision:null},{url:"assets/emacs-lisp-C9XAeP06.js",revision:null},{url:"assets/elm-DbKCFpqz.js",revision:null},{url:"assets/elixir-CDX3lj18.js",revision:null},{url:"assets/edge-BkV0erSs.js",revision:null},{url:"assets/dream-maker-BtqSS_iP.js",revision:null},{url:"assets/dracula-soft-BXkSAIEj.js",revision:null},{url:"assets/dracula-BzJJZx-M.js",revision:null},{url:"assets/dotenv-Da5cRb03.js",revision:null},{url:"assets/docker-BcOcwvcX.js",revision:null},{url:"assets/diff-D97Zzqfu.js",revision:null},{url:"assets/desktop-BmXAJ9_W.js",revision:null},{url:"assets/dax-CEL-wOlO.js",revision:null},{url:"assets/dart-CF10PKvl.js",revision:null},{url:"assets/dark-plus-eOWES_5F.js",revision:null},{url:"assets/d-85-TOEBH.js",revision:null},{url:"assets/cypher-COkxafJQ.js",revision:null},{url:"assets/cue-D82EKSYY.js",revision:null},{url:"assets/csv-fuZLfV_i.js",revision:null},{url:"assets/css-DPfMkruS.js",revision:null},{url:"assets/csharp-CX12Zw3r.js",revision:null},{url:"assets/crystal-tKQVLTB8.js",revision:null},{url:"assets/cpp-CofmeUqb.js",revision:null},{url:"assets/core-BSTVzpXI.js",revision:null},{url:"assets/coq-DkFqJrB1.js",revision:null},{url:"assets/common-lisp-Cg-RD9OK.js",revision:null},{url:"assets/coffee-Ch7k5sss.js",revision:null},{url:"assets/codeql-DsOJ9woJ.js",revision:null},{url:"assets/codeowners-Bp6g37R7.js",revision:null},{url:"assets/cobol-nwyudZeR.js",revision:null},{url:"assets/cmake-D1j8_8rp.js",revision:null},{url:"assets/clojure-P80f7IUj.js",revision:null},{url:"assets/clarity-D53aC0YG.js",revision:null},{url:"assets/catppuccin-mocha-D87Tk5Gz.js",revision:null},{url:"assets/catppuccin-macchiato-DQyhUUbL.js",revision:null},{url:"assets/catppuccin-latte-C9dUb6Cb.js",revision:null},{url:"assets/catppuccin-frappe-DFWUc33u.js",revision:null},{url:"assets/cairo-KRGpt6FW.js",revision:null},{url:"assets/cadence-Bv_4Rxtq.js",revision:null},{url:"assets/c-BIGW1oBm.js",revision:null},{url:"assets/bsl-BO_Y6i37.js",revision:null},{url:"assets/blade-DVc8C-J4.js",revision:null},{url:"assets/bicep-Bmn6On1c.js",revision:null},{url:"assets/bibtex-CHM0blh-.js",revision:null},{url:"assets/berry-D08WgyRC.js",revision:null},{url:"assets/beancount-k_qm7-4y.js",revision:null},{url:"assets/bat-BkioyH1T.js",revision:null},{url:"assets/ballerina-BFfxhgS-.js",revision:null},{url:"assets/ayu-dark-Cv9koXgw.js",revision:null},{url:"assets/awk-DMzUqQB5.js",revision:null},{url:"assets/aurora-x-D-2ljcwZ.js",revision:null},{url:"assets/astro-CbQHKStN.js",revision:null},{url:"assets/asm-D_Q5rh1f.js",revision:null},{url:"assets/asciidoc-Dv7Oe6Be.js",revision:null},{url:"assets/ara-BRHolxvo.js",revision:null},{url:"assets/applescript-Co6uUVPk.js",revision:null},{url:"assets/apl-dKokRX4l.js",revision:null},{url:"assets/apex-DhZLUxFE.js",revision:null},{url:"assets/apache-Pmp26Uib.js",revision:null},{url:"assets/angular-ts-BwZT4LLn.js",revision:null},{url:"assets/angular-html-CU67Zn6k.js",revision:null},{url:"assets/andromeeda-C-Jbm3Hp.js",revision:null},{url:"assets/ada-bCR0ucgS.js",revision:null},{url:"assets/actionscript-3-CfeIJUat.js",revision:null},{url:"assets/abap-BdImnpbu.js",revision:null},{url:"assets/CodeNomad-Icon-bmTWNPXy.png",revision:null},{url:"manifest.webmanifest",revision:"4f4aaad9542bc67e284999ba91ccc883"}],{}),s.cleanupOutdatedCaches(),s.registerRoute(({url:s,request:l})=>!s.pathname.startsWith("/api/")&&("document"!==l.destination&&["script","style","image","font"].includes(l.destination)),new s.CacheFirst({cacheName:"asset-cache",plugins:[new s.ExpirationPlugin({maxEntries:200,maxAgeSeconds:2592e3}),new s.CacheableResponsePlugin({statuses:[0,200]})]}),"GET")});
1
+ if(!self.define){let s,l={};const e=(e,r)=>(e=new URL(e+".js",r).href,l[e]||new Promise(l=>{if("document"in self){const s=document.createElement("script");s.src=e,s.onload=l,document.head.appendChild(s)}else s=e,importScripts(e),l()}).then(()=>{let s=l[e];if(!s)throw new Error(`Module ${e} didn’t register its module`);return s}));self.define=(r,i)=>{const n=s||("document"in self?document.currentScript.src:"")||location.href;if(l[n])return;let u={};const a=s=>e(s,n),t={module:{uri:n},exports:u,require:a};l[n]=Promise.all(r.map(s=>t[s]||a(s))).then(s=>(i(...s),u))}}define(["./workbox-60d14903"],function(s){"use strict";self.skipWaiting(),s.clientsClaim(),s.precacheAndRoute([{url:"ui-version.json",revision:"ffd3002f215166a9a2a09f979b55db75"},{url:"registerSW.js",revision:"1872c500de691dce40960bb85481de07"},{url:"pwa-64x64.png",revision:"55fc93ef82eddd32757c324c67f4d8f1"},{url:"pwa-512x512.png",revision:"d8a4783526b318a4a52001e9c9211042"},{url:"pwa-192x192.png",revision:"7d5867127ea8948a52f4eb886da6fca9"},{url:"monaco.worker.js",revision:"47d7f51fe40cc40367130f6c0e059886"},{url:"maskable-icon-512x512.png",revision:"976548c7ab2d6651e02451cffe3439ec"},{url:"manifest.webmanifest",revision:"3587c37712635a9f01cc4906c536d9a9"},{url:"logo.png",revision:"c88fc77a366f290821abc617e67e6674"},{url:"favicon.ico",revision:"62e8e3f28a97401e2932f62503cbdd6b"},{url:"apple-touch-icon-180x180.png",revision:"8252a341f68ec24937e6aaa5d422f87e"},{url:"assets/zig-VOosw3JB.js",revision:null},{url:"assets/zenscript-DVFEvuxE.js",revision:null},{url:"assets/yaml-Buea-lGh.js",revision:null},{url:"assets/xsl-CtQFsRM5.js",revision:null},{url:"assets/xml-sdJ4AIDG.js",revision:null},{url:"assets/wolfram-lXgVvXCa.js",revision:null},{url:"assets/wit-5i3qLPDT.js",revision:null},{url:"assets/wikitext-BhOHFoWU.js",revision:null},{url:"assets/wgsl-Dx-B1_4e.js",revision:null},{url:"assets/wenyan-BV7otONQ.js",revision:null},{url:"assets/wasm-MzD3tlZU.js",revision:null},{url:"assets/wasm-CG6Dc4jp.js",revision:null},{url:"assets/vyper-CDx5xZoG.js",revision:null},{url:"assets/vue-vine-_Ih-lPRR.js",revision:null},{url:"assets/vue-html-DAAvJJDi.js",revision:null},{url:"assets/vue-CCoi5OLL.js",revision:null},{url:"assets/vitesse-light-CVO1_9PV.js",revision:null},{url:"assets/vitesse-dark-D0r3Knsf.js",revision:null},{url:"assets/vitesse-black-Bkuqu6BP.js",revision:null},{url:"assets/viml-CJc9bBzg.js",revision:null},{url:"assets/vhdl-CeAyd5Ju.js",revision:null},{url:"assets/vesper-DU1UobuO.js",revision:null},{url:"assets/verilog-BQ8w6xss.js",revision:null},{url:"assets/vb-D17OF-Vu.js",revision:null},{url:"assets/vala-CsfeWuGM.js",revision:null},{url:"assets/v-BcVCzyr7.js",revision:null},{url:"assets/typst-DHCkPAjA.js",revision:null},{url:"assets/typespec-Df68jz8_.js",revision:null},{url:"assets/typescript-BPQ3VLAy.js",revision:null},{url:"assets/twig-CO9l9SDP.js",revision:null},{url:"assets/turtle-BsS91CYL.js",revision:null},{url:"assets/tsx-COt5Ahok.js",revision:null},{url:"assets/tsv-B_m7g4N7.js",revision:null},{url:"assets/ts-tags-zn1MmPIZ.js",revision:null},{url:"assets/toml-vGWfd6FD.js",revision:null},{url:"assets/tokyo-night-hegEt444.js",revision:null},{url:"assets/tex-Cppo0RY3.js",revision:null},{url:"assets/terraform-BETggiCN.js",revision:null},{url:"assets/templ-W15q3VgB.js",revision:null},{url:"assets/tcl-dwOrl1Do.js",revision:null},{url:"assets/tasl-QIJgUcNo.js",revision:null},{url:"assets/talonscript-CkByrt1z.js",revision:null},{url:"assets/systemd-4A_iFExJ.js",revision:null},{url:"assets/system-verilog-CnnmHF94.js",revision:null},{url:"assets/synthwave-84-CbfX1IO0.js",revision:null},{url:"assets/swift-Dg5xB15N.js",revision:null},{url:"assets/svelte-3Dk4HxPD.js",revision:null},{url:"assets/stylus-BEDo0Tqx.js",revision:null},{url:"assets/stata-BH5u7GGu.js",revision:null},{url:"assets/ssh-config-_ykCGR6B.js",revision:null},{url:"assets/sql-BLtJtn59.js",revision:null},{url:"assets/splunk-BtCnVYZw.js",revision:null},{url:"assets/sparql-rVzFXLq3.js",revision:null},{url:"assets/soy-Brmx7dQM.js",revision:null},{url:"assets/solidity-BbcW6ACK.js",revision:null},{url:"assets/solarized-light-L9t79GZl.js",revision:null},{url:"assets/solarized-dark-DXbdFlpD.js",revision:null},{url:"assets/snazzy-light-Bw305WKR.js",revision:null},{url:"assets/smalltalk-BERRCDM3.js",revision:null},{url:"assets/slack-ochin-DqwNpetd.js",revision:null},{url:"assets/slack-dark-BthQWCQV.js",revision:null},{url:"assets/shellsession-BADoaaVG.js",revision:null},{url:"assets/shellscript-Yzrsuije.js",revision:null},{url:"assets/shaderlab-Dg9Lc6iA.js",revision:null},{url:"assets/sdbl-DVxCFoDh.js",revision:null},{url:"assets/scss-OYdSNvt2.js",revision:null},{url:"assets/scheme-C98Dy4si.js",revision:null},{url:"assets/scala-C151Ov-r.js",revision:null},{url:"assets/sass-Cj5Yp3dK.js",revision:null},{url:"assets/sas-cz2c8ADy.js",revision:null},{url:"assets/rust-B1yitclQ.js",revision:null},{url:"assets/ruby-BvKwtOVI.js",revision:null},{url:"assets/rst-B0xPkSld.js",revision:null},{url:"assets/rosmsg-BJDFO7_C.js",revision:null},{url:"assets/rose-pine-moon-NleAzG8P.js",revision:null},{url:"assets/rose-pine-dawn-CnK8MTSM.js",revision:null},{url:"assets/rose-pine-BHrmToEH.js",revision:null},{url:"assets/riscv-BM1_JUlF.js",revision:null},{url:"assets/rel-C3B-1QV4.js",revision:null},{url:"assets/regexp-CDVJQ6XC.js",revision:null},{url:"assets/reg-C-SQnVFl.js",revision:null},{url:"assets/red-bN70gL4F.js",revision:null},{url:"assets/razor-WgofotgN.js",revision:null},{url:"assets/raku-DXvB9xmW.js",revision:null},{url:"assets/racket-BqYA7rlc.js",revision:null},{url:"assets/r-DiinP2Uv.js",revision:null},{url:"assets/qss-IeuSbFQv.js",revision:null},{url:"assets/qmldir-C8lEn-DE.js",revision:null},{url:"assets/qml-3beO22l8.js",revision:null},{url:"assets/python-B6aJPvgy.js",revision:null},{url:"assets/purescript-CklMAg4u.js",revision:null},{url:"assets/puppet-BMWR74SV.js",revision:null},{url:"assets/pug-CGlum2m_.js",revision:null},{url:"assets/proto-DyJlTyXw.js",revision:null},{url:"assets/prolog-CbFg5uaA.js",revision:null},{url:"assets/prisma-Dd19v3D-.js",revision:null},{url:"assets/powershell-Dpen1YoG.js",revision:null},{url:"assets/powerquery-CEu0bR-o.js",revision:null},{url:"assets/postcss-CXtECtnM.js",revision:null},{url:"assets/polar-C0HS_06l.js",revision:null},{url:"assets/poimandres-CS3Unz2-.js",revision:null},{url:"assets/po-BTJTHyun.js",revision:null},{url:"assets/plsql-ChMvpjG-.js",revision:null},{url:"assets/plastic-3e1v2bzS.js",revision:null},{url:"assets/pkl-u5AG7uiY.js",revision:null},{url:"assets/php-CDn_0X-4.js",revision:null},{url:"assets/perl-C0TMdlhV.js",revision:null},{url:"assets/pascal-D93ZcfNL.js",revision:null},{url:"assets/one-light-PoHY5YXO.js",revision:null},{url:"assets/one-dark-pro-DVMEJ2y_.js",revision:null},{url:"assets/ocaml-C0hk2d4L.js",revision:null},{url:"assets/objective-cpp-CLxacb5B.js",revision:null},{url:"assets/objective-c-DXmwc3jG.js",revision:null},{url:"assets/nushell-C-sUppwS.js",revision:null},{url:"assets/nord-Ddv68eIx.js",revision:null},{url:"assets/nix-BbRYJGeE.js",revision:null},{url:"assets/nim-CVrawwO9.js",revision:null},{url:"assets/night-owl-C39BiMTA.js",revision:null},{url:"assets/nginx-DknmC5AR.js",revision:null},{url:"assets/nextflow-CUEJCptM.js",revision:null},{url:"assets/narrat-DRg8JJMk.js",revision:null},{url:"assets/move-Bu9oaDYs.js",revision:null},{url:"assets/monokai-D4h5O-jR.js",revision:null},{url:"assets/mojo-1DNp92w6.js",revision:null},{url:"assets/mipsasm-CKIfxQSi.js",revision:null},{url:"assets/min-light-CTRr51gU.js",revision:null},{url:"assets/min-dark-CafNBF8u.js",revision:null},{url:"assets/mermaid-DKYwYmdq.js",revision:null},{url:"assets/mdx-Cmh6b_Ma.js",revision:null},{url:"assets/mdc-DUICxH0z.js",revision:null},{url:"assets/matlab-D7o27uSR.js",revision:null},{url:"assets/material-theme-palenight-Csfq5Kiy.js",revision:null},{url:"assets/material-theme-ocean-CyktbL80.js",revision:null},{url:"assets/material-theme-lighter-B0m2ddpp.js",revision:null},{url:"assets/material-theme-darker-BfHTSMKl.js",revision:null},{url:"assets/material-theme-D5KoaKCx.js",revision:null},{url:"assets/marko-CPi9NSCl.js",revision:null},{url:"assets/markdown-Cvjx9yec.js",revision:null},{url:"assets/make-CHLpvVh8.js",revision:null},{url:"assets/main-HAZkIolJ.css",revision:null},{url:"assets/main-DN9FHv8o.js",revision:null},{url:"assets/luau-CXu1NL6O.js",revision:null},{url:"assets/lua-BbnMAYS6.js",revision:null},{url:"assets/logo-BtOb2qkB.js",revision:null},{url:"assets/log-2UxHyX5q.js",revision:null},{url:"assets/loading-CmEVQgyj.css",revision:null},{url:"assets/loading-Bu5vIfc2.js",revision:null},{url:"assets/llvm-BtvRca6l.js",revision:null},{url:"assets/liquid-DYVedYrR.js",revision:null},{url:"assets/light-plus-B7mTdjB0.js",revision:null},{url:"assets/less-B1dDrJ26.js",revision:null},{url:"assets/lean-DP1Csr6i.js",revision:null},{url:"assets/latex-BUKiar2Z.js",revision:null},{url:"assets/laserwave-DUszq2jm.js",revision:null},{url:"assets/kusto-BvAqAH-y.js",revision:null},{url:"assets/kotlin-BdnUsdx6.js",revision:null},{url:"assets/kdl-DV7GczEv.js",revision:null},{url:"assets/kanagawa-wave-DWedfzmr.js",revision:null},{url:"assets/kanagawa-lotus-CfQXZHmo.js",revision:null},{url:"assets/kanagawa-dragon-CkXjmgJE.js",revision:null},{url:"assets/julia-C8NyazO9.js",revision:null},{url:"assets/jsx-g9-lgVsj.js",revision:null},{url:"assets/jssm-C2t-YnRu.js",revision:null},{url:"assets/jsonnet-DFQXde-d.js",revision:null},{url:"assets/jsonl-DcaNXYhu.js",revision:null},{url:"assets/jsonc-Des-eS-w.js",revision:null},{url:"assets/json5-C9tS-k6U.js",revision:null},{url:"assets/json-Cp-IABpG.js",revision:null},{url:"assets/jison-wvAkD_A8.js",revision:null},{url:"assets/jinja-4LBKfQ-Z.js",revision:null},{url:"assets/javascript-wDzz0qaB.js",revision:null},{url:"assets/java-CylS5w8V.js",revision:null},{url:"assets/ini-BEwlwnbL.js",revision:null},{url:"assets/index-IUhCGbWv.css",revision:null},{url:"assets/index-DdQ7zIzB.js",revision:null},{url:"assets/index-DN20ggb1.js",revision:null},{url:"assets/index-D4PT0yE4.js",revision:null},{url:"assets/index-Bi993fzd.js",revision:null},{url:"assets/imba-DGztddWO.js",revision:null},{url:"assets/hy-DFXneXwc.js",revision:null},{url:"assets/hxml-Bvhsp5Yf.js",revision:null},{url:"assets/hurl-irOxFIW8.js",revision:null},{url:"assets/http-jrhK8wxY.js",revision:null},{url:"assets/html-derivative-BFtXZ54Q.js",revision:null},{url:"assets/html-GMplVEZG.js",revision:null},{url:"assets/houston-DnULxvSX.js",revision:null},{url:"assets/hlsl-D3lLCCz7.js",revision:null},{url:"assets/hjson-D5-asLiD.js",revision:null},{url:"assets/hcl-BWvSN4gD.js",revision:null},{url:"assets/haxe-CzTSHFRz.js",revision:null},{url:"assets/haskell-Df6bDoY_.js",revision:null},{url:"assets/handlebars-BL8al0AC.js",revision:null},{url:"assets/haml-B8DHNrY2.js",revision:null},{url:"assets/hack-CaT9iCJl.js",revision:null},{url:"assets/gruvbox-light-soft-hJgmCMqR.js",revision:null},{url:"assets/gruvbox-light-medium-DRw_LuNl.js",revision:null},{url:"assets/gruvbox-light-hard-CH1njM8p.js",revision:null},{url:"assets/gruvbox-dark-soft-CVdnzihN.js",revision:null},{url:"assets/gruvbox-dark-medium-GsRaNv29.js",revision:null},{url:"assets/gruvbox-dark-hard-CFHQjOhq.js",revision:null},{url:"assets/groovy-gcz8RCvz.js",revision:null},{url:"assets/graphql-ChdNCCLP.js",revision:null},{url:"assets/go-Dn2_MT6a.js",revision:null},{url:"assets/gnuplot-DdkO51Og.js",revision:null},{url:"assets/glsl-DplSGwfg.js",revision:null},{url:"assets/glimmer-ts-U6CK756n.js",revision:null},{url:"assets/glimmer-js-Rg0-pVw9.js",revision:null},{url:"assets/gleam-BspZqrRM.js",revision:null},{url:"assets/github-light-high-contrast-BfjtVDDH.js",revision:null},{url:"assets/github-light-default-D7oLnXFd.js",revision:null},{url:"assets/github-light-DAi9KRSo.js",revision:null},{url:"assets/github-dark-high-contrast-E3gJ1_iC.js",revision:null},{url:"assets/github-dark-dimmed-DH5Ifo-i.js",revision:null},{url:"assets/github-dark-default-Cuk6v7N8.js",revision:null},{url:"assets/github-dark-DHJKELXO.js",revision:null},{url:"assets/git-rebase-r7XF79zn.js",revision:null},{url:"assets/git-commit-F4YmCXRG.js",revision:null},{url:"assets/gherkin-DyxjwDmM.js",revision:null},{url:"assets/genie-D0YGMca9.js",revision:null},{url:"assets/gdshader-DkwncUOv.js",revision:null},{url:"assets/gdscript-DTMYz4Jt.js",revision:null},{url:"assets/gdresource-B7Tvp0Sc.js",revision:null},{url:"assets/fsharp-CXgrBDvD.js",revision:null},{url:"assets/fortran-free-form-D22FLkUw.js",revision:null},{url:"assets/fortran-fixed-form-BZjJHVRy.js",revision:null},{url:"assets/fluent-C4IJs8-o.js",revision:null},{url:"assets/fish-BvzEVeQv.js",revision:null},{url:"assets/fennel-BYunw83y.js",revision:null},{url:"assets/everforest-light-C8M2exoo.js",revision:null},{url:"assets/everforest-dark-BgDCqdQA.js",revision:null},{url:"assets/erlang-DsQrWhSR.js",revision:null},{url:"assets/erb-BOJIQeun.js",revision:null},{url:"assets/emacs-lisp-C9XAeP06.js",revision:null},{url:"assets/elm-DbKCFpqz.js",revision:null},{url:"assets/elixir-CDX3lj18.js",revision:null},{url:"assets/edge-BkV0erSs.js",revision:null},{url:"assets/dream-maker-BtqSS_iP.js",revision:null},{url:"assets/dracula-soft-BXkSAIEj.js",revision:null},{url:"assets/dracula-BzJJZx-M.js",revision:null},{url:"assets/dotenv-Da5cRb03.js",revision:null},{url:"assets/docker-BcOcwvcX.js",revision:null},{url:"assets/diff-D97Zzqfu.js",revision:null},{url:"assets/desktop-BmXAJ9_W.js",revision:null},{url:"assets/dax-CEL-wOlO.js",revision:null},{url:"assets/dart-CF10PKvl.js",revision:null},{url:"assets/dark-plus-eOWES_5F.js",revision:null},{url:"assets/d-85-TOEBH.js",revision:null},{url:"assets/cypher-COkxafJQ.js",revision:null},{url:"assets/cue-D82EKSYY.js",revision:null},{url:"assets/csv-fuZLfV_i.js",revision:null},{url:"assets/css-DPfMkruS.js",revision:null},{url:"assets/csharp-CX12Zw3r.js",revision:null},{url:"assets/crystal-tKQVLTB8.js",revision:null},{url:"assets/cpp-CofmeUqb.js",revision:null},{url:"assets/core-BSTVzpXI.js",revision:null},{url:"assets/coq-DkFqJrB1.js",revision:null},{url:"assets/common-lisp-Cg-RD9OK.js",revision:null},{url:"assets/coffee-Ch7k5sss.js",revision:null},{url:"assets/codeql-DsOJ9woJ.js",revision:null},{url:"assets/codeowners-Bp6g37R7.js",revision:null},{url:"assets/cobol-nwyudZeR.js",revision:null},{url:"assets/cmake-D1j8_8rp.js",revision:null},{url:"assets/clojure-P80f7IUj.js",revision:null},{url:"assets/clarity-D53aC0YG.js",revision:null},{url:"assets/catppuccin-mocha-D87Tk5Gz.js",revision:null},{url:"assets/catppuccin-macchiato-DQyhUUbL.js",revision:null},{url:"assets/catppuccin-latte-C9dUb6Cb.js",revision:null},{url:"assets/catppuccin-frappe-DFWUc33u.js",revision:null},{url:"assets/cairo-KRGpt6FW.js",revision:null},{url:"assets/cadence-Bv_4Rxtq.js",revision:null},{url:"assets/c-BIGW1oBm.js",revision:null},{url:"assets/bsl-BO_Y6i37.js",revision:null},{url:"assets/blade-DVc8C-J4.js",revision:null},{url:"assets/bicep-Bmn6On1c.js",revision:null},{url:"assets/bibtex-CHM0blh-.js",revision:null},{url:"assets/berry-D08WgyRC.js",revision:null},{url:"assets/beancount-k_qm7-4y.js",revision:null},{url:"assets/bat-BkioyH1T.js",revision:null},{url:"assets/ballerina-BFfxhgS-.js",revision:null},{url:"assets/ayu-dark-Cv9koXgw.js",revision:null},{url:"assets/awk-DMzUqQB5.js",revision:null},{url:"assets/aurora-x-D-2ljcwZ.js",revision:null},{url:"assets/astro-CbQHKStN.js",revision:null},{url:"assets/asm-D_Q5rh1f.js",revision:null},{url:"assets/asciidoc-Dv7Oe6Be.js",revision:null},{url:"assets/ara-BRHolxvo.js",revision:null},{url:"assets/applescript-Co6uUVPk.js",revision:null},{url:"assets/apl-dKokRX4l.js",revision:null},{url:"assets/apex-DhZLUxFE.js",revision:null},{url:"assets/apache-Pmp26Uib.js",revision:null},{url:"assets/angular-ts-BwZT4LLn.js",revision:null},{url:"assets/angular-html-CU67Zn6k.js",revision:null},{url:"assets/andromeeda-C-Jbm3Hp.js",revision:null},{url:"assets/ada-bCR0ucgS.js",revision:null},{url:"assets/actionscript-3-CfeIJUat.js",revision:null},{url:"assets/abap-BdImnpbu.js",revision:null},{url:"assets/CodeNomad-Icon-bmTWNPXy.png",revision:null},{url:"manifest.webmanifest",revision:"4f4aaad9542bc67e284999ba91ccc883"}],{}),s.cleanupOutdatedCaches(),s.registerRoute(({url:s,request:l})=>!s.pathname.startsWith("/api/")&&("document"!==l.destination&&["script","style","image","font"].includes(l.destination)),new s.CacheFirst({cacheName:"asset-cache",plugins:[new s.ExpirationPlugin({maxEntries:200,maxAgeSeconds:2592e3}),new s.CacheableResponsePlugin({statuses:[0,200]})]}),"GET")});
@@ -1,3 +1,3 @@
1
1
  {
2
- "uiVersion": "0.10.3"
2
+ "uiVersion": "0.11.1"
3
3
  }
@@ -1,148 +0,0 @@
1
- import { spawnSync } from "child_process";
2
- import { buildSpawnSpec } from "../workspaces/runtime";
3
- export class BinaryRegistry {
4
- constructor(configStore, eventBus, logger) {
5
- this.configStore = configStore;
6
- this.eventBus = eventBus;
7
- this.logger = logger;
8
- }
9
- list() {
10
- return this.mapRecords();
11
- }
12
- resolveDefault() {
13
- const binaries = this.mapRecords();
14
- if (binaries.length === 0) {
15
- this.logger.warn("No configured binaries found, falling back to opencode");
16
- return this.buildFallbackRecord("opencode");
17
- }
18
- return binaries.find((binary) => binary.isDefault) ?? binaries[0];
19
- }
20
- create(request) {
21
- this.logger.debug({ path: request.path }, "Registering OpenCode binary");
22
- const entry = {
23
- path: request.path,
24
- version: undefined,
25
- lastUsed: Date.now(),
26
- label: request.label,
27
- };
28
- const config = this.configStore.get();
29
- const nextConfig = this.cloneConfig(config);
30
- const deduped = nextConfig.opencodeBinaries.filter((binary) => binary.path !== request.path);
31
- nextConfig.opencodeBinaries = [entry, ...deduped];
32
- if (request.makeDefault) {
33
- nextConfig.preferences.lastUsedBinary = request.path;
34
- }
35
- this.configStore.replace(nextConfig);
36
- const record = this.getById(request.path);
37
- this.emitChange();
38
- return record;
39
- }
40
- update(id, updates) {
41
- this.logger.debug({ id }, "Updating OpenCode binary");
42
- const config = this.configStore.get();
43
- const nextConfig = this.cloneConfig(config);
44
- nextConfig.opencodeBinaries = nextConfig.opencodeBinaries.map((binary) => binary.path === id ? { ...binary, label: updates.label ?? binary.label } : binary);
45
- if (updates.makeDefault) {
46
- nextConfig.preferences.lastUsedBinary = id;
47
- }
48
- this.configStore.replace(nextConfig);
49
- const record = this.getById(id);
50
- this.emitChange();
51
- return record;
52
- }
53
- remove(id) {
54
- this.logger.debug({ id }, "Removing OpenCode binary");
55
- const config = this.configStore.get();
56
- const nextConfig = this.cloneConfig(config);
57
- const remaining = nextConfig.opencodeBinaries.filter((binary) => binary.path !== id);
58
- nextConfig.opencodeBinaries = remaining;
59
- if (nextConfig.preferences.lastUsedBinary === id) {
60
- nextConfig.preferences.lastUsedBinary = remaining[0]?.path;
61
- }
62
- this.configStore.replace(nextConfig);
63
- this.emitChange();
64
- }
65
- validatePath(path) {
66
- this.logger.debug({ path }, "Validating OpenCode binary path");
67
- return this.validateRecord({
68
- id: path,
69
- path,
70
- label: this.prettyLabel(path),
71
- isDefault: false,
72
- });
73
- }
74
- cloneConfig(config) {
75
- return JSON.parse(JSON.stringify(config));
76
- }
77
- mapRecords() {
78
- const config = this.configStore.get();
79
- const configuredBinaries = config.opencodeBinaries.map((binary) => ({
80
- id: binary.path,
81
- path: binary.path,
82
- label: binary.label ?? this.prettyLabel(binary.path),
83
- version: binary.version,
84
- isDefault: false,
85
- }));
86
- const defaultPath = config.preferences.lastUsedBinary ?? configuredBinaries[0]?.path ?? "opencode";
87
- const annotated = configuredBinaries.map((binary) => ({
88
- ...binary,
89
- isDefault: binary.path === defaultPath,
90
- }));
91
- if (!annotated.some((binary) => binary.path === defaultPath)) {
92
- annotated.unshift(this.buildFallbackRecord(defaultPath));
93
- }
94
- return annotated;
95
- }
96
- getById(id) {
97
- return this.mapRecords().find((binary) => binary.id === id) ?? this.buildFallbackRecord(id);
98
- }
99
- emitChange() {
100
- this.logger.debug("Emitting binaries changed event");
101
- this.eventBus?.publish({ type: "config.binariesChanged", binaries: this.mapRecords() });
102
- }
103
- validateRecord(record) {
104
- const inputPath = record.path;
105
- if (!inputPath) {
106
- return { valid: false, error: "Missing binary path" };
107
- }
108
- const spec = buildSpawnSpec(inputPath, ["--version"]);
109
- try {
110
- const result = spawnSync(spec.command, spec.args, {
111
- encoding: "utf8",
112
- windowsVerbatimArguments: Boolean(spec.options.windowsVerbatimArguments),
113
- });
114
- if (result.error) {
115
- return { valid: false, error: result.error.message };
116
- }
117
- if (result.status !== 0) {
118
- const stderr = result.stderr?.trim();
119
- const stdout = result.stdout?.trim();
120
- const combined = stderr || stdout;
121
- const error = combined ? `Exited with code ${result.status}: ${combined}` : `Exited with code ${result.status}`;
122
- return { valid: false, error };
123
- }
124
- const stdout = (result.stdout ?? "").trim();
125
- const firstLine = stdout.split(/\r?\n/).find((line) => line.trim().length > 0);
126
- const normalized = firstLine?.trim();
127
- const versionMatch = normalized?.match(/([0-9]+\.[0-9]+\.[0-9A-Za-z.-]+)/);
128
- const version = versionMatch?.[1];
129
- return { valid: true, version };
130
- }
131
- catch (error) {
132
- return { valid: false, error: error instanceof Error ? error.message : String(error) };
133
- }
134
- }
135
- buildFallbackRecord(path) {
136
- return {
137
- id: path,
138
- path,
139
- label: this.prettyLabel(path),
140
- isDefault: true,
141
- };
142
- }
143
- prettyLabel(path) {
144
- const parts = path.split(/[\\/]/);
145
- const last = parts[parts.length - 1] || path;
146
- return last || path;
147
- }
148
- }
@@ -1,200 +0,0 @@
1
- import fs from "fs";
2
- import path from "path";
3
- import { parse as parseYaml, stringify as stringifyYaml } from "yaml";
4
- import { ConfigFileSchema, ConfigYamlSchema, DEFAULT_CONFIG, DEFAULT_CONFIG_YAML, DEFAULT_STATE, StateFileSchema, } from "./schema";
5
- export class ConfigStore {
6
- constructor(location, eventBus, logger) {
7
- this.location = location;
8
- this.eventBus = eventBus;
9
- this.logger = logger;
10
- this.cache = DEFAULT_CONFIG;
11
- this.state = DEFAULT_STATE;
12
- this.loaded = false;
13
- }
14
- load() {
15
- if (this.loaded) {
16
- return this.cache;
17
- }
18
- try {
19
- const configYamlPath = this.location.configYamlPath;
20
- const stateYamlPath = this.location.stateYamlPath;
21
- const legacyJsonPath = this.location.legacyJsonPath;
22
- if (fs.existsSync(configYamlPath)) {
23
- const configDoc = this.readYamlFile(configYamlPath, DEFAULT_CONFIG_YAML, ConfigYamlSchema, "config");
24
- const stateDoc = fs.existsSync(stateYamlPath)
25
- ? this.readYamlFile(stateYamlPath, DEFAULT_STATE, StateFileSchema, "state")
26
- : DEFAULT_STATE;
27
- this.state = stateDoc;
28
- this.cache = this.mergeDocs(configDoc, stateDoc);
29
- this.logger.debug({ configYamlPath, stateYamlPath }, "Loaded existing YAML config/state");
30
- }
31
- else if (fs.existsSync(legacyJsonPath)) {
32
- const migrated = this.migrateFromLegacyJson(legacyJsonPath);
33
- this.state = migrated.state;
34
- this.cache = migrated.config;
35
- }
36
- else {
37
- // Fresh install: write defaults.
38
- this.state = DEFAULT_STATE;
39
- this.cache = this.mergeDocs(DEFAULT_CONFIG_YAML, DEFAULT_STATE);
40
- this.persist();
41
- this.logger.debug({ configYamlPath, stateYamlPath }, "No config files found, created default YAML config/state");
42
- }
43
- }
44
- catch (error) {
45
- this.logger.warn({ err: error }, "Failed to load config/state, using defaults");
46
- this.state = DEFAULT_STATE;
47
- this.cache = this.mergeDocs(DEFAULT_CONFIG_YAML, DEFAULT_STATE);
48
- }
49
- this.loaded = true;
50
- return this.cache;
51
- }
52
- get() {
53
- return this.load();
54
- }
55
- replace(config) {
56
- const validated = ConfigFileSchema.parse(config);
57
- this.commit(validated);
58
- }
59
- /**
60
- * Apply a merge-patch update to the current config.
61
- * - Missing keys are preserved.
62
- * - Object values are merged recursively.
63
- * - Explicit `null` deletes keys.
64
- * - Arrays are replaced.
65
- */
66
- mergePatch(patch) {
67
- if (!patch || typeof patch !== "object" || Array.isArray(patch)) {
68
- throw new Error("Config patch must be a JSON object");
69
- }
70
- const current = this.get();
71
- const next = applyMergePatch(current, patch);
72
- const validated = ConfigFileSchema.parse(next);
73
- this.commit(validated);
74
- }
75
- commit(next) {
76
- this.cache = next;
77
- this.loaded = true;
78
- this.state = {
79
- ...this.state,
80
- recentFolders: next.recentFolders,
81
- };
82
- this.persist();
83
- const published = Boolean(this.eventBus);
84
- this.eventBus?.publish({ type: "config.appChanged", config: this.cache });
85
- this.logger.debug({ broadcast: published }, "Config SSE event emitted");
86
- this.logger.trace({ config: this.cache }, "Config payload");
87
- }
88
- persist() {
89
- try {
90
- const configYamlPath = this.location.configYamlPath;
91
- const stateYamlPath = this.location.stateYamlPath;
92
- fs.mkdirSync(this.location.baseDir, { recursive: true });
93
- fs.mkdirSync(path.dirname(configYamlPath), { recursive: true });
94
- const configYaml = stringifyYaml(stripRecentFolders(this.cache));
95
- const stateYaml = stringifyYaml(this.state);
96
- fs.writeFileSync(configYamlPath, ensureTrailingNewline(configYaml), "utf-8");
97
- fs.writeFileSync(stateYamlPath, ensureTrailingNewline(stateYaml), "utf-8");
98
- this.logger.debug({ configYamlPath, stateYamlPath }, "Persisted YAML config/state");
99
- }
100
- catch (error) {
101
- this.logger.warn({ err: error }, "Failed to persist config");
102
- }
103
- }
104
- mergeDocs(configDoc, stateDoc) {
105
- const merged = {
106
- ...configDoc,
107
- // State wins for recent folders.
108
- recentFolders: stateDoc.recentFolders ?? [],
109
- };
110
- return ConfigFileSchema.parse(merged);
111
- }
112
- readYamlFile(filePath, fallback, schema, label) {
113
- try {
114
- const content = fs.readFileSync(filePath, "utf-8");
115
- const parsed = parseYaml(content);
116
- return schema.parse(parsed ?? {});
117
- }
118
- catch (error) {
119
- this.logger.warn({ err: error, filePath, label }, "Failed to read YAML file, using defaults");
120
- return fallback;
121
- }
122
- }
123
- migrateFromLegacyJson(legacyJsonPath) {
124
- const configYamlPath = this.location.configYamlPath;
125
- const stateYamlPath = this.location.stateYamlPath;
126
- const content = fs.readFileSync(legacyJsonPath, "utf-8");
127
- const parsed = JSON.parse(content);
128
- const legacy = ConfigFileSchema.parse(parsed);
129
- const state = StateFileSchema.parse({
130
- ...DEFAULT_STATE,
131
- recentFolders: legacy.recentFolders ?? [],
132
- });
133
- const merged = this.mergeDocs(stripRecentFolders(legacy), state);
134
- // Persist YAML docs first, then move legacy aside.
135
- try {
136
- fs.mkdirSync(this.location.baseDir, { recursive: true });
137
- fs.writeFileSync(configYamlPath, ensureTrailingNewline(stringifyYaml(stripRecentFolders(merged))), "utf-8");
138
- fs.writeFileSync(stateYamlPath, ensureTrailingNewline(stringifyYaml(state)), "utf-8");
139
- this.logger.info({ legacyJsonPath, configYamlPath, stateYamlPath }, "Migrated config.json -> YAML");
140
- }
141
- catch (error) {
142
- this.logger.warn({ err: error }, "Failed to persist migrated YAML config/state");
143
- }
144
- try {
145
- const bakPath = pickBackupPath(legacyJsonPath);
146
- fs.renameSync(legacyJsonPath, bakPath);
147
- this.logger.info({ legacyJsonPath, bakPath }, "Moved legacy config.json to backup");
148
- }
149
- catch (error) {
150
- this.logger.warn({ err: error, legacyJsonPath }, "Failed to rename legacy config.json to backup");
151
- }
152
- return { config: merged, state };
153
- }
154
- }
155
- function ensureTrailingNewline(content) {
156
- if (!content)
157
- return "\n";
158
- return content.endsWith("\n") ? content : `${content}\n`;
159
- }
160
- function stripRecentFolders(config) {
161
- const clone = { ...config };
162
- delete clone.recentFolders;
163
- return clone;
164
- }
165
- function isPlainObject(value) {
166
- if (!value || typeof value !== "object")
167
- return false;
168
- if (Array.isArray(value))
169
- return false;
170
- const proto = Object.getPrototypeOf(value);
171
- return proto === Object.prototype || proto === null;
172
- }
173
- function applyMergePatch(current, patch) {
174
- // RFC 7396-ish merge patch with explicit null deletes.
175
- if (!isPlainObject(patch)) {
176
- return patch;
177
- }
178
- const base = isPlainObject(current) ? { ...current } : {};
179
- for (const [key, value] of Object.entries(patch)) {
180
- if (value === null) {
181
- delete base[key];
182
- continue;
183
- }
184
- if (isPlainObject(value) && isPlainObject(base[key])) {
185
- base[key] = applyMergePatch(base[key], value);
186
- continue;
187
- }
188
- // Arrays and scalars replace.
189
- base[key] = value;
190
- }
191
- return base;
192
- }
193
- function pickBackupPath(legacyJsonPath) {
194
- const base = legacyJsonPath.endsWith(".json") ? legacyJsonPath.slice(0, -".json".length) : legacyJsonPath;
195
- const preferred = `${base}.json.bak`;
196
- if (!fs.existsSync(preferred)) {
197
- return preferred;
198
- }
199
- return `${base}.json.bak.${Date.now()}`;
200
- }
@@ -1,59 +0,0 @@
1
- import { z } from "zod";
2
- const BinaryCreateSchema = z.object({
3
- path: z.string(),
4
- label: z.string().optional(),
5
- makeDefault: z.boolean().optional(),
6
- });
7
- const BinaryUpdateSchema = z.object({
8
- label: z.string().optional(),
9
- makeDefault: z.boolean().optional(),
10
- });
11
- const BinaryValidateSchema = z.object({
12
- path: z.string(),
13
- });
14
- export function registerConfigRoutes(app, deps) {
15
- app.get("/api/config/app", async () => deps.configStore.get());
16
- app.put("/api/config/app", async (request, reply) => {
17
- // Backwards compatible: treat PUT as a merge-patch update.
18
- try {
19
- deps.configStore.mergePatch(request.body ?? {});
20
- return deps.configStore.get();
21
- }
22
- catch (error) {
23
- reply.code(400);
24
- return { error: error instanceof Error ? error.message : "Invalid config patch" };
25
- }
26
- });
27
- app.patch("/api/config/app", async (request, reply) => {
28
- try {
29
- deps.configStore.mergePatch(request.body ?? {});
30
- return deps.configStore.get();
31
- }
32
- catch (error) {
33
- reply.code(400);
34
- return { error: error instanceof Error ? error.message : "Invalid config patch" };
35
- }
36
- });
37
- app.get("/api/config/binaries", async () => {
38
- return { binaries: deps.binaryRegistry.list() };
39
- });
40
- app.post("/api/config/binaries", async (request, reply) => {
41
- const body = BinaryCreateSchema.parse(request.body ?? {});
42
- const binary = deps.binaryRegistry.create(body);
43
- reply.code(201);
44
- return { binary };
45
- });
46
- app.patch("/api/config/binaries/:id", async (request) => {
47
- const body = BinaryUpdateSchema.parse(request.body ?? {});
48
- const binary = deps.binaryRegistry.update(request.params.id, body);
49
- return { binary };
50
- });
51
- app.delete("/api/config/binaries/:id", async (request, reply) => {
52
- deps.binaryRegistry.remove(request.params.id);
53
- reply.code(204);
54
- });
55
- app.post("/api/config/binaries/validate", async (request) => {
56
- const body = BinaryValidateSchema.parse(request.body ?? {});
57
- return deps.binaryRegistry.validatePath(body.path);
58
- });
59
- }