From 9d4b9a57053192949333d6598ebef0170ac8b1e1 Mon Sep 17 00:00:00 2001 From: cmpadden Date: Sat, 16 Nov 2024 18:33:55 +0000 Subject: [PATCH] deploy: 7e7df8f9be55d1aa39f64893967a7f7b30a0aa3a --- 200.html | 12 ++--- 404.html | 12 ++--- _nuxt/{C7UpC-lS.js => 1Kbf3Vya.js} | 2 +- _nuxt/{CgY3mHZ8.js => 2mhhWi5_.js} | 2 +- _nuxt/{Bdy-hNpQ.js => BBSVol53.js} | 2 +- _nuxt/{Brnt98Va.js => BLsxZwe9.js} | 2 +- _nuxt/{yJKb_Bp-.js => BMXP4Aud.js} | 2 +- _nuxt/{d0UAFwvu.js => BY70Pa1e.js} | 2 +- _nuxt/{DZkks4hD.js => BZWGQyMp.js} | 2 +- _nuxt/{BGikLauJ.js => Bb-NKJJJ.js} | 6 +-- _nuxt/{-8n5e5Fj.js => Bbro_qAu.js} | 2 +- _nuxt/{Dz1t7Gnd.js => BenPNx-m.js} | 2 +- _nuxt/{BdJQj3nn.js => BfDVGm6D.js} | 2 +- _nuxt/{RX6TINi_.js => Bj-Uk9MO.js} | 2 +- _nuxt/{DO9Ai2bz.js => BlMa8jv2.js} | 2 +- _nuxt/{DCgnBWC9.js => Bl_PaSYd.js} | 2 +- _nuxt/{Dkf0i8RZ.js => BmBFiau8.js} | 2 +- _nuxt/{CruaMelD.js => BmkabxXg.js} | 2 +- _nuxt/{CnOV4Igh.js => BqebzOVN.js} | 2 +- _nuxt/{6mOJeL1q.js => BsGN5WoF.js} | 2 +- _nuxt/BxFzRrKR.js | 1 + _nuxt/{VwgkOAbI.js => C-O5BI11.js} | 2 +- _nuxt/{-R_hya3P.js => C6rjZEvg.js} | 2 +- _nuxt/{BRTC5WMg.js => CCH9cHpw.js} | 2 +- _nuxt/{CUESTjlD.js => CMNpy2Is.js} | 4 +- _nuxt/{D0Omvic7.js => CQnAi_qF.js} | 2 +- _nuxt/{yGJPFCFo.js => CS06-0Od.js} | 2 +- _nuxt/CYeLrQ9C.js | 1 - _nuxt/{CDVU8JqM.js => Cai_qPmv.js} | 2 +- _nuxt/{qIrQuZow.js => CgzK-JD6.js} | 2 +- _nuxt/{BMI0969X.js => Co_4egda.js} | 2 +- _nuxt/{B3q-b4GE.js => CyLrJzPW.js} | 2 +- _nuxt/{D-AOKchP.js => D-cHg-Fi.js} | 2 +- _nuxt/{C_llEvAK.js => D-lCVgpc.js} | 2 +- _nuxt/{DX7ac3Mz.js => D1bB9L70.js} | 2 +- _nuxt/{CjTcA1Xt.js => D7KuDJ9z.js} | 2 +- _nuxt/{D7IuvTrr.js => D9IbAJ0u.js} | 2 +- _nuxt/{UcoZFNdm.js => DA5xXx2M.js} | 2 +- _nuxt/{CzhJn57k.js => DAkPvNyh.js} | 2 +- _nuxt/{_kju5I3X.js => DEdI-0CJ.js} | 2 +- _nuxt/DS100L4J.js | 1 + _nuxt/{m3fkSCjB.js => DVb7Ucrz.js} | 2 +- _nuxt/{DbdbR-pn.js => DXvJDNSx.js} | 2 +- _nuxt/{DsIJUosc.js => DZAhYQpW.js} | 2 +- _nuxt/{CDv6ev47.js => DcUhz6y1.js} | 2 +- _nuxt/{BYwurayZ.js => DcYzECSc.js} | 2 +- _nuxt/{B9UQgCMx.js => Dcfpj4_h.js} | 2 +- _nuxt/{ZO3tSbNI.js => DjUAd0VA.js} | 2 +- _nuxt/Dn7xXnlv.js | 1 + _nuxt/{M6Zd1ArL.js => Dp7uR_kE.js} | 2 +- _nuxt/{DukX5DGJ.js => DrabSix5.js} | 2 +- _nuxt/{BEt4369s.js => DvMJuzSW.js} | 2 +- _nuxt/{CJwA9VmX.js => DvojTKw_.js} | 2 +- _nuxt/{BUYHwnpH.js => Dz8Mq6rn.js} | 2 +- _nuxt/{BnWdfnc6.js => DzMxnq8n.js} | 2 +- _nuxt/{jcdaHq-b.js => F1DGrIW6.js} | 2 +- _nuxt/{Do41mdVm.js => I6a6eUEM.js} | 2 +- _nuxt/KhAE7j4n.js | 1 - _nuxt/{D3_S2KpX.js => R23NweeK.js} | 2 +- _nuxt/{Do_OiPaD.js => UzigjR5s.js} | 2 +- _nuxt/{CQCYFxb2.js => ZRsJ-Dry.js} | 2 +- _nuxt/{DrbA_f4z.js => ZcTbvTx_.js} | 2 +- _nuxt/builds/latest.json | 2 +- .../2efca5fe-2565-4a9a-b7e7-d2f9211a5691.json | 1 + .../e6f03e61-3022-4913-8d71-aebf9c20aa41.json | 1 - _nuxt/{AY-jWpd0.js => cufizuWq.js} | 2 +- _nuxt/dVJITDym.js | 1 - _nuxt/{UWyBikFq.js => jaxqT3Hp.js} | 4 +- _nuxt/{BQ6KdT-G.js => mIisoFHp.js} | 2 +- _nuxt/{DJ6fbLCw.js => mjvvwssR.js} | 2 +- _nuxt/{D4ZOq_QP.js => oLoEepVu.js} | 2 +- _nuxt/{Btpue746.js => tatS9fxq.js} | 2 +- _nuxt/{DSaqcxQi.js => uL54ePkM.js} | 2 +- _nuxt/{DJ0wDsqg.js => xkKG0dwU.js} | 2 +- _payload.json | 2 +- ...81838504.json => cache.1731782019206.json} | 2 +- ...504.json => 6WfgQ5T9tH.1731782019206.json} | 0 ...504.json => 7TfxHWYxZH.1731782019206.json} | 0 ...504.json => 98CVAb0zLR.1731782019206.json} | 0 ...504.json => DIau8q3IMV.1731782019206.json} | 0 ...504.json => LcWrOc5HNX.1731782019206.json} | 0 ...504.json => Nr5UObwduV.1731782019206.json} | 0 ...504.json => QmL7G3Pk7i.1731782019206.json} | 0 ...504.json => TDaCLaQ73L.1731782019206.json} | 0 ...504.json => XgcK3x9EBy.1731782019206.json} | 0 ...504.json => ZzD9WRl1Uk.1731782019206.json} | 0 ...504.json => bXj5vj6Ts0.1731782019206.json} | 0 ...504.json => d7v45RMayO.1731782019206.json} | 0 ...504.json => pfbAdBSC9a.1731782019206.json} | 0 ...504.json => tGrf8kFZOz.1731782019206.json} | 0 ...504.json => ucEXmLbw2Z.1731782019206.json} | 0 ...504.json => v3HQ7aAWkW.1731782019206.json} | 0 articles/_payload.json | 2 +- articles/apu2-firmware-upgrade/_payload.json | 2 +- articles/apu2-firmware-upgrade/index.html | 50 +++++++++--------- articles/docker-selinux-volumes/_payload.json | 2 +- articles/docker-selinux-volumes/index.html | 52 +++++++++---------- articles/doctl/_payload.json | 2 +- articles/doctl/index.html | 44 ++++++++-------- .../fennel-initial-exploration/_payload.json | 2 +- .../fennel-initial-exploration/index.html | 50 +++++++++--------- articles/index.html | 38 +++++++------- .../_payload.json | 2 +- .../index.html | 42 +++++++-------- articles/nuxt-content-rss-feed/_payload.json | 2 +- articles/nuxt-content-rss-feed/index.html | 44 ++++++++-------- articles/nuxt-v3-migration/_payload.json | 2 +- articles/nuxt-v3-migration/index.html | 40 +++++++------- .../persistent-archlinux-usb/_payload.json | 2 +- articles/persistent-archlinux-usb/index.html | 52 +++++++++---------- .../_payload.json | 2 +- .../index.html | 44 ++++++++-------- .../_payload.json | 2 +- .../index.html | 44 ++++++++-------- .../_payload.json | 2 +- .../index.html | 44 ++++++++-------- articles/ssh-ed25519-sk-yubikey/_payload.json | 2 +- articles/ssh-ed25519-sk-yubikey/index.html | 46 ++++++++-------- .../_payload.json | 2 +- .../index.html | 50 +++++++++--------- .../vim-fugitive-gpg-pinentry/_payload.json | 2 +- articles/vim-fugitive-gpg-pinentry/index.html | 44 ++++++++-------- atom/index.html | 2 +- card/_payload.json | 2 +- card/index.html | 16 +++--- examples/nested_transitions/_payload.json | 2 +- examples/nested_transitions/index.html | 16 +++--- index.html | 22 ++++---- playground/_payload.json | 2 +- playground/audio/_payload.json | 2 +- playground/audio/index.html | 16 +++--- playground/chords/_payload.json | 2 +- playground/chords/index.html | 16 +++--- playground/conway/_payload.json | 2 +- playground/conway/index.html | 16 +++--- playground/french/_payload.json | 2 +- playground/french/index.html | 18 +++---- playground/index.html | 18 +++---- playground/matrix/_payload.json | 2 +- playground/matrix/index.html | 16 +++--- playground/metronome/_payload.json | 2 +- playground/metronome/index.html | 16 +++--- playground/midi/_payload.json | 2 +- playground/midi/index.html | 16 +++--- playground/palettes/mountains/_payload.json | 2 +- playground/palettes/mountains/index.html | 16 +++--- playground/palettes/variance/_payload.json | 2 +- playground/palettes/variance/index.html | 16 +++--- playground/plotter/_payload.json | 2 +- playground/plotter/index.html | 16 +++--- playground/tiling/_payload.json | 2 +- playground/tiling/index.html | 16 +++--- playground/waves/_payload.json | 2 +- playground/waves/index.html | 16 +++--- talks/_payload.json | 2 +- talks/index.html | 16 +++--- 156 files changed, 601 insertions(+), 601 deletions(-) rename _nuxt/{C7UpC-lS.js => 1Kbf3Vya.js} (65%) rename _nuxt/{CgY3mHZ8.js => 2mhhWi5_.js} (97%) rename _nuxt/{Bdy-hNpQ.js => BBSVol53.js} (64%) rename _nuxt/{Brnt98Va.js => BLsxZwe9.js} (61%) rename _nuxt/{yJKb_Bp-.js => BMXP4Aud.js} (79%) rename _nuxt/{d0UAFwvu.js => BY70Pa1e.js} (92%) rename _nuxt/{DZkks4hD.js => BZWGQyMp.js} (99%) rename _nuxt/{BGikLauJ.js => Bb-NKJJJ.js} (98%) rename _nuxt/{-8n5e5Fj.js => Bbro_qAu.js} (98%) rename _nuxt/{Dz1t7Gnd.js => BenPNx-m.js} (82%) rename _nuxt/{BdJQj3nn.js => BfDVGm6D.js} (88%) rename _nuxt/{RX6TINi_.js => Bj-Uk9MO.js} (89%) rename _nuxt/{DO9Ai2bz.js => BlMa8jv2.js} (81%) rename _nuxt/{DCgnBWC9.js => Bl_PaSYd.js} (78%) rename _nuxt/{Dkf0i8RZ.js => BmBFiau8.js} (90%) rename _nuxt/{CruaMelD.js => BmkabxXg.js} (65%) rename _nuxt/{CnOV4Igh.js => BqebzOVN.js} (94%) rename _nuxt/{6mOJeL1q.js => BsGN5WoF.js} (65%) create mode 100644 _nuxt/BxFzRrKR.js rename _nuxt/{VwgkOAbI.js => C-O5BI11.js} (87%) rename _nuxt/{-R_hya3P.js => C6rjZEvg.js} (79%) rename _nuxt/{BRTC5WMg.js => CCH9cHpw.js} (97%) rename _nuxt/{CUESTjlD.js => CMNpy2Is.js} (75%) rename _nuxt/{D0Omvic7.js => CQnAi_qF.js} (99%) rename _nuxt/{yGJPFCFo.js => CS06-0Od.js} (99%) delete mode 100644 _nuxt/CYeLrQ9C.js rename _nuxt/{CDVU8JqM.js => Cai_qPmv.js} (98%) rename _nuxt/{qIrQuZow.js => CgzK-JD6.js} (98%) rename _nuxt/{BMI0969X.js => Co_4egda.js} (63%) rename _nuxt/{B3q-b4GE.js => CyLrJzPW.js} (56%) rename _nuxt/{D-AOKchP.js => D-cHg-Fi.js} (65%) rename _nuxt/{C_llEvAK.js => D-lCVgpc.js} (96%) rename _nuxt/{DX7ac3Mz.js => D1bB9L70.js} (88%) rename _nuxt/{CjTcA1Xt.js => D7KuDJ9z.js} (64%) rename _nuxt/{D7IuvTrr.js => D9IbAJ0u.js} (53%) rename _nuxt/{UcoZFNdm.js => DA5xXx2M.js} (96%) rename _nuxt/{CzhJn57k.js => DAkPvNyh.js} (99%) rename _nuxt/{_kju5I3X.js => DEdI-0CJ.js} (88%) create mode 100644 _nuxt/DS100L4J.js rename _nuxt/{m3fkSCjB.js => DVb7Ucrz.js} (86%) rename _nuxt/{DbdbR-pn.js => DXvJDNSx.js} (64%) rename _nuxt/{DsIJUosc.js => DZAhYQpW.js} (88%) rename _nuxt/{CDv6ev47.js => DcUhz6y1.js} (97%) rename _nuxt/{BYwurayZ.js => DcYzECSc.js} (65%) rename _nuxt/{B9UQgCMx.js => Dcfpj4_h.js} (64%) rename _nuxt/{ZO3tSbNI.js => DjUAd0VA.js} (98%) create mode 100644 _nuxt/Dn7xXnlv.js rename _nuxt/{M6Zd1ArL.js => Dp7uR_kE.js} (58%) rename _nuxt/{DukX5DGJ.js => DrabSix5.js} (84%) rename _nuxt/{BEt4369s.js => DvMJuzSW.js} (64%) rename _nuxt/{CJwA9VmX.js => DvojTKw_.js} (65%) rename _nuxt/{BUYHwnpH.js => Dz8Mq6rn.js} (88%) rename _nuxt/{BnWdfnc6.js => DzMxnq8n.js} (77%) rename _nuxt/{jcdaHq-b.js => F1DGrIW6.js} (86%) rename _nuxt/{Do41mdVm.js => I6a6eUEM.js} (88%) delete mode 100644 _nuxt/KhAE7j4n.js rename _nuxt/{D3_S2KpX.js => R23NweeK.js} (95%) rename _nuxt/{Do_OiPaD.js => UzigjR5s.js} (78%) rename _nuxt/{CQCYFxb2.js => ZRsJ-Dry.js} (88%) rename _nuxt/{DrbA_f4z.js => ZcTbvTx_.js} (99%) create mode 100644 _nuxt/builds/meta/2efca5fe-2565-4a9a-b7e7-d2f9211a5691.json delete mode 100644 _nuxt/builds/meta/e6f03e61-3022-4913-8d71-aebf9c20aa41.json rename _nuxt/{AY-jWpd0.js => cufizuWq.js} (92%) delete mode 100644 _nuxt/dVJITDym.js rename _nuxt/{UWyBikFq.js => jaxqT3Hp.js} (91%) rename _nuxt/{BQ6KdT-G.js => mIisoFHp.js} (64%) rename _nuxt/{DJ6fbLCw.js => mjvvwssR.js} (98%) rename _nuxt/{D4ZOq_QP.js => oLoEepVu.js} (64%) rename _nuxt/{Btpue746.js => tatS9fxq.js} (93%) rename _nuxt/{DSaqcxQi.js => uL54ePkM.js} (93%) rename _nuxt/{DJ0wDsqg.js => xkKG0dwU.js} (94%) rename api/_content/{cache.1731781838504.json => cache.1731782019206.json} (99%) rename api/_content/query/{6WfgQ5T9tH.1731781838504.json => 6WfgQ5T9tH.1731782019206.json} (100%) rename api/_content/query/{7TfxHWYxZH.1731781838504.json => 7TfxHWYxZH.1731782019206.json} (100%) rename api/_content/query/{98CVAb0zLR.1731781838504.json => 98CVAb0zLR.1731782019206.json} (100%) rename api/_content/query/{DIau8q3IMV.1731781838504.json => DIau8q3IMV.1731782019206.json} (100%) rename api/_content/query/{LcWrOc5HNX.1731781838504.json => LcWrOc5HNX.1731782019206.json} (100%) rename api/_content/query/{Nr5UObwduV.1731781838504.json => Nr5UObwduV.1731782019206.json} (100%) rename api/_content/query/{QmL7G3Pk7i.1731781838504.json => QmL7G3Pk7i.1731782019206.json} (100%) rename api/_content/query/{TDaCLaQ73L.1731781838504.json => TDaCLaQ73L.1731782019206.json} (100%) rename api/_content/query/{XgcK3x9EBy.1731781838504.json => XgcK3x9EBy.1731782019206.json} (100%) rename api/_content/query/{ZzD9WRl1Uk.1731781838504.json => ZzD9WRl1Uk.1731782019206.json} (100%) rename api/_content/query/{bXj5vj6Ts0.1731781838504.json => bXj5vj6Ts0.1731782019206.json} (100%) rename api/_content/query/{d7v45RMayO.1731781838504.json => d7v45RMayO.1731782019206.json} (100%) rename api/_content/query/{pfbAdBSC9a.1731781838504.json => pfbAdBSC9a.1731782019206.json} (100%) rename api/_content/query/{tGrf8kFZOz.1731781838504.json => tGrf8kFZOz.1731782019206.json} (100%) rename api/_content/query/{ucEXmLbw2Z.1731781838504.json => ucEXmLbw2Z.1731782019206.json} (100%) rename api/_content/query/{v3HQ7aAWkW.1731781838504.json => v3HQ7aAWkW.1731782019206.json} (100%) diff --git a/200.html b/200.html index 6750d963..3883eca2 100644 --- a/200.html +++ b/200.html @@ -1,9 +1,9 @@ - - - - -
- \ No newline at end of file + + + + +
+ \ No newline at end of file diff --git a/404.html b/404.html index 49b8c010..5c944249 100644 --- a/404.html +++ b/404.html @@ -1,9 +1,9 @@ - - - - -
- \ No newline at end of file + + + + +
+ \ No newline at end of file diff --git a/_nuxt/C7UpC-lS.js b/_nuxt/1Kbf3Vya.js similarity index 65% rename from _nuxt/C7UpC-lS.js rename to _nuxt/1Kbf3Vya.js index 92d6c823..a78761fe 100644 --- a/_nuxt/C7UpC-lS.js +++ b/_nuxt/1Kbf3Vya.js @@ -1 +1 @@ -import{m as o,o as r,c as t,a8 as s}from"./BGikLauJ.js";const c={};function n(e,a){return r(),t("tbody",null,[s(e.$slots,"default")])}const d=o(c,[["render",n]]);export{d as default}; +import{m as o,o as r,c as t,a8 as s}from"./Bb-NKJJJ.js";const c={};function n(e,a){return r(),t("tbody",null,[s(e.$slots,"default")])}const d=o(c,[["render",n]]);export{d as default}; diff --git a/_nuxt/CgY3mHZ8.js b/_nuxt/2mhhWi5_.js similarity index 97% rename from _nuxt/CgY3mHZ8.js rename to _nuxt/2mhhWi5_.js index 0a87bb45..9076b90d 100644 --- a/_nuxt/CgY3mHZ8.js +++ b/_nuxt/2mhhWi5_.js @@ -1 +1 @@ -import{N as b,W as d,j as D,X as O,O as B,Y as M,y as E,v as H,Z as p,$ as R,a as S,a0 as V,a1 as j}from"./BGikLauJ.js";const z=s=>s==="defer"||s===!1;function F(...s){var v;const i=typeof s[s.length-1]=="string"?s.pop():void 0;typeof s[0]!="string"&&s.unshift(i);let[t,u,a={}]=s;if(typeof t!="string")throw new TypeError("[nuxt] [asyncData] key must be a string.");if(typeof u!="function")throw new TypeError("[nuxt] [asyncData] handler must be a function.");const e=b(),g=u,P=()=>d.value,C=()=>e.isHydrating?e.payload.data[t]:e.static.data[t];a.server=a.server??!0,a.default=a.default??P,a.getCachedData=a.getCachedData??C,a.lazy=a.lazy??!1,a.immediate=a.immediate??!0,a.deep=a.deep??d.deep,a.dedupe=a.dedupe??"cancel";const f=a.getCachedData(t,e),h=f!=null;if(!e._asyncData[t]||!a.immediate){(v=e.payload._errors)[t]??(v[t]=d.errorValue);const o=a.deep?D:O;e._asyncData[t]={data:o(h?f:a.default()),pending:D(!h),error:B(e.payload._errors,t),status:D("idle"),_default:a.default}}const r={...e._asyncData[t]};delete r._default,r.refresh=r.execute=(o={})=>{if(e._asyncDataPromises[t]){if(z(o.dedupe??a.dedupe))return e._asyncDataPromises[t];e._asyncDataPromises[t].cancelled=!0}if(o._initial||e.isHydrating&&o._initial!==!1){const c=o._initial?f:a.getCachedData(t,e);if(c!=null)return Promise.resolve(c)}r.pending.value=!0,r.status.value="pending";const l=new Promise((c,n)=>{try{c(g(e))}catch(y){n(y)}}).then(async c=>{if(l.cancelled)return e._asyncDataPromises[t];let n=c;a.transform&&(n=await a.transform(c)),a.pick&&(n=K(n,a.pick)),e.payload.data[t]=n,r.data.value=n,r.error.value=d.errorValue,r.status.value="success"}).catch(c=>{if(l.cancelled)return e._asyncDataPromises[t];r.error.value=R(c),r.data.value=S(a.default()),r.status.value="error"}).finally(()=>{l.cancelled||(r.pending.value=!1,delete e._asyncDataPromises[t])});return e._asyncDataPromises[t]=l,e._asyncDataPromises[t]},r.clear=()=>N(e,t);const _=()=>r.refresh({_initial:!0}),w=a.server!==!1&&e.payload.serverRendered;{const o=V();if(o&&!o._nuxtOnBeforeMountCbs){o._nuxtOnBeforeMountCbs=[];const n=o._nuxtOnBeforeMountCbs;M(()=>{n.forEach(y=>{y()}),n.splice(0,n.length)}),E(()=>n.splice(0,n.length))}w&&e.isHydrating&&(r.error.value||f!=null)?(r.pending.value=!1,r.status.value=r.error.value?"error":"success"):o&&(e.payload.serverRendered&&e.isHydrating||a.lazy)&&a.immediate?o._nuxtOnBeforeMountCbs.push(_):a.immediate&&_();const l=j();if(a.watch){const n=H(a.watch,()=>r.refresh());l&&p(n)}const c=e.hook("app:data:refresh",async n=>{(!n||n.includes(t))&&await r.refresh()});l&&p(c)}const m=Promise.resolve(e._asyncDataPromises[t]).then(()=>r);return Object.assign(m,r),m}function N(s,i){i in s.payload.data&&(s.payload.data[i]=void 0),i in s.payload._errors&&(s.payload._errors[i]=d.errorValue),s._asyncData[i]&&(s._asyncData[i].data.value=void 0,s._asyncData[i].error.value=d.errorValue,s._asyncData[i].pending.value=!1,s._asyncData[i].status.value="idle"),i in s._asyncDataPromises&&(s._asyncDataPromises[i]&&(s._asyncDataPromises[i].cancelled=!0),s._asyncDataPromises[i]=void 0)}function K(s,i){const t={};for(const u of i)t[u]=s[u];return t}export{F as u}; +import{N as b,W as d,j as D,X as O,O as B,Y as M,y as E,v as H,Z as p,$ as R,a as S,a0 as V,a1 as j}from"./Bb-NKJJJ.js";const z=s=>s==="defer"||s===!1;function F(...s){var v;const i=typeof s[s.length-1]=="string"?s.pop():void 0;typeof s[0]!="string"&&s.unshift(i);let[t,u,a={}]=s;if(typeof t!="string")throw new TypeError("[nuxt] [asyncData] key must be a string.");if(typeof u!="function")throw new TypeError("[nuxt] [asyncData] handler must be a function.");const e=b(),g=u,P=()=>d.value,C=()=>e.isHydrating?e.payload.data[t]:e.static.data[t];a.server=a.server??!0,a.default=a.default??P,a.getCachedData=a.getCachedData??C,a.lazy=a.lazy??!1,a.immediate=a.immediate??!0,a.deep=a.deep??d.deep,a.dedupe=a.dedupe??"cancel";const f=a.getCachedData(t,e),h=f!=null;if(!e._asyncData[t]||!a.immediate){(v=e.payload._errors)[t]??(v[t]=d.errorValue);const o=a.deep?D:O;e._asyncData[t]={data:o(h?f:a.default()),pending:D(!h),error:B(e.payload._errors,t),status:D("idle"),_default:a.default}}const r={...e._asyncData[t]};delete r._default,r.refresh=r.execute=(o={})=>{if(e._asyncDataPromises[t]){if(z(o.dedupe??a.dedupe))return e._asyncDataPromises[t];e._asyncDataPromises[t].cancelled=!0}if(o._initial||e.isHydrating&&o._initial!==!1){const c=o._initial?f:a.getCachedData(t,e);if(c!=null)return Promise.resolve(c)}r.pending.value=!0,r.status.value="pending";const l=new Promise((c,n)=>{try{c(g(e))}catch(y){n(y)}}).then(async c=>{if(l.cancelled)return e._asyncDataPromises[t];let n=c;a.transform&&(n=await a.transform(c)),a.pick&&(n=K(n,a.pick)),e.payload.data[t]=n,r.data.value=n,r.error.value=d.errorValue,r.status.value="success"}).catch(c=>{if(l.cancelled)return e._asyncDataPromises[t];r.error.value=R(c),r.data.value=S(a.default()),r.status.value="error"}).finally(()=>{l.cancelled||(r.pending.value=!1,delete e._asyncDataPromises[t])});return e._asyncDataPromises[t]=l,e._asyncDataPromises[t]},r.clear=()=>N(e,t);const _=()=>r.refresh({_initial:!0}),w=a.server!==!1&&e.payload.serverRendered;{const o=V();if(o&&!o._nuxtOnBeforeMountCbs){o._nuxtOnBeforeMountCbs=[];const n=o._nuxtOnBeforeMountCbs;M(()=>{n.forEach(y=>{y()}),n.splice(0,n.length)}),E(()=>n.splice(0,n.length))}w&&e.isHydrating&&(r.error.value||f!=null)?(r.pending.value=!1,r.status.value=r.error.value?"error":"success"):o&&(e.payload.serverRendered&&e.isHydrating||a.lazy)&&a.immediate?o._nuxtOnBeforeMountCbs.push(_):a.immediate&&_();const l=j();if(a.watch){const n=H(a.watch,()=>r.refresh());l&&p(n)}const c=e.hook("app:data:refresh",async n=>{(!n||n.includes(t))&&await r.refresh()});l&&p(c)}const m=Promise.resolve(e._asyncDataPromises[t]).then(()=>r);return Object.assign(m,r),m}function N(s,i){i in s.payload.data&&(s.payload.data[i]=void 0),i in s.payload._errors&&(s.payload._errors[i]=d.errorValue),s._asyncData[i]&&(s._asyncData[i].data.value=void 0,s._asyncData[i].error.value=d.errorValue,s._asyncData[i].pending.value=!1,s._asyncData[i].status.value="idle"),i in s._asyncDataPromises&&(s._asyncDataPromises[i]&&(s._asyncDataPromises[i].cancelled=!0),s._asyncDataPromises[i]=void 0)}function K(s,i){const t={};for(const u of i)t[u]=s[u];return t}export{F as u}; diff --git a/_nuxt/Bdy-hNpQ.js b/_nuxt/BBSVol53.js similarity index 64% rename from _nuxt/Bdy-hNpQ.js rename to _nuxt/BBSVol53.js index 3f55c808..a427f6c6 100644 --- a/_nuxt/Bdy-hNpQ.js +++ b/_nuxt/BBSVol53.js @@ -1 +1 @@ -import{m as o,o as r,c as s,a8 as t}from"./BGikLauJ.js";const c={};function n(e,a){return r(),s("li",null,[t(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; +import{m as o,o as r,c as s,a8 as t}from"./Bb-NKJJJ.js";const c={};function n(e,a){return r(),s("li",null,[t(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; diff --git a/_nuxt/Brnt98Va.js b/_nuxt/BLsxZwe9.js similarity index 61% rename from _nuxt/Brnt98Va.js rename to _nuxt/BLsxZwe9.js index ae7b65da..dc6687b1 100644 --- a/_nuxt/Brnt98Va.js +++ b/_nuxt/BLsxZwe9.js @@ -1 +1 @@ -import s from"./d0UAFwvu.js";import{d as o,J as u,k as f,a0 as m}from"./BGikLauJ.js";import"./Dnd51l0P.js";const d=o({name:"Markdown",extends:s,setup(t){const{parent:e}=m(),{between:n,default:a}=u(),r=f(()=>typeof t.unwrap=="string"?t.unwrap.split(" "):["*"]);return{fallbackSlot:a,tags:r,between:n,parent:e}}});export{d as default}; +import s from"./BY70Pa1e.js";import{d as o,J as u,k as f,a0 as m}from"./Bb-NKJJJ.js";import"./Dnd51l0P.js";const d=o({name:"Markdown",extends:s,setup(t){const{parent:e}=m(),{between:n,default:a}=u(),r=f(()=>typeof t.unwrap=="string"?t.unwrap.split(" "):["*"]);return{fallbackSlot:a,tags:r,between:n,parent:e}}});export{d as default}; diff --git a/_nuxt/yJKb_Bp-.js b/_nuxt/BMXP4Aud.js similarity index 79% rename from _nuxt/yJKb_Bp-.js rename to _nuxt/BMXP4Aud.js index 753dccbb..2881259e 100644 --- a/_nuxt/yJKb_Bp-.js +++ b/_nuxt/BMXP4Aud.js @@ -1 +1 @@ -import{d as t,a8 as a}from"./BGikLauJ.js";const o=t({__name:"ProseCode",props:{code:{type:String,default:""},language:{type:String,default:null},filename:{type:String,default:null},highlights:{type:Array,default:()=>[]},meta:{type:String,default:null}},setup(n){return(e,r)=>a(e.$slots,"default")}});export{o as _}; +import{d as t,a8 as a}from"./Bb-NKJJJ.js";const o=t({__name:"ProseCode",props:{code:{type:String,default:""},language:{type:String,default:null},filename:{type:String,default:null},highlights:{type:Array,default:()=>[]},meta:{type:String,default:null}},setup(n){return(e,r)=>a(e.$slots,"default")}});export{o as _}; diff --git a/_nuxt/d0UAFwvu.js b/_nuxt/BY70Pa1e.js similarity index 92% rename from _nuxt/d0UAFwvu.js rename to _nuxt/BY70Pa1e.js index 672977e2..f254c2dd 100644 --- a/_nuxt/d0UAFwvu.js +++ b/_nuxt/BY70Pa1e.js @@ -1 +1 @@ -import{f as i}from"./Dnd51l0P.js";import{d as u,J as f,k as c,L as a,a0 as d}from"./BGikLauJ.js";const p=u({name:"MDCSlot",functional:!0,props:{name:{type:String,default:"default"},unwrap:{type:[Boolean,String],default:!1},use:{type:Function,default:void 0}},setup(t){const{parent:s}=d(),{default:n}=f(),r=c(()=>typeof t.unwrap=="string"?t.unwrap.split(" "):["*"]);return{fallbackSlot:n,tags:r,parent:s}},render({use:t,unwrap:s,fallbackSlot:n,tags:r,parent:e}){var l;try{let o=t;return typeof t=="string"&&(o=(e==null?void 0:e.slots[t])||((l=e==null?void 0:e.parent)==null?void 0:l.slots[t]),console.warn(`Please set :use="$slots.${t}" in component to enable reactivity`)),o?s?i(o(),r):[o()]:n?n():a("div")}catch{return a("div")}}}),g=u({props:{use:{type:Function,default:void 0},unwrap:{type:[Boolean,String],default:!1}},render(t){return a(p,t)}});export{g as default}; +import{f as i}from"./Dnd51l0P.js";import{d as u,J as f,k as c,L as a,a0 as d}from"./Bb-NKJJJ.js";const p=u({name:"MDCSlot",functional:!0,props:{name:{type:String,default:"default"},unwrap:{type:[Boolean,String],default:!1},use:{type:Function,default:void 0}},setup(t){const{parent:s}=d(),{default:n}=f(),r=c(()=>typeof t.unwrap=="string"?t.unwrap.split(" "):["*"]);return{fallbackSlot:n,tags:r,parent:s}},render({use:t,unwrap:s,fallbackSlot:n,tags:r,parent:e}){var l;try{let o=t;return typeof t=="string"&&(o=(e==null?void 0:e.slots[t])||((l=e==null?void 0:e.parent)==null?void 0:l.slots[t]),console.warn(`Please set :use="$slots.${t}" in component to enable reactivity`)),o?s?i(o(),r):[o()]:n?n():a("div")}catch{return a("div")}}}),g=u({props:{use:{type:Function,default:void 0},unwrap:{type:[Boolean,String],default:!1}},render(t){return a(p,t)}});export{g as default}; diff --git a/_nuxt/DZkks4hD.js b/_nuxt/BZWGQyMp.js similarity index 99% rename from _nuxt/DZkks4hD.js rename to _nuxt/BZWGQyMp.js index fa0b19ad..9d9465f5 100644 --- a/_nuxt/DZkks4hD.js +++ b/_nuxt/BZWGQyMp.js @@ -1 +1 @@ -import{m as w,o as c,c as f,p as y,j as n,q as v,a as s,l as p,i as u,h as x,b as t,t as i,F as _}from"./BGikLauJ.js";const V={},k={version:"1.1",id:"Layer_1",xmlns:"http://www.w3.org/2000/svg","xmlns:xlink":"http://www.w3.org/1999/xlink",x:"0px",y:"0px",viewBox:"0 0 1160 1160","enable-background":"new 0 0 1160 1160","xml:space":"preserve"};function H(o,e){return c(),f("svg",k,e[0]||(e[0]=[y('',2)]))}const M=w(V,[["render",H]]),S=window.setInterval,C={class:"bg-slate-800"},P={class:"flex h-screen items-center justify-center"},z={class:"mb-4 space-y-2 text-sm"},B={class:"flex items-center"},j={class:"flex-none pl-2"},F={class:"flex items-center"},N={class:"flex-none pl-2"},W="TL;DR",$={__name:"card",setup(o){const e=n(!1),l=n(0),a=n(0),h=(g,r)=>g[r%g.length];v(()=>{S(function(){l.value>a.value?a.value+=1:l.value+=1},1500)});const m=["Skydiving","Bungee jumping","Rock climbing","White water rafting","Paragliding","Scuba diving","Base jumping","Mountain biking","Surfing","Snowboarding","Wakeboarding","Motorsports","Parkour","Zip-lining","Off-roading","Cave diving","Kiteboarding","Hang gliding","Free solo climbing","Bull riding","Ice climbing","Wing suit flying","Extreme skiing","Cliff jumping","Cave exploring"],d=["Astronaut","Archaeologist","Professional athlete","Chef","Fashion designer","Film director","Investigative journalist","Marine biologist","Musician","Photographer","Pilot","Professional gamer","Professor","Sculptor","Spy","Stand-up comedian","Surfing instructor","Travel writer","Wildlife photographer","Wine connoisseur","Yacht captain","Yoga instructor","Zookeeper","Adventure tour guide","Art curator"];return(g,r)=>{const b=M;return c(),f(_,null,[s(e)?(c(),f("div",{key:0,class:p(["absolute bottom-2 right-2 h-24 w-24",g.dynamic_bg])},[u(b)],2)):x("",!0),t("main",C,[t("div",P,[t("div",{id:"_card",onClick:r[0]||(r[0]=A=>e.value=!s(e)),class:"md:w-512px w-[350px] border border-white bg-black p-4 font-mono uppercase text-white shadow-xl shadow-yellow-950 transition-all hover:scale-102 hover:cursor-pointer"},[t("div",{class:"mb-2 font-extrabold tracking-wide"},i(W)),t("div",z,[r[5]||(r[5]=t("div",{class:"flex items-center"},[t("div",{class:"flex-none pr-2"},"Name"),t("div",{class:"flex-grow border-b border-dotted"}),t("div",{class:"flex-none pl-2"},"Colton")],-1)),t("div",B,[r[1]||(r[1]=t("div",{class:"flex-none pr-2"},"Profession",-1)),r[2]||(r[2]=t("div",{class:"flex-grow border-b border-dotted"},null,-1)),t("div",j,i(h(d,s(l))),1)]),t("div",F,[r[3]||(r[3]=t("div",{class:"flex-none pr-2"},"Hobby",-1)),r[4]||(r[4]=t("div",{class:"flex-grow border-b border-dotted"},null,-1)),t("div",N,i(h(m,s(a))),1)])])])])])],64)}}};export{$ as default}; +import{m as w,o as c,c as f,p as y,j as n,q as v,a as s,l as p,i as u,h as x,b as t,t as i,F as _}from"./Bb-NKJJJ.js";const V={},k={version:"1.1",id:"Layer_1",xmlns:"http://www.w3.org/2000/svg","xmlns:xlink":"http://www.w3.org/1999/xlink",x:"0px",y:"0px",viewBox:"0 0 1160 1160","enable-background":"new 0 0 1160 1160","xml:space":"preserve"};function H(o,e){return c(),f("svg",k,e[0]||(e[0]=[y('',2)]))}const M=w(V,[["render",H]]),S=window.setInterval,C={class:"bg-slate-800"},P={class:"flex h-screen items-center justify-center"},z={class:"mb-4 space-y-2 text-sm"},B={class:"flex items-center"},j={class:"flex-none pl-2"},F={class:"flex items-center"},N={class:"flex-none pl-2"},W="TL;DR",$={__name:"card",setup(o){const e=n(!1),l=n(0),a=n(0),h=(g,r)=>g[r%g.length];v(()=>{S(function(){l.value>a.value?a.value+=1:l.value+=1},1500)});const m=["Skydiving","Bungee jumping","Rock climbing","White water rafting","Paragliding","Scuba diving","Base jumping","Mountain biking","Surfing","Snowboarding","Wakeboarding","Motorsports","Parkour","Zip-lining","Off-roading","Cave diving","Kiteboarding","Hang gliding","Free solo climbing","Bull riding","Ice climbing","Wing suit flying","Extreme skiing","Cliff jumping","Cave exploring"],d=["Astronaut","Archaeologist","Professional athlete","Chef","Fashion designer","Film director","Investigative journalist","Marine biologist","Musician","Photographer","Pilot","Professional gamer","Professor","Sculptor","Spy","Stand-up comedian","Surfing instructor","Travel writer","Wildlife photographer","Wine connoisseur","Yacht captain","Yoga instructor","Zookeeper","Adventure tour guide","Art curator"];return(g,r)=>{const b=M;return c(),f(_,null,[s(e)?(c(),f("div",{key:0,class:p(["absolute bottom-2 right-2 h-24 w-24",g.dynamic_bg])},[u(b)],2)):x("",!0),t("main",C,[t("div",P,[t("div",{id:"_card",onClick:r[0]||(r[0]=A=>e.value=!s(e)),class:"md:w-512px w-[350px] border border-white bg-black p-4 font-mono uppercase text-white shadow-xl shadow-yellow-950 transition-all hover:scale-102 hover:cursor-pointer"},[t("div",{class:"mb-2 font-extrabold tracking-wide"},i(W)),t("div",z,[r[5]||(r[5]=t("div",{class:"flex items-center"},[t("div",{class:"flex-none pr-2"},"Name"),t("div",{class:"flex-grow border-b border-dotted"}),t("div",{class:"flex-none pl-2"},"Colton")],-1)),t("div",B,[r[1]||(r[1]=t("div",{class:"flex-none pr-2"},"Profession",-1)),r[2]||(r[2]=t("div",{class:"flex-grow border-b border-dotted"},null,-1)),t("div",j,i(h(d,s(l))),1)]),t("div",F,[r[3]||(r[3]=t("div",{class:"flex-none pr-2"},"Hobby",-1)),r[4]||(r[4]=t("div",{class:"flex-grow border-b border-dotted"},null,-1)),t("div",N,i(h(m,s(a))),1)])])])])])],64)}}};export{$ as default}; diff --git a/_nuxt/BGikLauJ.js b/_nuxt/Bb-NKJJJ.js similarity index 98% rename from _nuxt/BGikLauJ.js rename to _nuxt/Bb-NKJJJ.js index 0dff66f8..7eb13a69 100644 --- a/_nuxt/BGikLauJ.js +++ b/_nuxt/Bb-NKJJJ.js @@ -1,4 +1,4 @@ -const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["./Do41mdVm.js","./yGJPFCFo.js","./C-v3KzvZ.js","./Dnd51l0P.js","./DJ6fbLCw.js","./IShi1APO.js","./DukX5DGJ.js","./CgY3mHZ8.js","./UWyBikFq.js","./_...DjO8DLlz.css","./Dkf0i8RZ.js","./CnOV4Igh.js","./-8n5e5Fj.js","./BRTC5WMg.js","./CYeLrQ9C.js","./VwgkOAbI.js","./AY-jWpd0.js","./Do_OiPaD.js","./CUESTjlD.js","./KhAE7j4n.js","./d0UAFwvu.js","./Brnt98Va.js","./dVJITDym.js","./yJKb_Bp-.js","./ProseCode.B_fgAJq0.css","./Dz1t7Gnd.js"])))=>i.map(i=>d[i]); +const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["./I6a6eUEM.js","./CS06-0Od.js","./C-v3KzvZ.js","./Dnd51l0P.js","./mjvvwssR.js","./IShi1APO.js","./DrabSix5.js","./2mhhWi5_.js","./jaxqT3Hp.js","./_...DjO8DLlz.css","./BmBFiau8.js","./BqebzOVN.js","./Bbro_qAu.js","./CCH9cHpw.js","./Dn7xXnlv.js","./C-O5BI11.js","./cufizuWq.js","./UzigjR5s.js","./CMNpy2Is.js","./BxFzRrKR.js","./BY70Pa1e.js","./BLsxZwe9.js","./DS100L4J.js","./BMXP4Aud.js","./ProseCode.B_fgAJq0.css","./BenPNx-m.js"])))=>i.map(i=>d[i]); /** * @vue/shared v3.5.12 * (c) 2018-present Yuxi (Evan) You and Vue contributors @@ -20,7 +20,7 @@ const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["./Do41mdVm.js","./yGJP * vue-router v4.4.5 * (c) 2024 Eduardo San Martin Morote * @license MIT - */const io=typeof document<"u";function Wf(r){return typeof r=="object"||"displayName"in r||"props"in r||"__vccOpts"in r}function Zv(r){return r.__esModule||r[Symbol.toStringTag]==="Module"||r.default&&Wf(r.default)}const jt=Object.assign;function $i(r,a){const x={};for(const t in a){const w=a[t];x[t]=Wr(w)?w.map(r):r(w)}return x}const qo=()=>{},Wr=Array.isArray,qf=/#/g,Qv=/&/g,Jv=/\//g,e0=/=/g,t0=/\?/g,Xf=/\+/g,r0=/%5B/g,n0=/%5D/g,$f=/%5E/g,o0=/%60/g,Yf=/%7B/g,s0=/%7C/g,Kf=/%7D/g,i0=/%20/g;function fl(r){return encodeURI(""+r).replace(s0,"|").replace(r0,"[").replace(n0,"]")}function a0(r){return fl(r).replace(Yf,"{").replace(Kf,"}").replace($f,"^")}function Aa(r){return fl(r).replace(Xf,"%2B").replace(i0,"+").replace(qf,"%23").replace(Qv,"%26").replace(o0,"`").replace(Yf,"{").replace(Kf,"}").replace($f,"^")}function l0(r){return Aa(r).replace(e0,"%3D")}function u0(r){return fl(r).replace(qf,"%23").replace(t0,"%3F")}function c0(r){return r==null?"":u0(r).replace(Jv,"%2F")}function ts(r){try{return decodeURIComponent(""+r)}catch{}return""+r}const d0=/\/$/,f0=r=>r.replace(d0,"");function Yi(r,a,x="/"){let t,w={},g="",s="";const u=a.indexOf("#");let o=a.indexOf("?");return u=0&&(o=-1),o>-1&&(t=a.slice(0,o),g=a.slice(o+1,u>-1?u:a.length),w=r(g)),u>-1&&(t=t||a.slice(0,u),s=a.slice(u,a.length)),t=y0(t??a,x),{fullPath:t+(g&&"?")+g+s,path:t,query:w,hash:ts(s)}}function h0(r,a){const x=a.query?r(a.query):"";return a.path+(x&&"?")+x+(a.hash||"")}function ac(r,a){return!a||!r.toLowerCase().startsWith(a.toLowerCase())?r:r.slice(a.length)||"/"}function p0(r,a,x){const t=a.matched.length-1,w=x.matched.length-1;return t>-1&&t===w&&_o(a.matched[t],x.matched[w])&&Zf(a.params,x.params)&&r(a.query)===r(x.query)&&a.hash===x.hash}function _o(r,a){return(r.aliasOf||r)===(a.aliasOf||a)}function Zf(r,a){if(Object.keys(r).length!==Object.keys(a).length)return!1;for(const x in r)if(!m0(r[x],a[x]))return!1;return!0}function m0(r,a){return Wr(r)?lc(r,a):Wr(a)?lc(a,r):r===a}function lc(r,a){return Wr(a)?r.length===a.length&&r.every((x,t)=>x===a[t]):r.length===1&&r[0]===a}function y0(r,a){if(r.startsWith("/"))return r;if(!r)return a;const x=a.split("/"),t=r.split("/"),w=t[t.length-1];(w===".."||w===".")&&t.push("");let g=x.length-1,s,u;for(s=0;s1&&g--;else break;return x.slice(0,g).join("/")+"/"+t.slice(s).join("/")}const Br={path:"/",name:void 0,params:{},query:{},hash:"",fullPath:"/",matched:[],meta:{},redirectedFrom:void 0};var rs;(function(r){r.pop="pop",r.push="push"})(rs||(rs={}));var Xo;(function(r){r.back="back",r.forward="forward",r.unknown=""})(Xo||(Xo={}));function g0(r){if(!r)if(io){const a=document.querySelector("base");r=a&&a.getAttribute("href")||"/",r=r.replace(/^\w+:\/\/[^\/]+/,"")}else r="/";return r[0]!=="/"&&r[0]!=="#"&&(r="/"+r),f0(r)}const v0=/^[^#]+#/;function b0(r,a){return r.replace(v0,"#")+a}function _0(r,a){const x=document.documentElement.getBoundingClientRect(),t=r.getBoundingClientRect();return{behavior:a.behavior,left:t.left-x.left-(a.left||0),top:t.top-x.top-(a.top||0)}}const yi=()=>({left:window.scrollX,top:window.scrollY});function x0(r){let a;if("el"in r){const x=r.el,t=typeof x=="string"&&x.startsWith("#"),w=typeof x=="string"?t?document.getElementById(x.slice(1)):document.querySelector(x):x;if(!w)return;a=_0(w,r)}else a=r;"scrollBehavior"in document.documentElement.style?window.scrollTo(a):window.scrollTo(a.left!=null?a.left:window.scrollX,a.top!=null?a.top:window.scrollY)}function uc(r,a){return(history.state?history.state.position-a:-1)+r}const La=new Map;function w0(r,a){La.set(r,a)}function j0(r){const a=La.get(r);return La.delete(r),a}let S0=()=>location.protocol+"//"+location.host;function Qf(r,a){const{pathname:x,search:t,hash:w}=a,g=r.indexOf("#");if(g>-1){let u=w.includes(r.slice(g))?r.slice(g).length:1,o=w.slice(u);return o[0]!=="/"&&(o="/"+o),ac(o,"")}return ac(x,r)+t+w}function E0(r,a,x,t){let w=[],g=[],s=null;const u=({state:n})=>{const i=Qf(r,location),d=x.value,f=a.value;let p=0;if(n){if(x.value=i,a.value=n,s&&s===d){s=null;return}p=f?n.position-f.position:0}else t(i);w.forEach(c=>{c(x.value,d,{delta:p,type:rs.pop,direction:p?p>0?Xo.forward:Xo.back:Xo.unknown})})};function o(){s=x.value}function h(n){w.push(n);const i=()=>{const d=w.indexOf(n);d>-1&&w.splice(d,1)};return g.push(i),i}function y(){const{history:n}=window;n.state&&n.replaceState(jt({},n.state,{scroll:yi()}),"")}function m(){for(const n of g)n();g=[],window.removeEventListener("popstate",u),window.removeEventListener("beforeunload",y)}return window.addEventListener("popstate",u),window.addEventListener("beforeunload",y,{passive:!0}),{pauseListeners:o,listen:h,destroy:m}}function cc(r,a,x,t=!1,w=!1){return{back:r,current:a,forward:x,replaced:t,position:window.history.length,scroll:w?yi():null}}function T0(r){const{history:a,location:x}=window,t={value:Qf(r,x)},w={value:a.state};w.value||g(t.value,{back:null,current:t.value,forward:null,position:a.length-1,replaced:!0,scroll:null},!0);function g(o,h,y){const m=r.indexOf("#"),n=m>-1?(x.host&&document.querySelector("base")?r:r.slice(m))+o:S0()+r+o;try{a[y?"replaceState":"pushState"](h,"",n),w.value=h}catch(i){console.error(i),x[y?"replace":"assign"](n)}}function s(o,h){const y=jt({},a.state,cc(w.value.back,o,w.value.forward,!0),h,{position:w.value.position});g(o,y,!0),t.value=o}function u(o,h){const y=jt({},w.value,a.state,{forward:o,scroll:yi()});g(y.current,y,!0);const m=jt({},cc(t.value,o,null),{position:y.position+1},h);g(o,m,!1),t.value=o}return{location:t,state:w,push:u,replace:s}}function Jf(r){r=g0(r);const a=T0(r),x=E0(r,a.state,a.location,a.replace);function t(g,s=!0){s||x.pauseListeners(),history.go(g)}const w=jt({location:"",base:r,go:t,createHref:b0.bind(null,r)},a,x);return Object.defineProperty(w,"location",{enumerable:!0,get:()=>a.location.value}),Object.defineProperty(w,"state",{enumerable:!0,get:()=>a.state.value}),w}function k0(r){return r=location.host?r||location.pathname+location.search:"",r.includes("#")||(r+="#"),Jf(r)}function M0(r){return typeof r=="string"||r&&typeof r=="object"}function eh(r){return typeof r=="string"||typeof r=="symbol"}const th=Symbol("");var dc;(function(r){r[r.aborted=4]="aborted",r[r.cancelled=8]="cancelled",r[r.duplicated=16]="duplicated"})(dc||(dc={}));function xo(r,a){return jt(new Error,{type:r,[th]:!0},a)}function sn(r,a){return r instanceof Error&&th in r&&(a==null||!!(r.type&a))}const fc="[^/]+?",C0={sensitive:!1,strict:!1,start:!0,end:!0},O0=/[.+*?^${}()[\]/\\]/g;function P0(r,a){const x=jt({},C0,a),t=[];let w=x.start?"^":"";const g=[];for(const h of r){const y=h.length?[]:[90];x.strict&&!h.length&&(w+="/");for(let m=0;ma.length?a.length===1&&a[0]===80?1:-1:0}function rh(r,a){let x=0;const t=r.score,w=a.score;for(;x0&&a[a.length-1]<0}const L0={type:0,value:""},R0=/[a-zA-Z0-9_]/;function I0(r){if(!r)return[[]];if(r==="/")return[[L0]];if(!r.startsWith("/"))throw new Error(`Invalid path "${r}"`);function a(i){throw new Error(`ERR (${x})/"${h}": ${i}`)}let x=0,t=x;const w=[];let g;function s(){g&&w.push(g),g=[]}let u=0,o,h="",y="";function m(){h&&(x===0?g.push({type:0,value:h}):x===1||x===2||x===3?(g.length>1&&(o==="*"||o==="+")&&a(`A repeatable param (${h}) must be alone in its segment. eg: '/:ids+.`),g.push({type:1,value:h,regexp:y,repeatable:o==="*"||o==="+",optional:o==="*"||o==="?"})):a("Invalid state to consume buffer"),h="")}function n(){h+=o}for(;u{s(b)}:qo}function s(m){if(eh(m)){const n=t.get(m);n&&(t.delete(m),x.splice(x.indexOf(n),1),n.children.forEach(s),n.alias.forEach(s))}else{const n=x.indexOf(m);n>-1&&(x.splice(n,1),m.record.name&&t.delete(m.record.name),m.children.forEach(s),m.alias.forEach(s))}}function u(){return x}function o(m){const n=B0(m,x);x.splice(n,0,m),m.record.name&&!yc(m)&&t.set(m.record.name,m)}function h(m,n){let i,d={},f,p;if("name"in m&&m.name){if(i=t.get(m.name),!i)throw xo(1,{location:m});p=i.record.name,d=jt(pc(n.params,i.keys.filter(b=>!b.optional).concat(i.parent?i.parent.keys.filter(b=>b.optional):[]).map(b=>b.name)),m.params&&pc(m.params,i.keys.map(b=>b.name))),f=i.stringify(d)}else if(m.path!=null)f=m.path,i=x.find(b=>b.re.test(f)),i&&(d=i.parse(f),p=i.record.name);else{if(i=n.name?t.get(n.name):x.find(b=>b.re.test(n.path)),!i)throw xo(1,{location:m,currentLocation:n});p=i.record.name,d=jt({},n.params,m.params),f=i.stringify(d)}const c=[];let l=i;for(;l;)c.unshift(l.record),l=l.parent;return{name:p,path:f,params:d,matched:c,meta:U0(c)}}r.forEach(m=>g(m));function y(){x.length=0,t.clear()}return{addRoute:g,resolve:h,removeRoute:s,clearRoutes:y,getRoutes:u,getRecordMatcher:w}}function pc(r,a){const x={};for(const t of a)t in r&&(x[t]=r[t]);return x}function mc(r){const a={path:r.path,redirect:r.redirect,name:r.name,meta:r.meta||{},aliasOf:r.aliasOf,beforeEnter:r.beforeEnter,props:N0(r),children:r.children||[],instances:{},leaveGuards:new Set,updateGuards:new Set,enterCallbacks:{},components:"components"in r?r.components||null:r.component&&{default:r.component}};return Object.defineProperty(a,"mods",{value:{}}),a}function N0(r){const a={},x=r.props||!1;if("component"in r)a.default=x;else for(const t in r.components)a[t]=typeof x=="object"?x[t]:x;return a}function yc(r){for(;r;){if(r.record.aliasOf)return!0;r=r.parent}return!1}function U0(r){return r.reduce((a,x)=>jt(a,x.meta),{})}function gc(r,a){const x={};for(const t in r)x[t]=t in a?a[t]:r[t];return x}function B0(r,a){let x=0,t=a.length;for(;x!==t;){const g=x+t>>1;rh(r,a[g])<0?t=g:x=g+1}const w=G0(r);return w&&(t=a.lastIndexOf(w,t-1)),t}function G0(r){let a=r;for(;a=a.parent;)if(nh(a)&&rh(r,a)===0)return a}function nh({record:r}){return!!(r.name||r.components&&Object.keys(r.components).length||r.redirect)}function V0(r){const a={};if(r===""||r==="?")return a;const t=(r[0]==="?"?r.slice(1):r).split("&");for(let w=0;wg&&Aa(g)):[t&&Aa(t)]).forEach(g=>{g!==void 0&&(a+=(a.length?"&":"")+x,g!=null&&(a+="="+g))})}return a}function H0(r){const a={};for(const x in r){const t=r[x];t!==void 0&&(a[x]=Wr(t)?t.map(w=>w==null?null:""+w):t==null?t:""+t)}return a}const z0=Symbol(""),bc=Symbol(""),hl=Symbol(""),pl=Symbol(""),Ra=Symbol("");function Do(){let r=[];function a(t){return r.push(t),()=>{const w=r.indexOf(t);w>-1&&r.splice(w,1)}}function x(){r=[]}return{add:a,list:()=>r.slice(),reset:x}}function Sn(r,a,x,t,w,g=s=>s()){const s=t&&(t.enterCallbacks[w]=t.enterCallbacks[w]||[]);return()=>new Promise((u,o)=>{const h=n=>{n===!1?o(xo(4,{from:x,to:a})):n instanceof Error?o(n):M0(n)?o(xo(2,{from:a,to:n})):(s&&t.enterCallbacks[w]===s&&typeof n=="function"&&s.push(n),u())},y=g(()=>r.call(t&&t.instances[w],a,x,h));let m=Promise.resolve(y);r.length<3&&(m=m.then(h)),m.catch(n=>o(n))})}function Ki(r,a,x,t,w=g=>g()){const g=[];for(const s of r)for(const u in s.components){let o=s.components[u];if(!(a!=="beforeRouteEnter"&&!s.instances[u]))if(Wf(o)){const y=(o.__vccOpts||o)[a];y&&g.push(Sn(y,x,t,s,u,w))}else{let h=o();g.push(()=>h.then(y=>{if(!y)throw new Error(`Couldn't resolve component "${u}" at "${s.path}"`);const m=Zv(y)?y.default:y;s.mods[u]=y,s.components[u]=m;const i=(m.__vccOpts||m)[a];return i&&Sn(i,x,t,s,u,w)()}))}}return g}function _c(r){const a=zt(hl),x=zt(pl),t=vt(()=>{const o=Nt(r.to);return a.resolve(o)}),w=vt(()=>{const{matched:o}=t.value,{length:h}=o,y=o[h-1],m=x.matched;if(!y||!m.length)return-1;const n=m.findIndex(_o.bind(null,y));if(n>-1)return n;const i=xc(o[h-2]);return h>1&&xc(y)===i&&m[m.length-1].path!==i?m.findIndex(_o.bind(null,o[h-2])):n}),g=vt(()=>w.value>-1&&$0(x.params,t.value.params)),s=vt(()=>w.value>-1&&w.value===x.matched.length-1&&Zf(x.params,t.value.params));function u(o={}){return X0(o)?a[Nt(r.replace)?"replace":"push"](Nt(r.to)).catch(qo):Promise.resolve()}return{route:t,href:vt(()=>t.value.href),isActive:g,isExactActive:s,navigate:u}}const W0=gr({name:"RouterLink",compatConfig:{MODE:3},props:{to:{type:[String,Object],required:!0},replace:Boolean,activeClass:String,exactActiveClass:String,custom:Boolean,ariaCurrentValue:{type:String,default:"page"}},useLink:_c,setup(r,{slots:a}){const x=Rn(_c(r)),{options:t}=zt(hl),w=vt(()=>({[wc(r.activeClass,t.linkActiveClass,"router-link-active")]:x.isActive,[wc(r.exactActiveClass,t.linkExactActiveClass,"router-link-exact-active")]:x.isExactActive}));return()=>{const g=a.default&&a.default(x);return r.custom?g:Jt("a",{"aria-current":x.isExactActive?r.ariaCurrentValue:null,href:x.href,onClick:x.navigate,class:w.value},g)}}}),q0=W0;function X0(r){if(!(r.metaKey||r.altKey||r.ctrlKey||r.shiftKey)&&!r.defaultPrevented&&!(r.button!==void 0&&r.button!==0)){if(r.currentTarget&&r.currentTarget.getAttribute){const a=r.currentTarget.getAttribute("target");if(/\b_blank\b/i.test(a))return}return r.preventDefault&&r.preventDefault(),!0}}function $0(r,a){for(const x in a){const t=a[x],w=r[x];if(typeof t=="string"){if(t!==w)return!1}else if(!Wr(w)||w.length!==t.length||t.some((g,s)=>g!==w[s]))return!1}return!0}function xc(r){return r?r.aliasOf?r.aliasOf.path:r.path:""}const wc=(r,a,x)=>r??a??x,Y0=gr({name:"RouterView",inheritAttrs:!1,props:{name:{type:String,default:"default"},route:Object},compatConfig:{MODE:3},setup(r,{attrs:a,slots:x}){const t=zt(Ra),w=vt(()=>r.route||t.value),g=zt(bc,0),s=vt(()=>{let h=Nt(g);const{matched:y}=w.value;let m;for(;(m=y[h])&&!m.components;)h++;return h}),u=vt(()=>w.value.matched[s.value]);Ir(bc,vt(()=>s.value+1)),Ir(z0,u),Ir(Ra,w);const o=mt();return On(()=>[o.value,u.value,r.name],([h,y,m],[n,i,d])=>{y&&(y.instances[m]=h,i&&i!==y&&h&&h===n&&(y.leaveGuards.size||(y.leaveGuards=i.leaveGuards),y.updateGuards.size||(y.updateGuards=i.updateGuards))),h&&y&&(!i||!_o(y,i)||!n)&&(y.enterCallbacks[m]||[]).forEach(f=>f(h))},{flush:"post"}),()=>{const h=w.value,y=r.name,m=u.value,n=m&&m.components[y];if(!n)return jc(x.default,{Component:n,route:h});const i=m.props[y],d=i?i===!0?h.params:typeof i=="function"?i(h):i:null,p=Jt(n,jt({},d,a,{onVnodeUnmounted:c=>{c.component.isUnmounted&&(m.instances[y]=null)},ref:o}));return jc(x.default,{Component:p,route:h})||p}}});function jc(r,a){if(!r)return null;const x=r(a);return x.length===1?x[0]:x}const oh=Y0;function K0(r){const a=F0(r.routes,r),x=r.parseQuery||V0,t=r.stringifyQuery||vc,w=r.history,g=Do(),s=Do(),u=Do(),o=Zo(Br);let h=Br;io&&r.scrollBehavior&&"scrollRestoration"in history&&(history.scrollRestoration="manual");const y=$i.bind(null,K=>""+K),m=$i.bind(null,c0),n=$i.bind(null,ts);function i(K,J){let ne,ue;return eh(K)?(ne=a.getRecordMatcher(K),ue=J):ue=K,a.addRoute(ue,ne)}function d(K){const J=a.getRecordMatcher(K);J&&a.removeRoute(J)}function f(){return a.getRoutes().map(K=>K.record)}function p(K){return!!a.getRecordMatcher(K)}function c(K,J){if(J=jt({},J||o.value),typeof K=="string"){const z=Yi(x,K,J.path),te=a.resolve({path:z.path},J),ce=w.createHref(z.fullPath);return jt(z,te,{params:n(te.params),hash:ts(z.hash),redirectedFrom:void 0,href:ce})}let ne;if(K.path!=null)ne=jt({},K,{path:Yi(x,K.path,J.path).path});else{const z=jt({},K.params);for(const te in z)z[te]==null&&delete z[te];ne=jt({},K,{params:m(z)}),J.params=m(J.params)}const ue=a.resolve(ne,J),ie=K.hash||"";ue.params=y(n(ue.params));const he=h0(t,jt({},K,{hash:a0(ie),path:ue.path})),Y=w.createHref(he);return jt({fullPath:he,hash:ie,query:t===vc?H0(K.query):K.query||{}},ue,{redirectedFrom:void 0,href:Y})}function l(K){return typeof K=="string"?Yi(x,K,o.value.path):jt({},K)}function b(K,J){if(h!==K)return xo(8,{from:J,to:K})}function v(K){return E(K)}function j(K){return v(jt(l(K),{replace:!0}))}function T(K){const J=K.matched[K.matched.length-1];if(J&&J.redirect){const{redirect:ne}=J;let ue=typeof ne=="function"?ne(K):ne;return typeof ue=="string"&&(ue=ue.includes("?")||ue.includes("#")?ue=l(ue):{path:ue},ue.params={}),jt({query:K.query,hash:K.hash,params:ue.path!=null?{}:K.params},ue)}}function E(K,J){const ne=h=c(K),ue=o.value,ie=K.state,he=K.force,Y=K.replace===!0,z=T(ne);if(z)return E(jt(l(z),{state:typeof z=="object"?jt({},ie,z.state):ie,force:he,replace:Y}),J||ne);const te=ne;te.redirectedFrom=J;let ce;return!he&&p0(t,ue,ne)&&(ce=xo(16,{to:te,from:ue}),ee(ue,ue,!0,!1)),(ce?Promise.resolve(ce):L(te,ue)).catch(fe=>sn(fe)?sn(fe,2)?fe:Q(fe):V(fe,te,ue)).then(fe=>{if(fe){if(sn(fe,2))return E(jt({replace:Y},l(fe.to),{state:typeof fe.to=="object"?jt({},ie,fe.to.state):ie,force:he}),J||te)}else fe=I(te,ue,!0,Y,ie);return C(te,ue,fe),fe})}function M(K,J){const ne=b(K,J);return ne?Promise.reject(ne):Promise.resolve()}function P(K){const J=D.values().next().value;return J&&typeof J.runWithContext=="function"?J.runWithContext(K):K()}function L(K,J){let ne;const[ue,ie,he]=Z0(K,J);ne=Ki(ue.reverse(),"beforeRouteLeave",K,J);for(const z of ue)z.leaveGuards.forEach(te=>{ne.push(Sn(te,K,J))});const Y=M.bind(null,K,J);return ne.push(Y),X(ne).then(()=>{ne=[];for(const z of g.list())ne.push(Sn(z,K,J));return ne.push(Y),X(ne)}).then(()=>{ne=Ki(ie,"beforeRouteUpdate",K,J);for(const z of ie)z.updateGuards.forEach(te=>{ne.push(Sn(te,K,J))});return ne.push(Y),X(ne)}).then(()=>{ne=[];for(const z of he)if(z.beforeEnter)if(Wr(z.beforeEnter))for(const te of z.beforeEnter)ne.push(Sn(te,K,J));else ne.push(Sn(z.beforeEnter,K,J));return ne.push(Y),X(ne)}).then(()=>(K.matched.forEach(z=>z.enterCallbacks={}),ne=Ki(he,"beforeRouteEnter",K,J,P),ne.push(Y),X(ne))).then(()=>{ne=[];for(const z of s.list())ne.push(Sn(z,K,J));return ne.push(Y),X(ne)}).catch(z=>sn(z,8)?z:Promise.reject(z))}function C(K,J,ne){u.list().forEach(ue=>P(()=>ue(K,J,ne)))}function I(K,J,ne,ue,ie){const he=b(K,J);if(he)return he;const Y=J===Br,z=io?history.state:{};ne&&(ue||Y?w.replace(K.fullPath,jt({scroll:Y&&z&&z.scroll},ie)):w.push(K.fullPath,ie)),o.value=K,ee(K,J,ne,Y),Q()}let A;function N(){A||(A=w.listen((K,J,ne)=>{if(!G.listening)return;const ue=c(K),ie=T(ue);if(ie){E(jt(ie,{replace:!0}),ue).catch(qo);return}h=ue;const he=o.value;io&&w0(uc(he.fullPath,ne.delta),yi()),L(ue,he).catch(Y=>sn(Y,12)?Y:sn(Y,2)?(E(Y.to,ue).then(z=>{sn(z,20)&&!ne.delta&&ne.type===rs.pop&&w.go(-1,!1)}).catch(qo),Promise.reject()):(ne.delta&&w.go(-ne.delta,!1),V(Y,ue,he))).then(Y=>{Y=Y||I(ue,he,!1),Y&&(ne.delta&&!sn(Y,8)?w.go(-ne.delta,!1):ne.type===rs.pop&&sn(Y,20)&&w.go(-1,!1)),C(ue,he,Y)}).catch(qo)}))}let F=Do(),U=Do(),W;function V(K,J,ne){Q(K);const ue=U.list();return ue.length?ue.forEach(ie=>ie(K,J,ne)):console.error(K),Promise.reject(K)}function H(){return W&&o.value!==Br?Promise.resolve():new Promise((K,J)=>{F.add([K,J])})}function Q(K){return W||(W=!K,N(),F.list().forEach(([J,ne])=>K?ne(K):J()),F.reset()),K}function ee(K,J,ne,ue){const{scrollBehavior:ie}=r;if(!io||!ie)return Promise.resolve();const he=!ne&&j0(uc(K.fullPath,0))||(ue||!ne)&&history.state&&history.state.scroll||null;return Pr().then(()=>ie(K,J,he)).then(Y=>Y&&x0(Y)).catch(Y=>V(Y,K,J))}const ae=K=>w.go(K);let k;const D=new Set,G={currentRoute:o,listening:!0,addRoute:i,removeRoute:d,clearRoutes:a.clearRoutes,hasRoute:p,getRoutes:f,resolve:c,options:r,push:v,replace:j,go:ae,back:()=>ae(-1),forward:()=>ae(1),beforeEach:g.add,beforeResolve:s.add,afterEach:u.add,onError:U.add,isReady:H,install(K){const J=this;K.component("RouterLink",q0),K.component("RouterView",oh),K.config.globalProperties.$router=J,Object.defineProperty(K.config.globalProperties,"$route",{enumerable:!0,get:()=>Nt(o)}),io&&!k&&o.value===Br&&(k=!0,v(w.location).catch(ie=>{}));const ne={};for(const ie in Br)Object.defineProperty(ne,ie,{get:()=>o.value[ie],enumerable:!0});K.provide(hl,J),K.provide(pl,cn(ne)),K.provide(Ra,o);const ue=K.unmount;D.add(K),K.unmount=function(){D.delete(K),D.size<1&&(h=Br,A&&A(),A=null,o.value=Br,k=!1,W=!1),ue()}}};function X(K){return K.reduce((J,ne)=>J.then(()=>P(ne)),Promise.resolve())}return G}function Z0(r,a){const x=[],t=[],w=[],g=Math.max(a.matched.length,r.matched.length);for(let s=0;s_o(h,u))?t.push(u):x.push(u));const o=r.matched[s];o&&(a.matched.find(h=>_o(h,o))||w.push(o))}return[x,t,w]}function Q0(r){return zt(pl)}const J0=(r,a)=>a.path.replace(/(:\w+)\([^)]+\)/g,"$1").replace(/(:\w+)[?+*]/g,"$1").replace(/:\w+/g,x=>{var t;return((t=r.params[x.slice(1)])==null?void 0:t.toString())||""}),Ia=(r,a)=>{const x=r.route.matched.find(w=>{var g;return((g=w.components)==null?void 0:g.default)===r.Component.type}),t=a??(x==null?void 0:x.meta.key)??(x&&J0(r.route,x));return typeof t=="function"?t(r.route):t},eb=(r,a)=>({default:()=>r?Jt(Hp,r===!0?{}:r,a):a});function ml(r){return Array.isArray(r)?r:[r]}const tb="modulepreload",rb=function(r,a){return new URL(r,a).href},Sc={},rt=function(a,x,t){let w=Promise.resolve();if(x&&x.length>0){const s=document.getElementsByTagName("link"),u=document.querySelector("meta[property=csp-nonce]"),o=(u==null?void 0:u.nonce)||(u==null?void 0:u.getAttribute("nonce"));w=Promise.allSettled(x.map(h=>{if(h=rb(h,t),h in Sc)return;Sc[h]=!0;const y=h.endsWith(".css"),m=y?'[rel="stylesheet"]':"";if(!!t)for(let d=s.length-1;d>=0;d--){const f=s[d];if(f.href===h&&(!y||f.rel==="stylesheet"))return}else if(document.querySelector(`link[href="${h}"]${m}`))return;const i=document.createElement("link");if(i.rel=y?"stylesheet":tb,y||(i.as="script"),i.crossOrigin="",i.href=h,o&&i.setAttribute("nonce",o),document.head.appendChild(i),y)return new Promise((d,f)=>{i.addEventListener("load",d),i.addEventListener("error",()=>f(new Error(`Unable to preload CSS for ${h}`)))})}))}function g(s){const u=new Event("vite:preloadError",{cancelable:!0});if(u.payload=s,window.dispatchEvent(u),!u.defaultPrevented)throw s}return w.then(s=>{for(const u of s||[])u.status==="rejected"&&g(u.reason);return a().catch(g)})},nb={layout:"empty"},ob={layout:"light"},sb={layout:"light"},ib={layout:"light"},ab={layout:"light"},lb={layout:"light"},ub={layout:"light"},cb={layout:"light"},db={layout:"light"},fb={layout:"light"},hb={layout:"light"},pb={layout:"light"},Zi=[{name:"articles-slug",path:"/articles/:slug(.*)*",component:()=>rt(()=>import("./Do41mdVm.js"),__vite__mapDeps([0,1,2,3,4,5,6,7,8,9]),import.meta.url)},{name:"articles",path:"/articles",component:()=>rt(()=>import("./Dkf0i8RZ.js"),__vite__mapDeps([10,1,2,3,4,5,6,7,8]),import.meta.url)},{name:"card",path:"/card",meta:nb||{},component:()=>rt(()=>import("./DZkks4hD.js"),[],import.meta.url)},{name:"examples-nested_transitions",path:"/examples/nested_transitions",component:()=>rt(()=>import("./C_llEvAK.js"),[],import.meta.url)},{name:"index",path:"/",component:()=>rt(()=>import("./CnOV4Igh.js"),__vite__mapDeps([11,12,8,5,4]),import.meta.url)},{name:"playground-audio",path:"/playground/audio",component:()=>rt(()=>import("./CzhJn57k.js"),[],import.meta.url)},{name:"playground-chords",path:"/playground/chords",meta:ob||{},component:()=>rt(()=>import("./DrbA_f4z.js"),[],import.meta.url)},{name:"playground-conway",path:"/playground/conway",meta:sb||{},component:()=>rt(()=>import("./-R_hya3P.js"),[],import.meta.url)},{name:"playground-french",path:"/playground/french",meta:ib||{},component:()=>rt(()=>import("./BRTC5WMg.js"),__vite__mapDeps([13,5,7]),import.meta.url)},{name:"playground",path:"/playground",component:()=>rt(()=>import("./CYeLrQ9C.js"),__vite__mapDeps([14,12]),import.meta.url)},{name:"playground-matrix",path:"/playground/matrix",meta:ab||{},component:()=>rt(()=>import("./ZO3tSbNI.js"),[],import.meta.url)},{name:"playground-metronome",path:"/playground/metronome",meta:lb||{},component:()=>rt(()=>import("./DCgnBWC9.js"),[],import.meta.url)},{name:"playground-midi",path:"/playground/midi",meta:ub||{},component:()=>rt(()=>import("./CDVU8JqM.js"),[],import.meta.url)},{name:"playground-palettes-mountains",path:"/playground/palettes/mountains",meta:cb||{},component:()=>rt(()=>import("./D3_S2KpX.js"),[],import.meta.url)},{name:"playground-palettes-variance",path:"/playground/palettes/variance",meta:db||{},component:()=>rt(()=>import("./DSaqcxQi.js"),[],import.meta.url)},{name:"playground-plotter",path:"/playground/plotter",meta:fb||{},component:()=>rt(()=>import("./D0Omvic7.js"),[],import.meta.url)},{name:"playground-tiling",path:"/playground/tiling",meta:hb||{},component:()=>rt(()=>import("./CDv6ev47.js"),[],import.meta.url)},{name:"playground-waves",path:"/playground/waves",meta:pb||{},component:()=>rt(()=>import("./UcoZFNdm.js"),[],import.meta.url)},{name:"talks",path:"/talks",component:()=>rt(()=>import("./DJ0wDsqg.js"),[],import.meta.url)}],sh=(r,a,x)=>(a=a===!0?{}:a,{default:()=>{var t;return a?Jt(r,a,x):(t=x.default)==null?void 0:t.call(x)}});function Ec(r){const a=(r==null?void 0:r.meta.key)??r.path.replace(/(:\w+)\([^)]+\)/g,"$1").replace(/(:\w+)[?+*]/g,"$1").replace(/:\w+/g,x=>{var t;return((t=r.params[x.slice(1)])==null?void 0:t.toString())||""});return typeof a=="function"?a(r):a}function mb(r,a){return r===a||a===Br?!1:Ec(r)!==Ec(a)?!0:!r.matched.every((t,w)=>{var g,s;return t.components&&t.components.default===((s=(g=a.matched[w])==null?void 0:g.components)==null?void 0:s.default)})}const yb={scrollBehavior(r,a,x){var h;const t=Dt(),w=((h=jr().options)==null?void 0:h.scrollBehaviorType)??"auto";let g=x||void 0;const s=typeof r.meta.scrollToTop=="function"?r.meta.scrollToTop(r,a):r.meta.scrollToTop;if(!g&&a&&r&&s!==!1&&mb(r,a)&&(g={left:0,top:0}),r.path===a.path)return a.hash&&!r.hash?{left:0,top:0}:r.hash?{el:r.hash,top:Tc(r.hash),behavior:w}:!1;const u=y=>!!(y.meta.pageTransition??xa),o=u(a)&&u(r)?"page:transition:finish":"page:finish";return new Promise(y=>{t.hooks.hookOnce(o,async()=>{await new Promise(m=>setTimeout(m,0)),r.hash&&(g={el:r.hash,top:Tc(r.hash),behavior:w}),y(g)})})}};function Tc(r){try{const a=document.querySelector(r);if(a)return(Number.parseFloat(getComputedStyle(a).scrollMarginTop)||0)+(Number.parseFloat(getComputedStyle(document.documentElement).scrollPaddingTop)||0)}catch{}return 0}const gb={hashMode:!1,scrollBehaviorType:"auto"},Lr={...gb,...yb},vb=async r=>{var o;let a,x;if(!((o=r.meta)!=null&&o.validate))return;const t=Dt(),w=jr(),g=([a,x]=yo(()=>Promise.resolve(r.meta.validate(r))),a=await a,x(),a);if(g===!0)return;const s=pi({statusCode:g&&g.statusCode||404,statusMessage:g&&g.statusMessage||`Page Not Found: ${r.fullPath}`,data:{path:r.fullPath}}),u=w.beforeResolve(h=>{if(u(),h===r){const y=w.afterEach(async()=>{y(),await t.runWithContext(()=>lo(s)),window==null||window.history.pushState({},"",r.fullPath)});return!1}})},bb=async r=>{let a,x;const t=([a,x]=yo(()=>dl(r.path)),a=await a,x(),a);if(t.redirect)return pn(t.redirect,{acceptRelative:!0})?(window.location.href=t.redirect,!1):t.redirect},_b=[vb,bb],$o={};function xb(r,a,x){const{pathname:t,search:w,hash:g}=a,s=r.indexOf("#");if(s>-1){const h=g.includes(r.slice(s))?r.slice(s).length:1;let y=g.slice(h);return y[0]!=="/"&&(y="/"+y),zu(y,"")}const u=zu(t,r),o=!x||zy(u,x,{trailingSlash:!0})?u:x;return o+(o.includes("?")?"":w)+g}const wb=tn({name:"nuxt:router",enforce:"pre",async setup(r){var p;let a,x,t=us().app.baseURL;Lr.hashMode&&!t.includes("#")&&(t+="#");const w=((p=Lr.history)==null?void 0:p.call(Lr,t))??(Lr.hashMode?k0(t):Jf(t)),g=Lr.routes?([a,x]=yo(()=>Lr.routes(Zi)),a=await a,x(),a??Zi):Zi;let s;const u=K0({...Lr,scrollBehavior:(c,l,b)=>{if(l===Br){s=b;return}if(Lr.scrollBehavior){if(u.options.scrollBehavior=Lr.scrollBehavior,"scrollRestoration"in window.history){const v=u.beforeEach(()=>{v(),window.history.scrollRestoration="manual"})}return Lr.scrollBehavior(c,Br,s||b)}},history:w,routes:g});"scrollRestoration"in window.history&&(window.history.scrollRestoration="auto"),r.vueApp.use(u);const o=Zo(u.currentRoute.value);u.afterEach((c,l)=>{o.value=l}),Object.defineProperty(r.vueApp.config.globalProperties,"previousRoute",{get:()=>o.value});const h=xb(t,window.location,r.payload.path),y=Zo(u.currentRoute.value),m=()=>{y.value=u.currentRoute.value};r.hook("page:finish",m),u.afterEach((c,l)=>{var b,v,j,T;((v=(b=c.matched[0])==null?void 0:b.components)==null?void 0:v.default)===((T=(j=l.matched[0])==null?void 0:j.components)==null?void 0:T.default)&&m()});const n={};for(const c in y.value)Object.defineProperty(n,c,{get:()=>y.value[c],enumerable:!0});r._route=cn(n),r._middleware=r._middleware||{global:[],named:{}};const i=hi();u.afterEach(async(c,l,b)=>{delete r._processingMiddleware,!r.isHydrating&&i.value&&await r.runWithContext(Hg),b&&await r.callHook("page:loading:end"),c.matched.length===0&&await r.runWithContext(()=>lo(Sa({statusCode:404,fatal:!1,statusMessage:`Page not found: ${c.fullPath}`,data:{path:c.fullPath}})))});try{[a,x]=yo(()=>u.isReady()),await a,x()}catch(c){[a,x]=yo(()=>r.runWithContext(()=>lo(c))),await a,x()}const d=h!==u.currentRoute.value.fullPath?u.resolve(h):u.currentRoute.value;m();const f=r.payload.state._layout;return u.beforeEach(async(c,l)=>{var b;await r.callHook("page:loading:start"),c.meta=Rn(c.meta),r.isHydrating&&f&&!Pn(c.meta.layout)&&(c.meta.layout=f),r._processingMiddleware=!0;{const v=new Set([..._b,...r._middleware.global]);for(const j of c.matched){const T=j.meta.middleware;if(T)for(const E of ml(T))v.add(E)}{const j=await r.runWithContext(()=>dl(c.path));if(j.appMiddleware)for(const T in j.appMiddleware)j.appMiddleware[T]?v.add(T):v.delete(T)}for(const j of v){const T=typeof j=="string"?r._middleware.named[j]||await((b=$o[j])==null?void 0:b.call($o).then(M=>M.default||M)):j;if(!T)throw new Error(`Unknown route middleware: '${j}'.`);const E=await r.runWithContext(()=>T(c,l));if(!r.payload.serverRendered&&r.isHydrating&&(E===!1||E instanceof Error)){const M=E||Sa({statusCode:404,statusMessage:`Page Not Found: ${h}`});return await r.runWithContext(()=>lo(M)),!1}if(E!==!0&&(E||E===!1))return E}}}),u.onError(async()=>{delete r._processingMiddleware,await r.callHook("page:loading:end")}),r.hooks.hookOnce("app:created",async()=>{try{"name"in d&&(d.name=void 0),await u.replace({...d,force:!0}),u.options.scrollBehavior=Lr.scrollBehavior}catch(c){await r.runWithContext(()=>lo(c))}}),{provide:{router:u}}}}),Da=globalThis.requestIdleCallback||(r=>{const a=Date.now(),x={didTimeout:!1,timeRemaining:()=>Math.max(0,50-(Date.now()-a))};return setTimeout(()=>{r(x)},1)}),jb=globalThis.cancelIdleCallback||(r=>{clearTimeout(r)}),gi=r=>{const a=Dt();a.isHydrating?a.hooks.hookOnce("app:suspense:resolve",()=>{Da(()=>r())}):Da(()=>r())},Sb=tn({name:"nuxt:payload",setup(r){jr().beforeResolve(async(a,x)=>{if(a.path===x.path)return;const t=await sc(a.path);t&&Object.assign(r.static.data,t.data)}),gi(()=>{var a;r.hooks.hook("link:prefetch",async x=>{const{hostname:t}=new URL(x,window.location.href);t===window.location.hostname&&await sc(x)}),((a=navigator.connection)==null?void 0:a.effectiveType)!=="slow-2g"&&setTimeout(mi,1e3)})}}),Eb=tn(()=>{const r=jr();gi(()=>{r.beforeResolve(async()=>{await new Promise(a=>{setTimeout(a,100),requestAnimationFrame(()=>{setTimeout(a,0)})})})})}),Tb=tn(r=>{let a;async function x(){const t=await mi();a&&clearTimeout(a),a=setTimeout(x,Yu);try{const w=await $fetch(ll("builds/latest.json")+`?${Date.now()}`);w.id!==t.id&&r.hooks.callHook("app:manifest:update",w)}catch{}}gi(()=>{a=setTimeout(x,Yu)})});function kb(r={}){const a=r.path||window.location.pathname;let x={};try{x=qs(sessionStorage.getItem("nuxt:reload")||"{}")}catch{}if(r.force||(x==null?void 0:x.path)!==a||(x==null?void 0:x.expires){t.clear()}),r.hook("app:chunkError",({error:g})=>{t.add(g)});function w(g){const u="href"in g&&g.href[0]==="#"?x.app.baseURL+g.href:di(x.app.baseURL,g.fullPath);kb({path:u,persistState:!0})}r.hook("app:manifest:update",()=>{a.beforeResolve(w)}),a.onError((g,s)=>{t.has(g)&&w(s)})}}),Cb=ht(()=>rt(()=>import("./VwgkOAbI.js"),__vite__mapDeps([15,6,1,2,3,4,5,16,7,8]),import.meta.url).then(r=>r.default||r.default||r)),Ob=ht(()=>rt(()=>import("./Do_OiPaD.js"),__vite__mapDeps([17,16,5,7,8,4]),import.meta.url).then(r=>r.default||r.default||r)),Pb=ht(()=>rt(()=>import("./CUESTjlD.js"),__vite__mapDeps([18,5,7,8,4]),import.meta.url).then(r=>r.default||r.default||r)),Ab=ht(()=>rt(()=>import("./AY-jWpd0.js"),__vite__mapDeps([16,5,7,8,4]),import.meta.url).then(r=>r.default||r.default||r)),Lb=ht(()=>rt(()=>import("./DukX5DGJ.js"),__vite__mapDeps([6,1,2,3,4,5]),import.meta.url).then(r=>r.default||r.default||r)),Rb=ht(()=>rt(()=>import("./KhAE7j4n.js"),__vite__mapDeps([19,1,2,3,4,5]),import.meta.url).then(r=>r.default||r.default||r)),Ib=ht(()=>rt(()=>import("./d0UAFwvu.js"),__vite__mapDeps([20,3]),import.meta.url).then(r=>r.default||r.default||r)),Db=ht(()=>rt(()=>import("./BnWdfnc6.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Fb=ht(()=>rt(()=>import("./M6Zd1ArL.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Nb=ht(()=>rt(()=>import("./Brnt98Va.js"),__vite__mapDeps([21,20,3]),import.meta.url).then(r=>r.default||r.default||r)),Ub=ht(()=>rt(()=>import("./dVJITDym.js"),__vite__mapDeps([22,23,24]),import.meta.url).then(r=>r.default||r.default||r)),Bb=ht(()=>rt(()=>import("./CruaMelD.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Gb=ht(()=>rt(()=>import("./Dz1t7Gnd.js"),__vite__mapDeps([25,23,24]),import.meta.url).then(r=>r.default||r.default||r)),Vb=ht(()=>rt(()=>import("./DO9Ai2bz.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Hb=ht(()=>rt(()=>import("./D-AOKchP.js"),[],import.meta.url).then(r=>r.default||r.default||r)),zb=ht(()=>rt(()=>import("./BQ6KdT-G.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Wb=ht(()=>rt(()=>import("./BdJQj3nn.js"),[],import.meta.url).then(r=>r.default||r.default||r)),qb=ht(()=>rt(()=>import("./DX7ac3Mz.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Xb=ht(()=>rt(()=>import("./_kju5I3X.js"),[],import.meta.url).then(r=>r.default||r.default||r)),$b=ht(()=>rt(()=>import("./DsIJUosc.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Yb=ht(()=>rt(()=>import("./CQCYFxb2.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Kb=ht(()=>rt(()=>import("./BUYHwnpH.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Zb=ht(()=>rt(()=>import("./D7IuvTrr.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Qb=ht(()=>rt(()=>import("./RX6TINi_.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Jb=ht(()=>rt(()=>import("./Bdy-hNpQ.js"),[],import.meta.url).then(r=>r.default||r.default||r)),e1=ht(()=>rt(()=>import("./DbdbR-pn.js"),[],import.meta.url).then(r=>r.default||r.default||r)),t1=ht(()=>rt(()=>import("./BEt4369s.js"),[],import.meta.url).then(r=>r.default||r.default||r)),r1=ht(()=>rt(()=>import("./m3fkSCjB.js"),[],import.meta.url).then(r=>r.default||r.default||r)),n1=ht(()=>rt(()=>import("./BYwurayZ.js"),[],import.meta.url).then(r=>r.default||r.default||r)),o1=ht(()=>rt(()=>import("./6mOJeL1q.js"),[],import.meta.url).then(r=>r.default||r.default||r)),s1=ht(()=>rt(()=>import("./C7UpC-lS.js"),[],import.meta.url).then(r=>r.default||r.default||r)),i1=ht(()=>rt(()=>import("./D4ZOq_QP.js"),[],import.meta.url).then(r=>r.default||r.default||r)),a1=ht(()=>rt(()=>import("./CjTcA1Xt.js"),[],import.meta.url).then(r=>r.default||r.default||r)),l1=ht(()=>rt(()=>import("./CJwA9VmX.js"),[],import.meta.url).then(r=>r.default||r.default||r)),u1=ht(()=>rt(()=>import("./BMI0969X.js"),[],import.meta.url).then(r=>r.default||r.default||r)),c1=ht(()=>rt(()=>import("./B9UQgCMx.js"),[],import.meta.url).then(r=>r.default||r.default||r)),d1=[["ContentDoc",Cb],["ContentList",Ob],["ContentNavigation",Pb],["ContentQuery",Ab],["ContentRenderer",Lb],["ContentRendererMarkdown",Rb],["MDCSlot",Ib],["DocumentDrivenEmpty",Db],["DocumentDrivenNotFound",Fb],["Markdown",Nb],["ProseCode",Ub],["ProseCodeInline",Bb],["ProsePre",Gb],["ProseA",Vb],["ProseBlockquote",Hb],["ProseEm",zb],["ProseH1",Wb],["ProseH2",qb],["ProseH3",Xb],["ProseH4",$b],["ProseH5",Yb],["ProseH6",Kb],["ProseHr",Zb],["ProseImg",Qb],["ProseLi",Jb],["ProseOl",e1],["ProseP",t1],["ProseScript",r1],["ProseStrong",n1],["ProseTable",o1],["ProseTbody",s1],["ProseTd",i1],["ProseTh",a1],["ProseThead",l1],["ProseTr",u1],["ProseUl",c1]],f1=tn({name:"nuxt:global-components",setup(r){for(const[a,x]of d1)r.vueApp.component(a,x),r.vueApp.component("Lazy"+a,x)}}),Tn={default:()=>rt(()=>import("./jcdaHq-b.js"),[],import.meta.url),empty:()=>rt(()=>import("./B3q-b4GE.js"),[],import.meta.url),light:()=>rt(()=>import("./Btpue746.js"),[],import.meta.url)},h1=tn({name:"nuxt:prefetch",setup(r){const a=jr();r.hooks.hook("app:mounted",()=>{a.beforeEach(async x=>{var w;const t=(w=x==null?void 0:x.meta)==null?void 0:w.layout;t&&typeof Tn[t]=="function"&&await Tn[t]()})}),r.hooks.hook("link:prefetch",x=>{if(pn(x))return;const t=a.resolve(x);if(!t)return;const w=t.meta.layout;let g=ml(t.meta.middleware);g=g.filter(s=>typeof s=="string");for(const s of g)typeof $o[s]=="function"&&$o[s]();w&&typeof Tn[w]=="function"&&Tn[w]()})}});var ks=typeof globalThis<"u"?globalThis:typeof window<"u"?window:typeof global<"u"?global:typeof self<"u"?self:{};function p1(r){return r&&r.__esModule&&Object.prototype.hasOwnProperty.call(r,"default")?r.default:r}function Ms(r){throw new Error('Could not dynamically require "'+r+'". Please configure the dynamicRequireTargets or/and ignoreDynamicRequires option of @rollup/plugin-commonjs appropriately for this require call to work.')}var ih={exports:{}};/*! p5.js v1.11.0 September 25, 2024 */(function(r,a){(function(x){r.exports=x()})(function(){var x;return function t(w,g,s){function u(y,m){if(!g[y]){if(!w[y]){var n=typeof Ms=="function"&&Ms;if(!m&&n)return n(y,!0);if(o)return o(y,!0);throw(m=new Error("Cannot find module '"+y+"'")).code="MODULE_NOT_FOUND",m}n=g[y]={exports:{}},w[y][0].call(n.exports,function(i){return u(w[y][1][i]||i)},n,n.exports,t,w,g,s)}return g[y].exports}for(var o=typeof Ms=="function"&&Ms,h=0;h>16&255,l[b++]=d>>8&255,l[b++]=255&d;return c===2&&(d=u[i.charCodeAt(f)]<<2|u[i.charCodeAt(f+1)]>>4,l[b++]=255&d),c===1&&(d=u[i.charCodeAt(f)]<<10|u[i.charCodeAt(f+1)]<<4|u[i.charCodeAt(f+2)]>>2,l[b++]=d>>8&255,l[b++]=255&d),l},g.fromByteArray=function(i){for(var d,f=i.length,p=f%3,c=[],l=0,b=f-p;l>18&63]+s[L>>12&63]+s[L>>6&63]+s[63&L]}(E));return M.join("")}(i,l,b>2]+s[d<<4&63]+"==")):p==2&&(d=(i[f-2]<<8)+i[f-1],c.push(s[d>>10]+s[d>>4&63]+s[d<<2&63]+"=")),c.join("")};for(var s=[],u=[],o=typeof Uint8Array<"u"?Uint8Array:Array,h="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",y=0,m=h.length;y>>1;case"base64":return V(k).length;default:if(K)return X?-1:W(k).length;D=(""+D).toLowerCase(),K=!0}}function b(k,D,G){var X,K=!1;if((D=D===void 0||D<0?0:D)>this.length||(G=G===void 0||G>this.length?this.length:G)<=0||(G>>>=0)<=(D>>>=0))return"";for(k=k||"utf8";;)switch(k){case"hex":var J=this,de=D,ne=G,Y=J.length;(!ne||ne<0||Y=k.length){if(K)return-1;G=k.length-1}else if(G<0){if(!K)return-1;G=0}if(typeof D=="string"&&(D=n.from(D,X)),n.isBuffer(D))return D.length===0?-1:T(k,D,G,X,K);if(typeof D=="number")return D&=255,typeof Uint8Array.prototype.indexOf=="function"?(K?Uint8Array.prototype.indexOf:Uint8Array.prototype.lastIndexOf).call(k,D,G):T(k,[D],G,X,K);throw new TypeError("val must be string, number or Buffer")}function T(k,D,G,X,K){var J=1,ne=k.length,ue=D.length;if(X!==void 0&&((X=String(X).toLowerCase())==="ucs2"||X==="ucs-2"||X==="utf16le"||X==="utf-16le")){if(k.length<2||D.length<2)return-1;ne/=J=2,ue/=2,G/=2}function ie(ce,fe){return J===1?ce[fe]:ce.readUInt16BE(fe*J)}if(K)for(var he=-1,Y=G;Y>8,ue=ue%256,ie.push(ue),ie.push(ne);return ie}(D,k.length-G),k,G,X)}function P(k,D,G){G=Math.min(k.length,G);for(var X=[],K=D;K>>10&1023|55296),Y=56320|1023&Y),X.push(Y),K+=z}var te=X,ce=te.length;if(ce<=L)return String.fromCharCode.apply(String,te);for(var fe="",de=0;de"u"||typeof console.error!="function"||console.error("This browser lacks typed array (Uint8Array) support which is required by `buffer` v5.x. Use `buffer` v4.x if you require old browser support."),Object.defineProperty(n.prototype,"parent",{enumerable:!0,get:function(){if(n.isBuffer(this))return this.buffer}}),Object.defineProperty(n.prototype,"offset",{enumerable:!0,get:function(){if(n.isBuffer(this))return this.byteOffset}}),typeof Symbol<"u"&&Symbol.species!=null&&n[Symbol.species]===n&&Object.defineProperty(n,Symbol.species,{value:null,configurable:!0,enumerable:!1,writable:!1}),n.poolSize=8192,n.from=i,Object.setPrototypeOf(n.prototype,Uint8Array.prototype),Object.setPrototypeOf(n,Uint8Array),n.alloc=function(k,D,G){return D=D,G=G,d(k=k),!(k<=0)&&D!==void 0?typeof G=="string"?m(k).fill(D,G):m(k).fill(D):m(k)},n.allocUnsafe=f,n.allocUnsafeSlow=f,n.isBuffer=function(k){return k!=null&&k._isBuffer===!0&&k!==n.prototype},n.compare=function(k,D){if(Q(k,Uint8Array)&&(k=n.from(k,k.offset,k.byteLength)),Q(D,Uint8Array)&&(D=n.from(D,D.offset,D.byteLength)),!n.isBuffer(k)||!n.isBuffer(D))throw new TypeError('The "buf1", "buf2" arguments must be one of type Buffer or Uint8Array');if(k===D)return 0;for(var G=k.length,X=D.length,K=0,J=Math.min(G,X);Kk&&(D+=" ... "),""},h&&(n.prototype[h]=n.prototype.inspect),n.prototype.compare=function(k,D,G,X,K){if(Q(k,Uint8Array)&&(k=n.from(k,k.offset,k.byteLength)),!n.isBuffer(k))throw new TypeError('The "target" argument must be one of type Buffer or Uint8Array. Received type '+typeof k);if(G===void 0&&(G=k?k.length:0),X===void 0&&(X=0),K===void 0&&(K=this.length),(D=D===void 0?0:D)<0||G>k.length||X<0||K>this.length)throw new RangeError("out of range index");if(K<=X&&G<=D)return 0;if(K<=X)return-1;if(G<=D)return 1;if(this===k)return 0;for(var J=(K>>>=0)-(X>>>=0),ne=(G>>>=0)-(D>>>=0),ue=Math.min(J,ne),ie=this.slice(X,K),he=k.slice(D,G),Y=0;Y>>=0,isFinite(G)?(G>>>=0,X===void 0&&(X="utf8")):(X=G,G=void 0)}var K=this.length-D;if((G===void 0||Kthis.length)throw new RangeError("Attempt to write outside buffer bounds");X=X||"utf8";for(var J,ne,ue,ie=!1;;)switch(X){case"hex":var he=this,Y=k,z=D,te=G,ce=(z=Number(z)||0,he.length-z);(!te||ce<(te=Number(te)))&&(te=ce),(ce=Y.length)/2k.length)throw new RangeError("Index out of range")}function A(k,D,G,X){if(G+X>k.length)throw new RangeError("Index out of range");if(G<0)throw new RangeError("Index out of range")}function N(k,D,G,X,K){return D=+D,G>>>=0,K||A(k,0,G,4),o.write(k,D,G,X,23,4),G+4}function F(k,D,G,X,K){return D=+D,G>>>=0,K||A(k,0,G,8),o.write(k,D,G,X,52,8),G+8}n.prototype.slice=function(k,D){var G=this.length,G=((k=~~k)<0?(k+=G)<0&&(k=0):G>>=0,D>>>=0,G||C(k,D,this.length);for(var X=this[k],K=1,J=0;++J>>=0,D>>>=0,G||C(k,D,this.length);for(var X=this[k+--D],K=1;0>>=0,D||C(k,1,this.length),this[k]},n.prototype.readUInt16LE=function(k,D){return k>>>=0,D||C(k,2,this.length),this[k]|this[k+1]<<8},n.prototype.readUInt16BE=function(k,D){return k>>>=0,D||C(k,2,this.length),this[k]<<8|this[k+1]},n.prototype.readUInt32LE=function(k,D){return k>>>=0,D||C(k,4,this.length),(this[k]|this[k+1]<<8|this[k+2]<<16)+16777216*this[k+3]},n.prototype.readUInt32BE=function(k,D){return k>>>=0,D||C(k,4,this.length),16777216*this[k]+(this[k+1]<<16|this[k+2]<<8|this[k+3])},n.prototype.readIntLE=function(k,D,G){k>>>=0,D>>>=0,G||C(k,D,this.length);for(var X=this[k],K=1,J=0;++J>>=0,D>>>=0,G||C(k,D,this.length);for(var X=D,K=1,J=this[k+--X];0>>=0,D||C(k,1,this.length),128&this[k]?-1*(255-this[k]+1):this[k]},n.prototype.readInt16LE=function(k,D){return k>>>=0,D||C(k,2,this.length),D=this[k]|this[k+1]<<8,32768&D?4294901760|D:D},n.prototype.readInt16BE=function(k,D){return k>>>=0,D||C(k,2,this.length),D=this[k+1]|this[k]<<8,32768&D?4294901760|D:D},n.prototype.readInt32LE=function(k,D){return k>>>=0,D||C(k,4,this.length),this[k]|this[k+1]<<8|this[k+2]<<16|this[k+3]<<24},n.prototype.readInt32BE=function(k,D){return k>>>=0,D||C(k,4,this.length),this[k]<<24|this[k+1]<<16|this[k+2]<<8|this[k+3]},n.prototype.readFloatLE=function(k,D){return k>>>=0,D||C(k,4,this.length),o.read(this,k,!0,23,4)},n.prototype.readFloatBE=function(k,D){return k>>>=0,D||C(k,4,this.length),o.read(this,k,!1,23,4)},n.prototype.readDoubleLE=function(k,D){return k>>>=0,D||C(k,8,this.length),o.read(this,k,!0,52,8)},n.prototype.readDoubleBE=function(k,D){return k>>>=0,D||C(k,8,this.length),o.read(this,k,!1,52,8)},n.prototype.writeUIntLE=function(k,D,G,X){k=+k,D>>>=0,G>>>=0,X||I(this,k,D,G,Math.pow(2,8*G)-1,0);var K=1,J=0;for(this[D]=255&k;++J>>=0,G>>>=0,X||I(this,k,D,G,Math.pow(2,8*G)-1,0);var K=G-1,J=1;for(this[D+K]=255&k;0<=--K&&(J*=256);)this[D+K]=k/J&255;return D+G},n.prototype.writeUInt8=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,1,255,0),this[D]=255&k,D+1},n.prototype.writeUInt16LE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,2,65535,0),this[D]=255&k,this[D+1]=k>>>8,D+2},n.prototype.writeUInt16BE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,2,65535,0),this[D]=k>>>8,this[D+1]=255&k,D+2},n.prototype.writeUInt32LE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,4,4294967295,0),this[D+3]=k>>>24,this[D+2]=k>>>16,this[D+1]=k>>>8,this[D]=255&k,D+4},n.prototype.writeUInt32BE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,4,4294967295,0),this[D]=k>>>24,this[D+1]=k>>>16,this[D+2]=k>>>8,this[D+3]=255&k,D+4},n.prototype.writeIntLE=function(k,D,G,X){k=+k,D>>>=0,X||I(this,k,D,G,(X=Math.pow(2,8*G-1))-1,-X);var K=0,J=1,ne=0;for(this[D]=255&k;++K>0)-ne&255;return D+G},n.prototype.writeIntBE=function(k,D,G,X){k=+k,D>>>=0,X||I(this,k,D,G,(X=Math.pow(2,8*G-1))-1,-X);var K=G-1,J=1,ne=0;for(this[D+K]=255&k;0<=--K&&(J*=256);)k<0&&ne===0&&this[D+K+1]!==0&&(ne=1),this[D+K]=(k/J>>0)-ne&255;return D+G},n.prototype.writeInt8=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,1,127,-128),this[D]=255&(k=k<0?255+k+1:k),D+1},n.prototype.writeInt16LE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,2,32767,-32768),this[D]=255&k,this[D+1]=k>>>8,D+2},n.prototype.writeInt16BE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,2,32767,-32768),this[D]=k>>>8,this[D+1]=255&k,D+2},n.prototype.writeInt32LE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,4,2147483647,-2147483648),this[D]=255&k,this[D+1]=k>>>8,this[D+2]=k>>>16,this[D+3]=k>>>24,D+4},n.prototype.writeInt32BE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,4,2147483647,-2147483648),this[D]=(k=k<0?4294967295+k+1:k)>>>24,this[D+1]=k>>>16,this[D+2]=k>>>8,this[D+3]=255&k,D+4},n.prototype.writeFloatLE=function(k,D,G){return N(this,k,D,!0,G)},n.prototype.writeFloatBE=function(k,D,G){return N(this,k,D,!1,G)},n.prototype.writeDoubleLE=function(k,D,G){return F(this,k,D,!0,G)},n.prototype.writeDoubleBE=function(k,D,G){return F(this,k,D,!1,G)},n.prototype.copy=function(k,D,G,X){if(!n.isBuffer(k))throw new TypeError("argument should be a Buffer");if(G=G||0,X||X===0||(X=this.length),D>=k.length&&(D=k.length),(X=0=this.length)throw new RangeError("Index out of range");if(X<0)throw new RangeError("sourceEnd out of bounds");X>this.length&&(X=this.length);var K=(X=k.length-D>>=0,G=G===void 0?this.length:G>>>0,typeof(k=k||0)=="number")for(J=D;J>6|192,63&G|128)}else if(G<65536){if((D-=3)<0)break;J.push(G>>12|224,G>>6&63|128,63&G|128)}else{if(!(G<1114112))throw new Error("Invalid code point");if((D-=4)<0)break;J.push(G>>18|240,G>>12&63|128,G>>6&63|128,63&G|128)}}return J}function V(k){return u.toByteArray(function(D){if((D=(D=D.split("=")[0]).trim().replace(U,"")).length<2)return"";for(;D.length%4!=0;)D+="=";return D}(k))}function H(k,D,G,X){for(var K=0;K=D.length||K>=k.length);++K)D[K+G]=k[K];return K}function Q(k,D){return k instanceof D||k!=null&&k.constructor!=null&&k.constructor.name!=null&&k.constructor.name===D.name}function ee(k){return k!=k}var ae=function(){for(var k="0123456789abcdef",D=new Array(256),G=0;G<16;++G)for(var X=16*G,K=0;K<16;++K)D[X+K]=k[G]+k[K];return D}()}).call(this,t("buffer").Buffer)},{"base64-js":1,buffer:4,ieee754:254}],5:[function(t,w,g){w.exports=function(s){if(typeof s!="function")throw TypeError(String(s)+" is not a function");return s}},{}],6:[function(t,w,g){var s=t("../internals/is-object");w.exports=function(u){if(s(u)||u===null)return u;throw TypeError("Can't set "+String(u)+" as a prototype")}},{"../internals/is-object":75}],7:[function(o,w,g){var s=o("../internals/well-known-symbol"),u=o("../internals/object-create"),o=o("../internals/object-define-property"),h=s("unscopables"),y=Array.prototype;y[h]==null&&o.f(y,h,{configurable:!0,value:u(null)}),w.exports=function(m){y[h][m]=!0}},{"../internals/object-create":91,"../internals/object-define-property":93,"../internals/well-known-symbol":150}],8:[function(t,w,g){var s=t("../internals/string-multibyte").charAt;w.exports=function(u,o,h){return o+(h?s(u,o).length:1)}},{"../internals/string-multibyte":125}],9:[function(t,w,g){w.exports=function(s,u,o){if(s instanceof u)return s;throw TypeError("Incorrect "+(o?o+" ":"")+"invocation")}},{}],10:[function(t,w,g){var s=t("../internals/is-object");w.exports=function(u){if(s(u))return u;throw TypeError(String(u)+" is not an object")}},{"../internals/is-object":75}],11:[function(t,w,g){w.exports=typeof ArrayBuffer<"u"&&typeof DataView<"u"},{}],12:[function(A,w,g){function s(F){return m(F)&&n(N,i(F))}var u,o=A("../internals/array-buffer-native"),h=A("../internals/descriptors"),y=A("../internals/global"),m=A("../internals/is-object"),n=A("../internals/has"),i=A("../internals/classof"),d=A("../internals/create-non-enumerable-property"),f=A("../internals/redefine"),p=A("../internals/object-define-property").f,c=A("../internals/object-get-prototype-of"),l=A("../internals/object-set-prototype-of"),L=A("../internals/well-known-symbol"),A=A("../internals/uid"),b=y.Int8Array,v=b&&b.prototype,j=y.Uint8ClampedArray,j=j&&j.prototype,T=b&&c(b),E=v&&c(v),M=Object.prototype,P=M.isPrototypeOf,L=L("toStringTag"),C=A("TYPED_ARRAY_TAG"),I=o&&!!l&&i(y.opera)!=="Opera",A=!1,N={Int8Array:1,Uint8Array:1,Uint8ClampedArray:1,Int16Array:2,Uint16Array:2,Int32Array:4,Uint32Array:4,Float32Array:4,Float64Array:8};for(u in N)y[u]||(I=!1);if((!I||typeof T!="function"||T===Function.prototype)&&(T=function(){throw TypeError("Incorrect invocation")},I))for(u in N)y[u]&&l(y[u],T);if((!I||!E||E===M)&&(E=T.prototype,I))for(u in N)y[u]&&l(y[u].prototype,E);if(I&&c(j)!==E&&l(j,E),h&&!n(E,L))for(u in A=!0,p(E,L,{get:function(){return m(this)?this[C]:void 0}}),N)y[u]&&d(y[u],C,u);w.exports={NATIVE_ARRAY_BUFFER_VIEWS:I,TYPED_ARRAY_TAG:A&&C,aTypedArray:function(F){if(s(F))return F;throw TypeError("Target is not a typed array")},aTypedArrayConstructor:function(F){if(l){if(P.call(T,F))return F}else for(var U in N)if(n(N,u)&&(U=y[U],U&&(F===U||P.call(U,F))))return F;throw TypeError("Target is not a typed array constructor")},exportTypedArrayMethod:function(F,U,W){if(h){if(W)for(var V in N)V=y[V],V&&n(V.prototype,F)&&delete V.prototype[F];E[F]&&!W||f(E,F,!W&&I&&v[F]||U)}},exportTypedArrayStaticMethod:function(F,U,W){var V,H;if(h){if(l){if(W)for(V in N)(H=y[V])&&n(H,F)&&delete H[F];if(T[F]&&!W)return;try{return f(T,F,!W&&I&&b[F]||U)}catch{}}for(V in N)!(H=y[V])||H[F]&&!W||f(H,F,U)}},isView:function(F){return F=i(F),F==="DataView"||n(N,F)},isTypedArray:s,TypedArray:T,TypedArrayPrototype:E}},{"../internals/array-buffer-native":11,"../internals/classof":29,"../internals/create-non-enumerable-property":38,"../internals/descriptors":43,"../internals/global":60,"../internals/has":61,"../internals/is-object":75,"../internals/object-define-property":93,"../internals/object-get-prototype-of":98,"../internals/object-set-prototype-of":102,"../internals/redefine":110,"../internals/uid":147,"../internals/well-known-symbol":150}],13:[function(G,w,g){function s(z){return[255&z]}function u(z){return[255&z,z>>8&255]}function o(z){return[255&z,z>>8&255,z>>16&255,z>>24&255]}function h(z){return z[3]<<24|z[2]<<16|z[1]<<8|z[0]}function y(z){return J(z,23,4)}function m(z){return J(z,52,8)}function n(z,te){A(z[Q],te,{get:function(){return U(this)[te]}})}function i(ve,te,ye,fe){if(ye=M(ye),ve=U(ve),ye+te>ve.byteLength)throw K(ee);var de=U(ve.buffer).bytes,ye=ye+ve.byteOffset,ve=de.slice(ye,ye+te);return fe?ve:ve.reverse()}function d(z,te,ce,fe,de,ye){if(ce=M(ce),z=U(z),ce+te>z.byteLength)throw K(ee);for(var ve=U(z.buffer).bytes,Ee=ce+z.byteOffset,ge=fe(+de),Ne=0;Nehe;)(ue=ie[he++])in k||l(k,ue,ae[ue]);f.constructor=k}C&&L(G)!==X&&C(G,X);var P=new D(new k(2)),Y=G.setInt8;P.setInt8(0,2147483648),P.setInt8(1,2147483649),!P.getInt8(0)&&P.getInt8(1)||b(G,{setInt8:function(te,ce){Y.call(this,te,ce<<24>>24)},setUint8:function(te,ce){Y.call(this,te,ce<<24>>24)}},{unsafe:!0})}else k=function(z){j(this,k,V),z=M(z),W(this,{bytes:N.call(new Array(z),0),byteLength:z}),p||(this.byteLength=z)},D=function(z,de,ce){j(this,D,H),j(z,k,H);var fe=U(z).byteLength,de=T(de);if(de<0||fe>24},getUint8:function(z){return i(this,1,z)[0]},getInt16:function(z){return z=i(this,2,z,1>16},getUint16:function(z){return z=i(this,2,z,1>>0},getFloat32:function(z){return ne(i(this,4,z,1"+n+""}},{"../internals/require-object-coercible":115}],37:[function(t,w,g){function s(){return this}var u=t("../internals/iterators-core").IteratorPrototype,o=t("../internals/object-create"),h=t("../internals/create-property-descriptor"),y=t("../internals/set-to-string-tag"),m=t("../internals/iterators");w.exports=function(n,i,d){return i+=" Iterator",n.prototype=o(u,{next:h(1,d)}),y(n,i,!1,!0),m[i]=s,n}},{"../internals/create-property-descriptor":39,"../internals/iterators":80,"../internals/iterators-core":79,"../internals/object-create":91,"../internals/set-to-string-tag":119}],38:[function(t,w,g){var s=t("../internals/descriptors"),u=t("../internals/object-define-property"),o=t("../internals/create-property-descriptor");w.exports=s?function(h,y,m){return u.f(h,y,o(1,m))}:function(h,y,m){return h[y]=m,h}},{"../internals/create-property-descriptor":39,"../internals/descriptors":43,"../internals/object-define-property":93}],39:[function(t,w,g){w.exports=function(s,u){return{enumerable:!(1&s),configurable:!(2&s),writable:!(4&s),value:u}}},{}],40:[function(t,w,g){var s=t("../internals/to-primitive"),u=t("../internals/object-define-property"),o=t("../internals/create-property-descriptor");w.exports=function(h,y,m){y=s(y),y in h?u.f(h,y,o(0,m)):h[y]=m}},{"../internals/create-property-descriptor":39,"../internals/object-define-property":93,"../internals/to-primitive":142}],41:[function(c,w,g){function s(){return this}var u=c("../internals/export"),o=c("../internals/create-iterator-constructor"),h=c("../internals/object-get-prototype-of"),y=c("../internals/object-set-prototype-of"),m=c("../internals/set-to-string-tag"),n=c("../internals/create-non-enumerable-property"),i=c("../internals/redefine"),d=c("../internals/well-known-symbol"),f=c("../internals/is-pure"),p=c("../internals/iterators"),c=c("../internals/iterators-core"),l=c.IteratorPrototype,b=c.BUGGY_SAFARI_ITERATORS,v=d("iterator"),j="values",T="entries";w.exports=function(E,M,P,W,C,I,A){o(P,M,W);function N(k){if(k===C&&ee)return ee;if(!b&&k in H)return H[k];switch(k){case"keys":case j:case T:return function(){return new P(this,k)}}return function(){return new P(this)}}var F,U,W=M+" Iterator",V=!1,H=E.prototype,Q=H[v]||H["@@iterator"]||C&&H[C],ee=!b&&Q||N(C),ae=M=="Array"&&H.entries||Q;if(ae&&(ae=h(ae.call(new E)),l!==Object.prototype&&ae.next&&(f||h(ae)===l||(y?y(ae,l):typeof ae[v]!="function"&&n(ae,v,s)),m(ae,W,!0,!0),f&&(p[W]=s))),C==j&&Q&&Q.name!==j&&(V=!0,ee=function(){return Q.call(this)}),f&&!A||H[v]===ee||n(H,v,ee),p[M]=ee,C)if(F={values:N(j),keys:I?ee:N("keys"),entries:N(T)},A)for(U in F)!b&&!V&&U in H||i(H,U,F[U]);else u({target:M,proto:!0,forced:b||V},F);return F}},{"../internals/create-iterator-constructor":37,"../internals/create-non-enumerable-property":38,"../internals/export":50,"../internals/is-pure":76,"../internals/iterators":80,"../internals/iterators-core":79,"../internals/object-get-prototype-of":98,"../internals/object-set-prototype-of":102,"../internals/redefine":110,"../internals/set-to-string-tag":119,"../internals/well-known-symbol":150}],42:[function(t,w,g){var s=t("../internals/path"),u=t("../internals/has"),o=t("../internals/well-known-symbol-wrapped"),h=t("../internals/object-define-property").f;w.exports=function(y){var m=s.Symbol||(s.Symbol={});u(m,y)||h(m,y,{value:o.f(y)})}},{"../internals/has":61,"../internals/object-define-property":93,"../internals/path":106,"../internals/well-known-symbol-wrapped":149}],43:[function(t,w,g){t=t("../internals/fails"),w.exports=!t(function(){return Object.defineProperty({},1,{get:function(){return 7}})[1]!=7})},{"../internals/fails":51}],44:[function(u,w,g){var s=u("../internals/global"),u=u("../internals/is-object"),o=s.document,h=u(o)&&u(o.createElement);w.exports=function(y){return h?o.createElement(y):{}}},{"../internals/global":60,"../internals/is-object":75}],45:[function(t,w,g){w.exports={CSSRuleList:0,CSSStyleDeclaration:0,CSSValueList:0,ClientRectList:0,DOMRectList:0,DOMStringList:0,DOMTokenList:1,DataTransferItemList:0,FileList:0,HTMLAllCollection:0,HTMLCollection:0,HTMLFormElement:0,HTMLSelectElement:0,MediaList:0,MimeTypeArray:0,NamedNodeMap:0,NodeList:1,PaintRequestList:0,Plugin:0,PluginArray:0,SVGLengthList:0,SVGNumberList:0,SVGPathSegList:0,SVGPointList:0,SVGStringList:0,SVGTransformList:0,SourceBufferList:0,StyleSheetList:0,TextTrackCueList:0,TextTrackList:0,TouchList:0}},{}],46:[function(t,w,g){t=t("../internals/engine-user-agent"),w.exports=/(iphone|ipod|ipad).*applewebkit/i.test(t)},{"../internals/engine-user-agent":47}],47:[function(t,w,g){t=t("../internals/get-built-in"),w.exports=t("navigator","userAgent")||""},{"../internals/get-built-in":57}],48:[function(o,w,g){var s,u,h=o("../internals/global"),o=o("../internals/engine-user-agent"),h=h.process,h=h&&h.versions,h=h&&h.v8;h?u=(s=h.split("."))[0]+s[1]:o&&(!(s=o.match(/Edge\/(\d+)/))||74<=s[1])&&(s=o.match(/Chrome\/(\d+)/))&&(u=s[1]),w.exports=u&&+u},{"../internals/engine-user-agent":47,"../internals/global":60}],49:[function(t,w,g){w.exports=["constructor","hasOwnProperty","isPrototypeOf","propertyIsEnumerable","toLocaleString","toString","valueOf"]},{}],50:[function(t,w,g){var s=t("../internals/global"),u=t("../internals/object-get-own-property-descriptor").f,o=t("../internals/create-non-enumerable-property"),h=t("../internals/redefine"),y=t("../internals/set-global"),m=t("../internals/copy-constructor-properties"),n=t("../internals/is-forced");w.exports=function(i,d){var f,p,c,l=i.target,b=i.global,v=i.stat,j=b?s:v?s[l]||y(l,{}):(s[l]||{}).prototype;if(j)for(f in d){if(p=d[f],c=i.noTargetGet?(c=u(j,f))&&c.value:j[f],!n(b?f:l+(v?".":"#")+f,i.forced)&&c!==void 0){if(typeof p==typeof c)continue;m(p,c)}(i.sham||c&&c.sham)&&o(p,"sham",!0),h(j,f,p,i)}}},{"../internals/copy-constructor-properties":33,"../internals/create-non-enumerable-property":38,"../internals/global":60,"../internals/is-forced":74,"../internals/object-get-own-property-descriptor":94,"../internals/redefine":110,"../internals/set-global":117}],51:[function(t,w,g){w.exports=function(s){try{return!!s()}catch{return!0}}},{}],52:[function(d,w,g){d("../modules/es.regexp.exec");var s=d("../internals/redefine"),u=d("../internals/fails"),o=d("../internals/well-known-symbol"),h=d("../internals/regexp-exec"),y=d("../internals/create-non-enumerable-property"),m=o("species"),n=!u(function(){var c=/./;return c.exec=function(){var l=[];return l.groups={a:"7"},l},"".replace(c,"$")!=="7"}),i="a".replace(/./,"$0")==="$0",d=o("replace"),f=!!/./[d]&&/./[d]("a","$0")==="",p=!u(function(){var l=/(?:)/,c=l.exec,l=(l.exec=function(){return c.apply(this,arguments)},"ab".split(l));return l.length!==2||l[0]!=="a"||l[1]!=="b"});w.exports=function(c,l,b,v){var j,T,E=o(c),M=!u(function(){var L={};return L[E]=function(){return 7},""[c](L)!=7}),P=M&&!u(function(){var L=!1,C=/a/;return c==="split"&&((C={constructor:{}}).constructor[m]=function(){return C},C.flags="",C[E]=/./[E]),C.exec=function(){return L=!0,null},C[E](""),!L});M&&P&&(c!=="replace"||n&&i&&!f)&&(c!=="split"||p)||(j=/./[E],b=(P=b(E,""[c],function(L,C,I,A,N){return C.exec===h?M&&!N?{done:!0,value:j.call(C,I,A)}:{done:!0,value:L.call(I,C,A)}:{done:!1}},{REPLACE_KEEPS_$0:i,REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE:f}))[0],T=P[1],s(String.prototype,c,b),s(RegExp.prototype,E,l==2?function(L,C){return T.call(L,this,C)}:function(L){return T.call(L,this)})),v&&y(RegExp.prototype[E],"sham",!0)}},{"../internals/create-non-enumerable-property":38,"../internals/fails":51,"../internals/redefine":110,"../internals/regexp-exec":112,"../internals/well-known-symbol":150,"../modules/es.regexp.exec":195}],53:[function(t,w,g){function s(y,m,n,i,d,f,p,c){for(var l,b=d,v=0,j=!!p&&h(p,c,3);v>1,j=n===23?u(2,-24)-u(2,-77):0,T=m<0||m===0&&1/m<0?1:0,E=0;for((m=s(m))!=m||m===1/0?(f=m!=m?1:0,d=b):(d=o(h(m)/y),m*(p=u(2,-d))<1&&(d--,p*=2),2<=(m+=1<=d+v?j/p:j*u(2,1-v))*p&&(d++,p/=2),b<=d+v?(f=0,d=b):1<=d+v?(f=(m*p-1)*u(2,n),d+=v):(f=m*u(2,v-1)*u(2,n),d=0));8<=n;c[E++]=255&f,f/=256,n-=8);for(d=d<>1,c=b-7,l=d-1,b=m[l--],v=127&b;for(b>>=7;0>=-c,c+=n;0"+v+""},b=function(){try{u=document.domain&&new ActiveXObject("htmlfile")}catch{}b=u?((v=u).write(l("")),v.close(),j=v.parentWindow.Object,v=null,j):(v=i("iframe"),j="java"+p+":",v.style.display="none",n.appendChild(v),v.src=String(j),(j=v.contentWindow.document).open(),j.write(l("document.F=Object")),j.close(),j.F);for(var v,j,T=y.length;T--;)delete b[f][y[T]];return b()};m[c]=!0,w.exports=Object.create||function(v,j){var T;return v!==null?(s[f]=o(v),T=new s,s[f]=null,T[c]=v):T=b(),j===void 0?T:h(T,j)}},{"../internals/an-object":10,"../internals/document-create-element":44,"../internals/enum-bug-keys":49,"../internals/hidden-keys":62,"../internals/html":64,"../internals/object-define-properties":92,"../internals/shared-key":120}],92:[function(t,w,g){var s=t("../internals/descriptors"),u=t("../internals/object-define-property"),o=t("../internals/an-object"),h=t("../internals/object-keys");w.exports=s?Object.defineProperties:function(y,m){o(y);for(var n,i=h(m),d=i.length,f=0;fd;)!s(i,n=m[d++])||~o(f,n)||f.push(n);return f}},{"../internals/array-includes":18,"../internals/has":61,"../internals/hidden-keys":62,"../internals/to-indexed-object":136}],100:[function(t,w,g){var s=t("../internals/object-keys-internal"),u=t("../internals/enum-bug-keys");w.exports=Object.keys||function(o){return s(o,u)}},{"../internals/enum-bug-keys":49,"../internals/object-keys-internal":99}],101:[function(t,w,g){var s={}.propertyIsEnumerable,u=Object.getOwnPropertyDescriptor,o=u&&!s.call({1:2},1);g.f=o?function(h){return h=u(this,h),!!h&&h.enumerable}:s},{}],102:[function(t,w,g){var s=t("../internals/an-object"),u=t("../internals/a-possible-prototype");w.exports=Object.setPrototypeOf||("__proto__"in{}?function(){var o,h=!1,y={};try{(o=Object.getOwnPropertyDescriptor(Object.prototype,"__proto__").set).call(y,[]),h=y instanceof Array}catch{}return function(m,n){return s(m),u(n),h?o.call(m,n):m.__proto__=n,m}}():void 0)},{"../internals/a-possible-prototype":6,"../internals/an-object":10}],103:[function(t,w,g){function s(m){return function(n){for(var i,d=h(n),f=o(d),p=f.length,c=0,l=[];c{},Wr=Array.isArray,qf=/#/g,Qv=/&/g,Jv=/\//g,e0=/=/g,t0=/\?/g,Xf=/\+/g,r0=/%5B/g,n0=/%5D/g,$f=/%5E/g,o0=/%60/g,Yf=/%7B/g,s0=/%7C/g,Kf=/%7D/g,i0=/%20/g;function fl(r){return encodeURI(""+r).replace(s0,"|").replace(r0,"[").replace(n0,"]")}function a0(r){return fl(r).replace(Yf,"{").replace(Kf,"}").replace($f,"^")}function Aa(r){return fl(r).replace(Xf,"%2B").replace(i0,"+").replace(qf,"%23").replace(Qv,"%26").replace(o0,"`").replace(Yf,"{").replace(Kf,"}").replace($f,"^")}function l0(r){return Aa(r).replace(e0,"%3D")}function u0(r){return fl(r).replace(qf,"%23").replace(t0,"%3F")}function c0(r){return r==null?"":u0(r).replace(Jv,"%2F")}function ts(r){try{return decodeURIComponent(""+r)}catch{}return""+r}const d0=/\/$/,f0=r=>r.replace(d0,"");function Yi(r,a,x="/"){let t,w={},g="",s="";const u=a.indexOf("#");let o=a.indexOf("?");return u=0&&(o=-1),o>-1&&(t=a.slice(0,o),g=a.slice(o+1,u>-1?u:a.length),w=r(g)),u>-1&&(t=t||a.slice(0,u),s=a.slice(u,a.length)),t=y0(t??a,x),{fullPath:t+(g&&"?")+g+s,path:t,query:w,hash:ts(s)}}function h0(r,a){const x=a.query?r(a.query):"";return a.path+(x&&"?")+x+(a.hash||"")}function ac(r,a){return!a||!r.toLowerCase().startsWith(a.toLowerCase())?r:r.slice(a.length)||"/"}function p0(r,a,x){const t=a.matched.length-1,w=x.matched.length-1;return t>-1&&t===w&&_o(a.matched[t],x.matched[w])&&Zf(a.params,x.params)&&r(a.query)===r(x.query)&&a.hash===x.hash}function _o(r,a){return(r.aliasOf||r)===(a.aliasOf||a)}function Zf(r,a){if(Object.keys(r).length!==Object.keys(a).length)return!1;for(const x in r)if(!m0(r[x],a[x]))return!1;return!0}function m0(r,a){return Wr(r)?lc(r,a):Wr(a)?lc(a,r):r===a}function lc(r,a){return Wr(a)?r.length===a.length&&r.every((x,t)=>x===a[t]):r.length===1&&r[0]===a}function y0(r,a){if(r.startsWith("/"))return r;if(!r)return a;const x=a.split("/"),t=r.split("/"),w=t[t.length-1];(w===".."||w===".")&&t.push("");let g=x.length-1,s,u;for(s=0;s1&&g--;else break;return x.slice(0,g).join("/")+"/"+t.slice(s).join("/")}const Br={path:"/",name:void 0,params:{},query:{},hash:"",fullPath:"/",matched:[],meta:{},redirectedFrom:void 0};var rs;(function(r){r.pop="pop",r.push="push"})(rs||(rs={}));var Xo;(function(r){r.back="back",r.forward="forward",r.unknown=""})(Xo||(Xo={}));function g0(r){if(!r)if(io){const a=document.querySelector("base");r=a&&a.getAttribute("href")||"/",r=r.replace(/^\w+:\/\/[^\/]+/,"")}else r="/";return r[0]!=="/"&&r[0]!=="#"&&(r="/"+r),f0(r)}const v0=/^[^#]+#/;function b0(r,a){return r.replace(v0,"#")+a}function _0(r,a){const x=document.documentElement.getBoundingClientRect(),t=r.getBoundingClientRect();return{behavior:a.behavior,left:t.left-x.left-(a.left||0),top:t.top-x.top-(a.top||0)}}const yi=()=>({left:window.scrollX,top:window.scrollY});function x0(r){let a;if("el"in r){const x=r.el,t=typeof x=="string"&&x.startsWith("#"),w=typeof x=="string"?t?document.getElementById(x.slice(1)):document.querySelector(x):x;if(!w)return;a=_0(w,r)}else a=r;"scrollBehavior"in document.documentElement.style?window.scrollTo(a):window.scrollTo(a.left!=null?a.left:window.scrollX,a.top!=null?a.top:window.scrollY)}function uc(r,a){return(history.state?history.state.position-a:-1)+r}const La=new Map;function w0(r,a){La.set(r,a)}function j0(r){const a=La.get(r);return La.delete(r),a}let S0=()=>location.protocol+"//"+location.host;function Qf(r,a){const{pathname:x,search:t,hash:w}=a,g=r.indexOf("#");if(g>-1){let u=w.includes(r.slice(g))?r.slice(g).length:1,o=w.slice(u);return o[0]!=="/"&&(o="/"+o),ac(o,"")}return ac(x,r)+t+w}function E0(r,a,x,t){let w=[],g=[],s=null;const u=({state:n})=>{const i=Qf(r,location),d=x.value,f=a.value;let p=0;if(n){if(x.value=i,a.value=n,s&&s===d){s=null;return}p=f?n.position-f.position:0}else t(i);w.forEach(c=>{c(x.value,d,{delta:p,type:rs.pop,direction:p?p>0?Xo.forward:Xo.back:Xo.unknown})})};function o(){s=x.value}function h(n){w.push(n);const i=()=>{const d=w.indexOf(n);d>-1&&w.splice(d,1)};return g.push(i),i}function y(){const{history:n}=window;n.state&&n.replaceState(jt({},n.state,{scroll:yi()}),"")}function m(){for(const n of g)n();g=[],window.removeEventListener("popstate",u),window.removeEventListener("beforeunload",y)}return window.addEventListener("popstate",u),window.addEventListener("beforeunload",y,{passive:!0}),{pauseListeners:o,listen:h,destroy:m}}function cc(r,a,x,t=!1,w=!1){return{back:r,current:a,forward:x,replaced:t,position:window.history.length,scroll:w?yi():null}}function T0(r){const{history:a,location:x}=window,t={value:Qf(r,x)},w={value:a.state};w.value||g(t.value,{back:null,current:t.value,forward:null,position:a.length-1,replaced:!0,scroll:null},!0);function g(o,h,y){const m=r.indexOf("#"),n=m>-1?(x.host&&document.querySelector("base")?r:r.slice(m))+o:S0()+r+o;try{a[y?"replaceState":"pushState"](h,"",n),w.value=h}catch(i){console.error(i),x[y?"replace":"assign"](n)}}function s(o,h){const y=jt({},a.state,cc(w.value.back,o,w.value.forward,!0),h,{position:w.value.position});g(o,y,!0),t.value=o}function u(o,h){const y=jt({},w.value,a.state,{forward:o,scroll:yi()});g(y.current,y,!0);const m=jt({},cc(t.value,o,null),{position:y.position+1},h);g(o,m,!1),t.value=o}return{location:t,state:w,push:u,replace:s}}function Jf(r){r=g0(r);const a=T0(r),x=E0(r,a.state,a.location,a.replace);function t(g,s=!0){s||x.pauseListeners(),history.go(g)}const w=jt({location:"",base:r,go:t,createHref:b0.bind(null,r)},a,x);return Object.defineProperty(w,"location",{enumerable:!0,get:()=>a.location.value}),Object.defineProperty(w,"state",{enumerable:!0,get:()=>a.state.value}),w}function k0(r){return r=location.host?r||location.pathname+location.search:"",r.includes("#")||(r+="#"),Jf(r)}function M0(r){return typeof r=="string"||r&&typeof r=="object"}function eh(r){return typeof r=="string"||typeof r=="symbol"}const th=Symbol("");var dc;(function(r){r[r.aborted=4]="aborted",r[r.cancelled=8]="cancelled",r[r.duplicated=16]="duplicated"})(dc||(dc={}));function xo(r,a){return jt(new Error,{type:r,[th]:!0},a)}function sn(r,a){return r instanceof Error&&th in r&&(a==null||!!(r.type&a))}const fc="[^/]+?",C0={sensitive:!1,strict:!1,start:!0,end:!0},O0=/[.+*?^${}()[\]/\\]/g;function P0(r,a){const x=jt({},C0,a),t=[];let w=x.start?"^":"";const g=[];for(const h of r){const y=h.length?[]:[90];x.strict&&!h.length&&(w+="/");for(let m=0;ma.length?a.length===1&&a[0]===80?1:-1:0}function rh(r,a){let x=0;const t=r.score,w=a.score;for(;x0&&a[a.length-1]<0}const L0={type:0,value:""},R0=/[a-zA-Z0-9_]/;function I0(r){if(!r)return[[]];if(r==="/")return[[L0]];if(!r.startsWith("/"))throw new Error(`Invalid path "${r}"`);function a(i){throw new Error(`ERR (${x})/"${h}": ${i}`)}let x=0,t=x;const w=[];let g;function s(){g&&w.push(g),g=[]}let u=0,o,h="",y="";function m(){h&&(x===0?g.push({type:0,value:h}):x===1||x===2||x===3?(g.length>1&&(o==="*"||o==="+")&&a(`A repeatable param (${h}) must be alone in its segment. eg: '/:ids+.`),g.push({type:1,value:h,regexp:y,repeatable:o==="*"||o==="+",optional:o==="*"||o==="?"})):a("Invalid state to consume buffer"),h="")}function n(){h+=o}for(;u{s(b)}:qo}function s(m){if(eh(m)){const n=t.get(m);n&&(t.delete(m),x.splice(x.indexOf(n),1),n.children.forEach(s),n.alias.forEach(s))}else{const n=x.indexOf(m);n>-1&&(x.splice(n,1),m.record.name&&t.delete(m.record.name),m.children.forEach(s),m.alias.forEach(s))}}function u(){return x}function o(m){const n=B0(m,x);x.splice(n,0,m),m.record.name&&!yc(m)&&t.set(m.record.name,m)}function h(m,n){let i,d={},f,p;if("name"in m&&m.name){if(i=t.get(m.name),!i)throw xo(1,{location:m});p=i.record.name,d=jt(pc(n.params,i.keys.filter(b=>!b.optional).concat(i.parent?i.parent.keys.filter(b=>b.optional):[]).map(b=>b.name)),m.params&&pc(m.params,i.keys.map(b=>b.name))),f=i.stringify(d)}else if(m.path!=null)f=m.path,i=x.find(b=>b.re.test(f)),i&&(d=i.parse(f),p=i.record.name);else{if(i=n.name?t.get(n.name):x.find(b=>b.re.test(n.path)),!i)throw xo(1,{location:m,currentLocation:n});p=i.record.name,d=jt({},n.params,m.params),f=i.stringify(d)}const c=[];let l=i;for(;l;)c.unshift(l.record),l=l.parent;return{name:p,path:f,params:d,matched:c,meta:U0(c)}}r.forEach(m=>g(m));function y(){x.length=0,t.clear()}return{addRoute:g,resolve:h,removeRoute:s,clearRoutes:y,getRoutes:u,getRecordMatcher:w}}function pc(r,a){const x={};for(const t of a)t in r&&(x[t]=r[t]);return x}function mc(r){const a={path:r.path,redirect:r.redirect,name:r.name,meta:r.meta||{},aliasOf:r.aliasOf,beforeEnter:r.beforeEnter,props:N0(r),children:r.children||[],instances:{},leaveGuards:new Set,updateGuards:new Set,enterCallbacks:{},components:"components"in r?r.components||null:r.component&&{default:r.component}};return Object.defineProperty(a,"mods",{value:{}}),a}function N0(r){const a={},x=r.props||!1;if("component"in r)a.default=x;else for(const t in r.components)a[t]=typeof x=="object"?x[t]:x;return a}function yc(r){for(;r;){if(r.record.aliasOf)return!0;r=r.parent}return!1}function U0(r){return r.reduce((a,x)=>jt(a,x.meta),{})}function gc(r,a){const x={};for(const t in r)x[t]=t in a?a[t]:r[t];return x}function B0(r,a){let x=0,t=a.length;for(;x!==t;){const g=x+t>>1;rh(r,a[g])<0?t=g:x=g+1}const w=G0(r);return w&&(t=a.lastIndexOf(w,t-1)),t}function G0(r){let a=r;for(;a=a.parent;)if(nh(a)&&rh(r,a)===0)return a}function nh({record:r}){return!!(r.name||r.components&&Object.keys(r.components).length||r.redirect)}function V0(r){const a={};if(r===""||r==="?")return a;const t=(r[0]==="?"?r.slice(1):r).split("&");for(let w=0;wg&&Aa(g)):[t&&Aa(t)]).forEach(g=>{g!==void 0&&(a+=(a.length?"&":"")+x,g!=null&&(a+="="+g))})}return a}function H0(r){const a={};for(const x in r){const t=r[x];t!==void 0&&(a[x]=Wr(t)?t.map(w=>w==null?null:""+w):t==null?t:""+t)}return a}const z0=Symbol(""),bc=Symbol(""),hl=Symbol(""),pl=Symbol(""),Ra=Symbol("");function Do(){let r=[];function a(t){return r.push(t),()=>{const w=r.indexOf(t);w>-1&&r.splice(w,1)}}function x(){r=[]}return{add:a,list:()=>r.slice(),reset:x}}function Sn(r,a,x,t,w,g=s=>s()){const s=t&&(t.enterCallbacks[w]=t.enterCallbacks[w]||[]);return()=>new Promise((u,o)=>{const h=n=>{n===!1?o(xo(4,{from:x,to:a})):n instanceof Error?o(n):M0(n)?o(xo(2,{from:a,to:n})):(s&&t.enterCallbacks[w]===s&&typeof n=="function"&&s.push(n),u())},y=g(()=>r.call(t&&t.instances[w],a,x,h));let m=Promise.resolve(y);r.length<3&&(m=m.then(h)),m.catch(n=>o(n))})}function Ki(r,a,x,t,w=g=>g()){const g=[];for(const s of r)for(const u in s.components){let o=s.components[u];if(!(a!=="beforeRouteEnter"&&!s.instances[u]))if(Wf(o)){const y=(o.__vccOpts||o)[a];y&&g.push(Sn(y,x,t,s,u,w))}else{let h=o();g.push(()=>h.then(y=>{if(!y)throw new Error(`Couldn't resolve component "${u}" at "${s.path}"`);const m=Zv(y)?y.default:y;s.mods[u]=y,s.components[u]=m;const i=(m.__vccOpts||m)[a];return i&&Sn(i,x,t,s,u,w)()}))}}return g}function _c(r){const a=zt(hl),x=zt(pl),t=vt(()=>{const o=Nt(r.to);return a.resolve(o)}),w=vt(()=>{const{matched:o}=t.value,{length:h}=o,y=o[h-1],m=x.matched;if(!y||!m.length)return-1;const n=m.findIndex(_o.bind(null,y));if(n>-1)return n;const i=xc(o[h-2]);return h>1&&xc(y)===i&&m[m.length-1].path!==i?m.findIndex(_o.bind(null,o[h-2])):n}),g=vt(()=>w.value>-1&&$0(x.params,t.value.params)),s=vt(()=>w.value>-1&&w.value===x.matched.length-1&&Zf(x.params,t.value.params));function u(o={}){return X0(o)?a[Nt(r.replace)?"replace":"push"](Nt(r.to)).catch(qo):Promise.resolve()}return{route:t,href:vt(()=>t.value.href),isActive:g,isExactActive:s,navigate:u}}const W0=gr({name:"RouterLink",compatConfig:{MODE:3},props:{to:{type:[String,Object],required:!0},replace:Boolean,activeClass:String,exactActiveClass:String,custom:Boolean,ariaCurrentValue:{type:String,default:"page"}},useLink:_c,setup(r,{slots:a}){const x=Rn(_c(r)),{options:t}=zt(hl),w=vt(()=>({[wc(r.activeClass,t.linkActiveClass,"router-link-active")]:x.isActive,[wc(r.exactActiveClass,t.linkExactActiveClass,"router-link-exact-active")]:x.isExactActive}));return()=>{const g=a.default&&a.default(x);return r.custom?g:Jt("a",{"aria-current":x.isExactActive?r.ariaCurrentValue:null,href:x.href,onClick:x.navigate,class:w.value},g)}}}),q0=W0;function X0(r){if(!(r.metaKey||r.altKey||r.ctrlKey||r.shiftKey)&&!r.defaultPrevented&&!(r.button!==void 0&&r.button!==0)){if(r.currentTarget&&r.currentTarget.getAttribute){const a=r.currentTarget.getAttribute("target");if(/\b_blank\b/i.test(a))return}return r.preventDefault&&r.preventDefault(),!0}}function $0(r,a){for(const x in a){const t=a[x],w=r[x];if(typeof t=="string"){if(t!==w)return!1}else if(!Wr(w)||w.length!==t.length||t.some((g,s)=>g!==w[s]))return!1}return!0}function xc(r){return r?r.aliasOf?r.aliasOf.path:r.path:""}const wc=(r,a,x)=>r??a??x,Y0=gr({name:"RouterView",inheritAttrs:!1,props:{name:{type:String,default:"default"},route:Object},compatConfig:{MODE:3},setup(r,{attrs:a,slots:x}){const t=zt(Ra),w=vt(()=>r.route||t.value),g=zt(bc,0),s=vt(()=>{let h=Nt(g);const{matched:y}=w.value;let m;for(;(m=y[h])&&!m.components;)h++;return h}),u=vt(()=>w.value.matched[s.value]);Ir(bc,vt(()=>s.value+1)),Ir(z0,u),Ir(Ra,w);const o=mt();return On(()=>[o.value,u.value,r.name],([h,y,m],[n,i,d])=>{y&&(y.instances[m]=h,i&&i!==y&&h&&h===n&&(y.leaveGuards.size||(y.leaveGuards=i.leaveGuards),y.updateGuards.size||(y.updateGuards=i.updateGuards))),h&&y&&(!i||!_o(y,i)||!n)&&(y.enterCallbacks[m]||[]).forEach(f=>f(h))},{flush:"post"}),()=>{const h=w.value,y=r.name,m=u.value,n=m&&m.components[y];if(!n)return jc(x.default,{Component:n,route:h});const i=m.props[y],d=i?i===!0?h.params:typeof i=="function"?i(h):i:null,p=Jt(n,jt({},d,a,{onVnodeUnmounted:c=>{c.component.isUnmounted&&(m.instances[y]=null)},ref:o}));return jc(x.default,{Component:p,route:h})||p}}});function jc(r,a){if(!r)return null;const x=r(a);return x.length===1?x[0]:x}const oh=Y0;function K0(r){const a=F0(r.routes,r),x=r.parseQuery||V0,t=r.stringifyQuery||vc,w=r.history,g=Do(),s=Do(),u=Do(),o=Zo(Br);let h=Br;io&&r.scrollBehavior&&"scrollRestoration"in history&&(history.scrollRestoration="manual");const y=$i.bind(null,K=>""+K),m=$i.bind(null,c0),n=$i.bind(null,ts);function i(K,J){let ne,ue;return eh(K)?(ne=a.getRecordMatcher(K),ue=J):ue=K,a.addRoute(ue,ne)}function d(K){const J=a.getRecordMatcher(K);J&&a.removeRoute(J)}function f(){return a.getRoutes().map(K=>K.record)}function p(K){return!!a.getRecordMatcher(K)}function c(K,J){if(J=jt({},J||o.value),typeof K=="string"){const z=Yi(x,K,J.path),te=a.resolve({path:z.path},J),ce=w.createHref(z.fullPath);return jt(z,te,{params:n(te.params),hash:ts(z.hash),redirectedFrom:void 0,href:ce})}let ne;if(K.path!=null)ne=jt({},K,{path:Yi(x,K.path,J.path).path});else{const z=jt({},K.params);for(const te in z)z[te]==null&&delete z[te];ne=jt({},K,{params:m(z)}),J.params=m(J.params)}const ue=a.resolve(ne,J),ie=K.hash||"";ue.params=y(n(ue.params));const he=h0(t,jt({},K,{hash:a0(ie),path:ue.path})),Y=w.createHref(he);return jt({fullPath:he,hash:ie,query:t===vc?H0(K.query):K.query||{}},ue,{redirectedFrom:void 0,href:Y})}function l(K){return typeof K=="string"?Yi(x,K,o.value.path):jt({},K)}function b(K,J){if(h!==K)return xo(8,{from:J,to:K})}function v(K){return E(K)}function j(K){return v(jt(l(K),{replace:!0}))}function T(K){const J=K.matched[K.matched.length-1];if(J&&J.redirect){const{redirect:ne}=J;let ue=typeof ne=="function"?ne(K):ne;return typeof ue=="string"&&(ue=ue.includes("?")||ue.includes("#")?ue=l(ue):{path:ue},ue.params={}),jt({query:K.query,hash:K.hash,params:ue.path!=null?{}:K.params},ue)}}function E(K,J){const ne=h=c(K),ue=o.value,ie=K.state,he=K.force,Y=K.replace===!0,z=T(ne);if(z)return E(jt(l(z),{state:typeof z=="object"?jt({},ie,z.state):ie,force:he,replace:Y}),J||ne);const te=ne;te.redirectedFrom=J;let ce;return!he&&p0(t,ue,ne)&&(ce=xo(16,{to:te,from:ue}),ee(ue,ue,!0,!1)),(ce?Promise.resolve(ce):L(te,ue)).catch(fe=>sn(fe)?sn(fe,2)?fe:Q(fe):V(fe,te,ue)).then(fe=>{if(fe){if(sn(fe,2))return E(jt({replace:Y},l(fe.to),{state:typeof fe.to=="object"?jt({},ie,fe.to.state):ie,force:he}),J||te)}else fe=I(te,ue,!0,Y,ie);return C(te,ue,fe),fe})}function M(K,J){const ne=b(K,J);return ne?Promise.reject(ne):Promise.resolve()}function P(K){const J=D.values().next().value;return J&&typeof J.runWithContext=="function"?J.runWithContext(K):K()}function L(K,J){let ne;const[ue,ie,he]=Z0(K,J);ne=Ki(ue.reverse(),"beforeRouteLeave",K,J);for(const z of ue)z.leaveGuards.forEach(te=>{ne.push(Sn(te,K,J))});const Y=M.bind(null,K,J);return ne.push(Y),X(ne).then(()=>{ne=[];for(const z of g.list())ne.push(Sn(z,K,J));return ne.push(Y),X(ne)}).then(()=>{ne=Ki(ie,"beforeRouteUpdate",K,J);for(const z of ie)z.updateGuards.forEach(te=>{ne.push(Sn(te,K,J))});return ne.push(Y),X(ne)}).then(()=>{ne=[];for(const z of he)if(z.beforeEnter)if(Wr(z.beforeEnter))for(const te of z.beforeEnter)ne.push(Sn(te,K,J));else ne.push(Sn(z.beforeEnter,K,J));return ne.push(Y),X(ne)}).then(()=>(K.matched.forEach(z=>z.enterCallbacks={}),ne=Ki(he,"beforeRouteEnter",K,J,P),ne.push(Y),X(ne))).then(()=>{ne=[];for(const z of s.list())ne.push(Sn(z,K,J));return ne.push(Y),X(ne)}).catch(z=>sn(z,8)?z:Promise.reject(z))}function C(K,J,ne){u.list().forEach(ue=>P(()=>ue(K,J,ne)))}function I(K,J,ne,ue,ie){const he=b(K,J);if(he)return he;const Y=J===Br,z=io?history.state:{};ne&&(ue||Y?w.replace(K.fullPath,jt({scroll:Y&&z&&z.scroll},ie)):w.push(K.fullPath,ie)),o.value=K,ee(K,J,ne,Y),Q()}let A;function N(){A||(A=w.listen((K,J,ne)=>{if(!G.listening)return;const ue=c(K),ie=T(ue);if(ie){E(jt(ie,{replace:!0}),ue).catch(qo);return}h=ue;const he=o.value;io&&w0(uc(he.fullPath,ne.delta),yi()),L(ue,he).catch(Y=>sn(Y,12)?Y:sn(Y,2)?(E(Y.to,ue).then(z=>{sn(z,20)&&!ne.delta&&ne.type===rs.pop&&w.go(-1,!1)}).catch(qo),Promise.reject()):(ne.delta&&w.go(-ne.delta,!1),V(Y,ue,he))).then(Y=>{Y=Y||I(ue,he,!1),Y&&(ne.delta&&!sn(Y,8)?w.go(-ne.delta,!1):ne.type===rs.pop&&sn(Y,20)&&w.go(-1,!1)),C(ue,he,Y)}).catch(qo)}))}let F=Do(),U=Do(),W;function V(K,J,ne){Q(K);const ue=U.list();return ue.length?ue.forEach(ie=>ie(K,J,ne)):console.error(K),Promise.reject(K)}function H(){return W&&o.value!==Br?Promise.resolve():new Promise((K,J)=>{F.add([K,J])})}function Q(K){return W||(W=!K,N(),F.list().forEach(([J,ne])=>K?ne(K):J()),F.reset()),K}function ee(K,J,ne,ue){const{scrollBehavior:ie}=r;if(!io||!ie)return Promise.resolve();const he=!ne&&j0(uc(K.fullPath,0))||(ue||!ne)&&history.state&&history.state.scroll||null;return Pr().then(()=>ie(K,J,he)).then(Y=>Y&&x0(Y)).catch(Y=>V(Y,K,J))}const ae=K=>w.go(K);let k;const D=new Set,G={currentRoute:o,listening:!0,addRoute:i,removeRoute:d,clearRoutes:a.clearRoutes,hasRoute:p,getRoutes:f,resolve:c,options:r,push:v,replace:j,go:ae,back:()=>ae(-1),forward:()=>ae(1),beforeEach:g.add,beforeResolve:s.add,afterEach:u.add,onError:U.add,isReady:H,install(K){const J=this;K.component("RouterLink",q0),K.component("RouterView",oh),K.config.globalProperties.$router=J,Object.defineProperty(K.config.globalProperties,"$route",{enumerable:!0,get:()=>Nt(o)}),io&&!k&&o.value===Br&&(k=!0,v(w.location).catch(ie=>{}));const ne={};for(const ie in Br)Object.defineProperty(ne,ie,{get:()=>o.value[ie],enumerable:!0});K.provide(hl,J),K.provide(pl,cn(ne)),K.provide(Ra,o);const ue=K.unmount;D.add(K),K.unmount=function(){D.delete(K),D.size<1&&(h=Br,A&&A(),A=null,o.value=Br,k=!1,W=!1),ue()}}};function X(K){return K.reduce((J,ne)=>J.then(()=>P(ne)),Promise.resolve())}return G}function Z0(r,a){const x=[],t=[],w=[],g=Math.max(a.matched.length,r.matched.length);for(let s=0;s_o(h,u))?t.push(u):x.push(u));const o=r.matched[s];o&&(a.matched.find(h=>_o(h,o))||w.push(o))}return[x,t,w]}function Q0(r){return zt(pl)}const J0=(r,a)=>a.path.replace(/(:\w+)\([^)]+\)/g,"$1").replace(/(:\w+)[?+*]/g,"$1").replace(/:\w+/g,x=>{var t;return((t=r.params[x.slice(1)])==null?void 0:t.toString())||""}),Ia=(r,a)=>{const x=r.route.matched.find(w=>{var g;return((g=w.components)==null?void 0:g.default)===r.Component.type}),t=a??(x==null?void 0:x.meta.key)??(x&&J0(r.route,x));return typeof t=="function"?t(r.route):t},eb=(r,a)=>({default:()=>r?Jt(Hp,r===!0?{}:r,a):a});function ml(r){return Array.isArray(r)?r:[r]}const tb="modulepreload",rb=function(r,a){return new URL(r,a).href},Sc={},rt=function(a,x,t){let w=Promise.resolve();if(x&&x.length>0){const s=document.getElementsByTagName("link"),u=document.querySelector("meta[property=csp-nonce]"),o=(u==null?void 0:u.nonce)||(u==null?void 0:u.getAttribute("nonce"));w=Promise.allSettled(x.map(h=>{if(h=rb(h,t),h in Sc)return;Sc[h]=!0;const y=h.endsWith(".css"),m=y?'[rel="stylesheet"]':"";if(!!t)for(let d=s.length-1;d>=0;d--){const f=s[d];if(f.href===h&&(!y||f.rel==="stylesheet"))return}else if(document.querySelector(`link[href="${h}"]${m}`))return;const i=document.createElement("link");if(i.rel=y?"stylesheet":tb,y||(i.as="script"),i.crossOrigin="",i.href=h,o&&i.setAttribute("nonce",o),document.head.appendChild(i),y)return new Promise((d,f)=>{i.addEventListener("load",d),i.addEventListener("error",()=>f(new Error(`Unable to preload CSS for ${h}`)))})}))}function g(s){const u=new Event("vite:preloadError",{cancelable:!0});if(u.payload=s,window.dispatchEvent(u),!u.defaultPrevented)throw s}return w.then(s=>{for(const u of s||[])u.status==="rejected"&&g(u.reason);return a().catch(g)})},nb={layout:"empty"},ob={layout:"light"},sb={layout:"light"},ib={layout:"light"},ab={layout:"light"},lb={layout:"light"},ub={layout:"light"},cb={layout:"light"},db={layout:"light"},fb={layout:"light"},hb={layout:"light"},pb={layout:"light"},Zi=[{name:"articles-slug",path:"/articles/:slug(.*)*",component:()=>rt(()=>import("./I6a6eUEM.js"),__vite__mapDeps([0,1,2,3,4,5,6,7,8,9]),import.meta.url)},{name:"articles",path:"/articles",component:()=>rt(()=>import("./BmBFiau8.js"),__vite__mapDeps([10,1,2,3,4,5,6,7,8]),import.meta.url)},{name:"card",path:"/card",meta:nb||{},component:()=>rt(()=>import("./BZWGQyMp.js"),[],import.meta.url)},{name:"examples-nested_transitions",path:"/examples/nested_transitions",component:()=>rt(()=>import("./D-lCVgpc.js"),[],import.meta.url)},{name:"index",path:"/",component:()=>rt(()=>import("./BqebzOVN.js"),__vite__mapDeps([11,12,8,5,4]),import.meta.url)},{name:"playground-audio",path:"/playground/audio",component:()=>rt(()=>import("./DAkPvNyh.js"),[],import.meta.url)},{name:"playground-chords",path:"/playground/chords",meta:ob||{},component:()=>rt(()=>import("./ZcTbvTx_.js"),[],import.meta.url)},{name:"playground-conway",path:"/playground/conway",meta:sb||{},component:()=>rt(()=>import("./C6rjZEvg.js"),[],import.meta.url)},{name:"playground-french",path:"/playground/french",meta:ib||{},component:()=>rt(()=>import("./CCH9cHpw.js"),__vite__mapDeps([13,5,7]),import.meta.url)},{name:"playground",path:"/playground",component:()=>rt(()=>import("./Dn7xXnlv.js"),__vite__mapDeps([14,12]),import.meta.url)},{name:"playground-matrix",path:"/playground/matrix",meta:ab||{},component:()=>rt(()=>import("./DjUAd0VA.js"),[],import.meta.url)},{name:"playground-metronome",path:"/playground/metronome",meta:lb||{},component:()=>rt(()=>import("./Bl_PaSYd.js"),[],import.meta.url)},{name:"playground-midi",path:"/playground/midi",meta:ub||{},component:()=>rt(()=>import("./Cai_qPmv.js"),[],import.meta.url)},{name:"playground-palettes-mountains",path:"/playground/palettes/mountains",meta:cb||{},component:()=>rt(()=>import("./R23NweeK.js"),[],import.meta.url)},{name:"playground-palettes-variance",path:"/playground/palettes/variance",meta:db||{},component:()=>rt(()=>import("./uL54ePkM.js"),[],import.meta.url)},{name:"playground-plotter",path:"/playground/plotter",meta:fb||{},component:()=>rt(()=>import("./CQnAi_qF.js"),[],import.meta.url)},{name:"playground-tiling",path:"/playground/tiling",meta:hb||{},component:()=>rt(()=>import("./DcUhz6y1.js"),[],import.meta.url)},{name:"playground-waves",path:"/playground/waves",meta:pb||{},component:()=>rt(()=>import("./DA5xXx2M.js"),[],import.meta.url)},{name:"talks",path:"/talks",component:()=>rt(()=>import("./xkKG0dwU.js"),[],import.meta.url)}],sh=(r,a,x)=>(a=a===!0?{}:a,{default:()=>{var t;return a?Jt(r,a,x):(t=x.default)==null?void 0:t.call(x)}});function Ec(r){const a=(r==null?void 0:r.meta.key)??r.path.replace(/(:\w+)\([^)]+\)/g,"$1").replace(/(:\w+)[?+*]/g,"$1").replace(/:\w+/g,x=>{var t;return((t=r.params[x.slice(1)])==null?void 0:t.toString())||""});return typeof a=="function"?a(r):a}function mb(r,a){return r===a||a===Br?!1:Ec(r)!==Ec(a)?!0:!r.matched.every((t,w)=>{var g,s;return t.components&&t.components.default===((s=(g=a.matched[w])==null?void 0:g.components)==null?void 0:s.default)})}const yb={scrollBehavior(r,a,x){var h;const t=Dt(),w=((h=jr().options)==null?void 0:h.scrollBehaviorType)??"auto";let g=x||void 0;const s=typeof r.meta.scrollToTop=="function"?r.meta.scrollToTop(r,a):r.meta.scrollToTop;if(!g&&a&&r&&s!==!1&&mb(r,a)&&(g={left:0,top:0}),r.path===a.path)return a.hash&&!r.hash?{left:0,top:0}:r.hash?{el:r.hash,top:Tc(r.hash),behavior:w}:!1;const u=y=>!!(y.meta.pageTransition??xa),o=u(a)&&u(r)?"page:transition:finish":"page:finish";return new Promise(y=>{t.hooks.hookOnce(o,async()=>{await new Promise(m=>setTimeout(m,0)),r.hash&&(g={el:r.hash,top:Tc(r.hash),behavior:w}),y(g)})})}};function Tc(r){try{const a=document.querySelector(r);if(a)return(Number.parseFloat(getComputedStyle(a).scrollMarginTop)||0)+(Number.parseFloat(getComputedStyle(document.documentElement).scrollPaddingTop)||0)}catch{}return 0}const gb={hashMode:!1,scrollBehaviorType:"auto"},Lr={...gb,...yb},vb=async r=>{var o;let a,x;if(!((o=r.meta)!=null&&o.validate))return;const t=Dt(),w=jr(),g=([a,x]=yo(()=>Promise.resolve(r.meta.validate(r))),a=await a,x(),a);if(g===!0)return;const s=pi({statusCode:g&&g.statusCode||404,statusMessage:g&&g.statusMessage||`Page Not Found: ${r.fullPath}`,data:{path:r.fullPath}}),u=w.beforeResolve(h=>{if(u(),h===r){const y=w.afterEach(async()=>{y(),await t.runWithContext(()=>lo(s)),window==null||window.history.pushState({},"",r.fullPath)});return!1}})},bb=async r=>{let a,x;const t=([a,x]=yo(()=>dl(r.path)),a=await a,x(),a);if(t.redirect)return pn(t.redirect,{acceptRelative:!0})?(window.location.href=t.redirect,!1):t.redirect},_b=[vb,bb],$o={};function xb(r,a,x){const{pathname:t,search:w,hash:g}=a,s=r.indexOf("#");if(s>-1){const h=g.includes(r.slice(s))?r.slice(s).length:1;let y=g.slice(h);return y[0]!=="/"&&(y="/"+y),zu(y,"")}const u=zu(t,r),o=!x||zy(u,x,{trailingSlash:!0})?u:x;return o+(o.includes("?")?"":w)+g}const wb=tn({name:"nuxt:router",enforce:"pre",async setup(r){var p;let a,x,t=us().app.baseURL;Lr.hashMode&&!t.includes("#")&&(t+="#");const w=((p=Lr.history)==null?void 0:p.call(Lr,t))??(Lr.hashMode?k0(t):Jf(t)),g=Lr.routes?([a,x]=yo(()=>Lr.routes(Zi)),a=await a,x(),a??Zi):Zi;let s;const u=K0({...Lr,scrollBehavior:(c,l,b)=>{if(l===Br){s=b;return}if(Lr.scrollBehavior){if(u.options.scrollBehavior=Lr.scrollBehavior,"scrollRestoration"in window.history){const v=u.beforeEach(()=>{v(),window.history.scrollRestoration="manual"})}return Lr.scrollBehavior(c,Br,s||b)}},history:w,routes:g});"scrollRestoration"in window.history&&(window.history.scrollRestoration="auto"),r.vueApp.use(u);const o=Zo(u.currentRoute.value);u.afterEach((c,l)=>{o.value=l}),Object.defineProperty(r.vueApp.config.globalProperties,"previousRoute",{get:()=>o.value});const h=xb(t,window.location,r.payload.path),y=Zo(u.currentRoute.value),m=()=>{y.value=u.currentRoute.value};r.hook("page:finish",m),u.afterEach((c,l)=>{var b,v,j,T;((v=(b=c.matched[0])==null?void 0:b.components)==null?void 0:v.default)===((T=(j=l.matched[0])==null?void 0:j.components)==null?void 0:T.default)&&m()});const n={};for(const c in y.value)Object.defineProperty(n,c,{get:()=>y.value[c],enumerable:!0});r._route=cn(n),r._middleware=r._middleware||{global:[],named:{}};const i=hi();u.afterEach(async(c,l,b)=>{delete r._processingMiddleware,!r.isHydrating&&i.value&&await r.runWithContext(Hg),b&&await r.callHook("page:loading:end"),c.matched.length===0&&await r.runWithContext(()=>lo(Sa({statusCode:404,fatal:!1,statusMessage:`Page not found: ${c.fullPath}`,data:{path:c.fullPath}})))});try{[a,x]=yo(()=>u.isReady()),await a,x()}catch(c){[a,x]=yo(()=>r.runWithContext(()=>lo(c))),await a,x()}const d=h!==u.currentRoute.value.fullPath?u.resolve(h):u.currentRoute.value;m();const f=r.payload.state._layout;return u.beforeEach(async(c,l)=>{var b;await r.callHook("page:loading:start"),c.meta=Rn(c.meta),r.isHydrating&&f&&!Pn(c.meta.layout)&&(c.meta.layout=f),r._processingMiddleware=!0;{const v=new Set([..._b,...r._middleware.global]);for(const j of c.matched){const T=j.meta.middleware;if(T)for(const E of ml(T))v.add(E)}{const j=await r.runWithContext(()=>dl(c.path));if(j.appMiddleware)for(const T in j.appMiddleware)j.appMiddleware[T]?v.add(T):v.delete(T)}for(const j of v){const T=typeof j=="string"?r._middleware.named[j]||await((b=$o[j])==null?void 0:b.call($o).then(M=>M.default||M)):j;if(!T)throw new Error(`Unknown route middleware: '${j}'.`);const E=await r.runWithContext(()=>T(c,l));if(!r.payload.serverRendered&&r.isHydrating&&(E===!1||E instanceof Error)){const M=E||Sa({statusCode:404,statusMessage:`Page Not Found: ${h}`});return await r.runWithContext(()=>lo(M)),!1}if(E!==!0&&(E||E===!1))return E}}}),u.onError(async()=>{delete r._processingMiddleware,await r.callHook("page:loading:end")}),r.hooks.hookOnce("app:created",async()=>{try{"name"in d&&(d.name=void 0),await u.replace({...d,force:!0}),u.options.scrollBehavior=Lr.scrollBehavior}catch(c){await r.runWithContext(()=>lo(c))}}),{provide:{router:u}}}}),Da=globalThis.requestIdleCallback||(r=>{const a=Date.now(),x={didTimeout:!1,timeRemaining:()=>Math.max(0,50-(Date.now()-a))};return setTimeout(()=>{r(x)},1)}),jb=globalThis.cancelIdleCallback||(r=>{clearTimeout(r)}),gi=r=>{const a=Dt();a.isHydrating?a.hooks.hookOnce("app:suspense:resolve",()=>{Da(()=>r())}):Da(()=>r())},Sb=tn({name:"nuxt:payload",setup(r){jr().beforeResolve(async(a,x)=>{if(a.path===x.path)return;const t=await sc(a.path);t&&Object.assign(r.static.data,t.data)}),gi(()=>{var a;r.hooks.hook("link:prefetch",async x=>{const{hostname:t}=new URL(x,window.location.href);t===window.location.hostname&&await sc(x)}),((a=navigator.connection)==null?void 0:a.effectiveType)!=="slow-2g"&&setTimeout(mi,1e3)})}}),Eb=tn(()=>{const r=jr();gi(()=>{r.beforeResolve(async()=>{await new Promise(a=>{setTimeout(a,100),requestAnimationFrame(()=>{setTimeout(a,0)})})})})}),Tb=tn(r=>{let a;async function x(){const t=await mi();a&&clearTimeout(a),a=setTimeout(x,Yu);try{const w=await $fetch(ll("builds/latest.json")+`?${Date.now()}`);w.id!==t.id&&r.hooks.callHook("app:manifest:update",w)}catch{}}gi(()=>{a=setTimeout(x,Yu)})});function kb(r={}){const a=r.path||window.location.pathname;let x={};try{x=qs(sessionStorage.getItem("nuxt:reload")||"{}")}catch{}if(r.force||(x==null?void 0:x.path)!==a||(x==null?void 0:x.expires){t.clear()}),r.hook("app:chunkError",({error:g})=>{t.add(g)});function w(g){const u="href"in g&&g.href[0]==="#"?x.app.baseURL+g.href:di(x.app.baseURL,g.fullPath);kb({path:u,persistState:!0})}r.hook("app:manifest:update",()=>{a.beforeResolve(w)}),a.onError((g,s)=>{t.has(g)&&w(s)})}}),Cb=ht(()=>rt(()=>import("./C-O5BI11.js"),__vite__mapDeps([15,6,1,2,3,4,5,16,7,8]),import.meta.url).then(r=>r.default||r.default||r)),Ob=ht(()=>rt(()=>import("./UzigjR5s.js"),__vite__mapDeps([17,16,5,7,8,4]),import.meta.url).then(r=>r.default||r.default||r)),Pb=ht(()=>rt(()=>import("./CMNpy2Is.js"),__vite__mapDeps([18,5,7,8,4]),import.meta.url).then(r=>r.default||r.default||r)),Ab=ht(()=>rt(()=>import("./cufizuWq.js"),__vite__mapDeps([16,5,7,8,4]),import.meta.url).then(r=>r.default||r.default||r)),Lb=ht(()=>rt(()=>import("./DrabSix5.js"),__vite__mapDeps([6,1,2,3,4,5]),import.meta.url).then(r=>r.default||r.default||r)),Rb=ht(()=>rt(()=>import("./BxFzRrKR.js"),__vite__mapDeps([19,1,2,3,4,5]),import.meta.url).then(r=>r.default||r.default||r)),Ib=ht(()=>rt(()=>import("./BY70Pa1e.js"),__vite__mapDeps([20,3]),import.meta.url).then(r=>r.default||r.default||r)),Db=ht(()=>rt(()=>import("./DzMxnq8n.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Fb=ht(()=>rt(()=>import("./Dp7uR_kE.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Nb=ht(()=>rt(()=>import("./BLsxZwe9.js"),__vite__mapDeps([21,20,3]),import.meta.url).then(r=>r.default||r.default||r)),Ub=ht(()=>rt(()=>import("./DS100L4J.js"),__vite__mapDeps([22,23,24]),import.meta.url).then(r=>r.default||r.default||r)),Bb=ht(()=>rt(()=>import("./BmkabxXg.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Gb=ht(()=>rt(()=>import("./BenPNx-m.js"),__vite__mapDeps([25,23,24]),import.meta.url).then(r=>r.default||r.default||r)),Vb=ht(()=>rt(()=>import("./BlMa8jv2.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Hb=ht(()=>rt(()=>import("./D-cHg-Fi.js"),[],import.meta.url).then(r=>r.default||r.default||r)),zb=ht(()=>rt(()=>import("./mIisoFHp.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Wb=ht(()=>rt(()=>import("./BfDVGm6D.js"),[],import.meta.url).then(r=>r.default||r.default||r)),qb=ht(()=>rt(()=>import("./D1bB9L70.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Xb=ht(()=>rt(()=>import("./DEdI-0CJ.js"),[],import.meta.url).then(r=>r.default||r.default||r)),$b=ht(()=>rt(()=>import("./DZAhYQpW.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Yb=ht(()=>rt(()=>import("./ZRsJ-Dry.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Kb=ht(()=>rt(()=>import("./Dz8Mq6rn.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Zb=ht(()=>rt(()=>import("./D9IbAJ0u.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Qb=ht(()=>rt(()=>import("./Bj-Uk9MO.js"),[],import.meta.url).then(r=>r.default||r.default||r)),Jb=ht(()=>rt(()=>import("./BBSVol53.js"),[],import.meta.url).then(r=>r.default||r.default||r)),e1=ht(()=>rt(()=>import("./DXvJDNSx.js"),[],import.meta.url).then(r=>r.default||r.default||r)),t1=ht(()=>rt(()=>import("./DvMJuzSW.js"),[],import.meta.url).then(r=>r.default||r.default||r)),r1=ht(()=>rt(()=>import("./DVb7Ucrz.js"),[],import.meta.url).then(r=>r.default||r.default||r)),n1=ht(()=>rt(()=>import("./DcYzECSc.js"),[],import.meta.url).then(r=>r.default||r.default||r)),o1=ht(()=>rt(()=>import("./BsGN5WoF.js"),[],import.meta.url).then(r=>r.default||r.default||r)),s1=ht(()=>rt(()=>import("./1Kbf3Vya.js"),[],import.meta.url).then(r=>r.default||r.default||r)),i1=ht(()=>rt(()=>import("./oLoEepVu.js"),[],import.meta.url).then(r=>r.default||r.default||r)),a1=ht(()=>rt(()=>import("./D7KuDJ9z.js"),[],import.meta.url).then(r=>r.default||r.default||r)),l1=ht(()=>rt(()=>import("./DvojTKw_.js"),[],import.meta.url).then(r=>r.default||r.default||r)),u1=ht(()=>rt(()=>import("./Co_4egda.js"),[],import.meta.url).then(r=>r.default||r.default||r)),c1=ht(()=>rt(()=>import("./Dcfpj4_h.js"),[],import.meta.url).then(r=>r.default||r.default||r)),d1=[["ContentDoc",Cb],["ContentList",Ob],["ContentNavigation",Pb],["ContentQuery",Ab],["ContentRenderer",Lb],["ContentRendererMarkdown",Rb],["MDCSlot",Ib],["DocumentDrivenEmpty",Db],["DocumentDrivenNotFound",Fb],["Markdown",Nb],["ProseCode",Ub],["ProseCodeInline",Bb],["ProsePre",Gb],["ProseA",Vb],["ProseBlockquote",Hb],["ProseEm",zb],["ProseH1",Wb],["ProseH2",qb],["ProseH3",Xb],["ProseH4",$b],["ProseH5",Yb],["ProseH6",Kb],["ProseHr",Zb],["ProseImg",Qb],["ProseLi",Jb],["ProseOl",e1],["ProseP",t1],["ProseScript",r1],["ProseStrong",n1],["ProseTable",o1],["ProseTbody",s1],["ProseTd",i1],["ProseTh",a1],["ProseThead",l1],["ProseTr",u1],["ProseUl",c1]],f1=tn({name:"nuxt:global-components",setup(r){for(const[a,x]of d1)r.vueApp.component(a,x),r.vueApp.component("Lazy"+a,x)}}),Tn={default:()=>rt(()=>import("./F1DGrIW6.js"),[],import.meta.url),empty:()=>rt(()=>import("./CyLrJzPW.js"),[],import.meta.url),light:()=>rt(()=>import("./tatS9fxq.js"),[],import.meta.url)},h1=tn({name:"nuxt:prefetch",setup(r){const a=jr();r.hooks.hook("app:mounted",()=>{a.beforeEach(async x=>{var w;const t=(w=x==null?void 0:x.meta)==null?void 0:w.layout;t&&typeof Tn[t]=="function"&&await Tn[t]()})}),r.hooks.hook("link:prefetch",x=>{if(pn(x))return;const t=a.resolve(x);if(!t)return;const w=t.meta.layout;let g=ml(t.meta.middleware);g=g.filter(s=>typeof s=="string");for(const s of g)typeof $o[s]=="function"&&$o[s]();w&&typeof Tn[w]=="function"&&Tn[w]()})}});var ks=typeof globalThis<"u"?globalThis:typeof window<"u"?window:typeof global<"u"?global:typeof self<"u"?self:{};function p1(r){return r&&r.__esModule&&Object.prototype.hasOwnProperty.call(r,"default")?r.default:r}function Ms(r){throw new Error('Could not dynamically require "'+r+'". Please configure the dynamicRequireTargets or/and ignoreDynamicRequires option of @rollup/plugin-commonjs appropriately for this require call to work.')}var ih={exports:{}};/*! p5.js v1.11.0 September 25, 2024 */(function(r,a){(function(x){r.exports=x()})(function(){var x;return function t(w,g,s){function u(y,m){if(!g[y]){if(!w[y]){var n=typeof Ms=="function"&&Ms;if(!m&&n)return n(y,!0);if(o)return o(y,!0);throw(m=new Error("Cannot find module '"+y+"'")).code="MODULE_NOT_FOUND",m}n=g[y]={exports:{}},w[y][0].call(n.exports,function(i){return u(w[y][1][i]||i)},n,n.exports,t,w,g,s)}return g[y].exports}for(var o=typeof Ms=="function"&&Ms,h=0;h>16&255,l[b++]=d>>8&255,l[b++]=255&d;return c===2&&(d=u[i.charCodeAt(f)]<<2|u[i.charCodeAt(f+1)]>>4,l[b++]=255&d),c===1&&(d=u[i.charCodeAt(f)]<<10|u[i.charCodeAt(f+1)]<<4|u[i.charCodeAt(f+2)]>>2,l[b++]=d>>8&255,l[b++]=255&d),l},g.fromByteArray=function(i){for(var d,f=i.length,p=f%3,c=[],l=0,b=f-p;l>18&63]+s[L>>12&63]+s[L>>6&63]+s[63&L]}(E));return M.join("")}(i,l,b>2]+s[d<<4&63]+"==")):p==2&&(d=(i[f-2]<<8)+i[f-1],c.push(s[d>>10]+s[d>>4&63]+s[d<<2&63]+"=")),c.join("")};for(var s=[],u=[],o=typeof Uint8Array<"u"?Uint8Array:Array,h="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",y=0,m=h.length;y>>1;case"base64":return V(k).length;default:if(K)return X?-1:W(k).length;D=(""+D).toLowerCase(),K=!0}}function b(k,D,G){var X,K=!1;if((D=D===void 0||D<0?0:D)>this.length||(G=G===void 0||G>this.length?this.length:G)<=0||(G>>>=0)<=(D>>>=0))return"";for(k=k||"utf8";;)switch(k){case"hex":var J=this,de=D,ne=G,Y=J.length;(!ne||ne<0||Y=k.length){if(K)return-1;G=k.length-1}else if(G<0){if(!K)return-1;G=0}if(typeof D=="string"&&(D=n.from(D,X)),n.isBuffer(D))return D.length===0?-1:T(k,D,G,X,K);if(typeof D=="number")return D&=255,typeof Uint8Array.prototype.indexOf=="function"?(K?Uint8Array.prototype.indexOf:Uint8Array.prototype.lastIndexOf).call(k,D,G):T(k,[D],G,X,K);throw new TypeError("val must be string, number or Buffer")}function T(k,D,G,X,K){var J=1,ne=k.length,ue=D.length;if(X!==void 0&&((X=String(X).toLowerCase())==="ucs2"||X==="ucs-2"||X==="utf16le"||X==="utf-16le")){if(k.length<2||D.length<2)return-1;ne/=J=2,ue/=2,G/=2}function ie(ce,fe){return J===1?ce[fe]:ce.readUInt16BE(fe*J)}if(K)for(var he=-1,Y=G;Y>8,ue=ue%256,ie.push(ue),ie.push(ne);return ie}(D,k.length-G),k,G,X)}function P(k,D,G){G=Math.min(k.length,G);for(var X=[],K=D;K>>10&1023|55296),Y=56320|1023&Y),X.push(Y),K+=z}var te=X,ce=te.length;if(ce<=L)return String.fromCharCode.apply(String,te);for(var fe="",de=0;de"u"||typeof console.error!="function"||console.error("This browser lacks typed array (Uint8Array) support which is required by `buffer` v5.x. Use `buffer` v4.x if you require old browser support."),Object.defineProperty(n.prototype,"parent",{enumerable:!0,get:function(){if(n.isBuffer(this))return this.buffer}}),Object.defineProperty(n.prototype,"offset",{enumerable:!0,get:function(){if(n.isBuffer(this))return this.byteOffset}}),typeof Symbol<"u"&&Symbol.species!=null&&n[Symbol.species]===n&&Object.defineProperty(n,Symbol.species,{value:null,configurable:!0,enumerable:!1,writable:!1}),n.poolSize=8192,n.from=i,Object.setPrototypeOf(n.prototype,Uint8Array.prototype),Object.setPrototypeOf(n,Uint8Array),n.alloc=function(k,D,G){return D=D,G=G,d(k=k),!(k<=0)&&D!==void 0?typeof G=="string"?m(k).fill(D,G):m(k).fill(D):m(k)},n.allocUnsafe=f,n.allocUnsafeSlow=f,n.isBuffer=function(k){return k!=null&&k._isBuffer===!0&&k!==n.prototype},n.compare=function(k,D){if(Q(k,Uint8Array)&&(k=n.from(k,k.offset,k.byteLength)),Q(D,Uint8Array)&&(D=n.from(D,D.offset,D.byteLength)),!n.isBuffer(k)||!n.isBuffer(D))throw new TypeError('The "buf1", "buf2" arguments must be one of type Buffer or Uint8Array');if(k===D)return 0;for(var G=k.length,X=D.length,K=0,J=Math.min(G,X);Kk&&(D+=" ... "),""},h&&(n.prototype[h]=n.prototype.inspect),n.prototype.compare=function(k,D,G,X,K){if(Q(k,Uint8Array)&&(k=n.from(k,k.offset,k.byteLength)),!n.isBuffer(k))throw new TypeError('The "target" argument must be one of type Buffer or Uint8Array. Received type '+typeof k);if(G===void 0&&(G=k?k.length:0),X===void 0&&(X=0),K===void 0&&(K=this.length),(D=D===void 0?0:D)<0||G>k.length||X<0||K>this.length)throw new RangeError("out of range index");if(K<=X&&G<=D)return 0;if(K<=X)return-1;if(G<=D)return 1;if(this===k)return 0;for(var J=(K>>>=0)-(X>>>=0),ne=(G>>>=0)-(D>>>=0),ue=Math.min(J,ne),ie=this.slice(X,K),he=k.slice(D,G),Y=0;Y>>=0,isFinite(G)?(G>>>=0,X===void 0&&(X="utf8")):(X=G,G=void 0)}var K=this.length-D;if((G===void 0||Kthis.length)throw new RangeError("Attempt to write outside buffer bounds");X=X||"utf8";for(var J,ne,ue,ie=!1;;)switch(X){case"hex":var he=this,Y=k,z=D,te=G,ce=(z=Number(z)||0,he.length-z);(!te||ce<(te=Number(te)))&&(te=ce),(ce=Y.length)/2k.length)throw new RangeError("Index out of range")}function A(k,D,G,X){if(G+X>k.length)throw new RangeError("Index out of range");if(G<0)throw new RangeError("Index out of range")}function N(k,D,G,X,K){return D=+D,G>>>=0,K||A(k,0,G,4),o.write(k,D,G,X,23,4),G+4}function F(k,D,G,X,K){return D=+D,G>>>=0,K||A(k,0,G,8),o.write(k,D,G,X,52,8),G+8}n.prototype.slice=function(k,D){var G=this.length,G=((k=~~k)<0?(k+=G)<0&&(k=0):G>>=0,D>>>=0,G||C(k,D,this.length);for(var X=this[k],K=1,J=0;++J>>=0,D>>>=0,G||C(k,D,this.length);for(var X=this[k+--D],K=1;0>>=0,D||C(k,1,this.length),this[k]},n.prototype.readUInt16LE=function(k,D){return k>>>=0,D||C(k,2,this.length),this[k]|this[k+1]<<8},n.prototype.readUInt16BE=function(k,D){return k>>>=0,D||C(k,2,this.length),this[k]<<8|this[k+1]},n.prototype.readUInt32LE=function(k,D){return k>>>=0,D||C(k,4,this.length),(this[k]|this[k+1]<<8|this[k+2]<<16)+16777216*this[k+3]},n.prototype.readUInt32BE=function(k,D){return k>>>=0,D||C(k,4,this.length),16777216*this[k]+(this[k+1]<<16|this[k+2]<<8|this[k+3])},n.prototype.readIntLE=function(k,D,G){k>>>=0,D>>>=0,G||C(k,D,this.length);for(var X=this[k],K=1,J=0;++J>>=0,D>>>=0,G||C(k,D,this.length);for(var X=D,K=1,J=this[k+--X];0>>=0,D||C(k,1,this.length),128&this[k]?-1*(255-this[k]+1):this[k]},n.prototype.readInt16LE=function(k,D){return k>>>=0,D||C(k,2,this.length),D=this[k]|this[k+1]<<8,32768&D?4294901760|D:D},n.prototype.readInt16BE=function(k,D){return k>>>=0,D||C(k,2,this.length),D=this[k+1]|this[k]<<8,32768&D?4294901760|D:D},n.prototype.readInt32LE=function(k,D){return k>>>=0,D||C(k,4,this.length),this[k]|this[k+1]<<8|this[k+2]<<16|this[k+3]<<24},n.prototype.readInt32BE=function(k,D){return k>>>=0,D||C(k,4,this.length),this[k]<<24|this[k+1]<<16|this[k+2]<<8|this[k+3]},n.prototype.readFloatLE=function(k,D){return k>>>=0,D||C(k,4,this.length),o.read(this,k,!0,23,4)},n.prototype.readFloatBE=function(k,D){return k>>>=0,D||C(k,4,this.length),o.read(this,k,!1,23,4)},n.prototype.readDoubleLE=function(k,D){return k>>>=0,D||C(k,8,this.length),o.read(this,k,!0,52,8)},n.prototype.readDoubleBE=function(k,D){return k>>>=0,D||C(k,8,this.length),o.read(this,k,!1,52,8)},n.prototype.writeUIntLE=function(k,D,G,X){k=+k,D>>>=0,G>>>=0,X||I(this,k,D,G,Math.pow(2,8*G)-1,0);var K=1,J=0;for(this[D]=255&k;++J>>=0,G>>>=0,X||I(this,k,D,G,Math.pow(2,8*G)-1,0);var K=G-1,J=1;for(this[D+K]=255&k;0<=--K&&(J*=256);)this[D+K]=k/J&255;return D+G},n.prototype.writeUInt8=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,1,255,0),this[D]=255&k,D+1},n.prototype.writeUInt16LE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,2,65535,0),this[D]=255&k,this[D+1]=k>>>8,D+2},n.prototype.writeUInt16BE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,2,65535,0),this[D]=k>>>8,this[D+1]=255&k,D+2},n.prototype.writeUInt32LE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,4,4294967295,0),this[D+3]=k>>>24,this[D+2]=k>>>16,this[D+1]=k>>>8,this[D]=255&k,D+4},n.prototype.writeUInt32BE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,4,4294967295,0),this[D]=k>>>24,this[D+1]=k>>>16,this[D+2]=k>>>8,this[D+3]=255&k,D+4},n.prototype.writeIntLE=function(k,D,G,X){k=+k,D>>>=0,X||I(this,k,D,G,(X=Math.pow(2,8*G-1))-1,-X);var K=0,J=1,ne=0;for(this[D]=255&k;++K>0)-ne&255;return D+G},n.prototype.writeIntBE=function(k,D,G,X){k=+k,D>>>=0,X||I(this,k,D,G,(X=Math.pow(2,8*G-1))-1,-X);var K=G-1,J=1,ne=0;for(this[D+K]=255&k;0<=--K&&(J*=256);)k<0&&ne===0&&this[D+K+1]!==0&&(ne=1),this[D+K]=(k/J>>0)-ne&255;return D+G},n.prototype.writeInt8=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,1,127,-128),this[D]=255&(k=k<0?255+k+1:k),D+1},n.prototype.writeInt16LE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,2,32767,-32768),this[D]=255&k,this[D+1]=k>>>8,D+2},n.prototype.writeInt16BE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,2,32767,-32768),this[D]=k>>>8,this[D+1]=255&k,D+2},n.prototype.writeInt32LE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,4,2147483647,-2147483648),this[D]=255&k,this[D+1]=k>>>8,this[D+2]=k>>>16,this[D+3]=k>>>24,D+4},n.prototype.writeInt32BE=function(k,D,G){return k=+k,D>>>=0,G||I(this,k,D,4,2147483647,-2147483648),this[D]=(k=k<0?4294967295+k+1:k)>>>24,this[D+1]=k>>>16,this[D+2]=k>>>8,this[D+3]=255&k,D+4},n.prototype.writeFloatLE=function(k,D,G){return N(this,k,D,!0,G)},n.prototype.writeFloatBE=function(k,D,G){return N(this,k,D,!1,G)},n.prototype.writeDoubleLE=function(k,D,G){return F(this,k,D,!0,G)},n.prototype.writeDoubleBE=function(k,D,G){return F(this,k,D,!1,G)},n.prototype.copy=function(k,D,G,X){if(!n.isBuffer(k))throw new TypeError("argument should be a Buffer");if(G=G||0,X||X===0||(X=this.length),D>=k.length&&(D=k.length),(X=0=this.length)throw new RangeError("Index out of range");if(X<0)throw new RangeError("sourceEnd out of bounds");X>this.length&&(X=this.length);var K=(X=k.length-D>>=0,G=G===void 0?this.length:G>>>0,typeof(k=k||0)=="number")for(J=D;J>6|192,63&G|128)}else if(G<65536){if((D-=3)<0)break;J.push(G>>12|224,G>>6&63|128,63&G|128)}else{if(!(G<1114112))throw new Error("Invalid code point");if((D-=4)<0)break;J.push(G>>18|240,G>>12&63|128,G>>6&63|128,63&G|128)}}return J}function V(k){return u.toByteArray(function(D){if((D=(D=D.split("=")[0]).trim().replace(U,"")).length<2)return"";for(;D.length%4!=0;)D+="=";return D}(k))}function H(k,D,G,X){for(var K=0;K=D.length||K>=k.length);++K)D[K+G]=k[K];return K}function Q(k,D){return k instanceof D||k!=null&&k.constructor!=null&&k.constructor.name!=null&&k.constructor.name===D.name}function ee(k){return k!=k}var ae=function(){for(var k="0123456789abcdef",D=new Array(256),G=0;G<16;++G)for(var X=16*G,K=0;K<16;++K)D[X+K]=k[G]+k[K];return D}()}).call(this,t("buffer").Buffer)},{"base64-js":1,buffer:4,ieee754:254}],5:[function(t,w,g){w.exports=function(s){if(typeof s!="function")throw TypeError(String(s)+" is not a function");return s}},{}],6:[function(t,w,g){var s=t("../internals/is-object");w.exports=function(u){if(s(u)||u===null)return u;throw TypeError("Can't set "+String(u)+" as a prototype")}},{"../internals/is-object":75}],7:[function(o,w,g){var s=o("../internals/well-known-symbol"),u=o("../internals/object-create"),o=o("../internals/object-define-property"),h=s("unscopables"),y=Array.prototype;y[h]==null&&o.f(y,h,{configurable:!0,value:u(null)}),w.exports=function(m){y[h][m]=!0}},{"../internals/object-create":91,"../internals/object-define-property":93,"../internals/well-known-symbol":150}],8:[function(t,w,g){var s=t("../internals/string-multibyte").charAt;w.exports=function(u,o,h){return o+(h?s(u,o).length:1)}},{"../internals/string-multibyte":125}],9:[function(t,w,g){w.exports=function(s,u,o){if(s instanceof u)return s;throw TypeError("Incorrect "+(o?o+" ":"")+"invocation")}},{}],10:[function(t,w,g){var s=t("../internals/is-object");w.exports=function(u){if(s(u))return u;throw TypeError(String(u)+" is not an object")}},{"../internals/is-object":75}],11:[function(t,w,g){w.exports=typeof ArrayBuffer<"u"&&typeof DataView<"u"},{}],12:[function(A,w,g){function s(F){return m(F)&&n(N,i(F))}var u,o=A("../internals/array-buffer-native"),h=A("../internals/descriptors"),y=A("../internals/global"),m=A("../internals/is-object"),n=A("../internals/has"),i=A("../internals/classof"),d=A("../internals/create-non-enumerable-property"),f=A("../internals/redefine"),p=A("../internals/object-define-property").f,c=A("../internals/object-get-prototype-of"),l=A("../internals/object-set-prototype-of"),L=A("../internals/well-known-symbol"),A=A("../internals/uid"),b=y.Int8Array,v=b&&b.prototype,j=y.Uint8ClampedArray,j=j&&j.prototype,T=b&&c(b),E=v&&c(v),M=Object.prototype,P=M.isPrototypeOf,L=L("toStringTag"),C=A("TYPED_ARRAY_TAG"),I=o&&!!l&&i(y.opera)!=="Opera",A=!1,N={Int8Array:1,Uint8Array:1,Uint8ClampedArray:1,Int16Array:2,Uint16Array:2,Int32Array:4,Uint32Array:4,Float32Array:4,Float64Array:8};for(u in N)y[u]||(I=!1);if((!I||typeof T!="function"||T===Function.prototype)&&(T=function(){throw TypeError("Incorrect invocation")},I))for(u in N)y[u]&&l(y[u],T);if((!I||!E||E===M)&&(E=T.prototype,I))for(u in N)y[u]&&l(y[u].prototype,E);if(I&&c(j)!==E&&l(j,E),h&&!n(E,L))for(u in A=!0,p(E,L,{get:function(){return m(this)?this[C]:void 0}}),N)y[u]&&d(y[u],C,u);w.exports={NATIVE_ARRAY_BUFFER_VIEWS:I,TYPED_ARRAY_TAG:A&&C,aTypedArray:function(F){if(s(F))return F;throw TypeError("Target is not a typed array")},aTypedArrayConstructor:function(F){if(l){if(P.call(T,F))return F}else for(var U in N)if(n(N,u)&&(U=y[U],U&&(F===U||P.call(U,F))))return F;throw TypeError("Target is not a typed array constructor")},exportTypedArrayMethod:function(F,U,W){if(h){if(W)for(var V in N)V=y[V],V&&n(V.prototype,F)&&delete V.prototype[F];E[F]&&!W||f(E,F,!W&&I&&v[F]||U)}},exportTypedArrayStaticMethod:function(F,U,W){var V,H;if(h){if(l){if(W)for(V in N)(H=y[V])&&n(H,F)&&delete H[F];if(T[F]&&!W)return;try{return f(T,F,!W&&I&&b[F]||U)}catch{}}for(V in N)!(H=y[V])||H[F]&&!W||f(H,F,U)}},isView:function(F){return F=i(F),F==="DataView"||n(N,F)},isTypedArray:s,TypedArray:T,TypedArrayPrototype:E}},{"../internals/array-buffer-native":11,"../internals/classof":29,"../internals/create-non-enumerable-property":38,"../internals/descriptors":43,"../internals/global":60,"../internals/has":61,"../internals/is-object":75,"../internals/object-define-property":93,"../internals/object-get-prototype-of":98,"../internals/object-set-prototype-of":102,"../internals/redefine":110,"../internals/uid":147,"../internals/well-known-symbol":150}],13:[function(G,w,g){function s(z){return[255&z]}function u(z){return[255&z,z>>8&255]}function o(z){return[255&z,z>>8&255,z>>16&255,z>>24&255]}function h(z){return z[3]<<24|z[2]<<16|z[1]<<8|z[0]}function y(z){return J(z,23,4)}function m(z){return J(z,52,8)}function n(z,te){A(z[Q],te,{get:function(){return U(this)[te]}})}function i(ve,te,ye,fe){if(ye=M(ye),ve=U(ve),ye+te>ve.byteLength)throw K(ee);var de=U(ve.buffer).bytes,ye=ye+ve.byteOffset,ve=de.slice(ye,ye+te);return fe?ve:ve.reverse()}function d(z,te,ce,fe,de,ye){if(ce=M(ce),z=U(z),ce+te>z.byteLength)throw K(ee);for(var ve=U(z.buffer).bytes,Ee=ce+z.byteOffset,ge=fe(+de),Ne=0;Nehe;)(ue=ie[he++])in k||l(k,ue,ae[ue]);f.constructor=k}C&&L(G)!==X&&C(G,X);var P=new D(new k(2)),Y=G.setInt8;P.setInt8(0,2147483648),P.setInt8(1,2147483649),!P.getInt8(0)&&P.getInt8(1)||b(G,{setInt8:function(te,ce){Y.call(this,te,ce<<24>>24)},setUint8:function(te,ce){Y.call(this,te,ce<<24>>24)}},{unsafe:!0})}else k=function(z){j(this,k,V),z=M(z),W(this,{bytes:N.call(new Array(z),0),byteLength:z}),p||(this.byteLength=z)},D=function(z,de,ce){j(this,D,H),j(z,k,H);var fe=U(z).byteLength,de=T(de);if(de<0||fe>24},getUint8:function(z){return i(this,1,z)[0]},getInt16:function(z){return z=i(this,2,z,1>16},getUint16:function(z){return z=i(this,2,z,1>>0},getFloat32:function(z){return ne(i(this,4,z,1"+n+""}},{"../internals/require-object-coercible":115}],37:[function(t,w,g){function s(){return this}var u=t("../internals/iterators-core").IteratorPrototype,o=t("../internals/object-create"),h=t("../internals/create-property-descriptor"),y=t("../internals/set-to-string-tag"),m=t("../internals/iterators");w.exports=function(n,i,d){return i+=" Iterator",n.prototype=o(u,{next:h(1,d)}),y(n,i,!1,!0),m[i]=s,n}},{"../internals/create-property-descriptor":39,"../internals/iterators":80,"../internals/iterators-core":79,"../internals/object-create":91,"../internals/set-to-string-tag":119}],38:[function(t,w,g){var s=t("../internals/descriptors"),u=t("../internals/object-define-property"),o=t("../internals/create-property-descriptor");w.exports=s?function(h,y,m){return u.f(h,y,o(1,m))}:function(h,y,m){return h[y]=m,h}},{"../internals/create-property-descriptor":39,"../internals/descriptors":43,"../internals/object-define-property":93}],39:[function(t,w,g){w.exports=function(s,u){return{enumerable:!(1&s),configurable:!(2&s),writable:!(4&s),value:u}}},{}],40:[function(t,w,g){var s=t("../internals/to-primitive"),u=t("../internals/object-define-property"),o=t("../internals/create-property-descriptor");w.exports=function(h,y,m){y=s(y),y in h?u.f(h,y,o(0,m)):h[y]=m}},{"../internals/create-property-descriptor":39,"../internals/object-define-property":93,"../internals/to-primitive":142}],41:[function(c,w,g){function s(){return this}var u=c("../internals/export"),o=c("../internals/create-iterator-constructor"),h=c("../internals/object-get-prototype-of"),y=c("../internals/object-set-prototype-of"),m=c("../internals/set-to-string-tag"),n=c("../internals/create-non-enumerable-property"),i=c("../internals/redefine"),d=c("../internals/well-known-symbol"),f=c("../internals/is-pure"),p=c("../internals/iterators"),c=c("../internals/iterators-core"),l=c.IteratorPrototype,b=c.BUGGY_SAFARI_ITERATORS,v=d("iterator"),j="values",T="entries";w.exports=function(E,M,P,W,C,I,A){o(P,M,W);function N(k){if(k===C&&ee)return ee;if(!b&&k in H)return H[k];switch(k){case"keys":case j:case T:return function(){return new P(this,k)}}return function(){return new P(this)}}var F,U,W=M+" Iterator",V=!1,H=E.prototype,Q=H[v]||H["@@iterator"]||C&&H[C],ee=!b&&Q||N(C),ae=M=="Array"&&H.entries||Q;if(ae&&(ae=h(ae.call(new E)),l!==Object.prototype&&ae.next&&(f||h(ae)===l||(y?y(ae,l):typeof ae[v]!="function"&&n(ae,v,s)),m(ae,W,!0,!0),f&&(p[W]=s))),C==j&&Q&&Q.name!==j&&(V=!0,ee=function(){return Q.call(this)}),f&&!A||H[v]===ee||n(H,v,ee),p[M]=ee,C)if(F={values:N(j),keys:I?ee:N("keys"),entries:N(T)},A)for(U in F)!b&&!V&&U in H||i(H,U,F[U]);else u({target:M,proto:!0,forced:b||V},F);return F}},{"../internals/create-iterator-constructor":37,"../internals/create-non-enumerable-property":38,"../internals/export":50,"../internals/is-pure":76,"../internals/iterators":80,"../internals/iterators-core":79,"../internals/object-get-prototype-of":98,"../internals/object-set-prototype-of":102,"../internals/redefine":110,"../internals/set-to-string-tag":119,"../internals/well-known-symbol":150}],42:[function(t,w,g){var s=t("../internals/path"),u=t("../internals/has"),o=t("../internals/well-known-symbol-wrapped"),h=t("../internals/object-define-property").f;w.exports=function(y){var m=s.Symbol||(s.Symbol={});u(m,y)||h(m,y,{value:o.f(y)})}},{"../internals/has":61,"../internals/object-define-property":93,"../internals/path":106,"../internals/well-known-symbol-wrapped":149}],43:[function(t,w,g){t=t("../internals/fails"),w.exports=!t(function(){return Object.defineProperty({},1,{get:function(){return 7}})[1]!=7})},{"../internals/fails":51}],44:[function(u,w,g){var s=u("../internals/global"),u=u("../internals/is-object"),o=s.document,h=u(o)&&u(o.createElement);w.exports=function(y){return h?o.createElement(y):{}}},{"../internals/global":60,"../internals/is-object":75}],45:[function(t,w,g){w.exports={CSSRuleList:0,CSSStyleDeclaration:0,CSSValueList:0,ClientRectList:0,DOMRectList:0,DOMStringList:0,DOMTokenList:1,DataTransferItemList:0,FileList:0,HTMLAllCollection:0,HTMLCollection:0,HTMLFormElement:0,HTMLSelectElement:0,MediaList:0,MimeTypeArray:0,NamedNodeMap:0,NodeList:1,PaintRequestList:0,Plugin:0,PluginArray:0,SVGLengthList:0,SVGNumberList:0,SVGPathSegList:0,SVGPointList:0,SVGStringList:0,SVGTransformList:0,SourceBufferList:0,StyleSheetList:0,TextTrackCueList:0,TextTrackList:0,TouchList:0}},{}],46:[function(t,w,g){t=t("../internals/engine-user-agent"),w.exports=/(iphone|ipod|ipad).*applewebkit/i.test(t)},{"../internals/engine-user-agent":47}],47:[function(t,w,g){t=t("../internals/get-built-in"),w.exports=t("navigator","userAgent")||""},{"../internals/get-built-in":57}],48:[function(o,w,g){var s,u,h=o("../internals/global"),o=o("../internals/engine-user-agent"),h=h.process,h=h&&h.versions,h=h&&h.v8;h?u=(s=h.split("."))[0]+s[1]:o&&(!(s=o.match(/Edge\/(\d+)/))||74<=s[1])&&(s=o.match(/Chrome\/(\d+)/))&&(u=s[1]),w.exports=u&&+u},{"../internals/engine-user-agent":47,"../internals/global":60}],49:[function(t,w,g){w.exports=["constructor","hasOwnProperty","isPrototypeOf","propertyIsEnumerable","toLocaleString","toString","valueOf"]},{}],50:[function(t,w,g){var s=t("../internals/global"),u=t("../internals/object-get-own-property-descriptor").f,o=t("../internals/create-non-enumerable-property"),h=t("../internals/redefine"),y=t("../internals/set-global"),m=t("../internals/copy-constructor-properties"),n=t("../internals/is-forced");w.exports=function(i,d){var f,p,c,l=i.target,b=i.global,v=i.stat,j=b?s:v?s[l]||y(l,{}):(s[l]||{}).prototype;if(j)for(f in d){if(p=d[f],c=i.noTargetGet?(c=u(j,f))&&c.value:j[f],!n(b?f:l+(v?".":"#")+f,i.forced)&&c!==void 0){if(typeof p==typeof c)continue;m(p,c)}(i.sham||c&&c.sham)&&o(p,"sham",!0),h(j,f,p,i)}}},{"../internals/copy-constructor-properties":33,"../internals/create-non-enumerable-property":38,"../internals/global":60,"../internals/is-forced":74,"../internals/object-get-own-property-descriptor":94,"../internals/redefine":110,"../internals/set-global":117}],51:[function(t,w,g){w.exports=function(s){try{return!!s()}catch{return!0}}},{}],52:[function(d,w,g){d("../modules/es.regexp.exec");var s=d("../internals/redefine"),u=d("../internals/fails"),o=d("../internals/well-known-symbol"),h=d("../internals/regexp-exec"),y=d("../internals/create-non-enumerable-property"),m=o("species"),n=!u(function(){var c=/./;return c.exec=function(){var l=[];return l.groups={a:"7"},l},"".replace(c,"$")!=="7"}),i="a".replace(/./,"$0")==="$0",d=o("replace"),f=!!/./[d]&&/./[d]("a","$0")==="",p=!u(function(){var l=/(?:)/,c=l.exec,l=(l.exec=function(){return c.apply(this,arguments)},"ab".split(l));return l.length!==2||l[0]!=="a"||l[1]!=="b"});w.exports=function(c,l,b,v){var j,T,E=o(c),M=!u(function(){var L={};return L[E]=function(){return 7},""[c](L)!=7}),P=M&&!u(function(){var L=!1,C=/a/;return c==="split"&&((C={constructor:{}}).constructor[m]=function(){return C},C.flags="",C[E]=/./[E]),C.exec=function(){return L=!0,null},C[E](""),!L});M&&P&&(c!=="replace"||n&&i&&!f)&&(c!=="split"||p)||(j=/./[E],b=(P=b(E,""[c],function(L,C,I,A,N){return C.exec===h?M&&!N?{done:!0,value:j.call(C,I,A)}:{done:!0,value:L.call(I,C,A)}:{done:!1}},{REPLACE_KEEPS_$0:i,REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE:f}))[0],T=P[1],s(String.prototype,c,b),s(RegExp.prototype,E,l==2?function(L,C){return T.call(L,this,C)}:function(L){return T.call(L,this)})),v&&y(RegExp.prototype[E],"sham",!0)}},{"../internals/create-non-enumerable-property":38,"../internals/fails":51,"../internals/redefine":110,"../internals/regexp-exec":112,"../internals/well-known-symbol":150,"../modules/es.regexp.exec":195}],53:[function(t,w,g){function s(y,m,n,i,d,f,p,c){for(var l,b=d,v=0,j=!!p&&h(p,c,3);v>1,j=n===23?u(2,-24)-u(2,-77):0,T=m<0||m===0&&1/m<0?1:0,E=0;for((m=s(m))!=m||m===1/0?(f=m!=m?1:0,d=b):(d=o(h(m)/y),m*(p=u(2,-d))<1&&(d--,p*=2),2<=(m+=1<=d+v?j/p:j*u(2,1-v))*p&&(d++,p/=2),b<=d+v?(f=0,d=b):1<=d+v?(f=(m*p-1)*u(2,n),d+=v):(f=m*u(2,v-1)*u(2,n),d=0));8<=n;c[E++]=255&f,f/=256,n-=8);for(d=d<>1,c=b-7,l=d-1,b=m[l--],v=127&b;for(b>>=7;0>=-c,c+=n;0"+v+""},b=function(){try{u=document.domain&&new ActiveXObject("htmlfile")}catch{}b=u?((v=u).write(l("")),v.close(),j=v.parentWindow.Object,v=null,j):(v=i("iframe"),j="java"+p+":",v.style.display="none",n.appendChild(v),v.src=String(j),(j=v.contentWindow.document).open(),j.write(l("document.F=Object")),j.close(),j.F);for(var v,j,T=y.length;T--;)delete b[f][y[T]];return b()};m[c]=!0,w.exports=Object.create||function(v,j){var T;return v!==null?(s[f]=o(v),T=new s,s[f]=null,T[c]=v):T=b(),j===void 0?T:h(T,j)}},{"../internals/an-object":10,"../internals/document-create-element":44,"../internals/enum-bug-keys":49,"../internals/hidden-keys":62,"../internals/html":64,"../internals/object-define-properties":92,"../internals/shared-key":120}],92:[function(t,w,g){var s=t("../internals/descriptors"),u=t("../internals/object-define-property"),o=t("../internals/an-object"),h=t("../internals/object-keys");w.exports=s?Object.defineProperties:function(y,m){o(y);for(var n,i=h(m),d=i.length,f=0;fd;)!s(i,n=m[d++])||~o(f,n)||f.push(n);return f}},{"../internals/array-includes":18,"../internals/has":61,"../internals/hidden-keys":62,"../internals/to-indexed-object":136}],100:[function(t,w,g){var s=t("../internals/object-keys-internal"),u=t("../internals/enum-bug-keys");w.exports=Object.keys||function(o){return s(o,u)}},{"../internals/enum-bug-keys":49,"../internals/object-keys-internal":99}],101:[function(t,w,g){var s={}.propertyIsEnumerable,u=Object.getOwnPropertyDescriptor,o=u&&!s.call({1:2},1);g.f=o?function(h){return h=u(this,h),!!h&&h.enumerable}:s},{}],102:[function(t,w,g){var s=t("../internals/an-object"),u=t("../internals/a-possible-prototype");w.exports=Object.setPrototypeOf||("__proto__"in{}?function(){var o,h=!1,y={};try{(o=Object.getOwnPropertyDescriptor(Object.prototype,"__proto__").set).call(y,[]),h=y instanceof Array}catch{}return function(m,n){return s(m),u(n),h?o.call(m,n):m.__proto__=n,m}}():void 0)},{"../internals/a-possible-prototype":6,"../internals/an-object":10}],103:[function(t,w,g){function s(m){return function(n){for(var i,d=h(n),f=o(d),p=f.length,c=0,l=[];cl&&(p=p.slice(0,l)),m?f+p:p+f)}}var u=t("../internals/to-length"),o=t("../internals/string-repeat"),h=t("../internals/require-object-coercible"),y=Math.ceil;w.exports={start:s(!1),end:s(!0)}},{"../internals/require-object-coercible":115,"../internals/string-repeat":129,"../internals/to-length":138}],128:[function(t,w,g){function s(b){return b+22+75*(b<26)}function u(b){var v,j=[],T=(b=function(Q){for(var ee=[],ae=0,k=Q.length;aec((o-M)/N))throw RangeError(f);for(M+=(I-E)*N,E=I,A=0;Ao)throw RangeError(f);if(v==E){for(var F=M,U=h;;U+=h){var W=U<=P?1:P+y<=U?y:U-P;if(F>1,Q+=c(Q/ee);p*y>>1>>=1)&&(h+=h))1&m&&(y+=h);return y}},{"../internals/require-object-coercible":115,"../internals/to-integer":137}],130:[function(t,w,g){var s=t("../internals/fails"),u=t("../internals/whitespaces");w.exports=function(o){return s(function(){return!!u[o]()||"​…᠎"[o]()!="​…᠎"||u[o].name!==o})}},{"../internals/fails":51,"../internals/whitespaces":151}],131:[function(o,w,g){function s(m){return function(n){return n=String(u(n)),1&m&&(n=n.replace(h,"")),n=2&m?n.replace(y,""):n}}var u=o("../internals/require-object-coercible"),o="["+o("../internals/whitespaces")+"]",h=RegExp("^"+o+o+"*"),y=RegExp(o+o+"*$");w.exports={start:s(1),end:s(2),trim:s(3)}},{"../internals/require-object-coercible":115,"../internals/whitespaces":151}],132:[function(p,w,g){function s(C){return function(){L(C)}}function u(C){L(C.data)}function o(C){y.postMessage(C+"",c.protocol+"//"+c.host)}var h,y=p("../internals/global"),m=p("../internals/fails"),n=p("../internals/classof-raw"),i=p("../internals/function-bind-context"),d=p("../internals/html"),f=p("../internals/document-create-element"),p=p("../internals/engine-is-ios"),c=y.location,l=y.setImmediate,b=y.clearImmediate,v=y.process,j=y.MessageChannel,T=y.Dispatch,E=0,M={},P="onreadystatechange",L=function(C){var I;M.hasOwnProperty(C)&&(I=M[C],delete M[C],I())};l&&b||(l=function(C){for(var I=[],A=1;A=f.length?{value:d.target=void 0,done:!0}:p=="keys"?{value:c,done:!1}:p=="values"?{value:f[c],done:!1}:{value:[c,f[c]],done:!1}},"values"),o.Arguments=o.Array,u("keys"),u("values"),u("entries")},{"../internals/add-to-unscopables":7,"../internals/define-iterator":41,"../internals/internal-state":71,"../internals/iterators":80,"../internals/to-indexed-object":136}],166:[function(y,w,g){var s=y("../internals/export"),h=y("../internals/indexed-object"),u=y("../internals/to-indexed-object"),y=y("../internals/array-method-is-strict"),o=[].join,h=h!=Object,y=y("join",",");s({target:"Array",proto:!0,forced:h||!y},{join:function(m){return o.call(u(this),m===void 0?",":m)}})},{"../internals/array-method-is-strict":22,"../internals/export":50,"../internals/indexed-object":67,"../internals/to-indexed-object":136}],167:[function(u,w,g){var s=u("../internals/export"),u=u("../internals/array-last-index-of");s({target:"Array",proto:!0,forced:u!==[].lastIndexOf},{lastIndexOf:u})},{"../internals/array-last-index-of":20,"../internals/export":50}],168:[function(h,w,g){var s=h("../internals/export"),u=h("../internals/array-iteration").map,o=h("../internals/array-method-has-species-support"),h=h("../internals/array-method-uses-to-length"),o=o("map"),h=h("map");s({target:"Array",proto:!0,forced:!o||!h},{map:function(y){return u(this,y,1I;I++)m(T,P=C[I])&&!m(L,P)&&b(L,P,l(T,P));(L.prototype=E).constructor=L,y(o,j,L)}},{"../internals/classof-raw":28,"../internals/descriptors":43,"../internals/fails":51,"../internals/global":60,"../internals/has":61,"../internals/inherit-if-required":68,"../internals/is-forced":74,"../internals/object-create":91,"../internals/object-define-property":93,"../internals/object-get-own-property-descriptor":94,"../internals/object-get-own-property-names":96,"../internals/redefine":110,"../internals/string-trim":131,"../internals/to-primitive":142}],180:[function(t,w,g){t("../internals/export")({target:"Number",stat:!0},{EPSILON:Math.pow(2,-52)})},{"../internals/export":50}],181:[function(t,w,g){t("../internals/export")({target:"Number",stat:!0},{isFinite:t("../internals/number-is-finite")})},{"../internals/export":50,"../internals/number-is-finite":89}],182:[function(m,w,g){function s(d,f,p){return f===0?p:f%2==1?s(d,f-1,p*d):s(d*d,f/2,p)}var u=m("../internals/export"),o=m("../internals/to-integer"),h=m("../internals/this-number-value"),y=m("../internals/string-repeat"),m=m("../internals/fails"),n=1 .toFixed,i=Math.floor;u({target:"Number",proto:!0,forced:n&&(8e-5.toFixed(3)!=="0.000"||.9.toFixed(0)!=="1"||1.255.toFixed(2)!=="1.25"||0xde0b6b3a7640080.toFixed(0)!=="1000000000000000128")||!m(function(){n.call({})})},{toFixed:function(j){function f(P,L){for(var C=-1,I=L;++C<6;)I+=P*T[C],T[C]=I%1e7,I=i(I/1e7)}function p(P){for(var L=6,C=0;0<=--L;)C+=T[L],T[L]=i(C/P),C=C%P*1e7}function c(){for(var P,L=6,C="";0<=--L;)C===""&&L!==0&&T[L]===0||(P=String(T[L]),C=C===""?P:C+y.call("0",7-P.length)+P);return C}var l,b,v=h(this),j=o(j),T=[0,0,0,0,0,0],E="",M="0";if(j<0||20Pe;){var He,We,qe,$e=se[Pe++],lt=Se?$e.ok:$e.fail,ot=$e.resolve,Ze=$e.reject,it=$e.domain;try{lt?(Se||(Te.rejection===ye&&function(nt,Ct){C.call(n,function(){he?J.emit("rejectionHandled",nt):Le(te,nt,Ct.value)})}(Re,Te),Te.rejection=de),lt===!0?He=xe:(it&&it.enter(),He=lt(xe),it&&(it.exit(),qe=!0)),He===$e.promise?Ze(X("Promise-chain cycle")):(We=ge(He))?We.call(He,ot,Ze):ot(He)):Ze(xe)}catch(nt){it&&!qe&&it.exit(),Ze(nt)}}Te.reactions=[],Te.notified=!1,re&&!Te.rejection&&(Z=Re,le=Te,C.call(n,function(){var nt=le.value,Ct=Ie(le);if(Ct&&(Ct=U(function(){he?J.emit("unhandledRejection",nt,Z):Le(z,Z,nt)}),le.rejection=he||Ie(le)?ye:de,Ct.error))throw Ct.value}))}))},Le=function(Re,Te,re){var se;Y?((se=K.createEvent("Event")).promise=Te,se.reason=re,se.initEvent(Re,!1,!0),n.dispatchEvent(se)):se={promise:Te,reason:re},(Te=n["on"+Re])?Te(se):Re===z&&N("Unhandled promise rejection",re)},Ie=function(Re){return Re.rejection!==de&&!Re.parent},Fe=function(Re,Te,re,se){return function(Z){Re(Te,re,Z,se)}},Ge=function(Re,Te,re,se){Te.done||(Te.done=!0,(Te=se||Te).value=re,Te.state=fe,Ne(Re,Te,!0))},we=function(Re,Te,re,se){if(!Te.done){Te.done=!0,se&&(Te=se);try{if(Re===re)throw X("Promise can't be resolved itself");var Z=ge(re);Z?I(function(){var le={done:!1};try{Z.call(re,Fe(we,Re,le,Te),Fe(Ge,Re,le,Te))}catch(xe){Ge(Re,le,xe,Te)}}):(Te.value=re,Te.state=ce,Ne(Re,Te,!1))}catch(le){Ge(Re,{done:!1},le,Te)}}};ve&&(G=function(Re){j(this,G,ee),v(Re),s.call(this);var Te=ae(this);try{Re(Fe(we,this,Te),Fe(Ge,this,Te))}catch(re){Ge(this,Te,re)}},(s=function(Re){k(this,{type:ee,done:!1,notified:!1,parent:!1,reactions:[],rejection:!1,state:0,value:void 0})}).prototype=p(G.prototype,{then:function(Re,Te){var re=D(this),se=ue(L(this,G));return se.ok=typeof Re!="function"||Re,se.fail=typeof Te=="function"&&Te,se.domain=he?J.domain:void 0,re.parent=!0,re.reactions.push(se),re.state!=0&&Ne(this,re,!1),se.promise},catch:function(Re){return this.then(void 0,Re)}}),u=function(){var Re=new s,Te=ae(Re);this.promise=Re,this.resolve=Fe(we,Re,Te),this.reject=Fe(Ge,Re,Te)},F.f=ue=function(Re){return Re===G||Re===o?new u:ie(Re)},m||typeof d!="function"||(h=d.prototype.then,f(d.prototype,"then",function(Re,Te){var re=this;return new G(function(se,Z){h.call(re,se,Z)}).then(Re,Te)},{unsafe:!0}),typeof ne=="function"&&y({global:!0,enumerable:!0,forced:!0},{fetch:function(Re){return A(G,ne.apply(n,arguments))}}))),y({global:!0,wrap:!0,forced:ve},{Promise:G}),c(G,ee,!1,!0),l(ee),o=i(ee),y({target:ee,stat:!0,forced:ve},{reject:function(Re){var Te=ue(this);return Te.reject.call(void 0,Re),Te.promise}}),y({target:ee,stat:!0,forced:m||ve},{resolve:function(Re){return A(m&&this===o?G:this,Re)}}),y({target:ee,stat:!0,forced:Ee},{all:function(Re){var Te=this,re=ue(Te),se=re.resolve,Z=re.reject,le=U(function(){var xe=v(Te.resolve),Se=[],Pe=0,He=1;M(Re,function(We){var qe=Pe++,$e=!1;Se.push(void 0),He++,xe.call(Te,We).then(function(lt){$e||($e=!0,Se[qe]=lt,--He||se(Se))},Z)}),--He||se(Se)});return le.error&&Z(le.value),re.promise},race:function(Re){var Te=this,re=ue(Te),se=re.reject,Z=U(function(){var le=v(Te.resolve);M(Re,function(xe){le.call(Te,xe).then(re.resolve,se)})});return Z.error&&se(Z.value),re.promise}})},{"../internals/a-function":5,"../internals/an-instance":9,"../internals/check-correctness-of-iteration":27,"../internals/classof-raw":28,"../internals/engine-v8-version":48,"../internals/export":50,"../internals/get-built-in":57,"../internals/global":60,"../internals/host-report-errors":63,"../internals/inspect-source":69,"../internals/internal-state":71,"../internals/is-forced":74,"../internals/is-object":75,"../internals/is-pure":76,"../internals/iterate":78,"../internals/microtask":82,"../internals/native-promise-constructor":83,"../internals/new-promise-capability":87,"../internals/perform":107,"../internals/promise-resolve":108,"../internals/redefine":110,"../internals/redefine-all":109,"../internals/set-species":118,"../internals/set-to-string-tag":119,"../internals/species-constructor":123,"../internals/task":132,"../internals/well-known-symbol":150}],192:[function(n,w,g){var s=n("../internals/export"),p=n("../internals/get-built-in"),u=n("../internals/a-function"),o=n("../internals/an-object"),h=n("../internals/is-object"),y=n("../internals/object-create"),m=n("../internals/function-bind"),n=n("../internals/fails"),i=p("Reflect","construct"),d=n(function(){function c(){}return!(i(function(){},[],c)instanceof c)}),f=!n(function(){i(function(){})}),p=d||f;s({target:"Reflect",stat:!0,forced:p,sham:p},{construct:function(c,l){u(c),o(l);var b=arguments.length<3?c:u(arguments[2]);if(f&&!d)return i(c,l,b);if(c==b){switch(l.length){case 0:return new c;case 1:return new c(l[0]);case 2:return new c(l[0],l[1]);case 3:return new c(l[0],l[1],l[2]);case 4:return new c(l[0],l[1],l[2],l[3])}var v=[null];return v.push.apply(v,l),new(m.apply(c,v))}return v=b.prototype,b=y(h(v)?v:Object.prototype),v=Function.apply.call(c,b,l),h(v)?v:b}})},{"../internals/a-function":5,"../internals/an-object":10,"../internals/export":50,"../internals/fails":51,"../internals/function-bind":56,"../internals/get-built-in":57,"../internals/is-object":75,"../internals/object-create":91}],193:[function(t,w,g){var s=t("../internals/export"),u=t("../internals/is-object"),o=t("../internals/an-object"),h=t("../internals/has"),y=t("../internals/object-get-own-property-descriptor"),m=t("../internals/object-get-prototype-of");s({target:"Reflect",stat:!0},{get:function n(i,d){var f,p=arguments.length<3?i:arguments[2];return o(i)===p?i[d]:(f=y.f(i,d))?h(f,"value")?f.value:f.get===void 0?void 0:f.get.call(p):u(f=m(i))?n(f,d,p):void 0}})},{"../internals/an-object":10,"../internals/export":50,"../internals/has":61,"../internals/is-object":75,"../internals/object-get-own-property-descriptor":94,"../internals/object-get-prototype-of":98}],194:[function(t,w,g){var s=t("../internals/descriptors"),u=t("../internals/global"),o=t("../internals/is-forced"),h=t("../internals/inherit-if-required"),y=t("../internals/object-define-property").f,m=t("../internals/object-get-own-property-names").f,n=t("../internals/is-regexp"),i=t("../internals/regexp-flags"),d=t("../internals/regexp-sticky-helpers"),f=t("../internals/redefine"),p=t("../internals/fails"),c=t("../internals/internal-state").set,l=t("../internals/set-species"),b=t("../internals/well-known-symbol")("match"),v=u.RegExp,j=v.prototype,T=/a/g,E=/a/g,M=new v(T)!==T,P=d.UNSUPPORTED_Y;if(s&&o("RegExp",!M||P||p(function(){return E[b]=!1,v(T)!=T||v(E)==E||v(T,"i")!="/a/i"}))){for(var L=function(A,N){var F,U=this instanceof L,W=n(A),V=N===void 0;return!U&&W&&A.constructor===L&&V?A:(M?W&&!V&&(A=A.source):A instanceof L&&(V&&(N=i.call(A)),A=A.source),P&&(F=!!N&&-1I;)(function(A){A in L||y(L,A,{configurable:!0,get:function(){return v[A]},set:function(N){v[A]=N}})})(C[I++]);(j.constructor=L).prototype=j,f(u,"RegExp",L)}l("RegExp")},{"../internals/descriptors":43,"../internals/fails":51,"../internals/global":60,"../internals/inherit-if-required":68,"../internals/internal-state":71,"../internals/is-forced":74,"../internals/is-regexp":77,"../internals/object-define-property":93,"../internals/object-get-own-property-names":96,"../internals/redefine":110,"../internals/regexp-flags":113,"../internals/regexp-sticky-helpers":114,"../internals/set-species":118,"../internals/well-known-symbol":150}],195:[function(u,w,g){var s=u("../internals/export"),u=u("../internals/regexp-exec");s({target:"RegExp",proto:!0,forced:/./.exec!==u},{exec:u})},{"../internals/export":50,"../internals/regexp-exec":112}],196:[function(h,w,g){var s=h("../internals/redefine"),u=h("../internals/an-object"),n=h("../internals/fails"),o=h("../internals/regexp-flags"),h="toString",y=RegExp.prototype,m=y[h],n=n(function(){return m.call({source:"a",flags:"b"})!="/a/b"}),i=m.name!=h;(n||i)&&s(RegExp.prototype,h,function(){var d=u(this),f=String(d.source),p=d.flags;return"/"+f+"/"+String(p===void 0&&d instanceof RegExp&&!("flags"in y)?o.call(d):p)},{unsafe:!0})},{"../internals/an-object":10,"../internals/fails":51,"../internals/redefine":110,"../internals/regexp-flags":113}],197:[function(u,w,g){var s=u("../internals/collection"),u=u("../internals/collection-strong");w.exports=s("Set",function(o){return function(){return o(this,arguments.length?arguments[0]:void 0)}},u)},{"../internals/collection":32,"../internals/collection-strong":30}],198:[function(m,w,g){var s=m("../internals/export"),u=m("../internals/object-get-own-property-descriptor").f,o=m("../internals/to-length"),h=m("../internals/not-a-regexp"),y=m("../internals/require-object-coercible"),d=m("../internals/correct-is-regexp-logic"),m=m("../internals/is-pure"),n="".endsWith,i=Math.min,d=d("endsWith");s({target:"String",proto:!0,forced:!!(m||d||!(s=u(String.prototype,"endsWith"))||s.writable)&&!d},{endsWith:function(f){var p=String(y(this)),c=(h(f),1=i.length?{value:void 0,done:!0}:(i=s(i,d),n.index+=i.length,{value:i,done:!1})})},{"../internals/define-iterator":41,"../internals/internal-state":71,"../internals/string-multibyte":125}],201:[function(t,w,g){var s=t("../internals/fix-regexp-well-known-symbol-logic"),u=t("../internals/an-object"),o=t("../internals/to-length"),h=t("../internals/require-object-coercible"),y=t("../internals/advance-string-index"),m=t("../internals/regexp-exec-abstract");s("match",1,function(n,i,d){return[function(f){var p=h(this),c=f==null?void 0:f[n];return c!==void 0?c.call(f,p):new RegExp(f)[n](String(p))},function(f){var p=d(i,f,this);if(p.done)return p.value;var c=u(f),l=String(this);if(!c.global)return m(c,l);for(var b=c.unicode,v=[],j=c.lastIndex=0;(T=m(c,l))!==null;){var T=String(T[0]);(v[j]=T)===""&&(c.lastIndex=y(l,o(c.lastIndex),b)),j++}return j===0?null:v}]})},{"../internals/advance-string-index":8,"../internals/an-object":10,"../internals/fix-regexp-well-known-symbol-logic":52,"../internals/regexp-exec-abstract":111,"../internals/require-object-coercible":115,"../internals/to-length":138}],202:[function(t,w,g){var s=t("../internals/export"),u=t("../internals/string-pad").start;s({target:"String",proto:!0,forced:t("../internals/string-pad-webkit-bug")},{padStart:function(o){return u(this,o,1]*>)/g,l=/\$([$&'`]|\d\d?)/g;s("replace",2,function(b,v,j,T){var E=T.REGEXP_REPLACE_SUBSTITUTES_UNDEFINED_CAPTURE,M=T.REPLACE_KEEPS_$0,P=E?"$":"$0";return[function(L,C){var I=m(this),A=L==null?void 0:L[b];return A!==void 0?A.call(L,I,C):v.call(String(I),L,C)},function(L,C){if(!E&&M||typeof C=="string"&&C.indexOf(P)===-1){var I=j(v,L,this,C);if(I.done)return I.value}for(var A,N=u(L),F=String(this),U=typeof C=="function",W=(U||(C=String(C)),N.global),V=(W&&(A=N.unicode,N.lastIndex=0),[]);(k=i(N,F))!==null&&(V.push(k),W);)String(k[0])===""&&(N.lastIndex=n(F,h(N.lastIndex),A));for(var H,Q="",ee=0,ae=0;ae>>0;if(C==0)return[];if(M===void 0)return[L];if(!u(M))return j.call(L,M,C);for(var I,A,N,F=[],P=(M.ignoreCase?"i":"")+(M.multiline?"m":"")+(M.unicode?"u":"")+(M.sticky?"y":""),U=0,W=new RegExp(M.source,P+"g");(I=d.call(W,L))&&!(U<(A=W.lastIndex)&&(F.push(L.slice(U,I.index)),1=C));)W.lastIndex===I.index&&W.lastIndex++;return U===L.length?!N&&W.test("")||F.push(""):F.push(L.slice(U)),F.length>C?F.slice(0,C):F}:"0".split(void 0,0).length?function(M,P){return M===void 0&&P===0?[]:j.call(this,M,P)}:j;return[function(M,P){var L=h(this),C=M==null?void 0:M[v];return C!==void 0?C.call(M,L,P):E.call(String(L),M,P)},function(I,P){var L=T(E,I,this,P,E!==j);if(L.done)return L.value;var L=o(I),C=String(this),I=y(L,RegExp),A=L.unicode,N=(L.ignoreCase?"i":"")+(L.multiline?"m":"")+(L.unicode?"u":"")+(b?"y":"g"),F=new I(b?L:"^(?:"+L.source+")",N),U=P===void 0?l:P>>>0;if(U==0)return[];if(C.length===0)return i(F,C)===null?[C]:[];for(var W=0,V=0,H=[];Vne.key){he.splice(ue,0,ne);break}ue===z&&he.push(ne)}ie.updateURL()},forEach:function(ne){for(var ue,ie=H(this).entries,he=j(ne,16))return;for(Gt=0;rr();){if(tr=null,Gt>0)if(rr()=="."&&Gt<4)Ye++;else return;if(!G.test(rr()))return;for(;G.test(rr());){if(qr=parseInt(rr(),10),tr===null)tr=qr;else{if(tr==0)return;tr=tr*10+qr}if(tr>255)return;Ye++}gt[Xe]=gt[Xe]*256+tr,Gt++,(Gt==2||Gt==4)&&Xe++}if(Gt!=4)return;break}else if(rr()==":"){if(Ye++,!rr())return}else if(rr())return;gt[Xe++]=Ht}if(pt!==null)for(kr=Xe-pt,Xe=7;Xe!=0&&kr>0;)Xr=gt[Xe],gt[Xe--]=gt[pt+kr-1],gt[pt+--kr]=Xr;else if(Xe!=8)return;return gt}(ze.slice(1,-1)))?void(_e.host=Ke):ee;if(ge(_e))return ze=M(ze),ue.test(ze)||(Ke=function(Be){var gt=Be.split("."),Xe,pt,Ye,Ht,Yt,Gt,tr;if(gt.length&>[gt.length-1]==""&>.pop(),(Xe=gt.length)>4)return Be;for(pt=[],Ye=0;Ye1&&Ht.charAt(0)=="0"&&(Yt=X.test(Ht)?16:8,Ht=Ht.slice(Yt==8?1:2)),Ht==="")Gt=0;else{if(!(Yt==10?J:Yt==8?K:ne).test(Ht))return Be;Gt=parseInt(Ht,Yt)}pt.push(Gt)}for(Ye=0;Ye=V(256,5-Xe))return null}else if(Gt>255)return null;for(tr=pt.pop(),Ye=0;Ye":1,"`":1}),de=j({},fe,{"#":1,"?":1,"{":1,"}":1}),ye=j({},de,{"/":1,":":1,";":1,"=":1,"@":1,"[":1,"\\":1,"]":1,"^":1,"|":1}),ve=function(_e,ze){var Ke=E(_e,0);return 32"u"&&s!==void 0&&{}.toString.call(s)==="[object process]",b=typeof Uint8ClampedArray<"u"&&typeof importScripts<"u"&&typeof MessageChannel<"u";function v(){var z=setTimeout;return function(){return z(T,1)}}var j=new Array(1e3);function T(){for(var z=0;zL,applyPalette:()=>function(U,W,V="rgb565"){if(!U||!U.buffer)throw new Error("quantize() expected RGBA Uint8Array data");if(!(U instanceof Uint8Array||U instanceof Uint8ClampedArray))throw new Error("quantize() expected RGBA Uint8Array data");if(256>24&255,G=K>>16&255,X=K>>8&255,K=255&K,J=f(K,X,G,D),J=J in k?k[J]:k[J]=function(z,te,ce,fe,de){let ye=0,ve=1e100;for(let Le=0;Leve||(Ee=ge[0],(Ne+=E(Ee-z))>ve||(Ee=ge[1],(Ne+=E(Ee-te))>ve||(Ee=ge[2],(Ne+=E(Ee-ce))>ve||(ve=Ne,ye=Le))))}return ye}(K,X,G,D,W);ae[Y]=J}else{const Y=V==="rgb444"?p:d;for(let z=0;z>16&255,ue=ie>>8&255,ie=255&ie,he=Y(ie,ue,ne),he=he in k?k[he]:k[he]=function(te,ce,fe,de){let ye=0,ve=1e100;for(let Le=0;Leve||(Ee=ge[1],(Ne+=E(Ee-ce))>ve||(Ee=ge[2],(Ne+=E(Ee-fe))>ve||(ve=Ne,ye=Le)))}return ye}(ie,ue,ne,W);ae[z]=he}}return ae},default:()=>F,nearestColor:()=>function(U,W,V=j){return U[M(U,W,V)]},nearestColorIndex:()=>M,nearestColorIndexWithDistance:()=>P,prequantize:()=>function(U,{roundRGB:W=5,roundAlpha:V=10,oneBitAlpha:H=null}={}){const Q=new Uint32Array(U.buffer);for(let G=0;G>24&255;var ee,ae=D>>16&255,k=D>>8&255,D=255&D;X=T(X,V),H&&(ee=typeof H=="number"?H:127,X=X<=ee?0:255),D=T(D,W),k=T(k,W),ae=T(ae,W),Q[G]=X<<24|ae<<16|k<<8|D<<0}},quantize:()=>function(U,W,V={}){var{format:H="rgb565",clearAlpha:Q=!0,clearAlphaColor:ee=0,clearAlphaThreshold:ae=0,oneBitAlpha:k=!1}=V;if(!U||!U.buffer)throw new Error("quantize() expected RGBA Uint8Array data");if(!(U instanceof Uint8Array||U instanceof Uint8ClampedArray))throw new Error("quantize() expected RGBA Uint8Array data");U=new Uint32Array(U.buffer);let D=V.useSqrt!==!1;const G=H==="rgba4444",X=function(Ie,Fe){const Ge=Fe==="rgb444"?4096:65536,we=new Array(Ge),Re=Ie.length;if(Fe==="rgba4444")for(let ot=0;ot>24&255,re=Z>>16&255,se=Z>>8&255,Z=255&Z,le=f(Z,se,re,Te);let Ze=le in we?we[le]:we[le]=v();Ze.rc+=Z,Ze.gc+=se,Ze.bc+=re,Ze.ac+=Te,Ze.cnt++}else if(Fe==="rgb444")for(let ot=0;ot>16&255,Se=Pe>>8&255,Pe=255&Pe,He=p(Pe,Se,xe);let Ze=He in we?we[He]:we[He]=v();Ze.rc+=Pe,Ze.gc+=Se,Ze.bc+=xe,Ze.cnt++}else for(let ot=0;ot>16&255,qe=$e>>8&255,$e=255&$e,lt=d($e,qe,We);let Ze=lt in we?we[lt]:we[lt]=v();Ze.rc+=$e,Ze.gc+=qe,Ze.bc+=We,Ze.cnt++}return we}(U,H),K=X.length,J=K-1,ne=new Uint32Array(K+1);for(var ue=0,ie=0;ie>1,!(X[he=ne[z]].err<=te));Y=z)ne[Y]=he;ne[Y]=ie}var ce,fe=ue-W;for(ie=0;ie=ce.mtm&&X[ce.nn].mtm<=ce.tm)break;for(ce.mtm==J?de=ne[1]=ne[ne[0]--]:(b(X,de),ce.tm=ie),te=X[de].err,Y=1;(z=Y+Y)<=ne[0]&&(zX[ne[z+1]].err&&z++,!(te<=X[he=ne[z]].err));Y=z)ne[Y]=he;ne[Y]=de}var ye=X[ce.nn],ve=ce.cnt,Ee=ye.cnt,ge=1/(ve+Ee);G&&(ce.ac=ge*(ve*ce.ac+Ee*ye.ac)),ce.rc=ge*(ve*ce.rc+Ee*ye.rc),ce.gc=ge*(ve*ce.gc+Ee*ye.gc),ce.bc=ge*(ve*ce.bc+Ee*ye.bc),ce.cnt+=ye.cnt,ce.mtm=++ie,X[ye.bk].fw=ye.fw,X[ye.fw].bk=ye.bk,ye.mtm=J}let Ne=[];for(ie=0;;0){let Ie=c(Math.round(X[ie].rc),0,255),Fe=c(Math.round(X[ie].gc),0,255),Ge=c(Math.round(X[ie].bc),0,255),we=255;G&&(we=c(Math.round(X[ie].ac),0,255),k&&(Le=typeof k=="number"?k:127,we=we<=Le?0:255),Q&&we<=ae&&(Ie=Fe=Ge=ee,we=0));var Le=G?[Ie,Fe,Ge,we]:[Ie,Fe,Ge];if(function(Re,Te){for(let Z=0;Zfunction(U,W,V=5){if(U.length&&W.length){var H=U.map(D=>D.slice(0,3)),Q=V*V,ee=U[0].length;for(let D=0;Dee?G.slice(0,3):G.slice();var k=P(H,G.slice(0,3),j),ae=k[0],k=k[1];0>>0),ee!=0&&(Q=Math.max(Q,256));const ae=V;V=new Uint8Array(Q),0>=8,K-=8;if((te>he||ue)&&(ue?(ie=ne,he=(1<>=8,K-=8;0>3}function f(U,W,V,H){return U>>4|240&W|(240&V)<<4|(240&H)<<8}function p(U,W,V){return U>>4<<8|240&W|V>>4}function c(U,W,V){return U>8&255)}function A(U,W){for(var V=0;V>1,p=-7,c=o?b-1:0,l=o?-1:1,b=s[u+c];for(c+=l,m=b&(1<<-p)-1,b>>=-p,p+=i;0>=-p,p+=h;0>1,c=y===23?Math.pow(2,-24)-Math.pow(2,-77):0,l=h?0:v-1,b=h?1:-1,v=u<0||u===0&&1/u<0?1:0;for(u=Math.abs(u),isNaN(u)||u===1/0?(i=isNaN(u)?1:0,n=f):(n=Math.floor(Math.log(u)/Math.LN2),u*(h=Math.pow(2,-n))<1&&(n--,h*=2),2<=(u+=1<=n+p?c/h:c*Math.pow(2,1-p))*h&&(n++,h/=2),f<=n+p?(i=0,n=f):1<=n+p?(i=(u*h-1)*Math.pow(2,y),n+=p):(i=u*Math.pow(2,p-1)*Math.pow(2,y),n=0));8<=y;s[o+l]=255&i,l+=b,i/=256,y-=8);for(n=n<Math.abs(re[0])&&(se=1),se=Math.abs(re[2])>Math.abs(re[se])?2:se}function I(re,se){re.f+=se.f,re.b.f+=se.b.f}function A(re,se,Z){return re=re.a,se=se.a,Z=Z.a,se.b.a===re?Z.b.a===re?o(se.a,Z.a)?y(Z.b.a,se.a,Z.a)<=0:0<=y(se.b.a,Z.a,se.a):y(Z.b.a,re,Z.a)<=0:Z.b.a===re?0<=y(se.b.a,re,se.a):(se=h(se.b.a,re,se.a),(re=h(Z.b.a,re,Z.a))<=se)}function N(re){re.a.i=null;var se=re.e;se.a.c=se.c,se.c.a=se.a,re.e=null}function F(re,se){c(re.a),re.c=!1,(re.a=se).i=re}function U(re){for(var se=re.a.a;(re=Te(re)).a.a===se;);return re.c&&(F(re,se=b(Re(re).a.b,re.a.e)),re=Te(re)),re}function W(re,se,Z){var le=new we;return le.a=Z,le.e=ue(re.f,se.e,le),Z.i=le}function V(re,se){switch(re.s){case 100130:return(1&se)!=0;case 100131:return se!==0;case 100132:return 0>1]],He[Pe[qe]])?Fe:Ge)(Z,qe),He[Se]=null,We[Se]=Z.b,Z.b=Se}else for(Z.c[-(Se+1)]=null;0Math.max(Pe.a,We.a))){if(o(Se,Pe)){if(0Z.f&&(Z.f*=2,Z.c=Ne(Z.c,Z.f+1)),Z.b===0?xe=le:(xe=Z.b,Z.b=Z.c[Z.b]),Z.e[xe]=se,Z.c[xe]=le,Z.d[le]=xe,Z.h&&Ge(Z,le),xe):(Z=re.a++,re.c[Z]=se,-(Z+1))}function Ee(re){if(re.a===0)return Ie(re.b);var se=re.c[re.d[re.a-1]];if(re.b.a!==0&&o(Le(re.b),se))return Ie(re.b);for(;--re.a,0re.a||o(le[Pe],le[We])){xe[Z[Se]=Pe]=Se;break}xe[Z[Se]=We]=Se,Se=He}}function Ge(re,se){for(var Z=re.d,le=re.e,xe=re.c,Se=se,Pe=Z[Se];;){var He=Se>>1,We=Z[He];if(He==0||o(le[We],le[Pe])){xe[Z[Se]=Pe]=Se;break}xe[Z[Se]=We]=Se,Se=He}}function we(){this.e=this.a=null,this.f=0,this.c=this.b=this.h=this.d=!1}function Re(re){return re.e.c.b}function Te(re){return re.e.a.b}(s=he.prototype).x=function(){Y(this,0)},s.B=function(re,se){switch(re){case 100142:return;case 100140:switch(se){case 100130:case 100131:case 100132:case 100133:case 100134:return void(this.s=se)}break;case 100141:return void(this.m=!!se);default:return void z(this,100900)}z(this,100901)},s.y=function(re){switch(re){case 100142:return 0;case 100140:return this.s;case 100141:return this.m;default:z(this,100900)}return!1},s.A=function(re,se,Z){this.j[0]=re,this.j[1]=se,this.j[2]=Z},s.z=function(re,se){var Z=se||null;switch(re){case 100100:case 100106:this.h=Z;break;case 100104:case 100110:this.l=Z;break;case 100101:case 100107:this.k=Z;break;case 100102:case 100108:this.i=Z;break;case 100103:case 100109:this.p=Z;break;case 100105:case 100111:this.o=Z;break;case 100112:this.r=Z;break;default:z(this,100900)}},s.C=function(re,se){var Z=!1,le=[0,0,0];Y(this,2);for(var xe=0;xe<3;++xe){var Se=re[xe];Se<-1e150&&(Se=-1e150,Z=!0),1e150le[qe]&&(le[qe]=$e,xe[qe]=Z)}if(le[1]-He[1]>le[Z=0]-He[0]&&(Z=1),He[Z=le[2]-He[2]>le[Z]-He[Z]?2:Z]>=le[Z])Pe[0]=0,Pe[1]=0,Pe[2]=1;else{for(He=We[Z],xe=xe[Z],We=[le=0,0,0],He=[He.g[0]-xe.g[0],He.g[1]-xe.g[1],He.g[2]-xe.g[2]],qe=[0,0,0],Z=Se.e;Z!==Se;Z=Z.e)qe[0]=Z.g[0]-xe.g[0],qe[1]=Z.g[1]-xe.g[1],qe[2]=Z.g[2]-xe.g[2],We[0]=He[1]*qe[2]-He[2]*qe[1],We[1]=He[2]*qe[0]-He[0]*qe[2],We[2]=He[0]*qe[1]-He[1]*qe[0],le<($e=We[0]*We[0]+We[1]*We[1]+We[2]*We[2])&&(le=$e,Pe[0]=We[0],Pe[1]=We[1],Pe[2]=We[2]);le<=0&&(Pe[0]=Pe[1]=Pe[2]=0,Pe[C(He)]=1)}Se=!0}for(We=C(Pe),Z=this.b.c,le=(We+1)%3,xe=(We+2)%3,We=0>=1;)++p;if(l=1<>8&255,o[n++]=255&y,o[n++]=y>>8&255,o[n++]=(d!==null?128:0)|p,o[n++]=c,o[n++]=0,d!==null)for(var b=0,v=d.length;b>16&255,o[n++]=j>>8&255,o[n++]=255&j}if(i!==null){if(i<0||65535>8&255,o[n++]=0}var T=!1;this.addFrame=function(E,M,P,L,C,I){if(T===!0&&(--n,T=!1),I=I===void 0?{}:I,E<0||M<0||65535>=1;)++F;var U=1<>8&255,o[n++]=Q,o[n++]=0),o[n++]=44,o[n++]=255&E,o[n++]=E>>8&255,o[n++]=255&M,o[n++]=M>>8&255,o[n++]=255&P,o[n++]=P>>8&255,o[n++]=255&L,o[n++]=L>>8&255,o[n++]=A===!0?128|F-1:0,A===!0)for(var ee=0,ae=N.length;ee>16&255,o[n++]=k>>8&255,o[n++]=255&k}return n=function(D,G,X,K){D[G++]=X;var J=G++,ne=1<>=8,z-=8,G===J+256&&(D[J]=255,J=G++)}function fe(Ie){te|=Ie<>=8,z-=8,G===J+256&&(D[J]=255,J=G++);he===4096?(fe(ne),he=1+ie,Y=X+1,ye={}):(1<>=p,l-=p,M==i)f=1+d,c=(1<<(p=n+1))-1,E=null;else{if(M==d)break;for(var P=M>8,++L;var I=C;if(m>=8;E!==null&&f<4096&&(T[f++]=E<<8|I,c+1<=f&&p<12&&(++p,c=c<<1|1)),E=M}}v!==m&&console.log("Warning, gif stream shorter than expected.")}try{g.GifWriter=s,g.GifReader=function(o){var h=0;if(o[h++]!==71||o[h++]!==73||o[h++]!==70||o[h++]!==56||(o[h++]+1&253)!=56||o[h++]!==97)throw new Error("Invalid GIF 87a/89a header.");var y=o[h++]|o[h++]<<8,m=o[h++]|o[h++]<<8,n=o[h++],i=1<<1+(7&n),d=(o[h++],o[h++],null),f=null,p=(n>>7&&(d=h,h+=3*(f=i)),!0),c=[],l=0,b=null,v=0,j=null;for(this.width=y,this.height=m;p&&h>2&7,h++;break;case 254:for(;;){if(!(0<=(E=o[h++])))throw Error("Invalid block size");if(E===0)break;h+=E}break;default:throw new Error("Unknown graphic control label: 0x"+o[h-1].toString(16))}break;case 44:var E,M=o[h++]|o[h++]<<8,P=o[h++]|o[h++]<<8,L=o[h++]|o[h++]<<8,C=o[h++]|o[h++]<<8,W=o[h++],I=W>>6&1,A=1<<1+(7&W),N=d,F=f,U=!1,W=(W>>7&&(U=!0,N=h,h+=3*(F=A)),h);for(h++;;){if(!(0<=(E=o[h++])))throw Error("Invalid block size");if(E===0)break;h+=E}c.push({x:M,y:P,width:L,height:C,has_local_palette:U,palette_offset:N,palette_size:F,data_offset:W,data_length:h-W,transparent_index:b,interlaced:!!I,delay:l,disposal:v});break;case 59:p=!1;break;default:throw new Error("Unknown gif block: 0x"+o[h-1].toString(16))}this.numFrames=function(){return c.length},this.loopCount=function(){return j},this.frameInfo=function(V){if(V<0||V>=c.length)throw new Error("Frame index out of range.");return c[V]},this.decodeAndBlitFrameBGRA=function(V,H){for(var V=this.frameInfo(V),Q=V.width*V.height,ee=new Uint8Array(Q),ae=(u(o,V.data_offset,ee,Q),V.palette_offset),k=V.transparent_index,D=(k===null&&(k=256),V.width),G=y-D,X=D,K=4*(V.y*y+V.x),J=4*((V.y+V.height)*y+V.x),ne=K,ue=4*G,ie=(V.interlaced===!0&&(ue+=4*y*7),8),he=0,Y=ee.length;he>=1)),ce===k?ne+=4:(z=o[ae+3*ce],te=o[ae+3*ce+1],ce=o[ae+3*ce+2],H[ne++]=ce,H[ne++]=te,H[ne++]=z,H[ne++]=255),--X}},this.decodeAndBlitFrameRGBA=function(V,H){for(var V=this.frameInfo(V),Q=V.width*V.height,ee=new Uint8Array(Q),ae=(u(o,V.data_offset,ee,Q),V.palette_offset),k=V.transparent_index,D=(k===null&&(k=256),V.width),G=y-D,X=D,K=4*(V.y*y+V.x),J=4*((V.y+V.height)*y+V.x),ne=K,ue=4*G,ie=(V.interlaced===!0&&(ue+=4*y*7),8),he=0,Y=ee.length;he>=1)),ce===k?ne+=4:(z=o[ae+3*ce],te=o[ae+3*ce+1],ce=o[ae+3*ce+2],H[ne++]=z,H[ne++]=te,H[ne++]=ce,H[ne++]=255),--X}}}}catch{}},{}],257:[function(t,w,g){(function(s){var u,o;u=this,o=function(h){function y(B){if(this==null)throw TypeError();var S,O=String(this),R=O.length,B=B?Number(B):0;if(!((B=B!=B?0:B)<0||R<=B))return 55296<=(S=O.charCodeAt(B))&&S<=56319&&B+1>>16-S;return _.tag>>>=S,_.bitcount-=S,R+O}function A(_,S){for(;_.bitcount<24;)_.tag|=_.source[_.sourceIndex++]<<_.bitcount,_.bitcount+=8;for(var O=0,R=0,B=0,q=_.tag;R=2*R+(1&q),q>>>=1,O+=S.table[++B],0<=(R-=S.table[B]););return _.tag=q,_.bitcount-=B,S.trans[O+R]}function N(_,S,O){for(;;){var R=A(_,S);if(R===256)return n;if(R<256)_.dest[_.destLen++]=R;else for(var B,q=I(_,l[R-=257],b[R]),R=A(_,O),$=B=_.destLen-I(_,v[R],j[R]);$>>=1,R=B,I(q,2,0)){case 0:O=function(Oe){for(var Je,ut;8this.x2&&(this.x2=_)),typeof S=="number"&&((isNaN(this.y1)||isNaN(this.y2))&&(this.y1=S,this.y2=S),Sthis.y2&&(this.y2=S))},Q.prototype.addX=function(_){this.addPoint(_,null)},Q.prototype.addY=function(_){this.addPoint(null,_)},Q.prototype.addBezier=function(_,S,O,R,B,q,$,oe){var me=[_,S],pe=[O,R],je=[B,q],Me=[$,oe];this.addPoint(_,S),this.addPoint($,oe);for(var ke=0;ke<=1;ke++){var be,Ue=6*me[ke]-12*pe[ke]+6*je[ke],Ae=-3*me[ke]+9*pe[ke]-9*je[ke]+3*Me[ke],De=3*pe[ke]-3*me[ke];Ae==0?Ue==0||0<(be=-De/Ue)&&be<1&&(ke===0&&this.addX(H(me[ke],pe[ke],je[ke],Me[ke],be)),ke===1&&this.addY(H(me[ke],pe[ke],je[ke],Me[ke],be))):(be=Math.pow(Ue,2)-4*De*Ae)<0||(0<(De=(-Ue+Math.sqrt(be))/(2*Ae))&&De<1&&(ke===0&&this.addX(H(me[ke],pe[ke],je[ke],Me[ke],De)),ke===1&&this.addY(H(me[ke],pe[ke],je[ke],Me[ke],De))),0<(De=(-Ue-Math.sqrt(be))/(2*Ae))&&De<1&&(ke===0&&this.addX(H(me[ke],pe[ke],je[ke],Me[ke],De)),ke===1&&this.addY(H(me[ke],pe[ke],je[ke],Me[ke],De))))}},Q.prototype.addQuad=function(_,S,O,R,B,q){O=_+2/3*(O-_),R=S+2/3*(R-S),this.addBezier(_,S,O,R,O+1/3*(B-_),R+1/3*(q-S),B,q)},ee.prototype.moveTo=function(_,S){this.commands.push({type:"M",x:_,y:S})},ee.prototype.lineTo=function(_,S){this.commands.push({type:"L",x:_,y:S})},ee.prototype.curveTo=ee.prototype.bezierCurveTo=function(_,S,O,R,B,q){this.commands.push({type:"C",x1:_,y1:S,x2:O,y2:R,x:B,y:q})},ee.prototype.quadTo=ee.prototype.quadraticCurveTo=function(_,S,O,R){this.commands.push({type:"Q",x1:_,y1:S,x:O,y:R})},ee.prototype.close=ee.prototype.closePath=function(){this.commands.push({type:"Z"})},ee.prototype.extend=function(_){var S;if(_.commands)_=_.commands;else if(_ instanceof Q)return S=_,this.moveTo(S.x1,S.y1),this.lineTo(S.x2,S.y1),this.lineTo(S.x2,S.y2),this.lineTo(S.x1,S.y2),void this.close();Array.prototype.push.apply(this.commands,_)},ee.prototype.getBoundingBox=function(){for(var _=new Q,S=0,O=0,R=0,B=0,q=0;q>8&255,255&_]},K.USHORT=J(2),X.SHORT=function(_){return[(_=32768<=_?-(65536-_):_)>>8&255,255&_]},K.SHORT=J(2),X.UINT24=function(_){return[_>>16&255,_>>8&255,255&_]},K.UINT24=J(3),X.ULONG=function(_){return[_>>24&255,_>>16&255,_>>8&255,255&_]},K.ULONG=J(4),X.LONG=function(_){return[(_=2147483648<=_?-(4294967296-_):_)>>24&255,_>>16&255,_>>8&255,255&_]},K.LONG=J(4),X.FIXED=X.ULONG,K.FIXED=K.ULONG,X.FWORD=X.SHORT,K.FWORD=K.SHORT,X.UFWORD=X.USHORT,K.UFWORD=K.USHORT,X.LONGDATETIME=function(_){return[0,0,0,0,_>>24&255,_>>16&255,_>>8&255,255&_]},K.LONGDATETIME=J(8),X.TAG=function(_){return D.argument(_.length===4,"Tag should be exactly 4 ASCII characters."),[_.charCodeAt(0),_.charCodeAt(1),_.charCodeAt(2),_.charCodeAt(3)]},K.TAG=J(4),X.Card8=X.BYTE,K.Card8=K.BYTE,X.Card16=X.USHORT,K.Card16=K.USHORT,X.OffSize=X.BYTE,K.OffSize=K.BYTE,X.SID=X.USHORT,K.SID=K.USHORT,X.NUMBER=function(_){return-107<=_&&_<=107?[_+139]:108<=_&&_<=1131?[247+((_-=108)>>8),255&_]:-1131<=_&&_<=-108?[251+((_=-_-108)>>8),255&_]:-32768<=_&&_<=32767?X.NUMBER16(_):X.NUMBER32(_)},K.NUMBER=function(_){return X.NUMBER(_).length},X.NUMBER16=function(_){return[28,_>>8&255,255&_]},K.NUMBER16=J(3),X.NUMBER32=function(_){return[29,_>>24&255,_>>16&255,_>>8&255,255&_]},K.NUMBER32=J(5),X.REAL=function(_){for(var S=_.toString(),O=/\.(\d*?)(?:9{5,20}|0{5,20})\d{0,2}(?:e(.+)|$)/.exec(S),R=(O&&(O=parseFloat("1e"+((O[2]?+O[2]:0)+O[1].length)),S=(Math.round(_*O)/O).toString()),""),B=0,q=S.length;B>8&255,S[S.length]=255&R}return S},K.UTF16=function(_){return 2*_.length};var ne,ue={"x-mac-croatian":"ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®Š™´¨≠ŽØ∞±≤≥∆µ∂∑∏š∫ªºΩžø¿¡¬√ƒ≈Ć«Č… ÀÃÕŒœĐ—“”‘’÷◊©⁄€‹›Æ»–·‚„‰ÂćÁčÈÍÎÏÌÓÔđÒÚÛÙıˆ˜¯πË˚¸Êæˇ","x-mac-cyrillic":"АБВГДЕЖЗИЙКЛМНОПРСТУФХЦЧШЩЪЫЬЭЮЯ†°Ґ£§•¶І®©™Ђђ≠Ѓѓ∞±≤≥іµґЈЄєЇїЉљЊњјЅ¬√ƒ≈∆«»… ЋћЌќѕ–—“”‘’÷„ЎўЏџ№Ёёяабвгдежзийклмнопрстуфхцчшщъыьэю","x-mac-gaelic":"ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ÆØḂ±≤≥ḃĊċḊḋḞḟĠġṀæøṁṖṗɼƒſṠ«»… ÀÃÕŒœ–—“”‘’ṡẛÿŸṪ€‹›Ŷŷṫ·Ỳỳ⁊ÂÊÁËÈÍÎÏÌÓÔ♣ÒÚÛÙıÝýŴŵẄẅẀẁẂẃ","x-mac-greek":"Ĺ²É³ÖÜ΅àâä΄¨çéèê룙î‰ôö¦€ùûü†ΓΔΘΛΞΠß®©ΣΪ§≠°·Α±≤≥¥ΒΕΖΗΙΚΜΦΫΨΩάΝ¬ΟΡ≈Τ«»… ΥΧΆΈœ–―“”‘’÷ΉΊΌΎέήίόΏύαβψδεφγηιξκλμνοπώρστθωςχυζϊϋΐΰ­","x-mac-icelandic":"ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûüÝ°¢£§•¶ß®©™´¨≠ÆØ∞±≤≥¥µ∂∑∏π∫ªºΩæø¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸ⁄€ÐðÞþý·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙıˆ˜¯˘˙˚¸˝˛ˇ","x-mac-inuit":"ᐃᐄᐅᐆᐊᐋᐱᐲᐳᐴᐸᐹᑉᑎᑏᑐᑑᑕᑖᑦᑭᑮᑯᑰᑲᑳᒃᒋᒌᒍᒎᒐᒑ°ᒡᒥᒦ•¶ᒧ®©™ᒨᒪᒫᒻᓂᓃᓄᓅᓇᓈᓐᓯᓰᓱᓲᓴᓵᔅᓕᓖᓗᓘᓚᓛᓪᔨᔩᔪᔫᔭ… ᔮᔾᕕᕖᕗ–—“”‘’ᕘᕙᕚᕝᕆᕇᕈᕉᕋᕌᕐᕿᖀᖁᖂᖃᖄᖅᖏᖐᖑᖒᖓᖔᖕᙱᙲᙳᙴᙵᙶᖖᖠᖡᖢᖣᖤᖥᖦᕼŁł","x-mac-ce":"ÄĀāÉĄÖÜáąČäčĆć鏟ĎíďĒēĖóėôöõúĚěü†°Ę£§•¶ß®©™ę¨≠ģĮįĪ≤≥īĶ∂∑łĻļĽľĹĺŅņѬ√ńŇ∆«»… ňŐÕőŌ–—“”‘’÷◊ōŔŕŘ‹›řŖŗŠ‚„šŚśÁŤťÍŽžŪÓÔūŮÚůŰűŲųÝýķŻŁżĢˇ",macintosh:"ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ÆØ∞±≤≥¥µ∂∑∏π∫ªºΩæø¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸ⁄€‹›fifl‡·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙıˆ˜¯˘˙˚¸˝˛ˇ","x-mac-romanian":"ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ĂȘ∞±≤≥¥µ∂∑∏π∫ªºΩăș¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸ⁄€‹›Țț‡·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙıˆ˜¯˘˙˚¸˝˛ˇ","x-mac-turkish":"ÄÅÇÉÑÖÜáàâäãåçéèêëíìîïñóòôöõúùûü†°¢£§•¶ß®©™´¨≠ÆØ∞±≤≥¥µ∂∑∏π∫ªºΩæø¿¡¬√ƒ≈∆«»… ÀÃÕŒœ–—“”‘’÷◊ÿŸĞğİıŞş‡·‚„‰ÂÊÁËÈÍÎÏÌÓÔÒÚÛÙˆ˜¯˘˙˚¸˝˛ˇ"},ie=(G.MACSTRING=function(_,S,O,R){var B=ue[R];if(B!==void 0){for(var q="",$=0;$>8&255,ke+256&255)}return pe})(_,R,S);return S},X.INDEX=function(_){for(var S=1,O=[S],R=[],B=0;B<_.length;B+=1){var q=X.OBJECT(_[B]);Array.prototype.push.apply(R,q),S+=q.length,O.push(S)}if(R.length===0)return[0,0];for(var $=[],oe=1+Math.floor(Math.log(S)/Math.log(2))/8|0,me=[void 0,X.BYTE,X.USHORT,X.UINT24,X.ULONG][oe],pe=0;pe>8,S[je+1]=255&Me,S=S.concat(R[pe])}return S},K.TABLE=function(_){for(var S=0,O=_.fields.length,R=0;R>1,oe.skip("uShort",3),De.glyphIndexMap={};for(var ut,et=new Te.Parser(Ce,Oe+Je+14),tt=new Te.Parser(Ce,Oe+Je+16+2*ut),ct=new Te.Parser(Ce,Oe+Je+16+4*ut),Ot=new Te.Parser(Ce,Oe+Je+16+6*ut),Vt=Oe+Je+16+8*ut,Xt=0;Xt>4,$=15&$;if(q==15||(R+=B[q],$==15))break;R+=B[$]}return parseFloat(R)}if(32<=S&&S<=246)return S-139;if(247<=S&&S<=250)return 256*(S-247)+_.parseByte()+108;if(251<=S&&S<=254)return 256*-(S-251)-_.parseByte()-108;throw new Error("Invalid b0 "+S)}function dr(_,S,O){var R=new Te.Parser(_,S=S!==void 0?S:0),B=[],q=[];for(O=O!==void 0?O:_.length;R.relativeOffset>1,be.length=0,Ae=!0}return function tt(ct){for(var Ot,Vt,Xt,fr,hr,Ur,kt,Rt,wt,pr,Ft,ar,It=0;ItMath.abs(ar-Oe)?Ce=Ft+be.shift():Oe=ar+be.shift(),ke.curveTo(R,B,q,$,kt,Rt),ke.curveTo(wt,pr,Ft,ar,Ce,Oe);break;default:console.log("Glyph "+S.index+": unknown operator 1200"+nr),be.length=0}break;case 14:0>3;break;case 21:2>16),It+=2;break;case 29:hr=be.pop()+_.gsubrsBias,(Ur=_.gsubrs[hr])&&tt(Ur);break;case 30:for(;0=O.begin&&_=xe.length&&(q=R.parseChar(),O.names.push(R.parseString(q)));break;case 2.5:O.numberOfGlyphs=R.parseUShort(),O.offset=new Array(O.numberOfGlyphs);for(var oe=0;oeMe.value.tag?1:-1}),S.fields=S.fields.concat(R),S.fields=S.fields.concat(B),S}function Cl(_,S,O){for(var R=0;R 123 are reserved for internal usage");be|=1<>>1,q=_[B].tag;if(q===S)return B;q>>1,q=_[B];if(q===S)return B;q>>1,$=(B=_[q]).start;if($===S)return B;$(B=_[O-1]).end?0:B}function hs(_,S){this.font=_,this.tableName=S}function ps(_){hs.call(this,_,"gpos")}function $r(_){hs.call(this,_,"gsub")}function Al(_,S,O){for(var R=_.subtables,B=0;BS.points.length-1||R.matchedPoints[1]>B.points.length-1)throw Error("Matched points out of range in "+S.name);var $=S.points[R.matchedPoints[0]],oe=B.points[R.matchedPoints[1]],R={xScale:R.xScale,scale01:R.scale01,scale10:R.scale10,yScale:R.yScale,dx:0,dy:0},oe=Si([oe],R)[0];R.dx=$.x-oe.x,R.dy=$.y-oe.y,q=Si(B.points,R)}S.points=S.points.concat(q)}}return Il(S.points)}(ps.prototype=hs.prototype={searchTag:ji,binSearch:Ol,getTable:function(_){var S=this.font.tables[this.tableName];return S=!S&&_?this.font.tables[this.tableName]=this.createDefaultTable():S},getScriptNames:function(){var _=this.getTable();return _?_.scripts.map(function(S){return S.tag}):[]},getDefaultScriptName:function(){var _=this.getTable();if(_){for(var S=!1,O=0;O<_.scripts.length;O++){var R=_.scripts[O].tag;if(R==="DFLT")return R;R==="latn"&&(S=!0)}return S?"latn":void 0}},getScriptTable:function(_,S){var O,R=this.getTable(S);if(R)return O=R.scripts,0<=(R=ji(R.scripts,_=_||"DFLT"))?O[R].script:S?(O.splice(-1-R,0,S={tag:_,script:{defaultLangSys:{reserved:0,reqFeatureIndex:65535,featureIndexes:[]},langSysRecords:[]}}),S.script):void 0},getLangSysTable:function(B,S,O){var R,B=this.getScriptTable(B,O);if(B)return S&&S!=="dflt"&&S!=="DFLT"?0<=(R=ji(B.langSysRecords,S))?B.langSysRecords[R].langSys:O?(B.langSysRecords.splice(-1-R,0,O={tag:S,langSys:{reserved:0,reqFeatureIndex:65535,featureIndexes:[]}}),O.langSys):void 0:B.defaultLangSys},getFeatureTable:function(_,S,O,R){if(_=this.getLangSysTable(_,S,R),_){for(var B,q=_.featureIndexes,$=this.font.tables[this.tableName].features,oe=0;oe=$[S-1].tag,"Features must be added in alphabetical order."),$.push(B={tag:O,feature:{params:0,lookupListIndexes:[]}}),q.push(S),B.feature}},getLookupTables:function(q,S,O,R,B){var q=this.getFeatureTable(q,S,O,B),$=[];if(q){for(var oe,me=q.lookupListIndexes,pe=this.font.tables[this.tableName].lookups,je=0;je",q),S.stack.push(Math.round(64*q))}function Ti(_,S){var O=S.stack,R=O.pop(),B=S.fv,q=S.pv,$=S.ppem,oe=S.deltaBase+16*(_-1),me=S.deltaShift,pe=S.z0;h.DEBUG&&console.log(S.step,"DELTAP["+_+"]",R,O);for(var je=0;je>4)===$&&(0<=(ke=(15&ke)-8)&&ke++,h.DEBUG&&console.log(S.step,"DELTAPFIX",Me,"by",ke*me),Me=pe[Me],B.setRelative(Me,Me,ke*me,q))}}function ys(_,S){var O=S.stack,R=O.pop();h.DEBUG&&console.log(S.step,"ROUND[]"),O.push(64*S.round(R/64))}function ki(_,S){var O=S.stack,R=O.pop(),B=S.ppem,q=S.deltaBase+16*(_-1),$=S.deltaShift;h.DEBUG&&console.log(S.step,"DELTAC["+_+"]",R,O);for(var oe=0;oe>4)===B&&(0<=(pe=(15&pe)-8)&&pe++,pe=pe*$,h.DEBUG&&console.log(S.step,"DELTACFIX",me,"by",pe),S.cvt[me]+=pe)}}function tu(_,S){var O,B=S.stack,R=B.pop(),B=B.pop(),q=S.z2[R],$=S.z1[B];h.DEBUG&&console.log(S.step,"SDPVTL["+_+"]",R,B),R=_?(O=q.y-$.y,$.x-q.x):(O=$.x-q.x,$.y-q.y),S.dpv=Po(O,R)}function yn(_,S){var O=S.stack,R=S.prog,B=S.ip;h.DEBUG&&console.log(S.step,"PUSHB["+_+"]");for(var q=0;q<_;q++)O.push(R[++B]);S.ip=B}function gn(_,S){var O=S.ip,R=S.prog,B=S.stack;h.DEBUG&&console.log(S.ip,"PUSHW["+_+"]");for(var q=0;q<_;q++){var $=R[++O]<<8|R[++O];32768&$&&($=-(1+(65535^$))),B.push($)}S.ip=O}function Qe(_,S,O,R,B,q){var $,oe,pe=q.stack,me=_&&pe.pop(),pe=pe.pop(),je=q.rp0,je=q.z0[je],Me=q.z1[pe],ke=q.minDis,be=q.fv,Ue=q.dpv,Ae=$=Ue.distance(Me,je,!0,!0),De=0<=Ae?1:-1;Ae=Math.abs(Ae),_&&(oe=q.cvt[me],R&&Math.abs(Ae-oe)":"_")+(R?"R":"_")+(B===0?"Gr":B===1?"Bl":B===2?"Wh":"")+"]",_?me+"("+q.cvt[me]+","+oe+")":"",pe,"(d =",$,"->",De*Ae,")"),q.rp1=q.rp0,q.rp2=pe,S&&(q.rp0=pe)}Fl.prototype.exec=function(_,S){if(typeof S!="number")throw new Error("Point size is not a number!");if(!(2",R),oe.interpolate(Me,q,$,me),oe.touch(Me)}_.loop=1},Zl.bind(void 0,0),Zl.bind(void 0,1),function(_){for(var S=_.stack,O=_.rp0,R=_.z0[O],B=_.loop,q=_.fv,$=_.pv,oe=_.z1;B--;){var me=S.pop(),pe=oe[me];h.DEBUG&&console.log(_.step,(1<_.loop?"loop "+(_.loop-B)+": ":"")+"ALIGNRP[]",me),q.setRelative(pe,R,0,$),q.touch(pe)}_.loop=1},function(_){h.DEBUG&&console.log(_.step,"RTDG[]"),_.round=Eh},Ql.bind(void 0,0),Ql.bind(void 0,1),function(_){var S=_.prog,O=_.ip,R=_.stack,B=S[++O];h.DEBUG&&console.log(_.step,"NPUSHB[]",B);for(var q=0;q"u"?Nh:Uh)(_,function(O,R){if(O)return S(O);var B;try{B=Mi(R)}catch(q){return S(q,null)}return S(null,B)})},h.loadSync=function(_){return Mi(Ll(t("fs").readFileSync(_)))},Object.defineProperty(h,"__esModule",{value:!0})},o(typeof g=="object"&&w!==void 0?g:u.opentype={})}).call(this,t("buffer").Buffer)},{buffer:4,fs:2}],258:[function(t,w,g){(function(s){function u(y,m){for(var n=0,i=y.length-1;0<=i;i--){var d=y[i];d==="."?y.splice(i,1):d===".."?(y.splice(i,1),n++):n&&(y.splice(i,1),n--)}if(m)for(;n--;)y.unshift("..");return y}function o(y,m){if(y.filter)return y.filter(m);for(var n=[],i=0;i'.concat(l,"").concat(f,""),this.dummyDOM||(this.dummyDOM=document.getElementById(c).parentNode),this.descriptions?this.descriptions.fallbackElements||(this.descriptions.fallbackElements={}):this.descriptions={fallbackElements:{}},this.descriptions.fallbackElements[d]?this.descriptions.fallbackElements[d].innerHTML!==l&&(this.descriptions.fallbackElements[d].innerHTML=l):this._describeElementHTML("fallback",d,l),p===this.LABEL&&(this.descriptions.labelElements||(this.descriptions.labelElements={}),this.descriptions.labelElements[d]?this.descriptions.labelElements[d].innerHTML!==l&&(this.descriptions.labelElements[d].innerHTML=l):this._describeElementHTML("label",d,l)))},s.default.prototype._describeHTML=function(d,f){var p,c=this.canvas.id;d==="fallback"?(this.dummyDOM.querySelector("#".concat(c+u))?this.dummyDOM.querySelector("#"+c+h).insertAdjacentHTML("beforebegin",'

')):(p='

'),this.dummyDOM.querySelector("#".concat(c,"accessibleOutput"))?this.dummyDOM.querySelector("#".concat(c,"accessibleOutput")).insertAdjacentHTML("beforebegin",p):this.dummyDOM.querySelector("#".concat(c)).innerHTML=p),this.descriptions.fallback=this.dummyDOM.querySelector("#".concat(c).concat(o)),this.descriptions.fallback.innerHTML=f):d==="label"&&(this.dummyDOM.querySelector("#".concat(c+y))?this.dummyDOM.querySelector("#".concat(c+n))&&this.dummyDOM.querySelector("#".concat(c+n)).insertAdjacentHTML("beforebegin",'

')):(p='

'),this.dummyDOM.querySelector("#".concat(c,"accessibleOutputLabel"))?this.dummyDOM.querySelector("#".concat(c,"accessibleOutputLabel")).insertAdjacentHTML("beforebegin",p):this.dummyDOM.querySelector("#"+c).insertAdjacentHTML("afterend",p)),this.descriptions.label=this.dummyDOM.querySelector("#"+c+m),this.descriptions.label.innerHTML=f)},s.default.prototype._describeElementHTML=function(d,f,p){var c,l=this.canvas.id;d==="fallback"?(this.dummyDOM.querySelector("#".concat(l+u))?this.dummyDOM.querySelector("#"+l+h)||this.dummyDOM.querySelector("#"+l+o).insertAdjacentHTML("afterend",'
Canvas elements and their descriptions
')):(c='
Canvas elements and their descriptions
'),this.dummyDOM.querySelector("#".concat(l,"accessibleOutput"))?this.dummyDOM.querySelector("#".concat(l,"accessibleOutput")).insertAdjacentHTML("beforebegin",c):this.dummyDOM.querySelector("#"+l).innerHTML=c),(c=document.createElement("tr")).id=l+"_fte_"+f,this.dummyDOM.querySelector("#"+l+h).appendChild(c),this.descriptions.fallbackElements[f]=this.dummyDOM.querySelector("#".concat(l).concat("_fte_").concat(f)),this.descriptions.fallbackElements[f].innerHTML=p):d==="label"&&(this.dummyDOM.querySelector("#".concat(l+y))?this.dummyDOM.querySelector("#".concat(l+n))||this.dummyDOM.querySelector("#"+l+m).insertAdjacentHTML("afterend",'
')):(c='
'),this.dummyDOM.querySelector("#".concat(l,"accessibleOutputLabel"))?this.dummyDOM.querySelector("#".concat(l,"accessibleOutputLabel")).insertAdjacentHTML("beforebegin",c):this.dummyDOM.querySelector("#"+l).insertAdjacentHTML("afterend",c)),(d=document.createElement("tr")).id=l+"_lte_"+f,this.dummyDOM.querySelector("#"+l+n).appendChild(d),this.descriptions.labelElements[f]=this.dummyDOM.querySelector("#".concat(l).concat("_lte_").concat(f)),this.descriptions.labelElements[f].innerHTML=p)},t=s.default,g.default=t},{"../core/main":283,"core-js/modules/es.array.concat":153,"core-js/modules/es.regexp.exec":195,"core-js/modules/es.string.ends-with":198,"core-js/modules/es.string.replace":204}],264:[function(t,w,g){t("core-js/modules/es.array.concat"),t("core-js/modules/es.array.from"),t("core-js/modules/es.array.map"),t("core-js/modules/es.string.iterator"),t("core-js/modules/es.array.concat"),t("core-js/modules/es.array.from"),t("core-js/modules/es.array.map"),t("core-js/modules/es.string.iterator"),Object.defineProperty(g,"__esModule",{value:!0}),g.default=void 0,t=(t=t("../core/main"))&&t.__esModule?t:{default:t},t.default.prototype._updateGridOutput=function(s){var u,o,h,y;this.dummyDOM.querySelector("#".concat(s,"_summary"))&&(u=this._accessibleOutputs[s],h=function(m,n,i,d){return n="".concat(n," canvas, ").concat(i," by ").concat(d," pixels, contains ").concat(m[0]),n=(m[0]===1?"".concat(n," shape: "):"".concat(n," shapes: ")).concat(m[1]),n}((o=function(m,n){var i,d="",f="",p=0;for(i in n){var c,l=0;for(c in n[i]){var b='
  • ').concat(n[i][c].color," ").concat(i,",");i==="line"?b+=" location = ".concat(n[i][c].pos,", length = ").concat(n[i][c].length," pixels"):(b+=" location = ".concat(n[i][c].pos),i!=="point"&&(b+=", area = ".concat(n[i][c].area," %")),b+="
  • "),d+=b,l++,p++}f=1').concat(n[i][l].color," ").concat(i,"
    "):'').concat(n[i][l].color," ").concat(i," midpoint"),n[i][l].loc.locY";p=p+j+""}return p}(s,this.ingredients.shapes),h!==u.summary.innerHTML&&(u.summary.innerHTML=h),y!==u.map.innerHTML&&(u.map.innerHTML=y),o.details!==u.shapeDetails.innerHTML&&(u.shapeDetails.innerHTML=o.details),this._accessibleOutputs[s]=u)},t=t.default,g.default=t},{"../core/main":283,"core-js/modules/es.array.concat":153,"core-js/modules/es.array.from":162,"core-js/modules/es.array.map":168,"core-js/modules/es.string.iterator":200}],265:[function(t,w,g){t("core-js/modules/es.symbol"),t("core-js/modules/es.symbol.description"),t("core-js/modules/es.symbol.iterator"),t("core-js/modules/es.array.concat"),t("core-js/modules/es.array.fill"),t("core-js/modules/es.array.from"),t("core-js/modules/es.array.iterator"),t("core-js/modules/es.array.map"),t("core-js/modules/es.number.to-fixed"),t("core-js/modules/es.object.to-string"),t("core-js/modules/es.regexp.to-string"),t("core-js/modules/es.string.iterator"),t("core-js/modules/web.dom-collections.iterator"),t("core-js/modules/es.array.concat"),t("core-js/modules/es.array.fill"),t("core-js/modules/es.array.map"),t("core-js/modules/es.number.to-fixed"),Object.defineProperty(g,"__esModule",{value:!0}),g.default=void 0;var s=(t=t("../core/main"))&&t.__esModule?t:{default:t};function u(h){return function(y){if(Array.isArray(y)){for(var m=0,n=new Array(y.length);m')):this.dummyDOM.querySelector("#".concat(d)).innerHTML='
    '))):y==="Label"&&(m=d+h+(f=y),this.dummyDOM.querySelector("#".concat(n=d+"accessibleOutput"+y))||(this.dummyDOM.querySelector("#".concat(d,"_Label"))?this.dummyDOM.querySelector("#".concat(d,"_Label")):this.dummyDOM.querySelector("#".concat(d))).insertAdjacentHTML("afterend",'
    '))),this._accessibleOutputs[m]={},h==="textOutput"?(f="#".concat(d,"gridOutput").concat(f),i='
    Text Output

      '),this.dummyDOM.querySelector(f)?this.dummyDOM.querySelector(f).insertAdjacentHTML("beforebegin",i):this.dummyDOM.querySelector("#".concat(n)).innerHTML=i,this._accessibleOutputs[m].list=this.dummyDOM.querySelector("#".concat(m,"_list"))):h==="gridOutput"&&(f="#".concat(d,"textOutput").concat(f),i='
      Grid Output

        '),this.dummyDOM.querySelector(f)?this.dummyDOM.querySelector(f).insertAdjacentHTML("afterend",i):this.dummyDOM.querySelector("#".concat(n)).innerHTML=i,this._accessibleOutputs[m].map=this.dummyDOM.querySelector("#".concat(m,"_map"))),this._accessibleOutputs[m].shapeDetails=this.dummyDOM.querySelector("#".concat(m,"_shapeDetails")),this._accessibleOutputs[m].summary=this.dummyDOM.querySelector("#".concat(m,"_summary"))},s.default.prototype._updateAccsOutput=function(){var h=this.canvas.id;JSON.stringify(this.ingredients.shapes)===this.ingredients.pShapes&&this.ingredients.colors.background===this.ingredients.pBackground||(this.ingredients.pShapes=JSON.stringify(this.ingredients.shapes),this._accessibleOutputs.text&&this._updateTextOutput(h+"textOutput"),this._accessibleOutputs.grid&&this._updateGridOutput(h+"gridOutput"),this._accessibleOutputs.textLabel&&this._updateTextOutput(h+"textOutputLabel"),this._accessibleOutputs.gridLabel&&this._updateGridOutput(h+"gridOutputLabel"))},s.default.prototype._accsBackground=function(h){this.ingredients.pShapes=JSON.stringify(this.ingredients.shapes),this.ingredients.pBackground=this.ingredients.colors.background,this.ingredients.shapes={},this.ingredients.colors.backgroundRGBA!==h&&(this.ingredients.colors.backgroundRGBA=h,this.ingredients.colors.background=this._rgbColorName(h))},s.default.prototype._accsCanvasColors=function(h,y){h==="fill"?this.ingredients.colors.fillRGBA!==y&&(this.ingredients.colors.fillRGBA=y,this.ingredients.colors.fill=this._rgbColorName(y)):h==="stroke"&&this.ingredients.colors.strokeRGBA!==y&&(this.ingredients.colors.strokeRGBA=y,this.ingredients.colors.stroke=this._rgbColorName(y))},s.default.prototype._accsOutput=function(h,y){h==="ellipse"&&y[2]===y[3]?h="circle":h==="rectangle"&&y[2]===y[3]&&(h="square");var m,n,i={},d=!0,f=function(c,l){var b;return c=c==="rectangle"||c==="ellipse"||c==="arc"||c==="circle"||c==="square"?(b=Math.round(l[0]+l[2]/2),Math.round(l[1]+l[3]/2)):c==="triangle"?(b=(l[0]+l[2]+l[4])/3,(l[1]+l[3]+l[5])/3):c==="quadrilateral"?(b=(l[0]+l[2]+l[4]+l[6])/4,(l[1]+l[3]+l[5]+l[7])/4):c==="line"?(b=(l[0]+l[2])/2,(l[1]+l[3])/2):(b=l[0],l[1]),[b,c]}(h,y);if(h==="line"?(i.color=this.ingredients.colors.stroke,i.length=Math.round(this.dist(y[0],y[1],y[2],y[3])),m=this._getPos(y[0],[1]),n=this._getPos(y[2],[3]),i.loc=o(f,this.width,this.height),i.pos=m===n?"at ".concat(m):"from ".concat(m," to ").concat(n)):(h==="point"?i.color=this.ingredients.colors.stroke:(i.color=this.ingredients.colors.fill,i.area=this._getArea(h,y)),i.pos=this._getPos.apply(this,u(f)),i.loc=o(f,this.width,this.height)),this.ingredients.shapes[h]){if(this.ingredients.shapes[h]!==[i]){for(var p in this.ingredients.shapes[h])JSON.stringify(this.ingredients.shapes[h][p])===JSON.stringify(i)&&(d=!1);d===!0&&this.ingredients.shapes[h].push(i)}}else this.ingredients.shapes[h]=[i]},s.default.prototype._getPos=function(n,m){var n=new DOMPointReadOnly(n,m),m=this._renderer.isP3D?new DOMMatrix(this._renderer.uMVMatrix.mat4):this.drawingContext.getTransform(),n=n.matrixTransform(m),m=n.x,n=n.y,i=this.width*this._pixelDensity,d=this.height*this._pixelDensity;return m<.4*i?n<.4*d?"top left":.6*dMath.PI?i+=n:i-=n)):h==="ellipse"||h==="circle"?i=3.14*y[2]/2*y[3]/2:h==="line"||h==="point"?i=0:h==="quadrilateral"?i=Math.abs((y[6]+y[0])*(y[7]-y[1])+(y[0]+y[2])*(y[1]-y[3])+(y[2]+y[4])*(y[3]-y[5])+(y[4]+y[6])*(y[5]-y[7]))/2:h==="rectangle"||h==="square"?i=y[2]*y[3]:h==="triangle"&&(i=Math.abs(y[0]*(y[3]-y[5])+y[2]*(y[5]-y[1])+y[4]*(y[1]-y[3]))/2),this.width*this._pixelDensity),f=this.height*this._pixelDensity,p=[new DOMPoint(0,0),new DOMPoint(d,0),new DOMPoint(d,f),new DOMPoint(0,f)],c=(this._renderer.isP3D?new DOMMatrix(this._renderer.uMVMatrix.mat4):this.drawingContext.getTransform()).inverse(),l=p.map(function(v){return v.matrixTransform(c)}),b=Math.abs((l[3].x+l[0].x)*(l[3].y-l[0].y)+(l[0].x+l[1].x)*(l[0].y-l[1].y)+(l[1].x+l[2].x)*(l[1].y-l[2].y)+(l[2].x+l[3].x)*(l[2].y-l[3].y))/2;return Math.round(100*i/b)},t=s.default,g.default=t},{"../core/main":283,"core-js/modules/es.array.concat":153,"core-js/modules/es.array.fill":156,"core-js/modules/es.array.from":162,"core-js/modules/es.array.iterator":165,"core-js/modules/es.array.map":168,"core-js/modules/es.number.to-fixed":182,"core-js/modules/es.object.to-string":190,"core-js/modules/es.regexp.to-string":196,"core-js/modules/es.string.iterator":200,"core-js/modules/es.symbol":212,"core-js/modules/es.symbol.description":210,"core-js/modules/es.symbol.iterator":211,"core-js/modules/web.dom-collections.iterator":246}],266:[function(t,w,g){t("core-js/modules/es.array.concat"),t("core-js/modules/es.array.concat"),Object.defineProperty(g,"__esModule",{value:!0}),g.default=void 0,t=(t=t("../core/main"))&&t.__esModule?t:{default:t},t.default.prototype._updateTextOutput=function(s){var u,o,h,y;this.dummyDOM.querySelector("#".concat(s,"_summary"))&&(u=this._accessibleOutputs[s],h=function(m,n,i,d){return i="Your output is a, ".concat(i," by ").concat(d," pixels, ").concat(n," canvas containing the following"),i=m===1?"".concat(i," shape:"):"".concat(i," ").concat(m," shapes:"),i}((o=function(m,n){var i,d="",f=0;for(i in n)for(var p in n[i]){var c='
      • ').concat(n[i][p].color," ").concat(i,"");i==="line"?c+=", ".concat(n[i][p].pos,", ").concat(n[i][p].length," pixels long.
      • "):(c+=", at ".concat(n[i][p].pos),i!=="point"&&(c+=", covering ".concat(n[i][p].area,"% of the canvas")),c+="."),d+=c,f++}return{numShapes:f,listShapes:d}}(s,this.ingredients.shapes)).numShapes,this.ingredients.colors.background,this.width,this.height),y=function(m,n){var i,d="",f=0;for(i in n)for(var p in n[i]){var c='').concat(n[i][p].color," ").concat(i,"");i==="line"?c+="location = ".concat(n[i][p].pos,"length = ").concat(n[i][p].length," pixels"):(c+="location = ".concat(n[i][p].pos,""),i!=="point"&&(c+=" area = ".concat(n[i][p].area,"%")),c+=""),d+=c,f++}return d}(s,this.ingredients.shapes),h!==u.summary.innerHTML&&(u.summary.innerHTML=h),o.listShapes!==u.list.innerHTML&&(u.list.innerHTML=o.listShapes),y!==u.shapeDetails.innerHTML&&(u.shapeDetails.innerHTML=y),this._accessibleOutputs[s]=u)},t=t.default,g.default=t},{"../core/main":283,"core-js/modules/es.array.concat":153}],267:[function(t,w,g){var s=(s=t("./core/main"))&&s.__esModule?s:{default:s};t("./core/constants"),t("./core/environment"),t("./core/friendly_errors/stacktrace"),t("./core/friendly_errors/validate_params"),t("./core/friendly_errors/file_errors"),t("./core/friendly_errors/fes_core"),t("./core/friendly_errors/sketch_reader"),t("./core/helpers"),t("./core/legacy"),t("./core/preload"),t("./core/p5.Element"),t("./core/p5.Graphics"),t("./core/p5.Renderer"),t("./core/p5.Renderer2D"),t("./core/rendering"),t("./core/shim"),t("./core/structure"),t("./core/transform"),t("./core/shape/2d_primitives"),t("./core/shape/attributes"),t("./core/shape/curves"),t("./core/shape/vertex"),t("./accessibility/outputs"),t("./accessibility/textOutput"),t("./accessibility/gridOutput"),t("./accessibility/color_namer"),t("./color/color_conversion"),t("./color/creating_reading"),t("./color/p5.Color"),t("./color/setting"),t("./data/p5.TypedDict"),t("./data/local_storage.js"),t("./dom/dom"),t("./accessibility/describe"),t("./events/acceleration"),t("./events/keyboard"),t("./events/mouse"),t("./events/touch"),t("./image/filters"),t("./image/image"),t("./image/loading_displaying"),t("./image/p5.Image"),t("./image/pixels"),t("./io/files"),t("./io/p5.Table"),t("./io/p5.TableRow"),t("./io/p5.XML"),t("./math/calculation"),t("./math/math"),t("./math/noise"),t("./math/p5.Vector"),t("./math/random"),t("./math/trigonometry"),t("./typography/attributes"),t("./typography/loading_displaying"),t("./typography/p5.Font"),t("./utilities/array_functions"),t("./utilities/conversion"),t("./utilities/string_functions"),t("./utilities/time_date"),t("./webgl/3d_primitives"),t("./webgl/interaction"),t("./webgl/light"),t("./webgl/loading"),t("./webgl/material"),t("./webgl/p5.Camera"),t("./webgl/p5.DataArray"),t("./webgl/p5.Geometry"),t("./webgl/p5.Matrix"),t("./webgl/p5.Quat"),t("./webgl/p5.RendererGL.Immediate"),t("./webgl/p5.RendererGL"),t("./webgl/p5.RendererGL.Retained"),t("./webgl/p5.Framebuffer"),t("./webgl/p5.Shader"),t("./webgl/p5.RenderBuffer"),t("./webgl/p5.Texture"),t("./webgl/text"),t("./core/init"),w.exports=s.default},{"./accessibility/color_namer":262,"./accessibility/describe":263,"./accessibility/gridOutput":264,"./accessibility/outputs":265,"./accessibility/textOutput":266,"./color/color_conversion":268,"./color/creating_reading":269,"./color/p5.Color":270,"./color/setting":271,"./core/constants":272,"./core/environment":273,"./core/friendly_errors/fes_core":274,"./core/friendly_errors/file_errors":275,"./core/friendly_errors/sketch_reader":276,"./core/friendly_errors/stacktrace":277,"./core/friendly_errors/validate_params":278,"./core/helpers":279,"./core/init":280,"./core/legacy":282,"./core/main":283,"./core/p5.Element":284,"./core/p5.Graphics":285,"./core/p5.Renderer":286,"./core/p5.Renderer2D":287,"./core/preload":288,"./core/rendering":289,"./core/shape/2d_primitives":290,"./core/shape/attributes":291,"./core/shape/curves":292,"./core/shape/vertex":293,"./core/shim":294,"./core/structure":295,"./core/transform":296,"./data/local_storage.js":297,"./data/p5.TypedDict":298,"./dom/dom":299,"./events/acceleration":300,"./events/keyboard":301,"./events/mouse":302,"./events/touch":303,"./image/filters":304,"./image/image":305,"./image/loading_displaying":306,"./image/p5.Image":307,"./image/pixels":308,"./io/files":309,"./io/p5.Table":310,"./io/p5.TableRow":311,"./io/p5.XML":312,"./math/calculation":313,"./math/math":314,"./math/noise":315,"./math/p5.Vector":316,"./math/random":317,"./math/trigonometry":318,"./typography/attributes":319,"./typography/loading_displaying":320,"./typography/p5.Font":321,"./utilities/array_functions":322,"./utilities/conversion":323,"./utilities/string_functions":324,"./utilities/time_date":325,"./webgl/3d_primitives":326,"./webgl/interaction":328,"./webgl/light":329,"./webgl/loading":330,"./webgl/material":331,"./webgl/p5.Camera":332,"./webgl/p5.DataArray":333,"./webgl/p5.Framebuffer":334,"./webgl/p5.Geometry":335,"./webgl/p5.Matrix":336,"./webgl/p5.Quat":337,"./webgl/p5.RenderBuffer":338,"./webgl/p5.RendererGL":341,"./webgl/p5.RendererGL.Immediate":339,"./webgl/p5.RendererGL.Retained":340,"./webgl/p5.Shader":342,"./webgl/p5.Texture":343,"./webgl/text":344}],268:[function(t,w,g){Object.defineProperty(g,"__esModule",{value:!0}),g.default=void 0,t=(t=t("../core/main"))&&t.__esModule?t:{default:t},t.default.ColorConversion={_hsbaToHSLA:function(s){var u=s[0],o=s[1],h=s[2],y=(2-o)*h/2;return y!=0&&(y==1?o=0:y<.5?o/=2-o:o=o*h/(2-2*y)),[u,o,y,s[3]]},_hsbaToRGBA:function(s){var u,o,h,y,m,n=6*s[0],i=s[1],d=s[2];return i===0?[d,d,d,s[3]]:(o=d*(1-i),h=d*(1-i*(n-(u=Math.floor(n)))),i=d*(1-i*(1+u-n)),n=u===1?(y=h,m=d,o):u===2?(y=o,m=d,i):u===3?(y=o,m=h,d):u===4?(y=i,m=o,d):u===5?(y=d,m=o,h):(y=d,m=i,o),[y,m,n,s[3]])},_hslaToHSBA:function(s){var u=s[0],o=s[1],h=s[2],y=h<.5?(1+o)*h:h+o-h*o;return[u,o=2*(y-h)/y,y,s[3]]},_hslaToRGBA:function(s){var u,o=6*s[0],h=s[1],y=s[2];return h===0?[y,y,y,s[3]]:[(u=function(m,n,i){return m<0?m+=6:6<=m&&(m-=6),m<1?n+(i-n)*m:m<3?i:m<4?n+(i-n)*(4-m):n})(2+o,h=2*y-(y=y<.5?(1+h)*y:y+h-y*h),y),u(o,h,y),u(o-2,h,y),s[3]]},_rgbaToHSBA:function(s){var u,o,h=s[0],y=s[1],m=s[2],n=Math.max(h,y,m),i=n-Math.min(h,y,m);return i==0?o=u=0:(o=i/n,h===n?u=(y-m)/i:y===n?u=2+(m-h)/i:m===n&&(u=4+(h-y)/i),u<0?u+=6:6<=u&&(u-=6)),[u/6,o,n,s[3]]},_rgbaToHSLA:function(s){var u,o,h=s[0],y=s[1],m=s[2],n=Math.max(h,y,m),d=Math.min(h,y,m),i=n+d,d=n-d;return d==0?o=u=0:(o=i<1?d/i:d/(2-i),h===n?u=(y-m)/d:y===n?u=2+(m-h)/d:m===n&&(u=4+(h-y)/d),u<0?u+=6:6<=u&&(u-=6)),[u/6,o,i/2,s[3]]}},t=t.default.ColorConversion,g.default=t},{"../core/main":283}],269:[function(t,w,g){function s(n){return(s=typeof Symbol=="function"&&typeof Symbol.iterator=="symbol"?function(i){return typeof i}:function(i){return i&&typeof Symbol=="function"&&i.constructor===Symbol&&i!==Symbol.prototype?"symbol":typeof i})(n)}function u(n){return(u=typeof Symbol=="function"&&s(Symbol.iterator)==="symbol"?function(i){return s(i)}:function(i){return i&&typeof Symbol=="function"&&i.constructor===Symbol&&i!==Symbol.prototype?"symbol":s(i)})(n)}t("core-js/modules/es.symbol"),t("core-js/modules/es.symbol.description"),t("core-js/modules/es.symbol.iterator"),t("core-js/modules/es.array.iterator"),t("core-js/modules/es.array.map"),t("core-js/modules/es.object.get-own-property-descriptor"),t("core-js/modules/es.object.to-string"),t("core-js/modules/es.string.iterator"),t("core-js/modules/es.weak-map"),t("core-js/modules/web.dom-collections.iterator"),t("core-js/modules/es.array.map"),Object.defineProperty(g,"__esModule",{value:!0}),g.default=void 0;var o=(m=t("../core/main"))&&m.__esModule?m:{default:m},h=function(n){if(n&&n.__esModule)return n;if(n===null||u(n)!=="object"&&typeof n!="function")return{default:n};var i=y();if(i&&i.has(n))return i.get(n);var d,f={},p=Object.defineProperty&&Object.getOwnPropertyDescriptor;for(d in n){var c;Object.prototype.hasOwnProperty.call(n,d)&&((c=p?Object.getOwnPropertyDescriptor(n,d):null)&&(c.get||c.set)?Object.defineProperty(f,d,c):f[d]=n[d])}return f.default=n,i&&i.set(n,f),f}(t("../core/constants"));function y(){var n;return typeof WeakMap!="function"?null:(n=new WeakMap,y=function(){return n},n)}t("./p5.Color"),t("../core/friendly_errors/validate_params"),t("../core/friendly_errors/file_errors"),t("../core/friendly_errors/fes_core"),o.default.prototype.alpha=function(n){return o.default._validateParameters("alpha",arguments),this.color(n)._getAlpha()},o.default.prototype.blue=function(n){return o.default._validateParameters("blue",arguments),this.color(n)._getBlue()},o.default.prototype.brightness=function(n){return o.default._validateParameters("brightness",arguments),this.color(n)._getBrightness()},o.default.prototype.color=function(){for(var n,i=arguments.length,d=new Array(i),f=0;fc[0]?c[0]+=1:p[0]+=1),1<=(f=this.lerp(p[0],c[0],d))&&--f),n=this.lerp(p[1],c[1],d),i=this.lerp(p[2],c[2],d),p=this.lerp(p[3],c[3],d),f*=b[l][0],n*=b[l][1],i*=b[l][2],p*=b[l][3],this.color(f,n,i,p)},o.default.prototype.paletteLerp=function(n,i){var d=n[0];if(i>7,127&xe,le>>7,127&le);for(var Se=0;Se>7,127&Pe,0,0)}}return{cellImageInfo:re,dimOffset:Ie,dimImageInfo:we}}}}]),b}();h.default.RendererGL.prototype._renderText=function(b,v,j,T,E){if(this._textFont&&typeof this._textFont!="string"){if(!(E<=T)&&this._doFill){if(this._isOpenType()){b.push();var E=this._doStroke,M=this.drawMode,P=(this._doStroke=!1,this.drawMode=y.TEXTURE,this._textFont.font),L=(L=this._textFont._fontInfo)||(this._textFont._fontInfo=new l(P)),j=this._textFont._handleAlignment(this,v,j,T),T=this._textSize/P.unitsPerEm,C=(this.translate(j.x,j.y,0),this.scale(T,T,1),this.GL),j=!this._defaultFontShader,I=this._getFontShader(),A=(I.init(),I.bindShader(),j&&(I.setUniform("uGridImageSize",[64,64]),I.setUniform("uCellsImageSize",[64,64]),I.setUniform("uStrokeImageSize",[64,64]),I.setUniform("uGridSize",[9,9])),this._applyColorBlend(this.curFillColor),this.retainedMode.geometry.glyph),N=(A||((T=this._textGeom=new h.default.Geometry(1,1,function(){for(var he=0;he<=1;he++)for(var Y=0;Y<=1;Y++)this.vertices.push(new h.default.Vector(Y,he,0)),this.uvs.push(Y,he)})).computeFaces().computeNormals(),A=this.createBuffers("glyph",T)),!0),j=!1,T=void 0;try{for(var F,U=this.retainedMode.buffers.text[Symbol.iterator]();!(N=(F=U.next()).done);N=!0)F.value._prepareBuffer(A,I)}catch(he){j=!0,T=he}finally{try{N||U.return==null||U.return()}finally{if(j)throw T}}this._bindBuffer(A.indexBuffer,C.ELEMENT_ARRAY_BUFFER),I.setUniform("uMaterialColor",this.curFillColor),C.pixelStorei(C.UNPACK_PREMULTIPLY_ALPHA_WEBGL,!1);try{var W=0,V=null,H=P.stringToGlyphs(v),Q=!0,ee=!1,ae=void 0;try{for(var k,D=H[Symbol.iterator]();!(Q=(k=D.next()).done);Q=!0){var G,X,K=k.value,J=(V&&(W+=P.getKerningValue(V,K)),L.getGlyphInfo(K));J.uGlyphRect&&(G=J.rowInfo,X=J.colInfo,I.setUniform("uSamplerStrokes",J.strokeImageInfo.imageData),I.setUniform("uSamplerRowStrokes",G.cellImageInfo.imageData),I.setUniform("uSamplerRows",G.dimImageInfo.imageData),I.setUniform("uSamplerColStrokes",X.cellImageInfo.imageData),I.setUniform("uSamplerCols",X.dimImageInfo.imageData),I.setUniform("uGridOffset",J.uGridOffset),I.setUniform("uGlyphRect",J.uGlyphRect),I.setUniform("uGlyphOffset",W),I.bindTextures(),C.drawElements(C.TRIANGLES,6,this.GL.UNSIGNED_SHORT,0)),W+=K.advanceWidth,V=K}}catch(he){ee=!0,ae=he}finally{try{Q||D.return==null||D.return()}finally{if(ee)throw ae}}}finally{I.unbindShader(),this._doStroke=E,this.drawMode=M,C.pixelStorei(C.UNPACK_PREMULTIPLY_ALPHA_WEBGL,!0),b.pop()}}else console.log("WEBGL: only Opentype (.otf) and Truetype (.ttf) fonts are supported");return b}}else console.log("WEBGL: you must load and set a font before drawing text. See `loadFont` and `textFont` for more details.")}},{"../core/constants":272,"../core/main":283,"./p5.RendererGL.Retained":340,"./p5.Shader":342,"core-js/modules/es.array.iterator":165,"core-js/modules/es.object.get-own-property-descriptor":186,"core-js/modules/es.object.to-string":190,"core-js/modules/es.regexp.exec":195,"core-js/modules/es.string.iterator":200,"core-js/modules/es.string.split":206,"core-js/modules/es.string.sub":208,"core-js/modules/es.symbol":212,"core-js/modules/es.symbol.description":210,"core-js/modules/es.symbol.iterator":211,"core-js/modules/es.weak-map":244,"core-js/modules/web.dom-collections.iterator":246}]},{},[267])(267)})})(ih);var m1=ih.exports;const kc=p1(m1),y1=tn(()=>({provide:{p5:kc,p5:kc}})),g1=[$v,Kv,wb,Sb,Eb,Tb,Mb,f1,h1,y1],v1=gr({props:{vnode:{type:Object,required:!0},route:{type:Object,required:!0},vnodeRef:Object,renderKey:String,trackRootNodes:Boolean},setup(r){const a=r.renderKey,x=r.route,t={};for(const w in r.route)Object.defineProperty(t,w,{get:()=>a===r.renderKey?r.route[w]:x[w],enumerable:!0});return Ir(cs,cn(t)),()=>Jt(r.vnode,{ref:r.vnodeRef})}}),b1=gr({name:"NuxtPage",inheritAttrs:!1,props:{name:{type:String},transition:{type:[Boolean,Object],default:void 0},keepalive:{type:[Boolean,Object],default:void 0},route:{type:Object},pageKey:{type:[Function,String],default:null}},setup(r,{attrs:a,slots:x,expose:t}){const w=Dt(),g=mt(),s=zt(cs,null);let u;t({pageRef:g});const o=zt(Tf,null);let h;const y=w.deferHydration();if(w.isHydrating){const m=w.hooks.hookOnce("app:error",y);jr().beforeEach(m)}return r.pageKey&&On(()=>r.pageKey,(m,n)=>{m!==n&&w.callHook("page:loading:start")}),()=>Jt(oh,{name:r.name,route:r.route,...a},{default:m=>{const n=x1(s,m.route,m.Component),i=s&&s.matched.length===m.route.matched.length;if(!m.Component){if(h&&!i)return h;y();return}if(h&&o&&!o.isCurrent(m.route))return h;if(n&&s&&(!o||o!=null&&o.isCurrent(s)))return i?h:null;const d=Ia(m,r.pageKey);!w.isHydrating&&!w1(s,m.route,m.Component)&&u===d&&w.callHook("page:loading:end"),u=d;const f=!!(r.transition??m.route.meta.pageTransition??xa),p=f&&_1([r.transition,m.route.meta.pageTransition,xa,{onAfterLeave:()=>{w.callHook("page:transition:finish",m.Component)}}].filter(Boolean)),c=r.keepalive??m.route.meta.keepalive??wg;return h=sh(il,f&&p,eb(c,Jt(ol,{suspensible:!0,onPending:()=>w.callHook("page:start",m.Component),onResolve:()=>{Pr(()=>w.callHook("page:finish",m.Component).then(()=>w.callHook("page:loading:end")).finally(y))}},{default:()=>{const l=Jt(v1,{key:d||void 0,vnode:x.default?Jt(cr,void 0,x.default(m)):m.Component,route:m.route,renderKey:d||void 0,trackRootNodes:f,vnodeRef:g});return c&&(l.type.name=m.Component.type.name||m.Component.type.__name||"RouteProvider"),l}}))).default(),h}})}});function _1(r){const a=r.map(x=>({...x,onAfterLeave:x.onAfterLeave?ml(x.onAfterLeave):void 0}));return Sf(...a)}function x1(r,a,x){if(!r)return!1;const t=a.matched.findIndex(w=>{var g;return((g=w.components)==null?void 0:g.default)===(x==null?void 0:x.type)});return!t||t===-1?!1:a.matched.slice(0,t).some((w,g)=>{var s,u,o;return((s=w.components)==null?void 0:s.default)!==((o=(u=r.matched[g])==null?void 0:u.components)==null?void 0:o.default)})||x&&Ia({route:a,Component:x})!==Ia({route:r,Component:x})}function w1(r,a,x){return r?a.matched.findIndex(w=>{var g;return((g=w.components)==null?void 0:g.default)===(x==null?void 0:x.type)})t.default||t);return()=>Jt(x,r.layoutProps,a.slots)}}),S1=gr({name:"NuxtLayout",inheritAttrs:!1,props:{name:{type:[String,Boolean,Object],default:null},fallback:{type:[String,Object],default:null}},setup(r,a){const x=Dt(),t=zt(cs),w=t===fi()?Q0():t,g=vt(()=>{let o=Nt(r.name)??w.meta.layout??"default";return o&&!(o in Tn)&&r.fallback&&(o=Nt(r.fallback)),o}),s=mt();a.expose({layoutRef:s});const u=x.deferHydration();if(x.isHydrating){const o=x.hooks.hookOnce("app:error",u);jr().beforeEach(o)}return()=>{const o=g.value&&g.value in Tn,h=w.meta.layoutTransition??xg;return sh(il,o&&h,{default:()=>Jt(ol,{suspensible:!0,onResolve:()=>{Pr(u)}},{default:()=>Jt(E1,{layoutProps:ef(a.attrs,{ref:s}),key:g.value||void 0,name:g.value,shouldProvide:!r.name,hasTransition:!!h},a.slots)})}).default()}}}),E1=gr({name:"NuxtLayoutProvider",inheritAttrs:!1,props:{name:{type:[String,Boolean]},layoutProps:{type:Object},hasTransition:{type:Boolean},shouldProvide:{type:Boolean}},setup(r,a){const x=r.name;return r.shouldProvide&&Ir(Tf,{isCurrent:t=>x===(t.meta.layout??"default")}),()=>{var t,w;return!x||typeof x=="string"&&!(x in Tn)?(w=(t=a.slots).default)==null?void 0:w.call(t):Jt(j1,{key:x,layoutProps:r.layoutProps,name:x},a.slots)}}}),T1={class:"bg-background"},k1=gr({__name:"app",setup(r){return Vv({title:"Colton Padden",ogTitle:"Colton Padden",description:"This is where you can find the blog, portfolio, and experiments of Colton Padden.",ogDescription:"This is where you can find the blog, portfolio, and experiments of Colton Padden."}),(a,x)=>{const t=b1,w=S1;return wr(),jo("body",T1,[dt(w,null,{default:Sr(()=>[dt(t)]),_:1})])}}}),ah=(r,a)=>{const x=r.__vccOpts||r;for(const[t,w]of a)x[t]=w;return x},M1={},C1={class:"hover:text-orange-500",width:"366",height:"24",viewBox:"0 0 366 24",fill:"none",xmlns:"http://www.w3.org/2000/svg"};function O1(r,a){return wr(),jo("svg",C1,a[0]||(a[0]=[Am('',6)]))}const P1=ah(M1,[["render",O1]]);async function lh(r,a=jr()){const{path:x,matched:t}=a.resolve(r);if(!t.length||(a._routePreloaded||(a._routePreloaded=new Set),a._routePreloaded.has(x)))return;const w=a._preloadPromises=a._preloadPromises||[];if(w.length>4)return Promise.all(w).then(()=>lh(r,a));a._routePreloaded.add(x);const g=t.map(s=>{var u;return(u=s.components)==null?void 0:u.default}).filter(s=>typeof s=="function");for(const s of g){const u=Promise.resolve(s()).catch(()=>{}).finally(()=>w.splice(w.indexOf(u)));w.push(u)}await Promise.all(w)}const A1=(...r)=>r.find(a=>a!==void 0);function L1(r){const a=r.componentName||"NuxtLink";function x(w,g){if(!w||r.trailingSlash!=="append"&&r.trailingSlash!=="remove")return w;if(typeof w=="string")return Mc(w,r.trailingSlash);const s="path"in w&&w.path!==void 0?w.path:g(w).path;return{...w,name:void 0,path:Mc(s,r.trailingSlash)}}function t(w){const g=jr(),s=us(),u=vt(()=>!!w.target&&w.target!=="_self"),o=vt(()=>{const f=w.to||w.href||"";return typeof f=="string"&&pn(f,{acceptRelative:!0})}),h=hu("RouterLink"),y=h&&typeof h!="string"?h.useLink:void 0,m=vt(()=>{if(w.external)return!0;const f=w.to||w.href||"";return typeof f=="object"?!1:f===""||o.value}),n=vt(()=>{const f=w.to||w.href||"";return m.value?f:x(f,g.resolve)}),i=m.value||y==null?void 0:y({...w,to:n}),d=vt(()=>{var f;if(!n.value||o.value)return n.value;if(m.value){const p=typeof n.value=="object"&&"path"in n.value?Ta(n.value):n.value,c=typeof p=="object"?g.resolve(p).href:p;return x(c,g.resolve)}return typeof n.value=="object"?((f=g.resolve(n.value))==null?void 0:f.href)??null:x(di(s.app.baseURL,n.value),g.resolve)});return{to:n,hasTarget:u,isAbsoluteUrl:o,isExternal:m,href:d,isActive:(i==null?void 0:i.isActive)??vt(()=>n.value===g.currentRoute.value.path),isExactActive:(i==null?void 0:i.isExactActive)??vt(()=>n.value===g.currentRoute.value.path),route:(i==null?void 0:i.route)??vt(()=>g.resolve(n.value)),async navigate(){await Vg(d.value,{replace:w.replace,external:m.value||u.value})}}}return gr({name:a,props:{to:{type:[String,Object],default:void 0,required:!1},href:{type:[String,Object],default:void 0,required:!1},target:{type:String,default:void 0,required:!1},rel:{type:String,default:void 0,required:!1},noRel:{type:Boolean,default:void 0,required:!1},prefetch:{type:Boolean,default:void 0,required:!1},prefetchOn:{type:[String,Object],default:void 0,required:!1},noPrefetch:{type:Boolean,default:void 0,required:!1},activeClass:{type:String,default:void 0,required:!1},exactActiveClass:{type:String,default:void 0,required:!1},prefetchedClass:{type:String,default:void 0,required:!1},replace:{type:Boolean,default:void 0,required:!1},ariaCurrentValue:{type:String,default:void 0,required:!1},external:{type:Boolean,default:void 0,required:!1},custom:{type:Boolean,default:void 0,required:!1}},useLink:t,setup(w,{slots:g}){const s=jr(),{to:u,href:o,navigate:h,isExternal:y,hasTarget:m,isAbsoluteUrl:n}=t(w),i=mt(!1),d=mt(null),f=l=>{var b;d.value=w.custom?(b=l==null?void 0:l.$el)==null?void 0:b.nextElementSibling:l==null?void 0:l.$el};function p(l){var b,v;return!i.value&&(typeof w.prefetchOn=="string"?w.prefetchOn===l:((b=w.prefetchOn)==null?void 0:b[l])??((v=r.prefetchOn)==null?void 0:v[l]))&&(w.prefetch??r.prefetch)!==!1&&w.noPrefetch!==!0&&w.target!=="_blank"&&!D1()}async function c(l=Dt()){if(i.value)return;i.value=!0;const b=typeof u.value=="string"?u.value:y.value?Ta(u.value):s.resolve(u.value).fullPath;await Promise.all([l.hooks.callHook("link:prefetch",b).catch(()=>{}),!y.value&&!m.value&&lh(u.value,s).catch(()=>{})])}if(p("visibility")){const l=Dt();let b,v=null;Hr(()=>{const j=I1();gi(()=>{b=Da(()=>{var T;(T=d==null?void 0:d.value)!=null&&T.tagName&&(v=j.observe(d.value,async()=>{v==null||v(),v=null,await c(l)}))})})}),as(()=>{b&&jb(b),v==null||v(),v=null})}return()=>{var v;if(!y.value&&!m.value){const j={ref:f,to:u.value,activeClass:w.activeClass||r.activeClass,exactActiveClass:w.exactActiveClass||r.exactActiveClass,replace:w.replace,ariaCurrentValue:w.ariaCurrentValue,custom:w.custom};return w.custom||(p("interaction")&&(j.onPointerenter=c.bind(null,void 0),j.onFocus=c.bind(null,void 0)),i.value&&(j.class=w.prefetchedClass||r.prefetchedClass),j.rel=w.rel||void 0),Jt(hu("RouterLink"),j,g.default)}const l=w.target||null,b=A1(w.noRel?"":w.rel,r.externalRelAttribute,n.value||m.value?"noopener noreferrer":"")||null;return w.custom?g.default?g.default({href:o.value,navigate:h,prefetch:c,get route(){if(!o.value)return;const j=new URL(o.value,window.location.href);return{path:j.pathname,fullPath:j.pathname,get query(){return df(j.search)},hash:j.hash,params:{},name:void 0,matched:[],redirectedFrom:void 0,meta:{},href:o.value}},rel:b,target:l,isExternal:y.value||m.value,isActive:!1,isExactActive:!1}):null:Jt("a",{ref:d,href:o.value||null,rel:b,target:l},(v=g.default)==null?void 0:v.call(g))}}})}const R1=L1(jg);function Mc(r,a){const x=a==="append"?$s:ci;return pn(r)&&!r.startsWith("http")?r:x(r,!0)}function I1(){const r=Dt();if(r._observer)return r._observer;let a=null;const x=new Map,t=(g,s)=>(a||(a=new IntersectionObserver(u=>{for(const o of u){const h=x.get(o.target);(o.isIntersecting||o.intersectionRatio>0)&&h&&h()}})),x.set(g,s),a.observe(g),()=>{x.delete(g),a.unobserve(g),x.size===0&&(a.disconnect(),a=null)});return r._observer={observe:t}}function D1(){const r=navigator.connection;return!!(r&&(r.saveData||/2g/.test(r.effectiveType)))}var Cc;let F1=Symbol("headlessui.useid"),N1=0;const vi=(Cc=Dp)!=null?Cc:function(){return zt(F1,()=>`${++N1}`)()};function Ut(r){var a;if(r==null||r.value==null)return null;let x=(a=r.value.$el)!=null?a:r.value;return x instanceof Node?x:null}function Yn(r,a,...x){if(r in a){let w=a[r];return typeof w=="function"?w(...x):w}let t=new Error(`Tried to handle "${r}" but there is no handler defined. Only defined handlers are: ${Object.keys(a).map(w=>`"${w}"`).join(", ")}.`);throw Error.captureStackTrace&&Error.captureStackTrace(t,Yn),t}var U1=Object.defineProperty,B1=(r,a,x)=>a in r?U1(r,a,{enumerable:!0,configurable:!0,writable:!0,value:x}):r[a]=x,Oc=(r,a,x)=>(B1(r,typeof a!="symbol"?a+"":a,x),x);let G1=class{constructor(){Oc(this,"current",this.detect()),Oc(this,"currentId",0)}set(a){this.current!==a&&(this.currentId=0,this.current=a)}reset(){this.set(this.detect())}nextId(){return++this.currentId}get isServer(){return this.current==="server"}get isClient(){return this.current==="client"}detect(){return typeof window>"u"||typeof document>"u"?"server":"client"}},bi=new G1;function yl(r){if(bi.isServer)return null;if(r instanceof Node)return r.ownerDocument;if(r!=null&&r.hasOwnProperty("value")){let a=Ut(r);if(a)return a.ownerDocument}return document}let Fa=["[contentEditable=true]","[tabindex]","a[href]","area[href]","button:not([disabled])","iframe","input:not([disabled])","select:not([disabled])","textarea:not([disabled])"].map(r=>`${r}:not([tabindex='-1'])`).join(",");var Na=(r=>(r[r.First=1]="First",r[r.Previous=2]="Previous",r[r.Next=4]="Next",r[r.Last=8]="Last",r[r.WrapAround=16]="WrapAround",r[r.NoScroll=32]="NoScroll",r))(Na||{}),V1=(r=>(r[r.Error=0]="Error",r[r.Overflow=1]="Overflow",r[r.Success=2]="Success",r[r.Underflow=3]="Underflow",r))(V1||{}),H1=(r=>(r[r.Previous=-1]="Previous",r[r.Next=1]="Next",r))(H1||{});function uh(r=document.body){return r==null?[]:Array.from(r.querySelectorAll(Fa)).sort((a,x)=>Math.sign((a.tabIndex||Number.MAX_SAFE_INTEGER)-(x.tabIndex||Number.MAX_SAFE_INTEGER)))}var gl=(r=>(r[r.Strict=0]="Strict",r[r.Loose=1]="Loose",r))(gl||{});function vl(r,a=0){var x;return r===((x=yl(r))==null?void 0:x.body)?!1:Yn(a,{0(){return r.matches(Fa)},1(){let t=r;for(;t!==null;){if(t.matches(Fa))return!0;t=t.parentElement}return!1}})}function ch(r){let a=yl(r);Pr(()=>{a&&!vl(a.activeElement,0)&&W1(r)})}var z1=(r=>(r[r.Keyboard=0]="Keyboard",r[r.Mouse=1]="Mouse",r))(z1||{});typeof window<"u"&&typeof document<"u"&&(document.addEventListener("keydown",r=>{r.metaKey||r.altKey||r.ctrlKey||(document.documentElement.dataset.headlessuiFocusVisible="")},!0),document.addEventListener("click",r=>{r.detail===1?delete document.documentElement.dataset.headlessuiFocusVisible:r.detail===0&&(document.documentElement.dataset.headlessuiFocusVisible="")},!0));function W1(r){r==null||r.focus({preventScroll:!0})}let q1=["textarea","input"].join(",");function X1(r){var a,x;return(x=(a=r==null?void 0:r.matches)==null?void 0:a.call(r,q1))!=null?x:!1}function dh(r,a=x=>x){return r.slice().sort((x,t)=>{let w=a(x),g=a(t);if(w===null||g===null)return 0;let s=w.compareDocumentPosition(g);return s&Node.DOCUMENT_POSITION_FOLLOWING?-1:s&Node.DOCUMENT_POSITION_PRECEDING?1:0})}function $1(r,a){return Y1(uh(),a,{relativeTo:r})}function Y1(r,a,{sorted:x=!0,relativeTo:t=null,skipElements:w=[]}={}){var g;let s=(g=Array.isArray(r)?r.length>0?r[0].ownerDocument:document:r==null?void 0:r.ownerDocument)!=null?g:document,u=Array.isArray(r)?x?dh(r):r:uh(r);w.length>0&&u.length>1&&(u=u.filter(d=>!w.includes(d))),t=t??s.activeElement;let o=(()=>{if(a&5)return 1;if(a&10)return-1;throw new Error("Missing Focus.First, Focus.Previous, Focus.Next or Focus.Last")})(),h=(()=>{if(a&1)return 0;if(a&2)return Math.max(0,u.indexOf(t))-1;if(a&4)return Math.max(0,u.indexOf(t))+1;if(a&8)return u.length-1;throw new Error("Missing Focus.First, Focus.Previous, Focus.Next or Focus.Last")})(),y=a&32?{preventScroll:!0}:{},m=0,n=u.length,i;do{if(m>=n||m+n<=0)return 0;let d=h+m;if(a&16)d=(d+n)%n;else{if(d<0)return 3;if(d>=n)return 1}i=u[d],i==null||i.focus(y),m+=o}while(i!==s.activeElement);return a&6&&X1(i)&&i.select(),2}function K1(){return/iPhone/gi.test(window.navigator.platform)||/Mac/gi.test(window.navigator.platform)&&window.navigator.maxTouchPoints>0}function Z1(){return/Android/gi.test(window.navigator.userAgent)}function Q1(){return K1()||Z1()}function Cs(r,a,x){bi.isServer||Jr(t=>{document.addEventListener(r,a,x),t(()=>document.removeEventListener(r,a,x))})}function J1(r,a,x){bi.isServer||Jr(t=>{window.addEventListener(r,a,x),t(()=>window.removeEventListener(r,a,x))})}function e_(r,a,x=vt(()=>!0)){function t(g,s){if(!x.value||g.defaultPrevented)return;let u=s(g);if(u===null||!u.getRootNode().contains(u))return;let o=function h(y){return typeof y=="function"?h(y()):Array.isArray(y)||y instanceof Set?y:[y]}(r);for(let h of o){if(h===null)continue;let y=h instanceof HTMLElement?h:Ut(h);if(y!=null&&y.contains(u)||g.composed&&g.composedPath().includes(y))return}return!vl(u,gl.Loose)&&u.tabIndex!==-1&&g.preventDefault(),a(g,u)}let w=mt(null);Cs("pointerdown",g=>{var s,u;x.value&&(w.value=((u=(s=g.composedPath)==null?void 0:s.call(g))==null?void 0:u[0])||g.target)},!0),Cs("mousedown",g=>{var s,u;x.value&&(w.value=((u=(s=g.composedPath)==null?void 0:s.call(g))==null?void 0:u[0])||g.target)},!0),Cs("click",g=>{Q1()||w.value&&(t(g,()=>w.value),w.value=null)},!0),Cs("touchend",g=>t(g,()=>g.target instanceof HTMLElement?g.target:null),!0),J1("blur",g=>t(g,()=>window.document.activeElement instanceof HTMLIFrameElement?window.document.activeElement:null),!0)}function Pc(r,a){if(r)return r;let x=a??"button";if(typeof x=="string"&&x.toLowerCase()==="button")return"button"}function t_(r,a){let x=mt(Pc(r.value.type,r.value.as));return Hr(()=>{x.value=Pc(r.value.type,r.value.as)}),Jr(()=>{var t;x.value||Ut(a)&&Ut(a)instanceof HTMLButtonElement&&!((t=Ut(a))!=null&&t.hasAttribute("type"))&&(x.value="button")}),x}let Ac=/([\u2700-\u27BF]|[\uE000-\uF8FF]|\uD83C[\uDC00-\uDFFF]|\uD83D[\uDC00-\uDFFF]|[\u2011-\u26FF]|\uD83E[\uDD10-\uDDFF])/g;function Lc(r){var a,x;let t=(a=r.innerText)!=null?a:"",w=r.cloneNode(!0);if(!(w instanceof HTMLElement))return t;let g=!1;for(let u of w.querySelectorAll('[hidden],[aria-hidden],[role="img"]'))u.remove(),g=!0;let s=g?(x=w.innerText)!=null?x:"":t;return Ac.test(s)&&(s=s.replace(Ac,"")),s}function r_(r){let a=r.getAttribute("aria-label");if(typeof a=="string")return a.trim();let x=r.getAttribute("aria-labelledby");if(x){let t=x.split(" ").map(w=>{let g=document.getElementById(w);if(g){let s=g.getAttribute("aria-label");return typeof s=="string"?s.trim():Lc(g).trim()}return null}).filter(Boolean);if(t.length>0)return t.join(", ")}return Lc(r).trim()}function n_(r){let a=mt(""),x=mt("");return()=>{let t=Ut(r);if(!t)return"";let w=t.innerText;if(a.value===w)return x.value;let g=r_(t).trim().toLowerCase();return a.value=w,x.value=g,g}}function Rc(r){return[r.screenX,r.screenY]}function o_(){let r=mt([-1,-1]);return{wasMoved(a){let x=Rc(a);return r.value[0]===x[0]&&r.value[1]===x[1]?!1:(r.value=x,!0)},update(a){r.value=Rc(a)}}}function s_({container:r,accept:a,walk:x,enabled:t}){Jr(()=>{let w=r.value;if(!w||t!==void 0&&!t.value)return;let g=yl(r);if(!g)return;let s=Object.assign(o=>a(o),{acceptNode:a}),u=g.createTreeWalker(w,NodeFilter.SHOW_ELEMENT,s,!1);for(;u.nextNode();)x(u.currentNode)})}let fh=Symbol("Context");var Cr=(r=>(r[r.Open=1]="Open",r[r.Closed=2]="Closed",r[r.Closing=4]="Closing",r[r.Opening=8]="Opening",r))(Cr||{});function i_(){return bl()!==null}function bl(){return zt(fh,null)}function hh(r){Ir(fh,r)}var ur=(r=>(r.Space=" ",r.Enter="Enter",r.Escape="Escape",r.Backspace="Backspace",r.Delete="Delete",r.ArrowLeft="ArrowLeft",r.ArrowUp="ArrowUp",r.ArrowRight="ArrowRight",r.ArrowDown="ArrowDown",r.Home="Home",r.End="End",r.PageUp="PageUp",r.PageDown="PageDown",r.Tab="Tab",r))(ur||{});function a_(r){throw new Error("Unexpected object: "+r)}var Or=(r=>(r[r.First=0]="First",r[r.Previous=1]="Previous",r[r.Next=2]="Next",r[r.Last=3]="Last",r[r.Specific=4]="Specific",r[r.Nothing=5]="Nothing",r))(Or||{});function l_(r,a){let x=a.resolveItems();if(x.length<=0)return null;let t=a.resolveActiveIndex(),w=t??-1;switch(r.focus){case 0:{for(let g=0;g=0;--g)if(!a.resolveDisabled(x[g],g,x))return g;return t}case 2:{for(let g=w+1;g=0;--g)if(!a.resolveDisabled(x[g],g,x))return g;return t}case 4:{for(let g=0;g(r[r.None=0]="None",r[r.RenderStrategy=1]="RenderStrategy",r[r.Static=2]="Static",r))(ti||{}),kn=(r=>(r[r.Unmount=0]="Unmount",r[r.Hidden=1]="Hidden",r))(kn||{});function So({visible:r=!0,features:a=0,ourProps:x,theirProps:t,...w}){var g;let s=mh(t,x),u=Object.assign(w,{props:s});if(r||a&2&&s.static)return Qi(u);if(a&1){let o=(g=s.unmount)==null||g?0:1;return Yn(o,{0(){return null},1(){return Qi({...w,props:{...s,hidden:!0,style:{display:"none"}}})}})}return Qi(u)}function Qi({props:r,attrs:a,slots:x,slot:t,name:w}){var g,s;let{as:u,...o}=yh(r,["unmount","static"]),h=(g=x.default)==null?void 0:g.call(x,t),y={};if(t){let m=!1,n=[];for(let[i,d]of Object.entries(t))typeof d=="boolean"&&(m=!0),d===!0&&n.push(i);m&&(y["data-headlessui-state"]=n.join(" "))}if(u==="template"){if(h=ph(h??[]),Object.keys(o).length>0||Object.keys(a).length>0){let[m,...n]=h??[];if(!u_(m)||n.length>0)throw new Error(['Passing props on "template"!',"",`The current component <${w} /> is rendering a "template".`,"However we need to passthrough the following props:",Object.keys(o).concat(Object.keys(a)).map(f=>f.trim()).filter((f,p,c)=>c.indexOf(f)===p).sort((f,p)=>f.localeCompare(p)).map(f=>` - ${f}`).join(` `),"","You can apply a few solutions:",['Add an `as="..."` prop, to ensure that we render an actual element instead of a "template".',"Render a single element as the child so that we can forward the props onto that element."].map(f=>` - ${f}`).join(` `)].join(` -`));let i=mh((s=m.props)!=null?s:{},o,y),d=en(m,i,!0);for(let f in i)f.startsWith("on")&&(d.props||(d.props={}),d.props[f]=i[f]);return d}return Array.isArray(h)&&h.length===1?h[0]:h}return Jt(u,Object.assign({},o,y),{default:()=>h})}function ph(r){return r.flatMap(a=>a.type===cr?ph(a.children):[a])}function mh(...r){if(r.length===0)return{};if(r.length===1)return r[0];let a={},x={};for(let t of r)for(let w in t)w.startsWith("on")&&typeof t[w]=="function"?(x[w]!=null||(x[w]=[]),x[w].push(t[w])):a[w]=t[w];if(a.disabled||a["aria-disabled"])return Object.assign(a,Object.fromEntries(Object.keys(x).map(t=>[t,void 0])));for(let t in x)Object.assign(a,{[t](w,...g){let s=x[t];for(let u of s){if(w instanceof Event&&w.defaultPrevented)return;u(w,...g)}}});return a}function yh(r,a=[]){let x=Object.assign({},r);for(let t of a)t in x&&delete x[t];return x}function u_(r){return r==null?!1:typeof r.type=="string"||typeof r.type=="object"||typeof r.type=="function"}var c_=(r=>(r[r.Open=0]="Open",r[r.Closed=1]="Closed",r))(c_||{}),d_=(r=>(r[r.Pointer=0]="Pointer",r[r.Other=1]="Other",r))(d_||{});function f_(r){requestAnimationFrame(()=>requestAnimationFrame(r))}let gh=Symbol("MenuContext");function _i(r){let a=zt(gh,null);if(a===null){let x=new Error(`<${r} /> is missing a parent component.`);throw Error.captureStackTrace&&Error.captureStackTrace(x,_i),x}return a}let h_=gr({name:"Menu",props:{as:{type:[Object,String],default:"template"}},setup(r,{slots:a,attrs:x}){let t=mt(1),w=mt(null),g=mt(null),s=mt([]),u=mt(""),o=mt(null),h=mt(1);function y(n=i=>i){let i=o.value!==null?s.value[o.value]:null,d=dh(n(s.value.slice()),p=>Ut(p.dataRef.domRef)),f=i?d.indexOf(i):null;return f===-1&&(f=null),{items:d,activeItemIndex:f}}let m={menuState:t,buttonRef:w,itemsRef:g,items:s,searchQuery:u,activeItemIndex:o,activationTrigger:h,closeMenu:()=>{t.value=1,o.value=null},openMenu:()=>t.value=0,goToItem(n,i,d){let f=y(),p=l_(n===Or.Specific?{focus:Or.Specific,id:i}:{focus:n},{resolveItems:()=>f.items,resolveActiveIndex:()=>f.activeItemIndex,resolveId:c=>c.id,resolveDisabled:c=>c.dataRef.disabled});u.value="",o.value=p,h.value=d??1,s.value=f.items},search(n){let i=u.value!==""?0:1;u.value+=n.toLowerCase();let d=(o.value!==null?s.value.slice(o.value+i).concat(s.value.slice(0,o.value+i)):s.value).find(p=>p.dataRef.textValue.startsWith(u.value)&&!p.dataRef.disabled),f=d?s.value.indexOf(d):-1;f===-1||f===o.value||(o.value=f,h.value=1)},clearSearch(){u.value=""},registerItem(n,i){let d=y(f=>[...f,{id:n,dataRef:i}]);s.value=d.items,o.value=d.activeItemIndex,h.value=1},unregisterItem(n){let i=y(d=>{let f=d.findIndex(p=>p.id===n);return f!==-1&&d.splice(f,1),d});s.value=i.items,o.value=i.activeItemIndex,h.value=1}};return e_([w,g],(n,i)=>{var d;m.closeMenu(),vl(i,gl.Loose)||(n.preventDefault(),(d=Ut(w))==null||d.focus())},vt(()=>t.value===0)),Ir(gh,m),hh(vt(()=>Yn(t.value,{0:Cr.Open,1:Cr.Closed}))),()=>{let n={open:t.value===0,close:m.closeMenu};return So({ourProps:{},theirProps:r,slot:n,slots:a,attrs:x,name:"Menu"})}}}),p_=gr({name:"MenuButton",props:{disabled:{type:Boolean,default:!1},as:{type:[Object,String],default:"button"},id:{type:String,default:null}},setup(r,{attrs:a,slots:x,expose:t}){var w;let g=(w=r.id)!=null?w:`headlessui-menu-button-${vi()}`,s=_i("MenuButton");t({el:s.buttonRef,$el:s.buttonRef});function u(m){switch(m.key){case ur.Space:case ur.Enter:case ur.ArrowDown:m.preventDefault(),m.stopPropagation(),s.openMenu(),Pr(()=>{var n;(n=Ut(s.itemsRef))==null||n.focus({preventScroll:!0}),s.goToItem(Or.First)});break;case ur.ArrowUp:m.preventDefault(),m.stopPropagation(),s.openMenu(),Pr(()=>{var n;(n=Ut(s.itemsRef))==null||n.focus({preventScroll:!0}),s.goToItem(Or.Last)});break}}function o(m){switch(m.key){case ur.Space:m.preventDefault();break}}function h(m){r.disabled||(s.menuState.value===0?(s.closeMenu(),Pr(()=>{var n;return(n=Ut(s.buttonRef))==null?void 0:n.focus({preventScroll:!0})})):(m.preventDefault(),s.openMenu(),f_(()=>{var n;return(n=Ut(s.itemsRef))==null?void 0:n.focus({preventScroll:!0})})))}let y=t_(vt(()=>({as:r.as,type:a.type})),s.buttonRef);return()=>{var m;let n={open:s.menuState.value===0},{...i}=r,d={ref:s.buttonRef,id:g,type:y.value,"aria-haspopup":"menu","aria-controls":(m=Ut(s.itemsRef))==null?void 0:m.id,"aria-expanded":s.menuState.value===0,onKeydown:u,onKeyup:o,onClick:h};return So({ourProps:d,theirProps:i,slot:n,attrs:a,slots:x,name:"MenuButton"})}}}),m_=gr({name:"MenuItems",props:{as:{type:[Object,String],default:"div"},static:{type:Boolean,default:!1},unmount:{type:Boolean,default:!0},id:{type:String,default:null}},setup(r,{attrs:a,slots:x,expose:t}){var w;let g=(w=r.id)!=null?w:`headlessui-menu-items-${vi()}`,s=_i("MenuItems"),u=mt(null);t({el:s.itemsRef,$el:s.itemsRef}),s_({container:vt(()=>Ut(s.itemsRef)),enabled:vt(()=>s.menuState.value===0),accept(n){return n.getAttribute("role")==="menuitem"?NodeFilter.FILTER_REJECT:n.hasAttribute("role")?NodeFilter.FILTER_SKIP:NodeFilter.FILTER_ACCEPT},walk(n){n.setAttribute("role","none")}});function o(n){var i;switch(u.value&&clearTimeout(u.value),n.key){case ur.Space:if(s.searchQuery.value!=="")return n.preventDefault(),n.stopPropagation(),s.search(n.key);case ur.Enter:if(n.preventDefault(),n.stopPropagation(),s.activeItemIndex.value!==null){let d=s.items.value[s.activeItemIndex.value];(i=Ut(d.dataRef.domRef))==null||i.click()}s.closeMenu(),ch(Ut(s.buttonRef));break;case ur.ArrowDown:return n.preventDefault(),n.stopPropagation(),s.goToItem(Or.Next);case ur.ArrowUp:return n.preventDefault(),n.stopPropagation(),s.goToItem(Or.Previous);case ur.Home:case ur.PageUp:return n.preventDefault(),n.stopPropagation(),s.goToItem(Or.First);case ur.End:case ur.PageDown:return n.preventDefault(),n.stopPropagation(),s.goToItem(Or.Last);case ur.Escape:n.preventDefault(),n.stopPropagation(),s.closeMenu(),Pr(()=>{var d;return(d=Ut(s.buttonRef))==null?void 0:d.focus({preventScroll:!0})});break;case ur.Tab:n.preventDefault(),n.stopPropagation(),s.closeMenu(),Pr(()=>$1(Ut(s.buttonRef),n.shiftKey?Na.Previous:Na.Next));break;default:n.key.length===1&&(s.search(n.key),u.value=setTimeout(()=>s.clearSearch(),350));break}}function h(n){switch(n.key){case ur.Space:n.preventDefault();break}}let y=bl(),m=vt(()=>y!==null?(y.value&Cr.Open)===Cr.Open:s.menuState.value===0);return()=>{var n,i;let d={open:s.menuState.value===0},{...f}=r,p={"aria-activedescendant":s.activeItemIndex.value===null||(n=s.items.value[s.activeItemIndex.value])==null?void 0:n.id,"aria-labelledby":(i=Ut(s.buttonRef))==null?void 0:i.id,id:g,onKeydown:o,onKeyup:h,role:"menu",tabIndex:0,ref:s.itemsRef};return So({ourProps:p,theirProps:f,slot:d,attrs:a,slots:x,features:ti.RenderStrategy|ti.Static,visible:m.value,name:"MenuItems"})}}}),y_=gr({name:"MenuItem",inheritAttrs:!1,props:{as:{type:[Object,String],default:"template"},disabled:{type:Boolean,default:!1},id:{type:String,default:null}},setup(r,{slots:a,attrs:x,expose:t}){var w;let g=(w=r.id)!=null?w:`headlessui-menu-item-${vi()}`,s=_i("MenuItem"),u=mt(null);t({el:u,$el:u});let o=vt(()=>s.activeItemIndex.value!==null?s.items.value[s.activeItemIndex.value].id===g:!1),h=n_(u),y=vt(()=>({disabled:r.disabled,get textValue(){return h()},domRef:u}));Hr(()=>s.registerItem(g,y)),ls(()=>s.unregisterItem(g)),Jr(()=>{s.menuState.value===0&&o.value&&s.activationTrigger.value!==0&&Pr(()=>{var c,l;return(l=(c=Ut(u))==null?void 0:c.scrollIntoView)==null?void 0:l.call(c,{block:"nearest"})})});function m(c){if(r.disabled)return c.preventDefault();s.closeMenu(),ch(Ut(s.buttonRef))}function n(){if(r.disabled)return s.goToItem(Or.Nothing);s.goToItem(Or.Specific,g)}let i=o_();function d(c){i.update(c)}function f(c){i.wasMoved(c)&&(r.disabled||o.value||s.goToItem(Or.Specific,g,0))}function p(c){i.wasMoved(c)&&(r.disabled||o.value&&s.goToItem(Or.Nothing))}return()=>{let{disabled:c,...l}=r,b={active:o.value,disabled:c,close:s.closeMenu};return So({ourProps:{id:g,ref:u,role:"menuitem",tabIndex:c===!0?void 0:-1,"aria-disabled":c===!0?!0:void 0,onClick:m,onFocus:n,onPointerenter:d,onMouseenter:d,onPointermove:f,onMousemove:f,onPointerleave:p,onMouseleave:p},theirProps:{...x,...l},slot:b,attrs:x,slots:a,name:"MenuItem"})}}});function g_(r,a){return wr(),jo("svg",{xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 20 20",fill:"currentColor","aria-hidden":"true","data-slot":"icon"},[Et("path",{"fill-rule":"evenodd",d:"M2 4.75A.75.75 0 0 1 2.75 4h14.5a.75.75 0 0 1 0 1.5H2.75A.75.75 0 0 1 2 4.75ZM2 10a.75.75 0 0 1 .75-.75h14.5a.75.75 0 0 1 0 1.5H2.75A.75.75 0 0 1 2 10Zm0 5.25a.75.75 0 0 1 .75-.75h14.5a.75.75 0 0 1 0 1.5H2.75a.75.75 0 0 1-.75-.75Z","clip-rule":"evenodd"})])}const v_={class:"container mx-auto py-6"},b_={class:"flex items-center text-gray-200"},__={class:"flex items-center"},x_={class:"px-1 py-1"},w_={href:"/articles"},j_={href:"/talks"},S_={href:"/playground"},E_={__name:"Header",setup(r){return fi(),(a,x)=>{const t=P1,w=R1,g=p_,s=y_,u=m_,o=h_;return wr(),jo("nav",v_,[Et("div",b_,[dt(w,{to:"/",class:"flex-1"},{default:Sr(()=>[dt(t,{class:"w-32 md:w-64"})]),_:1}),Et("div",__,[dt(w,{to:"https://bsky.app/colton.boo"},{default:Sr(()=>x[0]||(x[0]=[Et("svg",{class:"mr-2 inline h-6 w-6 hover:text-orange-300",xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 600 530",fill:"currentColor"},[Et("path",{d:"m135.72 44.03c66.496 49.921 138.02 151.14 164.28 205.46 26.262-54.316 97.782-155.54 164.28-205.46 47.98-36.021 125.72-63.892 125.72 24.795 0 17.712-10.155 148.79-16.111 170.07-20.703 73.984-96.144 92.854-163.25 81.433 117.3 19.964 147.14 86.092 82.697 152.22-122.39 125.59-175.91-31.511-189.63-71.766-2.514-7.3797-3.6904-10.832-3.7077-7.8964-0.0174-2.9357-1.1937 0.51669-3.7077 7.8964-13.714 40.255-67.233 197.36-189.63 71.766-64.444-66.128-34.605-132.26 82.697-152.22-67.108 11.421-142.55-7.4491-163.25-81.433-5.9562-21.282-16.111-152.36-16.111-170.07 0-88.687 77.742-60.816 125.72-24.795z"})],-1)])),_:1}),dt(w,{to:"https://github.com/cmpadden"},{default:Sr(()=>x[1]||(x[1]=[Et("svg",{class:"mr-2 inline h-6 w-6 hover:text-orange-300",xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 24 24",fill:"currentColor"},[Et("path",{d:"M12.001 2C6.47598 2 2.00098 6.475 2.00098 12C2.00098 16.425 4.86348 20.1625 8.83848 21.4875C9.33848 21.575 9.52598 21.275 9.52598 21.0125C9.52598 20.775 9.51348 19.9875 9.51348 19.15C7.00098 19.6125 6.35098 18.5375 6.15098 17.975C6.03848 17.6875 5.55098 16.8 5.12598 16.5625C4.77598 16.375 4.27598 15.9125 5.11348 15.9C5.90098 15.8875 6.46348 16.625 6.65098 16.925C7.55098 18.4375 8.98848 18.0125 9.56348 17.75C9.65098 17.1 9.91348 16.6625 10.201 16.4125C7.97598 16.1625 5.65098 15.3 5.65098 11.475C5.65098 10.3875 6.03848 9.4875 6.67598 8.7875C6.57598 8.5375 6.22598 7.5125 6.77598 6.1375C6.77598 6.1375 7.61348 5.875 9.52598 7.1625C10.326 6.9375 11.176 6.825 12.026 6.825C12.876 6.825 13.726 6.9375 14.526 7.1625C16.4385 5.8625 17.276 6.1375 17.276 6.1375C17.826 7.5125 17.476 8.5375 17.376 8.7875C18.0135 9.4875 18.401 10.375 18.401 11.475C18.401 15.3125 16.0635 16.1625 13.8385 16.4125C14.201 16.725 14.5135 17.325 14.5135 18.2625C14.5135 19.6 14.501 20.675 14.501 21.0125C14.501 21.275 14.6885 21.5875 15.1885 21.4875C19.259 20.1133 21.9999 16.2963 22.001 12C22.001 6.475 17.526 2 12.001 2Z"})],-1)])),_:1}),dt(o,{as:"div",class:"relative z-50 inline-block text-left"},{default:Sr(()=>[Et("div",null,[dt(g,{class:"inline-flex w-full justify-center rounded-md bg-black/50 p-2 text-sm font-medium text-white hover:bg-opacity-30 focus:outline-none focus-visible:ring-2 focus-visible:ring-white focus-visible:ring-opacity-75"},{default:Sr(()=>[dt(Nt(g_),{class:"h-5 w-5 text-white hover:text-orange-500","aria-hidden":"true"})]),_:1})]),dt(il,{"enter-active-class":"transition duration-100 ease-out","enter-from-class":"transform scale-95 opacity-0","enter-to-class":"transform scale-100 opacity-100","leave-active-class":"transition duration-75 ease-in","leave-from-class":"transform scale-100 opacity-100","leave-to-class":"transform scale-95 opacity-0"},{default:Sr(()=>[dt(u,{class:"absolute right-0 mt-2 w-56 origin-top-right divide-y divide-gray-100 rounded-md bg-black/80 text-white shadow-lg ring-1 ring-black ring-opacity-5 backdrop-blur-sm focus:outline-none"},{default:Sr(()=>[Et("div",x_,[dt(s,null,{default:Sr(({active:h})=>[Et("a",w_,[Et("button",{class:Vn([h?"bg-slate-700 text-white":"text-white","group flex w-full items-center rounded-md px-2 py-2 text-sm"])},x[2]||(x[2]=[Et("svg",{class:"mr-2 inline h-5 w-5",xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24","stroke-width":"1.5",stroke:"currentColor"},[Et("path",{"stroke-linecap":"round","stroke-linejoin":"round",d:"M16.862 4.487l1.687-1.688a1.875 1.875 0 112.652 2.652L10.582 16.07a4.5 4.5 0 01-1.897 1.13L6 18l.8-2.685a4.5 4.5 0 011.13-1.897l8.932-8.931zm0 0L19.5 7.125M18 14v4.75A2.25 2.25 0 0115.75 21H5.25A2.25 2.25 0 013 18.75V8.25A2.25 2.25 0 015.25 6H10"})],-1),zo(" Blog ")]),2)])]),_:1}),dt(s,null,{default:Sr(({active:h})=>[Et("a",j_,[Et("button",{class:Vn([h?"bg-slate-700":"","group flex w-full items-center rounded-md px-2 py-2 text-sm"])},x[3]||(x[3]=[Et("svg",{class:"mr-2 inline h-5 w-5",xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24","stroke-width":"1.5",stroke:"currentColor"},[Et("path",{"stroke-linecap":"round","stroke-linejoin":"round",d:"M3.75 3v11.25A2.25 2.25 0 0 0 6 16.5h2.25M3.75 3h-1.5m1.5 0h16.5m0 0h1.5m-1.5 0v11.25A2.25 2.25 0 0 1 18 16.5h-2.25m-7.5 0h7.5m-7.5 0-1 3m8.5-3 1 3m0 0 .5 1.5m-.5-1.5h-9.5m0 0-.5 1.5M9 11.25v1.5M12 9v3.75m3-6v6"})],-1),zo(" Talks ")]),2)])]),_:1}),dt(s,null,{default:Sr(({active:h})=>[Et("a",S_,[Et("button",{class:Vn([h?"bg-slate-700 text-white":"text-white","group flex w-full items-center rounded-md px-2 py-2 text-sm"])},x[4]||(x[4]=[Et("svg",{class:"mr-2 inline h-5 w-5",xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24","stroke-width":"1.5",stroke:"currentColor"},[Et("path",{"stroke-linecap":"round","stroke-linejoin":"round",d:"M9.75 3.104v5.714a2.25 2.25 0 01-.659 1.591L5 14.5M9.75 3.104c-.251.023-.501.05-.75.082m.75-.082a24.301 24.301 0 014.5 0m0 0v5.714c0 .597.237 1.17.659 1.591L19.8 15.3M14.25 3.104c.251.023.501.05.75.082M19.8 15.3l-1.57.393A9.065 9.065 0 0112 15a9.065 9.065 0 00-6.23-.693L5 14.5m14.8.8l1.402 1.402c1.232 1.232.65 3.318-1.067 3.611A48.309 48.309 0 0112 21c-2.773 0-5.491-.235-8.135-.687-1.718-.293-2.3-2.379-1.067-3.61L5 14.5"})],-1),zo(" Experiments ")]),2)])]),_:1})])]),_:1})]),_:1})]),_:1})])])])}}};function T_(r){typeof queueMicrotask=="function"?queueMicrotask(r):Promise.resolve().then(r).catch(a=>setTimeout(()=>{throw a}))}function _l(){let r=[],a={addEventListener(x,t,w,g){return x.addEventListener(t,w,g),a.add(()=>x.removeEventListener(t,w,g))},requestAnimationFrame(...x){let t=requestAnimationFrame(...x);a.add(()=>cancelAnimationFrame(t))},nextFrame(...x){a.requestAnimationFrame(()=>{a.requestAnimationFrame(...x)})},setTimeout(...x){let t=setTimeout(...x);a.add(()=>clearTimeout(t))},microTask(...x){let t={current:!0};return T_(()=>{t.current&&x[0]()}),a.add(()=>{t.current=!1})},style(x,t,w){let g=x.style.getPropertyValue(t);return Object.assign(x.style,{[t]:w}),this.add(()=>{Object.assign(x.style,{[t]:g})})},group(x){let t=_l();return x(t),this.add(()=>t.dispose())},add(x){return r.push(x),()=>{let t=r.indexOf(x);if(t>=0)for(let w of r.splice(t,1))w()}},dispose(){for(let x of r.splice(0))x()}};return a}function k_(r){let a={called:!1};return(...x)=>{if(!a.called)return a.called=!0,r(...x)}}function Ji(r,...a){r&&a.length>0&&r.classList.add(...a)}function Os(r,...a){r&&a.length>0&&r.classList.remove(...a)}var Ua=(r=>(r.Finished="finished",r.Cancelled="cancelled",r))(Ua||{});function M_(r,a){let x=_l();if(!r)return x.dispose;let{transitionDuration:t,transitionDelay:w}=getComputedStyle(r),[g,s]=[t,w].map(u=>{let[o=0]=u.split(",").filter(Boolean).map(h=>h.includes("ms")?parseFloat(h):parseFloat(h)*1e3).sort((h,y)=>y-h);return o});return g!==0?x.setTimeout(()=>a("finished"),g+s):a("finished"),x.add(()=>a("cancelled")),x.dispose}function Ic(r,a,x,t,w,g){let s=_l(),u=g!==void 0?k_(g):()=>{};return Os(r,...w),Ji(r,...a,...x),s.nextFrame(()=>{Os(r,...x),Ji(r,...t),s.add(M_(r,o=>(Os(r,...t,...a),Ji(r,...w),u(o))))}),s.add(()=>Os(r,...a,...x,...t,...w)),s.add(()=>u("cancelled")),s.dispose}function Un(r=""){return r.split(/\s+/).filter(a=>a.length>1)}let xl=Symbol("TransitionContext");var C_=(r=>(r.Visible="visible",r.Hidden="hidden",r))(C_||{});function O_(){return zt(xl,null)!==null}function P_(){let r=zt(xl,null);if(r===null)throw new Error("A is used but it is missing a parent .");return r}function A_(){let r=zt(wl,null);if(r===null)throw new Error("A is used but it is missing a parent .");return r}let wl=Symbol("NestingContext");function xi(r){return"children"in r?xi(r.children):r.value.filter(({state:a})=>a==="visible").length>0}function vh(r){let a=mt([]),x=mt(!1);Hr(()=>x.value=!0),ls(()=>x.value=!1);function t(g,s=kn.Hidden){let u=a.value.findIndex(({id:o})=>o===g);u!==-1&&(Yn(s,{[kn.Unmount](){a.value.splice(u,1)},[kn.Hidden](){a.value[u].state="hidden"}}),!xi(a)&&x.value&&(r==null||r()))}function w(g){let s=a.value.find(({id:u})=>u===g);return s?s.state!=="visible"&&(s.state="visible"):a.value.push({id:g,state:"visible"}),()=>t(g,kn.Unmount)}return{children:a,register:w,unregister:t}}let bh=ti.RenderStrategy,L_=gr({props:{as:{type:[Object,String],default:"div"},show:{type:[Boolean],default:null},unmount:{type:[Boolean],default:!0},appear:{type:[Boolean],default:!1},enter:{type:[String],default:""},enterFrom:{type:[String],default:""},enterTo:{type:[String],default:""},entered:{type:[String],default:""},leave:{type:[String],default:""},leaveFrom:{type:[String],default:""},leaveTo:{type:[String],default:""}},emits:{beforeEnter:()=>!0,afterEnter:()=>!0,beforeLeave:()=>!0,afterLeave:()=>!0},setup(r,{emit:a,attrs:x,slots:t,expose:w}){let g=mt(0);function s(){g.value|=Cr.Opening,a("beforeEnter")}function u(){g.value&=~Cr.Opening,a("afterEnter")}function o(){g.value|=Cr.Closing,a("beforeLeave")}function h(){g.value&=~Cr.Closing,a("afterLeave")}if(!O_()&&i_())return()=>Jt(_h,{...r,onBeforeEnter:s,onAfterEnter:u,onBeforeLeave:o,onAfterLeave:h},t);let y=mt(null),m=vt(()=>r.unmount?kn.Unmount:kn.Hidden);w({el:y,$el:y});let{show:n,appear:i}=P_(),{register:d,unregister:f}=A_(),p=mt(n.value?"visible":"hidden"),c={value:!0},l=vi(),b={value:!1},v=vh(()=>{!b.value&&p.value!=="hidden"&&(p.value="hidden",f(l),h())});Hr(()=>{let A=d(l);ls(A)}),Jr(()=>{if(m.value===kn.Hidden&&l){if(n.value&&p.value!=="visible"){p.value="visible";return}Yn(p.value,{hidden:()=>f(l),visible:()=>d(l)})}});let j=Un(r.enter),T=Un(r.enterFrom),E=Un(r.enterTo),M=Un(r.entered),P=Un(r.leave),L=Un(r.leaveFrom),C=Un(r.leaveTo);Hr(()=>{Jr(()=>{if(p.value==="visible"){let A=Ut(y);if(A instanceof Comment&&A.data==="")throw new Error("Did you forget to passthrough the `ref` to the actual DOM node?")}})});function I(A){let N=c.value&&!i.value,F=Ut(y);!F||!(F instanceof HTMLElement)||N||(b.value=!0,n.value&&s(),n.value||o(),A(n.value?Ic(F,j,T,E,M,U=>{b.value=!1,U===Ua.Finished&&u()}):Ic(F,P,L,C,M,U=>{b.value=!1,U===Ua.Finished&&(xi(v)||(p.value="hidden",f(l),h()))})))}return Hr(()=>{On([n],(A,N,F)=>{I(F),c.value=!1},{immediate:!0})}),Ir(wl,v),hh(vt(()=>Yn(p.value,{visible:Cr.Open,hidden:Cr.Closed})|g.value)),()=>{let{appear:A,show:N,enter:F,enterFrom:U,enterTo:W,entered:V,leave:H,leaveFrom:Q,leaveTo:ee,...ae}=r,k={ref:y},D={...ae,...i.value&&n.value&&bi.isServer?{class:Vn([x.class,ae.class,...j,...T])}:{}};return So({theirProps:D,ourProps:k,slot:{},slots:t,attrs:x,features:bh,visible:p.value==="visible",name:"TransitionChild"})}}}),R_=L_,_h=gr({inheritAttrs:!1,props:{as:{type:[Object,String],default:"div"},show:{type:[Boolean],default:null},unmount:{type:[Boolean],default:!0},appear:{type:[Boolean],default:!1},enter:{type:[String],default:""},enterFrom:{type:[String],default:""},enterTo:{type:[String],default:""},entered:{type:[String],default:""},leave:{type:[String],default:""},leaveFrom:{type:[String],default:""},leaveTo:{type:[String],default:""}},emits:{beforeEnter:()=>!0,afterEnter:()=>!0,beforeLeave:()=>!0,afterLeave:()=>!0},setup(r,{emit:a,attrs:x,slots:t}){let w=bl(),g=vt(()=>r.show===null&&w!==null?(w.value&Cr.Open)===Cr.Open:r.show);Jr(()=>{if(![!0,!1].includes(g.value))throw new Error('A is used but it is missing a `:show="true | false"` prop.')});let s=mt(g.value?"visible":"hidden"),u=vh(()=>{s.value="hidden"}),o=mt(!0),h={show:g,appear:vt(()=>r.appear||!o.value)};return Hr(()=>{Jr(()=>{o.value=!1,g.value?s.value="visible":xi(u)||(s.value="hidden")})}),Ir(wl,u),Ir(xl,h),()=>{let y=yh(r,["show","appear","unmount","onBeforeEnter","onBeforeLeave","onAfterEnter","onAfterLeave"]),m={unmount:r.unmount};return So({ourProps:{...m,as:"template"},theirProps:{},slot:{},slots:{...t,default:()=>[Jt(R_,{onBeforeEnter:()=>a("beforeEnter"),onAfterEnter:()=>a("afterEnter"),onBeforeLeave:()=>a("beforeLeave"),onAfterLeave:()=>a("afterLeave"),...x,...m,...y},t.default)]},attrs:{},features:bh,visible:s.value==="visible",name:"Transition"})}}});const I_={},D_={class:"flex min-h-screen flex-col bg-background font-display"},F_={class:"mt-4 flex-1"},N_={class:"flex justify-center"};function U_(r,a){const x=E_,t=_h;return wr(),jo("body",D_,[dt(x),Et("main",F_,[Et("div",N_,[dt(t,{show:"",appear:"",as:"template",enter:"transition transform duration-300 ease-out","enter-from":"translate-x-8 opacity-0","enter-to":"translate-x-0 opacity-100",leave:"transition transform duration-300 ease-in","leave-from":"opacity-100","leave-to":"-translate-x-8 opacity-0"},{default:Sr(()=>a[0]||(a[0]=[Et("div",{class:"rounded-xl bg-black/50 p-6 text-center shadow-lg"},[Et("div",{class:"text-6xl font-extrabold tracking-widest text-white"}," Yikes! "),Et("div",{class:"text-base text-gray-200"}," Either the page you're looking for doesn't exist, or something has gone terribly wrong. ")],-1)])),_:1})])])])}const B_=ah(I_,[["render",U_]]),G_={key:0},Dc={__name:"nuxt-root",setup(r){const a=()=>null,x=Dt(),t=x.deferHydration();if(x.isHydrating){const o=x.hooks.hookOnce("app:error",t);jr().beforeEach(o)}const w=!1;Ir(cs,fi()),x.hooks.callHookWith(o=>o.map(h=>h()),"vue:setup");const g=hi(),s=!1;Td((o,h,y)=>{if(x.hooks.callHook("vue:error",o,h,y).catch(m=>console.error("[nuxt] Error in `vue:error` hook",m)),zg(o)&&(o.fatal||o.unhandled))return x.runWithContext(()=>lo(o)),!1});const u=!1;return(o,h)=>(wr(),En(ol,{onResolve:Nt(t)},{default:Sr(()=>[Nt(s)?(wr(),jo("div",G_)):Nt(g)?(wr(),En(Nt(B_),{key:1,error:Nt(g)},null,8,["error"])):Nt(u)?(wr(),En(Nt(a),{key:2,context:Nt(u)},null,8,["context"])):Nt(w)?(wr(),En(Kp(Nt(w)),{key:3})):(wr(),En(Nt(k1),{key:4}))]),_:1},8,["onResolve"]))}};let Fc;{let r;Fc=async function(){var s,u;if(r)return r;const t=!!(((s=window.__NUXT__)==null?void 0:s.serverRendered)??((u=document.getElementById("__NUXT_DATA__"))==null?void 0:u.dataset.ssr)==="true")?my(Dc):py(Dc),w=Mg({vueApp:t});async function g(o){await w.callHook("app:error",o),w.payload.error=w.payload.error||pi(o)}t.config.errorHandler=g;try{await Pg(w,g1)}catch(o){g(o)}try{await w.hooks.callHook("app:created",t),await w.hooks.callHook("app:beforeMount",t),t.mount(Eg),await w.hooks.callHook("app:mounted",t),await Pr()}catch(o){g(o)}return t.config.errorHandler===g&&(t.config.errorHandler=void 0),t},r=Fc().catch(a=>{throw console.error("Error while mounting app:",a),a})}export{pi as $,Rn as A,ex as B,q_ as C,Q_ as D,or as E,cr as F,Pr as G,Bv as H,us as I,Y_ as J,$s as K,Jt as L,rt as M,Dt as N,Ep as O,W_ as P,Vy as Q,Hu as R,_h as S,il as T,di as U,ci as V,J_ as W,Zo as X,Wp as Y,V_ as Z,R1 as _,Nt as a,Zn as a0,Xc as a1,bt as a2,hu as a3,qn as a4,Kt as a5,qs as a6,z_ as a7,$_ as a8,Wa as a9,Kp as aa,E_ as ab,ks as ac,Et as b,jo as c,gr as d,En as e,Sr as f,zo as g,Z_ as h,dt as i,mt as j,vt as k,Vn as l,ah as m,Vg as n,wr as o,Am as p,Hr as q,X_ as r,L_ as s,Qh as t,fi as u,On as v,K_ as w,as as x,ls as y,H_ as z}; +`));let i=mh((s=m.props)!=null?s:{},o,y),d=en(m,i,!0);for(let f in i)f.startsWith("on")&&(d.props||(d.props={}),d.props[f]=i[f]);return d}return Array.isArray(h)&&h.length===1?h[0]:h}return Jt(u,Object.assign({},o,y),{default:()=>h})}function ph(r){return r.flatMap(a=>a.type===cr?ph(a.children):[a])}function mh(...r){if(r.length===0)return{};if(r.length===1)return r[0];let a={},x={};for(let t of r)for(let w in t)w.startsWith("on")&&typeof t[w]=="function"?(x[w]!=null||(x[w]=[]),x[w].push(t[w])):a[w]=t[w];if(a.disabled||a["aria-disabled"])return Object.assign(a,Object.fromEntries(Object.keys(x).map(t=>[t,void 0])));for(let t in x)Object.assign(a,{[t](w,...g){let s=x[t];for(let u of s){if(w instanceof Event&&w.defaultPrevented)return;u(w,...g)}}});return a}function yh(r,a=[]){let x=Object.assign({},r);for(let t of a)t in x&&delete x[t];return x}function u_(r){return r==null?!1:typeof r.type=="string"||typeof r.type=="object"||typeof r.type=="function"}var c_=(r=>(r[r.Open=0]="Open",r[r.Closed=1]="Closed",r))(c_||{}),d_=(r=>(r[r.Pointer=0]="Pointer",r[r.Other=1]="Other",r))(d_||{});function f_(r){requestAnimationFrame(()=>requestAnimationFrame(r))}let gh=Symbol("MenuContext");function _i(r){let a=zt(gh,null);if(a===null){let x=new Error(`<${r} /> is missing a parent component.`);throw Error.captureStackTrace&&Error.captureStackTrace(x,_i),x}return a}let h_=gr({name:"Menu",props:{as:{type:[Object,String],default:"template"}},setup(r,{slots:a,attrs:x}){let t=mt(1),w=mt(null),g=mt(null),s=mt([]),u=mt(""),o=mt(null),h=mt(1);function y(n=i=>i){let i=o.value!==null?s.value[o.value]:null,d=dh(n(s.value.slice()),p=>Ut(p.dataRef.domRef)),f=i?d.indexOf(i):null;return f===-1&&(f=null),{items:d,activeItemIndex:f}}let m={menuState:t,buttonRef:w,itemsRef:g,items:s,searchQuery:u,activeItemIndex:o,activationTrigger:h,closeMenu:()=>{t.value=1,o.value=null},openMenu:()=>t.value=0,goToItem(n,i,d){let f=y(),p=l_(n===Or.Specific?{focus:Or.Specific,id:i}:{focus:n},{resolveItems:()=>f.items,resolveActiveIndex:()=>f.activeItemIndex,resolveId:c=>c.id,resolveDisabled:c=>c.dataRef.disabled});u.value="",o.value=p,h.value=d??1,s.value=f.items},search(n){let i=u.value!==""?0:1;u.value+=n.toLowerCase();let d=(o.value!==null?s.value.slice(o.value+i).concat(s.value.slice(0,o.value+i)):s.value).find(p=>p.dataRef.textValue.startsWith(u.value)&&!p.dataRef.disabled),f=d?s.value.indexOf(d):-1;f===-1||f===o.value||(o.value=f,h.value=1)},clearSearch(){u.value=""},registerItem(n,i){let d=y(f=>[...f,{id:n,dataRef:i}]);s.value=d.items,o.value=d.activeItemIndex,h.value=1},unregisterItem(n){let i=y(d=>{let f=d.findIndex(p=>p.id===n);return f!==-1&&d.splice(f,1),d});s.value=i.items,o.value=i.activeItemIndex,h.value=1}};return e_([w,g],(n,i)=>{var d;m.closeMenu(),vl(i,gl.Loose)||(n.preventDefault(),(d=Ut(w))==null||d.focus())},vt(()=>t.value===0)),Ir(gh,m),hh(vt(()=>Yn(t.value,{0:Cr.Open,1:Cr.Closed}))),()=>{let n={open:t.value===0,close:m.closeMenu};return So({ourProps:{},theirProps:r,slot:n,slots:a,attrs:x,name:"Menu"})}}}),p_=gr({name:"MenuButton",props:{disabled:{type:Boolean,default:!1},as:{type:[Object,String],default:"button"},id:{type:String,default:null}},setup(r,{attrs:a,slots:x,expose:t}){var w;let g=(w=r.id)!=null?w:`headlessui-menu-button-${vi()}`,s=_i("MenuButton");t({el:s.buttonRef,$el:s.buttonRef});function u(m){switch(m.key){case ur.Space:case ur.Enter:case ur.ArrowDown:m.preventDefault(),m.stopPropagation(),s.openMenu(),Pr(()=>{var n;(n=Ut(s.itemsRef))==null||n.focus({preventScroll:!0}),s.goToItem(Or.First)});break;case ur.ArrowUp:m.preventDefault(),m.stopPropagation(),s.openMenu(),Pr(()=>{var n;(n=Ut(s.itemsRef))==null||n.focus({preventScroll:!0}),s.goToItem(Or.Last)});break}}function o(m){switch(m.key){case ur.Space:m.preventDefault();break}}function h(m){r.disabled||(s.menuState.value===0?(s.closeMenu(),Pr(()=>{var n;return(n=Ut(s.buttonRef))==null?void 0:n.focus({preventScroll:!0})})):(m.preventDefault(),s.openMenu(),f_(()=>{var n;return(n=Ut(s.itemsRef))==null?void 0:n.focus({preventScroll:!0})})))}let y=t_(vt(()=>({as:r.as,type:a.type})),s.buttonRef);return()=>{var m;let n={open:s.menuState.value===0},{...i}=r,d={ref:s.buttonRef,id:g,type:y.value,"aria-haspopup":"menu","aria-controls":(m=Ut(s.itemsRef))==null?void 0:m.id,"aria-expanded":s.menuState.value===0,onKeydown:u,onKeyup:o,onClick:h};return So({ourProps:d,theirProps:i,slot:n,attrs:a,slots:x,name:"MenuButton"})}}}),m_=gr({name:"MenuItems",props:{as:{type:[Object,String],default:"div"},static:{type:Boolean,default:!1},unmount:{type:Boolean,default:!0},id:{type:String,default:null}},setup(r,{attrs:a,slots:x,expose:t}){var w;let g=(w=r.id)!=null?w:`headlessui-menu-items-${vi()}`,s=_i("MenuItems"),u=mt(null);t({el:s.itemsRef,$el:s.itemsRef}),s_({container:vt(()=>Ut(s.itemsRef)),enabled:vt(()=>s.menuState.value===0),accept(n){return n.getAttribute("role")==="menuitem"?NodeFilter.FILTER_REJECT:n.hasAttribute("role")?NodeFilter.FILTER_SKIP:NodeFilter.FILTER_ACCEPT},walk(n){n.setAttribute("role","none")}});function o(n){var i;switch(u.value&&clearTimeout(u.value),n.key){case ur.Space:if(s.searchQuery.value!=="")return n.preventDefault(),n.stopPropagation(),s.search(n.key);case ur.Enter:if(n.preventDefault(),n.stopPropagation(),s.activeItemIndex.value!==null){let d=s.items.value[s.activeItemIndex.value];(i=Ut(d.dataRef.domRef))==null||i.click()}s.closeMenu(),ch(Ut(s.buttonRef));break;case ur.ArrowDown:return n.preventDefault(),n.stopPropagation(),s.goToItem(Or.Next);case ur.ArrowUp:return n.preventDefault(),n.stopPropagation(),s.goToItem(Or.Previous);case ur.Home:case ur.PageUp:return n.preventDefault(),n.stopPropagation(),s.goToItem(Or.First);case ur.End:case ur.PageDown:return n.preventDefault(),n.stopPropagation(),s.goToItem(Or.Last);case ur.Escape:n.preventDefault(),n.stopPropagation(),s.closeMenu(),Pr(()=>{var d;return(d=Ut(s.buttonRef))==null?void 0:d.focus({preventScroll:!0})});break;case ur.Tab:n.preventDefault(),n.stopPropagation(),s.closeMenu(),Pr(()=>$1(Ut(s.buttonRef),n.shiftKey?Na.Previous:Na.Next));break;default:n.key.length===1&&(s.search(n.key),u.value=setTimeout(()=>s.clearSearch(),350));break}}function h(n){switch(n.key){case ur.Space:n.preventDefault();break}}let y=bl(),m=vt(()=>y!==null?(y.value&Cr.Open)===Cr.Open:s.menuState.value===0);return()=>{var n,i;let d={open:s.menuState.value===0},{...f}=r,p={"aria-activedescendant":s.activeItemIndex.value===null||(n=s.items.value[s.activeItemIndex.value])==null?void 0:n.id,"aria-labelledby":(i=Ut(s.buttonRef))==null?void 0:i.id,id:g,onKeydown:o,onKeyup:h,role:"menu",tabIndex:0,ref:s.itemsRef};return So({ourProps:p,theirProps:f,slot:d,attrs:a,slots:x,features:ti.RenderStrategy|ti.Static,visible:m.value,name:"MenuItems"})}}}),y_=gr({name:"MenuItem",inheritAttrs:!1,props:{as:{type:[Object,String],default:"template"},disabled:{type:Boolean,default:!1},id:{type:String,default:null}},setup(r,{slots:a,attrs:x,expose:t}){var w;let g=(w=r.id)!=null?w:`headlessui-menu-item-${vi()}`,s=_i("MenuItem"),u=mt(null);t({el:u,$el:u});let o=vt(()=>s.activeItemIndex.value!==null?s.items.value[s.activeItemIndex.value].id===g:!1),h=n_(u),y=vt(()=>({disabled:r.disabled,get textValue(){return h()},domRef:u}));Hr(()=>s.registerItem(g,y)),ls(()=>s.unregisterItem(g)),Jr(()=>{s.menuState.value===0&&o.value&&s.activationTrigger.value!==0&&Pr(()=>{var c,l;return(l=(c=Ut(u))==null?void 0:c.scrollIntoView)==null?void 0:l.call(c,{block:"nearest"})})});function m(c){if(r.disabled)return c.preventDefault();s.closeMenu(),ch(Ut(s.buttonRef))}function n(){if(r.disabled)return s.goToItem(Or.Nothing);s.goToItem(Or.Specific,g)}let i=o_();function d(c){i.update(c)}function f(c){i.wasMoved(c)&&(r.disabled||o.value||s.goToItem(Or.Specific,g,0))}function p(c){i.wasMoved(c)&&(r.disabled||o.value&&s.goToItem(Or.Nothing))}return()=>{let{disabled:c,...l}=r,b={active:o.value,disabled:c,close:s.closeMenu};return So({ourProps:{id:g,ref:u,role:"menuitem",tabIndex:c===!0?void 0:-1,"aria-disabled":c===!0?!0:void 0,onClick:m,onFocus:n,onPointerenter:d,onMouseenter:d,onPointermove:f,onMousemove:f,onPointerleave:p,onMouseleave:p},theirProps:{...x,...l},slot:b,attrs:x,slots:a,name:"MenuItem"})}}});function g_(r,a){return wr(),jo("svg",{xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 20 20",fill:"currentColor","aria-hidden":"true","data-slot":"icon"},[Et("path",{"fill-rule":"evenodd",d:"M2 4.75A.75.75 0 0 1 2.75 4h14.5a.75.75 0 0 1 0 1.5H2.75A.75.75 0 0 1 2 4.75ZM2 10a.75.75 0 0 1 .75-.75h14.5a.75.75 0 0 1 0 1.5H2.75A.75.75 0 0 1 2 10Zm0 5.25a.75.75 0 0 1 .75-.75h14.5a.75.75 0 0 1 0 1.5H2.75a.75.75 0 0 1-.75-.75Z","clip-rule":"evenodd"})])}const v_={class:"container mx-auto py-6"},b_={class:"flex items-center text-gray-200"},__={class:"flex items-center"},x_={class:"px-1 py-1"},w_={href:"/articles"},j_={href:"/talks"},S_={href:"/playground"},E_={__name:"Header",setup(r){return fi(),(a,x)=>{const t=P1,w=R1,g=p_,s=y_,u=m_,o=h_;return wr(),jo("nav",v_,[Et("div",b_,[dt(w,{to:"/",class:"flex-1"},{default:Sr(()=>[dt(t,{class:"w-32 md:w-64"})]),_:1}),Et("div",__,[dt(w,{to:"https://bsky.app/profile/colton.boo"},{default:Sr(()=>x[0]||(x[0]=[Et("svg",{class:"mr-2 inline h-6 w-6 hover:text-orange-300",xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 600 530",fill:"currentColor"},[Et("path",{d:"m135.72 44.03c66.496 49.921 138.02 151.14 164.28 205.46 26.262-54.316 97.782-155.54 164.28-205.46 47.98-36.021 125.72-63.892 125.72 24.795 0 17.712-10.155 148.79-16.111 170.07-20.703 73.984-96.144 92.854-163.25 81.433 117.3 19.964 147.14 86.092 82.697 152.22-122.39 125.59-175.91-31.511-189.63-71.766-2.514-7.3797-3.6904-10.832-3.7077-7.8964-0.0174-2.9357-1.1937 0.51669-3.7077 7.8964-13.714 40.255-67.233 197.36-189.63 71.766-64.444-66.128-34.605-132.26 82.697-152.22-67.108 11.421-142.55-7.4491-163.25-81.433-5.9562-21.282-16.111-152.36-16.111-170.07 0-88.687 77.742-60.816 125.72-24.795z"})],-1)])),_:1}),dt(w,{to:"https://github.com/cmpadden"},{default:Sr(()=>x[1]||(x[1]=[Et("svg",{class:"mr-2 inline h-6 w-6 hover:text-orange-300",xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 24 24",fill:"currentColor"},[Et("path",{d:"M12.001 2C6.47598 2 2.00098 6.475 2.00098 12C2.00098 16.425 4.86348 20.1625 8.83848 21.4875C9.33848 21.575 9.52598 21.275 9.52598 21.0125C9.52598 20.775 9.51348 19.9875 9.51348 19.15C7.00098 19.6125 6.35098 18.5375 6.15098 17.975C6.03848 17.6875 5.55098 16.8 5.12598 16.5625C4.77598 16.375 4.27598 15.9125 5.11348 15.9C5.90098 15.8875 6.46348 16.625 6.65098 16.925C7.55098 18.4375 8.98848 18.0125 9.56348 17.75C9.65098 17.1 9.91348 16.6625 10.201 16.4125C7.97598 16.1625 5.65098 15.3 5.65098 11.475C5.65098 10.3875 6.03848 9.4875 6.67598 8.7875C6.57598 8.5375 6.22598 7.5125 6.77598 6.1375C6.77598 6.1375 7.61348 5.875 9.52598 7.1625C10.326 6.9375 11.176 6.825 12.026 6.825C12.876 6.825 13.726 6.9375 14.526 7.1625C16.4385 5.8625 17.276 6.1375 17.276 6.1375C17.826 7.5125 17.476 8.5375 17.376 8.7875C18.0135 9.4875 18.401 10.375 18.401 11.475C18.401 15.3125 16.0635 16.1625 13.8385 16.4125C14.201 16.725 14.5135 17.325 14.5135 18.2625C14.5135 19.6 14.501 20.675 14.501 21.0125C14.501 21.275 14.6885 21.5875 15.1885 21.4875C19.259 20.1133 21.9999 16.2963 22.001 12C22.001 6.475 17.526 2 12.001 2Z"})],-1)])),_:1}),dt(o,{as:"div",class:"relative z-50 inline-block text-left"},{default:Sr(()=>[Et("div",null,[dt(g,{class:"inline-flex w-full justify-center rounded-md bg-black/50 p-2 text-sm font-medium text-white hover:bg-opacity-30 focus:outline-none focus-visible:ring-2 focus-visible:ring-white focus-visible:ring-opacity-75"},{default:Sr(()=>[dt(Nt(g_),{class:"h-5 w-5 text-white hover:text-orange-500","aria-hidden":"true"})]),_:1})]),dt(il,{"enter-active-class":"transition duration-100 ease-out","enter-from-class":"transform scale-95 opacity-0","enter-to-class":"transform scale-100 opacity-100","leave-active-class":"transition duration-75 ease-in","leave-from-class":"transform scale-100 opacity-100","leave-to-class":"transform scale-95 opacity-0"},{default:Sr(()=>[dt(u,{class:"absolute right-0 mt-2 w-56 origin-top-right divide-y divide-gray-100 rounded-md bg-black/80 text-white shadow-lg ring-1 ring-black ring-opacity-5 backdrop-blur-sm focus:outline-none"},{default:Sr(()=>[Et("div",x_,[dt(s,null,{default:Sr(({active:h})=>[Et("a",w_,[Et("button",{class:Vn([h?"bg-slate-700 text-white":"text-white","group flex w-full items-center rounded-md px-2 py-2 text-sm"])},x[2]||(x[2]=[Et("svg",{class:"mr-2 inline h-5 w-5",xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24","stroke-width":"1.5",stroke:"currentColor"},[Et("path",{"stroke-linecap":"round","stroke-linejoin":"round",d:"M16.862 4.487l1.687-1.688a1.875 1.875 0 112.652 2.652L10.582 16.07a4.5 4.5 0 01-1.897 1.13L6 18l.8-2.685a4.5 4.5 0 011.13-1.897l8.932-8.931zm0 0L19.5 7.125M18 14v4.75A2.25 2.25 0 0115.75 21H5.25A2.25 2.25 0 013 18.75V8.25A2.25 2.25 0 015.25 6H10"})],-1),zo(" Blog ")]),2)])]),_:1}),dt(s,null,{default:Sr(({active:h})=>[Et("a",j_,[Et("button",{class:Vn([h?"bg-slate-700":"","group flex w-full items-center rounded-md px-2 py-2 text-sm"])},x[3]||(x[3]=[Et("svg",{class:"mr-2 inline h-5 w-5",xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24","stroke-width":"1.5",stroke:"currentColor"},[Et("path",{"stroke-linecap":"round","stroke-linejoin":"round",d:"M3.75 3v11.25A2.25 2.25 0 0 0 6 16.5h2.25M3.75 3h-1.5m1.5 0h16.5m0 0h1.5m-1.5 0v11.25A2.25 2.25 0 0 1 18 16.5h-2.25m-7.5 0h7.5m-7.5 0-1 3m8.5-3 1 3m0 0 .5 1.5m-.5-1.5h-9.5m0 0-.5 1.5M9 11.25v1.5M12 9v3.75m3-6v6"})],-1),zo(" Talks ")]),2)])]),_:1}),dt(s,null,{default:Sr(({active:h})=>[Et("a",S_,[Et("button",{class:Vn([h?"bg-slate-700 text-white":"text-white","group flex w-full items-center rounded-md px-2 py-2 text-sm"])},x[4]||(x[4]=[Et("svg",{class:"mr-2 inline h-5 w-5",xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24","stroke-width":"1.5",stroke:"currentColor"},[Et("path",{"stroke-linecap":"round","stroke-linejoin":"round",d:"M9.75 3.104v5.714a2.25 2.25 0 01-.659 1.591L5 14.5M9.75 3.104c-.251.023-.501.05-.75.082m.75-.082a24.301 24.301 0 014.5 0m0 0v5.714c0 .597.237 1.17.659 1.591L19.8 15.3M14.25 3.104c.251.023.501.05.75.082M19.8 15.3l-1.57.393A9.065 9.065 0 0112 15a9.065 9.065 0 00-6.23-.693L5 14.5m14.8.8l1.402 1.402c1.232 1.232.65 3.318-1.067 3.611A48.309 48.309 0 0112 21c-2.773 0-5.491-.235-8.135-.687-1.718-.293-2.3-2.379-1.067-3.61L5 14.5"})],-1),zo(" Experiments ")]),2)])]),_:1})])]),_:1})]),_:1})]),_:1})])])])}}};function T_(r){typeof queueMicrotask=="function"?queueMicrotask(r):Promise.resolve().then(r).catch(a=>setTimeout(()=>{throw a}))}function _l(){let r=[],a={addEventListener(x,t,w,g){return x.addEventListener(t,w,g),a.add(()=>x.removeEventListener(t,w,g))},requestAnimationFrame(...x){let t=requestAnimationFrame(...x);a.add(()=>cancelAnimationFrame(t))},nextFrame(...x){a.requestAnimationFrame(()=>{a.requestAnimationFrame(...x)})},setTimeout(...x){let t=setTimeout(...x);a.add(()=>clearTimeout(t))},microTask(...x){let t={current:!0};return T_(()=>{t.current&&x[0]()}),a.add(()=>{t.current=!1})},style(x,t,w){let g=x.style.getPropertyValue(t);return Object.assign(x.style,{[t]:w}),this.add(()=>{Object.assign(x.style,{[t]:g})})},group(x){let t=_l();return x(t),this.add(()=>t.dispose())},add(x){return r.push(x),()=>{let t=r.indexOf(x);if(t>=0)for(let w of r.splice(t,1))w()}},dispose(){for(let x of r.splice(0))x()}};return a}function k_(r){let a={called:!1};return(...x)=>{if(!a.called)return a.called=!0,r(...x)}}function Ji(r,...a){r&&a.length>0&&r.classList.add(...a)}function Os(r,...a){r&&a.length>0&&r.classList.remove(...a)}var Ua=(r=>(r.Finished="finished",r.Cancelled="cancelled",r))(Ua||{});function M_(r,a){let x=_l();if(!r)return x.dispose;let{transitionDuration:t,transitionDelay:w}=getComputedStyle(r),[g,s]=[t,w].map(u=>{let[o=0]=u.split(",").filter(Boolean).map(h=>h.includes("ms")?parseFloat(h):parseFloat(h)*1e3).sort((h,y)=>y-h);return o});return g!==0?x.setTimeout(()=>a("finished"),g+s):a("finished"),x.add(()=>a("cancelled")),x.dispose}function Ic(r,a,x,t,w,g){let s=_l(),u=g!==void 0?k_(g):()=>{};return Os(r,...w),Ji(r,...a,...x),s.nextFrame(()=>{Os(r,...x),Ji(r,...t),s.add(M_(r,o=>(Os(r,...t,...a),Ji(r,...w),u(o))))}),s.add(()=>Os(r,...a,...x,...t,...w)),s.add(()=>u("cancelled")),s.dispose}function Un(r=""){return r.split(/\s+/).filter(a=>a.length>1)}let xl=Symbol("TransitionContext");var C_=(r=>(r.Visible="visible",r.Hidden="hidden",r))(C_||{});function O_(){return zt(xl,null)!==null}function P_(){let r=zt(xl,null);if(r===null)throw new Error("A is used but it is missing a parent .");return r}function A_(){let r=zt(wl,null);if(r===null)throw new Error("A is used but it is missing a parent .");return r}let wl=Symbol("NestingContext");function xi(r){return"children"in r?xi(r.children):r.value.filter(({state:a})=>a==="visible").length>0}function vh(r){let a=mt([]),x=mt(!1);Hr(()=>x.value=!0),ls(()=>x.value=!1);function t(g,s=kn.Hidden){let u=a.value.findIndex(({id:o})=>o===g);u!==-1&&(Yn(s,{[kn.Unmount](){a.value.splice(u,1)},[kn.Hidden](){a.value[u].state="hidden"}}),!xi(a)&&x.value&&(r==null||r()))}function w(g){let s=a.value.find(({id:u})=>u===g);return s?s.state!=="visible"&&(s.state="visible"):a.value.push({id:g,state:"visible"}),()=>t(g,kn.Unmount)}return{children:a,register:w,unregister:t}}let bh=ti.RenderStrategy,L_=gr({props:{as:{type:[Object,String],default:"div"},show:{type:[Boolean],default:null},unmount:{type:[Boolean],default:!0},appear:{type:[Boolean],default:!1},enter:{type:[String],default:""},enterFrom:{type:[String],default:""},enterTo:{type:[String],default:""},entered:{type:[String],default:""},leave:{type:[String],default:""},leaveFrom:{type:[String],default:""},leaveTo:{type:[String],default:""}},emits:{beforeEnter:()=>!0,afterEnter:()=>!0,beforeLeave:()=>!0,afterLeave:()=>!0},setup(r,{emit:a,attrs:x,slots:t,expose:w}){let g=mt(0);function s(){g.value|=Cr.Opening,a("beforeEnter")}function u(){g.value&=~Cr.Opening,a("afterEnter")}function o(){g.value|=Cr.Closing,a("beforeLeave")}function h(){g.value&=~Cr.Closing,a("afterLeave")}if(!O_()&&i_())return()=>Jt(_h,{...r,onBeforeEnter:s,onAfterEnter:u,onBeforeLeave:o,onAfterLeave:h},t);let y=mt(null),m=vt(()=>r.unmount?kn.Unmount:kn.Hidden);w({el:y,$el:y});let{show:n,appear:i}=P_(),{register:d,unregister:f}=A_(),p=mt(n.value?"visible":"hidden"),c={value:!0},l=vi(),b={value:!1},v=vh(()=>{!b.value&&p.value!=="hidden"&&(p.value="hidden",f(l),h())});Hr(()=>{let A=d(l);ls(A)}),Jr(()=>{if(m.value===kn.Hidden&&l){if(n.value&&p.value!=="visible"){p.value="visible";return}Yn(p.value,{hidden:()=>f(l),visible:()=>d(l)})}});let j=Un(r.enter),T=Un(r.enterFrom),E=Un(r.enterTo),M=Un(r.entered),P=Un(r.leave),L=Un(r.leaveFrom),C=Un(r.leaveTo);Hr(()=>{Jr(()=>{if(p.value==="visible"){let A=Ut(y);if(A instanceof Comment&&A.data==="")throw new Error("Did you forget to passthrough the `ref` to the actual DOM node?")}})});function I(A){let N=c.value&&!i.value,F=Ut(y);!F||!(F instanceof HTMLElement)||N||(b.value=!0,n.value&&s(),n.value||o(),A(n.value?Ic(F,j,T,E,M,U=>{b.value=!1,U===Ua.Finished&&u()}):Ic(F,P,L,C,M,U=>{b.value=!1,U===Ua.Finished&&(xi(v)||(p.value="hidden",f(l),h()))})))}return Hr(()=>{On([n],(A,N,F)=>{I(F),c.value=!1},{immediate:!0})}),Ir(wl,v),hh(vt(()=>Yn(p.value,{visible:Cr.Open,hidden:Cr.Closed})|g.value)),()=>{let{appear:A,show:N,enter:F,enterFrom:U,enterTo:W,entered:V,leave:H,leaveFrom:Q,leaveTo:ee,...ae}=r,k={ref:y},D={...ae,...i.value&&n.value&&bi.isServer?{class:Vn([x.class,ae.class,...j,...T])}:{}};return So({theirProps:D,ourProps:k,slot:{},slots:t,attrs:x,features:bh,visible:p.value==="visible",name:"TransitionChild"})}}}),R_=L_,_h=gr({inheritAttrs:!1,props:{as:{type:[Object,String],default:"div"},show:{type:[Boolean],default:null},unmount:{type:[Boolean],default:!0},appear:{type:[Boolean],default:!1},enter:{type:[String],default:""},enterFrom:{type:[String],default:""},enterTo:{type:[String],default:""},entered:{type:[String],default:""},leave:{type:[String],default:""},leaveFrom:{type:[String],default:""},leaveTo:{type:[String],default:""}},emits:{beforeEnter:()=>!0,afterEnter:()=>!0,beforeLeave:()=>!0,afterLeave:()=>!0},setup(r,{emit:a,attrs:x,slots:t}){let w=bl(),g=vt(()=>r.show===null&&w!==null?(w.value&Cr.Open)===Cr.Open:r.show);Jr(()=>{if(![!0,!1].includes(g.value))throw new Error('A is used but it is missing a `:show="true | false"` prop.')});let s=mt(g.value?"visible":"hidden"),u=vh(()=>{s.value="hidden"}),o=mt(!0),h={show:g,appear:vt(()=>r.appear||!o.value)};return Hr(()=>{Jr(()=>{o.value=!1,g.value?s.value="visible":xi(u)||(s.value="hidden")})}),Ir(wl,u),Ir(xl,h),()=>{let y=yh(r,["show","appear","unmount","onBeforeEnter","onBeforeLeave","onAfterEnter","onAfterLeave"]),m={unmount:r.unmount};return So({ourProps:{...m,as:"template"},theirProps:{},slot:{},slots:{...t,default:()=>[Jt(R_,{onBeforeEnter:()=>a("beforeEnter"),onAfterEnter:()=>a("afterEnter"),onBeforeLeave:()=>a("beforeLeave"),onAfterLeave:()=>a("afterLeave"),...x,...m,...y},t.default)]},attrs:{},features:bh,visible:s.value==="visible",name:"Transition"})}}});const I_={},D_={class:"flex min-h-screen flex-col bg-background font-display"},F_={class:"mt-4 flex-1"},N_={class:"flex justify-center"};function U_(r,a){const x=E_,t=_h;return wr(),jo("body",D_,[dt(x),Et("main",F_,[Et("div",N_,[dt(t,{show:"",appear:"",as:"template",enter:"transition transform duration-300 ease-out","enter-from":"translate-x-8 opacity-0","enter-to":"translate-x-0 opacity-100",leave:"transition transform duration-300 ease-in","leave-from":"opacity-100","leave-to":"-translate-x-8 opacity-0"},{default:Sr(()=>a[0]||(a[0]=[Et("div",{class:"rounded-xl bg-black/50 p-6 text-center shadow-lg"},[Et("div",{class:"text-6xl font-extrabold tracking-widest text-white"}," Yikes! "),Et("div",{class:"text-base text-gray-200"}," Either the page you're looking for doesn't exist, or something has gone terribly wrong. ")],-1)])),_:1})])])])}const B_=ah(I_,[["render",U_]]),G_={key:0},Dc={__name:"nuxt-root",setup(r){const a=()=>null,x=Dt(),t=x.deferHydration();if(x.isHydrating){const o=x.hooks.hookOnce("app:error",t);jr().beforeEach(o)}const w=!1;Ir(cs,fi()),x.hooks.callHookWith(o=>o.map(h=>h()),"vue:setup");const g=hi(),s=!1;Td((o,h,y)=>{if(x.hooks.callHook("vue:error",o,h,y).catch(m=>console.error("[nuxt] Error in `vue:error` hook",m)),zg(o)&&(o.fatal||o.unhandled))return x.runWithContext(()=>lo(o)),!1});const u=!1;return(o,h)=>(wr(),En(ol,{onResolve:Nt(t)},{default:Sr(()=>[Nt(s)?(wr(),jo("div",G_)):Nt(g)?(wr(),En(Nt(B_),{key:1,error:Nt(g)},null,8,["error"])):Nt(u)?(wr(),En(Nt(a),{key:2,context:Nt(u)},null,8,["context"])):Nt(w)?(wr(),En(Kp(Nt(w)),{key:3})):(wr(),En(Nt(k1),{key:4}))]),_:1},8,["onResolve"]))}};let Fc;{let r;Fc=async function(){var s,u;if(r)return r;const t=!!(((s=window.__NUXT__)==null?void 0:s.serverRendered)??((u=document.getElementById("__NUXT_DATA__"))==null?void 0:u.dataset.ssr)==="true")?my(Dc):py(Dc),w=Mg({vueApp:t});async function g(o){await w.callHook("app:error",o),w.payload.error=w.payload.error||pi(o)}t.config.errorHandler=g;try{await Pg(w,g1)}catch(o){g(o)}try{await w.hooks.callHook("app:created",t),await w.hooks.callHook("app:beforeMount",t),t.mount(Eg),await w.hooks.callHook("app:mounted",t),await Pr()}catch(o){g(o)}return t.config.errorHandler===g&&(t.config.errorHandler=void 0),t},r=Fc().catch(a=>{throw console.error("Error while mounting app:",a),a})}export{pi as $,Rn as A,ex as B,q_ as C,Q_ as D,or as E,cr as F,Pr as G,Bv as H,us as I,Y_ as J,$s as K,Jt as L,rt as M,Dt as N,Ep as O,W_ as P,Vy as Q,Hu as R,_h as S,il as T,di as U,ci as V,J_ as W,Zo as X,Wp as Y,V_ as Z,R1 as _,Nt as a,Zn as a0,Xc as a1,bt as a2,hu as a3,qn as a4,Kt as a5,qs as a6,z_ as a7,$_ as a8,Wa as a9,Kp as aa,E_ as ab,ks as ac,Et as b,jo as c,gr as d,En as e,Sr as f,zo as g,Z_ as h,dt as i,mt as j,vt as k,Vn as l,ah as m,Vg as n,wr as o,Am as p,Hr as q,X_ as r,L_ as s,Qh as t,fi as u,On as v,K_ as w,as as x,ls as y,H_ as z}; diff --git a/_nuxt/-8n5e5Fj.js b/_nuxt/Bbro_qAu.js similarity index 98% rename from _nuxt/-8n5e5Fj.js rename to _nuxt/Bbro_qAu.js index c0d2637d..7d0677d5 100644 --- a/_nuxt/-8n5e5Fj.js +++ b/_nuxt/Bbro_qAu.js @@ -1 +1 @@ -import{o as i,c as s,b as e,i as a,f as c,g as m,a as u,_ as d,k as w,t as g,F as v,r as y,e as x,h as k}from"./BGikLauJ.js";function b(o,t){return i(),s("svg",{xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 16 16",fill:"currentColor","aria-hidden":"true","data-slot":"icon"},[e("path",{"fill-rule":"evenodd",d:"M8.914 6.025a.75.75 0 0 1 1.06 0 3.5 3.5 0 0 1 0 4.95l-2 2a3.5 3.5 0 0 1-5.396-4.402.75.75 0 0 1 1.251.827 2 2 0 0 0 3.085 2.514l2-2a2 2 0 0 0 0-2.828.75.75 0 0 1 0-1.06Z","clip-rule":"evenodd"}),e("path",{"fill-rule":"evenodd",d:"M7.086 9.975a.75.75 0 0 1-1.06 0 3.5 3.5 0 0 1 0-4.95l2-2a3.5 3.5 0 0 1 5.396 4.402.75.75 0 0 1-1.251-.827 2 2 0 0 0-3.085-2.514l-2 2a2 2 0 0 0 0 2.828.75.75 0 0 1 0 1.06Z","clip-rule":"evenodd"})])}const M={class:"flex justify-end"},I={__name:"MoreLink",props:{to:{type:String,required:!0}},setup(o){const t=o;return(l,r)=>{const p=d;return i(),s("div",M,[a(p,{to:t.to,class:"flex items-center text-sm font-bold text-white hover:text-orange-500"},{default:c(()=>[r[0]||(r[0]=m(" More ")),a(u(b),{class:"ml-1 h-5 w-5","aria-hidden":"true"})]),_:1},8,["to"])])}}},j={class:"bg-gradient-to-b from-transparent to-background text-white"},D={class:"container mx-auto space-y-4 py-8 text-white"},L={class:"mb-4 grid grid-cols-1 gap-4 sm:grid-cols-2 lg:grid-cols-3"},P=["src"],V={class:"absolute top-0 flex h-full w-full items-center justify-center bg-gray-500/50"},B={class:"m-2 bg-black/70 p-2"},C={class:"text-xl font-bold text-orange-500"},N=["innerHTML"],S={key:0},T="Experiments",E={__name:"Playground",props:{showImages:{type:Boolean,default:!1},limit:{type:Number,default:null},linkToPlayground:{type:Boolean,default:!1}},setup(o){const t=o,l=[{title:"Conway",description:"Conway's game of life written in vanilla Javascript",link:"/playground/conway",img:"/images/previews/conway.png"},{title:"wm.spoon",description:"Window manager for Hammer Spoon written in Lua",link:"https://github.com/cmpadden/wm.spoon",img:"/images/previews/wm.spoon.png"},{title:"Metronome",description:"A simple metronome for tracking tempo in the browser",link:"/playground/metronome",img:"/images/previews/metronome.png"},{title:"Conjugations",description:"Search and explore the conjugations of 1000 French verbs",link:"/playground/french",img:"/images/previews/french-conjugations.png"},{title:"Mountains",description:"Visualize a gradient of colored waves generated with Perlin noise",link:"/playground/palettes/mountains",img:"/images/previews/noise.png"},{title:"Spectrogram",description:"Visualize the audio from your microphone as a waveform, frequency bars, and a spectrogram",link:"/playground/audio",img:"/images/previews/microphone.png"},{title:"Waves",description:"Demonstration of using p5.js within Vue.js to visualize trigonometric functions",link:"/playground/waves",img:"/images/previews/waves.png"},{title:"MIDI Chords",description:"Identify the chords being played by your attached MIDI device",link:"/playground/chords",img:"/images/previews/chord-identifier.png"},{title:"MIDI Events",description:"View the MIDI events triggered by a MIDI-controller through the Web MIDI API",link:"/playground/midi",img:"/images/previews/midi-events.png"},{title:"Matrix Multiplication",description:"Step through the process of matrix multiplication",link:"/playground/matrix",img:"/images/previews/matrix-multiplication.png"},{title:"Sequence Plotter",description:"Plot the fist 10,000 digits of Pi, or any sequence of digits, in 2-dimensional space",link:"/playground/plotter",img:"/images/previews/plotter.png"}],r=w(()=>t.limit===null||t.limit<=0?l:l.slice(0,t.limit));return(p,q)=>{const h=d,f=d,_=I;return i(),s("section",j,[e("div",D,[a(h,{to:"/playground",class:"font-mono text-3xl font-semibold lowercase underline decoration-orange-500 underline-offset-4 hover:text-orange-500"},{default:c(()=>[m(g(T))]),_:1}),e("div",L,[(i(!0),s(v,null,y(u(r),n=>(i(),x(f,{class:"relative drop-shadow-lg hover:ring-1 hover:ring-white",key:n.title,to:n.link},{default:c(()=>[e("img",{class:"grayscale-1 h-full w-full object-cover",src:n.img||"images/placeholder.png"},null,8,P),e("div",V,[e("div",B,[e("h3",C,g(n.title),1),e("div",{class:"line-clamp-2 text-sm text-gray-300",innerHTML:n.description},null,8,N)])])]),_:2},1032,["to"]))),128))]),o.linkToPlayground?(i(),s("div",S,[a(_,{to:"/playground"})])):k("",!0)])])}}};export{I as _,E as a}; +import{o as i,c as s,b as e,i as a,f as c,g as m,a as u,_ as d,k as w,t as g,F as v,r as y,e as x,h as k}from"./Bb-NKJJJ.js";function b(o,t){return i(),s("svg",{xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 16 16",fill:"currentColor","aria-hidden":"true","data-slot":"icon"},[e("path",{"fill-rule":"evenodd",d:"M8.914 6.025a.75.75 0 0 1 1.06 0 3.5 3.5 0 0 1 0 4.95l-2 2a3.5 3.5 0 0 1-5.396-4.402.75.75 0 0 1 1.251.827 2 2 0 0 0 3.085 2.514l2-2a2 2 0 0 0 0-2.828.75.75 0 0 1 0-1.06Z","clip-rule":"evenodd"}),e("path",{"fill-rule":"evenodd",d:"M7.086 9.975a.75.75 0 0 1-1.06 0 3.5 3.5 0 0 1 0-4.95l2-2a3.5 3.5 0 0 1 5.396 4.402.75.75 0 0 1-1.251-.827 2 2 0 0 0-3.085-2.514l-2 2a2 2 0 0 0 0 2.828.75.75 0 0 1 0 1.06Z","clip-rule":"evenodd"})])}const M={class:"flex justify-end"},I={__name:"MoreLink",props:{to:{type:String,required:!0}},setup(o){const t=o;return(l,r)=>{const p=d;return i(),s("div",M,[a(p,{to:t.to,class:"flex items-center text-sm font-bold text-white hover:text-orange-500"},{default:c(()=>[r[0]||(r[0]=m(" More ")),a(u(b),{class:"ml-1 h-5 w-5","aria-hidden":"true"})]),_:1},8,["to"])])}}},j={class:"bg-gradient-to-b from-transparent to-background text-white"},D={class:"container mx-auto space-y-4 py-8 text-white"},L={class:"mb-4 grid grid-cols-1 gap-4 sm:grid-cols-2 lg:grid-cols-3"},P=["src"],V={class:"absolute top-0 flex h-full w-full items-center justify-center bg-gray-500/50"},B={class:"m-2 bg-black/70 p-2"},C={class:"text-xl font-bold text-orange-500"},N=["innerHTML"],S={key:0},T="Experiments",E={__name:"Playground",props:{showImages:{type:Boolean,default:!1},limit:{type:Number,default:null},linkToPlayground:{type:Boolean,default:!1}},setup(o){const t=o,l=[{title:"Conway",description:"Conway's game of life written in vanilla Javascript",link:"/playground/conway",img:"/images/previews/conway.png"},{title:"wm.spoon",description:"Window manager for Hammer Spoon written in Lua",link:"https://github.com/cmpadden/wm.spoon",img:"/images/previews/wm.spoon.png"},{title:"Metronome",description:"A simple metronome for tracking tempo in the browser",link:"/playground/metronome",img:"/images/previews/metronome.png"},{title:"Conjugations",description:"Search and explore the conjugations of 1000 French verbs",link:"/playground/french",img:"/images/previews/french-conjugations.png"},{title:"Mountains",description:"Visualize a gradient of colored waves generated with Perlin noise",link:"/playground/palettes/mountains",img:"/images/previews/noise.png"},{title:"Spectrogram",description:"Visualize the audio from your microphone as a waveform, frequency bars, and a spectrogram",link:"/playground/audio",img:"/images/previews/microphone.png"},{title:"Waves",description:"Demonstration of using p5.js within Vue.js to visualize trigonometric functions",link:"/playground/waves",img:"/images/previews/waves.png"},{title:"MIDI Chords",description:"Identify the chords being played by your attached MIDI device",link:"/playground/chords",img:"/images/previews/chord-identifier.png"},{title:"MIDI Events",description:"View the MIDI events triggered by a MIDI-controller through the Web MIDI API",link:"/playground/midi",img:"/images/previews/midi-events.png"},{title:"Matrix Multiplication",description:"Step through the process of matrix multiplication",link:"/playground/matrix",img:"/images/previews/matrix-multiplication.png"},{title:"Sequence Plotter",description:"Plot the fist 10,000 digits of Pi, or any sequence of digits, in 2-dimensional space",link:"/playground/plotter",img:"/images/previews/plotter.png"}],r=w(()=>t.limit===null||t.limit<=0?l:l.slice(0,t.limit));return(p,q)=>{const h=d,f=d,_=I;return i(),s("section",j,[e("div",D,[a(h,{to:"/playground",class:"font-mono text-3xl font-semibold lowercase underline decoration-orange-500 underline-offset-4 hover:text-orange-500"},{default:c(()=>[m(g(T))]),_:1}),e("div",L,[(i(!0),s(v,null,y(u(r),n=>(i(),x(f,{class:"relative drop-shadow-lg hover:ring-1 hover:ring-white",key:n.title,to:n.link},{default:c(()=>[e("img",{class:"grayscale-1 h-full w-full object-cover",src:n.img||"images/placeholder.png"},null,8,P),e("div",V,[e("div",B,[e("h3",C,g(n.title),1),e("div",{class:"line-clamp-2 text-sm text-gray-300",innerHTML:n.description},null,8,N)])])]),_:2},1032,["to"]))),128))]),o.linkToPlayground?(i(),s("div",S,[a(_,{to:"/playground"})])):k("",!0)])])}}};export{I as _,E as a}; diff --git a/_nuxt/Dz1t7Gnd.js b/_nuxt/BenPNx-m.js similarity index 82% rename from _nuxt/Dz1t7Gnd.js rename to _nuxt/BenPNx-m.js index 82b519a2..af9a1c5b 100644 --- a/_nuxt/Dz1t7Gnd.js +++ b/_nuxt/BenPNx-m.js @@ -1 +1 @@ -import{_ as l}from"./yJKb_Bp-.js";import{d as n,o as s,e as i,f as o,b as r,l as g,a9 as u,a8 as f}from"./BGikLauJ.js";const h=n({__name:"ProsePre",props:{code:{type:String,default:""},language:{type:String,default:null},filename:{type:String,default:null},highlights:{type:Array,default:()=>[]},meta:{type:String,default:null},class:{type:String,default:null},style:{type:[String,Object],default:null}},setup(e){return(a,d)=>{const t=l;return s(),i(t,{code:e.code,language:e.language,filename:e.filename,highlights:e.highlights,meta:e.meta},{default:o(()=>[r("pre",{class:g(a.$props.class),style:u(e.style)},[f(a.$slots,"default")],6)]),_:3},8,["code","language","filename","highlights","meta"])}}});export{h as default}; +import{_ as l}from"./BMXP4Aud.js";import{d as n,o as s,e as i,f as o,b as r,l as g,a9 as u,a8 as f}from"./Bb-NKJJJ.js";const h=n({__name:"ProsePre",props:{code:{type:String,default:""},language:{type:String,default:null},filename:{type:String,default:null},highlights:{type:Array,default:()=>[]},meta:{type:String,default:null},class:{type:String,default:null},style:{type:[String,Object],default:null}},setup(e){return(a,d)=>{const t=l;return s(),i(t,{code:e.code,language:e.language,filename:e.filename,highlights:e.highlights,meta:e.meta},{default:o(()=>[r("pre",{class:g(a.$props.class),style:u(e.style)},[f(a.$slots,"default")],6)]),_:3},8,["code","language","filename","highlights","meta"])}}});export{h as default}; diff --git a/_nuxt/BdJQj3nn.js b/_nuxt/BfDVGm6D.js similarity index 88% rename from _nuxt/BdJQj3nn.js rename to _nuxt/BfDVGm6D.js index fb032bbe..6821e875 100644 --- a/_nuxt/BdJQj3nn.js +++ b/_nuxt/BfDVGm6D.js @@ -1 +1 @@ -import{d as p,I as f,k as u,o as s,c as n,a as i,a8 as r}from"./BGikLauJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH1",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=u(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h1))});return(e,k)=>(s(),n("h1",{id:t.id},[i(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; +import{d as p,I as f,k as u,o as s,c as n,a as i,a8 as r}from"./Bb-NKJJJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH1",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=u(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h1))});return(e,k)=>(s(),n("h1",{id:t.id},[i(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; diff --git a/_nuxt/RX6TINi_.js b/_nuxt/Bj-Uk9MO.js similarity index 89% rename from _nuxt/RX6TINi_.js rename to _nuxt/Bj-Uk9MO.js index faf2df6a..9bdbdfee 100644 --- a/_nuxt/RX6TINi_.js +++ b/_nuxt/Bj-Uk9MO.js @@ -1 +1 @@ -import{d as n,k as o,R as c,K as h,I as p,U as l,o as u,e as d,a,aa as m}from"./BGikLauJ.js";const g="img",S=n({__name:"ProseImg",props:{src:{type:String,default:""},alt:{type:String,default:""},width:{type:[String,Number],default:void 0},height:{type:[String,Number],default:void 0}},setup(r){const t=r,i=o(()=>{var s;if((s=t.src)!=null&&s.startsWith("/")&&!t.src.startsWith("//")){const e=c(h(p().app.baseURL));if(e!=="/"&&!t.src.startsWith(e))return l(e,t.src)}return t.src});return(s,e)=>(u(),d(m(a(g)),{src:a(i),alt:t.alt,width:t.width,height:t.height},null,8,["src","alt","width","height"]))}});export{S as default}; +import{d as n,k as o,R as c,K as h,I as p,U as l,o as u,e as d,a,aa as m}from"./Bb-NKJJJ.js";const g="img",S=n({__name:"ProseImg",props:{src:{type:String,default:""},alt:{type:String,default:""},width:{type:[String,Number],default:void 0},height:{type:[String,Number],default:void 0}},setup(r){const t=r,i=o(()=>{var s;if((s=t.src)!=null&&s.startsWith("/")&&!t.src.startsWith("//")){const e=c(h(p().app.baseURL));if(e!=="/"&&!t.src.startsWith(e))return l(e,t.src)}return t.src});return(s,e)=>(u(),d(m(a(g)),{src:a(i),alt:t.alt,width:t.width,height:t.height},null,8,["src","alt","width","height"]))}});export{S as default}; diff --git a/_nuxt/DO9Ai2bz.js b/_nuxt/BlMa8jv2.js similarity index 81% rename from _nuxt/DO9Ai2bz.js rename to _nuxt/BlMa8jv2.js index 2788d082..e67eedfd 100644 --- a/_nuxt/DO9Ai2bz.js +++ b/_nuxt/BlMa8jv2.js @@ -1 +1 @@ -import{d as o,o as n,e as s,f,a8 as p,_}from"./BGikLauJ.js";const d=o({__name:"ProseA",props:{href:{type:String,default:""},target:{type:String,default:void 0,required:!1}},setup(t){const e=t;return(r,c)=>{const a=_;return n(),s(a,{href:e.href,target:e.target},{default:f(()=>[p(r.$slots,"default")]),_:3},8,["href","target"])}}});export{d as default}; +import{d as o,o as n,e as s,f,a8 as p,_}from"./Bb-NKJJJ.js";const d=o({__name:"ProseA",props:{href:{type:String,default:""},target:{type:String,default:void 0,required:!1}},setup(t){const e=t;return(r,c)=>{const a=_;return n(),s(a,{href:e.href,target:e.target},{default:f(()=>[p(r.$slots,"default")]),_:3},8,["href","target"])}}});export{d as default}; diff --git a/_nuxt/DCgnBWC9.js b/_nuxt/Bl_PaSYd.js similarity index 78% rename from _nuxt/DCgnBWC9.js rename to _nuxt/Bl_PaSYd.js index 63457968..87af2ac4 100644 --- a/_nuxt/DCgnBWC9.js +++ b/_nuxt/Bl_PaSYd.js @@ -1 +1 @@ -import{c as t,b as r,o}from"./BGikLauJ.js";const s={class:"flex h-[768px] justify-center border-2 border-gray-500"},c={__name:"metronome",setup(a){return(m,e)=>(o(),t("main",s,e[0]||(e[0]=[r("iframe",{src:"https://simple-tempo.com",width:"100%",height:"100%"},null,-1)])))}};export{c as default}; +import{c as t,b as r,o}from"./Bb-NKJJJ.js";const s={class:"flex h-[768px] justify-center border-2 border-gray-500"},c={__name:"metronome",setup(a){return(m,e)=>(o(),t("main",s,e[0]||(e[0]=[r("iframe",{src:"https://simple-tempo.com",width:"100%",height:"100%"},null,-1)])))}};export{c as default}; diff --git a/_nuxt/Dkf0i8RZ.js b/_nuxt/BmBFiau8.js similarity index 90% rename from _nuxt/Dkf0i8RZ.js rename to _nuxt/BmBFiau8.js index d97b9c6d..c7e478da 100644 --- a/_nuxt/Dkf0i8RZ.js +++ b/_nuxt/BmBFiau8.js @@ -1 +1 @@ -import{u as A,j as g,w as R,k as m,c as r,b as o,F as d,r as v,a as i,h as S,o as l,t as p,i as f,f as h,g as D,l as C,_ as F}from"./BGikLauJ.js";import{_ as L}from"./yGJPFCFo.js";import j from"./DukX5DGJ.js";import{u as z}from"./CgY3mHZ8.js";import{q as E}from"./UWyBikFq.js";import"./C-v3KzvZ.js";import"./Dnd51l0P.js";import"./DJ6fbLCw.js";import"./IShi1APO.js";const M={class:"container mx-auto font-mono text-white"},G={class:"grid grid-cols-4 gap-4"},H={class:"col-span-4 lg:col-span-3"},I={class:"grid grid-cols-10 gap-y-4 lg:gap-y-6"},J={class:"col-span-10 lg:col-span-2"},K={class:"col-span-10 lg:col-span-8"},O={class:"flex-col space-y-2"},P={class:"col-span-4 border-l border-gray-700 pl-4 lg:col-span-1"},Q={class:"my-2 space-y-2"},U=["onClick"],W={key:0,class:"my-2 space-y-2"},X=["onClick"],ce={__name:"index",async setup(Y){let c,x;const u=A(),a=g([]);u.query.category&&(a.value=[u.query.category]);const n=g([]);u.query.tag&&(n.value=[u.query.tag]);const k=g(!0),{data:_}=([c,x]=R(()=>z("articles",()=>E().only(["_id","_path","title","description","date","img","author","tags","categories","img","excerpt","summary"]).sort({date:-1}).find())),c=await c,x(),c),w=m(()=>[...new Set(_.value.map(e=>e.categories).flat().sort())]),b=m(()=>[...new Set(_.value.map(e=>e.tags).flat().sort())]),q=m(()=>_.value.filter(e=>!(a.value.length>0&&!a.value.every(s=>e.categories.includes(s))||n.value.length>0&&!n.value.every(s=>e.tags.includes(s)))));function N(e){n.value.includes(e)?n.value=n.value.filter(s=>s!==e):n.value.push(e)}function T(e){a.value.includes(e)?a.value=a.value.filter(s=>s!==e):a.value.push(e)}return(e,s)=>{const B=F,V=L,$=j;return l(),r("section",M,[s[2]||(s[2]=o("h1",{class:"my-6 text-2xl font-extrabold"},"Blog",-1)),o("div",G,[o("div",H,[o("div",I,[(l(!0),r(d,null,v(i(q),t=>(l(),r(d,{key:t._id},[o("div",J,p(t.date),1),o("div",K,[o("div",O,[f(B,{class:"text-orange-500",to:t._path},{default:h(()=>[D(p(t.title),1)]),_:2},1032,["to"]),f($,{value:t},{default:h(()=>[f(V,{class:"line-clamp-5 text-xs text-gray-400",value:{body:t.excerpt}},null,8,["value"])]),_:2},1032,["value"])])])],64))),128))])]),o("div",P,[o("div",Q,[s[0]||(s[0]=o("p",{class:"text-xl font-bold"},"Categories",-1)),(l(!0),r(d,null,v(i(w),t=>(l(),r("div",{key:t,class:C([{"bg-orange-500rtext-white":i(a).includes(t)},"cursor-pointer p-1 text-sm text-gray-400 hover:bg-orange-500 hover:text-white"]),onClick:y=>T(t)},p(t),11,U))),128))]),i(k)?(l(),r("div",W,[s[1]||(s[1]=o("p",{class:"text-xl font-bold"},"Tags",-1)),(l(!0),r(d,null,v(i(b),(t,y)=>(l(),r("div",{key:y,class:C([{"bg-orange-500 text-white":i(n).includes(t)},"cursor-pointer select-none p-1 text-sm text-gray-400 hover:bg-orange-500 hover:text-white"]),onClick:Z=>N(t)},p(t),11,X))),128))])):S("",!0)])])])}}};export{ce as default}; +import{u as A,j as g,w as R,k as m,c as r,b as o,F as d,r as v,a as i,h as S,o as l,t as p,i as f,f as h,g as D,l as C,_ as F}from"./Bb-NKJJJ.js";import{_ as L}from"./CS06-0Od.js";import j from"./DrabSix5.js";import{u as z}from"./2mhhWi5_.js";import{q as E}from"./jaxqT3Hp.js";import"./C-v3KzvZ.js";import"./Dnd51l0P.js";import"./mjvvwssR.js";import"./IShi1APO.js";const M={class:"container mx-auto font-mono text-white"},G={class:"grid grid-cols-4 gap-4"},H={class:"col-span-4 lg:col-span-3"},I={class:"grid grid-cols-10 gap-y-4 lg:gap-y-6"},J={class:"col-span-10 lg:col-span-2"},K={class:"col-span-10 lg:col-span-8"},O={class:"flex-col space-y-2"},P={class:"col-span-4 border-l border-gray-700 pl-4 lg:col-span-1"},Q={class:"my-2 space-y-2"},U=["onClick"],W={key:0,class:"my-2 space-y-2"},X=["onClick"],ce={__name:"index",async setup(Y){let c,x;const u=A(),a=g([]);u.query.category&&(a.value=[u.query.category]);const n=g([]);u.query.tag&&(n.value=[u.query.tag]);const k=g(!0),{data:_}=([c,x]=R(()=>z("articles",()=>E().only(["_id","_path","title","description","date","img","author","tags","categories","img","excerpt","summary"]).sort({date:-1}).find())),c=await c,x(),c),w=m(()=>[...new Set(_.value.map(e=>e.categories).flat().sort())]),b=m(()=>[...new Set(_.value.map(e=>e.tags).flat().sort())]),q=m(()=>_.value.filter(e=>!(a.value.length>0&&!a.value.every(s=>e.categories.includes(s))||n.value.length>0&&!n.value.every(s=>e.tags.includes(s)))));function N(e){n.value.includes(e)?n.value=n.value.filter(s=>s!==e):n.value.push(e)}function T(e){a.value.includes(e)?a.value=a.value.filter(s=>s!==e):a.value.push(e)}return(e,s)=>{const B=F,V=L,$=j;return l(),r("section",M,[s[2]||(s[2]=o("h1",{class:"my-6 text-2xl font-extrabold"},"Blog",-1)),o("div",G,[o("div",H,[o("div",I,[(l(!0),r(d,null,v(i(q),t=>(l(),r(d,{key:t._id},[o("div",J,p(t.date),1),o("div",K,[o("div",O,[f(B,{class:"text-orange-500",to:t._path},{default:h(()=>[D(p(t.title),1)]),_:2},1032,["to"]),f($,{value:t},{default:h(()=>[f(V,{class:"line-clamp-5 text-xs text-gray-400",value:{body:t.excerpt}},null,8,["value"])]),_:2},1032,["value"])])])],64))),128))])]),o("div",P,[o("div",Q,[s[0]||(s[0]=o("p",{class:"text-xl font-bold"},"Categories",-1)),(l(!0),r(d,null,v(i(w),t=>(l(),r("div",{key:t,class:C([{"bg-orange-500rtext-white":i(a).includes(t)},"cursor-pointer p-1 text-sm text-gray-400 hover:bg-orange-500 hover:text-white"]),onClick:y=>T(t)},p(t),11,U))),128))]),i(k)?(l(),r("div",W,[s[1]||(s[1]=o("p",{class:"text-xl font-bold"},"Tags",-1)),(l(!0),r(d,null,v(i(b),(t,y)=>(l(),r("div",{key:y,class:C([{"bg-orange-500 text-white":i(n).includes(t)},"cursor-pointer select-none p-1 text-sm text-gray-400 hover:bg-orange-500 hover:text-white"]),onClick:Z=>N(t)},p(t),11,X))),128))])):S("",!0)])])])}}};export{ce as default}; diff --git a/_nuxt/CruaMelD.js b/_nuxt/BmkabxXg.js similarity index 65% rename from _nuxt/CruaMelD.js rename to _nuxt/BmkabxXg.js index 130d40e1..7fe45893 100644 --- a/_nuxt/CruaMelD.js +++ b/_nuxt/BmkabxXg.js @@ -1 +1 @@ -import{m as o,o as n,c as r,a8 as c}from"./BGikLauJ.js";const s={};function t(e,a){return n(),r("code",null,[c(e.$slots,"default")])}const d=o(s,[["render",t]]);export{d as default}; +import{m as o,o as n,c as r,a8 as c}from"./Bb-NKJJJ.js";const s={};function t(e,a){return n(),r("code",null,[c(e.$slots,"default")])}const d=o(s,[["render",t]]);export{d as default}; diff --git a/_nuxt/CnOV4Igh.js b/_nuxt/BqebzOVN.js similarity index 94% rename from _nuxt/CnOV4Igh.js rename to _nuxt/BqebzOVN.js index e51c74a3..f73618c9 100644 --- a/_nuxt/CnOV4Igh.js +++ b/_nuxt/BqebzOVN.js @@ -1 +1 @@ -import{d as f,o as n,c as a,b as t,h as u,g as s,i as r,f as m,F as b,r as x,t as _,_ as v,w,a as y}from"./BGikLauJ.js";import{_ as k,a as $}from"./-8n5e5Fj.js";import{q as C}from"./UWyBikFq.js";import"./IShi1APO.js";import"./DJ6fbLCw.js";const D={class:"container mx-auto pb-8"},S={class:"flex flex-wrap"},A={class:"w-full"},B={class:"lg:text-normal space-y-4 text-justify font-medium tracking-wide text-white sm:text-base"},I=f({__name:"AboutMe",setup(d){return(o,e)=>(n(),a("section",D,[t("div",S,[t("div",A,[t("div",B,[u("",!0),e[1]||(e[1]=t("div",null,[s(" I help educate data engineers while building the future of data orchestration at "),t("a",{class:"border-b-2 border-orange-500",target:"_blank",rel:"noopener noreferrer",href:"https://dagster.io/"},"Dagster"),s(". ")],-1)),e[2]||(e[2]=t("div",null,[s(" Previously, I worked at "),t("a",{class:"border-b-2 border-orange-500",target:"_blank",rel:"noopener noreferrer",href:"https://www.gemini.com/"},"Gemini"),s(" building the data platform that provided company-wide insights into the exchange and business. At Georgetown University's "),t("a",{class:"border-b-2 border-orange-500",target:"_blank",rel:"noopener noreferrer",href:"https://mccourt.georgetown.edu/research/the-massive-data-institute/"},"Massive Data Institute"),s(" building data warehousing, processing solutions, and portals to aid social scientists and researchers to leverage large-scale organic data. And previously I provided consulting for financial institutions and government agencies in the D.C. area around data practices, and identity and access management. ")],-1))])])])]))}}),N={class:"bg-gradient-to-b from-background to-transparent"},P={class:"container mx-auto space-y-4"},L={class:"mb-4 grid grid-cols-1 gap-4 md:grid-cols-2"},M={class:"h-full space-y-4 bg-black/70 p-4 text-white drop-shadow-lg hover:ring-1 hover:ring-white"},V={class:"flex-1 text-lg font-bold md:text-xl"},j={key:0,class:"text-sm font-light text-gray-400"},q={class:"line-clamp-3 text-sm text-gray-300"},F={__name:"BlogPosts",props:{articles:Object,show_dates:!1},setup(d){return(o,e)=>{const l=v,g=k;return n(),a("section",N,[t("div",P,[r(l,{to:"/articles",class:"font-mono text-3xl font-semibold text-white underline decoration-orange-500 underline-offset-4 hover:text-orange-500"},{default:m(()=>e[0]||(e[0]=[s(" blog posts ")])),_:1}),t("div",L,[(n(!0),a(b,null,x(d.articles,(i,c)=>(n(),a("div",{key:c},[r(l,{to:i._path,external:""},{default:m(()=>[t("div",M,[t("div",null,[t("div",V,_(i.title),1),d.show_dates?(n(),a("div",j,_(new Date(i.date).toLocaleDateString("en-US",{month:"long",day:"numeric",year:"numeric"})),1)):u("",!0)]),t("div",q,_(i.description),1)])]),_:2},1032,["to"])]))),128))]),t("div",null,[r(g,{to:"/articles"})])])])}}},G={class:"bg-emerald-950 bg-[url('/images/noise.svg')]"},H={__name:"index",async setup(d){let o,e;const l=([o,e]=w(()=>C().only(["_id","_path","title","description","date","img","author","tags"]).sort({date:-1}).limit(4).find()),o=await o,e(),o);return(g,i)=>{const c=I,p=F,h=$;return n(),a("div",null,[r(c),t("div",G,[u("",!0),r(p,{articles:y(l),show_dates:!0},null,8,["articles"]),r(h,{limit:9,showImages:"",linkToPlayground:""})])])}}};export{H as default}; +import{d as f,o as n,c as a,b as t,h as u,g as s,i as r,f as m,F as b,r as x,t as _,_ as v,w,a as y}from"./Bb-NKJJJ.js";import{_ as k,a as $}from"./Bbro_qAu.js";import{q as C}from"./jaxqT3Hp.js";import"./IShi1APO.js";import"./mjvvwssR.js";const D={class:"container mx-auto pb-8"},S={class:"flex flex-wrap"},A={class:"w-full"},B={class:"lg:text-normal space-y-4 text-justify font-medium tracking-wide text-white sm:text-base"},I=f({__name:"AboutMe",setup(d){return(o,e)=>(n(),a("section",D,[t("div",S,[t("div",A,[t("div",B,[u("",!0),e[1]||(e[1]=t("div",null,[s(" I help educate data engineers while building the future of data orchestration at "),t("a",{class:"border-b-2 border-orange-500",target:"_blank",rel:"noopener noreferrer",href:"https://dagster.io/"},"Dagster"),s(". ")],-1)),e[2]||(e[2]=t("div",null,[s(" Previously, I worked at "),t("a",{class:"border-b-2 border-orange-500",target:"_blank",rel:"noopener noreferrer",href:"https://www.gemini.com/"},"Gemini"),s(" building the data platform that provided company-wide insights into the exchange and business. At Georgetown University's "),t("a",{class:"border-b-2 border-orange-500",target:"_blank",rel:"noopener noreferrer",href:"https://mccourt.georgetown.edu/research/the-massive-data-institute/"},"Massive Data Institute"),s(" building data warehousing, processing solutions, and portals to aid social scientists and researchers to leverage large-scale organic data. And previously I provided consulting for financial institutions and government agencies in the D.C. area around data practices, and identity and access management. ")],-1))])])])]))}}),N={class:"bg-gradient-to-b from-background to-transparent"},P={class:"container mx-auto space-y-4"},L={class:"mb-4 grid grid-cols-1 gap-4 md:grid-cols-2"},M={class:"h-full space-y-4 bg-black/70 p-4 text-white drop-shadow-lg hover:ring-1 hover:ring-white"},V={class:"flex-1 text-lg font-bold md:text-xl"},j={key:0,class:"text-sm font-light text-gray-400"},q={class:"line-clamp-3 text-sm text-gray-300"},F={__name:"BlogPosts",props:{articles:Object,show_dates:!1},setup(d){return(o,e)=>{const l=v,g=k;return n(),a("section",N,[t("div",P,[r(l,{to:"/articles",class:"font-mono text-3xl font-semibold text-white underline decoration-orange-500 underline-offset-4 hover:text-orange-500"},{default:m(()=>e[0]||(e[0]=[s(" blog posts ")])),_:1}),t("div",L,[(n(!0),a(b,null,x(d.articles,(i,c)=>(n(),a("div",{key:c},[r(l,{to:i._path,external:""},{default:m(()=>[t("div",M,[t("div",null,[t("div",V,_(i.title),1),d.show_dates?(n(),a("div",j,_(new Date(i.date).toLocaleDateString("en-US",{month:"long",day:"numeric",year:"numeric"})),1)):u("",!0)]),t("div",q,_(i.description),1)])]),_:2},1032,["to"])]))),128))]),t("div",null,[r(g,{to:"/articles"})])])])}}},G={class:"bg-emerald-950 bg-[url('/images/noise.svg')]"},H={__name:"index",async setup(d){let o,e;const l=([o,e]=w(()=>C().only(["_id","_path","title","description","date","img","author","tags"]).sort({date:-1}).limit(4).find()),o=await o,e(),o);return(g,i)=>{const c=I,p=F,h=$;return n(),a("div",null,[r(c),t("div",G,[u("",!0),r(p,{articles:y(l),show_dates:!0},null,8,["articles"]),r(h,{limit:9,showImages:"",linkToPlayground:""})])])}}};export{H as default}; diff --git a/_nuxt/6mOJeL1q.js b/_nuxt/BsGN5WoF.js similarity index 65% rename from _nuxt/6mOJeL1q.js rename to _nuxt/BsGN5WoF.js index 2bf119d0..518aaf2f 100644 --- a/_nuxt/6mOJeL1q.js +++ b/_nuxt/BsGN5WoF.js @@ -1 +1 @@ -import{m as o,o as r,c as t,a8 as a}from"./BGikLauJ.js";const s={};function c(e,n){return r(),t("table",null,[a(e.$slots,"default")])}const f=o(s,[["render",c]]);export{f as default}; +import{m as o,o as r,c as t,a8 as a}from"./Bb-NKJJJ.js";const s={};function c(e,n){return r(),t("table",null,[a(e.$slots,"default")])}const f=o(s,[["render",c]]);export{f as default}; diff --git a/_nuxt/BxFzRrKR.js b/_nuxt/BxFzRrKR.js new file mode 100644 index 00000000..15ef6e23 --- /dev/null +++ b/_nuxt/BxFzRrKR.js @@ -0,0 +1 @@ +import{_ as o}from"./CS06-0Od.js";import"./Bb-NKJJJ.js";import"./C-v3KzvZ.js";import"./Dnd51l0P.js";import"./mjvvwssR.js";import"./IShi1APO.js";export{o as default}; diff --git a/_nuxt/VwgkOAbI.js b/_nuxt/C-O5BI11.js similarity index 87% rename from _nuxt/VwgkOAbI.js rename to _nuxt/C-O5BI11.js index f147ee36..c18fc2ba 100644 --- a/_nuxt/VwgkOAbI.js +++ b/_nuxt/C-O5BI11.js @@ -1 +1 @@ -import{a as d,v as w,u as y,G as D,H,I as g,d as S,J as _,K as b,L as p}from"./BGikLauJ.js";import q from"./DukX5DGJ.js";import x from"./AY-jWpd0.js";import"./yGJPFCFo.js";import"./C-v3KzvZ.js";import"./Dnd51l0P.js";import"./DJ6fbLCw.js";import"./IShi1APO.js";import"./CgY3mHZ8.js";import"./UWyBikFq.js";const a=(u,s=y())=>{const e=d(u),m=g();w(()=>d(u),(n=e)=>{if(!s.path||!n)return;const t=Object.assign({},(n==null?void 0:n.head)||{});t.meta=[...t.meta||[]],t.link=[...t.link||[]];const r=t.title||(n==null?void 0:n.title);r&&(t.title=r),m.public.content.host;const c=(t==null?void 0:t.description)||(n==null?void 0:n.description);c&&t.meta.filter(l=>l.name==="description").length===0&&t.meta.push({name:"description",content:c}),t!=null&&t.image||(n==null||n.image),D(()=>H(t))},{immediate:!0})},$=S({name:"ContentDoc",props:{tag:{type:String,required:!1,default:"div"},excerpt:{type:Boolean,default:!1},path:{type:String,required:!1,default:void 0},query:{type:Object,required:!1,default:void 0},head:{type:Boolean,required:!1,default:void 0}},render(u){const{contentHead:s}=g().public.content,e=_(),{tag:m,excerpt:f,path:n,query:t,head:r}=u,c=r===void 0?s:r,l={...t||{},path:n||(t==null?void 0:t.path)||b(y().path),find:"one"},v=(o,i)=>p("pre",null,JSON.stringify({message:"You should use slots with ",slot:o,data:i},null,2));return p(x,l,{default:e!=null&&e.default?({data:o,refresh:i,isPartial:C})=>{var h;return c&&a(o),(h=e.default)==null?void 0:h.call(e,{doc:o,refresh:i,isPartial:C,excerpt:f,...this.$attrs})}:({data:o})=>(c&&a(o),p(q,{value:o,excerpt:f,tag:m,...this.$attrs},{empty:i=>e!=null&&e.empty?e.empty(i):v("default",o)})),empty:o=>{var i;return((i=e==null?void 0:e.empty)==null?void 0:i.call(e,o))||p("p",null,"Document is empty, overwrite this content with #empty slot in .")},"not-found":o=>{var i;return((i=e==null?void 0:e["not-found"])==null?void 0:i.call(e,o))||p("p",null,"Document not found, overwrite this content with #not-found slot in .")}})}}),K=$;export{K as default}; +import{a as d,v as w,u as y,G as D,H,I as g,d as S,J as _,K as b,L as p}from"./Bb-NKJJJ.js";import q from"./DrabSix5.js";import x from"./cufizuWq.js";import"./CS06-0Od.js";import"./C-v3KzvZ.js";import"./Dnd51l0P.js";import"./mjvvwssR.js";import"./IShi1APO.js";import"./2mhhWi5_.js";import"./jaxqT3Hp.js";const a=(u,s=y())=>{const e=d(u),m=g();w(()=>d(u),(n=e)=>{if(!s.path||!n)return;const t=Object.assign({},(n==null?void 0:n.head)||{});t.meta=[...t.meta||[]],t.link=[...t.link||[]];const r=t.title||(n==null?void 0:n.title);r&&(t.title=r),m.public.content.host;const c=(t==null?void 0:t.description)||(n==null?void 0:n.description);c&&t.meta.filter(l=>l.name==="description").length===0&&t.meta.push({name:"description",content:c}),t!=null&&t.image||(n==null||n.image),D(()=>H(t))},{immediate:!0})},$=S({name:"ContentDoc",props:{tag:{type:String,required:!1,default:"div"},excerpt:{type:Boolean,default:!1},path:{type:String,required:!1,default:void 0},query:{type:Object,required:!1,default:void 0},head:{type:Boolean,required:!1,default:void 0}},render(u){const{contentHead:s}=g().public.content,e=_(),{tag:m,excerpt:f,path:n,query:t,head:r}=u,c=r===void 0?s:r,l={...t||{},path:n||(t==null?void 0:t.path)||b(y().path),find:"one"},v=(o,i)=>p("pre",null,JSON.stringify({message:"You should use slots with ",slot:o,data:i},null,2));return p(x,l,{default:e!=null&&e.default?({data:o,refresh:i,isPartial:C})=>{var h;return c&&a(o),(h=e.default)==null?void 0:h.call(e,{doc:o,refresh:i,isPartial:C,excerpt:f,...this.$attrs})}:({data:o})=>(c&&a(o),p(q,{value:o,excerpt:f,tag:m,...this.$attrs},{empty:i=>e!=null&&e.empty?e.empty(i):v("default",o)})),empty:o=>{var i;return((i=e==null?void 0:e.empty)==null?void 0:i.call(e,o))||p("p",null,"Document is empty, overwrite this content with #empty slot in .")},"not-found":o=>{var i;return((i=e==null?void 0:e["not-found"])==null?void 0:i.call(e,o))||p("p",null,"Document not found, overwrite this content with #not-found slot in .")}})}}),K=$;export{K as default}; diff --git a/_nuxt/-R_hya3P.js b/_nuxt/C6rjZEvg.js similarity index 79% rename from _nuxt/-R_hya3P.js rename to _nuxt/C6rjZEvg.js index bbe625c6..f7ebffbb 100644 --- a/_nuxt/-R_hya3P.js +++ b/_nuxt/C6rjZEvg.js @@ -1 +1 @@ -import{c as t,b as r,o}from"./BGikLauJ.js";const a={class:"flex h-[1024px] justify-center border-2 border-gray-500"},i={__name:"conway",setup(s){return(n,e)=>(o(),t("main",a,e[0]||(e[0]=[r("iframe",{src:"https://cmpadden.github.io/conway/",width:"100%",height:"100%"},null,-1)])))}};export{i as default}; +import{c as t,b as r,o}from"./Bb-NKJJJ.js";const a={class:"flex h-[1024px] justify-center border-2 border-gray-500"},i={__name:"conway",setup(s){return(n,e)=>(o(),t("main",a,e[0]||(e[0]=[r("iframe",{src:"https://cmpadden.github.io/conway/",width:"100%",height:"100%"},null,-1)])))}};export{i as default}; diff --git a/_nuxt/BRTC5WMg.js b/_nuxt/CCH9cHpw.js similarity index 97% rename from _nuxt/BRTC5WMg.js rename to _nuxt/CCH9cHpw.js index b8b92a08..ba37f670 100644 --- a/_nuxt/BRTC5WMg.js +++ b/_nuxt/CCH9cHpw.js @@ -1 +1 @@ -import{k as C,z as g,A as B,B as E,d as O,j as q,w as U,v as V,o as n,c as l,b as o,a as r,F as y,C as I,D as K,E as S,r as k,g as z,t as x,h as W}from"./BGikLauJ.js";import{h as M}from"./IShi1APO.js";import{u as G}from"./CgY3mHZ8.js";function H(v,i,p){const[e={},c]=typeof i=="string"?[{},i]:[i,p],s=C(()=>g(v)),t=e.key||M([c,typeof s.value=="string"?s.value:"",...P(e)]);if(!t||typeof t!="string")throw new TypeError("[nuxt] [useFetch] key must be a string: "+t);const h=t===c?"$f"+t:t;if(!e.baseURL&&typeof s.value=="string"&&s.value[0]==="/"&&s.value[1]==="/")throw new Error('[nuxt] [useFetch] the request URL must not start with "//".');const{server:w,lazy:b,default:j,transform:D,pick:F,watch:d,immediate:a,getCachedData:u,deep:m,dedupe:T,...L}=e,_=B({...E,...L,cache:typeof e.cache=="boolean"?void 0:e.cache}),R={server:w,lazy:b,default:j,transform:D,pick:F,immediate:a,getCachedData:u,deep:m,dedupe:T,watch:d===!1?[]:[_,s,...d||[]]};let f;return G(h,()=>{var N;(N=f==null?void 0:f.abort)==null||N.call(f,"Request aborted as another request to the same endpoint was initiated."),f=typeof AbortController<"u"?new AbortController:{};const A=g(e.timeout);let $;return A&&($=setTimeout(()=>f.abort("Request aborted due to timeout."),A),f.signal.onabort=()=>clearTimeout($)),(e.$fetch||globalThis.$fetch)(s.value,{signal:f.signal,..._}).finally(()=>{clearTimeout($)})},R)}function P(v){var p;const i=[((p=g(v.method))==null?void 0:p.toUpperCase())||"GET",g(v.baseURL)];for(const e of[v.params||v.query]){const c=g(e);if(!c)continue;const s={};for(const[t,h]of Object.entries(c))s[g(t)]=g(h);i.push(s)}return i}const Q={class:"flex h-screen flex-col font-mono"},J={class:"flex flex-1 overflow-hidden"},X={class:"flex flex-1 flex-col overflow-y-auto bg-amber-100 p-8"},Y={class:"inline-flex flex-col justify-center py-4 text-gray-600"},Z={key:0,class:"mt-2 w-full border border-gray-100 bg-white"},ee=["onClick"],te={class:"flex p-4"},oe={class:"text-cyan-600 flex-1 text-center text-3xl font-bold"},se={class:"text-amber-500"},ae={class:"grid grid-cols-1 gap-8 md:grid-cols-3 xl:grid-cols-4 xl:gap-12"},re={class:"border-cyan-600 h-full space-y-3 rounded-md border-2 bg-white p-4 shadow shadow-amber-500"},ne={class:"text-2xl font-semibold capitalize text-gray-700"},le={class:"text-gray-500"},ie={key:1,class:"italic"},ve=O({__name:"index",async setup(v){let i,p;const e=q(0),c=q(""),s=q([]),{data:t}=([i,p]=U(()=>H("/1000_french_conjugations.json",{transform:d=>d},"$lRKQKt9Al2")),i=await i,p(),i),h=C(()=>{if(t.value)return t.value[e.value]}),w=C(()=>{if(t.value!==null)return e.value===0?void 0:t.value[e.value-1]}),b=C(()=>{if(t.value!==null)return e.value+1===t.value.length?void 0:t.value[e.value+1]});V(c,d=>{if(s.value=[],!d)return;let a=0;s.value=t.value.filter(u=>u.word.normalize("NFD").replace(new RegExp("\\p{Diacritic}","gu"),"").includes(d)&&a<=5?(a++,!0):!1)});const j=()=>{t.value&&e.value{e.value>0&&(e.value-=1)},F=d=>{c.value="",s.value=[],e.value=d.word_popularity-1};return(d,a)=>(n(),l("div",Q,[o("div",J,[a[3]||(a[3]=o("div",{class:"flex w-20 bg-amber-500"},[o("div",{class:"absolute bottom-10 w-20 -rotate-90 transform whitespace-nowrap"},[o("div",{class:"text-4xl font-bold text-white"}," 1000 French Conjugations ")])],-1)),o("div",X,[r(t)&&r(t).length>0&&r(h)?(n(),l(y,{key:0},[a[2]||(a[2]=o("div",{class:"flex rounded-xl bg-amber-500 p-4"},[o("div",{class:"text-center text-xl italic text-white"}," Click a word to the left or right to cycle through the 1000 most popular French verb conjugations! ")],-1)),o("div",Y,[I(o("input",{"onUpdate:modelValue":a[0]||(a[0]=u=>S(c)?c.value=u:null),class:"rounded-xl border border-2 border-gray-400 p-2 pl-8 focus:border-amber-500 focus:outline-none",type:"search",autocomplete:"off",placeholder:"Search Words"},null,512),[[K,r(c)]]),r(s).length>0?(n(),l("ul",Z,[(n(!0),l(y,null,k(r(s),u=>(n(),l("li",{key:u.word,class:"relative cursor-pointer py-1 pl-8 pr-2 hover:bg-yellow-50 hover:text-gray-900",onClick:m=>F(u)},[a[1]||(a[1]=o("svg",{class:"absolute left-2 top-2 h-4 w-4",xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 20 20",fill:"currentColor"},[o("path",{d:"M12.293 5.293a1 1 0 011.414 0l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414-1.414L14.586 11H3a1 1 0 110-2h11.586l-2.293-2.293a1 1 0 010-1.414z"})],-1)),z(" "+x(u.word),1)],8,ee))),128))])):W("",!0)]),o("div",te,[o("div",{class:"flex-1 cursor-pointer select-none text-center text-2xl text-gray-400",onClick:D},x(r(w)?r(w).word:"-"),1),o("div",oe,[o("span",se," #"+x(r(h).word_popularity),1),z(" "+x(r(h).word),1)]),o("div",{class:"flex-1 cursor-pointer select-none text-center text-2xl text-gray-400",onClick:j},x(r(b)?r(b).word:"-"),1)]),o("div",ae,[(n(!0),l(y,null,k(r(h).conjugations,(u,m)=>(n(),l("div",{key:m},[o("div",re,[o("h1",ne,x(m),1),o("div",le,[(n(!0),l(y,null,k(u,(T,L)=>(n(),l("p",{key:L},[(n(!0),l(y,null,k(T,(_,R)=>(n(),l("i",{key:R},x(_.text),1))),128))]))),128))])])]))),128))])],64)):(n(),l("div",ie,"Loading..."))])])]))}});export{ve as default}; +import{k as C,z as g,A as B,B as E,d as O,j as q,w as U,v as V,o as n,c as l,b as o,a as r,F as y,C as I,D as K,E as S,r as k,g as z,t as x,h as W}from"./Bb-NKJJJ.js";import{h as M}from"./IShi1APO.js";import{u as G}from"./2mhhWi5_.js";function H(v,i,p){const[e={},c]=typeof i=="string"?[{},i]:[i,p],s=C(()=>g(v)),t=e.key||M([c,typeof s.value=="string"?s.value:"",...P(e)]);if(!t||typeof t!="string")throw new TypeError("[nuxt] [useFetch] key must be a string: "+t);const h=t===c?"$f"+t:t;if(!e.baseURL&&typeof s.value=="string"&&s.value[0]==="/"&&s.value[1]==="/")throw new Error('[nuxt] [useFetch] the request URL must not start with "//".');const{server:w,lazy:b,default:j,transform:D,pick:F,watch:d,immediate:a,getCachedData:u,deep:m,dedupe:T,...L}=e,_=B({...E,...L,cache:typeof e.cache=="boolean"?void 0:e.cache}),R={server:w,lazy:b,default:j,transform:D,pick:F,immediate:a,getCachedData:u,deep:m,dedupe:T,watch:d===!1?[]:[_,s,...d||[]]};let f;return G(h,()=>{var N;(N=f==null?void 0:f.abort)==null||N.call(f,"Request aborted as another request to the same endpoint was initiated."),f=typeof AbortController<"u"?new AbortController:{};const A=g(e.timeout);let $;return A&&($=setTimeout(()=>f.abort("Request aborted due to timeout."),A),f.signal.onabort=()=>clearTimeout($)),(e.$fetch||globalThis.$fetch)(s.value,{signal:f.signal,..._}).finally(()=>{clearTimeout($)})},R)}function P(v){var p;const i=[((p=g(v.method))==null?void 0:p.toUpperCase())||"GET",g(v.baseURL)];for(const e of[v.params||v.query]){const c=g(e);if(!c)continue;const s={};for(const[t,h]of Object.entries(c))s[g(t)]=g(h);i.push(s)}return i}const Q={class:"flex h-screen flex-col font-mono"},J={class:"flex flex-1 overflow-hidden"},X={class:"flex flex-1 flex-col overflow-y-auto bg-amber-100 p-8"},Y={class:"inline-flex flex-col justify-center py-4 text-gray-600"},Z={key:0,class:"mt-2 w-full border border-gray-100 bg-white"},ee=["onClick"],te={class:"flex p-4"},oe={class:"text-cyan-600 flex-1 text-center text-3xl font-bold"},se={class:"text-amber-500"},ae={class:"grid grid-cols-1 gap-8 md:grid-cols-3 xl:grid-cols-4 xl:gap-12"},re={class:"border-cyan-600 h-full space-y-3 rounded-md border-2 bg-white p-4 shadow shadow-amber-500"},ne={class:"text-2xl font-semibold capitalize text-gray-700"},le={class:"text-gray-500"},ie={key:1,class:"italic"},ve=O({__name:"index",async setup(v){let i,p;const e=q(0),c=q(""),s=q([]),{data:t}=([i,p]=U(()=>H("/1000_french_conjugations.json",{transform:d=>d},"$lRKQKt9Al2")),i=await i,p(),i),h=C(()=>{if(t.value)return t.value[e.value]}),w=C(()=>{if(t.value!==null)return e.value===0?void 0:t.value[e.value-1]}),b=C(()=>{if(t.value!==null)return e.value+1===t.value.length?void 0:t.value[e.value+1]});V(c,d=>{if(s.value=[],!d)return;let a=0;s.value=t.value.filter(u=>u.word.normalize("NFD").replace(new RegExp("\\p{Diacritic}","gu"),"").includes(d)&&a<=5?(a++,!0):!1)});const j=()=>{t.value&&e.value{e.value>0&&(e.value-=1)},F=d=>{c.value="",s.value=[],e.value=d.word_popularity-1};return(d,a)=>(n(),l("div",Q,[o("div",J,[a[3]||(a[3]=o("div",{class:"flex w-20 bg-amber-500"},[o("div",{class:"absolute bottom-10 w-20 -rotate-90 transform whitespace-nowrap"},[o("div",{class:"text-4xl font-bold text-white"}," 1000 French Conjugations ")])],-1)),o("div",X,[r(t)&&r(t).length>0&&r(h)?(n(),l(y,{key:0},[a[2]||(a[2]=o("div",{class:"flex rounded-xl bg-amber-500 p-4"},[o("div",{class:"text-center text-xl italic text-white"}," Click a word to the left or right to cycle through the 1000 most popular French verb conjugations! ")],-1)),o("div",Y,[I(o("input",{"onUpdate:modelValue":a[0]||(a[0]=u=>S(c)?c.value=u:null),class:"rounded-xl border border-2 border-gray-400 p-2 pl-8 focus:border-amber-500 focus:outline-none",type:"search",autocomplete:"off",placeholder:"Search Words"},null,512),[[K,r(c)]]),r(s).length>0?(n(),l("ul",Z,[(n(!0),l(y,null,k(r(s),u=>(n(),l("li",{key:u.word,class:"relative cursor-pointer py-1 pl-8 pr-2 hover:bg-yellow-50 hover:text-gray-900",onClick:m=>F(u)},[a[1]||(a[1]=o("svg",{class:"absolute left-2 top-2 h-4 w-4",xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 20 20",fill:"currentColor"},[o("path",{d:"M12.293 5.293a1 1 0 011.414 0l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414-1.414L14.586 11H3a1 1 0 110-2h11.586l-2.293-2.293a1 1 0 010-1.414z"})],-1)),z(" "+x(u.word),1)],8,ee))),128))])):W("",!0)]),o("div",te,[o("div",{class:"flex-1 cursor-pointer select-none text-center text-2xl text-gray-400",onClick:D},x(r(w)?r(w).word:"-"),1),o("div",oe,[o("span",se," #"+x(r(h).word_popularity),1),z(" "+x(r(h).word),1)]),o("div",{class:"flex-1 cursor-pointer select-none text-center text-2xl text-gray-400",onClick:j},x(r(b)?r(b).word:"-"),1)]),o("div",ae,[(n(!0),l(y,null,k(r(h).conjugations,(u,m)=>(n(),l("div",{key:m},[o("div",re,[o("h1",ne,x(m),1),o("div",le,[(n(!0),l(y,null,k(u,(T,L)=>(n(),l("p",{key:L},[(n(!0),l(y,null,k(T,(_,R)=>(n(),l("i",{key:R},x(_.text),1))),128))]))),128))])])]))),128))])],64)):(n(),l("div",ie,"Loading..."))])])]))}});export{ve as default}; diff --git a/_nuxt/CUESTjlD.js b/_nuxt/CMNpy2Is.js similarity index 75% rename from _nuxt/CUESTjlD.js rename to _nuxt/CMNpy2Is.js index c45e6895..9c581975 100644 --- a/_nuxt/CUESTjlD.js +++ b/_nuxt/CMNpy2Is.js @@ -1,2 +1,2 @@ -const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["./qIrQuZow.js","./BGikLauJ.js","./UWyBikFq.js","./IShi1APO.js","./DJ6fbLCw.js","./C-v3KzvZ.js"])))=>i.map(i=>d[i]); -import{h as f}from"./IShi1APO.js";import{u as m}from"./CgY3mHZ8.js";import{I as v,M as l,N as d,O as g,E as y,d as h,P as _,k as w,J as C,L as p,_ as P}from"./BGikLauJ.js";import{q as N,w as c,e as $,s as x,j as E,u as T}from"./UWyBikFq.js";import{u as j}from"./DJ6fbLCw.js";const S=async t=>{const{content:e}=v().public;typeof(t==null?void 0:t.params)!="function"&&(t=N(t));const n=t.params(),o=e.experimental.stripQueryParameters?c(`/navigation/${`${f(n)}.${e.integrity}`}/${$(n)}.json`):c(`/navigation/${f(n)}.${e.integrity}.json`);if(x())return(await l(()=>import("./qIrQuZow.js"),__vite__mapDeps([0,1,2,3,4,5]),import.meta.url).then(i=>i.generateNavigation))(n);const a=await $fetch(o,{method:"GET",responseType:"json",params:e.experimental.stripQueryParameters?void 0:{_params:E(n),previewToken:j().getPreviewToken()}});if(typeof a=="string"&&a.startsWith(""))throw new Error("Not found");return a},b="$s";function D(...t){const e=typeof t[t.length-1]=="string"?t.pop():void 0;typeof t[0]!="string"&&t.unshift(e);const[n,o]=t;if(!n||typeof n!="string")throw new TypeError("[nuxt] [useState] key must be a string: "+n);if(o!==void 0&&typeof o!="function")throw new Error("[nuxt] [useState] init must be a function: "+o);const a=b+n,r=d(),i=g(r.payload.state,a);if(i.value===void 0&&o){const s=o();if(y(s))return r.payload.state[a]=s,s;i.value=s}return i}const R=h({name:"ContentNavigation",props:{query:{type:Object,required:!1,default:void 0}},async setup(t){const{query:e}=_(t),n=w(()=>{var a;return typeof((a=e.value)==null?void 0:a.params)=="function"?e.value.params():e.value});if(!n.value&&D("dd-navigation").value){const{navigation:a}=T();return{navigation:a}}const{data:o}=await m(`content-navigation-${f(n.value)}`,()=>S(n.value));return{navigation:o}},render(t){const e=C(),{navigation:n}=t,o=i=>p(P,{to:i._path},()=>i.title),a=(i,s)=>p("ul",s?{"data-level":s}:null,i.map(u=>u.children?p("li",null,[o(u),a(u.children,s+1)]):p("li",null,o(u)))),r=i=>a(i,0);return e!=null&&e.default?e.default({navigation:n,...this.$attrs}):r(n)}}),I=R;export{I as default}; +const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["./CgzK-JD6.js","./Bb-NKJJJ.js","./jaxqT3Hp.js","./IShi1APO.js","./mjvvwssR.js","./C-v3KzvZ.js"])))=>i.map(i=>d[i]); +import{h as f}from"./IShi1APO.js";import{u as m}from"./2mhhWi5_.js";import{I as v,M as l,N as d,O as g,E as y,d as h,P as _,k as w,J as C,L as p,_ as P}from"./Bb-NKJJJ.js";import{q as N,w as c,e as $,s as x,j as E,u as T}from"./jaxqT3Hp.js";import{u as j}from"./mjvvwssR.js";const S=async t=>{const{content:e}=v().public;typeof(t==null?void 0:t.params)!="function"&&(t=N(t));const n=t.params(),o=e.experimental.stripQueryParameters?c(`/navigation/${`${f(n)}.${e.integrity}`}/${$(n)}.json`):c(`/navigation/${f(n)}.${e.integrity}.json`);if(x())return(await l(()=>import("./CgzK-JD6.js"),__vite__mapDeps([0,1,2,3,4,5]),import.meta.url).then(i=>i.generateNavigation))(n);const a=await $fetch(o,{method:"GET",responseType:"json",params:e.experimental.stripQueryParameters?void 0:{_params:E(n),previewToken:j().getPreviewToken()}});if(typeof a=="string"&&a.startsWith(""))throw new Error("Not found");return a},b="$s";function D(...t){const e=typeof t[t.length-1]=="string"?t.pop():void 0;typeof t[0]!="string"&&t.unshift(e);const[n,o]=t;if(!n||typeof n!="string")throw new TypeError("[nuxt] [useState] key must be a string: "+n);if(o!==void 0&&typeof o!="function")throw new Error("[nuxt] [useState] init must be a function: "+o);const a=b+n,r=d(),i=g(r.payload.state,a);if(i.value===void 0&&o){const s=o();if(y(s))return r.payload.state[a]=s,s;i.value=s}return i}const R=h({name:"ContentNavigation",props:{query:{type:Object,required:!1,default:void 0}},async setup(t){const{query:e}=_(t),n=w(()=>{var a;return typeof((a=e.value)==null?void 0:a.params)=="function"?e.value.params():e.value});if(!n.value&&D("dd-navigation").value){const{navigation:a}=T();return{navigation:a}}const{data:o}=await m(`content-navigation-${f(n.value)}`,()=>S(n.value));return{navigation:o}},render(t){const e=C(),{navigation:n}=t,o=i=>p(P,{to:i._path},()=>i.title),a=(i,s)=>p("ul",s?{"data-level":s}:null,i.map(u=>u.children?p("li",null,[o(u),a(u.children,s+1)]):p("li",null,o(u)))),r=i=>a(i,0);return e!=null&&e.default?e.default({navigation:n,...this.$attrs}):r(n)}}),I=R;export{I as default}; diff --git a/_nuxt/D0Omvic7.js b/_nuxt/CQnAi_qF.js similarity index 99% rename from _nuxt/D0Omvic7.js rename to _nuxt/CQnAi_qF.js index aac263a1..86240719 100644 --- a/_nuxt/D0Omvic7.js +++ b/_nuxt/CQnAi_qF.js @@ -1 +1 @@ -import{m as g,c as u,b as n,C as d,D as a,t as c,o as h}from"./BGikLauJ.js";const p={data(){return{height:0,width:0,distance:3,strokeWidth:1,startingX:125,startingY:150,startingAngle:0,sequence:"3.1415926535897932384626433832795028841971693993751058209749445923078164062862089986280348253421170679821480865132823066470938446095505822317253594081284811174502841027019385211055596446229489549303819644288109756659334461284756482337867831652712019091456485669234603486104543266482133936072602491412737245870066063155881748815209209628292540917153643678925903600113305305488204665213841469519415116094330572703657595919530921861173819326117931051185480744623799627495673518857527248912279381830119491298336733624406566430860213949463952247371907021798609437027705392171762931767523846748184676694051320005681271452635608277857713427577896091736371787214684409012249534301465495853710507922796892589235420199561121290219608640344181598136297747713099605187072113499999983729780499510597317328160963185950244594553469083026425223082533446850352619311881710100031378387528865875332083814206171776691473035982534904287554687311595628638823537875937519577818577805321712268066130019278766111959092164201989380952572010654858632788659361533818279682303019520353018529689957736225994138912497217752834791315155748572424541506959508295331168617278558890750983817546374649393192550604009277016711390098488240128583616035637076601047101819429555961989467678374494482553797747268471040475346462080466842590694912933136770289891521047521620569660240580381501935112533824300355876402474964732639141992726042699227967823547816360093417216412199245863150302861829745557067498385054945885869269956909272107975093029553211653449872027559602364806654991198818347977535663698074265425278625518184175746728909777727938000816470600161452491921732172147723501414419735685481613611573525521334757418494684385233239073941433345477624168625189835694855620992192221842725502542568876717904946016534668049886272327917860857843838279679766814541009538837863609506800642251252051173929848960841284886269456042419652850222106611863067442786220391949450471237137869609563643719172874677646575739624138908658326459958133904780275900994657640789512694683983525957098258226205224894077267194782684826014769909026401363944374553050682034962524517493996514314298091906592509372216964615157098583874105978859597729754989301617539284681382686838689427741559918559252459539594310499725246808459872736446958486538367362226260991246080512438843904512441365497627807977156914359977001296160894416948685558484063534220722258284886481584560285060168427394522674676788952521385225499546667278239864565961163548862305774564980355936345681743241125150760694794510965960940252288797108931456691368672287489405601015033086179286809208747609178249385890097149096759852613655497818931297848216829989487226588048575640142704775551323796414515237462343645428584447952658678210511413547357395231134271661021359695362314429524849371871101457654035902799344037420073105785390621983874478084784896833214457138687519435064302184531910484810053706146806749192781911979399520614196634287544406437451237181921799983910159195618146751426912397489409071864942319615679452080951465502252316038819301420937621378559566389377870830390697920773467221825625996615014215030680384477345492026054146659252014974428507325186660021324340881907104863317346496514539057962685610055081066587969981635747363840525714591028970641401109712062804390397595156771577004203378699360072305587631763594218731251471205329281918261861258673215791984148488291644706095752706957220917567116722910981690915280173506712748583222871835209353965725121083579151369882091444210067510334671103141267111369908658516398315019701651511685171437657618351556508849099898599823873455283316355076479185358932261854896321329330898570642046752590709154814165498594616371802709819943099244889575712828905923233260972997120844335732654893823911932597463667305836041428138830320382490375898524374417029132765618093773444030707469211201913020330380197621101100449293215160842444859637669838952286847831235526582131449576857262433441893039686426243410773226978028073189154411010446823252716201052652272111660396665573092547110557853763466820653109896526918620564769312570586356620185581007293606598764861179104533488503461136576867532494416680396265797877185560845529654126654085306143444318586769751456614068007002378776591344017127494704205622305389945613140711270004078547332699390814546646458807972708266830634328587856983052358089330657574067954571637752542021149557615814002501262285941302164715509792592309907965473761255176567513575178296664547791745011299614890304639947132962107340437518957359614589019389713111790429782856475032031986915140287080859904801094121472213179476477726224142548545403321571853061422881375850430633217518297986622371721591607716692547487389866549494501146540628433663937900397692656721463853067360965712091807638327166416274888800786925602902284721040317211860820419000422966171196377921337575114959501566049631862947265473642523081770367515906735023507283540567040386743513622224771589150495309844489333096340878076932599397805419341447377441842631298608099888687413260472156951623965864573021631598193195167353812974167729478672422924654366800980676928238280689964004824354037014163149658979409243237896907069779422362508221688957383798623001593776471651228935786015881617557829735233446042815126272037343146531977774160319906655418763979293344195215413418994854447345673831624993419131814809277771038638773431772075456545322077709212019051660962804909263601975988281613323166636528619326686336062735676303544776280350450777235547105859548702790814356240145171806246436267945612753181340783303362542327839449753824372058353114771199260638133467768796959703098339130771098704085913374641442822772634659470474587847787201927715280731767907707157213444730605700733492436931138350493163128404251219256517980694113528013147013047816437885185290928545201165839341965621349143415956258658655705526904965209858033850722426482939728584783163057777560688876446248246857926039535277348030480290058760758251047470916439613626760449256274204208320856611906254543372131535958450687724602901618766795240616342522577195429162991930645537799140373404328752628889639958794757291746426357455254079091451357111369410911939325191076020825202618798531887705842972591677813149699009019211697173727847684726860849003377024242916513005005168323364350389517029893922334517220138128069650117844087451960121228599371623130171144484640903890644954440061986907548516026327505298349187407866808818338510228334508504860825039302133219715518430635455007668282949304137765527939751754613953984683393638304746119966538581538420568533862186725233402830871123282789212507712629463229563989898935821167456270102183564622013496715188190973038119800497340723961036854066431939509790190699639552453005450580685501956730229219139339185680344903982059551002263535361920419947455385938102343955449597783779023742161727111723643435439478221818528624085140066604433258885698670543154706965747458550332323342107301545940516553790686627333799585115625784322988273723198987571415957811196358330059408730681216028764962867446047746491599505497374256269010490377819868359381465741268049256487985561453723478673303904688383436346553794986419270563872931748723320837601123029911367938627089438799362016295154133714248928307220126901475466847653576164773794675200490757155527819653621323926406160136358155907422020203187277605277219005561484255518792530343513984425322341576233610642506390497500865627109535919465897514131034822769306247435363256916078154781811528436679570611086153315044521274739245449454236828860613408414863776700961207151249140430272538607648236341433462351897576645216413767969031495019108575984423919862916421939949072362346468441173940326591840443780513338945257423995082965912285085558215725031071257012668302402929525220118726767562204154205161841634847565169998116141010029960783869092916030288400269104140792886215078424516709087000699282120660418371806535567252532567532861291042487761825829765157959847035622262934860034158722980534989650226291748788202734209222245339856264766914905562842503912757710284027998066365825488926488025456610172967026640765590429099456815065265305371829412703369313785178609040708667114965583434347693385781711386455873678123014587687126603489139095620099393610310291616152881384379099042317473363948045759314931405297634757481193567091101377517210080315590248530906692037671922033229094334676851422144773793937517034436619910403375111735471918550464490263655128162288244625759163330391072253837421821408835086573917715096828874782656995995744906617583441375223970968340800535598491754173818839994469748676265516582765848358845314277568790029095170283529716344562129640435231176006651012412006597558512761785838292041974844236080071930457618932349229279650198751872127267507981255470958904556357921221033346697499235630254947802490114195212382815309114079073860251522742995818072471625916685451333123948049470791191532673430282441860414263639548000448002670496248201792896476697583183271314251702969234889627668440323260927524960357996469256504936818360900323809293459588970695365349406034021665443755890045632882250545255640564482465151875471196218443965825337543885690941130315095261793780029741207665147939425902989695946995565761218656196733786236256125216320862869222103274889218654364802296780705765615144632046927906821207388377814233562823608963208068222468012248261177185896381409183903673672220888321513755600372798394004152970028783076670944474560134556417254370906979396122571429894671543578468788614445812314593571984922528471605049221242470141214780573455105008019086996033027634787081081754501193071412233908663938339529425786905076431006383519834389341596131854347546495569781038293097164651438407007073604112373599843452251610507027056235266012764848308407611830130527932054274628654036036745328651057065874882256981579367897669742205750596834408697350201410206723585020072452256326513410559240190274216248439140359989535394590944070469120914093870012645600162374288021092764579310657922955249887275846101264836999892256959688159205600101655256375678",sketch:void 0}},methods:{render(i){const e="0123456789",o=["#FFBA08","#FAA307","#F48C06","#E85D04","#DC2F02","#D00000","#9D0208","#6A040F","#370617","#03071E"],r={x:this.startingX,y:this.startingY,a:this.startingAngle};i.strokeWeight(this.strokeWidth),i.clear(),i.background(o[9]),i.beginShape(),i.vertex(r.x,r.y);for(let t=0;t{o.setup=()=>{o.createCanvas(this.width,this.height),o.noLoop(),o.noFill(),o.stroke(255),this.sketch=o,this.render(this.sketch)}};new this.$p5(e,"canvas")}},m={class:"select-none bg-gradient-to-b from-green-800 to-gray-800"},x={class:"m-8 grid place-items-center"},b={class:"font-mono text-white"},f={class:"my-2 md:w-full"},v={class:"grid-rows-auto my-2 grid grid-cols-2 gap-4 border-2 border-white p-2 md:grid-rows-1"},k={class:"flex flex-col"},y={for:"distanceInput",class:"text-right"},w={class:"flex flex-col"},A={for:"strokeInput",class:"text-right"},_={class:"flex flex-col"},I={for:"startingXInput",class:"text-right"},C={class:"flex flex-col"},D={for:"startingYInput",class:"text-right"},V={class:"flex flex-col"},W={for:"startingAngleInput",class:"text-right"};function X(i,e,o,r,t,l){return h(),u("div",m,[n("div",x,[n("div",b,[e[11]||(e[11]=n("div",{id:"canvas-container",class:"mb-2 h-96 border-2 border-white"},[n("div",{id:"canvas"})],-1)),n("div",f,[d(n("textarea",{readonly:"","onUpdate:modelValue":e[0]||(e[0]=s=>t.sequence=s),class:"block h-24 w-full border-2 border-white bg-black p-2 text-white placeholder:italic placeholder:text-slate-400 focus:border-2 focus:border-yellow-500 focus:outline-none sm:text-sm",placeholder:"Enter a sequence of numbers...",type:"text"},null,512),[[a,t.sequence]])]),n("div",v,[n("div",k,[n("label",y,"Segment Length ["+c(t.distance)+"]",1),d(n("input",{id:"distanceInput","onUpdate:modelValue":e[1]||(e[1]=s=>t.distance=s),onChange:e[2]||(e[2]=s=>l.render(t.sketch)),type:"range",min:"1",max:"25",class:"accent-black"},null,544),[[a,t.distance,void 0,{number:!0}]])]),n("div",w,[n("label",A,"Stroke Width ["+c(t.strokeWidth)+"]",1),d(n("input",{id:"strokeInput","onUpdate:modelValue":e[3]||(e[3]=s=>t.strokeWidth=s),onChange:e[4]||(e[4]=s=>l.render(t.sketch)),type:"range",min:"1",max:"10",class:"accent-black"},null,544),[[a,t.strokeWidth,void 0,{number:!0}]])]),n("div",_,[n("label",I,"X Position ["+c(t.startingX)+"]",1),d(n("input",{id:"startingXInput","onUpdate:modelValue":e[5]||(e[5]=s=>t.startingX=s),onChange:e[6]||(e[6]=s=>l.render(t.sketch)),type:"range",min:"0",max:"500",class:"accent-black"},null,544),[[a,t.startingX,void 0,{number:!0}]])]),n("div",C,[n("label",D,"Y Position ["+c(t.startingY)+"]",1),d(n("input",{id:"startingYInput","onUpdate:modelValue":e[7]||(e[7]=s=>t.startingY=s),onChange:e[8]||(e[8]=s=>l.render(t.sketch)),type:"range",min:"0",max:"500",class:"accent-black"},null,544),[[a,t.startingY,void 0,{number:!0}]])]),n("div",V,[n("label",W,"Angle ["+c(t.startingAngle)+"]",1),d(n("input",{id:"startingAngleInput","onUpdate:modelValue":e[9]||(e[9]=s=>t.startingAngle=s),onChange:e[10]||(e[10]=s=>l.render(t.sketch)),type:"range",min:"-180",max:"180",class:"accent-black"},null,544),[[a,t.startingAngle,void 0,{number:!0}]])])])])])])}const E=g(p,[["render",X]]);export{E as default}; +import{m as g,c as u,b as n,C as d,D as a,t as c,o as h}from"./Bb-NKJJJ.js";const p={data(){return{height:0,width:0,distance:3,strokeWidth:1,startingX:125,startingY:150,startingAngle:0,sequence:"3.1415926535897932384626433832795028841971693993751058209749445923078164062862089986280348253421170679821480865132823066470938446095505822317253594081284811174502841027019385211055596446229489549303819644288109756659334461284756482337867831652712019091456485669234603486104543266482133936072602491412737245870066063155881748815209209628292540917153643678925903600113305305488204665213841469519415116094330572703657595919530921861173819326117931051185480744623799627495673518857527248912279381830119491298336733624406566430860213949463952247371907021798609437027705392171762931767523846748184676694051320005681271452635608277857713427577896091736371787214684409012249534301465495853710507922796892589235420199561121290219608640344181598136297747713099605187072113499999983729780499510597317328160963185950244594553469083026425223082533446850352619311881710100031378387528865875332083814206171776691473035982534904287554687311595628638823537875937519577818577805321712268066130019278766111959092164201989380952572010654858632788659361533818279682303019520353018529689957736225994138912497217752834791315155748572424541506959508295331168617278558890750983817546374649393192550604009277016711390098488240128583616035637076601047101819429555961989467678374494482553797747268471040475346462080466842590694912933136770289891521047521620569660240580381501935112533824300355876402474964732639141992726042699227967823547816360093417216412199245863150302861829745557067498385054945885869269956909272107975093029553211653449872027559602364806654991198818347977535663698074265425278625518184175746728909777727938000816470600161452491921732172147723501414419735685481613611573525521334757418494684385233239073941433345477624168625189835694855620992192221842725502542568876717904946016534668049886272327917860857843838279679766814541009538837863609506800642251252051173929848960841284886269456042419652850222106611863067442786220391949450471237137869609563643719172874677646575739624138908658326459958133904780275900994657640789512694683983525957098258226205224894077267194782684826014769909026401363944374553050682034962524517493996514314298091906592509372216964615157098583874105978859597729754989301617539284681382686838689427741559918559252459539594310499725246808459872736446958486538367362226260991246080512438843904512441365497627807977156914359977001296160894416948685558484063534220722258284886481584560285060168427394522674676788952521385225499546667278239864565961163548862305774564980355936345681743241125150760694794510965960940252288797108931456691368672287489405601015033086179286809208747609178249385890097149096759852613655497818931297848216829989487226588048575640142704775551323796414515237462343645428584447952658678210511413547357395231134271661021359695362314429524849371871101457654035902799344037420073105785390621983874478084784896833214457138687519435064302184531910484810053706146806749192781911979399520614196634287544406437451237181921799983910159195618146751426912397489409071864942319615679452080951465502252316038819301420937621378559566389377870830390697920773467221825625996615014215030680384477345492026054146659252014974428507325186660021324340881907104863317346496514539057962685610055081066587969981635747363840525714591028970641401109712062804390397595156771577004203378699360072305587631763594218731251471205329281918261861258673215791984148488291644706095752706957220917567116722910981690915280173506712748583222871835209353965725121083579151369882091444210067510334671103141267111369908658516398315019701651511685171437657618351556508849099898599823873455283316355076479185358932261854896321329330898570642046752590709154814165498594616371802709819943099244889575712828905923233260972997120844335732654893823911932597463667305836041428138830320382490375898524374417029132765618093773444030707469211201913020330380197621101100449293215160842444859637669838952286847831235526582131449576857262433441893039686426243410773226978028073189154411010446823252716201052652272111660396665573092547110557853763466820653109896526918620564769312570586356620185581007293606598764861179104533488503461136576867532494416680396265797877185560845529654126654085306143444318586769751456614068007002378776591344017127494704205622305389945613140711270004078547332699390814546646458807972708266830634328587856983052358089330657574067954571637752542021149557615814002501262285941302164715509792592309907965473761255176567513575178296664547791745011299614890304639947132962107340437518957359614589019389713111790429782856475032031986915140287080859904801094121472213179476477726224142548545403321571853061422881375850430633217518297986622371721591607716692547487389866549494501146540628433663937900397692656721463853067360965712091807638327166416274888800786925602902284721040317211860820419000422966171196377921337575114959501566049631862947265473642523081770367515906735023507283540567040386743513622224771589150495309844489333096340878076932599397805419341447377441842631298608099888687413260472156951623965864573021631598193195167353812974167729478672422924654366800980676928238280689964004824354037014163149658979409243237896907069779422362508221688957383798623001593776471651228935786015881617557829735233446042815126272037343146531977774160319906655418763979293344195215413418994854447345673831624993419131814809277771038638773431772075456545322077709212019051660962804909263601975988281613323166636528619326686336062735676303544776280350450777235547105859548702790814356240145171806246436267945612753181340783303362542327839449753824372058353114771199260638133467768796959703098339130771098704085913374641442822772634659470474587847787201927715280731767907707157213444730605700733492436931138350493163128404251219256517980694113528013147013047816437885185290928545201165839341965621349143415956258658655705526904965209858033850722426482939728584783163057777560688876446248246857926039535277348030480290058760758251047470916439613626760449256274204208320856611906254543372131535958450687724602901618766795240616342522577195429162991930645537799140373404328752628889639958794757291746426357455254079091451357111369410911939325191076020825202618798531887705842972591677813149699009019211697173727847684726860849003377024242916513005005168323364350389517029893922334517220138128069650117844087451960121228599371623130171144484640903890644954440061986907548516026327505298349187407866808818338510228334508504860825039302133219715518430635455007668282949304137765527939751754613953984683393638304746119966538581538420568533862186725233402830871123282789212507712629463229563989898935821167456270102183564622013496715188190973038119800497340723961036854066431939509790190699639552453005450580685501956730229219139339185680344903982059551002263535361920419947455385938102343955449597783779023742161727111723643435439478221818528624085140066604433258885698670543154706965747458550332323342107301545940516553790686627333799585115625784322988273723198987571415957811196358330059408730681216028764962867446047746491599505497374256269010490377819868359381465741268049256487985561453723478673303904688383436346553794986419270563872931748723320837601123029911367938627089438799362016295154133714248928307220126901475466847653576164773794675200490757155527819653621323926406160136358155907422020203187277605277219005561484255518792530343513984425322341576233610642506390497500865627109535919465897514131034822769306247435363256916078154781811528436679570611086153315044521274739245449454236828860613408414863776700961207151249140430272538607648236341433462351897576645216413767969031495019108575984423919862916421939949072362346468441173940326591840443780513338945257423995082965912285085558215725031071257012668302402929525220118726767562204154205161841634847565169998116141010029960783869092916030288400269104140792886215078424516709087000699282120660418371806535567252532567532861291042487761825829765157959847035622262934860034158722980534989650226291748788202734209222245339856264766914905562842503912757710284027998066365825488926488025456610172967026640765590429099456815065265305371829412703369313785178609040708667114965583434347693385781711386455873678123014587687126603489139095620099393610310291616152881384379099042317473363948045759314931405297634757481193567091101377517210080315590248530906692037671922033229094334676851422144773793937517034436619910403375111735471918550464490263655128162288244625759163330391072253837421821408835086573917715096828874782656995995744906617583441375223970968340800535598491754173818839994469748676265516582765848358845314277568790029095170283529716344562129640435231176006651012412006597558512761785838292041974844236080071930457618932349229279650198751872127267507981255470958904556357921221033346697499235630254947802490114195212382815309114079073860251522742995818072471625916685451333123948049470791191532673430282441860414263639548000448002670496248201792896476697583183271314251702969234889627668440323260927524960357996469256504936818360900323809293459588970695365349406034021665443755890045632882250545255640564482465151875471196218443965825337543885690941130315095261793780029741207665147939425902989695946995565761218656196733786236256125216320862869222103274889218654364802296780705765615144632046927906821207388377814233562823608963208068222468012248261177185896381409183903673672220888321513755600372798394004152970028783076670944474560134556417254370906979396122571429894671543578468788614445812314593571984922528471605049221242470141214780573455105008019086996033027634787081081754501193071412233908663938339529425786905076431006383519834389341596131854347546495569781038293097164651438407007073604112373599843452251610507027056235266012764848308407611830130527932054274628654036036745328651057065874882256981579367897669742205750596834408697350201410206723585020072452256326513410559240190274216248439140359989535394590944070469120914093870012645600162374288021092764579310657922955249887275846101264836999892256959688159205600101655256375678",sketch:void 0}},methods:{render(i){const e="0123456789",o=["#FFBA08","#FAA307","#F48C06","#E85D04","#DC2F02","#D00000","#9D0208","#6A040F","#370617","#03071E"],r={x:this.startingX,y:this.startingY,a:this.startingAngle};i.strokeWeight(this.strokeWidth),i.clear(),i.background(o[9]),i.beginShape(),i.vertex(r.x,r.y);for(let t=0;t{o.setup=()=>{o.createCanvas(this.width,this.height),o.noLoop(),o.noFill(),o.stroke(255),this.sketch=o,this.render(this.sketch)}};new this.$p5(e,"canvas")}},m={class:"select-none bg-gradient-to-b from-green-800 to-gray-800"},x={class:"m-8 grid place-items-center"},b={class:"font-mono text-white"},f={class:"my-2 md:w-full"},v={class:"grid-rows-auto my-2 grid grid-cols-2 gap-4 border-2 border-white p-2 md:grid-rows-1"},k={class:"flex flex-col"},y={for:"distanceInput",class:"text-right"},w={class:"flex flex-col"},A={for:"strokeInput",class:"text-right"},_={class:"flex flex-col"},I={for:"startingXInput",class:"text-right"},C={class:"flex flex-col"},D={for:"startingYInput",class:"text-right"},V={class:"flex flex-col"},W={for:"startingAngleInput",class:"text-right"};function X(i,e,o,r,t,l){return h(),u("div",m,[n("div",x,[n("div",b,[e[11]||(e[11]=n("div",{id:"canvas-container",class:"mb-2 h-96 border-2 border-white"},[n("div",{id:"canvas"})],-1)),n("div",f,[d(n("textarea",{readonly:"","onUpdate:modelValue":e[0]||(e[0]=s=>t.sequence=s),class:"block h-24 w-full border-2 border-white bg-black p-2 text-white placeholder:italic placeholder:text-slate-400 focus:border-2 focus:border-yellow-500 focus:outline-none sm:text-sm",placeholder:"Enter a sequence of numbers...",type:"text"},null,512),[[a,t.sequence]])]),n("div",v,[n("div",k,[n("label",y,"Segment Length ["+c(t.distance)+"]",1),d(n("input",{id:"distanceInput","onUpdate:modelValue":e[1]||(e[1]=s=>t.distance=s),onChange:e[2]||(e[2]=s=>l.render(t.sketch)),type:"range",min:"1",max:"25",class:"accent-black"},null,544),[[a,t.distance,void 0,{number:!0}]])]),n("div",w,[n("label",A,"Stroke Width ["+c(t.strokeWidth)+"]",1),d(n("input",{id:"strokeInput","onUpdate:modelValue":e[3]||(e[3]=s=>t.strokeWidth=s),onChange:e[4]||(e[4]=s=>l.render(t.sketch)),type:"range",min:"1",max:"10",class:"accent-black"},null,544),[[a,t.strokeWidth,void 0,{number:!0}]])]),n("div",_,[n("label",I,"X Position ["+c(t.startingX)+"]",1),d(n("input",{id:"startingXInput","onUpdate:modelValue":e[5]||(e[5]=s=>t.startingX=s),onChange:e[6]||(e[6]=s=>l.render(t.sketch)),type:"range",min:"0",max:"500",class:"accent-black"},null,544),[[a,t.startingX,void 0,{number:!0}]])]),n("div",C,[n("label",D,"Y Position ["+c(t.startingY)+"]",1),d(n("input",{id:"startingYInput","onUpdate:modelValue":e[7]||(e[7]=s=>t.startingY=s),onChange:e[8]||(e[8]=s=>l.render(t.sketch)),type:"range",min:"0",max:"500",class:"accent-black"},null,544),[[a,t.startingY,void 0,{number:!0}]])]),n("div",V,[n("label",W,"Angle ["+c(t.startingAngle)+"]",1),d(n("input",{id:"startingAngleInput","onUpdate:modelValue":e[9]||(e[9]=s=>t.startingAngle=s),onChange:e[10]||(e[10]=s=>l.render(t.sketch)),type:"range",min:"-180",max:"180",class:"accent-black"},null,544),[[a,t.startingAngle,void 0,{number:!0}]])])])])])])}const E=g(p,[["render",X]]);export{E as default}; diff --git a/_nuxt/yGJPFCFo.js b/_nuxt/CS06-0Od.js similarity index 99% rename from _nuxt/yGJPFCFo.js rename to _nuxt/CS06-0Od.js index fb4bf249..19ae993e 100644 --- a/_nuxt/yGJPFCFo.js +++ b/_nuxt/CS06-0Od.js @@ -1 +1 @@ -import{d as H,a0 as ln,a2 as en,k as w,L as D,a3 as on,a4 as P,a5 as tn,a6 as rn,o as an,e as un,a as sn}from"./BGikLauJ.js";import{p as F,k as cn}from"./C-v3KzvZ.js";import{f as pn}from"./Dnd51l0P.js";import{u as dn}from"./DJ6fbLCw.js";class S{constructor(l,o,t){this.property=l,this.normal=o,t&&(this.space=t)}}S.prototype.property={};S.prototype.normal={};S.prototype.space=null;function V(n,l){const o={},t={};let r=-1;for(;++r4&&o.slice(0,4)==="data"&&yn.test(l)){if(l.charAt(4)==="-"){const a=l.slice(5).replace(j,Cn);t="data"+a.charAt(0).toUpperCase()+a.slice(1)}else{const a=l.slice(4);if(!j.test(a)){let i=a.replace(vn,kn);i.charAt(0)!=="-"&&(i="-"+i),l="data"+i}}r=E}return new r(t,l)}function kn(n){return"-"+n.toLowerCase()}function Cn(n){return n.charAt(1).toUpperCase()}const Sn=V([W,q,Y,$,mn],"html");V([W,q,Y,$,fn],"svg");const B=["a","abbr","address","area","article","aside","audio","b","base","bdi","bdo","blockquote","body","br","button","canvas","caption","cite","code","col","colgroup","data","datalist","dd","del","details","dfn","dialog","div","dl","dt","em","embed","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","head","header","hgroup","hr","html","i","iframe","img","input","ins","kbd","label","legend","li","link","main","map","mark","math","menu","menuitem","meta","meter","nav","noscript","object","ol","optgroup","option","output","p","param","picture","pre","progress","q","rb","rp","rt","rtc","ruby","s","samp","script","section","select","slot","small","source","span","strong","style","sub","summary","sup","svg","table","tbody","td","template","textarea","tfoot","th","thead","time","title","tr","track","u","ul","var","video","wbr"],T="default",Z=/^@|^v-on:/,J=/^:|^v-bind:/,G=/^v-model/,xn=["select","textarea","input"],wn=Object.fromEntries(["p","a","blockquote","code","pre","code","em","h1","h2","h3","h4","h5","h6","hr","img","ul","ol","li","strong","table","thead","tbody","td","th","tr","script"].map(n=>[n,`prose-${n}`])),Pn=H({name:"MDCRenderer",props:{body:{type:Object,required:!0},data:{type:Object,default:()=>({})},class:{type:[String,Object],default:void 0},tag:{type:[String,Boolean],default:void 0},prose:{type:Boolean,default:void 0},components:{type:Object,default:()=>({})},unwrap:{type:[Boolean,String],default:!1}},async setup(n){var i,s,c,h,m,p,y,z;const l=(c=(s=(i=ln())==null?void 0:i.appContext)==null?void 0:s.app)==null?void 0:c.$nuxt,o=(l==null?void 0:l.$route)||(l==null?void 0:l._route),{mdc:t}=((h=l==null?void 0:l.$config)==null?void 0:h.public)||{},r={...(m=t==null?void 0:t.components)!=null&&m.prose&&n.prose!==!1?wn:{},...((p=t==null?void 0:t.components)==null?void 0:p.map)||{},...en(((z=(y=n.data)==null?void 0:y.mdc)==null?void 0:z.components)||{}),...n.components},a=w(()=>{var I;const nn=(((I=n.body)==null?void 0:I.children)||[]).map(x=>x.tag||x.type).filter(x=>!B.includes(x));return Array.from(new Set(nn)).sort().join(".")});return await zn(n.body,{tags:r}),{tags:r,contentKey:a,route:o}},render(n){var p,y;const{tags:l,tag:o,body:t,data:r,contentKey:a,route:i,unwrap:s}=n;if(!t)return null;const c={...r,tags:l,$route:i},h=o!==!1?U(o||((p=c.component)==null?void 0:p.name)||c.component||"div"):void 0;return h?D(h,{...(y=c.component)==null?void 0:y.props,class:n.class,...this.$attrs,key:a},{default:m}):m==null?void 0:m();function m(){return s?pn(A(t,D,c,c).default(),typeof s=="string"?s.split(" "):["*"]):A(t,D,c,c).default()}}});function On(n,l,o,t={}){if(n.type==="text")return l(P,n.value);if(n.type==="comment")return l(tn,null,n.value);const r=n.tag,a=Q(n,o.tags);if(n.tag==="binding")return Dn(n,l,o,t);const i=U(a);typeof i=="object"&&(i.tag=r);const s=Ln(n,o);return l(i,s,A(n,l,o,{...t,...s}))}function Dn(n,l,o,t={}){var h,m;const r={...t,$document:o,$doc:o},a=/\.|\[(\d+)\]/,s=((h=n.props)==null?void 0:h.value.trim().split(a).filter(Boolean)).reduce((p,y)=>{if(p&&y in p)return typeof p[y]=="function"?p[y]():p[y]},r),c=(m=n.props)==null?void 0:m.defaultValue;return l(P,s??c??"")}function A(n,l,o,t){const a=(n.children||[]).reduce((s,c)=>{if(!Bn(c))return s[T].push(c),s;const h=En(c);return s[h]=s[h]||[],c.type==="element"&&s[h].push(...c.children||[]),s},{[T]:[]});return Object.entries(a).reduce((s,[c,h])=>(h.length&&(s[c]=()=>{const m=h.map(p=>On(p,l,o,t));return Un(m)}),s),{})}function Ln(n,l){const{tag:o="",props:t={}}=n;return Object.keys(t).reduce(function(r,a){if(a==="__ignoreMap")return r;const i=t[a];if(G.test(a)&&!xn.includes(o))return Mn(a,i,r,l);if(a==="v-bind")return Rn(a,i,r,l);if(Z.test(a))return Tn(a,i,r,l);if(J.test(a))return An(a,i,r,l);const{attribute:s}=bn(Sn,a);return Array.isArray(i)&&i.every(c=>typeof c=="string")?(r[s]=i.join(" "),r):(r[s]=i,r)},{})}function Mn(n,l,o,t){const r=p=>+p,a=p=>p.trim(),i=p=>p,s=n.replace(G,"").split(".").filter(p=>p).reduce((p,y)=>(p[y]=!0,p),{}),c="value",h=s.lazy?"change":"input",m=s.number?r:s.trim?a:i;return o[c]=O(l,t),o.on=o.on||{},o.on[h]=p=>t[l]=m(p),o}function Rn(n,l,o,t){const r=O(l,t);return o=Object.assign(o,r),o}function Tn(n,l,o,t){return n=n.replace(Z,""),o.on=o.on||{},o.on[n]=()=>O(l,t),o}function An(n,l,o,t){return n=n.replace(J,""),o[n]=O(l,t),o}const U=n=>typeof n=="string"?B.includes(n)?n:on(F(n),!1):n;function O(n,l){const o=n.split(".").reduce((t,r)=>typeof t=="object"?t[r]:void 0,l);return typeof o>"u"?rn(n):o}function En(n){let l="";for(const o of Object.keys(n.props||{}))if(!(!o.startsWith("#")&&!o.startsWith("v-slot:"))){l=o.split(/[:#]/,2)[1];break}return l||T}function Bn(n){return n.tag==="template"}function Un(n){const l=[];for(const o of n){const t=l[l.length-1];o.type===P&&(t==null?void 0:t.type)===P?t.children=t.children+o.children:l.push(o)}return l}async function zn(n,l){if(!n)return;const o=Array.from(new Set(t(n,l)));await Promise.all(o.map(async r=>{if(r!=null&&r.render||r!=null&&r.ssrRender||r!=null&&r.__ssrInlineRender)return;const a=U(r);a!=null&&a.__asyncLoader&&!a.__asyncResolved&&await a.__asyncLoader()}));function t(r,a){const i=r.tag;if(r.type==="text"||i==="binding"||r.type==="comment")return[];const s=Q(r,a.tags),c=[];r.type!=="root"&&!B.includes(s)&&c.push(s);for(const h of r.children||[])c.push(...t(h,a));return c}}function Q(n,l){var t;const o=n.tag;return!o||typeof((t=n.props)==null?void 0:t.__ignoreMap)<"u"?o:l[o]||l[F(o)]||l[cn(n.tag)]||o}const Fn=H({__name:"ContentRendererMarkdown",props:{value:{type:Object,required:!0},excerpt:{type:Boolean,default:!1},tag:{type:String,default:"div"},components:{type:Object,default:()=>({})},data:{type:Object,default:()=>({})}},setup(n){const l=n,o=dn().isEnabled(),t=w(()=>{let i=l.value.body||l.value;return l.excerpt&&l.value.excerpt&&(i=l.value.excerpt),i}),r=w(()=>{const{body:i,excerpt:s,...c}=l.value;return{...c,...l.data}}),a=w(()=>({...l.components,...r.value._components||{}}));return(i,s)=>{const c=Pn;return an(),un(c,{body:t.value,data:r.value,tag:n.tag,components:a.value,"data-content-id":sn(o)?n.value._id:void 0},null,8,["body","data","tag","components","data-content-id"])}}});export{Fn as _}; +import{d as H,a0 as ln,a2 as en,k as w,L as D,a3 as on,a4 as P,a5 as tn,a6 as rn,o as an,e as un,a as sn}from"./Bb-NKJJJ.js";import{p as F,k as cn}from"./C-v3KzvZ.js";import{f as pn}from"./Dnd51l0P.js";import{u as dn}from"./mjvvwssR.js";class S{constructor(l,o,t){this.property=l,this.normal=o,t&&(this.space=t)}}S.prototype.property={};S.prototype.normal={};S.prototype.space=null;function V(n,l){const o={},t={};let r=-1;for(;++r4&&o.slice(0,4)==="data"&&yn.test(l)){if(l.charAt(4)==="-"){const a=l.slice(5).replace(j,Cn);t="data"+a.charAt(0).toUpperCase()+a.slice(1)}else{const a=l.slice(4);if(!j.test(a)){let i=a.replace(vn,kn);i.charAt(0)!=="-"&&(i="-"+i),l="data"+i}}r=E}return new r(t,l)}function kn(n){return"-"+n.toLowerCase()}function Cn(n){return n.charAt(1).toUpperCase()}const Sn=V([W,q,Y,$,mn],"html");V([W,q,Y,$,fn],"svg");const B=["a","abbr","address","area","article","aside","audio","b","base","bdi","bdo","blockquote","body","br","button","canvas","caption","cite","code","col","colgroup","data","datalist","dd","del","details","dfn","dialog","div","dl","dt","em","embed","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","head","header","hgroup","hr","html","i","iframe","img","input","ins","kbd","label","legend","li","link","main","map","mark","math","menu","menuitem","meta","meter","nav","noscript","object","ol","optgroup","option","output","p","param","picture","pre","progress","q","rb","rp","rt","rtc","ruby","s","samp","script","section","select","slot","small","source","span","strong","style","sub","summary","sup","svg","table","tbody","td","template","textarea","tfoot","th","thead","time","title","tr","track","u","ul","var","video","wbr"],T="default",Z=/^@|^v-on:/,J=/^:|^v-bind:/,G=/^v-model/,xn=["select","textarea","input"],wn=Object.fromEntries(["p","a","blockquote","code","pre","code","em","h1","h2","h3","h4","h5","h6","hr","img","ul","ol","li","strong","table","thead","tbody","td","th","tr","script"].map(n=>[n,`prose-${n}`])),Pn=H({name:"MDCRenderer",props:{body:{type:Object,required:!0},data:{type:Object,default:()=>({})},class:{type:[String,Object],default:void 0},tag:{type:[String,Boolean],default:void 0},prose:{type:Boolean,default:void 0},components:{type:Object,default:()=>({})},unwrap:{type:[Boolean,String],default:!1}},async setup(n){var i,s,c,h,m,p,y,z;const l=(c=(s=(i=ln())==null?void 0:i.appContext)==null?void 0:s.app)==null?void 0:c.$nuxt,o=(l==null?void 0:l.$route)||(l==null?void 0:l._route),{mdc:t}=((h=l==null?void 0:l.$config)==null?void 0:h.public)||{},r={...(m=t==null?void 0:t.components)!=null&&m.prose&&n.prose!==!1?wn:{},...((p=t==null?void 0:t.components)==null?void 0:p.map)||{},...en(((z=(y=n.data)==null?void 0:y.mdc)==null?void 0:z.components)||{}),...n.components},a=w(()=>{var I;const nn=(((I=n.body)==null?void 0:I.children)||[]).map(x=>x.tag||x.type).filter(x=>!B.includes(x));return Array.from(new Set(nn)).sort().join(".")});return await zn(n.body,{tags:r}),{tags:r,contentKey:a,route:o}},render(n){var p,y;const{tags:l,tag:o,body:t,data:r,contentKey:a,route:i,unwrap:s}=n;if(!t)return null;const c={...r,tags:l,$route:i},h=o!==!1?U(o||((p=c.component)==null?void 0:p.name)||c.component||"div"):void 0;return h?D(h,{...(y=c.component)==null?void 0:y.props,class:n.class,...this.$attrs,key:a},{default:m}):m==null?void 0:m();function m(){return s?pn(A(t,D,c,c).default(),typeof s=="string"?s.split(" "):["*"]):A(t,D,c,c).default()}}});function On(n,l,o,t={}){if(n.type==="text")return l(P,n.value);if(n.type==="comment")return l(tn,null,n.value);const r=n.tag,a=Q(n,o.tags);if(n.tag==="binding")return Dn(n,l,o,t);const i=U(a);typeof i=="object"&&(i.tag=r);const s=Ln(n,o);return l(i,s,A(n,l,o,{...t,...s}))}function Dn(n,l,o,t={}){var h,m;const r={...t,$document:o,$doc:o},a=/\.|\[(\d+)\]/,s=((h=n.props)==null?void 0:h.value.trim().split(a).filter(Boolean)).reduce((p,y)=>{if(p&&y in p)return typeof p[y]=="function"?p[y]():p[y]},r),c=(m=n.props)==null?void 0:m.defaultValue;return l(P,s??c??"")}function A(n,l,o,t){const a=(n.children||[]).reduce((s,c)=>{if(!Bn(c))return s[T].push(c),s;const h=En(c);return s[h]=s[h]||[],c.type==="element"&&s[h].push(...c.children||[]),s},{[T]:[]});return Object.entries(a).reduce((s,[c,h])=>(h.length&&(s[c]=()=>{const m=h.map(p=>On(p,l,o,t));return Un(m)}),s),{})}function Ln(n,l){const{tag:o="",props:t={}}=n;return Object.keys(t).reduce(function(r,a){if(a==="__ignoreMap")return r;const i=t[a];if(G.test(a)&&!xn.includes(o))return Mn(a,i,r,l);if(a==="v-bind")return Rn(a,i,r,l);if(Z.test(a))return Tn(a,i,r,l);if(J.test(a))return An(a,i,r,l);const{attribute:s}=bn(Sn,a);return Array.isArray(i)&&i.every(c=>typeof c=="string")?(r[s]=i.join(" "),r):(r[s]=i,r)},{})}function Mn(n,l,o,t){const r=p=>+p,a=p=>p.trim(),i=p=>p,s=n.replace(G,"").split(".").filter(p=>p).reduce((p,y)=>(p[y]=!0,p),{}),c="value",h=s.lazy?"change":"input",m=s.number?r:s.trim?a:i;return o[c]=O(l,t),o.on=o.on||{},o.on[h]=p=>t[l]=m(p),o}function Rn(n,l,o,t){const r=O(l,t);return o=Object.assign(o,r),o}function Tn(n,l,o,t){return n=n.replace(Z,""),o.on=o.on||{},o.on[n]=()=>O(l,t),o}function An(n,l,o,t){return n=n.replace(J,""),o[n]=O(l,t),o}const U=n=>typeof n=="string"?B.includes(n)?n:on(F(n),!1):n;function O(n,l){const o=n.split(".").reduce((t,r)=>typeof t=="object"?t[r]:void 0,l);return typeof o>"u"?rn(n):o}function En(n){let l="";for(const o of Object.keys(n.props||{}))if(!(!o.startsWith("#")&&!o.startsWith("v-slot:"))){l=o.split(/[:#]/,2)[1];break}return l||T}function Bn(n){return n.tag==="template"}function Un(n){const l=[];for(const o of n){const t=l[l.length-1];o.type===P&&(t==null?void 0:t.type)===P?t.children=t.children+o.children:l.push(o)}return l}async function zn(n,l){if(!n)return;const o=Array.from(new Set(t(n,l)));await Promise.all(o.map(async r=>{if(r!=null&&r.render||r!=null&&r.ssrRender||r!=null&&r.__ssrInlineRender)return;const a=U(r);a!=null&&a.__asyncLoader&&!a.__asyncResolved&&await a.__asyncLoader()}));function t(r,a){const i=r.tag;if(r.type==="text"||i==="binding"||r.type==="comment")return[];const s=Q(r,a.tags),c=[];r.type!=="root"&&!B.includes(s)&&c.push(s);for(const h of r.children||[])c.push(...t(h,a));return c}}function Q(n,l){var t;const o=n.tag;return!o||typeof((t=n.props)==null?void 0:t.__ignoreMap)<"u"?o:l[o]||l[F(o)]||l[cn(n.tag)]||o}const Fn=H({__name:"ContentRendererMarkdown",props:{value:{type:Object,required:!0},excerpt:{type:Boolean,default:!1},tag:{type:String,default:"div"},components:{type:Object,default:()=>({})},data:{type:Object,default:()=>({})}},setup(n){const l=n,o=dn().isEnabled(),t=w(()=>{let i=l.value.body||l.value;return l.excerpt&&l.value.excerpt&&(i=l.value.excerpt),i}),r=w(()=>{const{body:i,excerpt:s,...c}=l.value;return{...c,...l.data}}),a=w(()=>({...l.components,...r.value._components||{}}));return(i,s)=>{const c=Pn;return an(),un(c,{body:t.value,data:r.value,tag:n.tag,components:a.value,"data-content-id":sn(o)?n.value._id:void 0},null,8,["body","data","tag","components","data-content-id"])}}});export{Fn as _}; diff --git a/_nuxt/CYeLrQ9C.js b/_nuxt/CYeLrQ9C.js deleted file mode 100644 index 883ee770..00000000 --- a/_nuxt/CYeLrQ9C.js +++ /dev/null @@ -1 +0,0 @@ -import{a as o}from"./-8n5e5Fj.js";import{m as c,c as n,i as t,o as a}from"./BGikLauJ.js";const r={};function s(_,m){const e=o;return a(),n("div",null,[t(e,{showImages:""})])}const d=c(r,[["render",s]]);export{d as default}; diff --git a/_nuxt/CDVU8JqM.js b/_nuxt/Cai_qPmv.js similarity index 98% rename from _nuxt/CDVU8JqM.js rename to _nuxt/Cai_qPmv.js index f962fee8..083650fa 100644 --- a/_nuxt/CDVU8JqM.js +++ b/_nuxt/Cai_qPmv.js @@ -1 +1 @@ -import{m,c as o,g as a,b as t,l as h,t as d,F as c,r as p,h as v,o as i}from"./BGikLauJ.js";const b={mounted(){typeof navigator.requestMIDIAccess<"u"&&navigator.requestMIDIAccess().then(n=>{this.midi=n,this.midi.inputs.forEach(e=>{e.onmidimessage=u=>{this.events.push(u),this.events.length>15&&this.events.shift()}})},n=>{console.error(n)})},data(){return{tooltip:!1,midi:void 0,events:[]}},computed:{inputs(){if(typeof this.midi<"u")return Array.from(this.midi.inputs.values())},outputs(){if(typeof this.midi<"u")return Array.from(this.midi.outputs.values())}},filters:{midiCommand:n=>{switch(n){case 144:return"Note On";case 128:return"Note Off";default:return n}},midiNote:n=>`${["C","C# / Db","D","D# / Eb","E","F","F# / Bb","G","G# / Ab","A","A# / Bb","B"][n%12]} (${Math.floor(n/12)-2})`}},g={class:"h-2/3 bg-gradient-to-b from-green-100 to-green-50"},x={key:0,class:"p-4 text-center font-light tracking-wide"},_={key:1},y={class:"flex flex-wrap"},w={class:"absolute bottom-16 right-2"},k={class:"p-3 font-mono text-orange-900"},I={class:"mb-2 font-bold"},M={class:"mb-2"},N={key:0,class:"p-4 text-center italic"},D={key:1},C={class:"flex-1"},A={class:"flex-1"},z={class:"mb-2"},E={key:0,class:"p-4 text-center italic"},V={key:1},B={class:"flex-1"},F={class:"flex-1"},S={class:"absolute bottom-2 right-2"},O={class:"container mx-auto p-8"},P={class:"w-full table-auto border-2 border-green-800 bg-green-50 text-sm"},q={class:"divide-y divide-gray-100"},G={key:0},T={class:"p-2 text-left"},U={class:"p-2 text-left"},W={class:"p-2 text-left"},H={class:"p-2 text-left"},L={key:0,class:"p-2 text-left"},R={key:1,class:"p-2 text-left"},Y={class:"p-2 text-left"},j={class:"p-2 text-right"};function J(n,e,u,K,l,r){return i(),o("div",g,[typeof l.midi>"u"?(i(),o("div",x,e[1]||(e[1]=[a(" Unfortunately, the Web MIDI API is "),t("a",{class:"text-blue-500 underline",href:"https://developer.mozilla.org/en-US/docs/Web/API/MIDIMessageEvent#browser_compatibility"},"not supported",-1),a(" in all browsers... ")]))):(i(),o("div",_,[t("div",y,[t("div",w,[t("div",{class:h([{hidden:!l.tooltip,block:l.tooltip},"z-50 max-w-md break-words rounded-lg border-2 border-green-800 bg-yellow-200 text-sm font-normal leading-normal"])},[t("div",null,[e[5]||(e[5]=t("div",{class:"mb-0 rounded-t-lg border-b border-solid bg-green-600 p-3 font-semibold uppercase text-white opacity-75"}," MIDI Status ",-1)),t("div",k,[t("div",I,[e[2]||(e[2]=a(" Enabled: ")),t("span",null,d(typeof l.midi<"u"?"Yep!":"Nope"),1)]),t("div",M,[e[3]||(e[3]=t("div",{class:"font-bold"},"Inputs:",-1)),r.inputs.length===0?(i(),o("div",N," No input devices detected :( ")):(i(),o("div",D,[(i(!0),o(c,null,p(r.inputs,s=>(i(),o("div",{key:s.id,class:"flex"},[t("div",C,d(s.manufacturer),1),t("div",A,d(s.name),1)]))),128))]))]),t("div",z,[e[4]||(e[4]=t("div",{class:"font-bold"},"Outputs:",-1)),r.outputs.length===0?(i(),o("div",E," No output devices detected :( ")):(i(),o("div",V,[(i(!0),o(c,null,p(r.outputs,s=>(i(),o("div",{key:s.id,class:"flex"},[t("div",B,d(s.manufacturer),1),t("div",F,d(s.name),1)]))),128))]))])])])],2)]),t("div",S,[t("button",{ref:"btnRef",onClick:e[0]||(e[0]=s=>l.tooltip=!l.tooltip),class:"rounded-lg bg-green-800 px-2 py-1 text-white shadow hover:text-yellow-200 hover:shadow-lg",type:"button"},e[6]||(e[6]=[t("svg",{xmlns:"http://www.w3.org/2000/svg","xmlns:xlink":"http://www.w3.org/1999/xlink",viewBox:"0 0 20 20",class:"h-10 w-10"},[t("g",{fill:"none"},[t("path",{d:"M2.5 4a.5.5 0 0 0-.5.5v11a.5.5 0 0 0 .5.5h15a.5.5 0 0 0 .5-.5v-11a.5.5 0 0 0-.5-.5h-15zm.5 6h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v5H3v-5zm2.75-2.5a.75.75 0 1 1 0-1.5a.75.75 0 0 1 0 1.5zm6.25-1a.5.5 0 0 1 .5-.5h2a.5.5 0 0 1 0 1h-2a.5.5 0 0 1-.5-.5zm-3 .251a.75.75 0 1 1-1.5 0a.75.75 0 0 1 1.5 0z",fill:"currentColor"})])],-1)]),512)])]),t("div",O,[e[9]||(e[9]=t("div",{class:"font-display text-4xl font-light tracking-wide"}," MIDI Events ",-1)),t("table",P,[e[8]||(e[8]=t("thead",{class:"bg-green-800 font-semibold uppercase text-white"},[t("tr",null,[t("th",{class:"p-2 text-left"},"Timestamp"),t("th",{class:"p-2 text-left"},"Command #"),t("th",{class:"p-2 text-left"},"Command"),t("th",{class:"p-2 text-left"},"Note #"),t("th",{class:"p-2 text-left"},"Note"),t("th",{class:"p-2 text-left"},"Velocity"),t("th",{class:"p-2 text-right"},"MIDI Source")])],-1)),t("tbody",q,[l.events.length===0?(i(),o("tr",G,e[7]||(e[7]=[t("td",{class:"p-4 text-center italic",colspan:"7"}," Press a key, or turn a knob! ",-1)]))):v("",!0),(i(!0),o(c,null,p(l.events,(s,f)=>(i(),o("tr",{key:f},[t("td",T,d(s.timeStamp.toFixed(2)),1),t("td",U,d(s.data[0]),1),t("td",W,d(s.data[0]|n.midiCommand),1),t("td",H,d(s.data[1]),1),s.data[0]===144||s.data[0]==128?(i(),o("td",L,d(s.data[1]|n.midiNote),1)):(i(),o("td",R,"-")),t("td",Y,d(s.data[2]),1),t("td",j,d(s.srcElement.name),1)]))),128))])])])]))])}const X=m(b,[["render",J]]);export{X as default}; +import{m,c as o,g as a,b as t,l as h,t as d,F as c,r as p,h as v,o as i}from"./Bb-NKJJJ.js";const b={mounted(){typeof navigator.requestMIDIAccess<"u"&&navigator.requestMIDIAccess().then(n=>{this.midi=n,this.midi.inputs.forEach(e=>{e.onmidimessage=u=>{this.events.push(u),this.events.length>15&&this.events.shift()}})},n=>{console.error(n)})},data(){return{tooltip:!1,midi:void 0,events:[]}},computed:{inputs(){if(typeof this.midi<"u")return Array.from(this.midi.inputs.values())},outputs(){if(typeof this.midi<"u")return Array.from(this.midi.outputs.values())}},filters:{midiCommand:n=>{switch(n){case 144:return"Note On";case 128:return"Note Off";default:return n}},midiNote:n=>`${["C","C# / Db","D","D# / Eb","E","F","F# / Bb","G","G# / Ab","A","A# / Bb","B"][n%12]} (${Math.floor(n/12)-2})`}},g={class:"h-2/3 bg-gradient-to-b from-green-100 to-green-50"},x={key:0,class:"p-4 text-center font-light tracking-wide"},_={key:1},y={class:"flex flex-wrap"},w={class:"absolute bottom-16 right-2"},k={class:"p-3 font-mono text-orange-900"},I={class:"mb-2 font-bold"},M={class:"mb-2"},N={key:0,class:"p-4 text-center italic"},D={key:1},C={class:"flex-1"},A={class:"flex-1"},z={class:"mb-2"},E={key:0,class:"p-4 text-center italic"},V={key:1},B={class:"flex-1"},F={class:"flex-1"},S={class:"absolute bottom-2 right-2"},O={class:"container mx-auto p-8"},P={class:"w-full table-auto border-2 border-green-800 bg-green-50 text-sm"},q={class:"divide-y divide-gray-100"},G={key:0},T={class:"p-2 text-left"},U={class:"p-2 text-left"},W={class:"p-2 text-left"},H={class:"p-2 text-left"},L={key:0,class:"p-2 text-left"},R={key:1,class:"p-2 text-left"},Y={class:"p-2 text-left"},j={class:"p-2 text-right"};function J(n,e,u,K,l,r){return i(),o("div",g,[typeof l.midi>"u"?(i(),o("div",x,e[1]||(e[1]=[a(" Unfortunately, the Web MIDI API is "),t("a",{class:"text-blue-500 underline",href:"https://developer.mozilla.org/en-US/docs/Web/API/MIDIMessageEvent#browser_compatibility"},"not supported",-1),a(" in all browsers... ")]))):(i(),o("div",_,[t("div",y,[t("div",w,[t("div",{class:h([{hidden:!l.tooltip,block:l.tooltip},"z-50 max-w-md break-words rounded-lg border-2 border-green-800 bg-yellow-200 text-sm font-normal leading-normal"])},[t("div",null,[e[5]||(e[5]=t("div",{class:"mb-0 rounded-t-lg border-b border-solid bg-green-600 p-3 font-semibold uppercase text-white opacity-75"}," MIDI Status ",-1)),t("div",k,[t("div",I,[e[2]||(e[2]=a(" Enabled: ")),t("span",null,d(typeof l.midi<"u"?"Yep!":"Nope"),1)]),t("div",M,[e[3]||(e[3]=t("div",{class:"font-bold"},"Inputs:",-1)),r.inputs.length===0?(i(),o("div",N," No input devices detected :( ")):(i(),o("div",D,[(i(!0),o(c,null,p(r.inputs,s=>(i(),o("div",{key:s.id,class:"flex"},[t("div",C,d(s.manufacturer),1),t("div",A,d(s.name),1)]))),128))]))]),t("div",z,[e[4]||(e[4]=t("div",{class:"font-bold"},"Outputs:",-1)),r.outputs.length===0?(i(),o("div",E," No output devices detected :( ")):(i(),o("div",V,[(i(!0),o(c,null,p(r.outputs,s=>(i(),o("div",{key:s.id,class:"flex"},[t("div",B,d(s.manufacturer),1),t("div",F,d(s.name),1)]))),128))]))])])])],2)]),t("div",S,[t("button",{ref:"btnRef",onClick:e[0]||(e[0]=s=>l.tooltip=!l.tooltip),class:"rounded-lg bg-green-800 px-2 py-1 text-white shadow hover:text-yellow-200 hover:shadow-lg",type:"button"},e[6]||(e[6]=[t("svg",{xmlns:"http://www.w3.org/2000/svg","xmlns:xlink":"http://www.w3.org/1999/xlink",viewBox:"0 0 20 20",class:"h-10 w-10"},[t("g",{fill:"none"},[t("path",{d:"M2.5 4a.5.5 0 0 0-.5.5v11a.5.5 0 0 0 .5.5h15a.5.5 0 0 0 .5-.5v-11a.5.5 0 0 0-.5-.5h-15zm.5 6h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v5H3v-5zm2.75-2.5a.75.75 0 1 1 0-1.5a.75.75 0 0 1 0 1.5zm6.25-1a.5.5 0 0 1 .5-.5h2a.5.5 0 0 1 0 1h-2a.5.5 0 0 1-.5-.5zm-3 .251a.75.75 0 1 1-1.5 0a.75.75 0 0 1 1.5 0z",fill:"currentColor"})])],-1)]),512)])]),t("div",O,[e[9]||(e[9]=t("div",{class:"font-display text-4xl font-light tracking-wide"}," MIDI Events ",-1)),t("table",P,[e[8]||(e[8]=t("thead",{class:"bg-green-800 font-semibold uppercase text-white"},[t("tr",null,[t("th",{class:"p-2 text-left"},"Timestamp"),t("th",{class:"p-2 text-left"},"Command #"),t("th",{class:"p-2 text-left"},"Command"),t("th",{class:"p-2 text-left"},"Note #"),t("th",{class:"p-2 text-left"},"Note"),t("th",{class:"p-2 text-left"},"Velocity"),t("th",{class:"p-2 text-right"},"MIDI Source")])],-1)),t("tbody",q,[l.events.length===0?(i(),o("tr",G,e[7]||(e[7]=[t("td",{class:"p-4 text-center italic",colspan:"7"}," Press a key, or turn a knob! ",-1)]))):v("",!0),(i(!0),o(c,null,p(l.events,(s,f)=>(i(),o("tr",{key:f},[t("td",T,d(s.timeStamp.toFixed(2)),1),t("td",U,d(s.data[0]),1),t("td",W,d(s.data[0]|n.midiCommand),1),t("td",H,d(s.data[1]),1),s.data[0]===144||s.data[0]==128?(i(),o("td",L,d(s.data[1]|n.midiNote),1)):(i(),o("td",R,"-")),t("td",Y,d(s.data[2]),1),t("td",j,d(s.srcElement.name),1)]))),128))])])])]))])}const X=m(b,[["render",J]]);export{X as default}; diff --git a/_nuxt/qIrQuZow.js b/_nuxt/CgzK-JD6.js similarity index 98% rename from _nuxt/qIrQuZow.js rename to _nuxt/CgzK-JD6.js index e9ff5986..a278b253 100644 --- a/_nuxt/qIrQuZow.js +++ b/_nuxt/CgzK-JD6.js @@ -1 +1 @@ -import{a6 as _,U as T,ac as W,I as M,N as J,Q as H}from"./BGikLauJ.js";import{g as k,a as j,b as O,o as b,c as G,d as $,f as P,h as D,i as Z}from"./UWyBikFq.js";import{p as q}from"./C-v3KzvZ.js";import{u as U}from"./DJ6fbLCw.js";import"./IShi1APO.js";const F="memory",V=()=>{const t=new Map;return{name:F,getInstance:()=>t,hasItem(r){return t.has(r)},getItem(r){return t.get(r)??null},getItemRaw(r){return t.get(r)??null},setItem(r,n){t.set(r,n)},setItemRaw(r,n){t.set(r,n)},removeItem(r){t.delete(r)},getKeys(){return[...t.keys()]},clear(){t.clear()},dispose(){t.clear()}}};function Q(t){return!t||typeof t.then!="function"?Promise.resolve(t):t}function p(t,...r){try{return Q(t(...r))}catch(n){return Promise.reject(n)}}function X(t){const r=typeof t;return t===null||r!=="object"&&r!=="function"}function ee(t){const r=Object.getPrototypeOf(t);return!r||r.isPrototypeOf(Object)}function S(t){if(X(t))return String(t);if(ee(t)||Array.isArray(t))return JSON.stringify(t);if(typeof t.toJSON=="function")return S(t.toJSON());throw new Error("[unstorage] Cannot stringify value!")}function z(){if(typeof Buffer>"u")throw new TypeError("[unstorage] Buffer is not supported!")}const R="base64:";function te(t){if(typeof t=="string")return t;z();const r=Buffer.from(t).toString("base64");return R+r}function re(t){return typeof t!="string"||!t.startsWith(R)?t:(z(),Buffer.from(t.slice(R.length),"base64"))}const ne=["hasItem","getItem","getItemRaw","setItem","setItemRaw","removeItem","getMeta","setMeta","removeMeta","getKeys","clear","mount","unmount"];function ie(t,r){if(r=E(r),!r)return t;const n={...t};for(const a of ne)n[a]=(c="",...l)=>t[a](r+c,...l);return n.getKeys=(a="",...c)=>t.getKeys(r+a,...c).then(l=>l.map(o=>o.slice(r.length))),n}function y(t){return t?t.split("?")[0].replace(/[/\\]/g,":").replace(/:+/g,":").replace(/^:|:$/g,""):""}function ae(...t){return y(t.join(":"))}function E(t){return t=y(t),t?t+":":""}const se="memory",oe=()=>{const t=new Map;return{name:se,getInstance:()=>t,hasItem(r){return t.has(r)},getItem(r){return t.get(r)??null},getItemRaw(r){return t.get(r)??null},setItem(r,n){t.set(r,n)},setItemRaw(r,n){t.set(r,n)},removeItem(r){t.delete(r)},getKeys(){return[...t.keys()]},clear(){t.clear()},dispose(){t.clear()}}};function ue(t={}){const r={mounts:{"":t.driver||oe()},mountpoints:[""],watching:!1,watchListeners:[],unwatch:{}},n=e=>{for(const i of r.mountpoints)if(e.startsWith(i))return{base:i,relativeKey:e.slice(i.length),driver:r.mounts[i]};return{base:"",relativeKey:e,driver:r.mounts[""]}},a=(e,i)=>r.mountpoints.filter(s=>s.startsWith(e)||i&&e.startsWith(s)).map(s=>({relativeBase:e.length>s.length?e.slice(s.length):void 0,mountpoint:s,driver:r.mounts[s]})),c=(e,i)=>{if(r.watching){i=y(i);for(const s of r.watchListeners)s(e,i)}},l=async()=>{if(!r.watching){r.watching=!0;for(const e in r.mounts)r.unwatch[e]=await x(r.mounts[e],c,e)}},o=async()=>{if(r.watching){for(const e in r.unwatch)await r.unwatch[e]();r.unwatch={},r.watching=!1}},g=(e,i,s)=>{const u=new Map,m=f=>{let d=u.get(f.base);return d||(d={driver:f.driver,base:f.base,items:[]},u.set(f.base,d)),d};for(const f of e){const d=typeof f=="string",v=y(d?f:f.key),I=d?void 0:f.value,w=d||!f.options?i:{...i,...f.options},A=n(v);m(A).items.push({key:v,value:I,relativeKey:A.relativeKey,options:w})}return Promise.all([...u.values()].map(f=>s(f))).then(f=>f.flat())},h={hasItem(e,i={}){e=y(e);const{relativeKey:s,driver:u}=n(e);return p(u.hasItem,s,i)},getItem(e,i={}){e=y(e);const{relativeKey:s,driver:u}=n(e);return p(u.getItem,s,i).then(m=>_(m))},getItems(e,i){return g(e,i,s=>s.driver.getItems?p(s.driver.getItems,s.items.map(u=>({key:u.relativeKey,options:u.options})),i).then(u=>u.map(m=>({key:ae(s.base,m.key),value:_(m.value)}))):Promise.all(s.items.map(u=>p(s.driver.getItem,u.relativeKey,u.options).then(m=>({key:u.key,value:_(m)})))))},getItemRaw(e,i={}){e=y(e);const{relativeKey:s,driver:u}=n(e);return u.getItemRaw?p(u.getItemRaw,s,i):p(u.getItem,s,i).then(m=>re(m))},async setItem(e,i,s={}){if(i===void 0)return h.removeItem(e);e=y(e);const{relativeKey:u,driver:m}=n(e);m.setItem&&(await p(m.setItem,u,S(i),s),m.watch||c("update",e))},async setItems(e,i){await g(e,i,async s=>{if(s.driver.setItems)return p(s.driver.setItems,s.items.map(u=>({key:u.relativeKey,value:S(u.value),options:u.options})),i);s.driver.setItem&&await Promise.all(s.items.map(u=>p(s.driver.setItem,u.relativeKey,S(u.value),u.options)))})},async setItemRaw(e,i,s={}){if(i===void 0)return h.removeItem(e,s);e=y(e);const{relativeKey:u,driver:m}=n(e);if(m.setItemRaw)await p(m.setItemRaw,u,i,s);else if(m.setItem)await p(m.setItem,u,te(i),s);else return;m.watch||c("update",e)},async removeItem(e,i={}){typeof i=="boolean"&&(i={removeMeta:i}),e=y(e);const{relativeKey:s,driver:u}=n(e);u.removeItem&&(await p(u.removeItem,s,i),(i.removeMeta||i.removeMata)&&await p(u.removeItem,s+"$",i),u.watch||c("remove",e))},async getMeta(e,i={}){typeof i=="boolean"&&(i={nativeOnly:i}),e=y(e);const{relativeKey:s,driver:u}=n(e),m=Object.create(null);if(u.getMeta&&Object.assign(m,await p(u.getMeta,s,i)),!i.nativeOnly){const f=await p(u.getItem,s+"$",i).then(d=>_(d));f&&typeof f=="object"&&(typeof f.atime=="string"&&(f.atime=new Date(f.atime)),typeof f.mtime=="string"&&(f.mtime=new Date(f.mtime)),Object.assign(m,f))}return m},setMeta(e,i,s={}){return this.setItem(e+"$",i,s)},removeMeta(e,i={}){return this.removeItem(e+"$",i)},async getKeys(e,i={}){e=E(e);const s=a(e,!0);let u=[];const m=[];for(const f of s){const d=await p(f.driver.getKeys,f.relativeBase,i);for(const v of d){const I=f.mountpoint+y(v);u.some(w=>I.startsWith(w))||m.push(I)}u=[f.mountpoint,...u.filter(v=>!v.startsWith(f.mountpoint))]}return e?m.filter(f=>f.startsWith(e)&&f[f.length-1]!=="$"):m.filter(f=>f[f.length-1]!=="$")},async clear(e,i={}){e=E(e),await Promise.all(a(e,!1).map(async s=>{if(s.driver.clear)return p(s.driver.clear,s.relativeBase,i);if(s.driver.removeItem){const u=await s.driver.getKeys(s.relativeBase||"",i);return Promise.all(u.map(m=>s.driver.removeItem(m,i)))}}))},async dispose(){await Promise.all(Object.values(r.mounts).map(e=>L(e)))},async watch(e){return await l(),r.watchListeners.push(e),async()=>{r.watchListeners=r.watchListeners.filter(i=>i!==e),r.watchListeners.length===0&&await o()}},async unwatch(){r.watchListeners=[],await o()},mount(e,i){if(e=E(e),e&&r.mounts[e])throw new Error(`already mounted at ${e}`);return e&&(r.mountpoints.push(e),r.mountpoints.sort((s,u)=>u.length-s.length)),r.mounts[e]=i,r.watching&&Promise.resolve(x(i,c,e)).then(s=>{r.unwatch[e]=s}).catch(console.error),h},async unmount(e,i=!0){e=E(e),!(!e||!r.mounts[e])&&(r.watching&&e in r.unwatch&&(r.unwatch[e](),delete r.unwatch[e]),i&&await L(r.mounts[e]),r.mountpoints=r.mountpoints.filter(s=>s!==e),delete r.mounts[e])},getMount(e=""){e=y(e)+":";const i=n(e);return{driver:i.driver,base:i.base}},getMounts(e="",i={}){return e=y(e),a(e,i.parents).map(u=>({driver:u.driver,base:u.mountpoint}))},keys:(e,i={})=>h.getKeys(e,i),get:(e,i={})=>h.getItem(e,i),set:(e,i,s={})=>h.setItem(e,i,s),has:(e,i={})=>h.hasItem(e,i),del:(e,i={})=>h.removeItem(e,i),remove:(e,i={})=>h.removeItem(e,i)};return h}function x(t,r,n){return t.watch?t.watch((a,c)=>r(a,n+c)):()=>{}}async function L(t){typeof t.dispose=="function"&&await p(t.dispose)}function ce(t={}){const r=le(n,t.operators);function n(a,c){return typeof c!="object"||c instanceof RegExp?r.$eq(a,c):Object.keys(c||{}).every(l=>{const o=c[l];if(l.startsWith("$")&&r[l]){const g=r[l];return typeof g=="function"?g(a,o):!1}return n(k(a,l),o)})}return n}function le(t,r={}){return{$match:(n,a)=>t(n,a),$eq:(n,a)=>a instanceof RegExp?a.test(n):n===a,$ne:(n,a)=>a instanceof RegExp?!a.test(n):n!==a,$not:(n,a)=>!t(n,a),$and:(n,a)=>(j(a,"$and requires an array as condition"),a.every(c=>t(n,c))),$or:(n,a)=>(j(a,"$or requires an array as condition"),a.some(c=>t(n,c))),$in:(n,a)=>O(a).some(c=>Array.isArray(n)?t(n,{$contains:c}):t(n,c)),$contains:(n,a)=>(n=Array.isArray(n)?n:String(n),O(a).every(c=>n.includes(c))),$icontains:(n,a)=>{if(typeof a!="string")throw new TypeError("$icontains requires a string, use $contains instead");return n=String(n).toLocaleLowerCase(),O(a).every(c=>n.includes(c.toLocaleLowerCase()))},$containsAny:(n,a)=>(j(a,"$containsAny requires an array as condition"),n=Array.isArray(n)?n:String(n),a.some(c=>n.includes(c))),$exists:(n,a)=>a?typeof n<"u":typeof n>"u",$type:(n,a)=>typeof n===String(a),$regex:(n,a)=>{if(!(a instanceof RegExp)){const c=String(a).match(/\/(.*)\/([dgimsuy]*)$/);a=c!=null&&c[1]?new RegExp(c[1],c[2]||""):new RegExp(a)}return a.test(String(n||""))},$lt:(n,a)=>nn<=a,$gt:(n,a)=>n>a,$gte:(n,a)=>n>=a,...r||{}}}function fe(t){const r=ce(),n=(l,{query:o,before:g,after:h})=>{const e=typeof o=="string"?{_path:o}:o,i=l.findIndex(u=>r(u,e));g=g??1,h=h??1;const s=new Array(g+h).fill(null,0);return i===-1?s:s.map((u,m)=>l[i-g+m+ +(m>=g)]||null)},a=[(l,o)=>{const g=l.result.filter(h=>O(o.where).every(e=>r(h,e)));return{...l,result:g,total:g.length}},(l,o)=>O(o.sort).forEach(g=>G(l.result,g)),function(o,g,h){var e;if(g.surround){let i=n(((e=o.result)==null?void 0:e.length)===1?h:o.result,g.surround);i=$(P(g.without))(i),i=$(D(g.only))(i),o.surround=i}return o}],c=[(l,o)=>{if(o.skip)return{...l,result:l.result.slice(o.skip),skip:o.skip}},(l,o)=>{if(o.limit)return{...l,result:l.result.slice(0,o.limit),limit:o.limit}},function(o,g,h){var e,i,s;if(g.dirConfig){const u=((e=o.result[0])==null?void 0:e._path)||((s=(i=g.where)==null?void 0:i.find(m=>m._path))==null?void 0:s._path);if(typeof u=="string"){const m=h.find(f=>f._path===T(u,"_dir"));m&&(o.dirConfig={_path:m._path,...P(["_"])(m)})}}return o},(l,o)=>({...l,result:$(P(o.without))(l.result)}),(l,o)=>({...l,result:$(D(o.only))(l.result)})];return async l=>{const o=await t(),g=l.params(),h={result:o,limit:0,skip:0,total:o.length},e=a.reduce((s,u)=>u(s,g,o)||s,h);if(g.count)return{result:e.result.length};const i=c.reduce((s,u)=>u(s,g,o)||s,e);return g.first?{...b(["skip","limit","total"])(i),result:i.result[0]}:i}}function N(t){const r=fe(t);return async n=>{var l;n.params().first&&n.withDirConfig();const a=n.params(),c=await r(n);return a.surround?c==null?void 0:c.surround:(c!=null&&c.dirConfig&&(c.result={_path:(l=c.dirConfig)==null?void 0:l._path,...c.result,_dir:c.dirConfig}),c==null?void 0:c.result)}}var me={exports:{}};(function(t,r){(function(n,a,c){t.exports=c(),t.exports.default=c()})("slugify",W,function(){var n=JSON.parse(`{"$":"dollar","%":"percent","&":"and","<":"less",">":"greater","|":"or","¢":"cent","£":"pound","¤":"currency","¥":"yen","©":"(c)","ª":"a","®":"(r)","º":"o","À":"A","Á":"A","Â":"A","Ã":"A","Ä":"A","Å":"A","Æ":"AE","Ç":"C","È":"E","É":"E","Ê":"E","Ë":"E","Ì":"I","Í":"I","Î":"I","Ï":"I","Ð":"D","Ñ":"N","Ò":"O","Ó":"O","Ô":"O","Õ":"O","Ö":"O","Ø":"O","Ù":"U","Ú":"U","Û":"U","Ü":"U","Ý":"Y","Þ":"TH","ß":"ss","à":"a","á":"a","â":"a","ã":"a","ä":"a","å":"a","æ":"ae","ç":"c","è":"e","é":"e","ê":"e","ë":"e","ì":"i","í":"i","î":"i","ï":"i","ð":"d","ñ":"n","ò":"o","ó":"o","ô":"o","õ":"o","ö":"o","ø":"o","ù":"u","ú":"u","û":"u","ü":"u","ý":"y","þ":"th","ÿ":"y","Ā":"A","ā":"a","Ă":"A","ă":"a","Ą":"A","ą":"a","Ć":"C","ć":"c","Č":"C","č":"c","Ď":"D","ď":"d","Đ":"DJ","đ":"dj","Ē":"E","ē":"e","Ė":"E","ė":"e","Ę":"e","ę":"e","Ě":"E","ě":"e","Ğ":"G","ğ":"g","Ģ":"G","ģ":"g","Ĩ":"I","ĩ":"i","Ī":"i","ī":"i","Į":"I","į":"i","İ":"I","ı":"i","Ķ":"k","ķ":"k","Ļ":"L","ļ":"l","Ľ":"L","ľ":"l","Ł":"L","ł":"l","Ń":"N","ń":"n","Ņ":"N","ņ":"n","Ň":"N","ň":"n","Ō":"O","ō":"o","Ő":"O","ő":"o","Œ":"OE","œ":"oe","Ŕ":"R","ŕ":"r","Ř":"R","ř":"r","Ś":"S","ś":"s","Ş":"S","ş":"s","Š":"S","š":"s","Ţ":"T","ţ":"t","Ť":"T","ť":"t","Ũ":"U","ũ":"u","Ū":"u","ū":"u","Ů":"U","ů":"u","Ű":"U","ű":"u","Ų":"U","ų":"u","Ŵ":"W","ŵ":"w","Ŷ":"Y","ŷ":"y","Ÿ":"Y","Ź":"Z","ź":"z","Ż":"Z","ż":"z","Ž":"Z","ž":"z","Ə":"E","ƒ":"f","Ơ":"O","ơ":"o","Ư":"U","ư":"u","Lj":"LJ","lj":"lj","Nj":"NJ","nj":"nj","Ș":"S","ș":"s","Ț":"T","ț":"t","ə":"e","˚":"o","Ά":"A","Έ":"E","Ή":"H","Ί":"I","Ό":"O","Ύ":"Y","Ώ":"W","ΐ":"i","Α":"A","Β":"B","Γ":"G","Δ":"D","Ε":"E","Ζ":"Z","Η":"H","Θ":"8","Ι":"I","Κ":"K","Λ":"L","Μ":"M","Ν":"N","Ξ":"3","Ο":"O","Π":"P","Ρ":"R","Σ":"S","Τ":"T","Υ":"Y","Φ":"F","Χ":"X","Ψ":"PS","Ω":"W","Ϊ":"I","Ϋ":"Y","ά":"a","έ":"e","ή":"h","ί":"i","ΰ":"y","α":"a","β":"b","γ":"g","δ":"d","ε":"e","ζ":"z","η":"h","θ":"8","ι":"i","κ":"k","λ":"l","μ":"m","ν":"n","ξ":"3","ο":"o","π":"p","ρ":"r","ς":"s","σ":"s","τ":"t","υ":"y","φ":"f","χ":"x","ψ":"ps","ω":"w","ϊ":"i","ϋ":"y","ό":"o","ύ":"y","ώ":"w","Ё":"Yo","Ђ":"DJ","Є":"Ye","І":"I","Ї":"Yi","Ј":"J","Љ":"LJ","Њ":"NJ","Ћ":"C","Џ":"DZ","А":"A","Б":"B","В":"V","Г":"G","Д":"D","Е":"E","Ж":"Zh","З":"Z","И":"I","Й":"J","К":"K","Л":"L","М":"M","Н":"N","О":"O","П":"P","Р":"R","С":"S","Т":"T","У":"U","Ф":"F","Х":"H","Ц":"C","Ч":"Ch","Ш":"Sh","Щ":"Sh","Ъ":"U","Ы":"Y","Ь":"","Э":"E","Ю":"Yu","Я":"Ya","а":"a","б":"b","в":"v","г":"g","д":"d","е":"e","ж":"zh","з":"z","и":"i","й":"j","к":"k","л":"l","м":"m","н":"n","о":"o","п":"p","р":"r","с":"s","т":"t","у":"u","ф":"f","х":"h","ц":"c","ч":"ch","ш":"sh","щ":"sh","ъ":"u","ы":"y","ь":"","э":"e","ю":"yu","я":"ya","ё":"yo","ђ":"dj","є":"ye","і":"i","ї":"yi","ј":"j","љ":"lj","њ":"nj","ћ":"c","ѝ":"u","џ":"dz","Ґ":"G","ґ":"g","Ғ":"GH","ғ":"gh","Қ":"KH","қ":"kh","Ң":"NG","ң":"ng","Ү":"UE","ү":"ue","Ұ":"U","ұ":"u","Һ":"H","һ":"h","Ә":"AE","ә":"ae","Ө":"OE","ө":"oe","Ա":"A","Բ":"B","Գ":"G","Դ":"D","Ե":"E","Զ":"Z","Է":"E'","Ը":"Y'","Թ":"T'","Ժ":"JH","Ի":"I","Լ":"L","Խ":"X","Ծ":"C'","Կ":"K","Հ":"H","Ձ":"D'","Ղ":"GH","Ճ":"TW","Մ":"M","Յ":"Y","Ն":"N","Շ":"SH","Չ":"CH","Պ":"P","Ջ":"J","Ռ":"R'","Ս":"S","Վ":"V","Տ":"T","Ր":"R","Ց":"C","Փ":"P'","Ք":"Q'","Օ":"O''","Ֆ":"F","և":"EV","ء":"a","آ":"aa","أ":"a","ؤ":"u","إ":"i","ئ":"e","ا":"a","ب":"b","ة":"h","ت":"t","ث":"th","ج":"j","ح":"h","خ":"kh","د":"d","ذ":"th","ر":"r","ز":"z","س":"s","ش":"sh","ص":"s","ض":"dh","ط":"t","ظ":"z","ع":"a","غ":"gh","ف":"f","ق":"q","ك":"k","ل":"l","م":"m","ن":"n","ه":"h","و":"w","ى":"a","ي":"y","ً":"an","ٌ":"on","ٍ":"en","َ":"a","ُ":"u","ِ":"e","ْ":"","٠":"0","١":"1","٢":"2","٣":"3","٤":"4","٥":"5","٦":"6","٧":"7","٨":"8","٩":"9","پ":"p","چ":"ch","ژ":"zh","ک":"k","گ":"g","ی":"y","۰":"0","۱":"1","۲":"2","۳":"3","۴":"4","۵":"5","۶":"6","۷":"7","۸":"8","۹":"9","฿":"baht","ა":"a","ბ":"b","გ":"g","დ":"d","ე":"e","ვ":"v","ზ":"z","თ":"t","ი":"i","კ":"k","ლ":"l","მ":"m","ნ":"n","ო":"o","პ":"p","ჟ":"zh","რ":"r","ს":"s","ტ":"t","უ":"u","ფ":"f","ქ":"k","ღ":"gh","ყ":"q","შ":"sh","ჩ":"ch","ც":"ts","ძ":"dz","წ":"ts","ჭ":"ch","ხ":"kh","ჯ":"j","ჰ":"h","Ṣ":"S","ṣ":"s","Ẁ":"W","ẁ":"w","Ẃ":"W","ẃ":"w","Ẅ":"W","ẅ":"w","ẞ":"SS","Ạ":"A","ạ":"a","Ả":"A","ả":"a","Ấ":"A","ấ":"a","Ầ":"A","ầ":"a","Ẩ":"A","ẩ":"a","Ẫ":"A","ẫ":"a","Ậ":"A","ậ":"a","Ắ":"A","ắ":"a","Ằ":"A","ằ":"a","Ẳ":"A","ẳ":"a","Ẵ":"A","ẵ":"a","Ặ":"A","ặ":"a","Ẹ":"E","ẹ":"e","Ẻ":"E","ẻ":"e","Ẽ":"E","ẽ":"e","Ế":"E","ế":"e","Ề":"E","ề":"e","Ể":"E","ể":"e","Ễ":"E","ễ":"e","Ệ":"E","ệ":"e","Ỉ":"I","ỉ":"i","Ị":"I","ị":"i","Ọ":"O","ọ":"o","Ỏ":"O","ỏ":"o","Ố":"O","ố":"o","Ồ":"O","ồ":"o","Ổ":"O","ổ":"o","Ỗ":"O","ỗ":"o","Ộ":"O","ộ":"o","Ớ":"O","ớ":"o","Ờ":"O","ờ":"o","Ở":"O","ở":"o","Ỡ":"O","ỡ":"o","Ợ":"O","ợ":"o","Ụ":"U","ụ":"u","Ủ":"U","ủ":"u","Ứ":"U","ứ":"u","Ừ":"U","ừ":"u","Ử":"U","ử":"u","Ữ":"U","ữ":"u","Ự":"U","ự":"u","Ỳ":"Y","ỳ":"y","Ỵ":"Y","ỵ":"y","Ỷ":"Y","ỷ":"y","Ỹ":"Y","ỹ":"y","–":"-","‘":"'","’":"'","“":"\\"","”":"\\"","„":"\\"","†":"+","•":"*","…":"...","₠":"ecu","₢":"cruzeiro","₣":"french franc","₤":"lira","₥":"mill","₦":"naira","₧":"peseta","₨":"rupee","₩":"won","₪":"new shequel","₫":"dong","€":"euro","₭":"kip","₮":"tugrik","₯":"drachma","₰":"penny","₱":"peso","₲":"guarani","₳":"austral","₴":"hryvnia","₵":"cedi","₸":"kazakhstani tenge","₹":"indian rupee","₺":"turkish lira","₽":"russian ruble","₿":"bitcoin","℠":"sm","™":"tm","∂":"d","∆":"delta","∑":"sum","∞":"infinity","♥":"love","元":"yuan","円":"yen","﷼":"rial","ﻵ":"laa","ﻷ":"laa","ﻹ":"lai","ﻻ":"la"}`),a=JSON.parse('{"bg":{"Й":"Y","Ц":"Ts","Щ":"Sht","Ъ":"A","Ь":"Y","й":"y","ц":"ts","щ":"sht","ъ":"a","ь":"y"},"de":{"Ä":"AE","ä":"ae","Ö":"OE","ö":"oe","Ü":"UE","ü":"ue","ß":"ss","%":"prozent","&":"und","|":"oder","∑":"summe","∞":"unendlich","♥":"liebe"},"es":{"%":"por ciento","&":"y","<":"menor que",">":"mayor que","|":"o","¢":"centavos","£":"libras","¤":"moneda","₣":"francos","∑":"suma","∞":"infinito","♥":"amor"},"fr":{"%":"pourcent","&":"et","<":"plus petit",">":"plus grand","|":"ou","¢":"centime","£":"livre","¤":"devise","₣":"franc","∑":"somme","∞":"infini","♥":"amour"},"pt":{"%":"porcento","&":"e","<":"menor",">":"maior","|":"ou","¢":"centavo","∑":"soma","£":"libra","∞":"infinito","♥":"amor"},"uk":{"И":"Y","и":"y","Й":"Y","й":"y","Ц":"Ts","ц":"ts","Х":"Kh","х":"kh","Щ":"Shch","щ":"shch","Г":"H","г":"h"},"vi":{"Đ":"D","đ":"d"},"da":{"Ø":"OE","ø":"oe","Å":"AA","å":"aa","%":"procent","&":"og","|":"eller","$":"dollar","<":"mindre end",">":"større end"},"nb":{"&":"og","Å":"AA","Æ":"AE","Ø":"OE","å":"aa","æ":"ae","ø":"oe"},"it":{"&":"e"},"nl":{"&":"en"},"sv":{"&":"och","Å":"AA","Ä":"AE","Ö":"OE","å":"aa","ä":"ae","ö":"oe"}}');function c(l,o){if(typeof l!="string")throw new Error("slugify: string argument expected");o=typeof o=="string"?{replacement:o}:o||{};var g=a[o.locale]||{},h=o.replacement===void 0?"-":o.replacement,e=o.trim===void 0?!0:o.trim,i=l.normalize().split("").reduce(function(s,u){var m=g[u];return m===void 0&&(m=n[u]),m===void 0&&(m=u),m===h&&(m=" "),s+m.replace(o.remove||/[^\w\s$*_+~.()'"!\-:@]+/g,"")},"");return o.strict&&(i=i.replace(/[^A-Za-z0-9\s]/g,"")),e&&(i=i.trim()),i=i.replace(/\s+/g,h),o.lower&&(i=i.toLowerCase()),i}return c.extend=function(l){Object.assign(n,l)},c})})(me);const he=t=>t.split(/[\s-]/g).map(q).join(" ");function ge(t,r){const{navigation:n}=M().public.content;if(n===!1)return[];const a=l=>({...de(["title",...n.fields])(l),...ye(l==null?void 0:l.navigation)?l.navigation:{}}),c=t.sort((l,o)=>l._path.localeCompare(o._path)).reduce((l,o)=>{var m;const g=o._path.substring(1).split("/"),h=o._id.split(":").slice(1),e=!!((m=h[h.length-1])!=null&&m.match(/([1-9][0-9]*\.)?index.md/g)),i=f=>({title:f.title,_path:f._path,_file:f._file,children:[],...a(f),...f._draft?{_draft:!0}:{}}),s=i(o);if(e){const f=r[s._path];if(typeof(f==null?void 0:f.navigation)<"u"&&!(f!=null&&f.navigation))return l;if(o._path!=="/"){const d=i(o);s.children.push(d)}f&&Object.assign(s,a(f))}return g.length===1?(l.push(s),l):(g.slice(0,-1).reduce((f,d,v)=>{const I="/"+g.slice(0,v+1).join("/"),w=r[I];if(typeof(w==null?void 0:w.navigation)<"u"&&!w.navigation)return[];let A=f.find(B=>B._path===I);return A||(A={title:he(d),_path:I,_file:o._file,children:[],...w&&a(w)},f.push(A)),A.children},l).push(s),l)},[]);return Y(c)}const pe=new Intl.Collator(void 0,{numeric:!0,sensitivity:"base"});function Y(t){var n;t.forEach(a=>{a._file=a._file.split(".").slice(0,-1).join(".")});const r=t.sort((a,c)=>pe.compare(a._file,c._file));for(const a of r)(n=a.children)!=null&&n.length?Y(a.children):delete a.children,delete a._file;return t}function de(t){return r=>(r=r||{},t&&t.length?t.filter(n=>typeof r[n]<"u").reduce((n,a)=>Object.assign(n,{[a]:r[a]}),{}):r)}function ye(t){return Object.prototype.toString.call(t)==="[object Object]"}const we=t=>H(t,M().public.content.api.baseURL),ve=ie(ue({driver:V()}),"@content");function Ie(t){async function r(){const n=new Set(await t.getKeys("cache:")),a=U().getPreviewToken();if(a){const l=await t.getItem(`${a}$`).then(h=>h||{});if(Array.isArray(l.ignoreSources)){const h=l.ignoreSources.map(e=>`cache:${e.trim()}:`);for(const e of n)h.some(i=>e.startsWith(i))&&n.delete(e)}const o=await t.getKeys(`${a}:`),g=await Promise.all(o.map(h=>t.getItem(h)));for(const h of g)n.delete(`cache:${h._id}`),h.__deleted||n.add(`${a}:${h._id}`)}return await Promise.all(Array.from(n).map(l=>t.getItem(l)))}return{storage:t,fetch:N(r),query:n=>Z(N(r),{initialParams:n,legacy:!0})}}let C=null,K=null;async function Ae(){return K?await K:C||(K=Ee(),C=await K),C}async function Ee(){const t=J(),{content:r}=M().public,n=Ie(ve),a=await n.storage.getItem("integrity");if(r.integrity!==+(a||0)){const{contents:c,navigation:l}=await $fetch(we(r.integrity?`cache.${r.integrity}.json`:"cache.json"));await Promise.all(c.map(o=>n.storage.setItem(`cache:${o._id}`,o))),await n.storage.setItem("navigation",l),await n.storage.setItem("integrity",r.integrity)}return await t.callHook("content:storage",n.storage),n}async function je(t){const r=await Ae();if(!U().getPreviewToken()&&Object.keys(t||{}).length===0)return r.storage.getItem("navigation");const n=await r.query(t).where({_partial:!1,navigation:{$ne:!1}}).find(),c=(await r.query().where({_path:/\/_dir$/i,_partial:!0}).find()).reduce((l,o)=>{var h;((h=o.title)==null?void 0:h.toLowerCase())==="dir"&&(o.title=void 0);const g=o._path.split("/").slice(0,-1).join("/")||"/";return l[g]={...o,...o.body},l},{});return ge(n,c)}export{ve as contentStorage,Ie as createDB,je as generateNavigation,Ae as useContentDatabase}; +import{a6 as _,U as T,ac as W,I as M,N as J,Q as H}from"./Bb-NKJJJ.js";import{g as k,a as j,b as O,o as b,c as G,d as $,f as P,h as D,i as Z}from"./jaxqT3Hp.js";import{p as q}from"./C-v3KzvZ.js";import{u as U}from"./mjvvwssR.js";import"./IShi1APO.js";const F="memory",V=()=>{const t=new Map;return{name:F,getInstance:()=>t,hasItem(r){return t.has(r)},getItem(r){return t.get(r)??null},getItemRaw(r){return t.get(r)??null},setItem(r,n){t.set(r,n)},setItemRaw(r,n){t.set(r,n)},removeItem(r){t.delete(r)},getKeys(){return[...t.keys()]},clear(){t.clear()},dispose(){t.clear()}}};function Q(t){return!t||typeof t.then!="function"?Promise.resolve(t):t}function p(t,...r){try{return Q(t(...r))}catch(n){return Promise.reject(n)}}function X(t){const r=typeof t;return t===null||r!=="object"&&r!=="function"}function ee(t){const r=Object.getPrototypeOf(t);return!r||r.isPrototypeOf(Object)}function S(t){if(X(t))return String(t);if(ee(t)||Array.isArray(t))return JSON.stringify(t);if(typeof t.toJSON=="function")return S(t.toJSON());throw new Error("[unstorage] Cannot stringify value!")}function z(){if(typeof Buffer>"u")throw new TypeError("[unstorage] Buffer is not supported!")}const R="base64:";function te(t){if(typeof t=="string")return t;z();const r=Buffer.from(t).toString("base64");return R+r}function re(t){return typeof t!="string"||!t.startsWith(R)?t:(z(),Buffer.from(t.slice(R.length),"base64"))}const ne=["hasItem","getItem","getItemRaw","setItem","setItemRaw","removeItem","getMeta","setMeta","removeMeta","getKeys","clear","mount","unmount"];function ie(t,r){if(r=E(r),!r)return t;const n={...t};for(const a of ne)n[a]=(c="",...l)=>t[a](r+c,...l);return n.getKeys=(a="",...c)=>t.getKeys(r+a,...c).then(l=>l.map(o=>o.slice(r.length))),n}function y(t){return t?t.split("?")[0].replace(/[/\\]/g,":").replace(/:+/g,":").replace(/^:|:$/g,""):""}function ae(...t){return y(t.join(":"))}function E(t){return t=y(t),t?t+":":""}const se="memory",oe=()=>{const t=new Map;return{name:se,getInstance:()=>t,hasItem(r){return t.has(r)},getItem(r){return t.get(r)??null},getItemRaw(r){return t.get(r)??null},setItem(r,n){t.set(r,n)},setItemRaw(r,n){t.set(r,n)},removeItem(r){t.delete(r)},getKeys(){return[...t.keys()]},clear(){t.clear()},dispose(){t.clear()}}};function ue(t={}){const r={mounts:{"":t.driver||oe()},mountpoints:[""],watching:!1,watchListeners:[],unwatch:{}},n=e=>{for(const i of r.mountpoints)if(e.startsWith(i))return{base:i,relativeKey:e.slice(i.length),driver:r.mounts[i]};return{base:"",relativeKey:e,driver:r.mounts[""]}},a=(e,i)=>r.mountpoints.filter(s=>s.startsWith(e)||i&&e.startsWith(s)).map(s=>({relativeBase:e.length>s.length?e.slice(s.length):void 0,mountpoint:s,driver:r.mounts[s]})),c=(e,i)=>{if(r.watching){i=y(i);for(const s of r.watchListeners)s(e,i)}},l=async()=>{if(!r.watching){r.watching=!0;for(const e in r.mounts)r.unwatch[e]=await x(r.mounts[e],c,e)}},o=async()=>{if(r.watching){for(const e in r.unwatch)await r.unwatch[e]();r.unwatch={},r.watching=!1}},g=(e,i,s)=>{const u=new Map,m=f=>{let d=u.get(f.base);return d||(d={driver:f.driver,base:f.base,items:[]},u.set(f.base,d)),d};for(const f of e){const d=typeof f=="string",v=y(d?f:f.key),I=d?void 0:f.value,w=d||!f.options?i:{...i,...f.options},A=n(v);m(A).items.push({key:v,value:I,relativeKey:A.relativeKey,options:w})}return Promise.all([...u.values()].map(f=>s(f))).then(f=>f.flat())},h={hasItem(e,i={}){e=y(e);const{relativeKey:s,driver:u}=n(e);return p(u.hasItem,s,i)},getItem(e,i={}){e=y(e);const{relativeKey:s,driver:u}=n(e);return p(u.getItem,s,i).then(m=>_(m))},getItems(e,i){return g(e,i,s=>s.driver.getItems?p(s.driver.getItems,s.items.map(u=>({key:u.relativeKey,options:u.options})),i).then(u=>u.map(m=>({key:ae(s.base,m.key),value:_(m.value)}))):Promise.all(s.items.map(u=>p(s.driver.getItem,u.relativeKey,u.options).then(m=>({key:u.key,value:_(m)})))))},getItemRaw(e,i={}){e=y(e);const{relativeKey:s,driver:u}=n(e);return u.getItemRaw?p(u.getItemRaw,s,i):p(u.getItem,s,i).then(m=>re(m))},async setItem(e,i,s={}){if(i===void 0)return h.removeItem(e);e=y(e);const{relativeKey:u,driver:m}=n(e);m.setItem&&(await p(m.setItem,u,S(i),s),m.watch||c("update",e))},async setItems(e,i){await g(e,i,async s=>{if(s.driver.setItems)return p(s.driver.setItems,s.items.map(u=>({key:u.relativeKey,value:S(u.value),options:u.options})),i);s.driver.setItem&&await Promise.all(s.items.map(u=>p(s.driver.setItem,u.relativeKey,S(u.value),u.options)))})},async setItemRaw(e,i,s={}){if(i===void 0)return h.removeItem(e,s);e=y(e);const{relativeKey:u,driver:m}=n(e);if(m.setItemRaw)await p(m.setItemRaw,u,i,s);else if(m.setItem)await p(m.setItem,u,te(i),s);else return;m.watch||c("update",e)},async removeItem(e,i={}){typeof i=="boolean"&&(i={removeMeta:i}),e=y(e);const{relativeKey:s,driver:u}=n(e);u.removeItem&&(await p(u.removeItem,s,i),(i.removeMeta||i.removeMata)&&await p(u.removeItem,s+"$",i),u.watch||c("remove",e))},async getMeta(e,i={}){typeof i=="boolean"&&(i={nativeOnly:i}),e=y(e);const{relativeKey:s,driver:u}=n(e),m=Object.create(null);if(u.getMeta&&Object.assign(m,await p(u.getMeta,s,i)),!i.nativeOnly){const f=await p(u.getItem,s+"$",i).then(d=>_(d));f&&typeof f=="object"&&(typeof f.atime=="string"&&(f.atime=new Date(f.atime)),typeof f.mtime=="string"&&(f.mtime=new Date(f.mtime)),Object.assign(m,f))}return m},setMeta(e,i,s={}){return this.setItem(e+"$",i,s)},removeMeta(e,i={}){return this.removeItem(e+"$",i)},async getKeys(e,i={}){e=E(e);const s=a(e,!0);let u=[];const m=[];for(const f of s){const d=await p(f.driver.getKeys,f.relativeBase,i);for(const v of d){const I=f.mountpoint+y(v);u.some(w=>I.startsWith(w))||m.push(I)}u=[f.mountpoint,...u.filter(v=>!v.startsWith(f.mountpoint))]}return e?m.filter(f=>f.startsWith(e)&&f[f.length-1]!=="$"):m.filter(f=>f[f.length-1]!=="$")},async clear(e,i={}){e=E(e),await Promise.all(a(e,!1).map(async s=>{if(s.driver.clear)return p(s.driver.clear,s.relativeBase,i);if(s.driver.removeItem){const u=await s.driver.getKeys(s.relativeBase||"",i);return Promise.all(u.map(m=>s.driver.removeItem(m,i)))}}))},async dispose(){await Promise.all(Object.values(r.mounts).map(e=>L(e)))},async watch(e){return await l(),r.watchListeners.push(e),async()=>{r.watchListeners=r.watchListeners.filter(i=>i!==e),r.watchListeners.length===0&&await o()}},async unwatch(){r.watchListeners=[],await o()},mount(e,i){if(e=E(e),e&&r.mounts[e])throw new Error(`already mounted at ${e}`);return e&&(r.mountpoints.push(e),r.mountpoints.sort((s,u)=>u.length-s.length)),r.mounts[e]=i,r.watching&&Promise.resolve(x(i,c,e)).then(s=>{r.unwatch[e]=s}).catch(console.error),h},async unmount(e,i=!0){e=E(e),!(!e||!r.mounts[e])&&(r.watching&&e in r.unwatch&&(r.unwatch[e](),delete r.unwatch[e]),i&&await L(r.mounts[e]),r.mountpoints=r.mountpoints.filter(s=>s!==e),delete r.mounts[e])},getMount(e=""){e=y(e)+":";const i=n(e);return{driver:i.driver,base:i.base}},getMounts(e="",i={}){return e=y(e),a(e,i.parents).map(u=>({driver:u.driver,base:u.mountpoint}))},keys:(e,i={})=>h.getKeys(e,i),get:(e,i={})=>h.getItem(e,i),set:(e,i,s={})=>h.setItem(e,i,s),has:(e,i={})=>h.hasItem(e,i),del:(e,i={})=>h.removeItem(e,i),remove:(e,i={})=>h.removeItem(e,i)};return h}function x(t,r,n){return t.watch?t.watch((a,c)=>r(a,n+c)):()=>{}}async function L(t){typeof t.dispose=="function"&&await p(t.dispose)}function ce(t={}){const r=le(n,t.operators);function n(a,c){return typeof c!="object"||c instanceof RegExp?r.$eq(a,c):Object.keys(c||{}).every(l=>{const o=c[l];if(l.startsWith("$")&&r[l]){const g=r[l];return typeof g=="function"?g(a,o):!1}return n(k(a,l),o)})}return n}function le(t,r={}){return{$match:(n,a)=>t(n,a),$eq:(n,a)=>a instanceof RegExp?a.test(n):n===a,$ne:(n,a)=>a instanceof RegExp?!a.test(n):n!==a,$not:(n,a)=>!t(n,a),$and:(n,a)=>(j(a,"$and requires an array as condition"),a.every(c=>t(n,c))),$or:(n,a)=>(j(a,"$or requires an array as condition"),a.some(c=>t(n,c))),$in:(n,a)=>O(a).some(c=>Array.isArray(n)?t(n,{$contains:c}):t(n,c)),$contains:(n,a)=>(n=Array.isArray(n)?n:String(n),O(a).every(c=>n.includes(c))),$icontains:(n,a)=>{if(typeof a!="string")throw new TypeError("$icontains requires a string, use $contains instead");return n=String(n).toLocaleLowerCase(),O(a).every(c=>n.includes(c.toLocaleLowerCase()))},$containsAny:(n,a)=>(j(a,"$containsAny requires an array as condition"),n=Array.isArray(n)?n:String(n),a.some(c=>n.includes(c))),$exists:(n,a)=>a?typeof n<"u":typeof n>"u",$type:(n,a)=>typeof n===String(a),$regex:(n,a)=>{if(!(a instanceof RegExp)){const c=String(a).match(/\/(.*)\/([dgimsuy]*)$/);a=c!=null&&c[1]?new RegExp(c[1],c[2]||""):new RegExp(a)}return a.test(String(n||""))},$lt:(n,a)=>nn<=a,$gt:(n,a)=>n>a,$gte:(n,a)=>n>=a,...r||{}}}function fe(t){const r=ce(),n=(l,{query:o,before:g,after:h})=>{const e=typeof o=="string"?{_path:o}:o,i=l.findIndex(u=>r(u,e));g=g??1,h=h??1;const s=new Array(g+h).fill(null,0);return i===-1?s:s.map((u,m)=>l[i-g+m+ +(m>=g)]||null)},a=[(l,o)=>{const g=l.result.filter(h=>O(o.where).every(e=>r(h,e)));return{...l,result:g,total:g.length}},(l,o)=>O(o.sort).forEach(g=>G(l.result,g)),function(o,g,h){var e;if(g.surround){let i=n(((e=o.result)==null?void 0:e.length)===1?h:o.result,g.surround);i=$(P(g.without))(i),i=$(D(g.only))(i),o.surround=i}return o}],c=[(l,o)=>{if(o.skip)return{...l,result:l.result.slice(o.skip),skip:o.skip}},(l,o)=>{if(o.limit)return{...l,result:l.result.slice(0,o.limit),limit:o.limit}},function(o,g,h){var e,i,s;if(g.dirConfig){const u=((e=o.result[0])==null?void 0:e._path)||((s=(i=g.where)==null?void 0:i.find(m=>m._path))==null?void 0:s._path);if(typeof u=="string"){const m=h.find(f=>f._path===T(u,"_dir"));m&&(o.dirConfig={_path:m._path,...P(["_"])(m)})}}return o},(l,o)=>({...l,result:$(P(o.without))(l.result)}),(l,o)=>({...l,result:$(D(o.only))(l.result)})];return async l=>{const o=await t(),g=l.params(),h={result:o,limit:0,skip:0,total:o.length},e=a.reduce((s,u)=>u(s,g,o)||s,h);if(g.count)return{result:e.result.length};const i=c.reduce((s,u)=>u(s,g,o)||s,e);return g.first?{...b(["skip","limit","total"])(i),result:i.result[0]}:i}}function N(t){const r=fe(t);return async n=>{var l;n.params().first&&n.withDirConfig();const a=n.params(),c=await r(n);return a.surround?c==null?void 0:c.surround:(c!=null&&c.dirConfig&&(c.result={_path:(l=c.dirConfig)==null?void 0:l._path,...c.result,_dir:c.dirConfig}),c==null?void 0:c.result)}}var me={exports:{}};(function(t,r){(function(n,a,c){t.exports=c(),t.exports.default=c()})("slugify",W,function(){var n=JSON.parse(`{"$":"dollar","%":"percent","&":"and","<":"less",">":"greater","|":"or","¢":"cent","£":"pound","¤":"currency","¥":"yen","©":"(c)","ª":"a","®":"(r)","º":"o","À":"A","Á":"A","Â":"A","Ã":"A","Ä":"A","Å":"A","Æ":"AE","Ç":"C","È":"E","É":"E","Ê":"E","Ë":"E","Ì":"I","Í":"I","Î":"I","Ï":"I","Ð":"D","Ñ":"N","Ò":"O","Ó":"O","Ô":"O","Õ":"O","Ö":"O","Ø":"O","Ù":"U","Ú":"U","Û":"U","Ü":"U","Ý":"Y","Þ":"TH","ß":"ss","à":"a","á":"a","â":"a","ã":"a","ä":"a","å":"a","æ":"ae","ç":"c","è":"e","é":"e","ê":"e","ë":"e","ì":"i","í":"i","î":"i","ï":"i","ð":"d","ñ":"n","ò":"o","ó":"o","ô":"o","õ":"o","ö":"o","ø":"o","ù":"u","ú":"u","û":"u","ü":"u","ý":"y","þ":"th","ÿ":"y","Ā":"A","ā":"a","Ă":"A","ă":"a","Ą":"A","ą":"a","Ć":"C","ć":"c","Č":"C","č":"c","Ď":"D","ď":"d","Đ":"DJ","đ":"dj","Ē":"E","ē":"e","Ė":"E","ė":"e","Ę":"e","ę":"e","Ě":"E","ě":"e","Ğ":"G","ğ":"g","Ģ":"G","ģ":"g","Ĩ":"I","ĩ":"i","Ī":"i","ī":"i","Į":"I","į":"i","İ":"I","ı":"i","Ķ":"k","ķ":"k","Ļ":"L","ļ":"l","Ľ":"L","ľ":"l","Ł":"L","ł":"l","Ń":"N","ń":"n","Ņ":"N","ņ":"n","Ň":"N","ň":"n","Ō":"O","ō":"o","Ő":"O","ő":"o","Œ":"OE","œ":"oe","Ŕ":"R","ŕ":"r","Ř":"R","ř":"r","Ś":"S","ś":"s","Ş":"S","ş":"s","Š":"S","š":"s","Ţ":"T","ţ":"t","Ť":"T","ť":"t","Ũ":"U","ũ":"u","Ū":"u","ū":"u","Ů":"U","ů":"u","Ű":"U","ű":"u","Ų":"U","ų":"u","Ŵ":"W","ŵ":"w","Ŷ":"Y","ŷ":"y","Ÿ":"Y","Ź":"Z","ź":"z","Ż":"Z","ż":"z","Ž":"Z","ž":"z","Ə":"E","ƒ":"f","Ơ":"O","ơ":"o","Ư":"U","ư":"u","Lj":"LJ","lj":"lj","Nj":"NJ","nj":"nj","Ș":"S","ș":"s","Ț":"T","ț":"t","ə":"e","˚":"o","Ά":"A","Έ":"E","Ή":"H","Ί":"I","Ό":"O","Ύ":"Y","Ώ":"W","ΐ":"i","Α":"A","Β":"B","Γ":"G","Δ":"D","Ε":"E","Ζ":"Z","Η":"H","Θ":"8","Ι":"I","Κ":"K","Λ":"L","Μ":"M","Ν":"N","Ξ":"3","Ο":"O","Π":"P","Ρ":"R","Σ":"S","Τ":"T","Υ":"Y","Φ":"F","Χ":"X","Ψ":"PS","Ω":"W","Ϊ":"I","Ϋ":"Y","ά":"a","έ":"e","ή":"h","ί":"i","ΰ":"y","α":"a","β":"b","γ":"g","δ":"d","ε":"e","ζ":"z","η":"h","θ":"8","ι":"i","κ":"k","λ":"l","μ":"m","ν":"n","ξ":"3","ο":"o","π":"p","ρ":"r","ς":"s","σ":"s","τ":"t","υ":"y","φ":"f","χ":"x","ψ":"ps","ω":"w","ϊ":"i","ϋ":"y","ό":"o","ύ":"y","ώ":"w","Ё":"Yo","Ђ":"DJ","Є":"Ye","І":"I","Ї":"Yi","Ј":"J","Љ":"LJ","Њ":"NJ","Ћ":"C","Џ":"DZ","А":"A","Б":"B","В":"V","Г":"G","Д":"D","Е":"E","Ж":"Zh","З":"Z","И":"I","Й":"J","К":"K","Л":"L","М":"M","Н":"N","О":"O","П":"P","Р":"R","С":"S","Т":"T","У":"U","Ф":"F","Х":"H","Ц":"C","Ч":"Ch","Ш":"Sh","Щ":"Sh","Ъ":"U","Ы":"Y","Ь":"","Э":"E","Ю":"Yu","Я":"Ya","а":"a","б":"b","в":"v","г":"g","д":"d","е":"e","ж":"zh","з":"z","и":"i","й":"j","к":"k","л":"l","м":"m","н":"n","о":"o","п":"p","р":"r","с":"s","т":"t","у":"u","ф":"f","х":"h","ц":"c","ч":"ch","ш":"sh","щ":"sh","ъ":"u","ы":"y","ь":"","э":"e","ю":"yu","я":"ya","ё":"yo","ђ":"dj","є":"ye","і":"i","ї":"yi","ј":"j","љ":"lj","њ":"nj","ћ":"c","ѝ":"u","џ":"dz","Ґ":"G","ґ":"g","Ғ":"GH","ғ":"gh","Қ":"KH","қ":"kh","Ң":"NG","ң":"ng","Ү":"UE","ү":"ue","Ұ":"U","ұ":"u","Һ":"H","һ":"h","Ә":"AE","ә":"ae","Ө":"OE","ө":"oe","Ա":"A","Բ":"B","Գ":"G","Դ":"D","Ե":"E","Զ":"Z","Է":"E'","Ը":"Y'","Թ":"T'","Ժ":"JH","Ի":"I","Լ":"L","Խ":"X","Ծ":"C'","Կ":"K","Հ":"H","Ձ":"D'","Ղ":"GH","Ճ":"TW","Մ":"M","Յ":"Y","Ն":"N","Շ":"SH","Չ":"CH","Պ":"P","Ջ":"J","Ռ":"R'","Ս":"S","Վ":"V","Տ":"T","Ր":"R","Ց":"C","Փ":"P'","Ք":"Q'","Օ":"O''","Ֆ":"F","և":"EV","ء":"a","آ":"aa","أ":"a","ؤ":"u","إ":"i","ئ":"e","ا":"a","ب":"b","ة":"h","ت":"t","ث":"th","ج":"j","ح":"h","خ":"kh","د":"d","ذ":"th","ر":"r","ز":"z","س":"s","ش":"sh","ص":"s","ض":"dh","ط":"t","ظ":"z","ع":"a","غ":"gh","ف":"f","ق":"q","ك":"k","ل":"l","م":"m","ن":"n","ه":"h","و":"w","ى":"a","ي":"y","ً":"an","ٌ":"on","ٍ":"en","َ":"a","ُ":"u","ِ":"e","ْ":"","٠":"0","١":"1","٢":"2","٣":"3","٤":"4","٥":"5","٦":"6","٧":"7","٨":"8","٩":"9","پ":"p","چ":"ch","ژ":"zh","ک":"k","گ":"g","ی":"y","۰":"0","۱":"1","۲":"2","۳":"3","۴":"4","۵":"5","۶":"6","۷":"7","۸":"8","۹":"9","฿":"baht","ა":"a","ბ":"b","გ":"g","დ":"d","ე":"e","ვ":"v","ზ":"z","თ":"t","ი":"i","კ":"k","ლ":"l","მ":"m","ნ":"n","ო":"o","პ":"p","ჟ":"zh","რ":"r","ს":"s","ტ":"t","უ":"u","ფ":"f","ქ":"k","ღ":"gh","ყ":"q","შ":"sh","ჩ":"ch","ც":"ts","ძ":"dz","წ":"ts","ჭ":"ch","ხ":"kh","ჯ":"j","ჰ":"h","Ṣ":"S","ṣ":"s","Ẁ":"W","ẁ":"w","Ẃ":"W","ẃ":"w","Ẅ":"W","ẅ":"w","ẞ":"SS","Ạ":"A","ạ":"a","Ả":"A","ả":"a","Ấ":"A","ấ":"a","Ầ":"A","ầ":"a","Ẩ":"A","ẩ":"a","Ẫ":"A","ẫ":"a","Ậ":"A","ậ":"a","Ắ":"A","ắ":"a","Ằ":"A","ằ":"a","Ẳ":"A","ẳ":"a","Ẵ":"A","ẵ":"a","Ặ":"A","ặ":"a","Ẹ":"E","ẹ":"e","Ẻ":"E","ẻ":"e","Ẽ":"E","ẽ":"e","Ế":"E","ế":"e","Ề":"E","ề":"e","Ể":"E","ể":"e","Ễ":"E","ễ":"e","Ệ":"E","ệ":"e","Ỉ":"I","ỉ":"i","Ị":"I","ị":"i","Ọ":"O","ọ":"o","Ỏ":"O","ỏ":"o","Ố":"O","ố":"o","Ồ":"O","ồ":"o","Ổ":"O","ổ":"o","Ỗ":"O","ỗ":"o","Ộ":"O","ộ":"o","Ớ":"O","ớ":"o","Ờ":"O","ờ":"o","Ở":"O","ở":"o","Ỡ":"O","ỡ":"o","Ợ":"O","ợ":"o","Ụ":"U","ụ":"u","Ủ":"U","ủ":"u","Ứ":"U","ứ":"u","Ừ":"U","ừ":"u","Ử":"U","ử":"u","Ữ":"U","ữ":"u","Ự":"U","ự":"u","Ỳ":"Y","ỳ":"y","Ỵ":"Y","ỵ":"y","Ỷ":"Y","ỷ":"y","Ỹ":"Y","ỹ":"y","–":"-","‘":"'","’":"'","“":"\\"","”":"\\"","„":"\\"","†":"+","•":"*","…":"...","₠":"ecu","₢":"cruzeiro","₣":"french franc","₤":"lira","₥":"mill","₦":"naira","₧":"peseta","₨":"rupee","₩":"won","₪":"new shequel","₫":"dong","€":"euro","₭":"kip","₮":"tugrik","₯":"drachma","₰":"penny","₱":"peso","₲":"guarani","₳":"austral","₴":"hryvnia","₵":"cedi","₸":"kazakhstani tenge","₹":"indian rupee","₺":"turkish lira","₽":"russian ruble","₿":"bitcoin","℠":"sm","™":"tm","∂":"d","∆":"delta","∑":"sum","∞":"infinity","♥":"love","元":"yuan","円":"yen","﷼":"rial","ﻵ":"laa","ﻷ":"laa","ﻹ":"lai","ﻻ":"la"}`),a=JSON.parse('{"bg":{"Й":"Y","Ц":"Ts","Щ":"Sht","Ъ":"A","Ь":"Y","й":"y","ц":"ts","щ":"sht","ъ":"a","ь":"y"},"de":{"Ä":"AE","ä":"ae","Ö":"OE","ö":"oe","Ü":"UE","ü":"ue","ß":"ss","%":"prozent","&":"und","|":"oder","∑":"summe","∞":"unendlich","♥":"liebe"},"es":{"%":"por ciento","&":"y","<":"menor que",">":"mayor que","|":"o","¢":"centavos","£":"libras","¤":"moneda","₣":"francos","∑":"suma","∞":"infinito","♥":"amor"},"fr":{"%":"pourcent","&":"et","<":"plus petit",">":"plus grand","|":"ou","¢":"centime","£":"livre","¤":"devise","₣":"franc","∑":"somme","∞":"infini","♥":"amour"},"pt":{"%":"porcento","&":"e","<":"menor",">":"maior","|":"ou","¢":"centavo","∑":"soma","£":"libra","∞":"infinito","♥":"amor"},"uk":{"И":"Y","и":"y","Й":"Y","й":"y","Ц":"Ts","ц":"ts","Х":"Kh","х":"kh","Щ":"Shch","щ":"shch","Г":"H","г":"h"},"vi":{"Đ":"D","đ":"d"},"da":{"Ø":"OE","ø":"oe","Å":"AA","å":"aa","%":"procent","&":"og","|":"eller","$":"dollar","<":"mindre end",">":"større end"},"nb":{"&":"og","Å":"AA","Æ":"AE","Ø":"OE","å":"aa","æ":"ae","ø":"oe"},"it":{"&":"e"},"nl":{"&":"en"},"sv":{"&":"och","Å":"AA","Ä":"AE","Ö":"OE","å":"aa","ä":"ae","ö":"oe"}}');function c(l,o){if(typeof l!="string")throw new Error("slugify: string argument expected");o=typeof o=="string"?{replacement:o}:o||{};var g=a[o.locale]||{},h=o.replacement===void 0?"-":o.replacement,e=o.trim===void 0?!0:o.trim,i=l.normalize().split("").reduce(function(s,u){var m=g[u];return m===void 0&&(m=n[u]),m===void 0&&(m=u),m===h&&(m=" "),s+m.replace(o.remove||/[^\w\s$*_+~.()'"!\-:@]+/g,"")},"");return o.strict&&(i=i.replace(/[^A-Za-z0-9\s]/g,"")),e&&(i=i.trim()),i=i.replace(/\s+/g,h),o.lower&&(i=i.toLowerCase()),i}return c.extend=function(l){Object.assign(n,l)},c})})(me);const he=t=>t.split(/[\s-]/g).map(q).join(" ");function ge(t,r){const{navigation:n}=M().public.content;if(n===!1)return[];const a=l=>({...de(["title",...n.fields])(l),...ye(l==null?void 0:l.navigation)?l.navigation:{}}),c=t.sort((l,o)=>l._path.localeCompare(o._path)).reduce((l,o)=>{var m;const g=o._path.substring(1).split("/"),h=o._id.split(":").slice(1),e=!!((m=h[h.length-1])!=null&&m.match(/([1-9][0-9]*\.)?index.md/g)),i=f=>({title:f.title,_path:f._path,_file:f._file,children:[],...a(f),...f._draft?{_draft:!0}:{}}),s=i(o);if(e){const f=r[s._path];if(typeof(f==null?void 0:f.navigation)<"u"&&!(f!=null&&f.navigation))return l;if(o._path!=="/"){const d=i(o);s.children.push(d)}f&&Object.assign(s,a(f))}return g.length===1?(l.push(s),l):(g.slice(0,-1).reduce((f,d,v)=>{const I="/"+g.slice(0,v+1).join("/"),w=r[I];if(typeof(w==null?void 0:w.navigation)<"u"&&!w.navigation)return[];let A=f.find(B=>B._path===I);return A||(A={title:he(d),_path:I,_file:o._file,children:[],...w&&a(w)},f.push(A)),A.children},l).push(s),l)},[]);return Y(c)}const pe=new Intl.Collator(void 0,{numeric:!0,sensitivity:"base"});function Y(t){var n;t.forEach(a=>{a._file=a._file.split(".").slice(0,-1).join(".")});const r=t.sort((a,c)=>pe.compare(a._file,c._file));for(const a of r)(n=a.children)!=null&&n.length?Y(a.children):delete a.children,delete a._file;return t}function de(t){return r=>(r=r||{},t&&t.length?t.filter(n=>typeof r[n]<"u").reduce((n,a)=>Object.assign(n,{[a]:r[a]}),{}):r)}function ye(t){return Object.prototype.toString.call(t)==="[object Object]"}const we=t=>H(t,M().public.content.api.baseURL),ve=ie(ue({driver:V()}),"@content");function Ie(t){async function r(){const n=new Set(await t.getKeys("cache:")),a=U().getPreviewToken();if(a){const l=await t.getItem(`${a}$`).then(h=>h||{});if(Array.isArray(l.ignoreSources)){const h=l.ignoreSources.map(e=>`cache:${e.trim()}:`);for(const e of n)h.some(i=>e.startsWith(i))&&n.delete(e)}const o=await t.getKeys(`${a}:`),g=await Promise.all(o.map(h=>t.getItem(h)));for(const h of g)n.delete(`cache:${h._id}`),h.__deleted||n.add(`${a}:${h._id}`)}return await Promise.all(Array.from(n).map(l=>t.getItem(l)))}return{storage:t,fetch:N(r),query:n=>Z(N(r),{initialParams:n,legacy:!0})}}let C=null,K=null;async function Ae(){return K?await K:C||(K=Ee(),C=await K),C}async function Ee(){const t=J(),{content:r}=M().public,n=Ie(ve),a=await n.storage.getItem("integrity");if(r.integrity!==+(a||0)){const{contents:c,navigation:l}=await $fetch(we(r.integrity?`cache.${r.integrity}.json`:"cache.json"));await Promise.all(c.map(o=>n.storage.setItem(`cache:${o._id}`,o))),await n.storage.setItem("navigation",l),await n.storage.setItem("integrity",r.integrity)}return await t.callHook("content:storage",n.storage),n}async function je(t){const r=await Ae();if(!U().getPreviewToken()&&Object.keys(t||{}).length===0)return r.storage.getItem("navigation");const n=await r.query(t).where({_partial:!1,navigation:{$ne:!1}}).find(),c=(await r.query().where({_path:/\/_dir$/i,_partial:!0}).find()).reduce((l,o)=>{var h;((h=o.title)==null?void 0:h.toLowerCase())==="dir"&&(o.title=void 0);const g=o._path.split("/").slice(0,-1).join("/")||"/";return l[g]={...o,...o.body},l},{});return ge(n,c)}export{ve as contentStorage,Ie as createDB,je as generateNavigation,Ae as useContentDatabase}; diff --git a/_nuxt/BMI0969X.js b/_nuxt/Co_4egda.js similarity index 63% rename from _nuxt/BMI0969X.js rename to _nuxt/Co_4egda.js index 4ea366da..2a0c3f22 100644 --- a/_nuxt/BMI0969X.js +++ b/_nuxt/Co_4egda.js @@ -1 +1 @@ -import{m as r,o,c as t,a8 as s}from"./BGikLauJ.js";const c={};function n(e,a){return o(),t("tr",null,[s(e.$slots,"default")])}const f=r(c,[["render",n]]);export{f as default}; +import{m as r,o,c as t,a8 as s}from"./Bb-NKJJJ.js";const c={};function n(e,a){return o(),t("tr",null,[s(e.$slots,"default")])}const f=r(c,[["render",n]]);export{f as default}; diff --git a/_nuxt/B3q-b4GE.js b/_nuxt/CyLrJzPW.js similarity index 56% rename from _nuxt/B3q-b4GE.js rename to _nuxt/CyLrJzPW.js index cd52142f..a33af9c4 100644 --- a/_nuxt/B3q-b4GE.js +++ b/_nuxt/CyLrJzPW.js @@ -1 +1 @@ -import{m as r,a8 as t}from"./BGikLauJ.js";const s={};function n(e,o){return t(e.$slots,"default")}const c=r(s,[["render",n]]);export{c as default}; +import{m as r,a8 as t}from"./Bb-NKJJJ.js";const s={};function n(e,o){return t(e.$slots,"default")}const c=r(s,[["render",n]]);export{c as default}; diff --git a/_nuxt/D-AOKchP.js b/_nuxt/D-cHg-Fi.js similarity index 65% rename from _nuxt/D-AOKchP.js rename to _nuxt/D-cHg-Fi.js index 5e08b4f9..5d5380c4 100644 --- a/_nuxt/D-AOKchP.js +++ b/_nuxt/D-cHg-Fi.js @@ -1 +1 @@ -import{m as o,o as t,c,a8 as r}from"./BGikLauJ.js";const s={};function n(e,a){return t(),c("blockquote",null,[r(e.$slots,"default")])}const f=o(s,[["render",n]]);export{f as default}; +import{m as o,o as t,c,a8 as r}from"./Bb-NKJJJ.js";const s={};function n(e,a){return t(),c("blockquote",null,[r(e.$slots,"default")])}const f=o(s,[["render",n]]);export{f as default}; diff --git a/_nuxt/C_llEvAK.js b/_nuxt/D-lCVgpc.js similarity index 96% rename from _nuxt/C_llEvAK.js rename to _nuxt/D-lCVgpc.js index 8c07e96b..df80c528 100644 --- a/_nuxt/C_llEvAK.js +++ b/_nuxt/D-lCVgpc.js @@ -1 +1 @@ -import{m as l,j as c,c as d,i as a,f as n,b as e,t as f,S as m,o as p,s as u}from"./BGikLauJ.js";const y={setup(){return{show:c(!1)}}},g={class:"bg-gradeint-to-r flex min-h-screen items-center justify-center from-blue-400 to-indigo-500"},h={class:"flex w-96 items-center space-x-4 rounded-2xl bg-white p-4 shadow-xl"},x={class:"absolute inset-x-0 top-0 my-4 flex w-full items-center justify-center"};function _(r,t,v,o,w,b){const s=u,i=m;return p(),d("div",g,[a(i,{as:"template",show:o.show,enter:"transition transform duration-300 ease-out","enter-from":"translate-x-4 opacity-0","enter-to":"translate-x-0 opacity-100",leave:"transition transform duration-300 ease-in","leave-from":"opacity-100","leave-to":"opacity-0"},{default:n(()=>[e("div",h,[a(s,{as:"img",enter:"transition transform duration-300 ease-out","enter-from":"rotate-90 scale-50 opacity-0","enter-to":"rotate-0 scale-100 opacity-100",leave:"transition transform duration-300 ease-in","leave-from":"opacity-100","leave-to":"opacity-0",class:"h-16 w-16 rounded-full ring ring-indigo-500 ring-opacity-50 ring-offset-2",src:"/images/placeholder.png"}),a(s,{enter:"transition transform duration-300 ease-out","enter-from":"translate-x-12 opacity-0","enter-to":"translate-x-0 opacity-100",leave:"transition transform duration-300 ease-in","leave-from":"opacity-100","leave-to":"opacity-0"},{default:n(()=>t[1]||(t[1]=[e("h1",{class:"font-medium text-gray-900"},"Floofy McFloof",-1),e("p",{class:"mt-0.5 text-gray-500"}," Hey! I'm McFloof and I have been a very good boi this year. ",-1)])),_:1})])]),_:1},8,["show"]),e("div",x,[e("button",{class:"rounded-lg bg-blue-200 px-4 py-1.5 text-sm font-medium text-indigo-900 hover:bg-blue-100 focus:outline-none focus:ring-2 focus:ring-indigo-600",onClick:t[0]||(t[0]=H=>o.show=!o.show)},f(o.show?"Hide":"Show")+" profile ",1)])])}const j=l(y,[["render",_]]);export{j as default}; +import{m as l,j as c,c as d,i as a,f as n,b as e,t as f,S as m,o as p,s as u}from"./Bb-NKJJJ.js";const y={setup(){return{show:c(!1)}}},g={class:"bg-gradeint-to-r flex min-h-screen items-center justify-center from-blue-400 to-indigo-500"},h={class:"flex w-96 items-center space-x-4 rounded-2xl bg-white p-4 shadow-xl"},x={class:"absolute inset-x-0 top-0 my-4 flex w-full items-center justify-center"};function _(r,t,v,o,w,b){const s=u,i=m;return p(),d("div",g,[a(i,{as:"template",show:o.show,enter:"transition transform duration-300 ease-out","enter-from":"translate-x-4 opacity-0","enter-to":"translate-x-0 opacity-100",leave:"transition transform duration-300 ease-in","leave-from":"opacity-100","leave-to":"opacity-0"},{default:n(()=>[e("div",h,[a(s,{as:"img",enter:"transition transform duration-300 ease-out","enter-from":"rotate-90 scale-50 opacity-0","enter-to":"rotate-0 scale-100 opacity-100",leave:"transition transform duration-300 ease-in","leave-from":"opacity-100","leave-to":"opacity-0",class:"h-16 w-16 rounded-full ring ring-indigo-500 ring-opacity-50 ring-offset-2",src:"/images/placeholder.png"}),a(s,{enter:"transition transform duration-300 ease-out","enter-from":"translate-x-12 opacity-0","enter-to":"translate-x-0 opacity-100",leave:"transition transform duration-300 ease-in","leave-from":"opacity-100","leave-to":"opacity-0"},{default:n(()=>t[1]||(t[1]=[e("h1",{class:"font-medium text-gray-900"},"Floofy McFloof",-1),e("p",{class:"mt-0.5 text-gray-500"}," Hey! I'm McFloof and I have been a very good boi this year. ",-1)])),_:1})])]),_:1},8,["show"]),e("div",x,[e("button",{class:"rounded-lg bg-blue-200 px-4 py-1.5 text-sm font-medium text-indigo-900 hover:bg-blue-100 focus:outline-none focus:ring-2 focus:ring-indigo-600",onClick:t[0]||(t[0]=H=>o.show=!o.show)},f(o.show?"Hide":"Show")+" profile ",1)])])}const j=l(y,[["render",_]]);export{j as default}; diff --git a/_nuxt/DX7ac3Mz.js b/_nuxt/D1bB9L70.js similarity index 88% rename from _nuxt/DX7ac3Mz.js rename to _nuxt/D1bB9L70.js index 39b79de1..251503ef 100644 --- a/_nuxt/DX7ac3Mz.js +++ b/_nuxt/D1bB9L70.js @@ -1 +1 @@ -import{d as p,I as f,k as i,o as s,c as n,a as u,a8 as r}from"./BGikLauJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH2",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=i(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h2))});return(e,k)=>(s(),n("h2",{id:t.id},[t.id&&u(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; +import{d as p,I as f,k as i,o as s,c as n,a as u,a8 as r}from"./Bb-NKJJJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH2",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=i(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h2))});return(e,k)=>(s(),n("h2",{id:t.id},[t.id&&u(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; diff --git a/_nuxt/CjTcA1Xt.js b/_nuxt/D7KuDJ9z.js similarity index 64% rename from _nuxt/CjTcA1Xt.js rename to _nuxt/D7KuDJ9z.js index c3ac9971..5b2aa62c 100644 --- a/_nuxt/CjTcA1Xt.js +++ b/_nuxt/D7KuDJ9z.js @@ -1 +1 @@ -import{m as o,o as r,c as t,a8 as s}from"./BGikLauJ.js";const c={};function n(e,a){return r(),t("th",null,[s(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; +import{m as o,o as r,c as t,a8 as s}from"./Bb-NKJJJ.js";const c={};function n(e,a){return r(),t("th",null,[s(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; diff --git a/_nuxt/D7IuvTrr.js b/_nuxt/D9IbAJ0u.js similarity index 53% rename from _nuxt/D7IuvTrr.js rename to _nuxt/D9IbAJ0u.js index eb787c6a..0f1ebbe5 100644 --- a/_nuxt/D7IuvTrr.js +++ b/_nuxt/D9IbAJ0u.js @@ -1 +1 @@ -import{m as e,o as r,c}from"./BGikLauJ.js";const o={};function t(n,s){return r(),c("hr")}const _=e(o,[["render",t]]);export{_ as default}; +import{m as e,o as r,c}from"./Bb-NKJJJ.js";const o={};function t(n,s){return r(),c("hr")}const _=e(o,[["render",t]]);export{_ as default}; diff --git a/_nuxt/UcoZFNdm.js b/_nuxt/DA5xXx2M.js similarity index 96% rename from _nuxt/UcoZFNdm.js rename to _nuxt/DA5xXx2M.js index b8128db7..4932faa9 100644 --- a/_nuxt/UcoZFNdm.js +++ b/_nuxt/DA5xXx2M.js @@ -1 +1 @@ -import{m as p,o as b,c,b as i,C as r,D as m,i as u}from"./BGikLauJ.js";const w={props:{id:{type:String,default:"canvas"},canvasHeight:{type:Number,default:250},type:{type:String,default:"sin"},diameter:{type:Number,default:15},amplitude:{type:Number,default:75},lambda:{type:Number,default:266}},mounted(){const o=e=>{const a=e.windowWidth>=450?this.canvasHeight:200,s=e.windowWidth>=450?800:300;let t=0,d=a/2,n=0;const l=100,v=125;e.setup=()=>{e.createCanvas(s,a),t=-this.diameter},e.draw=()=>{this.type==="sin"?d=this.amplitude*e.sin(t*(e.TWO_PI/this.lambda))+a/2:this.type==="cos"?d=this.amplitude*e.cos(t*(e.TWO_PI/this.lambda))+a/2:this.type==="tan"?d=this.amplitude*e.tan(t*(e.TWO_PI/this.lambda))+a/2:d=a/2,n=Math.round(e.map(d,-this.amplitude+a/2,this.amplitude+a/2,0,255)),t=t>=s+this.diameter?-this.diameter:t+1,e.noStroke(),e.fill(e.color(n,l,v)),e.ellipse(t,d,this.diameter,this.diameter)},e.mousePressed=()=>{e.clear()}};new this.$p5(o,this.id)},data(){return{}}},f=["id"];function h(o,e,a,s,t,d){return b(),c("div",{id:a.id},null,8,f)}const y=p(w,[["render",h]]),_={data(){return{wave:{amplitude:75,lambda:266,diameter:15}}}},x={class:"bg-gradient-to-b from-green-800 to-gray-800"},g={class:"grid h-screen place-items-center"},I={class:"font-mono text-white"},k={class:"my-2 items-end md:flex"},W={class:"hidden md:block"},N={class:"mb-2 border-2 border-white"},T={class:"mb-2 border-2 border-white"},H={class:"border-2 border-white"};function P(o,e,a,s,t,d){const n=y;return b(),c("div",x,[i("div",g,[i("div",I,[i("div",k,[e[6]||(e[6]=i("div",{class:"flex-none md:flex-1"},"y(x) = A sin((2π / λ) x)",-1)),i("div",W,[e[3]||(e[3]=i("label",{for:"aInput",class:"form-label inline-block"},"A",-1)),r(i("input",{id:"aInput","onUpdate:modelValue":e[0]||(e[0]=l=>t.wave.amplitude=l),type:"number",class:"w-16 border-2 border-white bg-transparent p-2 focus:outline-none"},null,512),[[m,t.wave.amplitude,void 0,{number:!0}]]),e[4]||(e[4]=i("label",{for:"lInput",class:"form-label inline-block"},"λ",-1)),r(i("input",{id:"lInput","onUpdate:modelValue":e[1]||(e[1]=l=>t.wave.lambda=l),type:"number",class:"w-16 border-2 border-white bg-transparent p-2 focus:outline-none"},null,512),[[m,t.wave.lambda,void 0,{number:!0}]]),e[5]||(e[5]=i("label",{for:"dInput",class:"form-label inline-block"},"◒",-1)),r(i("input",{id:"dInput","onUpdate:modelValue":e[2]||(e[2]=l=>t.wave.diameter=l),type:"number",class:"w-16 border-2 border-white bg-transparent p-2 focus:outline-none"},null,512),[[m,t.wave.diameter,void 0,{number:!0}]])])]),i("div",N,[u(n,{id:"canvas1",type:"sin",diameter:t.wave.diameter,amplitude:t.wave.amplitude,lambda:t.wave.lambda},null,8,["diameter","amplitude","lambda"])]),e[7]||(e[7]=i("div",null,"y(x) = A cos((2π / λ) x)",-1)),i("div",T,[u(n,{id:"canvas2",type:"cos",diameter:t.wave.diameter,amplitude:t.wave.amplitude,lambda:t.wave.lambda},null,8,["diameter","amplitude","lambda"])]),e[8]||(e[8]=i("div",null,"y(x) = A tan((2π / λ) x)",-1)),i("div",H,[u(n,{id:"canvas3",type:"tan",diameter:t.wave.diameter,amplitude:t.wave.amplitude,lambda:t.wave.lambda},null,8,["diameter","amplitude","lambda"])]),e[9]||(e[9]=i("div",{class:"my-2 w-48 md:w-full"}," Click or tap anywhere to clear the canvas! ",-1))])])])}const A=p(_,[["render",P]]);export{A as default}; +import{m as p,o as b,c,b as i,C as r,D as m,i as u}from"./Bb-NKJJJ.js";const w={props:{id:{type:String,default:"canvas"},canvasHeight:{type:Number,default:250},type:{type:String,default:"sin"},diameter:{type:Number,default:15},amplitude:{type:Number,default:75},lambda:{type:Number,default:266}},mounted(){const o=e=>{const a=e.windowWidth>=450?this.canvasHeight:200,s=e.windowWidth>=450?800:300;let t=0,d=a/2,n=0;const l=100,v=125;e.setup=()=>{e.createCanvas(s,a),t=-this.diameter},e.draw=()=>{this.type==="sin"?d=this.amplitude*e.sin(t*(e.TWO_PI/this.lambda))+a/2:this.type==="cos"?d=this.amplitude*e.cos(t*(e.TWO_PI/this.lambda))+a/2:this.type==="tan"?d=this.amplitude*e.tan(t*(e.TWO_PI/this.lambda))+a/2:d=a/2,n=Math.round(e.map(d,-this.amplitude+a/2,this.amplitude+a/2,0,255)),t=t>=s+this.diameter?-this.diameter:t+1,e.noStroke(),e.fill(e.color(n,l,v)),e.ellipse(t,d,this.diameter,this.diameter)},e.mousePressed=()=>{e.clear()}};new this.$p5(o,this.id)},data(){return{}}},f=["id"];function h(o,e,a,s,t,d){return b(),c("div",{id:a.id},null,8,f)}const y=p(w,[["render",h]]),_={data(){return{wave:{amplitude:75,lambda:266,diameter:15}}}},x={class:"bg-gradient-to-b from-green-800 to-gray-800"},g={class:"grid h-screen place-items-center"},I={class:"font-mono text-white"},k={class:"my-2 items-end md:flex"},W={class:"hidden md:block"},N={class:"mb-2 border-2 border-white"},T={class:"mb-2 border-2 border-white"},H={class:"border-2 border-white"};function P(o,e,a,s,t,d){const n=y;return b(),c("div",x,[i("div",g,[i("div",I,[i("div",k,[e[6]||(e[6]=i("div",{class:"flex-none md:flex-1"},"y(x) = A sin((2π / λ) x)",-1)),i("div",W,[e[3]||(e[3]=i("label",{for:"aInput",class:"form-label inline-block"},"A",-1)),r(i("input",{id:"aInput","onUpdate:modelValue":e[0]||(e[0]=l=>t.wave.amplitude=l),type:"number",class:"w-16 border-2 border-white bg-transparent p-2 focus:outline-none"},null,512),[[m,t.wave.amplitude,void 0,{number:!0}]]),e[4]||(e[4]=i("label",{for:"lInput",class:"form-label inline-block"},"λ",-1)),r(i("input",{id:"lInput","onUpdate:modelValue":e[1]||(e[1]=l=>t.wave.lambda=l),type:"number",class:"w-16 border-2 border-white bg-transparent p-2 focus:outline-none"},null,512),[[m,t.wave.lambda,void 0,{number:!0}]]),e[5]||(e[5]=i("label",{for:"dInput",class:"form-label inline-block"},"◒",-1)),r(i("input",{id:"dInput","onUpdate:modelValue":e[2]||(e[2]=l=>t.wave.diameter=l),type:"number",class:"w-16 border-2 border-white bg-transparent p-2 focus:outline-none"},null,512),[[m,t.wave.diameter,void 0,{number:!0}]])])]),i("div",N,[u(n,{id:"canvas1",type:"sin",diameter:t.wave.diameter,amplitude:t.wave.amplitude,lambda:t.wave.lambda},null,8,["diameter","amplitude","lambda"])]),e[7]||(e[7]=i("div",null,"y(x) = A cos((2π / λ) x)",-1)),i("div",T,[u(n,{id:"canvas2",type:"cos",diameter:t.wave.diameter,amplitude:t.wave.amplitude,lambda:t.wave.lambda},null,8,["diameter","amplitude","lambda"])]),e[8]||(e[8]=i("div",null,"y(x) = A tan((2π / λ) x)",-1)),i("div",H,[u(n,{id:"canvas3",type:"tan",diameter:t.wave.diameter,amplitude:t.wave.amplitude,lambda:t.wave.lambda},null,8,["diameter","amplitude","lambda"])]),e[9]||(e[9]=i("div",{class:"my-2 w-48 md:w-full"}," Click or tap anywhere to clear the canvas! ",-1))])])])}const A=p(_,[["render",P]]);export{A as default}; diff --git a/_nuxt/CzhJn57k.js b/_nuxt/DAkPvNyh.js similarity index 99% rename from _nuxt/CzhJn57k.js rename to _nuxt/DAkPvNyh.js index e1feb08a..a057280d 100644 --- a/_nuxt/CzhJn57k.js +++ b/_nuxt/DAkPvNyh.js @@ -1 +1 @@ -import{q,v as D,x as T,o as b,c as p,b as t,d as E,j as W,y as R,i as C,a as _,F as A,r as F,t as h}from"./BGikLauJ.js";const M={id:"waveform-canvas-container",class:"bg-green h-full w-full"},N=["width","height"],P={__name:"Waveform",props:{timeDomainBufferHistory:{type:Array},canvasWidth:{type:Number},canvasHeight:{type:Number},fillStyle:{type:String,default:"rgba(0,0,0)"},strokeStyle:{type:String,default:"rgb(255, 255, 255)"}},setup(f){const l=f;return q(()=>{const d=document.getElementById("waveformCanvas2"),e=d.getContext("2d"),s=document.getElementById("waveform-canvas-container"),c=l.canvasHeight?l.canvasHeight:s.clientHeight,o=l.canvasWidth?l.canvasWidth:s.clientWidth;d.height=c,d.width=o,e.fillRect(0,0,o,c),D(l.timeDomainBufferHistory,a=>{e.fillStyle=l.fillStyle,e.fillRect(0,0,o,c),e.lineWidth=2,e.strokeStyle=l.strokeStyle,e.beginPath();const u=a.slice(-1)[0],v=u.length,m=o/v;let g=0;for(let i=0;i{}),(d,e)=>(b(),p("div",M,[t("canvas",{id:"waveformCanvas2",width:f.canvasWidth,height:f.canvasHeight},null,8,N)]))}},U={id:"spectrogram-canvas-container",class:"h-full w-full"},O=["width","height"],k=36,G={__name:"Spectrogram",props:{frequencyDomainBufferHistory:{type:Array},canvasWidth:{type:Number},canvasHeight:{type:Number},fillStyle:{type:String,default:"rgba(0,0,0)"}},setup(f){const l=f,d=(e,s,c,o,a)=>(e-s)*(a-s)/(c-s)+o;return q(()=>{const e=document.getElementById("spectrogramCanvas"),s=e.getContext("2d"),c=document.getElementById("spectrogram-canvas-container"),o=l.canvasHeight?l.canvasHeight:c.clientHeight,a=l.canvasWidth?l.canvasWidth:c.clientWidth;e.height=o,e.width=a,s.fillRect(0,0,a,o),D(l.frequencyDomainBufferHistory,u=>{s.fillStyle=l.fillStyle,s.fillRect(0,0,a,o);const v=u.slice(-k),m=a/128,g=o/k;for(let i=0;i<128;i++)for(let n=0;n(b(),p("div",null,[t("div",U,[t("canvas",{id:"spectrogramCanvas",width:f.canvasWidth,height:f.canvasHeight},null,8,O)])]))}},L={id:"bar-graph-container",class:"h-full w-full"},V=["width","height"],I=2,Y={__name:"FrequencyBarGraph",props:{audioBufferHistory:{type:Array},canvasWidth:{type:Number},canvasHeight:{type:Number},fillStyle:{type:String,default:"rgba(0,0,0)"},strokeStyle:{type:String,default:"rgb(255, 255, 255)"}},setup(f){const l=f,d=(e,s,c,o,a)=>(e-s)*(a-s)/(c-s)+o;return q(()=>{const e=document.getElementById("frequencyBarGraphCanvas"),s=e.getContext("2d"),c=document.getElementById("bar-graph-container"),o=l.canvasHeight?l.canvasHeight:c.clientHeight,a=l.canvasWidth?l.canvasWidth:c.clientWidth;e.height=o,e.width=a,s.fillRect(0,0,a,o),D(l.audioBufferHistory,u=>{s.fillStyle=l.fillStyle,s.fillRect(0,0,a,o);const v=u.slice(-1)[0],m=a/v.length-I;let g=0;for(let i=0;i(b(),p("div",null,[t("div",L,[t("canvas",{id:"frequencyBarGraphCanvas",width:f.canvasWidth,height:f.canvasHeight},null,8,V)])]))}},j={class:"container mx-auto font-mono"},z={class:"bg-background text-white"},K={class:"grid grid-cols-3 gap-4"},X={class:"col-span-3 rounded-xl bg-black/75 p-4 lg:col-span-1"},Z={class:"col-span-3 rounded-xl bg-black/75 p-4 lg:col-span-1"},J={class:"col-span-3 rounded-xl bg-black/75 p-4 lg:col-span-1"},Q={class:"col-span-3 rounded-xl bg-black/75 p-4"},tt={class:"w-full table-fixed"},et={class:"text-left font-bold"},nt={class:"text-left"},st={class:"text-left"},lt={class:"text-left"},ot={class:"text-left"},at={class:"truncate text-right"},it={class:"col-span-3 rounded-xl bg-black/75 p-4"},ct={class:"w-full table-fixed"},rt={class:"text-left font-bold"},dt={class:"text-left"},ut={class:"text-left"},ht={class:"text-left"},ft={class:"text-left"},gt={class:"truncate text-right"},B=256,yt=E({__name:"index",setup(f){let l=null,d=null,e=null,s=null,c=null,o=W([]),a=null,u=W([]);W(!1);const v=()=>{requestAnimationFrame(v),e.getByteTimeDomainData(c),c!==null&&(o.value.push(c.slice()),o.value.length>B&&o.value.shift()),e.getByteFrequencyData(a),a!==null&&(u.value.push(a.slice()),u.value.length>B&&u.value.shift())},m=()=>{navigator.mediaDevices.getUserMedia({audio:!0}).then(i=>{l=i,d=new(window.AudioContext||window.webkitAudioContext),e=d.createAnalyser(),s=d.createMediaStreamSource(l),s.connect(e),e.fftSize=2048,c=new Uint8Array(e.frequencyBinCount),a=new Uint8Array(e.frequencyBinCount);for(let n=0;n{l!==null&&l.active&&l.getAudioTracks().forEach(i=>{i.stop()})};return R(()=>{console.log("disabling"),g()}),(i,n)=>{const y=P,w=G,$=Y;return b(),p("div",j,[t("div",z,[t("div",K,[t("div",{class:"col-span-3 space-x-2"},[t("div",{class:"inline-block rounded-lg bg-black px-6 py-2 text-sm font-bold uppercase text-white shadow-lg shadow-green-400/25 transition duration-150 ease-in-out hover:cursor-pointer hover:shadow-white/25",onClick:m}," Enable "),t("div",{class:"inline-block rounded-lg bg-black px-6 py-2 text-sm font-bold uppercase text-white shadow-lg shadow-red-400/25 transition duration-150 ease-in-out hover:cursor-pointer hover:shadow-white/25",onClick:g}," Disable ")]),t("div",X,[n[0]||(n[0]=t("div",{class:"mb-2 text-xl font-bold"},"Time Domain Waveform",-1)),C(y,{timeDomainBufferHistory:_(o),strokeStyle:"rgb(255, 0, 255)",class:"h-72 border-2 border-gray-400"},null,8,["timeDomainBufferHistory"])]),t("div",Z,[n[1]||(n[1]=t("div",{class:"mb-2 text-xl font-bold"},"Frequency Spectrogram",-1)),C(w,{frequencyDomainBufferHistory:_(u),class:"h-72 border-2 border-gray-400"},null,8,["frequencyDomainBufferHistory"])]),t("div",J,[n[2]||(n[2]=t("div",{class:"mb-2 text-xl font-bold"},"Frequency Bar Chart",-1)),C($,{audioBufferHistory:_(u),class:"h-72 border-2 border-gray-400"},null,8,["audioBufferHistory"])]),t("div",Q,[n[4]||(n[4]=t("div",{class:"mb-2 text-xl font-bold"},"Frequency Buffer History",-1)),t("div",null,[t("table",tt,[n[3]||(n[3]=t("thead",{class:"collapse border-b-2 md:visible"},[t("tr",null,[t("th",{class:"text-left"},"Index"),t("th",{class:"text-left"},"Mean"),t("th",{class:"text-left"},"Min"),t("th",{class:"text-left"},"Max"),t("th",{class:"text-left"},"FFT"),t("th",{class:"text-right"},"Buffer")])],-1)),t("tbody",null,[(b(!0),p(A,null,F(_(u).slice(-10),(r,x)=>(b(),p("tr",{key:x},[t("td",et,h(x),1),t("td",nt,h((r.reduce((H,S)=>H+S)/r.length).toFixed(2)),1),t("td",st,h(Math.min(...r)),1),t("td",lt,h(Math.max(...r)),1),t("td",ot,h(r.length),1),t("td",at,h(r.slice(0,4))+" ... "+h(r.slice(-4)),1)]))),128))])])])]),t("div",it,[n[6]||(n[6]=t("div",{class:"mb-2 text-xl font-bold"},"Time Domain Buffer History",-1)),t("div",null,[t("table",ct,[n[5]||(n[5]=t("thead",{class:"collapse border-b-2 md:visible"},[t("tr",null,[t("th",{class:"text-left"},"Index"),t("th",{class:"text-left"},"Mean"),t("th",{class:"text-left"},"Min"),t("th",{class:"text-left"},"Max"),t("th",{class:"text-left"},"FFT"),t("th",{class:"text-right"},"Buffer")])],-1)),t("tbody",null,[(b(!0),p(A,null,F(_(o).slice(-10),(r,x)=>(b(),p("tr",{key:x},[t("td",rt,h(x),1),t("td",dt,h((r.reduce((H,S)=>H+S)/r.length).toFixed(2)),1),t("td",ut,h(Math.min(...r)),1),t("td",ht,h(Math.max(...r)),1),t("td",ft,h(r.length),1),t("td",gt,h(r.slice(0,4))+" ... "+h(r.slice(-4)),1)]))),128))])])])])])])])}}});export{yt as default}; +import{q,v as D,x as T,o as b,c as p,b as t,d as E,j as W,y as R,i as C,a as _,F as A,r as F,t as h}from"./Bb-NKJJJ.js";const M={id:"waveform-canvas-container",class:"bg-green h-full w-full"},N=["width","height"],P={__name:"Waveform",props:{timeDomainBufferHistory:{type:Array},canvasWidth:{type:Number},canvasHeight:{type:Number},fillStyle:{type:String,default:"rgba(0,0,0)"},strokeStyle:{type:String,default:"rgb(255, 255, 255)"}},setup(f){const l=f;return q(()=>{const d=document.getElementById("waveformCanvas2"),e=d.getContext("2d"),s=document.getElementById("waveform-canvas-container"),c=l.canvasHeight?l.canvasHeight:s.clientHeight,o=l.canvasWidth?l.canvasWidth:s.clientWidth;d.height=c,d.width=o,e.fillRect(0,0,o,c),D(l.timeDomainBufferHistory,a=>{e.fillStyle=l.fillStyle,e.fillRect(0,0,o,c),e.lineWidth=2,e.strokeStyle=l.strokeStyle,e.beginPath();const u=a.slice(-1)[0],v=u.length,m=o/v;let g=0;for(let i=0;i{}),(d,e)=>(b(),p("div",M,[t("canvas",{id:"waveformCanvas2",width:f.canvasWidth,height:f.canvasHeight},null,8,N)]))}},U={id:"spectrogram-canvas-container",class:"h-full w-full"},O=["width","height"],k=36,G={__name:"Spectrogram",props:{frequencyDomainBufferHistory:{type:Array},canvasWidth:{type:Number},canvasHeight:{type:Number},fillStyle:{type:String,default:"rgba(0,0,0)"}},setup(f){const l=f,d=(e,s,c,o,a)=>(e-s)*(a-s)/(c-s)+o;return q(()=>{const e=document.getElementById("spectrogramCanvas"),s=e.getContext("2d"),c=document.getElementById("spectrogram-canvas-container"),o=l.canvasHeight?l.canvasHeight:c.clientHeight,a=l.canvasWidth?l.canvasWidth:c.clientWidth;e.height=o,e.width=a,s.fillRect(0,0,a,o),D(l.frequencyDomainBufferHistory,u=>{s.fillStyle=l.fillStyle,s.fillRect(0,0,a,o);const v=u.slice(-k),m=a/128,g=o/k;for(let i=0;i<128;i++)for(let n=0;n(b(),p("div",null,[t("div",U,[t("canvas",{id:"spectrogramCanvas",width:f.canvasWidth,height:f.canvasHeight},null,8,O)])]))}},L={id:"bar-graph-container",class:"h-full w-full"},V=["width","height"],I=2,Y={__name:"FrequencyBarGraph",props:{audioBufferHistory:{type:Array},canvasWidth:{type:Number},canvasHeight:{type:Number},fillStyle:{type:String,default:"rgba(0,0,0)"},strokeStyle:{type:String,default:"rgb(255, 255, 255)"}},setup(f){const l=f,d=(e,s,c,o,a)=>(e-s)*(a-s)/(c-s)+o;return q(()=>{const e=document.getElementById("frequencyBarGraphCanvas"),s=e.getContext("2d"),c=document.getElementById("bar-graph-container"),o=l.canvasHeight?l.canvasHeight:c.clientHeight,a=l.canvasWidth?l.canvasWidth:c.clientWidth;e.height=o,e.width=a,s.fillRect(0,0,a,o),D(l.audioBufferHistory,u=>{s.fillStyle=l.fillStyle,s.fillRect(0,0,a,o);const v=u.slice(-1)[0],m=a/v.length-I;let g=0;for(let i=0;i(b(),p("div",null,[t("div",L,[t("canvas",{id:"frequencyBarGraphCanvas",width:f.canvasWidth,height:f.canvasHeight},null,8,V)])]))}},j={class:"container mx-auto font-mono"},z={class:"bg-background text-white"},K={class:"grid grid-cols-3 gap-4"},X={class:"col-span-3 rounded-xl bg-black/75 p-4 lg:col-span-1"},Z={class:"col-span-3 rounded-xl bg-black/75 p-4 lg:col-span-1"},J={class:"col-span-3 rounded-xl bg-black/75 p-4 lg:col-span-1"},Q={class:"col-span-3 rounded-xl bg-black/75 p-4"},tt={class:"w-full table-fixed"},et={class:"text-left font-bold"},nt={class:"text-left"},st={class:"text-left"},lt={class:"text-left"},ot={class:"text-left"},at={class:"truncate text-right"},it={class:"col-span-3 rounded-xl bg-black/75 p-4"},ct={class:"w-full table-fixed"},rt={class:"text-left font-bold"},dt={class:"text-left"},ut={class:"text-left"},ht={class:"text-left"},ft={class:"text-left"},gt={class:"truncate text-right"},B=256,yt=E({__name:"index",setup(f){let l=null,d=null,e=null,s=null,c=null,o=W([]),a=null,u=W([]);W(!1);const v=()=>{requestAnimationFrame(v),e.getByteTimeDomainData(c),c!==null&&(o.value.push(c.slice()),o.value.length>B&&o.value.shift()),e.getByteFrequencyData(a),a!==null&&(u.value.push(a.slice()),u.value.length>B&&u.value.shift())},m=()=>{navigator.mediaDevices.getUserMedia({audio:!0}).then(i=>{l=i,d=new(window.AudioContext||window.webkitAudioContext),e=d.createAnalyser(),s=d.createMediaStreamSource(l),s.connect(e),e.fftSize=2048,c=new Uint8Array(e.frequencyBinCount),a=new Uint8Array(e.frequencyBinCount);for(let n=0;n{l!==null&&l.active&&l.getAudioTracks().forEach(i=>{i.stop()})};return R(()=>{console.log("disabling"),g()}),(i,n)=>{const y=P,w=G,$=Y;return b(),p("div",j,[t("div",z,[t("div",K,[t("div",{class:"col-span-3 space-x-2"},[t("div",{class:"inline-block rounded-lg bg-black px-6 py-2 text-sm font-bold uppercase text-white shadow-lg shadow-green-400/25 transition duration-150 ease-in-out hover:cursor-pointer hover:shadow-white/25",onClick:m}," Enable "),t("div",{class:"inline-block rounded-lg bg-black px-6 py-2 text-sm font-bold uppercase text-white shadow-lg shadow-red-400/25 transition duration-150 ease-in-out hover:cursor-pointer hover:shadow-white/25",onClick:g}," Disable ")]),t("div",X,[n[0]||(n[0]=t("div",{class:"mb-2 text-xl font-bold"},"Time Domain Waveform",-1)),C(y,{timeDomainBufferHistory:_(o),strokeStyle:"rgb(255, 0, 255)",class:"h-72 border-2 border-gray-400"},null,8,["timeDomainBufferHistory"])]),t("div",Z,[n[1]||(n[1]=t("div",{class:"mb-2 text-xl font-bold"},"Frequency Spectrogram",-1)),C(w,{frequencyDomainBufferHistory:_(u),class:"h-72 border-2 border-gray-400"},null,8,["frequencyDomainBufferHistory"])]),t("div",J,[n[2]||(n[2]=t("div",{class:"mb-2 text-xl font-bold"},"Frequency Bar Chart",-1)),C($,{audioBufferHistory:_(u),class:"h-72 border-2 border-gray-400"},null,8,["audioBufferHistory"])]),t("div",Q,[n[4]||(n[4]=t("div",{class:"mb-2 text-xl font-bold"},"Frequency Buffer History",-1)),t("div",null,[t("table",tt,[n[3]||(n[3]=t("thead",{class:"collapse border-b-2 md:visible"},[t("tr",null,[t("th",{class:"text-left"},"Index"),t("th",{class:"text-left"},"Mean"),t("th",{class:"text-left"},"Min"),t("th",{class:"text-left"},"Max"),t("th",{class:"text-left"},"FFT"),t("th",{class:"text-right"},"Buffer")])],-1)),t("tbody",null,[(b(!0),p(A,null,F(_(u).slice(-10),(r,x)=>(b(),p("tr",{key:x},[t("td",et,h(x),1),t("td",nt,h((r.reduce((H,S)=>H+S)/r.length).toFixed(2)),1),t("td",st,h(Math.min(...r)),1),t("td",lt,h(Math.max(...r)),1),t("td",ot,h(r.length),1),t("td",at,h(r.slice(0,4))+" ... "+h(r.slice(-4)),1)]))),128))])])])]),t("div",it,[n[6]||(n[6]=t("div",{class:"mb-2 text-xl font-bold"},"Time Domain Buffer History",-1)),t("div",null,[t("table",ct,[n[5]||(n[5]=t("thead",{class:"collapse border-b-2 md:visible"},[t("tr",null,[t("th",{class:"text-left"},"Index"),t("th",{class:"text-left"},"Mean"),t("th",{class:"text-left"},"Min"),t("th",{class:"text-left"},"Max"),t("th",{class:"text-left"},"FFT"),t("th",{class:"text-right"},"Buffer")])],-1)),t("tbody",null,[(b(!0),p(A,null,F(_(o).slice(-10),(r,x)=>(b(),p("tr",{key:x},[t("td",rt,h(x),1),t("td",dt,h((r.reduce((H,S)=>H+S)/r.length).toFixed(2)),1),t("td",ut,h(Math.min(...r)),1),t("td",ht,h(Math.max(...r)),1),t("td",ft,h(r.length),1),t("td",gt,h(r.slice(0,4))+" ... "+h(r.slice(-4)),1)]))),128))])])])])])])])}}});export{yt as default}; diff --git a/_nuxt/_kju5I3X.js b/_nuxt/DEdI-0CJ.js similarity index 88% rename from _nuxt/_kju5I3X.js rename to _nuxt/DEdI-0CJ.js index 89c7f99c..f4e426d1 100644 --- a/_nuxt/_kju5I3X.js +++ b/_nuxt/DEdI-0CJ.js @@ -1 +1 @@ -import{d as p,I as f,k as i,o as s,c as n,a as u,a8 as r}from"./BGikLauJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH3",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=i(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h3))});return(e,k)=>(s(),n("h3",{id:t.id},[t.id&&u(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; +import{d as p,I as f,k as i,o as s,c as n,a as u,a8 as r}from"./Bb-NKJJJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH3",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=i(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h3))});return(e,k)=>(s(),n("h3",{id:t.id},[t.id&&u(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; diff --git a/_nuxt/DS100L4J.js b/_nuxt/DS100L4J.js new file mode 100644 index 00000000..28dd53df --- /dev/null +++ b/_nuxt/DS100L4J.js @@ -0,0 +1 @@ +import{_ as m}from"./BMXP4Aud.js";import"./Bb-NKJJJ.js";export{m as default}; diff --git a/_nuxt/m3fkSCjB.js b/_nuxt/DVb7Ucrz.js similarity index 86% rename from _nuxt/m3fkSCjB.js rename to _nuxt/DVb7Ucrz.js index 1f5185be..1af7a0fa 100644 --- a/_nuxt/m3fkSCjB.js +++ b/_nuxt/DVb7Ucrz.js @@ -1 +1 @@ -import{d as r,a as n,o as s,c as a,g as e,b as o,h as i}from"./BGikLauJ.js";const d={key:0},u=r({__name:"ProseScript",props:{src:{type:String,default:""}},setup(c){return(p,t)=>n(!1)?(s(),a("div",d,t[0]||(t[0]=[e(" Rendering the "),o("code",null,"script",-1),e(" element is dangerous and is disabled by default. Consider implementing your own "),o("code",null,"ProseScript",-1),e(" element to have control over script rendering. ")]))):i("",!0)}});export{u as default}; +import{d as r,a as n,o as s,c as a,g as e,b as o,h as i}from"./Bb-NKJJJ.js";const d={key:0},u=r({__name:"ProseScript",props:{src:{type:String,default:""}},setup(c){return(p,t)=>n(!1)?(s(),a("div",d,t[0]||(t[0]=[e(" Rendering the "),o("code",null,"script",-1),e(" element is dangerous and is disabled by default. Consider implementing your own "),o("code",null,"ProseScript",-1),e(" element to have control over script rendering. ")]))):i("",!0)}});export{u as default}; diff --git a/_nuxt/DbdbR-pn.js b/_nuxt/DXvJDNSx.js similarity index 64% rename from _nuxt/DbdbR-pn.js rename to _nuxt/DXvJDNSx.js index f92fadf9..9cda8594 100644 --- a/_nuxt/DbdbR-pn.js +++ b/_nuxt/DXvJDNSx.js @@ -1 +1 @@ -import{m as o,o as r,c as s,a8 as t}from"./BGikLauJ.js";const c={};function n(e,a){return r(),s("ol",null,[t(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; +import{m as o,o as r,c as s,a8 as t}from"./Bb-NKJJJ.js";const c={};function n(e,a){return r(),s("ol",null,[t(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; diff --git a/_nuxt/DsIJUosc.js b/_nuxt/DZAhYQpW.js similarity index 88% rename from _nuxt/DsIJUosc.js rename to _nuxt/DZAhYQpW.js index 19ef7990..5bfa7ab1 100644 --- a/_nuxt/DsIJUosc.js +++ b/_nuxt/DZAhYQpW.js @@ -1 +1 @@ -import{d as p,I as f,k as i,o as s,c as n,a as u,a8 as r}from"./BGikLauJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH4",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=i(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h4))});return(e,k)=>(s(),n("h4",{id:t.id},[t.id&&u(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; +import{d as p,I as f,k as i,o as s,c as n,a as u,a8 as r}from"./Bb-NKJJJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH4",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=i(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h4))});return(e,k)=>(s(),n("h4",{id:t.id},[t.id&&u(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; diff --git a/_nuxt/CDv6ev47.js b/_nuxt/DcUhz6y1.js similarity index 97% rename from _nuxt/CDv6ev47.js rename to _nuxt/DcUhz6y1.js index c0217858..28644d41 100644 --- a/_nuxt/CDv6ev47.js +++ b/_nuxt/DcUhz6y1.js @@ -1 +1 @@ -import{m as s,c as h,p as c,o as l}from"./BGikLauJ.js";const a=["#FFBA08","#FAA307","#F48C06","#E85D04","#DC2F02","#D00000","#9D0208","#6A040F","#370617","#03071E"];class o{constructor(){this.parent=void 0,this.children=[],this.orientation="h",this.x=0,this.y=0,this.w=500,this.h=500}add(i){i.parent=this,this.children.push(i),this.children.forEach((n,r)=>{if(this.orientation==="h"){const e=this.w/this.children.length,t=this.x+e*r;n.x=t,n.y=this.y,n.w=e,n.h=this.h}else if(this.orientation==="v"){const e=this.h/this.children.length,t=this.y+e*r;n.x=this.x,n.y=t,n.w=this.w,n.h=e}})}render(i){i.rect(this.x,this.y,this.w,this.h),this.children.forEach((n,r)=>{i.fill(a[r%9]),n.render(i)})}traverseForPoint(i,n){if(this.children.length===0)return this;for(let r=0;r=e.x&&i<=e.x+e.w&&n>=e.y&&n<=e.y+e.h)return e.children.length===0?e:e.traverseForPoint(i,n)}}}const f={data(){return{}},mounted(){const d=i=>{const e=new o;for(let t=0;t<5;t++)e.add(new o);e.children[0].orientation="v";for(let t=0;t<3;t++)e.children[0].add(new o);e.children[0].children[2].orientation="v";for(let t=0;t<4;t++)e.children[0].children[2].add(new o);e.children[1].orientation="v";for(let t=0;t<3;t++)e.children[1].add(new o);for(let t=0;t<9;t++)e.children[1].children[0].add(new o);e.children[3].orientation="v";for(let t=0;t<3;t++)e.children[3].add(new o);e.children[3].children[2].orientation="v";for(let t=0;t<9;t++)e.children[3].children[2].add(new o);i.setup=()=>{i.createCanvas(500,500),i.noLoop(),i.stroke(255),i.strokeWeight(2),e.render(i)},i.mouseClicked=()=>{const t=e.traverseForPoint(i.mouseX,i.mouseY);t&&(t.add(new o),t.add(new o),t.render(i))}};new this.$p5(d,"canvas")}},v={class:"select-none bg-gradient-to-b from-green-800 to-gray-800"};function w(d,i,n,r,e,t){return l(),h("div",v,i[0]||(i[0]=[c('
        m-ary tree based tiling
        ',1)]))}const m=s(f,[["render",w]]);export{m as default}; +import{m as s,c as h,p as c,o as l}from"./Bb-NKJJJ.js";const a=["#FFBA08","#FAA307","#F48C06","#E85D04","#DC2F02","#D00000","#9D0208","#6A040F","#370617","#03071E"];class o{constructor(){this.parent=void 0,this.children=[],this.orientation="h",this.x=0,this.y=0,this.w=500,this.h=500}add(i){i.parent=this,this.children.push(i),this.children.forEach((n,r)=>{if(this.orientation==="h"){const e=this.w/this.children.length,t=this.x+e*r;n.x=t,n.y=this.y,n.w=e,n.h=this.h}else if(this.orientation==="v"){const e=this.h/this.children.length,t=this.y+e*r;n.x=this.x,n.y=t,n.w=this.w,n.h=e}})}render(i){i.rect(this.x,this.y,this.w,this.h),this.children.forEach((n,r)=>{i.fill(a[r%9]),n.render(i)})}traverseForPoint(i,n){if(this.children.length===0)return this;for(let r=0;r=e.x&&i<=e.x+e.w&&n>=e.y&&n<=e.y+e.h)return e.children.length===0?e:e.traverseForPoint(i,n)}}}const f={data(){return{}},mounted(){const d=i=>{const e=new o;for(let t=0;t<5;t++)e.add(new o);e.children[0].orientation="v";for(let t=0;t<3;t++)e.children[0].add(new o);e.children[0].children[2].orientation="v";for(let t=0;t<4;t++)e.children[0].children[2].add(new o);e.children[1].orientation="v";for(let t=0;t<3;t++)e.children[1].add(new o);for(let t=0;t<9;t++)e.children[1].children[0].add(new o);e.children[3].orientation="v";for(let t=0;t<3;t++)e.children[3].add(new o);e.children[3].children[2].orientation="v";for(let t=0;t<9;t++)e.children[3].children[2].add(new o);i.setup=()=>{i.createCanvas(500,500),i.noLoop(),i.stroke(255),i.strokeWeight(2),e.render(i)},i.mouseClicked=()=>{const t=e.traverseForPoint(i.mouseX,i.mouseY);t&&(t.add(new o),t.add(new o),t.render(i))}};new this.$p5(d,"canvas")}},v={class:"select-none bg-gradient-to-b from-green-800 to-gray-800"};function w(d,i,n,r,e,t){return l(),h("div",v,i[0]||(i[0]=[c('
        m-ary tree based tiling
        ',1)]))}const m=s(f,[["render",w]]);export{m as default}; diff --git a/_nuxt/BYwurayZ.js b/_nuxt/DcYzECSc.js similarity index 65% rename from _nuxt/BYwurayZ.js rename to _nuxt/DcYzECSc.js index e1daf3b2..d0db4793 100644 --- a/_nuxt/BYwurayZ.js +++ b/_nuxt/DcYzECSc.js @@ -1 +1 @@ -import{m as o,o as r,c as t,a8 as n}from"./BGikLauJ.js";const s={};function c(e,a){return r(),t("strong",null,[n(e.$slots,"default")])}const f=o(s,[["render",c]]);export{f as default}; +import{m as o,o as r,c as t,a8 as n}from"./Bb-NKJJJ.js";const s={};function c(e,a){return r(),t("strong",null,[n(e.$slots,"default")])}const f=o(s,[["render",c]]);export{f as default}; diff --git a/_nuxt/B9UQgCMx.js b/_nuxt/Dcfpj4_h.js similarity index 64% rename from _nuxt/B9UQgCMx.js rename to _nuxt/Dcfpj4_h.js index 862edeff..60acd9b4 100644 --- a/_nuxt/B9UQgCMx.js +++ b/_nuxt/Dcfpj4_h.js @@ -1 +1 @@ -import{m as o,o as r,c as s,a8 as t}from"./BGikLauJ.js";const c={};function n(e,a){return r(),s("ul",null,[t(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; +import{m as o,o as r,c as s,a8 as t}from"./Bb-NKJJJ.js";const c={};function n(e,a){return r(),s("ul",null,[t(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; diff --git a/_nuxt/ZO3tSbNI.js b/_nuxt/DjUAd0VA.js similarity index 98% rename from _nuxt/ZO3tSbNI.js rename to _nuxt/DjUAd0VA.js index 48128f35..b5d17389 100644 --- a/_nuxt/ZO3tSbNI.js +++ b/_nuxt/DjUAd0VA.js @@ -1 +1 @@ -import{m as _,c as l,b as s,g as y,l as p,h as g,F as b,r as m,o as n,t as f}from"./BGikLauJ.js";const v={data(){return{ix:0,a:[],b:[],timeline:[]}},computed:{step(){return this.timeline.length>0?this.timeline[this.ix]:{}},done(){return this.timeline.length-1===this.ix},started(){return this.timeline.length>0}},mounted(){this.generateRandomMatrices()},created(){typeof window>"u"||window.addEventListener("keydown",o=>{o.key===" "?this.incStep():o.key==="Escape"&&(this.ix=0)})},methods:{randomMatrix(o,t){const c=[];for(let h=0;hMath.round(Math.random()*10)));return c},generateRandomMatrices(){this.a=this.randomMatrix(3,4),this.b=this.randomMatrix(4,3),this.timeline=[],this.ix=0},matmul(o,t){this.timeline=[];const c=[];for(let i=0;i{t[e].forEach((a,r)=>{const x=[];o[e].forEach((d,u)=>{x.push(`${o[r][u]}*${t[u][e]}`),c[r][e]+=o[r][u]*t[u][e],h[e*o.length+r]=`${x.join(" + ")} = ${c[r][e]}`;const w=JSON.parse(JSON.stringify(c));this.timeline.push({row_a:r,col_a:u,row_b:u,col_b:e,row_c:r,col_c:e,cur_a:o[r][u],cur_b:t[u][e],cur_c:c[r][e],c:w,all_steps:h.slice()})})})})},incStep(){this.ix!==this.timeline.length-1&&(this.ix+=1)},decStep(){this.ix>0&&(this.ix-=1)}}},k={class:"m-4 space-y-4"},M={class:"m-2 inline-flex space-x-2"},S=["disabled"],C=["disabled"],B={class:"flex flex-wrap"},E={class:"w-34 m-2 border border-gray-400 sm:w-min"},A={class:"flex items-center justify-center p-4"},N={class:"text-center"},R={class:"w-34 m-2 border border-gray-400 sm:w-min"},V={class:"flex items-center justify-center p-4"},D={class:"text-center"},F={class:"flex items-center justify-center p-4"},J={class:"text-center"},L={key:1,class:"m-2 w-72 border border-gray-400 bg-gray-100 p-2"},O={class:"font-mono"},z={key:0,class:"italic text-green-600"};function P(o,t,c,h,i,e){return n(),l("div",k,[t[10]||(t[10]=s("div",{class:"m-2 w-4/5"},[y(" Walk through the steps of matrix multiplication with randomly generated matrices. Press the "),s("em",null,"Spacebar"),y(" or click the "),s("em",{class:"bg-blue-200 text-blue-800"},"buttons"),y(" to iterate through the steps. ")],-1)),s("div",M,[s("button",{class:p(["bg-blue-200 px-4 py-2 font-bold text-blue-800 hover:bg-blue-800 hover:text-blue-200",{"cursor-not-allowed opacity-25":!e.started}]),disabled:!e.started,onClick:t[0]||(t[0]=(...a)=>e.decStep&&e.decStep(...a))},t[4]||(t[4]=[s("svg",{xmlns:"http://www.w3.org/2000/svg",class:"h-4 w-4",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor"},[s("path",{"stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M11 19l-7-7 7-7m8 14l-7-7 7-7"})],-1)]),10,S),e.done?g("",!0):(n(),l("button",{key:0,class:"bg-blue-200 px-4 py-2 font-bold text-blue-800 hover:bg-blue-800 hover:text-blue-200",onClick:t[1]||(t[1]=a=>e.matmul(i.a,i.b))}," Start ")),e.done?(n(),l("button",{key:1,class:"bg-orange-200 px-4 py-2 font-bold text-orange-800 hover:bg-blue-800 hover:text-blue-200",onClick:t[2]||(t[2]=(...a)=>e.generateRandomMatrices&&e.generateRandomMatrices(...a))}," Again! ")):g("",!0),s("button",{class:p(["bg-blue-200 px-4 py-2 font-bold text-blue-800 hover:bg-blue-800 hover:text-blue-200",{"cursor-not-allowed opacity-25":!e.started}]),disabled:!e.started,onClick:t[3]||(t[3]=(...a)=>e.incStep&&e.incStep(...a))},t[5]||(t[5]=[s("svg",{xmlns:"http://www.w3.org/2000/svg",class:"h-4 w-4",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor"},[s("path",{"stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M13 5l7 7-7 7M5 5l7 7-7 7"})],-1)]),10,C)]),s("div",B,[s("div",E,[t[6]||(t[6]=s("div",{class:"bg-gray-200 p-4 text-center font-bold"},"Matrix A",-1)),s("div",A,[s("table",N,[(n(!0),l(b,null,m(i.a,(a,r)=>(n(),l("tr",{key:r},[(n(!0),l(b,null,m(i.a[r],(x,d)=>(n(),l("td",{key:d,class:p([{"text-green-600":e.step&&r==e.step.row_a&&d==e.step.col_a,"bg-orange-100":e.step&&r==e.step.row_a},"p-2"])},f(i.a[r][d]),3))),128))]))),128))])])]),s("div",R,[t[7]||(t[7]=s("div",{class:"bg-gray-200 p-4 text-center font-bold"},"Matrix B",-1)),s("div",V,[s("table",D,[(n(!0),l(b,null,m(i.b,(a,r)=>(n(),l("tr",{key:r},[(n(!0),l(b,null,m(i.b[r],(x,d)=>(n(),l("td",{key:d,class:p([{"text-green-600":e.step&&r==e.step.row_b&&d==e.step.col_b,"bg-orange-100":e.step&&d==e.step.col_b},"p-2"])},f(i.b[r][d]),3))),128))]))),128))])])]),e.started?(n(),l("div",{key:0,class:p(["m-2 w-72 border border-gray-400 sm:w-min",{"bg-green-200":e.done}])},[t[8]||(t[8]=s("div",{class:"bg-gray-200 p-4 text-center font-bold"},"Result",-1)),s("div",F,[s("table",J,[(n(!0),l(b,null,m(e.step.c,(a,r)=>(n(),l("tr",{key:r},[(n(!0),l(b,null,m(e.step.c[r],(x,d)=>(n(),l("td",{key:d,class:"p-2"},f(e.step.c[r][d]),1))),128))]))),128))])])],2)):g("",!0),e.started?(n(),l("div",L,[t[9]||(t[9]=s("div",{class:"text-center font-bold"},"Steps",-1)),s("div",O,[(n(!0),l(b,null,m(e.step.all_steps,(a,r)=>(n(),l("div",{key:r},f(a),1))),128)),e.done?(n(),l("div",z,"Done!")):g("",!0)])])):g("",!0)]),g("",!0)])}const W=_(v,[["render",P]]);export{W as default}; +import{m as _,c as l,b as s,g as y,l as p,h as g,F as b,r as m,o as n,t as f}from"./Bb-NKJJJ.js";const v={data(){return{ix:0,a:[],b:[],timeline:[]}},computed:{step(){return this.timeline.length>0?this.timeline[this.ix]:{}},done(){return this.timeline.length-1===this.ix},started(){return this.timeline.length>0}},mounted(){this.generateRandomMatrices()},created(){typeof window>"u"||window.addEventListener("keydown",o=>{o.key===" "?this.incStep():o.key==="Escape"&&(this.ix=0)})},methods:{randomMatrix(o,t){const c=[];for(let h=0;hMath.round(Math.random()*10)));return c},generateRandomMatrices(){this.a=this.randomMatrix(3,4),this.b=this.randomMatrix(4,3),this.timeline=[],this.ix=0},matmul(o,t){this.timeline=[];const c=[];for(let i=0;i{t[e].forEach((a,r)=>{const x=[];o[e].forEach((d,u)=>{x.push(`${o[r][u]}*${t[u][e]}`),c[r][e]+=o[r][u]*t[u][e],h[e*o.length+r]=`${x.join(" + ")} = ${c[r][e]}`;const w=JSON.parse(JSON.stringify(c));this.timeline.push({row_a:r,col_a:u,row_b:u,col_b:e,row_c:r,col_c:e,cur_a:o[r][u],cur_b:t[u][e],cur_c:c[r][e],c:w,all_steps:h.slice()})})})})},incStep(){this.ix!==this.timeline.length-1&&(this.ix+=1)},decStep(){this.ix>0&&(this.ix-=1)}}},k={class:"m-4 space-y-4"},M={class:"m-2 inline-flex space-x-2"},S=["disabled"],C=["disabled"],B={class:"flex flex-wrap"},E={class:"w-34 m-2 border border-gray-400 sm:w-min"},A={class:"flex items-center justify-center p-4"},N={class:"text-center"},R={class:"w-34 m-2 border border-gray-400 sm:w-min"},V={class:"flex items-center justify-center p-4"},D={class:"text-center"},F={class:"flex items-center justify-center p-4"},J={class:"text-center"},L={key:1,class:"m-2 w-72 border border-gray-400 bg-gray-100 p-2"},O={class:"font-mono"},z={key:0,class:"italic text-green-600"};function P(o,t,c,h,i,e){return n(),l("div",k,[t[10]||(t[10]=s("div",{class:"m-2 w-4/5"},[y(" Walk through the steps of matrix multiplication with randomly generated matrices. Press the "),s("em",null,"Spacebar"),y(" or click the "),s("em",{class:"bg-blue-200 text-blue-800"},"buttons"),y(" to iterate through the steps. ")],-1)),s("div",M,[s("button",{class:p(["bg-blue-200 px-4 py-2 font-bold text-blue-800 hover:bg-blue-800 hover:text-blue-200",{"cursor-not-allowed opacity-25":!e.started}]),disabled:!e.started,onClick:t[0]||(t[0]=(...a)=>e.decStep&&e.decStep(...a))},t[4]||(t[4]=[s("svg",{xmlns:"http://www.w3.org/2000/svg",class:"h-4 w-4",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor"},[s("path",{"stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M11 19l-7-7 7-7m8 14l-7-7 7-7"})],-1)]),10,S),e.done?g("",!0):(n(),l("button",{key:0,class:"bg-blue-200 px-4 py-2 font-bold text-blue-800 hover:bg-blue-800 hover:text-blue-200",onClick:t[1]||(t[1]=a=>e.matmul(i.a,i.b))}," Start ")),e.done?(n(),l("button",{key:1,class:"bg-orange-200 px-4 py-2 font-bold text-orange-800 hover:bg-blue-800 hover:text-blue-200",onClick:t[2]||(t[2]=(...a)=>e.generateRandomMatrices&&e.generateRandomMatrices(...a))}," Again! ")):g("",!0),s("button",{class:p(["bg-blue-200 px-4 py-2 font-bold text-blue-800 hover:bg-blue-800 hover:text-blue-200",{"cursor-not-allowed opacity-25":!e.started}]),disabled:!e.started,onClick:t[3]||(t[3]=(...a)=>e.incStep&&e.incStep(...a))},t[5]||(t[5]=[s("svg",{xmlns:"http://www.w3.org/2000/svg",class:"h-4 w-4",fill:"none",viewBox:"0 0 24 24",stroke:"currentColor"},[s("path",{"stroke-linecap":"round","stroke-linejoin":"round","stroke-width":"2",d:"M13 5l7 7-7 7M5 5l7 7-7 7"})],-1)]),10,C)]),s("div",B,[s("div",E,[t[6]||(t[6]=s("div",{class:"bg-gray-200 p-4 text-center font-bold"},"Matrix A",-1)),s("div",A,[s("table",N,[(n(!0),l(b,null,m(i.a,(a,r)=>(n(),l("tr",{key:r},[(n(!0),l(b,null,m(i.a[r],(x,d)=>(n(),l("td",{key:d,class:p([{"text-green-600":e.step&&r==e.step.row_a&&d==e.step.col_a,"bg-orange-100":e.step&&r==e.step.row_a},"p-2"])},f(i.a[r][d]),3))),128))]))),128))])])]),s("div",R,[t[7]||(t[7]=s("div",{class:"bg-gray-200 p-4 text-center font-bold"},"Matrix B",-1)),s("div",V,[s("table",D,[(n(!0),l(b,null,m(i.b,(a,r)=>(n(),l("tr",{key:r},[(n(!0),l(b,null,m(i.b[r],(x,d)=>(n(),l("td",{key:d,class:p([{"text-green-600":e.step&&r==e.step.row_b&&d==e.step.col_b,"bg-orange-100":e.step&&d==e.step.col_b},"p-2"])},f(i.b[r][d]),3))),128))]))),128))])])]),e.started?(n(),l("div",{key:0,class:p(["m-2 w-72 border border-gray-400 sm:w-min",{"bg-green-200":e.done}])},[t[8]||(t[8]=s("div",{class:"bg-gray-200 p-4 text-center font-bold"},"Result",-1)),s("div",F,[s("table",J,[(n(!0),l(b,null,m(e.step.c,(a,r)=>(n(),l("tr",{key:r},[(n(!0),l(b,null,m(e.step.c[r],(x,d)=>(n(),l("td",{key:d,class:"p-2"},f(e.step.c[r][d]),1))),128))]))),128))])])],2)):g("",!0),e.started?(n(),l("div",L,[t[9]||(t[9]=s("div",{class:"text-center font-bold"},"Steps",-1)),s("div",O,[(n(!0),l(b,null,m(e.step.all_steps,(a,r)=>(n(),l("div",{key:r},f(a),1))),128)),e.done?(n(),l("div",z,"Done!")):g("",!0)])])):g("",!0)]),g("",!0)])}const W=_(v,[["render",P]]);export{W as default}; diff --git a/_nuxt/Dn7xXnlv.js b/_nuxt/Dn7xXnlv.js new file mode 100644 index 00000000..348edf31 --- /dev/null +++ b/_nuxt/Dn7xXnlv.js @@ -0,0 +1 @@ +import{a as o}from"./Bbro_qAu.js";import{m as c,c as n,i as t,o as a}from"./Bb-NKJJJ.js";const r={};function s(_,m){const e=o;return a(),n("div",null,[t(e,{showImages:""})])}const d=c(r,[["render",s]]);export{d as default}; diff --git a/_nuxt/M6Zd1ArL.js b/_nuxt/Dp7uR_kE.js similarity index 58% rename from _nuxt/M6Zd1ArL.js rename to _nuxt/Dp7uR_kE.js index 273fb610..f170a0f0 100644 --- a/_nuxt/M6Zd1ArL.js +++ b/_nuxt/Dp7uR_kE.js @@ -1 +1 @@ -import{d as n,L as e}from"./BGikLauJ.js";const t=n({name:"DocumentDrivenNotFound",render(){return e("div","Document not found")}});export{t as default}; +import{d as n,L as e}from"./Bb-NKJJJ.js";const t=n({name:"DocumentDrivenNotFound",render(){return e("div","Document not found")}});export{t as default}; diff --git a/_nuxt/DukX5DGJ.js b/_nuxt/DrabSix5.js similarity index 84% rename from _nuxt/DukX5DGJ.js rename to _nuxt/DrabSix5.js index bbf73d84..1ce913cc 100644 --- a/_nuxt/DukX5DGJ.js +++ b/_nuxt/DrabSix5.js @@ -1 +1 @@ -import{_ as f}from"./yGJPFCFo.js";import{d as l,v as s,J as d,L as c}from"./BGikLauJ.js";import"./C-v3KzvZ.js";import"./Dnd51l0P.js";import"./DJ6fbLCw.js";import"./IShi1APO.js";const $=l({name:"ContentRenderer",props:{value:{type:Object,required:!1,default:()=>({})},excerpt:{type:Boolean,default:!1},tag:{type:String,default:"div"}},setup(t){s(()=>t.excerpt,n=>{var e,i,a;n&&!((e=t.value)!=null&&e.excerpt)&&(console.warn(`No excerpt found for document content/${(i=t==null?void 0:t.value)==null?void 0:i._path}.${(a=t==null?void 0:t.value)==null?void 0:a._extension}!`),console.warn("Make sure to use in your content if you want to use excerpt feature."))},{immediate:!0})},render(t){var u,o;const n=d(),{value:e,excerpt:i,tag:a}=t,r=i?e==null?void 0:e.excerpt:e==null?void 0:e.body;return!((u=r==null?void 0:r.children)!=null&&u.length)&&(n!=null&&n.empty)?n.empty({value:e,excerpt:i,tag:a,...this.$attrs}):n!=null&&n.default?n.default({value:e,excerpt:i,tag:a,...this.$attrs}):(r==null?void 0:r.type)==="root"&&((o=r==null?void 0:r.children)!=null&&o.length)?c(f,{value:e,excerpt:i,tag:a,...this.$attrs}):c("pre",null,JSON.stringify({message:"You should use slots with ",value:e,excerpt:i,tag:a},null,2))}});export{$ as default}; +import{_ as f}from"./CS06-0Od.js";import{d as l,v as s,J as d,L as c}from"./Bb-NKJJJ.js";import"./C-v3KzvZ.js";import"./Dnd51l0P.js";import"./mjvvwssR.js";import"./IShi1APO.js";const $=l({name:"ContentRenderer",props:{value:{type:Object,required:!1,default:()=>({})},excerpt:{type:Boolean,default:!1},tag:{type:String,default:"div"}},setup(t){s(()=>t.excerpt,n=>{var e,i,a;n&&!((e=t.value)!=null&&e.excerpt)&&(console.warn(`No excerpt found for document content/${(i=t==null?void 0:t.value)==null?void 0:i._path}.${(a=t==null?void 0:t.value)==null?void 0:a._extension}!`),console.warn("Make sure to use in your content if you want to use excerpt feature."))},{immediate:!0})},render(t){var u,o;const n=d(),{value:e,excerpt:i,tag:a}=t,r=i?e==null?void 0:e.excerpt:e==null?void 0:e.body;return!((u=r==null?void 0:r.children)!=null&&u.length)&&(n!=null&&n.empty)?n.empty({value:e,excerpt:i,tag:a,...this.$attrs}):n!=null&&n.default?n.default({value:e,excerpt:i,tag:a,...this.$attrs}):(r==null?void 0:r.type)==="root"&&((o=r==null?void 0:r.children)!=null&&o.length)?c(f,{value:e,excerpt:i,tag:a,...this.$attrs}):c("pre",null,JSON.stringify({message:"You should use slots with ",value:e,excerpt:i,tag:a},null,2))}});export{$ as default}; diff --git a/_nuxt/BEt4369s.js b/_nuxt/DvMJuzSW.js similarity index 64% rename from _nuxt/BEt4369s.js rename to _nuxt/DvMJuzSW.js index 1e08b373..8daea2b7 100644 --- a/_nuxt/BEt4369s.js +++ b/_nuxt/DvMJuzSW.js @@ -1 +1 @@ -import{m as o,o as r,c as s,a8 as t}from"./BGikLauJ.js";const c={};function n(e,a){return r(),s("p",null,[t(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; +import{m as o,o as r,c as s,a8 as t}from"./Bb-NKJJJ.js";const c={};function n(e,a){return r(),s("p",null,[t(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; diff --git a/_nuxt/CJwA9VmX.js b/_nuxt/DvojTKw_.js similarity index 65% rename from _nuxt/CJwA9VmX.js rename to _nuxt/DvojTKw_.js index bdd13b2c..822c3efe 100644 --- a/_nuxt/CJwA9VmX.js +++ b/_nuxt/DvojTKw_.js @@ -1 +1 @@ -import{m as o,o as r,c as t,a8 as a}from"./BGikLauJ.js";const s={};function c(e,n){return r(),t("thead",null,[a(e.$slots,"default")])}const d=o(s,[["render",c]]);export{d as default}; +import{m as o,o as r,c as t,a8 as a}from"./Bb-NKJJJ.js";const s={};function c(e,n){return r(),t("thead",null,[a(e.$slots,"default")])}const d=o(s,[["render",c]]);export{d as default}; diff --git a/_nuxt/BUYHwnpH.js b/_nuxt/Dz8Mq6rn.js similarity index 88% rename from _nuxt/BUYHwnpH.js rename to _nuxt/Dz8Mq6rn.js index 3140e794..69e8a8e4 100644 --- a/_nuxt/BUYHwnpH.js +++ b/_nuxt/Dz8Mq6rn.js @@ -1 +1 @@ -import{d as p,I as f,k as i,o as s,c as n,a as u,a8 as r}from"./BGikLauJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH6",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=i(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h6))});return(e,k)=>(s(),n("h6",{id:t.id},[t.id&&u(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; +import{d as p,I as f,k as i,o as s,c as n,a as u,a8 as r}from"./Bb-NKJJJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH6",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=i(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h6))});return(e,k)=>(s(),n("h6",{id:t.id},[t.id&&u(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; diff --git a/_nuxt/BnWdfnc6.js b/_nuxt/DzMxnq8n.js similarity index 77% rename from _nuxt/BnWdfnc6.js rename to _nuxt/DzMxnq8n.js index 0fb1c429..b455f0f7 100644 --- a/_nuxt/BnWdfnc6.js +++ b/_nuxt/DzMxnq8n.js @@ -1 +1 @@ -import{d as n,L as e}from"./BGikLauJ.js";const r=n({name:"DocumentDrivenEmpty",props:{value:{type:Object,required:!0}},render({value:t}){return e("div",void 0,[e("p","Document is empty"),e("p",`Add content to it by opening ${t._source}/${t._file} file.`)])}});export{r as default}; +import{d as n,L as e}from"./Bb-NKJJJ.js";const r=n({name:"DocumentDrivenEmpty",props:{value:{type:Object,required:!0}},render({value:t}){return e("div",void 0,[e("p","Document is empty"),e("p",`Add content to it by opening ${t._source}/${t._file} file.`)])}});export{r as default}; diff --git a/_nuxt/jcdaHq-b.js b/_nuxt/F1DGrIW6.js similarity index 86% rename from _nuxt/jcdaHq-b.js rename to _nuxt/F1DGrIW6.js index 022e936b..50e42d8c 100644 --- a/_nuxt/jcdaHq-b.js +++ b/_nuxt/F1DGrIW6.js @@ -1 +1 @@ -import{m as n,o as r,c,b as e,i as s,a8 as d,ab as _}from"./BGikLauJ.js";const i={},f={class:"pin-b w-full p-4 text-xs font-bold text-white"};function u(t,o){return r(),c("footer",f,o[0]||(o[0]=[e("div",{class:"flex justify-center space-x-2"},[e("a",{class:"hover:cursor-pointer hover:text-blue-200",href:"https://github.com/cmpadden/cmpadden.github.io"}," © Colton Padden"),e("span",null,"•"),e("a",{class:"hover:cursor-pointer hover:text-blue-200",href:"/sitemap.xml"},"Sitemap"),e("span",null,"•"),e("a",{class:"hover:cursor-pointer hover:text-blue-200",href:"/atom"},"RSS")],-1)]))}const p=n(i,[["render",u]]),m={},h={class:"bg-orange-500"},x={class:"rounded-3xl border-[10px] border-orange-500"},b={class:"flex min-h-screen flex-col rounded-3xl bg-background font-display"},v={class:"flex-1"};function g(t,o){const a=_,l=p;return r(),c("div",h,[e("div",x,[e("main",b,[s(a),e("main",v,[d(t.$slots,"default")]),s(l)])])])}const S=n(m,[["render",g]]);export{S as default}; +import{m as n,o as r,c,b as e,i as s,a8 as d,ab as _}from"./Bb-NKJJJ.js";const i={},f={class:"pin-b w-full p-4 text-xs font-bold text-white"};function u(t,o){return r(),c("footer",f,o[0]||(o[0]=[e("div",{class:"flex justify-center space-x-2"},[e("a",{class:"hover:cursor-pointer hover:text-blue-200",href:"https://github.com/cmpadden/cmpadden.github.io"}," © Colton Padden"),e("span",null,"•"),e("a",{class:"hover:cursor-pointer hover:text-blue-200",href:"/sitemap.xml"},"Sitemap"),e("span",null,"•"),e("a",{class:"hover:cursor-pointer hover:text-blue-200",href:"/atom"},"RSS")],-1)]))}const p=n(i,[["render",u]]),m={},h={class:"bg-orange-500"},x={class:"rounded-3xl border-[10px] border-orange-500"},b={class:"flex min-h-screen flex-col rounded-3xl bg-background font-display"},v={class:"flex-1"};function g(t,o){const a=_,l=p;return r(),c("div",h,[e("div",x,[e("main",b,[s(a),e("main",v,[d(t.$slots,"default")]),s(l)])])])}const S=n(m,[["render",g]]);export{S as default}; diff --git a/_nuxt/Do41mdVm.js b/_nuxt/I6a6eUEM.js similarity index 88% rename from _nuxt/Do41mdVm.js rename to _nuxt/I6a6eUEM.js index e75a3f80..c0943caa 100644 --- a/_nuxt/Do41mdVm.js +++ b/_nuxt/I6a6eUEM.js @@ -1 +1 @@ -import{d as N,u as R,w as g,n as B,a as o,o as r,c,b as e,F as f,r as h,e as v,f as m,g as b,t as _,h as y,i as w,_ as T}from"./BGikLauJ.js";import{_ as V}from"./yGJPFCFo.js";import $ from"./DukX5DGJ.js";import{u as D}from"./CgY3mHZ8.js";import{q as E}from"./UWyBikFq.js";import"./C-v3KzvZ.js";import"./Dnd51l0P.js";import"./DJ6fbLCw.js";import"./IShi1APO.js";const q={key:0,class:"container mx-auto"},A={class:"mb-10 bg-gray-200 p-10 shadow-lg"},F={class:"space-y-6"},L={class:"flex space-x-2"},S={class:"flex flex-col space-y-2 md:flex md:flex-row md:space-x-2 md:space-y-0"},I={class:"flex flex-col space-y-2 md:flex md:flex-row md:space-x-2 md:space-y-0"},M={class:"flex"},O=["src"],j={class:"text-xl font-bold text-gray-700 md:text-3xl"},z={class:"prose max-w-none prose-a:font-bold prose-a:no-underline hover:prose-a:text-orange-500 prose-pre:bg-white prose-pre:text-black"},Z=N({__name:"[...slug]",async setup(G){let s,l;const d=R(),{data:t}=([s,l]=g(()=>D("page-data",()=>E(d.path).findOne())),s=await s,l(),s),u={};if(t.value===null||d.path in u){const i=u[d.path]||"/";console.log(i),[s,l]=g(()=>B({path:i})),await s,l()}return(i,a)=>{const x=T,k=V,C=$;return o(t)?(r(),c("div",q,[e("article",A,[e("div",F,[e("div",L,[e("div",null,[a[0]||(a[0]=e("div",{class:"text-xs uppercase text-gray-400"},"Category",-1)),e("div",S,[(r(!0),c(f,null,h(o(t).categories,(n,p)=>(r(),v(x,{class:"w-min rounded-md bg-background px-2 text-sm font-bold text-white hover:cursor-pointer hover:bg-orange-500",key:p,to:`/articles?category=${n}`},{default:m(()=>[b(_(n),1)]),_:2},1032,["to"]))),128))])]),a[2]||(a[2]=e("div",{class:"border-r border-gray-400"},null,-1)),e("div",null,[a[1]||(a[1]=e("div",{class:"text-xs uppercase text-gray-400"},"Tags",-1)),e("div",I,[(r(!0),c(f,null,h(o(t).tags,(n,p)=>(r(),v(x,{class:"w-min rounded-md bg-background px-2 text-sm font-bold text-white hover:cursor-pointer hover:bg-orange-500",key:p,to:`/articles?tag=${n}`},{default:m(()=>[b(_(n),1)]),_:2},1032,["to"]))),128))])])]),e("div",M,[o(t).cover_image?(r(),c("img",{key:0,class:"mr-4 h-16 border-2 border-black",src:o(t).cover_image},null,8,O)):y("",!0),e("h1",j,_(o(t).title),1)])]),e("article",z,[w(C,null,{default:m(()=>[w(k,{value:o(t)},null,8,["value"])]),_:1})])])])):y("",!0)}}});export{Z as default}; +import{d as N,u as R,w as g,n as B,a as o,o as r,c,b as e,F as f,r as h,e as v,f as m,g as b,t as _,h as y,i as w,_ as T}from"./Bb-NKJJJ.js";import{_ as V}from"./CS06-0Od.js";import $ from"./DrabSix5.js";import{u as D}from"./2mhhWi5_.js";import{q as E}from"./jaxqT3Hp.js";import"./C-v3KzvZ.js";import"./Dnd51l0P.js";import"./mjvvwssR.js";import"./IShi1APO.js";const q={key:0,class:"container mx-auto"},A={class:"mb-10 bg-gray-200 p-10 shadow-lg"},F={class:"space-y-6"},L={class:"flex space-x-2"},S={class:"flex flex-col space-y-2 md:flex md:flex-row md:space-x-2 md:space-y-0"},I={class:"flex flex-col space-y-2 md:flex md:flex-row md:space-x-2 md:space-y-0"},M={class:"flex"},O=["src"],j={class:"text-xl font-bold text-gray-700 md:text-3xl"},z={class:"prose max-w-none prose-a:font-bold prose-a:no-underline hover:prose-a:text-orange-500 prose-pre:bg-white prose-pre:text-black"},Z=N({__name:"[...slug]",async setup(G){let s,l;const d=R(),{data:t}=([s,l]=g(()=>D("page-data",()=>E(d.path).findOne())),s=await s,l(),s),u={};if(t.value===null||d.path in u){const i=u[d.path]||"/";console.log(i),[s,l]=g(()=>B({path:i})),await s,l()}return(i,a)=>{const x=T,k=V,C=$;return o(t)?(r(),c("div",q,[e("article",A,[e("div",F,[e("div",L,[e("div",null,[a[0]||(a[0]=e("div",{class:"text-xs uppercase text-gray-400"},"Category",-1)),e("div",S,[(r(!0),c(f,null,h(o(t).categories,(n,p)=>(r(),v(x,{class:"w-min rounded-md bg-background px-2 text-sm font-bold text-white hover:cursor-pointer hover:bg-orange-500",key:p,to:`/articles?category=${n}`},{default:m(()=>[b(_(n),1)]),_:2},1032,["to"]))),128))])]),a[2]||(a[2]=e("div",{class:"border-r border-gray-400"},null,-1)),e("div",null,[a[1]||(a[1]=e("div",{class:"text-xs uppercase text-gray-400"},"Tags",-1)),e("div",I,[(r(!0),c(f,null,h(o(t).tags,(n,p)=>(r(),v(x,{class:"w-min rounded-md bg-background px-2 text-sm font-bold text-white hover:cursor-pointer hover:bg-orange-500",key:p,to:`/articles?tag=${n}`},{default:m(()=>[b(_(n),1)]),_:2},1032,["to"]))),128))])])]),e("div",M,[o(t).cover_image?(r(),c("img",{key:0,class:"mr-4 h-16 border-2 border-black",src:o(t).cover_image},null,8,O)):y("",!0),e("h1",j,_(o(t).title),1)])]),e("article",z,[w(C,null,{default:m(()=>[w(k,{value:o(t)},null,8,["value"])]),_:1})])])])):y("",!0)}}});export{Z as default}; diff --git a/_nuxt/KhAE7j4n.js b/_nuxt/KhAE7j4n.js deleted file mode 100644 index ee480e6e..00000000 --- a/_nuxt/KhAE7j4n.js +++ /dev/null @@ -1 +0,0 @@ -import{_ as o}from"./yGJPFCFo.js";import"./BGikLauJ.js";import"./C-v3KzvZ.js";import"./Dnd51l0P.js";import"./DJ6fbLCw.js";import"./IShi1APO.js";export{o as default}; diff --git a/_nuxt/D3_S2KpX.js b/_nuxt/R23NweeK.js similarity index 95% rename from _nuxt/D3_S2KpX.js rename to _nuxt/R23NweeK.js index 3586bd23..466cc80e 100644 --- a/_nuxt/D3_S2KpX.js +++ b/_nuxt/R23NweeK.js @@ -1 +1 @@ -import{m as u,c as v,b as c,o as h}from"./BGikLauJ.js";const y={data(){return{}},mounted(){document.getElementById("canvas").textContent="";const o=["#FFBA08","#FAA307","#F48C06","#E85D04","#DC2F02","#D00000","#9D0208","#6A040F","#370617","#03071E"],a=e=>{const m=document.getElementById("canvas"),r=m.clientWidth,n=m.clientHeight,f=[{color:o[1],min:0,max:.2,yoff:2},{color:o[2],min:.2,max:.4,yoff:2},{color:o[3],min:.4,max:.6,yoff:2},{color:o[4],min:.6,max:.8,yoff:2},{color:o[5],min:.8,max:1,yoff:2}],l=t=>{e.fill(t.color),e.beginShape();let s=1;for(let i=0;i<=r;i+=10){const x=e.map(e.noise(s,t.yoff),0,1,t.min*n,t.max*n);e.curveVertex(i,x),s+=e.random(.5)}t.yoff+=e.random(-.5,.5),e.curveVertex(r,n),e.curveVertex(0,n),e.endShape(e.CLOSE)},d=()=>{e.clear(),e.background(o[0]),f.forEach(t=>l(t))};e.setup=()=>{e.createCanvas(r,n),e.frameRate(5),e.smooth(),e.stroke(255),e.strokeWeight(3),d(),e.noLoop()},e.mousePressed=()=>{d()}};new this.$p5(a,"canvas")}},p={class:"select-none bg-gradient-to-b from-green-800 to-gray-800"};function g(o,a,e,m,r,n){return h(),v("div",p,a[0]||(a[0]=[c("div",{class:"m-6 grid h-2/3 place-items-center"},[c("div",{class:"font-mono text-white"},[c("div",{id:"canvas",class:"mb-2 h-96 border-2 border-white"}),c("div",{class:"my-2 w-48 text-center md:w-full"}," Click or tap anywhere to redraw the waves! ")])],-1)]))}const E=u(y,[["render",g]]);export{E as default}; +import{m as u,c as v,b as c,o as h}from"./Bb-NKJJJ.js";const y={data(){return{}},mounted(){document.getElementById("canvas").textContent="";const o=["#FFBA08","#FAA307","#F48C06","#E85D04","#DC2F02","#D00000","#9D0208","#6A040F","#370617","#03071E"],a=e=>{const m=document.getElementById("canvas"),r=m.clientWidth,n=m.clientHeight,f=[{color:o[1],min:0,max:.2,yoff:2},{color:o[2],min:.2,max:.4,yoff:2},{color:o[3],min:.4,max:.6,yoff:2},{color:o[4],min:.6,max:.8,yoff:2},{color:o[5],min:.8,max:1,yoff:2}],l=t=>{e.fill(t.color),e.beginShape();let s=1;for(let i=0;i<=r;i+=10){const x=e.map(e.noise(s,t.yoff),0,1,t.min*n,t.max*n);e.curveVertex(i,x),s+=e.random(.5)}t.yoff+=e.random(-.5,.5),e.curveVertex(r,n),e.curveVertex(0,n),e.endShape(e.CLOSE)},d=()=>{e.clear(),e.background(o[0]),f.forEach(t=>l(t))};e.setup=()=>{e.createCanvas(r,n),e.frameRate(5),e.smooth(),e.stroke(255),e.strokeWeight(3),d(),e.noLoop()},e.mousePressed=()=>{d()}};new this.$p5(a,"canvas")}},p={class:"select-none bg-gradient-to-b from-green-800 to-gray-800"};function g(o,a,e,m,r,n){return h(),v("div",p,a[0]||(a[0]=[c("div",{class:"m-6 grid h-2/3 place-items-center"},[c("div",{class:"font-mono text-white"},[c("div",{id:"canvas",class:"mb-2 h-96 border-2 border-white"}),c("div",{class:"my-2 w-48 text-center md:w-full"}," Click or tap anywhere to redraw the waves! ")])],-1)]))}const E=u(y,[["render",g]]);export{E as default}; diff --git a/_nuxt/Do_OiPaD.js b/_nuxt/UzigjR5s.js similarity index 78% rename from _nuxt/Do_OiPaD.js rename to _nuxt/UzigjR5s.js index 56cd17f0..dabd4e2a 100644 --- a/_nuxt/Do_OiPaD.js +++ b/_nuxt/UzigjR5s.js @@ -1 +1 @@ -import n from"./AY-jWpd0.js";import{d as c,J as l,L as u}from"./BGikLauJ.js";import"./IShi1APO.js";import"./CgY3mHZ8.js";import"./UWyBikFq.js";import"./DJ6fbLCw.js";const p=(r,t)=>u("pre",null,JSON.stringify({message:"You should use slots with ",slot:r,data:t},null,2)),h=c({name:"ContentList",props:{path:{type:String,required:!1,default:void 0},query:{type:Object,required:!1,default:void 0}},render(r){const t=l(),{path:f,query:a}=r,m={...a||{},path:f||(a==null?void 0:a.path)||"/"};return u(n,m,{default:t!=null&&t.default?({data:e,refresh:o,isPartial:d})=>t.default({list:e,refresh:o,isPartial:d,...this.$attrs}):e=>p("default",e.data),empty:e=>t!=null&&t.empty?t.empty(e):p("default",e==null?void 0:e.data),"not-found":e=>{var o;return t!=null&&t["not-found"]?(o=t==null?void 0:t["not-found"])==null?void 0:o.call(t,e):p("not-found",e==null?void 0:e.data)}})}}),i=h;export{i as default}; +import n from"./cufizuWq.js";import{d as c,J as l,L as u}from"./Bb-NKJJJ.js";import"./IShi1APO.js";import"./2mhhWi5_.js";import"./jaxqT3Hp.js";import"./mjvvwssR.js";const p=(r,t)=>u("pre",null,JSON.stringify({message:"You should use slots with ",slot:r,data:t},null,2)),h=c({name:"ContentList",props:{path:{type:String,required:!1,default:void 0},query:{type:Object,required:!1,default:void 0}},render(r){const t=l(),{path:f,query:a}=r,m={...a||{},path:f||(a==null?void 0:a.path)||"/"};return u(n,m,{default:t!=null&&t.default?({data:e,refresh:o,isPartial:d})=>t.default({list:e,refresh:o,isPartial:d,...this.$attrs}):e=>p("default",e.data),empty:e=>t!=null&&t.empty?t.empty(e):p("default",e==null?void 0:e.data),"not-found":e=>{var o;return t!=null&&t["not-found"]?(o=t==null?void 0:t["not-found"])==null?void 0:o.call(t,e):p("not-found",e==null?void 0:e.data)}})}}),i=h;export{i as default}; diff --git a/_nuxt/CQCYFxb2.js b/_nuxt/ZRsJ-Dry.js similarity index 88% rename from _nuxt/CQCYFxb2.js rename to _nuxt/ZRsJ-Dry.js index 44f8e3d5..ae3f39d6 100644 --- a/_nuxt/CQCYFxb2.js +++ b/_nuxt/ZRsJ-Dry.js @@ -1 +1 @@ -import{d as p,I as f,k as i,o as s,c as n,a as u,a8 as r}from"./BGikLauJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH5",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=i(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h5))});return(e,k)=>(s(),n("h5",{id:t.id},[t.id&&u(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; +import{d as p,I as f,k as i,o as s,c as n,a as u,a8 as r}from"./Bb-NKJJJ.js";const l=["id"],d=["href"],_=p({__name:"ProseH5",props:{id:{}},setup(c){const t=c,{headings:o}=f().public.mdc,a=i(()=>{var e;return t.id&&(typeof(o==null?void 0:o.anchorLinks)=="boolean"&&(o==null?void 0:o.anchorLinks)===!0||typeof(o==null?void 0:o.anchorLinks)=="object"&&((e=o==null?void 0:o.anchorLinks)==null?void 0:e.h5))});return(e,k)=>(s(),n("h5",{id:t.id},[t.id&&u(a)?(s(),n("a",{key:0,href:`#${t.id}`},[r(e.$slots,"default")],8,d)):r(e.$slots,"default",{key:1})],8,l))}});export{_ as default}; diff --git a/_nuxt/DrbA_f4z.js b/_nuxt/ZcTbvTx_.js similarity index 99% rename from _nuxt/DrbA_f4z.js rename to _nuxt/ZcTbvTx_.js index afca3d40..e4fe51a8 100644 --- a/_nuxt/DrbA_f4z.js +++ b/_nuxt/ZcTbvTx_.js @@ -1 +1 @@ -import{m as v,c as r,g as l,b as e,l as b,t as d,F as c,r as u,i as _,f as y,T as M,o as n,h as g}from"./BGikLauJ.js";var p={};Object.defineProperty(p,"__esModule",{value:!0});var h=p.identify=void 0;const O=["C","C# / Db","D","D# / Eb","E","F","F# / Gb","G","G# / Ab","A","A# / Bb","B"],x={"0 1":{name:"No 3 Major 7",rootOffset:1},"0 2":{name:"No 3 7",rootOffset:2},"0 3":{name:"Minor",rootOffset:0},"0 4":{name:"",rootOffset:0},"0 5":{name:"5",rootOffset:5},"0 6":{name:"7",rootOffset:2},"0 7":{name:"5",rootOffset:0},"0 8":{name:"",rootOffset:8},"0 9":{name:"Minor",rootOffset:9},"0 10":{name:"No 3 7",rootOffset:0},"0 11":{name:"No 3 Major 7",rootOffset:0},"0 2 7":{name:"Suspended 2",rootOffset:0},"0 3 6 10":{name:"Half-diminished 7",rootOffset:0},"0 3 6 9":{name:"Diminished 7",rootOffset:0},"0 3 6":{name:"Diminished",rootOffset:0},"0 3 7 10":{name:"Minor 7",rootOffset:0},"0 3 7 11":{name:"Minor-major 7",rootOffset:0},"0 3 7 9":{name:"Minor 6",rootOffset:0},"0 3 7":{name:"Minor",rootOffset:0},"0 4 10":{name:"7",rootOffset:0},"0 4 11":{name:"Major 7",rootOffset:0},"0 4 5 7":{name:"Add 11",rootOffset:0},"0 4 6":{name:"Flat 5",rootOffset:0},"0 4 7 10":{name:"Dominant 7",rootOffset:0},"0 4 7 11":{name:"Major 7",rootOffset:0},"0 4 7 9":{name:"Major 6",rootOffset:0},"0 4 7":{name:"",rootOffset:0},"0 4 8 10":{name:"Augmented 7",rootOffset:0},"0 4 8":{name:"Augmented",rootOffset:0},"0 5 7":{name:"Suspended 4",rootOffset:0},"0 5 8":{name:"Minor",rootOffset:5},"0 5 9":{name:"",rootOffset:5},"0 4 9":{name:"Minor",rootOffset:9},"0 3 8":{name:"",rootOffset:8},"0 1 5":{name:"Major 7",rootOffset:1},"0 2 6":{name:"Major 7",rootOffset:2},"0 4 5":{name:"5 Major 7",rootOffset:5},"0 1 2":{name:"No 3 7 Major 7",rootOffset:2},"0 1 3":{name:"Minor Add Flat 9",rootOffset:0},"0 1 6":{name:"5 Sharp 11",rootOffset:6},"0 1 7":{name:"5 Add Flat 9",rootOffset:0},"0 1 8":{name:"5 Major 7",rootOffset:1},"0 1 9":{name:"Sharp 9",rootOffset:9},"0 1 10":{name:"Minor Add 9",rootOffset:10},"0 1 11":{name:"No 3 7 Major 7",rootOffset:1},"0 2 10":{name:"Add 9",rootOffset:10},"0 2 11":{name:"Minor Add Flat 9",rootOffset:11},"0 2 3":{name:"Minor Add 9",rootOffset:0},"0 2 4":{name:"Add 9",rootOffset:0},"0 2 5":{name:"Minor 7",rootOffset:2},"0 2 8":{name:"Flat 5",rootOffset:8},"0 2 9":{name:"5 7",rootOffset:2},"0 3 10":{name:"Minor 7",rootOffset:0},"0 3 4":{name:"Sharp 9",rootOffset:0},"0 3 5":{name:"5 7",rootOffset:5},"0 3 9":{name:"Minor 6",rootOffset:0},"0 5 10":{name:"Suspended 2",rootOffset:10},"0 5 11":{name:"5 Sharp 11",rootOffset:5},"0 5 6":{name:"5 Add Flat 9",rootOffset:5},"0 6 10":{name:"Flat 5",rootOffset:6},"0 6 7":{name:"5 Sharp 11",rootOffset:0},"0 6 8":{name:"7",rootOffset:8},"0 6 9":{name:"Minor 6",rootOffset:9},"0 7 10":{name:"5 7",rootOffset:0},"0 7 11":{name:"5 Major 7",rootOffset:0},"0 7 8":{name:"Major 7",rootOffset:8},"0 7 9":{name:"Minor 7",rootOffset:9},"0 8 10":{name:"Add 9",rootOffset:8},"0 8 11":{name:"Sharp 9",rootOffset:8},"0 9 10":{name:"Minor Add Flat 9",rootOffset:9},"0 9 11":{name:"Minor Add 9",rootOffset:9},"0 10 11":{name:"No 3 7 Major 7",rootOffset:0},"0 1 4":{name:"Sharp 9",rootOffset:9},"0 8 9":{name:"Sharp 9",rootOffset:5},"0 6 11":{name:"7/13",rootOffset:2},"0 3 11":{name:"Sharp 9",rootOffset:8}};function w(i){const o=i.map(t=>typeof t=="number"?t:O.indexOf(t));if(o.some(t=>t<0))throw new Error("Unsupported note letter or number provided");if(new Set(o.map(t=>t%12)).size===1)return{name:O[o[0]%12]};const f=o.sort((t,s)=>t-s)[0]%12,m=[...new Set(o.map(t=>(t-f)%12).sort((t,s)=>t-s))],a=x[m.join(" ")];if(a){const t=(f+a.rootOffset)%12,s=O[t];return{name:a.name!==""?`${s} ${a.name}`:s,interval:m,root:t}}else return{name:void 0}}h=p.identify=w;const A={mounted(){typeof navigator.requestMIDIAccess<"u"&&navigator.requestMIDIAccess().then(i=>{this.midi=i,this.midi.inputs.forEach(o=>{o.onmidimessage=f=>{f.data[0]===144&&(this.activeKeys.set(f.data[1],f.data),this.$forceUpdate()),f.data[0]===128&&(this.activeKeys.delete(f.data[1]),this.$forceUpdate())}})},i=>{console.error(i)})},data(){return{tooltip:!1,midi:void 0,activeKeys:new Map}},computed:{inputs(){if(typeof this.midi<"u")return Array.from(this.midi.inputs.values())},outputs(){if(typeof this.midi<"u")return Array.from(this.midi.outputs.values())}},methods:{orderedNotes(){return Array.from(this.activeKeys.keys()).sort()},chord(){const i=Array.from(this.activeKeys.keys());return h(i).name}}},k={class:"h-2/3 bg-gradient-to-b from-green-900 via-purple-900 to-indigo-900 text-white"},N={key:0,class:"p-4 text-center font-light tracking-wide"},S={key:1},j={class:"flex flex-wrap"},I={class:"absolute bottom-16 right-2"},D={class:"p-3 font-mono text-orange-900"},F={class:"mb-2 font-bold"},E={class:"mb-2"},z={key:0,class:"p-4 text-center italic"},V={key:1},C={class:"flex-1"},B={class:"flex-1"},K={class:"mb-2"},T={key:0,class:"p-4 text-center italic"},P={key:1},U={class:"flex-1"},G={class:"flex-1"},L={class:"absolute bottom-2 right-2"},q={class:"absolute bottom-4 left-4"},H={class:"flex"},R={class:"grid h-screen place-items-center"},W={class:"text-center text-4xl font-semibold tracking-wide"},Y={key:0};function J(i,o,f,m,a,t){return n(),r("div",k,[typeof a.midi>"u"?(n(),r("div",N,o[1]||(o[1]=[l(" Unfortunately, the Web MIDI API is "),e("a",{class:"text-blue-500 underline",href:"https://developer.mozilla.org/en-US/docs/Web/API/MIDIMessageEvent#browser_compatibility"},"not supported",-1),l(" in all browsers... ")]))):(n(),r("div",S,[e("div",j,[e("div",I,[e("div",{class:b([{hidden:!a.tooltip,block:a.tooltip},"z-50 max-w-md break-words rounded-lg border-2 border-green-800 bg-yellow-200 text-sm font-normal leading-normal"])},[e("div",null,[o[5]||(o[5]=e("div",{class:"mb-0 rounded-t-lg border-b border-solid bg-green-600 p-3 font-semibold uppercase text-white opacity-75"}," MIDI Status ",-1)),e("div",D,[e("div",F,[o[2]||(o[2]=l(" Enabled: ")),e("span",null,d(typeof a.midi<"u"?"Yep!":"Nope"),1)]),e("div",E,[o[3]||(o[3]=e("div",{class:"font-bold"},"Inputs:",-1)),t.inputs.length===0?(n(),r("div",z," No input devices detected :( ")):(n(),r("div",V,[(n(!0),r(c,null,u(t.inputs,s=>(n(),r("div",{key:s.id,class:"flex"},[e("div",C,d(s.manufacturer),1),e("div",B,d(s.name),1)]))),128))]))]),e("div",K,[o[4]||(o[4]=e("div",{class:"font-bold"},"Outputs:",-1)),t.outputs.length===0?(n(),r("div",T," No output devices detected :( ")):(n(),r("div",P,[(n(!0),r(c,null,u(t.outputs,s=>(n(),r("div",{key:s.id,class:"flex"},[e("div",U,d(s.manufacturer),1),e("div",G,d(s.name),1)]))),128))]))])])])],2)]),e("div",L,[e("button",{onClick:o[0]||(o[0]=s=>a.tooltip=!a.tooltip),class:"border-3 rounded-lg border-green-600 bg-green-50 px-1 text-green-600 shadow hover:text-green-500 hover:shadow-lg",type:"button"},o[6]||(o[6]=[e("svg",{xmlns:"http://www.w3.org/2000/svg","xmlns:xlink":"http://www.w3.org/1999/xlink",viewBox:"0 0 20 20",class:"h-10 w-10"},[e("g",{fill:"none"},[e("path",{d:"M2.5 4a.5.5 0 0 0-.5.5v11a.5.5 0 0 0 .5.5h15a.5.5 0 0 0 .5-.5v-11a.5.5 0 0 0-.5-.5h-15zm.5 6h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v5H3v-5zm2.75-2.5a.75.75 0 1 1 0-1.5a.75.75 0 0 1 0 1.5zm6.25-1a.5.5 0 0 1 .5-.5h2a.5.5 0 0 1 0 1h-2a.5.5 0 0 1-.5-.5zm-3 .251a.75.75 0 1 1-1.5 0a.75.75 0 0 1 1.5 0z",fill:"currentColor"})])],-1)]))])]),e("div",q,[e("div",H,[(n(!0),r(c,null,u(t.orderedNotes(),s=>(n(),r("div",{key:s,class:"p-4"},d(s),1))),128))])]),e("div",R,[e("div",null,[e("div",W,[_(M,{"enter-active-class":"duration-500 ease-out","enter-class":"opacity-0 transform","enter-to-class":"opacity-100","leave-active-class":"duration-500","leave-class":"opacity-100","leave-to-class":"opacity-0 transform"},{default:y(()=>[a.activeKeys.size>=1?(n(),r("div",Y,d(t.chord()||"?"),1)):g("",!0)]),_:1})])])])]))])}const X=v(A,[["render",J]]);export{X as default}; +import{m as v,c as r,g as l,b as e,l as b,t as d,F as c,r as u,i as _,f as y,T as M,o as n,h as g}from"./Bb-NKJJJ.js";var p={};Object.defineProperty(p,"__esModule",{value:!0});var h=p.identify=void 0;const O=["C","C# / Db","D","D# / Eb","E","F","F# / Gb","G","G# / Ab","A","A# / Bb","B"],x={"0 1":{name:"No 3 Major 7",rootOffset:1},"0 2":{name:"No 3 7",rootOffset:2},"0 3":{name:"Minor",rootOffset:0},"0 4":{name:"",rootOffset:0},"0 5":{name:"5",rootOffset:5},"0 6":{name:"7",rootOffset:2},"0 7":{name:"5",rootOffset:0},"0 8":{name:"",rootOffset:8},"0 9":{name:"Minor",rootOffset:9},"0 10":{name:"No 3 7",rootOffset:0},"0 11":{name:"No 3 Major 7",rootOffset:0},"0 2 7":{name:"Suspended 2",rootOffset:0},"0 3 6 10":{name:"Half-diminished 7",rootOffset:0},"0 3 6 9":{name:"Diminished 7",rootOffset:0},"0 3 6":{name:"Diminished",rootOffset:0},"0 3 7 10":{name:"Minor 7",rootOffset:0},"0 3 7 11":{name:"Minor-major 7",rootOffset:0},"0 3 7 9":{name:"Minor 6",rootOffset:0},"0 3 7":{name:"Minor",rootOffset:0},"0 4 10":{name:"7",rootOffset:0},"0 4 11":{name:"Major 7",rootOffset:0},"0 4 5 7":{name:"Add 11",rootOffset:0},"0 4 6":{name:"Flat 5",rootOffset:0},"0 4 7 10":{name:"Dominant 7",rootOffset:0},"0 4 7 11":{name:"Major 7",rootOffset:0},"0 4 7 9":{name:"Major 6",rootOffset:0},"0 4 7":{name:"",rootOffset:0},"0 4 8 10":{name:"Augmented 7",rootOffset:0},"0 4 8":{name:"Augmented",rootOffset:0},"0 5 7":{name:"Suspended 4",rootOffset:0},"0 5 8":{name:"Minor",rootOffset:5},"0 5 9":{name:"",rootOffset:5},"0 4 9":{name:"Minor",rootOffset:9},"0 3 8":{name:"",rootOffset:8},"0 1 5":{name:"Major 7",rootOffset:1},"0 2 6":{name:"Major 7",rootOffset:2},"0 4 5":{name:"5 Major 7",rootOffset:5},"0 1 2":{name:"No 3 7 Major 7",rootOffset:2},"0 1 3":{name:"Minor Add Flat 9",rootOffset:0},"0 1 6":{name:"5 Sharp 11",rootOffset:6},"0 1 7":{name:"5 Add Flat 9",rootOffset:0},"0 1 8":{name:"5 Major 7",rootOffset:1},"0 1 9":{name:"Sharp 9",rootOffset:9},"0 1 10":{name:"Minor Add 9",rootOffset:10},"0 1 11":{name:"No 3 7 Major 7",rootOffset:1},"0 2 10":{name:"Add 9",rootOffset:10},"0 2 11":{name:"Minor Add Flat 9",rootOffset:11},"0 2 3":{name:"Minor Add 9",rootOffset:0},"0 2 4":{name:"Add 9",rootOffset:0},"0 2 5":{name:"Minor 7",rootOffset:2},"0 2 8":{name:"Flat 5",rootOffset:8},"0 2 9":{name:"5 7",rootOffset:2},"0 3 10":{name:"Minor 7",rootOffset:0},"0 3 4":{name:"Sharp 9",rootOffset:0},"0 3 5":{name:"5 7",rootOffset:5},"0 3 9":{name:"Minor 6",rootOffset:0},"0 5 10":{name:"Suspended 2",rootOffset:10},"0 5 11":{name:"5 Sharp 11",rootOffset:5},"0 5 6":{name:"5 Add Flat 9",rootOffset:5},"0 6 10":{name:"Flat 5",rootOffset:6},"0 6 7":{name:"5 Sharp 11",rootOffset:0},"0 6 8":{name:"7",rootOffset:8},"0 6 9":{name:"Minor 6",rootOffset:9},"0 7 10":{name:"5 7",rootOffset:0},"0 7 11":{name:"5 Major 7",rootOffset:0},"0 7 8":{name:"Major 7",rootOffset:8},"0 7 9":{name:"Minor 7",rootOffset:9},"0 8 10":{name:"Add 9",rootOffset:8},"0 8 11":{name:"Sharp 9",rootOffset:8},"0 9 10":{name:"Minor Add Flat 9",rootOffset:9},"0 9 11":{name:"Minor Add 9",rootOffset:9},"0 10 11":{name:"No 3 7 Major 7",rootOffset:0},"0 1 4":{name:"Sharp 9",rootOffset:9},"0 8 9":{name:"Sharp 9",rootOffset:5},"0 6 11":{name:"7/13",rootOffset:2},"0 3 11":{name:"Sharp 9",rootOffset:8}};function w(i){const o=i.map(t=>typeof t=="number"?t:O.indexOf(t));if(o.some(t=>t<0))throw new Error("Unsupported note letter or number provided");if(new Set(o.map(t=>t%12)).size===1)return{name:O[o[0]%12]};const f=o.sort((t,s)=>t-s)[0]%12,m=[...new Set(o.map(t=>(t-f)%12).sort((t,s)=>t-s))],a=x[m.join(" ")];if(a){const t=(f+a.rootOffset)%12,s=O[t];return{name:a.name!==""?`${s} ${a.name}`:s,interval:m,root:t}}else return{name:void 0}}h=p.identify=w;const A={mounted(){typeof navigator.requestMIDIAccess<"u"&&navigator.requestMIDIAccess().then(i=>{this.midi=i,this.midi.inputs.forEach(o=>{o.onmidimessage=f=>{f.data[0]===144&&(this.activeKeys.set(f.data[1],f.data),this.$forceUpdate()),f.data[0]===128&&(this.activeKeys.delete(f.data[1]),this.$forceUpdate())}})},i=>{console.error(i)})},data(){return{tooltip:!1,midi:void 0,activeKeys:new Map}},computed:{inputs(){if(typeof this.midi<"u")return Array.from(this.midi.inputs.values())},outputs(){if(typeof this.midi<"u")return Array.from(this.midi.outputs.values())}},methods:{orderedNotes(){return Array.from(this.activeKeys.keys()).sort()},chord(){const i=Array.from(this.activeKeys.keys());return h(i).name}}},k={class:"h-2/3 bg-gradient-to-b from-green-900 via-purple-900 to-indigo-900 text-white"},N={key:0,class:"p-4 text-center font-light tracking-wide"},S={key:1},j={class:"flex flex-wrap"},I={class:"absolute bottom-16 right-2"},D={class:"p-3 font-mono text-orange-900"},F={class:"mb-2 font-bold"},E={class:"mb-2"},z={key:0,class:"p-4 text-center italic"},V={key:1},C={class:"flex-1"},B={class:"flex-1"},K={class:"mb-2"},T={key:0,class:"p-4 text-center italic"},P={key:1},U={class:"flex-1"},G={class:"flex-1"},L={class:"absolute bottom-2 right-2"},q={class:"absolute bottom-4 left-4"},H={class:"flex"},R={class:"grid h-screen place-items-center"},W={class:"text-center text-4xl font-semibold tracking-wide"},Y={key:0};function J(i,o,f,m,a,t){return n(),r("div",k,[typeof a.midi>"u"?(n(),r("div",N,o[1]||(o[1]=[l(" Unfortunately, the Web MIDI API is "),e("a",{class:"text-blue-500 underline",href:"https://developer.mozilla.org/en-US/docs/Web/API/MIDIMessageEvent#browser_compatibility"},"not supported",-1),l(" in all browsers... ")]))):(n(),r("div",S,[e("div",j,[e("div",I,[e("div",{class:b([{hidden:!a.tooltip,block:a.tooltip},"z-50 max-w-md break-words rounded-lg border-2 border-green-800 bg-yellow-200 text-sm font-normal leading-normal"])},[e("div",null,[o[5]||(o[5]=e("div",{class:"mb-0 rounded-t-lg border-b border-solid bg-green-600 p-3 font-semibold uppercase text-white opacity-75"}," MIDI Status ",-1)),e("div",D,[e("div",F,[o[2]||(o[2]=l(" Enabled: ")),e("span",null,d(typeof a.midi<"u"?"Yep!":"Nope"),1)]),e("div",E,[o[3]||(o[3]=e("div",{class:"font-bold"},"Inputs:",-1)),t.inputs.length===0?(n(),r("div",z," No input devices detected :( ")):(n(),r("div",V,[(n(!0),r(c,null,u(t.inputs,s=>(n(),r("div",{key:s.id,class:"flex"},[e("div",C,d(s.manufacturer),1),e("div",B,d(s.name),1)]))),128))]))]),e("div",K,[o[4]||(o[4]=e("div",{class:"font-bold"},"Outputs:",-1)),t.outputs.length===0?(n(),r("div",T," No output devices detected :( ")):(n(),r("div",P,[(n(!0),r(c,null,u(t.outputs,s=>(n(),r("div",{key:s.id,class:"flex"},[e("div",U,d(s.manufacturer),1),e("div",G,d(s.name),1)]))),128))]))])])])],2)]),e("div",L,[e("button",{onClick:o[0]||(o[0]=s=>a.tooltip=!a.tooltip),class:"border-3 rounded-lg border-green-600 bg-green-50 px-1 text-green-600 shadow hover:text-green-500 hover:shadow-lg",type:"button"},o[6]||(o[6]=[e("svg",{xmlns:"http://www.w3.org/2000/svg","xmlns:xlink":"http://www.w3.org/1999/xlink",viewBox:"0 0 20 20",class:"h-10 w-10"},[e("g",{fill:"none"},[e("path",{d:"M2.5 4a.5.5 0 0 0-.5.5v11a.5.5 0 0 0 .5.5h15a.5.5 0 0 0 .5-.5v-11a.5.5 0 0 0-.5-.5h-15zm.5 6h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v3.5a.5.5 0 0 0 1 0V10h2v5H3v-5zm2.75-2.5a.75.75 0 1 1 0-1.5a.75.75 0 0 1 0 1.5zm6.25-1a.5.5 0 0 1 .5-.5h2a.5.5 0 0 1 0 1h-2a.5.5 0 0 1-.5-.5zm-3 .251a.75.75 0 1 1-1.5 0a.75.75 0 0 1 1.5 0z",fill:"currentColor"})])],-1)]))])]),e("div",q,[e("div",H,[(n(!0),r(c,null,u(t.orderedNotes(),s=>(n(),r("div",{key:s,class:"p-4"},d(s),1))),128))])]),e("div",R,[e("div",null,[e("div",W,[_(M,{"enter-active-class":"duration-500 ease-out","enter-class":"opacity-0 transform","enter-to-class":"opacity-100","leave-active-class":"duration-500","leave-class":"opacity-100","leave-to-class":"opacity-0 transform"},{default:y(()=>[a.activeKeys.size>=1?(n(),r("div",Y,d(t.chord()||"?"),1)):g("",!0)]),_:1})])])])]))])}const X=v(A,[["render",J]]);export{X as default}; diff --git a/_nuxt/builds/latest.json b/_nuxt/builds/latest.json index 70d73518..13ea7850 100644 --- a/_nuxt/builds/latest.json +++ b/_nuxt/builds/latest.json @@ -1 +1 @@ -{"id":"e6f03e61-3022-4913-8d71-aebf9c20aa41","timestamp":1731781839142} \ No newline at end of file +{"id":"2efca5fe-2565-4a9a-b7e7-d2f9211a5691","timestamp":1731782019828} \ No newline at end of file diff --git a/_nuxt/builds/meta/2efca5fe-2565-4a9a-b7e7-d2f9211a5691.json b/_nuxt/builds/meta/2efca5fe-2565-4a9a-b7e7-d2f9211a5691.json new file mode 100644 index 00000000..c376d26a --- /dev/null +++ b/_nuxt/builds/meta/2efca5fe-2565-4a9a-b7e7-d2f9211a5691.json @@ -0,0 +1 @@ +{"id":"2efca5fe-2565-4a9a-b7e7-d2f9211a5691","timestamp":1731782019828,"matcher":{"static":{},"wildcard":{},"dynamic":{}},"prerendered":["/","/playground","/playground/audio","/playground/conway","/playground/chords","/playground/metronome","/playground/waves","/playground/midi","/playground/french","/playground/palettes/mountains","/articles","/card","/examples/nested_transitions","/playground/matrix","/playground/palettes/variance","/playground/plotter","/talks","/playground/tiling","/articles/nuxt-v3-migration","/articles/migrate-truenas-from-core-to-scale","/articles/vim-fugitive-gpg-pinentry","/articles/podcast-transcription-whispercpp","/articles/nuxt-content-rss-feed","/articles/doctl","/articles/reset-ipmi-password-from-host-os","/articles/quick-tip-rerunning-bash-commands","/articles/fennel-initial-exploration","/articles/ssh-ed25519-sk-yubikey","/articles/apu2-firmware-upgrade","/articles/unit-testing-micropython-with-mocks","/articles/persistent-archlinux-usb","/articles/docker-selinux-volumes"]} \ No newline at end of file diff --git a/_nuxt/builds/meta/e6f03e61-3022-4913-8d71-aebf9c20aa41.json b/_nuxt/builds/meta/e6f03e61-3022-4913-8d71-aebf9c20aa41.json deleted file mode 100644 index 7bcec36f..00000000 --- a/_nuxt/builds/meta/e6f03e61-3022-4913-8d71-aebf9c20aa41.json +++ /dev/null @@ -1 +0,0 @@ -{"id":"e6f03e61-3022-4913-8d71-aebf9c20aa41","timestamp":1731781839142,"matcher":{"static":{},"wildcard":{},"dynamic":{}},"prerendered":["/","/playground","/playground/audio","/playground/conway","/playground/waves","/playground/metronome","/playground/chords","/playground/midi","/playground/french","/playground/palettes/mountains","/articles","/card","/examples/nested_transitions","/playground/matrix","/playground/plotter","/playground/tiling","/playground/palettes/variance","/talks","/articles/migrate-truenas-from-core-to-scale","/articles/nuxt-v3-migration","/articles/vim-fugitive-gpg-pinentry","/articles/podcast-transcription-whispercpp","/articles/nuxt-content-rss-feed","/articles/doctl","/articles/reset-ipmi-password-from-host-os","/articles/quick-tip-rerunning-bash-commands","/articles/ssh-ed25519-sk-yubikey","/articles/fennel-initial-exploration","/articles/apu2-firmware-upgrade","/articles/unit-testing-micropython-with-mocks","/articles/persistent-archlinux-usb","/articles/docker-selinux-volumes"]} \ No newline at end of file diff --git a/_nuxt/AY-jWpd0.js b/_nuxt/cufizuWq.js similarity index 92% rename from _nuxt/AY-jWpd0.js rename to _nuxt/cufizuWq.js index afbb7c17..705b18d0 100644 --- a/_nuxt/AY-jWpd0.js +++ b/_nuxt/cufizuWq.js @@ -1 +1 @@ -import{h as g}from"./IShi1APO.js";import{u as C}from"./CgY3mHZ8.js";import{q as m}from"./UWyBikFq.js";import{d as S,P as b,k,I as _,v as A,J as N,L as O}from"./BGikLauJ.js";import"./DJ6fbLCw.js";const Q=S({name:"ContentQuery",props:{path:{type:String,required:!1,default:void 0},only:{type:Array,required:!1,default:void 0},without:{type:Array,required:!1,default:void 0},where:{type:Object,required:!1,default:void 0},sort:{type:Object,required:!1,default:void 0},limit:{type:Number,required:!1,default:void 0},skip:{type:Number,required:!1,default:void 0},locale:{type:String,required:!1,default:void 0},find:{type:String,required:!1,default:void 0}},async setup(u){const{path:t,only:r,without:o,where:a,sort:f,limit:l,skip:d,locale:s,find:p}=b(u),y=k(()=>{var e;return(e=t.value)==null?void 0:e.includes("/_")}),h=!_().public.content.experimental.advanceQuery;A(()=>u,()=>n(),{deep:!0});const i=e=>h?e!=null&&e.surround?e.surround:e!=null&&e._id||Array.isArray(e)?e:e==null?void 0:e.result:e.result,{data:v,refresh:n}=await C(`content-query-${g(u)}`,()=>{let e;return t.value?e=m(t.value):e=m(),r.value&&(e=e.only(r.value)),o.value&&(e=e.without(o.value)),a.value&&(e=e.where(a.value)),f.value&&(e=e.sort(f.value)),l.value&&(e=e.limit(l.value)),d.value&&(e=e.skip(d.value)),s.value&&(e=e.where({_locale:s.value})),p.value==="one"?e.findOne().then(i):p.value==="surround"?t.value?h?e.findSurround(t.value):e.withSurround(t.value).findOne().then(i):(console.warn("[Content] Surround queries requires `path` prop to be set."),console.warn("[Content] Query without `path` will return regular `find()` results."),e.find().then(i)):e.find().then(i)});return{isPartial:y,data:v,refresh:n}},render(u){var c;const t=N(),{data:r,refresh:o,isPartial:a,path:f,only:l,without:d,where:s,sort:p,limit:y,skip:h,locale:i,find:v}=u,n={path:f,only:l,without:d,where:s,sort:p,limit:y,skip:h,locale:i,find:v};if(n.find==="one"){if(!r&&(t!=null&&t["not-found"]))return t["not-found"]({props:n,...this.$attrs});if(t!=null&&t.empty&&(r==null?void 0:r._type)==="markdown"&&!((c=r==null?void 0:r.body)!=null&&c.children.length))return t.empty({props:n,...this.$attrs})}else if((!r||!r.length)&&t!=null&&t["not-found"])return t["not-found"]({props:n,...this.$attrs});return t!=null&&t.default?t.default({data:r,refresh:o,isPartial:a,props:n,...this.$attrs}):((w,q)=>O("pre",null,JSON.stringify({message:"You should use slots with !",slot:w,data:q},null,2)))("default",{data:r,props:n,isPartial:a})}}),J=Q;export{J as default}; +import{h as g}from"./IShi1APO.js";import{u as C}from"./2mhhWi5_.js";import{q as m}from"./jaxqT3Hp.js";import{d as S,P as b,k,I as _,v as A,J as N,L as O}from"./Bb-NKJJJ.js";import"./mjvvwssR.js";const Q=S({name:"ContentQuery",props:{path:{type:String,required:!1,default:void 0},only:{type:Array,required:!1,default:void 0},without:{type:Array,required:!1,default:void 0},where:{type:Object,required:!1,default:void 0},sort:{type:Object,required:!1,default:void 0},limit:{type:Number,required:!1,default:void 0},skip:{type:Number,required:!1,default:void 0},locale:{type:String,required:!1,default:void 0},find:{type:String,required:!1,default:void 0}},async setup(u){const{path:t,only:r,without:o,where:a,sort:f,limit:l,skip:d,locale:s,find:p}=b(u),y=k(()=>{var e;return(e=t.value)==null?void 0:e.includes("/_")}),h=!_().public.content.experimental.advanceQuery;A(()=>u,()=>n(),{deep:!0});const i=e=>h?e!=null&&e.surround?e.surround:e!=null&&e._id||Array.isArray(e)?e:e==null?void 0:e.result:e.result,{data:v,refresh:n}=await C(`content-query-${g(u)}`,()=>{let e;return t.value?e=m(t.value):e=m(),r.value&&(e=e.only(r.value)),o.value&&(e=e.without(o.value)),a.value&&(e=e.where(a.value)),f.value&&(e=e.sort(f.value)),l.value&&(e=e.limit(l.value)),d.value&&(e=e.skip(d.value)),s.value&&(e=e.where({_locale:s.value})),p.value==="one"?e.findOne().then(i):p.value==="surround"?t.value?h?e.findSurround(t.value):e.withSurround(t.value).findOne().then(i):(console.warn("[Content] Surround queries requires `path` prop to be set."),console.warn("[Content] Query without `path` will return regular `find()` results."),e.find().then(i)):e.find().then(i)});return{isPartial:y,data:v,refresh:n}},render(u){var c;const t=N(),{data:r,refresh:o,isPartial:a,path:f,only:l,without:d,where:s,sort:p,limit:y,skip:h,locale:i,find:v}=u,n={path:f,only:l,without:d,where:s,sort:p,limit:y,skip:h,locale:i,find:v};if(n.find==="one"){if(!r&&(t!=null&&t["not-found"]))return t["not-found"]({props:n,...this.$attrs});if(t!=null&&t.empty&&(r==null?void 0:r._type)==="markdown"&&!((c=r==null?void 0:r.body)!=null&&c.children.length))return t.empty({props:n,...this.$attrs})}else if((!r||!r.length)&&t!=null&&t["not-found"])return t["not-found"]({props:n,...this.$attrs});return t!=null&&t.default?t.default({data:r,refresh:o,isPartial:a,props:n,...this.$attrs}):((w,q)=>O("pre",null,JSON.stringify({message:"You should use slots with !",slot:w,data:q},null,2)))("default",{data:r,props:n,isPartial:a})}}),J=Q;export{J as default}; diff --git a/_nuxt/dVJITDym.js b/_nuxt/dVJITDym.js deleted file mode 100644 index 732e1d8d..00000000 --- a/_nuxt/dVJITDym.js +++ /dev/null @@ -1 +0,0 @@ -import{_ as m}from"./yJKb_Bp-.js";import"./BGikLauJ.js";export{m as default}; diff --git a/_nuxt/UWyBikFq.js b/_nuxt/jaxqT3Hp.js similarity index 91% rename from _nuxt/UWyBikFq.js rename to _nuxt/jaxqT3Hp.js index 3bdfcce1..1420dec3 100644 --- a/_nuxt/UWyBikFq.js +++ b/_nuxt/jaxqT3Hp.js @@ -1,2 +1,2 @@ -const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["./qIrQuZow.js","./BGikLauJ.js","./C-v3KzvZ.js","./DJ6fbLCw.js","./IShi1APO.js"])))=>i.map(i=>d[i]); -import{Q as $,I as h,R as P,U as b,V as C,M as E}from"./BGikLauJ.js";import{h as l}from"./IShi1APO.js";import{u as m}from"./DJ6fbLCw.js";const d=(t,r)=>r.split(".").reduce((n,i)=>n&&n[i],t),p=(t,r)=>Object.keys(t).filter(r).reduce((n,i)=>Object.assign(n,{[i]:t[i]}),{}),B=t=>r=>t&&t.length?p(r,n=>!t.includes(n)):r,j=t=>r=>Array.isArray(r)?r.map(n=>t(n)):t(r),w=t=>{const r=[],n=[];for(const i of t)["$","_"].includes(i)?r.push(i):n.push(i);return{prefixes:r,properties:n}},q=(t=[])=>r=>{if(t.length===0||!r)return r;const{prefixes:n,properties:i}=w(t);return p(r,o=>!i.includes(o)&&!n.includes(o[0]))},I=(t=[])=>r=>{if(t.length===0||!r)return r;const{prefixes:n,properties:i}=w(t);return p(r,o=>i.includes(o)||n.includes(o[0]))},Q=(t,r)=>{const n=new Intl.Collator(r.$locale,{numeric:r.$numeric,caseFirst:r.$caseFirst,sensitivity:r.$sensitivity}),i=Object.keys(r).filter(o=>!o.startsWith("$"));for(const o of i)t=t.sort((a,e)=>{const s=[d(a,o),d(e,o)].map(c=>{if(c!==null)return c instanceof Date?c.toISOString():c});return r[o]===-1&&s.reverse(),n.compare(s[0],s[1])});return t},F=(t,r="Expected an array")=>{if(!Array.isArray(t))throw new TypeError(r)},u=t=>Array.isArray(t)?t:[void 0,null].includes(t)?[]:[t],S=["sort","where","only","without"];function x(t,r={}){const n={};for(const e of Object.keys(r.initialParams||{}))n[e]=S.includes(e)?u(r.initialParams[e]):r.initialParams[e];const i=(e,s=c=>c)=>(...c)=>(n[e]=s(...c),a),o=e=>{var s;return r.legacy?e!=null&&e.surround?e.surround:e&&(e!=null&&e.dirConfig&&(e.result={_path:(s=e.dirConfig)==null?void 0:s._path,...e.result,_dir:e.dirConfig}),e!=null&&e._path||Array.isArray(e)||!Object.prototype.hasOwnProperty.call(e,"result")?e:e==null?void 0:e.result):e},a={params:()=>({...n,...n.where?{where:[...u(n.where)]}:{},...n.sort?{sort:[...u(n.sort)]}:{}}),only:i("only",u),without:i("without",u),where:i("where",e=>[...u(n.where),...u(e)]),sort:i("sort",e=>[...u(n.sort),...u(e)]),limit:i("limit",e=>parseInt(String(e),10)),skip:i("skip",e=>parseInt(String(e),10)),find:()=>t(a).then(o),findOne:()=>t(i("first")(!0)).then(o),count:()=>t(i("count")(!0)).then(o),locale:e=>a.where({_locale:e}),withSurround:i("surround",(e,s)=>({query:e,...s})),withDirConfig:()=>i("dirConfig")(!0)};return r.legacy&&(a.findSurround=(e,s)=>a.withSurround(e,s).find().then(o)),a}function g(t){return JSON.stringify(t,A)}function A(t,r){return r instanceof RegExp?`--REGEX ${r.toString()}`:r}const O=t=>{let r=g(t);return r=typeof Buffer<"u"?Buffer.from(r).toString("base64"):btoa(r),r=r.replace(/\+/g,"-").replace(/\//g,"_").replace(/=/g,""),(r.match(/.{1,100}/g)||[]).join("/")},y=t=>$(t,h().public.content.api.baseURL),U=()=>{throw console.warn("useContent is only accessible when you are using `documentDriven` mode."),console.warn("Learn more by visiting: https://content.nuxt.com/document-driven"),new Error("useContent is only accessible when you are using `documentDriven` mode.")},D=()=>{const{experimental:t}=h().public.content;return t.clientDB?!0:m().isEnabled()},R=()=>async t=>{const{content:r}=h().public,n=t.params(),i=r.experimental.stripQueryParameters?y(`/query/${`${l(n)}.${r.integrity}`}/${O(n)}.json`):y(`/query/${l(n)}.${r.integrity}.json`);if(D())return(await E(()=>import("./qIrQuZow.js"),__vite__mapDeps([0,1,2,3,4]),import.meta.url).then(e=>e.useContentDatabase())).fetch(t);const o=await $fetch(i,{method:"GET",responseType:"json",params:r.experimental.stripQueryParameters?void 0:{_params:g(n),previewToken:m().getPreviewToken()}});if(typeof o=="string"&&o.startsWith(""))throw new Error("Not found");return o};function G(t,...r){const{content:n}=h().public,i=x(R(),{initialParams:typeof t!="string"?t:{},legacy:!0});let o;typeof t=="string"&&(o=P(b(t,...r)));const a=i.params;return i.params=()=>{var s,c,f;const e=a();return o&&(e.where=e.where||[],e.first&&(e.where||[]).length===0?e.where.push({_path:C(o)}):e.where.push({_path:new RegExp(`^${o.replace(/[-[\]{}()*+.,^$\s/]/g,"\\$&")}`)})),(s=e.sort)!=null&&s.length||(e.sort=[{_stem:1,$numeric:!0}]),n.locales.length&&((f=(c=e.where)==null?void 0:c.find(_=>_._locale))!=null&&f._locale||(e.where=e.where||[],e.where.push({_locale:n.defaultLocale}))),e},i}export{F as a,u as b,Q as c,j as d,O as e,q as f,d as g,I as h,x as i,g as j,B as o,G as q,D as s,U as u,y as w}; +const __vite__mapDeps=(i,m=__vite__mapDeps,d=(m.f||(m.f=["./CgzK-JD6.js","./Bb-NKJJJ.js","./C-v3KzvZ.js","./mjvvwssR.js","./IShi1APO.js"])))=>i.map(i=>d[i]); +import{Q as $,I as h,R as P,U as b,V as C,M as E}from"./Bb-NKJJJ.js";import{h as l}from"./IShi1APO.js";import{u as m}from"./mjvvwssR.js";const d=(t,r)=>r.split(".").reduce((n,i)=>n&&n[i],t),p=(t,r)=>Object.keys(t).filter(r).reduce((n,i)=>Object.assign(n,{[i]:t[i]}),{}),B=t=>r=>t&&t.length?p(r,n=>!t.includes(n)):r,j=t=>r=>Array.isArray(r)?r.map(n=>t(n)):t(r),w=t=>{const r=[],n=[];for(const i of t)["$","_"].includes(i)?r.push(i):n.push(i);return{prefixes:r,properties:n}},q=(t=[])=>r=>{if(t.length===0||!r)return r;const{prefixes:n,properties:i}=w(t);return p(r,o=>!i.includes(o)&&!n.includes(o[0]))},I=(t=[])=>r=>{if(t.length===0||!r)return r;const{prefixes:n,properties:i}=w(t);return p(r,o=>i.includes(o)||n.includes(o[0]))},Q=(t,r)=>{const n=new Intl.Collator(r.$locale,{numeric:r.$numeric,caseFirst:r.$caseFirst,sensitivity:r.$sensitivity}),i=Object.keys(r).filter(o=>!o.startsWith("$"));for(const o of i)t=t.sort((a,e)=>{const s=[d(a,o),d(e,o)].map(c=>{if(c!==null)return c instanceof Date?c.toISOString():c});return r[o]===-1&&s.reverse(),n.compare(s[0],s[1])});return t},F=(t,r="Expected an array")=>{if(!Array.isArray(t))throw new TypeError(r)},u=t=>Array.isArray(t)?t:[void 0,null].includes(t)?[]:[t],S=["sort","where","only","without"];function x(t,r={}){const n={};for(const e of Object.keys(r.initialParams||{}))n[e]=S.includes(e)?u(r.initialParams[e]):r.initialParams[e];const i=(e,s=c=>c)=>(...c)=>(n[e]=s(...c),a),o=e=>{var s;return r.legacy?e!=null&&e.surround?e.surround:e&&(e!=null&&e.dirConfig&&(e.result={_path:(s=e.dirConfig)==null?void 0:s._path,...e.result,_dir:e.dirConfig}),e!=null&&e._path||Array.isArray(e)||!Object.prototype.hasOwnProperty.call(e,"result")?e:e==null?void 0:e.result):e},a={params:()=>({...n,...n.where?{where:[...u(n.where)]}:{},...n.sort?{sort:[...u(n.sort)]}:{}}),only:i("only",u),without:i("without",u),where:i("where",e=>[...u(n.where),...u(e)]),sort:i("sort",e=>[...u(n.sort),...u(e)]),limit:i("limit",e=>parseInt(String(e),10)),skip:i("skip",e=>parseInt(String(e),10)),find:()=>t(a).then(o),findOne:()=>t(i("first")(!0)).then(o),count:()=>t(i("count")(!0)).then(o),locale:e=>a.where({_locale:e}),withSurround:i("surround",(e,s)=>({query:e,...s})),withDirConfig:()=>i("dirConfig")(!0)};return r.legacy&&(a.findSurround=(e,s)=>a.withSurround(e,s).find().then(o)),a}function g(t){return JSON.stringify(t,A)}function A(t,r){return r instanceof RegExp?`--REGEX ${r.toString()}`:r}const O=t=>{let r=g(t);return r=typeof Buffer<"u"?Buffer.from(r).toString("base64"):btoa(r),r=r.replace(/\+/g,"-").replace(/\//g,"_").replace(/=/g,""),(r.match(/.{1,100}/g)||[]).join("/")},y=t=>$(t,h().public.content.api.baseURL),U=()=>{throw console.warn("useContent is only accessible when you are using `documentDriven` mode."),console.warn("Learn more by visiting: https://content.nuxt.com/document-driven"),new Error("useContent is only accessible when you are using `documentDriven` mode.")},D=()=>{const{experimental:t}=h().public.content;return t.clientDB?!0:m().isEnabled()},R=()=>async t=>{const{content:r}=h().public,n=t.params(),i=r.experimental.stripQueryParameters?y(`/query/${`${l(n)}.${r.integrity}`}/${O(n)}.json`):y(`/query/${l(n)}.${r.integrity}.json`);if(D())return(await E(()=>import("./CgzK-JD6.js"),__vite__mapDeps([0,1,2,3,4]),import.meta.url).then(e=>e.useContentDatabase())).fetch(t);const o=await $fetch(i,{method:"GET",responseType:"json",params:r.experimental.stripQueryParameters?void 0:{_params:g(n),previewToken:m().getPreviewToken()}});if(typeof o=="string"&&o.startsWith(""))throw new Error("Not found");return o};function G(t,...r){const{content:n}=h().public,i=x(R(),{initialParams:typeof t!="string"?t:{},legacy:!0});let o;typeof t=="string"&&(o=P(b(t,...r)));const a=i.params;return i.params=()=>{var s,c,f;const e=a();return o&&(e.where=e.where||[],e.first&&(e.where||[]).length===0?e.where.push({_path:C(o)}):e.where.push({_path:new RegExp(`^${o.replace(/[-[\]{}()*+.,^$\s/]/g,"\\$&")}`)})),(s=e.sort)!=null&&s.length||(e.sort=[{_stem:1,$numeric:!0}]),n.locales.length&&((f=(c=e.where)==null?void 0:c.find(_=>_._locale))!=null&&f._locale||(e.where=e.where||[],e.where.push({_locale:n.defaultLocale}))),e},i}export{F as a,u as b,Q as c,j as d,O as e,q as f,d as g,I as h,x as i,g as j,B as o,G as q,D as s,U as u,y as w}; diff --git a/_nuxt/BQ6KdT-G.js b/_nuxt/mIisoFHp.js similarity index 64% rename from _nuxt/BQ6KdT-G.js rename to _nuxt/mIisoFHp.js index 3e84ab86..c240d928 100644 --- a/_nuxt/BQ6KdT-G.js +++ b/_nuxt/mIisoFHp.js @@ -1 +1 @@ -import{m as o,o as r,c as s,a8 as t}from"./BGikLauJ.js";const c={};function n(e,a){return r(),s("em",null,[t(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; +import{m as o,o as r,c as s,a8 as t}from"./Bb-NKJJJ.js";const c={};function n(e,a){return r(),s("em",null,[t(e.$slots,"default")])}const f=o(c,[["render",n]]);export{f as default}; diff --git a/_nuxt/DJ6fbLCw.js b/_nuxt/mjvvwssR.js similarity index 98% rename from _nuxt/DJ6fbLCw.js rename to _nuxt/mjvvwssR.js index de36f345..fef6160f 100644 --- a/_nuxt/DJ6fbLCw.js +++ b/_nuxt/mjvvwssR.js @@ -1 +1 @@ -import{j as A,Z as b,v as O,a1 as P,a7 as R,a6 as D,G as x,u as T}from"./BGikLauJ.js";import{i as N}from"./IShi1APO.js";function L(e,o){if(typeof e!="string")throw new TypeError("argument str must be a string");const i={},t=o||{},s=t.decode||M;let a=0;for(;aD(decodeURIComponent(e)),encode:e=>encodeURIComponent(typeof e=="string"?e:JSON.stringify(e))},y=window.cookieStore;function k(e,o){var f;const i={...q,...o};i.filter??(i.filter=c=>c===e);const t=C(i)||{};let s;i.maxAge!==void 0?s=i.maxAge*1e3:i.expires&&(s=i.expires.getTime()-Date.now());const a=s!==void 0&&s<=0,r=u(a?void 0:t[e]??((f=i.default)==null?void 0:f.call(i))),n=s&&!a?H(r,s,i.watch&&i.watch!=="shallow"):A(r);{let c=null;try{!y&&typeof BroadcastChannel<"u"&&(c=new BroadcastChannel(`nuxt:cookies:${e}`))}catch{}const l=()=>{i.readonly||N(n.value,t[e])||(F(e,n.value,i),t[e]=u(n.value),c==null||c.postMessage({value:i.encode(n.value)}))},p=d=>{var m;const w=d.refresh?(m=C(i))==null?void 0:m[e]:i.decode(d.value);v=!0,n.value=w,t[e]=u(w),x(()=>{v=!1})};let v=!1;const S=!!P();if(S&&b(()=>{v=!0,l(),c==null||c.close()}),y){const d=w=>{const m=w.changed.find(g=>g.name===e),I=w.deleted.find(g=>g.name===e);m&&p({value:m.value}),I&&p({value:null})};y.addEventListener("change",d),S&&b(()=>y.removeEventListener("change",d))}else c&&(c.onmessage=({data:d})=>p(d));i.watch?O(n,()=>{v||l()},{deep:i.watch!=="shallow"}):l()}return n}function C(e={}){return L(document.cookie,e)}function B(e,o,i={}){return o==null?E(e,o,{...i,maxAge:-1}):E(e,o,i)}function F(e,o,i={}){document.cookie=B(e,o,i)}const j=2147483647;function H(e,o,i){let t,s,a=0;const r=i?A(e):{value:e};return P()&&b(()=>{s==null||s(),clearTimeout(t)}),R((n,f)=>{i&&(s=O(r,f));function c(){a=0,clearTimeout(t);const l=o-a,p=l{if(a+=p,a({isEnabled:()=>{const t=T().query;return Object.prototype.hasOwnProperty.call(t,"preview")&&!t.preview?!1:!!(t.preview||k("previewToken").value||sessionStorage.getItem("previewToken"))},getPreviewToken:()=>k("previewToken").value||sessionStorage.getItem("previewToken")||void 0,setPreviewToken:t=>{k("previewToken").value=t,T().query.preview=t||"",t?sessionStorage.setItem("previewToken",t):sessionStorage.removeItem("previewToken"),window.location.reload()}});export{$ as u}; +import{j as A,Z as b,v as O,a1 as P,a7 as R,a6 as D,G as x,u as T}from"./Bb-NKJJJ.js";import{i as N}from"./IShi1APO.js";function L(e,o){if(typeof e!="string")throw new TypeError("argument str must be a string");const i={},t=o||{},s=t.decode||M;let a=0;for(;aD(decodeURIComponent(e)),encode:e=>encodeURIComponent(typeof e=="string"?e:JSON.stringify(e))},y=window.cookieStore;function k(e,o){var f;const i={...q,...o};i.filter??(i.filter=c=>c===e);const t=C(i)||{};let s;i.maxAge!==void 0?s=i.maxAge*1e3:i.expires&&(s=i.expires.getTime()-Date.now());const a=s!==void 0&&s<=0,r=u(a?void 0:t[e]??((f=i.default)==null?void 0:f.call(i))),n=s&&!a?H(r,s,i.watch&&i.watch!=="shallow"):A(r);{let c=null;try{!y&&typeof BroadcastChannel<"u"&&(c=new BroadcastChannel(`nuxt:cookies:${e}`))}catch{}const l=()=>{i.readonly||N(n.value,t[e])||(F(e,n.value,i),t[e]=u(n.value),c==null||c.postMessage({value:i.encode(n.value)}))},p=d=>{var m;const w=d.refresh?(m=C(i))==null?void 0:m[e]:i.decode(d.value);v=!0,n.value=w,t[e]=u(w),x(()=>{v=!1})};let v=!1;const S=!!P();if(S&&b(()=>{v=!0,l(),c==null||c.close()}),y){const d=w=>{const m=w.changed.find(g=>g.name===e),I=w.deleted.find(g=>g.name===e);m&&p({value:m.value}),I&&p({value:null})};y.addEventListener("change",d),S&&b(()=>y.removeEventListener("change",d))}else c&&(c.onmessage=({data:d})=>p(d));i.watch?O(n,()=>{v||l()},{deep:i.watch!=="shallow"}):l()}return n}function C(e={}){return L(document.cookie,e)}function B(e,o,i={}){return o==null?E(e,o,{...i,maxAge:-1}):E(e,o,i)}function F(e,o,i={}){document.cookie=B(e,o,i)}const j=2147483647;function H(e,o,i){let t,s,a=0;const r=i?A(e):{value:e};return P()&&b(()=>{s==null||s(),clearTimeout(t)}),R((n,f)=>{i&&(s=O(r,f));function c(){a=0,clearTimeout(t);const l=o-a,p=l{if(a+=p,a({isEnabled:()=>{const t=T().query;return Object.prototype.hasOwnProperty.call(t,"preview")&&!t.preview?!1:!!(t.preview||k("previewToken").value||sessionStorage.getItem("previewToken"))},getPreviewToken:()=>k("previewToken").value||sessionStorage.getItem("previewToken")||void 0,setPreviewToken:t=>{k("previewToken").value=t,T().query.preview=t||"",t?sessionStorage.setItem("previewToken",t):sessionStorage.removeItem("previewToken"),window.location.reload()}});export{$ as u}; diff --git a/_nuxt/D4ZOq_QP.js b/_nuxt/oLoEepVu.js similarity index 64% rename from _nuxt/D4ZOq_QP.js rename to _nuxt/oLoEepVu.js index 61aa0ac3..c204a518 100644 --- a/_nuxt/D4ZOq_QP.js +++ b/_nuxt/oLoEepVu.js @@ -1 +1 @@ -import{m as o,o as r,c as t,a8 as s}from"./BGikLauJ.js";const c={};function n(e,a){return r(),t("td",null,[s(e.$slots,"default")])}const d=o(c,[["render",n]]);export{d as default}; +import{m as o,o as r,c as t,a8 as s}from"./Bb-NKJJJ.js";const c={};function n(e,a){return r(),t("td",null,[s(e.$slots,"default")])}const d=o(c,[["render",n]]);export{d as default}; diff --git a/_nuxt/Btpue746.js b/_nuxt/tatS9fxq.js similarity index 93% rename from _nuxt/Btpue746.js rename to _nuxt/tatS9fxq.js index 406840d6..71cb6c99 100644 --- a/_nuxt/Btpue746.js +++ b/_nuxt/tatS9fxq.js @@ -1 +1 @@ -import{m as s,c as l,b as t,a8 as a,i as r,f as c,_ as i,o as d}from"./BGikLauJ.js";const _={},u={class:"container mx-auto my-12 h-screen bg-background font-display lg:my-16"},f={class:"flex flex-col border-2 bg-white"},m={class:"absolute left-2 top-2"};function h(o,e){const n=i;return d(),l("main",null,[t("div",u,[t("div",f,[a(o.$slots,"default")])]),t("div",m,[r(n,{as:"button",to:"/playground",class:"inline-flex items-center rounded px-4 py-2 font-bold text-white hover:text-cyan"},{default:c(()=>e[0]||(e[0]=[t("svg",{xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24","stroke-width":"1.5",stroke:"currentColor",class:"mr-2 h-6 w-6"},[t("path",{"stroke-linecap":"round","stroke-linejoin":"round",d:"M15.75 9V5.25A2.25 2.25 0 0013.5 3h-6a2.25 2.25 0 00-2.25 2.25v13.5A2.25 2.25 0 007.5 21h6a2.25 2.25 0 002.25-2.25V15M12 9l-3 3m0 0l3 3m-3-3h12.75"})],-1),t("span",null,"Back",-1)])),_:1})])])}const x=s(_,[["render",h]]);export{x as default}; +import{m as s,c as l,b as t,a8 as a,i as r,f as c,_ as i,o as d}from"./Bb-NKJJJ.js";const _={},u={class:"container mx-auto my-12 h-screen bg-background font-display lg:my-16"},f={class:"flex flex-col border-2 bg-white"},m={class:"absolute left-2 top-2"};function h(o,e){const n=i;return d(),l("main",null,[t("div",u,[t("div",f,[a(o.$slots,"default")])]),t("div",m,[r(n,{as:"button",to:"/playground",class:"inline-flex items-center rounded px-4 py-2 font-bold text-white hover:text-cyan"},{default:c(()=>e[0]||(e[0]=[t("svg",{xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24","stroke-width":"1.5",stroke:"currentColor",class:"mr-2 h-6 w-6"},[t("path",{"stroke-linecap":"round","stroke-linejoin":"round",d:"M15.75 9V5.25A2.25 2.25 0 0013.5 3h-6a2.25 2.25 0 00-2.25 2.25v13.5A2.25 2.25 0 007.5 21h6a2.25 2.25 0 002.25-2.25V15M12 9l-3 3m0 0l3 3m-3-3h12.75"})],-1),t("span",null,"Back",-1)])),_:1})])])}const x=s(_,[["render",h]]);export{x as default}; diff --git a/_nuxt/DSaqcxQi.js b/_nuxt/uL54ePkM.js similarity index 93% rename from _nuxt/DSaqcxQi.js rename to _nuxt/uL54ePkM.js index 62239f74..1920be3e 100644 --- a/_nuxt/DSaqcxQi.js +++ b/_nuxt/uL54ePkM.js @@ -1 +1 @@ -import{m as u,c as m,b as c,o as p}from"./BGikLauJ.js";const v={data(){return{}},mounted(){const a=["#FFBA08","#FAA307","#F48C06","#E85D04","#DC2F02","#D00000","#9D0208","#6A040F","#370617","#03071E"],s=e=>{const i=t=>e.unhex([t.slice(1,3),t.slice(3,5),t.slice(5,7)]),d=(t,o)=>t.map(n=>{const r=e.floor(n+e.random(-o,o));return r<0?0:r>255?255:r});e.setup=()=>{e.createCanvas(400,400),e.frameRate(5)},e.draw=()=>{e.background(0,0,0),e.noStroke(),e.noFill();const t=9,o=400/t;for(let n=0;n<=t;n++)for(let r=0;r<=t;r++){const f=a[r%9];e.fill(d(i(f),15)),e.square(n*o,r*o,o)}}};new this.$p5(s,"canvas")}},I={class:"bg-gradient-to-b from-green-800 to-gray-800"};function b(a,s,e,i,d,l){return p(),m("div",I,s[0]||(s[0]=[c("div",{class:"grid h-screen place-items-center"},[c("div",{class:"font-mono text-white"},[c("div",{class:"mb-2 border-2 border-white"},[c("div",{id:"canvas"})])])],-1)]))}const _=u(v,[["render",b]]);export{_ as default}; +import{m as u,c as m,b as c,o as p}from"./Bb-NKJJJ.js";const v={data(){return{}},mounted(){const a=["#FFBA08","#FAA307","#F48C06","#E85D04","#DC2F02","#D00000","#9D0208","#6A040F","#370617","#03071E"],s=e=>{const i=t=>e.unhex([t.slice(1,3),t.slice(3,5),t.slice(5,7)]),d=(t,o)=>t.map(n=>{const r=e.floor(n+e.random(-o,o));return r<0?0:r>255?255:r});e.setup=()=>{e.createCanvas(400,400),e.frameRate(5)},e.draw=()=>{e.background(0,0,0),e.noStroke(),e.noFill();const t=9,o=400/t;for(let n=0;n<=t;n++)for(let r=0;r<=t;r++){const f=a[r%9];e.fill(d(i(f),15)),e.square(n*o,r*o,o)}}};new this.$p5(s,"canvas")}},I={class:"bg-gradient-to-b from-green-800 to-gray-800"};function b(a,s,e,i,d,l){return p(),m("div",I,s[0]||(s[0]=[c("div",{class:"grid h-screen place-items-center"},[c("div",{class:"font-mono text-white"},[c("div",{class:"mb-2 border-2 border-white"},[c("div",{id:"canvas"})])])],-1)]))}const _=u(v,[["render",b]]);export{_ as default}; diff --git a/_nuxt/DJ0wDsqg.js b/_nuxt/xkKG0dwU.js similarity index 94% rename from _nuxt/DJ0wDsqg.js rename to _nuxt/xkKG0dwU.js index cd0d8ebd..d40d1760 100644 --- a/_nuxt/DJ0wDsqg.js +++ b/_nuxt/xkKG0dwU.js @@ -1 +1 @@ -import{c as s,b as t,F as c,r as u,o,t as r,h as a}from"./BGikLauJ.js";const _={class:"container mx-auto font-mono text-white"},p={class:"grid grid-cols-10 gap-y-6"},g={class:"col-span-2"},m={class:"col-span-8"},f={class:"flex-col space-y-2"},v={class:"text-orange-500"},b={class:"flex"},x=["href"],y=["href"],D=["href"],M={__name:"talks",setup(w){const h=[{title:"Dagster Deep Dive",subtitle:"Orchestrating ML Workloads with Dagster & Modal",date:new Date("2024-9-24"),href_slides:"https://github.com/dagster-io/talks/blob/main/slides/deep-dive-dagster-modal-demo.pdf",href_code:"https://github.com/dagster-io/dagster-modal-demo"},{title:"Dagster Deep Dive",subtitle:"Data Quality: Building Reliable Data Platforms",date:new Date("2024-08-06"),href_slides:"https://github.com/dagster-io/talks/blob/main/slides/deep-dive-data-quality.pdf",href_video:"https://www.youtube.com/watch?v=vT0sSKEPE3A"},{title:"MotherDuck and Dagster",subtitle:"From local development to production",date:new Date("2024-04-18"),href_slides:"https://github.com/dagster-io/talks/blob/main/slides/motherduck-dagster-evidence-hybrid-compute.pdf",href_video:"https://www.youtube.com/watch?v=cOSiMMb_rjk",href_code:"https://github.com/dagster-io/talks/tree/main/motherduck-dagster-hybrid-compute"},{title:"Dagster Deep Dive",subtitle:"Configurations and Resources",date:new Date("2024-03-05"),href_slides:"https://github.com/dagster-io/talks/blob/main/slides/02-deep-dive-resources.pdf",href_video:"https://www.youtube.com/watch?v=i6m7k16W-yg",href_code:"https://github.com/dagster-io/talks/tree/main/dagster-deep-dives/dagster_deep_dives/resources_and_configurations"}],n=function(i){return i.toLocaleDateString("en-US",{month:"short",day:"2-digit",year:"numeric"})};return(i,d)=>(o(),s("div",_,[d[0]||(d[0]=t("h1",{class:"my-6 text-2xl font-extrabold"},"Talks",-1)),t("div",p,[(o(),s(c,null,u(h,(e,l)=>(o(),s(c,{key:l},[t("div",g,r(n(e.date)),1),t("div",m,[t("div",f,[t("div",v,r(e.title),1),t("div",null,r(e.subtitle),1),t("div",b,[e.href_slides?(o(),s("a",{key:0,class:"text-xs uppercase text-gray-400 hover:cursor-pointer hover:text-orange-500",href:e.href_slides}," [slide_deck] ",8,x)):a("",!0),e.href_video?(o(),s("a",{key:1,class:"text-xs uppercase text-gray-400 hover:cursor-pointer hover:text-orange-500",href:e.href_video}," [video] ",8,y)):a("",!0),e.href_code?(o(),s("a",{key:2,class:"text-xs uppercase text-gray-400 hover:cursor-pointer hover:text-orange-500",href:e.href_code}," [source_code] ",8,D)):a("",!0)])])])],64))),64))])]))}};export{M as default}; +import{c as s,b as t,F as c,r as u,o,t as r,h as a}from"./Bb-NKJJJ.js";const _={class:"container mx-auto font-mono text-white"},p={class:"grid grid-cols-10 gap-y-6"},g={class:"col-span-2"},m={class:"col-span-8"},f={class:"flex-col space-y-2"},v={class:"text-orange-500"},b={class:"flex"},x=["href"],y=["href"],D=["href"],M={__name:"talks",setup(w){const h=[{title:"Dagster Deep Dive",subtitle:"Orchestrating ML Workloads with Dagster & Modal",date:new Date("2024-9-24"),href_slides:"https://github.com/dagster-io/talks/blob/main/slides/deep-dive-dagster-modal-demo.pdf",href_code:"https://github.com/dagster-io/dagster-modal-demo"},{title:"Dagster Deep Dive",subtitle:"Data Quality: Building Reliable Data Platforms",date:new Date("2024-08-06"),href_slides:"https://github.com/dagster-io/talks/blob/main/slides/deep-dive-data-quality.pdf",href_video:"https://www.youtube.com/watch?v=vT0sSKEPE3A"},{title:"MotherDuck and Dagster",subtitle:"From local development to production",date:new Date("2024-04-18"),href_slides:"https://github.com/dagster-io/talks/blob/main/slides/motherduck-dagster-evidence-hybrid-compute.pdf",href_video:"https://www.youtube.com/watch?v=cOSiMMb_rjk",href_code:"https://github.com/dagster-io/talks/tree/main/motherduck-dagster-hybrid-compute"},{title:"Dagster Deep Dive",subtitle:"Configurations and Resources",date:new Date("2024-03-05"),href_slides:"https://github.com/dagster-io/talks/blob/main/slides/02-deep-dive-resources.pdf",href_video:"https://www.youtube.com/watch?v=i6m7k16W-yg",href_code:"https://github.com/dagster-io/talks/tree/main/dagster-deep-dives/dagster_deep_dives/resources_and_configurations"}],n=function(i){return i.toLocaleDateString("en-US",{month:"short",day:"2-digit",year:"numeric"})};return(i,d)=>(o(),s("div",_,[d[0]||(d[0]=t("h1",{class:"my-6 text-2xl font-extrabold"},"Talks",-1)),t("div",p,[(o(),s(c,null,u(h,(e,l)=>(o(),s(c,{key:l},[t("div",g,r(n(e.date)),1),t("div",m,[t("div",f,[t("div",v,r(e.title),1),t("div",null,r(e.subtitle),1),t("div",b,[e.href_slides?(o(),s("a",{key:0,class:"text-xs uppercase text-gray-400 hover:cursor-pointer hover:text-orange-500",href:e.href_slides}," [slide_deck] ",8,x)):a("",!0),e.href_video?(o(),s("a",{key:1,class:"text-xs uppercase text-gray-400 hover:cursor-pointer hover:text-orange-500",href:e.href_video}," [video] ",8,y)):a("",!0),e.href_code?(o(),s("a",{key:2,class:"text-xs uppercase text-gray-400 hover:cursor-pointer hover:text-orange-500",href:e.href_code}," [source_code] ",8,D)):a("",!0)])])])],64))),64))])]))}};export{M as default}; diff --git a/_payload.json b/_payload.json index 95a58b1f..0cfef5ce 100644 --- a/_payload.json +++ b/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781852082] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782032338] \ No newline at end of file diff --git a/api/_content/cache.1731781838504.json b/api/_content/cache.1731782019206.json similarity index 99% rename from api/_content/cache.1731781838504.json rename to api/_content/cache.1731782019206.json index 55cafe1b..9e6a815d 100644 --- a/api/_content/cache.1731781838504.json +++ b/api/_content/cache.1731782019206.json @@ -1 +1 @@ -{"generatedAt":1731781852413,"generateTime":572,"contents":[{"_path":"/articles/apu2-firmware-upgrade","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Upgrading the Firmware on the PCEngines APU2","description":"I've had a PCEngines APU2 gathering dust for a\nfew years, and have recently decided to dust it off.","date":"2019-12-21","draft":false,"tags":["pcengine","apu"],"categories":["linux"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I've had a "},{"type":"element","tag":"a","props":{"href":"https://pcengines.ch/apu2.htm","rel":["nofollow"]},"children":[{"type":"text","value":"PCEngines APU2"}]},{"type":"text","value":" gathering dust for a\nfew years, and have recently decided to dust it off."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Since the last time the device has been powered on, there have been many great\nimprovements to the firmware, and it was very-much due for an upgrade. The\nfollowing steps outline how the firmware was upgraded on the APU from the\nalready-installed operating system -- CentOS."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I've had a "},{"type":"element","tag":"a","props":{"href":"https://pcengines.ch/apu2.htm","rel":["nofollow"]},"children":[{"type":"text","value":"PCEngines APU2"}]},{"type":"text","value":" gathering dust for a\nfew years, and have recently decided to dust it off."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Since the last time the device has been powered on, there have been many great\nimprovements to the firmware, and it was very-much due for an upgrade. The\nfollowing steps outline how the firmware was upgraded on the APU from the\nalready-installed operating system -- CentOS."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First, we will connect to the device over the serial port"}]},{"type":"element","tag":"pre","props":{"code":"screen /dev/tty.usbserial 115200\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"screen /dev/tty.usbserial 115200\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, we will install the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"flashrom"}]},{"type":"text","value":" utility that is needed to update the\nfirmware. Because it is not available in the default "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"yum"}]},{"type":"text","value":" repositories, we\nwill enable the "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Extra Packages for Enterprise Linux"}]},{"type":"text","value":" (EPEL) repository before\ninstallation."}]},{"type":"element","tag":"pre","props":{"code":"sudo yum install epel-release\nsudo yum install flashrom\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sudo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" yum"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" install"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" epel-release\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sudo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" yum"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" install"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" flashrom\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Next, we will download the latest version of the firmware that is compatible\nwith the APU2 device from the PC Engines release website:\n"},{"type":"element","tag":"a","props":{"href":"https://pcengines.github.io/","rel":["nofollow"]},"children":[{"type":"text","value":"https://pcengines.github.io/"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"code":"curl -O https://3mdeb.com/open-source-firmware/pcengines/apu2/apu2_v4.11.0.1.rom\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"curl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -O"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" https://3mdeb.com/open-source-firmware/pcengines/apu2/apu2_v4.11.0.1.rom\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And finally, we will flash the firmware..."}]},{"type":"element","tag":"pre","props":{"code":"sudo flashrom -w apu2_v4.11.0.1.rom -p internal:boardmismatch=force\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sudo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" flashrom"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -w"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" apu2_v4.11.0.1.rom"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -p"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" internal:boardmismatch=force\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"element","tag":"strong","props":{},"children":[{"type":"text","value":"References:"}]}]},{"type":"element","tag":"ul","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://pcengines.ch/apu2.htm","rel":["nofollow"]},"children":[{"type":"text","value":"https://pcengines.ch/apu2.htm"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://pcengines.github.io/","rel":["nofollow"]},"children":[{"type":"text","value":"https://pcengines.github.io/"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/pcengines/apu2-documentation/blob/master/docs/firmware_flashing.md","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/pcengines/apu2-documentation/blob/master/docs/firmware_flashing.md"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/pcengines/apu2-documentation/blob/master/docs/cold_reset.md","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/pcengines/apu2-documentation/blob/master/docs/cold_reset.md"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/elad/openbsd-apu2","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/elad/openbsd-apu2"}]}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:apu2-firmware-upgrade.md","_source":"content","_file":"articles/apu2-firmware-upgrade.md","_stem":"articles/apu2-firmware-upgrade","_extension":"md"},{"_path":"/articles/docker-selinux-volumes","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Docker Volume Permissions with SELinux","description":"Unfamiliar with running Docker on a SELinux enabled system, I found myself\nrunning into a bunch of file permission errors while creating volumes.","date":"2019-12-26","draft":false,"tags":["docker","selinux"],"categories":["linux"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Unfamiliar with running Docker on a SELinux enabled system, I found myself\nrunning into a bunch of file permission errors while creating volumes."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Unfamiliar with running Docker on a SELinux enabled system, I found myself\nrunning into a bunch of file permission errors while creating volumes."}]},{"type":"element","tag":"pre","props":{"code":"mkdir: can't create directory '/data': Permission denied\n","language":"txt","meta":"","className":"language-txt shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"mkdir: can't create directory '/data': Permission denied\n"}]}]}]}]},{"type":"element","tag":"hr","props":{},"children":[]},{"type":"element","tag":"pre","props":{"code":"$ docker info --format '{{json .SecurityOptions}}'\n[\n \"name=seccomp,profile=/etc/docker/seccomp.json\",\n \"name=selinux\"\n]\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" docker"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" info"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --format"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" '{{json .SecurityOptions}}'\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"name=seccomp,profile=/etc/docker/seccomp.json\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"name=selinux\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"It turns out that this can be resolved by appending the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":":z"}]},{"type":"text","value":" flag to the volume\nmappings in the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"docker-compose.yml"}]},{"type":"text","value":" file, indicating that the volume content\nis shared."}]},{"type":"element","tag":"pre","props":{"code":"services:\n server:\n volumes:\n - ./data:/data:z\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"services:\n server:\n volumes:\n - ./data:/data:z\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"From the Docker documentation:"}]},{"type":"element","tag":"blockquote","props":{},"children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"z"}]},{"type":"text","value":" option tells Docker that two containers share the volume content. As\na result, Docker labels the content with a shared content label. Shared\nvolume labels allow all containers to read/write content."}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"element","tag":"strong","props":{},"children":[{"type":"text","value":"References"}]}]},{"type":"element","tag":"ul","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://docs.docker.com/engine/reference/commandline/info/","rel":["nofollow"]},"children":[{"type":"text","value":"Docker Docs: Docker Info"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://docs.docker.com/engine/reference/commandline/run/#mount-volumes-from-container---volumes-from","rel":["nofollow"]},"children":[{"type":"text","value":"Docker Docs: Mounting Volumes"}]}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:docker-selinux-volumes.md","_source":"content","_file":"articles/docker-selinux-volumes.md","_stem":"articles/docker-selinux-volumes","_extension":"md"},{"_path":"/articles/doctl","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Exploring the Digital Ocean `doctl` Utility","description":"I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean doctl command line utility.\nThis proved to be an extremely easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future.","date":"2023-01-01","tags":["linux","digital-ocean"],"categories":["linux"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"doctl"}]},{"type":"text","value":" command line utility.\nThis proved to be an "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"extremely"}]},{"type":"text","value":" easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"doctl"}]},{"type":"text","value":" command line utility.\nThis proved to be an "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"extremely"}]},{"type":"text","value":" easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"To start things off, I had to install and setup authentication to Digital Ocean. Doing\nthis on my Mac machine, I opted to use "},{"type":"element","tag":"a","props":{"href":"https://brew.sh/","rel":["nofollow"]},"children":[{"type":"text","value":"Homebrew"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"# install `doctl`\nbrew install doctl\n\n# setup authentication\ndoctl auth init\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# install `doctl`\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"brew"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" install"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" doctl\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# setup authentication\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" auth"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" init\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"While the online "},{"type":"element","tag":"a","props":{"href":"https://docs.digitalocean.com/reference/doctl/reference/compute/droplet/create/","rel":["nofollow"]},"children":[{"type":"text","value":"documentation"}]},{"type":"text","value":" is fantastic, I instead found myself mostly referencing the outputs of "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"--help"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute droplet create --help\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" droplet"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" create"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --help\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I had to find the image name of the version of Ubuntu I wanted to install:"}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute image list --public | grep ubuntu-22\n\n# 118857366 22.04 (LTS) x64 snapshot Ubuntu ubuntu-22-04-x64 true 7\n# 119383150 22.10 x64 snapshot Ubuntu ubuntu-22-10-x64 true 7\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" image"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" list"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --public"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" |"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" grep"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ubuntu-22\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# 118857366 22.04 (LTS) x64 snapshot Ubuntu ubuntu-22-04-x64 true 7\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# 119383150 22.10 x64 snapshot Ubuntu ubuntu-22-10-x64 true 7\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And also the slug of the compute size:"}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute size list\n\n# Slug Description Memory VCPUs Disk Price Monthly Price Hourly\n# s-1vcpu-512mb-10gb Basic 512 1 10 4.00 0.005950\n# s-1vcpu-1gb Basic 1024 1 25 6.00 0.008930\n# s-1vcpu-1gb-amd Basic AMD 1024 1 25 7.00 0.010420\n# s-1vcpu-1gb-intel Basic Intel 1024 1 25 7.00 0.010420\n# s-1vcpu-2gb Basic 2048 1 50 12.00 0.017860\n# s-1vcpu-2gb-amd Basic AMD 2048 1 50 14.00 0.020830\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" size"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" list\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# Slug Description Memory VCPUs Disk Price Monthly Price Hourly\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-512mb-10gb Basic 512 1 10 4.00 0.005950\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-1gb Basic 1024 1 25 6.00 0.008930\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-1gb-amd Basic AMD 1024 1 25 7.00 0.010420\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-1gb-intel Basic Intel 1024 1 25 7.00 0.010420\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-2gb Basic 2048 1 50 12.00 0.017860\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-2gb-amd Basic AMD 2048 1 50 14.00 0.020830\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I've also configured a few SSH keys with Digital Ocean, and I can have the key (specified by ID) provisioned to the machine using the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"--ssh-keys"}]},{"type":"text","value":" flag."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute ssh-key list\n\n# ID Name FingerPrint\n# 1234 mini \n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ssh-key"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" list\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# ID Name FingerPrint\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# 1234 mini \n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Also, I wanted to install a few packages to the box upon creation, this can be done easily with the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"--user-data-file"}]},{"type":"text","value":" flag to run an initialization script."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"echo 'apt install -y imagemagick zip' > bootstrap.sh\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"echo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 'apt install -y imagemagick zip'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" >"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" bootstrap.sh\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Putting it all together, here is the simple command for creating a small compute instance!"}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute droplet create \\\n --image ubuntu-22-10-x64 \\\n --size s-1vcpu-512mb-10gb \\\n --region nyc1 \\\n --ssh-keys 1234 \\\n --user-data-file boostrap.sh \\\n ephemeral\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" droplet"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" create"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --image"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ubuntu-22-10-x64"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --size"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" s-1vcpu-512mb-10gb"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --region"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" nyc1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --ssh-keys"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 1234"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --user-data-file"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" boostrap.sh"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ephemeral\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Finally, I can connect, do my thing, and destroy the instance."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute ssh ephemeral\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ssh"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ephemeral\n"}]}]}]}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute droplet delete --force ephemeral\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" droplet"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" delete"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --force"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ephemeral\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"All-in-all, I was up and running in about 20 minutes. What a handy utility!"}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:doctl.md","_source":"content","_file":"articles/doctl.md","_stem":"articles/doctl","_extension":"md"},{"_path":"/articles/fennel-initial-exploration","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Impressions of Fennel with Hammerspoon","description":"A while back I read an introductory book on Lisp programming titled, \"The Little\nSchemer\". This book opened my mind to new (to me) programming paradigms, and left me\nwith a strong desire to find a way to incorporate Lisp programming into my every day\nlife. It took some time, but I believe I've found what I've been looking for: Fennel.","draft":false,"date":"2023-10-22","tags":["lisp","hammerspoon","fennel"],"categories":["lisp"],"cover_image":"/images/dall-e-fennel-hammer.jpeg","excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"A while back I read an introductory book on Lisp programming titled, \"The Little\nSchemer\". This book opened my mind to new (to me) programming paradigms, and left me\nwith a strong desire to find a way to incorporate Lisp programming into my every day\nlife. It took some time, but I believe I've found what I've been looking for: Fennel."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"A while back I read an introductory book on Lisp programming titled, \"The Little\nSchemer\". This book opened my mind to new (to me) programming paradigms, and left me\nwith a strong desire to find a way to incorporate Lisp programming into my every day\nlife. It took some time, but I believe I've found what I've been looking for: Fennel."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The "},{"type":"element","tag":"a","props":{"href":"https://fennel-lang.org/","rel":["nofollow"]},"children":[{"type":"text","value":"Fennel"}]},{"type":"text","value":" programming language is a dialect of Lisp that boasts compatibility with\nLua, and it just so happens that two of my favorite applications are configured with\nexactly that language: "},{"type":"element","tag":"a","props":{"href":"https://www.hammerspoon.org/","rel":["nofollow"]},"children":[{"type":"text","value":"Hammerspoon"}]},{"type":"text","value":", and "},{"type":"element","tag":"a","props":{"href":"https://neovim.io/","rel":["nofollow"]},"children":[{"type":"text","value":"Neovim"}]},{"type":"text","value":"."}]},{"type":"element","tag":"h2","props":{"id":"initial-observations"},"children":[{"type":"text","value":"Initial Observations"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"To initially explore Fennel, I wanted to start small. My Hammerspoon configuration\nconsists of 7 "},{"type":"element","tag":"a","props":{"href":"https://github.com/cmpadden/dotfiles/tree/795749fa17e1310bb001bb7deaa22be8689f0027/hammerspoon/.hammerspoon/modules","rel":["nofollow"]},"children":[{"type":"text","value":"modules"}]},{"type":"text","value":" that I use for operations such as: launching applications,\nmanaging windows, keeping my computer from going to sleep, and general operating system\nautomation. So the plan is to translate these modules into Fennel, while maintaining\nwithout breaking the existing functionality. However, at this point, I wasn't even sure\nhow to embed Fennel into my project..."}]},{"type":"element","tag":"h2","props":{"id":"integrating-fennel-with-hammerspoon"},"children":[{"type":"text","value":"Integrating Fennel with Hammerspoon"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"While official documentation exists describing how to "},{"type":"element","tag":"a","props":{"href":"https://fennel-lang.org/setup#embedding-fennel","rel":["nofollow"]},"children":[{"type":"text","value":"embed fennel"}]},{"type":"text","value":" into your\nproject; it didn't provide me with enough clarity to know my next steps on integrating\nit with Hammerspoon. I found a few resources online demonstrating how to extend the\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"package.path"}]},{"type":"text","value":" and "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"package.cpath"}]},{"type":"text","value":" properties in Lua, but I was unable to get this to\nwork."}]},{"type":"element","tag":"ul","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://blog.exupero.org/hammerspoon-with-fennel/","rel":["nofollow"]},"children":[{"type":"text","value":"https://blog.exupero.org/hammerspoon-with-fennel/"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/Hammerspoon/hammerspoon/issues/2377#issuecomment-636331435","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/Hammerspoon/hammerspoon/issues/2377#issuecomment-636331435"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/agzam/spacehammer/blob/master/init.lua","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/agzam/spacehammer/blob/master/init.lua"}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Ultimately, I opted to include the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"fennel.lua"}]},{"type":"text","value":" file to my Hammerspoon configuration,\nand while not ideal, it does make the configuration nicely self-contained. I'll leave it\nas a future task to include the module installed with LuaRocks."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"With Fennel now included in my Hammerspoon configuration, all I need to do is configure\nthe "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"fennel.path"}]},{"type":"text","value":" to point to the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"*.fnl"}]},{"type":"text","value":" files in the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":".hammerspoon/"}]},{"type":"text","value":" directory, and\nttranslating these modules can begin!"}]},{"type":"element","tag":"pre","props":{"className":"language-lua shiki shiki-themes github-light","code":"-- init.lua\n\nlocal fennel = require('fennel')\n\nfennel.path = package.path .. \";\" .. os.getenv(\"HOME\") .. \"/.hammerspoon/?.fnl\"\n\ntable.insert(package.loaders or package.searchers, fennel.searcher)\n\nrequire 'main'\n","language":"lua","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"-- init.lua\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"local"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" fennel "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" require"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'fennel'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"fennel."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"path"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" package.path"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" .."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \";\" "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":".."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" os.getenv"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"HOME\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":".."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"/.hammerspoon/?.fnl\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"table.insert"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"package.loaders"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" or"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" package.searchers"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", fennel."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"searcher"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"require"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 'main'\n"}]}]}]}]},{"type":"element","tag":"h2","props":{"id":"translating-lua-to-fennel"},"children":[{"type":"text","value":"Translating Lua to Fennel"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"As a Fennel novice, I was happy to see that the Fennel project provides an online\ncross-compiler for Lua and Fennel called "},{"type":"element","tag":"a","props":{"href":"https://fennel-lang.org/see","rel":["nofollow"]},"children":[{"type":"text","value":"anti-fennel"}]},{"type":"text","value":", and while it can generate some\nstrange-looking Fennel code, it was an extremely useful tool for me to get\nup-and-running right away. For example, by pasting the simple "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"sleep"}]},{"type":"text","value":" function\nfrom the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"helpers"}]},{"type":"text","value":" module into the compiler:"}]},{"type":"element","tag":"pre","props":{"className":"language-lua shiki shiki-themes github-light","code":"function sleep(ms)\n os.execute(\"sleep \" .. tonumber(ms) / 1000)\nend\n","language":"lua","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"function"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" sleep"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(ms)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" os.execute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"sleep \" "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":".."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" tonumber"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(ms) "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"/"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 1000"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"end\n"}]}]}]}]},{"type":"element","tag":"pre","props":{"className":"language-lisp shiki shiki-themes github-light","code":"(fn sleep [ms]\n (os.execute (.. \"sleep \" (/ (tonumber ms) 1000))))\n","language":"lisp","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(fn "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"sleep"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" [ms]\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (os.execute (.. "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"sleep \""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"/"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (tonumber ms) "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"1000"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"))))\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"As another example, here is the output for my "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"caffeine"}]},{"type":"text","value":" toggle:"}]},{"type":"element","tag":"pre","props":{"className":"language-lua shiki shiki-themes github-light","code":"hs.hotkey.bind(HYPER, \"0\", function()\n hs.caffeinate.toggle(\"displayIdle\")\n if hs.caffeinate.get(\"displayIdle\") then\n helpers:show(\"Caffeine Enabled\", nil, helpers.styles.success, helpers.assets.check)\n else\n helpers:show(\"Caffeine Disabled\", nil, helpers.styles.error, helpers.assets.ban)\n end\nend)\n","language":"lua","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"hs."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"hotkey"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"bind"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(HYPER, "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"0\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"function"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"()\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" hs."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"caffeinate"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"toggle"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"displayIdle\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" if"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" hs."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"caffeinate"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"get"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"displayIdle\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"then\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" helpers"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":":"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"show"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"Caffeine Enabled\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"nil"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", helpers."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"styles"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"success"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", helpers."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"assets"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"check"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" else\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" helpers"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":":"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"show"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"Caffeine Disabled\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"nil"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", helpers."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"styles"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"error"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", helpers."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"assets"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"ban"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" end\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"end"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]}]}]},{"type":"element","tag":"pre","props":{"className":"language-lisp shiki shiki-themes github-light","code":"(hs.hotkey.bind HYPER :0\n (fn [] (hs.caffeinate.toggle :displayIdle)\n (if (hs.caffeinate.get :displayIdle)\n (helpers:show \"Caffeine Enabled\" nil helpers.styles.success helpers.assets.check)\n (helpers:show \"Caffeine Disabled\" nil helpers.styles.error helpers.assets.ban)))) \n","language":"lisp","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(hs.hotkey.bind HYPER "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":":0\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (fn [] (hs.caffeinate.toggle "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":":displayIdle"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"if"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (hs.caffeinate.get "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":":displayIdle"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"helpers"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":":show "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"Caffeine Enabled\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" nil"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" helpers.styles.success helpers.assets.check)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"helpers"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":":show "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"Caffeine Disabled\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" nil"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" helpers.styles.error helpers.assets.ban))))\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"This was especially helpful for more gnarly modules like the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"window"}]},{"type":"text","value":" module used for\nwindow management, and seeing the Lua and Fennel code side-by-side was a kick starter in\nlearning the language!"}]},{"type":"element","tag":"h2","props":{"id":"next-steps"},"children":[{"type":"text","value":"Next Steps"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"While my Fennel Hammerspoon configuration now works with parity to its Lua counterpart,\nI have not yet added new features or modules. I look forward to writing new Fennel code,\nand deepen my understanding of Lisp and the Fennel programming language."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Additionally, before beginning this endeavor, I was already aware of projects like\n"},{"type":"element","tag":"a","props":{"href":"https://github.com/agzam/spacehammer","rel":["nofollow"]},"children":[{"type":"text","value":"spacehammer"}]},{"type":"text","value":"; a wildly impressive Hammerspoon configuration written in Fennel, but,\nI wanted to start small and learn the integration myself. However, with the basics out\nof the way, I hope to explore this project further, and seek lessons-learned for the\nconfiguration of my own."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The full pull-request for translating my Lua Hammerspoon configuration to Fennel can be\nfound here: "},{"type":"element","tag":"a","props":{"href":"https://github.com/cmpadden/dotfiles/pull/19/files","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/cmpadden/dotfiles/pull/19/files"}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"initial-observations","depth":2,"text":"Initial Observations"},{"id":"integrating-fennel-with-hammerspoon","depth":2,"text":"Integrating Fennel with Hammerspoon"},{"id":"translating-lua-to-fennel","depth":2,"text":"Translating Lua to Fennel"},{"id":"next-steps","depth":2,"text":"Next Steps"}]}},"_type":"markdown","_id":"content:articles:fennel-initial-exploration.md","_source":"content","_file":"articles/fennel-initial-exploration.md","_stem":"articles/fennel-initial-exploration","_extension":"md"},{"_path":"/articles/migrate-truenas-from-core-to-scale","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Migrate to TrueNAS Scale from TrueNAS Core","description":"TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale.","date":"2021-12-28","tags":["homelab","supermicro","truenas"],"categories":["homelab"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Thankfully, the upgrade procedure to migrate from TrueNAS Core to TrueNAS Scale is relatively straight forward. All it requires is to create a bootable USB of the TrueNAS Scale image, boot the USB, and select "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Install/Upgrade"}]},{"type":"text","value":" in the installation wizard. But for the sake of being thorough, you can find instructions on how to backup system configurations and install the OS below."}]},{"type":"element","tag":"ol","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Make a backup of your system’s configuration\n"},{"type":"element","tag":"ol","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Navigate to "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"System > General"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Click "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Save Config"}]},{"type":"text","value":", check the "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Export Secret Seed"}]},{"type":"text","value":" box, and click "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Save"}]}]}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Export dataset keys for the encrypted pools\n"},{"type":"element","tag":"ol","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Navigate to "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Storage > Pools"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Click the cog icon, and select "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Export Dataset Keys"}]}]}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Insert the TrueNAS Core bootable USB into the NAS"}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"From the Supermicro IPMI interface select "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Remote Control"}]},{"type":"text","value":" and "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"iKVM/HTML5"}]},{"type":"text","value":" and select "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Reboot"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Select the bootable USB as the boot device"}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"From the TrueNAS installation wizard, select "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Install/Upgrade"}]},{"type":"text","value":", select the drive that contains the TrueNAS installation, and select "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Upgrade Install"}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Reboot the device, and voila — you should be up-and-running! Give the system a quick rundown to validate that your settings and pools have transferred correctly, and then enjoy all the container goodness!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"For a breakdown of the differences between TrueNAS Core, Enterprise, and Scale, you can reference "},{"type":"element","tag":"a","props":{"href":"https://www.truenas.com/help-me-choose/","rel":["nofollow"]},"children":[{"type":"text","value":"this table"}]},{"type":"text","value":"."}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:migrate-truenas-from-core-to-scale.md","_source":"content","_file":"articles/migrate-truenas-from-core-to-scale.md","_stem":"articles/migrate-truenas-from-core-to-scale","_extension":"md"},{"_path":"/articles/nuxt-content-rss-feed","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"How To Add an RSS Feed to a Nuxt Website","description":"If you are a user of Nuxt Content and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.","draft":false,"date":"2024-01-06","tags":["nuxt","rss"],"categories":["programming"],"cover_image":"/images/nuxt-content-rss-feed.jpg","excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you are a user of "},{"type":"element","tag":"a","props":{"href":"https://content.nuxt.com/","rel":["nofollow"]},"children":[{"type":"text","value":"Nuxt Content"}]},{"type":"text","value":" and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you are a user of "},{"type":"element","tag":"a","props":{"href":"https://content.nuxt.com/","rel":["nofollow"]},"children":[{"type":"text","value":"Nuxt Content"}]},{"type":"text","value":" and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application."}]},{"type":"element","tag":"h2","props":{"id":"preface"},"children":[{"type":"text","value":"Preface"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In version 2 of Nuxt, the community module, "},{"type":"element","tag":"a","props":{"href":"https://github.com/nuxt-community/feed-module","rel":["nofollow"]},"children":[{"type":"text","value":"nuxt-community/feed-module"}]},{"type":"text","value":" was a popular choice for adding an RSS feed to your website. However, there has been an unresolved "},{"type":"element","tag":"a","props":{"href":"https://github.com/nuxt-community/feed-module/issues/106","rel":["nofollow"]},"children":[{"type":"text","value":"open issue"}]},{"type":"text","value":" since April 1st, 2022 to add support for Nuxt v3. Thankfully, implementing this feature without a module is relatively straight forward."}]},{"type":"element","tag":"h2","props":{"id":"instructions"},"children":[{"type":"text","value":"Instructions"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First, install the "},{"type":"element","tag":"a","props":{"href":"https://www.npmjs.com/package/feed","rel":["nofollow"]},"children":[{"type":"text","value":"feed"}]},{"type":"text","value":" library into your project:"}]},{"type":"element","tag":"pre","props":{"className":"language-shell shiki shiki-themes github-light","code":"npm i -D feed\n","language":"shell","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"npm"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" i"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -D"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" feed\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, create a "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"server/"}]},{"type":"text","value":" folder in your project if it does not already exist, and create a file named "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"server/routes/atom.ts"}]},{"type":"text","value":"."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Here, we will leverage the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"feed"}]},{"type":"text","value":" library and construct an XML representation of our Nuxt content. As you can see, we first define our "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"feed"}]},{"type":"text","value":" with metadata associated with our RSS feed. This will be used by RSS readers to provide context to the end user. Then, we query our Nuxt content with "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"serverQueryContent"}]},{"type":"text","value":" and append a "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"feed.addItem"}]},{"type":"text","value":" for each article."}]},{"type":"element","tag":"pre","props":{"className":"language-ts shiki shiki-themes github-light","code":"import { serverQueryContent } from '#content/server';\nimport { Feed } from 'feed';\n\nconst BASE_URL = \"https://mywebsite.com\"\nconst AUTHOR_NAME = \"Firstname Lastname\"\n\nexport default defineEventHandler(async (event) => {\n\n const feed = new Feed({\n title: \"My Title\",\n description: \"My Description\",\n id: BASE_URL,\n link: BASE_URL,\n language: \"en\",\n image: `${BASE_URL}/images/placeholder.png`,\n favicon: `${BASE_URL}/favicon.ico`,\n copyright: `All rights reserved ${new Date().getFullYear()}, ${AUTHOR_NAME}`,\n updated: new Date(),\n generator: \"Nuxt static site generation + Feed for Node.js\",\n feedLinks: {\n atom: `${BASE_URL}/atom`\n },\n author: {\n name: AUTHOR_NAME,\n }\n });\n\n const articles = await serverQueryContent(event).find();\n\n articles.forEach((article) => {\n feed.addItem({\n title: article.title ? article.title : \"Missing Title\",\n id: article._path,\n link: `${BASE_URL}${article._path}`,\n description: article.description,\n author: [\n {\n name: AUTHOR_NAME,\n },\n ],\n date: new Date(article.date),\n image: article.cover_image ? `${BASE_URL}/${article.cover_image}` : undefined\n });\n });\n\n return feed.atom1();\n});\n","language":"ts","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" { serverQueryContent } "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" '#content/server'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":";\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" { Feed } "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 'feed'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":";\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"https://mywebsite.com\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" AUTHOR_NAME"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"Firstname Lastname\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"export"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" default"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" defineEventHandler"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"async"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#E36209"},"children":[{"type":"text","value":"event"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"=>"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" feed"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" new"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" Feed"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" title: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"My Title\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" description: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"My Description\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":12},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" id: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":13},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" link: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":14},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" language: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"en\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":15},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" image: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"`${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}/images/placeholder.png`"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":16},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" favicon: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"`${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}/favicon.ico`"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":17},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" copyright: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"`All rights reserved ${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"new"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" Date"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"()."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"getFullYear"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"()"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}, ${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"AUTHOR_NAME"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}`"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":18},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" updated: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"new"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" Date"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(),\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":19},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" generator: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"Nuxt static site generation + Feed for Node.js\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":20},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" feedLinks: {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":21},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" atom: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"`${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}/atom`\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":22},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" },\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":23},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" author: {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":24},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" name: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"AUTHOR_NAME"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":25},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" }\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":26},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":27},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":28},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" articles"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" await"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" serverQueryContent"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(event)."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"find"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"();\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":29},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":30},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" articles."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"forEach"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#E36209"},"children":[{"type":"text","value":"article"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"=>"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":31},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" feed."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"addItem"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":32},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" title: article.title "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"?"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" article.title "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":":"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"Missing Title\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":33},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" id: article._path,\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":34},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" link: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"`${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"article"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"_path"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}`"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":35},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" description: article.description,\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":36},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" author: [\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":37},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":38},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" name: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"AUTHOR_NAME"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":39},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" },\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":40},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ],\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":41},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" date: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"new"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" Date"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(article.date),\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":42},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" image: article.cover_image "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"?"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" `${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}/${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"article"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"cover_image"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}`"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" :"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" undefined\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":43},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":44},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":45},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":46},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" return"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" feed."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"atom1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"();\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":47},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"});\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And that's just about it! Except, if you are statically generating your website with the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"nuxt generate"}]},{"type":"text","value":" command, you will need to configure this server-side route to be pre-rendered on site generation. This is as simple as adding a "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"nitro"}]},{"type":"text","value":" "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"prerender"}]},{"type":"text","value":" definition in your "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"nuxt.config.ts"}]},{"type":"text","value":" file, like so:"}]},{"type":"element","tag":"pre","props":{"className":"language-ts shiki shiki-themes github-light","code":"nitro: {\n prerender: {\n routes: ['/atom']\n }\n}\n","language":"ts","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"nitro"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" prerender"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" routes"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": ["}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'/atom'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" }\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"}\n"}]}]}]}]},{"type":"element","tag":"h2","props":{"id":"bonus"},"children":[{"type":"text","value":"Bonus"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"You may also be interested in adding a "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"sitemap.xml"}]},{"type":"text","value":" to your website. This can be done in almost an identical fashion!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Install the dependency:"}]},{"type":"element","tag":"pre","props":{"className":"language-shell shiki shiki-themes github-light","code":"npm i -D sitemap\n","language":"shell","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"npm"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" i"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -D"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" sitemap\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Create a route at "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"server/routes/sitemap.xml.ts"}]},{"type":"text","value":":"}]},{"type":"element","tag":"pre","props":{"className":"language-ts shiki shiki-themes github-light","code":"import { serverQueryContent } from '#content/server';\nimport { SitemapStream, streamToPromise } from 'sitemap';\n\nexport default defineEventHandler(async (event) => {\n const articles = await serverQueryContent(event).find();\n\n const sitemap = new SitemapStream({ hostname: 'https://my-website.com/' });\n\n // Add non nuxt content endpoints here\n sitemap.write({ url: '/' });\n sitemap.write({ url: '/articles' });\n\n // Dynamically generate routes for Nuxt markdown content\n articles.forEach((article) => sitemap.write({ url: article._path, changefreq: 'monthly' }));\n sitemap.end();\n\n return (await streamToPromise(sitemap));\n});\n","language":"ts","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" { serverQueryContent } "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" '#content/server'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":";\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" { SitemapStream, streamToPromise } "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 'sitemap'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":";\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"export"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" default"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" defineEventHandler"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"async"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#E36209"},"children":[{"type":"text","value":"event"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"=>"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" articles"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" await"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" serverQueryContent"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(event)."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"find"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"();\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" sitemap"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" new"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" SitemapStream"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({ hostname: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'https://my-website.com/'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":" // Add non nuxt content endpoints here\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" sitemap."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"write"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({ url: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'/'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" sitemap."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"write"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({ url: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'/articles'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":12},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":13},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":" // Dynamically generate routes for Nuxt markdown content\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":14},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" articles."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"forEach"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#E36209"},"children":[{"type":"text","value":"article"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"=>"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" sitemap."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"write"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({ url: article._path, changefreq: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'monthly'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" }));\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":15},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" sitemap."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"end"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"();\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":16},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":17},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" return"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"await"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" streamToPromise"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(sitemap));\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":18},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"});\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And add the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"prerender"}]},{"type":"text","value":" entry in your "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"nuxt.config.ts"}]},{"type":"text","value":":"}]},{"type":"element","tag":"pre","props":{"className":"language-ts shiki shiki-themes github-light","code":"nitro: {\n prerender: {\n routes: ['/sitemap.xml', '/atom']\n }\n}\n","language":"ts","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"nitro"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" prerender"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" routes"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": ["}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'/sitemap.xml'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'/atom'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" }\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"}\n"}]}]}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"preface","depth":2,"text":"Preface"},{"id":"instructions","depth":2,"text":"Instructions"},{"id":"bonus","depth":2,"text":"Bonus"}]}},"_type":"markdown","_id":"content:articles:nuxt-content-rss-feed.md","_source":"content","_file":"articles/nuxt-content-rss-feed.md","_stem":"articles/nuxt-content-rss-feed","_extension":"md"},{"_path":"/articles/nuxt-v3-migration","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"This Website Has Been Migrated to Nuxt 3 🎉","description":"This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years.\nNot to mention, all of the great plugins in the Vue ecosystem!","date":"2022-12-31","tags":["nuxt"],"categories":["web"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years.\nNot to mention, all of the great plugins in the Vue ecosystem!"}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years.\nNot to mention, all of the great plugins in the Vue ecosystem!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you're curious what changes were required to make the migration, you can check out "},{"type":"element","tag":"a","props":{"href":"https://github.com/cmpadden/cmpadden.github.io/pull/3","rel":["nofollow"]},"children":[{"type":"text","value":"pull request #3"}]},{"type":"text","value":" in the GitHub repository."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"element","tag":"img","props":{"alt":"Screenshot of Nuxt Migration Pull Request","src":"/images/nuxt-migration-pr.png"},"children":[]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"While the documentation for making this migration is great, there were many breaking changes, and the overall process was quite tedious.\nFor this reason, I opted to generate a new project entirely, and port existing code to this clean slate.\nI believe that this resulted in a project with a bit less cruft."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The most valuable resources for making these changes include:"}]},{"type":"element","tag":"ul","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://nuxt.com/docs/migration/overview","rel":["nofollow"]},"children":[{"type":"text","value":"The Nuxt Migration Guide"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://nuxt.com/docs/getting-started/introduction","rel":["nofollow"]},"children":[{"type":"text","value":"The Nuxt Framework Guide"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://tailwindcss.nuxt.dev/","rel":["nofollow"]},"children":[{"type":"text","value":"Nuxt Tailwind Module Documentation"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://content.nuxtjs.org/","rel":["nofollow"]},"children":[{"type":"text","value":"Nuxt Content Module Documentation"}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Part of the delay for doing this upgrade was in waiting for module developers to support this major release.\nI'm super thankful for all of the hard work they've don, and I'm excited to explore all of the new features available!\nI just hope that the breaking changes in this release don't cause too much fracturing of the community, as it does feel like déjà vu of Python 2 and 3."}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:nuxt-v3-migration.md","_source":"content","_file":"articles/nuxt-v3-migration.md","_stem":"articles/nuxt-v3-migration","_extension":"md"},{"_path":"/articles/persistent-archlinux-usb","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Create a Persistent Arch Linux Bootable USB with Vagrant","description":"When installing a linux distribution, it is common for the instructions to have\nthe user create a bootable USB, boot from the device, and proceed with the\ninstallation procedure from that live medium. However, this blog post will\noutline an alternative approach where a virtual machine created with Vagrant\nwill be used in favor of the live medium.","date":"2020-01-09","draft":false,"tags":["vagrant","archlinux"],"categories":["linux"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"When installing a linux distribution, it is common for the instructions to have\nthe user create a bootable USB, boot from the device, and proceed with the\ninstallation procedure from that live medium. However, this blog post will\noutline an alternative approach where a virtual machine created with Vagrant\nwill be used in favor of the live medium."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"When installing a linux distribution, it is common for the instructions to have\nthe user create a bootable USB, boot from the device, and proceed with the\ninstallation procedure from that live medium. However, this blog post will\noutline an alternative approach where a virtual machine created with Vagrant\nwill be used in favor of the live medium."}]},{"type":"element","tag":"h1","props":{"id":"preface"},"children":[{"type":"text","value":"Preface"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The original intention was to use Docker for this process -- leveraging the\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"--device"}]},{"type":"text","value":" flag and mounting the target USB device in the Docker container,\nbut the underlying hypervisor in Docker Desktop for Mac does not support this.\n"},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://docs.docker.com/docker-for-mac/docker-toolbox/","rel":["nofollow"]},"children":[{"type":"text","value":"1"}]},{"type":"text","value":" "},{"type":"element","tag":"a","props":{"href":"https://github.com/moby/hyperkit","rel":["nofollow"]},"children":[{"type":"text","value":"2"}]},{"type":"text","value":" "},{"type":"element","tag":"a","props":{"href":"https://github.com/docker/for-mac/issues/900","rel":["nofollow"]},"children":[{"type":"text","value":"3"}]}]},{"type":"text","value":" While there are workarounds using Docker\nMachine, Vagrant felt like the path of least resistance."}]},{"type":"element","tag":"h1","props":{"id":"instructions"},"children":[{"type":"text","value":"Instructions"}]},{"type":"element","tag":"h2","props":{"id":"create-an-arch-linux-virtual-machine-with-vagrant"},"children":[{"type":"text","value":"Create an Arch Linux Virtual Machine with Vagrant"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Get the latest Arch Linux image "},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://app.vagrantup.com/archlinux/boxes/archlinux","rel":["nofollow"]},"children":[{"type":"text","value":"4"}]}]},{"type":"text","value":" from the Vagrant Cloud Box\nCatalog."}]},{"type":"element","tag":"pre","props":{"code":"vagrant box add archlinux/archlinux\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"vagrant"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" box"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" add"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" archlinux/archlinux\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Determine the USB vendor information for the thumb-drive that we will\npass-through to the virtual machine. Using the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"VBoxManage"}]},{"type":"text","value":" utility that comes\nwith Virtual Box, list the devices, and make note of the Vendor and Product ID."}]},{"type":"element","tag":"pre","props":{"code":" VBoxManage list usbhost\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" VBoxManage"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" list"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" usbhost\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Create a "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Vagrantfile"}]},{"type":"text","value":" with "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"archlinx/archlinux"}]},{"type":"text","value":" as the target box, and the USB\ndevice information that is passed through. "},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://antonyjepson.wordpress.com/2012/01/26/quickly-attaching-usb-devices-to-virtualbox-guests-using-vboxmanage/","rel":["nofollow"]},"children":[{"type":"text","value":"5"}]},{"type":"text","value":" "},{"type":"element","tag":"a","props":{"href":"https://gist.github.com/dscape/7d829c0c116ef419f963","rel":["nofollow"]},"children":[{"type":"text","value":"6"}]}]},{"type":"text","value":" Vagrant\noffers a handy customization parameter "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"vb.customize"}]},{"type":"text","value":" that calls the\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"VBoxManage"}]},{"type":"text","value":" command under-the-hood, allowing one to enable the guest machine\nto access the host machine's USB devices."}]},{"type":"element","tag":"pre","props":{"code":"# -*- mode: ruby -*-\n# vi: set ft=ruby :\n\nVagrant.configure(\"2\") do |config|\n config.vm.box = \"archlinux/archlinux\"\n config.vm.provider \"virtualbox\" do |vb|\n vb.name = \"archlinux\"\n vb.customize ['modifyvm', :id, '--usb', 'on']\n vb.customize ['usbfilter', 'add', '1', '--target', :id, '--name', 'SanDisk Ultra Fit', '--vendorid', '0x0781', '--productid', '0x5583']\n end\nend\n","language":"txt","meta":"","className":"language-txt shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"# -*- mode: ruby -*-\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"# vi: set ft=ruby :\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Vagrant.configure(\"2\") do |config|\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" config.vm.box = \"archlinux/archlinux\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" config.vm.provider \"virtualbox\" do |vb|\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" vb.name = \"archlinux\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" vb.customize ['modifyvm', :id, '--usb', 'on']\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" vb.customize ['usbfilter', 'add', '1', '--target', :id, '--name', 'SanDisk Ultra Fit', '--vendorid', '0x0781', '--productid', '0x5583']\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" end\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"end\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"When virtual machine is brought up, the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"usbfilter"}]},{"type":"text","value":" is applied, and the guest\nis able to access to the host machine's USB device that was specified in the\nfilter."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Start the machine, "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"ssh"}]},{"type":"text","value":" into the guest, and list the devices to confirm that\nthe USB device is available (see: "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"/dev/sdb"}]},{"type":"text","value":")."}]},{"type":"element","tag":"pre","props":{"code":"$ vagrant up\n$ vagrant ssh\n[vagrant@archlinux ~]$ lsblk\nNAME MAJ:MIN RM SIZE RO TYPE MOUNTPOINT\nsda 8:0 0 20G 0 disk\n├─sda1 8:1 0 1.9G 0 part [SWAP]\n└─sda2 8:2 0 18.1G 0 part /\nsdb 8:16 1 28.7G 0 disk\n└─sdb1 8:17 1 8G 0 part\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" vagrant"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" up\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" vagrant"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ssh\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[vagrant@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]$ lsblk\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"NAME"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" MAJ:MIN"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" RM"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" SIZE"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" RO"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" TYPE"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" MOUNTPOINT\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sda"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8:0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 20G"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" disk\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"├─sda1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8:1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 1.9G"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" part"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" [SWAP]\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"└─sda2"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8:2"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 18.1G"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" part"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" /\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sdb"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8:16"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 28.7G"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" disk\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"└─sdb1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8:17"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8G"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" part\n"}]}]}]}]},{"type":"element","tag":"h2","props":{"id":"install-arch-linux-on-the-usb-drive"},"children":[{"type":"text","value":"Install Arch Linux on the USB Drive"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The "},{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/Installation_guide","rel":["nofollow"]},"children":[{"type":"text","value":"Arch Linux Installation Guide"}]},{"type":"text","value":" outlines the installation procedure in\ngreat detail -- the following steps follow this closely with a few alteration\ndue to installing onto removable media."}]},{"type":"element","tag":"h3","props":{"id":"partition-the-disk-uefi-with-gpt"},"children":[{"type":"text","value":"Partition the Disk (UEFI with GPT)"}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# fdisk /dev/sdb\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# fdisk /dev/sdb\n"}]}]}]}]},{"type":"element","tag":"pre","props":{"code":"Command (m for help): p\nDisk /dev/sdb: 28.66 GiB, 30752636928 bytes, 60063744 sectors\nDisk model: Ultra Fit\nUnits: sectors of 1 * 512 = 512 bytes\nSector size (logical/physical): 512 bytes / 512 bytes\nI/O size (minimum/optimal): 512 bytes / 512 bytes\nDisklabel type: gpt\nDisk identifier: E1D6C445-1B79-AB4D-A442-FA4AD6DF4ECC\n\nDevice Start End Sectors Size Type\n/dev/sdb1 2048 1050623 1048576 512M EFI System\n/dev/sdb2 1050624 60063710 59013087 28.1G Linux filesystem\n\nFilesystem/RAID signature on partition 1 will be wiped.\n","language":"txt","meta":"","className":"language-txt shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Command (m for help): p\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Disk /dev/sdb: 28.66 GiB, 30752636928 bytes, 60063744 sectors\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Disk model: Ultra Fit\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Units: sectors of 1 * 512 = 512 bytes\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Sector size (logical/physical): 512 bytes / 512 bytes\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"I/O size (minimum/optimal): 512 bytes / 512 bytes\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Disklabel type: gpt\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Disk identifier: E1D6C445-1B79-AB4D-A442-FA4AD6DF4ECC\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Device Start End Sectors Size Type\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"/dev/sdb1 2048 1050623 1048576 512M EFI System\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":12},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"/dev/sdb2 1050624 60063710 59013087 28.1G Linux filesystem\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":13},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":14},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Filesystem/RAID signature on partition 1 will be wiped.\n"}]}]}]}]},{"type":"element","tag":"h3","props":{"id":"format-the-partitions"},"children":[{"type":"text","value":"Format the Partitions"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The UEFI specification mandates support for FAT file-systems, and FAT32 is\nrecommended for removable media. "},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/EFI_system_partition#Format_the_partition","rel":["nofollow"]},"children":[{"type":"text","value":"7"}]}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# pacman -Sy dosfstools\n[root@archlinux ~]# mkfs.fat -F32 /dev/sdb1\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# pacman -Sy dosfstools\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# mkfs.fat -F32 /dev/sdb1\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"As for the root partition, it is recommended to use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"ext4"}]},{"type":"text","value":" without a journal to\nreduce the reads and writes to the file-system as this is detrimental to the\nflash-based USB drive. "},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/Install_Arch_Linux_on_a_USB_key#Installation_tweaks","rel":["nofollow"]},"children":[{"type":"text","value":"8"}]}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# mkfs.ext4 -O \"^has_journal\" /dev/sdb2\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# mkfs.ext4 -O "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"^has_journal\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" /dev/sdb2\n"}]}]}]}]},{"type":"element","tag":"h3","props":{"id":"mount-the-partitions-and-bootstrap-the-environment"},"children":[{"type":"text","value":"Mount the Partitions and Bootstrap the Environment"}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# mount /dev/sdb2 /mnt\n[root@archlinux ~]# mkdir -p /mnt/boot/efi\n[root@archlinux ~]# mount /dev/sdb1 /mnt/boot/efi\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# mount /dev/sdb2 /mnt\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# mkdir -p /mnt/boot/efi\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# mount /dev/sdb1 /mnt/boot/efi\n"}]}]}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# pacman -S arch-install-scripts\n[root@archlinux ~]# pacstrap /mnt base linux linux-firmware\n[root@archlinux ~]# genfstab -U /mnt >> /mnt/etc/fstab\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# pacman -S arch-install-scripts\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# pacstrap /mnt base linux linux-firmware\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# genfstab -U /mnt "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":">>"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" /mnt/etc/fstab\n"}]}]}]}]},{"type":"element","tag":"h3","props":{"id":"configure-the-new-environment"},"children":[{"type":"text","value":"Configure the New Environment"}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# arch-chroot /mnt\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"[root@archlinux ~]# arch-chroot /mnt\n"}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux /]# ln -sf /usr/share/zoneinfo/US/Eastern /etc/localtime\n[root@archlinux /]# hwclock --systohc\n[root@archlinux /]# sed -i 's/#en_US.UTF-8/en_US.UTF-8/' /etc/locale.gen\n[root@archlinux /]# locale-gen\n[root@archlinux /]# echo \"LANG=en_US.UTF-8\" > /etc/locale.conf\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"[root@archlinux /]# ln -sf /usr/share/zoneinfo/US/Eastern /etc/localtime\n[root@archlinux /]# hwclock --systohc\n[root@archlinux /]# sed -i 's/#en_US.UTF-8/en_US.UTF-8/' /etc/locale.gen\n[root@archlinux /]# locale-gen\n[root@archlinux /]# echo \"LANG=en_US.UTF-8\" > /etc/locale.conf\n"}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux /]# hostnamectl set-hostname usb\n[root@archlinux /]# echo \"127.0.0.1 localhost\" >> /etc/hosts\n[root@archlinux /]# echo \"::1 localhost\" >> /etc/hosts\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"[root@archlinux /]# hostnamectl set-hostname usb\n[root@archlinux /]# echo \"127.0.0.1 localhost\" >> /etc/hosts\n[root@archlinux /]# echo \"::1 localhost\" >> /etc/hosts\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Note, one difference here from a standard installation is that the\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"--removable"}]},{"type":"text","value":" flag is specified when installing the GRUB bootloader.\n"},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/GRUB#UEFI_systems","rel":["nofollow"]},"children":[{"type":"text","value":"10"}]}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux /]# pacman -S grub\n[root@archlinux /]# grub-install --target=x86_64-efi --efi-directory=/boot/efi --removable --recheck\n[root@usb /]# grub-mkconfig -o /boot/grub/grub.cfg\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"[root@archlinux /]# pacman -S grub\n[root@archlinux /]# grub-install --target=x86_64-efi --efi-directory=/boot/efi --removable --recheck\n[root@usb /]# grub-mkconfig -o /boot/grub/grub.cfg\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Shutdown the virtual machine, restart the host machine, and boot the newly\ncreated Arch Linux thumb-drive!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"🎉"}]},{"type":"element","tag":"h2","props":{"id":"side-note"},"children":[{"type":"text","value":"Side-note"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"It was attempted to use the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"controlvm usbattach"}]},{"type":"text","value":" command to pass the USB\ndevice to the guest machine, but this did not work as it expects the virtual\nmachine to already be running, and the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"vb.customize"}]},{"type":"text","value":" option runs prior to\nbooting the machine. "},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://www.vagrantup.com/docs/virtualbox/configuration.html#vboxmanage-customizations","rel":["nofollow"]},"children":[{"type":"text","value":"11"}]}]}]},{"type":"element","tag":"pre","props":{"code":"Command: [\"controlvm\", \"060a716b-1965-49e2-bc56-12beed5df716\", \"usbattach36fc9e60-c465-11cf-8056-444553540000\"]\n\nStderr: VBoxManage.exe: error: Machine '060a716b-1965-49e2-bc56-12beed5df716' is not currently running.\n","language":"txt","meta":"","className":"language-txt shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Command: [\"controlvm\", \"060a716b-1965-49e2-bc56-12beed5df716\", \"usbattach36fc9e60-c465-11cf-8056-444553540000\"]\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Stderr: VBoxManage.exe: error: Machine '060a716b-1965-49e2-bc56-12beed5df716' is not currently running.\n"}]}]}]}]},{"type":"element","tag":"h2","props":{"id":"references"},"children":[{"type":"text","value":"References"}]},{"type":"element","tag":"ol","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://docs.docker.com/docker-for-mac/docker-toolbox/","rel":["nofollow"]},"children":[{"type":"text","value":"Docker Desktop on Mac vs. Docker Toolbox"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/moby/hyperkit","rel":["nofollow"]},"children":[{"type":"text","value":"GitHub - HyperKit"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/docker/for-mac/issues/900","rel":["nofollow"]},"children":[{"type":"text","value":"GitHub - Docker for Mac - Issue #900"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://app.vagrantup.com/archlinux/boxes/archlinux","rel":["nofollow"]},"children":[{"type":"text","value":"Vagrant Cloud - Arch Linux"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://antonyjepson.wordpress.com/2012/01/26/quickly-attaching-usb-devices-to-virtualbox-guests-using-vboxmanage/","rel":["nofollow"]},"children":[{"type":"text","value":"Attaching USB Devices to VirtualBox Guests using VBoxManage"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://gist.github.com/dscape/7d829c0c116ef419f963","rel":["nofollow"]},"children":[{"type":"text","value":"GitHub Gist - Vagrant USB Filter"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/EFI_system_partition#Format_the_partition","rel":["nofollow"]},"children":[{"type":"text","value":"Arch Linux Wiki - EFI System Partition - Format Partitions"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/Install_Arch_Linux_on_a_USB_key#Installation_tweaks","rel":["nofollow"]},"children":[{"type":"text","value":"Arch Linux Wiki - Arch Linux on USB - Installation Tweaks"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/Installation_guide","rel":["nofollow"]},"children":[{"type":"text","value":"Arch Linux Wiki - Installation Guide"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/GRUB#UEFI_systems","rel":["nofollow"]},"children":[{"type":"text","value":"Arch Linux Wiki - GRUB - UEFI Systems"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://www.vagrantup.com/docs/virtualbox/configuration.html#vboxmanage-customizations","rel":["nofollow"]},"children":[{"type":"text","value":"Vagrant VBoxManage Customizations "}]}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"create-an-arch-linux-virtual-machine-with-vagrant","depth":2,"text":"Create an Arch Linux Virtual Machine with Vagrant"},{"id":"install-arch-linux-on-the-usb-drive","depth":2,"text":"Install Arch Linux on the USB Drive","children":[{"id":"partition-the-disk-uefi-with-gpt","depth":3,"text":"Partition the Disk (UEFI with GPT)"},{"id":"format-the-partitions","depth":3,"text":"Format the Partitions"},{"id":"mount-the-partitions-and-bootstrap-the-environment","depth":3,"text":"Mount the Partitions and Bootstrap the Environment"},{"id":"configure-the-new-environment","depth":3,"text":"Configure the New Environment"}]},{"id":"side-note","depth":2,"text":"Side-note"},{"id":"references","depth":2,"text":"References"}]}},"_type":"markdown","_id":"content:articles:persistent-archlinux-usb.md","_source":"content","_file":"articles/persistent-archlinux-usb.md","_stem":"articles/persistent-archlinux-usb","_extension":"md"},{"_path":"/articles/podcast-transcription-whispercpp","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Easily Transcribe Podcasts with Whisper.cpp","description":"If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive whisper.cpp project. This high-performance fork of OpenAI's Whisper can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the Alter Everything podcast.","draft":false,"date":"2024-01-08","tags":["whisper.cpp","ml"],"categories":["programming"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive "},{"type":"element","tag":"a","props":{"href":"https://github.com/ggerganov/whisper.cpp","rel":["nofollow"]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":" project. This high-performance fork of "},{"type":"element","tag":"a","props":{"href":"https://github.com/openai/whisper","rel":["nofollow"]},"children":[{"type":"text","value":"OpenAI's Whisper"}]},{"type":"text","value":" can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the "},{"type":"element","tag":"a","props":{"href":"https://podcasts.apple.com/us/podcast/alter-everything/id1356137854","rel":["nofollow"]},"children":[{"type":"text","value":"Alter Everything"}]},{"type":"text","value":" podcast."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive "},{"type":"element","tag":"a","props":{"href":"https://github.com/ggerganov/whisper.cpp","rel":["nofollow"]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":" project. This high-performance fork of "},{"type":"element","tag":"a","props":{"href":"https://github.com/openai/whisper","rel":["nofollow"]},"children":[{"type":"text","value":"OpenAI's Whisper"}]},{"type":"text","value":" can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the "},{"type":"element","tag":"a","props":{"href":"https://podcasts.apple.com/us/podcast/alter-everything/id1356137854","rel":["nofollow"]},"children":[{"type":"text","value":"Alter Everything"}]},{"type":"text","value":" podcast."}]},{"type":"element","tag":"h2","props":{"id":"obtain-audio-files"},"children":[{"type":"text","value":"Obtain Audio File(s)"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First, let's get the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"wav"}]},{"type":"text","value":" file from YouTube using the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"youtube-dl"}]},{"type":"text","value":" utility. It should be noted that "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":" expects "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"wav"}]},{"type":"text","value":" filetypes, and this utility defaults to "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"mp3"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"code":" $ youtube-dl \\\n --extract-audio \\\n --audio-format wav \\\n --output podcast.wav \\\n \"https://www.youtube.com/watch?v=CoUN690wSYQ\"\n","language":"sh","meta":"","className":"language-sh shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" youtube-dl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --extract-audio"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --audio-format"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" wav"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --output"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" podcast.wav"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"https://www.youtube.com/watch?v=CoUN690wSYQ\"\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"This file has a 44.1 kHz sample rate, and "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":" expects 16 kHz, so let's go ahead and convert that."}]},{"type":"element","tag":"pre","props":{"code":" $ file podcast.wav\npodcast.wav: RIFF (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 44100 Hz\n\n $ ffmpeg -i podcast.wav -ar 16000 podcast-16khz.wav\n\n $ file podcast-16khz.wav\npodcast-16khz.wav: RIFF (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 16000 Hz\n\n# NOTE: it looks like it's possible to specify this conversion as a post-process as a\n# flag to the `youtube-dl` command -- I will explore this further next time...\n# youtube-dl --extract-audio --audio-quality 0 --audio-format mp3 --postprocessor-args \"-ar 44100\" %dl%\n","language":"sh","meta":"","className":"language-sh shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" file"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" podcast.wav\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"podcast.wav:"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" RIFF"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 44100 Hz\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ffmpeg"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -i"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" podcast.wav"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -ar"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 16000"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" podcast-16khz.wav\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" file"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" podcast-16khz.wav\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"podcast-16khz.wav:"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" RIFF"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 16000 Hz\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# NOTE: it looks like it's possible to specify this conversion as a post-process as a\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# flag to the `youtube-dl` command -- I will explore this further next time...\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# youtube-dl --extract-audio --audio-quality 0 --audio-format mp3 --postprocessor-args \"-ar 44100\" %dl%\n"}]}]}]}]},{"type":"element","tag":"h2","props":{"id":"build-whispercpp-transcribe-audio"},"children":[{"type":"text","value":"Build "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":" & Transcribe Audio"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, let's get the latest version of "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":", download the English Whisper model, and build the example."}]},{"type":"element","tag":"pre","props":{"code":"# Clone the `whisper.cpp` repository\n $ git clone --depth 1 git@github.com:ggerganov/whisper.cpp && cd whisper.cpp\n\n# Download the English Whisper model in `ggml` format\n $ bash ./models/download-ggml-model.sh base.en\n\n# Build the main example\n $ make\n","language":"sh","meta":"","className":"language-sh shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# Clone the `whisper.cpp` repository\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" git"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" clone"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --depth"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" git@github.com:ggerganov/whisper.cpp"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" && "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"cd"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" whisper.cpp\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# Download the English Whisper model in `ggml` format\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" bash"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ./models/download-ggml-model.sh"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" base.en\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# Build the main example\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" make\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And finally, let's transcribe that podcast!"}]},{"type":"element","tag":"pre","props":{"code":" $ ./main \\\n -m ~/workspace/whisper.cpp/models/ggml-base.en.bin \\\n -f ~/Downloads/podcast-16khz.wav \\\n --output-vtt \\\n --output-file out\n\n# whisper_print_timings: load time = 114.71 ms\n# whisper_print_timings: fallbacks = 0 p / 0 h\n# whisper_print_timings: mel time = 692.20 ms\n# whisper_print_timings: sample time = 22278.10 ms / 27893 runs ( 0.80 ms per run)\n# whisper_print_timings: encode time = 10000.75 ms / 55 runs ( 181.83 ms per run)\n# whisper_print_timings: decode time = 331.77 ms / 54 runs ( 6.14 ms per run)\n# whisper_print_timings: batchd time = 45236.73 ms / 27566 runs ( 1.64 ms per run)\n# whisper_print_timings: prompt time = 1921.90 ms / 11832 runs ( 0.16 ms per run)\n# whisper_print_timings: total time = 80709.54 ms\n","language":"sh","meta":"","className":"language-sh shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ./main"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -m"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ~/workspace/whisper.cpp/models/ggml-base.en.bin"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -f"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ~/Downloads/podcast-16khz.wav"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --output-vtt"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --output-file"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" out\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: load time = 114.71 ms\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: fallbacks = 0 p / 0 h\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: mel time = 692.20 ms\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: sample time = 22278.10 ms / 27893 runs ( 0.80 ms per run)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: encode time = 10000.75 ms / 55 runs ( 181.83 ms per run)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":12},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: decode time = 331.77 ms / 54 runs ( 6.14 ms per run)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":13},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: batchd time = 45236.73 ms / 27566 runs ( 1.64 ms per run)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":14},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: prompt time = 1921.90 ms / 11832 runs ( 0.16 ms per run)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":15},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: total time = 80709.54 ms\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"A full podcast transcribed in ~80 seconds on an M1 Mac Mini -- not too bad!"}]},{"type":"element","tag":"pre","props":{"code":"# out.vtt\n\n00:00:00.000 --> 00:00:06.480\n >> Hi everyone. We recently launched a short engagement feedback survey for the Alter Everything\n\n00:00:06.480 --> 00:00:11.360\n podcast. Click the link in the episode description wherever you're listening to let us know what\n\n00:00:11.360 --> 00:00:16.320\n you think and help us improve our show.\n\n00:00:16.320 --> 00:00:21.200\n Welcome to Alter Everything, a podcast about data science and analytics culture. I'm Megan\n\n00:00:21.200 --> 00:00:26.440\n Dibble and today I'm talking with Nick Schrock, CTO and founder of Dagster Labs. We discussed\n\n00:00:26.440 --> 00:00:31.560\n data engineering trends, challenges in the field, why he started his company, and what\n\n00:00:31.560 --> 00:00:38.960\n makes him excited about the future of data engineering. Let's get started.\n\n00:00:38.960 --> 00:00:42.720\n >> Hi, Nick. It's great to have you on our show today. Thanks for being here.\n\n00:00:42.720 --> 00:00:43.920\n >> Thanks for having me.\n\n00:00:43.920 --> 00:00:48.280\n >> Yeah. Could you start off by giving an introduction to yourself for our listeners?\n\n00:00:48.280 --> 00:00:52.920\n >> Sure. My name is Nick Schrock. I'm the CTO and founder of Dagster Labs. There's the\n\n00:00:52.920 --> 00:00:59.520\n company behind Dagster, which is a data orchestration framework. Prior to doing this, I was an engineer\n\n00:00:59.520 --> 00:01:05.960\n at Facebook from 2009, 2017. While I was there, I found a team called product infrastructure\n\n00:01:05.960 --> 00:01:09.800\n whose goal was to make our application developers more efficient and productive, and a bunch\n\n00:01:09.800 --> 00:01:13.840\n of open source work came out of that actually, one of which was React, which I had nothing\n\n00:01:13.840 --> 00:01:18.040\n to do with, but actually the CEO of Dagster Labs co-created and I personally co-created\n\n00:01:18.040 --> 00:01:22.640\n GraphQL. So as I like to say, Pete and I were present at the creation of the full hipster\n\n00:01:22.640 --> 00:01:28.680\n stack. I moved on to Facebook in 2017, figuring out what to do next, and this data engineering\n\n00:01:28.680 --> 00:01:32.960\n and data orchestration problem really got me hooked actually quite soon after I left,\n\n00:01:32.960 --> 00:01:36.280\n and the rest is history. I'm sure we'll get into that more.\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"# out.vtt\n\n00:00:00.000 --> 00:00:06.480\n >> Hi everyone. We recently launched a short engagement feedback survey for the Alter Everything\n\n00:00:06.480 --> 00:00:11.360\n podcast. Click the link in the episode description wherever you're listening to let us know what\n\n00:00:11.360 --> 00:00:16.320\n you think and help us improve our show.\n\n00:00:16.320 --> 00:00:21.200\n Welcome to Alter Everything, a podcast about data science and analytics culture. I'm Megan\n\n00:00:21.200 --> 00:00:26.440\n Dibble and today I'm talking with Nick Schrock, CTO and founder of Dagster Labs. We discussed\n\n00:00:26.440 --> 00:00:31.560\n data engineering trends, challenges in the field, why he started his company, and what\n\n00:00:31.560 --> 00:00:38.960\n makes him excited about the future of data engineering. Let's get started.\n\n00:00:38.960 --> 00:00:42.720\n >> Hi, Nick. It's great to have you on our show today. Thanks for being here.\n\n00:00:42.720 --> 00:00:43.920\n >> Thanks for having me.\n\n00:00:43.920 --> 00:00:48.280\n >> Yeah. Could you start off by giving an introduction to yourself for our listeners?\n\n00:00:48.280 --> 00:00:52.920\n >> Sure. My name is Nick Schrock. I'm the CTO and founder of Dagster Labs. There's the\n\n00:00:52.920 --> 00:00:59.520\n company behind Dagster, which is a data orchestration framework. Prior to doing this, I was an engineer\n\n00:00:59.520 --> 00:01:05.960\n at Facebook from 2009, 2017. While I was there, I found a team called product infrastructure\n\n00:01:05.960 --> 00:01:09.800\n whose goal was to make our application developers more efficient and productive, and a bunch\n\n00:01:09.800 --> 00:01:13.840\n of open source work came out of that actually, one of which was React, which I had nothing\n\n00:01:13.840 --> 00:01:18.040\n to do with, but actually the CEO of Dagster Labs co-created and I personally co-created\n\n00:01:18.040 --> 00:01:22.640\n GraphQL. So as I like to say, Pete and I were present at the creation of the full hipster\n\n00:01:22.640 --> 00:01:28.680\n stack. I moved on to Facebook in 2017, figuring out what to do next, and this data engineering\n\n00:01:28.680 --> 00:01:32.960\n and data orchestration problem really got me hooked actually quite soon after I left,\n\n00:01:32.960 --> 00:01:36.280\n and the rest is history. I'm sure we'll get into that more.\n"}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"obtain-audio-files","depth":2,"text":"Obtain Audio File(s)"},{"id":"build-whispercpp-transcribe-audio","depth":2,"text":"Build whisper.cpp & Transcribe Audio"}]}},"_type":"markdown","_id":"content:articles:podcast-transcription-whispercpp.md","_source":"content","_file":"articles/podcast-transcription-whispercpp.md","_stem":"articles/podcast-transcription-whispercpp","_extension":"md"},{"_path":"/articles/quick-tip-rerunning-bash-commands","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Tip: Re-running Bash Commands","description":"Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use sudo for a command that requires\nroot privileges.","date":"2021-09-22","draft":false,"tags":["tip","bash"],"categories":["tips"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"sudo"}]},{"type":"text","value":" for a command that requires\nroot privileges."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"sudo"}]},{"type":"text","value":" for a command that requires\nroot privileges."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ pacman -Syu\nerror: you cannot perform this operation unless you are root.\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" pacman"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -Syu\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"error:"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" you"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" cannot"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" perform"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" this"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" operation"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" unless"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" you"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" are"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" root.\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Well, I have good news for you -- you can easily re-issue a command with the\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!!"}]},{"type":"text","value":" designator! Simply type "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"sudo"}]},{"type":"text","value":" followed by "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!!"}]},{"type":"text","value":" and you're good to go."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ sudo !!\nsudo pacman -Syu\n[sudo] password for colton:\n:: Synchronizing package databases...\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" sudo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" !!\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sudo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" pacman"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -Syu\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[sudo] password "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"for"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" colton:\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"::"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" Synchronizing"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" package"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" databases...\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"--"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Commands that are prefixed with a bang, "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!"}]},{"type":"text","value":", are considered "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Event\nDesignators,"}]},{"type":"text","value":" and are references to your command-line history. You can take a\nlook at your history with the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"history"}]},{"type":"text","value":" command."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ history\n 1021 touch hello_world.txt\n 1022 ls\n 1023 echo \"Here we go again!\"\n 1024 find . -name *.py\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" history\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" 1021"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" touch"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" hello_world.txt\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" 1022"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ls\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" 1023"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" echo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"Here we go again!\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" 1024"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" find"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -name"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" *"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":".py\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"There are many ways to use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!"}]},{"type":"text","value":" in your shell. For example, if you wanted to\nre-issue a specific command in your history, you could use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!n"}]},{"type":"text","value":" where "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"n"}]},{"type":"text","value":" is\nthe number next to the command in your history."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ !1023\necho \"Here we go again!\"\nHere we go again!\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" !1023\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"echo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"Here we go again!\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"Here"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" we"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" go"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" again!\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Or... if you wanted to run the command you issued 4-commands ago, you can use\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!-4"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ !-4\nls\nhello_world.txt\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" !-4\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"ls\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"hello_world.txt\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Or... if you wanted to run the last command that started with the string\n"},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"find"}]},{"type":"text","value":", you can use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!find"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ !find\nfind . -name *.py\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" !find\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"find"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -name"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" *"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":".py\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Be sure to check out the "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Event Designators"}]},{"type":"text","value":" section of the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"bash"}]},{"type":"text","value":" "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"man"}]},{"type":"text","value":" pages\nfor more information!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"--"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"As an aside, for even faster command-line history navigation, be sure to check\nout the excellent "},{"type":"element","tag":"a","props":{"href":"https://github.com/junegunn/fzf","rel":["nofollow"]},"children":[{"type":"text","value":"fzf"}]},{"type":"text","value":" utility by "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"junegunn."}]},{"type":"text","value":"\nOne of the many features of "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"fzf"}]},{"type":"text","value":" is browsing and re-issuing commands from your\ncommand-line history with a fuzzy-finder!"}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:quick-tip-rerunning-bash-commands.md","_source":"content","_file":"articles/quick-tip-rerunning-bash-commands.md","_stem":"articles/quick-tip-rerunning-bash-commands","_extension":"md"},{"_path":"/articles/reset-ipmi-password-from-host-os","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Reset IPMI Credentials from the Host OS","description":"If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these\ninstructions will guide your through the process of resetting the credentials to their default values from the host\noperating system.","date":"2021-12-27","tags":["homelab","supermicro","truenas"],"categories":["homelab"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these\ninstructions will guide your through the process of resetting the credentials to their default values from the host\noperating system."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these\ninstructions will guide your through the process of resetting the credentials to their default values from the host\noperating system."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you are using an operating system like TrueNAS -- good news! It's possible to reset the IPMI password directly from\nthe web interface. This is done by navigating to "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Network > IPMI"}]},{"type":"text","value":", and simply entering a new value in the "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"IPMI\nPassword Reset"}]},{"type":"text","value":" field."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you are using some other OS that doesn't have this feature, you can achieve similar results by using the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"ipmitool"}]},{"type":"text","value":"\ncommand-line utility."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First, you'll want to determine the user ID associated with the user for whom you'd like to reset the password."}]},{"type":"element","tag":"pre","props":{"code":"# ipmitool user list\nID Name Callin Link Auth IPMI Msg Channel Priv Limit\n1 true false false Unknown (0x00)\n2 ADMIN true false false Unknown (0x00)\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"# ipmitool user list\nID Name Callin Link Auth IPMI Msg Channel Priv Limit\n1 true false false Unknown (0x00)\n2 ADMIN true false false Unknown (0x00)\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In this case, we will be resetting the password for "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"ADMIN"}]},{"type":"text","value":" who has a user ID of "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"2"}]},{"type":"text","value":". Then we'll assign the new\npassword like so:"}]},{"type":"element","tag":"pre","props":{"code":"# ipmitool user set password 2 \n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"# ipmitool user set password 2 \n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And you should be good to go!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"..."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Alternatively, if you'd like to factory reset the baseboard management controller (BMC), which will reset the IPMI\ncredentials to their default value, you can issue the following command."}]},{"type":"element","tag":"pre","props":{"code":"# ipmitool raw 0x3c 0x40\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"# ipmitool raw 0x3c 0x40\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Where "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"0x3c"}]},{"type":"text","value":" is the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":""}]},{"type":"text","value":" argument, a.k.a. the network function code that defines the functional routing for\nmessages, and "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"0x40"}]},{"type":"text","value":" is the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":""}]},{"type":"text","value":". According to section 5.1 of the "},{"type":"element","tag":"a","props":{"href":"https://www.intel.com/content/dam/www/public/us/en/documents/product-briefs/ipmi-second-gen-interface-spec-v2-rev1-1.pdf","rel":["nofollow"]},"children":[{"type":"text","value":"IPMI interface\nspecification"}]},{"type":"text","value":",\n"},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"netfn"}]},{"type":"text","value":" codes ranging from 0x30 to 0x3F are reserved for vendor specific functions. I searched around for some\nSupermicro references on these vendor specific network functions without much luck other than various "},{"type":"element","tag":"a","props":{"href":"https://www.supermicro.com/support/faqs/faq.cfm?faq=15448","rel":["nofollow"]},"children":[{"type":"text","value":"support\nresponses"}]},{"type":"text","value":" on how to reset a device. Bummer!"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:reset-ipmi-password-from-host-os.md","_source":"content","_file":"articles/reset-ipmi-password-from-host-os.md","_stem":"articles/reset-ipmi-password-from-host-os","_extension":"md"},{"_path":"/articles/ssh-ed25519-sk-yubikey","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Configuring a YubiKey for use with OpenSSH","description":"YubiKey's are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" ed25519-sk key type that supports FIDO compliant hardware keys.","draft":false,"date":"2024-06-09","tags":["unix","configurations"],"categories":["tooling","tips"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://www.yubico.com/","rel":["nofollow"]},"children":[{"type":"text","value":"YubiKey's"}]},{"type":"text","value":" are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"ed25519-sk"}]},{"type":"text","value":" key type that supports FIDO compliant hardware keys."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://www.yubico.com/","rel":["nofollow"]},"children":[{"type":"text","value":"YubiKey's"}]},{"type":"text","value":" are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"ed25519-sk"}]},{"type":"text","value":" key type that supports FIDO compliant hardware keys."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In "},{"type":"element","tag":"a","props":{"href":"https://www.openssh.com/txt/release-8.2","rel":["nofollow"]},"children":[{"type":"text","value":"release 8.2 of OpenSSH"}]},{"type":"text","value":" support for FIDO devices was added with public key types \"ecdsa-sk\" and \"ed25519-sk\" (-sk standing for \"security key\"). This key type is supported by YubiKey's with firmware version 5.2.3 or higher."}]},{"type":"element","tag":"blockquote","props":{},"children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"This release adds support for FIDO/U2F hardware authenticators to OpenSSH. U2F/FIDO are open standards for inexpensive two-factor authentication hardware that are widely used for website authentication. In OpenSSH FIDO devices are supported by new public key types \"ecdsa-sk\" and \"ed25519-sk\", along with corresponding certificate types."}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Let's get started by installing the latest version of OpenSSH via "},{"type":"element","tag":"a","props":{"href":"https://brew.sh/","rel":["nofollow"]},"children":[{"type":"text","value":"Homebrew"}]},{"type":"text","value":", along with the YubiKey Manager (ykman) CLI. The version of OpenSSH included with macOS is not compatible."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"$ brew install openssh ykman\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" brew"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" install"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" openssh"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ykman\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, let's confirm that our YubiKey has a firmware that is greater than 5.2.3:"}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"$ ykman list\nYubiKey 5Ci (5.4.3) [OTP+FIDO+CCID]\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ykman"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" list\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"YubiKey"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 5Ci"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (5.4.3) [OTP+FIDO+CCID]\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Next, we'll go ahead and enable a pin on our device via the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"change-pin"}]},{"type":"text","value":" command, as this a requirement for our use."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"$ ykman fido access change-pin\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ykman"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" fido"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" access"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" change-pin\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And last, we'll generate the key on our device!"}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"$ ssh-keygen -t ed25519-sk -O resident\nGenerating public/private ed25519-sk key pair.\nYou may need to touch your authenticator to authorize key generation.\n...\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ssh-keygen"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -t"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ed25519-sk"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -O"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" resident\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"Generating"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" public/private"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ed25519-sk"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" key"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" pair.\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"You"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" may"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" need"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" to"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" touch"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" your"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" authenticator"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" to"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" authorize"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" key"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" generation.\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"...\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"We specify "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"resident"}]},{"type":"text","value":" to indicate that the key handle is to be stored on the YubiKey itself, since we will be using this device with multiple computers."}]},{"type":"element","tag":"pre","props":{"className":"language-txt shiki shiki-themes github-light","code":"resident\n Indicate that the key handle should be stored on the FIDO\n authenticator itself. This makes it easier to use the\n authenticator on multiple computers. Resident keys may be\n supported on FIDO2 authenticators and typically require that a PIN\n be set on the authenticator prior to generation. Resident keys\n may be loaded off the authenticator using ssh-add(1). Storing\n both parts of a key on a FIDO authenticator increases the\n likelihood of an attacker being able to use a stolen authenticator\n device.\n","language":"txt","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"resident\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" Indicate that the key handle should be stored on the FIDO\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" authenticator itself. This makes it easier to use the\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" authenticator on multiple computers. Resident keys may be\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" supported on FIDO2 authenticators and typically require that a PIN\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" be set on the authenticator prior to generation. Resident keys\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" may be loaded off the authenticator using ssh-add(1). Storing\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" both parts of a key on a FIDO authenticator increases the\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" likelihood of an attacker being able to use a stolen authenticator\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" device.\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And that's all it takes -- simple enough. Now, when interacting with "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"ssh"}]},{"type":"text","value":" or "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"git"}]},{"type":"text","value":" you will be prompted to touch the YubiKey to bring that little bit of physical 2FA."}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:ssh-ed25519-sk-yubikey.md","_source":"content","_file":"articles/ssh-ed25519-sk-yubikey.md","_stem":"articles/ssh-ed25519-sk-yubikey","_extension":"md"},{"_path":"/articles/unit-testing-micropython-with-mocks","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Unit Testing in MicroPython with Mocks","description":"Unit testing code for embedded systems can be challenging. While it's possible\nto leverage emulators, write side-effect free code, or run tests on the\nhardware itself, it's often easiest to unit test the code on your personal\ncomputer with mocked hardware functionality.","date":"2020-02-07","draft":false,"tags":["micropython","testing","mocks","tutorial"],"categories":["python","embedded"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Unit testing code for embedded systems can be challenging. While it's possible\nto leverage emulators, write side-effect free code, or run tests on the\nhardware itself, it's often easiest to unit test the code on your personal\ncomputer with mocked hardware functionality."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Unit testing code for embedded systems can be challenging. While it's possible\nto leverage emulators, write side-effect free code, or run tests on the\nhardware itself, it's often easiest to unit test the code on your personal\ncomputer with mocked hardware functionality."}]},{"type":"element","tag":"h1","props":{"id":"mocking"},"children":[{"type":"text","value":"Mocking"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Mocks allow us to replace the hardware interfacing functionality under-the-hood\nwith predefined results and side-effects. For example, if there is a piece of\nlogic that retrieves values from an accelerometer to get a device's\norientation, it would be possible to mock the returned values of the\naccelerometer -- allowing us to run the unit tests on a device that does not\nhave an accelerometer sensor installed."}]},{"type":"element","tag":"h1","props":{"id":"a-micropython-mocking-example"},"children":[{"type":"text","value":"A MicroPython Mocking Example"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In this example, we will be unit testing a module named "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"time_logger"}]},{"type":"text","value":", that\ndepends on the MicroPython library "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"utime"}]},{"type":"text","value":" to log the most recent Epoch time to\na file."}]},{"type":"element","tag":"pre","props":{"className":"language-python shiki shiki-themes github-light","code":"# time_logger.py\n\nclass TimeLogger(object):\n\n def save_time(self):\n \"\"\" Overwrite a file with the most recent Epoch timestamp from `utime`\n \"\"\"\n with open(\"LAST_KNOWN_TIME\", \"w+\") as f:\n f.write(str(utime.time()))\n","language":"python","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# time_logger.py\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"class"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" TimeLogger"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"object"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"):\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" def"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" save_time"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(self):\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"\"\" Overwrite a file with the most recent Epoch timestamp from `utime`\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"\"\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" with"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" open"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"LAST_KNOWN_TIME\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"w+\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"as"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" f:\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" f.write("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"str"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(utime.time()))\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First, because the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"utime"}]},{"type":"text","value":" module is not installed on the machine that the unit\ntests on, we must mock "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"utime"}]},{"type":"text","value":" module before importing "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"time_logger"}]},{"type":"text","value":" in our\nunit test file."}]},{"type":"element","tag":"pre","props":{"className":"language-python shiki shiki-themes github-light","code":"# test_time_logger.py\n\nimport unittest\n\nfrom unittest.mock import MagicMock\n\nsys.modules['utime'] = MagicMock()\nfrom time_logger import TimeLogger\n","language":"python","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# test_time_logger.py\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" unittest\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" unittest.mock "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" MagicMock\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"sys.modules["}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'utime'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"] "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" MagicMock()\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" time_logger "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" TimeLogger\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, we can write a test that patches the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"utime.time"}]},{"type":"text","value":" functionality so that\nit returns a value of our choosing -- in this case, "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"1234"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"className":"language-python shiki shiki-themes github-light","code":"class TestTimeLogger(unittest.TestCase):\n\n def test_save_time(self):\n \"\"\" Verify that the Epoch time is written to file\n \"\"\"\n with unittest.mock.patch(\"utime.time\", return_value=1234):\n t = TimeLogger()\n t.save_time()\n with open(\"LAST_KNOWN_TIME\") as f:\n self.assertEqual(\"1234\", f.read())\n","language":"python","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"class"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" TestTimeLogger"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"unittest"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"TestCase"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"):\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" def"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" test_save_time"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(self):\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"\"\" Verify that the Epoch time is written to file\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"\"\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" with"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" unittest.mock.patch("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"utime.time\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#E36209"},"children":[{"type":"text","value":"return_value"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"1234"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"):\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" t "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" TimeLogger()\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" t.save_time()\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" with"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" open"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"LAST_KNOWN_TIME\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"as"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" f:\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" self"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":".assertEqual("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"1234\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", f.read())\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Now, when the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"save_time"}]},{"type":"text","value":" method gets the latest time from "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"utime.time()"}]},{"type":"text","value":", the\nvalue will be patched to return "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"1234"}]},{"type":"text","value":". That value will be written to a file,\nand our unit test will pass!"}]},{"type":"element","tag":"h2","props":{"id":"references"},"children":[{"type":"text","value":"References"}]},{"type":"element","tag":"ol","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://docs.python.org/3/library/unittest.html","rel":["nofollow"]},"children":[{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"unittest"}]},{"type":"text","value":" — Unit testing framework"}]}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"references","depth":2,"text":"References"}]}},"_type":"markdown","_id":"content:articles:unit-testing-micropython-with-mocks.md","_source":"content","_file":"articles/unit-testing-micropython-with-mocks.md","_stem":"articles/unit-testing-micropython-with-mocks","_extension":"md"},{"_path":"/articles/vim-fugitive-gpg-pinentry","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Using pinentry-mac to sign commits from vim-fugitive","description":"In order to sign git commits from within Vim using a plugin like tpope/vim-fugitive, it is necessary to configure the gpg-agent to use a GUI based pinentry-program.","draft":false,"date":"2024-05-11","tags":["vim","tip"],"categories":["tooling","tips"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In order to sign git commits from within Vim using a plugin like "},{"type":"element","tag":"a","props":{"href":"https://github.com/tpope/vim-fugitive","rel":["nofollow"]},"children":[{"type":"text","value":"tpope/vim-fugitive"}]},{"type":"text","value":", it is necessary to configure the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"gpg-agent"}]},{"type":"text","value":" to use a GUI based "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"pinentry-program"}]},{"type":"text","value":"."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In order to sign git commits from within Vim using a plugin like "},{"type":"element","tag":"a","props":{"href":"https://github.com/tpope/vim-fugitive","rel":["nofollow"]},"children":[{"type":"text","value":"tpope/vim-fugitive"}]},{"type":"text","value":", it is necessary to configure the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"gpg-agent"}]},{"type":"text","value":" to use a GUI based "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"pinentry-program"}]},{"type":"text","value":"."}]},{"type":"element","tag":"h2","props":{"id":"preface"},"children":[{"type":"text","value":"Preface"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The man, the myth, the legend, Timothy Popallopollis himself "},{"type":"element","tag":"a","props":{"href":"https://github.com/tpope/vim-fugitive/issues/846#issuecomment-253816577","rel":["nofollow"]},"children":[{"type":"text","value":"recommends"}]},{"type":"text","value":" configuring your "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"gpg-agent"}]},{"type":"text","value":" to use a GUI based "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"pinentry-program"}]},{"type":"text","value":". On macOS this can be done quite by simply installing "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"pinentry-mac"}]},{"type":"text","value":", and updating your "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"gpg-agent.conf"}]},{"type":"text","value":"."}]},{"type":"element","tag":"h2","props":{"id":"configuration"},"children":[{"type":"text","value":"Configuration"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First things first, let's install the pinentry program."}]},{"type":"element","tag":"pre","props":{"code":"$ brew install pinentry-mac\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" brew"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" install"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" pinentry-mac\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, all we need to do is set the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"pinentry-program"}]},{"type":"text","value":" option in your "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"~/.gnupg/gpg-agent.conf"}]},{"type":"text","value":" file."}]},{"type":"element","tag":"pre","props":{"code":"default-cache-ttl 600\nmax-cache-ttl 7200\npinentry-program /opt/homebrew/bin/pinentry-mac\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"default-cache-ttl 600\nmax-cache-ttl 7200\npinentry-program /opt/homebrew/bin/pinentry-mac\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If your don't know the path to your pinentry program, you can throw down a quick "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"which"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"code":"$ which pinentry-mac\n/opt/homebrew/bin/pinentry-mac\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" which"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" pinentry-mac\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"/opt/homebrew/bin/pinentry-mac\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Or use your Homebrew prefix."}]},{"type":"element","tag":"pre","props":{"code":"$ echo $(brew --prefix)/bin/pinentry-mac\n/opt/homebrew/bin/pinentry-mac\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" echo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" $("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"brew"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --prefix"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"/bin/pinentry-mac\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"/opt/homebrew/bin/pinentry-mac\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"But that's all it takes. Now, you should be prompted to enter your gpg pin in an external window when signing commits from vim."}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"preface","depth":2,"text":"Preface"},{"id":"configuration","depth":2,"text":"Configuration"}]}},"_type":"markdown","_id":"content:articles:vim-fugitive-gpg-pinentry.md","_source":"content","_file":"articles/vim-fugitive-gpg-pinentry.md","_stem":"articles/vim-fugitive-gpg-pinentry","_extension":"md"}],"navigation":[{"title":"Articles","_path":"/articles","children":[{"title":"Upgrading the Firmware on the PCEngines APU2","_path":"/articles/apu2-firmware-upgrade"},{"title":"Docker Volume Permissions with SELinux","_path":"/articles/docker-selinux-volumes"},{"title":"Exploring the Digital Ocean `doctl` Utility","_path":"/articles/doctl"},{"title":"Impressions of Fennel with Hammerspoon","_path":"/articles/fennel-initial-exploration"},{"title":"Migrate to TrueNAS Scale from TrueNAS Core","_path":"/articles/migrate-truenas-from-core-to-scale"},{"title":"How To Add an RSS Feed to a Nuxt Website","_path":"/articles/nuxt-content-rss-feed"},{"title":"This Website Has Been Migrated to Nuxt 3 🎉","_path":"/articles/nuxt-v3-migration"},{"title":"Create a Persistent Arch Linux Bootable USB with Vagrant","_path":"/articles/persistent-archlinux-usb"},{"title":"Easily Transcribe Podcasts with Whisper.cpp","_path":"/articles/podcast-transcription-whispercpp"},{"title":"Tip: Re-running Bash Commands","_path":"/articles/quick-tip-rerunning-bash-commands"},{"title":"Reset IPMI Credentials from the Host OS","_path":"/articles/reset-ipmi-password-from-host-os"},{"title":"Configuring a YubiKey for use with OpenSSH","_path":"/articles/ssh-ed25519-sk-yubikey"},{"title":"Unit Testing in MicroPython with Mocks","_path":"/articles/unit-testing-micropython-with-mocks"},{"title":"Using pinentry-mac to sign commits from vim-fugitive","_path":"/articles/vim-fugitive-gpg-pinentry"}]}]} \ No newline at end of file +{"generatedAt":1731782032663,"generateTime":571,"contents":[{"_path":"/articles/apu2-firmware-upgrade","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Upgrading the Firmware on the PCEngines APU2","description":"I've had a PCEngines APU2 gathering dust for a\nfew years, and have recently decided to dust it off.","date":"2019-12-21","draft":false,"tags":["pcengine","apu"],"categories":["linux"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I've had a "},{"type":"element","tag":"a","props":{"href":"https://pcengines.ch/apu2.htm","rel":["nofollow"]},"children":[{"type":"text","value":"PCEngines APU2"}]},{"type":"text","value":" gathering dust for a\nfew years, and have recently decided to dust it off."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Since the last time the device has been powered on, there have been many great\nimprovements to the firmware, and it was very-much due for an upgrade. The\nfollowing steps outline how the firmware was upgraded on the APU from the\nalready-installed operating system -- CentOS."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I've had a "},{"type":"element","tag":"a","props":{"href":"https://pcengines.ch/apu2.htm","rel":["nofollow"]},"children":[{"type":"text","value":"PCEngines APU2"}]},{"type":"text","value":" gathering dust for a\nfew years, and have recently decided to dust it off."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Since the last time the device has been powered on, there have been many great\nimprovements to the firmware, and it was very-much due for an upgrade. The\nfollowing steps outline how the firmware was upgraded on the APU from the\nalready-installed operating system -- CentOS."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First, we will connect to the device over the serial port"}]},{"type":"element","tag":"pre","props":{"code":"screen /dev/tty.usbserial 115200\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"screen /dev/tty.usbserial 115200\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, we will install the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"flashrom"}]},{"type":"text","value":" utility that is needed to update the\nfirmware. Because it is not available in the default "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"yum"}]},{"type":"text","value":" repositories, we\nwill enable the "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Extra Packages for Enterprise Linux"}]},{"type":"text","value":" (EPEL) repository before\ninstallation."}]},{"type":"element","tag":"pre","props":{"code":"sudo yum install epel-release\nsudo yum install flashrom\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sudo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" yum"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" install"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" epel-release\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sudo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" yum"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" install"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" flashrom\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Next, we will download the latest version of the firmware that is compatible\nwith the APU2 device from the PC Engines release website:\n"},{"type":"element","tag":"a","props":{"href":"https://pcengines.github.io/","rel":["nofollow"]},"children":[{"type":"text","value":"https://pcengines.github.io/"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"code":"curl -O https://3mdeb.com/open-source-firmware/pcengines/apu2/apu2_v4.11.0.1.rom\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"curl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -O"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" https://3mdeb.com/open-source-firmware/pcengines/apu2/apu2_v4.11.0.1.rom\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And finally, we will flash the firmware..."}]},{"type":"element","tag":"pre","props":{"code":"sudo flashrom -w apu2_v4.11.0.1.rom -p internal:boardmismatch=force\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sudo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" flashrom"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -w"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" apu2_v4.11.0.1.rom"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -p"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" internal:boardmismatch=force\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"element","tag":"strong","props":{},"children":[{"type":"text","value":"References:"}]}]},{"type":"element","tag":"ul","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://pcengines.ch/apu2.htm","rel":["nofollow"]},"children":[{"type":"text","value":"https://pcengines.ch/apu2.htm"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://pcengines.github.io/","rel":["nofollow"]},"children":[{"type":"text","value":"https://pcengines.github.io/"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/pcengines/apu2-documentation/blob/master/docs/firmware_flashing.md","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/pcengines/apu2-documentation/blob/master/docs/firmware_flashing.md"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/pcengines/apu2-documentation/blob/master/docs/cold_reset.md","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/pcengines/apu2-documentation/blob/master/docs/cold_reset.md"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/elad/openbsd-apu2","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/elad/openbsd-apu2"}]}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:apu2-firmware-upgrade.md","_source":"content","_file":"articles/apu2-firmware-upgrade.md","_stem":"articles/apu2-firmware-upgrade","_extension":"md"},{"_path":"/articles/docker-selinux-volumes","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Docker Volume Permissions with SELinux","description":"Unfamiliar with running Docker on a SELinux enabled system, I found myself\nrunning into a bunch of file permission errors while creating volumes.","date":"2019-12-26","draft":false,"tags":["docker","selinux"],"categories":["linux"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Unfamiliar with running Docker on a SELinux enabled system, I found myself\nrunning into a bunch of file permission errors while creating volumes."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Unfamiliar with running Docker on a SELinux enabled system, I found myself\nrunning into a bunch of file permission errors while creating volumes."}]},{"type":"element","tag":"pre","props":{"code":"mkdir: can't create directory '/data': Permission denied\n","language":"txt","meta":"","className":"language-txt shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"mkdir: can't create directory '/data': Permission denied\n"}]}]}]}]},{"type":"element","tag":"hr","props":{},"children":[]},{"type":"element","tag":"pre","props":{"code":"$ docker info --format '{{json .SecurityOptions}}'\n[\n \"name=seccomp,profile=/etc/docker/seccomp.json\",\n \"name=selinux\"\n]\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" docker"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" info"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --format"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" '{{json .SecurityOptions}}'\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"name=seccomp,profile=/etc/docker/seccomp.json\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"name=selinux\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"It turns out that this can be resolved by appending the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":":z"}]},{"type":"text","value":" flag to the volume\nmappings in the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"docker-compose.yml"}]},{"type":"text","value":" file, indicating that the volume content\nis shared."}]},{"type":"element","tag":"pre","props":{"code":"services:\n server:\n volumes:\n - ./data:/data:z\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"services:\n server:\n volumes:\n - ./data:/data:z\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"From the Docker documentation:"}]},{"type":"element","tag":"blockquote","props":{},"children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"z"}]},{"type":"text","value":" option tells Docker that two containers share the volume content. As\na result, Docker labels the content with a shared content label. Shared\nvolume labels allow all containers to read/write content."}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"element","tag":"strong","props":{},"children":[{"type":"text","value":"References"}]}]},{"type":"element","tag":"ul","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://docs.docker.com/engine/reference/commandline/info/","rel":["nofollow"]},"children":[{"type":"text","value":"Docker Docs: Docker Info"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://docs.docker.com/engine/reference/commandline/run/#mount-volumes-from-container---volumes-from","rel":["nofollow"]},"children":[{"type":"text","value":"Docker Docs: Mounting Volumes"}]}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:docker-selinux-volumes.md","_source":"content","_file":"articles/docker-selinux-volumes.md","_stem":"articles/docker-selinux-volumes","_extension":"md"},{"_path":"/articles/doctl","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Exploring the Digital Ocean `doctl` Utility","description":"I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean doctl command line utility.\nThis proved to be an extremely easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future.","date":"2023-01-01","tags":["linux","digital-ocean"],"categories":["linux"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"doctl"}]},{"type":"text","value":" command line utility.\nThis proved to be an "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"extremely"}]},{"type":"text","value":" easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"doctl"}]},{"type":"text","value":" command line utility.\nThis proved to be an "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"extremely"}]},{"type":"text","value":" easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"To start things off, I had to install and setup authentication to Digital Ocean. Doing\nthis on my Mac machine, I opted to use "},{"type":"element","tag":"a","props":{"href":"https://brew.sh/","rel":["nofollow"]},"children":[{"type":"text","value":"Homebrew"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"# install `doctl`\nbrew install doctl\n\n# setup authentication\ndoctl auth init\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# install `doctl`\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"brew"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" install"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" doctl\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# setup authentication\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" auth"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" init\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"While the online "},{"type":"element","tag":"a","props":{"href":"https://docs.digitalocean.com/reference/doctl/reference/compute/droplet/create/","rel":["nofollow"]},"children":[{"type":"text","value":"documentation"}]},{"type":"text","value":" is fantastic, I instead found myself mostly referencing the outputs of "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"--help"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute droplet create --help\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" droplet"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" create"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --help\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I had to find the image name of the version of Ubuntu I wanted to install:"}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute image list --public | grep ubuntu-22\n\n# 118857366 22.04 (LTS) x64 snapshot Ubuntu ubuntu-22-04-x64 true 7\n# 119383150 22.10 x64 snapshot Ubuntu ubuntu-22-10-x64 true 7\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" image"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" list"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --public"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" |"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" grep"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ubuntu-22\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# 118857366 22.04 (LTS) x64 snapshot Ubuntu ubuntu-22-04-x64 true 7\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# 119383150 22.10 x64 snapshot Ubuntu ubuntu-22-10-x64 true 7\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And also the slug of the compute size:"}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute size list\n\n# Slug Description Memory VCPUs Disk Price Monthly Price Hourly\n# s-1vcpu-512mb-10gb Basic 512 1 10 4.00 0.005950\n# s-1vcpu-1gb Basic 1024 1 25 6.00 0.008930\n# s-1vcpu-1gb-amd Basic AMD 1024 1 25 7.00 0.010420\n# s-1vcpu-1gb-intel Basic Intel 1024 1 25 7.00 0.010420\n# s-1vcpu-2gb Basic 2048 1 50 12.00 0.017860\n# s-1vcpu-2gb-amd Basic AMD 2048 1 50 14.00 0.020830\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" size"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" list\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# Slug Description Memory VCPUs Disk Price Monthly Price Hourly\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-512mb-10gb Basic 512 1 10 4.00 0.005950\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-1gb Basic 1024 1 25 6.00 0.008930\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-1gb-amd Basic AMD 1024 1 25 7.00 0.010420\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-1gb-intel Basic Intel 1024 1 25 7.00 0.010420\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-2gb Basic 2048 1 50 12.00 0.017860\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# s-1vcpu-2gb-amd Basic AMD 2048 1 50 14.00 0.020830\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"I've also configured a few SSH keys with Digital Ocean, and I can have the key (specified by ID) provisioned to the machine using the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"--ssh-keys"}]},{"type":"text","value":" flag."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute ssh-key list\n\n# ID Name FingerPrint\n# 1234 mini \n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ssh-key"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" list\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# ID Name FingerPrint\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# 1234 mini \n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Also, I wanted to install a few packages to the box upon creation, this can be done easily with the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"--user-data-file"}]},{"type":"text","value":" flag to run an initialization script."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"echo 'apt install -y imagemagick zip' > bootstrap.sh\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"echo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 'apt install -y imagemagick zip'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" >"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" bootstrap.sh\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Putting it all together, here is the simple command for creating a small compute instance!"}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute droplet create \\\n --image ubuntu-22-10-x64 \\\n --size s-1vcpu-512mb-10gb \\\n --region nyc1 \\\n --ssh-keys 1234 \\\n --user-data-file boostrap.sh \\\n ephemeral\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" droplet"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" create"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --image"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ubuntu-22-10-x64"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --size"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" s-1vcpu-512mb-10gb"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --region"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" nyc1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --ssh-keys"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 1234"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --user-data-file"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" boostrap.sh"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ephemeral\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Finally, I can connect, do my thing, and destroy the instance."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute ssh ephemeral\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ssh"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ephemeral\n"}]}]}]}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"doctl compute droplet delete --force ephemeral\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"doctl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" compute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" droplet"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" delete"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --force"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ephemeral\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"All-in-all, I was up and running in about 20 minutes. What a handy utility!"}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:doctl.md","_source":"content","_file":"articles/doctl.md","_stem":"articles/doctl","_extension":"md"},{"_path":"/articles/fennel-initial-exploration","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Impressions of Fennel with Hammerspoon","description":"A while back I read an introductory book on Lisp programming titled, \"The Little\nSchemer\". This book opened my mind to new (to me) programming paradigms, and left me\nwith a strong desire to find a way to incorporate Lisp programming into my every day\nlife. It took some time, but I believe I've found what I've been looking for: Fennel.","draft":false,"date":"2023-10-22","tags":["lisp","hammerspoon","fennel"],"categories":["lisp"],"cover_image":"/images/dall-e-fennel-hammer.jpeg","excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"A while back I read an introductory book on Lisp programming titled, \"The Little\nSchemer\". This book opened my mind to new (to me) programming paradigms, and left me\nwith a strong desire to find a way to incorporate Lisp programming into my every day\nlife. It took some time, but I believe I've found what I've been looking for: Fennel."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"A while back I read an introductory book on Lisp programming titled, \"The Little\nSchemer\". This book opened my mind to new (to me) programming paradigms, and left me\nwith a strong desire to find a way to incorporate Lisp programming into my every day\nlife. It took some time, but I believe I've found what I've been looking for: Fennel."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The "},{"type":"element","tag":"a","props":{"href":"https://fennel-lang.org/","rel":["nofollow"]},"children":[{"type":"text","value":"Fennel"}]},{"type":"text","value":" programming language is a dialect of Lisp that boasts compatibility with\nLua, and it just so happens that two of my favorite applications are configured with\nexactly that language: "},{"type":"element","tag":"a","props":{"href":"https://www.hammerspoon.org/","rel":["nofollow"]},"children":[{"type":"text","value":"Hammerspoon"}]},{"type":"text","value":", and "},{"type":"element","tag":"a","props":{"href":"https://neovim.io/","rel":["nofollow"]},"children":[{"type":"text","value":"Neovim"}]},{"type":"text","value":"."}]},{"type":"element","tag":"h2","props":{"id":"initial-observations"},"children":[{"type":"text","value":"Initial Observations"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"To initially explore Fennel, I wanted to start small. My Hammerspoon configuration\nconsists of 7 "},{"type":"element","tag":"a","props":{"href":"https://github.com/cmpadden/dotfiles/tree/795749fa17e1310bb001bb7deaa22be8689f0027/hammerspoon/.hammerspoon/modules","rel":["nofollow"]},"children":[{"type":"text","value":"modules"}]},{"type":"text","value":" that I use for operations such as: launching applications,\nmanaging windows, keeping my computer from going to sleep, and general operating system\nautomation. So the plan is to translate these modules into Fennel, while maintaining\nwithout breaking the existing functionality. However, at this point, I wasn't even sure\nhow to embed Fennel into my project..."}]},{"type":"element","tag":"h2","props":{"id":"integrating-fennel-with-hammerspoon"},"children":[{"type":"text","value":"Integrating Fennel with Hammerspoon"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"While official documentation exists describing how to "},{"type":"element","tag":"a","props":{"href":"https://fennel-lang.org/setup#embedding-fennel","rel":["nofollow"]},"children":[{"type":"text","value":"embed fennel"}]},{"type":"text","value":" into your\nproject; it didn't provide me with enough clarity to know my next steps on integrating\nit with Hammerspoon. I found a few resources online demonstrating how to extend the\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"package.path"}]},{"type":"text","value":" and "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"package.cpath"}]},{"type":"text","value":" properties in Lua, but I was unable to get this to\nwork."}]},{"type":"element","tag":"ul","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://blog.exupero.org/hammerspoon-with-fennel/","rel":["nofollow"]},"children":[{"type":"text","value":"https://blog.exupero.org/hammerspoon-with-fennel/"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/Hammerspoon/hammerspoon/issues/2377#issuecomment-636331435","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/Hammerspoon/hammerspoon/issues/2377#issuecomment-636331435"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/agzam/spacehammer/blob/master/init.lua","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/agzam/spacehammer/blob/master/init.lua"}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Ultimately, I opted to include the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"fennel.lua"}]},{"type":"text","value":" file to my Hammerspoon configuration,\nand while not ideal, it does make the configuration nicely self-contained. I'll leave it\nas a future task to include the module installed with LuaRocks."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"With Fennel now included in my Hammerspoon configuration, all I need to do is configure\nthe "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"fennel.path"}]},{"type":"text","value":" to point to the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"*.fnl"}]},{"type":"text","value":" files in the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":".hammerspoon/"}]},{"type":"text","value":" directory, and\nttranslating these modules can begin!"}]},{"type":"element","tag":"pre","props":{"className":"language-lua shiki shiki-themes github-light","code":"-- init.lua\n\nlocal fennel = require('fennel')\n\nfennel.path = package.path .. \";\" .. os.getenv(\"HOME\") .. \"/.hammerspoon/?.fnl\"\n\ntable.insert(package.loaders or package.searchers, fennel.searcher)\n\nrequire 'main'\n","language":"lua","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"-- init.lua\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"local"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" fennel "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" require"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'fennel'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"fennel."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"path"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" package.path"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" .."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \";\" "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":".."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" os.getenv"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"HOME\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":".."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"/.hammerspoon/?.fnl\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"table.insert"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"package.loaders"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" or"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" package.searchers"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", fennel."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"searcher"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"require"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 'main'\n"}]}]}]}]},{"type":"element","tag":"h2","props":{"id":"translating-lua-to-fennel"},"children":[{"type":"text","value":"Translating Lua to Fennel"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"As a Fennel novice, I was happy to see that the Fennel project provides an online\ncross-compiler for Lua and Fennel called "},{"type":"element","tag":"a","props":{"href":"https://fennel-lang.org/see","rel":["nofollow"]},"children":[{"type":"text","value":"anti-fennel"}]},{"type":"text","value":", and while it can generate some\nstrange-looking Fennel code, it was an extremely useful tool for me to get\nup-and-running right away. For example, by pasting the simple "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"sleep"}]},{"type":"text","value":" function\nfrom the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"helpers"}]},{"type":"text","value":" module into the compiler:"}]},{"type":"element","tag":"pre","props":{"className":"language-lua shiki shiki-themes github-light","code":"function sleep(ms)\n os.execute(\"sleep \" .. tonumber(ms) / 1000)\nend\n","language":"lua","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"function"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" sleep"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(ms)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" os.execute"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"sleep \" "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":".."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" tonumber"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(ms) "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"/"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 1000"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"end\n"}]}]}]}]},{"type":"element","tag":"pre","props":{"className":"language-lisp shiki shiki-themes github-light","code":"(fn sleep [ms]\n (os.execute (.. \"sleep \" (/ (tonumber ms) 1000))))\n","language":"lisp","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(fn "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"sleep"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" [ms]\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (os.execute (.. "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"sleep \""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"/"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (tonumber ms) "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"1000"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"))))\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"As another example, here is the output for my "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"caffeine"}]},{"type":"text","value":" toggle:"}]},{"type":"element","tag":"pre","props":{"className":"language-lua shiki shiki-themes github-light","code":"hs.hotkey.bind(HYPER, \"0\", function()\n hs.caffeinate.toggle(\"displayIdle\")\n if hs.caffeinate.get(\"displayIdle\") then\n helpers:show(\"Caffeine Enabled\", nil, helpers.styles.success, helpers.assets.check)\n else\n helpers:show(\"Caffeine Disabled\", nil, helpers.styles.error, helpers.assets.ban)\n end\nend)\n","language":"lua","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"hs."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"hotkey"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"bind"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(HYPER, "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"0\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"function"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"()\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" hs."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"caffeinate"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"toggle"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"displayIdle\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" if"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" hs."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"caffeinate"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"get"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"displayIdle\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"then\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" helpers"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":":"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"show"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"Caffeine Enabled\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"nil"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", helpers."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"styles"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"success"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", helpers."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"assets"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"check"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" else\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" helpers"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":":"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"show"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"Caffeine Disabled\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"nil"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", helpers."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"styles"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"error"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", helpers."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"assets"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"ban"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" end\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"end"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]}]}]},{"type":"element","tag":"pre","props":{"className":"language-lisp shiki shiki-themes github-light","code":"(hs.hotkey.bind HYPER :0\n (fn [] (hs.caffeinate.toggle :displayIdle)\n (if (hs.caffeinate.get :displayIdle)\n (helpers:show \"Caffeine Enabled\" nil helpers.styles.success helpers.assets.check)\n (helpers:show \"Caffeine Disabled\" nil helpers.styles.error helpers.assets.ban)))) \n","language":"lisp","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(hs.hotkey.bind HYPER "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":":0\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (fn [] (hs.caffeinate.toggle "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":":displayIdle"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"if"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (hs.caffeinate.get "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":":displayIdle"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"helpers"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":":show "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"Caffeine Enabled\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" nil"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" helpers.styles.success helpers.assets.check)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"helpers"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":":show "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"Caffeine Disabled\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" nil"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" helpers.styles.error helpers.assets.ban))))\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"This was especially helpful for more gnarly modules like the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"window"}]},{"type":"text","value":" module used for\nwindow management, and seeing the Lua and Fennel code side-by-side was a kick starter in\nlearning the language!"}]},{"type":"element","tag":"h2","props":{"id":"next-steps"},"children":[{"type":"text","value":"Next Steps"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"While my Fennel Hammerspoon configuration now works with parity to its Lua counterpart,\nI have not yet added new features or modules. I look forward to writing new Fennel code,\nand deepen my understanding of Lisp and the Fennel programming language."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Additionally, before beginning this endeavor, I was already aware of projects like\n"},{"type":"element","tag":"a","props":{"href":"https://github.com/agzam/spacehammer","rel":["nofollow"]},"children":[{"type":"text","value":"spacehammer"}]},{"type":"text","value":"; a wildly impressive Hammerspoon configuration written in Fennel, but,\nI wanted to start small and learn the integration myself. However, with the basics out\nof the way, I hope to explore this project further, and seek lessons-learned for the\nconfiguration of my own."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The full pull-request for translating my Lua Hammerspoon configuration to Fennel can be\nfound here: "},{"type":"element","tag":"a","props":{"href":"https://github.com/cmpadden/dotfiles/pull/19/files","rel":["nofollow"]},"children":[{"type":"text","value":"https://github.com/cmpadden/dotfiles/pull/19/files"}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"initial-observations","depth":2,"text":"Initial Observations"},{"id":"integrating-fennel-with-hammerspoon","depth":2,"text":"Integrating Fennel with Hammerspoon"},{"id":"translating-lua-to-fennel","depth":2,"text":"Translating Lua to Fennel"},{"id":"next-steps","depth":2,"text":"Next Steps"}]}},"_type":"markdown","_id":"content:articles:fennel-initial-exploration.md","_source":"content","_file":"articles/fennel-initial-exploration.md","_stem":"articles/fennel-initial-exploration","_extension":"md"},{"_path":"/articles/migrate-truenas-from-core-to-scale","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Migrate to TrueNAS Scale from TrueNAS Core","description":"TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale.","date":"2021-12-28","tags":["homelab","supermicro","truenas"],"categories":["homelab"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Thankfully, the upgrade procedure to migrate from TrueNAS Core to TrueNAS Scale is relatively straight forward. All it requires is to create a bootable USB of the TrueNAS Scale image, boot the USB, and select "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Install/Upgrade"}]},{"type":"text","value":" in the installation wizard. But for the sake of being thorough, you can find instructions on how to backup system configurations and install the OS below."}]},{"type":"element","tag":"ol","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Make a backup of your system’s configuration\n"},{"type":"element","tag":"ol","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Navigate to "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"System > General"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Click "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Save Config"}]},{"type":"text","value":", check the "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Export Secret Seed"}]},{"type":"text","value":" box, and click "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Save"}]}]}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Export dataset keys for the encrypted pools\n"},{"type":"element","tag":"ol","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Navigate to "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Storage > Pools"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Click the cog icon, and select "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Export Dataset Keys"}]}]}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Insert the TrueNAS Core bootable USB into the NAS"}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"From the Supermicro IPMI interface select "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Remote Control"}]},{"type":"text","value":" and "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"iKVM/HTML5"}]},{"type":"text","value":" and select "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Reboot"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"Select the bootable USB as the boot device"}]},{"type":"element","tag":"li","props":{},"children":[{"type":"text","value":"From the TrueNAS installation wizard, select "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Install/Upgrade"}]},{"type":"text","value":", select the drive that contains the TrueNAS installation, and select "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Upgrade Install"}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Reboot the device, and voila — you should be up-and-running! Give the system a quick rundown to validate that your settings and pools have transferred correctly, and then enjoy all the container goodness!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"For a breakdown of the differences between TrueNAS Core, Enterprise, and Scale, you can reference "},{"type":"element","tag":"a","props":{"href":"https://www.truenas.com/help-me-choose/","rel":["nofollow"]},"children":[{"type":"text","value":"this table"}]},{"type":"text","value":"."}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:migrate-truenas-from-core-to-scale.md","_source":"content","_file":"articles/migrate-truenas-from-core-to-scale.md","_stem":"articles/migrate-truenas-from-core-to-scale","_extension":"md"},{"_path":"/articles/nuxt-content-rss-feed","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"How To Add an RSS Feed to a Nuxt Website","description":"If you are a user of Nuxt Content and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.","draft":false,"date":"2024-01-06","tags":["nuxt","rss"],"categories":["programming"],"cover_image":"/images/nuxt-content-rss-feed.jpg","excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you are a user of "},{"type":"element","tag":"a","props":{"href":"https://content.nuxt.com/","rel":["nofollow"]},"children":[{"type":"text","value":"Nuxt Content"}]},{"type":"text","value":" and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you are a user of "},{"type":"element","tag":"a","props":{"href":"https://content.nuxt.com/","rel":["nofollow"]},"children":[{"type":"text","value":"Nuxt Content"}]},{"type":"text","value":" and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application."}]},{"type":"element","tag":"h2","props":{"id":"preface"},"children":[{"type":"text","value":"Preface"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In version 2 of Nuxt, the community module, "},{"type":"element","tag":"a","props":{"href":"https://github.com/nuxt-community/feed-module","rel":["nofollow"]},"children":[{"type":"text","value":"nuxt-community/feed-module"}]},{"type":"text","value":" was a popular choice for adding an RSS feed to your website. However, there has been an unresolved "},{"type":"element","tag":"a","props":{"href":"https://github.com/nuxt-community/feed-module/issues/106","rel":["nofollow"]},"children":[{"type":"text","value":"open issue"}]},{"type":"text","value":" since April 1st, 2022 to add support for Nuxt v3. Thankfully, implementing this feature without a module is relatively straight forward."}]},{"type":"element","tag":"h2","props":{"id":"instructions"},"children":[{"type":"text","value":"Instructions"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First, install the "},{"type":"element","tag":"a","props":{"href":"https://www.npmjs.com/package/feed","rel":["nofollow"]},"children":[{"type":"text","value":"feed"}]},{"type":"text","value":" library into your project:"}]},{"type":"element","tag":"pre","props":{"className":"language-shell shiki shiki-themes github-light","code":"npm i -D feed\n","language":"shell","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"npm"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" i"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -D"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" feed\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, create a "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"server/"}]},{"type":"text","value":" folder in your project if it does not already exist, and create a file named "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"server/routes/atom.ts"}]},{"type":"text","value":"."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Here, we will leverage the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"feed"}]},{"type":"text","value":" library and construct an XML representation of our Nuxt content. As you can see, we first define our "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"feed"}]},{"type":"text","value":" with metadata associated with our RSS feed. This will be used by RSS readers to provide context to the end user. Then, we query our Nuxt content with "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"serverQueryContent"}]},{"type":"text","value":" and append a "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"feed.addItem"}]},{"type":"text","value":" for each article."}]},{"type":"element","tag":"pre","props":{"className":"language-ts shiki shiki-themes github-light","code":"import { serverQueryContent } from '#content/server';\nimport { Feed } from 'feed';\n\nconst BASE_URL = \"https://mywebsite.com\"\nconst AUTHOR_NAME = \"Firstname Lastname\"\n\nexport default defineEventHandler(async (event) => {\n\n const feed = new Feed({\n title: \"My Title\",\n description: \"My Description\",\n id: BASE_URL,\n link: BASE_URL,\n language: \"en\",\n image: `${BASE_URL}/images/placeholder.png`,\n favicon: `${BASE_URL}/favicon.ico`,\n copyright: `All rights reserved ${new Date().getFullYear()}, ${AUTHOR_NAME}`,\n updated: new Date(),\n generator: \"Nuxt static site generation + Feed for Node.js\",\n feedLinks: {\n atom: `${BASE_URL}/atom`\n },\n author: {\n name: AUTHOR_NAME,\n }\n });\n\n const articles = await serverQueryContent(event).find();\n\n articles.forEach((article) => {\n feed.addItem({\n title: article.title ? article.title : \"Missing Title\",\n id: article._path,\n link: `${BASE_URL}${article._path}`,\n description: article.description,\n author: [\n {\n name: AUTHOR_NAME,\n },\n ],\n date: new Date(article.date),\n image: article.cover_image ? `${BASE_URL}/${article.cover_image}` : undefined\n });\n });\n\n return feed.atom1();\n});\n","language":"ts","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" { serverQueryContent } "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" '#content/server'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":";\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" { Feed } "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 'feed'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":";\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"https://mywebsite.com\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" AUTHOR_NAME"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"Firstname Lastname\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"export"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" default"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" defineEventHandler"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"async"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#E36209"},"children":[{"type":"text","value":"event"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"=>"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" feed"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" new"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" Feed"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" title: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"My Title\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" description: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"My Description\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":12},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" id: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":13},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" link: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":14},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" language: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"en\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":15},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" image: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"`${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}/images/placeholder.png`"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":16},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" favicon: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"`${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}/favicon.ico`"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":17},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" copyright: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"`All rights reserved ${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"new"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" Date"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"()."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"getFullYear"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"()"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}, ${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"AUTHOR_NAME"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}`"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":18},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" updated: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"new"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" Date"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(),\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":19},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" generator: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"Nuxt static site generation + Feed for Node.js\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":20},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" feedLinks: {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":21},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" atom: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"`${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}/atom`\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":22},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" },\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":23},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" author: {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":24},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" name: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"AUTHOR_NAME"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":25},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" }\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":26},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":27},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":28},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" articles"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" await"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" serverQueryContent"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(event)."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"find"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"();\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":29},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":30},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" articles."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"forEach"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#E36209"},"children":[{"type":"text","value":"article"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"=>"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":31},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" feed."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"addItem"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":32},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" title: article.title "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"?"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" article.title "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":":"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"Missing Title\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":33},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" id: article._path,\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":34},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" link: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"`${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"article"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"_path"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}`"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":35},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" description: article.description,\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":36},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" author: [\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":37},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":38},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" name: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"AUTHOR_NAME"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":",\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":39},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" },\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":40},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ],\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":41},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" date: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"new"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" Date"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(article.date),\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":42},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" image: article.cover_image "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"?"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" `${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"BASE_URL"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}/${"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"article"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"cover_image"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"}`"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" :"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" undefined\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":43},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":44},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":45},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":46},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" return"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" feed."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"atom1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"();\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":47},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"});\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And that's just about it! Except, if you are statically generating your website with the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"nuxt generate"}]},{"type":"text","value":" command, you will need to configure this server-side route to be pre-rendered on site generation. This is as simple as adding a "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"nitro"}]},{"type":"text","value":" "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"prerender"}]},{"type":"text","value":" definition in your "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"nuxt.config.ts"}]},{"type":"text","value":" file, like so:"}]},{"type":"element","tag":"pre","props":{"className":"language-ts shiki shiki-themes github-light","code":"nitro: {\n prerender: {\n routes: ['/atom']\n }\n}\n","language":"ts","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"nitro"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" prerender"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" routes"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": ["}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'/atom'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" }\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"}\n"}]}]}]}]},{"type":"element","tag":"h2","props":{"id":"bonus"},"children":[{"type":"text","value":"Bonus"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"You may also be interested in adding a "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"sitemap.xml"}]},{"type":"text","value":" to your website. This can be done in almost an identical fashion!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Install the dependency:"}]},{"type":"element","tag":"pre","props":{"className":"language-shell shiki shiki-themes github-light","code":"npm i -D sitemap\n","language":"shell","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"npm"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" i"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -D"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" sitemap\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Create a route at "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"server/routes/sitemap.xml.ts"}]},{"type":"text","value":":"}]},{"type":"element","tag":"pre","props":{"className":"language-ts shiki shiki-themes github-light","code":"import { serverQueryContent } from '#content/server';\nimport { SitemapStream, streamToPromise } from 'sitemap';\n\nexport default defineEventHandler(async (event) => {\n const articles = await serverQueryContent(event).find();\n\n const sitemap = new SitemapStream({ hostname: 'https://my-website.com/' });\n\n // Add non nuxt content endpoints here\n sitemap.write({ url: '/' });\n sitemap.write({ url: '/articles' });\n\n // Dynamically generate routes for Nuxt markdown content\n articles.forEach((article) => sitemap.write({ url: article._path, changefreq: 'monthly' }));\n sitemap.end();\n\n return (await streamToPromise(sitemap));\n});\n","language":"ts","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" { serverQueryContent } "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" '#content/server'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":";\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" { SitemapStream, streamToPromise } "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 'sitemap'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":";\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"export"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" default"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" defineEventHandler"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"async"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#E36209"},"children":[{"type":"text","value":"event"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"=>"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" articles"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" await"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" serverQueryContent"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(event)."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"find"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"();\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" const"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" sitemap"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" ="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" new"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" SitemapStream"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({ hostname: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'https://my-website.com/'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":" // Add non nuxt content endpoints here\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" sitemap."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"write"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({ url: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'/'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" sitemap."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"write"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({ url: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'/articles'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" });\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":12},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":13},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":" // Dynamically generate routes for Nuxt markdown content\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":14},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" articles."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"forEach"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#E36209"},"children":[{"type":"text","value":"article"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"=>"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" sitemap."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"write"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"({ url: article._path, changefreq: "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'monthly'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" }));\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":15},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" sitemap."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"end"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"();\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":16},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":17},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" return"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" ("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"await"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" streamToPromise"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(sitemap));\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":18},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"});\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And add the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"prerender"}]},{"type":"text","value":" entry in your "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"nuxt.config.ts"}]},{"type":"text","value":":"}]},{"type":"element","tag":"pre","props":{"className":"language-ts shiki shiki-themes github-light","code":"nitro: {\n prerender: {\n routes: ['/sitemap.xml', '/atom']\n }\n}\n","language":"ts","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"nitro"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" prerender"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": {\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" routes"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":": ["}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'/sitemap.xml'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'/atom'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" }\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"}\n"}]}]}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"preface","depth":2,"text":"Preface"},{"id":"instructions","depth":2,"text":"Instructions"},{"id":"bonus","depth":2,"text":"Bonus"}]}},"_type":"markdown","_id":"content:articles:nuxt-content-rss-feed.md","_source":"content","_file":"articles/nuxt-content-rss-feed.md","_stem":"articles/nuxt-content-rss-feed","_extension":"md"},{"_path":"/articles/nuxt-v3-migration","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"This Website Has Been Migrated to Nuxt 3 🎉","description":"This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years.\nNot to mention, all of the great plugins in the Vue ecosystem!","date":"2022-12-31","tags":["nuxt"],"categories":["web"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years.\nNot to mention, all of the great plugins in the Vue ecosystem!"}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years.\nNot to mention, all of the great plugins in the Vue ecosystem!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you're curious what changes were required to make the migration, you can check out "},{"type":"element","tag":"a","props":{"href":"https://github.com/cmpadden/cmpadden.github.io/pull/3","rel":["nofollow"]},"children":[{"type":"text","value":"pull request #3"}]},{"type":"text","value":" in the GitHub repository."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"element","tag":"img","props":{"alt":"Screenshot of Nuxt Migration Pull Request","src":"/images/nuxt-migration-pr.png"},"children":[]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"While the documentation for making this migration is great, there were many breaking changes, and the overall process was quite tedious.\nFor this reason, I opted to generate a new project entirely, and port existing code to this clean slate.\nI believe that this resulted in a project with a bit less cruft."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The most valuable resources for making these changes include:"}]},{"type":"element","tag":"ul","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://nuxt.com/docs/migration/overview","rel":["nofollow"]},"children":[{"type":"text","value":"The Nuxt Migration Guide"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://nuxt.com/docs/getting-started/introduction","rel":["nofollow"]},"children":[{"type":"text","value":"The Nuxt Framework Guide"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://tailwindcss.nuxt.dev/","rel":["nofollow"]},"children":[{"type":"text","value":"Nuxt Tailwind Module Documentation"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://content.nuxtjs.org/","rel":["nofollow"]},"children":[{"type":"text","value":"Nuxt Content Module Documentation"}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Part of the delay for doing this upgrade was in waiting for module developers to support this major release.\nI'm super thankful for all of the hard work they've don, and I'm excited to explore all of the new features available!\nI just hope that the breaking changes in this release don't cause too much fracturing of the community, as it does feel like déjà vu of Python 2 and 3."}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:nuxt-v3-migration.md","_source":"content","_file":"articles/nuxt-v3-migration.md","_stem":"articles/nuxt-v3-migration","_extension":"md"},{"_path":"/articles/persistent-archlinux-usb","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Create a Persistent Arch Linux Bootable USB with Vagrant","description":"When installing a linux distribution, it is common for the instructions to have\nthe user create a bootable USB, boot from the device, and proceed with the\ninstallation procedure from that live medium. However, this blog post will\noutline an alternative approach where a virtual machine created with Vagrant\nwill be used in favor of the live medium.","date":"2020-01-09","draft":false,"tags":["vagrant","archlinux"],"categories":["linux"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"When installing a linux distribution, it is common for the instructions to have\nthe user create a bootable USB, boot from the device, and proceed with the\ninstallation procedure from that live medium. However, this blog post will\noutline an alternative approach where a virtual machine created with Vagrant\nwill be used in favor of the live medium."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"When installing a linux distribution, it is common for the instructions to have\nthe user create a bootable USB, boot from the device, and proceed with the\ninstallation procedure from that live medium. However, this blog post will\noutline an alternative approach where a virtual machine created with Vagrant\nwill be used in favor of the live medium."}]},{"type":"element","tag":"h1","props":{"id":"preface"},"children":[{"type":"text","value":"Preface"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The original intention was to use Docker for this process -- leveraging the\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"--device"}]},{"type":"text","value":" flag and mounting the target USB device in the Docker container,\nbut the underlying hypervisor in Docker Desktop for Mac does not support this.\n"},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://docs.docker.com/docker-for-mac/docker-toolbox/","rel":["nofollow"]},"children":[{"type":"text","value":"1"}]},{"type":"text","value":" "},{"type":"element","tag":"a","props":{"href":"https://github.com/moby/hyperkit","rel":["nofollow"]},"children":[{"type":"text","value":"2"}]},{"type":"text","value":" "},{"type":"element","tag":"a","props":{"href":"https://github.com/docker/for-mac/issues/900","rel":["nofollow"]},"children":[{"type":"text","value":"3"}]}]},{"type":"text","value":" While there are workarounds using Docker\nMachine, Vagrant felt like the path of least resistance."}]},{"type":"element","tag":"h1","props":{"id":"instructions"},"children":[{"type":"text","value":"Instructions"}]},{"type":"element","tag":"h2","props":{"id":"create-an-arch-linux-virtual-machine-with-vagrant"},"children":[{"type":"text","value":"Create an Arch Linux Virtual Machine with Vagrant"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Get the latest Arch Linux image "},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://app.vagrantup.com/archlinux/boxes/archlinux","rel":["nofollow"]},"children":[{"type":"text","value":"4"}]}]},{"type":"text","value":" from the Vagrant Cloud Box\nCatalog."}]},{"type":"element","tag":"pre","props":{"code":"vagrant box add archlinux/archlinux\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"vagrant"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" box"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" add"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" archlinux/archlinux\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Determine the USB vendor information for the thumb-drive that we will\npass-through to the virtual machine. Using the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"VBoxManage"}]},{"type":"text","value":" utility that comes\nwith Virtual Box, list the devices, and make note of the Vendor and Product ID."}]},{"type":"element","tag":"pre","props":{"code":" VBoxManage list usbhost\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" VBoxManage"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" list"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" usbhost\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Create a "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Vagrantfile"}]},{"type":"text","value":" with "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"archlinx/archlinux"}]},{"type":"text","value":" as the target box, and the USB\ndevice information that is passed through. "},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://antonyjepson.wordpress.com/2012/01/26/quickly-attaching-usb-devices-to-virtualbox-guests-using-vboxmanage/","rel":["nofollow"]},"children":[{"type":"text","value":"5"}]},{"type":"text","value":" "},{"type":"element","tag":"a","props":{"href":"https://gist.github.com/dscape/7d829c0c116ef419f963","rel":["nofollow"]},"children":[{"type":"text","value":"6"}]}]},{"type":"text","value":" Vagrant\noffers a handy customization parameter "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"vb.customize"}]},{"type":"text","value":" that calls the\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"VBoxManage"}]},{"type":"text","value":" command under-the-hood, allowing one to enable the guest machine\nto access the host machine's USB devices."}]},{"type":"element","tag":"pre","props":{"code":"# -*- mode: ruby -*-\n# vi: set ft=ruby :\n\nVagrant.configure(\"2\") do |config|\n config.vm.box = \"archlinux/archlinux\"\n config.vm.provider \"virtualbox\" do |vb|\n vb.name = \"archlinux\"\n vb.customize ['modifyvm', :id, '--usb', 'on']\n vb.customize ['usbfilter', 'add', '1', '--target', :id, '--name', 'SanDisk Ultra Fit', '--vendorid', '0x0781', '--productid', '0x5583']\n end\nend\n","language":"txt","meta":"","className":"language-txt shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"# -*- mode: ruby -*-\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"# vi: set ft=ruby :\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Vagrant.configure(\"2\") do |config|\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" config.vm.box = \"archlinux/archlinux\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" config.vm.provider \"virtualbox\" do |vb|\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" vb.name = \"archlinux\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" vb.customize ['modifyvm', :id, '--usb', 'on']\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" vb.customize ['usbfilter', 'add', '1', '--target', :id, '--name', 'SanDisk Ultra Fit', '--vendorid', '0x0781', '--productid', '0x5583']\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" end\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"end\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"When virtual machine is brought up, the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"usbfilter"}]},{"type":"text","value":" is applied, and the guest\nis able to access to the host machine's USB device that was specified in the\nfilter."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Start the machine, "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"ssh"}]},{"type":"text","value":" into the guest, and list the devices to confirm that\nthe USB device is available (see: "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"/dev/sdb"}]},{"type":"text","value":")."}]},{"type":"element","tag":"pre","props":{"code":"$ vagrant up\n$ vagrant ssh\n[vagrant@archlinux ~]$ lsblk\nNAME MAJ:MIN RM SIZE RO TYPE MOUNTPOINT\nsda 8:0 0 20G 0 disk\n├─sda1 8:1 0 1.9G 0 part [SWAP]\n└─sda2 8:2 0 18.1G 0 part /\nsdb 8:16 1 28.7G 0 disk\n└─sdb1 8:17 1 8G 0 part\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" vagrant"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" up\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" vagrant"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ssh\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[vagrant@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]$ lsblk\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"NAME"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" MAJ:MIN"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" RM"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" SIZE"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" RO"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" TYPE"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" MOUNTPOINT\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sda"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8:0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 20G"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" disk\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"├─sda1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8:1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 1.9G"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" part"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" [SWAP]\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"└─sda2"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8:2"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 18.1G"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" part"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" /\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sdb"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8:16"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 28.7G"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" disk\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"└─sdb1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8:17"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 8G"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 0"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" part\n"}]}]}]}]},{"type":"element","tag":"h2","props":{"id":"install-arch-linux-on-the-usb-drive"},"children":[{"type":"text","value":"Install Arch Linux on the USB Drive"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The "},{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/Installation_guide","rel":["nofollow"]},"children":[{"type":"text","value":"Arch Linux Installation Guide"}]},{"type":"text","value":" outlines the installation procedure in\ngreat detail -- the following steps follow this closely with a few alteration\ndue to installing onto removable media."}]},{"type":"element","tag":"h3","props":{"id":"partition-the-disk-uefi-with-gpt"},"children":[{"type":"text","value":"Partition the Disk (UEFI with GPT)"}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# fdisk /dev/sdb\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# fdisk /dev/sdb\n"}]}]}]}]},{"type":"element","tag":"pre","props":{"code":"Command (m for help): p\nDisk /dev/sdb: 28.66 GiB, 30752636928 bytes, 60063744 sectors\nDisk model: Ultra Fit\nUnits: sectors of 1 * 512 = 512 bytes\nSector size (logical/physical): 512 bytes / 512 bytes\nI/O size (minimum/optimal): 512 bytes / 512 bytes\nDisklabel type: gpt\nDisk identifier: E1D6C445-1B79-AB4D-A442-FA4AD6DF4ECC\n\nDevice Start End Sectors Size Type\n/dev/sdb1 2048 1050623 1048576 512M EFI System\n/dev/sdb2 1050624 60063710 59013087 28.1G Linux filesystem\n\nFilesystem/RAID signature on partition 1 will be wiped.\n","language":"txt","meta":"","className":"language-txt shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Command (m for help): p\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Disk /dev/sdb: 28.66 GiB, 30752636928 bytes, 60063744 sectors\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Disk model: Ultra Fit\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Units: sectors of 1 * 512 = 512 bytes\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Sector size (logical/physical): 512 bytes / 512 bytes\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"I/O size (minimum/optimal): 512 bytes / 512 bytes\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Disklabel type: gpt\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Disk identifier: E1D6C445-1B79-AB4D-A442-FA4AD6DF4ECC\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Device Start End Sectors Size Type\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"/dev/sdb1 2048 1050623 1048576 512M EFI System\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":12},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"/dev/sdb2 1050624 60063710 59013087 28.1G Linux filesystem\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":13},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":14},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Filesystem/RAID signature on partition 1 will be wiped.\n"}]}]}]}]},{"type":"element","tag":"h3","props":{"id":"format-the-partitions"},"children":[{"type":"text","value":"Format the Partitions"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The UEFI specification mandates support for FAT file-systems, and FAT32 is\nrecommended for removable media. "},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/EFI_system_partition#Format_the_partition","rel":["nofollow"]},"children":[{"type":"text","value":"7"}]}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# pacman -Sy dosfstools\n[root@archlinux ~]# mkfs.fat -F32 /dev/sdb1\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# pacman -Sy dosfstools\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# mkfs.fat -F32 /dev/sdb1\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"As for the root partition, it is recommended to use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"ext4"}]},{"type":"text","value":" without a journal to\nreduce the reads and writes to the file-system as this is detrimental to the\nflash-based USB drive. "},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/Install_Arch_Linux_on_a_USB_key#Installation_tweaks","rel":["nofollow"]},"children":[{"type":"text","value":"8"}]}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# mkfs.ext4 -O \"^has_journal\" /dev/sdb2\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# mkfs.ext4 -O "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"^has_journal\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" /dev/sdb2\n"}]}]}]}]},{"type":"element","tag":"h3","props":{"id":"mount-the-partitions-and-bootstrap-the-environment"},"children":[{"type":"text","value":"Mount the Partitions and Bootstrap the Environment"}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# mount /dev/sdb2 /mnt\n[root@archlinux ~]# mkdir -p /mnt/boot/efi\n[root@archlinux ~]# mount /dev/sdb1 /mnt/boot/efi\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# mount /dev/sdb2 /mnt\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# mkdir -p /mnt/boot/efi\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# mount /dev/sdb1 /mnt/boot/efi\n"}]}]}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# pacman -S arch-install-scripts\n[root@archlinux ~]# pacstrap /mnt base linux linux-firmware\n[root@archlinux ~]# genfstab -U /mnt >> /mnt/etc/fstab\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# pacman -S arch-install-scripts\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# pacstrap /mnt base linux linux-firmware\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[root@archlinux "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"~"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"]# genfstab -U /mnt "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":">>"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" /mnt/etc/fstab\n"}]}]}]}]},{"type":"element","tag":"h3","props":{"id":"configure-the-new-environment"},"children":[{"type":"text","value":"Configure the New Environment"}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux ~]# arch-chroot /mnt\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"[root@archlinux ~]# arch-chroot /mnt\n"}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux /]# ln -sf /usr/share/zoneinfo/US/Eastern /etc/localtime\n[root@archlinux /]# hwclock --systohc\n[root@archlinux /]# sed -i 's/#en_US.UTF-8/en_US.UTF-8/' /etc/locale.gen\n[root@archlinux /]# locale-gen\n[root@archlinux /]# echo \"LANG=en_US.UTF-8\" > /etc/locale.conf\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"[root@archlinux /]# ln -sf /usr/share/zoneinfo/US/Eastern /etc/localtime\n[root@archlinux /]# hwclock --systohc\n[root@archlinux /]# sed -i 's/#en_US.UTF-8/en_US.UTF-8/' /etc/locale.gen\n[root@archlinux /]# locale-gen\n[root@archlinux /]# echo \"LANG=en_US.UTF-8\" > /etc/locale.conf\n"}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux /]# hostnamectl set-hostname usb\n[root@archlinux /]# echo \"127.0.0.1 localhost\" >> /etc/hosts\n[root@archlinux /]# echo \"::1 localhost\" >> /etc/hosts\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"[root@archlinux /]# hostnamectl set-hostname usb\n[root@archlinux /]# echo \"127.0.0.1 localhost\" >> /etc/hosts\n[root@archlinux /]# echo \"::1 localhost\" >> /etc/hosts\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Note, one difference here from a standard installation is that the\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"--removable"}]},{"type":"text","value":" flag is specified when installing the GRUB bootloader.\n"},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/GRUB#UEFI_systems","rel":["nofollow"]},"children":[{"type":"text","value":"10"}]}]}]},{"type":"element","tag":"pre","props":{"code":"[root@archlinux /]# pacman -S grub\n[root@archlinux /]# grub-install --target=x86_64-efi --efi-directory=/boot/efi --removable --recheck\n[root@usb /]# grub-mkconfig -o /boot/grub/grub.cfg\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"[root@archlinux /]# pacman -S grub\n[root@archlinux /]# grub-install --target=x86_64-efi --efi-directory=/boot/efi --removable --recheck\n[root@usb /]# grub-mkconfig -o /boot/grub/grub.cfg\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Shutdown the virtual machine, restart the host machine, and boot the newly\ncreated Arch Linux thumb-drive!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"🎉"}]},{"type":"element","tag":"h2","props":{"id":"side-note"},"children":[{"type":"text","value":"Side-note"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"It was attempted to use the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"controlvm usbattach"}]},{"type":"text","value":" command to pass the USB\ndevice to the guest machine, but this did not work as it expects the virtual\nmachine to already be running, and the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"vb.customize"}]},{"type":"text","value":" option runs prior to\nbooting the machine. "},{"type":"element","tag":"sup","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://www.vagrantup.com/docs/virtualbox/configuration.html#vboxmanage-customizations","rel":["nofollow"]},"children":[{"type":"text","value":"11"}]}]}]},{"type":"element","tag":"pre","props":{"code":"Command: [\"controlvm\", \"060a716b-1965-49e2-bc56-12beed5df716\", \"usbattach36fc9e60-c465-11cf-8056-444553540000\"]\n\nStderr: VBoxManage.exe: error: Machine '060a716b-1965-49e2-bc56-12beed5df716' is not currently running.\n","language":"txt","meta":"","className":"language-txt shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Command: [\"controlvm\", \"060a716b-1965-49e2-bc56-12beed5df716\", \"usbattach36fc9e60-c465-11cf-8056-444553540000\"]\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"Stderr: VBoxManage.exe: error: Machine '060a716b-1965-49e2-bc56-12beed5df716' is not currently running.\n"}]}]}]}]},{"type":"element","tag":"h2","props":{"id":"references"},"children":[{"type":"text","value":"References"}]},{"type":"element","tag":"ol","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://docs.docker.com/docker-for-mac/docker-toolbox/","rel":["nofollow"]},"children":[{"type":"text","value":"Docker Desktop on Mac vs. Docker Toolbox"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/moby/hyperkit","rel":["nofollow"]},"children":[{"type":"text","value":"GitHub - HyperKit"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://github.com/docker/for-mac/issues/900","rel":["nofollow"]},"children":[{"type":"text","value":"GitHub - Docker for Mac - Issue #900"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://app.vagrantup.com/archlinux/boxes/archlinux","rel":["nofollow"]},"children":[{"type":"text","value":"Vagrant Cloud - Arch Linux"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://antonyjepson.wordpress.com/2012/01/26/quickly-attaching-usb-devices-to-virtualbox-guests-using-vboxmanage/","rel":["nofollow"]},"children":[{"type":"text","value":"Attaching USB Devices to VirtualBox Guests using VBoxManage"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://gist.github.com/dscape/7d829c0c116ef419f963","rel":["nofollow"]},"children":[{"type":"text","value":"GitHub Gist - Vagrant USB Filter"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/EFI_system_partition#Format_the_partition","rel":["nofollow"]},"children":[{"type":"text","value":"Arch Linux Wiki - EFI System Partition - Format Partitions"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/Install_Arch_Linux_on_a_USB_key#Installation_tweaks","rel":["nofollow"]},"children":[{"type":"text","value":"Arch Linux Wiki - Arch Linux on USB - Installation Tweaks"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/Installation_guide","rel":["nofollow"]},"children":[{"type":"text","value":"Arch Linux Wiki - Installation Guide"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://wiki.archlinux.org/index.php/GRUB#UEFI_systems","rel":["nofollow"]},"children":[{"type":"text","value":"Arch Linux Wiki - GRUB - UEFI Systems"}]}]},{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://www.vagrantup.com/docs/virtualbox/configuration.html#vboxmanage-customizations","rel":["nofollow"]},"children":[{"type":"text","value":"Vagrant VBoxManage Customizations "}]}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"create-an-arch-linux-virtual-machine-with-vagrant","depth":2,"text":"Create an Arch Linux Virtual Machine with Vagrant"},{"id":"install-arch-linux-on-the-usb-drive","depth":2,"text":"Install Arch Linux on the USB Drive","children":[{"id":"partition-the-disk-uefi-with-gpt","depth":3,"text":"Partition the Disk (UEFI with GPT)"},{"id":"format-the-partitions","depth":3,"text":"Format the Partitions"},{"id":"mount-the-partitions-and-bootstrap-the-environment","depth":3,"text":"Mount the Partitions and Bootstrap the Environment"},{"id":"configure-the-new-environment","depth":3,"text":"Configure the New Environment"}]},{"id":"side-note","depth":2,"text":"Side-note"},{"id":"references","depth":2,"text":"References"}]}},"_type":"markdown","_id":"content:articles:persistent-archlinux-usb.md","_source":"content","_file":"articles/persistent-archlinux-usb.md","_stem":"articles/persistent-archlinux-usb","_extension":"md"},{"_path":"/articles/podcast-transcription-whispercpp","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Easily Transcribe Podcasts with Whisper.cpp","description":"If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive whisper.cpp project. This high-performance fork of OpenAI's Whisper can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the Alter Everything podcast.","draft":false,"date":"2024-01-08","tags":["whisper.cpp","ml"],"categories":["programming"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive "},{"type":"element","tag":"a","props":{"href":"https://github.com/ggerganov/whisper.cpp","rel":["nofollow"]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":" project. This high-performance fork of "},{"type":"element","tag":"a","props":{"href":"https://github.com/openai/whisper","rel":["nofollow"]},"children":[{"type":"text","value":"OpenAI's Whisper"}]},{"type":"text","value":" can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the "},{"type":"element","tag":"a","props":{"href":"https://podcasts.apple.com/us/podcast/alter-everything/id1356137854","rel":["nofollow"]},"children":[{"type":"text","value":"Alter Everything"}]},{"type":"text","value":" podcast."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive "},{"type":"element","tag":"a","props":{"href":"https://github.com/ggerganov/whisper.cpp","rel":["nofollow"]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":" project. This high-performance fork of "},{"type":"element","tag":"a","props":{"href":"https://github.com/openai/whisper","rel":["nofollow"]},"children":[{"type":"text","value":"OpenAI's Whisper"}]},{"type":"text","value":" can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the "},{"type":"element","tag":"a","props":{"href":"https://podcasts.apple.com/us/podcast/alter-everything/id1356137854","rel":["nofollow"]},"children":[{"type":"text","value":"Alter Everything"}]},{"type":"text","value":" podcast."}]},{"type":"element","tag":"h2","props":{"id":"obtain-audio-files"},"children":[{"type":"text","value":"Obtain Audio File(s)"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First, let's get the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"wav"}]},{"type":"text","value":" file from YouTube using the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"youtube-dl"}]},{"type":"text","value":" utility. It should be noted that "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":" expects "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"wav"}]},{"type":"text","value":" filetypes, and this utility defaults to "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"mp3"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"code":" $ youtube-dl \\\n --extract-audio \\\n --audio-format wav \\\n --output podcast.wav \\\n \"https://www.youtube.com/watch?v=CoUN690wSYQ\"\n","language":"sh","meta":"","className":"language-sh shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" youtube-dl"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --extract-audio"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --audio-format"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" wav"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --output"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" podcast.wav"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"https://www.youtube.com/watch?v=CoUN690wSYQ\"\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"This file has a 44.1 kHz sample rate, and "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":" expects 16 kHz, so let's go ahead and convert that."}]},{"type":"element","tag":"pre","props":{"code":" $ file podcast.wav\npodcast.wav: RIFF (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 44100 Hz\n\n $ ffmpeg -i podcast.wav -ar 16000 podcast-16khz.wav\n\n $ file podcast-16khz.wav\npodcast-16khz.wav: RIFF (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 16000 Hz\n\n# NOTE: it looks like it's possible to specify this conversion as a post-process as a\n# flag to the `youtube-dl` command -- I will explore this further next time...\n# youtube-dl --extract-audio --audio-quality 0 --audio-format mp3 --postprocessor-args \"-ar 44100\" %dl%\n","language":"sh","meta":"","className":"language-sh shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" file"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" podcast.wav\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"podcast.wav:"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" RIFF"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 44100 Hz\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ffmpeg"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -i"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" podcast.wav"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -ar"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 16000"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" podcast-16khz.wav\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" file"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" podcast-16khz.wav\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"podcast-16khz.wav:"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" RIFF"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 16000 Hz\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# NOTE: it looks like it's possible to specify this conversion as a post-process as a\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# flag to the `youtube-dl` command -- I will explore this further next time...\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# youtube-dl --extract-audio --audio-quality 0 --audio-format mp3 --postprocessor-args \"-ar 44100\" %dl%\n"}]}]}]}]},{"type":"element","tag":"h2","props":{"id":"build-whispercpp-transcribe-audio"},"children":[{"type":"text","value":"Build "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":" & Transcribe Audio"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, let's get the latest version of "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"whisper.cpp"}]},{"type":"text","value":", download the English Whisper model, and build the example."}]},{"type":"element","tag":"pre","props":{"code":"# Clone the `whisper.cpp` repository\n $ git clone --depth 1 git@github.com:ggerganov/whisper.cpp && cd whisper.cpp\n\n# Download the English Whisper model in `ggml` format\n $ bash ./models/download-ggml-model.sh base.en\n\n# Build the main example\n $ make\n","language":"sh","meta":"","className":"language-sh shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# Clone the `whisper.cpp` repository\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" git"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" clone"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --depth"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" 1"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" git@github.com:ggerganov/whisper.cpp"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" && "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"cd"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" whisper.cpp\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# Download the English Whisper model in `ggml` format\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" bash"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ./models/download-ggml-model.sh"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" base.en\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# Build the main example\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" make\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And finally, let's transcribe that podcast!"}]},{"type":"element","tag":"pre","props":{"code":" $ ./main \\\n -m ~/workspace/whisper.cpp/models/ggml-base.en.bin \\\n -f ~/Downloads/podcast-16khz.wav \\\n --output-vtt \\\n --output-file out\n\n# whisper_print_timings: load time = 114.71 ms\n# whisper_print_timings: fallbacks = 0 p / 0 h\n# whisper_print_timings: mel time = 692.20 ms\n# whisper_print_timings: sample time = 22278.10 ms / 27893 runs ( 0.80 ms per run)\n# whisper_print_timings: encode time = 10000.75 ms / 55 runs ( 181.83 ms per run)\n# whisper_print_timings: decode time = 331.77 ms / 54 runs ( 6.14 ms per run)\n# whisper_print_timings: batchd time = 45236.73 ms / 27566 runs ( 1.64 ms per run)\n# whisper_print_timings: prompt time = 1921.90 ms / 11832 runs ( 0.16 ms per run)\n# whisper_print_timings: total time = 80709.54 ms\n","language":"sh","meta":"","className":"language-sh shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" $"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ./main"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -m"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ~/workspace/whisper.cpp/models/ggml-base.en.bin"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -f"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ~/Downloads/podcast-16khz.wav"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --output-vtt"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" \\\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --output-file"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" out\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: load time = 114.71 ms\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: fallbacks = 0 p / 0 h\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: mel time = 692.20 ms\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: sample time = 22278.10 ms / 27893 runs ( 0.80 ms per run)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":11},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: encode time = 10000.75 ms / 55 runs ( 181.83 ms per run)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":12},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: decode time = 331.77 ms / 54 runs ( 6.14 ms per run)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":13},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: batchd time = 45236.73 ms / 27566 runs ( 1.64 ms per run)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":14},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: prompt time = 1921.90 ms / 11832 runs ( 0.16 ms per run)\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":15},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# whisper_print_timings: total time = 80709.54 ms\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"A full podcast transcribed in ~80 seconds on an M1 Mac Mini -- not too bad!"}]},{"type":"element","tag":"pre","props":{"code":"# out.vtt\n\n00:00:00.000 --> 00:00:06.480\n >> Hi everyone. We recently launched a short engagement feedback survey for the Alter Everything\n\n00:00:06.480 --> 00:00:11.360\n podcast. Click the link in the episode description wherever you're listening to let us know what\n\n00:00:11.360 --> 00:00:16.320\n you think and help us improve our show.\n\n00:00:16.320 --> 00:00:21.200\n Welcome to Alter Everything, a podcast about data science and analytics culture. I'm Megan\n\n00:00:21.200 --> 00:00:26.440\n Dibble and today I'm talking with Nick Schrock, CTO and founder of Dagster Labs. We discussed\n\n00:00:26.440 --> 00:00:31.560\n data engineering trends, challenges in the field, why he started his company, and what\n\n00:00:31.560 --> 00:00:38.960\n makes him excited about the future of data engineering. Let's get started.\n\n00:00:38.960 --> 00:00:42.720\n >> Hi, Nick. It's great to have you on our show today. Thanks for being here.\n\n00:00:42.720 --> 00:00:43.920\n >> Thanks for having me.\n\n00:00:43.920 --> 00:00:48.280\n >> Yeah. Could you start off by giving an introduction to yourself for our listeners?\n\n00:00:48.280 --> 00:00:52.920\n >> Sure. My name is Nick Schrock. I'm the CTO and founder of Dagster Labs. There's the\n\n00:00:52.920 --> 00:00:59.520\n company behind Dagster, which is a data orchestration framework. Prior to doing this, I was an engineer\n\n00:00:59.520 --> 00:01:05.960\n at Facebook from 2009, 2017. While I was there, I found a team called product infrastructure\n\n00:01:05.960 --> 00:01:09.800\n whose goal was to make our application developers more efficient and productive, and a bunch\n\n00:01:09.800 --> 00:01:13.840\n of open source work came out of that actually, one of which was React, which I had nothing\n\n00:01:13.840 --> 00:01:18.040\n to do with, but actually the CEO of Dagster Labs co-created and I personally co-created\n\n00:01:18.040 --> 00:01:22.640\n GraphQL. So as I like to say, Pete and I were present at the creation of the full hipster\n\n00:01:22.640 --> 00:01:28.680\n stack. I moved on to Facebook in 2017, figuring out what to do next, and this data engineering\n\n00:01:28.680 --> 00:01:32.960\n and data orchestration problem really got me hooked actually quite soon after I left,\n\n00:01:32.960 --> 00:01:36.280\n and the rest is history. I'm sure we'll get into that more.\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"# out.vtt\n\n00:00:00.000 --> 00:00:06.480\n >> Hi everyone. We recently launched a short engagement feedback survey for the Alter Everything\n\n00:00:06.480 --> 00:00:11.360\n podcast. Click the link in the episode description wherever you're listening to let us know what\n\n00:00:11.360 --> 00:00:16.320\n you think and help us improve our show.\n\n00:00:16.320 --> 00:00:21.200\n Welcome to Alter Everything, a podcast about data science and analytics culture. I'm Megan\n\n00:00:21.200 --> 00:00:26.440\n Dibble and today I'm talking with Nick Schrock, CTO and founder of Dagster Labs. We discussed\n\n00:00:26.440 --> 00:00:31.560\n data engineering trends, challenges in the field, why he started his company, and what\n\n00:00:31.560 --> 00:00:38.960\n makes him excited about the future of data engineering. Let's get started.\n\n00:00:38.960 --> 00:00:42.720\n >> Hi, Nick. It's great to have you on our show today. Thanks for being here.\n\n00:00:42.720 --> 00:00:43.920\n >> Thanks for having me.\n\n00:00:43.920 --> 00:00:48.280\n >> Yeah. Could you start off by giving an introduction to yourself for our listeners?\n\n00:00:48.280 --> 00:00:52.920\n >> Sure. My name is Nick Schrock. I'm the CTO and founder of Dagster Labs. There's the\n\n00:00:52.920 --> 00:00:59.520\n company behind Dagster, which is a data orchestration framework. Prior to doing this, I was an engineer\n\n00:00:59.520 --> 00:01:05.960\n at Facebook from 2009, 2017. While I was there, I found a team called product infrastructure\n\n00:01:05.960 --> 00:01:09.800\n whose goal was to make our application developers more efficient and productive, and a bunch\n\n00:01:09.800 --> 00:01:13.840\n of open source work came out of that actually, one of which was React, which I had nothing\n\n00:01:13.840 --> 00:01:18.040\n to do with, but actually the CEO of Dagster Labs co-created and I personally co-created\n\n00:01:18.040 --> 00:01:22.640\n GraphQL. So as I like to say, Pete and I were present at the creation of the full hipster\n\n00:01:22.640 --> 00:01:28.680\n stack. I moved on to Facebook in 2017, figuring out what to do next, and this data engineering\n\n00:01:28.680 --> 00:01:32.960\n and data orchestration problem really got me hooked actually quite soon after I left,\n\n00:01:32.960 --> 00:01:36.280\n and the rest is history. I'm sure we'll get into that more.\n"}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"obtain-audio-files","depth":2,"text":"Obtain Audio File(s)"},{"id":"build-whispercpp-transcribe-audio","depth":2,"text":"Build whisper.cpp & Transcribe Audio"}]}},"_type":"markdown","_id":"content:articles:podcast-transcription-whispercpp.md","_source":"content","_file":"articles/podcast-transcription-whispercpp.md","_stem":"articles/podcast-transcription-whispercpp","_extension":"md"},{"_path":"/articles/quick-tip-rerunning-bash-commands","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Tip: Re-running Bash Commands","description":"Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use sudo for a command that requires\nroot privileges.","date":"2021-09-22","draft":false,"tags":["tip","bash"],"categories":["tips"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"sudo"}]},{"type":"text","value":" for a command that requires\nroot privileges."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"sudo"}]},{"type":"text","value":" for a command that requires\nroot privileges."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ pacman -Syu\nerror: you cannot perform this operation unless you are root.\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" pacman"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -Syu\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"error:"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" you"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" cannot"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" perform"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" this"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" operation"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" unless"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" you"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" are"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" root.\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Well, I have good news for you -- you can easily re-issue a command with the\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!!"}]},{"type":"text","value":" designator! Simply type "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"sudo"}]},{"type":"text","value":" followed by "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!!"}]},{"type":"text","value":" and you're good to go."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ sudo !!\nsudo pacman -Syu\n[sudo] password for colton:\n:: Synchronizing package databases...\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" sudo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" !!\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"sudo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" pacman"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -Syu\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"[sudo] password "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"for"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" colton:\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"::"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" Synchronizing"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" package"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" databases...\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"--"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Commands that are prefixed with a bang, "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!"}]},{"type":"text","value":", are considered "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Event\nDesignators,"}]},{"type":"text","value":" and are references to your command-line history. You can take a\nlook at your history with the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"history"}]},{"type":"text","value":" command."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ history\n 1021 touch hello_world.txt\n 1022 ls\n 1023 echo \"Here we go again!\"\n 1024 find . -name *.py\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" history\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" 1021"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" touch"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" hello_world.txt\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" 1022"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ls\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" 1023"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" echo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"Here we go again!\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" 1024"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" find"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -name"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" *"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":".py\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"There are many ways to use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!"}]},{"type":"text","value":" in your shell. For example, if you wanted to\nre-issue a specific command in your history, you could use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!n"}]},{"type":"text","value":" where "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"n"}]},{"type":"text","value":" is\nthe number next to the command in your history."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ !1023\necho \"Here we go again!\"\nHere we go again!\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" !1023\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"echo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"Here we go again!\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"Here"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" we"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" go"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" again!\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Or... if you wanted to run the command you issued 4-commands ago, you can use\n"},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!-4"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ !-4\nls\nhello_world.txt\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" !-4\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"ls\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"hello_world.txt\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Or... if you wanted to run the last command that started with the string\n"},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"find"}]},{"type":"text","value":", you can use "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"!find"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"className":"language-bash shiki shiki-themes github-light","code":"$ !find\nfind . -name *.py\n","language":"bash","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" !find\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"find"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -name"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" *"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":".py\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Be sure to check out the "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"Event Designators"}]},{"type":"text","value":" section of the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"bash"}]},{"type":"text","value":" "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"man"}]},{"type":"text","value":" pages\nfor more information!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"--"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"As an aside, for even faster command-line history navigation, be sure to check\nout the excellent "},{"type":"element","tag":"a","props":{"href":"https://github.com/junegunn/fzf","rel":["nofollow"]},"children":[{"type":"text","value":"fzf"}]},{"type":"text","value":" utility by "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"junegunn."}]},{"type":"text","value":"\nOne of the many features of "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"fzf"}]},{"type":"text","value":" is browsing and re-issuing commands from your\ncommand-line history with a fuzzy-finder!"}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:quick-tip-rerunning-bash-commands.md","_source":"content","_file":"articles/quick-tip-rerunning-bash-commands.md","_stem":"articles/quick-tip-rerunning-bash-commands","_extension":"md"},{"_path":"/articles/reset-ipmi-password-from-host-os","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Reset IPMI Credentials from the Host OS","description":"If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these\ninstructions will guide your through the process of resetting the credentials to their default values from the host\noperating system.","date":"2021-12-27","tags":["homelab","supermicro","truenas"],"categories":["homelab"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these\ninstructions will guide your through the process of resetting the credentials to their default values from the host\noperating system."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these\ninstructions will guide your through the process of resetting the credentials to their default values from the host\noperating system."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you are using an operating system like TrueNAS -- good news! It's possible to reset the IPMI password directly from\nthe web interface. This is done by navigating to "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"Network > IPMI"}]},{"type":"text","value":", and simply entering a new value in the "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"IPMI\nPassword Reset"}]},{"type":"text","value":" field."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If you are using some other OS that doesn't have this feature, you can achieve similar results by using the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"ipmitool"}]},{"type":"text","value":"\ncommand-line utility."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First, you'll want to determine the user ID associated with the user for whom you'd like to reset the password."}]},{"type":"element","tag":"pre","props":{"code":"# ipmitool user list\nID Name Callin Link Auth IPMI Msg Channel Priv Limit\n1 true false false Unknown (0x00)\n2 ADMIN true false false Unknown (0x00)\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"# ipmitool user list\nID Name Callin Link Auth IPMI Msg Channel Priv Limit\n1 true false false Unknown (0x00)\n2 ADMIN true false false Unknown (0x00)\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In this case, we will be resetting the password for "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"ADMIN"}]},{"type":"text","value":" who has a user ID of "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"2"}]},{"type":"text","value":". Then we'll assign the new\npassword like so:"}]},{"type":"element","tag":"pre","props":{"code":"# ipmitool user set password 2 \n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"# ipmitool user set password 2 \n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And you should be good to go!"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"..."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Alternatively, if you'd like to factory reset the baseboard management controller (BMC), which will reset the IPMI\ncredentials to their default value, you can issue the following command."}]},{"type":"element","tag":"pre","props":{"code":"# ipmitool raw 0x3c 0x40\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"# ipmitool raw 0x3c 0x40\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Where "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"0x3c"}]},{"type":"text","value":" is the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":""}]},{"type":"text","value":" argument, a.k.a. the network function code that defines the functional routing for\nmessages, and "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"0x40"}]},{"type":"text","value":" is the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":""}]},{"type":"text","value":". According to section 5.1 of the "},{"type":"element","tag":"a","props":{"href":"https://www.intel.com/content/dam/www/public/us/en/documents/product-briefs/ipmi-second-gen-interface-spec-v2-rev1-1.pdf","rel":["nofollow"]},"children":[{"type":"text","value":"IPMI interface\nspecification"}]},{"type":"text","value":",\n"},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"netfn"}]},{"type":"text","value":" codes ranging from 0x30 to 0x3F are reserved for vendor specific functions. I searched around for some\nSupermicro references on these vendor specific network functions without much luck other than various "},{"type":"element","tag":"a","props":{"href":"https://www.supermicro.com/support/faqs/faq.cfm?faq=15448","rel":["nofollow"]},"children":[{"type":"text","value":"support\nresponses"}]},{"type":"text","value":" on how to reset a device. Bummer!"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:reset-ipmi-password-from-host-os.md","_source":"content","_file":"articles/reset-ipmi-password-from-host-os.md","_stem":"articles/reset-ipmi-password-from-host-os","_extension":"md"},{"_path":"/articles/ssh-ed25519-sk-yubikey","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Configuring a YubiKey for use with OpenSSH","description":"YubiKey's are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" ed25519-sk key type that supports FIDO compliant hardware keys.","draft":false,"date":"2024-06-09","tags":["unix","configurations"],"categories":["tooling","tips"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://www.yubico.com/","rel":["nofollow"]},"children":[{"type":"text","value":"YubiKey's"}]},{"type":"text","value":" are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"ed25519-sk"}]},{"type":"text","value":" key type that supports FIDO compliant hardware keys."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://www.yubico.com/","rel":["nofollow"]},"children":[{"type":"text","value":"YubiKey's"}]},{"type":"text","value":" are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"ed25519-sk"}]},{"type":"text","value":" key type that supports FIDO compliant hardware keys."}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In "},{"type":"element","tag":"a","props":{"href":"https://www.openssh.com/txt/release-8.2","rel":["nofollow"]},"children":[{"type":"text","value":"release 8.2 of OpenSSH"}]},{"type":"text","value":" support for FIDO devices was added with public key types \"ecdsa-sk\" and \"ed25519-sk\" (-sk standing for \"security key\"). This key type is supported by YubiKey's with firmware version 5.2.3 or higher."}]},{"type":"element","tag":"blockquote","props":{},"children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"This release adds support for FIDO/U2F hardware authenticators to OpenSSH. U2F/FIDO are open standards for inexpensive two-factor authentication hardware that are widely used for website authentication. In OpenSSH FIDO devices are supported by new public key types \"ecdsa-sk\" and \"ed25519-sk\", along with corresponding certificate types."}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Let's get started by installing the latest version of OpenSSH via "},{"type":"element","tag":"a","props":{"href":"https://brew.sh/","rel":["nofollow"]},"children":[{"type":"text","value":"Homebrew"}]},{"type":"text","value":", along with the YubiKey Manager (ykman) CLI. The version of OpenSSH included with macOS is not compatible."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"$ brew install openssh ykman\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" brew"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" install"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" openssh"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ykman\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, let's confirm that our YubiKey has a firmware that is greater than 5.2.3:"}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"$ ykman list\nYubiKey 5Ci (5.4.3) [OTP+FIDO+CCID]\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ykman"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" list\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"YubiKey"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" 5Ci"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" (5.4.3) [OTP+FIDO+CCID]\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Next, we'll go ahead and enable a pin on our device via the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"change-pin"}]},{"type":"text","value":" command, as this a requirement for our use."}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"$ ykman fido access change-pin\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ykman"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" fido"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" access"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" change-pin\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And last, we'll generate the key on our device!"}]},{"type":"element","tag":"pre","props":{"className":"language-sh shiki shiki-themes github-light","code":"$ ssh-keygen -t ed25519-sk -O resident\nGenerating public/private ed25519-sk key pair.\nYou may need to touch your authenticator to authorize key generation.\n...\n","language":"sh","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ssh-keygen"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -t"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ed25519-sk"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" -O"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" resident\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"Generating"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" public/private"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" ed25519-sk"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" key"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" pair.\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"You"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" may"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" need"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" to"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" touch"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" your"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" authenticator"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" to"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" authorize"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" key"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" generation.\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"...\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"We specify "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"resident"}]},{"type":"text","value":" to indicate that the key handle is to be stored on the YubiKey itself, since we will be using this device with multiple computers."}]},{"type":"element","tag":"pre","props":{"className":"language-txt shiki shiki-themes github-light","code":"resident\n Indicate that the key handle should be stored on the FIDO\n authenticator itself. This makes it easier to use the\n authenticator on multiple computers. Resident keys may be\n supported on FIDO2 authenticators and typically require that a PIN\n be set on the authenticator prior to generation. Resident keys\n may be loaded off the authenticator using ssh-add(1). Storing\n both parts of a key on a FIDO authenticator increases the\n likelihood of an attacker being able to use a stolen authenticator\n device.\n","language":"txt","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":"resident\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" Indicate that the key handle should be stored on the FIDO\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" authenticator itself. This makes it easier to use the\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" authenticator on multiple computers. Resident keys may be\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" supported on FIDO2 authenticators and typically require that a PIN\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" be set on the authenticator prior to generation. Resident keys\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" may be loaded off the authenticator using ssh-add(1). Storing\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" both parts of a key on a FIDO authenticator increases the\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" likelihood of an attacker being able to use a stolen authenticator\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{},"children":[{"type":"text","value":" device.\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"And that's all it takes -- simple enough. Now, when interacting with "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"ssh"}]},{"type":"text","value":" or "},{"type":"element","tag":"em","props":{},"children":[{"type":"text","value":"git"}]},{"type":"text","value":" you will be prompted to touch the YubiKey to bring that little bit of physical 2FA."}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[]}},"_type":"markdown","_id":"content:articles:ssh-ed25519-sk-yubikey.md","_source":"content","_file":"articles/ssh-ed25519-sk-yubikey.md","_stem":"articles/ssh-ed25519-sk-yubikey","_extension":"md"},{"_path":"/articles/unit-testing-micropython-with-mocks","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Unit Testing in MicroPython with Mocks","description":"Unit testing code for embedded systems can be challenging. While it's possible\nto leverage emulators, write side-effect free code, or run tests on the\nhardware itself, it's often easiest to unit test the code on your personal\ncomputer with mocked hardware functionality.","date":"2020-02-07","draft":false,"tags":["micropython","testing","mocks","tutorial"],"categories":["python","embedded"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Unit testing code for embedded systems can be challenging. While it's possible\nto leverage emulators, write side-effect free code, or run tests on the\nhardware itself, it's often easiest to unit test the code on your personal\ncomputer with mocked hardware functionality."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Unit testing code for embedded systems can be challenging. While it's possible\nto leverage emulators, write side-effect free code, or run tests on the\nhardware itself, it's often easiest to unit test the code on your personal\ncomputer with mocked hardware functionality."}]},{"type":"element","tag":"h1","props":{"id":"mocking"},"children":[{"type":"text","value":"Mocking"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Mocks allow us to replace the hardware interfacing functionality under-the-hood\nwith predefined results and side-effects. For example, if there is a piece of\nlogic that retrieves values from an accelerometer to get a device's\norientation, it would be possible to mock the returned values of the\naccelerometer -- allowing us to run the unit tests on a device that does not\nhave an accelerometer sensor installed."}]},{"type":"element","tag":"h1","props":{"id":"a-micropython-mocking-example"},"children":[{"type":"text","value":"A MicroPython Mocking Example"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In this example, we will be unit testing a module named "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"time_logger"}]},{"type":"text","value":", that\ndepends on the MicroPython library "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"utime"}]},{"type":"text","value":" to log the most recent Epoch time to\na file."}]},{"type":"element","tag":"pre","props":{"className":"language-python shiki shiki-themes github-light","code":"# time_logger.py\n\nclass TimeLogger(object):\n\n def save_time(self):\n \"\"\" Overwrite a file with the most recent Epoch timestamp from `utime`\n \"\"\"\n with open(\"LAST_KNOWN_TIME\", \"w+\") as f:\n f.write(str(utime.time()))\n","language":"python","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# time_logger.py\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"class"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" TimeLogger"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"object"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"):\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" def"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" save_time"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(self):\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"\"\" Overwrite a file with the most recent Epoch timestamp from `utime`\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"\"\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" with"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" open"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"LAST_KNOWN_TIME\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"w+\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"as"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" f:\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" f.write("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"str"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(utime.time()))\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First, because the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"utime"}]},{"type":"text","value":" module is not installed on the machine that the unit\ntests on, we must mock "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"utime"}]},{"type":"text","value":" module before importing "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"time_logger"}]},{"type":"text","value":" in our\nunit test file."}]},{"type":"element","tag":"pre","props":{"className":"language-python shiki shiki-themes github-light","code":"# test_time_logger.py\n\nimport unittest\n\nfrom unittest.mock import MagicMock\n\nsys.modules['utime'] = MagicMock()\nfrom time_logger import TimeLogger\n","language":"python","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6A737D"},"children":[{"type":"text","value":"# test_time_logger.py\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" unittest\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" unittest.mock "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" MagicMock\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"sys.modules["}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"'utime'"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"] "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" MagicMock()\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"from"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" time_logger "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"import"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" TimeLogger\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, we can write a test that patches the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"utime.time"}]},{"type":"text","value":" functionality so that\nit returns a value of our choosing -- in this case, "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"1234"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"className":"language-python shiki shiki-themes github-light","code":"class TestTimeLogger(unittest.TestCase):\n\n def test_save_time(self):\n \"\"\" Verify that the Epoch time is written to file\n \"\"\"\n with unittest.mock.patch(\"utime.time\", return_value=1234):\n t = TimeLogger()\n t.save_time()\n with open(\"LAST_KNOWN_TIME\") as f:\n self.assertEqual(\"1234\", f.read())\n","language":"python","meta":"","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"class"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" TestTimeLogger"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"unittest"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"."}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"TestCase"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"):\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"emptyLinePlaceholder":true},"children":[{"type":"text","value":"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":3},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" def"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":" test_save_time"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"(self):\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":4},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"\"\" Verify that the Epoch time is written to file\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":5},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" \"\"\"\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":6},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" with"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" unittest.mock.patch("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"utime.time\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#E36209"},"children":[{"type":"text","value":"return_value"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":"1234"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"):\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":7},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" t "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"="}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" TimeLogger()\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":8},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" t.save_time()\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":9},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":" with"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" open"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":"("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"LAST_KNOWN_TIME\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":") "}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#D73A49"},"children":[{"type":"text","value":"as"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" f:\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":10},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" self"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":".assertEqual("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"\"1234\""}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":", f.read())\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Now, when the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"save_time"}]},{"type":"text","value":" method gets the latest time from "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"utime.time()"}]},{"type":"text","value":", the\nvalue will be patched to return "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"1234"}]},{"type":"text","value":". That value will be written to a file,\nand our unit test will pass!"}]},{"type":"element","tag":"h2","props":{"id":"references"},"children":[{"type":"text","value":"References"}]},{"type":"element","tag":"ol","props":{},"children":[{"type":"element","tag":"li","props":{},"children":[{"type":"element","tag":"a","props":{"href":"https://docs.python.org/3/library/unittest.html","rel":["nofollow"]},"children":[{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"unittest"}]},{"type":"text","value":" — Unit testing framework"}]}]}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"references","depth":2,"text":"References"}]}},"_type":"markdown","_id":"content:articles:unit-testing-micropython-with-mocks.md","_source":"content","_file":"articles/unit-testing-micropython-with-mocks.md","_stem":"articles/unit-testing-micropython-with-mocks","_extension":"md"},{"_path":"/articles/vim-fugitive-gpg-pinentry","_dir":"articles","_draft":false,"_partial":false,"_locale":"","title":"Using pinentry-mac to sign commits from vim-fugitive","description":"In order to sign git commits from within Vim using a plugin like tpope/vim-fugitive, it is necessary to configure the gpg-agent to use a GUI based pinentry-program.","draft":false,"date":"2024-05-11","tags":["vim","tip"],"categories":["tooling","tips"],"excerpt":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In order to sign git commits from within Vim using a plugin like "},{"type":"element","tag":"a","props":{"href":"https://github.com/tpope/vim-fugitive","rel":["nofollow"]},"children":[{"type":"text","value":"tpope/vim-fugitive"}]},{"type":"text","value":", it is necessary to configure the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"gpg-agent"}]},{"type":"text","value":" to use a GUI based "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"pinentry-program"}]},{"type":"text","value":"."}]}]},"body":{"type":"root","children":[{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"In order to sign git commits from within Vim using a plugin like "},{"type":"element","tag":"a","props":{"href":"https://github.com/tpope/vim-fugitive","rel":["nofollow"]},"children":[{"type":"text","value":"tpope/vim-fugitive"}]},{"type":"text","value":", it is necessary to configure the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"gpg-agent"}]},{"type":"text","value":" to use a GUI based "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"pinentry-program"}]},{"type":"text","value":"."}]},{"type":"element","tag":"h2","props":{"id":"preface"},"children":[{"type":"text","value":"Preface"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"The man, the myth, the legend, Timothy Popallopollis himself "},{"type":"element","tag":"a","props":{"href":"https://github.com/tpope/vim-fugitive/issues/846#issuecomment-253816577","rel":["nofollow"]},"children":[{"type":"text","value":"recommends"}]},{"type":"text","value":" configuring your "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"gpg-agent"}]},{"type":"text","value":" to use a GUI based "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"pinentry-program"}]},{"type":"text","value":". On macOS this can be done quite by simply installing "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"pinentry-mac"}]},{"type":"text","value":", and updating your "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"gpg-agent.conf"}]},{"type":"text","value":"."}]},{"type":"element","tag":"h2","props":{"id":"configuration"},"children":[{"type":"text","value":"Configuration"}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"First things first, let's install the pinentry program."}]},{"type":"element","tag":"pre","props":{"code":"$ brew install pinentry-mac\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" brew"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" install"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" pinentry-mac\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Then, all we need to do is set the "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"pinentry-program"}]},{"type":"text","value":" option in your "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"~/.gnupg/gpg-agent.conf"}]},{"type":"text","value":" file."}]},{"type":"element","tag":"pre","props":{"code":"default-cache-ttl 600\nmax-cache-ttl 7200\npinentry-program /opt/homebrew/bin/pinentry-mac\n"},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"text","value":"default-cache-ttl 600\nmax-cache-ttl 7200\npinentry-program /opt/homebrew/bin/pinentry-mac\n"}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"If your don't know the path to your pinentry program, you can throw down a quick "},{"type":"element","tag":"code","props":{"className":[]},"children":[{"type":"text","value":"which"}]},{"type":"text","value":"."}]},{"type":"element","tag":"pre","props":{"code":"$ which pinentry-mac\n/opt/homebrew/bin/pinentry-mac\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" which"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" pinentry-mac\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"/opt/homebrew/bin/pinentry-mac\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"Or use your Homebrew prefix."}]},{"type":"element","tag":"pre","props":{"code":"$ echo $(brew --prefix)/bin/pinentry-mac\n/opt/homebrew/bin/pinentry-mac\n","language":"bash","meta":"","className":"language-bash shiki shiki-themes github-light","style":""},"children":[{"type":"element","tag":"code","props":{"__ignoreMap":""},"children":[{"type":"element","tag":"span","props":{"class":"line","line":1},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"$"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":" echo"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":" $("}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"brew"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#005CC5"},"children":[{"type":"text","value":" --prefix"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#24292E"},"children":[{"type":"text","value":")"}]},{"type":"element","tag":"span","props":{"style":"--shiki-default:#032F62"},"children":[{"type":"text","value":"/bin/pinentry-mac\n"}]}]},{"type":"element","tag":"span","props":{"class":"line","line":2},"children":[{"type":"element","tag":"span","props":{"style":"--shiki-default:#6F42C1"},"children":[{"type":"text","value":"/opt/homebrew/bin/pinentry-mac\n"}]}]}]}]},{"type":"element","tag":"p","props":{},"children":[{"type":"text","value":"But that's all it takes. Now, you should be prompted to enter your gpg pin in an external window when signing commits from vim."}]},{"type":"element","tag":"style","props":{},"children":[{"type":"text","value":"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}"}]}],"toc":{"title":"","searchDepth":2,"depth":2,"links":[{"id":"preface","depth":2,"text":"Preface"},{"id":"configuration","depth":2,"text":"Configuration"}]}},"_type":"markdown","_id":"content:articles:vim-fugitive-gpg-pinentry.md","_source":"content","_file":"articles/vim-fugitive-gpg-pinentry.md","_stem":"articles/vim-fugitive-gpg-pinentry","_extension":"md"}],"navigation":[{"title":"Articles","_path":"/articles","children":[{"title":"Upgrading the Firmware on the PCEngines APU2","_path":"/articles/apu2-firmware-upgrade"},{"title":"Docker Volume Permissions with SELinux","_path":"/articles/docker-selinux-volumes"},{"title":"Exploring the Digital Ocean `doctl` Utility","_path":"/articles/doctl"},{"title":"Impressions of Fennel with Hammerspoon","_path":"/articles/fennel-initial-exploration"},{"title":"Migrate to TrueNAS Scale from TrueNAS Core","_path":"/articles/migrate-truenas-from-core-to-scale"},{"title":"How To Add an RSS Feed to a Nuxt Website","_path":"/articles/nuxt-content-rss-feed"},{"title":"This Website Has Been Migrated to Nuxt 3 🎉","_path":"/articles/nuxt-v3-migration"},{"title":"Create a Persistent Arch Linux Bootable USB with Vagrant","_path":"/articles/persistent-archlinux-usb"},{"title":"Easily Transcribe Podcasts with Whisper.cpp","_path":"/articles/podcast-transcription-whispercpp"},{"title":"Tip: Re-running Bash Commands","_path":"/articles/quick-tip-rerunning-bash-commands"},{"title":"Reset IPMI Credentials from the Host OS","_path":"/articles/reset-ipmi-password-from-host-os"},{"title":"Configuring a YubiKey for use with OpenSSH","_path":"/articles/ssh-ed25519-sk-yubikey"},{"title":"Unit Testing in MicroPython with Mocks","_path":"/articles/unit-testing-micropython-with-mocks"},{"title":"Using pinentry-mac to sign commits from vim-fugitive","_path":"/articles/vim-fugitive-gpg-pinentry"}]}]} \ No newline at end of file diff --git a/api/_content/query/6WfgQ5T9tH.1731781838504.json b/api/_content/query/6WfgQ5T9tH.1731782019206.json similarity index 100% rename from api/_content/query/6WfgQ5T9tH.1731781838504.json rename to api/_content/query/6WfgQ5T9tH.1731782019206.json diff --git a/api/_content/query/7TfxHWYxZH.1731781838504.json b/api/_content/query/7TfxHWYxZH.1731782019206.json similarity index 100% rename from api/_content/query/7TfxHWYxZH.1731781838504.json rename to api/_content/query/7TfxHWYxZH.1731782019206.json diff --git a/api/_content/query/98CVAb0zLR.1731781838504.json b/api/_content/query/98CVAb0zLR.1731782019206.json similarity index 100% rename from api/_content/query/98CVAb0zLR.1731781838504.json rename to api/_content/query/98CVAb0zLR.1731782019206.json diff --git a/api/_content/query/DIau8q3IMV.1731781838504.json b/api/_content/query/DIau8q3IMV.1731782019206.json similarity index 100% rename from api/_content/query/DIau8q3IMV.1731781838504.json rename to api/_content/query/DIau8q3IMV.1731782019206.json diff --git a/api/_content/query/LcWrOc5HNX.1731781838504.json b/api/_content/query/LcWrOc5HNX.1731782019206.json similarity index 100% rename from api/_content/query/LcWrOc5HNX.1731781838504.json rename to api/_content/query/LcWrOc5HNX.1731782019206.json diff --git a/api/_content/query/Nr5UObwduV.1731781838504.json b/api/_content/query/Nr5UObwduV.1731782019206.json similarity index 100% rename from api/_content/query/Nr5UObwduV.1731781838504.json rename to api/_content/query/Nr5UObwduV.1731782019206.json diff --git a/api/_content/query/QmL7G3Pk7i.1731781838504.json b/api/_content/query/QmL7G3Pk7i.1731782019206.json similarity index 100% rename from api/_content/query/QmL7G3Pk7i.1731781838504.json rename to api/_content/query/QmL7G3Pk7i.1731782019206.json diff --git a/api/_content/query/TDaCLaQ73L.1731781838504.json b/api/_content/query/TDaCLaQ73L.1731782019206.json similarity index 100% rename from api/_content/query/TDaCLaQ73L.1731781838504.json rename to api/_content/query/TDaCLaQ73L.1731782019206.json diff --git a/api/_content/query/XgcK3x9EBy.1731781838504.json b/api/_content/query/XgcK3x9EBy.1731782019206.json similarity index 100% rename from api/_content/query/XgcK3x9EBy.1731781838504.json rename to api/_content/query/XgcK3x9EBy.1731782019206.json diff --git a/api/_content/query/ZzD9WRl1Uk.1731781838504.json b/api/_content/query/ZzD9WRl1Uk.1731782019206.json similarity index 100% rename from api/_content/query/ZzD9WRl1Uk.1731781838504.json rename to api/_content/query/ZzD9WRl1Uk.1731782019206.json diff --git a/api/_content/query/bXj5vj6Ts0.1731781838504.json b/api/_content/query/bXj5vj6Ts0.1731782019206.json similarity index 100% rename from api/_content/query/bXj5vj6Ts0.1731781838504.json rename to api/_content/query/bXj5vj6Ts0.1731782019206.json diff --git a/api/_content/query/d7v45RMayO.1731781838504.json b/api/_content/query/d7v45RMayO.1731782019206.json similarity index 100% rename from api/_content/query/d7v45RMayO.1731781838504.json rename to api/_content/query/d7v45RMayO.1731782019206.json diff --git a/api/_content/query/pfbAdBSC9a.1731781838504.json b/api/_content/query/pfbAdBSC9a.1731782019206.json similarity index 100% rename from api/_content/query/pfbAdBSC9a.1731781838504.json rename to api/_content/query/pfbAdBSC9a.1731782019206.json diff --git a/api/_content/query/tGrf8kFZOz.1731781838504.json b/api/_content/query/tGrf8kFZOz.1731782019206.json similarity index 100% rename from api/_content/query/tGrf8kFZOz.1731781838504.json rename to api/_content/query/tGrf8kFZOz.1731782019206.json diff --git a/api/_content/query/ucEXmLbw2Z.1731781838504.json b/api/_content/query/ucEXmLbw2Z.1731782019206.json similarity index 100% rename from api/_content/query/ucEXmLbw2Z.1731781838504.json rename to api/_content/query/ucEXmLbw2Z.1731782019206.json diff --git a/api/_content/query/v3HQ7aAWkW.1731781838504.json b/api/_content/query/v3HQ7aAWkW.1731782019206.json similarity index 100% rename from api/_content/query/v3HQ7aAWkW.1731781838504.json rename to api/_content/query/v3HQ7aAWkW.1731782019206.json diff --git a/articles/_payload.json b/articles/_payload.json index 2cec598a..0b8d1a54 100644 --- a/articles/_payload.json +++ b/articles/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":360},["ShallowReactive",2],{"articles":3},[4,45,87,131,157,174,207,222,239,253,277,297,313,329],{"_path":5,"title":6,"description":7,"date":8,"tags":9,"categories":12,"excerpt":15,"_id":44},"/articles/ssh-ed25519-sk-yubikey","Configuring a YubiKey for use with OpenSSH","YubiKey's are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" ed25519-sk key type that supports FIDO compliant hardware keys.","2024-06-09",[10,11],"unix","configurations",[13,14],"tooling","tips",{"type":16,"children":17},"root",[18],{"type":19,"tag":20,"props":21,"children":22},"element","p",{},[23,33,35,42],{"type":19,"tag":24,"props":25,"children":29},"a",{"href":26,"rel":27},"https://www.yubico.com/",[28],"nofollow",[30],{"type":31,"value":32},"text","YubiKey's",{"type":31,"value":34}," are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" ",{"type":19,"tag":36,"props":37,"children":39},"code",{"className":38},[],[40],{"type":31,"value":41},"ed25519-sk",{"type":31,"value":43}," key type that supports FIDO compliant hardware keys.","content:articles:ssh-ed25519-sk-yubikey.md",{"_path":46,"title":47,"description":48,"date":49,"tags":50,"categories":53,"excerpt":54,"_id":86},"/articles/vim-fugitive-gpg-pinentry","Using pinentry-mac to sign commits from vim-fugitive","In order to sign git commits from within Vim using a plugin like tpope/vim-fugitive, it is necessary to configure the gpg-agent to use a GUI based pinentry-program.","2024-05-11",[51,52],"vim","tip",[13,14],{"type":16,"children":55},[56],{"type":19,"tag":20,"props":57,"children":58},{},[59,61,68,70,76,78,84],{"type":31,"value":60},"In order to sign git commits from within Vim using a plugin like ",{"type":19,"tag":24,"props":62,"children":65},{"href":63,"rel":64},"https://github.com/tpope/vim-fugitive",[28],[66],{"type":31,"value":67},"tpope/vim-fugitive",{"type":31,"value":69},", it is necessary to configure the ",{"type":19,"tag":36,"props":71,"children":73},{"className":72},[],[74],{"type":31,"value":75},"gpg-agent",{"type":31,"value":77}," to use a GUI based ",{"type":19,"tag":36,"props":79,"children":81},{"className":80},[],[82],{"type":31,"value":83},"pinentry-program",{"type":31,"value":85},".","content:articles:vim-fugitive-gpg-pinentry.md",{"_path":88,"title":89,"description":90,"date":91,"tags":92,"categories":95,"excerpt":97,"_id":130},"/articles/podcast-transcription-whispercpp","Easily Transcribe Podcasts with Whisper.cpp","If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive whisper.cpp project. This high-performance fork of OpenAI's Whisper can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the Alter Everything podcast.","2024-01-08",[93,94],"whisper.cpp","ml",[96],"programming",{"type":16,"children":98},[99],{"type":19,"tag":20,"props":100,"children":101},{},[102,104,110,112,119,121,128],{"type":31,"value":103},"If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive ",{"type":19,"tag":24,"props":105,"children":108},{"href":106,"rel":107},"https://github.com/ggerganov/whisper.cpp",[28],[109],{"type":31,"value":93},{"type":31,"value":111}," project. This high-performance fork of ",{"type":19,"tag":24,"props":113,"children":116},{"href":114,"rel":115},"https://github.com/openai/whisper",[28],[117],{"type":31,"value":118},"OpenAI's Whisper",{"type":31,"value":120}," can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the ",{"type":19,"tag":24,"props":122,"children":125},{"href":123,"rel":124},"https://podcasts.apple.com/us/podcast/alter-everything/id1356137854",[28],[126],{"type":31,"value":127},"Alter Everything",{"type":31,"value":129}," podcast.","content:articles:podcast-transcription-whispercpp.md",{"_path":132,"title":133,"description":134,"date":135,"tags":136,"categories":139,"excerpt":140,"_id":156},"/articles/nuxt-content-rss-feed","How To Add an RSS Feed to a Nuxt Website","If you are a user of Nuxt Content and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.","2024-01-06",[137,138],"nuxt","rss",[96],{"type":16,"children":141},[142],{"type":19,"tag":20,"props":143,"children":144},{},[145,147,154],{"type":31,"value":146},"If you are a user of ",{"type":19,"tag":24,"props":148,"children":151},{"href":149,"rel":150},"https://content.nuxt.com/",[28],[152],{"type":31,"value":153},"Nuxt Content",{"type":31,"value":155}," and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.","content:articles:nuxt-content-rss-feed.md",{"_path":158,"title":159,"description":160,"date":161,"tags":162,"categories":166,"excerpt":167,"_id":173},"/articles/fennel-initial-exploration","Impressions of Fennel with Hammerspoon","A while back I read an introductory book on Lisp programming titled, \"The Little\nSchemer\". This book opened my mind to new (to me) programming paradigms, and left me\nwith a strong desire to find a way to incorporate Lisp programming into my every day\nlife. It took some time, but I believe I've found what I've been looking for: Fennel.","2023-10-22",[163,164,165],"lisp","hammerspoon","fennel",[163],{"type":16,"children":168},[169],{"type":19,"tag":20,"props":170,"children":171},{},[172],{"type":31,"value":160},"content:articles:fennel-initial-exploration.md",{"_path":175,"title":176,"description":177,"date":178,"tags":179,"categories":182,"excerpt":183,"_id":206},"/articles/doctl","Exploring the Digital Ocean `doctl` Utility","I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean doctl command line utility.\nThis proved to be an extremely easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future.","2023-01-01",[180,181],"linux","digital-ocean",[180],{"type":16,"children":184},[185],{"type":19,"tag":20,"props":186,"children":187},{},[188,190,196,198,204],{"type":31,"value":189},"I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean ",{"type":19,"tag":36,"props":191,"children":193},{"className":192},[],[194],{"type":31,"value":195},"doctl",{"type":31,"value":197}," command line utility.\nThis proved to be an ",{"type":19,"tag":199,"props":200,"children":201},"em",{},[202],{"type":31,"value":203},"extremely",{"type":31,"value":205}," easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future.","content:articles:doctl.md",{"_path":208,"title":209,"description":210,"date":211,"tags":212,"categories":213,"excerpt":215,"_id":221},"/articles/nuxt-v3-migration","This Website Has Been Migrated to Nuxt 3 🎉","This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years.\nNot to mention, all of the great plugins in the Vue ecosystem!","2022-12-31",[137],[214],"web",{"type":16,"children":216},[217],{"type":19,"tag":20,"props":218,"children":219},{},[220],{"type":31,"value":210},"content:articles:nuxt-v3-migration.md",{"_path":223,"title":224,"description":225,"date":226,"tags":227,"categories":231,"excerpt":232,"_id":238},"/articles/migrate-truenas-from-core-to-scale","Migrate to TrueNAS Scale from TrueNAS Core","TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale.","2021-12-28",[228,229,230],"homelab","supermicro","truenas",[228],{"type":16,"children":233},[234],{"type":19,"tag":20,"props":235,"children":236},{},[237],{"type":31,"value":225},"content:articles:migrate-truenas-from-core-to-scale.md",{"_path":240,"title":241,"description":242,"date":243,"tags":244,"categories":245,"excerpt":246,"_id":252},"/articles/reset-ipmi-password-from-host-os","Reset IPMI Credentials from the Host OS","If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these\ninstructions will guide your through the process of resetting the credentials to their default values from the host\noperating system.","2021-12-27",[228,229,230],[228],{"type":16,"children":247},[248],{"type":19,"tag":20,"props":249,"children":250},{},[251],{"type":31,"value":242},"content:articles:reset-ipmi-password-from-host-os.md",{"_path":254,"title":255,"description":256,"date":257,"tags":258,"categories":260,"excerpt":261,"_id":276},"/articles/quick-tip-rerunning-bash-commands","Tip: Re-running Bash Commands","Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use sudo for a command that requires\nroot privileges.","2021-09-22",[52,259],"bash",[14],{"type":16,"children":262},[263],{"type":19,"tag":20,"props":264,"children":265},{},[266,268,274],{"type":31,"value":267},"Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use ",{"type":19,"tag":36,"props":269,"children":271},{"className":270},[],[272],{"type":31,"value":273},"sudo",{"type":31,"value":275}," for a command that requires\nroot privileges.","content:articles:quick-tip-rerunning-bash-commands.md",{"_path":278,"title":279,"description":280,"date":281,"tags":282,"categories":287,"excerpt":290,"_id":296},"/articles/unit-testing-micropython-with-mocks","Unit Testing in MicroPython with Mocks","Unit testing code for embedded systems can be challenging. While it's possible\nto leverage emulators, write side-effect free code, or run tests on the\nhardware itself, it's often easiest to unit test the code on your personal\ncomputer with mocked hardware functionality.","2020-02-07",[283,284,285,286],"micropython","testing","mocks","tutorial",[288,289],"python","embedded",{"type":16,"children":291},[292],{"type":19,"tag":20,"props":293,"children":294},{},[295],{"type":31,"value":280},"content:articles:unit-testing-micropython-with-mocks.md",{"_path":298,"title":299,"description":300,"date":301,"tags":302,"categories":305,"excerpt":306,"_id":312},"/articles/persistent-archlinux-usb","Create a Persistent Arch Linux Bootable USB with Vagrant","When installing a linux distribution, it is common for the instructions to have\nthe user create a bootable USB, boot from the device, and proceed with the\ninstallation procedure from that live medium. However, this blog post will\noutline an alternative approach where a virtual machine created with Vagrant\nwill be used in favor of the live medium.","2020-01-09",[303,304],"vagrant","archlinux",[180],{"type":16,"children":307},[308],{"type":19,"tag":20,"props":309,"children":310},{},[311],{"type":31,"value":300},"content:articles:persistent-archlinux-usb.md",{"_path":314,"title":315,"description":316,"date":317,"tags":318,"categories":321,"excerpt":322,"_id":328},"/articles/docker-selinux-volumes","Docker Volume Permissions with SELinux","Unfamiliar with running Docker on a SELinux enabled system, I found myself\nrunning into a bunch of file permission errors while creating volumes.","2019-12-26",[319,320],"docker","selinux",[180],{"type":16,"children":323},[324],{"type":19,"tag":20,"props":325,"children":326},{},[327],{"type":31,"value":316},"content:articles:docker-selinux-volumes.md",{"_path":330,"title":331,"description":332,"date":333,"tags":334,"categories":337,"excerpt":338,"_id":359},"/articles/apu2-firmware-upgrade","Upgrading the Firmware on the PCEngines APU2","I've had a PCEngines APU2 gathering dust for a\nfew years, and have recently decided to dust it off.","2019-12-21",[335,336],"pcengine","apu",[180],{"type":16,"children":339},[340,354],{"type":19,"tag":20,"props":341,"children":342},{},[343,345,352],{"type":31,"value":344},"I've had a ",{"type":19,"tag":24,"props":346,"children":349},{"href":347,"rel":348},"https://pcengines.ch/apu2.htm",[28],[350],{"type":31,"value":351},"PCEngines APU2",{"type":31,"value":353}," gathering dust for a\nfew years, and have recently decided to dust it off.",{"type":19,"tag":20,"props":355,"children":356},{},[357],{"type":31,"value":358},"Since the last time the device has been powered on, there have been many great\nimprovements to the firmware, and it was very-much due for an upgrade. The\nfollowing steps outline how the firmware was upgraded on the APU from the\nalready-installed operating system -- CentOS.","content:articles:apu2-firmware-upgrade.md",1731781852990] \ No newline at end of file +[{"data":1,"prerenderedAt":360},["ShallowReactive",2],{"articles":3},[4,45,87,131,157,174,207,222,239,253,277,297,313,329],{"_path":5,"title":6,"description":7,"date":8,"tags":9,"categories":12,"excerpt":15,"_id":44},"/articles/ssh-ed25519-sk-yubikey","Configuring a YubiKey for use with OpenSSH","YubiKey's are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" ed25519-sk key type that supports FIDO compliant hardware keys.","2024-06-09",[10,11],"unix","configurations",[13,14],"tooling","tips",{"type":16,"children":17},"root",[18],{"type":19,"tag":20,"props":21,"children":22},"element","p",{},[23,33,35,42],{"type":19,"tag":24,"props":25,"children":29},"a",{"href":26,"rel":27},"https://www.yubico.com/",[28],"nofollow",[30],{"type":31,"value":32},"text","YubiKey's",{"type":31,"value":34}," are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" ",{"type":19,"tag":36,"props":37,"children":39},"code",{"className":38},[],[40],{"type":31,"value":41},"ed25519-sk",{"type":31,"value":43}," key type that supports FIDO compliant hardware keys.","content:articles:ssh-ed25519-sk-yubikey.md",{"_path":46,"title":47,"description":48,"date":49,"tags":50,"categories":53,"excerpt":54,"_id":86},"/articles/vim-fugitive-gpg-pinentry","Using pinentry-mac to sign commits from vim-fugitive","In order to sign git commits from within Vim using a plugin like tpope/vim-fugitive, it is necessary to configure the gpg-agent to use a GUI based pinentry-program.","2024-05-11",[51,52],"vim","tip",[13,14],{"type":16,"children":55},[56],{"type":19,"tag":20,"props":57,"children":58},{},[59,61,68,70,76,78,84],{"type":31,"value":60},"In order to sign git commits from within Vim using a plugin like ",{"type":19,"tag":24,"props":62,"children":65},{"href":63,"rel":64},"https://github.com/tpope/vim-fugitive",[28],[66],{"type":31,"value":67},"tpope/vim-fugitive",{"type":31,"value":69},", it is necessary to configure the ",{"type":19,"tag":36,"props":71,"children":73},{"className":72},[],[74],{"type":31,"value":75},"gpg-agent",{"type":31,"value":77}," to use a GUI based ",{"type":19,"tag":36,"props":79,"children":81},{"className":80},[],[82],{"type":31,"value":83},"pinentry-program",{"type":31,"value":85},".","content:articles:vim-fugitive-gpg-pinentry.md",{"_path":88,"title":89,"description":90,"date":91,"tags":92,"categories":95,"excerpt":97,"_id":130},"/articles/podcast-transcription-whispercpp","Easily Transcribe Podcasts with Whisper.cpp","If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive whisper.cpp project. This high-performance fork of OpenAI's Whisper can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the Alter Everything podcast.","2024-01-08",[93,94],"whisper.cpp","ml",[96],"programming",{"type":16,"children":98},[99],{"type":19,"tag":20,"props":100,"children":101},{},[102,104,110,112,119,121,128],{"type":31,"value":103},"If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive ",{"type":19,"tag":24,"props":105,"children":108},{"href":106,"rel":107},"https://github.com/ggerganov/whisper.cpp",[28],[109],{"type":31,"value":93},{"type":31,"value":111}," project. This high-performance fork of ",{"type":19,"tag":24,"props":113,"children":116},{"href":114,"rel":115},"https://github.com/openai/whisper",[28],[117],{"type":31,"value":118},"OpenAI's Whisper",{"type":31,"value":120}," can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the ",{"type":19,"tag":24,"props":122,"children":125},{"href":123,"rel":124},"https://podcasts.apple.com/us/podcast/alter-everything/id1356137854",[28],[126],{"type":31,"value":127},"Alter Everything",{"type":31,"value":129}," podcast.","content:articles:podcast-transcription-whispercpp.md",{"_path":132,"title":133,"description":134,"date":135,"tags":136,"categories":139,"excerpt":140,"_id":156},"/articles/nuxt-content-rss-feed","How To Add an RSS Feed to a Nuxt Website","If you are a user of Nuxt Content and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.","2024-01-06",[137,138],"nuxt","rss",[96],{"type":16,"children":141},[142],{"type":19,"tag":20,"props":143,"children":144},{},[145,147,154],{"type":31,"value":146},"If you are a user of ",{"type":19,"tag":24,"props":148,"children":151},{"href":149,"rel":150},"https://content.nuxt.com/",[28],[152],{"type":31,"value":153},"Nuxt Content",{"type":31,"value":155}," and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.","content:articles:nuxt-content-rss-feed.md",{"_path":158,"title":159,"description":160,"date":161,"tags":162,"categories":166,"excerpt":167,"_id":173},"/articles/fennel-initial-exploration","Impressions of Fennel with Hammerspoon","A while back I read an introductory book on Lisp programming titled, \"The Little\nSchemer\". This book opened my mind to new (to me) programming paradigms, and left me\nwith a strong desire to find a way to incorporate Lisp programming into my every day\nlife. It took some time, but I believe I've found what I've been looking for: Fennel.","2023-10-22",[163,164,165],"lisp","hammerspoon","fennel",[163],{"type":16,"children":168},[169],{"type":19,"tag":20,"props":170,"children":171},{},[172],{"type":31,"value":160},"content:articles:fennel-initial-exploration.md",{"_path":175,"title":176,"description":177,"date":178,"tags":179,"categories":182,"excerpt":183,"_id":206},"/articles/doctl","Exploring the Digital Ocean `doctl` Utility","I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean doctl command line utility.\nThis proved to be an extremely easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future.","2023-01-01",[180,181],"linux","digital-ocean",[180],{"type":16,"children":184},[185],{"type":19,"tag":20,"props":186,"children":187},{},[188,190,196,198,204],{"type":31,"value":189},"I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean ",{"type":19,"tag":36,"props":191,"children":193},{"className":192},[],[194],{"type":31,"value":195},"doctl",{"type":31,"value":197}," command line utility.\nThis proved to be an ",{"type":19,"tag":199,"props":200,"children":201},"em",{},[202],{"type":31,"value":203},"extremely",{"type":31,"value":205}," easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future.","content:articles:doctl.md",{"_path":208,"title":209,"description":210,"date":211,"tags":212,"categories":213,"excerpt":215,"_id":221},"/articles/nuxt-v3-migration","This Website Has Been Migrated to Nuxt 3 🎉","This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years.\nNot to mention, all of the great plugins in the Vue ecosystem!","2022-12-31",[137],[214],"web",{"type":16,"children":216},[217],{"type":19,"tag":20,"props":218,"children":219},{},[220],{"type":31,"value":210},"content:articles:nuxt-v3-migration.md",{"_path":223,"title":224,"description":225,"date":226,"tags":227,"categories":231,"excerpt":232,"_id":238},"/articles/migrate-truenas-from-core-to-scale","Migrate to TrueNAS Scale from TrueNAS Core","TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale.","2021-12-28",[228,229,230],"homelab","supermicro","truenas",[228],{"type":16,"children":233},[234],{"type":19,"tag":20,"props":235,"children":236},{},[237],{"type":31,"value":225},"content:articles:migrate-truenas-from-core-to-scale.md",{"_path":240,"title":241,"description":242,"date":243,"tags":244,"categories":245,"excerpt":246,"_id":252},"/articles/reset-ipmi-password-from-host-os","Reset IPMI Credentials from the Host OS","If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these\ninstructions will guide your through the process of resetting the credentials to their default values from the host\noperating system.","2021-12-27",[228,229,230],[228],{"type":16,"children":247},[248],{"type":19,"tag":20,"props":249,"children":250},{},[251],{"type":31,"value":242},"content:articles:reset-ipmi-password-from-host-os.md",{"_path":254,"title":255,"description":256,"date":257,"tags":258,"categories":260,"excerpt":261,"_id":276},"/articles/quick-tip-rerunning-bash-commands","Tip: Re-running Bash Commands","Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use sudo for a command that requires\nroot privileges.","2021-09-22",[52,259],"bash",[14],{"type":16,"children":262},[263],{"type":19,"tag":20,"props":264,"children":265},{},[266,268,274],{"type":31,"value":267},"Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use ",{"type":19,"tag":36,"props":269,"children":271},{"className":270},[],[272],{"type":31,"value":273},"sudo",{"type":31,"value":275}," for a command that requires\nroot privileges.","content:articles:quick-tip-rerunning-bash-commands.md",{"_path":278,"title":279,"description":280,"date":281,"tags":282,"categories":287,"excerpt":290,"_id":296},"/articles/unit-testing-micropython-with-mocks","Unit Testing in MicroPython with Mocks","Unit testing code for embedded systems can be challenging. While it's possible\nto leverage emulators, write side-effect free code, or run tests on the\nhardware itself, it's often easiest to unit test the code on your personal\ncomputer with mocked hardware functionality.","2020-02-07",[283,284,285,286],"micropython","testing","mocks","tutorial",[288,289],"python","embedded",{"type":16,"children":291},[292],{"type":19,"tag":20,"props":293,"children":294},{},[295],{"type":31,"value":280},"content:articles:unit-testing-micropython-with-mocks.md",{"_path":298,"title":299,"description":300,"date":301,"tags":302,"categories":305,"excerpt":306,"_id":312},"/articles/persistent-archlinux-usb","Create a Persistent Arch Linux Bootable USB with Vagrant","When installing a linux distribution, it is common for the instructions to have\nthe user create a bootable USB, boot from the device, and proceed with the\ninstallation procedure from that live medium. However, this blog post will\noutline an alternative approach where a virtual machine created with Vagrant\nwill be used in favor of the live medium.","2020-01-09",[303,304],"vagrant","archlinux",[180],{"type":16,"children":307},[308],{"type":19,"tag":20,"props":309,"children":310},{},[311],{"type":31,"value":300},"content:articles:persistent-archlinux-usb.md",{"_path":314,"title":315,"description":316,"date":317,"tags":318,"categories":321,"excerpt":322,"_id":328},"/articles/docker-selinux-volumes","Docker Volume Permissions with SELinux","Unfamiliar with running Docker on a SELinux enabled system, I found myself\nrunning into a bunch of file permission errors while creating volumes.","2019-12-26",[319,320],"docker","selinux",[180],{"type":16,"children":323},[324],{"type":19,"tag":20,"props":325,"children":326},{},[327],{"type":31,"value":316},"content:articles:docker-selinux-volumes.md",{"_path":330,"title":331,"description":332,"date":333,"tags":334,"categories":337,"excerpt":338,"_id":359},"/articles/apu2-firmware-upgrade","Upgrading the Firmware on the PCEngines APU2","I've had a PCEngines APU2 gathering dust for a\nfew years, and have recently decided to dust it off.","2019-12-21",[335,336],"pcengine","apu",[180],{"type":16,"children":339},[340,354],{"type":19,"tag":20,"props":341,"children":342},{},[343,345,352],{"type":31,"value":344},"I've had a ",{"type":19,"tag":24,"props":346,"children":349},{"href":347,"rel":348},"https://pcengines.ch/apu2.htm",[28],[350],{"type":31,"value":351},"PCEngines APU2",{"type":31,"value":353}," gathering dust for a\nfew years, and have recently decided to dust it off.",{"type":19,"tag":20,"props":355,"children":356},{},[357],{"type":31,"value":358},"Since the last time the device has been powered on, there have been many great\nimprovements to the firmware, and it was very-much due for an upgrade. The\nfollowing steps outline how the firmware was upgraded on the APU from the\nalready-installed operating system -- CentOS.","content:articles:apu2-firmware-upgrade.md",1731782033239] \ No newline at end of file diff --git a/articles/apu2-firmware-upgrade/_payload.json b/articles/apu2-firmware-upgrade/_payload.json index 7affaded..ea2fa57c 100644 --- a/articles/apu2-firmware-upgrade/_payload.json +++ b/articles/apu2-firmware-upgrade/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":315},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"draft":6,"tags":11,"categories":14,"excerpt":16,"body":43,"_type":309,"_id":310,"_source":311,"_file":312,"_stem":313,"_extension":314},"/articles/apu2-firmware-upgrade","articles",false,"","Upgrading the Firmware on the PCEngines APU2","I've had a PCEngines APU2 gathering dust for a\nfew years, and have recently decided to dust it off.","2019-12-21",[12,13],"pcengine","apu",[15],"linux",{"type":17,"children":18},"root",[19,38],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24,27,36],{"type":25,"value":26},"text","I've had a ",{"type":20,"tag":28,"props":29,"children":33},"a",{"href":30,"rel":31},"https://pcengines.ch/apu2.htm",[32],"nofollow",[34],{"type":25,"value":35},"PCEngines APU2",{"type":25,"value":37}," gathering dust for a\nfew years, and have recently decided to dust it off.",{"type":20,"tag":21,"props":39,"children":40},{},[41],{"type":25,"value":42},"Since the last time the device has been powered on, there have been many great\nimprovements to the firmware, and it was very-much due for an upgrade. The\nfollowing steps outline how the firmware was upgraded on the APU from the\nalready-installed operating system -- CentOS.",{"type":17,"children":44,"toc":307},[45,55,59,64,74,103,161,174,200,205,244,253,301],{"type":20,"tag":21,"props":46,"children":47},{},[48,49,54],{"type":25,"value":26},{"type":20,"tag":28,"props":50,"children":52},{"href":30,"rel":51},[32],[53],{"type":25,"value":35},{"type":25,"value":37},{"type":20,"tag":21,"props":56,"children":57},{},[58],{"type":25,"value":42},{"type":20,"tag":21,"props":60,"children":61},{},[62],{"type":25,"value":63},"First, we will connect to the device over the serial port",{"type":20,"tag":65,"props":66,"children":68},"pre",{"code":67},"screen /dev/tty.usbserial 115200\n",[69],{"type":20,"tag":70,"props":71,"children":72},"code",{"__ignoreMap":7},[73],{"type":25,"value":67},{"type":20,"tag":21,"props":75,"children":76},{},[77,79,85,87,93,95,101],{"type":25,"value":78},"Then, we will install the ",{"type":20,"tag":70,"props":80,"children":82},{"className":81},[],[83],{"type":25,"value":84},"flashrom",{"type":25,"value":86}," utility that is needed to update the\nfirmware. Because it is not available in the default ",{"type":20,"tag":70,"props":88,"children":90},{"className":89},[],[91],{"type":25,"value":92},"yum",{"type":25,"value":94}," repositories, we\nwill enable the ",{"type":20,"tag":96,"props":97,"children":98},"em",{},[99],{"type":25,"value":100},"Extra Packages for Enterprise Linux",{"type":25,"value":102}," (EPEL) repository before\ninstallation.",{"type":20,"tag":65,"props":104,"children":108},{"code":105,"language":106,"meta":7,"className":107,"style":7},"sudo yum install epel-release\nsudo yum install flashrom\n","bash","language-bash shiki shiki-themes github-light",[109],{"type":20,"tag":70,"props":110,"children":111},{"__ignoreMap":7},[112,140],{"type":20,"tag":113,"props":114,"children":117},"span",{"class":115,"line":116},"line",1,[118,124,130,135],{"type":20,"tag":113,"props":119,"children":121},{"style":120},"--shiki-default:#6F42C1",[122],{"type":25,"value":123},"sudo",{"type":20,"tag":113,"props":125,"children":127},{"style":126},"--shiki-default:#032F62",[128],{"type":25,"value":129}," yum",{"type":20,"tag":113,"props":131,"children":132},{"style":126},[133],{"type":25,"value":134}," install",{"type":20,"tag":113,"props":136,"children":137},{"style":126},[138],{"type":25,"value":139}," epel-release\n",{"type":20,"tag":113,"props":141,"children":143},{"class":115,"line":142},2,[144,148,152,156],{"type":20,"tag":113,"props":145,"children":146},{"style":120},[147],{"type":25,"value":123},{"type":20,"tag":113,"props":149,"children":150},{"style":126},[151],{"type":25,"value":129},{"type":20,"tag":113,"props":153,"children":154},{"style":126},[155],{"type":25,"value":134},{"type":20,"tag":113,"props":157,"children":158},{"style":126},[159],{"type":25,"value":160}," flashrom\n",{"type":20,"tag":21,"props":162,"children":163},{},[164,166,172],{"type":25,"value":165},"Next, we will download the latest version of the firmware that is compatible\nwith the APU2 device from the PC Engines release website:\n",{"type":20,"tag":28,"props":167,"children":170},{"href":168,"rel":169},"https://pcengines.github.io/",[32],[171],{"type":25,"value":168},{"type":25,"value":173},".",{"type":20,"tag":65,"props":175,"children":177},{"code":176,"language":106,"meta":7,"className":107,"style":7},"curl -O https://3mdeb.com/open-source-firmware/pcengines/apu2/apu2_v4.11.0.1.rom\n",[178],{"type":20,"tag":70,"props":179,"children":180},{"__ignoreMap":7},[181],{"type":20,"tag":113,"props":182,"children":183},{"class":115,"line":116},[184,189,195],{"type":20,"tag":113,"props":185,"children":186},{"style":120},[187],{"type":25,"value":188},"curl",{"type":20,"tag":113,"props":190,"children":192},{"style":191},"--shiki-default:#005CC5",[193],{"type":25,"value":194}," -O",{"type":20,"tag":113,"props":196,"children":197},{"style":126},[198],{"type":25,"value":199}," https://3mdeb.com/open-source-firmware/pcengines/apu2/apu2_v4.11.0.1.rom\n",{"type":20,"tag":21,"props":201,"children":202},{},[203],{"type":25,"value":204},"And finally, we will flash the firmware...",{"type":20,"tag":65,"props":206,"children":208},{"code":207,"language":106,"meta":7,"className":107,"style":7},"sudo flashrom -w apu2_v4.11.0.1.rom -p internal:boardmismatch=force\n",[209],{"type":20,"tag":70,"props":210,"children":211},{"__ignoreMap":7},[212],{"type":20,"tag":113,"props":213,"children":214},{"class":115,"line":116},[215,219,224,229,234,239],{"type":20,"tag":113,"props":216,"children":217},{"style":120},[218],{"type":25,"value":123},{"type":20,"tag":113,"props":220,"children":221},{"style":126},[222],{"type":25,"value":223}," flashrom",{"type":20,"tag":113,"props":225,"children":226},{"style":191},[227],{"type":25,"value":228}," -w",{"type":20,"tag":113,"props":230,"children":231},{"style":126},[232],{"type":25,"value":233}," apu2_v4.11.0.1.rom",{"type":20,"tag":113,"props":235,"children":236},{"style":191},[237],{"type":25,"value":238}," -p",{"type":20,"tag":113,"props":240,"children":241},{"style":126},[242],{"type":25,"value":243}," internal:boardmismatch=force\n",{"type":20,"tag":21,"props":245,"children":246},{},[247],{"type":20,"tag":248,"props":249,"children":250},"strong",{},[251],{"type":25,"value":252},"References:",{"type":20,"tag":254,"props":255,"children":256},"ul",{},[257,266,274,283,292],{"type":20,"tag":258,"props":259,"children":260},"li",{},[261],{"type":20,"tag":28,"props":262,"children":264},{"href":30,"rel":263},[32],[265],{"type":25,"value":30},{"type":20,"tag":258,"props":267,"children":268},{},[269],{"type":20,"tag":28,"props":270,"children":272},{"href":168,"rel":271},[32],[273],{"type":25,"value":168},{"type":20,"tag":258,"props":275,"children":276},{},[277],{"type":20,"tag":28,"props":278,"children":281},{"href":279,"rel":280},"https://github.com/pcengines/apu2-documentation/blob/master/docs/firmware_flashing.md",[32],[282],{"type":25,"value":279},{"type":20,"tag":258,"props":284,"children":285},{},[286],{"type":20,"tag":28,"props":287,"children":290},{"href":288,"rel":289},"https://github.com/pcengines/apu2-documentation/blob/master/docs/cold_reset.md",[32],[291],{"type":25,"value":288},{"type":20,"tag":258,"props":293,"children":294},{},[295],{"type":20,"tag":28,"props":296,"children":299},{"href":297,"rel":298},"https://github.com/elad/openbsd-apu2",[32],[300],{"type":25,"value":297},{"type":20,"tag":302,"props":303,"children":304},"style",{},[305],{"type":25,"value":306},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":142,"depth":142,"links":308},[],"markdown","content:articles:apu2-firmware-upgrade.md","content","articles/apu2-firmware-upgrade.md","articles/apu2-firmware-upgrade","md",1731781853329] \ No newline at end of file +[{"data":1,"prerenderedAt":315},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"draft":6,"tags":11,"categories":14,"excerpt":16,"body":43,"_type":309,"_id":310,"_source":311,"_file":312,"_stem":313,"_extension":314},"/articles/apu2-firmware-upgrade","articles",false,"","Upgrading the Firmware on the PCEngines APU2","I've had a PCEngines APU2 gathering dust for a\nfew years, and have recently decided to dust it off.","2019-12-21",[12,13],"pcengine","apu",[15],"linux",{"type":17,"children":18},"root",[19,38],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24,27,36],{"type":25,"value":26},"text","I've had a ",{"type":20,"tag":28,"props":29,"children":33},"a",{"href":30,"rel":31},"https://pcengines.ch/apu2.htm",[32],"nofollow",[34],{"type":25,"value":35},"PCEngines APU2",{"type":25,"value":37}," gathering dust for a\nfew years, and have recently decided to dust it off.",{"type":20,"tag":21,"props":39,"children":40},{},[41],{"type":25,"value":42},"Since the last time the device has been powered on, there have been many great\nimprovements to the firmware, and it was very-much due for an upgrade. The\nfollowing steps outline how the firmware was upgraded on the APU from the\nalready-installed operating system -- CentOS.",{"type":17,"children":44,"toc":307},[45,55,59,64,74,103,161,174,200,205,244,253,301],{"type":20,"tag":21,"props":46,"children":47},{},[48,49,54],{"type":25,"value":26},{"type":20,"tag":28,"props":50,"children":52},{"href":30,"rel":51},[32],[53],{"type":25,"value":35},{"type":25,"value":37},{"type":20,"tag":21,"props":56,"children":57},{},[58],{"type":25,"value":42},{"type":20,"tag":21,"props":60,"children":61},{},[62],{"type":25,"value":63},"First, we will connect to the device over the serial port",{"type":20,"tag":65,"props":66,"children":68},"pre",{"code":67},"screen /dev/tty.usbserial 115200\n",[69],{"type":20,"tag":70,"props":71,"children":72},"code",{"__ignoreMap":7},[73],{"type":25,"value":67},{"type":20,"tag":21,"props":75,"children":76},{},[77,79,85,87,93,95,101],{"type":25,"value":78},"Then, we will install the ",{"type":20,"tag":70,"props":80,"children":82},{"className":81},[],[83],{"type":25,"value":84},"flashrom",{"type":25,"value":86}," utility that is needed to update the\nfirmware. Because it is not available in the default ",{"type":20,"tag":70,"props":88,"children":90},{"className":89},[],[91],{"type":25,"value":92},"yum",{"type":25,"value":94}," repositories, we\nwill enable the ",{"type":20,"tag":96,"props":97,"children":98},"em",{},[99],{"type":25,"value":100},"Extra Packages for Enterprise Linux",{"type":25,"value":102}," (EPEL) repository before\ninstallation.",{"type":20,"tag":65,"props":104,"children":108},{"code":105,"language":106,"meta":7,"className":107,"style":7},"sudo yum install epel-release\nsudo yum install flashrom\n","bash","language-bash shiki shiki-themes github-light",[109],{"type":20,"tag":70,"props":110,"children":111},{"__ignoreMap":7},[112,140],{"type":20,"tag":113,"props":114,"children":117},"span",{"class":115,"line":116},"line",1,[118,124,130,135],{"type":20,"tag":113,"props":119,"children":121},{"style":120},"--shiki-default:#6F42C1",[122],{"type":25,"value":123},"sudo",{"type":20,"tag":113,"props":125,"children":127},{"style":126},"--shiki-default:#032F62",[128],{"type":25,"value":129}," yum",{"type":20,"tag":113,"props":131,"children":132},{"style":126},[133],{"type":25,"value":134}," install",{"type":20,"tag":113,"props":136,"children":137},{"style":126},[138],{"type":25,"value":139}," epel-release\n",{"type":20,"tag":113,"props":141,"children":143},{"class":115,"line":142},2,[144,148,152,156],{"type":20,"tag":113,"props":145,"children":146},{"style":120},[147],{"type":25,"value":123},{"type":20,"tag":113,"props":149,"children":150},{"style":126},[151],{"type":25,"value":129},{"type":20,"tag":113,"props":153,"children":154},{"style":126},[155],{"type":25,"value":134},{"type":20,"tag":113,"props":157,"children":158},{"style":126},[159],{"type":25,"value":160}," flashrom\n",{"type":20,"tag":21,"props":162,"children":163},{},[164,166,172],{"type":25,"value":165},"Next, we will download the latest version of the firmware that is compatible\nwith the APU2 device from the PC Engines release website:\n",{"type":20,"tag":28,"props":167,"children":170},{"href":168,"rel":169},"https://pcengines.github.io/",[32],[171],{"type":25,"value":168},{"type":25,"value":173},".",{"type":20,"tag":65,"props":175,"children":177},{"code":176,"language":106,"meta":7,"className":107,"style":7},"curl -O https://3mdeb.com/open-source-firmware/pcengines/apu2/apu2_v4.11.0.1.rom\n",[178],{"type":20,"tag":70,"props":179,"children":180},{"__ignoreMap":7},[181],{"type":20,"tag":113,"props":182,"children":183},{"class":115,"line":116},[184,189,195],{"type":20,"tag":113,"props":185,"children":186},{"style":120},[187],{"type":25,"value":188},"curl",{"type":20,"tag":113,"props":190,"children":192},{"style":191},"--shiki-default:#005CC5",[193],{"type":25,"value":194}," -O",{"type":20,"tag":113,"props":196,"children":197},{"style":126},[198],{"type":25,"value":199}," https://3mdeb.com/open-source-firmware/pcengines/apu2/apu2_v4.11.0.1.rom\n",{"type":20,"tag":21,"props":201,"children":202},{},[203],{"type":25,"value":204},"And finally, we will flash the firmware...",{"type":20,"tag":65,"props":206,"children":208},{"code":207,"language":106,"meta":7,"className":107,"style":7},"sudo flashrom -w apu2_v4.11.0.1.rom -p internal:boardmismatch=force\n",[209],{"type":20,"tag":70,"props":210,"children":211},{"__ignoreMap":7},[212],{"type":20,"tag":113,"props":213,"children":214},{"class":115,"line":116},[215,219,224,229,234,239],{"type":20,"tag":113,"props":216,"children":217},{"style":120},[218],{"type":25,"value":123},{"type":20,"tag":113,"props":220,"children":221},{"style":126},[222],{"type":25,"value":223}," flashrom",{"type":20,"tag":113,"props":225,"children":226},{"style":191},[227],{"type":25,"value":228}," -w",{"type":20,"tag":113,"props":230,"children":231},{"style":126},[232],{"type":25,"value":233}," apu2_v4.11.0.1.rom",{"type":20,"tag":113,"props":235,"children":236},{"style":191},[237],{"type":25,"value":238}," -p",{"type":20,"tag":113,"props":240,"children":241},{"style":126},[242],{"type":25,"value":243}," internal:boardmismatch=force\n",{"type":20,"tag":21,"props":245,"children":246},{},[247],{"type":20,"tag":248,"props":249,"children":250},"strong",{},[251],{"type":25,"value":252},"References:",{"type":20,"tag":254,"props":255,"children":256},"ul",{},[257,266,274,283,292],{"type":20,"tag":258,"props":259,"children":260},"li",{},[261],{"type":20,"tag":28,"props":262,"children":264},{"href":30,"rel":263},[32],[265],{"type":25,"value":30},{"type":20,"tag":258,"props":267,"children":268},{},[269],{"type":20,"tag":28,"props":270,"children":272},{"href":168,"rel":271},[32],[273],{"type":25,"value":168},{"type":20,"tag":258,"props":275,"children":276},{},[277],{"type":20,"tag":28,"props":278,"children":281},{"href":279,"rel":280},"https://github.com/pcengines/apu2-documentation/blob/master/docs/firmware_flashing.md",[32],[282],{"type":25,"value":279},{"type":20,"tag":258,"props":284,"children":285},{},[286],{"type":20,"tag":28,"props":287,"children":290},{"href":288,"rel":289},"https://github.com/pcengines/apu2-documentation/blob/master/docs/cold_reset.md",[32],[291],{"type":25,"value":288},{"type":20,"tag":258,"props":293,"children":294},{},[295],{"type":20,"tag":28,"props":296,"children":299},{"href":297,"rel":298},"https://github.com/elad/openbsd-apu2",[32],[300],{"type":25,"value":297},{"type":20,"tag":302,"props":303,"children":304},"style",{},[305],{"type":25,"value":306},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":142,"depth":142,"links":308},[],"markdown","content:articles:apu2-firmware-upgrade.md","content","articles/apu2-firmware-upgrade.md","articles/apu2-firmware-upgrade","md",1731782033597] \ No newline at end of file diff --git a/articles/apu2-firmware-upgrade/index.html b/articles/apu2-firmware-upgrade/index.html index 7d2fd96d..210c83ac 100644 --- a/articles/apu2-firmware-upgrade/index.html +++ b/articles/apu2-firmware-upgrade/index.html @@ -5,35 +5,35 @@ - - - - - + + + + + - + - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + -
        Category

        Upgrading the Firmware on the PCEngines APU2

        I've had a PCEngines APU2 gathering dust for a +

        Category

        Upgrading the Firmware on the PCEngines APU2

        I've had a PCEngines APU2 gathering dust for a few years, and have recently decided to dust it off.

        Since the last time the device has been powered on, there have been many great improvements to the firmware, and it was very-much due for an upgrade. The following steps outline how the firmware was upgraded on the APU from the @@ -47,5 +47,5 @@ with the APU2 device from the PC Engines release website: https://pcengines.github.io/.

        curl -O https://3mdeb.com/open-source-firmware/pcengines/apu2/apu2_v4.11.0.1.rom
         

        And finally, we will flash the firmware...

        sudo flashrom -w apu2_v4.11.0.1.rom -p internal:boardmismatch=force
        -

        References:

        - \ No newline at end of file +

        References:

        + \ No newline at end of file diff --git a/articles/docker-selinux-volumes/_payload.json b/articles/docker-selinux-volumes/_payload.json index 85fda653..d5141f74 100644 --- a/articles/docker-selinux-volumes/_payload.json +++ b/articles/docker-selinux-volumes/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":240},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"draft":6,"tags":11,"categories":14,"excerpt":16,"body":26,"_type":234,"_id":235,"_source":236,"_file":237,"_stem":238,"_extension":239},"/articles/docker-selinux-volumes","articles",false,"","Docker Volume Permissions with SELinux","Unfamiliar with running Docker on a SELinux enabled system, I found myself\nrunning into a bunch of file permission errors while creating volumes.","2019-12-26",[12,13],"docker","selinux",[15],"linux",{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24],{"type":25,"value":9},"text",{"type":17,"children":27,"toc":232},[28,32,53,57,139,160,168,173,190,199,226],{"type":20,"tag":21,"props":29,"children":30},{},[31],{"type":25,"value":9},{"type":20,"tag":33,"props":34,"children":38},"pre",{"code":35,"language":36,"meta":7,"className":37,"style":7},"mkdir: can't create directory '/data': Permission denied\n","txt","language-txt shiki shiki-themes github-light",[39],{"type":20,"tag":40,"props":41,"children":42},"code",{"__ignoreMap":7},[43],{"type":20,"tag":44,"props":45,"children":48},"span",{"class":46,"line":47},"line",1,[49],{"type":20,"tag":44,"props":50,"children":51},{},[52],{"type":25,"value":35},{"type":20,"tag":54,"props":55,"children":56},"hr",{},[],{"type":20,"tag":33,"props":58,"children":62},{"code":59,"language":60,"meta":7,"className":61,"style":7},"$ docker info --format '{{json .SecurityOptions}}'\n[\n \"name=seccomp,profile=/etc/docker/seccomp.json\",\n \"name=selinux\"\n]\n","bash","language-bash shiki shiki-themes github-light",[63],{"type":20,"tag":40,"props":64,"children":65},{"__ignoreMap":7},[66,97,107,121,130],{"type":20,"tag":44,"props":67,"children":68},{"class":46,"line":47},[69,75,81,86,92],{"type":20,"tag":44,"props":70,"children":72},{"style":71},"--shiki-default:#6F42C1",[73],{"type":25,"value":74},"$",{"type":20,"tag":44,"props":76,"children":78},{"style":77},"--shiki-default:#032F62",[79],{"type":25,"value":80}," docker",{"type":20,"tag":44,"props":82,"children":83},{"style":77},[84],{"type":25,"value":85}," info",{"type":20,"tag":44,"props":87,"children":89},{"style":88},"--shiki-default:#005CC5",[90],{"type":25,"value":91}," --format",{"type":20,"tag":44,"props":93,"children":94},{"style":77},[95],{"type":25,"value":96}," '{{json .SecurityOptions}}'\n",{"type":20,"tag":44,"props":98,"children":100},{"class":46,"line":99},2,[101],{"type":20,"tag":44,"props":102,"children":104},{"style":103},"--shiki-default:#24292E",[105],{"type":25,"value":106},"[\n",{"type":20,"tag":44,"props":108,"children":110},{"class":46,"line":109},3,[111,116],{"type":20,"tag":44,"props":112,"children":113},{"style":77},[114],{"type":25,"value":115}," \"name=seccomp,profile=/etc/docker/seccomp.json\"",{"type":20,"tag":44,"props":117,"children":118},{"style":103},[119],{"type":25,"value":120},",\n",{"type":20,"tag":44,"props":122,"children":124},{"class":46,"line":123},4,[125],{"type":20,"tag":44,"props":126,"children":127},{"style":77},[128],{"type":25,"value":129}," \"name=selinux\"\n",{"type":20,"tag":44,"props":131,"children":133},{"class":46,"line":132},5,[134],{"type":20,"tag":44,"props":135,"children":136},{"style":103},[137],{"type":25,"value":138},"]\n",{"type":20,"tag":21,"props":140,"children":141},{},[142,144,150,152,158],{"type":25,"value":143},"It turns out that this can be resolved by appending the ",{"type":20,"tag":40,"props":145,"children":147},{"className":146},[],[148],{"type":25,"value":149},":z",{"type":25,"value":151}," flag to the volume\nmappings in the ",{"type":20,"tag":40,"props":153,"children":155},{"className":154},[],[156],{"type":25,"value":157},"docker-compose.yml",{"type":25,"value":159}," file, indicating that the volume content\nis shared.",{"type":20,"tag":33,"props":161,"children":163},{"code":162},"services:\n server:\n volumes:\n - ./data:/data:z\n",[164],{"type":20,"tag":40,"props":165,"children":166},{"__ignoreMap":7},[167],{"type":25,"value":162},{"type":20,"tag":21,"props":169,"children":170},{},[171],{"type":25,"value":172},"From the Docker documentation:",{"type":20,"tag":174,"props":175,"children":176},"blockquote",{},[177],{"type":20,"tag":21,"props":178,"children":179},{},[180,182,188],{"type":25,"value":181},"The ",{"type":20,"tag":40,"props":183,"children":185},{"className":184},[],[186],{"type":25,"value":187},"z",{"type":25,"value":189}," option tells Docker that two containers share the volume content. As\na result, Docker labels the content with a shared content label. Shared\nvolume labels allow all containers to read/write content.",{"type":20,"tag":21,"props":191,"children":192},{},[193],{"type":20,"tag":194,"props":195,"children":196},"strong",{},[197],{"type":25,"value":198},"References",{"type":20,"tag":200,"props":201,"children":202},"ul",{},[203,216],{"type":20,"tag":204,"props":205,"children":206},"li",{},[207],{"type":20,"tag":208,"props":209,"children":213},"a",{"href":210,"rel":211},"https://docs.docker.com/engine/reference/commandline/info/",[212],"nofollow",[214],{"type":25,"value":215},"Docker Docs: Docker Info",{"type":20,"tag":204,"props":217,"children":218},{},[219],{"type":20,"tag":208,"props":220,"children":223},{"href":221,"rel":222},"https://docs.docker.com/engine/reference/commandline/run/#mount-volumes-from-container---volumes-from",[212],[224],{"type":25,"value":225},"Docker Docs: Mounting Volumes",{"type":20,"tag":227,"props":228,"children":229},"style",{},[230],{"type":25,"value":231},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":99,"depth":99,"links":233},[],"markdown","content:articles:docker-selinux-volumes.md","content","articles/docker-selinux-volumes.md","articles/docker-selinux-volumes","md",1731781853294] \ No newline at end of file +[{"data":1,"prerenderedAt":240},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"draft":6,"tags":11,"categories":14,"excerpt":16,"body":26,"_type":234,"_id":235,"_source":236,"_file":237,"_stem":238,"_extension":239},"/articles/docker-selinux-volumes","articles",false,"","Docker Volume Permissions with SELinux","Unfamiliar with running Docker on a SELinux enabled system, I found myself\nrunning into a bunch of file permission errors while creating volumes.","2019-12-26",[12,13],"docker","selinux",[15],"linux",{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24],{"type":25,"value":9},"text",{"type":17,"children":27,"toc":232},[28,32,53,57,139,160,168,173,190,199,226],{"type":20,"tag":21,"props":29,"children":30},{},[31],{"type":25,"value":9},{"type":20,"tag":33,"props":34,"children":38},"pre",{"code":35,"language":36,"meta":7,"className":37,"style":7},"mkdir: can't create directory '/data': Permission denied\n","txt","language-txt shiki shiki-themes github-light",[39],{"type":20,"tag":40,"props":41,"children":42},"code",{"__ignoreMap":7},[43],{"type":20,"tag":44,"props":45,"children":48},"span",{"class":46,"line":47},"line",1,[49],{"type":20,"tag":44,"props":50,"children":51},{},[52],{"type":25,"value":35},{"type":20,"tag":54,"props":55,"children":56},"hr",{},[],{"type":20,"tag":33,"props":58,"children":62},{"code":59,"language":60,"meta":7,"className":61,"style":7},"$ docker info --format '{{json .SecurityOptions}}'\n[\n \"name=seccomp,profile=/etc/docker/seccomp.json\",\n \"name=selinux\"\n]\n","bash","language-bash shiki shiki-themes github-light",[63],{"type":20,"tag":40,"props":64,"children":65},{"__ignoreMap":7},[66,97,107,121,130],{"type":20,"tag":44,"props":67,"children":68},{"class":46,"line":47},[69,75,81,86,92],{"type":20,"tag":44,"props":70,"children":72},{"style":71},"--shiki-default:#6F42C1",[73],{"type":25,"value":74},"$",{"type":20,"tag":44,"props":76,"children":78},{"style":77},"--shiki-default:#032F62",[79],{"type":25,"value":80}," docker",{"type":20,"tag":44,"props":82,"children":83},{"style":77},[84],{"type":25,"value":85}," info",{"type":20,"tag":44,"props":87,"children":89},{"style":88},"--shiki-default:#005CC5",[90],{"type":25,"value":91}," --format",{"type":20,"tag":44,"props":93,"children":94},{"style":77},[95],{"type":25,"value":96}," '{{json .SecurityOptions}}'\n",{"type":20,"tag":44,"props":98,"children":100},{"class":46,"line":99},2,[101],{"type":20,"tag":44,"props":102,"children":104},{"style":103},"--shiki-default:#24292E",[105],{"type":25,"value":106},"[\n",{"type":20,"tag":44,"props":108,"children":110},{"class":46,"line":109},3,[111,116],{"type":20,"tag":44,"props":112,"children":113},{"style":77},[114],{"type":25,"value":115}," \"name=seccomp,profile=/etc/docker/seccomp.json\"",{"type":20,"tag":44,"props":117,"children":118},{"style":103},[119],{"type":25,"value":120},",\n",{"type":20,"tag":44,"props":122,"children":124},{"class":46,"line":123},4,[125],{"type":20,"tag":44,"props":126,"children":127},{"style":77},[128],{"type":25,"value":129}," \"name=selinux\"\n",{"type":20,"tag":44,"props":131,"children":133},{"class":46,"line":132},5,[134],{"type":20,"tag":44,"props":135,"children":136},{"style":103},[137],{"type":25,"value":138},"]\n",{"type":20,"tag":21,"props":140,"children":141},{},[142,144,150,152,158],{"type":25,"value":143},"It turns out that this can be resolved by appending the ",{"type":20,"tag":40,"props":145,"children":147},{"className":146},[],[148],{"type":25,"value":149},":z",{"type":25,"value":151}," flag to the volume\nmappings in the ",{"type":20,"tag":40,"props":153,"children":155},{"className":154},[],[156],{"type":25,"value":157},"docker-compose.yml",{"type":25,"value":159}," file, indicating that the volume content\nis shared.",{"type":20,"tag":33,"props":161,"children":163},{"code":162},"services:\n server:\n volumes:\n - ./data:/data:z\n",[164],{"type":20,"tag":40,"props":165,"children":166},{"__ignoreMap":7},[167],{"type":25,"value":162},{"type":20,"tag":21,"props":169,"children":170},{},[171],{"type":25,"value":172},"From the Docker documentation:",{"type":20,"tag":174,"props":175,"children":176},"blockquote",{},[177],{"type":20,"tag":21,"props":178,"children":179},{},[180,182,188],{"type":25,"value":181},"The ",{"type":20,"tag":40,"props":183,"children":185},{"className":184},[],[186],{"type":25,"value":187},"z",{"type":25,"value":189}," option tells Docker that two containers share the volume content. As\na result, Docker labels the content with a shared content label. Shared\nvolume labels allow all containers to read/write content.",{"type":20,"tag":21,"props":191,"children":192},{},[193],{"type":20,"tag":194,"props":195,"children":196},"strong",{},[197],{"type":25,"value":198},"References",{"type":20,"tag":200,"props":201,"children":202},"ul",{},[203,216],{"type":20,"tag":204,"props":205,"children":206},"li",{},[207],{"type":20,"tag":208,"props":209,"children":213},"a",{"href":210,"rel":211},"https://docs.docker.com/engine/reference/commandline/info/",[212],"nofollow",[214],{"type":25,"value":215},"Docker Docs: Docker Info",{"type":20,"tag":204,"props":217,"children":218},{},[219],{"type":20,"tag":208,"props":220,"children":223},{"href":221,"rel":222},"https://docs.docker.com/engine/reference/commandline/run/#mount-volumes-from-container---volumes-from",[212],[224],{"type":25,"value":225},"Docker Docs: Mounting Volumes",{"type":20,"tag":227,"props":228,"children":229},"style",{},[230],{"type":25,"value":231},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":99,"depth":99,"links":233},[],"markdown","content:articles:docker-selinux-volumes.md","content","articles/docker-selinux-volumes.md","articles/docker-selinux-volumes","md",1731782033584] \ No newline at end of file diff --git a/articles/docker-selinux-volumes/index.html b/articles/docker-selinux-volumes/index.html index 843bf547..4718f594 100644 --- a/articles/docker-selinux-volumes/index.html +++ b/articles/docker-selinux-volumes/index.html @@ -5,36 +5,36 @@ - - - - - + + + + + - + - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + -
        Category

        Docker Volume Permissions with SELinux

        Unfamiliar with running Docker on a SELinux enabled system, I found myself +

        Category

        Docker Volume Permissions with SELinux

        Unfamiliar with running Docker on a SELinux enabled system, I found myself running into a bunch of file permission errors while creating volumes.

        mkdir: can't create directory '/data': Permission denied
         

        $ docker info --format '{{json .SecurityOptions}}'
         [
        @@ -49,5 +49,5 @@
               - ./data:/data:z
         

        From the Docker documentation:

        The z option tells Docker that two containers share the volume content. As a result, Docker labels the content with a shared content label. Shared -volume labels allow all containers to read/write content.

        References

        - \ No newline at end of file +volume labels allow all containers to read/write content.

        References

        + \ No newline at end of file diff --git a/articles/doctl/_payload.json b/articles/doctl/_payload.json index db647247..0e66e892 100644 --- a/articles/doctl/_payload.json +++ b/articles/doctl/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":712},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"tags":11,"categories":14,"excerpt":15,"body":43,"_type":706,"_id":707,"_source":708,"_file":709,"_stem":710,"_extension":711},"/articles/doctl","articles",false,"","Exploring the Digital Ocean `doctl` Utility","I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean doctl command line utility.\nThis proved to be an extremely easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future.","2023-01-01",[12,13],"linux","digital-ocean",[12],{"type":16,"children":17},"root",[18],{"type":19,"tag":20,"props":21,"children":22},"element","p",{},[23,26,33,35,41],{"type":24,"value":25},"text","I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean ",{"type":19,"tag":27,"props":28,"children":30},"code",{"className":29},[],[31],{"type":24,"value":32},"doctl",{"type":24,"value":34}," command line utility.\nThis proved to be an ",{"type":19,"tag":36,"props":37,"children":38},"em",{},[39],{"type":24,"value":40},"extremely",{"type":24,"value":42}," easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future.",{"type":16,"children":44,"toc":704},[45,60,76,156,177,212,217,289,294,389,402,452,465,495,500,624,629,657,693,698],{"type":19,"tag":20,"props":46,"children":47},{},[48,49,54,55,59],{"type":24,"value":25},{"type":19,"tag":27,"props":50,"children":52},{"className":51},[],[53],{"type":24,"value":32},{"type":24,"value":34},{"type":19,"tag":36,"props":56,"children":57},{},[58],{"type":24,"value":40},{"type":24,"value":42},{"type":19,"tag":20,"props":61,"children":62},{},[63,65,74],{"type":24,"value":64},"To start things off, I had to install and setup authentication to Digital Ocean. Doing\nthis on my Mac machine, I opted to use ",{"type":19,"tag":66,"props":67,"children":71},"a",{"href":68,"rel":69},"https://brew.sh/",[70],"nofollow",[72],{"type":24,"value":73},"Homebrew",{"type":24,"value":75},".",{"type":19,"tag":77,"props":78,"children":82},"pre",{"className":79,"code":80,"language":81,"meta":7,"style":7},"language-sh shiki shiki-themes github-light","# install `doctl`\nbrew install doctl\n\n# setup authentication\ndoctl auth init\n","sh",[83],{"type":19,"tag":27,"props":84,"children":85},{"__ignoreMap":7},[86,98,119,129,138],{"type":19,"tag":87,"props":88,"children":91},"span",{"class":89,"line":90},"line",1,[92],{"type":19,"tag":87,"props":93,"children":95},{"style":94},"--shiki-default:#6A737D",[96],{"type":24,"value":97},"# install `doctl`\n",{"type":19,"tag":87,"props":99,"children":101},{"class":89,"line":100},2,[102,108,114],{"type":19,"tag":87,"props":103,"children":105},{"style":104},"--shiki-default:#6F42C1",[106],{"type":24,"value":107},"brew",{"type":19,"tag":87,"props":109,"children":111},{"style":110},"--shiki-default:#032F62",[112],{"type":24,"value":113}," install",{"type":19,"tag":87,"props":115,"children":116},{"style":110},[117],{"type":24,"value":118}," doctl\n",{"type":19,"tag":87,"props":120,"children":122},{"class":89,"line":121},3,[123],{"type":19,"tag":87,"props":124,"children":126},{"emptyLinePlaceholder":125},true,[127],{"type":24,"value":128},"\n",{"type":19,"tag":87,"props":130,"children":132},{"class":89,"line":131},4,[133],{"type":19,"tag":87,"props":134,"children":135},{"style":94},[136],{"type":24,"value":137},"# setup authentication\n",{"type":19,"tag":87,"props":139,"children":141},{"class":89,"line":140},5,[142,146,151],{"type":19,"tag":87,"props":143,"children":144},{"style":104},[145],{"type":24,"value":32},{"type":19,"tag":87,"props":147,"children":148},{"style":110},[149],{"type":24,"value":150}," auth",{"type":19,"tag":87,"props":152,"children":153},{"style":110},[154],{"type":24,"value":155}," init\n",{"type":19,"tag":20,"props":157,"children":158},{},[159,161,168,170,176],{"type":24,"value":160},"While the online ",{"type":19,"tag":66,"props":162,"children":165},{"href":163,"rel":164},"https://docs.digitalocean.com/reference/doctl/reference/compute/droplet/create/",[70],[166],{"type":24,"value":167},"documentation",{"type":24,"value":169}," is fantastic, I instead found myself mostly referencing the outputs of ",{"type":19,"tag":27,"props":171,"children":173},{"className":172},[],[174],{"type":24,"value":175},"--help",{"type":24,"value":75},{"type":19,"tag":77,"props":178,"children":180},{"className":79,"code":179,"language":81,"meta":7,"style":7},"doctl compute droplet create --help\n",[181],{"type":19,"tag":27,"props":182,"children":183},{"__ignoreMap":7},[184],{"type":19,"tag":87,"props":185,"children":186},{"class":89,"line":90},[187,191,196,201,206],{"type":19,"tag":87,"props":188,"children":189},{"style":104},[190],{"type":24,"value":32},{"type":19,"tag":87,"props":192,"children":193},{"style":110},[194],{"type":24,"value":195}," compute",{"type":19,"tag":87,"props":197,"children":198},{"style":110},[199],{"type":24,"value":200}," droplet",{"type":19,"tag":87,"props":202,"children":203},{"style":110},[204],{"type":24,"value":205}," create",{"type":19,"tag":87,"props":207,"children":209},{"style":208},"--shiki-default:#005CC5",[210],{"type":24,"value":211}," --help\n",{"type":19,"tag":20,"props":213,"children":214},{},[215],{"type":24,"value":216},"I had to find the image name of the version of Ubuntu I wanted to install:",{"type":19,"tag":77,"props":218,"children":220},{"className":79,"code":219,"language":81,"meta":7,"style":7},"doctl compute image list --public | grep ubuntu-22\n\n# 118857366 22.04 (LTS) x64 snapshot Ubuntu ubuntu-22-04-x64 true 7\n# 119383150 22.10 x64 snapshot Ubuntu ubuntu-22-10-x64 true 7\n",[221],{"type":19,"tag":27,"props":222,"children":223},{"__ignoreMap":7},[224,266,273,281],{"type":19,"tag":87,"props":225,"children":226},{"class":89,"line":90},[227,231,235,240,245,250,256,261],{"type":19,"tag":87,"props":228,"children":229},{"style":104},[230],{"type":24,"value":32},{"type":19,"tag":87,"props":232,"children":233},{"style":110},[234],{"type":24,"value":195},{"type":19,"tag":87,"props":236,"children":237},{"style":110},[238],{"type":24,"value":239}," image",{"type":19,"tag":87,"props":241,"children":242},{"style":110},[243],{"type":24,"value":244}," list",{"type":19,"tag":87,"props":246,"children":247},{"style":208},[248],{"type":24,"value":249}," --public",{"type":19,"tag":87,"props":251,"children":253},{"style":252},"--shiki-default:#D73A49",[254],{"type":24,"value":255}," |",{"type":19,"tag":87,"props":257,"children":258},{"style":104},[259],{"type":24,"value":260}," grep",{"type":19,"tag":87,"props":262,"children":263},{"style":110},[264],{"type":24,"value":265}," ubuntu-22\n",{"type":19,"tag":87,"props":267,"children":268},{"class":89,"line":100},[269],{"type":19,"tag":87,"props":270,"children":271},{"emptyLinePlaceholder":125},[272],{"type":24,"value":128},{"type":19,"tag":87,"props":274,"children":275},{"class":89,"line":121},[276],{"type":19,"tag":87,"props":277,"children":278},{"style":94},[279],{"type":24,"value":280},"# 118857366 22.04 (LTS) x64 snapshot Ubuntu ubuntu-22-04-x64 true 7\n",{"type":19,"tag":87,"props":282,"children":283},{"class":89,"line":131},[284],{"type":19,"tag":87,"props":285,"children":286},{"style":94},[287],{"type":24,"value":288},"# 119383150 22.10 x64 snapshot Ubuntu ubuntu-22-10-x64 true 7\n",{"type":19,"tag":20,"props":290,"children":291},{},[292],{"type":24,"value":293},"And also the slug of the compute size:",{"type":19,"tag":77,"props":295,"children":297},{"className":79,"code":296,"language":81,"meta":7,"style":7},"doctl compute size list\n\n# Slug Description Memory VCPUs Disk Price Monthly Price Hourly\n# s-1vcpu-512mb-10gb Basic 512 1 10 4.00 0.005950\n# s-1vcpu-1gb Basic 1024 1 25 6.00 0.008930\n# s-1vcpu-1gb-amd Basic AMD 1024 1 25 7.00 0.010420\n# s-1vcpu-1gb-intel Basic Intel 1024 1 25 7.00 0.010420\n# s-1vcpu-2gb Basic 2048 1 50 12.00 0.017860\n# s-1vcpu-2gb-amd Basic AMD 2048 1 50 14.00 0.020830\n",[298],{"type":19,"tag":27,"props":299,"children":300},{"__ignoreMap":7},[301,322,329,337,345,353,362,371,380],{"type":19,"tag":87,"props":302,"children":303},{"class":89,"line":90},[304,308,312,317],{"type":19,"tag":87,"props":305,"children":306},{"style":104},[307],{"type":24,"value":32},{"type":19,"tag":87,"props":309,"children":310},{"style":110},[311],{"type":24,"value":195},{"type":19,"tag":87,"props":313,"children":314},{"style":110},[315],{"type":24,"value":316}," size",{"type":19,"tag":87,"props":318,"children":319},{"style":110},[320],{"type":24,"value":321}," list\n",{"type":19,"tag":87,"props":323,"children":324},{"class":89,"line":100},[325],{"type":19,"tag":87,"props":326,"children":327},{"emptyLinePlaceholder":125},[328],{"type":24,"value":128},{"type":19,"tag":87,"props":330,"children":331},{"class":89,"line":121},[332],{"type":19,"tag":87,"props":333,"children":334},{"style":94},[335],{"type":24,"value":336},"# Slug Description Memory VCPUs Disk Price Monthly Price Hourly\n",{"type":19,"tag":87,"props":338,"children":339},{"class":89,"line":131},[340],{"type":19,"tag":87,"props":341,"children":342},{"style":94},[343],{"type":24,"value":344},"# s-1vcpu-512mb-10gb Basic 512 1 10 4.00 0.005950\n",{"type":19,"tag":87,"props":346,"children":347},{"class":89,"line":140},[348],{"type":19,"tag":87,"props":349,"children":350},{"style":94},[351],{"type":24,"value":352},"# s-1vcpu-1gb Basic 1024 1 25 6.00 0.008930\n",{"type":19,"tag":87,"props":354,"children":356},{"class":89,"line":355},6,[357],{"type":19,"tag":87,"props":358,"children":359},{"style":94},[360],{"type":24,"value":361},"# s-1vcpu-1gb-amd Basic AMD 1024 1 25 7.00 0.010420\n",{"type":19,"tag":87,"props":363,"children":365},{"class":89,"line":364},7,[366],{"type":19,"tag":87,"props":367,"children":368},{"style":94},[369],{"type":24,"value":370},"# s-1vcpu-1gb-intel Basic Intel 1024 1 25 7.00 0.010420\n",{"type":19,"tag":87,"props":372,"children":374},{"class":89,"line":373},8,[375],{"type":19,"tag":87,"props":376,"children":377},{"style":94},[378],{"type":24,"value":379},"# s-1vcpu-2gb Basic 2048 1 50 12.00 0.017860\n",{"type":19,"tag":87,"props":381,"children":383},{"class":89,"line":382},9,[384],{"type":19,"tag":87,"props":385,"children":386},{"style":94},[387],{"type":24,"value":388},"# s-1vcpu-2gb-amd Basic AMD 2048 1 50 14.00 0.020830\n",{"type":19,"tag":20,"props":390,"children":391},{},[392,394,400],{"type":24,"value":393},"I've also configured a few SSH keys with Digital Ocean, and I can have the key (specified by ID) provisioned to the machine using the ",{"type":19,"tag":27,"props":395,"children":397},{"className":396},[],[398],{"type":24,"value":399},"--ssh-keys",{"type":24,"value":401}," flag.",{"type":19,"tag":77,"props":403,"children":405},{"className":79,"code":404,"language":81,"meta":7,"style":7},"doctl compute ssh-key list\n\n# ID Name FingerPrint\n# 1234 mini \u003Credacted>\n",[406],{"type":19,"tag":27,"props":407,"children":408},{"__ignoreMap":7},[409,429,436,444],{"type":19,"tag":87,"props":410,"children":411},{"class":89,"line":90},[412,416,420,425],{"type":19,"tag":87,"props":413,"children":414},{"style":104},[415],{"type":24,"value":32},{"type":19,"tag":87,"props":417,"children":418},{"style":110},[419],{"type":24,"value":195},{"type":19,"tag":87,"props":421,"children":422},{"style":110},[423],{"type":24,"value":424}," ssh-key",{"type":19,"tag":87,"props":426,"children":427},{"style":110},[428],{"type":24,"value":321},{"type":19,"tag":87,"props":430,"children":431},{"class":89,"line":100},[432],{"type":19,"tag":87,"props":433,"children":434},{"emptyLinePlaceholder":125},[435],{"type":24,"value":128},{"type":19,"tag":87,"props":437,"children":438},{"class":89,"line":121},[439],{"type":19,"tag":87,"props":440,"children":441},{"style":94},[442],{"type":24,"value":443},"# ID Name FingerPrint\n",{"type":19,"tag":87,"props":445,"children":446},{"class":89,"line":131},[447],{"type":19,"tag":87,"props":448,"children":449},{"style":94},[450],{"type":24,"value":451},"# 1234 mini \u003Credacted>\n",{"type":19,"tag":20,"props":453,"children":454},{},[455,457,463],{"type":24,"value":456},"Also, I wanted to install a few packages to the box upon creation, this can be done easily with the ",{"type":19,"tag":27,"props":458,"children":460},{"className":459},[],[461],{"type":24,"value":462},"--user-data-file",{"type":24,"value":464}," flag to run an initialization script.",{"type":19,"tag":77,"props":466,"children":468},{"className":79,"code":467,"language":81,"meta":7,"style":7},"echo 'apt install -y imagemagick zip' > bootstrap.sh\n",[469],{"type":19,"tag":27,"props":470,"children":471},{"__ignoreMap":7},[472],{"type":19,"tag":87,"props":473,"children":474},{"class":89,"line":90},[475,480,485,490],{"type":19,"tag":87,"props":476,"children":477},{"style":208},[478],{"type":24,"value":479},"echo",{"type":19,"tag":87,"props":481,"children":482},{"style":110},[483],{"type":24,"value":484}," 'apt install -y imagemagick zip'",{"type":19,"tag":87,"props":486,"children":487},{"style":252},[488],{"type":24,"value":489}," >",{"type":19,"tag":87,"props":491,"children":492},{"style":110},[493],{"type":24,"value":494}," bootstrap.sh\n",{"type":19,"tag":20,"props":496,"children":497},{},[498],{"type":24,"value":499},"Putting it all together, here is the simple command for creating a small compute instance!",{"type":19,"tag":77,"props":501,"children":503},{"className":79,"code":502,"language":81,"meta":7,"style":7},"doctl compute droplet create \\\n --image ubuntu-22-10-x64 \\\n --size s-1vcpu-512mb-10gb \\\n --region nyc1 \\\n --ssh-keys 1234 \\\n --user-data-file boostrap.sh \\\n ephemeral\n",[504],{"type":19,"tag":27,"props":505,"children":506},{"__ignoreMap":7},[507,531,548,565,582,599,616],{"type":19,"tag":87,"props":508,"children":509},{"class":89,"line":90},[510,514,518,522,526],{"type":19,"tag":87,"props":511,"children":512},{"style":104},[513],{"type":24,"value":32},{"type":19,"tag":87,"props":515,"children":516},{"style":110},[517],{"type":24,"value":195},{"type":19,"tag":87,"props":519,"children":520},{"style":110},[521],{"type":24,"value":200},{"type":19,"tag":87,"props":523,"children":524},{"style":110},[525],{"type":24,"value":205},{"type":19,"tag":87,"props":527,"children":528},{"style":208},[529],{"type":24,"value":530}," \\\n",{"type":19,"tag":87,"props":532,"children":533},{"class":89,"line":100},[534,539,544],{"type":19,"tag":87,"props":535,"children":536},{"style":208},[537],{"type":24,"value":538}," --image",{"type":19,"tag":87,"props":540,"children":541},{"style":110},[542],{"type":24,"value":543}," ubuntu-22-10-x64",{"type":19,"tag":87,"props":545,"children":546},{"style":208},[547],{"type":24,"value":530},{"type":19,"tag":87,"props":549,"children":550},{"class":89,"line":121},[551,556,561],{"type":19,"tag":87,"props":552,"children":553},{"style":208},[554],{"type":24,"value":555}," --size",{"type":19,"tag":87,"props":557,"children":558},{"style":110},[559],{"type":24,"value":560}," s-1vcpu-512mb-10gb",{"type":19,"tag":87,"props":562,"children":563},{"style":208},[564],{"type":24,"value":530},{"type":19,"tag":87,"props":566,"children":567},{"class":89,"line":131},[568,573,578],{"type":19,"tag":87,"props":569,"children":570},{"style":208},[571],{"type":24,"value":572}," --region",{"type":19,"tag":87,"props":574,"children":575},{"style":110},[576],{"type":24,"value":577}," nyc1",{"type":19,"tag":87,"props":579,"children":580},{"style":208},[581],{"type":24,"value":530},{"type":19,"tag":87,"props":583,"children":584},{"class":89,"line":140},[585,590,595],{"type":19,"tag":87,"props":586,"children":587},{"style":208},[588],{"type":24,"value":589}," --ssh-keys",{"type":19,"tag":87,"props":591,"children":592},{"style":208},[593],{"type":24,"value":594}," 1234",{"type":19,"tag":87,"props":596,"children":597},{"style":208},[598],{"type":24,"value":530},{"type":19,"tag":87,"props":600,"children":601},{"class":89,"line":355},[602,607,612],{"type":19,"tag":87,"props":603,"children":604},{"style":208},[605],{"type":24,"value":606}," --user-data-file",{"type":19,"tag":87,"props":608,"children":609},{"style":110},[610],{"type":24,"value":611}," boostrap.sh",{"type":19,"tag":87,"props":613,"children":614},{"style":208},[615],{"type":24,"value":530},{"type":19,"tag":87,"props":617,"children":618},{"class":89,"line":364},[619],{"type":19,"tag":87,"props":620,"children":621},{"style":110},[622],{"type":24,"value":623}," ephemeral\n",{"type":19,"tag":20,"props":625,"children":626},{},[627],{"type":24,"value":628},"Finally, I can connect, do my thing, and destroy the instance.",{"type":19,"tag":77,"props":630,"children":632},{"className":79,"code":631,"language":81,"meta":7,"style":7},"doctl compute ssh ephemeral\n",[633],{"type":19,"tag":27,"props":634,"children":635},{"__ignoreMap":7},[636],{"type":19,"tag":87,"props":637,"children":638},{"class":89,"line":90},[639,643,647,652],{"type":19,"tag":87,"props":640,"children":641},{"style":104},[642],{"type":24,"value":32},{"type":19,"tag":87,"props":644,"children":645},{"style":110},[646],{"type":24,"value":195},{"type":19,"tag":87,"props":648,"children":649},{"style":110},[650],{"type":24,"value":651}," ssh",{"type":19,"tag":87,"props":653,"children":654},{"style":110},[655],{"type":24,"value":656}," ephemeral\n",{"type":19,"tag":77,"props":658,"children":660},{"className":79,"code":659,"language":81,"meta":7,"style":7},"doctl compute droplet delete --force ephemeral\n",[661],{"type":19,"tag":27,"props":662,"children":663},{"__ignoreMap":7},[664],{"type":19,"tag":87,"props":665,"children":666},{"class":89,"line":90},[667,671,675,679,684,689],{"type":19,"tag":87,"props":668,"children":669},{"style":104},[670],{"type":24,"value":32},{"type":19,"tag":87,"props":672,"children":673},{"style":110},[674],{"type":24,"value":195},{"type":19,"tag":87,"props":676,"children":677},{"style":110},[678],{"type":24,"value":200},{"type":19,"tag":87,"props":680,"children":681},{"style":110},[682],{"type":24,"value":683}," delete",{"type":19,"tag":87,"props":685,"children":686},{"style":208},[687],{"type":24,"value":688}," --force",{"type":19,"tag":87,"props":690,"children":691},{"style":110},[692],{"type":24,"value":656},{"type":19,"tag":20,"props":694,"children":695},{},[696],{"type":24,"value":697},"All-in-all, I was up and running in about 20 minutes. What a handy utility!",{"type":19,"tag":699,"props":700,"children":701},"style",{},[702],{"type":24,"value":703},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":100,"depth":100,"links":705},[],"markdown","content:articles:doctl.md","content","articles/doctl.md","articles/doctl","md",1731781853253] \ No newline at end of file +[{"data":1,"prerenderedAt":712},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"tags":11,"categories":14,"excerpt":15,"body":43,"_type":706,"_id":707,"_source":708,"_file":709,"_stem":710,"_extension":711},"/articles/doctl","articles",false,"","Exploring the Digital Ocean `doctl` Utility","I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean doctl command line utility.\nThis proved to be an extremely easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future.","2023-01-01",[12,13],"linux","digital-ocean",[12],{"type":16,"children":17},"root",[18],{"type":19,"tag":20,"props":21,"children":22},"element","p",{},[23,26,33,35,41],{"type":24,"value":25},"text","I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean ",{"type":19,"tag":27,"props":28,"children":30},"code",{"className":29},[],[31],{"type":24,"value":32},"doctl",{"type":24,"value":34}," command line utility.\nThis proved to be an ",{"type":19,"tag":36,"props":37,"children":38},"em",{},[39],{"type":24,"value":40},"extremely",{"type":24,"value":42}," easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future.",{"type":16,"children":44,"toc":704},[45,60,76,156,177,212,217,289,294,389,402,452,465,495,500,624,629,657,693,698],{"type":19,"tag":20,"props":46,"children":47},{},[48,49,54,55,59],{"type":24,"value":25},{"type":19,"tag":27,"props":50,"children":52},{"className":51},[],[53],{"type":24,"value":32},{"type":24,"value":34},{"type":19,"tag":36,"props":56,"children":57},{},[58],{"type":24,"value":40},{"type":24,"value":42},{"type":19,"tag":20,"props":61,"children":62},{},[63,65,74],{"type":24,"value":64},"To start things off, I had to install and setup authentication to Digital Ocean. Doing\nthis on my Mac machine, I opted to use ",{"type":19,"tag":66,"props":67,"children":71},"a",{"href":68,"rel":69},"https://brew.sh/",[70],"nofollow",[72],{"type":24,"value":73},"Homebrew",{"type":24,"value":75},".",{"type":19,"tag":77,"props":78,"children":82},"pre",{"className":79,"code":80,"language":81,"meta":7,"style":7},"language-sh shiki shiki-themes github-light","# install `doctl`\nbrew install doctl\n\n# setup authentication\ndoctl auth init\n","sh",[83],{"type":19,"tag":27,"props":84,"children":85},{"__ignoreMap":7},[86,98,119,129,138],{"type":19,"tag":87,"props":88,"children":91},"span",{"class":89,"line":90},"line",1,[92],{"type":19,"tag":87,"props":93,"children":95},{"style":94},"--shiki-default:#6A737D",[96],{"type":24,"value":97},"# install `doctl`\n",{"type":19,"tag":87,"props":99,"children":101},{"class":89,"line":100},2,[102,108,114],{"type":19,"tag":87,"props":103,"children":105},{"style":104},"--shiki-default:#6F42C1",[106],{"type":24,"value":107},"brew",{"type":19,"tag":87,"props":109,"children":111},{"style":110},"--shiki-default:#032F62",[112],{"type":24,"value":113}," install",{"type":19,"tag":87,"props":115,"children":116},{"style":110},[117],{"type":24,"value":118}," doctl\n",{"type":19,"tag":87,"props":120,"children":122},{"class":89,"line":121},3,[123],{"type":19,"tag":87,"props":124,"children":126},{"emptyLinePlaceholder":125},true,[127],{"type":24,"value":128},"\n",{"type":19,"tag":87,"props":130,"children":132},{"class":89,"line":131},4,[133],{"type":19,"tag":87,"props":134,"children":135},{"style":94},[136],{"type":24,"value":137},"# setup authentication\n",{"type":19,"tag":87,"props":139,"children":141},{"class":89,"line":140},5,[142,146,151],{"type":19,"tag":87,"props":143,"children":144},{"style":104},[145],{"type":24,"value":32},{"type":19,"tag":87,"props":147,"children":148},{"style":110},[149],{"type":24,"value":150}," auth",{"type":19,"tag":87,"props":152,"children":153},{"style":110},[154],{"type":24,"value":155}," init\n",{"type":19,"tag":20,"props":157,"children":158},{},[159,161,168,170,176],{"type":24,"value":160},"While the online ",{"type":19,"tag":66,"props":162,"children":165},{"href":163,"rel":164},"https://docs.digitalocean.com/reference/doctl/reference/compute/droplet/create/",[70],[166],{"type":24,"value":167},"documentation",{"type":24,"value":169}," is fantastic, I instead found myself mostly referencing the outputs of ",{"type":19,"tag":27,"props":171,"children":173},{"className":172},[],[174],{"type":24,"value":175},"--help",{"type":24,"value":75},{"type":19,"tag":77,"props":178,"children":180},{"className":79,"code":179,"language":81,"meta":7,"style":7},"doctl compute droplet create --help\n",[181],{"type":19,"tag":27,"props":182,"children":183},{"__ignoreMap":7},[184],{"type":19,"tag":87,"props":185,"children":186},{"class":89,"line":90},[187,191,196,201,206],{"type":19,"tag":87,"props":188,"children":189},{"style":104},[190],{"type":24,"value":32},{"type":19,"tag":87,"props":192,"children":193},{"style":110},[194],{"type":24,"value":195}," compute",{"type":19,"tag":87,"props":197,"children":198},{"style":110},[199],{"type":24,"value":200}," droplet",{"type":19,"tag":87,"props":202,"children":203},{"style":110},[204],{"type":24,"value":205}," create",{"type":19,"tag":87,"props":207,"children":209},{"style":208},"--shiki-default:#005CC5",[210],{"type":24,"value":211}," --help\n",{"type":19,"tag":20,"props":213,"children":214},{},[215],{"type":24,"value":216},"I had to find the image name of the version of Ubuntu I wanted to install:",{"type":19,"tag":77,"props":218,"children":220},{"className":79,"code":219,"language":81,"meta":7,"style":7},"doctl compute image list --public | grep ubuntu-22\n\n# 118857366 22.04 (LTS) x64 snapshot Ubuntu ubuntu-22-04-x64 true 7\n# 119383150 22.10 x64 snapshot Ubuntu ubuntu-22-10-x64 true 7\n",[221],{"type":19,"tag":27,"props":222,"children":223},{"__ignoreMap":7},[224,266,273,281],{"type":19,"tag":87,"props":225,"children":226},{"class":89,"line":90},[227,231,235,240,245,250,256,261],{"type":19,"tag":87,"props":228,"children":229},{"style":104},[230],{"type":24,"value":32},{"type":19,"tag":87,"props":232,"children":233},{"style":110},[234],{"type":24,"value":195},{"type":19,"tag":87,"props":236,"children":237},{"style":110},[238],{"type":24,"value":239}," image",{"type":19,"tag":87,"props":241,"children":242},{"style":110},[243],{"type":24,"value":244}," list",{"type":19,"tag":87,"props":246,"children":247},{"style":208},[248],{"type":24,"value":249}," --public",{"type":19,"tag":87,"props":251,"children":253},{"style":252},"--shiki-default:#D73A49",[254],{"type":24,"value":255}," |",{"type":19,"tag":87,"props":257,"children":258},{"style":104},[259],{"type":24,"value":260}," grep",{"type":19,"tag":87,"props":262,"children":263},{"style":110},[264],{"type":24,"value":265}," ubuntu-22\n",{"type":19,"tag":87,"props":267,"children":268},{"class":89,"line":100},[269],{"type":19,"tag":87,"props":270,"children":271},{"emptyLinePlaceholder":125},[272],{"type":24,"value":128},{"type":19,"tag":87,"props":274,"children":275},{"class":89,"line":121},[276],{"type":19,"tag":87,"props":277,"children":278},{"style":94},[279],{"type":24,"value":280},"# 118857366 22.04 (LTS) x64 snapshot Ubuntu ubuntu-22-04-x64 true 7\n",{"type":19,"tag":87,"props":282,"children":283},{"class":89,"line":131},[284],{"type":19,"tag":87,"props":285,"children":286},{"style":94},[287],{"type":24,"value":288},"# 119383150 22.10 x64 snapshot Ubuntu ubuntu-22-10-x64 true 7\n",{"type":19,"tag":20,"props":290,"children":291},{},[292],{"type":24,"value":293},"And also the slug of the compute size:",{"type":19,"tag":77,"props":295,"children":297},{"className":79,"code":296,"language":81,"meta":7,"style":7},"doctl compute size list\n\n# Slug Description Memory VCPUs Disk Price Monthly Price Hourly\n# s-1vcpu-512mb-10gb Basic 512 1 10 4.00 0.005950\n# s-1vcpu-1gb Basic 1024 1 25 6.00 0.008930\n# s-1vcpu-1gb-amd Basic AMD 1024 1 25 7.00 0.010420\n# s-1vcpu-1gb-intel Basic Intel 1024 1 25 7.00 0.010420\n# s-1vcpu-2gb Basic 2048 1 50 12.00 0.017860\n# s-1vcpu-2gb-amd Basic AMD 2048 1 50 14.00 0.020830\n",[298],{"type":19,"tag":27,"props":299,"children":300},{"__ignoreMap":7},[301,322,329,337,345,353,362,371,380],{"type":19,"tag":87,"props":302,"children":303},{"class":89,"line":90},[304,308,312,317],{"type":19,"tag":87,"props":305,"children":306},{"style":104},[307],{"type":24,"value":32},{"type":19,"tag":87,"props":309,"children":310},{"style":110},[311],{"type":24,"value":195},{"type":19,"tag":87,"props":313,"children":314},{"style":110},[315],{"type":24,"value":316}," size",{"type":19,"tag":87,"props":318,"children":319},{"style":110},[320],{"type":24,"value":321}," list\n",{"type":19,"tag":87,"props":323,"children":324},{"class":89,"line":100},[325],{"type":19,"tag":87,"props":326,"children":327},{"emptyLinePlaceholder":125},[328],{"type":24,"value":128},{"type":19,"tag":87,"props":330,"children":331},{"class":89,"line":121},[332],{"type":19,"tag":87,"props":333,"children":334},{"style":94},[335],{"type":24,"value":336},"# Slug Description Memory VCPUs Disk Price Monthly Price Hourly\n",{"type":19,"tag":87,"props":338,"children":339},{"class":89,"line":131},[340],{"type":19,"tag":87,"props":341,"children":342},{"style":94},[343],{"type":24,"value":344},"# s-1vcpu-512mb-10gb Basic 512 1 10 4.00 0.005950\n",{"type":19,"tag":87,"props":346,"children":347},{"class":89,"line":140},[348],{"type":19,"tag":87,"props":349,"children":350},{"style":94},[351],{"type":24,"value":352},"# s-1vcpu-1gb Basic 1024 1 25 6.00 0.008930\n",{"type":19,"tag":87,"props":354,"children":356},{"class":89,"line":355},6,[357],{"type":19,"tag":87,"props":358,"children":359},{"style":94},[360],{"type":24,"value":361},"# s-1vcpu-1gb-amd Basic AMD 1024 1 25 7.00 0.010420\n",{"type":19,"tag":87,"props":363,"children":365},{"class":89,"line":364},7,[366],{"type":19,"tag":87,"props":367,"children":368},{"style":94},[369],{"type":24,"value":370},"# s-1vcpu-1gb-intel Basic Intel 1024 1 25 7.00 0.010420\n",{"type":19,"tag":87,"props":372,"children":374},{"class":89,"line":373},8,[375],{"type":19,"tag":87,"props":376,"children":377},{"style":94},[378],{"type":24,"value":379},"# s-1vcpu-2gb Basic 2048 1 50 12.00 0.017860\n",{"type":19,"tag":87,"props":381,"children":383},{"class":89,"line":382},9,[384],{"type":19,"tag":87,"props":385,"children":386},{"style":94},[387],{"type":24,"value":388},"# s-1vcpu-2gb-amd Basic AMD 2048 1 50 14.00 0.020830\n",{"type":19,"tag":20,"props":390,"children":391},{},[392,394,400],{"type":24,"value":393},"I've also configured a few SSH keys with Digital Ocean, and I can have the key (specified by ID) provisioned to the machine using the ",{"type":19,"tag":27,"props":395,"children":397},{"className":396},[],[398],{"type":24,"value":399},"--ssh-keys",{"type":24,"value":401}," flag.",{"type":19,"tag":77,"props":403,"children":405},{"className":79,"code":404,"language":81,"meta":7,"style":7},"doctl compute ssh-key list\n\n# ID Name FingerPrint\n# 1234 mini \u003Credacted>\n",[406],{"type":19,"tag":27,"props":407,"children":408},{"__ignoreMap":7},[409,429,436,444],{"type":19,"tag":87,"props":410,"children":411},{"class":89,"line":90},[412,416,420,425],{"type":19,"tag":87,"props":413,"children":414},{"style":104},[415],{"type":24,"value":32},{"type":19,"tag":87,"props":417,"children":418},{"style":110},[419],{"type":24,"value":195},{"type":19,"tag":87,"props":421,"children":422},{"style":110},[423],{"type":24,"value":424}," ssh-key",{"type":19,"tag":87,"props":426,"children":427},{"style":110},[428],{"type":24,"value":321},{"type":19,"tag":87,"props":430,"children":431},{"class":89,"line":100},[432],{"type":19,"tag":87,"props":433,"children":434},{"emptyLinePlaceholder":125},[435],{"type":24,"value":128},{"type":19,"tag":87,"props":437,"children":438},{"class":89,"line":121},[439],{"type":19,"tag":87,"props":440,"children":441},{"style":94},[442],{"type":24,"value":443},"# ID Name FingerPrint\n",{"type":19,"tag":87,"props":445,"children":446},{"class":89,"line":131},[447],{"type":19,"tag":87,"props":448,"children":449},{"style":94},[450],{"type":24,"value":451},"# 1234 mini \u003Credacted>\n",{"type":19,"tag":20,"props":453,"children":454},{},[455,457,463],{"type":24,"value":456},"Also, I wanted to install a few packages to the box upon creation, this can be done easily with the ",{"type":19,"tag":27,"props":458,"children":460},{"className":459},[],[461],{"type":24,"value":462},"--user-data-file",{"type":24,"value":464}," flag to run an initialization script.",{"type":19,"tag":77,"props":466,"children":468},{"className":79,"code":467,"language":81,"meta":7,"style":7},"echo 'apt install -y imagemagick zip' > bootstrap.sh\n",[469],{"type":19,"tag":27,"props":470,"children":471},{"__ignoreMap":7},[472],{"type":19,"tag":87,"props":473,"children":474},{"class":89,"line":90},[475,480,485,490],{"type":19,"tag":87,"props":476,"children":477},{"style":208},[478],{"type":24,"value":479},"echo",{"type":19,"tag":87,"props":481,"children":482},{"style":110},[483],{"type":24,"value":484}," 'apt install -y imagemagick zip'",{"type":19,"tag":87,"props":486,"children":487},{"style":252},[488],{"type":24,"value":489}," >",{"type":19,"tag":87,"props":491,"children":492},{"style":110},[493],{"type":24,"value":494}," bootstrap.sh\n",{"type":19,"tag":20,"props":496,"children":497},{},[498],{"type":24,"value":499},"Putting it all together, here is the simple command for creating a small compute instance!",{"type":19,"tag":77,"props":501,"children":503},{"className":79,"code":502,"language":81,"meta":7,"style":7},"doctl compute droplet create \\\n --image ubuntu-22-10-x64 \\\n --size s-1vcpu-512mb-10gb \\\n --region nyc1 \\\n --ssh-keys 1234 \\\n --user-data-file boostrap.sh \\\n ephemeral\n",[504],{"type":19,"tag":27,"props":505,"children":506},{"__ignoreMap":7},[507,531,548,565,582,599,616],{"type":19,"tag":87,"props":508,"children":509},{"class":89,"line":90},[510,514,518,522,526],{"type":19,"tag":87,"props":511,"children":512},{"style":104},[513],{"type":24,"value":32},{"type":19,"tag":87,"props":515,"children":516},{"style":110},[517],{"type":24,"value":195},{"type":19,"tag":87,"props":519,"children":520},{"style":110},[521],{"type":24,"value":200},{"type":19,"tag":87,"props":523,"children":524},{"style":110},[525],{"type":24,"value":205},{"type":19,"tag":87,"props":527,"children":528},{"style":208},[529],{"type":24,"value":530}," \\\n",{"type":19,"tag":87,"props":532,"children":533},{"class":89,"line":100},[534,539,544],{"type":19,"tag":87,"props":535,"children":536},{"style":208},[537],{"type":24,"value":538}," --image",{"type":19,"tag":87,"props":540,"children":541},{"style":110},[542],{"type":24,"value":543}," ubuntu-22-10-x64",{"type":19,"tag":87,"props":545,"children":546},{"style":208},[547],{"type":24,"value":530},{"type":19,"tag":87,"props":549,"children":550},{"class":89,"line":121},[551,556,561],{"type":19,"tag":87,"props":552,"children":553},{"style":208},[554],{"type":24,"value":555}," --size",{"type":19,"tag":87,"props":557,"children":558},{"style":110},[559],{"type":24,"value":560}," s-1vcpu-512mb-10gb",{"type":19,"tag":87,"props":562,"children":563},{"style":208},[564],{"type":24,"value":530},{"type":19,"tag":87,"props":566,"children":567},{"class":89,"line":131},[568,573,578],{"type":19,"tag":87,"props":569,"children":570},{"style":208},[571],{"type":24,"value":572}," --region",{"type":19,"tag":87,"props":574,"children":575},{"style":110},[576],{"type":24,"value":577}," nyc1",{"type":19,"tag":87,"props":579,"children":580},{"style":208},[581],{"type":24,"value":530},{"type":19,"tag":87,"props":583,"children":584},{"class":89,"line":140},[585,590,595],{"type":19,"tag":87,"props":586,"children":587},{"style":208},[588],{"type":24,"value":589}," --ssh-keys",{"type":19,"tag":87,"props":591,"children":592},{"style":208},[593],{"type":24,"value":594}," 1234",{"type":19,"tag":87,"props":596,"children":597},{"style":208},[598],{"type":24,"value":530},{"type":19,"tag":87,"props":600,"children":601},{"class":89,"line":355},[602,607,612],{"type":19,"tag":87,"props":603,"children":604},{"style":208},[605],{"type":24,"value":606}," --user-data-file",{"type":19,"tag":87,"props":608,"children":609},{"style":110},[610],{"type":24,"value":611}," boostrap.sh",{"type":19,"tag":87,"props":613,"children":614},{"style":208},[615],{"type":24,"value":530},{"type":19,"tag":87,"props":617,"children":618},{"class":89,"line":364},[619],{"type":19,"tag":87,"props":620,"children":621},{"style":110},[622],{"type":24,"value":623}," ephemeral\n",{"type":19,"tag":20,"props":625,"children":626},{},[627],{"type":24,"value":628},"Finally, I can connect, do my thing, and destroy the instance.",{"type":19,"tag":77,"props":630,"children":632},{"className":79,"code":631,"language":81,"meta":7,"style":7},"doctl compute ssh ephemeral\n",[633],{"type":19,"tag":27,"props":634,"children":635},{"__ignoreMap":7},[636],{"type":19,"tag":87,"props":637,"children":638},{"class":89,"line":90},[639,643,647,652],{"type":19,"tag":87,"props":640,"children":641},{"style":104},[642],{"type":24,"value":32},{"type":19,"tag":87,"props":644,"children":645},{"style":110},[646],{"type":24,"value":195},{"type":19,"tag":87,"props":648,"children":649},{"style":110},[650],{"type":24,"value":651}," ssh",{"type":19,"tag":87,"props":653,"children":654},{"style":110},[655],{"type":24,"value":656}," ephemeral\n",{"type":19,"tag":77,"props":658,"children":660},{"className":79,"code":659,"language":81,"meta":7,"style":7},"doctl compute droplet delete --force ephemeral\n",[661],{"type":19,"tag":27,"props":662,"children":663},{"__ignoreMap":7},[664],{"type":19,"tag":87,"props":665,"children":666},{"class":89,"line":90},[667,671,675,679,684,689],{"type":19,"tag":87,"props":668,"children":669},{"style":104},[670],{"type":24,"value":32},{"type":19,"tag":87,"props":672,"children":673},{"style":110},[674],{"type":24,"value":195},{"type":19,"tag":87,"props":676,"children":677},{"style":110},[678],{"type":24,"value":200},{"type":19,"tag":87,"props":680,"children":681},{"style":110},[682],{"type":24,"value":683}," delete",{"type":19,"tag":87,"props":685,"children":686},{"style":208},[687],{"type":24,"value":688}," --force",{"type":19,"tag":87,"props":690,"children":691},{"style":110},[692],{"type":24,"value":656},{"type":19,"tag":20,"props":694,"children":695},{},[696],{"type":24,"value":697},"All-in-all, I was up and running in about 20 minutes. What a handy utility!",{"type":19,"tag":699,"props":700,"children":701},"style",{},[702],{"type":24,"value":703},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":100,"depth":100,"links":705},[],"markdown","content:articles:doctl.md","content","articles/doctl.md","articles/doctl","md",1731782033495] \ No newline at end of file diff --git a/articles/doctl/index.html b/articles/doctl/index.html index 4dbd3077..e883a0d2 100644 --- a/articles/doctl/index.html +++ b/articles/doctl/index.html @@ -5,32 +5,32 @@ - - - - - + + + + + - + - - - - - - - - - - - - - + + + + + + + + + + + + + -

        Exploring the Digital Ocean `doctl` Utility

        I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean doctl command line utility. +

        Exploring the Digital Ocean `doctl` Utility

        I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean doctl command line utility. This proved to be an extremely easy way to configure, create, connect, and destroy a Linux box, and I foresee myself using this service even more in the future.

        To start things off, I had to install and setup authentication to Digital Ocean. Doing this on my Mac machine, I opted to use Homebrew.

        # install `doctl`
         brew install doctl
        @@ -65,5 +65,5 @@
             ephemeral
         

        Finally, I can connect, do my thing, and destroy the instance.

        doctl compute ssh ephemeral
         
        doctl compute droplet delete --force ephemeral
        -

        All-in-all, I was up and running in about 20 minutes. What a handy utility!

        - \ No newline at end of file +

        All-in-all, I was up and running in about 20 minutes. What a handy utility!

        + \ No newline at end of file diff --git a/articles/fennel-initial-exploration/_payload.json b/articles/fennel-initial-exploration/_payload.json index 214455dc..f63933a6 100644 --- a/articles/fennel-initial-exploration/_payload.json +++ b/articles/fennel-initial-exploration/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":1106},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"draft":6,"date":10,"tags":11,"categories":15,"cover_image":16,"excerpt":17,"body":27,"_type":1100,"_id":1101,"_source":1102,"_file":1103,"_stem":1104,"_extension":1105},"/articles/fennel-initial-exploration","articles",false,"","Impressions of Fennel with Hammerspoon","A while back I read an introductory book on Lisp programming titled, \"The Little\nSchemer\". This book opened my mind to new (to me) programming paradigms, and left me\nwith a strong desire to find a way to incorporate Lisp programming into my every day\nlife. It took some time, but I believe I've found what I've been looking for: Fennel.","2023-10-22",[12,13,14],"lisp","hammerspoon","fennel",[12],"/images/dall-e-fennel-hammer.jpeg",{"type":18,"children":19},"root",[20],{"type":21,"tag":22,"props":23,"children":24},"element","p",{},[25],{"type":26,"value":9},"text",{"type":18,"children":28,"toc":1094},[29,33,67,74,88,94,125,157,170,199,422,428,458,536,598,611,917,1039,1052,1058,1063,1077,1088],{"type":21,"tag":22,"props":30,"children":31},{},[32],{"type":26,"value":9},{"type":21,"tag":22,"props":34,"children":35},{},[36,38,47,49,56,58,65],{"type":26,"value":37},"The ",{"type":21,"tag":39,"props":40,"children":44},"a",{"href":41,"rel":42},"https://fennel-lang.org/",[43],"nofollow",[45],{"type":26,"value":46},"Fennel",{"type":26,"value":48}," programming language is a dialect of Lisp that boasts compatibility with\nLua, and it just so happens that two of my favorite applications are configured with\nexactly that language: ",{"type":21,"tag":39,"props":50,"children":53},{"href":51,"rel":52},"https://www.hammerspoon.org/",[43],[54],{"type":26,"value":55},"Hammerspoon",{"type":26,"value":57},", and ",{"type":21,"tag":39,"props":59,"children":62},{"href":60,"rel":61},"https://neovim.io/",[43],[63],{"type":26,"value":64},"Neovim",{"type":26,"value":66},".",{"type":21,"tag":68,"props":69,"children":71},"h2",{"id":70},"initial-observations",[72],{"type":26,"value":73},"Initial Observations",{"type":21,"tag":22,"props":75,"children":76},{},[77,79,86],{"type":26,"value":78},"To initially explore Fennel, I wanted to start small. My Hammerspoon configuration\nconsists of 7 ",{"type":21,"tag":39,"props":80,"children":83},{"href":81,"rel":82},"https://github.com/cmpadden/dotfiles/tree/795749fa17e1310bb001bb7deaa22be8689f0027/hammerspoon/.hammerspoon/modules",[43],[84],{"type":26,"value":85},"modules",{"type":26,"value":87}," that I use for operations such as: launching applications,\nmanaging windows, keeping my computer from going to sleep, and general operating system\nautomation. So the plan is to translate these modules into Fennel, while maintaining\nwithout breaking the existing functionality. However, at this point, I wasn't even sure\nhow to embed Fennel into my project...",{"type":21,"tag":68,"props":89,"children":91},{"id":90},"integrating-fennel-with-hammerspoon",[92],{"type":26,"value":93},"Integrating Fennel with Hammerspoon",{"type":21,"tag":22,"props":95,"children":96},{},[97,99,106,108,115,117,123],{"type":26,"value":98},"While official documentation exists describing how to ",{"type":21,"tag":39,"props":100,"children":103},{"href":101,"rel":102},"https://fennel-lang.org/setup#embedding-fennel",[43],[104],{"type":26,"value":105},"embed fennel",{"type":26,"value":107}," into your\nproject; it didn't provide me with enough clarity to know my next steps on integrating\nit with Hammerspoon. I found a few resources online demonstrating how to extend the\n",{"type":21,"tag":109,"props":110,"children":112},"code",{"className":111},[],[113],{"type":26,"value":114},"package.path",{"type":26,"value":116}," and ",{"type":21,"tag":109,"props":118,"children":120},{"className":119},[],[121],{"type":26,"value":122},"package.cpath",{"type":26,"value":124}," properties in Lua, but I was unable to get this to\nwork.",{"type":21,"tag":126,"props":127,"children":128},"ul",{},[129,139,148],{"type":21,"tag":130,"props":131,"children":132},"li",{},[133],{"type":21,"tag":39,"props":134,"children":137},{"href":135,"rel":136},"https://blog.exupero.org/hammerspoon-with-fennel/",[43],[138],{"type":26,"value":135},{"type":21,"tag":130,"props":140,"children":141},{},[142],{"type":21,"tag":39,"props":143,"children":146},{"href":144,"rel":145},"https://github.com/Hammerspoon/hammerspoon/issues/2377#issuecomment-636331435",[43],[147],{"type":26,"value":144},{"type":21,"tag":130,"props":149,"children":150},{},[151],{"type":21,"tag":39,"props":152,"children":155},{"href":153,"rel":154},"https://github.com/agzam/spacehammer/blob/master/init.lua",[43],[156],{"type":26,"value":153},{"type":21,"tag":22,"props":158,"children":159},{},[160,162,168],{"type":26,"value":161},"Ultimately, I opted to include the ",{"type":21,"tag":109,"props":163,"children":165},{"className":164},[],[166],{"type":26,"value":167},"fennel.lua",{"type":26,"value":169}," file to my Hammerspoon configuration,\nand while not ideal, it does make the configuration nicely self-contained. I'll leave it\nas a future task to include the module installed with LuaRocks.",{"type":21,"tag":22,"props":171,"children":172},{},[173,175,181,183,189,191,197],{"type":26,"value":174},"With Fennel now included in my Hammerspoon configuration, all I need to do is configure\nthe ",{"type":21,"tag":109,"props":176,"children":178},{"className":177},[],[179],{"type":26,"value":180},"fennel.path",{"type":26,"value":182}," to point to the ",{"type":21,"tag":109,"props":184,"children":186},{"className":185},[],[187],{"type":26,"value":188},"*.fnl",{"type":26,"value":190}," files in the ",{"type":21,"tag":109,"props":192,"children":194},{"className":193},[],[195],{"type":26,"value":196},".hammerspoon/",{"type":26,"value":198}," directory, and\nttranslating these modules can begin!",{"type":21,"tag":200,"props":201,"children":205},"pre",{"className":202,"code":203,"language":204,"meta":7,"style":7},"language-lua shiki shiki-themes github-light","-- init.lua\n\nlocal fennel = require('fennel')\n\nfennel.path = package.path .. \";\" .. os.getenv(\"HOME\") .. \"/.hammerspoon/?.fnl\"\n\ntable.insert(package.loaders or package.searchers, fennel.searcher)\n\nrequire 'main'\n","lua",[206],{"type":21,"tag":109,"props":207,"children":208},{"__ignoreMap":7},[209,221,231,274,282,350,358,400,408],{"type":21,"tag":210,"props":211,"children":214},"span",{"class":212,"line":213},"line",1,[215],{"type":21,"tag":210,"props":216,"children":218},{"style":217},"--shiki-default:#6A737D",[219],{"type":26,"value":220},"-- init.lua\n",{"type":21,"tag":210,"props":222,"children":224},{"class":212,"line":223},2,[225],{"type":21,"tag":210,"props":226,"children":228},{"emptyLinePlaceholder":227},true,[229],{"type":26,"value":230},"\n",{"type":21,"tag":210,"props":232,"children":234},{"class":212,"line":233},3,[235,241,247,252,258,263,269],{"type":21,"tag":210,"props":236,"children":238},{"style":237},"--shiki-default:#D73A49",[239],{"type":26,"value":240},"local",{"type":21,"tag":210,"props":242,"children":244},{"style":243},"--shiki-default:#24292E",[245],{"type":26,"value":246}," fennel ",{"type":21,"tag":210,"props":248,"children":249},{"style":237},[250],{"type":26,"value":251},"=",{"type":21,"tag":210,"props":253,"children":255},{"style":254},"--shiki-default:#005CC5",[256],{"type":26,"value":257}," require",{"type":21,"tag":210,"props":259,"children":260},{"style":243},[261],{"type":26,"value":262},"(",{"type":21,"tag":210,"props":264,"children":266},{"style":265},"--shiki-default:#032F62",[267],{"type":26,"value":268},"'fennel'",{"type":21,"tag":210,"props":270,"children":271},{"style":243},[272],{"type":26,"value":273},")\n",{"type":21,"tag":210,"props":275,"children":277},{"class":212,"line":276},4,[278],{"type":21,"tag":210,"props":279,"children":280},{"emptyLinePlaceholder":227},[281],{"type":26,"value":230},{"type":21,"tag":210,"props":283,"children":285},{"class":212,"line":284},5,[286,291,297,302,307,312,317,322,327,331,336,341,345],{"type":21,"tag":210,"props":287,"children":288},{"style":243},[289],{"type":26,"value":290},"fennel.",{"type":21,"tag":210,"props":292,"children":294},{"style":293},"--shiki-default:#6F42C1",[295],{"type":26,"value":296},"path",{"type":21,"tag":210,"props":298,"children":299},{"style":237},[300],{"type":26,"value":301}," =",{"type":21,"tag":210,"props":303,"children":304},{"style":254},[305],{"type":26,"value":306}," package.path",{"type":21,"tag":210,"props":308,"children":309},{"style":237},[310],{"type":26,"value":311}," ..",{"type":21,"tag":210,"props":313,"children":314},{"style":265},[315],{"type":26,"value":316}," \";\" ",{"type":21,"tag":210,"props":318,"children":319},{"style":237},[320],{"type":26,"value":321},"..",{"type":21,"tag":210,"props":323,"children":324},{"style":254},[325],{"type":26,"value":326}," os.getenv",{"type":21,"tag":210,"props":328,"children":329},{"style":243},[330],{"type":26,"value":262},{"type":21,"tag":210,"props":332,"children":333},{"style":265},[334],{"type":26,"value":335},"\"HOME\"",{"type":21,"tag":210,"props":337,"children":338},{"style":243},[339],{"type":26,"value":340},") ",{"type":21,"tag":210,"props":342,"children":343},{"style":237},[344],{"type":26,"value":321},{"type":21,"tag":210,"props":346,"children":347},{"style":265},[348],{"type":26,"value":349}," \"/.hammerspoon/?.fnl\"\n",{"type":21,"tag":210,"props":351,"children":353},{"class":212,"line":352},6,[354],{"type":21,"tag":210,"props":355,"children":356},{"emptyLinePlaceholder":227},[357],{"type":26,"value":230},{"type":21,"tag":210,"props":359,"children":361},{"class":212,"line":360},7,[362,367,371,376,381,386,391,396],{"type":21,"tag":210,"props":363,"children":364},{"style":254},[365],{"type":26,"value":366},"table.insert",{"type":21,"tag":210,"props":368,"children":369},{"style":243},[370],{"type":26,"value":262},{"type":21,"tag":210,"props":372,"children":373},{"style":254},[374],{"type":26,"value":375},"package.loaders",{"type":21,"tag":210,"props":377,"children":378},{"style":237},[379],{"type":26,"value":380}," or",{"type":21,"tag":210,"props":382,"children":383},{"style":254},[384],{"type":26,"value":385}," package.searchers",{"type":21,"tag":210,"props":387,"children":388},{"style":243},[389],{"type":26,"value":390},", fennel.",{"type":21,"tag":210,"props":392,"children":393},{"style":293},[394],{"type":26,"value":395},"searcher",{"type":21,"tag":210,"props":397,"children":398},{"style":243},[399],{"type":26,"value":273},{"type":21,"tag":210,"props":401,"children":403},{"class":212,"line":402},8,[404],{"type":21,"tag":210,"props":405,"children":406},{"emptyLinePlaceholder":227},[407],{"type":26,"value":230},{"type":21,"tag":210,"props":409,"children":411},{"class":212,"line":410},9,[412,417],{"type":21,"tag":210,"props":413,"children":414},{"style":254},[415],{"type":26,"value":416},"require",{"type":21,"tag":210,"props":418,"children":419},{"style":265},[420],{"type":26,"value":421}," 'main'\n",{"type":21,"tag":68,"props":423,"children":425},{"id":424},"translating-lua-to-fennel",[426],{"type":26,"value":427},"Translating Lua to Fennel",{"type":21,"tag":22,"props":429,"children":430},{},[431,433,440,442,448,450,456],{"type":26,"value":432},"As a Fennel novice, I was happy to see that the Fennel project provides an online\ncross-compiler for Lua and Fennel called ",{"type":21,"tag":39,"props":434,"children":437},{"href":435,"rel":436},"https://fennel-lang.org/see",[43],[438],{"type":26,"value":439},"anti-fennel",{"type":26,"value":441},", and while it can generate some\nstrange-looking Fennel code, it was an extremely useful tool for me to get\nup-and-running right away. For example, by pasting the simple ",{"type":21,"tag":109,"props":443,"children":445},{"className":444},[],[446],{"type":26,"value":447},"sleep",{"type":26,"value":449}," function\nfrom the ",{"type":21,"tag":109,"props":451,"children":453},{"className":452},[],[454],{"type":26,"value":455},"helpers",{"type":26,"value":457}," module into the compiler:",{"type":21,"tag":200,"props":459,"children":461},{"className":202,"code":460,"language":204,"meta":7,"style":7},"function sleep(ms)\n os.execute(\"sleep \" .. tonumber(ms) / 1000)\nend\n",[462],{"type":21,"tag":109,"props":463,"children":464},{"__ignoreMap":7},[465,483,528],{"type":21,"tag":210,"props":466,"children":467},{"class":212,"line":213},[468,473,478],{"type":21,"tag":210,"props":469,"children":470},{"style":237},[471],{"type":26,"value":472},"function",{"type":21,"tag":210,"props":474,"children":475},{"style":293},[476],{"type":26,"value":477}," sleep",{"type":21,"tag":210,"props":479,"children":480},{"style":243},[481],{"type":26,"value":482},"(ms)\n",{"type":21,"tag":210,"props":484,"children":485},{"class":212,"line":223},[486,491,495,500,504,509,514,519,524],{"type":21,"tag":210,"props":487,"children":488},{"style":254},[489],{"type":26,"value":490}," os.execute",{"type":21,"tag":210,"props":492,"children":493},{"style":243},[494],{"type":26,"value":262},{"type":21,"tag":210,"props":496,"children":497},{"style":265},[498],{"type":26,"value":499},"\"sleep \" ",{"type":21,"tag":210,"props":501,"children":502},{"style":237},[503],{"type":26,"value":321},{"type":21,"tag":210,"props":505,"children":506},{"style":254},[507],{"type":26,"value":508}," tonumber",{"type":21,"tag":210,"props":510,"children":511},{"style":243},[512],{"type":26,"value":513},"(ms) ",{"type":21,"tag":210,"props":515,"children":516},{"style":237},[517],{"type":26,"value":518},"/",{"type":21,"tag":210,"props":520,"children":521},{"style":254},[522],{"type":26,"value":523}," 1000",{"type":21,"tag":210,"props":525,"children":526},{"style":243},[527],{"type":26,"value":273},{"type":21,"tag":210,"props":529,"children":530},{"class":212,"line":233},[531],{"type":21,"tag":210,"props":532,"children":533},{"style":237},[534],{"type":26,"value":535},"end\n",{"type":21,"tag":200,"props":537,"children":540},{"className":538,"code":539,"language":12,"meta":7,"style":7},"language-lisp shiki shiki-themes github-light","(fn sleep [ms]\n (os.execute (.. \"sleep \" (/ (tonumber ms) 1000))))\n",[541],{"type":21,"tag":109,"props":542,"children":543},{"__ignoreMap":7},[544,561],{"type":21,"tag":210,"props":545,"children":546},{"class":212,"line":213},[547,552,556],{"type":21,"tag":210,"props":548,"children":549},{"style":243},[550],{"type":26,"value":551},"(fn ",{"type":21,"tag":210,"props":553,"children":554},{"style":254},[555],{"type":26,"value":447},{"type":21,"tag":210,"props":557,"children":558},{"style":243},[559],{"type":26,"value":560}," [ms]\n",{"type":21,"tag":210,"props":562,"children":563},{"class":212,"line":223},[564,569,574,579,583,588,593],{"type":21,"tag":210,"props":565,"children":566},{"style":243},[567],{"type":26,"value":568}," (os.execute (.. ",{"type":21,"tag":210,"props":570,"children":571},{"style":265},[572],{"type":26,"value":573},"\"sleep \"",{"type":21,"tag":210,"props":575,"children":576},{"style":243},[577],{"type":26,"value":578}," (",{"type":21,"tag":210,"props":580,"children":581},{"style":254},[582],{"type":26,"value":518},{"type":21,"tag":210,"props":584,"children":585},{"style":243},[586],{"type":26,"value":587}," (tonumber ms) ",{"type":21,"tag":210,"props":589,"children":590},{"style":254},[591],{"type":26,"value":592},"1000",{"type":21,"tag":210,"props":594,"children":595},{"style":243},[596],{"type":26,"value":597},"))))\n",{"type":21,"tag":22,"props":599,"children":600},{},[601,603,609],{"type":26,"value":602},"As another example, here is the output for my ",{"type":21,"tag":604,"props":605,"children":606},"em",{},[607],{"type":26,"value":608},"caffeine",{"type":26,"value":610}," toggle:",{"type":21,"tag":200,"props":612,"children":614},{"className":202,"code":613,"language":204,"meta":7,"style":7},"hs.hotkey.bind(HYPER, \"0\", function()\n hs.caffeinate.toggle(\"displayIdle\")\n if hs.caffeinate.get(\"displayIdle\") then\n helpers:show(\"Caffeine Enabled\", nil, helpers.styles.success, helpers.assets.check)\n else\n helpers:show(\"Caffeine Disabled\", nil, helpers.styles.error, helpers.assets.ban)\n end\nend)\n",[615],{"type":21,"tag":109,"props":616,"children":617},{"__ignoreMap":7},[618,664,699,742,819,827,897,905],{"type":21,"tag":210,"props":619,"children":620},{"class":212,"line":213},[621,626,631,635,640,645,650,655,659],{"type":21,"tag":210,"props":622,"children":623},{"style":243},[624],{"type":26,"value":625},"hs.",{"type":21,"tag":210,"props":627,"children":628},{"style":293},[629],{"type":26,"value":630},"hotkey",{"type":21,"tag":210,"props":632,"children":633},{"style":243},[634],{"type":26,"value":66},{"type":21,"tag":210,"props":636,"children":637},{"style":254},[638],{"type":26,"value":639},"bind",{"type":21,"tag":210,"props":641,"children":642},{"style":243},[643],{"type":26,"value":644},"(HYPER, ",{"type":21,"tag":210,"props":646,"children":647},{"style":265},[648],{"type":26,"value":649},"\"0\"",{"type":21,"tag":210,"props":651,"children":652},{"style":243},[653],{"type":26,"value":654},", ",{"type":21,"tag":210,"props":656,"children":657},{"style":237},[658],{"type":26,"value":472},{"type":21,"tag":210,"props":660,"children":661},{"style":243},[662],{"type":26,"value":663},"()\n",{"type":21,"tag":210,"props":665,"children":666},{"class":212,"line":223},[667,672,677,681,686,690,695],{"type":21,"tag":210,"props":668,"children":669},{"style":243},[670],{"type":26,"value":671}," hs.",{"type":21,"tag":210,"props":673,"children":674},{"style":293},[675],{"type":26,"value":676},"caffeinate",{"type":21,"tag":210,"props":678,"children":679},{"style":243},[680],{"type":26,"value":66},{"type":21,"tag":210,"props":682,"children":683},{"style":254},[684],{"type":26,"value":685},"toggle",{"type":21,"tag":210,"props":687,"children":688},{"style":243},[689],{"type":26,"value":262},{"type":21,"tag":210,"props":691,"children":692},{"style":265},[693],{"type":26,"value":694},"\"displayIdle\"",{"type":21,"tag":210,"props":696,"children":697},{"style":243},[698],{"type":26,"value":273},{"type":21,"tag":210,"props":700,"children":701},{"class":212,"line":233},[702,707,712,716,720,725,729,733,737],{"type":21,"tag":210,"props":703,"children":704},{"style":237},[705],{"type":26,"value":706}," if",{"type":21,"tag":210,"props":708,"children":709},{"style":243},[710],{"type":26,"value":711}," hs.",{"type":21,"tag":210,"props":713,"children":714},{"style":293},[715],{"type":26,"value":676},{"type":21,"tag":210,"props":717,"children":718},{"style":243},[719],{"type":26,"value":66},{"type":21,"tag":210,"props":721,"children":722},{"style":254},[723],{"type":26,"value":724},"get",{"type":21,"tag":210,"props":726,"children":727},{"style":243},[728],{"type":26,"value":262},{"type":21,"tag":210,"props":730,"children":731},{"style":265},[732],{"type":26,"value":694},{"type":21,"tag":210,"props":734,"children":735},{"style":243},[736],{"type":26,"value":340},{"type":21,"tag":210,"props":738,"children":739},{"style":237},[740],{"type":26,"value":741},"then\n",{"type":21,"tag":210,"props":743,"children":744},{"class":212,"line":276},[745,750,755,760,764,769,773,778,783,788,792,797,801,806,810,815],{"type":21,"tag":210,"props":746,"children":747},{"style":293},[748],{"type":26,"value":749}," helpers",{"type":21,"tag":210,"props":751,"children":752},{"style":243},[753],{"type":26,"value":754},":",{"type":21,"tag":210,"props":756,"children":757},{"style":254},[758],{"type":26,"value":759},"show",{"type":21,"tag":210,"props":761,"children":762},{"style":243},[763],{"type":26,"value":262},{"type":21,"tag":210,"props":765,"children":766},{"style":265},[767],{"type":26,"value":768},"\"Caffeine Enabled\"",{"type":21,"tag":210,"props":770,"children":771},{"style":243},[772],{"type":26,"value":654},{"type":21,"tag":210,"props":774,"children":775},{"style":254},[776],{"type":26,"value":777},"nil",{"type":21,"tag":210,"props":779,"children":780},{"style":243},[781],{"type":26,"value":782},", helpers.",{"type":21,"tag":210,"props":784,"children":785},{"style":293},[786],{"type":26,"value":787},"styles",{"type":21,"tag":210,"props":789,"children":790},{"style":243},[791],{"type":26,"value":66},{"type":21,"tag":210,"props":793,"children":794},{"style":293},[795],{"type":26,"value":796},"success",{"type":21,"tag":210,"props":798,"children":799},{"style":243},[800],{"type":26,"value":782},{"type":21,"tag":210,"props":802,"children":803},{"style":293},[804],{"type":26,"value":805},"assets",{"type":21,"tag":210,"props":807,"children":808},{"style":243},[809],{"type":26,"value":66},{"type":21,"tag":210,"props":811,"children":812},{"style":293},[813],{"type":26,"value":814},"check",{"type":21,"tag":210,"props":816,"children":817},{"style":243},[818],{"type":26,"value":273},{"type":21,"tag":210,"props":820,"children":821},{"class":212,"line":284},[822],{"type":21,"tag":210,"props":823,"children":824},{"style":237},[825],{"type":26,"value":826}," else\n",{"type":21,"tag":210,"props":828,"children":829},{"class":212,"line":352},[830,834,838,842,846,851,855,859,863,867,871,876,880,884,888,893],{"type":21,"tag":210,"props":831,"children":832},{"style":293},[833],{"type":26,"value":749},{"type":21,"tag":210,"props":835,"children":836},{"style":243},[837],{"type":26,"value":754},{"type":21,"tag":210,"props":839,"children":840},{"style":254},[841],{"type":26,"value":759},{"type":21,"tag":210,"props":843,"children":844},{"style":243},[845],{"type":26,"value":262},{"type":21,"tag":210,"props":847,"children":848},{"style":265},[849],{"type":26,"value":850},"\"Caffeine Disabled\"",{"type":21,"tag":210,"props":852,"children":853},{"style":243},[854],{"type":26,"value":654},{"type":21,"tag":210,"props":856,"children":857},{"style":254},[858],{"type":26,"value":777},{"type":21,"tag":210,"props":860,"children":861},{"style":243},[862],{"type":26,"value":782},{"type":21,"tag":210,"props":864,"children":865},{"style":293},[866],{"type":26,"value":787},{"type":21,"tag":210,"props":868,"children":869},{"style":243},[870],{"type":26,"value":66},{"type":21,"tag":210,"props":872,"children":873},{"style":293},[874],{"type":26,"value":875},"error",{"type":21,"tag":210,"props":877,"children":878},{"style":243},[879],{"type":26,"value":782},{"type":21,"tag":210,"props":881,"children":882},{"style":293},[883],{"type":26,"value":805},{"type":21,"tag":210,"props":885,"children":886},{"style":243},[887],{"type":26,"value":66},{"type":21,"tag":210,"props":889,"children":890},{"style":293},[891],{"type":26,"value":892},"ban",{"type":21,"tag":210,"props":894,"children":895},{"style":243},[896],{"type":26,"value":273},{"type":21,"tag":210,"props":898,"children":899},{"class":212,"line":360},[900],{"type":21,"tag":210,"props":901,"children":902},{"style":237},[903],{"type":26,"value":904}," end\n",{"type":21,"tag":210,"props":906,"children":907},{"class":212,"line":402},[908,913],{"type":21,"tag":210,"props":909,"children":910},{"style":237},[911],{"type":26,"value":912},"end",{"type":21,"tag":210,"props":914,"children":915},{"style":243},[916],{"type":26,"value":273},{"type":21,"tag":200,"props":918,"children":920},{"className":538,"code":919,"language":12,"meta":7,"style":7},"(hs.hotkey.bind HYPER :0\n (fn [] (hs.caffeinate.toggle :displayIdle)\n (if (hs.caffeinate.get :displayIdle)\n (helpers:show \"Caffeine Enabled\" nil helpers.styles.success helpers.assets.check)\n (helpers:show \"Caffeine Disabled\" nil helpers.styles.error helpers.assets.ban)))) \n",[921],{"type":21,"tag":109,"props":922,"children":923},{"__ignoreMap":7},[924,937,954,980,1011],{"type":21,"tag":210,"props":925,"children":926},{"class":212,"line":213},[927,932],{"type":21,"tag":210,"props":928,"children":929},{"style":243},[930],{"type":26,"value":931},"(hs.hotkey.bind HYPER ",{"type":21,"tag":210,"props":933,"children":934},{"style":293},[935],{"type":26,"value":936},":0\n",{"type":21,"tag":210,"props":938,"children":939},{"class":212,"line":223},[940,945,950],{"type":21,"tag":210,"props":941,"children":942},{"style":243},[943],{"type":26,"value":944}," (fn [] (hs.caffeinate.toggle ",{"type":21,"tag":210,"props":946,"children":947},{"style":293},[948],{"type":26,"value":949},":displayIdle",{"type":21,"tag":210,"props":951,"children":952},{"style":243},[953],{"type":26,"value":273},{"type":21,"tag":210,"props":955,"children":956},{"class":212,"line":233},[957,962,967,972,976],{"type":21,"tag":210,"props":958,"children":959},{"style":243},[960],{"type":26,"value":961}," (",{"type":21,"tag":210,"props":963,"children":964},{"style":237},[965],{"type":26,"value":966},"if",{"type":21,"tag":210,"props":968,"children":969},{"style":243},[970],{"type":26,"value":971}," (hs.caffeinate.get ",{"type":21,"tag":210,"props":973,"children":974},{"style":293},[975],{"type":26,"value":949},{"type":21,"tag":210,"props":977,"children":978},{"style":243},[979],{"type":26,"value":273},{"type":21,"tag":210,"props":981,"children":982},{"class":212,"line":276},[983,988,992,997,1001,1006],{"type":21,"tag":210,"props":984,"children":985},{"style":243},[986],{"type":26,"value":987}," (",{"type":21,"tag":210,"props":989,"children":990},{"style":254},[991],{"type":26,"value":455},{"type":21,"tag":210,"props":993,"children":994},{"style":243},[995],{"type":26,"value":996},":show ",{"type":21,"tag":210,"props":998,"children":999},{"style":265},[1000],{"type":26,"value":768},{"type":21,"tag":210,"props":1002,"children":1003},{"style":254},[1004],{"type":26,"value":1005}," nil",{"type":21,"tag":210,"props":1007,"children":1008},{"style":243},[1009],{"type":26,"value":1010}," helpers.styles.success helpers.assets.check)\n",{"type":21,"tag":210,"props":1012,"children":1013},{"class":212,"line":284},[1014,1018,1022,1026,1030,1034],{"type":21,"tag":210,"props":1015,"children":1016},{"style":243},[1017],{"type":26,"value":987},{"type":21,"tag":210,"props":1019,"children":1020},{"style":254},[1021],{"type":26,"value":455},{"type":21,"tag":210,"props":1023,"children":1024},{"style":243},[1025],{"type":26,"value":996},{"type":21,"tag":210,"props":1027,"children":1028},{"style":265},[1029],{"type":26,"value":850},{"type":21,"tag":210,"props":1031,"children":1032},{"style":254},[1033],{"type":26,"value":1005},{"type":21,"tag":210,"props":1035,"children":1036},{"style":243},[1037],{"type":26,"value":1038}," helpers.styles.error helpers.assets.ban))))\n",{"type":21,"tag":22,"props":1040,"children":1041},{},[1042,1044,1050],{"type":26,"value":1043},"This was especially helpful for more gnarly modules like the ",{"type":21,"tag":109,"props":1045,"children":1047},{"className":1046},[],[1048],{"type":26,"value":1049},"window",{"type":26,"value":1051}," module used for\nwindow management, and seeing the Lua and Fennel code side-by-side was a kick starter in\nlearning the language!",{"type":21,"tag":68,"props":1053,"children":1055},{"id":1054},"next-steps",[1056],{"type":26,"value":1057},"Next Steps",{"type":21,"tag":22,"props":1059,"children":1060},{},[1061],{"type":26,"value":1062},"While my Fennel Hammerspoon configuration now works with parity to its Lua counterpart,\nI have not yet added new features or modules. I look forward to writing new Fennel code,\nand deepen my understanding of Lisp and the Fennel programming language.",{"type":21,"tag":22,"props":1064,"children":1065},{},[1066,1068,1075],{"type":26,"value":1067},"Additionally, before beginning this endeavor, I was already aware of projects like\n",{"type":21,"tag":39,"props":1069,"children":1072},{"href":1070,"rel":1071},"https://github.com/agzam/spacehammer",[43],[1073],{"type":26,"value":1074},"spacehammer",{"type":26,"value":1076},"; a wildly impressive Hammerspoon configuration written in Fennel, but,\nI wanted to start small and learn the integration myself. However, with the basics out\nof the way, I hope to explore this project further, and seek lessons-learned for the\nconfiguration of my own.",{"type":21,"tag":22,"props":1078,"children":1079},{},[1080,1082],{"type":26,"value":1081},"The full pull-request for translating my Lua Hammerspoon configuration to Fennel can be\nfound here: ",{"type":21,"tag":39,"props":1083,"children":1086},{"href":1084,"rel":1085},"https://github.com/cmpadden/dotfiles/pull/19/files",[43],[1087],{"type":26,"value":1084},{"type":21,"tag":1089,"props":1090,"children":1091},"style",{},[1092],{"type":26,"value":1093},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":223,"depth":223,"links":1095},[1096,1097,1098,1099],{"id":70,"depth":223,"text":73},{"id":90,"depth":223,"text":93},{"id":424,"depth":223,"text":427},{"id":1054,"depth":223,"text":1057},"markdown","content:articles:fennel-initial-exploration.md","content","articles/fennel-initial-exploration.md","articles/fennel-initial-exploration","md",1731781853249] \ No newline at end of file +[{"data":1,"prerenderedAt":1106},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"draft":6,"date":10,"tags":11,"categories":15,"cover_image":16,"excerpt":17,"body":27,"_type":1100,"_id":1101,"_source":1102,"_file":1103,"_stem":1104,"_extension":1105},"/articles/fennel-initial-exploration","articles",false,"","Impressions of Fennel with Hammerspoon","A while back I read an introductory book on Lisp programming titled, \"The Little\nSchemer\". This book opened my mind to new (to me) programming paradigms, and left me\nwith a strong desire to find a way to incorporate Lisp programming into my every day\nlife. It took some time, but I believe I've found what I've been looking for: Fennel.","2023-10-22",[12,13,14],"lisp","hammerspoon","fennel",[12],"/images/dall-e-fennel-hammer.jpeg",{"type":18,"children":19},"root",[20],{"type":21,"tag":22,"props":23,"children":24},"element","p",{},[25],{"type":26,"value":9},"text",{"type":18,"children":28,"toc":1094},[29,33,67,74,88,94,125,157,170,199,422,428,458,536,598,611,917,1039,1052,1058,1063,1077,1088],{"type":21,"tag":22,"props":30,"children":31},{},[32],{"type":26,"value":9},{"type":21,"tag":22,"props":34,"children":35},{},[36,38,47,49,56,58,65],{"type":26,"value":37},"The ",{"type":21,"tag":39,"props":40,"children":44},"a",{"href":41,"rel":42},"https://fennel-lang.org/",[43],"nofollow",[45],{"type":26,"value":46},"Fennel",{"type":26,"value":48}," programming language is a dialect of Lisp that boasts compatibility with\nLua, and it just so happens that two of my favorite applications are configured with\nexactly that language: ",{"type":21,"tag":39,"props":50,"children":53},{"href":51,"rel":52},"https://www.hammerspoon.org/",[43],[54],{"type":26,"value":55},"Hammerspoon",{"type":26,"value":57},", and ",{"type":21,"tag":39,"props":59,"children":62},{"href":60,"rel":61},"https://neovim.io/",[43],[63],{"type":26,"value":64},"Neovim",{"type":26,"value":66},".",{"type":21,"tag":68,"props":69,"children":71},"h2",{"id":70},"initial-observations",[72],{"type":26,"value":73},"Initial Observations",{"type":21,"tag":22,"props":75,"children":76},{},[77,79,86],{"type":26,"value":78},"To initially explore Fennel, I wanted to start small. My Hammerspoon configuration\nconsists of 7 ",{"type":21,"tag":39,"props":80,"children":83},{"href":81,"rel":82},"https://github.com/cmpadden/dotfiles/tree/795749fa17e1310bb001bb7deaa22be8689f0027/hammerspoon/.hammerspoon/modules",[43],[84],{"type":26,"value":85},"modules",{"type":26,"value":87}," that I use for operations such as: launching applications,\nmanaging windows, keeping my computer from going to sleep, and general operating system\nautomation. So the plan is to translate these modules into Fennel, while maintaining\nwithout breaking the existing functionality. However, at this point, I wasn't even sure\nhow to embed Fennel into my project...",{"type":21,"tag":68,"props":89,"children":91},{"id":90},"integrating-fennel-with-hammerspoon",[92],{"type":26,"value":93},"Integrating Fennel with Hammerspoon",{"type":21,"tag":22,"props":95,"children":96},{},[97,99,106,108,115,117,123],{"type":26,"value":98},"While official documentation exists describing how to ",{"type":21,"tag":39,"props":100,"children":103},{"href":101,"rel":102},"https://fennel-lang.org/setup#embedding-fennel",[43],[104],{"type":26,"value":105},"embed fennel",{"type":26,"value":107}," into your\nproject; it didn't provide me with enough clarity to know my next steps on integrating\nit with Hammerspoon. I found a few resources online demonstrating how to extend the\n",{"type":21,"tag":109,"props":110,"children":112},"code",{"className":111},[],[113],{"type":26,"value":114},"package.path",{"type":26,"value":116}," and ",{"type":21,"tag":109,"props":118,"children":120},{"className":119},[],[121],{"type":26,"value":122},"package.cpath",{"type":26,"value":124}," properties in Lua, but I was unable to get this to\nwork.",{"type":21,"tag":126,"props":127,"children":128},"ul",{},[129,139,148],{"type":21,"tag":130,"props":131,"children":132},"li",{},[133],{"type":21,"tag":39,"props":134,"children":137},{"href":135,"rel":136},"https://blog.exupero.org/hammerspoon-with-fennel/",[43],[138],{"type":26,"value":135},{"type":21,"tag":130,"props":140,"children":141},{},[142],{"type":21,"tag":39,"props":143,"children":146},{"href":144,"rel":145},"https://github.com/Hammerspoon/hammerspoon/issues/2377#issuecomment-636331435",[43],[147],{"type":26,"value":144},{"type":21,"tag":130,"props":149,"children":150},{},[151],{"type":21,"tag":39,"props":152,"children":155},{"href":153,"rel":154},"https://github.com/agzam/spacehammer/blob/master/init.lua",[43],[156],{"type":26,"value":153},{"type":21,"tag":22,"props":158,"children":159},{},[160,162,168],{"type":26,"value":161},"Ultimately, I opted to include the ",{"type":21,"tag":109,"props":163,"children":165},{"className":164},[],[166],{"type":26,"value":167},"fennel.lua",{"type":26,"value":169}," file to my Hammerspoon configuration,\nand while not ideal, it does make the configuration nicely self-contained. I'll leave it\nas a future task to include the module installed with LuaRocks.",{"type":21,"tag":22,"props":171,"children":172},{},[173,175,181,183,189,191,197],{"type":26,"value":174},"With Fennel now included in my Hammerspoon configuration, all I need to do is configure\nthe ",{"type":21,"tag":109,"props":176,"children":178},{"className":177},[],[179],{"type":26,"value":180},"fennel.path",{"type":26,"value":182}," to point to the ",{"type":21,"tag":109,"props":184,"children":186},{"className":185},[],[187],{"type":26,"value":188},"*.fnl",{"type":26,"value":190}," files in the ",{"type":21,"tag":109,"props":192,"children":194},{"className":193},[],[195],{"type":26,"value":196},".hammerspoon/",{"type":26,"value":198}," directory, and\nttranslating these modules can begin!",{"type":21,"tag":200,"props":201,"children":205},"pre",{"className":202,"code":203,"language":204,"meta":7,"style":7},"language-lua shiki shiki-themes github-light","-- init.lua\n\nlocal fennel = require('fennel')\n\nfennel.path = package.path .. \";\" .. os.getenv(\"HOME\") .. \"/.hammerspoon/?.fnl\"\n\ntable.insert(package.loaders or package.searchers, fennel.searcher)\n\nrequire 'main'\n","lua",[206],{"type":21,"tag":109,"props":207,"children":208},{"__ignoreMap":7},[209,221,231,274,282,350,358,400,408],{"type":21,"tag":210,"props":211,"children":214},"span",{"class":212,"line":213},"line",1,[215],{"type":21,"tag":210,"props":216,"children":218},{"style":217},"--shiki-default:#6A737D",[219],{"type":26,"value":220},"-- init.lua\n",{"type":21,"tag":210,"props":222,"children":224},{"class":212,"line":223},2,[225],{"type":21,"tag":210,"props":226,"children":228},{"emptyLinePlaceholder":227},true,[229],{"type":26,"value":230},"\n",{"type":21,"tag":210,"props":232,"children":234},{"class":212,"line":233},3,[235,241,247,252,258,263,269],{"type":21,"tag":210,"props":236,"children":238},{"style":237},"--shiki-default:#D73A49",[239],{"type":26,"value":240},"local",{"type":21,"tag":210,"props":242,"children":244},{"style":243},"--shiki-default:#24292E",[245],{"type":26,"value":246}," fennel ",{"type":21,"tag":210,"props":248,"children":249},{"style":237},[250],{"type":26,"value":251},"=",{"type":21,"tag":210,"props":253,"children":255},{"style":254},"--shiki-default:#005CC5",[256],{"type":26,"value":257}," require",{"type":21,"tag":210,"props":259,"children":260},{"style":243},[261],{"type":26,"value":262},"(",{"type":21,"tag":210,"props":264,"children":266},{"style":265},"--shiki-default:#032F62",[267],{"type":26,"value":268},"'fennel'",{"type":21,"tag":210,"props":270,"children":271},{"style":243},[272],{"type":26,"value":273},")\n",{"type":21,"tag":210,"props":275,"children":277},{"class":212,"line":276},4,[278],{"type":21,"tag":210,"props":279,"children":280},{"emptyLinePlaceholder":227},[281],{"type":26,"value":230},{"type":21,"tag":210,"props":283,"children":285},{"class":212,"line":284},5,[286,291,297,302,307,312,317,322,327,331,336,341,345],{"type":21,"tag":210,"props":287,"children":288},{"style":243},[289],{"type":26,"value":290},"fennel.",{"type":21,"tag":210,"props":292,"children":294},{"style":293},"--shiki-default:#6F42C1",[295],{"type":26,"value":296},"path",{"type":21,"tag":210,"props":298,"children":299},{"style":237},[300],{"type":26,"value":301}," =",{"type":21,"tag":210,"props":303,"children":304},{"style":254},[305],{"type":26,"value":306}," package.path",{"type":21,"tag":210,"props":308,"children":309},{"style":237},[310],{"type":26,"value":311}," ..",{"type":21,"tag":210,"props":313,"children":314},{"style":265},[315],{"type":26,"value":316}," \";\" ",{"type":21,"tag":210,"props":318,"children":319},{"style":237},[320],{"type":26,"value":321},"..",{"type":21,"tag":210,"props":323,"children":324},{"style":254},[325],{"type":26,"value":326}," os.getenv",{"type":21,"tag":210,"props":328,"children":329},{"style":243},[330],{"type":26,"value":262},{"type":21,"tag":210,"props":332,"children":333},{"style":265},[334],{"type":26,"value":335},"\"HOME\"",{"type":21,"tag":210,"props":337,"children":338},{"style":243},[339],{"type":26,"value":340},") ",{"type":21,"tag":210,"props":342,"children":343},{"style":237},[344],{"type":26,"value":321},{"type":21,"tag":210,"props":346,"children":347},{"style":265},[348],{"type":26,"value":349}," \"/.hammerspoon/?.fnl\"\n",{"type":21,"tag":210,"props":351,"children":353},{"class":212,"line":352},6,[354],{"type":21,"tag":210,"props":355,"children":356},{"emptyLinePlaceholder":227},[357],{"type":26,"value":230},{"type":21,"tag":210,"props":359,"children":361},{"class":212,"line":360},7,[362,367,371,376,381,386,391,396],{"type":21,"tag":210,"props":363,"children":364},{"style":254},[365],{"type":26,"value":366},"table.insert",{"type":21,"tag":210,"props":368,"children":369},{"style":243},[370],{"type":26,"value":262},{"type":21,"tag":210,"props":372,"children":373},{"style":254},[374],{"type":26,"value":375},"package.loaders",{"type":21,"tag":210,"props":377,"children":378},{"style":237},[379],{"type":26,"value":380}," or",{"type":21,"tag":210,"props":382,"children":383},{"style":254},[384],{"type":26,"value":385}," package.searchers",{"type":21,"tag":210,"props":387,"children":388},{"style":243},[389],{"type":26,"value":390},", fennel.",{"type":21,"tag":210,"props":392,"children":393},{"style":293},[394],{"type":26,"value":395},"searcher",{"type":21,"tag":210,"props":397,"children":398},{"style":243},[399],{"type":26,"value":273},{"type":21,"tag":210,"props":401,"children":403},{"class":212,"line":402},8,[404],{"type":21,"tag":210,"props":405,"children":406},{"emptyLinePlaceholder":227},[407],{"type":26,"value":230},{"type":21,"tag":210,"props":409,"children":411},{"class":212,"line":410},9,[412,417],{"type":21,"tag":210,"props":413,"children":414},{"style":254},[415],{"type":26,"value":416},"require",{"type":21,"tag":210,"props":418,"children":419},{"style":265},[420],{"type":26,"value":421}," 'main'\n",{"type":21,"tag":68,"props":423,"children":425},{"id":424},"translating-lua-to-fennel",[426],{"type":26,"value":427},"Translating Lua to Fennel",{"type":21,"tag":22,"props":429,"children":430},{},[431,433,440,442,448,450,456],{"type":26,"value":432},"As a Fennel novice, I was happy to see that the Fennel project provides an online\ncross-compiler for Lua and Fennel called ",{"type":21,"tag":39,"props":434,"children":437},{"href":435,"rel":436},"https://fennel-lang.org/see",[43],[438],{"type":26,"value":439},"anti-fennel",{"type":26,"value":441},", and while it can generate some\nstrange-looking Fennel code, it was an extremely useful tool for me to get\nup-and-running right away. For example, by pasting the simple ",{"type":21,"tag":109,"props":443,"children":445},{"className":444},[],[446],{"type":26,"value":447},"sleep",{"type":26,"value":449}," function\nfrom the ",{"type":21,"tag":109,"props":451,"children":453},{"className":452},[],[454],{"type":26,"value":455},"helpers",{"type":26,"value":457}," module into the compiler:",{"type":21,"tag":200,"props":459,"children":461},{"className":202,"code":460,"language":204,"meta":7,"style":7},"function sleep(ms)\n os.execute(\"sleep \" .. tonumber(ms) / 1000)\nend\n",[462],{"type":21,"tag":109,"props":463,"children":464},{"__ignoreMap":7},[465,483,528],{"type":21,"tag":210,"props":466,"children":467},{"class":212,"line":213},[468,473,478],{"type":21,"tag":210,"props":469,"children":470},{"style":237},[471],{"type":26,"value":472},"function",{"type":21,"tag":210,"props":474,"children":475},{"style":293},[476],{"type":26,"value":477}," sleep",{"type":21,"tag":210,"props":479,"children":480},{"style":243},[481],{"type":26,"value":482},"(ms)\n",{"type":21,"tag":210,"props":484,"children":485},{"class":212,"line":223},[486,491,495,500,504,509,514,519,524],{"type":21,"tag":210,"props":487,"children":488},{"style":254},[489],{"type":26,"value":490}," os.execute",{"type":21,"tag":210,"props":492,"children":493},{"style":243},[494],{"type":26,"value":262},{"type":21,"tag":210,"props":496,"children":497},{"style":265},[498],{"type":26,"value":499},"\"sleep \" ",{"type":21,"tag":210,"props":501,"children":502},{"style":237},[503],{"type":26,"value":321},{"type":21,"tag":210,"props":505,"children":506},{"style":254},[507],{"type":26,"value":508}," tonumber",{"type":21,"tag":210,"props":510,"children":511},{"style":243},[512],{"type":26,"value":513},"(ms) ",{"type":21,"tag":210,"props":515,"children":516},{"style":237},[517],{"type":26,"value":518},"/",{"type":21,"tag":210,"props":520,"children":521},{"style":254},[522],{"type":26,"value":523}," 1000",{"type":21,"tag":210,"props":525,"children":526},{"style":243},[527],{"type":26,"value":273},{"type":21,"tag":210,"props":529,"children":530},{"class":212,"line":233},[531],{"type":21,"tag":210,"props":532,"children":533},{"style":237},[534],{"type":26,"value":535},"end\n",{"type":21,"tag":200,"props":537,"children":540},{"className":538,"code":539,"language":12,"meta":7,"style":7},"language-lisp shiki shiki-themes github-light","(fn sleep [ms]\n (os.execute (.. \"sleep \" (/ (tonumber ms) 1000))))\n",[541],{"type":21,"tag":109,"props":542,"children":543},{"__ignoreMap":7},[544,561],{"type":21,"tag":210,"props":545,"children":546},{"class":212,"line":213},[547,552,556],{"type":21,"tag":210,"props":548,"children":549},{"style":243},[550],{"type":26,"value":551},"(fn ",{"type":21,"tag":210,"props":553,"children":554},{"style":254},[555],{"type":26,"value":447},{"type":21,"tag":210,"props":557,"children":558},{"style":243},[559],{"type":26,"value":560}," [ms]\n",{"type":21,"tag":210,"props":562,"children":563},{"class":212,"line":223},[564,569,574,579,583,588,593],{"type":21,"tag":210,"props":565,"children":566},{"style":243},[567],{"type":26,"value":568}," (os.execute (.. ",{"type":21,"tag":210,"props":570,"children":571},{"style":265},[572],{"type":26,"value":573},"\"sleep \"",{"type":21,"tag":210,"props":575,"children":576},{"style":243},[577],{"type":26,"value":578}," (",{"type":21,"tag":210,"props":580,"children":581},{"style":254},[582],{"type":26,"value":518},{"type":21,"tag":210,"props":584,"children":585},{"style":243},[586],{"type":26,"value":587}," (tonumber ms) ",{"type":21,"tag":210,"props":589,"children":590},{"style":254},[591],{"type":26,"value":592},"1000",{"type":21,"tag":210,"props":594,"children":595},{"style":243},[596],{"type":26,"value":597},"))))\n",{"type":21,"tag":22,"props":599,"children":600},{},[601,603,609],{"type":26,"value":602},"As another example, here is the output for my ",{"type":21,"tag":604,"props":605,"children":606},"em",{},[607],{"type":26,"value":608},"caffeine",{"type":26,"value":610}," toggle:",{"type":21,"tag":200,"props":612,"children":614},{"className":202,"code":613,"language":204,"meta":7,"style":7},"hs.hotkey.bind(HYPER, \"0\", function()\n hs.caffeinate.toggle(\"displayIdle\")\n if hs.caffeinate.get(\"displayIdle\") then\n helpers:show(\"Caffeine Enabled\", nil, helpers.styles.success, helpers.assets.check)\n else\n helpers:show(\"Caffeine Disabled\", nil, helpers.styles.error, helpers.assets.ban)\n end\nend)\n",[615],{"type":21,"tag":109,"props":616,"children":617},{"__ignoreMap":7},[618,664,699,742,819,827,897,905],{"type":21,"tag":210,"props":619,"children":620},{"class":212,"line":213},[621,626,631,635,640,645,650,655,659],{"type":21,"tag":210,"props":622,"children":623},{"style":243},[624],{"type":26,"value":625},"hs.",{"type":21,"tag":210,"props":627,"children":628},{"style":293},[629],{"type":26,"value":630},"hotkey",{"type":21,"tag":210,"props":632,"children":633},{"style":243},[634],{"type":26,"value":66},{"type":21,"tag":210,"props":636,"children":637},{"style":254},[638],{"type":26,"value":639},"bind",{"type":21,"tag":210,"props":641,"children":642},{"style":243},[643],{"type":26,"value":644},"(HYPER, ",{"type":21,"tag":210,"props":646,"children":647},{"style":265},[648],{"type":26,"value":649},"\"0\"",{"type":21,"tag":210,"props":651,"children":652},{"style":243},[653],{"type":26,"value":654},", ",{"type":21,"tag":210,"props":656,"children":657},{"style":237},[658],{"type":26,"value":472},{"type":21,"tag":210,"props":660,"children":661},{"style":243},[662],{"type":26,"value":663},"()\n",{"type":21,"tag":210,"props":665,"children":666},{"class":212,"line":223},[667,672,677,681,686,690,695],{"type":21,"tag":210,"props":668,"children":669},{"style":243},[670],{"type":26,"value":671}," hs.",{"type":21,"tag":210,"props":673,"children":674},{"style":293},[675],{"type":26,"value":676},"caffeinate",{"type":21,"tag":210,"props":678,"children":679},{"style":243},[680],{"type":26,"value":66},{"type":21,"tag":210,"props":682,"children":683},{"style":254},[684],{"type":26,"value":685},"toggle",{"type":21,"tag":210,"props":687,"children":688},{"style":243},[689],{"type":26,"value":262},{"type":21,"tag":210,"props":691,"children":692},{"style":265},[693],{"type":26,"value":694},"\"displayIdle\"",{"type":21,"tag":210,"props":696,"children":697},{"style":243},[698],{"type":26,"value":273},{"type":21,"tag":210,"props":700,"children":701},{"class":212,"line":233},[702,707,712,716,720,725,729,733,737],{"type":21,"tag":210,"props":703,"children":704},{"style":237},[705],{"type":26,"value":706}," if",{"type":21,"tag":210,"props":708,"children":709},{"style":243},[710],{"type":26,"value":711}," hs.",{"type":21,"tag":210,"props":713,"children":714},{"style":293},[715],{"type":26,"value":676},{"type":21,"tag":210,"props":717,"children":718},{"style":243},[719],{"type":26,"value":66},{"type":21,"tag":210,"props":721,"children":722},{"style":254},[723],{"type":26,"value":724},"get",{"type":21,"tag":210,"props":726,"children":727},{"style":243},[728],{"type":26,"value":262},{"type":21,"tag":210,"props":730,"children":731},{"style":265},[732],{"type":26,"value":694},{"type":21,"tag":210,"props":734,"children":735},{"style":243},[736],{"type":26,"value":340},{"type":21,"tag":210,"props":738,"children":739},{"style":237},[740],{"type":26,"value":741},"then\n",{"type":21,"tag":210,"props":743,"children":744},{"class":212,"line":276},[745,750,755,760,764,769,773,778,783,788,792,797,801,806,810,815],{"type":21,"tag":210,"props":746,"children":747},{"style":293},[748],{"type":26,"value":749}," helpers",{"type":21,"tag":210,"props":751,"children":752},{"style":243},[753],{"type":26,"value":754},":",{"type":21,"tag":210,"props":756,"children":757},{"style":254},[758],{"type":26,"value":759},"show",{"type":21,"tag":210,"props":761,"children":762},{"style":243},[763],{"type":26,"value":262},{"type":21,"tag":210,"props":765,"children":766},{"style":265},[767],{"type":26,"value":768},"\"Caffeine Enabled\"",{"type":21,"tag":210,"props":770,"children":771},{"style":243},[772],{"type":26,"value":654},{"type":21,"tag":210,"props":774,"children":775},{"style":254},[776],{"type":26,"value":777},"nil",{"type":21,"tag":210,"props":779,"children":780},{"style":243},[781],{"type":26,"value":782},", helpers.",{"type":21,"tag":210,"props":784,"children":785},{"style":293},[786],{"type":26,"value":787},"styles",{"type":21,"tag":210,"props":789,"children":790},{"style":243},[791],{"type":26,"value":66},{"type":21,"tag":210,"props":793,"children":794},{"style":293},[795],{"type":26,"value":796},"success",{"type":21,"tag":210,"props":798,"children":799},{"style":243},[800],{"type":26,"value":782},{"type":21,"tag":210,"props":802,"children":803},{"style":293},[804],{"type":26,"value":805},"assets",{"type":21,"tag":210,"props":807,"children":808},{"style":243},[809],{"type":26,"value":66},{"type":21,"tag":210,"props":811,"children":812},{"style":293},[813],{"type":26,"value":814},"check",{"type":21,"tag":210,"props":816,"children":817},{"style":243},[818],{"type":26,"value":273},{"type":21,"tag":210,"props":820,"children":821},{"class":212,"line":284},[822],{"type":21,"tag":210,"props":823,"children":824},{"style":237},[825],{"type":26,"value":826}," else\n",{"type":21,"tag":210,"props":828,"children":829},{"class":212,"line":352},[830,834,838,842,846,851,855,859,863,867,871,876,880,884,888,893],{"type":21,"tag":210,"props":831,"children":832},{"style":293},[833],{"type":26,"value":749},{"type":21,"tag":210,"props":835,"children":836},{"style":243},[837],{"type":26,"value":754},{"type":21,"tag":210,"props":839,"children":840},{"style":254},[841],{"type":26,"value":759},{"type":21,"tag":210,"props":843,"children":844},{"style":243},[845],{"type":26,"value":262},{"type":21,"tag":210,"props":847,"children":848},{"style":265},[849],{"type":26,"value":850},"\"Caffeine Disabled\"",{"type":21,"tag":210,"props":852,"children":853},{"style":243},[854],{"type":26,"value":654},{"type":21,"tag":210,"props":856,"children":857},{"style":254},[858],{"type":26,"value":777},{"type":21,"tag":210,"props":860,"children":861},{"style":243},[862],{"type":26,"value":782},{"type":21,"tag":210,"props":864,"children":865},{"style":293},[866],{"type":26,"value":787},{"type":21,"tag":210,"props":868,"children":869},{"style":243},[870],{"type":26,"value":66},{"type":21,"tag":210,"props":872,"children":873},{"style":293},[874],{"type":26,"value":875},"error",{"type":21,"tag":210,"props":877,"children":878},{"style":243},[879],{"type":26,"value":782},{"type":21,"tag":210,"props":881,"children":882},{"style":293},[883],{"type":26,"value":805},{"type":21,"tag":210,"props":885,"children":886},{"style":243},[887],{"type":26,"value":66},{"type":21,"tag":210,"props":889,"children":890},{"style":293},[891],{"type":26,"value":892},"ban",{"type":21,"tag":210,"props":894,"children":895},{"style":243},[896],{"type":26,"value":273},{"type":21,"tag":210,"props":898,"children":899},{"class":212,"line":360},[900],{"type":21,"tag":210,"props":901,"children":902},{"style":237},[903],{"type":26,"value":904}," end\n",{"type":21,"tag":210,"props":906,"children":907},{"class":212,"line":402},[908,913],{"type":21,"tag":210,"props":909,"children":910},{"style":237},[911],{"type":26,"value":912},"end",{"type":21,"tag":210,"props":914,"children":915},{"style":243},[916],{"type":26,"value":273},{"type":21,"tag":200,"props":918,"children":920},{"className":538,"code":919,"language":12,"meta":7,"style":7},"(hs.hotkey.bind HYPER :0\n (fn [] (hs.caffeinate.toggle :displayIdle)\n (if (hs.caffeinate.get :displayIdle)\n (helpers:show \"Caffeine Enabled\" nil helpers.styles.success helpers.assets.check)\n (helpers:show \"Caffeine Disabled\" nil helpers.styles.error helpers.assets.ban)))) \n",[921],{"type":21,"tag":109,"props":922,"children":923},{"__ignoreMap":7},[924,937,954,980,1011],{"type":21,"tag":210,"props":925,"children":926},{"class":212,"line":213},[927,932],{"type":21,"tag":210,"props":928,"children":929},{"style":243},[930],{"type":26,"value":931},"(hs.hotkey.bind HYPER ",{"type":21,"tag":210,"props":933,"children":934},{"style":293},[935],{"type":26,"value":936},":0\n",{"type":21,"tag":210,"props":938,"children":939},{"class":212,"line":223},[940,945,950],{"type":21,"tag":210,"props":941,"children":942},{"style":243},[943],{"type":26,"value":944}," (fn [] (hs.caffeinate.toggle ",{"type":21,"tag":210,"props":946,"children":947},{"style":293},[948],{"type":26,"value":949},":displayIdle",{"type":21,"tag":210,"props":951,"children":952},{"style":243},[953],{"type":26,"value":273},{"type":21,"tag":210,"props":955,"children":956},{"class":212,"line":233},[957,962,967,972,976],{"type":21,"tag":210,"props":958,"children":959},{"style":243},[960],{"type":26,"value":961}," (",{"type":21,"tag":210,"props":963,"children":964},{"style":237},[965],{"type":26,"value":966},"if",{"type":21,"tag":210,"props":968,"children":969},{"style":243},[970],{"type":26,"value":971}," (hs.caffeinate.get ",{"type":21,"tag":210,"props":973,"children":974},{"style":293},[975],{"type":26,"value":949},{"type":21,"tag":210,"props":977,"children":978},{"style":243},[979],{"type":26,"value":273},{"type":21,"tag":210,"props":981,"children":982},{"class":212,"line":276},[983,988,992,997,1001,1006],{"type":21,"tag":210,"props":984,"children":985},{"style":243},[986],{"type":26,"value":987}," (",{"type":21,"tag":210,"props":989,"children":990},{"style":254},[991],{"type":26,"value":455},{"type":21,"tag":210,"props":993,"children":994},{"style":243},[995],{"type":26,"value":996},":show ",{"type":21,"tag":210,"props":998,"children":999},{"style":265},[1000],{"type":26,"value":768},{"type":21,"tag":210,"props":1002,"children":1003},{"style":254},[1004],{"type":26,"value":1005}," nil",{"type":21,"tag":210,"props":1007,"children":1008},{"style":243},[1009],{"type":26,"value":1010}," helpers.styles.success helpers.assets.check)\n",{"type":21,"tag":210,"props":1012,"children":1013},{"class":212,"line":284},[1014,1018,1022,1026,1030,1034],{"type":21,"tag":210,"props":1015,"children":1016},{"style":243},[1017],{"type":26,"value":987},{"type":21,"tag":210,"props":1019,"children":1020},{"style":254},[1021],{"type":26,"value":455},{"type":21,"tag":210,"props":1023,"children":1024},{"style":243},[1025],{"type":26,"value":996},{"type":21,"tag":210,"props":1027,"children":1028},{"style":265},[1029],{"type":26,"value":850},{"type":21,"tag":210,"props":1031,"children":1032},{"style":254},[1033],{"type":26,"value":1005},{"type":21,"tag":210,"props":1035,"children":1036},{"style":243},[1037],{"type":26,"value":1038}," helpers.styles.error helpers.assets.ban))))\n",{"type":21,"tag":22,"props":1040,"children":1041},{},[1042,1044,1050],{"type":26,"value":1043},"This was especially helpful for more gnarly modules like the ",{"type":21,"tag":109,"props":1045,"children":1047},{"className":1046},[],[1048],{"type":26,"value":1049},"window",{"type":26,"value":1051}," module used for\nwindow management, and seeing the Lua and Fennel code side-by-side was a kick starter in\nlearning the language!",{"type":21,"tag":68,"props":1053,"children":1055},{"id":1054},"next-steps",[1056],{"type":26,"value":1057},"Next Steps",{"type":21,"tag":22,"props":1059,"children":1060},{},[1061],{"type":26,"value":1062},"While my Fennel Hammerspoon configuration now works with parity to its Lua counterpart,\nI have not yet added new features or modules. I look forward to writing new Fennel code,\nand deepen my understanding of Lisp and the Fennel programming language.",{"type":21,"tag":22,"props":1064,"children":1065},{},[1066,1068,1075],{"type":26,"value":1067},"Additionally, before beginning this endeavor, I was already aware of projects like\n",{"type":21,"tag":39,"props":1069,"children":1072},{"href":1070,"rel":1071},"https://github.com/agzam/spacehammer",[43],[1073],{"type":26,"value":1074},"spacehammer",{"type":26,"value":1076},"; a wildly impressive Hammerspoon configuration written in Fennel, but,\nI wanted to start small and learn the integration myself. However, with the basics out\nof the way, I hope to explore this project further, and seek lessons-learned for the\nconfiguration of my own.",{"type":21,"tag":22,"props":1078,"children":1079},{},[1080,1082],{"type":26,"value":1081},"The full pull-request for translating my Lua Hammerspoon configuration to Fennel can be\nfound here: ",{"type":21,"tag":39,"props":1083,"children":1086},{"href":1084,"rel":1085},"https://github.com/cmpadden/dotfiles/pull/19/files",[43],[1087],{"type":26,"value":1084},{"type":21,"tag":1089,"props":1090,"children":1091},"style",{},[1092],{"type":26,"value":1093},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":223,"depth":223,"links":1095},[1096,1097,1098,1099],{"id":70,"depth":223,"text":73},{"id":90,"depth":223,"text":93},{"id":424,"depth":223,"text":427},{"id":1054,"depth":223,"text":1057},"markdown","content:articles:fennel-initial-exploration.md","content","articles/fennel-initial-exploration.md","articles/fennel-initial-exploration","md",1731782033492] \ No newline at end of file diff --git a/articles/fennel-initial-exploration/index.html b/articles/fennel-initial-exploration/index.html index f2e35c85..0849a8b5 100644 --- a/articles/fennel-initial-exploration/index.html +++ b/articles/fennel-initial-exploration/index.html @@ -5,35 +5,35 @@ - - - - - + + + + + - + - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + -

        Impressions of Fennel with Hammerspoon

        A while back I read an introductory book on Lisp programming titled, "The Little +

        Impressions of Fennel with Hammerspoon

        A while back I read an introductory book on Lisp programming titled, "The Little Schemer". This book opened my mind to new (to me) programming paradigms, and left me with a strong desire to find a way to incorporate Lisp programming into my every day life. It took some time, but I believe I've found what I've been looking for: Fennel.

        The Fennel programming language is a dialect of Lisp that boasts compatibility with @@ -91,5 +91,5 @@ I wanted to start small and learn the integration myself. However, with the basics out of the way, I hope to explore this project further, and seek lessons-learned for the configuration of my own.

        The full pull-request for translating my Lua Hammerspoon configuration to Fennel can be -found here: https://github.com/cmpadden/dotfiles/pull/19/files

        - \ No newline at end of file +found here: https://github.com/cmpadden/dotfiles/pull/19/files

        + \ No newline at end of file diff --git a/articles/index.html b/articles/index.html index 2b6b9587..ae8ccbd0 100644 --- a/articles/index.html +++ b/articles/index.html @@ -2,29 +2,29 @@ Colton Padden - - - - - + + + + + - + - - - - - - - - - - + + + + + + + + + + -

        Blog

        2024-06-09
        Configuring a YubiKey for use with OpenSSH

        YubiKey's are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the "new" ed25519-sk key type that supports FIDO compliant hardware keys.

        2024-05-11
        Using pinentry-mac to sign commits from vim-fugitive

        In order to sign git commits from within Vim using a plugin like tpope/vim-fugitive, it is necessary to configure the gpg-agent to use a GUI based pinentry-program.

        2024-01-08
        Easily Transcribe Podcasts with Whisper.cpp

        If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive whisper.cpp project. This high-performance fork of OpenAI's Whisper can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the Alter Everything podcast.

        2024-01-06
        How To Add an RSS Feed to a Nuxt Website

        If you are a user of Nuxt Content and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.

        2023-10-22
        Impressions of Fennel with Hammerspoon

        A while back I read an introductory book on Lisp programming titled, "The Little +

        Blog

        2024-06-09
        Configuring a YubiKey for use with OpenSSH

        YubiKey's are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the "new" ed25519-sk key type that supports FIDO compliant hardware keys.

        2024-05-11
        Using pinentry-mac to sign commits from vim-fugitive

        In order to sign git commits from within Vim using a plugin like tpope/vim-fugitive, it is necessary to configure the gpg-agent to use a GUI based pinentry-program.

        2024-01-08
        Easily Transcribe Podcasts with Whisper.cpp

        If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive whisper.cpp project. This high-performance fork of OpenAI's Whisper can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the Alter Everything podcast.

        2024-01-06
        How To Add an RSS Feed to a Nuxt Website

        If you are a user of Nuxt Content and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.

        2023-10-22
        Impressions of Fennel with Hammerspoon

        A while back I read an introductory book on Lisp programming titled, "The Little Schemer". This book opened my mind to new (to me) programming paradigms, and left me with a strong desire to find a way to incorporate Lisp programming into my every day life. It took some time, but I believe I've found what I've been looking for: Fennel.

        2023-01-01
        Exploring the Digital Ocean `doctl` Utility

        I recently had the need to spool up a small ephemeral Linux instance, and wanted to try something new by exploring the Digital Ocean doctl command line utility. @@ -45,5 +45,5 @@ few years, and have recently decided to dust it off.

        Since the last time the device has been powered on, there have been many great improvements to the firmware, and it was very-much due for an upgrade. The following steps outline how the firmware was upgraded on the APU from the -already-installed operating system -- CentOS.

        Categories

        embedded
        homelab
        linux
        lisp
        programming
        python
        tips
        tooling
        web

        Tags

        apu
        archlinux
        bash
        configurations
        digital-ocean
        docker
        fennel
        hammerspoon
        homelab
        linux
        lisp
        micropython
        ml
        mocks
        nuxt
        pcengine
        rss
        selinux
        supermicro
        testing
        tip
        truenas
        tutorial
        unix
        vagrant
        vim
        whisper.cpp
        - \ No newline at end of file +already-installed operating system -- CentOS.

        Categories

        embedded
        homelab
        linux
        lisp
        programming
        python
        tips
        tooling
        web

        Tags

        apu
        archlinux
        bash
        configurations
        digital-ocean
        docker
        fennel
        hammerspoon
        homelab
        linux
        lisp
        micropython
        ml
        mocks
        nuxt
        pcengine
        rss
        selinux
        supermicro
        testing
        tip
        truenas
        tutorial
        unix
        vagrant
        vim
        whisper.cpp
        + \ No newline at end of file diff --git a/articles/migrate-truenas-from-core-to-scale/_payload.json b/articles/migrate-truenas-from-core-to-scale/_payload.json index 8571f4fa..5f00c163 100644 --- a/articles/migrate-truenas-from-core-to-scale/_payload.json +++ b/articles/migrate-truenas-from-core-to-scale/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":205},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"tags":11,"categories":15,"excerpt":16,"body":26,"_type":199,"_id":200,"_source":201,"_file":202,"_stem":203,"_extension":204},"/articles/migrate-truenas-from-core-to-scale","articles",false,"","Migrate to TrueNAS Scale from TrueNAS Core","TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale.","2021-12-28",[12,13,14],"homelab","supermicro","truenas",[12],{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24],{"type":25,"value":9},"text",{"type":17,"children":27,"toc":196},[28,32,45,175,180],{"type":20,"tag":21,"props":29,"children":30},{},[31],{"type":25,"value":9},{"type":20,"tag":21,"props":33,"children":34},{},[35,37,43],{"type":25,"value":36},"Thankfully, the upgrade procedure to migrate from TrueNAS Core to TrueNAS Scale is relatively straight forward. All it requires is to create a bootable USB of the TrueNAS Scale image, boot the USB, and select ",{"type":20,"tag":38,"props":39,"children":40},"em",{},[41],{"type":25,"value":42},"Install/Upgrade",{"type":25,"value":44}," in the installation wizard. But for the sake of being thorough, you can find instructions on how to backup system configurations and install the OS below.",{"type":20,"tag":46,"props":47,"children":48},"ol",{},[49,96,125,130,154,159],{"type":20,"tag":50,"props":51,"children":52},"li",{},[53,55],{"type":25,"value":54},"Make a backup of your system’s configuration\n",{"type":20,"tag":46,"props":56,"children":57},{},[58,70],{"type":20,"tag":50,"props":59,"children":60},{},[61,63],{"type":25,"value":62},"Navigate to ",{"type":20,"tag":64,"props":65,"children":67},"code",{"className":66},[],[68],{"type":25,"value":69},"System > General",{"type":20,"tag":50,"props":71,"children":72},{},[73,75,81,83,88,90],{"type":25,"value":74},"Click ",{"type":20,"tag":64,"props":76,"children":78},{"className":77},[],[79],{"type":25,"value":80},"Save Config",{"type":25,"value":82},", check the ",{"type":20,"tag":38,"props":84,"children":85},{},[86],{"type":25,"value":87},"Export Secret Seed",{"type":25,"value":89}," box, and click ",{"type":20,"tag":64,"props":91,"children":93},{"className":92},[],[94],{"type":25,"value":95},"Save",{"type":20,"tag":50,"props":97,"children":98},{},[99,101],{"type":25,"value":100},"Export dataset keys for the encrypted pools\n",{"type":20,"tag":46,"props":102,"children":103},{},[104,114],{"type":20,"tag":50,"props":105,"children":106},{},[107,108],{"type":25,"value":62},{"type":20,"tag":64,"props":109,"children":111},{"className":110},[],[112],{"type":25,"value":113},"Storage > Pools",{"type":20,"tag":50,"props":115,"children":116},{},[117,119],{"type":25,"value":118},"Click the cog icon, and select ",{"type":20,"tag":64,"props":120,"children":122},{"className":121},[],[123],{"type":25,"value":124},"Export Dataset Keys",{"type":20,"tag":50,"props":126,"children":127},{},[128],{"type":25,"value":129},"Insert the TrueNAS Core bootable USB into the NAS",{"type":20,"tag":50,"props":131,"children":132},{},[133,135,140,142,147,149],{"type":25,"value":134},"From the Supermicro IPMI interface select ",{"type":20,"tag":38,"props":136,"children":137},{},[138],{"type":25,"value":139},"Remote Control",{"type":25,"value":141}," and ",{"type":20,"tag":38,"props":143,"children":144},{},[145],{"type":25,"value":146},"iKVM/HTML5",{"type":25,"value":148}," and select ",{"type":20,"tag":38,"props":150,"children":151},{},[152],{"type":25,"value":153},"Reboot",{"type":20,"tag":50,"props":155,"children":156},{},[157],{"type":25,"value":158},"Select the bootable USB as the boot device",{"type":20,"tag":50,"props":160,"children":161},{},[162,164,168,170],{"type":25,"value":163},"From the TrueNAS installation wizard, select ",{"type":20,"tag":38,"props":165,"children":166},{},[167],{"type":25,"value":42},{"type":25,"value":169},", select the drive that contains the TrueNAS installation, and select ",{"type":20,"tag":38,"props":171,"children":172},{},[173],{"type":25,"value":174},"Upgrade Install",{"type":20,"tag":21,"props":176,"children":177},{},[178],{"type":25,"value":179},"Reboot the device, and voila — you should be up-and-running! Give the system a quick rundown to validate that your settings and pools have transferred correctly, and then enjoy all the container goodness!",{"type":20,"tag":21,"props":181,"children":182},{},[183,185,194],{"type":25,"value":184},"For a breakdown of the differences between TrueNAS Core, Enterprise, and Scale, you can reference ",{"type":20,"tag":186,"props":187,"children":191},"a",{"href":188,"rel":189},"https://www.truenas.com/help-me-choose/",[190],"nofollow",[192],{"type":25,"value":193},"this table",{"type":25,"value":195},".",{"title":7,"searchDepth":197,"depth":197,"links":198},2,[],"markdown","content:articles:migrate-truenas-from-core-to-scale.md","content","articles/migrate-truenas-from-core-to-scale.md","articles/migrate-truenas-from-core-to-scale","md",1731781853259] \ No newline at end of file +[{"data":1,"prerenderedAt":205},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"tags":11,"categories":15,"excerpt":16,"body":26,"_type":199,"_id":200,"_source":201,"_file":202,"_stem":203,"_extension":204},"/articles/migrate-truenas-from-core-to-scale","articles",false,"","Migrate to TrueNAS Scale from TrueNAS Core","TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale.","2021-12-28",[12,13,14],"homelab","supermicro","truenas",[12],{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24],{"type":25,"value":9},"text",{"type":17,"children":27,"toc":196},[28,32,45,175,180],{"type":20,"tag":21,"props":29,"children":30},{},[31],{"type":25,"value":9},{"type":20,"tag":21,"props":33,"children":34},{},[35,37,43],{"type":25,"value":36},"Thankfully, the upgrade procedure to migrate from TrueNAS Core to TrueNAS Scale is relatively straight forward. All it requires is to create a bootable USB of the TrueNAS Scale image, boot the USB, and select ",{"type":20,"tag":38,"props":39,"children":40},"em",{},[41],{"type":25,"value":42},"Install/Upgrade",{"type":25,"value":44}," in the installation wizard. But for the sake of being thorough, you can find instructions on how to backup system configurations and install the OS below.",{"type":20,"tag":46,"props":47,"children":48},"ol",{},[49,96,125,130,154,159],{"type":20,"tag":50,"props":51,"children":52},"li",{},[53,55],{"type":25,"value":54},"Make a backup of your system’s configuration\n",{"type":20,"tag":46,"props":56,"children":57},{},[58,70],{"type":20,"tag":50,"props":59,"children":60},{},[61,63],{"type":25,"value":62},"Navigate to ",{"type":20,"tag":64,"props":65,"children":67},"code",{"className":66},[],[68],{"type":25,"value":69},"System > General",{"type":20,"tag":50,"props":71,"children":72},{},[73,75,81,83,88,90],{"type":25,"value":74},"Click ",{"type":20,"tag":64,"props":76,"children":78},{"className":77},[],[79],{"type":25,"value":80},"Save Config",{"type":25,"value":82},", check the ",{"type":20,"tag":38,"props":84,"children":85},{},[86],{"type":25,"value":87},"Export Secret Seed",{"type":25,"value":89}," box, and click ",{"type":20,"tag":64,"props":91,"children":93},{"className":92},[],[94],{"type":25,"value":95},"Save",{"type":20,"tag":50,"props":97,"children":98},{},[99,101],{"type":25,"value":100},"Export dataset keys for the encrypted pools\n",{"type":20,"tag":46,"props":102,"children":103},{},[104,114],{"type":20,"tag":50,"props":105,"children":106},{},[107,108],{"type":25,"value":62},{"type":20,"tag":64,"props":109,"children":111},{"className":110},[],[112],{"type":25,"value":113},"Storage > Pools",{"type":20,"tag":50,"props":115,"children":116},{},[117,119],{"type":25,"value":118},"Click the cog icon, and select ",{"type":20,"tag":64,"props":120,"children":122},{"className":121},[],[123],{"type":25,"value":124},"Export Dataset Keys",{"type":20,"tag":50,"props":126,"children":127},{},[128],{"type":25,"value":129},"Insert the TrueNAS Core bootable USB into the NAS",{"type":20,"tag":50,"props":131,"children":132},{},[133,135,140,142,147,149],{"type":25,"value":134},"From the Supermicro IPMI interface select ",{"type":20,"tag":38,"props":136,"children":137},{},[138],{"type":25,"value":139},"Remote Control",{"type":25,"value":141}," and ",{"type":20,"tag":38,"props":143,"children":144},{},[145],{"type":25,"value":146},"iKVM/HTML5",{"type":25,"value":148}," and select ",{"type":20,"tag":38,"props":150,"children":151},{},[152],{"type":25,"value":153},"Reboot",{"type":20,"tag":50,"props":155,"children":156},{},[157],{"type":25,"value":158},"Select the bootable USB as the boot device",{"type":20,"tag":50,"props":160,"children":161},{},[162,164,168,170],{"type":25,"value":163},"From the TrueNAS installation wizard, select ",{"type":20,"tag":38,"props":165,"children":166},{},[167],{"type":25,"value":42},{"type":25,"value":169},", select the drive that contains the TrueNAS installation, and select ",{"type":20,"tag":38,"props":171,"children":172},{},[173],{"type":25,"value":174},"Upgrade Install",{"type":20,"tag":21,"props":176,"children":177},{},[178],{"type":25,"value":179},"Reboot the device, and voila — you should be up-and-running! Give the system a quick rundown to validate that your settings and pools have transferred correctly, and then enjoy all the container goodness!",{"type":20,"tag":21,"props":181,"children":182},{},[183,185,194],{"type":25,"value":184},"For a breakdown of the differences between TrueNAS Core, Enterprise, and Scale, you can reference ",{"type":20,"tag":186,"props":187,"children":191},"a",{"href":188,"rel":189},"https://www.truenas.com/help-me-choose/",[190],"nofollow",[192],{"type":25,"value":193},"this table",{"type":25,"value":195},".",{"title":7,"searchDepth":197,"depth":197,"links":198},2,[],"markdown","content:articles:migrate-truenas-from-core-to-scale.md","content","articles/migrate-truenas-from-core-to-scale.md","articles/migrate-truenas-from-core-to-scale","md",1731782033528] \ No newline at end of file diff --git a/articles/migrate-truenas-from-core-to-scale/index.html b/articles/migrate-truenas-from-core-to-scale/index.html index 5da37379..7f05675c 100644 --- a/articles/migrate-truenas-from-core-to-scale/index.html +++ b/articles/migrate-truenas-from-core-to-scale/index.html @@ -3,31 +3,31 @@ Colton Padden - - - - - + + + + + - + - - - - - - - - - - - - + + + + + + + + + + + + -

        Migrate to TrueNAS Scale from TrueNAS Core

        TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale.

        Thankfully, the upgrade procedure to migrate from TrueNAS Core to TrueNAS Scale is relatively straight forward. All it requires is to create a bootable USB of the TrueNAS Scale image, boot the USB, and select Install/Upgrade in the installation wizard. But for the sake of being thorough, you can find instructions on how to backup system configurations and install the OS below.

        1. Make a backup of your system’s configuration +

          Migrate to TrueNAS Scale from TrueNAS Core

          TrueNAS Scale offers a variety of interesting features that are not supported in its TrueNAS Core counterpart. Most notably, because the base operating system is Debian Linux, opposed to FreeBSD, there is native support for Linux containers via Docker, and Kubernetes Pods! This post walks through the steps required to upgrade from TrueNAS Core to TrueNAS Scale.

          Thankfully, the upgrade procedure to migrate from TrueNAS Core to TrueNAS Scale is relatively straight forward. All it requires is to create a bootable USB of the TrueNAS Scale image, boot the USB, and select Install/Upgrade in the installation wizard. But for the sake of being thorough, you can find instructions on how to backup system configurations and install the OS below.

          1. Make a backup of your system’s configuration
            1. Navigate to System > General
            2. Click Save Config, check the Export Secret Seed box, and click Save
          2. Export dataset keys for the encrypted pools -
            1. Navigate to Storage > Pools
            2. Click the cog icon, and select Export Dataset Keys
          3. Insert the TrueNAS Core bootable USB into the NAS
          4. From the Supermicro IPMI interface select Remote Control and iKVM/HTML5 and select Reboot
          5. Select the bootable USB as the boot device
          6. From the TrueNAS installation wizard, select Install/Upgrade, select the drive that contains the TrueNAS installation, and select Upgrade Install

          Reboot the device, and voila — you should be up-and-running! Give the system a quick rundown to validate that your settings and pools have transferred correctly, and then enjoy all the container goodness!

          For a breakdown of the differences between TrueNAS Core, Enterprise, and Scale, you can reference this table.

          - \ No newline at end of file +
          1. Navigate to Storage > Pools
          2. Click the cog icon, and select Export Dataset Keys
        2. Insert the TrueNAS Core bootable USB into the NAS
        3. From the Supermicro IPMI interface select Remote Control and iKVM/HTML5 and select Reboot
        4. Select the bootable USB as the boot device
        5. From the TrueNAS installation wizard, select Install/Upgrade, select the drive that contains the TrueNAS installation, and select Upgrade Install

        Reboot the device, and voila — you should be up-and-running! Give the system a quick rundown to validate that your settings and pools have transferred correctly, and then enjoy all the container goodness!

        For a breakdown of the differences between TrueNAS Core, Enterprise, and Scale, you can reference this table.

        + \ No newline at end of file diff --git a/articles/nuxt-content-rss-feed/_payload.json b/articles/nuxt-content-rss-feed/_payload.json index 94dba717..79f58279 100644 --- a/articles/nuxt-content-rss-feed/_payload.json +++ b/articles/nuxt-content-rss-feed/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":1792},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"draft":6,"date":10,"tags":11,"categories":14,"cover_image":16,"excerpt":17,"body":39,"_type":1786,"_id":1787,"_source":1788,"_file":1789,"_stem":1790,"_extension":1791},"/articles/nuxt-content-rss-feed","articles",false,"","How To Add an RSS Feed to a Nuxt Website","If you are a user of Nuxt Content and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.","2024-01-06",[12,13],"nuxt","rss",[15],"programming","/images/nuxt-content-rss-feed.jpg",{"type":18,"children":19},"root",[20],{"type":21,"tag":22,"props":23,"children":24},"element","p",{},[25,28,37],{"type":26,"value":27},"text","If you are a user of ",{"type":21,"tag":29,"props":30,"children":34},"a",{"href":31,"rel":32},"https://content.nuxt.com/",[33],"nofollow",[35],{"type":26,"value":36},"Nuxt Content",{"type":26,"value":38}," and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.",{"type":18,"children":40,"toc":1781},[41,51,58,81,87,101,141,162,197,1139,1176,1246,1252,1265,1270,1297,1309,1685,1703,1775],{"type":21,"tag":22,"props":42,"children":43},{},[44,45,50],{"type":26,"value":27},{"type":21,"tag":29,"props":46,"children":48},{"href":31,"rel":47},[33],[49],{"type":26,"value":36},{"type":26,"value":38},{"type":21,"tag":52,"props":53,"children":55},"h2",{"id":54},"preface",[56],{"type":26,"value":57},"Preface",{"type":21,"tag":22,"props":59,"children":60},{},[61,63,70,72,79],{"type":26,"value":62},"In version 2 of Nuxt, the community module, ",{"type":21,"tag":29,"props":64,"children":67},{"href":65,"rel":66},"https://github.com/nuxt-community/feed-module",[33],[68],{"type":26,"value":69},"nuxt-community/feed-module",{"type":26,"value":71}," was a popular choice for adding an RSS feed to your website. However, there has been an unresolved ",{"type":21,"tag":29,"props":73,"children":76},{"href":74,"rel":75},"https://github.com/nuxt-community/feed-module/issues/106",[33],[77],{"type":26,"value":78},"open issue",{"type":26,"value":80}," since April 1st, 2022 to add support for Nuxt v3. Thankfully, implementing this feature without a module is relatively straight forward.",{"type":21,"tag":52,"props":82,"children":84},{"id":83},"instructions",[85],{"type":26,"value":86},"Instructions",{"type":21,"tag":22,"props":88,"children":89},{},[90,92,99],{"type":26,"value":91},"First, install the ",{"type":21,"tag":29,"props":93,"children":96},{"href":94,"rel":95},"https://www.npmjs.com/package/feed",[33],[97],{"type":26,"value":98},"feed",{"type":26,"value":100}," library into your project:",{"type":21,"tag":102,"props":103,"children":107},"pre",{"className":104,"code":105,"language":106,"meta":7,"style":7},"language-shell shiki shiki-themes github-light","npm i -D feed\n","shell",[108],{"type":21,"tag":109,"props":110,"children":111},"code",{"__ignoreMap":7},[112],{"type":21,"tag":113,"props":114,"children":117},"span",{"class":115,"line":116},"line",1,[118,124,130,136],{"type":21,"tag":113,"props":119,"children":121},{"style":120},"--shiki-default:#6F42C1",[122],{"type":26,"value":123},"npm",{"type":21,"tag":113,"props":125,"children":127},{"style":126},"--shiki-default:#032F62",[128],{"type":26,"value":129}," i",{"type":21,"tag":113,"props":131,"children":133},{"style":132},"--shiki-default:#005CC5",[134],{"type":26,"value":135}," -D",{"type":21,"tag":113,"props":137,"children":138},{"style":126},[139],{"type":26,"value":140}," feed\n",{"type":21,"tag":22,"props":142,"children":143},{},[144,146,152,154,160],{"type":26,"value":145},"Then, create a ",{"type":21,"tag":109,"props":147,"children":149},{"className":148},[],[150],{"type":26,"value":151},"server/",{"type":26,"value":153}," folder in your project if it does not already exist, and create a file named ",{"type":21,"tag":109,"props":155,"children":157},{"className":156},[],[158],{"type":26,"value":159},"server/routes/atom.ts",{"type":26,"value":161},".",{"type":21,"tag":22,"props":163,"children":164},{},[165,167,172,174,179,181,187,189,195],{"type":26,"value":166},"Here, we will leverage the ",{"type":21,"tag":109,"props":168,"children":170},{"className":169},[],[171],{"type":26,"value":98},{"type":26,"value":173}," library and construct an XML representation of our Nuxt content. As you can see, we first define our ",{"type":21,"tag":109,"props":175,"children":177},{"className":176},[],[178],{"type":26,"value":98},{"type":26,"value":180}," with metadata associated with our RSS feed. This will be used by RSS readers to provide context to the end user. Then, we query our Nuxt content with ",{"type":21,"tag":109,"props":182,"children":184},{"className":183},[],[185],{"type":26,"value":186},"serverQueryContent",{"type":26,"value":188}," and append a ",{"type":21,"tag":109,"props":190,"children":192},{"className":191},[],[193],{"type":26,"value":194},"feed.addItem",{"type":26,"value":196}," for each article.",{"type":21,"tag":102,"props":198,"children":202},{"className":199,"code":200,"language":201,"meta":7,"style":7},"language-ts shiki shiki-themes github-light","import { serverQueryContent } from '#content/server';\nimport { Feed } from 'feed';\n\nconst BASE_URL = \"https://mywebsite.com\"\nconst AUTHOR_NAME = \"Firstname Lastname\"\n\nexport default defineEventHandler(async (event) => {\n\n const feed = new Feed({\n title: \"My Title\",\n description: \"My Description\",\n id: BASE_URL,\n link: BASE_URL,\n language: \"en\",\n image: `${BASE_URL}/images/placeholder.png`,\n favicon: `${BASE_URL}/favicon.ico`,\n copyright: `All rights reserved ${new Date().getFullYear()}, ${AUTHOR_NAME}`,\n updated: new Date(),\n generator: \"Nuxt static site generation + Feed for Node.js\",\n feedLinks: {\n atom: `${BASE_URL}/atom`\n },\n author: {\n name: AUTHOR_NAME,\n }\n });\n\n const articles = await serverQueryContent(event).find();\n\n articles.forEach((article) => {\n feed.addItem({\n title: article.title ? article.title : \"Missing Title\",\n id: article._path,\n link: `${BASE_URL}${article._path}`,\n description: article.description,\n author: [\n {\n name: AUTHOR_NAME,\n },\n ],\n date: new Date(article.date),\n image: article.cover_image ? `${BASE_URL}/${article.cover_image}` : undefined\n });\n });\n\n return feed.atom1();\n});\n","ts",[203],{"type":21,"tag":109,"props":204,"children":205},{"__ignoreMap":7},[206,236,262,272,296,318,326,381,389,422,441,459,477,494,512,539,565,623,645,663,672,694,703,712,729,738,747,755,797,805,841,859,892,901,944,953,962,971,988,997,1006,1028,1082,1091,1099,1107,1130],{"type":21,"tag":113,"props":207,"children":208},{"class":115,"line":116},[209,215,221,226,231],{"type":21,"tag":113,"props":210,"children":212},{"style":211},"--shiki-default:#D73A49",[213],{"type":26,"value":214},"import",{"type":21,"tag":113,"props":216,"children":218},{"style":217},"--shiki-default:#24292E",[219],{"type":26,"value":220}," { serverQueryContent } ",{"type":21,"tag":113,"props":222,"children":223},{"style":211},[224],{"type":26,"value":225},"from",{"type":21,"tag":113,"props":227,"children":228},{"style":126},[229],{"type":26,"value":230}," '#content/server'",{"type":21,"tag":113,"props":232,"children":233},{"style":217},[234],{"type":26,"value":235},";\n",{"type":21,"tag":113,"props":237,"children":239},{"class":115,"line":238},2,[240,244,249,253,258],{"type":21,"tag":113,"props":241,"children":242},{"style":211},[243],{"type":26,"value":214},{"type":21,"tag":113,"props":245,"children":246},{"style":217},[247],{"type":26,"value":248}," { Feed } ",{"type":21,"tag":113,"props":250,"children":251},{"style":211},[252],{"type":26,"value":225},{"type":21,"tag":113,"props":254,"children":255},{"style":126},[256],{"type":26,"value":257}," 'feed'",{"type":21,"tag":113,"props":259,"children":260},{"style":217},[261],{"type":26,"value":235},{"type":21,"tag":113,"props":263,"children":265},{"class":115,"line":264},3,[266],{"type":21,"tag":113,"props":267,"children":269},{"emptyLinePlaceholder":268},true,[270],{"type":26,"value":271},"\n",{"type":21,"tag":113,"props":273,"children":275},{"class":115,"line":274},4,[276,281,286,291],{"type":21,"tag":113,"props":277,"children":278},{"style":211},[279],{"type":26,"value":280},"const",{"type":21,"tag":113,"props":282,"children":283},{"style":132},[284],{"type":26,"value":285}," BASE_URL",{"type":21,"tag":113,"props":287,"children":288},{"style":211},[289],{"type":26,"value":290}," =",{"type":21,"tag":113,"props":292,"children":293},{"style":126},[294],{"type":26,"value":295}," \"https://mywebsite.com\"\n",{"type":21,"tag":113,"props":297,"children":299},{"class":115,"line":298},5,[300,304,309,313],{"type":21,"tag":113,"props":301,"children":302},{"style":211},[303],{"type":26,"value":280},{"type":21,"tag":113,"props":305,"children":306},{"style":132},[307],{"type":26,"value":308}," AUTHOR_NAME",{"type":21,"tag":113,"props":310,"children":311},{"style":211},[312],{"type":26,"value":290},{"type":21,"tag":113,"props":314,"children":315},{"style":126},[316],{"type":26,"value":317}," \"Firstname Lastname\"\n",{"type":21,"tag":113,"props":319,"children":321},{"class":115,"line":320},6,[322],{"type":21,"tag":113,"props":323,"children":324},{"emptyLinePlaceholder":268},[325],{"type":26,"value":271},{"type":21,"tag":113,"props":327,"children":329},{"class":115,"line":328},7,[330,335,340,345,350,355,360,366,371,376],{"type":21,"tag":113,"props":331,"children":332},{"style":211},[333],{"type":26,"value":334},"export",{"type":21,"tag":113,"props":336,"children":337},{"style":211},[338],{"type":26,"value":339}," default",{"type":21,"tag":113,"props":341,"children":342},{"style":120},[343],{"type":26,"value":344}," defineEventHandler",{"type":21,"tag":113,"props":346,"children":347},{"style":217},[348],{"type":26,"value":349},"(",{"type":21,"tag":113,"props":351,"children":352},{"style":211},[353],{"type":26,"value":354},"async",{"type":21,"tag":113,"props":356,"children":357},{"style":217},[358],{"type":26,"value":359}," (",{"type":21,"tag":113,"props":361,"children":363},{"style":362},"--shiki-default:#E36209",[364],{"type":26,"value":365},"event",{"type":21,"tag":113,"props":367,"children":368},{"style":217},[369],{"type":26,"value":370},") ",{"type":21,"tag":113,"props":372,"children":373},{"style":211},[374],{"type":26,"value":375},"=>",{"type":21,"tag":113,"props":377,"children":378},{"style":217},[379],{"type":26,"value":380}," {\n",{"type":21,"tag":113,"props":382,"children":384},{"class":115,"line":383},8,[385],{"type":21,"tag":113,"props":386,"children":387},{"emptyLinePlaceholder":268},[388],{"type":26,"value":271},{"type":21,"tag":113,"props":390,"children":392},{"class":115,"line":391},9,[393,398,403,407,412,417],{"type":21,"tag":113,"props":394,"children":395},{"style":211},[396],{"type":26,"value":397}," const",{"type":21,"tag":113,"props":399,"children":400},{"style":132},[401],{"type":26,"value":402}," feed",{"type":21,"tag":113,"props":404,"children":405},{"style":211},[406],{"type":26,"value":290},{"type":21,"tag":113,"props":408,"children":409},{"style":211},[410],{"type":26,"value":411}," new",{"type":21,"tag":113,"props":413,"children":414},{"style":120},[415],{"type":26,"value":416}," Feed",{"type":21,"tag":113,"props":418,"children":419},{"style":217},[420],{"type":26,"value":421},"({\n",{"type":21,"tag":113,"props":423,"children":425},{"class":115,"line":424},10,[426,431,436],{"type":21,"tag":113,"props":427,"children":428},{"style":217},[429],{"type":26,"value":430}," title: ",{"type":21,"tag":113,"props":432,"children":433},{"style":126},[434],{"type":26,"value":435},"\"My Title\"",{"type":21,"tag":113,"props":437,"children":438},{"style":217},[439],{"type":26,"value":440},",\n",{"type":21,"tag":113,"props":442,"children":444},{"class":115,"line":443},11,[445,450,455],{"type":21,"tag":113,"props":446,"children":447},{"style":217},[448],{"type":26,"value":449}," description: ",{"type":21,"tag":113,"props":451,"children":452},{"style":126},[453],{"type":26,"value":454},"\"My Description\"",{"type":21,"tag":113,"props":456,"children":457},{"style":217},[458],{"type":26,"value":440},{"type":21,"tag":113,"props":460,"children":462},{"class":115,"line":461},12,[463,468,473],{"type":21,"tag":113,"props":464,"children":465},{"style":217},[466],{"type":26,"value":467}," id: ",{"type":21,"tag":113,"props":469,"children":470},{"style":132},[471],{"type":26,"value":472},"BASE_URL",{"type":21,"tag":113,"props":474,"children":475},{"style":217},[476],{"type":26,"value":440},{"type":21,"tag":113,"props":478,"children":480},{"class":115,"line":479},13,[481,486,490],{"type":21,"tag":113,"props":482,"children":483},{"style":217},[484],{"type":26,"value":485}," link: ",{"type":21,"tag":113,"props":487,"children":488},{"style":132},[489],{"type":26,"value":472},{"type":21,"tag":113,"props":491,"children":492},{"style":217},[493],{"type":26,"value":440},{"type":21,"tag":113,"props":495,"children":497},{"class":115,"line":496},14,[498,503,508],{"type":21,"tag":113,"props":499,"children":500},{"style":217},[501],{"type":26,"value":502}," language: ",{"type":21,"tag":113,"props":504,"children":505},{"style":126},[506],{"type":26,"value":507},"\"en\"",{"type":21,"tag":113,"props":509,"children":510},{"style":217},[511],{"type":26,"value":440},{"type":21,"tag":113,"props":513,"children":515},{"class":115,"line":514},15,[516,521,526,530,535],{"type":21,"tag":113,"props":517,"children":518},{"style":217},[519],{"type":26,"value":520}," image: ",{"type":21,"tag":113,"props":522,"children":523},{"style":126},[524],{"type":26,"value":525},"`${",{"type":21,"tag":113,"props":527,"children":528},{"style":132},[529],{"type":26,"value":472},{"type":21,"tag":113,"props":531,"children":532},{"style":126},[533],{"type":26,"value":534},"}/images/placeholder.png`",{"type":21,"tag":113,"props":536,"children":537},{"style":217},[538],{"type":26,"value":440},{"type":21,"tag":113,"props":540,"children":542},{"class":115,"line":541},16,[543,548,552,556,561],{"type":21,"tag":113,"props":544,"children":545},{"style":217},[546],{"type":26,"value":547}," favicon: ",{"type":21,"tag":113,"props":549,"children":550},{"style":126},[551],{"type":26,"value":525},{"type":21,"tag":113,"props":553,"children":554},{"style":132},[555],{"type":26,"value":472},{"type":21,"tag":113,"props":557,"children":558},{"style":126},[559],{"type":26,"value":560},"}/favicon.ico`",{"type":21,"tag":113,"props":562,"children":563},{"style":217},[564],{"type":26,"value":440},{"type":21,"tag":113,"props":566,"children":568},{"class":115,"line":567},17,[569,574,579,584,589,594,599,604,609,614,619],{"type":21,"tag":113,"props":570,"children":571},{"style":217},[572],{"type":26,"value":573}," copyright: ",{"type":21,"tag":113,"props":575,"children":576},{"style":126},[577],{"type":26,"value":578},"`All rights reserved ${",{"type":21,"tag":113,"props":580,"children":581},{"style":211},[582],{"type":26,"value":583},"new",{"type":21,"tag":113,"props":585,"children":586},{"style":120},[587],{"type":26,"value":588}," Date",{"type":21,"tag":113,"props":590,"children":591},{"style":126},[592],{"type":26,"value":593},"().",{"type":21,"tag":113,"props":595,"children":596},{"style":120},[597],{"type":26,"value":598},"getFullYear",{"type":21,"tag":113,"props":600,"children":601},{"style":126},[602],{"type":26,"value":603},"()",{"type":21,"tag":113,"props":605,"children":606},{"style":126},[607],{"type":26,"value":608},"}, ${",{"type":21,"tag":113,"props":610,"children":611},{"style":132},[612],{"type":26,"value":613},"AUTHOR_NAME",{"type":21,"tag":113,"props":615,"children":616},{"style":126},[617],{"type":26,"value":618},"}`",{"type":21,"tag":113,"props":620,"children":621},{"style":217},[622],{"type":26,"value":440},{"type":21,"tag":113,"props":624,"children":626},{"class":115,"line":625},18,[627,632,636,640],{"type":21,"tag":113,"props":628,"children":629},{"style":217},[630],{"type":26,"value":631}," updated: ",{"type":21,"tag":113,"props":633,"children":634},{"style":211},[635],{"type":26,"value":583},{"type":21,"tag":113,"props":637,"children":638},{"style":120},[639],{"type":26,"value":588},{"type":21,"tag":113,"props":641,"children":642},{"style":217},[643],{"type":26,"value":644},"(),\n",{"type":21,"tag":113,"props":646,"children":648},{"class":115,"line":647},19,[649,654,659],{"type":21,"tag":113,"props":650,"children":651},{"style":217},[652],{"type":26,"value":653}," generator: ",{"type":21,"tag":113,"props":655,"children":656},{"style":126},[657],{"type":26,"value":658},"\"Nuxt static site generation + Feed for Node.js\"",{"type":21,"tag":113,"props":660,"children":661},{"style":217},[662],{"type":26,"value":440},{"type":21,"tag":113,"props":664,"children":666},{"class":115,"line":665},20,[667],{"type":21,"tag":113,"props":668,"children":669},{"style":217},[670],{"type":26,"value":671}," feedLinks: {\n",{"type":21,"tag":113,"props":673,"children":675},{"class":115,"line":674},21,[676,681,685,689],{"type":21,"tag":113,"props":677,"children":678},{"style":217},[679],{"type":26,"value":680}," atom: ",{"type":21,"tag":113,"props":682,"children":683},{"style":126},[684],{"type":26,"value":525},{"type":21,"tag":113,"props":686,"children":687},{"style":132},[688],{"type":26,"value":472},{"type":21,"tag":113,"props":690,"children":691},{"style":126},[692],{"type":26,"value":693},"}/atom`\n",{"type":21,"tag":113,"props":695,"children":697},{"class":115,"line":696},22,[698],{"type":21,"tag":113,"props":699,"children":700},{"style":217},[701],{"type":26,"value":702}," },\n",{"type":21,"tag":113,"props":704,"children":706},{"class":115,"line":705},23,[707],{"type":21,"tag":113,"props":708,"children":709},{"style":217},[710],{"type":26,"value":711}," author: {\n",{"type":21,"tag":113,"props":713,"children":715},{"class":115,"line":714},24,[716,721,725],{"type":21,"tag":113,"props":717,"children":718},{"style":217},[719],{"type":26,"value":720}," name: ",{"type":21,"tag":113,"props":722,"children":723},{"style":132},[724],{"type":26,"value":613},{"type":21,"tag":113,"props":726,"children":727},{"style":217},[728],{"type":26,"value":440},{"type":21,"tag":113,"props":730,"children":732},{"class":115,"line":731},25,[733],{"type":21,"tag":113,"props":734,"children":735},{"style":217},[736],{"type":26,"value":737}," }\n",{"type":21,"tag":113,"props":739,"children":741},{"class":115,"line":740},26,[742],{"type":21,"tag":113,"props":743,"children":744},{"style":217},[745],{"type":26,"value":746}," });\n",{"type":21,"tag":113,"props":748,"children":750},{"class":115,"line":749},27,[751],{"type":21,"tag":113,"props":752,"children":753},{"emptyLinePlaceholder":268},[754],{"type":26,"value":271},{"type":21,"tag":113,"props":756,"children":758},{"class":115,"line":757},28,[759,763,768,772,777,782,787,792],{"type":21,"tag":113,"props":760,"children":761},{"style":211},[762],{"type":26,"value":397},{"type":21,"tag":113,"props":764,"children":765},{"style":132},[766],{"type":26,"value":767}," articles",{"type":21,"tag":113,"props":769,"children":770},{"style":211},[771],{"type":26,"value":290},{"type":21,"tag":113,"props":773,"children":774},{"style":211},[775],{"type":26,"value":776}," await",{"type":21,"tag":113,"props":778,"children":779},{"style":120},[780],{"type":26,"value":781}," serverQueryContent",{"type":21,"tag":113,"props":783,"children":784},{"style":217},[785],{"type":26,"value":786},"(event).",{"type":21,"tag":113,"props":788,"children":789},{"style":120},[790],{"type":26,"value":791},"find",{"type":21,"tag":113,"props":793,"children":794},{"style":217},[795],{"type":26,"value":796},"();\n",{"type":21,"tag":113,"props":798,"children":800},{"class":115,"line":799},29,[801],{"type":21,"tag":113,"props":802,"children":803},{"emptyLinePlaceholder":268},[804],{"type":26,"value":271},{"type":21,"tag":113,"props":806,"children":808},{"class":115,"line":807},30,[809,814,819,824,829,833,837],{"type":21,"tag":113,"props":810,"children":811},{"style":217},[812],{"type":26,"value":813}," articles.",{"type":21,"tag":113,"props":815,"children":816},{"style":120},[817],{"type":26,"value":818},"forEach",{"type":21,"tag":113,"props":820,"children":821},{"style":217},[822],{"type":26,"value":823},"((",{"type":21,"tag":113,"props":825,"children":826},{"style":362},[827],{"type":26,"value":828},"article",{"type":21,"tag":113,"props":830,"children":831},{"style":217},[832],{"type":26,"value":370},{"type":21,"tag":113,"props":834,"children":835},{"style":211},[836],{"type":26,"value":375},{"type":21,"tag":113,"props":838,"children":839},{"style":217},[840],{"type":26,"value":380},{"type":21,"tag":113,"props":842,"children":844},{"class":115,"line":843},31,[845,850,855],{"type":21,"tag":113,"props":846,"children":847},{"style":217},[848],{"type":26,"value":849}," feed.",{"type":21,"tag":113,"props":851,"children":852},{"style":120},[853],{"type":26,"value":854},"addItem",{"type":21,"tag":113,"props":856,"children":857},{"style":217},[858],{"type":26,"value":421},{"type":21,"tag":113,"props":860,"children":862},{"class":115,"line":861},32,[863,868,873,878,883,888],{"type":21,"tag":113,"props":864,"children":865},{"style":217},[866],{"type":26,"value":867}," title: article.title ",{"type":21,"tag":113,"props":869,"children":870},{"style":211},[871],{"type":26,"value":872},"?",{"type":21,"tag":113,"props":874,"children":875},{"style":217},[876],{"type":26,"value":877}," article.title ",{"type":21,"tag":113,"props":879,"children":880},{"style":211},[881],{"type":26,"value":882},":",{"type":21,"tag":113,"props":884,"children":885},{"style":126},[886],{"type":26,"value":887}," \"Missing Title\"",{"type":21,"tag":113,"props":889,"children":890},{"style":217},[891],{"type":26,"value":440},{"type":21,"tag":113,"props":893,"children":895},{"class":115,"line":894},33,[896],{"type":21,"tag":113,"props":897,"children":898},{"style":217},[899],{"type":26,"value":900}," id: article._path,\n",{"type":21,"tag":113,"props":902,"children":904},{"class":115,"line":903},34,[905,910,914,918,923,927,931,936,940],{"type":21,"tag":113,"props":906,"children":907},{"style":217},[908],{"type":26,"value":909}," link: ",{"type":21,"tag":113,"props":911,"children":912},{"style":126},[913],{"type":26,"value":525},{"type":21,"tag":113,"props":915,"children":916},{"style":132},[917],{"type":26,"value":472},{"type":21,"tag":113,"props":919,"children":920},{"style":126},[921],{"type":26,"value":922},"}${",{"type":21,"tag":113,"props":924,"children":925},{"style":217},[926],{"type":26,"value":828},{"type":21,"tag":113,"props":928,"children":929},{"style":126},[930],{"type":26,"value":161},{"type":21,"tag":113,"props":932,"children":933},{"style":217},[934],{"type":26,"value":935},"_path",{"type":21,"tag":113,"props":937,"children":938},{"style":126},[939],{"type":26,"value":618},{"type":21,"tag":113,"props":941,"children":942},{"style":217},[943],{"type":26,"value":440},{"type":21,"tag":113,"props":945,"children":947},{"class":115,"line":946},35,[948],{"type":21,"tag":113,"props":949,"children":950},{"style":217},[951],{"type":26,"value":952}," description: article.description,\n",{"type":21,"tag":113,"props":954,"children":956},{"class":115,"line":955},36,[957],{"type":21,"tag":113,"props":958,"children":959},{"style":217},[960],{"type":26,"value":961}," author: [\n",{"type":21,"tag":113,"props":963,"children":965},{"class":115,"line":964},37,[966],{"type":21,"tag":113,"props":967,"children":968},{"style":217},[969],{"type":26,"value":970}," {\n",{"type":21,"tag":113,"props":972,"children":974},{"class":115,"line":973},38,[975,980,984],{"type":21,"tag":113,"props":976,"children":977},{"style":217},[978],{"type":26,"value":979}," name: ",{"type":21,"tag":113,"props":981,"children":982},{"style":132},[983],{"type":26,"value":613},{"type":21,"tag":113,"props":985,"children":986},{"style":217},[987],{"type":26,"value":440},{"type":21,"tag":113,"props":989,"children":991},{"class":115,"line":990},39,[992],{"type":21,"tag":113,"props":993,"children":994},{"style":217},[995],{"type":26,"value":996}," },\n",{"type":21,"tag":113,"props":998,"children":1000},{"class":115,"line":999},40,[1001],{"type":21,"tag":113,"props":1002,"children":1003},{"style":217},[1004],{"type":26,"value":1005}," ],\n",{"type":21,"tag":113,"props":1007,"children":1009},{"class":115,"line":1008},41,[1010,1015,1019,1023],{"type":21,"tag":113,"props":1011,"children":1012},{"style":217},[1013],{"type":26,"value":1014}," date: ",{"type":21,"tag":113,"props":1016,"children":1017},{"style":211},[1018],{"type":26,"value":583},{"type":21,"tag":113,"props":1020,"children":1021},{"style":120},[1022],{"type":26,"value":588},{"type":21,"tag":113,"props":1024,"children":1025},{"style":217},[1026],{"type":26,"value":1027},"(article.date),\n",{"type":21,"tag":113,"props":1029,"children":1031},{"class":115,"line":1030},42,[1032,1037,1041,1046,1050,1055,1059,1063,1068,1072,1077],{"type":21,"tag":113,"props":1033,"children":1034},{"style":217},[1035],{"type":26,"value":1036}," image: article.cover_image ",{"type":21,"tag":113,"props":1038,"children":1039},{"style":211},[1040],{"type":26,"value":872},{"type":21,"tag":113,"props":1042,"children":1043},{"style":126},[1044],{"type":26,"value":1045}," `${",{"type":21,"tag":113,"props":1047,"children":1048},{"style":132},[1049],{"type":26,"value":472},{"type":21,"tag":113,"props":1051,"children":1052},{"style":126},[1053],{"type":26,"value":1054},"}/${",{"type":21,"tag":113,"props":1056,"children":1057},{"style":217},[1058],{"type":26,"value":828},{"type":21,"tag":113,"props":1060,"children":1061},{"style":126},[1062],{"type":26,"value":161},{"type":21,"tag":113,"props":1064,"children":1065},{"style":217},[1066],{"type":26,"value":1067},"cover_image",{"type":21,"tag":113,"props":1069,"children":1070},{"style":126},[1071],{"type":26,"value":618},{"type":21,"tag":113,"props":1073,"children":1074},{"style":211},[1075],{"type":26,"value":1076}," :",{"type":21,"tag":113,"props":1078,"children":1079},{"style":132},[1080],{"type":26,"value":1081}," undefined\n",{"type":21,"tag":113,"props":1083,"children":1085},{"class":115,"line":1084},43,[1086],{"type":21,"tag":113,"props":1087,"children":1088},{"style":217},[1089],{"type":26,"value":1090}," });\n",{"type":21,"tag":113,"props":1092,"children":1094},{"class":115,"line":1093},44,[1095],{"type":21,"tag":113,"props":1096,"children":1097},{"style":217},[1098],{"type":26,"value":746},{"type":21,"tag":113,"props":1100,"children":1102},{"class":115,"line":1101},45,[1103],{"type":21,"tag":113,"props":1104,"children":1105},{"emptyLinePlaceholder":268},[1106],{"type":26,"value":271},{"type":21,"tag":113,"props":1108,"children":1110},{"class":115,"line":1109},46,[1111,1116,1121,1126],{"type":21,"tag":113,"props":1112,"children":1113},{"style":211},[1114],{"type":26,"value":1115}," return",{"type":21,"tag":113,"props":1117,"children":1118},{"style":217},[1119],{"type":26,"value":1120}," feed.",{"type":21,"tag":113,"props":1122,"children":1123},{"style":120},[1124],{"type":26,"value":1125},"atom1",{"type":21,"tag":113,"props":1127,"children":1128},{"style":217},[1129],{"type":26,"value":796},{"type":21,"tag":113,"props":1131,"children":1133},{"class":115,"line":1132},47,[1134],{"type":21,"tag":113,"props":1135,"children":1136},{"style":217},[1137],{"type":26,"value":1138},"});\n",{"type":21,"tag":22,"props":1140,"children":1141},{},[1142,1144,1150,1152,1158,1160,1166,1168,1174],{"type":26,"value":1143},"And that's just about it! Except, if you are statically generating your website with the ",{"type":21,"tag":109,"props":1145,"children":1147},{"className":1146},[],[1148],{"type":26,"value":1149},"nuxt generate",{"type":26,"value":1151}," command, you will need to configure this server-side route to be pre-rendered on site generation. This is as simple as adding a ",{"type":21,"tag":109,"props":1153,"children":1155},{"className":1154},[],[1156],{"type":26,"value":1157},"nitro",{"type":26,"value":1159}," ",{"type":21,"tag":109,"props":1161,"children":1163},{"className":1162},[],[1164],{"type":26,"value":1165},"prerender",{"type":26,"value":1167}," definition in your ",{"type":21,"tag":109,"props":1169,"children":1171},{"className":1170},[],[1172],{"type":26,"value":1173},"nuxt.config.ts",{"type":26,"value":1175}," file, like so:",{"type":21,"tag":102,"props":1177,"children":1179},{"className":199,"code":1178,"language":201,"meta":7,"style":7},"nitro: {\n prerender: {\n routes: ['/atom']\n }\n}\n",[1180],{"type":21,"tag":109,"props":1181,"children":1182},{"__ignoreMap":7},[1183,1195,1207,1230,1238],{"type":21,"tag":113,"props":1184,"children":1185},{"class":115,"line":116},[1186,1190],{"type":21,"tag":113,"props":1187,"children":1188},{"style":120},[1189],{"type":26,"value":1157},{"type":21,"tag":113,"props":1191,"children":1192},{"style":217},[1193],{"type":26,"value":1194},": {\n",{"type":21,"tag":113,"props":1196,"children":1197},{"class":115,"line":238},[1198,1203],{"type":21,"tag":113,"props":1199,"children":1200},{"style":120},[1201],{"type":26,"value":1202}," prerender",{"type":21,"tag":113,"props":1204,"children":1205},{"style":217},[1206],{"type":26,"value":1194},{"type":21,"tag":113,"props":1208,"children":1209},{"class":115,"line":264},[1210,1215,1220,1225],{"type":21,"tag":113,"props":1211,"children":1212},{"style":120},[1213],{"type":26,"value":1214}," routes",{"type":21,"tag":113,"props":1216,"children":1217},{"style":217},[1218],{"type":26,"value":1219},": [",{"type":21,"tag":113,"props":1221,"children":1222},{"style":126},[1223],{"type":26,"value":1224},"'/atom'",{"type":21,"tag":113,"props":1226,"children":1227},{"style":217},[1228],{"type":26,"value":1229},"]\n",{"type":21,"tag":113,"props":1231,"children":1232},{"class":115,"line":274},[1233],{"type":21,"tag":113,"props":1234,"children":1235},{"style":217},[1236],{"type":26,"value":1237}," }\n",{"type":21,"tag":113,"props":1239,"children":1240},{"class":115,"line":298},[1241],{"type":21,"tag":113,"props":1242,"children":1243},{"style":217},[1244],{"type":26,"value":1245},"}\n",{"type":21,"tag":52,"props":1247,"children":1249},{"id":1248},"bonus",[1250],{"type":26,"value":1251},"Bonus",{"type":21,"tag":22,"props":1253,"children":1254},{},[1255,1257,1263],{"type":26,"value":1256},"You may also be interested in adding a ",{"type":21,"tag":109,"props":1258,"children":1260},{"className":1259},[],[1261],{"type":26,"value":1262},"sitemap.xml",{"type":26,"value":1264}," to your website. This can be done in almost an identical fashion!",{"type":21,"tag":22,"props":1266,"children":1267},{},[1268],{"type":26,"value":1269},"Install the dependency:",{"type":21,"tag":102,"props":1271,"children":1273},{"className":104,"code":1272,"language":106,"meta":7,"style":7},"npm i -D sitemap\n",[1274],{"type":21,"tag":109,"props":1275,"children":1276},{"__ignoreMap":7},[1277],{"type":21,"tag":113,"props":1278,"children":1279},{"class":115,"line":116},[1280,1284,1288,1292],{"type":21,"tag":113,"props":1281,"children":1282},{"style":120},[1283],{"type":26,"value":123},{"type":21,"tag":113,"props":1285,"children":1286},{"style":126},[1287],{"type":26,"value":129},{"type":21,"tag":113,"props":1289,"children":1290},{"style":132},[1291],{"type":26,"value":135},{"type":21,"tag":113,"props":1293,"children":1294},{"style":126},[1295],{"type":26,"value":1296}," sitemap\n",{"type":21,"tag":22,"props":1298,"children":1299},{},[1300,1302,1308],{"type":26,"value":1301},"Create a route at ",{"type":21,"tag":109,"props":1303,"children":1305},{"className":1304},[],[1306],{"type":26,"value":1307},"server/routes/sitemap.xml.ts",{"type":26,"value":882},{"type":21,"tag":102,"props":1310,"children":1312},{"className":199,"code":1311,"language":201,"meta":7,"style":7},"import { serverQueryContent } from '#content/server';\nimport { SitemapStream, streamToPromise } from 'sitemap';\n\nexport default defineEventHandler(async (event) => {\n const articles = await serverQueryContent(event).find();\n\n const sitemap = new SitemapStream({ hostname: 'https://my-website.com/' });\n\n // Add non nuxt content endpoints here\n sitemap.write({ url: '/' });\n sitemap.write({ url: '/articles' });\n\n // Dynamically generate routes for Nuxt markdown content\n articles.forEach((article) => sitemap.write({ url: article._path, changefreq: 'monthly' }));\n sitemap.end();\n\n return (await streamToPromise(sitemap));\n});\n",[1313],{"type":21,"tag":109,"props":1314,"children":1315},{"__ignoreMap":7},[1316,1339,1364,1371,1414,1449,1456,1496,1503,1512,1539,1563,1570,1578,1629,1645,1652,1678],{"type":21,"tag":113,"props":1317,"children":1318},{"class":115,"line":116},[1319,1323,1327,1331,1335],{"type":21,"tag":113,"props":1320,"children":1321},{"style":211},[1322],{"type":26,"value":214},{"type":21,"tag":113,"props":1324,"children":1325},{"style":217},[1326],{"type":26,"value":220},{"type":21,"tag":113,"props":1328,"children":1329},{"style":211},[1330],{"type":26,"value":225},{"type":21,"tag":113,"props":1332,"children":1333},{"style":126},[1334],{"type":26,"value":230},{"type":21,"tag":113,"props":1336,"children":1337},{"style":217},[1338],{"type":26,"value":235},{"type":21,"tag":113,"props":1340,"children":1341},{"class":115,"line":238},[1342,1346,1351,1355,1360],{"type":21,"tag":113,"props":1343,"children":1344},{"style":211},[1345],{"type":26,"value":214},{"type":21,"tag":113,"props":1347,"children":1348},{"style":217},[1349],{"type":26,"value":1350}," { SitemapStream, streamToPromise } ",{"type":21,"tag":113,"props":1352,"children":1353},{"style":211},[1354],{"type":26,"value":225},{"type":21,"tag":113,"props":1356,"children":1357},{"style":126},[1358],{"type":26,"value":1359}," 'sitemap'",{"type":21,"tag":113,"props":1361,"children":1362},{"style":217},[1363],{"type":26,"value":235},{"type":21,"tag":113,"props":1365,"children":1366},{"class":115,"line":264},[1367],{"type":21,"tag":113,"props":1368,"children":1369},{"emptyLinePlaceholder":268},[1370],{"type":26,"value":271},{"type":21,"tag":113,"props":1372,"children":1373},{"class":115,"line":274},[1374,1378,1382,1386,1390,1394,1398,1402,1406,1410],{"type":21,"tag":113,"props":1375,"children":1376},{"style":211},[1377],{"type":26,"value":334},{"type":21,"tag":113,"props":1379,"children":1380},{"style":211},[1381],{"type":26,"value":339},{"type":21,"tag":113,"props":1383,"children":1384},{"style":120},[1385],{"type":26,"value":344},{"type":21,"tag":113,"props":1387,"children":1388},{"style":217},[1389],{"type":26,"value":349},{"type":21,"tag":113,"props":1391,"children":1392},{"style":211},[1393],{"type":26,"value":354},{"type":21,"tag":113,"props":1395,"children":1396},{"style":217},[1397],{"type":26,"value":359},{"type":21,"tag":113,"props":1399,"children":1400},{"style":362},[1401],{"type":26,"value":365},{"type":21,"tag":113,"props":1403,"children":1404},{"style":217},[1405],{"type":26,"value":370},{"type":21,"tag":113,"props":1407,"children":1408},{"style":211},[1409],{"type":26,"value":375},{"type":21,"tag":113,"props":1411,"children":1412},{"style":217},[1413],{"type":26,"value":380},{"type":21,"tag":113,"props":1415,"children":1416},{"class":115,"line":298},[1417,1421,1425,1429,1433,1437,1441,1445],{"type":21,"tag":113,"props":1418,"children":1419},{"style":211},[1420],{"type":26,"value":397},{"type":21,"tag":113,"props":1422,"children":1423},{"style":132},[1424],{"type":26,"value":767},{"type":21,"tag":113,"props":1426,"children":1427},{"style":211},[1428],{"type":26,"value":290},{"type":21,"tag":113,"props":1430,"children":1431},{"style":211},[1432],{"type":26,"value":776},{"type":21,"tag":113,"props":1434,"children":1435},{"style":120},[1436],{"type":26,"value":781},{"type":21,"tag":113,"props":1438,"children":1439},{"style":217},[1440],{"type":26,"value":786},{"type":21,"tag":113,"props":1442,"children":1443},{"style":120},[1444],{"type":26,"value":791},{"type":21,"tag":113,"props":1446,"children":1447},{"style":217},[1448],{"type":26,"value":796},{"type":21,"tag":113,"props":1450,"children":1451},{"class":115,"line":320},[1452],{"type":21,"tag":113,"props":1453,"children":1454},{"emptyLinePlaceholder":268},[1455],{"type":26,"value":271},{"type":21,"tag":113,"props":1457,"children":1458},{"class":115,"line":328},[1459,1463,1468,1472,1476,1481,1486,1491],{"type":21,"tag":113,"props":1460,"children":1461},{"style":211},[1462],{"type":26,"value":397},{"type":21,"tag":113,"props":1464,"children":1465},{"style":132},[1466],{"type":26,"value":1467}," sitemap",{"type":21,"tag":113,"props":1469,"children":1470},{"style":211},[1471],{"type":26,"value":290},{"type":21,"tag":113,"props":1473,"children":1474},{"style":211},[1475],{"type":26,"value":411},{"type":21,"tag":113,"props":1477,"children":1478},{"style":120},[1479],{"type":26,"value":1480}," SitemapStream",{"type":21,"tag":113,"props":1482,"children":1483},{"style":217},[1484],{"type":26,"value":1485},"({ hostname: ",{"type":21,"tag":113,"props":1487,"children":1488},{"style":126},[1489],{"type":26,"value":1490},"'https://my-website.com/'",{"type":21,"tag":113,"props":1492,"children":1493},{"style":217},[1494],{"type":26,"value":1495}," });\n",{"type":21,"tag":113,"props":1497,"children":1498},{"class":115,"line":383},[1499],{"type":21,"tag":113,"props":1500,"children":1501},{"emptyLinePlaceholder":268},[1502],{"type":26,"value":271},{"type":21,"tag":113,"props":1504,"children":1505},{"class":115,"line":391},[1506],{"type":21,"tag":113,"props":1507,"children":1509},{"style":1508},"--shiki-default:#6A737D",[1510],{"type":26,"value":1511}," // Add non nuxt content endpoints here\n",{"type":21,"tag":113,"props":1513,"children":1514},{"class":115,"line":424},[1515,1520,1525,1530,1535],{"type":21,"tag":113,"props":1516,"children":1517},{"style":217},[1518],{"type":26,"value":1519}," sitemap.",{"type":21,"tag":113,"props":1521,"children":1522},{"style":120},[1523],{"type":26,"value":1524},"write",{"type":21,"tag":113,"props":1526,"children":1527},{"style":217},[1528],{"type":26,"value":1529},"({ url: ",{"type":21,"tag":113,"props":1531,"children":1532},{"style":126},[1533],{"type":26,"value":1534},"'/'",{"type":21,"tag":113,"props":1536,"children":1537},{"style":217},[1538],{"type":26,"value":1495},{"type":21,"tag":113,"props":1540,"children":1541},{"class":115,"line":443},[1542,1546,1550,1554,1559],{"type":21,"tag":113,"props":1543,"children":1544},{"style":217},[1545],{"type":26,"value":1519},{"type":21,"tag":113,"props":1547,"children":1548},{"style":120},[1549],{"type":26,"value":1524},{"type":21,"tag":113,"props":1551,"children":1552},{"style":217},[1553],{"type":26,"value":1529},{"type":21,"tag":113,"props":1555,"children":1556},{"style":126},[1557],{"type":26,"value":1558},"'/articles'",{"type":21,"tag":113,"props":1560,"children":1561},{"style":217},[1562],{"type":26,"value":1495},{"type":21,"tag":113,"props":1564,"children":1565},{"class":115,"line":461},[1566],{"type":21,"tag":113,"props":1567,"children":1568},{"emptyLinePlaceholder":268},[1569],{"type":26,"value":271},{"type":21,"tag":113,"props":1571,"children":1572},{"class":115,"line":479},[1573],{"type":21,"tag":113,"props":1574,"children":1575},{"style":1508},[1576],{"type":26,"value":1577}," // Dynamically generate routes for Nuxt markdown content\n",{"type":21,"tag":113,"props":1579,"children":1580},{"class":115,"line":496},[1581,1585,1589,1593,1597,1601,1605,1610,1614,1619,1624],{"type":21,"tag":113,"props":1582,"children":1583},{"style":217},[1584],{"type":26,"value":813},{"type":21,"tag":113,"props":1586,"children":1587},{"style":120},[1588],{"type":26,"value":818},{"type":21,"tag":113,"props":1590,"children":1591},{"style":217},[1592],{"type":26,"value":823},{"type":21,"tag":113,"props":1594,"children":1595},{"style":362},[1596],{"type":26,"value":828},{"type":21,"tag":113,"props":1598,"children":1599},{"style":217},[1600],{"type":26,"value":370},{"type":21,"tag":113,"props":1602,"children":1603},{"style":211},[1604],{"type":26,"value":375},{"type":21,"tag":113,"props":1606,"children":1607},{"style":217},[1608],{"type":26,"value":1609}," sitemap.",{"type":21,"tag":113,"props":1611,"children":1612},{"style":120},[1613],{"type":26,"value":1524},{"type":21,"tag":113,"props":1615,"children":1616},{"style":217},[1617],{"type":26,"value":1618},"({ url: article._path, changefreq: ",{"type":21,"tag":113,"props":1620,"children":1621},{"style":126},[1622],{"type":26,"value":1623},"'monthly'",{"type":21,"tag":113,"props":1625,"children":1626},{"style":217},[1627],{"type":26,"value":1628}," }));\n",{"type":21,"tag":113,"props":1630,"children":1631},{"class":115,"line":514},[1632,1636,1641],{"type":21,"tag":113,"props":1633,"children":1634},{"style":217},[1635],{"type":26,"value":1519},{"type":21,"tag":113,"props":1637,"children":1638},{"style":120},[1639],{"type":26,"value":1640},"end",{"type":21,"tag":113,"props":1642,"children":1643},{"style":217},[1644],{"type":26,"value":796},{"type":21,"tag":113,"props":1646,"children":1647},{"class":115,"line":541},[1648],{"type":21,"tag":113,"props":1649,"children":1650},{"emptyLinePlaceholder":268},[1651],{"type":26,"value":271},{"type":21,"tag":113,"props":1653,"children":1654},{"class":115,"line":567},[1655,1659,1663,1668,1673],{"type":21,"tag":113,"props":1656,"children":1657},{"style":211},[1658],{"type":26,"value":1115},{"type":21,"tag":113,"props":1660,"children":1661},{"style":217},[1662],{"type":26,"value":359},{"type":21,"tag":113,"props":1664,"children":1665},{"style":211},[1666],{"type":26,"value":1667},"await",{"type":21,"tag":113,"props":1669,"children":1670},{"style":120},[1671],{"type":26,"value":1672}," streamToPromise",{"type":21,"tag":113,"props":1674,"children":1675},{"style":217},[1676],{"type":26,"value":1677},"(sitemap));\n",{"type":21,"tag":113,"props":1679,"children":1680},{"class":115,"line":625},[1681],{"type":21,"tag":113,"props":1682,"children":1683},{"style":217},[1684],{"type":26,"value":1138},{"type":21,"tag":22,"props":1686,"children":1687},{},[1688,1690,1695,1697,1702],{"type":26,"value":1689},"And add the ",{"type":21,"tag":109,"props":1691,"children":1693},{"className":1692},[],[1694],{"type":26,"value":1165},{"type":26,"value":1696}," entry in your ",{"type":21,"tag":109,"props":1698,"children":1700},{"className":1699},[],[1701],{"type":26,"value":1173},{"type":26,"value":882},{"type":21,"tag":102,"props":1704,"children":1706},{"className":199,"code":1705,"language":201,"meta":7,"style":7},"nitro: {\n prerender: {\n routes: ['/sitemap.xml', '/atom']\n }\n}\n",[1707],{"type":21,"tag":109,"props":1708,"children":1709},{"__ignoreMap":7},[1710,1721,1732,1761,1768],{"type":21,"tag":113,"props":1711,"children":1712},{"class":115,"line":116},[1713,1717],{"type":21,"tag":113,"props":1714,"children":1715},{"style":120},[1716],{"type":26,"value":1157},{"type":21,"tag":113,"props":1718,"children":1719},{"style":217},[1720],{"type":26,"value":1194},{"type":21,"tag":113,"props":1722,"children":1723},{"class":115,"line":238},[1724,1728],{"type":21,"tag":113,"props":1725,"children":1726},{"style":120},[1727],{"type":26,"value":1202},{"type":21,"tag":113,"props":1729,"children":1730},{"style":217},[1731],{"type":26,"value":1194},{"type":21,"tag":113,"props":1733,"children":1734},{"class":115,"line":264},[1735,1739,1743,1748,1753,1757],{"type":21,"tag":113,"props":1736,"children":1737},{"style":120},[1738],{"type":26,"value":1214},{"type":21,"tag":113,"props":1740,"children":1741},{"style":217},[1742],{"type":26,"value":1219},{"type":21,"tag":113,"props":1744,"children":1745},{"style":126},[1746],{"type":26,"value":1747},"'/sitemap.xml'",{"type":21,"tag":113,"props":1749,"children":1750},{"style":217},[1751],{"type":26,"value":1752},", ",{"type":21,"tag":113,"props":1754,"children":1755},{"style":126},[1756],{"type":26,"value":1224},{"type":21,"tag":113,"props":1758,"children":1759},{"style":217},[1760],{"type":26,"value":1229},{"type":21,"tag":113,"props":1762,"children":1763},{"class":115,"line":274},[1764],{"type":21,"tag":113,"props":1765,"children":1766},{"style":217},[1767],{"type":26,"value":1237},{"type":21,"tag":113,"props":1769,"children":1770},{"class":115,"line":298},[1771],{"type":21,"tag":113,"props":1772,"children":1773},{"style":217},[1774],{"type":26,"value":1245},{"type":21,"tag":1776,"props":1777,"children":1778},"style",{},[1779],{"type":26,"value":1780},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":238,"depth":238,"links":1782},[1783,1784,1785],{"id":54,"depth":238,"text":57},{"id":83,"depth":238,"text":86},{"id":1248,"depth":238,"text":1251},"markdown","content:articles:nuxt-content-rss-feed.md","content","articles/nuxt-content-rss-feed.md","articles/nuxt-content-rss-feed","md",1731781853002] \ No newline at end of file +[{"data":1,"prerenderedAt":1792},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"draft":6,"date":10,"tags":11,"categories":14,"cover_image":16,"excerpt":17,"body":39,"_type":1786,"_id":1787,"_source":1788,"_file":1789,"_stem":1790,"_extension":1791},"/articles/nuxt-content-rss-feed","articles",false,"","How To Add an RSS Feed to a Nuxt Website","If you are a user of Nuxt Content and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.","2024-01-06",[12,13],"nuxt","rss",[15],"programming","/images/nuxt-content-rss-feed.jpg",{"type":18,"children":19},"root",[20],{"type":21,"tag":22,"props":23,"children":24},"element","p",{},[25,28,37],{"type":26,"value":27},"text","If you are a user of ",{"type":21,"tag":29,"props":30,"children":34},"a",{"href":31,"rel":32},"https://content.nuxt.com/",[33],"nofollow",[35],{"type":26,"value":36},"Nuxt Content",{"type":26,"value":38}," and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.",{"type":18,"children":40,"toc":1781},[41,51,58,81,87,101,141,162,197,1139,1176,1246,1252,1265,1270,1297,1309,1685,1703,1775],{"type":21,"tag":22,"props":42,"children":43},{},[44,45,50],{"type":26,"value":27},{"type":21,"tag":29,"props":46,"children":48},{"href":31,"rel":47},[33],[49],{"type":26,"value":36},{"type":26,"value":38},{"type":21,"tag":52,"props":53,"children":55},"h2",{"id":54},"preface",[56],{"type":26,"value":57},"Preface",{"type":21,"tag":22,"props":59,"children":60},{},[61,63,70,72,79],{"type":26,"value":62},"In version 2 of Nuxt, the community module, ",{"type":21,"tag":29,"props":64,"children":67},{"href":65,"rel":66},"https://github.com/nuxt-community/feed-module",[33],[68],{"type":26,"value":69},"nuxt-community/feed-module",{"type":26,"value":71}," was a popular choice for adding an RSS feed to your website. However, there has been an unresolved ",{"type":21,"tag":29,"props":73,"children":76},{"href":74,"rel":75},"https://github.com/nuxt-community/feed-module/issues/106",[33],[77],{"type":26,"value":78},"open issue",{"type":26,"value":80}," since April 1st, 2022 to add support for Nuxt v3. Thankfully, implementing this feature without a module is relatively straight forward.",{"type":21,"tag":52,"props":82,"children":84},{"id":83},"instructions",[85],{"type":26,"value":86},"Instructions",{"type":21,"tag":22,"props":88,"children":89},{},[90,92,99],{"type":26,"value":91},"First, install the ",{"type":21,"tag":29,"props":93,"children":96},{"href":94,"rel":95},"https://www.npmjs.com/package/feed",[33],[97],{"type":26,"value":98},"feed",{"type":26,"value":100}," library into your project:",{"type":21,"tag":102,"props":103,"children":107},"pre",{"className":104,"code":105,"language":106,"meta":7,"style":7},"language-shell shiki shiki-themes github-light","npm i -D feed\n","shell",[108],{"type":21,"tag":109,"props":110,"children":111},"code",{"__ignoreMap":7},[112],{"type":21,"tag":113,"props":114,"children":117},"span",{"class":115,"line":116},"line",1,[118,124,130,136],{"type":21,"tag":113,"props":119,"children":121},{"style":120},"--shiki-default:#6F42C1",[122],{"type":26,"value":123},"npm",{"type":21,"tag":113,"props":125,"children":127},{"style":126},"--shiki-default:#032F62",[128],{"type":26,"value":129}," i",{"type":21,"tag":113,"props":131,"children":133},{"style":132},"--shiki-default:#005CC5",[134],{"type":26,"value":135}," -D",{"type":21,"tag":113,"props":137,"children":138},{"style":126},[139],{"type":26,"value":140}," feed\n",{"type":21,"tag":22,"props":142,"children":143},{},[144,146,152,154,160],{"type":26,"value":145},"Then, create a ",{"type":21,"tag":109,"props":147,"children":149},{"className":148},[],[150],{"type":26,"value":151},"server/",{"type":26,"value":153}," folder in your project if it does not already exist, and create a file named ",{"type":21,"tag":109,"props":155,"children":157},{"className":156},[],[158],{"type":26,"value":159},"server/routes/atom.ts",{"type":26,"value":161},".",{"type":21,"tag":22,"props":163,"children":164},{},[165,167,172,174,179,181,187,189,195],{"type":26,"value":166},"Here, we will leverage the ",{"type":21,"tag":109,"props":168,"children":170},{"className":169},[],[171],{"type":26,"value":98},{"type":26,"value":173}," library and construct an XML representation of our Nuxt content. As you can see, we first define our ",{"type":21,"tag":109,"props":175,"children":177},{"className":176},[],[178],{"type":26,"value":98},{"type":26,"value":180}," with metadata associated with our RSS feed. This will be used by RSS readers to provide context to the end user. Then, we query our Nuxt content with ",{"type":21,"tag":109,"props":182,"children":184},{"className":183},[],[185],{"type":26,"value":186},"serverQueryContent",{"type":26,"value":188}," and append a ",{"type":21,"tag":109,"props":190,"children":192},{"className":191},[],[193],{"type":26,"value":194},"feed.addItem",{"type":26,"value":196}," for each article.",{"type":21,"tag":102,"props":198,"children":202},{"className":199,"code":200,"language":201,"meta":7,"style":7},"language-ts shiki shiki-themes github-light","import { serverQueryContent } from '#content/server';\nimport { Feed } from 'feed';\n\nconst BASE_URL = \"https://mywebsite.com\"\nconst AUTHOR_NAME = \"Firstname Lastname\"\n\nexport default defineEventHandler(async (event) => {\n\n const feed = new Feed({\n title: \"My Title\",\n description: \"My Description\",\n id: BASE_URL,\n link: BASE_URL,\n language: \"en\",\n image: `${BASE_URL}/images/placeholder.png`,\n favicon: `${BASE_URL}/favicon.ico`,\n copyright: `All rights reserved ${new Date().getFullYear()}, ${AUTHOR_NAME}`,\n updated: new Date(),\n generator: \"Nuxt static site generation + Feed for Node.js\",\n feedLinks: {\n atom: `${BASE_URL}/atom`\n },\n author: {\n name: AUTHOR_NAME,\n }\n });\n\n const articles = await serverQueryContent(event).find();\n\n articles.forEach((article) => {\n feed.addItem({\n title: article.title ? article.title : \"Missing Title\",\n id: article._path,\n link: `${BASE_URL}${article._path}`,\n description: article.description,\n author: [\n {\n name: AUTHOR_NAME,\n },\n ],\n date: new Date(article.date),\n image: article.cover_image ? `${BASE_URL}/${article.cover_image}` : undefined\n });\n });\n\n return feed.atom1();\n});\n","ts",[203],{"type":21,"tag":109,"props":204,"children":205},{"__ignoreMap":7},[206,236,262,272,296,318,326,381,389,422,441,459,477,494,512,539,565,623,645,663,672,694,703,712,729,738,747,755,797,805,841,859,892,901,944,953,962,971,988,997,1006,1028,1082,1091,1099,1107,1130],{"type":21,"tag":113,"props":207,"children":208},{"class":115,"line":116},[209,215,221,226,231],{"type":21,"tag":113,"props":210,"children":212},{"style":211},"--shiki-default:#D73A49",[213],{"type":26,"value":214},"import",{"type":21,"tag":113,"props":216,"children":218},{"style":217},"--shiki-default:#24292E",[219],{"type":26,"value":220}," { serverQueryContent } ",{"type":21,"tag":113,"props":222,"children":223},{"style":211},[224],{"type":26,"value":225},"from",{"type":21,"tag":113,"props":227,"children":228},{"style":126},[229],{"type":26,"value":230}," '#content/server'",{"type":21,"tag":113,"props":232,"children":233},{"style":217},[234],{"type":26,"value":235},";\n",{"type":21,"tag":113,"props":237,"children":239},{"class":115,"line":238},2,[240,244,249,253,258],{"type":21,"tag":113,"props":241,"children":242},{"style":211},[243],{"type":26,"value":214},{"type":21,"tag":113,"props":245,"children":246},{"style":217},[247],{"type":26,"value":248}," { Feed } ",{"type":21,"tag":113,"props":250,"children":251},{"style":211},[252],{"type":26,"value":225},{"type":21,"tag":113,"props":254,"children":255},{"style":126},[256],{"type":26,"value":257}," 'feed'",{"type":21,"tag":113,"props":259,"children":260},{"style":217},[261],{"type":26,"value":235},{"type":21,"tag":113,"props":263,"children":265},{"class":115,"line":264},3,[266],{"type":21,"tag":113,"props":267,"children":269},{"emptyLinePlaceholder":268},true,[270],{"type":26,"value":271},"\n",{"type":21,"tag":113,"props":273,"children":275},{"class":115,"line":274},4,[276,281,286,291],{"type":21,"tag":113,"props":277,"children":278},{"style":211},[279],{"type":26,"value":280},"const",{"type":21,"tag":113,"props":282,"children":283},{"style":132},[284],{"type":26,"value":285}," BASE_URL",{"type":21,"tag":113,"props":287,"children":288},{"style":211},[289],{"type":26,"value":290}," =",{"type":21,"tag":113,"props":292,"children":293},{"style":126},[294],{"type":26,"value":295}," \"https://mywebsite.com\"\n",{"type":21,"tag":113,"props":297,"children":299},{"class":115,"line":298},5,[300,304,309,313],{"type":21,"tag":113,"props":301,"children":302},{"style":211},[303],{"type":26,"value":280},{"type":21,"tag":113,"props":305,"children":306},{"style":132},[307],{"type":26,"value":308}," AUTHOR_NAME",{"type":21,"tag":113,"props":310,"children":311},{"style":211},[312],{"type":26,"value":290},{"type":21,"tag":113,"props":314,"children":315},{"style":126},[316],{"type":26,"value":317}," \"Firstname Lastname\"\n",{"type":21,"tag":113,"props":319,"children":321},{"class":115,"line":320},6,[322],{"type":21,"tag":113,"props":323,"children":324},{"emptyLinePlaceholder":268},[325],{"type":26,"value":271},{"type":21,"tag":113,"props":327,"children":329},{"class":115,"line":328},7,[330,335,340,345,350,355,360,366,371,376],{"type":21,"tag":113,"props":331,"children":332},{"style":211},[333],{"type":26,"value":334},"export",{"type":21,"tag":113,"props":336,"children":337},{"style":211},[338],{"type":26,"value":339}," default",{"type":21,"tag":113,"props":341,"children":342},{"style":120},[343],{"type":26,"value":344}," defineEventHandler",{"type":21,"tag":113,"props":346,"children":347},{"style":217},[348],{"type":26,"value":349},"(",{"type":21,"tag":113,"props":351,"children":352},{"style":211},[353],{"type":26,"value":354},"async",{"type":21,"tag":113,"props":356,"children":357},{"style":217},[358],{"type":26,"value":359}," (",{"type":21,"tag":113,"props":361,"children":363},{"style":362},"--shiki-default:#E36209",[364],{"type":26,"value":365},"event",{"type":21,"tag":113,"props":367,"children":368},{"style":217},[369],{"type":26,"value":370},") ",{"type":21,"tag":113,"props":372,"children":373},{"style":211},[374],{"type":26,"value":375},"=>",{"type":21,"tag":113,"props":377,"children":378},{"style":217},[379],{"type":26,"value":380}," {\n",{"type":21,"tag":113,"props":382,"children":384},{"class":115,"line":383},8,[385],{"type":21,"tag":113,"props":386,"children":387},{"emptyLinePlaceholder":268},[388],{"type":26,"value":271},{"type":21,"tag":113,"props":390,"children":392},{"class":115,"line":391},9,[393,398,403,407,412,417],{"type":21,"tag":113,"props":394,"children":395},{"style":211},[396],{"type":26,"value":397}," const",{"type":21,"tag":113,"props":399,"children":400},{"style":132},[401],{"type":26,"value":402}," feed",{"type":21,"tag":113,"props":404,"children":405},{"style":211},[406],{"type":26,"value":290},{"type":21,"tag":113,"props":408,"children":409},{"style":211},[410],{"type":26,"value":411}," new",{"type":21,"tag":113,"props":413,"children":414},{"style":120},[415],{"type":26,"value":416}," Feed",{"type":21,"tag":113,"props":418,"children":419},{"style":217},[420],{"type":26,"value":421},"({\n",{"type":21,"tag":113,"props":423,"children":425},{"class":115,"line":424},10,[426,431,436],{"type":21,"tag":113,"props":427,"children":428},{"style":217},[429],{"type":26,"value":430}," title: ",{"type":21,"tag":113,"props":432,"children":433},{"style":126},[434],{"type":26,"value":435},"\"My Title\"",{"type":21,"tag":113,"props":437,"children":438},{"style":217},[439],{"type":26,"value":440},",\n",{"type":21,"tag":113,"props":442,"children":444},{"class":115,"line":443},11,[445,450,455],{"type":21,"tag":113,"props":446,"children":447},{"style":217},[448],{"type":26,"value":449}," description: ",{"type":21,"tag":113,"props":451,"children":452},{"style":126},[453],{"type":26,"value":454},"\"My Description\"",{"type":21,"tag":113,"props":456,"children":457},{"style":217},[458],{"type":26,"value":440},{"type":21,"tag":113,"props":460,"children":462},{"class":115,"line":461},12,[463,468,473],{"type":21,"tag":113,"props":464,"children":465},{"style":217},[466],{"type":26,"value":467}," id: ",{"type":21,"tag":113,"props":469,"children":470},{"style":132},[471],{"type":26,"value":472},"BASE_URL",{"type":21,"tag":113,"props":474,"children":475},{"style":217},[476],{"type":26,"value":440},{"type":21,"tag":113,"props":478,"children":480},{"class":115,"line":479},13,[481,486,490],{"type":21,"tag":113,"props":482,"children":483},{"style":217},[484],{"type":26,"value":485}," link: ",{"type":21,"tag":113,"props":487,"children":488},{"style":132},[489],{"type":26,"value":472},{"type":21,"tag":113,"props":491,"children":492},{"style":217},[493],{"type":26,"value":440},{"type":21,"tag":113,"props":495,"children":497},{"class":115,"line":496},14,[498,503,508],{"type":21,"tag":113,"props":499,"children":500},{"style":217},[501],{"type":26,"value":502}," language: ",{"type":21,"tag":113,"props":504,"children":505},{"style":126},[506],{"type":26,"value":507},"\"en\"",{"type":21,"tag":113,"props":509,"children":510},{"style":217},[511],{"type":26,"value":440},{"type":21,"tag":113,"props":513,"children":515},{"class":115,"line":514},15,[516,521,526,530,535],{"type":21,"tag":113,"props":517,"children":518},{"style":217},[519],{"type":26,"value":520}," image: ",{"type":21,"tag":113,"props":522,"children":523},{"style":126},[524],{"type":26,"value":525},"`${",{"type":21,"tag":113,"props":527,"children":528},{"style":132},[529],{"type":26,"value":472},{"type":21,"tag":113,"props":531,"children":532},{"style":126},[533],{"type":26,"value":534},"}/images/placeholder.png`",{"type":21,"tag":113,"props":536,"children":537},{"style":217},[538],{"type":26,"value":440},{"type":21,"tag":113,"props":540,"children":542},{"class":115,"line":541},16,[543,548,552,556,561],{"type":21,"tag":113,"props":544,"children":545},{"style":217},[546],{"type":26,"value":547}," favicon: ",{"type":21,"tag":113,"props":549,"children":550},{"style":126},[551],{"type":26,"value":525},{"type":21,"tag":113,"props":553,"children":554},{"style":132},[555],{"type":26,"value":472},{"type":21,"tag":113,"props":557,"children":558},{"style":126},[559],{"type":26,"value":560},"}/favicon.ico`",{"type":21,"tag":113,"props":562,"children":563},{"style":217},[564],{"type":26,"value":440},{"type":21,"tag":113,"props":566,"children":568},{"class":115,"line":567},17,[569,574,579,584,589,594,599,604,609,614,619],{"type":21,"tag":113,"props":570,"children":571},{"style":217},[572],{"type":26,"value":573}," copyright: ",{"type":21,"tag":113,"props":575,"children":576},{"style":126},[577],{"type":26,"value":578},"`All rights reserved ${",{"type":21,"tag":113,"props":580,"children":581},{"style":211},[582],{"type":26,"value":583},"new",{"type":21,"tag":113,"props":585,"children":586},{"style":120},[587],{"type":26,"value":588}," Date",{"type":21,"tag":113,"props":590,"children":591},{"style":126},[592],{"type":26,"value":593},"().",{"type":21,"tag":113,"props":595,"children":596},{"style":120},[597],{"type":26,"value":598},"getFullYear",{"type":21,"tag":113,"props":600,"children":601},{"style":126},[602],{"type":26,"value":603},"()",{"type":21,"tag":113,"props":605,"children":606},{"style":126},[607],{"type":26,"value":608},"}, ${",{"type":21,"tag":113,"props":610,"children":611},{"style":132},[612],{"type":26,"value":613},"AUTHOR_NAME",{"type":21,"tag":113,"props":615,"children":616},{"style":126},[617],{"type":26,"value":618},"}`",{"type":21,"tag":113,"props":620,"children":621},{"style":217},[622],{"type":26,"value":440},{"type":21,"tag":113,"props":624,"children":626},{"class":115,"line":625},18,[627,632,636,640],{"type":21,"tag":113,"props":628,"children":629},{"style":217},[630],{"type":26,"value":631}," updated: ",{"type":21,"tag":113,"props":633,"children":634},{"style":211},[635],{"type":26,"value":583},{"type":21,"tag":113,"props":637,"children":638},{"style":120},[639],{"type":26,"value":588},{"type":21,"tag":113,"props":641,"children":642},{"style":217},[643],{"type":26,"value":644},"(),\n",{"type":21,"tag":113,"props":646,"children":648},{"class":115,"line":647},19,[649,654,659],{"type":21,"tag":113,"props":650,"children":651},{"style":217},[652],{"type":26,"value":653}," generator: ",{"type":21,"tag":113,"props":655,"children":656},{"style":126},[657],{"type":26,"value":658},"\"Nuxt static site generation + Feed for Node.js\"",{"type":21,"tag":113,"props":660,"children":661},{"style":217},[662],{"type":26,"value":440},{"type":21,"tag":113,"props":664,"children":666},{"class":115,"line":665},20,[667],{"type":21,"tag":113,"props":668,"children":669},{"style":217},[670],{"type":26,"value":671}," feedLinks: {\n",{"type":21,"tag":113,"props":673,"children":675},{"class":115,"line":674},21,[676,681,685,689],{"type":21,"tag":113,"props":677,"children":678},{"style":217},[679],{"type":26,"value":680}," atom: ",{"type":21,"tag":113,"props":682,"children":683},{"style":126},[684],{"type":26,"value":525},{"type":21,"tag":113,"props":686,"children":687},{"style":132},[688],{"type":26,"value":472},{"type":21,"tag":113,"props":690,"children":691},{"style":126},[692],{"type":26,"value":693},"}/atom`\n",{"type":21,"tag":113,"props":695,"children":697},{"class":115,"line":696},22,[698],{"type":21,"tag":113,"props":699,"children":700},{"style":217},[701],{"type":26,"value":702}," },\n",{"type":21,"tag":113,"props":704,"children":706},{"class":115,"line":705},23,[707],{"type":21,"tag":113,"props":708,"children":709},{"style":217},[710],{"type":26,"value":711}," author: {\n",{"type":21,"tag":113,"props":713,"children":715},{"class":115,"line":714},24,[716,721,725],{"type":21,"tag":113,"props":717,"children":718},{"style":217},[719],{"type":26,"value":720}," name: ",{"type":21,"tag":113,"props":722,"children":723},{"style":132},[724],{"type":26,"value":613},{"type":21,"tag":113,"props":726,"children":727},{"style":217},[728],{"type":26,"value":440},{"type":21,"tag":113,"props":730,"children":732},{"class":115,"line":731},25,[733],{"type":21,"tag":113,"props":734,"children":735},{"style":217},[736],{"type":26,"value":737}," }\n",{"type":21,"tag":113,"props":739,"children":741},{"class":115,"line":740},26,[742],{"type":21,"tag":113,"props":743,"children":744},{"style":217},[745],{"type":26,"value":746}," });\n",{"type":21,"tag":113,"props":748,"children":750},{"class":115,"line":749},27,[751],{"type":21,"tag":113,"props":752,"children":753},{"emptyLinePlaceholder":268},[754],{"type":26,"value":271},{"type":21,"tag":113,"props":756,"children":758},{"class":115,"line":757},28,[759,763,768,772,777,782,787,792],{"type":21,"tag":113,"props":760,"children":761},{"style":211},[762],{"type":26,"value":397},{"type":21,"tag":113,"props":764,"children":765},{"style":132},[766],{"type":26,"value":767}," articles",{"type":21,"tag":113,"props":769,"children":770},{"style":211},[771],{"type":26,"value":290},{"type":21,"tag":113,"props":773,"children":774},{"style":211},[775],{"type":26,"value":776}," await",{"type":21,"tag":113,"props":778,"children":779},{"style":120},[780],{"type":26,"value":781}," serverQueryContent",{"type":21,"tag":113,"props":783,"children":784},{"style":217},[785],{"type":26,"value":786},"(event).",{"type":21,"tag":113,"props":788,"children":789},{"style":120},[790],{"type":26,"value":791},"find",{"type":21,"tag":113,"props":793,"children":794},{"style":217},[795],{"type":26,"value":796},"();\n",{"type":21,"tag":113,"props":798,"children":800},{"class":115,"line":799},29,[801],{"type":21,"tag":113,"props":802,"children":803},{"emptyLinePlaceholder":268},[804],{"type":26,"value":271},{"type":21,"tag":113,"props":806,"children":808},{"class":115,"line":807},30,[809,814,819,824,829,833,837],{"type":21,"tag":113,"props":810,"children":811},{"style":217},[812],{"type":26,"value":813}," articles.",{"type":21,"tag":113,"props":815,"children":816},{"style":120},[817],{"type":26,"value":818},"forEach",{"type":21,"tag":113,"props":820,"children":821},{"style":217},[822],{"type":26,"value":823},"((",{"type":21,"tag":113,"props":825,"children":826},{"style":362},[827],{"type":26,"value":828},"article",{"type":21,"tag":113,"props":830,"children":831},{"style":217},[832],{"type":26,"value":370},{"type":21,"tag":113,"props":834,"children":835},{"style":211},[836],{"type":26,"value":375},{"type":21,"tag":113,"props":838,"children":839},{"style":217},[840],{"type":26,"value":380},{"type":21,"tag":113,"props":842,"children":844},{"class":115,"line":843},31,[845,850,855],{"type":21,"tag":113,"props":846,"children":847},{"style":217},[848],{"type":26,"value":849}," feed.",{"type":21,"tag":113,"props":851,"children":852},{"style":120},[853],{"type":26,"value":854},"addItem",{"type":21,"tag":113,"props":856,"children":857},{"style":217},[858],{"type":26,"value":421},{"type":21,"tag":113,"props":860,"children":862},{"class":115,"line":861},32,[863,868,873,878,883,888],{"type":21,"tag":113,"props":864,"children":865},{"style":217},[866],{"type":26,"value":867}," title: article.title ",{"type":21,"tag":113,"props":869,"children":870},{"style":211},[871],{"type":26,"value":872},"?",{"type":21,"tag":113,"props":874,"children":875},{"style":217},[876],{"type":26,"value":877}," article.title ",{"type":21,"tag":113,"props":879,"children":880},{"style":211},[881],{"type":26,"value":882},":",{"type":21,"tag":113,"props":884,"children":885},{"style":126},[886],{"type":26,"value":887}," \"Missing Title\"",{"type":21,"tag":113,"props":889,"children":890},{"style":217},[891],{"type":26,"value":440},{"type":21,"tag":113,"props":893,"children":895},{"class":115,"line":894},33,[896],{"type":21,"tag":113,"props":897,"children":898},{"style":217},[899],{"type":26,"value":900}," id: article._path,\n",{"type":21,"tag":113,"props":902,"children":904},{"class":115,"line":903},34,[905,910,914,918,923,927,931,936,940],{"type":21,"tag":113,"props":906,"children":907},{"style":217},[908],{"type":26,"value":909}," link: ",{"type":21,"tag":113,"props":911,"children":912},{"style":126},[913],{"type":26,"value":525},{"type":21,"tag":113,"props":915,"children":916},{"style":132},[917],{"type":26,"value":472},{"type":21,"tag":113,"props":919,"children":920},{"style":126},[921],{"type":26,"value":922},"}${",{"type":21,"tag":113,"props":924,"children":925},{"style":217},[926],{"type":26,"value":828},{"type":21,"tag":113,"props":928,"children":929},{"style":126},[930],{"type":26,"value":161},{"type":21,"tag":113,"props":932,"children":933},{"style":217},[934],{"type":26,"value":935},"_path",{"type":21,"tag":113,"props":937,"children":938},{"style":126},[939],{"type":26,"value":618},{"type":21,"tag":113,"props":941,"children":942},{"style":217},[943],{"type":26,"value":440},{"type":21,"tag":113,"props":945,"children":947},{"class":115,"line":946},35,[948],{"type":21,"tag":113,"props":949,"children":950},{"style":217},[951],{"type":26,"value":952}," description: article.description,\n",{"type":21,"tag":113,"props":954,"children":956},{"class":115,"line":955},36,[957],{"type":21,"tag":113,"props":958,"children":959},{"style":217},[960],{"type":26,"value":961}," author: [\n",{"type":21,"tag":113,"props":963,"children":965},{"class":115,"line":964},37,[966],{"type":21,"tag":113,"props":967,"children":968},{"style":217},[969],{"type":26,"value":970}," {\n",{"type":21,"tag":113,"props":972,"children":974},{"class":115,"line":973},38,[975,980,984],{"type":21,"tag":113,"props":976,"children":977},{"style":217},[978],{"type":26,"value":979}," name: ",{"type":21,"tag":113,"props":981,"children":982},{"style":132},[983],{"type":26,"value":613},{"type":21,"tag":113,"props":985,"children":986},{"style":217},[987],{"type":26,"value":440},{"type":21,"tag":113,"props":989,"children":991},{"class":115,"line":990},39,[992],{"type":21,"tag":113,"props":993,"children":994},{"style":217},[995],{"type":26,"value":996}," },\n",{"type":21,"tag":113,"props":998,"children":1000},{"class":115,"line":999},40,[1001],{"type":21,"tag":113,"props":1002,"children":1003},{"style":217},[1004],{"type":26,"value":1005}," ],\n",{"type":21,"tag":113,"props":1007,"children":1009},{"class":115,"line":1008},41,[1010,1015,1019,1023],{"type":21,"tag":113,"props":1011,"children":1012},{"style":217},[1013],{"type":26,"value":1014}," date: ",{"type":21,"tag":113,"props":1016,"children":1017},{"style":211},[1018],{"type":26,"value":583},{"type":21,"tag":113,"props":1020,"children":1021},{"style":120},[1022],{"type":26,"value":588},{"type":21,"tag":113,"props":1024,"children":1025},{"style":217},[1026],{"type":26,"value":1027},"(article.date),\n",{"type":21,"tag":113,"props":1029,"children":1031},{"class":115,"line":1030},42,[1032,1037,1041,1046,1050,1055,1059,1063,1068,1072,1077],{"type":21,"tag":113,"props":1033,"children":1034},{"style":217},[1035],{"type":26,"value":1036}," image: article.cover_image ",{"type":21,"tag":113,"props":1038,"children":1039},{"style":211},[1040],{"type":26,"value":872},{"type":21,"tag":113,"props":1042,"children":1043},{"style":126},[1044],{"type":26,"value":1045}," `${",{"type":21,"tag":113,"props":1047,"children":1048},{"style":132},[1049],{"type":26,"value":472},{"type":21,"tag":113,"props":1051,"children":1052},{"style":126},[1053],{"type":26,"value":1054},"}/${",{"type":21,"tag":113,"props":1056,"children":1057},{"style":217},[1058],{"type":26,"value":828},{"type":21,"tag":113,"props":1060,"children":1061},{"style":126},[1062],{"type":26,"value":161},{"type":21,"tag":113,"props":1064,"children":1065},{"style":217},[1066],{"type":26,"value":1067},"cover_image",{"type":21,"tag":113,"props":1069,"children":1070},{"style":126},[1071],{"type":26,"value":618},{"type":21,"tag":113,"props":1073,"children":1074},{"style":211},[1075],{"type":26,"value":1076}," :",{"type":21,"tag":113,"props":1078,"children":1079},{"style":132},[1080],{"type":26,"value":1081}," undefined\n",{"type":21,"tag":113,"props":1083,"children":1085},{"class":115,"line":1084},43,[1086],{"type":21,"tag":113,"props":1087,"children":1088},{"style":217},[1089],{"type":26,"value":1090}," });\n",{"type":21,"tag":113,"props":1092,"children":1094},{"class":115,"line":1093},44,[1095],{"type":21,"tag":113,"props":1096,"children":1097},{"style":217},[1098],{"type":26,"value":746},{"type":21,"tag":113,"props":1100,"children":1102},{"class":115,"line":1101},45,[1103],{"type":21,"tag":113,"props":1104,"children":1105},{"emptyLinePlaceholder":268},[1106],{"type":26,"value":271},{"type":21,"tag":113,"props":1108,"children":1110},{"class":115,"line":1109},46,[1111,1116,1121,1126],{"type":21,"tag":113,"props":1112,"children":1113},{"style":211},[1114],{"type":26,"value":1115}," return",{"type":21,"tag":113,"props":1117,"children":1118},{"style":217},[1119],{"type":26,"value":1120}," feed.",{"type":21,"tag":113,"props":1122,"children":1123},{"style":120},[1124],{"type":26,"value":1125},"atom1",{"type":21,"tag":113,"props":1127,"children":1128},{"style":217},[1129],{"type":26,"value":796},{"type":21,"tag":113,"props":1131,"children":1133},{"class":115,"line":1132},47,[1134],{"type":21,"tag":113,"props":1135,"children":1136},{"style":217},[1137],{"type":26,"value":1138},"});\n",{"type":21,"tag":22,"props":1140,"children":1141},{},[1142,1144,1150,1152,1158,1160,1166,1168,1174],{"type":26,"value":1143},"And that's just about it! Except, if you are statically generating your website with the ",{"type":21,"tag":109,"props":1145,"children":1147},{"className":1146},[],[1148],{"type":26,"value":1149},"nuxt generate",{"type":26,"value":1151}," command, you will need to configure this server-side route to be pre-rendered on site generation. This is as simple as adding a ",{"type":21,"tag":109,"props":1153,"children":1155},{"className":1154},[],[1156],{"type":26,"value":1157},"nitro",{"type":26,"value":1159}," ",{"type":21,"tag":109,"props":1161,"children":1163},{"className":1162},[],[1164],{"type":26,"value":1165},"prerender",{"type":26,"value":1167}," definition in your ",{"type":21,"tag":109,"props":1169,"children":1171},{"className":1170},[],[1172],{"type":26,"value":1173},"nuxt.config.ts",{"type":26,"value":1175}," file, like so:",{"type":21,"tag":102,"props":1177,"children":1179},{"className":199,"code":1178,"language":201,"meta":7,"style":7},"nitro: {\n prerender: {\n routes: ['/atom']\n }\n}\n",[1180],{"type":21,"tag":109,"props":1181,"children":1182},{"__ignoreMap":7},[1183,1195,1207,1230,1238],{"type":21,"tag":113,"props":1184,"children":1185},{"class":115,"line":116},[1186,1190],{"type":21,"tag":113,"props":1187,"children":1188},{"style":120},[1189],{"type":26,"value":1157},{"type":21,"tag":113,"props":1191,"children":1192},{"style":217},[1193],{"type":26,"value":1194},": {\n",{"type":21,"tag":113,"props":1196,"children":1197},{"class":115,"line":238},[1198,1203],{"type":21,"tag":113,"props":1199,"children":1200},{"style":120},[1201],{"type":26,"value":1202}," prerender",{"type":21,"tag":113,"props":1204,"children":1205},{"style":217},[1206],{"type":26,"value":1194},{"type":21,"tag":113,"props":1208,"children":1209},{"class":115,"line":264},[1210,1215,1220,1225],{"type":21,"tag":113,"props":1211,"children":1212},{"style":120},[1213],{"type":26,"value":1214}," routes",{"type":21,"tag":113,"props":1216,"children":1217},{"style":217},[1218],{"type":26,"value":1219},": [",{"type":21,"tag":113,"props":1221,"children":1222},{"style":126},[1223],{"type":26,"value":1224},"'/atom'",{"type":21,"tag":113,"props":1226,"children":1227},{"style":217},[1228],{"type":26,"value":1229},"]\n",{"type":21,"tag":113,"props":1231,"children":1232},{"class":115,"line":274},[1233],{"type":21,"tag":113,"props":1234,"children":1235},{"style":217},[1236],{"type":26,"value":1237}," }\n",{"type":21,"tag":113,"props":1239,"children":1240},{"class":115,"line":298},[1241],{"type":21,"tag":113,"props":1242,"children":1243},{"style":217},[1244],{"type":26,"value":1245},"}\n",{"type":21,"tag":52,"props":1247,"children":1249},{"id":1248},"bonus",[1250],{"type":26,"value":1251},"Bonus",{"type":21,"tag":22,"props":1253,"children":1254},{},[1255,1257,1263],{"type":26,"value":1256},"You may also be interested in adding a ",{"type":21,"tag":109,"props":1258,"children":1260},{"className":1259},[],[1261],{"type":26,"value":1262},"sitemap.xml",{"type":26,"value":1264}," to your website. This can be done in almost an identical fashion!",{"type":21,"tag":22,"props":1266,"children":1267},{},[1268],{"type":26,"value":1269},"Install the dependency:",{"type":21,"tag":102,"props":1271,"children":1273},{"className":104,"code":1272,"language":106,"meta":7,"style":7},"npm i -D sitemap\n",[1274],{"type":21,"tag":109,"props":1275,"children":1276},{"__ignoreMap":7},[1277],{"type":21,"tag":113,"props":1278,"children":1279},{"class":115,"line":116},[1280,1284,1288,1292],{"type":21,"tag":113,"props":1281,"children":1282},{"style":120},[1283],{"type":26,"value":123},{"type":21,"tag":113,"props":1285,"children":1286},{"style":126},[1287],{"type":26,"value":129},{"type":21,"tag":113,"props":1289,"children":1290},{"style":132},[1291],{"type":26,"value":135},{"type":21,"tag":113,"props":1293,"children":1294},{"style":126},[1295],{"type":26,"value":1296}," sitemap\n",{"type":21,"tag":22,"props":1298,"children":1299},{},[1300,1302,1308],{"type":26,"value":1301},"Create a route at ",{"type":21,"tag":109,"props":1303,"children":1305},{"className":1304},[],[1306],{"type":26,"value":1307},"server/routes/sitemap.xml.ts",{"type":26,"value":882},{"type":21,"tag":102,"props":1310,"children":1312},{"className":199,"code":1311,"language":201,"meta":7,"style":7},"import { serverQueryContent } from '#content/server';\nimport { SitemapStream, streamToPromise } from 'sitemap';\n\nexport default defineEventHandler(async (event) => {\n const articles = await serverQueryContent(event).find();\n\n const sitemap = new SitemapStream({ hostname: 'https://my-website.com/' });\n\n // Add non nuxt content endpoints here\n sitemap.write({ url: '/' });\n sitemap.write({ url: '/articles' });\n\n // Dynamically generate routes for Nuxt markdown content\n articles.forEach((article) => sitemap.write({ url: article._path, changefreq: 'monthly' }));\n sitemap.end();\n\n return (await streamToPromise(sitemap));\n});\n",[1313],{"type":21,"tag":109,"props":1314,"children":1315},{"__ignoreMap":7},[1316,1339,1364,1371,1414,1449,1456,1496,1503,1512,1539,1563,1570,1578,1629,1645,1652,1678],{"type":21,"tag":113,"props":1317,"children":1318},{"class":115,"line":116},[1319,1323,1327,1331,1335],{"type":21,"tag":113,"props":1320,"children":1321},{"style":211},[1322],{"type":26,"value":214},{"type":21,"tag":113,"props":1324,"children":1325},{"style":217},[1326],{"type":26,"value":220},{"type":21,"tag":113,"props":1328,"children":1329},{"style":211},[1330],{"type":26,"value":225},{"type":21,"tag":113,"props":1332,"children":1333},{"style":126},[1334],{"type":26,"value":230},{"type":21,"tag":113,"props":1336,"children":1337},{"style":217},[1338],{"type":26,"value":235},{"type":21,"tag":113,"props":1340,"children":1341},{"class":115,"line":238},[1342,1346,1351,1355,1360],{"type":21,"tag":113,"props":1343,"children":1344},{"style":211},[1345],{"type":26,"value":214},{"type":21,"tag":113,"props":1347,"children":1348},{"style":217},[1349],{"type":26,"value":1350}," { SitemapStream, streamToPromise } ",{"type":21,"tag":113,"props":1352,"children":1353},{"style":211},[1354],{"type":26,"value":225},{"type":21,"tag":113,"props":1356,"children":1357},{"style":126},[1358],{"type":26,"value":1359}," 'sitemap'",{"type":21,"tag":113,"props":1361,"children":1362},{"style":217},[1363],{"type":26,"value":235},{"type":21,"tag":113,"props":1365,"children":1366},{"class":115,"line":264},[1367],{"type":21,"tag":113,"props":1368,"children":1369},{"emptyLinePlaceholder":268},[1370],{"type":26,"value":271},{"type":21,"tag":113,"props":1372,"children":1373},{"class":115,"line":274},[1374,1378,1382,1386,1390,1394,1398,1402,1406,1410],{"type":21,"tag":113,"props":1375,"children":1376},{"style":211},[1377],{"type":26,"value":334},{"type":21,"tag":113,"props":1379,"children":1380},{"style":211},[1381],{"type":26,"value":339},{"type":21,"tag":113,"props":1383,"children":1384},{"style":120},[1385],{"type":26,"value":344},{"type":21,"tag":113,"props":1387,"children":1388},{"style":217},[1389],{"type":26,"value":349},{"type":21,"tag":113,"props":1391,"children":1392},{"style":211},[1393],{"type":26,"value":354},{"type":21,"tag":113,"props":1395,"children":1396},{"style":217},[1397],{"type":26,"value":359},{"type":21,"tag":113,"props":1399,"children":1400},{"style":362},[1401],{"type":26,"value":365},{"type":21,"tag":113,"props":1403,"children":1404},{"style":217},[1405],{"type":26,"value":370},{"type":21,"tag":113,"props":1407,"children":1408},{"style":211},[1409],{"type":26,"value":375},{"type":21,"tag":113,"props":1411,"children":1412},{"style":217},[1413],{"type":26,"value":380},{"type":21,"tag":113,"props":1415,"children":1416},{"class":115,"line":298},[1417,1421,1425,1429,1433,1437,1441,1445],{"type":21,"tag":113,"props":1418,"children":1419},{"style":211},[1420],{"type":26,"value":397},{"type":21,"tag":113,"props":1422,"children":1423},{"style":132},[1424],{"type":26,"value":767},{"type":21,"tag":113,"props":1426,"children":1427},{"style":211},[1428],{"type":26,"value":290},{"type":21,"tag":113,"props":1430,"children":1431},{"style":211},[1432],{"type":26,"value":776},{"type":21,"tag":113,"props":1434,"children":1435},{"style":120},[1436],{"type":26,"value":781},{"type":21,"tag":113,"props":1438,"children":1439},{"style":217},[1440],{"type":26,"value":786},{"type":21,"tag":113,"props":1442,"children":1443},{"style":120},[1444],{"type":26,"value":791},{"type":21,"tag":113,"props":1446,"children":1447},{"style":217},[1448],{"type":26,"value":796},{"type":21,"tag":113,"props":1450,"children":1451},{"class":115,"line":320},[1452],{"type":21,"tag":113,"props":1453,"children":1454},{"emptyLinePlaceholder":268},[1455],{"type":26,"value":271},{"type":21,"tag":113,"props":1457,"children":1458},{"class":115,"line":328},[1459,1463,1468,1472,1476,1481,1486,1491],{"type":21,"tag":113,"props":1460,"children":1461},{"style":211},[1462],{"type":26,"value":397},{"type":21,"tag":113,"props":1464,"children":1465},{"style":132},[1466],{"type":26,"value":1467}," sitemap",{"type":21,"tag":113,"props":1469,"children":1470},{"style":211},[1471],{"type":26,"value":290},{"type":21,"tag":113,"props":1473,"children":1474},{"style":211},[1475],{"type":26,"value":411},{"type":21,"tag":113,"props":1477,"children":1478},{"style":120},[1479],{"type":26,"value":1480}," SitemapStream",{"type":21,"tag":113,"props":1482,"children":1483},{"style":217},[1484],{"type":26,"value":1485},"({ hostname: ",{"type":21,"tag":113,"props":1487,"children":1488},{"style":126},[1489],{"type":26,"value":1490},"'https://my-website.com/'",{"type":21,"tag":113,"props":1492,"children":1493},{"style":217},[1494],{"type":26,"value":1495}," });\n",{"type":21,"tag":113,"props":1497,"children":1498},{"class":115,"line":383},[1499],{"type":21,"tag":113,"props":1500,"children":1501},{"emptyLinePlaceholder":268},[1502],{"type":26,"value":271},{"type":21,"tag":113,"props":1504,"children":1505},{"class":115,"line":391},[1506],{"type":21,"tag":113,"props":1507,"children":1509},{"style":1508},"--shiki-default:#6A737D",[1510],{"type":26,"value":1511}," // Add non nuxt content endpoints here\n",{"type":21,"tag":113,"props":1513,"children":1514},{"class":115,"line":424},[1515,1520,1525,1530,1535],{"type":21,"tag":113,"props":1516,"children":1517},{"style":217},[1518],{"type":26,"value":1519}," sitemap.",{"type":21,"tag":113,"props":1521,"children":1522},{"style":120},[1523],{"type":26,"value":1524},"write",{"type":21,"tag":113,"props":1526,"children":1527},{"style":217},[1528],{"type":26,"value":1529},"({ url: ",{"type":21,"tag":113,"props":1531,"children":1532},{"style":126},[1533],{"type":26,"value":1534},"'/'",{"type":21,"tag":113,"props":1536,"children":1537},{"style":217},[1538],{"type":26,"value":1495},{"type":21,"tag":113,"props":1540,"children":1541},{"class":115,"line":443},[1542,1546,1550,1554,1559],{"type":21,"tag":113,"props":1543,"children":1544},{"style":217},[1545],{"type":26,"value":1519},{"type":21,"tag":113,"props":1547,"children":1548},{"style":120},[1549],{"type":26,"value":1524},{"type":21,"tag":113,"props":1551,"children":1552},{"style":217},[1553],{"type":26,"value":1529},{"type":21,"tag":113,"props":1555,"children":1556},{"style":126},[1557],{"type":26,"value":1558},"'/articles'",{"type":21,"tag":113,"props":1560,"children":1561},{"style":217},[1562],{"type":26,"value":1495},{"type":21,"tag":113,"props":1564,"children":1565},{"class":115,"line":461},[1566],{"type":21,"tag":113,"props":1567,"children":1568},{"emptyLinePlaceholder":268},[1569],{"type":26,"value":271},{"type":21,"tag":113,"props":1571,"children":1572},{"class":115,"line":479},[1573],{"type":21,"tag":113,"props":1574,"children":1575},{"style":1508},[1576],{"type":26,"value":1577}," // Dynamically generate routes for Nuxt markdown content\n",{"type":21,"tag":113,"props":1579,"children":1580},{"class":115,"line":496},[1581,1585,1589,1593,1597,1601,1605,1610,1614,1619,1624],{"type":21,"tag":113,"props":1582,"children":1583},{"style":217},[1584],{"type":26,"value":813},{"type":21,"tag":113,"props":1586,"children":1587},{"style":120},[1588],{"type":26,"value":818},{"type":21,"tag":113,"props":1590,"children":1591},{"style":217},[1592],{"type":26,"value":823},{"type":21,"tag":113,"props":1594,"children":1595},{"style":362},[1596],{"type":26,"value":828},{"type":21,"tag":113,"props":1598,"children":1599},{"style":217},[1600],{"type":26,"value":370},{"type":21,"tag":113,"props":1602,"children":1603},{"style":211},[1604],{"type":26,"value":375},{"type":21,"tag":113,"props":1606,"children":1607},{"style":217},[1608],{"type":26,"value":1609}," sitemap.",{"type":21,"tag":113,"props":1611,"children":1612},{"style":120},[1613],{"type":26,"value":1524},{"type":21,"tag":113,"props":1615,"children":1616},{"style":217},[1617],{"type":26,"value":1618},"({ url: article._path, changefreq: ",{"type":21,"tag":113,"props":1620,"children":1621},{"style":126},[1622],{"type":26,"value":1623},"'monthly'",{"type":21,"tag":113,"props":1625,"children":1626},{"style":217},[1627],{"type":26,"value":1628}," }));\n",{"type":21,"tag":113,"props":1630,"children":1631},{"class":115,"line":514},[1632,1636,1641],{"type":21,"tag":113,"props":1633,"children":1634},{"style":217},[1635],{"type":26,"value":1519},{"type":21,"tag":113,"props":1637,"children":1638},{"style":120},[1639],{"type":26,"value":1640},"end",{"type":21,"tag":113,"props":1642,"children":1643},{"style":217},[1644],{"type":26,"value":796},{"type":21,"tag":113,"props":1646,"children":1647},{"class":115,"line":541},[1648],{"type":21,"tag":113,"props":1649,"children":1650},{"emptyLinePlaceholder":268},[1651],{"type":26,"value":271},{"type":21,"tag":113,"props":1653,"children":1654},{"class":115,"line":567},[1655,1659,1663,1668,1673],{"type":21,"tag":113,"props":1656,"children":1657},{"style":211},[1658],{"type":26,"value":1115},{"type":21,"tag":113,"props":1660,"children":1661},{"style":217},[1662],{"type":26,"value":359},{"type":21,"tag":113,"props":1664,"children":1665},{"style":211},[1666],{"type":26,"value":1667},"await",{"type":21,"tag":113,"props":1669,"children":1670},{"style":120},[1671],{"type":26,"value":1672}," streamToPromise",{"type":21,"tag":113,"props":1674,"children":1675},{"style":217},[1676],{"type":26,"value":1677},"(sitemap));\n",{"type":21,"tag":113,"props":1679,"children":1680},{"class":115,"line":625},[1681],{"type":21,"tag":113,"props":1682,"children":1683},{"style":217},[1684],{"type":26,"value":1138},{"type":21,"tag":22,"props":1686,"children":1687},{},[1688,1690,1695,1697,1702],{"type":26,"value":1689},"And add the ",{"type":21,"tag":109,"props":1691,"children":1693},{"className":1692},[],[1694],{"type":26,"value":1165},{"type":26,"value":1696}," entry in your ",{"type":21,"tag":109,"props":1698,"children":1700},{"className":1699},[],[1701],{"type":26,"value":1173},{"type":26,"value":882},{"type":21,"tag":102,"props":1704,"children":1706},{"className":199,"code":1705,"language":201,"meta":7,"style":7},"nitro: {\n prerender: {\n routes: ['/sitemap.xml', '/atom']\n }\n}\n",[1707],{"type":21,"tag":109,"props":1708,"children":1709},{"__ignoreMap":7},[1710,1721,1732,1761,1768],{"type":21,"tag":113,"props":1711,"children":1712},{"class":115,"line":116},[1713,1717],{"type":21,"tag":113,"props":1714,"children":1715},{"style":120},[1716],{"type":26,"value":1157},{"type":21,"tag":113,"props":1718,"children":1719},{"style":217},[1720],{"type":26,"value":1194},{"type":21,"tag":113,"props":1722,"children":1723},{"class":115,"line":238},[1724,1728],{"type":21,"tag":113,"props":1725,"children":1726},{"style":120},[1727],{"type":26,"value":1202},{"type":21,"tag":113,"props":1729,"children":1730},{"style":217},[1731],{"type":26,"value":1194},{"type":21,"tag":113,"props":1733,"children":1734},{"class":115,"line":264},[1735,1739,1743,1748,1753,1757],{"type":21,"tag":113,"props":1736,"children":1737},{"style":120},[1738],{"type":26,"value":1214},{"type":21,"tag":113,"props":1740,"children":1741},{"style":217},[1742],{"type":26,"value":1219},{"type":21,"tag":113,"props":1744,"children":1745},{"style":126},[1746],{"type":26,"value":1747},"'/sitemap.xml'",{"type":21,"tag":113,"props":1749,"children":1750},{"style":217},[1751],{"type":26,"value":1752},", ",{"type":21,"tag":113,"props":1754,"children":1755},{"style":126},[1756],{"type":26,"value":1224},{"type":21,"tag":113,"props":1758,"children":1759},{"style":217},[1760],{"type":26,"value":1229},{"type":21,"tag":113,"props":1762,"children":1763},{"class":115,"line":274},[1764],{"type":21,"tag":113,"props":1765,"children":1766},{"style":217},[1767],{"type":26,"value":1237},{"type":21,"tag":113,"props":1769,"children":1770},{"class":115,"line":298},[1771],{"type":21,"tag":113,"props":1772,"children":1773},{"style":217},[1774],{"type":26,"value":1245},{"type":21,"tag":1776,"props":1777,"children":1778},"style",{},[1779],{"type":26,"value":1780},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":238,"depth":238,"links":1782},[1783,1784,1785],{"id":54,"depth":238,"text":57},{"id":83,"depth":238,"text":86},{"id":1248,"depth":238,"text":1251},"markdown","content:articles:nuxt-content-rss-feed.md","content","articles/nuxt-content-rss-feed.md","articles/nuxt-content-rss-feed","md",1731782033246] \ No newline at end of file diff --git a/articles/nuxt-content-rss-feed/index.html b/articles/nuxt-content-rss-feed/index.html index f65d47c1..6a351ac3 100644 --- a/articles/nuxt-content-rss-feed/index.html +++ b/articles/nuxt-content-rss-feed/index.html @@ -5,32 +5,32 @@ - - - - - + + + + + - + - - - - - - - - - - - - - + + + + + + + + + + + + + -
        Category

        How To Add an RSS Feed to a Nuxt Website

        If you are a user of Nuxt Content and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.

        Preface

        In version 2 of Nuxt, the community module, nuxt-community/feed-module was a popular choice for adding an RSS feed to your website. However, there has been an unresolved open issue since April 1st, 2022 to add support for Nuxt v3. Thankfully, implementing this feature without a module is relatively straight forward.

        Instructions

        First, install the feed library into your project:

        npm i -D feed
        +
        Category

        How To Add an RSS Feed to a Nuxt Website

        If you are a user of Nuxt Content and you wish to configure an RSS feed for your website, it's a surprisingly straight forward process! In this article, we will walk through the code changes required to host an RSS feed for both a statically generated, and Universal Nuxt application.

        Preface

        In version 2 of Nuxt, the community module, nuxt-community/feed-module was a popular choice for adding an RSS feed to your website. However, there has been an unresolved open issue since April 1st, 2022 to add support for Nuxt v3. Thankfully, implementing this feature without a module is relatively straight forward.

        Instructions

        First, install the feed library into your project:

        npm i -D feed
         

        Then, create a server/ folder in your project if it does not already exist, and create a file named server/routes/atom.ts.

        Here, we will leverage the feed library and construct an XML representation of our Nuxt content. As you can see, we first define our feed with metadata associated with our RSS feed. This will be used by RSS readers to provide context to the end user. Then, we query our Nuxt content with serverQueryContent and append a feed.addItem for each article.

        import { serverQueryContent } from '#content/server';
         import { Feed } from 'feed';
         
        @@ -107,5 +107,5 @@
             routes: ['/sitemap.xml', '/atom']
           }
         }
        -
        - \ No newline at end of file +
        + \ No newline at end of file diff --git a/articles/nuxt-v3-migration/_payload.json b/articles/nuxt-v3-migration/_payload.json index 1a2e1582..0278bd0e 100644 --- a/articles/nuxt-v3-migration/_payload.json +++ b/articles/nuxt-v3-migration/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":125},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"tags":11,"categories":13,"excerpt":15,"body":25,"_type":119,"_id":120,"_source":121,"_file":122,"_stem":123,"_extension":124},"/articles/nuxt-v3-migration","articles",false,"","This Website Has Been Migrated to Nuxt 3 🎉","This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years.\nNot to mention, all of the great plugins in the Vue ecosystem!","2022-12-31",[12],"nuxt",[14],"web",{"type":16,"children":17},"root",[18],{"type":19,"tag":20,"props":21,"children":22},"element","p",{},[23],{"type":24,"value":9},"text",{"type":16,"children":26,"toc":116},[27,31,47,56,61,66,111],{"type":19,"tag":20,"props":28,"children":29},{},[30],{"type":24,"value":9},{"type":19,"tag":20,"props":32,"children":33},{},[34,36,45],{"type":24,"value":35},"If you're curious what changes were required to make the migration, you can check out ",{"type":19,"tag":37,"props":38,"children":42},"a",{"href":39,"rel":40},"https://github.com/cmpadden/cmpadden.github.io/pull/3",[41],"nofollow",[43],{"type":24,"value":44},"pull request #3",{"type":24,"value":46}," in the GitHub repository.",{"type":19,"tag":20,"props":48,"children":49},{},[50],{"type":19,"tag":51,"props":52,"children":55},"img",{"alt":53,"src":54},"Screenshot of Nuxt Migration Pull Request","/images/nuxt-migration-pr.png",[],{"type":19,"tag":20,"props":57,"children":58},{},[59],{"type":24,"value":60},"While the documentation for making this migration is great, there were many breaking changes, and the overall process was quite tedious.\nFor this reason, I opted to generate a new project entirely, and port existing code to this clean slate.\nI believe that this resulted in a project with a bit less cruft.",{"type":19,"tag":20,"props":62,"children":63},{},[64],{"type":24,"value":65},"The most valuable resources for making these changes include:",{"type":19,"tag":67,"props":68,"children":69},"ul",{},[70,81,91,101],{"type":19,"tag":71,"props":72,"children":73},"li",{},[74],{"type":19,"tag":37,"props":75,"children":78},{"href":76,"rel":77},"https://nuxt.com/docs/migration/overview",[41],[79],{"type":24,"value":80},"The Nuxt Migration Guide",{"type":19,"tag":71,"props":82,"children":83},{},[84],{"type":19,"tag":37,"props":85,"children":88},{"href":86,"rel":87},"https://nuxt.com/docs/getting-started/introduction",[41],[89],{"type":24,"value":90},"The Nuxt Framework Guide",{"type":19,"tag":71,"props":92,"children":93},{},[94],{"type":19,"tag":37,"props":95,"children":98},{"href":96,"rel":97},"https://tailwindcss.nuxt.dev/",[41],[99],{"type":24,"value":100},"Nuxt Tailwind Module Documentation",{"type":19,"tag":71,"props":102,"children":103},{},[104],{"type":19,"tag":37,"props":105,"children":108},{"href":106,"rel":107},"https://content.nuxtjs.org/",[41],[109],{"type":24,"value":110},"Nuxt Content Module Documentation",{"type":19,"tag":20,"props":112,"children":113},{},[114],{"type":24,"value":115},"Part of the delay for doing this upgrade was in waiting for module developers to support this major release.\nI'm super thankful for all of the hard work they've don, and I'm excited to explore all of the new features available!\nI just hope that the breaking changes in this release don't cause too much fracturing of the community, as it does feel like déjà vu of Python 2 and 3.",{"title":7,"searchDepth":117,"depth":117,"links":118},2,[],"markdown","content:articles:nuxt-v3-migration.md","content","articles/nuxt-v3-migration.md","articles/nuxt-v3-migration","md",1731781853256] \ No newline at end of file +[{"data":1,"prerenderedAt":125},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"tags":11,"categories":13,"excerpt":15,"body":25,"_type":119,"_id":120,"_source":121,"_file":122,"_stem":123,"_extension":124},"/articles/nuxt-v3-migration","articles",false,"","This Website Has Been Migrated to Nuxt 3 🎉","This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years.\nNot to mention, all of the great plugins in the Vue ecosystem!","2022-12-31",[12],"nuxt",[14],"web",{"type":16,"children":17},"root",[18],{"type":19,"tag":20,"props":21,"children":22},"element","p",{},[23],{"type":24,"value":9},"text",{"type":16,"children":26,"toc":116},[27,31,47,56,61,66,111],{"type":19,"tag":20,"props":28,"children":29},{},[30],{"type":24,"value":9},{"type":19,"tag":20,"props":32,"children":33},{},[34,36,45],{"type":24,"value":35},"If you're curious what changes were required to make the migration, you can check out ",{"type":19,"tag":37,"props":38,"children":42},"a",{"href":39,"rel":40},"https://github.com/cmpadden/cmpadden.github.io/pull/3",[41],"nofollow",[43],{"type":24,"value":44},"pull request #3",{"type":24,"value":46}," in the GitHub repository.",{"type":19,"tag":20,"props":48,"children":49},{},[50],{"type":19,"tag":51,"props":52,"children":55},"img",{"alt":53,"src":54},"Screenshot of Nuxt Migration Pull Request","/images/nuxt-migration-pr.png",[],{"type":19,"tag":20,"props":57,"children":58},{},[59],{"type":24,"value":60},"While the documentation for making this migration is great, there were many breaking changes, and the overall process was quite tedious.\nFor this reason, I opted to generate a new project entirely, and port existing code to this clean slate.\nI believe that this resulted in a project with a bit less cruft.",{"type":19,"tag":20,"props":62,"children":63},{},[64],{"type":24,"value":65},"The most valuable resources for making these changes include:",{"type":19,"tag":67,"props":68,"children":69},"ul",{},[70,81,91,101],{"type":19,"tag":71,"props":72,"children":73},"li",{},[74],{"type":19,"tag":37,"props":75,"children":78},{"href":76,"rel":77},"https://nuxt.com/docs/migration/overview",[41],[79],{"type":24,"value":80},"The Nuxt Migration Guide",{"type":19,"tag":71,"props":82,"children":83},{},[84],{"type":19,"tag":37,"props":85,"children":88},{"href":86,"rel":87},"https://nuxt.com/docs/getting-started/introduction",[41],[89],{"type":24,"value":90},"The Nuxt Framework Guide",{"type":19,"tag":71,"props":92,"children":93},{},[94],{"type":19,"tag":37,"props":95,"children":98},{"href":96,"rel":97},"https://tailwindcss.nuxt.dev/",[41],[99],{"type":24,"value":100},"Nuxt Tailwind Module Documentation",{"type":19,"tag":71,"props":102,"children":103},{},[104],{"type":19,"tag":37,"props":105,"children":108},{"href":106,"rel":107},"https://content.nuxtjs.org/",[41],[109],{"type":24,"value":110},"Nuxt Content Module Documentation",{"type":19,"tag":20,"props":112,"children":113},{},[114],{"type":24,"value":115},"Part of the delay for doing this upgrade was in waiting for module developers to support this major release.\nI'm super thankful for all of the hard work they've don, and I'm excited to explore all of the new features available!\nI just hope that the breaking changes in this release don't cause too much fracturing of the community, as it does feel like déjà vu of Python 2 and 3.",{"title":7,"searchDepth":117,"depth":117,"links":118},2,[],"markdown","content:articles:nuxt-v3-migration.md","content","articles/nuxt-v3-migration.md","articles/nuxt-v3-migration","md",1731782033523] \ No newline at end of file diff --git a/articles/nuxt-v3-migration/index.html b/articles/nuxt-v3-migration/index.html index 48175bbd..4b0b3f3e 100644 --- a/articles/nuxt-v3-migration/index.html +++ b/articles/nuxt-v3-migration/index.html @@ -3,33 +3,33 @@ Colton Padden - - - - - + + + + + - + - - - - - - - - - - - + + + + + + + + + + + -
        Category
        Tags

        This Website Has Been Migrated to Nuxt 3 🎉

        This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years. +

        Category
        Tags

        This Website Has Been Migrated to Nuxt 3 🎉

        This website has been migrated to the latest version of the Nuxt web framework, and with it comes all of the exciting goodies that the Vue and Nuxt teams have been concocting over the past few years. Not to mention, all of the great plugins in the Vue ecosystem!

        If you're curious what changes were required to make the migration, you can check out pull request #3 in the GitHub repository.

        Screenshot of Nuxt Migration Pull Request

        While the documentation for making this migration is great, there were many breaking changes, and the overall process was quite tedious. For this reason, I opted to generate a new project entirely, and port existing code to this clean slate. I believe that this resulted in a project with a bit less cruft.

        The most valuable resources for making these changes include:

        Part of the delay for doing this upgrade was in waiting for module developers to support this major release. I'm super thankful for all of the hard work they've don, and I'm excited to explore all of the new features available! -I just hope that the breaking changes in this release don't cause too much fracturing of the community, as it does feel like déjà vu of Python 2 and 3.

        - \ No newline at end of file +I just hope that the breaking changes in this release don't cause too much fracturing of the community, as it does feel like déjà vu of Python 2 and 3.

        + \ No newline at end of file diff --git a/articles/persistent-archlinux-usb/_payload.json b/articles/persistent-archlinux-usb/_payload.json index 2a614e8f..7e179924 100644 --- a/articles/persistent-archlinux-usb/_payload.json +++ b/articles/persistent-archlinux-usb/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":1337},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"draft":6,"tags":11,"categories":14,"excerpt":16,"body":26,"_type":1331,"_id":1332,"_source":1333,"_file":1334,"_stem":1335,"_extension":1336},"/articles/persistent-archlinux-usb","articles",false,"","Create a Persistent Arch Linux Bootable USB with Vagrant","When installing a linux distribution, it is common for the instructions to have\nthe user create a bootable USB, boot from the device, and proceed with the\ninstallation procedure from that live medium. However, this blog post will\noutline an alternative approach where a virtual machine created with Vagrant\nwill be used in favor of the live medium.","2020-01-09",[12,13],"vagrant","archlinux",[15],"linux",{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24],{"type":25,"value":9},"text",{"type":17,"children":27,"toc":1320},[28,32,39,85,91,98,115,152,165,190,246,354,367,388,654,660,674,681,705,825,831,846,885,908,941,947,1002,1067,1073,1081,1089,1097,1120,1128,1133,1138,1144,1174,1204,1210,1314],{"type":20,"tag":21,"props":29,"children":30},{},[31],{"type":25,"value":9},{"type":20,"tag":33,"props":34,"children":36},"h1",{"id":35},"preface",[37],{"type":25,"value":38},"Preface",{"type":20,"tag":21,"props":40,"children":41},{},[42,44,51,53,83],{"type":25,"value":43},"The original intention was to use Docker for this process -- leveraging the\n",{"type":20,"tag":45,"props":46,"children":48},"code",{"className":47},[],[49],{"type":25,"value":50},"--device",{"type":25,"value":52}," flag and mounting the target USB device in the Docker container,\nbut the underlying hypervisor in Docker Desktop for Mac does not support this.\n",{"type":20,"tag":54,"props":55,"children":56},"sup",{},[57,66,68,75,76],{"type":20,"tag":58,"props":59,"children":63},"a",{"href":60,"rel":61},"https://docs.docker.com/docker-for-mac/docker-toolbox/",[62],"nofollow",[64],{"type":25,"value":65},"1",{"type":25,"value":67}," ",{"type":20,"tag":58,"props":69,"children":72},{"href":70,"rel":71},"https://github.com/moby/hyperkit",[62],[73],{"type":25,"value":74},"2",{"type":25,"value":67},{"type":20,"tag":58,"props":77,"children":80},{"href":78,"rel":79},"https://github.com/docker/for-mac/issues/900",[62],[81],{"type":25,"value":82},"3",{"type":25,"value":84}," While there are workarounds using Docker\nMachine, Vagrant felt like the path of least resistance.",{"type":20,"tag":33,"props":86,"children":88},{"id":87},"instructions",[89],{"type":25,"value":90},"Instructions",{"type":20,"tag":92,"props":93,"children":95},"h2",{"id":94},"create-an-arch-linux-virtual-machine-with-vagrant",[96],{"type":25,"value":97},"Create an Arch Linux Virtual Machine with Vagrant",{"type":20,"tag":21,"props":99,"children":100},{},[101,103,113],{"type":25,"value":102},"Get the latest Arch Linux image ",{"type":20,"tag":54,"props":104,"children":105},{},[106],{"type":20,"tag":58,"props":107,"children":110},{"href":108,"rel":109},"https://app.vagrantup.com/archlinux/boxes/archlinux",[62],[111],{"type":25,"value":112},"4",{"type":25,"value":114}," from the Vagrant Cloud Box\nCatalog.",{"type":20,"tag":116,"props":117,"children":121},"pre",{"code":118,"language":119,"meta":7,"className":120,"style":7},"vagrant box add archlinux/archlinux\n","bash","language-bash shiki shiki-themes github-light",[122],{"type":20,"tag":45,"props":123,"children":124},{"__ignoreMap":7},[125],{"type":20,"tag":126,"props":127,"children":130},"span",{"class":128,"line":129},"line",1,[131,136,142,147],{"type":20,"tag":126,"props":132,"children":134},{"style":133},"--shiki-default:#6F42C1",[135],{"type":25,"value":12},{"type":20,"tag":126,"props":137,"children":139},{"style":138},"--shiki-default:#032F62",[140],{"type":25,"value":141}," box",{"type":20,"tag":126,"props":143,"children":144},{"style":138},[145],{"type":25,"value":146}," add",{"type":20,"tag":126,"props":148,"children":149},{"style":138},[150],{"type":25,"value":151}," archlinux/archlinux\n",{"type":20,"tag":21,"props":153,"children":154},{},[155,157,163],{"type":25,"value":156},"Determine the USB vendor information for the thumb-drive that we will\npass-through to the virtual machine. Using the ",{"type":20,"tag":45,"props":158,"children":160},{"className":159},[],[161],{"type":25,"value":162},"VBoxManage",{"type":25,"value":164}," utility that comes\nwith Virtual Box, list the devices, and make note of the Vendor and Product ID.",{"type":20,"tag":116,"props":166,"children":168},{"code":167,"language":119,"meta":7,"className":120,"style":7}," VBoxManage list usbhost\n",[169],{"type":20,"tag":45,"props":170,"children":171},{"__ignoreMap":7},[172],{"type":20,"tag":126,"props":173,"children":174},{"class":128,"line":129},[175,180,185],{"type":20,"tag":126,"props":176,"children":177},{"style":133},[178],{"type":25,"value":179}," VBoxManage",{"type":20,"tag":126,"props":181,"children":182},{"style":138},[183],{"type":25,"value":184}," list",{"type":20,"tag":126,"props":186,"children":187},{"style":138},[188],{"type":25,"value":189}," usbhost\n",{"type":20,"tag":21,"props":191,"children":192},{},[193,195,201,203,209,211,229,231,237,239,244],{"type":25,"value":194},"Create a ",{"type":20,"tag":45,"props":196,"children":198},{"className":197},[],[199],{"type":25,"value":200},"Vagrantfile",{"type":25,"value":202}," with ",{"type":20,"tag":45,"props":204,"children":206},{"className":205},[],[207],{"type":25,"value":208},"archlinx/archlinux",{"type":25,"value":210}," as the target box, and the USB\ndevice information that is passed through. ",{"type":20,"tag":54,"props":212,"children":213},{},[214,221,222],{"type":20,"tag":58,"props":215,"children":218},{"href":216,"rel":217},"https://antonyjepson.wordpress.com/2012/01/26/quickly-attaching-usb-devices-to-virtualbox-guests-using-vboxmanage/",[62],[219],{"type":25,"value":220},"5",{"type":25,"value":67},{"type":20,"tag":58,"props":223,"children":226},{"href":224,"rel":225},"https://gist.github.com/dscape/7d829c0c116ef419f963",[62],[227],{"type":25,"value":228},"6",{"type":25,"value":230}," Vagrant\noffers a handy customization parameter ",{"type":20,"tag":45,"props":232,"children":234},{"className":233},[],[235],{"type":25,"value":236},"vb.customize",{"type":25,"value":238}," that calls the\n",{"type":20,"tag":45,"props":240,"children":242},{"className":241},[],[243],{"type":25,"value":162},{"type":25,"value":245}," command under-the-hood, allowing one to enable the guest machine\nto access the host machine's USB devices.",{"type":20,"tag":116,"props":247,"children":251},{"code":248,"language":249,"meta":7,"className":250,"style":7},"# -*- mode: ruby -*-\n# vi: set ft=ruby :\n\nVagrant.configure(\"2\") do |config|\n config.vm.box = \"archlinux/archlinux\"\n config.vm.provider \"virtualbox\" do |vb|\n vb.name = \"archlinux\"\n vb.customize ['modifyvm', :id, '--usb', 'on']\n vb.customize ['usbfilter', 'add', '1', '--target', :id, '--name', 'SanDisk Ultra Fit', '--vendorid', '0x0781', '--productid', '0x5583']\n end\nend\n","txt","language-txt shiki shiki-themes github-light",[252],{"type":20,"tag":45,"props":253,"children":254},{"__ignoreMap":7},[255,263,272,282,291,300,309,318,327,336,345],{"type":20,"tag":126,"props":256,"children":257},{"class":128,"line":129},[258],{"type":20,"tag":126,"props":259,"children":260},{},[261],{"type":25,"value":262},"# -*- mode: ruby -*-\n",{"type":20,"tag":126,"props":264,"children":266},{"class":128,"line":265},2,[267],{"type":20,"tag":126,"props":268,"children":269},{},[270],{"type":25,"value":271},"# vi: set ft=ruby :\n",{"type":20,"tag":126,"props":273,"children":275},{"class":128,"line":274},3,[276],{"type":20,"tag":126,"props":277,"children":279},{"emptyLinePlaceholder":278},true,[280],{"type":25,"value":281},"\n",{"type":20,"tag":126,"props":283,"children":285},{"class":128,"line":284},4,[286],{"type":20,"tag":126,"props":287,"children":288},{},[289],{"type":25,"value":290},"Vagrant.configure(\"2\") do |config|\n",{"type":20,"tag":126,"props":292,"children":294},{"class":128,"line":293},5,[295],{"type":20,"tag":126,"props":296,"children":297},{},[298],{"type":25,"value":299}," config.vm.box = \"archlinux/archlinux\"\n",{"type":20,"tag":126,"props":301,"children":303},{"class":128,"line":302},6,[304],{"type":20,"tag":126,"props":305,"children":306},{},[307],{"type":25,"value":308}," config.vm.provider \"virtualbox\" do |vb|\n",{"type":20,"tag":126,"props":310,"children":312},{"class":128,"line":311},7,[313],{"type":20,"tag":126,"props":314,"children":315},{},[316],{"type":25,"value":317}," vb.name = \"archlinux\"\n",{"type":20,"tag":126,"props":319,"children":321},{"class":128,"line":320},8,[322],{"type":20,"tag":126,"props":323,"children":324},{},[325],{"type":25,"value":326}," vb.customize ['modifyvm', :id, '--usb', 'on']\n",{"type":20,"tag":126,"props":328,"children":330},{"class":128,"line":329},9,[331],{"type":20,"tag":126,"props":332,"children":333},{},[334],{"type":25,"value":335}," vb.customize ['usbfilter', 'add', '1', '--target', :id, '--name', 'SanDisk Ultra Fit', '--vendorid', '0x0781', '--productid', '0x5583']\n",{"type":20,"tag":126,"props":337,"children":339},{"class":128,"line":338},10,[340],{"type":20,"tag":126,"props":341,"children":342},{},[343],{"type":25,"value":344}," end\n",{"type":20,"tag":126,"props":346,"children":348},{"class":128,"line":347},11,[349],{"type":20,"tag":126,"props":350,"children":351},{},[352],{"type":25,"value":353},"end\n",{"type":20,"tag":21,"props":355,"children":356},{},[357,359,365],{"type":25,"value":358},"When virtual machine is brought up, the ",{"type":20,"tag":45,"props":360,"children":362},{"className":361},[],[363],{"type":25,"value":364},"usbfilter",{"type":25,"value":366}," is applied, and the guest\nis able to access to the host machine's USB device that was specified in the\nfilter.",{"type":20,"tag":21,"props":368,"children":369},{},[370,372,378,380,386],{"type":25,"value":371},"Start the machine, ",{"type":20,"tag":45,"props":373,"children":375},{"className":374},[],[376],{"type":25,"value":377},"ssh",{"type":25,"value":379}," into the guest, and list the devices to confirm that\nthe USB device is available (see: ",{"type":20,"tag":45,"props":381,"children":383},{"className":382},[],[384],{"type":25,"value":385},"/dev/sdb",{"type":25,"value":387},").",{"type":20,"tag":116,"props":389,"children":391},{"code":390,"language":119,"meta":7,"className":120,"style":7},"$ vagrant up\n$ vagrant ssh\n[vagrant@archlinux ~]$ lsblk\nNAME MAJ:MIN RM SIZE RO TYPE MOUNTPOINT\nsda 8:0 0 20G 0 disk\n├─sda1 8:1 0 1.9G 0 part [SWAP]\n└─sda2 8:2 0 18.1G 0 part /\nsdb 8:16 1 28.7G 0 disk\n└─sdb1 8:17 1 8G 0 part\n",[392],{"type":20,"tag":45,"props":393,"children":394},{"__ignoreMap":7},[395,413,429,449,487,521,557,592,623],{"type":20,"tag":126,"props":396,"children":397},{"class":128,"line":129},[398,403,408],{"type":20,"tag":126,"props":399,"children":400},{"style":133},[401],{"type":25,"value":402},"$",{"type":20,"tag":126,"props":404,"children":405},{"style":138},[406],{"type":25,"value":407}," vagrant",{"type":20,"tag":126,"props":409,"children":410},{"style":138},[411],{"type":25,"value":412}," up\n",{"type":20,"tag":126,"props":414,"children":415},{"class":128,"line":265},[416,420,424],{"type":20,"tag":126,"props":417,"children":418},{"style":133},[419],{"type":25,"value":402},{"type":20,"tag":126,"props":421,"children":422},{"style":138},[423],{"type":25,"value":407},{"type":20,"tag":126,"props":425,"children":426},{"style":138},[427],{"type":25,"value":428}," ssh\n",{"type":20,"tag":126,"props":430,"children":431},{"class":128,"line":274},[432,438,444],{"type":20,"tag":126,"props":433,"children":435},{"style":434},"--shiki-default:#24292E",[436],{"type":25,"value":437},"[vagrant@archlinux ",{"type":20,"tag":126,"props":439,"children":441},{"style":440},"--shiki-default:#D73A49",[442],{"type":25,"value":443},"~",{"type":20,"tag":126,"props":445,"children":446},{"style":434},[447],{"type":25,"value":448},"]$ lsblk\n",{"type":20,"tag":126,"props":450,"children":451},{"class":128,"line":284},[452,457,462,467,472,477,482],{"type":20,"tag":126,"props":453,"children":454},{"style":133},[455],{"type":25,"value":456},"NAME",{"type":20,"tag":126,"props":458,"children":459},{"style":138},[460],{"type":25,"value":461}," MAJ:MIN",{"type":20,"tag":126,"props":463,"children":464},{"style":138},[465],{"type":25,"value":466}," RM",{"type":20,"tag":126,"props":468,"children":469},{"style":138},[470],{"type":25,"value":471}," SIZE",{"type":20,"tag":126,"props":473,"children":474},{"style":138},[475],{"type":25,"value":476}," RO",{"type":20,"tag":126,"props":478,"children":479},{"style":138},[480],{"type":25,"value":481}," TYPE",{"type":20,"tag":126,"props":483,"children":484},{"style":138},[485],{"type":25,"value":486}," MOUNTPOINT\n",{"type":20,"tag":126,"props":488,"children":489},{"class":128,"line":293},[490,495,500,506,511,516],{"type":20,"tag":126,"props":491,"children":492},{"style":133},[493],{"type":25,"value":494},"sda",{"type":20,"tag":126,"props":496,"children":497},{"style":138},[498],{"type":25,"value":499}," 8:0",{"type":20,"tag":126,"props":501,"children":503},{"style":502},"--shiki-default:#005CC5",[504],{"type":25,"value":505}," 0",{"type":20,"tag":126,"props":507,"children":508},{"style":138},[509],{"type":25,"value":510}," 20G",{"type":20,"tag":126,"props":512,"children":513},{"style":502},[514],{"type":25,"value":515}," 0",{"type":20,"tag":126,"props":517,"children":518},{"style":138},[519],{"type":25,"value":520}," disk\n",{"type":20,"tag":126,"props":522,"children":523},{"class":128,"line":302},[524,529,534,538,543,547,552],{"type":20,"tag":126,"props":525,"children":526},{"style":133},[527],{"type":25,"value":528},"├─sda1",{"type":20,"tag":126,"props":530,"children":531},{"style":138},[532],{"type":25,"value":533}," 8:1",{"type":20,"tag":126,"props":535,"children":536},{"style":502},[537],{"type":25,"value":505},{"type":20,"tag":126,"props":539,"children":540},{"style":138},[541],{"type":25,"value":542}," 1.9G",{"type":20,"tag":126,"props":544,"children":545},{"style":502},[546],{"type":25,"value":515},{"type":20,"tag":126,"props":548,"children":549},{"style":138},[550],{"type":25,"value":551}," part",{"type":20,"tag":126,"props":553,"children":554},{"style":434},[555],{"type":25,"value":556}," [SWAP]\n",{"type":20,"tag":126,"props":558,"children":559},{"class":128,"line":311},[560,565,570,574,579,583,587],{"type":20,"tag":126,"props":561,"children":562},{"style":133},[563],{"type":25,"value":564},"└─sda2",{"type":20,"tag":126,"props":566,"children":567},{"style":138},[568],{"type":25,"value":569}," 8:2",{"type":20,"tag":126,"props":571,"children":572},{"style":502},[573],{"type":25,"value":505},{"type":20,"tag":126,"props":575,"children":576},{"style":138},[577],{"type":25,"value":578}," 18.1G",{"type":20,"tag":126,"props":580,"children":581},{"style":502},[582],{"type":25,"value":515},{"type":20,"tag":126,"props":584,"children":585},{"style":138},[586],{"type":25,"value":551},{"type":20,"tag":126,"props":588,"children":589},{"style":138},[590],{"type":25,"value":591}," /\n",{"type":20,"tag":126,"props":593,"children":594},{"class":128,"line":320},[595,600,605,610,615,619],{"type":20,"tag":126,"props":596,"children":597},{"style":133},[598],{"type":25,"value":599},"sdb",{"type":20,"tag":126,"props":601,"children":602},{"style":138},[603],{"type":25,"value":604}," 8:16",{"type":20,"tag":126,"props":606,"children":607},{"style":502},[608],{"type":25,"value":609}," 1",{"type":20,"tag":126,"props":611,"children":612},{"style":138},[613],{"type":25,"value":614}," 28.7G",{"type":20,"tag":126,"props":616,"children":617},{"style":502},[618],{"type":25,"value":515},{"type":20,"tag":126,"props":620,"children":621},{"style":138},[622],{"type":25,"value":520},{"type":20,"tag":126,"props":624,"children":625},{"class":128,"line":329},[626,631,636,640,645,649],{"type":20,"tag":126,"props":627,"children":628},{"style":133},[629],{"type":25,"value":630},"└─sdb1",{"type":20,"tag":126,"props":632,"children":633},{"style":138},[634],{"type":25,"value":635}," 8:17",{"type":20,"tag":126,"props":637,"children":638},{"style":502},[639],{"type":25,"value":609},{"type":20,"tag":126,"props":641,"children":642},{"style":138},[643],{"type":25,"value":644}," 8G",{"type":20,"tag":126,"props":646,"children":647},{"style":502},[648],{"type":25,"value":515},{"type":20,"tag":126,"props":650,"children":651},{"style":138},[652],{"type":25,"value":653}," part\n",{"type":20,"tag":92,"props":655,"children":657},{"id":656},"install-arch-linux-on-the-usb-drive",[658],{"type":25,"value":659},"Install Arch Linux on the USB Drive",{"type":20,"tag":21,"props":661,"children":662},{},[663,665,672],{"type":25,"value":664},"The ",{"type":20,"tag":58,"props":666,"children":669},{"href":667,"rel":668},"https://wiki.archlinux.org/index.php/Installation_guide",[62],[670],{"type":25,"value":671},"Arch Linux Installation Guide",{"type":25,"value":673}," outlines the installation procedure in\ngreat detail -- the following steps follow this closely with a few alteration\ndue to installing onto removable media.",{"type":20,"tag":675,"props":676,"children":678},"h3",{"id":677},"partition-the-disk-uefi-with-gpt",[679],{"type":25,"value":680},"Partition the Disk (UEFI with GPT)",{"type":20,"tag":116,"props":682,"children":684},{"code":683,"language":119,"meta":7,"className":120,"style":7},"[root@archlinux ~]# fdisk /dev/sdb\n",[685],{"type":20,"tag":45,"props":686,"children":687},{"__ignoreMap":7},[688],{"type":20,"tag":126,"props":689,"children":690},{"class":128,"line":129},[691,696,700],{"type":20,"tag":126,"props":692,"children":693},{"style":434},[694],{"type":25,"value":695},"[root@archlinux ",{"type":20,"tag":126,"props":697,"children":698},{"style":440},[699],{"type":25,"value":443},{"type":20,"tag":126,"props":701,"children":702},{"style":434},[703],{"type":25,"value":704},"]# fdisk /dev/sdb\n",{"type":20,"tag":116,"props":706,"children":708},{"code":707,"language":249,"meta":7,"className":250,"style":7},"Command (m for help): p\nDisk /dev/sdb: 28.66 GiB, 30752636928 bytes, 60063744 sectors\nDisk model: Ultra Fit\nUnits: sectors of 1 * 512 = 512 bytes\nSector size (logical/physical): 512 bytes / 512 bytes\nI/O size (minimum/optimal): 512 bytes / 512 bytes\nDisklabel type: gpt\nDisk identifier: E1D6C445-1B79-AB4D-A442-FA4AD6DF4ECC\n\nDevice Start End Sectors Size Type\n/dev/sdb1 2048 1050623 1048576 512M EFI System\n/dev/sdb2 1050624 60063710 59013087 28.1G Linux filesystem\n\nFilesystem/RAID signature on partition 1 will be wiped.\n",[709],{"type":20,"tag":45,"props":710,"children":711},{"__ignoreMap":7},[712,720,728,736,744,752,760,768,776,783,791,799,808,816],{"type":20,"tag":126,"props":713,"children":714},{"class":128,"line":129},[715],{"type":20,"tag":126,"props":716,"children":717},{},[718],{"type":25,"value":719},"Command (m for help): p\n",{"type":20,"tag":126,"props":721,"children":722},{"class":128,"line":265},[723],{"type":20,"tag":126,"props":724,"children":725},{},[726],{"type":25,"value":727},"Disk /dev/sdb: 28.66 GiB, 30752636928 bytes, 60063744 sectors\n",{"type":20,"tag":126,"props":729,"children":730},{"class":128,"line":274},[731],{"type":20,"tag":126,"props":732,"children":733},{},[734],{"type":25,"value":735},"Disk model: Ultra Fit\n",{"type":20,"tag":126,"props":737,"children":738},{"class":128,"line":284},[739],{"type":20,"tag":126,"props":740,"children":741},{},[742],{"type":25,"value":743},"Units: sectors of 1 * 512 = 512 bytes\n",{"type":20,"tag":126,"props":745,"children":746},{"class":128,"line":293},[747],{"type":20,"tag":126,"props":748,"children":749},{},[750],{"type":25,"value":751},"Sector size (logical/physical): 512 bytes / 512 bytes\n",{"type":20,"tag":126,"props":753,"children":754},{"class":128,"line":302},[755],{"type":20,"tag":126,"props":756,"children":757},{},[758],{"type":25,"value":759},"I/O size (minimum/optimal): 512 bytes / 512 bytes\n",{"type":20,"tag":126,"props":761,"children":762},{"class":128,"line":311},[763],{"type":20,"tag":126,"props":764,"children":765},{},[766],{"type":25,"value":767},"Disklabel type: gpt\n",{"type":20,"tag":126,"props":769,"children":770},{"class":128,"line":320},[771],{"type":20,"tag":126,"props":772,"children":773},{},[774],{"type":25,"value":775},"Disk identifier: E1D6C445-1B79-AB4D-A442-FA4AD6DF4ECC\n",{"type":20,"tag":126,"props":777,"children":778},{"class":128,"line":329},[779],{"type":20,"tag":126,"props":780,"children":781},{"emptyLinePlaceholder":278},[782],{"type":25,"value":281},{"type":20,"tag":126,"props":784,"children":785},{"class":128,"line":338},[786],{"type":20,"tag":126,"props":787,"children":788},{},[789],{"type":25,"value":790},"Device Start End Sectors Size Type\n",{"type":20,"tag":126,"props":792,"children":793},{"class":128,"line":347},[794],{"type":20,"tag":126,"props":795,"children":796},{},[797],{"type":25,"value":798},"/dev/sdb1 2048 1050623 1048576 512M EFI System\n",{"type":20,"tag":126,"props":800,"children":802},{"class":128,"line":801},12,[803],{"type":20,"tag":126,"props":804,"children":805},{},[806],{"type":25,"value":807},"/dev/sdb2 1050624 60063710 59013087 28.1G Linux filesystem\n",{"type":20,"tag":126,"props":809,"children":811},{"class":128,"line":810},13,[812],{"type":20,"tag":126,"props":813,"children":814},{"emptyLinePlaceholder":278},[815],{"type":25,"value":281},{"type":20,"tag":126,"props":817,"children":819},{"class":128,"line":818},14,[820],{"type":20,"tag":126,"props":821,"children":822},{},[823],{"type":25,"value":824},"Filesystem/RAID signature on partition 1 will be wiped.\n",{"type":20,"tag":675,"props":826,"children":828},{"id":827},"format-the-partitions",[829],{"type":25,"value":830},"Format the Partitions",{"type":20,"tag":21,"props":832,"children":833},{},[834,836],{"type":25,"value":835},"The UEFI specification mandates support for FAT file-systems, and FAT32 is\nrecommended for removable media. ",{"type":20,"tag":54,"props":837,"children":838},{},[839],{"type":20,"tag":58,"props":840,"children":843},{"href":841,"rel":842},"https://wiki.archlinux.org/index.php/EFI_system_partition#Format_the_partition",[62],[844],{"type":25,"value":845},"7",{"type":20,"tag":116,"props":847,"children":849},{"code":848,"language":119,"meta":7,"className":120,"style":7},"[root@archlinux ~]# pacman -Sy dosfstools\n[root@archlinux ~]# mkfs.fat -F32 /dev/sdb1\n",[850],{"type":20,"tag":45,"props":851,"children":852},{"__ignoreMap":7},[853,869],{"type":20,"tag":126,"props":854,"children":855},{"class":128,"line":129},[856,860,864],{"type":20,"tag":126,"props":857,"children":858},{"style":434},[859],{"type":25,"value":695},{"type":20,"tag":126,"props":861,"children":862},{"style":440},[863],{"type":25,"value":443},{"type":20,"tag":126,"props":865,"children":866},{"style":434},[867],{"type":25,"value":868},"]# pacman -Sy dosfstools\n",{"type":20,"tag":126,"props":870,"children":871},{"class":128,"line":265},[872,876,880],{"type":20,"tag":126,"props":873,"children":874},{"style":434},[875],{"type":25,"value":695},{"type":20,"tag":126,"props":877,"children":878},{"style":440},[879],{"type":25,"value":443},{"type":20,"tag":126,"props":881,"children":882},{"style":434},[883],{"type":25,"value":884},"]# mkfs.fat -F32 /dev/sdb1\n",{"type":20,"tag":21,"props":886,"children":887},{},[888,890,896,898],{"type":25,"value":889},"As for the root partition, it is recommended to use ",{"type":20,"tag":45,"props":891,"children":893},{"className":892},[],[894],{"type":25,"value":895},"ext4",{"type":25,"value":897}," without a journal to\nreduce the reads and writes to the file-system as this is detrimental to the\nflash-based USB drive. ",{"type":20,"tag":54,"props":899,"children":900},{},[901],{"type":20,"tag":58,"props":902,"children":905},{"href":903,"rel":904},"https://wiki.archlinux.org/index.php/Install_Arch_Linux_on_a_USB_key#Installation_tweaks",[62],[906],{"type":25,"value":907},"8",{"type":20,"tag":116,"props":909,"children":911},{"code":910,"language":119,"meta":7,"className":120,"style":7},"[root@archlinux ~]# mkfs.ext4 -O \"^has_journal\" /dev/sdb2\n",[912],{"type":20,"tag":45,"props":913,"children":914},{"__ignoreMap":7},[915],{"type":20,"tag":126,"props":916,"children":917},{"class":128,"line":129},[918,922,926,931,936],{"type":20,"tag":126,"props":919,"children":920},{"style":434},[921],{"type":25,"value":695},{"type":20,"tag":126,"props":923,"children":924},{"style":440},[925],{"type":25,"value":443},{"type":20,"tag":126,"props":927,"children":928},{"style":434},[929],{"type":25,"value":930},"]# mkfs.ext4 -O ",{"type":20,"tag":126,"props":932,"children":933},{"style":138},[934],{"type":25,"value":935},"\"^has_journal\"",{"type":20,"tag":126,"props":937,"children":938},{"style":434},[939],{"type":25,"value":940}," /dev/sdb2\n",{"type":20,"tag":675,"props":942,"children":944},{"id":943},"mount-the-partitions-and-bootstrap-the-environment",[945],{"type":25,"value":946},"Mount the Partitions and Bootstrap the Environment",{"type":20,"tag":116,"props":948,"children":950},{"code":949,"language":119,"meta":7,"className":120,"style":7},"[root@archlinux ~]# mount /dev/sdb2 /mnt\n[root@archlinux ~]# mkdir -p /mnt/boot/efi\n[root@archlinux ~]# mount /dev/sdb1 /mnt/boot/efi\n",[951],{"type":20,"tag":45,"props":952,"children":953},{"__ignoreMap":7},[954,970,986],{"type":20,"tag":126,"props":955,"children":956},{"class":128,"line":129},[957,961,965],{"type":20,"tag":126,"props":958,"children":959},{"style":434},[960],{"type":25,"value":695},{"type":20,"tag":126,"props":962,"children":963},{"style":440},[964],{"type":25,"value":443},{"type":20,"tag":126,"props":966,"children":967},{"style":434},[968],{"type":25,"value":969},"]# mount /dev/sdb2 /mnt\n",{"type":20,"tag":126,"props":971,"children":972},{"class":128,"line":265},[973,977,981],{"type":20,"tag":126,"props":974,"children":975},{"style":434},[976],{"type":25,"value":695},{"type":20,"tag":126,"props":978,"children":979},{"style":440},[980],{"type":25,"value":443},{"type":20,"tag":126,"props":982,"children":983},{"style":434},[984],{"type":25,"value":985},"]# mkdir -p /mnt/boot/efi\n",{"type":20,"tag":126,"props":987,"children":988},{"class":128,"line":274},[989,993,997],{"type":20,"tag":126,"props":990,"children":991},{"style":434},[992],{"type":25,"value":695},{"type":20,"tag":126,"props":994,"children":995},{"style":440},[996],{"type":25,"value":443},{"type":20,"tag":126,"props":998,"children":999},{"style":434},[1000],{"type":25,"value":1001},"]# mount /dev/sdb1 /mnt/boot/efi\n",{"type":20,"tag":116,"props":1003,"children":1005},{"code":1004,"language":119,"meta":7,"className":120,"style":7},"[root@archlinux ~]# pacman -S arch-install-scripts\n[root@archlinux ~]# pacstrap /mnt base linux linux-firmware\n[root@archlinux ~]# genfstab -U /mnt >> /mnt/etc/fstab\n",[1006],{"type":20,"tag":45,"props":1007,"children":1008},{"__ignoreMap":7},[1009,1025,1041],{"type":20,"tag":126,"props":1010,"children":1011},{"class":128,"line":129},[1012,1016,1020],{"type":20,"tag":126,"props":1013,"children":1014},{"style":434},[1015],{"type":25,"value":695},{"type":20,"tag":126,"props":1017,"children":1018},{"style":440},[1019],{"type":25,"value":443},{"type":20,"tag":126,"props":1021,"children":1022},{"style":434},[1023],{"type":25,"value":1024},"]# pacman -S arch-install-scripts\n",{"type":20,"tag":126,"props":1026,"children":1027},{"class":128,"line":265},[1028,1032,1036],{"type":20,"tag":126,"props":1029,"children":1030},{"style":434},[1031],{"type":25,"value":695},{"type":20,"tag":126,"props":1033,"children":1034},{"style":440},[1035],{"type":25,"value":443},{"type":20,"tag":126,"props":1037,"children":1038},{"style":434},[1039],{"type":25,"value":1040},"]# pacstrap /mnt base linux linux-firmware\n",{"type":20,"tag":126,"props":1042,"children":1043},{"class":128,"line":274},[1044,1048,1052,1057,1062],{"type":20,"tag":126,"props":1045,"children":1046},{"style":434},[1047],{"type":25,"value":695},{"type":20,"tag":126,"props":1049,"children":1050},{"style":440},[1051],{"type":25,"value":443},{"type":20,"tag":126,"props":1053,"children":1054},{"style":434},[1055],{"type":25,"value":1056},"]# genfstab -U /mnt ",{"type":20,"tag":126,"props":1058,"children":1059},{"style":440},[1060],{"type":25,"value":1061},">>",{"type":20,"tag":126,"props":1063,"children":1064},{"style":434},[1065],{"type":25,"value":1066}," /mnt/etc/fstab\n",{"type":20,"tag":675,"props":1068,"children":1070},{"id":1069},"configure-the-new-environment",[1071],{"type":25,"value":1072},"Configure the New Environment",{"type":20,"tag":116,"props":1074,"children":1076},{"code":1075},"[root@archlinux ~]# arch-chroot /mnt\n",[1077],{"type":20,"tag":45,"props":1078,"children":1079},{"__ignoreMap":7},[1080],{"type":25,"value":1075},{"type":20,"tag":116,"props":1082,"children":1084},{"code":1083},"[root@archlinux /]# ln -sf /usr/share/zoneinfo/US/Eastern /etc/localtime\n[root@archlinux /]# hwclock --systohc\n[root@archlinux /]# sed -i 's/#en_US.UTF-8/en_US.UTF-8/' /etc/locale.gen\n[root@archlinux /]# locale-gen\n[root@archlinux /]# echo \"LANG=en_US.UTF-8\" > /etc/locale.conf\n",[1085],{"type":20,"tag":45,"props":1086,"children":1087},{"__ignoreMap":7},[1088],{"type":25,"value":1083},{"type":20,"tag":116,"props":1090,"children":1092},{"code":1091},"[root@archlinux /]# hostnamectl set-hostname usb\n[root@archlinux /]# echo \"127.0.0.1 localhost\" >> /etc/hosts\n[root@archlinux /]# echo \"::1 localhost\" >> /etc/hosts\n",[1093],{"type":20,"tag":45,"props":1094,"children":1095},{"__ignoreMap":7},[1096],{"type":25,"value":1091},{"type":20,"tag":21,"props":1098,"children":1099},{},[1100,1102,1108,1110],{"type":25,"value":1101},"Note, one difference here from a standard installation is that the\n",{"type":20,"tag":45,"props":1103,"children":1105},{"className":1104},[],[1106],{"type":25,"value":1107},"--removable",{"type":25,"value":1109}," flag is specified when installing the GRUB bootloader.\n",{"type":20,"tag":54,"props":1111,"children":1112},{},[1113],{"type":20,"tag":58,"props":1114,"children":1117},{"href":1115,"rel":1116},"https://wiki.archlinux.org/index.php/GRUB#UEFI_systems",[62],[1118],{"type":25,"value":1119},"10",{"type":20,"tag":116,"props":1121,"children":1123},{"code":1122},"[root@archlinux /]# pacman -S grub\n[root@archlinux /]# grub-install --target=x86_64-efi --efi-directory=/boot/efi --removable --recheck\n[root@usb /]# grub-mkconfig -o /boot/grub/grub.cfg\n",[1124],{"type":20,"tag":45,"props":1125,"children":1126},{"__ignoreMap":7},[1127],{"type":25,"value":1122},{"type":20,"tag":21,"props":1129,"children":1130},{},[1131],{"type":25,"value":1132},"Shutdown the virtual machine, restart the host machine, and boot the newly\ncreated Arch Linux thumb-drive!",{"type":20,"tag":21,"props":1134,"children":1135},{},[1136],{"type":25,"value":1137},"🎉",{"type":20,"tag":92,"props":1139,"children":1141},{"id":1140},"side-note",[1142],{"type":25,"value":1143},"Side-note",{"type":20,"tag":21,"props":1145,"children":1146},{},[1147,1149,1155,1157,1162,1164],{"type":25,"value":1148},"It was attempted to use the ",{"type":20,"tag":45,"props":1150,"children":1152},{"className":1151},[],[1153],{"type":25,"value":1154},"controlvm usbattach",{"type":25,"value":1156}," command to pass the USB\ndevice to the guest machine, but this did not work as it expects the virtual\nmachine to already be running, and the ",{"type":20,"tag":45,"props":1158,"children":1160},{"className":1159},[],[1161],{"type":25,"value":236},{"type":25,"value":1163}," option runs prior to\nbooting the machine. ",{"type":20,"tag":54,"props":1165,"children":1166},{},[1167],{"type":20,"tag":58,"props":1168,"children":1171},{"href":1169,"rel":1170},"https://www.vagrantup.com/docs/virtualbox/configuration.html#vboxmanage-customizations",[62],[1172],{"type":25,"value":1173},"11",{"type":20,"tag":116,"props":1175,"children":1177},{"code":1176,"language":249,"meta":7,"className":250,"style":7},"Command: [\"controlvm\", \"060a716b-1965-49e2-bc56-12beed5df716\", \"usbattach36fc9e60-c465-11cf-8056-444553540000\"]\n\nStderr: VBoxManage.exe: error: Machine '060a716b-1965-49e2-bc56-12beed5df716' is not currently running.\n",[1178],{"type":20,"tag":45,"props":1179,"children":1180},{"__ignoreMap":7},[1181,1189,1196],{"type":20,"tag":126,"props":1182,"children":1183},{"class":128,"line":129},[1184],{"type":20,"tag":126,"props":1185,"children":1186},{},[1187],{"type":25,"value":1188},"Command: [\"controlvm\", \"060a716b-1965-49e2-bc56-12beed5df716\", \"usbattach36fc9e60-c465-11cf-8056-444553540000\"]\n",{"type":20,"tag":126,"props":1190,"children":1191},{"class":128,"line":265},[1192],{"type":20,"tag":126,"props":1193,"children":1194},{"emptyLinePlaceholder":278},[1195],{"type":25,"value":281},{"type":20,"tag":126,"props":1197,"children":1198},{"class":128,"line":274},[1199],{"type":20,"tag":126,"props":1200,"children":1201},{},[1202],{"type":25,"value":1203},"Stderr: VBoxManage.exe: error: Machine '060a716b-1965-49e2-bc56-12beed5df716' is not currently running.\n",{"type":20,"tag":92,"props":1205,"children":1207},{"id":1206},"references",[1208],{"type":25,"value":1209},"References",{"type":20,"tag":1211,"props":1212,"children":1213},"ol",{},[1214,1224,1233,1242,1251,1260,1269,1278,1287,1296,1305],{"type":20,"tag":1215,"props":1216,"children":1217},"li",{},[1218],{"type":20,"tag":58,"props":1219,"children":1221},{"href":60,"rel":1220},[62],[1222],{"type":25,"value":1223},"Docker Desktop on Mac vs. Docker Toolbox",{"type":20,"tag":1215,"props":1225,"children":1226},{},[1227],{"type":20,"tag":58,"props":1228,"children":1230},{"href":70,"rel":1229},[62],[1231],{"type":25,"value":1232},"GitHub - HyperKit",{"type":20,"tag":1215,"props":1234,"children":1235},{},[1236],{"type":20,"tag":58,"props":1237,"children":1239},{"href":78,"rel":1238},[62],[1240],{"type":25,"value":1241},"GitHub - Docker for Mac - Issue #900",{"type":20,"tag":1215,"props":1243,"children":1244},{},[1245],{"type":20,"tag":58,"props":1246,"children":1248},{"href":108,"rel":1247},[62],[1249],{"type":25,"value":1250},"Vagrant Cloud - Arch Linux",{"type":20,"tag":1215,"props":1252,"children":1253},{},[1254],{"type":20,"tag":58,"props":1255,"children":1257},{"href":216,"rel":1256},[62],[1258],{"type":25,"value":1259},"Attaching USB Devices to VirtualBox Guests using VBoxManage",{"type":20,"tag":1215,"props":1261,"children":1262},{},[1263],{"type":20,"tag":58,"props":1264,"children":1266},{"href":224,"rel":1265},[62],[1267],{"type":25,"value":1268},"GitHub Gist - Vagrant USB Filter",{"type":20,"tag":1215,"props":1270,"children":1271},{},[1272],{"type":20,"tag":58,"props":1273,"children":1275},{"href":841,"rel":1274},[62],[1276],{"type":25,"value":1277},"Arch Linux Wiki - EFI System Partition - Format Partitions",{"type":20,"tag":1215,"props":1279,"children":1280},{},[1281],{"type":20,"tag":58,"props":1282,"children":1284},{"href":903,"rel":1283},[62],[1285],{"type":25,"value":1286},"Arch Linux Wiki - Arch Linux on USB - Installation Tweaks",{"type":20,"tag":1215,"props":1288,"children":1289},{},[1290],{"type":20,"tag":58,"props":1291,"children":1293},{"href":667,"rel":1292},[62],[1294],{"type":25,"value":1295},"Arch Linux Wiki - Installation Guide",{"type":20,"tag":1215,"props":1297,"children":1298},{},[1299],{"type":20,"tag":58,"props":1300,"children":1302},{"href":1115,"rel":1301},[62],[1303],{"type":25,"value":1304},"Arch Linux Wiki - GRUB - UEFI Systems",{"type":20,"tag":1215,"props":1306,"children":1307},{},[1308],{"type":20,"tag":58,"props":1309,"children":1311},{"href":1169,"rel":1310},[62],[1312],{"type":25,"value":1313},"Vagrant VBoxManage Customizations ",{"type":20,"tag":1315,"props":1316,"children":1317},"style",{},[1318],{"type":25,"value":1319},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":265,"depth":265,"links":1321},[1322,1323,1329,1330],{"id":94,"depth":265,"text":97},{"id":656,"depth":265,"text":659,"children":1324},[1325,1326,1327,1328],{"id":677,"depth":274,"text":680},{"id":827,"depth":274,"text":830},{"id":943,"depth":274,"text":946},{"id":1069,"depth":274,"text":1072},{"id":1140,"depth":265,"text":1143},{"id":1206,"depth":265,"text":1209},"markdown","content:articles:persistent-archlinux-usb.md","content","articles/persistent-archlinux-usb.md","articles/persistent-archlinux-usb","md",1731781853291] \ No newline at end of file +[{"data":1,"prerenderedAt":1337},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"draft":6,"tags":11,"categories":14,"excerpt":16,"body":26,"_type":1331,"_id":1332,"_source":1333,"_file":1334,"_stem":1335,"_extension":1336},"/articles/persistent-archlinux-usb","articles",false,"","Create a Persistent Arch Linux Bootable USB with Vagrant","When installing a linux distribution, it is common for the instructions to have\nthe user create a bootable USB, boot from the device, and proceed with the\ninstallation procedure from that live medium. However, this blog post will\noutline an alternative approach where a virtual machine created with Vagrant\nwill be used in favor of the live medium.","2020-01-09",[12,13],"vagrant","archlinux",[15],"linux",{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24],{"type":25,"value":9},"text",{"type":17,"children":27,"toc":1320},[28,32,39,85,91,98,115,152,165,190,246,354,367,388,654,660,674,681,705,825,831,846,885,908,941,947,1002,1067,1073,1081,1089,1097,1120,1128,1133,1138,1144,1174,1204,1210,1314],{"type":20,"tag":21,"props":29,"children":30},{},[31],{"type":25,"value":9},{"type":20,"tag":33,"props":34,"children":36},"h1",{"id":35},"preface",[37],{"type":25,"value":38},"Preface",{"type":20,"tag":21,"props":40,"children":41},{},[42,44,51,53,83],{"type":25,"value":43},"The original intention was to use Docker for this process -- leveraging the\n",{"type":20,"tag":45,"props":46,"children":48},"code",{"className":47},[],[49],{"type":25,"value":50},"--device",{"type":25,"value":52}," flag and mounting the target USB device in the Docker container,\nbut the underlying hypervisor in Docker Desktop for Mac does not support this.\n",{"type":20,"tag":54,"props":55,"children":56},"sup",{},[57,66,68,75,76],{"type":20,"tag":58,"props":59,"children":63},"a",{"href":60,"rel":61},"https://docs.docker.com/docker-for-mac/docker-toolbox/",[62],"nofollow",[64],{"type":25,"value":65},"1",{"type":25,"value":67}," ",{"type":20,"tag":58,"props":69,"children":72},{"href":70,"rel":71},"https://github.com/moby/hyperkit",[62],[73],{"type":25,"value":74},"2",{"type":25,"value":67},{"type":20,"tag":58,"props":77,"children":80},{"href":78,"rel":79},"https://github.com/docker/for-mac/issues/900",[62],[81],{"type":25,"value":82},"3",{"type":25,"value":84}," While there are workarounds using Docker\nMachine, Vagrant felt like the path of least resistance.",{"type":20,"tag":33,"props":86,"children":88},{"id":87},"instructions",[89],{"type":25,"value":90},"Instructions",{"type":20,"tag":92,"props":93,"children":95},"h2",{"id":94},"create-an-arch-linux-virtual-machine-with-vagrant",[96],{"type":25,"value":97},"Create an Arch Linux Virtual Machine with Vagrant",{"type":20,"tag":21,"props":99,"children":100},{},[101,103,113],{"type":25,"value":102},"Get the latest Arch Linux image ",{"type":20,"tag":54,"props":104,"children":105},{},[106],{"type":20,"tag":58,"props":107,"children":110},{"href":108,"rel":109},"https://app.vagrantup.com/archlinux/boxes/archlinux",[62],[111],{"type":25,"value":112},"4",{"type":25,"value":114}," from the Vagrant Cloud Box\nCatalog.",{"type":20,"tag":116,"props":117,"children":121},"pre",{"code":118,"language":119,"meta":7,"className":120,"style":7},"vagrant box add archlinux/archlinux\n","bash","language-bash shiki shiki-themes github-light",[122],{"type":20,"tag":45,"props":123,"children":124},{"__ignoreMap":7},[125],{"type":20,"tag":126,"props":127,"children":130},"span",{"class":128,"line":129},"line",1,[131,136,142,147],{"type":20,"tag":126,"props":132,"children":134},{"style":133},"--shiki-default:#6F42C1",[135],{"type":25,"value":12},{"type":20,"tag":126,"props":137,"children":139},{"style":138},"--shiki-default:#032F62",[140],{"type":25,"value":141}," box",{"type":20,"tag":126,"props":143,"children":144},{"style":138},[145],{"type":25,"value":146}," add",{"type":20,"tag":126,"props":148,"children":149},{"style":138},[150],{"type":25,"value":151}," archlinux/archlinux\n",{"type":20,"tag":21,"props":153,"children":154},{},[155,157,163],{"type":25,"value":156},"Determine the USB vendor information for the thumb-drive that we will\npass-through to the virtual machine. Using the ",{"type":20,"tag":45,"props":158,"children":160},{"className":159},[],[161],{"type":25,"value":162},"VBoxManage",{"type":25,"value":164}," utility that comes\nwith Virtual Box, list the devices, and make note of the Vendor and Product ID.",{"type":20,"tag":116,"props":166,"children":168},{"code":167,"language":119,"meta":7,"className":120,"style":7}," VBoxManage list usbhost\n",[169],{"type":20,"tag":45,"props":170,"children":171},{"__ignoreMap":7},[172],{"type":20,"tag":126,"props":173,"children":174},{"class":128,"line":129},[175,180,185],{"type":20,"tag":126,"props":176,"children":177},{"style":133},[178],{"type":25,"value":179}," VBoxManage",{"type":20,"tag":126,"props":181,"children":182},{"style":138},[183],{"type":25,"value":184}," list",{"type":20,"tag":126,"props":186,"children":187},{"style":138},[188],{"type":25,"value":189}," usbhost\n",{"type":20,"tag":21,"props":191,"children":192},{},[193,195,201,203,209,211,229,231,237,239,244],{"type":25,"value":194},"Create a ",{"type":20,"tag":45,"props":196,"children":198},{"className":197},[],[199],{"type":25,"value":200},"Vagrantfile",{"type":25,"value":202}," with ",{"type":20,"tag":45,"props":204,"children":206},{"className":205},[],[207],{"type":25,"value":208},"archlinx/archlinux",{"type":25,"value":210}," as the target box, and the USB\ndevice information that is passed through. ",{"type":20,"tag":54,"props":212,"children":213},{},[214,221,222],{"type":20,"tag":58,"props":215,"children":218},{"href":216,"rel":217},"https://antonyjepson.wordpress.com/2012/01/26/quickly-attaching-usb-devices-to-virtualbox-guests-using-vboxmanage/",[62],[219],{"type":25,"value":220},"5",{"type":25,"value":67},{"type":20,"tag":58,"props":223,"children":226},{"href":224,"rel":225},"https://gist.github.com/dscape/7d829c0c116ef419f963",[62],[227],{"type":25,"value":228},"6",{"type":25,"value":230}," Vagrant\noffers a handy customization parameter ",{"type":20,"tag":45,"props":232,"children":234},{"className":233},[],[235],{"type":25,"value":236},"vb.customize",{"type":25,"value":238}," that calls the\n",{"type":20,"tag":45,"props":240,"children":242},{"className":241},[],[243],{"type":25,"value":162},{"type":25,"value":245}," command under-the-hood, allowing one to enable the guest machine\nto access the host machine's USB devices.",{"type":20,"tag":116,"props":247,"children":251},{"code":248,"language":249,"meta":7,"className":250,"style":7},"# -*- mode: ruby -*-\n# vi: set ft=ruby :\n\nVagrant.configure(\"2\") do |config|\n config.vm.box = \"archlinux/archlinux\"\n config.vm.provider \"virtualbox\" do |vb|\n vb.name = \"archlinux\"\n vb.customize ['modifyvm', :id, '--usb', 'on']\n vb.customize ['usbfilter', 'add', '1', '--target', :id, '--name', 'SanDisk Ultra Fit', '--vendorid', '0x0781', '--productid', '0x5583']\n end\nend\n","txt","language-txt shiki shiki-themes github-light",[252],{"type":20,"tag":45,"props":253,"children":254},{"__ignoreMap":7},[255,263,272,282,291,300,309,318,327,336,345],{"type":20,"tag":126,"props":256,"children":257},{"class":128,"line":129},[258],{"type":20,"tag":126,"props":259,"children":260},{},[261],{"type":25,"value":262},"# -*- mode: ruby -*-\n",{"type":20,"tag":126,"props":264,"children":266},{"class":128,"line":265},2,[267],{"type":20,"tag":126,"props":268,"children":269},{},[270],{"type":25,"value":271},"# vi: set ft=ruby :\n",{"type":20,"tag":126,"props":273,"children":275},{"class":128,"line":274},3,[276],{"type":20,"tag":126,"props":277,"children":279},{"emptyLinePlaceholder":278},true,[280],{"type":25,"value":281},"\n",{"type":20,"tag":126,"props":283,"children":285},{"class":128,"line":284},4,[286],{"type":20,"tag":126,"props":287,"children":288},{},[289],{"type":25,"value":290},"Vagrant.configure(\"2\") do |config|\n",{"type":20,"tag":126,"props":292,"children":294},{"class":128,"line":293},5,[295],{"type":20,"tag":126,"props":296,"children":297},{},[298],{"type":25,"value":299}," config.vm.box = \"archlinux/archlinux\"\n",{"type":20,"tag":126,"props":301,"children":303},{"class":128,"line":302},6,[304],{"type":20,"tag":126,"props":305,"children":306},{},[307],{"type":25,"value":308}," config.vm.provider \"virtualbox\" do |vb|\n",{"type":20,"tag":126,"props":310,"children":312},{"class":128,"line":311},7,[313],{"type":20,"tag":126,"props":314,"children":315},{},[316],{"type":25,"value":317}," vb.name = \"archlinux\"\n",{"type":20,"tag":126,"props":319,"children":321},{"class":128,"line":320},8,[322],{"type":20,"tag":126,"props":323,"children":324},{},[325],{"type":25,"value":326}," vb.customize ['modifyvm', :id, '--usb', 'on']\n",{"type":20,"tag":126,"props":328,"children":330},{"class":128,"line":329},9,[331],{"type":20,"tag":126,"props":332,"children":333},{},[334],{"type":25,"value":335}," vb.customize ['usbfilter', 'add', '1', '--target', :id, '--name', 'SanDisk Ultra Fit', '--vendorid', '0x0781', '--productid', '0x5583']\n",{"type":20,"tag":126,"props":337,"children":339},{"class":128,"line":338},10,[340],{"type":20,"tag":126,"props":341,"children":342},{},[343],{"type":25,"value":344}," end\n",{"type":20,"tag":126,"props":346,"children":348},{"class":128,"line":347},11,[349],{"type":20,"tag":126,"props":350,"children":351},{},[352],{"type":25,"value":353},"end\n",{"type":20,"tag":21,"props":355,"children":356},{},[357,359,365],{"type":25,"value":358},"When virtual machine is brought up, the ",{"type":20,"tag":45,"props":360,"children":362},{"className":361},[],[363],{"type":25,"value":364},"usbfilter",{"type":25,"value":366}," is applied, and the guest\nis able to access to the host machine's USB device that was specified in the\nfilter.",{"type":20,"tag":21,"props":368,"children":369},{},[370,372,378,380,386],{"type":25,"value":371},"Start the machine, ",{"type":20,"tag":45,"props":373,"children":375},{"className":374},[],[376],{"type":25,"value":377},"ssh",{"type":25,"value":379}," into the guest, and list the devices to confirm that\nthe USB device is available (see: ",{"type":20,"tag":45,"props":381,"children":383},{"className":382},[],[384],{"type":25,"value":385},"/dev/sdb",{"type":25,"value":387},").",{"type":20,"tag":116,"props":389,"children":391},{"code":390,"language":119,"meta":7,"className":120,"style":7},"$ vagrant up\n$ vagrant ssh\n[vagrant@archlinux ~]$ lsblk\nNAME MAJ:MIN RM SIZE RO TYPE MOUNTPOINT\nsda 8:0 0 20G 0 disk\n├─sda1 8:1 0 1.9G 0 part [SWAP]\n└─sda2 8:2 0 18.1G 0 part /\nsdb 8:16 1 28.7G 0 disk\n└─sdb1 8:17 1 8G 0 part\n",[392],{"type":20,"tag":45,"props":393,"children":394},{"__ignoreMap":7},[395,413,429,449,487,521,557,592,623],{"type":20,"tag":126,"props":396,"children":397},{"class":128,"line":129},[398,403,408],{"type":20,"tag":126,"props":399,"children":400},{"style":133},[401],{"type":25,"value":402},"$",{"type":20,"tag":126,"props":404,"children":405},{"style":138},[406],{"type":25,"value":407}," vagrant",{"type":20,"tag":126,"props":409,"children":410},{"style":138},[411],{"type":25,"value":412}," up\n",{"type":20,"tag":126,"props":414,"children":415},{"class":128,"line":265},[416,420,424],{"type":20,"tag":126,"props":417,"children":418},{"style":133},[419],{"type":25,"value":402},{"type":20,"tag":126,"props":421,"children":422},{"style":138},[423],{"type":25,"value":407},{"type":20,"tag":126,"props":425,"children":426},{"style":138},[427],{"type":25,"value":428}," ssh\n",{"type":20,"tag":126,"props":430,"children":431},{"class":128,"line":274},[432,438,444],{"type":20,"tag":126,"props":433,"children":435},{"style":434},"--shiki-default:#24292E",[436],{"type":25,"value":437},"[vagrant@archlinux ",{"type":20,"tag":126,"props":439,"children":441},{"style":440},"--shiki-default:#D73A49",[442],{"type":25,"value":443},"~",{"type":20,"tag":126,"props":445,"children":446},{"style":434},[447],{"type":25,"value":448},"]$ lsblk\n",{"type":20,"tag":126,"props":450,"children":451},{"class":128,"line":284},[452,457,462,467,472,477,482],{"type":20,"tag":126,"props":453,"children":454},{"style":133},[455],{"type":25,"value":456},"NAME",{"type":20,"tag":126,"props":458,"children":459},{"style":138},[460],{"type":25,"value":461}," MAJ:MIN",{"type":20,"tag":126,"props":463,"children":464},{"style":138},[465],{"type":25,"value":466}," RM",{"type":20,"tag":126,"props":468,"children":469},{"style":138},[470],{"type":25,"value":471}," SIZE",{"type":20,"tag":126,"props":473,"children":474},{"style":138},[475],{"type":25,"value":476}," RO",{"type":20,"tag":126,"props":478,"children":479},{"style":138},[480],{"type":25,"value":481}," TYPE",{"type":20,"tag":126,"props":483,"children":484},{"style":138},[485],{"type":25,"value":486}," MOUNTPOINT\n",{"type":20,"tag":126,"props":488,"children":489},{"class":128,"line":293},[490,495,500,506,511,516],{"type":20,"tag":126,"props":491,"children":492},{"style":133},[493],{"type":25,"value":494},"sda",{"type":20,"tag":126,"props":496,"children":497},{"style":138},[498],{"type":25,"value":499}," 8:0",{"type":20,"tag":126,"props":501,"children":503},{"style":502},"--shiki-default:#005CC5",[504],{"type":25,"value":505}," 0",{"type":20,"tag":126,"props":507,"children":508},{"style":138},[509],{"type":25,"value":510}," 20G",{"type":20,"tag":126,"props":512,"children":513},{"style":502},[514],{"type":25,"value":515}," 0",{"type":20,"tag":126,"props":517,"children":518},{"style":138},[519],{"type":25,"value":520}," disk\n",{"type":20,"tag":126,"props":522,"children":523},{"class":128,"line":302},[524,529,534,538,543,547,552],{"type":20,"tag":126,"props":525,"children":526},{"style":133},[527],{"type":25,"value":528},"├─sda1",{"type":20,"tag":126,"props":530,"children":531},{"style":138},[532],{"type":25,"value":533}," 8:1",{"type":20,"tag":126,"props":535,"children":536},{"style":502},[537],{"type":25,"value":505},{"type":20,"tag":126,"props":539,"children":540},{"style":138},[541],{"type":25,"value":542}," 1.9G",{"type":20,"tag":126,"props":544,"children":545},{"style":502},[546],{"type":25,"value":515},{"type":20,"tag":126,"props":548,"children":549},{"style":138},[550],{"type":25,"value":551}," part",{"type":20,"tag":126,"props":553,"children":554},{"style":434},[555],{"type":25,"value":556}," [SWAP]\n",{"type":20,"tag":126,"props":558,"children":559},{"class":128,"line":311},[560,565,570,574,579,583,587],{"type":20,"tag":126,"props":561,"children":562},{"style":133},[563],{"type":25,"value":564},"└─sda2",{"type":20,"tag":126,"props":566,"children":567},{"style":138},[568],{"type":25,"value":569}," 8:2",{"type":20,"tag":126,"props":571,"children":572},{"style":502},[573],{"type":25,"value":505},{"type":20,"tag":126,"props":575,"children":576},{"style":138},[577],{"type":25,"value":578}," 18.1G",{"type":20,"tag":126,"props":580,"children":581},{"style":502},[582],{"type":25,"value":515},{"type":20,"tag":126,"props":584,"children":585},{"style":138},[586],{"type":25,"value":551},{"type":20,"tag":126,"props":588,"children":589},{"style":138},[590],{"type":25,"value":591}," /\n",{"type":20,"tag":126,"props":593,"children":594},{"class":128,"line":320},[595,600,605,610,615,619],{"type":20,"tag":126,"props":596,"children":597},{"style":133},[598],{"type":25,"value":599},"sdb",{"type":20,"tag":126,"props":601,"children":602},{"style":138},[603],{"type":25,"value":604}," 8:16",{"type":20,"tag":126,"props":606,"children":607},{"style":502},[608],{"type":25,"value":609}," 1",{"type":20,"tag":126,"props":611,"children":612},{"style":138},[613],{"type":25,"value":614}," 28.7G",{"type":20,"tag":126,"props":616,"children":617},{"style":502},[618],{"type":25,"value":515},{"type":20,"tag":126,"props":620,"children":621},{"style":138},[622],{"type":25,"value":520},{"type":20,"tag":126,"props":624,"children":625},{"class":128,"line":329},[626,631,636,640,645,649],{"type":20,"tag":126,"props":627,"children":628},{"style":133},[629],{"type":25,"value":630},"└─sdb1",{"type":20,"tag":126,"props":632,"children":633},{"style":138},[634],{"type":25,"value":635}," 8:17",{"type":20,"tag":126,"props":637,"children":638},{"style":502},[639],{"type":25,"value":609},{"type":20,"tag":126,"props":641,"children":642},{"style":138},[643],{"type":25,"value":644}," 8G",{"type":20,"tag":126,"props":646,"children":647},{"style":502},[648],{"type":25,"value":515},{"type":20,"tag":126,"props":650,"children":651},{"style":138},[652],{"type":25,"value":653}," part\n",{"type":20,"tag":92,"props":655,"children":657},{"id":656},"install-arch-linux-on-the-usb-drive",[658],{"type":25,"value":659},"Install Arch Linux on the USB Drive",{"type":20,"tag":21,"props":661,"children":662},{},[663,665,672],{"type":25,"value":664},"The ",{"type":20,"tag":58,"props":666,"children":669},{"href":667,"rel":668},"https://wiki.archlinux.org/index.php/Installation_guide",[62],[670],{"type":25,"value":671},"Arch Linux Installation Guide",{"type":25,"value":673}," outlines the installation procedure in\ngreat detail -- the following steps follow this closely with a few alteration\ndue to installing onto removable media.",{"type":20,"tag":675,"props":676,"children":678},"h3",{"id":677},"partition-the-disk-uefi-with-gpt",[679],{"type":25,"value":680},"Partition the Disk (UEFI with GPT)",{"type":20,"tag":116,"props":682,"children":684},{"code":683,"language":119,"meta":7,"className":120,"style":7},"[root@archlinux ~]# fdisk /dev/sdb\n",[685],{"type":20,"tag":45,"props":686,"children":687},{"__ignoreMap":7},[688],{"type":20,"tag":126,"props":689,"children":690},{"class":128,"line":129},[691,696,700],{"type":20,"tag":126,"props":692,"children":693},{"style":434},[694],{"type":25,"value":695},"[root@archlinux ",{"type":20,"tag":126,"props":697,"children":698},{"style":440},[699],{"type":25,"value":443},{"type":20,"tag":126,"props":701,"children":702},{"style":434},[703],{"type":25,"value":704},"]# fdisk /dev/sdb\n",{"type":20,"tag":116,"props":706,"children":708},{"code":707,"language":249,"meta":7,"className":250,"style":7},"Command (m for help): p\nDisk /dev/sdb: 28.66 GiB, 30752636928 bytes, 60063744 sectors\nDisk model: Ultra Fit\nUnits: sectors of 1 * 512 = 512 bytes\nSector size (logical/physical): 512 bytes / 512 bytes\nI/O size (minimum/optimal): 512 bytes / 512 bytes\nDisklabel type: gpt\nDisk identifier: E1D6C445-1B79-AB4D-A442-FA4AD6DF4ECC\n\nDevice Start End Sectors Size Type\n/dev/sdb1 2048 1050623 1048576 512M EFI System\n/dev/sdb2 1050624 60063710 59013087 28.1G Linux filesystem\n\nFilesystem/RAID signature on partition 1 will be wiped.\n",[709],{"type":20,"tag":45,"props":710,"children":711},{"__ignoreMap":7},[712,720,728,736,744,752,760,768,776,783,791,799,808,816],{"type":20,"tag":126,"props":713,"children":714},{"class":128,"line":129},[715],{"type":20,"tag":126,"props":716,"children":717},{},[718],{"type":25,"value":719},"Command (m for help): p\n",{"type":20,"tag":126,"props":721,"children":722},{"class":128,"line":265},[723],{"type":20,"tag":126,"props":724,"children":725},{},[726],{"type":25,"value":727},"Disk /dev/sdb: 28.66 GiB, 30752636928 bytes, 60063744 sectors\n",{"type":20,"tag":126,"props":729,"children":730},{"class":128,"line":274},[731],{"type":20,"tag":126,"props":732,"children":733},{},[734],{"type":25,"value":735},"Disk model: Ultra Fit\n",{"type":20,"tag":126,"props":737,"children":738},{"class":128,"line":284},[739],{"type":20,"tag":126,"props":740,"children":741},{},[742],{"type":25,"value":743},"Units: sectors of 1 * 512 = 512 bytes\n",{"type":20,"tag":126,"props":745,"children":746},{"class":128,"line":293},[747],{"type":20,"tag":126,"props":748,"children":749},{},[750],{"type":25,"value":751},"Sector size (logical/physical): 512 bytes / 512 bytes\n",{"type":20,"tag":126,"props":753,"children":754},{"class":128,"line":302},[755],{"type":20,"tag":126,"props":756,"children":757},{},[758],{"type":25,"value":759},"I/O size (minimum/optimal): 512 bytes / 512 bytes\n",{"type":20,"tag":126,"props":761,"children":762},{"class":128,"line":311},[763],{"type":20,"tag":126,"props":764,"children":765},{},[766],{"type":25,"value":767},"Disklabel type: gpt\n",{"type":20,"tag":126,"props":769,"children":770},{"class":128,"line":320},[771],{"type":20,"tag":126,"props":772,"children":773},{},[774],{"type":25,"value":775},"Disk identifier: E1D6C445-1B79-AB4D-A442-FA4AD6DF4ECC\n",{"type":20,"tag":126,"props":777,"children":778},{"class":128,"line":329},[779],{"type":20,"tag":126,"props":780,"children":781},{"emptyLinePlaceholder":278},[782],{"type":25,"value":281},{"type":20,"tag":126,"props":784,"children":785},{"class":128,"line":338},[786],{"type":20,"tag":126,"props":787,"children":788},{},[789],{"type":25,"value":790},"Device Start End Sectors Size Type\n",{"type":20,"tag":126,"props":792,"children":793},{"class":128,"line":347},[794],{"type":20,"tag":126,"props":795,"children":796},{},[797],{"type":25,"value":798},"/dev/sdb1 2048 1050623 1048576 512M EFI System\n",{"type":20,"tag":126,"props":800,"children":802},{"class":128,"line":801},12,[803],{"type":20,"tag":126,"props":804,"children":805},{},[806],{"type":25,"value":807},"/dev/sdb2 1050624 60063710 59013087 28.1G Linux filesystem\n",{"type":20,"tag":126,"props":809,"children":811},{"class":128,"line":810},13,[812],{"type":20,"tag":126,"props":813,"children":814},{"emptyLinePlaceholder":278},[815],{"type":25,"value":281},{"type":20,"tag":126,"props":817,"children":819},{"class":128,"line":818},14,[820],{"type":20,"tag":126,"props":821,"children":822},{},[823],{"type":25,"value":824},"Filesystem/RAID signature on partition 1 will be wiped.\n",{"type":20,"tag":675,"props":826,"children":828},{"id":827},"format-the-partitions",[829],{"type":25,"value":830},"Format the Partitions",{"type":20,"tag":21,"props":832,"children":833},{},[834,836],{"type":25,"value":835},"The UEFI specification mandates support for FAT file-systems, and FAT32 is\nrecommended for removable media. ",{"type":20,"tag":54,"props":837,"children":838},{},[839],{"type":20,"tag":58,"props":840,"children":843},{"href":841,"rel":842},"https://wiki.archlinux.org/index.php/EFI_system_partition#Format_the_partition",[62],[844],{"type":25,"value":845},"7",{"type":20,"tag":116,"props":847,"children":849},{"code":848,"language":119,"meta":7,"className":120,"style":7},"[root@archlinux ~]# pacman -Sy dosfstools\n[root@archlinux ~]# mkfs.fat -F32 /dev/sdb1\n",[850],{"type":20,"tag":45,"props":851,"children":852},{"__ignoreMap":7},[853,869],{"type":20,"tag":126,"props":854,"children":855},{"class":128,"line":129},[856,860,864],{"type":20,"tag":126,"props":857,"children":858},{"style":434},[859],{"type":25,"value":695},{"type":20,"tag":126,"props":861,"children":862},{"style":440},[863],{"type":25,"value":443},{"type":20,"tag":126,"props":865,"children":866},{"style":434},[867],{"type":25,"value":868},"]# pacman -Sy dosfstools\n",{"type":20,"tag":126,"props":870,"children":871},{"class":128,"line":265},[872,876,880],{"type":20,"tag":126,"props":873,"children":874},{"style":434},[875],{"type":25,"value":695},{"type":20,"tag":126,"props":877,"children":878},{"style":440},[879],{"type":25,"value":443},{"type":20,"tag":126,"props":881,"children":882},{"style":434},[883],{"type":25,"value":884},"]# mkfs.fat -F32 /dev/sdb1\n",{"type":20,"tag":21,"props":886,"children":887},{},[888,890,896,898],{"type":25,"value":889},"As for the root partition, it is recommended to use ",{"type":20,"tag":45,"props":891,"children":893},{"className":892},[],[894],{"type":25,"value":895},"ext4",{"type":25,"value":897}," without a journal to\nreduce the reads and writes to the file-system as this is detrimental to the\nflash-based USB drive. ",{"type":20,"tag":54,"props":899,"children":900},{},[901],{"type":20,"tag":58,"props":902,"children":905},{"href":903,"rel":904},"https://wiki.archlinux.org/index.php/Install_Arch_Linux_on_a_USB_key#Installation_tweaks",[62],[906],{"type":25,"value":907},"8",{"type":20,"tag":116,"props":909,"children":911},{"code":910,"language":119,"meta":7,"className":120,"style":7},"[root@archlinux ~]# mkfs.ext4 -O \"^has_journal\" /dev/sdb2\n",[912],{"type":20,"tag":45,"props":913,"children":914},{"__ignoreMap":7},[915],{"type":20,"tag":126,"props":916,"children":917},{"class":128,"line":129},[918,922,926,931,936],{"type":20,"tag":126,"props":919,"children":920},{"style":434},[921],{"type":25,"value":695},{"type":20,"tag":126,"props":923,"children":924},{"style":440},[925],{"type":25,"value":443},{"type":20,"tag":126,"props":927,"children":928},{"style":434},[929],{"type":25,"value":930},"]# mkfs.ext4 -O ",{"type":20,"tag":126,"props":932,"children":933},{"style":138},[934],{"type":25,"value":935},"\"^has_journal\"",{"type":20,"tag":126,"props":937,"children":938},{"style":434},[939],{"type":25,"value":940}," /dev/sdb2\n",{"type":20,"tag":675,"props":942,"children":944},{"id":943},"mount-the-partitions-and-bootstrap-the-environment",[945],{"type":25,"value":946},"Mount the Partitions and Bootstrap the Environment",{"type":20,"tag":116,"props":948,"children":950},{"code":949,"language":119,"meta":7,"className":120,"style":7},"[root@archlinux ~]# mount /dev/sdb2 /mnt\n[root@archlinux ~]# mkdir -p /mnt/boot/efi\n[root@archlinux ~]# mount /dev/sdb1 /mnt/boot/efi\n",[951],{"type":20,"tag":45,"props":952,"children":953},{"__ignoreMap":7},[954,970,986],{"type":20,"tag":126,"props":955,"children":956},{"class":128,"line":129},[957,961,965],{"type":20,"tag":126,"props":958,"children":959},{"style":434},[960],{"type":25,"value":695},{"type":20,"tag":126,"props":962,"children":963},{"style":440},[964],{"type":25,"value":443},{"type":20,"tag":126,"props":966,"children":967},{"style":434},[968],{"type":25,"value":969},"]# mount /dev/sdb2 /mnt\n",{"type":20,"tag":126,"props":971,"children":972},{"class":128,"line":265},[973,977,981],{"type":20,"tag":126,"props":974,"children":975},{"style":434},[976],{"type":25,"value":695},{"type":20,"tag":126,"props":978,"children":979},{"style":440},[980],{"type":25,"value":443},{"type":20,"tag":126,"props":982,"children":983},{"style":434},[984],{"type":25,"value":985},"]# mkdir -p /mnt/boot/efi\n",{"type":20,"tag":126,"props":987,"children":988},{"class":128,"line":274},[989,993,997],{"type":20,"tag":126,"props":990,"children":991},{"style":434},[992],{"type":25,"value":695},{"type":20,"tag":126,"props":994,"children":995},{"style":440},[996],{"type":25,"value":443},{"type":20,"tag":126,"props":998,"children":999},{"style":434},[1000],{"type":25,"value":1001},"]# mount /dev/sdb1 /mnt/boot/efi\n",{"type":20,"tag":116,"props":1003,"children":1005},{"code":1004,"language":119,"meta":7,"className":120,"style":7},"[root@archlinux ~]# pacman -S arch-install-scripts\n[root@archlinux ~]# pacstrap /mnt base linux linux-firmware\n[root@archlinux ~]# genfstab -U /mnt >> /mnt/etc/fstab\n",[1006],{"type":20,"tag":45,"props":1007,"children":1008},{"__ignoreMap":7},[1009,1025,1041],{"type":20,"tag":126,"props":1010,"children":1011},{"class":128,"line":129},[1012,1016,1020],{"type":20,"tag":126,"props":1013,"children":1014},{"style":434},[1015],{"type":25,"value":695},{"type":20,"tag":126,"props":1017,"children":1018},{"style":440},[1019],{"type":25,"value":443},{"type":20,"tag":126,"props":1021,"children":1022},{"style":434},[1023],{"type":25,"value":1024},"]# pacman -S arch-install-scripts\n",{"type":20,"tag":126,"props":1026,"children":1027},{"class":128,"line":265},[1028,1032,1036],{"type":20,"tag":126,"props":1029,"children":1030},{"style":434},[1031],{"type":25,"value":695},{"type":20,"tag":126,"props":1033,"children":1034},{"style":440},[1035],{"type":25,"value":443},{"type":20,"tag":126,"props":1037,"children":1038},{"style":434},[1039],{"type":25,"value":1040},"]# pacstrap /mnt base linux linux-firmware\n",{"type":20,"tag":126,"props":1042,"children":1043},{"class":128,"line":274},[1044,1048,1052,1057,1062],{"type":20,"tag":126,"props":1045,"children":1046},{"style":434},[1047],{"type":25,"value":695},{"type":20,"tag":126,"props":1049,"children":1050},{"style":440},[1051],{"type":25,"value":443},{"type":20,"tag":126,"props":1053,"children":1054},{"style":434},[1055],{"type":25,"value":1056},"]# genfstab -U /mnt ",{"type":20,"tag":126,"props":1058,"children":1059},{"style":440},[1060],{"type":25,"value":1061},">>",{"type":20,"tag":126,"props":1063,"children":1064},{"style":434},[1065],{"type":25,"value":1066}," /mnt/etc/fstab\n",{"type":20,"tag":675,"props":1068,"children":1070},{"id":1069},"configure-the-new-environment",[1071],{"type":25,"value":1072},"Configure the New Environment",{"type":20,"tag":116,"props":1074,"children":1076},{"code":1075},"[root@archlinux ~]# arch-chroot /mnt\n",[1077],{"type":20,"tag":45,"props":1078,"children":1079},{"__ignoreMap":7},[1080],{"type":25,"value":1075},{"type":20,"tag":116,"props":1082,"children":1084},{"code":1083},"[root@archlinux /]# ln -sf /usr/share/zoneinfo/US/Eastern /etc/localtime\n[root@archlinux /]# hwclock --systohc\n[root@archlinux /]# sed -i 's/#en_US.UTF-8/en_US.UTF-8/' /etc/locale.gen\n[root@archlinux /]# locale-gen\n[root@archlinux /]# echo \"LANG=en_US.UTF-8\" > /etc/locale.conf\n",[1085],{"type":20,"tag":45,"props":1086,"children":1087},{"__ignoreMap":7},[1088],{"type":25,"value":1083},{"type":20,"tag":116,"props":1090,"children":1092},{"code":1091},"[root@archlinux /]# hostnamectl set-hostname usb\n[root@archlinux /]# echo \"127.0.0.1 localhost\" >> /etc/hosts\n[root@archlinux /]# echo \"::1 localhost\" >> /etc/hosts\n",[1093],{"type":20,"tag":45,"props":1094,"children":1095},{"__ignoreMap":7},[1096],{"type":25,"value":1091},{"type":20,"tag":21,"props":1098,"children":1099},{},[1100,1102,1108,1110],{"type":25,"value":1101},"Note, one difference here from a standard installation is that the\n",{"type":20,"tag":45,"props":1103,"children":1105},{"className":1104},[],[1106],{"type":25,"value":1107},"--removable",{"type":25,"value":1109}," flag is specified when installing the GRUB bootloader.\n",{"type":20,"tag":54,"props":1111,"children":1112},{},[1113],{"type":20,"tag":58,"props":1114,"children":1117},{"href":1115,"rel":1116},"https://wiki.archlinux.org/index.php/GRUB#UEFI_systems",[62],[1118],{"type":25,"value":1119},"10",{"type":20,"tag":116,"props":1121,"children":1123},{"code":1122},"[root@archlinux /]# pacman -S grub\n[root@archlinux /]# grub-install --target=x86_64-efi --efi-directory=/boot/efi --removable --recheck\n[root@usb /]# grub-mkconfig -o /boot/grub/grub.cfg\n",[1124],{"type":20,"tag":45,"props":1125,"children":1126},{"__ignoreMap":7},[1127],{"type":25,"value":1122},{"type":20,"tag":21,"props":1129,"children":1130},{},[1131],{"type":25,"value":1132},"Shutdown the virtual machine, restart the host machine, and boot the newly\ncreated Arch Linux thumb-drive!",{"type":20,"tag":21,"props":1134,"children":1135},{},[1136],{"type":25,"value":1137},"🎉",{"type":20,"tag":92,"props":1139,"children":1141},{"id":1140},"side-note",[1142],{"type":25,"value":1143},"Side-note",{"type":20,"tag":21,"props":1145,"children":1146},{},[1147,1149,1155,1157,1162,1164],{"type":25,"value":1148},"It was attempted to use the ",{"type":20,"tag":45,"props":1150,"children":1152},{"className":1151},[],[1153],{"type":25,"value":1154},"controlvm usbattach",{"type":25,"value":1156}," command to pass the USB\ndevice to the guest machine, but this did not work as it expects the virtual\nmachine to already be running, and the ",{"type":20,"tag":45,"props":1158,"children":1160},{"className":1159},[],[1161],{"type":25,"value":236},{"type":25,"value":1163}," option runs prior to\nbooting the machine. ",{"type":20,"tag":54,"props":1165,"children":1166},{},[1167],{"type":20,"tag":58,"props":1168,"children":1171},{"href":1169,"rel":1170},"https://www.vagrantup.com/docs/virtualbox/configuration.html#vboxmanage-customizations",[62],[1172],{"type":25,"value":1173},"11",{"type":20,"tag":116,"props":1175,"children":1177},{"code":1176,"language":249,"meta":7,"className":250,"style":7},"Command: [\"controlvm\", \"060a716b-1965-49e2-bc56-12beed5df716\", \"usbattach36fc9e60-c465-11cf-8056-444553540000\"]\n\nStderr: VBoxManage.exe: error: Machine '060a716b-1965-49e2-bc56-12beed5df716' is not currently running.\n",[1178],{"type":20,"tag":45,"props":1179,"children":1180},{"__ignoreMap":7},[1181,1189,1196],{"type":20,"tag":126,"props":1182,"children":1183},{"class":128,"line":129},[1184],{"type":20,"tag":126,"props":1185,"children":1186},{},[1187],{"type":25,"value":1188},"Command: [\"controlvm\", \"060a716b-1965-49e2-bc56-12beed5df716\", \"usbattach36fc9e60-c465-11cf-8056-444553540000\"]\n",{"type":20,"tag":126,"props":1190,"children":1191},{"class":128,"line":265},[1192],{"type":20,"tag":126,"props":1193,"children":1194},{"emptyLinePlaceholder":278},[1195],{"type":25,"value":281},{"type":20,"tag":126,"props":1197,"children":1198},{"class":128,"line":274},[1199],{"type":20,"tag":126,"props":1200,"children":1201},{},[1202],{"type":25,"value":1203},"Stderr: VBoxManage.exe: error: Machine '060a716b-1965-49e2-bc56-12beed5df716' is not currently running.\n",{"type":20,"tag":92,"props":1205,"children":1207},{"id":1206},"references",[1208],{"type":25,"value":1209},"References",{"type":20,"tag":1211,"props":1212,"children":1213},"ol",{},[1214,1224,1233,1242,1251,1260,1269,1278,1287,1296,1305],{"type":20,"tag":1215,"props":1216,"children":1217},"li",{},[1218],{"type":20,"tag":58,"props":1219,"children":1221},{"href":60,"rel":1220},[62],[1222],{"type":25,"value":1223},"Docker Desktop on Mac vs. Docker Toolbox",{"type":20,"tag":1215,"props":1225,"children":1226},{},[1227],{"type":20,"tag":58,"props":1228,"children":1230},{"href":70,"rel":1229},[62],[1231],{"type":25,"value":1232},"GitHub - HyperKit",{"type":20,"tag":1215,"props":1234,"children":1235},{},[1236],{"type":20,"tag":58,"props":1237,"children":1239},{"href":78,"rel":1238},[62],[1240],{"type":25,"value":1241},"GitHub - Docker for Mac - Issue #900",{"type":20,"tag":1215,"props":1243,"children":1244},{},[1245],{"type":20,"tag":58,"props":1246,"children":1248},{"href":108,"rel":1247},[62],[1249],{"type":25,"value":1250},"Vagrant Cloud - Arch Linux",{"type":20,"tag":1215,"props":1252,"children":1253},{},[1254],{"type":20,"tag":58,"props":1255,"children":1257},{"href":216,"rel":1256},[62],[1258],{"type":25,"value":1259},"Attaching USB Devices to VirtualBox Guests using VBoxManage",{"type":20,"tag":1215,"props":1261,"children":1262},{},[1263],{"type":20,"tag":58,"props":1264,"children":1266},{"href":224,"rel":1265},[62],[1267],{"type":25,"value":1268},"GitHub Gist - Vagrant USB Filter",{"type":20,"tag":1215,"props":1270,"children":1271},{},[1272],{"type":20,"tag":58,"props":1273,"children":1275},{"href":841,"rel":1274},[62],[1276],{"type":25,"value":1277},"Arch Linux Wiki - EFI System Partition - Format Partitions",{"type":20,"tag":1215,"props":1279,"children":1280},{},[1281],{"type":20,"tag":58,"props":1282,"children":1284},{"href":903,"rel":1283},[62],[1285],{"type":25,"value":1286},"Arch Linux Wiki - Arch Linux on USB - Installation Tweaks",{"type":20,"tag":1215,"props":1288,"children":1289},{},[1290],{"type":20,"tag":58,"props":1291,"children":1293},{"href":667,"rel":1292},[62],[1294],{"type":25,"value":1295},"Arch Linux Wiki - Installation Guide",{"type":20,"tag":1215,"props":1297,"children":1298},{},[1299],{"type":20,"tag":58,"props":1300,"children":1302},{"href":1115,"rel":1301},[62],[1303],{"type":25,"value":1304},"Arch Linux Wiki - GRUB - UEFI Systems",{"type":20,"tag":1215,"props":1306,"children":1307},{},[1308],{"type":20,"tag":58,"props":1309,"children":1311},{"href":1169,"rel":1310},[62],[1312],{"type":25,"value":1313},"Vagrant VBoxManage Customizations ",{"type":20,"tag":1315,"props":1316,"children":1317},"style",{},[1318],{"type":25,"value":1319},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":265,"depth":265,"links":1321},[1322,1323,1329,1330],{"id":94,"depth":265,"text":97},{"id":656,"depth":265,"text":659,"children":1324},[1325,1326,1327,1328],{"id":677,"depth":274,"text":680},{"id":827,"depth":274,"text":830},{"id":943,"depth":274,"text":946},{"id":1069,"depth":274,"text":1072},{"id":1140,"depth":265,"text":1143},{"id":1206,"depth":265,"text":1209},"markdown","content:articles:persistent-archlinux-usb.md","content","articles/persistent-archlinux-usb.md","articles/persistent-archlinux-usb","md",1731782033546] \ No newline at end of file diff --git a/articles/persistent-archlinux-usb/index.html b/articles/persistent-archlinux-usb/index.html index b97f9ced..c92d2659 100644 --- a/articles/persistent-archlinux-usb/index.html +++ b/articles/persistent-archlinux-usb/index.html @@ -5,36 +5,36 @@ - - - - - + + + + + - + - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + -

        Create a Persistent Arch Linux Bootable USB with Vagrant

        When installing a linux distribution, it is common for the instructions to have +

        Create a Persistent Arch Linux Bootable USB with Vagrant

        When installing a linux distribution, it is common for the instructions to have the user create a bootable USB, boot from the device, and proceed with the installation procedure from that live medium. However, this blog post will outline an alternative approach where a virtual machine created with Vagrant @@ -124,5 +124,5 @@ booting the machine. 11

        Command: ["controlvm", "060a716b-1965-49e2-bc56-12beed5df716", "usbattach36fc9e60-c465-11cf-8056-444553540000"]
         
         Stderr: VBoxManage.exe: error: Machine '060a716b-1965-49e2-bc56-12beed5df716' is not currently running.
        -

        References

        1. Docker Desktop on Mac vs. Docker Toolbox
        2. GitHub - HyperKit
        3. GitHub - Docker for Mac - Issue #900
        4. Vagrant Cloud - Arch Linux
        5. Attaching USB Devices to VirtualBox Guests using VBoxManage
        6. GitHub Gist - Vagrant USB Filter
        7. Arch Linux Wiki - EFI System Partition - Format Partitions
        8. Arch Linux Wiki - Arch Linux on USB - Installation Tweaks
        9. Arch Linux Wiki - Installation Guide
        10. Arch Linux Wiki - GRUB - UEFI Systems
        11. Vagrant VBoxManage Customizations
        - \ No newline at end of file +

        References

        1. Docker Desktop on Mac vs. Docker Toolbox
        2. GitHub - HyperKit
        3. GitHub - Docker for Mac - Issue #900
        4. Vagrant Cloud - Arch Linux
        5. Attaching USB Devices to VirtualBox Guests using VBoxManage
        6. GitHub Gist - Vagrant USB Filter
        7. Arch Linux Wiki - EFI System Partition - Format Partitions
        8. Arch Linux Wiki - Arch Linux on USB - Installation Tweaks
        9. Arch Linux Wiki - Installation Guide
        10. Arch Linux Wiki - GRUB - UEFI Systems
        11. Vagrant VBoxManage Customizations
        + \ No newline at end of file diff --git a/articles/podcast-transcription-whispercpp/_payload.json b/articles/podcast-transcription-whispercpp/_payload.json index 10963cf4..6e390b7e 100644 --- a/articles/podcast-transcription-whispercpp/_payload.json +++ b/articles/podcast-transcription-whispercpp/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":750},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"draft":6,"date":10,"tags":11,"categories":14,"excerpt":16,"body":55,"_type":744,"_id":745,"_source":746,"_file":747,"_stem":748,"_extension":749},"/articles/podcast-transcription-whispercpp","articles",false,"","Easily Transcribe Podcasts with Whisper.cpp","If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive whisper.cpp project. This high-performance fork of OpenAI's Whisper can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the Alter Everything podcast.","2024-01-08",[12,13],"whisper.cpp","ml",[15],"programming",{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24,27,35,37,44,46,53],{"type":25,"value":26},"text","If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive ",{"type":20,"tag":28,"props":29,"children":33},"a",{"href":30,"rel":31},"https://github.com/ggerganov/whisper.cpp",[32],"nofollow",[34],{"type":25,"value":12},{"type":25,"value":36}," project. This high-performance fork of ",{"type":20,"tag":28,"props":38,"children":41},{"href":39,"rel":40},"https://github.com/openai/whisper",[32],[42],{"type":25,"value":43},"OpenAI's Whisper",{"type":25,"value":45}," can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the ",{"type":20,"tag":28,"props":47,"children":50},{"href":48,"rel":49},"https://podcasts.apple.com/us/podcast/alter-everything/id1356137854",[32],[51],{"type":25,"value":52},"Alter Everything",{"type":25,"value":54}," podcast.",{"type":17,"children":56,"toc":739},[57,79,86,130,222,234,399,412,424,550,555,720,725,733],{"type":20,"tag":21,"props":58,"children":59},{},[60,61,66,67,72,73,78],{"type":25,"value":26},{"type":20,"tag":28,"props":62,"children":64},{"href":30,"rel":63},[32],[65],{"type":25,"value":12},{"type":25,"value":36},{"type":20,"tag":28,"props":68,"children":70},{"href":39,"rel":69},[32],[71],{"type":25,"value":43},{"type":25,"value":45},{"type":20,"tag":28,"props":74,"children":76},{"href":48,"rel":75},[32],[77],{"type":25,"value":52},{"type":25,"value":54},{"type":20,"tag":80,"props":81,"children":83},"h2",{"id":82},"obtain-audio-files",[84],{"type":25,"value":85},"Obtain Audio File(s)",{"type":20,"tag":21,"props":87,"children":88},{},[89,91,98,100,106,108,113,115,120,122,128],{"type":25,"value":90},"First, let's get the ",{"type":20,"tag":92,"props":93,"children":95},"code",{"className":94},[],[96],{"type":25,"value":97},"wav",{"type":25,"value":99}," file from YouTube using the ",{"type":20,"tag":92,"props":101,"children":103},{"className":102},[],[104],{"type":25,"value":105},"youtube-dl",{"type":25,"value":107}," utility. It should be noted that ",{"type":20,"tag":92,"props":109,"children":111},{"className":110},[],[112],{"type":25,"value":12},{"type":25,"value":114}," expects ",{"type":20,"tag":92,"props":116,"children":118},{"className":117},[],[119],{"type":25,"value":97},{"type":25,"value":121}," filetypes, and this utility defaults to ",{"type":20,"tag":92,"props":123,"children":125},{"className":124},[],[126],{"type":25,"value":127},"mp3",{"type":25,"value":129},".",{"type":20,"tag":131,"props":132,"children":136},"pre",{"code":133,"language":134,"meta":7,"className":135,"style":7}," $ youtube-dl \\\n --extract-audio \\\n --audio-format wav \\\n --output podcast.wav \\\n \"https://www.youtube.com/watch?v=CoUN690wSYQ\"\n","sh","language-sh shiki shiki-themes github-light",[137],{"type":20,"tag":92,"props":138,"children":139},{"__ignoreMap":7},[140,164,177,195,213],{"type":20,"tag":141,"props":142,"children":145},"span",{"class":143,"line":144},"line",1,[146,152,158],{"type":20,"tag":141,"props":147,"children":149},{"style":148},"--shiki-default:#6F42C1",[150],{"type":25,"value":151}," $",{"type":20,"tag":141,"props":153,"children":155},{"style":154},"--shiki-default:#032F62",[156],{"type":25,"value":157}," youtube-dl",{"type":20,"tag":141,"props":159,"children":161},{"style":160},"--shiki-default:#005CC5",[162],{"type":25,"value":163}," \\\n",{"type":20,"tag":141,"props":165,"children":167},{"class":143,"line":166},2,[168,173],{"type":20,"tag":141,"props":169,"children":170},{"style":160},[171],{"type":25,"value":172}," --extract-audio",{"type":20,"tag":141,"props":174,"children":175},{"style":160},[176],{"type":25,"value":163},{"type":20,"tag":141,"props":178,"children":180},{"class":143,"line":179},3,[181,186,191],{"type":20,"tag":141,"props":182,"children":183},{"style":160},[184],{"type":25,"value":185}," --audio-format",{"type":20,"tag":141,"props":187,"children":188},{"style":154},[189],{"type":25,"value":190}," wav",{"type":20,"tag":141,"props":192,"children":193},{"style":160},[194],{"type":25,"value":163},{"type":20,"tag":141,"props":196,"children":198},{"class":143,"line":197},4,[199,204,209],{"type":20,"tag":141,"props":200,"children":201},{"style":160},[202],{"type":25,"value":203}," --output",{"type":20,"tag":141,"props":205,"children":206},{"style":154},[207],{"type":25,"value":208}," podcast.wav",{"type":20,"tag":141,"props":210,"children":211},{"style":160},[212],{"type":25,"value":163},{"type":20,"tag":141,"props":214,"children":216},{"class":143,"line":215},5,[217],{"type":20,"tag":141,"props":218,"children":219},{"style":154},[220],{"type":25,"value":221}," \"https://www.youtube.com/watch?v=CoUN690wSYQ\"\n",{"type":20,"tag":21,"props":223,"children":224},{},[225,227,232],{"type":25,"value":226},"This file has a 44.1 kHz sample rate, and ",{"type":20,"tag":92,"props":228,"children":230},{"className":229},[],[231],{"type":25,"value":12},{"type":25,"value":233}," expects 16 kHz, so let's go ahead and convert that.",{"type":20,"tag":131,"props":235,"children":237},{"code":236,"language":134,"meta":7,"className":135,"style":7}," $ file podcast.wav\npodcast.wav: RIFF (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 44100 Hz\n\n $ ffmpeg -i podcast.wav -ar 16000 podcast-16khz.wav\n\n $ file podcast-16khz.wav\npodcast-16khz.wav: RIFF (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 16000 Hz\n\n# NOTE: it looks like it's possible to specify this conversion as a post-process as a\n# flag to the `youtube-dl` command -- I will explore this further next time...\n# youtube-dl --extract-audio --audio-quality 0 --audio-format mp3 --postprocessor-args \"-ar 44100\" %dl%\n",[238],{"type":20,"tag":92,"props":239,"children":240},{"__ignoreMap":7},[241,258,277,286,322,329,345,363,371,381,390],{"type":20,"tag":141,"props":242,"children":243},{"class":143,"line":144},[244,248,253],{"type":20,"tag":141,"props":245,"children":246},{"style":148},[247],{"type":25,"value":151},{"type":20,"tag":141,"props":249,"children":250},{"style":154},[251],{"type":25,"value":252}," file",{"type":20,"tag":141,"props":254,"children":255},{"style":154},[256],{"type":25,"value":257}," podcast.wav\n",{"type":20,"tag":141,"props":259,"children":260},{"class":143,"line":166},[261,266,271],{"type":20,"tag":141,"props":262,"children":263},{"style":148},[264],{"type":25,"value":265},"podcast.wav:",{"type":20,"tag":141,"props":267,"children":268},{"style":154},[269],{"type":25,"value":270}," RIFF",{"type":20,"tag":141,"props":272,"children":274},{"style":273},"--shiki-default:#24292E",[275],{"type":25,"value":276}," (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 44100 Hz\n",{"type":20,"tag":141,"props":278,"children":279},{"class":143,"line":179},[280],{"type":20,"tag":141,"props":281,"children":283},{"emptyLinePlaceholder":282},true,[284],{"type":25,"value":285},"\n",{"type":20,"tag":141,"props":287,"children":288},{"class":143,"line":197},[289,293,298,303,307,312,317],{"type":20,"tag":141,"props":290,"children":291},{"style":148},[292],{"type":25,"value":151},{"type":20,"tag":141,"props":294,"children":295},{"style":154},[296],{"type":25,"value":297}," ffmpeg",{"type":20,"tag":141,"props":299,"children":300},{"style":160},[301],{"type":25,"value":302}," -i",{"type":20,"tag":141,"props":304,"children":305},{"style":154},[306],{"type":25,"value":208},{"type":20,"tag":141,"props":308,"children":309},{"style":160},[310],{"type":25,"value":311}," -ar",{"type":20,"tag":141,"props":313,"children":314},{"style":160},[315],{"type":25,"value":316}," 16000",{"type":20,"tag":141,"props":318,"children":319},{"style":154},[320],{"type":25,"value":321}," podcast-16khz.wav\n",{"type":20,"tag":141,"props":323,"children":324},{"class":143,"line":215},[325],{"type":20,"tag":141,"props":326,"children":327},{"emptyLinePlaceholder":282},[328],{"type":25,"value":285},{"type":20,"tag":141,"props":330,"children":332},{"class":143,"line":331},6,[333,337,341],{"type":20,"tag":141,"props":334,"children":335},{"style":148},[336],{"type":25,"value":151},{"type":20,"tag":141,"props":338,"children":339},{"style":154},[340],{"type":25,"value":252},{"type":20,"tag":141,"props":342,"children":343},{"style":154},[344],{"type":25,"value":321},{"type":20,"tag":141,"props":346,"children":348},{"class":143,"line":347},7,[349,354,358],{"type":20,"tag":141,"props":350,"children":351},{"style":148},[352],{"type":25,"value":353},"podcast-16khz.wav:",{"type":20,"tag":141,"props":355,"children":356},{"style":154},[357],{"type":25,"value":270},{"type":20,"tag":141,"props":359,"children":360},{"style":273},[361],{"type":25,"value":362}," (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 16000 Hz\n",{"type":20,"tag":141,"props":364,"children":366},{"class":143,"line":365},8,[367],{"type":20,"tag":141,"props":368,"children":369},{"emptyLinePlaceholder":282},[370],{"type":25,"value":285},{"type":20,"tag":141,"props":372,"children":374},{"class":143,"line":373},9,[375],{"type":20,"tag":141,"props":376,"children":378},{"style":377},"--shiki-default:#6A737D",[379],{"type":25,"value":380},"# NOTE: it looks like it's possible to specify this conversion as a post-process as a\n",{"type":20,"tag":141,"props":382,"children":384},{"class":143,"line":383},10,[385],{"type":20,"tag":141,"props":386,"children":387},{"style":377},[388],{"type":25,"value":389},"# flag to the `youtube-dl` command -- I will explore this further next time...\n",{"type":20,"tag":141,"props":391,"children":393},{"class":143,"line":392},11,[394],{"type":20,"tag":141,"props":395,"children":396},{"style":377},[397],{"type":25,"value":398},"# youtube-dl --extract-audio --audio-quality 0 --audio-format mp3 --postprocessor-args \"-ar 44100\" %dl%\n",{"type":20,"tag":80,"props":400,"children":402},{"id":401},"build-whispercpp-transcribe-audio",[403,405,410],{"type":25,"value":404},"Build ",{"type":20,"tag":92,"props":406,"children":408},{"className":407},[],[409],{"type":25,"value":12},{"type":25,"value":411}," & Transcribe Audio",{"type":20,"tag":21,"props":413,"children":414},{},[415,417,422],{"type":25,"value":416},"Then, let's get the latest version of ",{"type":20,"tag":92,"props":418,"children":420},{"className":419},[],[421],{"type":25,"value":12},{"type":25,"value":423},", download the English Whisper model, and build the example.",{"type":20,"tag":131,"props":425,"children":427},{"code":426,"language":134,"meta":7,"className":135,"style":7},"# Clone the `whisper.cpp` repository\n $ git clone --depth 1 git@github.com:ggerganov/whisper.cpp && cd whisper.cpp\n\n# Download the English Whisper model in `ggml` format\n $ bash ./models/download-ggml-model.sh base.en\n\n# Build the main example\n $ make\n",[428],{"type":20,"tag":92,"props":429,"children":430},{"__ignoreMap":7},[431,439,486,493,501,523,530,538],{"type":20,"tag":141,"props":432,"children":433},{"class":143,"line":144},[434],{"type":20,"tag":141,"props":435,"children":436},{"style":377},[437],{"type":25,"value":438},"# Clone the `whisper.cpp` repository\n",{"type":20,"tag":141,"props":440,"children":441},{"class":143,"line":166},[442,446,451,456,461,466,471,476,481],{"type":20,"tag":141,"props":443,"children":444},{"style":148},[445],{"type":25,"value":151},{"type":20,"tag":141,"props":447,"children":448},{"style":154},[449],{"type":25,"value":450}," git",{"type":20,"tag":141,"props":452,"children":453},{"style":154},[454],{"type":25,"value":455}," clone",{"type":20,"tag":141,"props":457,"children":458},{"style":160},[459],{"type":25,"value":460}," --depth",{"type":20,"tag":141,"props":462,"children":463},{"style":160},[464],{"type":25,"value":465}," 1",{"type":20,"tag":141,"props":467,"children":468},{"style":154},[469],{"type":25,"value":470}," git@github.com:ggerganov/whisper.cpp",{"type":20,"tag":141,"props":472,"children":473},{"style":273},[474],{"type":25,"value":475}," && ",{"type":20,"tag":141,"props":477,"children":478},{"style":160},[479],{"type":25,"value":480},"cd",{"type":20,"tag":141,"props":482,"children":483},{"style":154},[484],{"type":25,"value":485}," whisper.cpp\n",{"type":20,"tag":141,"props":487,"children":488},{"class":143,"line":179},[489],{"type":20,"tag":141,"props":490,"children":491},{"emptyLinePlaceholder":282},[492],{"type":25,"value":285},{"type":20,"tag":141,"props":494,"children":495},{"class":143,"line":197},[496],{"type":20,"tag":141,"props":497,"children":498},{"style":377},[499],{"type":25,"value":500},"# Download the English Whisper model in `ggml` format\n",{"type":20,"tag":141,"props":502,"children":503},{"class":143,"line":215},[504,508,513,518],{"type":20,"tag":141,"props":505,"children":506},{"style":148},[507],{"type":25,"value":151},{"type":20,"tag":141,"props":509,"children":510},{"style":154},[511],{"type":25,"value":512}," bash",{"type":20,"tag":141,"props":514,"children":515},{"style":154},[516],{"type":25,"value":517}," ./models/download-ggml-model.sh",{"type":20,"tag":141,"props":519,"children":520},{"style":154},[521],{"type":25,"value":522}," base.en\n",{"type":20,"tag":141,"props":524,"children":525},{"class":143,"line":331},[526],{"type":20,"tag":141,"props":527,"children":528},{"emptyLinePlaceholder":282},[529],{"type":25,"value":285},{"type":20,"tag":141,"props":531,"children":532},{"class":143,"line":347},[533],{"type":20,"tag":141,"props":534,"children":535},{"style":377},[536],{"type":25,"value":537},"# Build the main example\n",{"type":20,"tag":141,"props":539,"children":540},{"class":143,"line":365},[541,545],{"type":20,"tag":141,"props":542,"children":543},{"style":148},[544],{"type":25,"value":151},{"type":20,"tag":141,"props":546,"children":547},{"style":154},[548],{"type":25,"value":549}," make\n",{"type":20,"tag":21,"props":551,"children":552},{},[553],{"type":25,"value":554},"And finally, let's transcribe that podcast!",{"type":20,"tag":131,"props":556,"children":558},{"code":557,"language":134,"meta":7,"className":135,"style":7}," $ ./main \\\n -m ~/workspace/whisper.cpp/models/ggml-base.en.bin \\\n -f ~/Downloads/podcast-16khz.wav \\\n --output-vtt \\\n --output-file out\n\n# whisper_print_timings: load time = 114.71 ms\n# whisper_print_timings: fallbacks = 0 p / 0 h\n# whisper_print_timings: mel time = 692.20 ms\n# whisper_print_timings: sample time = 22278.10 ms / 27893 runs ( 0.80 ms per run)\n# whisper_print_timings: encode time = 10000.75 ms / 55 runs ( 181.83 ms per run)\n# whisper_print_timings: decode time = 331.77 ms / 54 runs ( 6.14 ms per run)\n# whisper_print_timings: batchd time = 45236.73 ms / 27566 runs ( 1.64 ms per run)\n# whisper_print_timings: prompt time = 1921.90 ms / 11832 runs ( 0.16 ms per run)\n# whisper_print_timings: total time = 80709.54 ms\n",[559],{"type":20,"tag":92,"props":560,"children":561},{"__ignoreMap":7},[562,578,595,612,624,637,644,652,660,668,676,684,693,702,711],{"type":20,"tag":141,"props":563,"children":564},{"class":143,"line":144},[565,569,574],{"type":20,"tag":141,"props":566,"children":567},{"style":148},[568],{"type":25,"value":151},{"type":20,"tag":141,"props":570,"children":571},{"style":154},[572],{"type":25,"value":573}," ./main",{"type":20,"tag":141,"props":575,"children":576},{"style":160},[577],{"type":25,"value":163},{"type":20,"tag":141,"props":579,"children":580},{"class":143,"line":166},[581,586,591],{"type":20,"tag":141,"props":582,"children":583},{"style":160},[584],{"type":25,"value":585}," -m",{"type":20,"tag":141,"props":587,"children":588},{"style":154},[589],{"type":25,"value":590}," ~/workspace/whisper.cpp/models/ggml-base.en.bin",{"type":20,"tag":141,"props":592,"children":593},{"style":160},[594],{"type":25,"value":163},{"type":20,"tag":141,"props":596,"children":597},{"class":143,"line":179},[598,603,608],{"type":20,"tag":141,"props":599,"children":600},{"style":160},[601],{"type":25,"value":602}," -f",{"type":20,"tag":141,"props":604,"children":605},{"style":154},[606],{"type":25,"value":607}," ~/Downloads/podcast-16khz.wav",{"type":20,"tag":141,"props":609,"children":610},{"style":160},[611],{"type":25,"value":163},{"type":20,"tag":141,"props":613,"children":614},{"class":143,"line":197},[615,620],{"type":20,"tag":141,"props":616,"children":617},{"style":160},[618],{"type":25,"value":619}," --output-vtt",{"type":20,"tag":141,"props":621,"children":622},{"style":160},[623],{"type":25,"value":163},{"type":20,"tag":141,"props":625,"children":626},{"class":143,"line":215},[627,632],{"type":20,"tag":141,"props":628,"children":629},{"style":160},[630],{"type":25,"value":631}," --output-file",{"type":20,"tag":141,"props":633,"children":634},{"style":154},[635],{"type":25,"value":636}," out\n",{"type":20,"tag":141,"props":638,"children":639},{"class":143,"line":331},[640],{"type":20,"tag":141,"props":641,"children":642},{"emptyLinePlaceholder":282},[643],{"type":25,"value":285},{"type":20,"tag":141,"props":645,"children":646},{"class":143,"line":347},[647],{"type":20,"tag":141,"props":648,"children":649},{"style":377},[650],{"type":25,"value":651},"# whisper_print_timings: load time = 114.71 ms\n",{"type":20,"tag":141,"props":653,"children":654},{"class":143,"line":365},[655],{"type":20,"tag":141,"props":656,"children":657},{"style":377},[658],{"type":25,"value":659},"# whisper_print_timings: fallbacks = 0 p / 0 h\n",{"type":20,"tag":141,"props":661,"children":662},{"class":143,"line":373},[663],{"type":20,"tag":141,"props":664,"children":665},{"style":377},[666],{"type":25,"value":667},"# whisper_print_timings: mel time = 692.20 ms\n",{"type":20,"tag":141,"props":669,"children":670},{"class":143,"line":383},[671],{"type":20,"tag":141,"props":672,"children":673},{"style":377},[674],{"type":25,"value":675},"# whisper_print_timings: sample time = 22278.10 ms / 27893 runs ( 0.80 ms per run)\n",{"type":20,"tag":141,"props":677,"children":678},{"class":143,"line":392},[679],{"type":20,"tag":141,"props":680,"children":681},{"style":377},[682],{"type":25,"value":683},"# whisper_print_timings: encode time = 10000.75 ms / 55 runs ( 181.83 ms per run)\n",{"type":20,"tag":141,"props":685,"children":687},{"class":143,"line":686},12,[688],{"type":20,"tag":141,"props":689,"children":690},{"style":377},[691],{"type":25,"value":692},"# whisper_print_timings: decode time = 331.77 ms / 54 runs ( 6.14 ms per run)\n",{"type":20,"tag":141,"props":694,"children":696},{"class":143,"line":695},13,[697],{"type":20,"tag":141,"props":698,"children":699},{"style":377},[700],{"type":25,"value":701},"# whisper_print_timings: batchd time = 45236.73 ms / 27566 runs ( 1.64 ms per run)\n",{"type":20,"tag":141,"props":703,"children":705},{"class":143,"line":704},14,[706],{"type":20,"tag":141,"props":707,"children":708},{"style":377},[709],{"type":25,"value":710},"# whisper_print_timings: prompt time = 1921.90 ms / 11832 runs ( 0.16 ms per run)\n",{"type":20,"tag":141,"props":712,"children":714},{"class":143,"line":713},15,[715],{"type":20,"tag":141,"props":716,"children":717},{"style":377},[718],{"type":25,"value":719},"# whisper_print_timings: total time = 80709.54 ms\n",{"type":20,"tag":21,"props":721,"children":722},{},[723],{"type":25,"value":724},"A full podcast transcribed in ~80 seconds on an M1 Mac Mini -- not too bad!",{"type":20,"tag":131,"props":726,"children":728},{"code":727},"# out.vtt\n\n00:00:00.000 --> 00:00:06.480\n >> Hi everyone. We recently launched a short engagement feedback survey for the Alter Everything\n\n00:00:06.480 --> 00:00:11.360\n podcast. Click the link in the episode description wherever you're listening to let us know what\n\n00:00:11.360 --> 00:00:16.320\n you think and help us improve our show.\n\n00:00:16.320 --> 00:00:21.200\n Welcome to Alter Everything, a podcast about data science and analytics culture. I'm Megan\n\n00:00:21.200 --> 00:00:26.440\n Dibble and today I'm talking with Nick Schrock, CTO and founder of Dagster Labs. We discussed\n\n00:00:26.440 --> 00:00:31.560\n data engineering trends, challenges in the field, why he started his company, and what\n\n00:00:31.560 --> 00:00:38.960\n makes him excited about the future of data engineering. Let's get started.\n\n00:00:38.960 --> 00:00:42.720\n >> Hi, Nick. It's great to have you on our show today. Thanks for being here.\n\n00:00:42.720 --> 00:00:43.920\n >> Thanks for having me.\n\n00:00:43.920 --> 00:00:48.280\n >> Yeah. Could you start off by giving an introduction to yourself for our listeners?\n\n00:00:48.280 --> 00:00:52.920\n >> Sure. My name is Nick Schrock. I'm the CTO and founder of Dagster Labs. There's the\n\n00:00:52.920 --> 00:00:59.520\n company behind Dagster, which is a data orchestration framework. Prior to doing this, I was an engineer\n\n00:00:59.520 --> 00:01:05.960\n at Facebook from 2009, 2017. While I was there, I found a team called product infrastructure\n\n00:01:05.960 --> 00:01:09.800\n whose goal was to make our application developers more efficient and productive, and a bunch\n\n00:01:09.800 --> 00:01:13.840\n of open source work came out of that actually, one of which was React, which I had nothing\n\n00:01:13.840 --> 00:01:18.040\n to do with, but actually the CEO of Dagster Labs co-created and I personally co-created\n\n00:01:18.040 --> 00:01:22.640\n GraphQL. So as I like to say, Pete and I were present at the creation of the full hipster\n\n00:01:22.640 --> 00:01:28.680\n stack. I moved on to Facebook in 2017, figuring out what to do next, and this data engineering\n\n00:01:28.680 --> 00:01:32.960\n and data orchestration problem really got me hooked actually quite soon after I left,\n\n00:01:32.960 --> 00:01:36.280\n and the rest is history. I'm sure we'll get into that more.\n",[729],{"type":20,"tag":92,"props":730,"children":731},{"__ignoreMap":7},[732],{"type":25,"value":727},{"type":20,"tag":734,"props":735,"children":736},"style",{},[737],{"type":25,"value":738},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":166,"depth":166,"links":740},[741,742],{"id":82,"depth":166,"text":85},{"id":401,"depth":166,"text":743},"Build whisper.cpp & Transcribe Audio","markdown","content:articles:podcast-transcription-whispercpp.md","content","articles/podcast-transcription-whispercpp.md","articles/podcast-transcription-whispercpp","md",1731781853000] \ No newline at end of file +[{"data":1,"prerenderedAt":750},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"draft":6,"date":10,"tags":11,"categories":14,"excerpt":16,"body":55,"_type":744,"_id":745,"_source":746,"_file":747,"_stem":748,"_extension":749},"/articles/podcast-transcription-whispercpp","articles",false,"","Easily Transcribe Podcasts with Whisper.cpp","If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive whisper.cpp project. This high-performance fork of OpenAI's Whisper can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the Alter Everything podcast.","2024-01-08",[12,13],"whisper.cpp","ml",[15],"programming",{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24,27,35,37,44,46,53],{"type":25,"value":26},"text","If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive ",{"type":20,"tag":28,"props":29,"children":33},"a",{"href":30,"rel":31},"https://github.com/ggerganov/whisper.cpp",[32],"nofollow",[34],{"type":25,"value":12},{"type":25,"value":36}," project. This high-performance fork of ",{"type":20,"tag":28,"props":38,"children":41},{"href":39,"rel":40},"https://github.com/openai/whisper",[32],[42],{"type":25,"value":43},"OpenAI's Whisper",{"type":25,"value":45}," can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the ",{"type":20,"tag":28,"props":47,"children":50},{"href":48,"rel":49},"https://podcasts.apple.com/us/podcast/alter-everything/id1356137854",[32],[51],{"type":25,"value":52},"Alter Everything",{"type":25,"value":54}," podcast.",{"type":17,"children":56,"toc":739},[57,79,86,130,222,234,399,412,424,550,555,720,725,733],{"type":20,"tag":21,"props":58,"children":59},{},[60,61,66,67,72,73,78],{"type":25,"value":26},{"type":20,"tag":28,"props":62,"children":64},{"href":30,"rel":63},[32],[65],{"type":25,"value":12},{"type":25,"value":36},{"type":20,"tag":28,"props":68,"children":70},{"href":39,"rel":69},[32],[71],{"type":25,"value":43},{"type":25,"value":45},{"type":20,"tag":28,"props":74,"children":76},{"href":48,"rel":75},[32],[77],{"type":25,"value":52},{"type":25,"value":54},{"type":20,"tag":80,"props":81,"children":83},"h2",{"id":82},"obtain-audio-files",[84],{"type":25,"value":85},"Obtain Audio File(s)",{"type":20,"tag":21,"props":87,"children":88},{},[89,91,98,100,106,108,113,115,120,122,128],{"type":25,"value":90},"First, let's get the ",{"type":20,"tag":92,"props":93,"children":95},"code",{"className":94},[],[96],{"type":25,"value":97},"wav",{"type":25,"value":99}," file from YouTube using the ",{"type":20,"tag":92,"props":101,"children":103},{"className":102},[],[104],{"type":25,"value":105},"youtube-dl",{"type":25,"value":107}," utility. It should be noted that ",{"type":20,"tag":92,"props":109,"children":111},{"className":110},[],[112],{"type":25,"value":12},{"type":25,"value":114}," expects ",{"type":20,"tag":92,"props":116,"children":118},{"className":117},[],[119],{"type":25,"value":97},{"type":25,"value":121}," filetypes, and this utility defaults to ",{"type":20,"tag":92,"props":123,"children":125},{"className":124},[],[126],{"type":25,"value":127},"mp3",{"type":25,"value":129},".",{"type":20,"tag":131,"props":132,"children":136},"pre",{"code":133,"language":134,"meta":7,"className":135,"style":7}," $ youtube-dl \\\n --extract-audio \\\n --audio-format wav \\\n --output podcast.wav \\\n \"https://www.youtube.com/watch?v=CoUN690wSYQ\"\n","sh","language-sh shiki shiki-themes github-light",[137],{"type":20,"tag":92,"props":138,"children":139},{"__ignoreMap":7},[140,164,177,195,213],{"type":20,"tag":141,"props":142,"children":145},"span",{"class":143,"line":144},"line",1,[146,152,158],{"type":20,"tag":141,"props":147,"children":149},{"style":148},"--shiki-default:#6F42C1",[150],{"type":25,"value":151}," $",{"type":20,"tag":141,"props":153,"children":155},{"style":154},"--shiki-default:#032F62",[156],{"type":25,"value":157}," youtube-dl",{"type":20,"tag":141,"props":159,"children":161},{"style":160},"--shiki-default:#005CC5",[162],{"type":25,"value":163}," \\\n",{"type":20,"tag":141,"props":165,"children":167},{"class":143,"line":166},2,[168,173],{"type":20,"tag":141,"props":169,"children":170},{"style":160},[171],{"type":25,"value":172}," --extract-audio",{"type":20,"tag":141,"props":174,"children":175},{"style":160},[176],{"type":25,"value":163},{"type":20,"tag":141,"props":178,"children":180},{"class":143,"line":179},3,[181,186,191],{"type":20,"tag":141,"props":182,"children":183},{"style":160},[184],{"type":25,"value":185}," --audio-format",{"type":20,"tag":141,"props":187,"children":188},{"style":154},[189],{"type":25,"value":190}," wav",{"type":20,"tag":141,"props":192,"children":193},{"style":160},[194],{"type":25,"value":163},{"type":20,"tag":141,"props":196,"children":198},{"class":143,"line":197},4,[199,204,209],{"type":20,"tag":141,"props":200,"children":201},{"style":160},[202],{"type":25,"value":203}," --output",{"type":20,"tag":141,"props":205,"children":206},{"style":154},[207],{"type":25,"value":208}," podcast.wav",{"type":20,"tag":141,"props":210,"children":211},{"style":160},[212],{"type":25,"value":163},{"type":20,"tag":141,"props":214,"children":216},{"class":143,"line":215},5,[217],{"type":20,"tag":141,"props":218,"children":219},{"style":154},[220],{"type":25,"value":221}," \"https://www.youtube.com/watch?v=CoUN690wSYQ\"\n",{"type":20,"tag":21,"props":223,"children":224},{},[225,227,232],{"type":25,"value":226},"This file has a 44.1 kHz sample rate, and ",{"type":20,"tag":92,"props":228,"children":230},{"className":229},[],[231],{"type":25,"value":12},{"type":25,"value":233}," expects 16 kHz, so let's go ahead and convert that.",{"type":20,"tag":131,"props":235,"children":237},{"code":236,"language":134,"meta":7,"className":135,"style":7}," $ file podcast.wav\npodcast.wav: RIFF (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 44100 Hz\n\n $ ffmpeg -i podcast.wav -ar 16000 podcast-16khz.wav\n\n $ file podcast-16khz.wav\npodcast-16khz.wav: RIFF (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 16000 Hz\n\n# NOTE: it looks like it's possible to specify this conversion as a post-process as a\n# flag to the `youtube-dl` command -- I will explore this further next time...\n# youtube-dl --extract-audio --audio-quality 0 --audio-format mp3 --postprocessor-args \"-ar 44100\" %dl%\n",[238],{"type":20,"tag":92,"props":239,"children":240},{"__ignoreMap":7},[241,258,277,286,322,329,345,363,371,381,390],{"type":20,"tag":141,"props":242,"children":243},{"class":143,"line":144},[244,248,253],{"type":20,"tag":141,"props":245,"children":246},{"style":148},[247],{"type":25,"value":151},{"type":20,"tag":141,"props":249,"children":250},{"style":154},[251],{"type":25,"value":252}," file",{"type":20,"tag":141,"props":254,"children":255},{"style":154},[256],{"type":25,"value":257}," podcast.wav\n",{"type":20,"tag":141,"props":259,"children":260},{"class":143,"line":166},[261,266,271],{"type":20,"tag":141,"props":262,"children":263},{"style":148},[264],{"type":25,"value":265},"podcast.wav:",{"type":20,"tag":141,"props":267,"children":268},{"style":154},[269],{"type":25,"value":270}," RIFF",{"type":20,"tag":141,"props":272,"children":274},{"style":273},"--shiki-default:#24292E",[275],{"type":25,"value":276}," (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 44100 Hz\n",{"type":20,"tag":141,"props":278,"children":279},{"class":143,"line":179},[280],{"type":20,"tag":141,"props":281,"children":283},{"emptyLinePlaceholder":282},true,[284],{"type":25,"value":285},"\n",{"type":20,"tag":141,"props":287,"children":288},{"class":143,"line":197},[289,293,298,303,307,312,317],{"type":20,"tag":141,"props":290,"children":291},{"style":148},[292],{"type":25,"value":151},{"type":20,"tag":141,"props":294,"children":295},{"style":154},[296],{"type":25,"value":297}," ffmpeg",{"type":20,"tag":141,"props":299,"children":300},{"style":160},[301],{"type":25,"value":302}," -i",{"type":20,"tag":141,"props":304,"children":305},{"style":154},[306],{"type":25,"value":208},{"type":20,"tag":141,"props":308,"children":309},{"style":160},[310],{"type":25,"value":311}," -ar",{"type":20,"tag":141,"props":313,"children":314},{"style":160},[315],{"type":25,"value":316}," 16000",{"type":20,"tag":141,"props":318,"children":319},{"style":154},[320],{"type":25,"value":321}," podcast-16khz.wav\n",{"type":20,"tag":141,"props":323,"children":324},{"class":143,"line":215},[325],{"type":20,"tag":141,"props":326,"children":327},{"emptyLinePlaceholder":282},[328],{"type":25,"value":285},{"type":20,"tag":141,"props":330,"children":332},{"class":143,"line":331},6,[333,337,341],{"type":20,"tag":141,"props":334,"children":335},{"style":148},[336],{"type":25,"value":151},{"type":20,"tag":141,"props":338,"children":339},{"style":154},[340],{"type":25,"value":252},{"type":20,"tag":141,"props":342,"children":343},{"style":154},[344],{"type":25,"value":321},{"type":20,"tag":141,"props":346,"children":348},{"class":143,"line":347},7,[349,354,358],{"type":20,"tag":141,"props":350,"children":351},{"style":148},[352],{"type":25,"value":353},"podcast-16khz.wav:",{"type":20,"tag":141,"props":355,"children":356},{"style":154},[357],{"type":25,"value":270},{"type":20,"tag":141,"props":359,"children":360},{"style":273},[361],{"type":25,"value":362}," (little-endian) data, WAVE audio, Microsoft PCM, 16 bit, stereo 16000 Hz\n",{"type":20,"tag":141,"props":364,"children":366},{"class":143,"line":365},8,[367],{"type":20,"tag":141,"props":368,"children":369},{"emptyLinePlaceholder":282},[370],{"type":25,"value":285},{"type":20,"tag":141,"props":372,"children":374},{"class":143,"line":373},9,[375],{"type":20,"tag":141,"props":376,"children":378},{"style":377},"--shiki-default:#6A737D",[379],{"type":25,"value":380},"# NOTE: it looks like it's possible to specify this conversion as a post-process as a\n",{"type":20,"tag":141,"props":382,"children":384},{"class":143,"line":383},10,[385],{"type":20,"tag":141,"props":386,"children":387},{"style":377},[388],{"type":25,"value":389},"# flag to the `youtube-dl` command -- I will explore this further next time...\n",{"type":20,"tag":141,"props":391,"children":393},{"class":143,"line":392},11,[394],{"type":20,"tag":141,"props":395,"children":396},{"style":377},[397],{"type":25,"value":398},"# youtube-dl --extract-audio --audio-quality 0 --audio-format mp3 --postprocessor-args \"-ar 44100\" %dl%\n",{"type":20,"tag":80,"props":400,"children":402},{"id":401},"build-whispercpp-transcribe-audio",[403,405,410],{"type":25,"value":404},"Build ",{"type":20,"tag":92,"props":406,"children":408},{"className":407},[],[409],{"type":25,"value":12},{"type":25,"value":411}," & Transcribe Audio",{"type":20,"tag":21,"props":413,"children":414},{},[415,417,422],{"type":25,"value":416},"Then, let's get the latest version of ",{"type":20,"tag":92,"props":418,"children":420},{"className":419},[],[421],{"type":25,"value":12},{"type":25,"value":423},", download the English Whisper model, and build the example.",{"type":20,"tag":131,"props":425,"children":427},{"code":426,"language":134,"meta":7,"className":135,"style":7},"# Clone the `whisper.cpp` repository\n $ git clone --depth 1 git@github.com:ggerganov/whisper.cpp && cd whisper.cpp\n\n# Download the English Whisper model in `ggml` format\n $ bash ./models/download-ggml-model.sh base.en\n\n# Build the main example\n $ make\n",[428],{"type":20,"tag":92,"props":429,"children":430},{"__ignoreMap":7},[431,439,486,493,501,523,530,538],{"type":20,"tag":141,"props":432,"children":433},{"class":143,"line":144},[434],{"type":20,"tag":141,"props":435,"children":436},{"style":377},[437],{"type":25,"value":438},"# Clone the `whisper.cpp` repository\n",{"type":20,"tag":141,"props":440,"children":441},{"class":143,"line":166},[442,446,451,456,461,466,471,476,481],{"type":20,"tag":141,"props":443,"children":444},{"style":148},[445],{"type":25,"value":151},{"type":20,"tag":141,"props":447,"children":448},{"style":154},[449],{"type":25,"value":450}," git",{"type":20,"tag":141,"props":452,"children":453},{"style":154},[454],{"type":25,"value":455}," clone",{"type":20,"tag":141,"props":457,"children":458},{"style":160},[459],{"type":25,"value":460}," --depth",{"type":20,"tag":141,"props":462,"children":463},{"style":160},[464],{"type":25,"value":465}," 1",{"type":20,"tag":141,"props":467,"children":468},{"style":154},[469],{"type":25,"value":470}," git@github.com:ggerganov/whisper.cpp",{"type":20,"tag":141,"props":472,"children":473},{"style":273},[474],{"type":25,"value":475}," && ",{"type":20,"tag":141,"props":477,"children":478},{"style":160},[479],{"type":25,"value":480},"cd",{"type":20,"tag":141,"props":482,"children":483},{"style":154},[484],{"type":25,"value":485}," whisper.cpp\n",{"type":20,"tag":141,"props":487,"children":488},{"class":143,"line":179},[489],{"type":20,"tag":141,"props":490,"children":491},{"emptyLinePlaceholder":282},[492],{"type":25,"value":285},{"type":20,"tag":141,"props":494,"children":495},{"class":143,"line":197},[496],{"type":20,"tag":141,"props":497,"children":498},{"style":377},[499],{"type":25,"value":500},"# Download the English Whisper model in `ggml` format\n",{"type":20,"tag":141,"props":502,"children":503},{"class":143,"line":215},[504,508,513,518],{"type":20,"tag":141,"props":505,"children":506},{"style":148},[507],{"type":25,"value":151},{"type":20,"tag":141,"props":509,"children":510},{"style":154},[511],{"type":25,"value":512}," bash",{"type":20,"tag":141,"props":514,"children":515},{"style":154},[516],{"type":25,"value":517}," ./models/download-ggml-model.sh",{"type":20,"tag":141,"props":519,"children":520},{"style":154},[521],{"type":25,"value":522}," base.en\n",{"type":20,"tag":141,"props":524,"children":525},{"class":143,"line":331},[526],{"type":20,"tag":141,"props":527,"children":528},{"emptyLinePlaceholder":282},[529],{"type":25,"value":285},{"type":20,"tag":141,"props":531,"children":532},{"class":143,"line":347},[533],{"type":20,"tag":141,"props":534,"children":535},{"style":377},[536],{"type":25,"value":537},"# Build the main example\n",{"type":20,"tag":141,"props":539,"children":540},{"class":143,"line":365},[541,545],{"type":20,"tag":141,"props":542,"children":543},{"style":148},[544],{"type":25,"value":151},{"type":20,"tag":141,"props":546,"children":547},{"style":154},[548],{"type":25,"value":549}," make\n",{"type":20,"tag":21,"props":551,"children":552},{},[553],{"type":25,"value":554},"And finally, let's transcribe that podcast!",{"type":20,"tag":131,"props":556,"children":558},{"code":557,"language":134,"meta":7,"className":135,"style":7}," $ ./main \\\n -m ~/workspace/whisper.cpp/models/ggml-base.en.bin \\\n -f ~/Downloads/podcast-16khz.wav \\\n --output-vtt \\\n --output-file out\n\n# whisper_print_timings: load time = 114.71 ms\n# whisper_print_timings: fallbacks = 0 p / 0 h\n# whisper_print_timings: mel time = 692.20 ms\n# whisper_print_timings: sample time = 22278.10 ms / 27893 runs ( 0.80 ms per run)\n# whisper_print_timings: encode time = 10000.75 ms / 55 runs ( 181.83 ms per run)\n# whisper_print_timings: decode time = 331.77 ms / 54 runs ( 6.14 ms per run)\n# whisper_print_timings: batchd time = 45236.73 ms / 27566 runs ( 1.64 ms per run)\n# whisper_print_timings: prompt time = 1921.90 ms / 11832 runs ( 0.16 ms per run)\n# whisper_print_timings: total time = 80709.54 ms\n",[559],{"type":20,"tag":92,"props":560,"children":561},{"__ignoreMap":7},[562,578,595,612,624,637,644,652,660,668,676,684,693,702,711],{"type":20,"tag":141,"props":563,"children":564},{"class":143,"line":144},[565,569,574],{"type":20,"tag":141,"props":566,"children":567},{"style":148},[568],{"type":25,"value":151},{"type":20,"tag":141,"props":570,"children":571},{"style":154},[572],{"type":25,"value":573}," ./main",{"type":20,"tag":141,"props":575,"children":576},{"style":160},[577],{"type":25,"value":163},{"type":20,"tag":141,"props":579,"children":580},{"class":143,"line":166},[581,586,591],{"type":20,"tag":141,"props":582,"children":583},{"style":160},[584],{"type":25,"value":585}," -m",{"type":20,"tag":141,"props":587,"children":588},{"style":154},[589],{"type":25,"value":590}," ~/workspace/whisper.cpp/models/ggml-base.en.bin",{"type":20,"tag":141,"props":592,"children":593},{"style":160},[594],{"type":25,"value":163},{"type":20,"tag":141,"props":596,"children":597},{"class":143,"line":179},[598,603,608],{"type":20,"tag":141,"props":599,"children":600},{"style":160},[601],{"type":25,"value":602}," -f",{"type":20,"tag":141,"props":604,"children":605},{"style":154},[606],{"type":25,"value":607}," ~/Downloads/podcast-16khz.wav",{"type":20,"tag":141,"props":609,"children":610},{"style":160},[611],{"type":25,"value":163},{"type":20,"tag":141,"props":613,"children":614},{"class":143,"line":197},[615,620],{"type":20,"tag":141,"props":616,"children":617},{"style":160},[618],{"type":25,"value":619}," --output-vtt",{"type":20,"tag":141,"props":621,"children":622},{"style":160},[623],{"type":25,"value":163},{"type":20,"tag":141,"props":625,"children":626},{"class":143,"line":215},[627,632],{"type":20,"tag":141,"props":628,"children":629},{"style":160},[630],{"type":25,"value":631}," --output-file",{"type":20,"tag":141,"props":633,"children":634},{"style":154},[635],{"type":25,"value":636}," out\n",{"type":20,"tag":141,"props":638,"children":639},{"class":143,"line":331},[640],{"type":20,"tag":141,"props":641,"children":642},{"emptyLinePlaceholder":282},[643],{"type":25,"value":285},{"type":20,"tag":141,"props":645,"children":646},{"class":143,"line":347},[647],{"type":20,"tag":141,"props":648,"children":649},{"style":377},[650],{"type":25,"value":651},"# whisper_print_timings: load time = 114.71 ms\n",{"type":20,"tag":141,"props":653,"children":654},{"class":143,"line":365},[655],{"type":20,"tag":141,"props":656,"children":657},{"style":377},[658],{"type":25,"value":659},"# whisper_print_timings: fallbacks = 0 p / 0 h\n",{"type":20,"tag":141,"props":661,"children":662},{"class":143,"line":373},[663],{"type":20,"tag":141,"props":664,"children":665},{"style":377},[666],{"type":25,"value":667},"# whisper_print_timings: mel time = 692.20 ms\n",{"type":20,"tag":141,"props":669,"children":670},{"class":143,"line":383},[671],{"type":20,"tag":141,"props":672,"children":673},{"style":377},[674],{"type":25,"value":675},"# whisper_print_timings: sample time = 22278.10 ms / 27893 runs ( 0.80 ms per run)\n",{"type":20,"tag":141,"props":677,"children":678},{"class":143,"line":392},[679],{"type":20,"tag":141,"props":680,"children":681},{"style":377},[682],{"type":25,"value":683},"# whisper_print_timings: encode time = 10000.75 ms / 55 runs ( 181.83 ms per run)\n",{"type":20,"tag":141,"props":685,"children":687},{"class":143,"line":686},12,[688],{"type":20,"tag":141,"props":689,"children":690},{"style":377},[691],{"type":25,"value":692},"# whisper_print_timings: decode time = 331.77 ms / 54 runs ( 6.14 ms per run)\n",{"type":20,"tag":141,"props":694,"children":696},{"class":143,"line":695},13,[697],{"type":20,"tag":141,"props":698,"children":699},{"style":377},[700],{"type":25,"value":701},"# whisper_print_timings: batchd time = 45236.73 ms / 27566 runs ( 1.64 ms per run)\n",{"type":20,"tag":141,"props":703,"children":705},{"class":143,"line":704},14,[706],{"type":20,"tag":141,"props":707,"children":708},{"style":377},[709],{"type":25,"value":710},"# whisper_print_timings: prompt time = 1921.90 ms / 11832 runs ( 0.16 ms per run)\n",{"type":20,"tag":141,"props":712,"children":714},{"class":143,"line":713},15,[715],{"type":20,"tag":141,"props":716,"children":717},{"style":377},[718],{"type":25,"value":719},"# whisper_print_timings: total time = 80709.54 ms\n",{"type":20,"tag":21,"props":721,"children":722},{},[723],{"type":25,"value":724},"A full podcast transcribed in ~80 seconds on an M1 Mac Mini -- not too bad!",{"type":20,"tag":131,"props":726,"children":728},{"code":727},"# out.vtt\n\n00:00:00.000 --> 00:00:06.480\n >> Hi everyone. We recently launched a short engagement feedback survey for the Alter Everything\n\n00:00:06.480 --> 00:00:11.360\n podcast. Click the link in the episode description wherever you're listening to let us know what\n\n00:00:11.360 --> 00:00:16.320\n you think and help us improve our show.\n\n00:00:16.320 --> 00:00:21.200\n Welcome to Alter Everything, a podcast about data science and analytics culture. I'm Megan\n\n00:00:21.200 --> 00:00:26.440\n Dibble and today I'm talking with Nick Schrock, CTO and founder of Dagster Labs. We discussed\n\n00:00:26.440 --> 00:00:31.560\n data engineering trends, challenges in the field, why he started his company, and what\n\n00:00:31.560 --> 00:00:38.960\n makes him excited about the future of data engineering. Let's get started.\n\n00:00:38.960 --> 00:00:42.720\n >> Hi, Nick. It's great to have you on our show today. Thanks for being here.\n\n00:00:42.720 --> 00:00:43.920\n >> Thanks for having me.\n\n00:00:43.920 --> 00:00:48.280\n >> Yeah. Could you start off by giving an introduction to yourself for our listeners?\n\n00:00:48.280 --> 00:00:52.920\n >> Sure. My name is Nick Schrock. I'm the CTO and founder of Dagster Labs. There's the\n\n00:00:52.920 --> 00:00:59.520\n company behind Dagster, which is a data orchestration framework. Prior to doing this, I was an engineer\n\n00:00:59.520 --> 00:01:05.960\n at Facebook from 2009, 2017. While I was there, I found a team called product infrastructure\n\n00:01:05.960 --> 00:01:09.800\n whose goal was to make our application developers more efficient and productive, and a bunch\n\n00:01:09.800 --> 00:01:13.840\n of open source work came out of that actually, one of which was React, which I had nothing\n\n00:01:13.840 --> 00:01:18.040\n to do with, but actually the CEO of Dagster Labs co-created and I personally co-created\n\n00:01:18.040 --> 00:01:22.640\n GraphQL. So as I like to say, Pete and I were present at the creation of the full hipster\n\n00:01:22.640 --> 00:01:28.680\n stack. I moved on to Facebook in 2017, figuring out what to do next, and this data engineering\n\n00:01:28.680 --> 00:01:32.960\n and data orchestration problem really got me hooked actually quite soon after I left,\n\n00:01:32.960 --> 00:01:36.280\n and the rest is history. I'm sure we'll get into that more.\n",[729],{"type":20,"tag":92,"props":730,"children":731},{"__ignoreMap":7},[732],{"type":25,"value":727},{"type":20,"tag":734,"props":735,"children":736},"style",{},[737],{"type":25,"value":738},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":166,"depth":166,"links":740},[741,742],{"id":82,"depth":166,"text":85},{"id":401,"depth":166,"text":743},"Build whisper.cpp & Transcribe Audio","markdown","content:articles:podcast-transcription-whispercpp.md","content","articles/podcast-transcription-whispercpp.md","articles/podcast-transcription-whispercpp","md",1731782033244] \ No newline at end of file diff --git a/articles/podcast-transcription-whispercpp/index.html b/articles/podcast-transcription-whispercpp/index.html index 677b5e3d..0d56bb21 100644 --- a/articles/podcast-transcription-whispercpp/index.html +++ b/articles/podcast-transcription-whispercpp/index.html @@ -5,32 +5,32 @@ - - - - - + + + + + - + - - - - - - - - - - - - - + + + + + + + + + + + + + -

        Easily Transcribe Podcasts with Whisper.cpp

        If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive whisper.cpp project. This high-performance fork of OpenAI's Whisper can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the Alter Everything podcast.

        Obtain Audio File(s)

        First, let's get the wav file from YouTube using the youtube-dl utility. It should be noted that whisper.cpp expects wav filetypes, and this utility defaults to mp3.

         $ youtube-dl \
        +

        Easily Transcribe Podcasts with Whisper.cpp

        If you've ever had the need to transcribe a podcast, lecture, or some other audio recording, it turns out it's surprisingly easy with the extremely impressive whisper.cpp project. This high-performance fork of OpenAI's Whisper can run on all sorts of hardware -- including my M1 Mac Mini. Let's walk through an example from start-to-finish of transcribing an episode of the Alter Everything podcast.

        Obtain Audio File(s)

        First, let's get the wav file from YouTube using the youtube-dl utility. It should be noted that whisper.cpp expects wav filetypes, and this utility defaults to mp3.

         $ youtube-dl \
             --extract-audio \
             --audio-format wav \
             --output podcast.wav \
        @@ -130,5 +130,5 @@
         
         00:01:32.960 --> 00:01:36.280
          and the rest is history. I'm sure we'll get into that more.
        -
        - \ No newline at end of file +
        + \ No newline at end of file diff --git a/articles/quick-tip-rerunning-bash-commands/_payload.json b/articles/quick-tip-rerunning-bash-commands/_payload.json index 428ca4cb..6088c906 100644 --- a/articles/quick-tip-rerunning-bash-commands/_payload.json +++ b/articles/quick-tip-rerunning-bash-commands/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":647},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"draft":6,"tags":11,"categories":14,"excerpt":16,"body":36,"_type":641,"_id":642,"_source":643,"_file":644,"_stem":645,"_extension":646},"/articles/quick-tip-rerunning-bash-commands","articles",false,"","Tip: Re-running Bash Commands","Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use sudo for a command that requires\nroot privileges.","2021-09-22",[12,13],"tip","bash",[15],"tips",{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24,27,34],{"type":25,"value":26},"text","Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use ",{"type":20,"tag":28,"props":29,"children":31},"code",{"className":30},[],[32],{"type":25,"value":33},"sudo",{"type":25,"value":35}," for a command that requires\nroot privileges.",{"type":17,"children":37,"toc":639},[38,48,134,161,245,250,279,381,409,463,476,511,530,572,599,603,633],{"type":20,"tag":21,"props":39,"children":40},{},[41,42,47],{"type":25,"value":26},{"type":20,"tag":28,"props":43,"children":45},{"className":44},[],[46],{"type":25,"value":33},{"type":25,"value":35},{"type":20,"tag":49,"props":50,"children":53},"pre",{"className":51,"code":52,"language":13,"meta":7,"style":7},"language-bash shiki shiki-themes github-light","$ pacman -Syu\nerror: you cannot perform this operation unless you are root.\n",[54],{"type":20,"tag":28,"props":55,"children":56},{"__ignoreMap":7},[57,81],{"type":20,"tag":58,"props":59,"children":62},"span",{"class":60,"line":61},"line",1,[63,69,75],{"type":20,"tag":58,"props":64,"children":66},{"style":65},"--shiki-default:#6F42C1",[67],{"type":25,"value":68},"$",{"type":20,"tag":58,"props":70,"children":72},{"style":71},"--shiki-default:#032F62",[73],{"type":25,"value":74}," pacman",{"type":20,"tag":58,"props":76,"children":78},{"style":77},"--shiki-default:#005CC5",[79],{"type":25,"value":80}," -Syu\n",{"type":20,"tag":58,"props":82,"children":84},{"class":60,"line":83},2,[85,90,95,100,105,110,115,120,124,129],{"type":20,"tag":58,"props":86,"children":87},{"style":65},[88],{"type":25,"value":89},"error:",{"type":20,"tag":58,"props":91,"children":92},{"style":71},[93],{"type":25,"value":94}," you",{"type":20,"tag":58,"props":96,"children":97},{"style":71},[98],{"type":25,"value":99}," cannot",{"type":20,"tag":58,"props":101,"children":102},{"style":71},[103],{"type":25,"value":104}," perform",{"type":20,"tag":58,"props":106,"children":107},{"style":71},[108],{"type":25,"value":109}," this",{"type":20,"tag":58,"props":111,"children":112},{"style":71},[113],{"type":25,"value":114}," operation",{"type":20,"tag":58,"props":116,"children":117},{"style":71},[118],{"type":25,"value":119}," unless",{"type":20,"tag":58,"props":121,"children":122},{"style":71},[123],{"type":25,"value":94},{"type":20,"tag":58,"props":125,"children":126},{"style":71},[127],{"type":25,"value":128}," are",{"type":20,"tag":58,"props":130,"children":131},{"style":71},[132],{"type":25,"value":133}," root.\n",{"type":20,"tag":21,"props":135,"children":136},{},[137,139,145,147,152,154,159],{"type":25,"value":138},"Well, I have good news for you -- you can easily re-issue a command with the\n",{"type":20,"tag":28,"props":140,"children":142},{"className":141},[],[143],{"type":25,"value":144},"!!",{"type":25,"value":146}," designator! Simply type ",{"type":20,"tag":28,"props":148,"children":150},{"className":149},[],[151],{"type":25,"value":33},{"type":25,"value":153}," followed by ",{"type":20,"tag":28,"props":155,"children":157},{"className":156},[],[158],{"type":25,"value":144},{"type":25,"value":160}," and you're good to go.",{"type":20,"tag":49,"props":162,"children":164},{"className":51,"code":163,"language":13,"meta":7,"style":7},"$ sudo !!\nsudo pacman -Syu\n[sudo] password for colton:\n:: Synchronizing package databases...\n",[165],{"type":20,"tag":28,"props":166,"children":167},{"__ignoreMap":7},[168,185,200,221],{"type":20,"tag":58,"props":169,"children":170},{"class":60,"line":61},[171,175,180],{"type":20,"tag":58,"props":172,"children":173},{"style":65},[174],{"type":25,"value":68},{"type":20,"tag":58,"props":176,"children":177},{"style":71},[178],{"type":25,"value":179}," sudo",{"type":20,"tag":58,"props":181,"children":182},{"style":71},[183],{"type":25,"value":184}," !!\n",{"type":20,"tag":58,"props":186,"children":187},{"class":60,"line":83},[188,192,196],{"type":20,"tag":58,"props":189,"children":190},{"style":65},[191],{"type":25,"value":33},{"type":20,"tag":58,"props":193,"children":194},{"style":71},[195],{"type":25,"value":74},{"type":20,"tag":58,"props":197,"children":198},{"style":77},[199],{"type":25,"value":80},{"type":20,"tag":58,"props":201,"children":203},{"class":60,"line":202},3,[204,210,216],{"type":20,"tag":58,"props":205,"children":207},{"style":206},"--shiki-default:#24292E",[208],{"type":25,"value":209},"[sudo] password ",{"type":20,"tag":58,"props":211,"children":213},{"style":212},"--shiki-default:#D73A49",[214],{"type":25,"value":215},"for",{"type":20,"tag":58,"props":217,"children":218},{"style":206},[219],{"type":25,"value":220}," colton:\n",{"type":20,"tag":58,"props":222,"children":224},{"class":60,"line":223},4,[225,230,235,240],{"type":20,"tag":58,"props":226,"children":227},{"style":77},[228],{"type":25,"value":229},"::",{"type":20,"tag":58,"props":231,"children":232},{"style":71},[233],{"type":25,"value":234}," Synchronizing",{"type":20,"tag":58,"props":236,"children":237},{"style":71},[238],{"type":25,"value":239}," package",{"type":20,"tag":58,"props":241,"children":242},{"style":71},[243],{"type":25,"value":244}," databases...\n",{"type":20,"tag":21,"props":246,"children":247},{},[248],{"type":25,"value":249},"--",{"type":20,"tag":21,"props":251,"children":252},{},[253,255,261,263,269,271,277],{"type":25,"value":254},"Commands that are prefixed with a bang, ",{"type":20,"tag":28,"props":256,"children":258},{"className":257},[],[259],{"type":25,"value":260},"!",{"type":25,"value":262},", are considered ",{"type":20,"tag":264,"props":265,"children":266},"em",{},[267],{"type":25,"value":268},"Event\nDesignators,",{"type":25,"value":270}," and are references to your command-line history. You can take a\nlook at your history with the ",{"type":20,"tag":28,"props":272,"children":274},{"className":273},[],[275],{"type":25,"value":276},"history",{"type":25,"value":278}," command.",{"type":20,"tag":49,"props":280,"children":282},{"className":51,"code":281,"language":13,"meta":7,"style":7},"$ history\n 1021 touch hello_world.txt\n 1022 ls\n 1023 echo \"Here we go again!\"\n 1024 find . -name *.py\n",[283],{"type":20,"tag":28,"props":284,"children":285},{"__ignoreMap":7},[286,298,316,329,347],{"type":20,"tag":58,"props":287,"children":288},{"class":60,"line":61},[289,293],{"type":20,"tag":58,"props":290,"children":291},{"style":65},[292],{"type":25,"value":68},{"type":20,"tag":58,"props":294,"children":295},{"style":71},[296],{"type":25,"value":297}," history\n",{"type":20,"tag":58,"props":299,"children":300},{"class":60,"line":83},[301,306,311],{"type":20,"tag":58,"props":302,"children":303},{"style":65},[304],{"type":25,"value":305}," 1021",{"type":20,"tag":58,"props":307,"children":308},{"style":71},[309],{"type":25,"value":310}," touch",{"type":20,"tag":58,"props":312,"children":313},{"style":71},[314],{"type":25,"value":315}," hello_world.txt\n",{"type":20,"tag":58,"props":317,"children":318},{"class":60,"line":202},[319,324],{"type":20,"tag":58,"props":320,"children":321},{"style":65},[322],{"type":25,"value":323}," 1022",{"type":20,"tag":58,"props":325,"children":326},{"style":71},[327],{"type":25,"value":328}," ls\n",{"type":20,"tag":58,"props":330,"children":331},{"class":60,"line":223},[332,337,342],{"type":20,"tag":58,"props":333,"children":334},{"style":65},[335],{"type":25,"value":336}," 1023",{"type":20,"tag":58,"props":338,"children":339},{"style":71},[340],{"type":25,"value":341}," echo",{"type":20,"tag":58,"props":343,"children":344},{"style":71},[345],{"type":25,"value":346}," \"Here we go again!\"\n",{"type":20,"tag":58,"props":348,"children":350},{"class":60,"line":349},5,[351,356,361,366,371,376],{"type":20,"tag":58,"props":352,"children":353},{"style":65},[354],{"type":25,"value":355}," 1024",{"type":20,"tag":58,"props":357,"children":358},{"style":71},[359],{"type":25,"value":360}," find",{"type":20,"tag":58,"props":362,"children":363},{"style":71},[364],{"type":25,"value":365}," .",{"type":20,"tag":58,"props":367,"children":368},{"style":77},[369],{"type":25,"value":370}," -name",{"type":20,"tag":58,"props":372,"children":373},{"style":77},[374],{"type":25,"value":375}," *",{"type":20,"tag":58,"props":377,"children":378},{"style":71},[379],{"type":25,"value":380},".py\n",{"type":20,"tag":21,"props":382,"children":383},{},[384,386,391,393,399,401,407],{"type":25,"value":385},"There are many ways to use ",{"type":20,"tag":28,"props":387,"children":389},{"className":388},[],[390],{"type":25,"value":260},{"type":25,"value":392}," in your shell. For example, if you wanted to\nre-issue a specific command in your history, you could use ",{"type":20,"tag":28,"props":394,"children":396},{"className":395},[],[397],{"type":25,"value":398},"!n",{"type":25,"value":400}," where ",{"type":20,"tag":28,"props":402,"children":404},{"className":403},[],[405],{"type":25,"value":406},"n",{"type":25,"value":408}," is\nthe number next to the command in your history.",{"type":20,"tag":49,"props":410,"children":412},{"className":51,"code":411,"language":13,"meta":7,"style":7},"$ !1023\necho \"Here we go again!\"\nHere we go again!\n",[413],{"type":20,"tag":28,"props":414,"children":415},{"__ignoreMap":7},[416,428,440],{"type":20,"tag":58,"props":417,"children":418},{"class":60,"line":61},[419,423],{"type":20,"tag":58,"props":420,"children":421},{"style":65},[422],{"type":25,"value":68},{"type":20,"tag":58,"props":424,"children":425},{"style":71},[426],{"type":25,"value":427}," !1023\n",{"type":20,"tag":58,"props":429,"children":430},{"class":60,"line":83},[431,436],{"type":20,"tag":58,"props":432,"children":433},{"style":77},[434],{"type":25,"value":435},"echo",{"type":20,"tag":58,"props":437,"children":438},{"style":71},[439],{"type":25,"value":346},{"type":20,"tag":58,"props":441,"children":442},{"class":60,"line":202},[443,448,453,458],{"type":20,"tag":58,"props":444,"children":445},{"style":65},[446],{"type":25,"value":447},"Here",{"type":20,"tag":58,"props":449,"children":450},{"style":71},[451],{"type":25,"value":452}," we",{"type":20,"tag":58,"props":454,"children":455},{"style":71},[456],{"type":25,"value":457}," go",{"type":20,"tag":58,"props":459,"children":460},{"style":71},[461],{"type":25,"value":462}," again!\n",{"type":20,"tag":21,"props":464,"children":465},{},[466,468,474],{"type":25,"value":467},"Or... if you wanted to run the command you issued 4-commands ago, you can use\n",{"type":20,"tag":28,"props":469,"children":471},{"className":470},[],[472],{"type":25,"value":473},"!-4",{"type":25,"value":475},".",{"type":20,"tag":49,"props":477,"children":479},{"className":51,"code":478,"language":13,"meta":7,"style":7},"$ !-4\nls\nhello_world.txt\n",[480],{"type":20,"tag":28,"props":481,"children":482},{"__ignoreMap":7},[483,495,503],{"type":20,"tag":58,"props":484,"children":485},{"class":60,"line":61},[486,490],{"type":20,"tag":58,"props":487,"children":488},{"style":65},[489],{"type":25,"value":68},{"type":20,"tag":58,"props":491,"children":492},{"style":71},[493],{"type":25,"value":494}," !-4\n",{"type":20,"tag":58,"props":496,"children":497},{"class":60,"line":83},[498],{"type":20,"tag":58,"props":499,"children":500},{"style":65},[501],{"type":25,"value":502},"ls\n",{"type":20,"tag":58,"props":504,"children":505},{"class":60,"line":202},[506],{"type":20,"tag":58,"props":507,"children":508},{"style":65},[509],{"type":25,"value":510},"hello_world.txt\n",{"type":20,"tag":21,"props":512,"children":513},{},[514,516,521,523,529],{"type":25,"value":515},"Or... if you wanted to run the last command that started with the string\n",{"type":20,"tag":264,"props":517,"children":518},{},[519],{"type":25,"value":520},"find",{"type":25,"value":522},", you can use ",{"type":20,"tag":28,"props":524,"children":526},{"className":525},[],[527],{"type":25,"value":528},"!find",{"type":25,"value":475},{"type":20,"tag":49,"props":531,"children":533},{"className":51,"code":532,"language":13,"meta":7,"style":7},"$ !find\nfind . -name *.py\n",[534],{"type":20,"tag":28,"props":535,"children":536},{"__ignoreMap":7},[537,549],{"type":20,"tag":58,"props":538,"children":539},{"class":60,"line":61},[540,544],{"type":20,"tag":58,"props":541,"children":542},{"style":65},[543],{"type":25,"value":68},{"type":20,"tag":58,"props":545,"children":546},{"style":71},[547],{"type":25,"value":548}," !find\n",{"type":20,"tag":58,"props":550,"children":551},{"class":60,"line":83},[552,556,560,564,568],{"type":20,"tag":58,"props":553,"children":554},{"style":65},[555],{"type":25,"value":520},{"type":20,"tag":58,"props":557,"children":558},{"style":71},[559],{"type":25,"value":365},{"type":20,"tag":58,"props":561,"children":562},{"style":77},[563],{"type":25,"value":370},{"type":20,"tag":58,"props":565,"children":566},{"style":77},[567],{"type":25,"value":375},{"type":20,"tag":58,"props":569,"children":570},{"style":71},[571],{"type":25,"value":380},{"type":20,"tag":21,"props":573,"children":574},{},[575,577,582,584,589,591,597],{"type":25,"value":576},"Be sure to check out the ",{"type":20,"tag":264,"props":578,"children":579},{},[580],{"type":25,"value":581},"Event Designators",{"type":25,"value":583}," section of the ",{"type":20,"tag":28,"props":585,"children":587},{"className":586},[],[588],{"type":25,"value":13},{"type":25,"value":590}," ",{"type":20,"tag":28,"props":592,"children":594},{"className":593},[],[595],{"type":25,"value":596},"man",{"type":25,"value":598}," pages\nfor more information!",{"type":20,"tag":21,"props":600,"children":601},{},[602],{"type":25,"value":249},{"type":20,"tag":21,"props":604,"children":605},{},[606,608,617,619,624,626,631],{"type":25,"value":607},"As an aside, for even faster command-line history navigation, be sure to check\nout the excellent ",{"type":20,"tag":609,"props":610,"children":614},"a",{"href":611,"rel":612},"https://github.com/junegunn/fzf",[613],"nofollow",[615],{"type":25,"value":616},"fzf",{"type":25,"value":618}," utility by ",{"type":20,"tag":264,"props":620,"children":621},{},[622],{"type":25,"value":623},"junegunn.",{"type":25,"value":625},"\nOne of the many features of ",{"type":20,"tag":28,"props":627,"children":629},{"className":628},[],[630],{"type":25,"value":616},{"type":25,"value":632}," is browsing and re-issuing commands from your\ncommand-line history with a fuzzy-finder!",{"type":20,"tag":634,"props":635,"children":636},"style",{},[637],{"type":25,"value":638},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":83,"depth":83,"links":640},[],"markdown","content:articles:quick-tip-rerunning-bash-commands.md","content","articles/quick-tip-rerunning-bash-commands.md","articles/quick-tip-rerunning-bash-commands","md",1731781853285] \ No newline at end of file +[{"data":1,"prerenderedAt":647},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"draft":6,"tags":11,"categories":14,"excerpt":16,"body":36,"_type":641,"_id":642,"_source":643,"_file":644,"_stem":645,"_extension":646},"/articles/quick-tip-rerunning-bash-commands","articles",false,"","Tip: Re-running Bash Commands","Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use sudo for a command that requires\nroot privileges.","2021-09-22",[12,13],"tip","bash",[15],"tips",{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24,27,34],{"type":25,"value":26},"text","Do you ever find yourself having to re-run a Bash command? I often find this\nhappening to myself when I neglect to use ",{"type":20,"tag":28,"props":29,"children":31},"code",{"className":30},[],[32],{"type":25,"value":33},"sudo",{"type":25,"value":35}," for a command that requires\nroot privileges.",{"type":17,"children":37,"toc":639},[38,48,134,161,245,250,279,381,409,463,476,511,530,572,599,603,633],{"type":20,"tag":21,"props":39,"children":40},{},[41,42,47],{"type":25,"value":26},{"type":20,"tag":28,"props":43,"children":45},{"className":44},[],[46],{"type":25,"value":33},{"type":25,"value":35},{"type":20,"tag":49,"props":50,"children":53},"pre",{"className":51,"code":52,"language":13,"meta":7,"style":7},"language-bash shiki shiki-themes github-light","$ pacman -Syu\nerror: you cannot perform this operation unless you are root.\n",[54],{"type":20,"tag":28,"props":55,"children":56},{"__ignoreMap":7},[57,81],{"type":20,"tag":58,"props":59,"children":62},"span",{"class":60,"line":61},"line",1,[63,69,75],{"type":20,"tag":58,"props":64,"children":66},{"style":65},"--shiki-default:#6F42C1",[67],{"type":25,"value":68},"$",{"type":20,"tag":58,"props":70,"children":72},{"style":71},"--shiki-default:#032F62",[73],{"type":25,"value":74}," pacman",{"type":20,"tag":58,"props":76,"children":78},{"style":77},"--shiki-default:#005CC5",[79],{"type":25,"value":80}," -Syu\n",{"type":20,"tag":58,"props":82,"children":84},{"class":60,"line":83},2,[85,90,95,100,105,110,115,120,124,129],{"type":20,"tag":58,"props":86,"children":87},{"style":65},[88],{"type":25,"value":89},"error:",{"type":20,"tag":58,"props":91,"children":92},{"style":71},[93],{"type":25,"value":94}," you",{"type":20,"tag":58,"props":96,"children":97},{"style":71},[98],{"type":25,"value":99}," cannot",{"type":20,"tag":58,"props":101,"children":102},{"style":71},[103],{"type":25,"value":104}," perform",{"type":20,"tag":58,"props":106,"children":107},{"style":71},[108],{"type":25,"value":109}," this",{"type":20,"tag":58,"props":111,"children":112},{"style":71},[113],{"type":25,"value":114}," operation",{"type":20,"tag":58,"props":116,"children":117},{"style":71},[118],{"type":25,"value":119}," unless",{"type":20,"tag":58,"props":121,"children":122},{"style":71},[123],{"type":25,"value":94},{"type":20,"tag":58,"props":125,"children":126},{"style":71},[127],{"type":25,"value":128}," are",{"type":20,"tag":58,"props":130,"children":131},{"style":71},[132],{"type":25,"value":133}," root.\n",{"type":20,"tag":21,"props":135,"children":136},{},[137,139,145,147,152,154,159],{"type":25,"value":138},"Well, I have good news for you -- you can easily re-issue a command with the\n",{"type":20,"tag":28,"props":140,"children":142},{"className":141},[],[143],{"type":25,"value":144},"!!",{"type":25,"value":146}," designator! Simply type ",{"type":20,"tag":28,"props":148,"children":150},{"className":149},[],[151],{"type":25,"value":33},{"type":25,"value":153}," followed by ",{"type":20,"tag":28,"props":155,"children":157},{"className":156},[],[158],{"type":25,"value":144},{"type":25,"value":160}," and you're good to go.",{"type":20,"tag":49,"props":162,"children":164},{"className":51,"code":163,"language":13,"meta":7,"style":7},"$ sudo !!\nsudo pacman -Syu\n[sudo] password for colton:\n:: Synchronizing package databases...\n",[165],{"type":20,"tag":28,"props":166,"children":167},{"__ignoreMap":7},[168,185,200,221],{"type":20,"tag":58,"props":169,"children":170},{"class":60,"line":61},[171,175,180],{"type":20,"tag":58,"props":172,"children":173},{"style":65},[174],{"type":25,"value":68},{"type":20,"tag":58,"props":176,"children":177},{"style":71},[178],{"type":25,"value":179}," sudo",{"type":20,"tag":58,"props":181,"children":182},{"style":71},[183],{"type":25,"value":184}," !!\n",{"type":20,"tag":58,"props":186,"children":187},{"class":60,"line":83},[188,192,196],{"type":20,"tag":58,"props":189,"children":190},{"style":65},[191],{"type":25,"value":33},{"type":20,"tag":58,"props":193,"children":194},{"style":71},[195],{"type":25,"value":74},{"type":20,"tag":58,"props":197,"children":198},{"style":77},[199],{"type":25,"value":80},{"type":20,"tag":58,"props":201,"children":203},{"class":60,"line":202},3,[204,210,216],{"type":20,"tag":58,"props":205,"children":207},{"style":206},"--shiki-default:#24292E",[208],{"type":25,"value":209},"[sudo] password ",{"type":20,"tag":58,"props":211,"children":213},{"style":212},"--shiki-default:#D73A49",[214],{"type":25,"value":215},"for",{"type":20,"tag":58,"props":217,"children":218},{"style":206},[219],{"type":25,"value":220}," colton:\n",{"type":20,"tag":58,"props":222,"children":224},{"class":60,"line":223},4,[225,230,235,240],{"type":20,"tag":58,"props":226,"children":227},{"style":77},[228],{"type":25,"value":229},"::",{"type":20,"tag":58,"props":231,"children":232},{"style":71},[233],{"type":25,"value":234}," Synchronizing",{"type":20,"tag":58,"props":236,"children":237},{"style":71},[238],{"type":25,"value":239}," package",{"type":20,"tag":58,"props":241,"children":242},{"style":71},[243],{"type":25,"value":244}," databases...\n",{"type":20,"tag":21,"props":246,"children":247},{},[248],{"type":25,"value":249},"--",{"type":20,"tag":21,"props":251,"children":252},{},[253,255,261,263,269,271,277],{"type":25,"value":254},"Commands that are prefixed with a bang, ",{"type":20,"tag":28,"props":256,"children":258},{"className":257},[],[259],{"type":25,"value":260},"!",{"type":25,"value":262},", are considered ",{"type":20,"tag":264,"props":265,"children":266},"em",{},[267],{"type":25,"value":268},"Event\nDesignators,",{"type":25,"value":270}," and are references to your command-line history. You can take a\nlook at your history with the ",{"type":20,"tag":28,"props":272,"children":274},{"className":273},[],[275],{"type":25,"value":276},"history",{"type":25,"value":278}," command.",{"type":20,"tag":49,"props":280,"children":282},{"className":51,"code":281,"language":13,"meta":7,"style":7},"$ history\n 1021 touch hello_world.txt\n 1022 ls\n 1023 echo \"Here we go again!\"\n 1024 find . -name *.py\n",[283],{"type":20,"tag":28,"props":284,"children":285},{"__ignoreMap":7},[286,298,316,329,347],{"type":20,"tag":58,"props":287,"children":288},{"class":60,"line":61},[289,293],{"type":20,"tag":58,"props":290,"children":291},{"style":65},[292],{"type":25,"value":68},{"type":20,"tag":58,"props":294,"children":295},{"style":71},[296],{"type":25,"value":297}," history\n",{"type":20,"tag":58,"props":299,"children":300},{"class":60,"line":83},[301,306,311],{"type":20,"tag":58,"props":302,"children":303},{"style":65},[304],{"type":25,"value":305}," 1021",{"type":20,"tag":58,"props":307,"children":308},{"style":71},[309],{"type":25,"value":310}," touch",{"type":20,"tag":58,"props":312,"children":313},{"style":71},[314],{"type":25,"value":315}," hello_world.txt\n",{"type":20,"tag":58,"props":317,"children":318},{"class":60,"line":202},[319,324],{"type":20,"tag":58,"props":320,"children":321},{"style":65},[322],{"type":25,"value":323}," 1022",{"type":20,"tag":58,"props":325,"children":326},{"style":71},[327],{"type":25,"value":328}," ls\n",{"type":20,"tag":58,"props":330,"children":331},{"class":60,"line":223},[332,337,342],{"type":20,"tag":58,"props":333,"children":334},{"style":65},[335],{"type":25,"value":336}," 1023",{"type":20,"tag":58,"props":338,"children":339},{"style":71},[340],{"type":25,"value":341}," echo",{"type":20,"tag":58,"props":343,"children":344},{"style":71},[345],{"type":25,"value":346}," \"Here we go again!\"\n",{"type":20,"tag":58,"props":348,"children":350},{"class":60,"line":349},5,[351,356,361,366,371,376],{"type":20,"tag":58,"props":352,"children":353},{"style":65},[354],{"type":25,"value":355}," 1024",{"type":20,"tag":58,"props":357,"children":358},{"style":71},[359],{"type":25,"value":360}," find",{"type":20,"tag":58,"props":362,"children":363},{"style":71},[364],{"type":25,"value":365}," .",{"type":20,"tag":58,"props":367,"children":368},{"style":77},[369],{"type":25,"value":370}," -name",{"type":20,"tag":58,"props":372,"children":373},{"style":77},[374],{"type":25,"value":375}," *",{"type":20,"tag":58,"props":377,"children":378},{"style":71},[379],{"type":25,"value":380},".py\n",{"type":20,"tag":21,"props":382,"children":383},{},[384,386,391,393,399,401,407],{"type":25,"value":385},"There are many ways to use ",{"type":20,"tag":28,"props":387,"children":389},{"className":388},[],[390],{"type":25,"value":260},{"type":25,"value":392}," in your shell. For example, if you wanted to\nre-issue a specific command in your history, you could use ",{"type":20,"tag":28,"props":394,"children":396},{"className":395},[],[397],{"type":25,"value":398},"!n",{"type":25,"value":400}," where ",{"type":20,"tag":28,"props":402,"children":404},{"className":403},[],[405],{"type":25,"value":406},"n",{"type":25,"value":408}," is\nthe number next to the command in your history.",{"type":20,"tag":49,"props":410,"children":412},{"className":51,"code":411,"language":13,"meta":7,"style":7},"$ !1023\necho \"Here we go again!\"\nHere we go again!\n",[413],{"type":20,"tag":28,"props":414,"children":415},{"__ignoreMap":7},[416,428,440],{"type":20,"tag":58,"props":417,"children":418},{"class":60,"line":61},[419,423],{"type":20,"tag":58,"props":420,"children":421},{"style":65},[422],{"type":25,"value":68},{"type":20,"tag":58,"props":424,"children":425},{"style":71},[426],{"type":25,"value":427}," !1023\n",{"type":20,"tag":58,"props":429,"children":430},{"class":60,"line":83},[431,436],{"type":20,"tag":58,"props":432,"children":433},{"style":77},[434],{"type":25,"value":435},"echo",{"type":20,"tag":58,"props":437,"children":438},{"style":71},[439],{"type":25,"value":346},{"type":20,"tag":58,"props":441,"children":442},{"class":60,"line":202},[443,448,453,458],{"type":20,"tag":58,"props":444,"children":445},{"style":65},[446],{"type":25,"value":447},"Here",{"type":20,"tag":58,"props":449,"children":450},{"style":71},[451],{"type":25,"value":452}," we",{"type":20,"tag":58,"props":454,"children":455},{"style":71},[456],{"type":25,"value":457}," go",{"type":20,"tag":58,"props":459,"children":460},{"style":71},[461],{"type":25,"value":462}," again!\n",{"type":20,"tag":21,"props":464,"children":465},{},[466,468,474],{"type":25,"value":467},"Or... if you wanted to run the command you issued 4-commands ago, you can use\n",{"type":20,"tag":28,"props":469,"children":471},{"className":470},[],[472],{"type":25,"value":473},"!-4",{"type":25,"value":475},".",{"type":20,"tag":49,"props":477,"children":479},{"className":51,"code":478,"language":13,"meta":7,"style":7},"$ !-4\nls\nhello_world.txt\n",[480],{"type":20,"tag":28,"props":481,"children":482},{"__ignoreMap":7},[483,495,503],{"type":20,"tag":58,"props":484,"children":485},{"class":60,"line":61},[486,490],{"type":20,"tag":58,"props":487,"children":488},{"style":65},[489],{"type":25,"value":68},{"type":20,"tag":58,"props":491,"children":492},{"style":71},[493],{"type":25,"value":494}," !-4\n",{"type":20,"tag":58,"props":496,"children":497},{"class":60,"line":83},[498],{"type":20,"tag":58,"props":499,"children":500},{"style":65},[501],{"type":25,"value":502},"ls\n",{"type":20,"tag":58,"props":504,"children":505},{"class":60,"line":202},[506],{"type":20,"tag":58,"props":507,"children":508},{"style":65},[509],{"type":25,"value":510},"hello_world.txt\n",{"type":20,"tag":21,"props":512,"children":513},{},[514,516,521,523,529],{"type":25,"value":515},"Or... if you wanted to run the last command that started with the string\n",{"type":20,"tag":264,"props":517,"children":518},{},[519],{"type":25,"value":520},"find",{"type":25,"value":522},", you can use ",{"type":20,"tag":28,"props":524,"children":526},{"className":525},[],[527],{"type":25,"value":528},"!find",{"type":25,"value":475},{"type":20,"tag":49,"props":531,"children":533},{"className":51,"code":532,"language":13,"meta":7,"style":7},"$ !find\nfind . -name *.py\n",[534],{"type":20,"tag":28,"props":535,"children":536},{"__ignoreMap":7},[537,549],{"type":20,"tag":58,"props":538,"children":539},{"class":60,"line":61},[540,544],{"type":20,"tag":58,"props":541,"children":542},{"style":65},[543],{"type":25,"value":68},{"type":20,"tag":58,"props":545,"children":546},{"style":71},[547],{"type":25,"value":548}," !find\n",{"type":20,"tag":58,"props":550,"children":551},{"class":60,"line":83},[552,556,560,564,568],{"type":20,"tag":58,"props":553,"children":554},{"style":65},[555],{"type":25,"value":520},{"type":20,"tag":58,"props":557,"children":558},{"style":71},[559],{"type":25,"value":365},{"type":20,"tag":58,"props":561,"children":562},{"style":77},[563],{"type":25,"value":370},{"type":20,"tag":58,"props":565,"children":566},{"style":77},[567],{"type":25,"value":375},{"type":20,"tag":58,"props":569,"children":570},{"style":71},[571],{"type":25,"value":380},{"type":20,"tag":21,"props":573,"children":574},{},[575,577,582,584,589,591,597],{"type":25,"value":576},"Be sure to check out the ",{"type":20,"tag":264,"props":578,"children":579},{},[580],{"type":25,"value":581},"Event Designators",{"type":25,"value":583}," section of the ",{"type":20,"tag":28,"props":585,"children":587},{"className":586},[],[588],{"type":25,"value":13},{"type":25,"value":590}," ",{"type":20,"tag":28,"props":592,"children":594},{"className":593},[],[595],{"type":25,"value":596},"man",{"type":25,"value":598}," pages\nfor more information!",{"type":20,"tag":21,"props":600,"children":601},{},[602],{"type":25,"value":249},{"type":20,"tag":21,"props":604,"children":605},{},[606,608,617,619,624,626,631],{"type":25,"value":607},"As an aside, for even faster command-line history navigation, be sure to check\nout the excellent ",{"type":20,"tag":609,"props":610,"children":614},"a",{"href":611,"rel":612},"https://github.com/junegunn/fzf",[613],"nofollow",[615],{"type":25,"value":616},"fzf",{"type":25,"value":618}," utility by ",{"type":20,"tag":264,"props":620,"children":621},{},[622],{"type":25,"value":623},"junegunn.",{"type":25,"value":625},"\nOne of the many features of ",{"type":20,"tag":28,"props":627,"children":629},{"className":628},[],[630],{"type":25,"value":616},{"type":25,"value":632}," is browsing and re-issuing commands from your\ncommand-line history with a fuzzy-finder!",{"type":20,"tag":634,"props":635,"children":636},"style",{},[637],{"type":25,"value":638},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":83,"depth":83,"links":640},[],"markdown","content:articles:quick-tip-rerunning-bash-commands.md","content","articles/quick-tip-rerunning-bash-commands.md","articles/quick-tip-rerunning-bash-commands","md",1731782033537] \ No newline at end of file diff --git a/articles/quick-tip-rerunning-bash-commands/index.html b/articles/quick-tip-rerunning-bash-commands/index.html index 7861cbeb..cb9e6758 100644 --- a/articles/quick-tip-rerunning-bash-commands/index.html +++ b/articles/quick-tip-rerunning-bash-commands/index.html @@ -5,32 +5,32 @@ - - - - - + + + + + - + - - - - - - - - - - - - - + + + + + + + + + + + + + -
        Category

        Tip: Re-running Bash Commands

        Do you ever find yourself having to re-run a Bash command? I often find this +

        Category

        Tip: Re-running Bash Commands

        Do you ever find yourself having to re-run a Bash command? I often find this happening to myself when I neglect to use sudo for a command that requires root privileges.

        $ pacman -Syu
         error: you cannot perform this operation unless you are root.
        @@ -62,5 +62,5 @@
         for more information!

        --

        As an aside, for even faster command-line history navigation, be sure to check out the excellent fzf utility by junegunn. One of the many features of fzf is browsing and re-issuing commands from your -command-line history with a fuzzy-finder!

        - \ No newline at end of file +command-line history with a fuzzy-finder!

        + \ No newline at end of file diff --git a/articles/reset-ipmi-password-from-host-os/_payload.json b/articles/reset-ipmi-password-from-host-os/_payload.json index b2bca4c8..a7360b02 100644 --- a/articles/reset-ipmi-password-from-host-os/_payload.json +++ b/articles/reset-ipmi-password-from-host-os/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":203},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"tags":11,"categories":15,"excerpt":16,"body":26,"_type":197,"_id":198,"_source":199,"_file":200,"_stem":201,"_extension":202},"/articles/reset-ipmi-password-from-host-os","articles",false,"","Reset IPMI Credentials from the Host OS","If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these\ninstructions will guide your through the process of resetting the credentials to their default values from the host\noperating system.","2021-12-27",[12,13,14],"homelab","supermicro","truenas",[12],{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24],{"type":25,"value":9},"text",{"type":17,"children":27,"toc":194},[28,32,54,67,72,81,100,108,113,118,123,131],{"type":20,"tag":21,"props":29,"children":30},{},[31],{"type":25,"value":9},{"type":20,"tag":21,"props":33,"children":34},{},[35,37,44,46,52],{"type":25,"value":36},"If you are using an operating system like TrueNAS -- good news! It's possible to reset the IPMI password directly from\nthe web interface. This is done by navigating to ",{"type":20,"tag":38,"props":39,"children":41},"code",{"className":40},[],[42],{"type":25,"value":43},"Network > IPMI",{"type":25,"value":45},", and simply entering a new value in the ",{"type":20,"tag":47,"props":48,"children":49},"em",{},[50],{"type":25,"value":51},"IPMI\nPassword Reset",{"type":25,"value":53}," field.",{"type":20,"tag":21,"props":55,"children":56},{},[57,59,65],{"type":25,"value":58},"If you are using some other OS that doesn't have this feature, you can achieve similar results by using the ",{"type":20,"tag":38,"props":60,"children":62},{"className":61},[],[63],{"type":25,"value":64},"ipmitool",{"type":25,"value":66},"\ncommand-line utility.",{"type":20,"tag":21,"props":68,"children":69},{},[70],{"type":25,"value":71},"First, you'll want to determine the user ID associated with the user for whom you'd like to reset the password.",{"type":20,"tag":73,"props":74,"children":76},"pre",{"code":75},"# ipmitool user list\nID Name Callin Link Auth IPMI Msg Channel Priv Limit\n1 true false false Unknown (0x00)\n2 ADMIN true false false Unknown (0x00)\n",[77],{"type":20,"tag":38,"props":78,"children":79},{"__ignoreMap":7},[80],{"type":25,"value":75},{"type":20,"tag":21,"props":82,"children":83},{},[84,86,91,93,98],{"type":25,"value":85},"In this case, we will be resetting the password for ",{"type":20,"tag":47,"props":87,"children":88},{},[89],{"type":25,"value":90},"ADMIN",{"type":25,"value":92}," who has a user ID of ",{"type":20,"tag":47,"props":94,"children":95},{},[96],{"type":25,"value":97},"2",{"type":25,"value":99},". Then we'll assign the new\npassword like so:",{"type":20,"tag":73,"props":101,"children":103},{"code":102},"# ipmitool user set password 2 \u003Cpassword>\n",[104],{"type":20,"tag":38,"props":105,"children":106},{"__ignoreMap":7},[107],{"type":25,"value":102},{"type":20,"tag":21,"props":109,"children":110},{},[111],{"type":25,"value":112},"And you should be good to go!",{"type":20,"tag":21,"props":114,"children":115},{},[116],{"type":25,"value":117},"...",{"type":20,"tag":21,"props":119,"children":120},{},[121],{"type":25,"value":122},"Alternatively, if you'd like to factory reset the baseboard management controller (BMC), which will reset the IPMI\ncredentials to their default value, you can issue the following command.",{"type":20,"tag":73,"props":124,"children":126},{"code":125},"# ipmitool raw 0x3c 0x40\n",[127],{"type":20,"tag":38,"props":128,"children":129},{"__ignoreMap":7},[130],{"type":25,"value":125},{"type":20,"tag":21,"props":132,"children":133},{},[134,136,142,144,150,152,158,159,165,167,176,178,183,185,192],{"type":25,"value":135},"Where ",{"type":20,"tag":38,"props":137,"children":139},{"className":138},[],[140],{"type":25,"value":141},"0x3c",{"type":25,"value":143}," is the ",{"type":20,"tag":38,"props":145,"children":147},{"className":146},[],[148],{"type":25,"value":149},"\u003Cnetfn>",{"type":25,"value":151}," argument, a.k.a. the network function code that defines the functional routing for\nmessages, and ",{"type":20,"tag":38,"props":153,"children":155},{"className":154},[],[156],{"type":25,"value":157},"0x40",{"type":25,"value":143},{"type":20,"tag":38,"props":160,"children":162},{"className":161},[],[163],{"type":25,"value":164},"\u003Ccmd>",{"type":25,"value":166},". According to section 5.1 of the ",{"type":20,"tag":168,"props":169,"children":173},"a",{"href":170,"rel":171},"https://www.intel.com/content/dam/www/public/us/en/documents/product-briefs/ipmi-second-gen-interface-spec-v2-rev1-1.pdf",[172],"nofollow",[174],{"type":25,"value":175},"IPMI interface\nspecification",{"type":25,"value":177},",\n",{"type":20,"tag":47,"props":179,"children":180},{},[181],{"type":25,"value":182},"netfn",{"type":25,"value":184}," codes ranging from 0x30 to 0x3F are reserved for vendor specific functions. I searched around for some\nSupermicro references on these vendor specific network functions without much luck other than various ",{"type":20,"tag":168,"props":186,"children":189},{"href":187,"rel":188},"https://www.supermicro.com/support/faqs/faq.cfm?faq=15448",[172],[190],{"type":25,"value":191},"support\nresponses",{"type":25,"value":193}," on how to reset a device. Bummer!",{"title":7,"searchDepth":195,"depth":195,"links":196},2,[],"markdown","content:articles:reset-ipmi-password-from-host-os.md","content","articles/reset-ipmi-password-from-host-os.md","articles/reset-ipmi-password-from-host-os","md",1731781853281] \ No newline at end of file +[{"data":1,"prerenderedAt":203},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"tags":11,"categories":15,"excerpt":16,"body":26,"_type":197,"_id":198,"_source":199,"_file":200,"_stem":201,"_extension":202},"/articles/reset-ipmi-password-from-host-os","articles",false,"","Reset IPMI Credentials from the Host OS","If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these\ninstructions will guide your through the process of resetting the credentials to their default values from the host\noperating system.","2021-12-27",[12,13,14],"homelab","supermicro","truenas",[12],{"type":17,"children":18},"root",[19],{"type":20,"tag":21,"props":22,"children":23},"element","p",{},[24],{"type":25,"value":9},"text",{"type":17,"children":27,"toc":194},[28,32,54,67,72,81,100,108,113,118,123,131],{"type":20,"tag":21,"props":29,"children":30},{},[31],{"type":25,"value":9},{"type":20,"tag":21,"props":33,"children":34},{},[35,37,44,46,52],{"type":25,"value":36},"If you are using an operating system like TrueNAS -- good news! It's possible to reset the IPMI password directly from\nthe web interface. This is done by navigating to ",{"type":20,"tag":38,"props":39,"children":41},"code",{"className":40},[],[42],{"type":25,"value":43},"Network > IPMI",{"type":25,"value":45},", and simply entering a new value in the ",{"type":20,"tag":47,"props":48,"children":49},"em",{},[50],{"type":25,"value":51},"IPMI\nPassword Reset",{"type":25,"value":53}," field.",{"type":20,"tag":21,"props":55,"children":56},{},[57,59,65],{"type":25,"value":58},"If you are using some other OS that doesn't have this feature, you can achieve similar results by using the ",{"type":20,"tag":38,"props":60,"children":62},{"className":61},[],[63],{"type":25,"value":64},"ipmitool",{"type":25,"value":66},"\ncommand-line utility.",{"type":20,"tag":21,"props":68,"children":69},{},[70],{"type":25,"value":71},"First, you'll want to determine the user ID associated with the user for whom you'd like to reset the password.",{"type":20,"tag":73,"props":74,"children":76},"pre",{"code":75},"# ipmitool user list\nID Name Callin Link Auth IPMI Msg Channel Priv Limit\n1 true false false Unknown (0x00)\n2 ADMIN true false false Unknown (0x00)\n",[77],{"type":20,"tag":38,"props":78,"children":79},{"__ignoreMap":7},[80],{"type":25,"value":75},{"type":20,"tag":21,"props":82,"children":83},{},[84,86,91,93,98],{"type":25,"value":85},"In this case, we will be resetting the password for ",{"type":20,"tag":47,"props":87,"children":88},{},[89],{"type":25,"value":90},"ADMIN",{"type":25,"value":92}," who has a user ID of ",{"type":20,"tag":47,"props":94,"children":95},{},[96],{"type":25,"value":97},"2",{"type":25,"value":99},". Then we'll assign the new\npassword like so:",{"type":20,"tag":73,"props":101,"children":103},{"code":102},"# ipmitool user set password 2 \u003Cpassword>\n",[104],{"type":20,"tag":38,"props":105,"children":106},{"__ignoreMap":7},[107],{"type":25,"value":102},{"type":20,"tag":21,"props":109,"children":110},{},[111],{"type":25,"value":112},"And you should be good to go!",{"type":20,"tag":21,"props":114,"children":115},{},[116],{"type":25,"value":117},"...",{"type":20,"tag":21,"props":119,"children":120},{},[121],{"type":25,"value":122},"Alternatively, if you'd like to factory reset the baseboard management controller (BMC), which will reset the IPMI\ncredentials to their default value, you can issue the following command.",{"type":20,"tag":73,"props":124,"children":126},{"code":125},"# ipmitool raw 0x3c 0x40\n",[127],{"type":20,"tag":38,"props":128,"children":129},{"__ignoreMap":7},[130],{"type":25,"value":125},{"type":20,"tag":21,"props":132,"children":133},{},[134,136,142,144,150,152,158,159,165,167,176,178,183,185,192],{"type":25,"value":135},"Where ",{"type":20,"tag":38,"props":137,"children":139},{"className":138},[],[140],{"type":25,"value":141},"0x3c",{"type":25,"value":143}," is the ",{"type":20,"tag":38,"props":145,"children":147},{"className":146},[],[148],{"type":25,"value":149},"\u003Cnetfn>",{"type":25,"value":151}," argument, a.k.a. the network function code that defines the functional routing for\nmessages, and ",{"type":20,"tag":38,"props":153,"children":155},{"className":154},[],[156],{"type":25,"value":157},"0x40",{"type":25,"value":143},{"type":20,"tag":38,"props":160,"children":162},{"className":161},[],[163],{"type":25,"value":164},"\u003Ccmd>",{"type":25,"value":166},". According to section 5.1 of the ",{"type":20,"tag":168,"props":169,"children":173},"a",{"href":170,"rel":171},"https://www.intel.com/content/dam/www/public/us/en/documents/product-briefs/ipmi-second-gen-interface-spec-v2-rev1-1.pdf",[172],"nofollow",[174],{"type":25,"value":175},"IPMI interface\nspecification",{"type":25,"value":177},",\n",{"type":20,"tag":47,"props":179,"children":180},{},[181],{"type":25,"value":182},"netfn",{"type":25,"value":184}," codes ranging from 0x30 to 0x3F are reserved for vendor specific functions. I searched around for some\nSupermicro references on these vendor specific network functions without much luck other than various ",{"type":20,"tag":168,"props":186,"children":189},{"href":187,"rel":188},"https://www.supermicro.com/support/faqs/faq.cfm?faq=15448",[172],[190],{"type":25,"value":191},"support\nresponses",{"type":25,"value":193}," on how to reset a device. Bummer!",{"title":7,"searchDepth":195,"depth":195,"links":196},2,[],"markdown","content:articles:reset-ipmi-password-from-host-os.md","content","articles/reset-ipmi-password-from-host-os.md","articles/reset-ipmi-password-from-host-os","md",1731782033532] \ No newline at end of file diff --git a/articles/reset-ipmi-password-from-host-os/index.html b/articles/reset-ipmi-password-from-host-os/index.html index 8f689b01..7cc87027 100644 --- a/articles/reset-ipmi-password-from-host-os/index.html +++ b/articles/reset-ipmi-password-from-host-os/index.html @@ -5,32 +5,32 @@ - - - - - + + + + + - + - - - - - - - - - - - - - + + + + + + + + + + + + + -

        Reset IPMI Credentials from the Host OS

        If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these +

        Reset IPMI Credentials from the Host OS

        If you ever find yourself locked out of the Intelligent Platform Management Interface (IPMI) of a server, these instructions will guide your through the process of resetting the credentials to their default values from the host operating system.

        If you are using an operating system like TrueNAS -- good news! It's possible to reset the IPMI password directly from the web interface. This is done by navigating to Network > IPMI, and simply entering a new value in the IPMI @@ -48,5 +48,5 @@ specification, netfn codes ranging from 0x30 to 0x3F are reserved for vendor specific functions. I searched around for some Supermicro references on these vendor specific network functions without much luck other than various support -responses on how to reset a device. Bummer!

        - \ No newline at end of file +responses on how to reset a device. Bummer!

        + \ No newline at end of file diff --git a/articles/ssh-ed25519-sk-yubikey/_payload.json b/articles/ssh-ed25519-sk-yubikey/_payload.json index f4621d98..4532aad3 100644 --- a/articles/ssh-ed25519-sk-yubikey/_payload.json +++ b/articles/ssh-ed25519-sk-yubikey/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":517},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"draft":6,"date":10,"tags":11,"categories":14,"excerpt":17,"body":46,"_type":511,"_id":512,"_source":513,"_file":514,"_stem":515,"_extension":516},"/articles/ssh-ed25519-sk-yubikey","articles",false,"","Configuring a YubiKey for use with OpenSSH","YubiKey's are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" ed25519-sk key type that supports FIDO compliant hardware keys.","2024-06-09",[12,13],"unix","configurations",[15,16],"tooling","tips",{"type":18,"children":19},"root",[20],{"type":21,"tag":22,"props":23,"children":24},"element","p",{},[25,35,37,44],{"type":21,"tag":26,"props":27,"children":31},"a",{"href":28,"rel":29},"https://www.yubico.com/",[30],"nofollow",[32],{"type":33,"value":34},"text","YubiKey's",{"type":33,"value":36}," are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" ",{"type":21,"tag":38,"props":39,"children":41},"code",{"className":40},[],[42],{"type":33,"value":43},"ed25519-sk",{"type":33,"value":45}," key type that supports FIDO compliant hardware keys.",{"type":18,"children":47,"toc":509},[48,63,77,86,100,143,148,192,205,238,243,376,389,484,503],{"type":21,"tag":22,"props":49,"children":50},{},[51,56,57,62],{"type":21,"tag":26,"props":52,"children":54},{"href":28,"rel":53},[30],[55],{"type":33,"value":34},{"type":33,"value":36},{"type":21,"tag":38,"props":58,"children":60},{"className":59},[],[61],{"type":33,"value":43},{"type":33,"value":45},{"type":21,"tag":22,"props":64,"children":65},{},[66,68,75],{"type":33,"value":67},"In ",{"type":21,"tag":26,"props":69,"children":72},{"href":70,"rel":71},"https://www.openssh.com/txt/release-8.2",[30],[73],{"type":33,"value":74},"release 8.2 of OpenSSH",{"type":33,"value":76}," support for FIDO devices was added with public key types \"ecdsa-sk\" and \"ed25519-sk\" (-sk standing for \"security key\"). This key type is supported by YubiKey's with firmware version 5.2.3 or higher.",{"type":21,"tag":78,"props":79,"children":80},"blockquote",{},[81],{"type":21,"tag":22,"props":82,"children":83},{},[84],{"type":33,"value":85},"This release adds support for FIDO/U2F hardware authenticators to OpenSSH. U2F/FIDO are open standards for inexpensive two-factor authentication hardware that are widely used for website authentication. In OpenSSH FIDO devices are supported by new public key types \"ecdsa-sk\" and \"ed25519-sk\", along with corresponding certificate types.",{"type":21,"tag":22,"props":87,"children":88},{},[89,91,98],{"type":33,"value":90},"Let's get started by installing the latest version of OpenSSH via ",{"type":21,"tag":26,"props":92,"children":95},{"href":93,"rel":94},"https://brew.sh/",[30],[96],{"type":33,"value":97},"Homebrew",{"type":33,"value":99},", along with the YubiKey Manager (ykman) CLI. The version of OpenSSH included with macOS is not compatible.",{"type":21,"tag":101,"props":102,"children":106},"pre",{"className":103,"code":104,"language":105,"meta":7,"style":7},"language-sh shiki shiki-themes github-light","$ brew install openssh ykman\n","sh",[107],{"type":21,"tag":38,"props":108,"children":109},{"__ignoreMap":7},[110],{"type":21,"tag":111,"props":112,"children":115},"span",{"class":113,"line":114},"line",1,[116,122,128,133,138],{"type":21,"tag":111,"props":117,"children":119},{"style":118},"--shiki-default:#6F42C1",[120],{"type":33,"value":121},"$",{"type":21,"tag":111,"props":123,"children":125},{"style":124},"--shiki-default:#032F62",[126],{"type":33,"value":127}," brew",{"type":21,"tag":111,"props":129,"children":130},{"style":124},[131],{"type":33,"value":132}," install",{"type":21,"tag":111,"props":134,"children":135},{"style":124},[136],{"type":33,"value":137}," openssh",{"type":21,"tag":111,"props":139,"children":140},{"style":124},[141],{"type":33,"value":142}," ykman\n",{"type":21,"tag":22,"props":144,"children":145},{},[146],{"type":33,"value":147},"Then, let's confirm that our YubiKey has a firmware that is greater than 5.2.3:",{"type":21,"tag":101,"props":149,"children":151},{"className":103,"code":150,"language":105,"meta":7,"style":7},"$ ykman list\nYubiKey 5Ci (5.4.3) [OTP+FIDO+CCID]\n",[152],{"type":21,"tag":38,"props":153,"children":154},{"__ignoreMap":7},[155,172],{"type":21,"tag":111,"props":156,"children":157},{"class":113,"line":114},[158,162,167],{"type":21,"tag":111,"props":159,"children":160},{"style":118},[161],{"type":33,"value":121},{"type":21,"tag":111,"props":163,"children":164},{"style":124},[165],{"type":33,"value":166}," ykman",{"type":21,"tag":111,"props":168,"children":169},{"style":124},[170],{"type":33,"value":171}," list\n",{"type":21,"tag":111,"props":173,"children":175},{"class":113,"line":174},2,[176,181,186],{"type":21,"tag":111,"props":177,"children":178},{"style":118},[179],{"type":33,"value":180},"YubiKey",{"type":21,"tag":111,"props":182,"children":183},{"style":124},[184],{"type":33,"value":185}," 5Ci",{"type":21,"tag":111,"props":187,"children":189},{"style":188},"--shiki-default:#24292E",[190],{"type":33,"value":191}," (5.4.3) [OTP+FIDO+CCID]\n",{"type":21,"tag":22,"props":193,"children":194},{},[195,197,203],{"type":33,"value":196},"Next, we'll go ahead and enable a pin on our device via the ",{"type":21,"tag":38,"props":198,"children":200},{"className":199},[],[201],{"type":33,"value":202},"change-pin",{"type":33,"value":204}," command, as this a requirement for our use.",{"type":21,"tag":101,"props":206,"children":208},{"className":103,"code":207,"language":105,"meta":7,"style":7},"$ ykman fido access change-pin\n",[209],{"type":21,"tag":38,"props":210,"children":211},{"__ignoreMap":7},[212],{"type":21,"tag":111,"props":213,"children":214},{"class":113,"line":114},[215,219,223,228,233],{"type":21,"tag":111,"props":216,"children":217},{"style":118},[218],{"type":33,"value":121},{"type":21,"tag":111,"props":220,"children":221},{"style":124},[222],{"type":33,"value":166},{"type":21,"tag":111,"props":224,"children":225},{"style":124},[226],{"type":33,"value":227}," fido",{"type":21,"tag":111,"props":229,"children":230},{"style":124},[231],{"type":33,"value":232}," access",{"type":21,"tag":111,"props":234,"children":235},{"style":124},[236],{"type":33,"value":237}," change-pin\n",{"type":21,"tag":22,"props":239,"children":240},{},[241],{"type":33,"value":242},"And last, we'll generate the key on our device!",{"type":21,"tag":101,"props":244,"children":246},{"className":103,"code":245,"language":105,"meta":7,"style":7},"$ ssh-keygen -t ed25519-sk -O resident\nGenerating public/private ed25519-sk key pair.\nYou may need to touch your authenticator to authorize key generation.\n...\n",[247],{"type":21,"tag":38,"props":248,"children":249},{"__ignoreMap":7},[250,283,310,367],{"type":21,"tag":111,"props":251,"children":252},{"class":113,"line":114},[253,257,262,268,273,278],{"type":21,"tag":111,"props":254,"children":255},{"style":118},[256],{"type":33,"value":121},{"type":21,"tag":111,"props":258,"children":259},{"style":124},[260],{"type":33,"value":261}," ssh-keygen",{"type":21,"tag":111,"props":263,"children":265},{"style":264},"--shiki-default:#005CC5",[266],{"type":33,"value":267}," -t",{"type":21,"tag":111,"props":269,"children":270},{"style":124},[271],{"type":33,"value":272}," ed25519-sk",{"type":21,"tag":111,"props":274,"children":275},{"style":264},[276],{"type":33,"value":277}," -O",{"type":21,"tag":111,"props":279,"children":280},{"style":124},[281],{"type":33,"value":282}," resident\n",{"type":21,"tag":111,"props":284,"children":285},{"class":113,"line":174},[286,291,296,300,305],{"type":21,"tag":111,"props":287,"children":288},{"style":118},[289],{"type":33,"value":290},"Generating",{"type":21,"tag":111,"props":292,"children":293},{"style":124},[294],{"type":33,"value":295}," public/private",{"type":21,"tag":111,"props":297,"children":298},{"style":124},[299],{"type":33,"value":272},{"type":21,"tag":111,"props":301,"children":302},{"style":124},[303],{"type":33,"value":304}," key",{"type":21,"tag":111,"props":306,"children":307},{"style":124},[308],{"type":33,"value":309}," pair.\n",{"type":21,"tag":111,"props":311,"children":313},{"class":113,"line":312},3,[314,319,324,329,334,339,344,349,353,358,362],{"type":21,"tag":111,"props":315,"children":316},{"style":118},[317],{"type":33,"value":318},"You",{"type":21,"tag":111,"props":320,"children":321},{"style":124},[322],{"type":33,"value":323}," may",{"type":21,"tag":111,"props":325,"children":326},{"style":124},[327],{"type":33,"value":328}," need",{"type":21,"tag":111,"props":330,"children":331},{"style":124},[332],{"type":33,"value":333}," to",{"type":21,"tag":111,"props":335,"children":336},{"style":124},[337],{"type":33,"value":338}," touch",{"type":21,"tag":111,"props":340,"children":341},{"style":124},[342],{"type":33,"value":343}," your",{"type":21,"tag":111,"props":345,"children":346},{"style":124},[347],{"type":33,"value":348}," authenticator",{"type":21,"tag":111,"props":350,"children":351},{"style":124},[352],{"type":33,"value":333},{"type":21,"tag":111,"props":354,"children":355},{"style":124},[356],{"type":33,"value":357}," authorize",{"type":21,"tag":111,"props":359,"children":360},{"style":124},[361],{"type":33,"value":304},{"type":21,"tag":111,"props":363,"children":364},{"style":124},[365],{"type":33,"value":366}," generation.\n",{"type":21,"tag":111,"props":368,"children":370},{"class":113,"line":369},4,[371],{"type":21,"tag":111,"props":372,"children":373},{"style":264},[374],{"type":33,"value":375},"...\n",{"type":21,"tag":22,"props":377,"children":378},{},[379,381,387],{"type":33,"value":380},"We specify ",{"type":21,"tag":382,"props":383,"children":384},"em",{},[385],{"type":33,"value":386},"resident",{"type":33,"value":388}," to indicate that the key handle is to be stored on the YubiKey itself, since we will be using this device with multiple computers.",{"type":21,"tag":101,"props":390,"children":394},{"className":391,"code":392,"language":393,"meta":7,"style":7},"language-txt shiki shiki-themes github-light","resident\n Indicate that the key handle should be stored on the FIDO\n authenticator itself. This makes it easier to use the\n authenticator on multiple computers. Resident keys may be\n supported on FIDO2 authenticators and typically require that a PIN\n be set on the authenticator prior to generation. Resident keys\n may be loaded off the authenticator using ssh-add(1). Storing\n both parts of a key on a FIDO authenticator increases the\n likelihood of an attacker being able to use a stolen authenticator\n device.\n","txt",[395],{"type":21,"tag":38,"props":396,"children":397},{"__ignoreMap":7},[398,406,414,422,430,439,448,457,466,475],{"type":21,"tag":111,"props":399,"children":400},{"class":113,"line":114},[401],{"type":21,"tag":111,"props":402,"children":403},{},[404],{"type":33,"value":405},"resident\n",{"type":21,"tag":111,"props":407,"children":408},{"class":113,"line":174},[409],{"type":21,"tag":111,"props":410,"children":411},{},[412],{"type":33,"value":413}," Indicate that the key handle should be stored on the FIDO\n",{"type":21,"tag":111,"props":415,"children":416},{"class":113,"line":312},[417],{"type":21,"tag":111,"props":418,"children":419},{},[420],{"type":33,"value":421}," authenticator itself. This makes it easier to use the\n",{"type":21,"tag":111,"props":423,"children":424},{"class":113,"line":369},[425],{"type":21,"tag":111,"props":426,"children":427},{},[428],{"type":33,"value":429}," authenticator on multiple computers. Resident keys may be\n",{"type":21,"tag":111,"props":431,"children":433},{"class":113,"line":432},5,[434],{"type":21,"tag":111,"props":435,"children":436},{},[437],{"type":33,"value":438}," supported on FIDO2 authenticators and typically require that a PIN\n",{"type":21,"tag":111,"props":440,"children":442},{"class":113,"line":441},6,[443],{"type":21,"tag":111,"props":444,"children":445},{},[446],{"type":33,"value":447}," be set on the authenticator prior to generation. Resident keys\n",{"type":21,"tag":111,"props":449,"children":451},{"class":113,"line":450},7,[452],{"type":21,"tag":111,"props":453,"children":454},{},[455],{"type":33,"value":456}," may be loaded off the authenticator using ssh-add(1). Storing\n",{"type":21,"tag":111,"props":458,"children":460},{"class":113,"line":459},8,[461],{"type":21,"tag":111,"props":462,"children":463},{},[464],{"type":33,"value":465}," both parts of a key on a FIDO authenticator increases the\n",{"type":21,"tag":111,"props":467,"children":469},{"class":113,"line":468},9,[470],{"type":21,"tag":111,"props":471,"children":472},{},[473],{"type":33,"value":474}," likelihood of an attacker being able to use a stolen authenticator\n",{"type":21,"tag":111,"props":476,"children":478},{"class":113,"line":477},10,[479],{"type":21,"tag":111,"props":480,"children":481},{},[482],{"type":33,"value":483}," device.\n",{"type":21,"tag":22,"props":485,"children":486},{},[487,489,494,496,501],{"type":33,"value":488},"And that's all it takes -- simple enough. Now, when interacting with ",{"type":21,"tag":382,"props":490,"children":491},{},[492],{"type":33,"value":493},"ssh",{"type":33,"value":495}," or ",{"type":21,"tag":382,"props":497,"children":498},{},[499],{"type":33,"value":500},"git",{"type":33,"value":502}," you will be prompted to touch the YubiKey to bring that little bit of physical 2FA.",{"type":21,"tag":504,"props":505,"children":506},"style",{},[507],{"type":33,"value":508},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":174,"depth":174,"links":510},[],"markdown","content:articles:ssh-ed25519-sk-yubikey.md","content","articles/ssh-ed25519-sk-yubikey.md","articles/ssh-ed25519-sk-yubikey","md",1731781852993] \ No newline at end of file +[{"data":1,"prerenderedAt":517},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"draft":6,"date":10,"tags":11,"categories":14,"excerpt":17,"body":46,"_type":511,"_id":512,"_source":513,"_file":514,"_stem":515,"_extension":516},"/articles/ssh-ed25519-sk-yubikey","articles",false,"","Configuring a YubiKey for use with OpenSSH","YubiKey's are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" ed25519-sk key type that supports FIDO compliant hardware keys.","2024-06-09",[12,13],"unix","configurations",[15,16],"tooling","tips",{"type":18,"children":19},"root",[20],{"type":21,"tag":22,"props":23,"children":24},"element","p",{},[25,35,37,44],{"type":21,"tag":26,"props":27,"children":31},"a",{"href":28,"rel":29},"https://www.yubico.com/",[30],"nofollow",[32],{"type":33,"value":34},"text","YubiKey's",{"type":33,"value":36}," are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the \"new\" ",{"type":21,"tag":38,"props":39,"children":41},"code",{"className":40},[],[42],{"type":33,"value":43},"ed25519-sk",{"type":33,"value":45}," key type that supports FIDO compliant hardware keys.",{"type":18,"children":47,"toc":509},[48,63,77,86,100,143,148,192,205,238,243,376,389,484,503],{"type":21,"tag":22,"props":49,"children":50},{},[51,56,57,62],{"type":21,"tag":26,"props":52,"children":54},{"href":28,"rel":53},[30],[55],{"type":33,"value":34},{"type":33,"value":36},{"type":21,"tag":38,"props":58,"children":60},{"className":59},[],[61],{"type":33,"value":43},{"type":33,"value":45},{"type":21,"tag":22,"props":64,"children":65},{},[66,68,75],{"type":33,"value":67},"In ",{"type":21,"tag":26,"props":69,"children":72},{"href":70,"rel":71},"https://www.openssh.com/txt/release-8.2",[30],[73],{"type":33,"value":74},"release 8.2 of OpenSSH",{"type":33,"value":76}," support for FIDO devices was added with public key types \"ecdsa-sk\" and \"ed25519-sk\" (-sk standing for \"security key\"). This key type is supported by YubiKey's with firmware version 5.2.3 or higher.",{"type":21,"tag":78,"props":79,"children":80},"blockquote",{},[81],{"type":21,"tag":22,"props":82,"children":83},{},[84],{"type":33,"value":85},"This release adds support for FIDO/U2F hardware authenticators to OpenSSH. U2F/FIDO are open standards for inexpensive two-factor authentication hardware that are widely used for website authentication. In OpenSSH FIDO devices are supported by new public key types \"ecdsa-sk\" and \"ed25519-sk\", along with corresponding certificate types.",{"type":21,"tag":22,"props":87,"children":88},{},[89,91,98],{"type":33,"value":90},"Let's get started by installing the latest version of OpenSSH via ",{"type":21,"tag":26,"props":92,"children":95},{"href":93,"rel":94},"https://brew.sh/",[30],[96],{"type":33,"value":97},"Homebrew",{"type":33,"value":99},", along with the YubiKey Manager (ykman) CLI. The version of OpenSSH included with macOS is not compatible.",{"type":21,"tag":101,"props":102,"children":106},"pre",{"className":103,"code":104,"language":105,"meta":7,"style":7},"language-sh shiki shiki-themes github-light","$ brew install openssh ykman\n","sh",[107],{"type":21,"tag":38,"props":108,"children":109},{"__ignoreMap":7},[110],{"type":21,"tag":111,"props":112,"children":115},"span",{"class":113,"line":114},"line",1,[116,122,128,133,138],{"type":21,"tag":111,"props":117,"children":119},{"style":118},"--shiki-default:#6F42C1",[120],{"type":33,"value":121},"$",{"type":21,"tag":111,"props":123,"children":125},{"style":124},"--shiki-default:#032F62",[126],{"type":33,"value":127}," brew",{"type":21,"tag":111,"props":129,"children":130},{"style":124},[131],{"type":33,"value":132}," install",{"type":21,"tag":111,"props":134,"children":135},{"style":124},[136],{"type":33,"value":137}," openssh",{"type":21,"tag":111,"props":139,"children":140},{"style":124},[141],{"type":33,"value":142}," ykman\n",{"type":21,"tag":22,"props":144,"children":145},{},[146],{"type":33,"value":147},"Then, let's confirm that our YubiKey has a firmware that is greater than 5.2.3:",{"type":21,"tag":101,"props":149,"children":151},{"className":103,"code":150,"language":105,"meta":7,"style":7},"$ ykman list\nYubiKey 5Ci (5.4.3) [OTP+FIDO+CCID]\n",[152],{"type":21,"tag":38,"props":153,"children":154},{"__ignoreMap":7},[155,172],{"type":21,"tag":111,"props":156,"children":157},{"class":113,"line":114},[158,162,167],{"type":21,"tag":111,"props":159,"children":160},{"style":118},[161],{"type":33,"value":121},{"type":21,"tag":111,"props":163,"children":164},{"style":124},[165],{"type":33,"value":166}," ykman",{"type":21,"tag":111,"props":168,"children":169},{"style":124},[170],{"type":33,"value":171}," list\n",{"type":21,"tag":111,"props":173,"children":175},{"class":113,"line":174},2,[176,181,186],{"type":21,"tag":111,"props":177,"children":178},{"style":118},[179],{"type":33,"value":180},"YubiKey",{"type":21,"tag":111,"props":182,"children":183},{"style":124},[184],{"type":33,"value":185}," 5Ci",{"type":21,"tag":111,"props":187,"children":189},{"style":188},"--shiki-default:#24292E",[190],{"type":33,"value":191}," (5.4.3) [OTP+FIDO+CCID]\n",{"type":21,"tag":22,"props":193,"children":194},{},[195,197,203],{"type":33,"value":196},"Next, we'll go ahead and enable a pin on our device via the ",{"type":21,"tag":38,"props":198,"children":200},{"className":199},[],[201],{"type":33,"value":202},"change-pin",{"type":33,"value":204}," command, as this a requirement for our use.",{"type":21,"tag":101,"props":206,"children":208},{"className":103,"code":207,"language":105,"meta":7,"style":7},"$ ykman fido access change-pin\n",[209],{"type":21,"tag":38,"props":210,"children":211},{"__ignoreMap":7},[212],{"type":21,"tag":111,"props":213,"children":214},{"class":113,"line":114},[215,219,223,228,233],{"type":21,"tag":111,"props":216,"children":217},{"style":118},[218],{"type":33,"value":121},{"type":21,"tag":111,"props":220,"children":221},{"style":124},[222],{"type":33,"value":166},{"type":21,"tag":111,"props":224,"children":225},{"style":124},[226],{"type":33,"value":227}," fido",{"type":21,"tag":111,"props":229,"children":230},{"style":124},[231],{"type":33,"value":232}," access",{"type":21,"tag":111,"props":234,"children":235},{"style":124},[236],{"type":33,"value":237}," change-pin\n",{"type":21,"tag":22,"props":239,"children":240},{},[241],{"type":33,"value":242},"And last, we'll generate the key on our device!",{"type":21,"tag":101,"props":244,"children":246},{"className":103,"code":245,"language":105,"meta":7,"style":7},"$ ssh-keygen -t ed25519-sk -O resident\nGenerating public/private ed25519-sk key pair.\nYou may need to touch your authenticator to authorize key generation.\n...\n",[247],{"type":21,"tag":38,"props":248,"children":249},{"__ignoreMap":7},[250,283,310,367],{"type":21,"tag":111,"props":251,"children":252},{"class":113,"line":114},[253,257,262,268,273,278],{"type":21,"tag":111,"props":254,"children":255},{"style":118},[256],{"type":33,"value":121},{"type":21,"tag":111,"props":258,"children":259},{"style":124},[260],{"type":33,"value":261}," ssh-keygen",{"type":21,"tag":111,"props":263,"children":265},{"style":264},"--shiki-default:#005CC5",[266],{"type":33,"value":267}," -t",{"type":21,"tag":111,"props":269,"children":270},{"style":124},[271],{"type":33,"value":272}," ed25519-sk",{"type":21,"tag":111,"props":274,"children":275},{"style":264},[276],{"type":33,"value":277}," -O",{"type":21,"tag":111,"props":279,"children":280},{"style":124},[281],{"type":33,"value":282}," resident\n",{"type":21,"tag":111,"props":284,"children":285},{"class":113,"line":174},[286,291,296,300,305],{"type":21,"tag":111,"props":287,"children":288},{"style":118},[289],{"type":33,"value":290},"Generating",{"type":21,"tag":111,"props":292,"children":293},{"style":124},[294],{"type":33,"value":295}," public/private",{"type":21,"tag":111,"props":297,"children":298},{"style":124},[299],{"type":33,"value":272},{"type":21,"tag":111,"props":301,"children":302},{"style":124},[303],{"type":33,"value":304}," key",{"type":21,"tag":111,"props":306,"children":307},{"style":124},[308],{"type":33,"value":309}," pair.\n",{"type":21,"tag":111,"props":311,"children":313},{"class":113,"line":312},3,[314,319,324,329,334,339,344,349,353,358,362],{"type":21,"tag":111,"props":315,"children":316},{"style":118},[317],{"type":33,"value":318},"You",{"type":21,"tag":111,"props":320,"children":321},{"style":124},[322],{"type":33,"value":323}," may",{"type":21,"tag":111,"props":325,"children":326},{"style":124},[327],{"type":33,"value":328}," need",{"type":21,"tag":111,"props":330,"children":331},{"style":124},[332],{"type":33,"value":333}," to",{"type":21,"tag":111,"props":335,"children":336},{"style":124},[337],{"type":33,"value":338}," touch",{"type":21,"tag":111,"props":340,"children":341},{"style":124},[342],{"type":33,"value":343}," your",{"type":21,"tag":111,"props":345,"children":346},{"style":124},[347],{"type":33,"value":348}," authenticator",{"type":21,"tag":111,"props":350,"children":351},{"style":124},[352],{"type":33,"value":333},{"type":21,"tag":111,"props":354,"children":355},{"style":124},[356],{"type":33,"value":357}," authorize",{"type":21,"tag":111,"props":359,"children":360},{"style":124},[361],{"type":33,"value":304},{"type":21,"tag":111,"props":363,"children":364},{"style":124},[365],{"type":33,"value":366}," generation.\n",{"type":21,"tag":111,"props":368,"children":370},{"class":113,"line":369},4,[371],{"type":21,"tag":111,"props":372,"children":373},{"style":264},[374],{"type":33,"value":375},"...\n",{"type":21,"tag":22,"props":377,"children":378},{},[379,381,387],{"type":33,"value":380},"We specify ",{"type":21,"tag":382,"props":383,"children":384},"em",{},[385],{"type":33,"value":386},"resident",{"type":33,"value":388}," to indicate that the key handle is to be stored on the YubiKey itself, since we will be using this device with multiple computers.",{"type":21,"tag":101,"props":390,"children":394},{"className":391,"code":392,"language":393,"meta":7,"style":7},"language-txt shiki shiki-themes github-light","resident\n Indicate that the key handle should be stored on the FIDO\n authenticator itself. This makes it easier to use the\n authenticator on multiple computers. Resident keys may be\n supported on FIDO2 authenticators and typically require that a PIN\n be set on the authenticator prior to generation. Resident keys\n may be loaded off the authenticator using ssh-add(1). Storing\n both parts of a key on a FIDO authenticator increases the\n likelihood of an attacker being able to use a stolen authenticator\n device.\n","txt",[395],{"type":21,"tag":38,"props":396,"children":397},{"__ignoreMap":7},[398,406,414,422,430,439,448,457,466,475],{"type":21,"tag":111,"props":399,"children":400},{"class":113,"line":114},[401],{"type":21,"tag":111,"props":402,"children":403},{},[404],{"type":33,"value":405},"resident\n",{"type":21,"tag":111,"props":407,"children":408},{"class":113,"line":174},[409],{"type":21,"tag":111,"props":410,"children":411},{},[412],{"type":33,"value":413}," Indicate that the key handle should be stored on the FIDO\n",{"type":21,"tag":111,"props":415,"children":416},{"class":113,"line":312},[417],{"type":21,"tag":111,"props":418,"children":419},{},[420],{"type":33,"value":421}," authenticator itself. This makes it easier to use the\n",{"type":21,"tag":111,"props":423,"children":424},{"class":113,"line":369},[425],{"type":21,"tag":111,"props":426,"children":427},{},[428],{"type":33,"value":429}," authenticator on multiple computers. Resident keys may be\n",{"type":21,"tag":111,"props":431,"children":433},{"class":113,"line":432},5,[434],{"type":21,"tag":111,"props":435,"children":436},{},[437],{"type":33,"value":438}," supported on FIDO2 authenticators and typically require that a PIN\n",{"type":21,"tag":111,"props":440,"children":442},{"class":113,"line":441},6,[443],{"type":21,"tag":111,"props":444,"children":445},{},[446],{"type":33,"value":447}," be set on the authenticator prior to generation. Resident keys\n",{"type":21,"tag":111,"props":449,"children":451},{"class":113,"line":450},7,[452],{"type":21,"tag":111,"props":453,"children":454},{},[455],{"type":33,"value":456}," may be loaded off the authenticator using ssh-add(1). Storing\n",{"type":21,"tag":111,"props":458,"children":460},{"class":113,"line":459},8,[461],{"type":21,"tag":111,"props":462,"children":463},{},[464],{"type":33,"value":465}," both parts of a key on a FIDO authenticator increases the\n",{"type":21,"tag":111,"props":467,"children":469},{"class":113,"line":468},9,[470],{"type":21,"tag":111,"props":471,"children":472},{},[473],{"type":33,"value":474}," likelihood of an attacker being able to use a stolen authenticator\n",{"type":21,"tag":111,"props":476,"children":478},{"class":113,"line":477},10,[479],{"type":21,"tag":111,"props":480,"children":481},{},[482],{"type":33,"value":483}," device.\n",{"type":21,"tag":22,"props":485,"children":486},{},[487,489,494,496,501],{"type":33,"value":488},"And that's all it takes -- simple enough. Now, when interacting with ",{"type":21,"tag":382,"props":490,"children":491},{},[492],{"type":33,"value":493},"ssh",{"type":33,"value":495}," or ",{"type":21,"tag":382,"props":497,"children":498},{},[499],{"type":33,"value":500},"git",{"type":33,"value":502}," you will be prompted to touch the YubiKey to bring that little bit of physical 2FA.",{"type":21,"tag":504,"props":505,"children":506},"style",{},[507],{"type":33,"value":508},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":174,"depth":174,"links":510},[],"markdown","content:articles:ssh-ed25519-sk-yubikey.md","content","articles/ssh-ed25519-sk-yubikey.md","articles/ssh-ed25519-sk-yubikey","md",1731782033241] \ No newline at end of file diff --git a/articles/ssh-ed25519-sk-yubikey/index.html b/articles/ssh-ed25519-sk-yubikey/index.html index e11750f5..73043d4c 100644 --- a/articles/ssh-ed25519-sk-yubikey/index.html +++ b/articles/ssh-ed25519-sk-yubikey/index.html @@ -5,33 +5,33 @@ - - - - - + + + + + - + - - - - - - - - - - - - - - + + + + + + + + + + + + + + -

        Configuring a YubiKey for use with OpenSSH

        YubiKey's are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the "new" ed25519-sk key type that supports FIDO compliant hardware keys.

        In release 8.2 of OpenSSH support for FIDO devices was added with public key types "ecdsa-sk" and "ed25519-sk" (-sk standing for "security key"). This key type is supported by YubiKey's with firmware version 5.2.3 or higher.

        This release adds support for FIDO/U2F hardware authenticators to OpenSSH. U2F/FIDO are open standards for inexpensive two-factor authentication hardware that are widely used for website authentication. In OpenSSH FIDO devices are supported by new public key types "ecdsa-sk" and "ed25519-sk", along with corresponding certificate types.

        Let's get started by installing the latest version of OpenSSH via Homebrew, along with the YubiKey Manager (ykman) CLI. The version of OpenSSH included with macOS is not compatible.

        $ brew install openssh ykman
        +

        Configuring a YubiKey for use with OpenSSH

        YubiKey's are a convenient way to introduce a physical form of two factor authentication into ones workflow. They support a variety of protocols, but in this guide we will walk through how to configure a YubiKey for use with OpenSSH via the "new" ed25519-sk key type that supports FIDO compliant hardware keys.

        In release 8.2 of OpenSSH support for FIDO devices was added with public key types "ecdsa-sk" and "ed25519-sk" (-sk standing for "security key"). This key type is supported by YubiKey's with firmware version 5.2.3 or higher.

        This release adds support for FIDO/U2F hardware authenticators to OpenSSH. U2F/FIDO are open standards for inexpensive two-factor authentication hardware that are widely used for website authentication. In OpenSSH FIDO devices are supported by new public key types "ecdsa-sk" and "ed25519-sk", along with corresponding certificate types.

        Let's get started by installing the latest version of OpenSSH via Homebrew, along with the YubiKey Manager (ykman) CLI. The version of OpenSSH included with macOS is not compatible.

        $ brew install openssh ykman
         

        Then, let's confirm that our YubiKey has a firmware that is greater than 5.2.3:

        $ ykman list
         YubiKey 5Ci (5.4.3) [OTP+FIDO+CCID]
         

        Next, we'll go ahead and enable a pin on our device via the change-pin command, as this a requirement for our use.

        $ ykman fido access change-pin
        @@ -49,5 +49,5 @@
                 both parts of a key on a FIDO authenticator increases the
                 likelihood of an attacker being able to use a stolen authenticator
                 device.
        -

        And that's all it takes -- simple enough. Now, when interacting with ssh or git you will be prompted to touch the YubiKey to bring that little bit of physical 2FA.

        - \ No newline at end of file +

        And that's all it takes -- simple enough. Now, when interacting with ssh or git you will be prompted to touch the YubiKey to bring that little bit of physical 2FA.

        + \ No newline at end of file diff --git a/articles/unit-testing-micropython-with-mocks/_payload.json b/articles/unit-testing-micropython-with-mocks/_payload.json index a0212985..63c21a15 100644 --- a/articles/unit-testing-micropython-with-mocks/_payload.json +++ b/articles/unit-testing-micropython-with-mocks/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":690},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"draft":6,"tags":11,"categories":16,"excerpt":19,"body":29,"_type":684,"_id":685,"_source":686,"_file":687,"_stem":688,"_extension":689},"/articles/unit-testing-micropython-with-mocks","articles",false,"","Unit Testing in MicroPython with Mocks","Unit testing code for embedded systems can be challenging. While it's possible\nto leverage emulators, write side-effect free code, or run tests on the\nhardware itself, it's often easiest to unit test the code on your personal\ncomputer with mocked hardware functionality.","2020-02-07",[12,13,14,15],"micropython","testing","mocks","tutorial",[17,18],"python","embedded",{"type":20,"children":21},"root",[22],{"type":23,"tag":24,"props":25,"children":26},"element","p",{},[27],{"type":28,"value":9},"text",{"type":20,"children":30,"toc":681},[31,35,42,47,53,75,252,278,398,419,618,646,653,675],{"type":23,"tag":24,"props":32,"children":33},{},[34],{"type":28,"value":9},{"type":23,"tag":36,"props":37,"children":39},"h1",{"id":38},"mocking",[40],{"type":28,"value":41},"Mocking",{"type":23,"tag":24,"props":43,"children":44},{},[45],{"type":28,"value":46},"Mocks allow us to replace the hardware interfacing functionality under-the-hood\nwith predefined results and side-effects. For example, if there is a piece of\nlogic that retrieves values from an accelerometer to get a device's\norientation, it would be possible to mock the returned values of the\naccelerometer -- allowing us to run the unit tests on a device that does not\nhave an accelerometer sensor installed.",{"type":23,"tag":36,"props":48,"children":50},{"id":49},"a-micropython-mocking-example",[51],{"type":28,"value":52},"A MicroPython Mocking Example",{"type":23,"tag":24,"props":54,"children":55},{},[56,58,65,67,73],{"type":28,"value":57},"In this example, we will be unit testing a module named ",{"type":23,"tag":59,"props":60,"children":62},"code",{"className":61},[],[63],{"type":28,"value":64},"time_logger",{"type":28,"value":66},", that\ndepends on the MicroPython library ",{"type":23,"tag":59,"props":68,"children":70},{"className":69},[],[71],{"type":28,"value":72},"utime",{"type":28,"value":74}," to log the most recent Epoch time to\na file.",{"type":23,"tag":76,"props":77,"children":80},"pre",{"className":78,"code":79,"language":17,"meta":7,"style":7},"language-python shiki shiki-themes github-light","# time_logger.py\n\nclass TimeLogger(object):\n\n def save_time(self):\n \"\"\" Overwrite a file with the most recent Epoch timestamp from `utime`\n \"\"\"\n with open(\"LAST_KNOWN_TIME\", \"w+\") as f:\n f.write(str(utime.time()))\n",[81],{"type":23,"tag":59,"props":82,"children":83},{"__ignoreMap":7},[84,96,106,139,147,166,176,185,233],{"type":23,"tag":85,"props":86,"children":89},"span",{"class":87,"line":88},"line",1,[90],{"type":23,"tag":85,"props":91,"children":93},{"style":92},"--shiki-default:#6A737D",[94],{"type":28,"value":95},"# time_logger.py\n",{"type":23,"tag":85,"props":97,"children":99},{"class":87,"line":98},2,[100],{"type":23,"tag":85,"props":101,"children":103},{"emptyLinePlaceholder":102},true,[104],{"type":28,"value":105},"\n",{"type":23,"tag":85,"props":107,"children":109},{"class":87,"line":108},3,[110,116,122,128,134],{"type":23,"tag":85,"props":111,"children":113},{"style":112},"--shiki-default:#D73A49",[114],{"type":28,"value":115},"class",{"type":23,"tag":85,"props":117,"children":119},{"style":118},"--shiki-default:#6F42C1",[120],{"type":28,"value":121}," TimeLogger",{"type":23,"tag":85,"props":123,"children":125},{"style":124},"--shiki-default:#24292E",[126],{"type":28,"value":127},"(",{"type":23,"tag":85,"props":129,"children":131},{"style":130},"--shiki-default:#005CC5",[132],{"type":28,"value":133},"object",{"type":23,"tag":85,"props":135,"children":136},{"style":124},[137],{"type":28,"value":138},"):\n",{"type":23,"tag":85,"props":140,"children":142},{"class":87,"line":141},4,[143],{"type":23,"tag":85,"props":144,"children":145},{"emptyLinePlaceholder":102},[146],{"type":28,"value":105},{"type":23,"tag":85,"props":148,"children":150},{"class":87,"line":149},5,[151,156,161],{"type":23,"tag":85,"props":152,"children":153},{"style":112},[154],{"type":28,"value":155}," def",{"type":23,"tag":85,"props":157,"children":158},{"style":118},[159],{"type":28,"value":160}," save_time",{"type":23,"tag":85,"props":162,"children":163},{"style":124},[164],{"type":28,"value":165},"(self):\n",{"type":23,"tag":85,"props":167,"children":169},{"class":87,"line":168},6,[170],{"type":23,"tag":85,"props":171,"children":173},{"style":172},"--shiki-default:#032F62",[174],{"type":28,"value":175}," \"\"\" Overwrite a file with the most recent Epoch timestamp from `utime`\n",{"type":23,"tag":85,"props":177,"children":179},{"class":87,"line":178},7,[180],{"type":23,"tag":85,"props":181,"children":182},{"style":172},[183],{"type":28,"value":184}," \"\"\"\n",{"type":23,"tag":85,"props":186,"children":188},{"class":87,"line":187},8,[189,194,199,203,208,213,218,223,228],{"type":23,"tag":85,"props":190,"children":191},{"style":112},[192],{"type":28,"value":193}," with",{"type":23,"tag":85,"props":195,"children":196},{"style":130},[197],{"type":28,"value":198}," open",{"type":23,"tag":85,"props":200,"children":201},{"style":124},[202],{"type":28,"value":127},{"type":23,"tag":85,"props":204,"children":205},{"style":172},[206],{"type":28,"value":207},"\"LAST_KNOWN_TIME\"",{"type":23,"tag":85,"props":209,"children":210},{"style":124},[211],{"type":28,"value":212},", ",{"type":23,"tag":85,"props":214,"children":215},{"style":172},[216],{"type":28,"value":217},"\"w+\"",{"type":23,"tag":85,"props":219,"children":220},{"style":124},[221],{"type":28,"value":222},") ",{"type":23,"tag":85,"props":224,"children":225},{"style":112},[226],{"type":28,"value":227},"as",{"type":23,"tag":85,"props":229,"children":230},{"style":124},[231],{"type":28,"value":232}," f:\n",{"type":23,"tag":85,"props":234,"children":236},{"class":87,"line":235},9,[237,242,247],{"type":23,"tag":85,"props":238,"children":239},{"style":124},[240],{"type":28,"value":241}," f.write(",{"type":23,"tag":85,"props":243,"children":244},{"style":130},[245],{"type":28,"value":246},"str",{"type":23,"tag":85,"props":248,"children":249},{"style":124},[250],{"type":28,"value":251},"(utime.time()))\n",{"type":23,"tag":24,"props":253,"children":254},{},[255,257,262,264,269,271,276],{"type":28,"value":256},"First, because the ",{"type":23,"tag":59,"props":258,"children":260},{"className":259},[],[261],{"type":28,"value":72},{"type":28,"value":263}," module is not installed on the machine that the unit\ntests on, we must mock ",{"type":23,"tag":59,"props":265,"children":267},{"className":266},[],[268],{"type":28,"value":72},{"type":28,"value":270}," module before importing ",{"type":23,"tag":59,"props":272,"children":274},{"className":273},[],[275],{"type":28,"value":64},{"type":28,"value":277}," in our\nunit test file.",{"type":23,"tag":76,"props":279,"children":281},{"className":78,"code":280,"language":17,"meta":7,"style":7},"# test_time_logger.py\n\nimport unittest\n\nfrom unittest.mock import MagicMock\n\nsys.modules['utime'] = MagicMock()\nfrom time_logger import TimeLogger\n",[282],{"type":23,"tag":59,"props":283,"children":284},{"__ignoreMap":7},[285,293,300,313,320,342,349,377],{"type":23,"tag":85,"props":286,"children":287},{"class":87,"line":88},[288],{"type":23,"tag":85,"props":289,"children":290},{"style":92},[291],{"type":28,"value":292},"# test_time_logger.py\n",{"type":23,"tag":85,"props":294,"children":295},{"class":87,"line":98},[296],{"type":23,"tag":85,"props":297,"children":298},{"emptyLinePlaceholder":102},[299],{"type":28,"value":105},{"type":23,"tag":85,"props":301,"children":302},{"class":87,"line":108},[303,308],{"type":23,"tag":85,"props":304,"children":305},{"style":112},[306],{"type":28,"value":307},"import",{"type":23,"tag":85,"props":309,"children":310},{"style":124},[311],{"type":28,"value":312}," unittest\n",{"type":23,"tag":85,"props":314,"children":315},{"class":87,"line":141},[316],{"type":23,"tag":85,"props":317,"children":318},{"emptyLinePlaceholder":102},[319],{"type":28,"value":105},{"type":23,"tag":85,"props":321,"children":322},{"class":87,"line":149},[323,328,333,337],{"type":23,"tag":85,"props":324,"children":325},{"style":112},[326],{"type":28,"value":327},"from",{"type":23,"tag":85,"props":329,"children":330},{"style":124},[331],{"type":28,"value":332}," unittest.mock ",{"type":23,"tag":85,"props":334,"children":335},{"style":112},[336],{"type":28,"value":307},{"type":23,"tag":85,"props":338,"children":339},{"style":124},[340],{"type":28,"value":341}," MagicMock\n",{"type":23,"tag":85,"props":343,"children":344},{"class":87,"line":168},[345],{"type":23,"tag":85,"props":346,"children":347},{"emptyLinePlaceholder":102},[348],{"type":28,"value":105},{"type":23,"tag":85,"props":350,"children":351},{"class":87,"line":178},[352,357,362,367,372],{"type":23,"tag":85,"props":353,"children":354},{"style":124},[355],{"type":28,"value":356},"sys.modules[",{"type":23,"tag":85,"props":358,"children":359},{"style":172},[360],{"type":28,"value":361},"'utime'",{"type":23,"tag":85,"props":363,"children":364},{"style":124},[365],{"type":28,"value":366},"] ",{"type":23,"tag":85,"props":368,"children":369},{"style":112},[370],{"type":28,"value":371},"=",{"type":23,"tag":85,"props":373,"children":374},{"style":124},[375],{"type":28,"value":376}," MagicMock()\n",{"type":23,"tag":85,"props":378,"children":379},{"class":87,"line":187},[380,384,389,393],{"type":23,"tag":85,"props":381,"children":382},{"style":112},[383],{"type":28,"value":327},{"type":23,"tag":85,"props":385,"children":386},{"style":124},[387],{"type":28,"value":388}," time_logger ",{"type":23,"tag":85,"props":390,"children":391},{"style":112},[392],{"type":28,"value":307},{"type":23,"tag":85,"props":394,"children":395},{"style":124},[396],{"type":28,"value":397}," TimeLogger\n",{"type":23,"tag":24,"props":399,"children":400},{},[401,403,409,411,417],{"type":28,"value":402},"Then, we can write a test that patches the ",{"type":23,"tag":59,"props":404,"children":406},{"className":405},[],[407],{"type":28,"value":408},"utime.time",{"type":28,"value":410}," functionality so that\nit returns a value of our choosing -- in this case, ",{"type":23,"tag":59,"props":412,"children":414},{"className":413},[],[415],{"type":28,"value":416},"1234",{"type":28,"value":418},".",{"type":23,"tag":76,"props":420,"children":422},{"className":78,"code":421,"language":17,"meta":7,"style":7},"class TestTimeLogger(unittest.TestCase):\n\n def test_save_time(self):\n \"\"\" Verify that the Epoch time is written to file\n \"\"\"\n with unittest.mock.patch(\"utime.time\", return_value=1234):\n t = TimeLogger()\n t.save_time()\n with open(\"LAST_KNOWN_TIME\") as f:\n self.assertEqual(\"1234\", f.read())\n",[423],{"type":23,"tag":59,"props":424,"children":425},{"__ignoreMap":7},[426,460,467,483,491,498,537,554,562,594],{"type":23,"tag":85,"props":427,"children":428},{"class":87,"line":88},[429,433,438,442,447,451,456],{"type":23,"tag":85,"props":430,"children":431},{"style":112},[432],{"type":28,"value":115},{"type":23,"tag":85,"props":434,"children":435},{"style":118},[436],{"type":28,"value":437}," TestTimeLogger",{"type":23,"tag":85,"props":439,"children":440},{"style":124},[441],{"type":28,"value":127},{"type":23,"tag":85,"props":443,"children":444},{"style":118},[445],{"type":28,"value":446},"unittest",{"type":23,"tag":85,"props":448,"children":449},{"style":124},[450],{"type":28,"value":418},{"type":23,"tag":85,"props":452,"children":453},{"style":118},[454],{"type":28,"value":455},"TestCase",{"type":23,"tag":85,"props":457,"children":458},{"style":124},[459],{"type":28,"value":138},{"type":23,"tag":85,"props":461,"children":462},{"class":87,"line":98},[463],{"type":23,"tag":85,"props":464,"children":465},{"emptyLinePlaceholder":102},[466],{"type":28,"value":105},{"type":23,"tag":85,"props":468,"children":469},{"class":87,"line":108},[470,474,479],{"type":23,"tag":85,"props":471,"children":472},{"style":112},[473],{"type":28,"value":155},{"type":23,"tag":85,"props":475,"children":476},{"style":118},[477],{"type":28,"value":478}," test_save_time",{"type":23,"tag":85,"props":480,"children":481},{"style":124},[482],{"type":28,"value":165},{"type":23,"tag":85,"props":484,"children":485},{"class":87,"line":141},[486],{"type":23,"tag":85,"props":487,"children":488},{"style":172},[489],{"type":28,"value":490}," \"\"\" Verify that the Epoch time is written to file\n",{"type":23,"tag":85,"props":492,"children":493},{"class":87,"line":149},[494],{"type":23,"tag":85,"props":495,"children":496},{"style":172},[497],{"type":28,"value":184},{"type":23,"tag":85,"props":499,"children":500},{"class":87,"line":168},[501,505,510,515,519,525,529,533],{"type":23,"tag":85,"props":502,"children":503},{"style":112},[504],{"type":28,"value":193},{"type":23,"tag":85,"props":506,"children":507},{"style":124},[508],{"type":28,"value":509}," unittest.mock.patch(",{"type":23,"tag":85,"props":511,"children":512},{"style":172},[513],{"type":28,"value":514},"\"utime.time\"",{"type":23,"tag":85,"props":516,"children":517},{"style":124},[518],{"type":28,"value":212},{"type":23,"tag":85,"props":520,"children":522},{"style":521},"--shiki-default:#E36209",[523],{"type":28,"value":524},"return_value",{"type":23,"tag":85,"props":526,"children":527},{"style":112},[528],{"type":28,"value":371},{"type":23,"tag":85,"props":530,"children":531},{"style":130},[532],{"type":28,"value":416},{"type":23,"tag":85,"props":534,"children":535},{"style":124},[536],{"type":28,"value":138},{"type":23,"tag":85,"props":538,"children":539},{"class":87,"line":178},[540,545,549],{"type":23,"tag":85,"props":541,"children":542},{"style":124},[543],{"type":28,"value":544}," t ",{"type":23,"tag":85,"props":546,"children":547},{"style":112},[548],{"type":28,"value":371},{"type":23,"tag":85,"props":550,"children":551},{"style":124},[552],{"type":28,"value":553}," TimeLogger()\n",{"type":23,"tag":85,"props":555,"children":556},{"class":87,"line":187},[557],{"type":23,"tag":85,"props":558,"children":559},{"style":124},[560],{"type":28,"value":561}," t.save_time()\n",{"type":23,"tag":85,"props":563,"children":564},{"class":87,"line":235},[565,570,574,578,582,586,590],{"type":23,"tag":85,"props":566,"children":567},{"style":112},[568],{"type":28,"value":569}," with",{"type":23,"tag":85,"props":571,"children":572},{"style":130},[573],{"type":28,"value":198},{"type":23,"tag":85,"props":575,"children":576},{"style":124},[577],{"type":28,"value":127},{"type":23,"tag":85,"props":579,"children":580},{"style":172},[581],{"type":28,"value":207},{"type":23,"tag":85,"props":583,"children":584},{"style":124},[585],{"type":28,"value":222},{"type":23,"tag":85,"props":587,"children":588},{"style":112},[589],{"type":28,"value":227},{"type":23,"tag":85,"props":591,"children":592},{"style":124},[593],{"type":28,"value":232},{"type":23,"tag":85,"props":595,"children":597},{"class":87,"line":596},10,[598,603,608,613],{"type":23,"tag":85,"props":599,"children":600},{"style":130},[601],{"type":28,"value":602}," self",{"type":23,"tag":85,"props":604,"children":605},{"style":124},[606],{"type":28,"value":607},".assertEqual(",{"type":23,"tag":85,"props":609,"children":610},{"style":172},[611],{"type":28,"value":612},"\"1234\"",{"type":23,"tag":85,"props":614,"children":615},{"style":124},[616],{"type":28,"value":617},", f.read())\n",{"type":23,"tag":24,"props":619,"children":620},{},[621,623,629,631,637,639,644],{"type":28,"value":622},"Now, when the ",{"type":23,"tag":59,"props":624,"children":626},{"className":625},[],[627],{"type":28,"value":628},"save_time",{"type":28,"value":630}," method gets the latest time from ",{"type":23,"tag":59,"props":632,"children":634},{"className":633},[],[635],{"type":28,"value":636},"utime.time()",{"type":28,"value":638},", the\nvalue will be patched to return ",{"type":23,"tag":59,"props":640,"children":642},{"className":641},[],[643],{"type":28,"value":416},{"type":28,"value":645},". That value will be written to a file,\nand our unit test will pass!",{"type":23,"tag":647,"props":648,"children":650},"h2",{"id":649},"references",[651],{"type":28,"value":652},"References",{"type":23,"tag":654,"props":655,"children":656},"ol",{},[657],{"type":23,"tag":658,"props":659,"children":660},"li",{},[661],{"type":23,"tag":662,"props":663,"children":667},"a",{"href":664,"rel":665},"https://docs.python.org/3/library/unittest.html",[666],"nofollow",[668,673],{"type":23,"tag":59,"props":669,"children":671},{"className":670},[],[672],{"type":28,"value":446},{"type":28,"value":674}," — Unit testing framework",{"type":23,"tag":676,"props":677,"children":678},"style",{},[679],{"type":28,"value":680},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":98,"depth":98,"links":682},[683],{"id":649,"depth":98,"text":652},"markdown","content:articles:unit-testing-micropython-with-mocks.md","content","articles/unit-testing-micropython-with-mocks.md","articles/unit-testing-micropython-with-mocks","md",1731781853288] \ No newline at end of file +[{"data":1,"prerenderedAt":690},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"date":10,"draft":6,"tags":11,"categories":16,"excerpt":19,"body":29,"_type":684,"_id":685,"_source":686,"_file":687,"_stem":688,"_extension":689},"/articles/unit-testing-micropython-with-mocks","articles",false,"","Unit Testing in MicroPython with Mocks","Unit testing code for embedded systems can be challenging. While it's possible\nto leverage emulators, write side-effect free code, or run tests on the\nhardware itself, it's often easiest to unit test the code on your personal\ncomputer with mocked hardware functionality.","2020-02-07",[12,13,14,15],"micropython","testing","mocks","tutorial",[17,18],"python","embedded",{"type":20,"children":21},"root",[22],{"type":23,"tag":24,"props":25,"children":26},"element","p",{},[27],{"type":28,"value":9},"text",{"type":20,"children":30,"toc":681},[31,35,42,47,53,75,252,278,398,419,618,646,653,675],{"type":23,"tag":24,"props":32,"children":33},{},[34],{"type":28,"value":9},{"type":23,"tag":36,"props":37,"children":39},"h1",{"id":38},"mocking",[40],{"type":28,"value":41},"Mocking",{"type":23,"tag":24,"props":43,"children":44},{},[45],{"type":28,"value":46},"Mocks allow us to replace the hardware interfacing functionality under-the-hood\nwith predefined results and side-effects. For example, if there is a piece of\nlogic that retrieves values from an accelerometer to get a device's\norientation, it would be possible to mock the returned values of the\naccelerometer -- allowing us to run the unit tests on a device that does not\nhave an accelerometer sensor installed.",{"type":23,"tag":36,"props":48,"children":50},{"id":49},"a-micropython-mocking-example",[51],{"type":28,"value":52},"A MicroPython Mocking Example",{"type":23,"tag":24,"props":54,"children":55},{},[56,58,65,67,73],{"type":28,"value":57},"In this example, we will be unit testing a module named ",{"type":23,"tag":59,"props":60,"children":62},"code",{"className":61},[],[63],{"type":28,"value":64},"time_logger",{"type":28,"value":66},", that\ndepends on the MicroPython library ",{"type":23,"tag":59,"props":68,"children":70},{"className":69},[],[71],{"type":28,"value":72},"utime",{"type":28,"value":74}," to log the most recent Epoch time to\na file.",{"type":23,"tag":76,"props":77,"children":80},"pre",{"className":78,"code":79,"language":17,"meta":7,"style":7},"language-python shiki shiki-themes github-light","# time_logger.py\n\nclass TimeLogger(object):\n\n def save_time(self):\n \"\"\" Overwrite a file with the most recent Epoch timestamp from `utime`\n \"\"\"\n with open(\"LAST_KNOWN_TIME\", \"w+\") as f:\n f.write(str(utime.time()))\n",[81],{"type":23,"tag":59,"props":82,"children":83},{"__ignoreMap":7},[84,96,106,139,147,166,176,185,233],{"type":23,"tag":85,"props":86,"children":89},"span",{"class":87,"line":88},"line",1,[90],{"type":23,"tag":85,"props":91,"children":93},{"style":92},"--shiki-default:#6A737D",[94],{"type":28,"value":95},"# time_logger.py\n",{"type":23,"tag":85,"props":97,"children":99},{"class":87,"line":98},2,[100],{"type":23,"tag":85,"props":101,"children":103},{"emptyLinePlaceholder":102},true,[104],{"type":28,"value":105},"\n",{"type":23,"tag":85,"props":107,"children":109},{"class":87,"line":108},3,[110,116,122,128,134],{"type":23,"tag":85,"props":111,"children":113},{"style":112},"--shiki-default:#D73A49",[114],{"type":28,"value":115},"class",{"type":23,"tag":85,"props":117,"children":119},{"style":118},"--shiki-default:#6F42C1",[120],{"type":28,"value":121}," TimeLogger",{"type":23,"tag":85,"props":123,"children":125},{"style":124},"--shiki-default:#24292E",[126],{"type":28,"value":127},"(",{"type":23,"tag":85,"props":129,"children":131},{"style":130},"--shiki-default:#005CC5",[132],{"type":28,"value":133},"object",{"type":23,"tag":85,"props":135,"children":136},{"style":124},[137],{"type":28,"value":138},"):\n",{"type":23,"tag":85,"props":140,"children":142},{"class":87,"line":141},4,[143],{"type":23,"tag":85,"props":144,"children":145},{"emptyLinePlaceholder":102},[146],{"type":28,"value":105},{"type":23,"tag":85,"props":148,"children":150},{"class":87,"line":149},5,[151,156,161],{"type":23,"tag":85,"props":152,"children":153},{"style":112},[154],{"type":28,"value":155}," def",{"type":23,"tag":85,"props":157,"children":158},{"style":118},[159],{"type":28,"value":160}," save_time",{"type":23,"tag":85,"props":162,"children":163},{"style":124},[164],{"type":28,"value":165},"(self):\n",{"type":23,"tag":85,"props":167,"children":169},{"class":87,"line":168},6,[170],{"type":23,"tag":85,"props":171,"children":173},{"style":172},"--shiki-default:#032F62",[174],{"type":28,"value":175}," \"\"\" Overwrite a file with the most recent Epoch timestamp from `utime`\n",{"type":23,"tag":85,"props":177,"children":179},{"class":87,"line":178},7,[180],{"type":23,"tag":85,"props":181,"children":182},{"style":172},[183],{"type":28,"value":184}," \"\"\"\n",{"type":23,"tag":85,"props":186,"children":188},{"class":87,"line":187},8,[189,194,199,203,208,213,218,223,228],{"type":23,"tag":85,"props":190,"children":191},{"style":112},[192],{"type":28,"value":193}," with",{"type":23,"tag":85,"props":195,"children":196},{"style":130},[197],{"type":28,"value":198}," open",{"type":23,"tag":85,"props":200,"children":201},{"style":124},[202],{"type":28,"value":127},{"type":23,"tag":85,"props":204,"children":205},{"style":172},[206],{"type":28,"value":207},"\"LAST_KNOWN_TIME\"",{"type":23,"tag":85,"props":209,"children":210},{"style":124},[211],{"type":28,"value":212},", ",{"type":23,"tag":85,"props":214,"children":215},{"style":172},[216],{"type":28,"value":217},"\"w+\"",{"type":23,"tag":85,"props":219,"children":220},{"style":124},[221],{"type":28,"value":222},") ",{"type":23,"tag":85,"props":224,"children":225},{"style":112},[226],{"type":28,"value":227},"as",{"type":23,"tag":85,"props":229,"children":230},{"style":124},[231],{"type":28,"value":232}," f:\n",{"type":23,"tag":85,"props":234,"children":236},{"class":87,"line":235},9,[237,242,247],{"type":23,"tag":85,"props":238,"children":239},{"style":124},[240],{"type":28,"value":241}," f.write(",{"type":23,"tag":85,"props":243,"children":244},{"style":130},[245],{"type":28,"value":246},"str",{"type":23,"tag":85,"props":248,"children":249},{"style":124},[250],{"type":28,"value":251},"(utime.time()))\n",{"type":23,"tag":24,"props":253,"children":254},{},[255,257,262,264,269,271,276],{"type":28,"value":256},"First, because the ",{"type":23,"tag":59,"props":258,"children":260},{"className":259},[],[261],{"type":28,"value":72},{"type":28,"value":263}," module is not installed on the machine that the unit\ntests on, we must mock ",{"type":23,"tag":59,"props":265,"children":267},{"className":266},[],[268],{"type":28,"value":72},{"type":28,"value":270}," module before importing ",{"type":23,"tag":59,"props":272,"children":274},{"className":273},[],[275],{"type":28,"value":64},{"type":28,"value":277}," in our\nunit test file.",{"type":23,"tag":76,"props":279,"children":281},{"className":78,"code":280,"language":17,"meta":7,"style":7},"# test_time_logger.py\n\nimport unittest\n\nfrom unittest.mock import MagicMock\n\nsys.modules['utime'] = MagicMock()\nfrom time_logger import TimeLogger\n",[282],{"type":23,"tag":59,"props":283,"children":284},{"__ignoreMap":7},[285,293,300,313,320,342,349,377],{"type":23,"tag":85,"props":286,"children":287},{"class":87,"line":88},[288],{"type":23,"tag":85,"props":289,"children":290},{"style":92},[291],{"type":28,"value":292},"# test_time_logger.py\n",{"type":23,"tag":85,"props":294,"children":295},{"class":87,"line":98},[296],{"type":23,"tag":85,"props":297,"children":298},{"emptyLinePlaceholder":102},[299],{"type":28,"value":105},{"type":23,"tag":85,"props":301,"children":302},{"class":87,"line":108},[303,308],{"type":23,"tag":85,"props":304,"children":305},{"style":112},[306],{"type":28,"value":307},"import",{"type":23,"tag":85,"props":309,"children":310},{"style":124},[311],{"type":28,"value":312}," unittest\n",{"type":23,"tag":85,"props":314,"children":315},{"class":87,"line":141},[316],{"type":23,"tag":85,"props":317,"children":318},{"emptyLinePlaceholder":102},[319],{"type":28,"value":105},{"type":23,"tag":85,"props":321,"children":322},{"class":87,"line":149},[323,328,333,337],{"type":23,"tag":85,"props":324,"children":325},{"style":112},[326],{"type":28,"value":327},"from",{"type":23,"tag":85,"props":329,"children":330},{"style":124},[331],{"type":28,"value":332}," unittest.mock ",{"type":23,"tag":85,"props":334,"children":335},{"style":112},[336],{"type":28,"value":307},{"type":23,"tag":85,"props":338,"children":339},{"style":124},[340],{"type":28,"value":341}," MagicMock\n",{"type":23,"tag":85,"props":343,"children":344},{"class":87,"line":168},[345],{"type":23,"tag":85,"props":346,"children":347},{"emptyLinePlaceholder":102},[348],{"type":28,"value":105},{"type":23,"tag":85,"props":350,"children":351},{"class":87,"line":178},[352,357,362,367,372],{"type":23,"tag":85,"props":353,"children":354},{"style":124},[355],{"type":28,"value":356},"sys.modules[",{"type":23,"tag":85,"props":358,"children":359},{"style":172},[360],{"type":28,"value":361},"'utime'",{"type":23,"tag":85,"props":363,"children":364},{"style":124},[365],{"type":28,"value":366},"] ",{"type":23,"tag":85,"props":368,"children":369},{"style":112},[370],{"type":28,"value":371},"=",{"type":23,"tag":85,"props":373,"children":374},{"style":124},[375],{"type":28,"value":376}," MagicMock()\n",{"type":23,"tag":85,"props":378,"children":379},{"class":87,"line":187},[380,384,389,393],{"type":23,"tag":85,"props":381,"children":382},{"style":112},[383],{"type":28,"value":327},{"type":23,"tag":85,"props":385,"children":386},{"style":124},[387],{"type":28,"value":388}," time_logger ",{"type":23,"tag":85,"props":390,"children":391},{"style":112},[392],{"type":28,"value":307},{"type":23,"tag":85,"props":394,"children":395},{"style":124},[396],{"type":28,"value":397}," TimeLogger\n",{"type":23,"tag":24,"props":399,"children":400},{},[401,403,409,411,417],{"type":28,"value":402},"Then, we can write a test that patches the ",{"type":23,"tag":59,"props":404,"children":406},{"className":405},[],[407],{"type":28,"value":408},"utime.time",{"type":28,"value":410}," functionality so that\nit returns a value of our choosing -- in this case, ",{"type":23,"tag":59,"props":412,"children":414},{"className":413},[],[415],{"type":28,"value":416},"1234",{"type":28,"value":418},".",{"type":23,"tag":76,"props":420,"children":422},{"className":78,"code":421,"language":17,"meta":7,"style":7},"class TestTimeLogger(unittest.TestCase):\n\n def test_save_time(self):\n \"\"\" Verify that the Epoch time is written to file\n \"\"\"\n with unittest.mock.patch(\"utime.time\", return_value=1234):\n t = TimeLogger()\n t.save_time()\n with open(\"LAST_KNOWN_TIME\") as f:\n self.assertEqual(\"1234\", f.read())\n",[423],{"type":23,"tag":59,"props":424,"children":425},{"__ignoreMap":7},[426,460,467,483,491,498,537,554,562,594],{"type":23,"tag":85,"props":427,"children":428},{"class":87,"line":88},[429,433,438,442,447,451,456],{"type":23,"tag":85,"props":430,"children":431},{"style":112},[432],{"type":28,"value":115},{"type":23,"tag":85,"props":434,"children":435},{"style":118},[436],{"type":28,"value":437}," TestTimeLogger",{"type":23,"tag":85,"props":439,"children":440},{"style":124},[441],{"type":28,"value":127},{"type":23,"tag":85,"props":443,"children":444},{"style":118},[445],{"type":28,"value":446},"unittest",{"type":23,"tag":85,"props":448,"children":449},{"style":124},[450],{"type":28,"value":418},{"type":23,"tag":85,"props":452,"children":453},{"style":118},[454],{"type":28,"value":455},"TestCase",{"type":23,"tag":85,"props":457,"children":458},{"style":124},[459],{"type":28,"value":138},{"type":23,"tag":85,"props":461,"children":462},{"class":87,"line":98},[463],{"type":23,"tag":85,"props":464,"children":465},{"emptyLinePlaceholder":102},[466],{"type":28,"value":105},{"type":23,"tag":85,"props":468,"children":469},{"class":87,"line":108},[470,474,479],{"type":23,"tag":85,"props":471,"children":472},{"style":112},[473],{"type":28,"value":155},{"type":23,"tag":85,"props":475,"children":476},{"style":118},[477],{"type":28,"value":478}," test_save_time",{"type":23,"tag":85,"props":480,"children":481},{"style":124},[482],{"type":28,"value":165},{"type":23,"tag":85,"props":484,"children":485},{"class":87,"line":141},[486],{"type":23,"tag":85,"props":487,"children":488},{"style":172},[489],{"type":28,"value":490}," \"\"\" Verify that the Epoch time is written to file\n",{"type":23,"tag":85,"props":492,"children":493},{"class":87,"line":149},[494],{"type":23,"tag":85,"props":495,"children":496},{"style":172},[497],{"type":28,"value":184},{"type":23,"tag":85,"props":499,"children":500},{"class":87,"line":168},[501,505,510,515,519,525,529,533],{"type":23,"tag":85,"props":502,"children":503},{"style":112},[504],{"type":28,"value":193},{"type":23,"tag":85,"props":506,"children":507},{"style":124},[508],{"type":28,"value":509}," unittest.mock.patch(",{"type":23,"tag":85,"props":511,"children":512},{"style":172},[513],{"type":28,"value":514},"\"utime.time\"",{"type":23,"tag":85,"props":516,"children":517},{"style":124},[518],{"type":28,"value":212},{"type":23,"tag":85,"props":520,"children":522},{"style":521},"--shiki-default:#E36209",[523],{"type":28,"value":524},"return_value",{"type":23,"tag":85,"props":526,"children":527},{"style":112},[528],{"type":28,"value":371},{"type":23,"tag":85,"props":530,"children":531},{"style":130},[532],{"type":28,"value":416},{"type":23,"tag":85,"props":534,"children":535},{"style":124},[536],{"type":28,"value":138},{"type":23,"tag":85,"props":538,"children":539},{"class":87,"line":178},[540,545,549],{"type":23,"tag":85,"props":541,"children":542},{"style":124},[543],{"type":28,"value":544}," t ",{"type":23,"tag":85,"props":546,"children":547},{"style":112},[548],{"type":28,"value":371},{"type":23,"tag":85,"props":550,"children":551},{"style":124},[552],{"type":28,"value":553}," TimeLogger()\n",{"type":23,"tag":85,"props":555,"children":556},{"class":87,"line":187},[557],{"type":23,"tag":85,"props":558,"children":559},{"style":124},[560],{"type":28,"value":561}," t.save_time()\n",{"type":23,"tag":85,"props":563,"children":564},{"class":87,"line":235},[565,570,574,578,582,586,590],{"type":23,"tag":85,"props":566,"children":567},{"style":112},[568],{"type":28,"value":569}," with",{"type":23,"tag":85,"props":571,"children":572},{"style":130},[573],{"type":28,"value":198},{"type":23,"tag":85,"props":575,"children":576},{"style":124},[577],{"type":28,"value":127},{"type":23,"tag":85,"props":579,"children":580},{"style":172},[581],{"type":28,"value":207},{"type":23,"tag":85,"props":583,"children":584},{"style":124},[585],{"type":28,"value":222},{"type":23,"tag":85,"props":587,"children":588},{"style":112},[589],{"type":28,"value":227},{"type":23,"tag":85,"props":591,"children":592},{"style":124},[593],{"type":28,"value":232},{"type":23,"tag":85,"props":595,"children":597},{"class":87,"line":596},10,[598,603,608,613],{"type":23,"tag":85,"props":599,"children":600},{"style":130},[601],{"type":28,"value":602}," self",{"type":23,"tag":85,"props":604,"children":605},{"style":124},[606],{"type":28,"value":607},".assertEqual(",{"type":23,"tag":85,"props":609,"children":610},{"style":172},[611],{"type":28,"value":612},"\"1234\"",{"type":23,"tag":85,"props":614,"children":615},{"style":124},[616],{"type":28,"value":617},", f.read())\n",{"type":23,"tag":24,"props":619,"children":620},{},[621,623,629,631,637,639,644],{"type":28,"value":622},"Now, when the ",{"type":23,"tag":59,"props":624,"children":626},{"className":625},[],[627],{"type":28,"value":628},"save_time",{"type":28,"value":630}," method gets the latest time from ",{"type":23,"tag":59,"props":632,"children":634},{"className":633},[],[635],{"type":28,"value":636},"utime.time()",{"type":28,"value":638},", the\nvalue will be patched to return ",{"type":23,"tag":59,"props":640,"children":642},{"className":641},[],[643],{"type":28,"value":416},{"type":28,"value":645},". That value will be written to a file,\nand our unit test will pass!",{"type":23,"tag":647,"props":648,"children":650},"h2",{"id":649},"references",[651],{"type":28,"value":652},"References",{"type":23,"tag":654,"props":655,"children":656},"ol",{},[657],{"type":23,"tag":658,"props":659,"children":660},"li",{},[661],{"type":23,"tag":662,"props":663,"children":667},"a",{"href":664,"rel":665},"https://docs.python.org/3/library/unittest.html",[666],"nofollow",[668,673],{"type":23,"tag":59,"props":669,"children":671},{"className":670},[],[672],{"type":28,"value":446},{"type":28,"value":674}," — Unit testing framework",{"type":23,"tag":676,"props":677,"children":678},"style",{},[679],{"type":28,"value":680},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":98,"depth":98,"links":682},[683],{"id":649,"depth":98,"text":652},"markdown","content:articles:unit-testing-micropython-with-mocks.md","content","articles/unit-testing-micropython-with-mocks.md","articles/unit-testing-micropython-with-mocks","md",1731782033542] \ No newline at end of file diff --git a/articles/unit-testing-micropython-with-mocks/index.html b/articles/unit-testing-micropython-with-mocks/index.html index badea74e..41f33956 100644 --- a/articles/unit-testing-micropython-with-mocks/index.html +++ b/articles/unit-testing-micropython-with-mocks/index.html @@ -5,35 +5,35 @@ - - - - - + + + + + - + - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + -

        Unit Testing in MicroPython with Mocks

        Unit testing code for embedded systems can be challenging. While it's possible +

        Unit Testing in MicroPython with Mocks

        Unit testing code for embedded systems can be challenging. While it's possible to leverage emulators, write side-effect free code, or run tests on the hardware itself, it's often easiest to unit test the code on your personal computer with mocked hardware functionality.

        Mocking

        Mocks allow us to replace the hardware interfacing functionality under-the-hood @@ -75,5 +75,5 @@ self.assertEqual("1234", f.read())

        Now, when the save_time method gets the latest time from utime.time(), the value will be patched to return 1234. That value will be written to a file, -and our unit test will pass!

        References

        1. unittest — Unit testing framework
        - \ No newline at end of file +and our unit test will pass!

        References

        1. unittest — Unit testing framework
        + \ No newline at end of file diff --git a/articles/vim-fugitive-gpg-pinentry/_payload.json b/articles/vim-fugitive-gpg-pinentry/_payload.json index ba01e4ae..38c305ec 100644 --- a/articles/vim-fugitive-gpg-pinentry/_payload.json +++ b/articles/vim-fugitive-gpg-pinentry/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":329},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"draft":6,"date":10,"tags":11,"categories":14,"excerpt":17,"body":56,"_type":323,"_id":324,"_source":325,"_file":326,"_stem":327,"_extension":328},"/articles/vim-fugitive-gpg-pinentry","articles",false,"","Using pinentry-mac to sign commits from vim-fugitive","In order to sign git commits from within Vim using a plugin like tpope/vim-fugitive, it is necessary to configure the gpg-agent to use a GUI based pinentry-program.","2024-05-11",[12,13],"vim","tip",[15,16],"tooling","tips",{"type":18,"children":19},"root",[20],{"type":21,"tag":22,"props":23,"children":24},"element","p",{},[25,28,37,39,46,48,54],{"type":26,"value":27},"text","In order to sign git commits from within Vim using a plugin like ",{"type":21,"tag":29,"props":30,"children":34},"a",{"href":31,"rel":32},"https://github.com/tpope/vim-fugitive",[33],"nofollow",[35],{"type":26,"value":36},"tpope/vim-fugitive",{"type":26,"value":38},", it is necessary to configure the ",{"type":21,"tag":40,"props":41,"children":43},"code",{"className":42},[],[44],{"type":26,"value":45},"gpg-agent",{"type":26,"value":47}," to use a GUI based ",{"type":21,"tag":40,"props":49,"children":51},{"className":50},[],[52],{"type":26,"value":53},"pinentry-program",{"type":26,"value":55},".",{"type":18,"children":57,"toc":319},[58,80,87,129,135,140,178,198,206,218,250,255,308,313],{"type":21,"tag":22,"props":59,"children":60},{},[61,62,67,68,73,74,79],{"type":26,"value":27},{"type":21,"tag":29,"props":63,"children":65},{"href":31,"rel":64},[33],[66],{"type":26,"value":36},{"type":26,"value":38},{"type":21,"tag":40,"props":69,"children":71},{"className":70},[],[72],{"type":26,"value":45},{"type":26,"value":47},{"type":21,"tag":40,"props":75,"children":77},{"className":76},[],[78],{"type":26,"value":53},{"type":26,"value":55},{"type":21,"tag":81,"props":82,"children":84},"h2",{"id":83},"preface",[85],{"type":26,"value":86},"Preface",{"type":21,"tag":22,"props":88,"children":89},{},[90,92,99,101,106,107,112,114,120,122,128],{"type":26,"value":91},"The man, the myth, the legend, Timothy Popallopollis himself ",{"type":21,"tag":29,"props":93,"children":96},{"href":94,"rel":95},"https://github.com/tpope/vim-fugitive/issues/846#issuecomment-253816577",[33],[97],{"type":26,"value":98},"recommends",{"type":26,"value":100}," configuring your ",{"type":21,"tag":40,"props":102,"children":104},{"className":103},[],[105],{"type":26,"value":45},{"type":26,"value":47},{"type":21,"tag":40,"props":108,"children":110},{"className":109},[],[111],{"type":26,"value":53},{"type":26,"value":113},". On macOS this can be done quite by simply installing ",{"type":21,"tag":40,"props":115,"children":117},{"className":116},[],[118],{"type":26,"value":119},"pinentry-mac",{"type":26,"value":121},", and updating your ",{"type":21,"tag":40,"props":123,"children":125},{"className":124},[],[126],{"type":26,"value":127},"gpg-agent.conf",{"type":26,"value":55},{"type":21,"tag":81,"props":130,"children":132},{"id":131},"configuration",[133],{"type":26,"value":134},"Configuration",{"type":21,"tag":22,"props":136,"children":137},{},[138],{"type":26,"value":139},"First things first, let's install the pinentry program.",{"type":21,"tag":141,"props":142,"children":146},"pre",{"code":143,"language":144,"meta":7,"className":145,"style":7},"$ brew install pinentry-mac\n","bash","language-bash shiki shiki-themes github-light",[147],{"type":21,"tag":40,"props":148,"children":149},{"__ignoreMap":7},[150],{"type":21,"tag":151,"props":152,"children":155},"span",{"class":153,"line":154},"line",1,[156,162,168,173],{"type":21,"tag":151,"props":157,"children":159},{"style":158},"--shiki-default:#6F42C1",[160],{"type":26,"value":161},"$",{"type":21,"tag":151,"props":163,"children":165},{"style":164},"--shiki-default:#032F62",[166],{"type":26,"value":167}," brew",{"type":21,"tag":151,"props":169,"children":170},{"style":164},[171],{"type":26,"value":172}," install",{"type":21,"tag":151,"props":174,"children":175},{"style":164},[176],{"type":26,"value":177}," pinentry-mac\n",{"type":21,"tag":22,"props":179,"children":180},{},[181,183,188,190,196],{"type":26,"value":182},"Then, all we need to do is set the ",{"type":21,"tag":40,"props":184,"children":186},{"className":185},[],[187],{"type":26,"value":53},{"type":26,"value":189}," option in your ",{"type":21,"tag":40,"props":191,"children":193},{"className":192},[],[194],{"type":26,"value":195},"~/.gnupg/gpg-agent.conf",{"type":26,"value":197}," file.",{"type":21,"tag":141,"props":199,"children":201},{"code":200},"default-cache-ttl 600\nmax-cache-ttl 7200\npinentry-program /opt/homebrew/bin/pinentry-mac\n",[202],{"type":21,"tag":40,"props":203,"children":204},{"__ignoreMap":7},[205],{"type":26,"value":200},{"type":21,"tag":22,"props":207,"children":208},{},[209,211,217],{"type":26,"value":210},"If your don't know the path to your pinentry program, you can throw down a quick ",{"type":21,"tag":40,"props":212,"children":214},{"className":213},[],[215],{"type":26,"value":216},"which",{"type":26,"value":55},{"type":21,"tag":141,"props":219,"children":221},{"code":220,"language":144,"meta":7,"className":145,"style":7},"$ which pinentry-mac\n/opt/homebrew/bin/pinentry-mac\n",[222],{"type":21,"tag":40,"props":223,"children":224},{"__ignoreMap":7},[225,241],{"type":21,"tag":151,"props":226,"children":227},{"class":153,"line":154},[228,232,237],{"type":21,"tag":151,"props":229,"children":230},{"style":158},[231],{"type":26,"value":161},{"type":21,"tag":151,"props":233,"children":234},{"style":164},[235],{"type":26,"value":236}," which",{"type":21,"tag":151,"props":238,"children":239},{"style":164},[240],{"type":26,"value":177},{"type":21,"tag":151,"props":242,"children":244},{"class":153,"line":243},2,[245],{"type":21,"tag":151,"props":246,"children":247},{"style":158},[248],{"type":26,"value":249},"/opt/homebrew/bin/pinentry-mac\n",{"type":21,"tag":22,"props":251,"children":252},{},[253],{"type":26,"value":254},"Or use your Homebrew prefix.",{"type":21,"tag":141,"props":256,"children":258},{"code":257,"language":144,"meta":7,"className":145,"style":7},"$ echo $(brew --prefix)/bin/pinentry-mac\n/opt/homebrew/bin/pinentry-mac\n",[259],{"type":21,"tag":40,"props":260,"children":261},{"__ignoreMap":7},[262,301],{"type":21,"tag":151,"props":263,"children":264},{"class":153,"line":154},[265,269,274,280,285,291,296],{"type":21,"tag":151,"props":266,"children":267},{"style":158},[268],{"type":26,"value":161},{"type":21,"tag":151,"props":270,"children":271},{"style":164},[272],{"type":26,"value":273}," echo",{"type":21,"tag":151,"props":275,"children":277},{"style":276},"--shiki-default:#24292E",[278],{"type":26,"value":279}," $(",{"type":21,"tag":151,"props":281,"children":282},{"style":158},[283],{"type":26,"value":284},"brew",{"type":21,"tag":151,"props":286,"children":288},{"style":287},"--shiki-default:#005CC5",[289],{"type":26,"value":290}," --prefix",{"type":21,"tag":151,"props":292,"children":293},{"style":276},[294],{"type":26,"value":295},")",{"type":21,"tag":151,"props":297,"children":298},{"style":164},[299],{"type":26,"value":300},"/bin/pinentry-mac\n",{"type":21,"tag":151,"props":302,"children":303},{"class":153,"line":243},[304],{"type":21,"tag":151,"props":305,"children":306},{"style":158},[307],{"type":26,"value":249},{"type":21,"tag":22,"props":309,"children":310},{},[311],{"type":26,"value":312},"But that's all it takes. Now, you should be prompted to enter your gpg pin in an external window when signing commits from vim.",{"type":21,"tag":314,"props":315,"children":316},"style",{},[317],{"type":26,"value":318},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":243,"depth":243,"links":320},[321,322],{"id":83,"depth":243,"text":86},{"id":131,"depth":243,"text":134},"markdown","content:articles:vim-fugitive-gpg-pinentry.md","content","articles/vim-fugitive-gpg-pinentry.md","articles/vim-fugitive-gpg-pinentry","md",1731781852998] \ No newline at end of file +[{"data":1,"prerenderedAt":329},["ShallowReactive",2],{"page-data":3},{"_path":4,"_dir":5,"_draft":6,"_partial":6,"_locale":7,"title":8,"description":9,"draft":6,"date":10,"tags":11,"categories":14,"excerpt":17,"body":56,"_type":323,"_id":324,"_source":325,"_file":326,"_stem":327,"_extension":328},"/articles/vim-fugitive-gpg-pinentry","articles",false,"","Using pinentry-mac to sign commits from vim-fugitive","In order to sign git commits from within Vim using a plugin like tpope/vim-fugitive, it is necessary to configure the gpg-agent to use a GUI based pinentry-program.","2024-05-11",[12,13],"vim","tip",[15,16],"tooling","tips",{"type":18,"children":19},"root",[20],{"type":21,"tag":22,"props":23,"children":24},"element","p",{},[25,28,37,39,46,48,54],{"type":26,"value":27},"text","In order to sign git commits from within Vim using a plugin like ",{"type":21,"tag":29,"props":30,"children":34},"a",{"href":31,"rel":32},"https://github.com/tpope/vim-fugitive",[33],"nofollow",[35],{"type":26,"value":36},"tpope/vim-fugitive",{"type":26,"value":38},", it is necessary to configure the ",{"type":21,"tag":40,"props":41,"children":43},"code",{"className":42},[],[44],{"type":26,"value":45},"gpg-agent",{"type":26,"value":47}," to use a GUI based ",{"type":21,"tag":40,"props":49,"children":51},{"className":50},[],[52],{"type":26,"value":53},"pinentry-program",{"type":26,"value":55},".",{"type":18,"children":57,"toc":319},[58,80,87,129,135,140,178,198,206,218,250,255,308,313],{"type":21,"tag":22,"props":59,"children":60},{},[61,62,67,68,73,74,79],{"type":26,"value":27},{"type":21,"tag":29,"props":63,"children":65},{"href":31,"rel":64},[33],[66],{"type":26,"value":36},{"type":26,"value":38},{"type":21,"tag":40,"props":69,"children":71},{"className":70},[],[72],{"type":26,"value":45},{"type":26,"value":47},{"type":21,"tag":40,"props":75,"children":77},{"className":76},[],[78],{"type":26,"value":53},{"type":26,"value":55},{"type":21,"tag":81,"props":82,"children":84},"h2",{"id":83},"preface",[85],{"type":26,"value":86},"Preface",{"type":21,"tag":22,"props":88,"children":89},{},[90,92,99,101,106,107,112,114,120,122,128],{"type":26,"value":91},"The man, the myth, the legend, Timothy Popallopollis himself ",{"type":21,"tag":29,"props":93,"children":96},{"href":94,"rel":95},"https://github.com/tpope/vim-fugitive/issues/846#issuecomment-253816577",[33],[97],{"type":26,"value":98},"recommends",{"type":26,"value":100}," configuring your ",{"type":21,"tag":40,"props":102,"children":104},{"className":103},[],[105],{"type":26,"value":45},{"type":26,"value":47},{"type":21,"tag":40,"props":108,"children":110},{"className":109},[],[111],{"type":26,"value":53},{"type":26,"value":113},". On macOS this can be done quite by simply installing ",{"type":21,"tag":40,"props":115,"children":117},{"className":116},[],[118],{"type":26,"value":119},"pinentry-mac",{"type":26,"value":121},", and updating your ",{"type":21,"tag":40,"props":123,"children":125},{"className":124},[],[126],{"type":26,"value":127},"gpg-agent.conf",{"type":26,"value":55},{"type":21,"tag":81,"props":130,"children":132},{"id":131},"configuration",[133],{"type":26,"value":134},"Configuration",{"type":21,"tag":22,"props":136,"children":137},{},[138],{"type":26,"value":139},"First things first, let's install the pinentry program.",{"type":21,"tag":141,"props":142,"children":146},"pre",{"code":143,"language":144,"meta":7,"className":145,"style":7},"$ brew install pinentry-mac\n","bash","language-bash shiki shiki-themes github-light",[147],{"type":21,"tag":40,"props":148,"children":149},{"__ignoreMap":7},[150],{"type":21,"tag":151,"props":152,"children":155},"span",{"class":153,"line":154},"line",1,[156,162,168,173],{"type":21,"tag":151,"props":157,"children":159},{"style":158},"--shiki-default:#6F42C1",[160],{"type":26,"value":161},"$",{"type":21,"tag":151,"props":163,"children":165},{"style":164},"--shiki-default:#032F62",[166],{"type":26,"value":167}," brew",{"type":21,"tag":151,"props":169,"children":170},{"style":164},[171],{"type":26,"value":172}," install",{"type":21,"tag":151,"props":174,"children":175},{"style":164},[176],{"type":26,"value":177}," pinentry-mac\n",{"type":21,"tag":22,"props":179,"children":180},{},[181,183,188,190,196],{"type":26,"value":182},"Then, all we need to do is set the ",{"type":21,"tag":40,"props":184,"children":186},{"className":185},[],[187],{"type":26,"value":53},{"type":26,"value":189}," option in your ",{"type":21,"tag":40,"props":191,"children":193},{"className":192},[],[194],{"type":26,"value":195},"~/.gnupg/gpg-agent.conf",{"type":26,"value":197}," file.",{"type":21,"tag":141,"props":199,"children":201},{"code":200},"default-cache-ttl 600\nmax-cache-ttl 7200\npinentry-program /opt/homebrew/bin/pinentry-mac\n",[202],{"type":21,"tag":40,"props":203,"children":204},{"__ignoreMap":7},[205],{"type":26,"value":200},{"type":21,"tag":22,"props":207,"children":208},{},[209,211,217],{"type":26,"value":210},"If your don't know the path to your pinentry program, you can throw down a quick ",{"type":21,"tag":40,"props":212,"children":214},{"className":213},[],[215],{"type":26,"value":216},"which",{"type":26,"value":55},{"type":21,"tag":141,"props":219,"children":221},{"code":220,"language":144,"meta":7,"className":145,"style":7},"$ which pinentry-mac\n/opt/homebrew/bin/pinentry-mac\n",[222],{"type":21,"tag":40,"props":223,"children":224},{"__ignoreMap":7},[225,241],{"type":21,"tag":151,"props":226,"children":227},{"class":153,"line":154},[228,232,237],{"type":21,"tag":151,"props":229,"children":230},{"style":158},[231],{"type":26,"value":161},{"type":21,"tag":151,"props":233,"children":234},{"style":164},[235],{"type":26,"value":236}," which",{"type":21,"tag":151,"props":238,"children":239},{"style":164},[240],{"type":26,"value":177},{"type":21,"tag":151,"props":242,"children":244},{"class":153,"line":243},2,[245],{"type":21,"tag":151,"props":246,"children":247},{"style":158},[248],{"type":26,"value":249},"/opt/homebrew/bin/pinentry-mac\n",{"type":21,"tag":22,"props":251,"children":252},{},[253],{"type":26,"value":254},"Or use your Homebrew prefix.",{"type":21,"tag":141,"props":256,"children":258},{"code":257,"language":144,"meta":7,"className":145,"style":7},"$ echo $(brew --prefix)/bin/pinentry-mac\n/opt/homebrew/bin/pinentry-mac\n",[259],{"type":21,"tag":40,"props":260,"children":261},{"__ignoreMap":7},[262,301],{"type":21,"tag":151,"props":263,"children":264},{"class":153,"line":154},[265,269,274,280,285,291,296],{"type":21,"tag":151,"props":266,"children":267},{"style":158},[268],{"type":26,"value":161},{"type":21,"tag":151,"props":270,"children":271},{"style":164},[272],{"type":26,"value":273}," echo",{"type":21,"tag":151,"props":275,"children":277},{"style":276},"--shiki-default:#24292E",[278],{"type":26,"value":279}," $(",{"type":21,"tag":151,"props":281,"children":282},{"style":158},[283],{"type":26,"value":284},"brew",{"type":21,"tag":151,"props":286,"children":288},{"style":287},"--shiki-default:#005CC5",[289],{"type":26,"value":290}," --prefix",{"type":21,"tag":151,"props":292,"children":293},{"style":276},[294],{"type":26,"value":295},")",{"type":21,"tag":151,"props":297,"children":298},{"style":164},[299],{"type":26,"value":300},"/bin/pinentry-mac\n",{"type":21,"tag":151,"props":302,"children":303},{"class":153,"line":243},[304],{"type":21,"tag":151,"props":305,"children":306},{"style":158},[307],{"type":26,"value":249},{"type":21,"tag":22,"props":309,"children":310},{},[311],{"type":26,"value":312},"But that's all it takes. Now, you should be prompted to enter your gpg pin in an external window when signing commits from vim.",{"type":21,"tag":314,"props":315,"children":316},"style",{},[317],{"type":26,"value":318},"html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}",{"title":7,"searchDepth":243,"depth":243,"links":320},[321,322],{"id":83,"depth":243,"text":86},{"id":131,"depth":243,"text":134},"markdown","content:articles:vim-fugitive-gpg-pinentry.md","content","articles/vim-fugitive-gpg-pinentry.md","articles/vim-fugitive-gpg-pinentry","md",1731782033242] \ No newline at end of file diff --git a/articles/vim-fugitive-gpg-pinentry/index.html b/articles/vim-fugitive-gpg-pinentry/index.html index 2ac598fe..0fa8f50f 100644 --- a/articles/vim-fugitive-gpg-pinentry/index.html +++ b/articles/vim-fugitive-gpg-pinentry/index.html @@ -5,32 +5,32 @@ - - - - - + + + + + - + - - - - - - - - - - - - - + + + + + + + + + + + + + -
        Category
        Tags

        Using pinentry-mac to sign commits from vim-fugitive

        In order to sign git commits from within Vim using a plugin like tpope/vim-fugitive, it is necessary to configure the gpg-agent to use a GUI based pinentry-program.

        Preface

        The man, the myth, the legend, Timothy Popallopollis himself recommends configuring your gpg-agent to use a GUI based pinentry-program. On macOS this can be done quite by simply installing pinentry-mac, and updating your gpg-agent.conf.

        Configuration

        First things first, let's install the pinentry program.

        $ brew install pinentry-mac
        +
        Category
        Tags

        Using pinentry-mac to sign commits from vim-fugitive

        In order to sign git commits from within Vim using a plugin like tpope/vim-fugitive, it is necessary to configure the gpg-agent to use a GUI based pinentry-program.

        Preface

        The man, the myth, the legend, Timothy Popallopollis himself recommends configuring your gpg-agent to use a GUI based pinentry-program. On macOS this can be done quite by simply installing pinentry-mac, and updating your gpg-agent.conf.

        Configuration

        First things first, let's install the pinentry program.

        $ brew install pinentry-mac
         

        Then, all we need to do is set the pinentry-program option in your ~/.gnupg/gpg-agent.conf file.

        default-cache-ttl 600
         max-cache-ttl 7200
         pinentry-program /opt/homebrew/bin/pinentry-mac
        @@ -38,5 +38,5 @@
         /opt/homebrew/bin/pinentry-mac
         

        Or use your Homebrew prefix.

        $ echo $(brew --prefix)/bin/pinentry-mac
         /opt/homebrew/bin/pinentry-mac
        -

        But that's all it takes. Now, you should be prompted to enter your gpg pin in an external window when signing commits from vim.

        - \ No newline at end of file +

        But that's all it takes. Now, you should be prompted to enter your gpg pin in an external window when signing commits from vim.

        + \ No newline at end of file diff --git a/atom/index.html b/atom/index.html index 61f8a4ec..822fd920 100644 --- a/atom/index.html +++ b/atom/index.html @@ -2,7 +2,7 @@ https://cmpadden.github.io cmpadden.github.io - 2024-11-16T18:30:52.413Z + 2024-11-16T18:33:52.662Z Nuxt static site generation + Feed for Node.js Colton Padden diff --git a/card/_payload.json b/card/_payload.json index 1160a8eb..0f08e087 100644 --- a/card/_payload.json +++ b/card/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853042] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033292] \ No newline at end of file diff --git a/card/index.html b/card/index.html index 639ae883..9aae6644 100644 --- a/card/index.html +++ b/card/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        TL;DR
        Name
        Colton
        Profession
        Astronaut
        Hobby
        Skydiving
        - \ No newline at end of file +
        TL;DR
        Name
        Colton
        Profession
        Astronaut
        Hobby
        Skydiving
        + \ No newline at end of file diff --git a/examples/nested_transitions/_payload.json b/examples/nested_transitions/_payload.json index bece40c3..320ef8dc 100644 --- a/examples/nested_transitions/_payload.json +++ b/examples/nested_transitions/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853048] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033296] \ No newline at end of file diff --git a/examples/nested_transitions/index.html b/examples/nested_transitions/index.html index 7700b4b6..f3e7a3b4 100644 --- a/examples/nested_transitions/index.html +++ b/examples/nested_transitions/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        - \ No newline at end of file +
        + \ No newline at end of file diff --git a/index.html b/index.html index 8f213511..48b0d21c 100644 --- a/index.html +++ b/index.html @@ -2,18 +2,18 @@ Colton Padden - - - - - - + + + + + + - - - + + + -
        I help educate data engineers while building the future of data orchestration at Dagster.
        Previously, I worked at Gemini building the data platform that provided company-wide insights into the exchange and business. At Georgetown University's Massive Data Institute building data warehousing, processing solutions, and portals to aid social scientists and researchers to leverage large-scale organic data. And previously I provided consulting for financial institutions and government agencies in the D.C. area around data practices, and identity and access management.
        - \ No newline at end of file +
        I help educate data engineers while building the future of data orchestration at Dagster.
        Previously, I worked at Gemini building the data platform that provided company-wide insights into the exchange and business. At Georgetown University's Massive Data Institute building data warehousing, processing solutions, and portals to aid social scientists and researchers to leverage large-scale organic data. And previously I provided consulting for financial institutions and government agencies in the D.C. area around data practices, and identity and access management.
        + \ No newline at end of file diff --git a/playground/_payload.json b/playground/_payload.json index dd37a3fb..4e2a5002 100644 --- a/playground/_payload.json +++ b/playground/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853003] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033247] \ No newline at end of file diff --git a/playground/audio/_payload.json b/playground/audio/_payload.json index e3fe7f1c..75e38db2 100644 --- a/playground/audio/_payload.json +++ b/playground/audio/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853014] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033256] \ No newline at end of file diff --git a/playground/audio/index.html b/playground/audio/index.html index c71f647d..b0fcd559 100644 --- a/playground/audio/index.html +++ b/playground/audio/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        Enable
        Disable
        Time Domain Waveform
        Frequency Spectrogram
        Frequency Bar Chart
        Frequency Buffer History
        IndexMeanMinMaxFFTBuffer
        Time Domain Buffer History
        IndexMeanMinMaxFFTBuffer
        - \ No newline at end of file +
        Enable
        Disable
        Time Domain Waveform
        Frequency Spectrogram
        Frequency Bar Chart
        Frequency Buffer History
        IndexMeanMinMaxFFTBuffer
        Time Domain Buffer History
        IndexMeanMinMaxFFTBuffer
        + \ No newline at end of file diff --git a/playground/chords/_payload.json b/playground/chords/_payload.json index c20a1491..c3281014 100644 --- a/playground/chords/_payload.json +++ b/playground/chords/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853017] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033261] \ No newline at end of file diff --git a/playground/chords/index.html b/playground/chords/index.html index d67b5f2d..66e7aabb 100644 --- a/playground/chords/index.html +++ b/playground/chords/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        Unfortunately, the Web MIDI API is not supported in all browsers...
        - \ No newline at end of file +
        Unfortunately, the Web MIDI API is not supported in all browsers...
        + \ No newline at end of file diff --git a/playground/conway/_payload.json b/playground/conway/_payload.json index 04870b49..4fb66d85 100644 --- a/playground/conway/_payload.json +++ b/playground/conway/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853005] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033249] \ No newline at end of file diff --git a/playground/conway/index.html b/playground/conway/index.html index 22b1f842..68038604 100644 --- a/playground/conway/index.html +++ b/playground/conway/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        - \ No newline at end of file +
        + \ No newline at end of file diff --git a/playground/french/_payload.json b/playground/french/_payload.json index 241882ca..2cb442b8 100644 --- a/playground/french/_payload.json +++ b/playground/french/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853010] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033252] \ No newline at end of file diff --git a/playground/french/index.html b/playground/french/index.html index abbcb2a5..a86b8f52 100644 --- a/playground/french/index.html +++ b/playground/french/index.html @@ -2,16 +2,16 @@ Colton Padden - - - - + + + + - - - + + + -
        1000 French Conjugations
        Loading...
        - \ No newline at end of file +
        1000 French Conjugations
        Loading...
        + \ No newline at end of file diff --git a/playground/index.html b/playground/index.html index ccfd8b3f..92921fe3 100644 --- a/playground/index.html +++ b/playground/index.html @@ -2,15 +2,15 @@ Colton Padden - - - - - - - + + + + + + + -
        - \ No newline at end of file +
        + \ No newline at end of file diff --git a/playground/matrix/_payload.json b/playground/matrix/_payload.json index 096bb000..a43e9ed8 100644 --- a/playground/matrix/_payload.json +++ b/playground/matrix/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853122] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033374] \ No newline at end of file diff --git a/playground/matrix/index.html b/playground/matrix/index.html index f7d1b286..8f226225 100644 --- a/playground/matrix/index.html +++ b/playground/matrix/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        Walk through the steps of matrix multiplication with randomly generated matrices. Press the Spacebar or click the buttons to iterate through the steps.
        Matrix A
        Matrix B
        - \ No newline at end of file +
        Walk through the steps of matrix multiplication with randomly generated matrices. Press the Spacebar or click the buttons to iterate through the steps.
        Matrix A
        Matrix B
        + \ No newline at end of file diff --git a/playground/metronome/_payload.json b/playground/metronome/_payload.json index fa7acfc1..f4119531 100644 --- a/playground/metronome/_payload.json +++ b/playground/metronome/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853007] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033250] \ No newline at end of file diff --git a/playground/metronome/index.html b/playground/metronome/index.html index a9a4e999..75d1b5fb 100644 --- a/playground/metronome/index.html +++ b/playground/metronome/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        - \ No newline at end of file +
        + \ No newline at end of file diff --git a/playground/midi/_payload.json b/playground/midi/_payload.json index a6e089f1..1b5aa392 100644 --- a/playground/midi/_payload.json +++ b/playground/midi/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853018] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033263] \ No newline at end of file diff --git a/playground/midi/index.html b/playground/midi/index.html index 8f6c70b8..f0b0b1b2 100644 --- a/playground/midi/index.html +++ b/playground/midi/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        Unfortunately, the Web MIDI API is not supported in all browsers...
        - \ No newline at end of file +
        Unfortunately, the Web MIDI API is not supported in all browsers...
        + \ No newline at end of file diff --git a/playground/palettes/mountains/_payload.json b/playground/palettes/mountains/_payload.json index c75e2041..ea0696b0 100644 --- a/playground/palettes/mountains/_payload.json +++ b/playground/palettes/mountains/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853012] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033254] \ No newline at end of file diff --git a/playground/palettes/mountains/index.html b/playground/palettes/mountains/index.html index 318911eb..8076ed37 100644 --- a/playground/palettes/mountains/index.html +++ b/playground/palettes/mountains/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        Click or tap anywhere to redraw the waves!
        - \ No newline at end of file +
        Click or tap anywhere to redraw the waves!
        + \ No newline at end of file diff --git a/playground/palettes/variance/_payload.json b/playground/palettes/variance/_payload.json index cc559da0..744eaa47 100644 --- a/playground/palettes/variance/_payload.json +++ b/playground/palettes/variance/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853167] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033411] \ No newline at end of file diff --git a/playground/palettes/variance/index.html b/playground/palettes/variance/index.html index 4dbf9446..a35b9bf6 100644 --- a/playground/palettes/variance/index.html +++ b/playground/palettes/variance/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        - \ No newline at end of file +
        + \ No newline at end of file diff --git a/playground/plotter/_payload.json b/playground/plotter/_payload.json index d184f682..2678d033 100644 --- a/playground/plotter/_payload.json +++ b/playground/plotter/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853165] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033409] \ No newline at end of file diff --git a/playground/plotter/index.html b/playground/plotter/index.html index 820a64fa..bbf7963d 100644 --- a/playground/plotter/index.html +++ b/playground/plotter/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        - \ No newline at end of file +
        + \ No newline at end of file diff --git a/playground/tiling/_payload.json b/playground/tiling/_payload.json index aece1f47..a71c61e3 100644 --- a/playground/tiling/_payload.json +++ b/playground/tiling/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853168] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033459] \ No newline at end of file diff --git a/playground/tiling/index.html b/playground/tiling/index.html index 8e41cdcd..0e3d2a87 100644 --- a/playground/tiling/index.html +++ b/playground/tiling/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        m-ary tree based tiling
        - \ No newline at end of file +
        m-ary tree based tiling
        + \ No newline at end of file diff --git a/playground/waves/_payload.json b/playground/waves/_payload.json index 8afa3e63..b611fd52 100644 --- a/playground/waves/_payload.json +++ b/playground/waves/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853015] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033258] \ No newline at end of file diff --git a/playground/waves/index.html b/playground/waves/index.html index bf53666c..f47809dd 100644 --- a/playground/waves/index.html +++ b/playground/waves/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -
        y(x) = A sin((2π / λ) x)
        y(x) = A cos((2π / λ) x)
        y(x) = A tan((2π / λ) x)
        Click or tap anywhere to clear the canvas!
        - \ No newline at end of file +
        y(x) = A sin((2π / λ) x)
        y(x) = A cos((2π / λ) x)
        y(x) = A tan((2π / λ) x)
        Click or tap anywhere to clear the canvas!
        + \ No newline at end of file diff --git a/talks/_payload.json b/talks/_payload.json index 99f53c00..08606484 100644 --- a/talks/_payload.json +++ b/talks/_payload.json @@ -1 +1 @@ -[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731781853216] \ No newline at end of file +[{"data":1,"prerenderedAt":3},["ShallowReactive",2],{},1731782033460] \ No newline at end of file diff --git a/talks/index.html b/talks/index.html index 4c0bac12..f55fb6c1 100644 --- a/talks/index.html +++ b/talks/index.html @@ -2,14 +2,14 @@ Colton Padden - - - - - - + + + + + + -

        Talks

        Sep 24, 2024
        Dagster Deep Dive
        Orchestrating ML Workloads with Dagster & Modal
        Aug 06, 2024
        Dagster Deep Dive
        Data Quality: Building Reliable Data Platforms
        Apr 18, 2024
        MotherDuck and Dagster
        From local development to production
        Mar 05, 2024
        Dagster Deep Dive
        Configurations and Resources
        - \ No newline at end of file +

        Talks

        Sep 24, 2024
        Dagster Deep Dive
        Orchestrating ML Workloads with Dagster & Modal
        Aug 06, 2024
        Dagster Deep Dive
        Data Quality: Building Reliable Data Platforms
        Apr 18, 2024
        MotherDuck and Dagster
        From local development to production
        Mar 05, 2024
        Dagster Deep Dive
        Configurations and Resources
        + \ No newline at end of file