flightdeck 0.2.6 → 0.2.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,8 +1,10 @@
1
1
  #!/usr/bin/env node
2
- var U=Object.defineProperty;var oe=(e,t)=>{for(var i in t)U(e,i,{get:t[i],enumerable:!0,configurable:!0,set:(o)=>t[i]=()=>o})};import*as I from"node:path";import{cli as Q,optional as W,parseBooleanOption as X,parseNumberOption as ee}from"comline";import{z as r}from"zod";import{execSync as D,spawn as M}from"node:child_process";import{createServer as G}from"node:http";import{homedir as V}from"node:os";import{resolve as _}from"node:path";import{inspect as F}from"node:util";import{Future as p}from"atom.io/internal";import{discoverType as B}from"atom.io/introspection";import{fromEntries as k,toEntries as f}from"atom.io/json";import{ChildSocket as J}from"atom.io/realtime-server";import{CronJob as z}from"cron";import{z as c}from"zod";import{existsSync as b,mkdirSync as T,readdirSync as A,readFileSync as j,rmSync as E,statSync as C,writeFileSync as O}from"node:fs";import{resolve as w}from"node:path";class S{rootDir;constructor(e){if(this.rootDir=e.path,!b(this.rootDir))T(this.rootDir,{recursive:!0})}getItem(e){let t=w(this.rootDir,e);if(b(t))return j(t,"utf-8");return null}setItem(e,t){let i=w(this.rootDir,e);O(i,t)}removeItem(e){let t=w(this.rootDir,e);if(b(t))E(t)}key(e){return A(this.rootDir).sort((o,g)=>{let a=C(o);return C(g).ctimeMs-a.ctimeMs})[e]??null}clear(){E(this.rootDir,{recursive:!0}),T(this.rootDir,{recursive:!0})}get length(){return A(this.rootDir).length}}import{createEnv as K}from"@t3-oss/env-core";import{z as H}from"zod";var R=K({server:{FLIGHTDECK_SECRET:H.string().optional()},clientPrefix:"NEVER",client:{},runtimeEnv:import.meta.env,emptyStringAsUndefined:!0});var Le=["downloaded","installed"],Pe=["notified","confirmed"];function q(e){return/^\d+\.\d+\.\d+$/.test(e)||!Number.isNaN(Number.parseFloat(e))}class L{options;safety=0;storage;webhookServer;services;serviceIdx;defaultServicesReadyToUpdate;servicesReadyToUpdate;autoRespawnDeadServices;logger;serviceLoggers;updateAvailabilityChecker=null;servicesLive;servicesDead;live=new p(()=>{});dead=new p(()=>{});restartTimes=[];constructor(e){this.options=e;let{FLIGHTDECK_SECRET:t}=R,{flightdeckRootDir:i=_(V(),".flightdeck")}=e,o=e.port??8080,g=`http://localhost:${o}`,a=f(e.services);if(this.services=k(a.map(([s])=>[s,null])),this.serviceIdx=k(a.map(([s],n)=>[s,n])),this.defaultServicesReadyToUpdate=k(a.map(([s,{waitFor:n}])=>[s,!n])),this.servicesReadyToUpdate={...this.defaultServicesReadyToUpdate},this.autoRespawnDeadServices=!0,this.logger=new u(this.options.packageName,process.pid,void 0,{jsonLogging:this.options.jsonLogging??!1}),this.serviceLoggers=k(a.map(([s])=>[s,new u(this.options.packageName,process.pid,s,{jsonLogging:this.options.jsonLogging??!1})])),this.servicesLive=a.map(()=>new p(()=>{})),this.servicesDead=a.map(()=>new p(()=>{})),this.live.use(Promise.all(this.servicesLive)),this.dead.use(Promise.all(this.servicesDead)),this.storage=new S({path:_(i,"storage",e.packageName)}),t===void 0)this.logger.warn("No FLIGHTDECK_SECRET environment variable found. FlightDeck will not run an update server.");else G((s,n)=>{let m=[];s.on("data",(l)=>{m.push(l instanceof Buffer?l:Buffer.from(l))}).on("end",()=>{let l=s.headers.authorization;try{if(typeof s.url==="undefined")throw 400;let h=`Bearer ${t}`;if(l!==`Bearer ${t}`)throw this.logger.info(`Unauthorized: needed \`${h}\`, got \`${l}\``),401;let y=new URL(s.url,g);this.logger.info(s.method,y.pathname);let v=Buffer.concat(m).toString();if(!q(v))throw 400;n.writeHead(200),n.end(),this.storage.setItem("updatePhase","notified"),this.storage.setItem("updateAwaitedVersion",v);let{checkAvailability:N}=e.scripts;if(N){this.updateAvailabilityChecker?.stop(),this.seekUpdate(v);let x=this.storage.getItem("updatePhase");if(this.logger.info('> storage("updatePhase") >',x),x==="notified")this.updateAvailabilityChecker=new z("30 * * * * *",()=>{this.seekUpdate(v)}),this.updateAvailabilityChecker.start()}else this.downloadPackage()}catch(h){if(this.logger.error(h,s.url),typeof h==="number")n.writeHead(h),n.end()}finally{m=[]}})}).listen(o,()=>{this.logger.info(`Server started on port ${o}`)});this.startAllServices().then(()=>{this.logger.info("All services started.")}).catch((s)=>{if(s instanceof Error)this.logger.error("Failed to start all services:",s.message)})}seekUpdate(e){this.logger.info("Checking for updates...");let{checkAvailability:t}=this.options.scripts;if(!t){this.logger.info("No checkAvailability script found.");return}try{let i=D(`${t} ${e}`);this.logger.info("Check stdout:",i.toString()),this.updateAvailabilityChecker?.stop(),this.storage.setItem("updatePhase","confirmed"),this.downloadPackage(),this.announceUpdate()}catch(i){if(i instanceof Error)this.logger.error("Check failed:",i.message);else{let o=B(i);this.logger.error("Check threw",o,i)}}}announceUpdate(){for(let e of f(this.services)){let[t,i]=e;if(i){if(this.options.services[t].waitFor)i.emit("updatesReady")}else this.startService(t)}}tryUpdate(){if(f(this.servicesReadyToUpdate).every(([,e])=>e))this.logger.info("All services are ready to update."),this.stopAllServices().then(()=>{this.logger.info("All services stopped; starting up fresh..."),this.startAllServices().then(()=>{this.logger.info("All services started; we're back online.")}).catch((e)=>{if(e instanceof Error)this.logger.error("Failed to start all services:",e.message)})}).catch((e)=>{if(e instanceof Error)this.logger.error("Failed to stop all services:",e.message)})}startAllServices(){this.logger.info("Starting all services..."),this.autoRespawnDeadServices=!0;let e=this.storage.getItem("setupPhase");switch(this.logger.info('> storage("setupPhase") >',e),e){case null:return this.logger.info("Starting from scratch."),this.downloadPackage(),this.installPackage(),this.startAllServices();case"downloaded":return this.logger.info("Found package downloaded but not installed."),this.installPackage(),this.startAllServices();case"installed":{for(let[t]of f(this.services))this.startService(t);return this.live}}}startService(e){if(this.logger.info(`Starting service ${this.options.packageName}::${e}, try ${this.safety}/2...`),this.safety>=2)throw new Error("Out of tries...");this.safety++;let[t,...i]=this.options.services[e].run.split(" "),o=M(t,i,{cwd:this.options.flightdeckRootDir,env:import.meta.env}),g=this.serviceLoggers[e],a=this.services[e]=new J(o,`${this.options.packageName}::${e}`,g);g.processCode=a.process.pid??-1,this.services[e].onAny((...s)=>{g.info("\uD83D\uDCAC",...s)}),this.services[e].on("readyToUpdate",()=>{this.logger.info(`Service "${e}" is ready to update.`),this.servicesReadyToUpdate[e]=!0,this.tryUpdate()}),this.services[e].on("alive",()=>{if(this.servicesLive[this.serviceIdx[e]].use(Promise.resolve()),this.servicesDead[this.serviceIdx[e]]=new p(()=>{}),this.dead.done)this.dead=new p(()=>{});this.dead.use(Promise.all(this.servicesDead))}),this.services[e].process.once("close",(s)=>{if(this.logger.info(`Auto-respawn saw "${e}" exit with code ${s}`),this.services[e]=null,!this.autoRespawnDeadServices){this.logger.info(`Auto-respawn is off; "${e}" rests.`);return}let n=this.storage.getItem("updatePhase");if(this.logger.info('> storage("updatePhase") >',n),n==="confirmed")this.serviceLoggers[e].info("Updating before startup..."),this.restartTimes=[],this.installPackage(),this.startService(e);else{let l=Date.now(),h=l-300000;if(this.restartTimes=this.restartTimes.filter((y)=>y>h),this.restartTimes.push(l),this.restartTimes.length<5)this.serviceLoggers[e].info("Crashed. Restarting..."),this.startService(e);else this.serviceLoggers[e].info("Crashed 5 times in 5 minutes. Not restarting.")}}),this.safety=0}downloadPackage(){this.logger.info("Downloading...");try{let e=D(this.options.scripts.download);this.logger.info("Download stdout:",e.toString()),this.storage.setItem("setupPhase","downloaded"),this.logger.info("Downloaded!")}catch(e){if(e instanceof Error)this.logger.error(`Failed to get the latest release: ${e.message}`);return}}installPackage(){this.logger.info("Installing...");try{let e=D(this.options.scripts.install);this.logger.info("Install stdout:",e.toString()),this.storage.setItem("setupPhase","installed"),this.logger.info("Installed!")}catch(e){if(e instanceof Error)this.logger.error(`Failed to get the latest release: ${e.message}`);return}}stopAllServices(){this.logger.info("Stopping all services... auto-respawn disabled."),this.autoRespawnDeadServices=!1;for(let[e]of f(this.services))this.stopService(e);return this.dead}stopService(e){let t=this.services[e];if(t){if(this.logger.info(`Stopping service "${e}"...`),this.servicesDead[this.serviceIdx[e]].use(new Promise((i)=>{t.emit("timeToStop"),t.process.once("close",(o)=>{this.logger.info(`Stopped service "${e}"; exited with code ${o}`),this.services[e]=null,i()})})),this.dead.use(Promise.all(this.servicesDead)),this.servicesLive[this.serviceIdx[e]]=new p(()=>{}),this.live.done)this.live=new p(()=>{});this.live.use(Promise.all(this.servicesLive))}else this.serviceLoggers[e].error("Tried to stop service, but it wasn't running.")}}var xe=c.object({level:c.union([c.literal("info"),c.literal("warn"),c.literal("ERR!")]),timestamp:c.number(),package:c.string(),service:c.string().optional(),process:c.number(),body:c.string()}),Y="line-format",Z="value",Te={title:"FlightDeck Log",description:"Format for events logged by the FlightDeck process manager.","file-type":"json","timestamp-field":"timestamp","subsecond-field":"subsecond","subsecond-units":"milli","opid-field":"service","level-field":"level",level:{info:"info",warning:"warn",error:"err!"},"ordered-by-time":!0,[Y]:[{prefix:" ",field:"__timestamp__","timestamp-format":"%Y-%m-%dT%H:%M:%S.%L%Z"},{prefix:" ",field:"process","min-width":5},{prefix:":",field:"package"},{prefix:":",field:"service","default-value":""},{prefix:"[",field:"__level__",suffix:"]"},{prefix:": ",field:"body"}],[Z]:{timestamp:{kind:"integer"},level:{kind:"string"},package:{kind:"string"},service:{kind:"string"},process:{kind:"integer"},body:{kind:"string"}}};class u{packageName;serviceName;jsonLogging;processCode;constructor(e,t,i,o){if(this.packageName=e,i)this.serviceName=i;this.processCode=t,this.jsonLogging=o?.jsonLogging??!1}info(...e){if(this.jsonLogging){let t={timestamp:Date.now(),level:"info",process:this.processCode,package:this.packageName,body:e.map((i)=>typeof i==="string"?i:F(i,!1,null,!0)).join(" ")};if(this.serviceName)t.service=this.serviceName;process.stdout.write(JSON.stringify(t)+`
3
- `)}else{let t=this.serviceName?`${this.packageName}::${this.serviceName}`:this.packageName;console.log(`${t}:`,...e)}}warn(...e){if(this.jsonLogging){let t={timestamp:Date.now(),level:"warn",process:this.processCode,package:this.packageName,body:e.map((i)=>typeof i==="string"?i:F(i,!1,null,!0)).join(" ")};if(this.serviceName)t.service=this.serviceName;process.stdout.write(JSON.stringify(t)+`
4
- `)}else{let t=this.serviceName?`${this.packageName}::${this.serviceName}`:this.packageName;console.warn(`${t}:`,...e)}}error(...e){if(this.jsonLogging){let t={timestamp:Date.now()+Math.floor(Math.random()*1000),level:"ERR!",process:this.processCode,package:this.packageName,body:e.map((i)=>typeof i==="string"?i:F(i,!1,null,!0)).join(" ")};if(this.serviceName)t.service=this.serviceName;process.stdout.write(JSON.stringify(t)+`
5
- `)}else{let t=this.serviceName?`${this.packageName}::${this.serviceName}`:this.packageName;console.error(`${t}:`,...e)}}}var d=new u("comline",process.pid,void 0,{jsonLogging:!0});Object.assign(console,{log:d.info.bind(d),info:d.info.bind(d),warn:d.warn.bind(d),error:d.error.bind(d)});var $={optionsSchema:r.object({port:r.number().optional(),packageName:r.string(),services:r.record(r.object({run:r.string(),waitFor:r.boolean()})),flightdeckRootDir:r.string(),scripts:r.object({download:r.string(),install:r.string(),checkAvailability:r.string()}),jsonLogging:r.boolean().optional()}),options:{port:{flag:"p",required:!1,description:"Port to run the flightdeck server on.",example:"--port=8080",parse:ee},packageName:{flag:"n",required:!0,description:"Name of the package.",example:'--packageName="my-app"'},services:{flag:"s",required:!0,description:"Map of service names to executables.",example:'--services="{\\"frontend\\":{\\"run\\":\\"./frontend\\",\\"waitFor\\":false},\\"backend\\":{\\"run\\":\\"./backend\\",\\"waitFor\\":true}}"',parse:JSON.parse},flightdeckRootDir:{flag:"d",required:!0,description:"Directory where the service is stored.",example:'--flightdeckRootDir="./services/sample/repo/my-app/current"'},scripts:{flag:"r",required:!0,description:"Map of scripts to run.",example:'--scripts="{\\"download\\":\\"npm i",\\"install\\":\\"npm run build\\"}"',parse:JSON.parse},jsonLogging:{flag:"j",required:!1,description:"Enable json logging.",example:"--jsonLogging",parse:X}}},te={optionsSchema:r.object({outdir:r.string().optional()}),options:{outdir:{flag:"o",required:!1,description:"Directory to write the schema to.",example:"--outdir=./dist"}}},ie=Q({cliName:"flightdeck",routes:W({schema:null,$configPath:null}),routeOptions:{"":$,$configPath:$,schema:te},debugOutput:!0,discoverConfigPath:(e)=>{if(e[0]==="schema")return;return e[0]??I.join(process.cwd(),"flightdeck.config.json")}},console),{inputs:P,writeJsonSchema:se}=ie(process.argv);switch(P.case){case"schema":{let{outdir:e}=P.opts;se(e??".")}break;default:{let e=new L(P.opts);process.on("close",async()=>{await e.stopAllServices()})}}
2
+ var K=Object.defineProperty;var ae=(e,t)=>{for(var i in t)K(e,i,{get:t[i],enumerable:!0,configurable:!0,set:(o)=>t[i]=()=>o})};import*as O from"node:path";import{cli as ee,optional as te,parseBooleanOption as ie,parseNumberOption as se}from"comline";import{z as r}from"zod";import{execSync as x,spawn as V}from"node:child_process";import{createServer as B}from"node:http";import{homedir as z}from"node:os";import{resolve as U}from"node:path";import{inspect as J}from"node:util";import{Future as d}from"atom.io/internal";import{discoverType as q}from"atom.io/introspection";import{fromEntries as k,toEntries as f}from"atom.io/json";import{ChildSocket as W}from"atom.io/realtime-server";import{CronJob as Y}from"cron";import{z as l}from"zod";import{existsSync as D,mkdirSync as A,readdirSync as C,readFileSync as H,rmSync as R,statSync as I,writeFileSync as G}from"node:fs";import{resolve as F}from"node:path";class L{rootDir;constructor(e){if(this.rootDir=e.path,!D(this.rootDir))A(this.rootDir,{recursive:!0})}getItem(e){let t=F(this.rootDir,e);if(D(t))return H(t,"utf-8");return null}setItem(e,t){let i=F(this.rootDir,e);G(i,t)}removeItem(e){let t=F(this.rootDir,e);if(D(t))R(t)}key(e){return C(this.rootDir).sort((o,n)=>{let c=I(o);return I(n).ctimeMs-c.ctimeMs})[e]??null}clear(){R(this.rootDir,{recursive:!0}),A(this.rootDir,{recursive:!0})}get length(){return C(this.rootDir).length}}import{createEnv as N}from"@t3-oss/env-core";import{z as M}from"zod";var _=N({server:{FLIGHTDECK_SECRET:M.string().optional()},clientPrefix:"NEVER",client:{},runtimeEnv:import.meta.env,emptyStringAsUndefined:!0});var Te=["downloaded","installed"],Ee=["notified","confirmed"];function Z(e){return/^\d+\.\d+\.\d+$/.test(e)||!Number.isNaN(Number.parseFloat(e))}class P{options;safety=0;storage;webhookServer;services;serviceIdx;defaultServicesReadyToUpdate;servicesReadyToUpdate;autoRespawnDeadServices;logger;serviceLoggers;updateAvailabilityChecker=null;servicesLive;servicesDead;live=new d(()=>{});dead=new d(()=>{});restartTimes=[];constructor(e){this.options=e;let{FLIGHTDECK_SECRET:t}=_,{flightdeckRootDir:i=U(z(),".flightdeck")}=e,o=e.port??8080,n=`http://localhost:${o}`,c=f(e.services);if(this.services=k(c.map(([s])=>[s,null])),this.serviceIdx=k(c.map(([s],a)=>[s,a])),this.defaultServicesReadyToUpdate=k(c.map(([s,{waitFor:a}])=>[s,!a])),this.servicesReadyToUpdate={...this.defaultServicesReadyToUpdate},this.autoRespawnDeadServices=!0,this.logger=new u(this.options.packageName,process.pid,void 0,{jsonLogging:this.options.jsonLogging??!1}),this.serviceLoggers=k(c.map(([s])=>[s,new u(this.options.packageName,process.pid,s,{jsonLogging:this.options.jsonLogging??!1})])),this.servicesLive=c.map(()=>new d(()=>{})),this.servicesDead=c.map(()=>new d(()=>{})),this.live.use(Promise.all(this.servicesLive)),this.dead.use(Promise.all(this.servicesDead)),this.storage=new L({path:U(i,"storage",e.packageName)}),t===void 0)this.logger.warn("No FLIGHTDECK_SECRET environment variable found. FlightDeck will not run an update server.");else B((s,a)=>{let m=[];s.on("data",(p)=>{m.push(p instanceof Buffer?p:Buffer.from(p))}).on("end",()=>{let p=s.headers.authorization;try{if(typeof s.url==="undefined")throw 400;let h=`Bearer ${t}`;if(p!==`Bearer ${t}`)throw this.logger.info(`Unauthorized: needed \`${h}\`, got \`${p}\``),401;let S=new URL(s.url,n);this.logger.info(s.method,S.pathname);let v=Buffer.concat(m).toString();if(!Z(v))throw 400;a.writeHead(200),a.end(),this.storage.setItem("updatePhase","notified"),this.storage.setItem("updateAwaitedVersion",v);let{checkAvailability:j}=e.scripts;if(j){this.updateAvailabilityChecker?.stop(),this.seekUpdate(v);let E=this.storage.getItem("updatePhase");if(this.logger.info('> storage("updatePhase") >',E),E==="notified")this.updateAvailabilityChecker=new Y("30 * * * * *",()=>{this.seekUpdate(v)}),this.updateAvailabilityChecker.start()}else this.downloadPackage()}catch(h){if(this.logger.error(h,s.url),typeof h==="number")a.writeHead(h),a.end()}finally{m=[]}})}).listen(o,()=>{this.logger.info(`Server started on port ${o}`)});this.startAllServices().then(()=>{this.logger.info("All services started.")}).catch((s)=>{if(s instanceof Error)this.logger.error("Failed to start all services:",s.message)})}seekUpdate(e){this.logger.info("Checking for updates...");let{checkAvailability:t}=this.options.scripts;if(!t){this.logger.info("No checkAvailability script found.");return}try{let i=x(`${t} ${e}`);this.logger.info("Check stdout:",i.toString()),this.updateAvailabilityChecker?.stop(),this.storage.setItem("updatePhase","confirmed"),this.downloadPackage(),this.announceUpdate()}catch(i){if(i instanceof Error)this.logger.error("Check failed:",i.message);else{let o=q(i);this.logger.error("Check threw",o,i)}}}announceUpdate(){for(let e of f(this.services)){let[t,i]=e;if(i){if(this.options.services[t].waitFor)i.emit("updatesReady")}else this.startService(t)}}tryUpdate(){if(f(this.servicesReadyToUpdate).every(([,e])=>e))this.logger.info("All services are ready to update."),this.stopAllServices().then(()=>{this.logger.info("All services stopped; starting up fresh..."),this.startAllServices().then(()=>{this.logger.info("All services started; we're back online.")}).catch((e)=>{if(e instanceof Error)this.logger.error("Failed to start all services:",e.message)})}).catch((e)=>{if(e instanceof Error)this.logger.error("Failed to stop all services:",e.message)})}startAllServices(){this.logger.info("Starting all services..."),this.autoRespawnDeadServices=!0;let e=this.storage.getItem("setupPhase");switch(this.logger.info('> storage("setupPhase") >',e),e){case null:return this.logger.info("Starting from scratch."),this.downloadPackage(),this.installPackage(),this.startAllServices();case"downloaded":return this.logger.info("Found package downloaded but not installed."),this.installPackage(),this.startAllServices();case"installed":{for(let[t]of f(this.services))this.startService(t);return this.live}}}startService(e){if(this.logger.info(`Starting service ${this.options.packageName}::${e}, try ${this.safety}/2...`),this.safety>=2)throw new Error("Out of tries...");this.safety++;let[t,...i]=this.options.services[e].run.split(" "),o=V(t,i,{cwd:this.options.flightdeckRootDir,env:import.meta.env}),n=this.serviceLoggers[e],c=this.services[e]=new W(o,`${this.options.packageName}::${e}`,n);n.processCode=c.process.pid??-1,this.services[e].onAny((...s)=>{n.info("\uD83D\uDCAC",...s)}),this.services[e].on("readyToUpdate",()=>{this.logger.info(`Service "${e}" is ready to update.`),this.servicesReadyToUpdate[e]=!0,this.tryUpdate()}),this.services[e].on("alive",()=>{if(this.servicesLive[this.serviceIdx[e]].use(Promise.resolve()),this.servicesDead[this.serviceIdx[e]]=new d(()=>{}),this.dead.done)this.dead=new d(()=>{});this.dead.use(Promise.all(this.servicesDead))}),this.services[e].process.once("close",(s)=>{if(this.logger.info(`Auto-respawn saw "${e}" exit with code ${s}`),this.services[e]=null,!this.autoRespawnDeadServices){this.logger.info(`Auto-respawn is off; "${e}" rests.`);return}let a=this.storage.getItem("updatePhase");if(this.logger.info('> storage("updatePhase") >',a),a==="confirmed")this.serviceLoggers[e].info("Updating before startup..."),this.restartTimes=[],this.installPackage(),this.startService(e);else{let p=Date.now(),h=p-300000;if(this.restartTimes=this.restartTimes.filter((S)=>S>h),this.restartTimes.push(p),this.restartTimes.length<5)this.serviceLoggers[e].info("Crashed. Restarting..."),this.startService(e);else this.serviceLoggers[e].info("Crashed 5 times in 5 minutes. Not restarting.")}}),this.safety=0}downloadPackage(){this.logger.info("Downloading...");try{let e=x(this.options.scripts.download);this.logger.info("Download stdout:",e.toString()),this.storage.setItem("setupPhase","downloaded"),this.logger.info("Downloaded!")}catch(e){if(e instanceof Error)this.logger.error(`Failed to get the latest release: ${e.message}`);return}}installPackage(){this.logger.info("Installing...");try{let e=x(this.options.scripts.install);this.logger.info("Install stdout:",e.toString()),this.storage.setItem("setupPhase","installed"),this.logger.info("Installed!")}catch(e){if(e instanceof Error)this.logger.error(`Failed to get the latest release: ${e.message}`);return}}stopAllServices(){this.logger.info("Stopping all services... auto-respawn disabled."),this.autoRespawnDeadServices=!1;for(let[e]of f(this.services))this.stopService(e);return this.dead}stopService(e){let t=this.services[e];if(t){if(this.logger.info(`Stopping service "${e}"...`),this.servicesDead[this.serviceIdx[e]].use(new Promise((i)=>{t.emit("timeToStop"),t.process.once("close",(o)=>{this.logger.info(`Stopped service "${e}"; exited with code ${o}`),this.services[e]=null,i()})})),this.dead.use(Promise.all(this.servicesDead)),this.servicesLive[this.serviceIdx[e]]=new d(()=>{}),this.live.done)this.live=new d(()=>{});this.live.use(Promise.all(this.servicesLive))}else this.serviceLoggers[e].error("Tried to stop service, but it wasn't running.")}}var y="info",b="warn",w="ERR!",Ae=l.object({level:l.union([l.literal(y),l.literal(b),l.literal(w)]),timestamp:l.number(),package:l.string(),service:l.string().optional(),process:l.number(),body:l.string()}),Q="line-format",X="value",Ce={title:"FlightDeck Log",description:"Format for events logged by the FlightDeck process manager.","file-type":"json","timestamp-field":"timestamp","timestamp-divisor":1000,"module-field":"package","opid-field":"service","level-field":"level",level:{info:y,warning:b,error:w},[Q]:[{field:"level"},{prefix:" ",field:"__timestamp__","timestamp-format":"%Y-%m-%dT%H:%M:%S.%L%Z"},{prefix:" ",field:"process","min-width":5},{prefix:":",field:"package"},{prefix:":",field:"service","default-value":""},{prefix:": ",field:"body"}],[X]:{timestamp:{kind:"integer"},level:{kind:"string"},package:{kind:"string"},service:{kind:"string"},process:{kind:"integer"},body:{kind:"string"}}};class u{packageName;serviceName;jsonLogging;processCode;constructor(e,t,i,o){if(this.packageName=e,i)this.serviceName=i;this.processCode=t,this.jsonLogging=o?.jsonLogging??!1}log(e,...t){if(this.jsonLogging){let i=t.map((n)=>typeof n==="string"?n:J(n,!1,null,!0)).join(" ");if(i.includes(`
3
+ `))i=`
4
+ ${i.split(`
5
+ `).join(`
6
+ `)}`;let o={timestamp:Date.now(),level:e,process:this.processCode,package:this.packageName,body:i};if(this.serviceName)o.service=this.serviceName;process.stdout.write(JSON.stringify(o)+`
7
+ `)}else{let i=this.serviceName?`${this.packageName}:${this.serviceName}`:this.packageName;switch(e){case y:console.log(`${i}:`,...t);break;case b:console.warn(`${i}:`,...t);break;case w:console.error(`${i}:`,...t);break}}}info(...e){this.log(y,...e)}warn(...e){this.log(b,...e)}error(...e){this.log(w,...e)}}var g=new u("comline",process.pid,void 0,{jsonLogging:!0});Object.assign(console,{log:g.info.bind(g),info:g.info.bind(g),warn:g.warn.bind(g),error:g.error.bind(g)});var $={optionsSchema:r.object({port:r.number().optional(),packageName:r.string(),services:r.record(r.object({run:r.string(),waitFor:r.boolean()})),flightdeckRootDir:r.string(),scripts:r.object({download:r.string(),install:r.string(),checkAvailability:r.string()}),jsonLogging:r.boolean().optional()}),options:{port:{flag:"p",required:!1,description:"Port to run the flightdeck server on.",example:"--port=8080",parse:se},packageName:{flag:"n",required:!0,description:"Name of the package.",example:'--packageName="my-app"'},services:{flag:"s",required:!0,description:"Map of service names to executables.",example:'--services="{\\"frontend\\":{\\"run\\":\\"./frontend\\",\\"waitFor\\":false},\\"backend\\":{\\"run\\":\\"./backend\\",\\"waitFor\\":true}}"',parse:JSON.parse},flightdeckRootDir:{flag:"d",required:!0,description:"Directory where the service is stored.",example:'--flightdeckRootDir="./services/sample/repo/my-app/current"'},scripts:{flag:"r",required:!0,description:"Map of scripts to run.",example:'--scripts="{\\"download\\":\\"npm i",\\"install\\":\\"npm run build\\"}"',parse:JSON.parse},jsonLogging:{flag:"j",required:!1,description:"Enable json logging.",example:"--jsonLogging",parse:ie}}},oe={optionsSchema:r.object({outdir:r.string().optional()}),options:{outdir:{flag:"o",required:!1,description:"Directory to write the schema to.",example:"--outdir=./dist"}}},re=ee({cliName:"flightdeck",routes:te({schema:null,$configPath:null}),routeOptions:{"":$,$configPath:$,schema:oe},debugOutput:!0,discoverConfigPath:(e)=>{if(e[0]==="schema")return;return e[0]??O.join(process.cwd(),"flightdeck.config.json")}},console),{inputs:T,writeJsonSchema:ne}=re(process.argv);switch(T.case){case"schema":{let{outdir:e}=T.opts;ne(e??".")}break;default:{let e=new P(T.opts);process.on("close",async()=>{await e.stopAllServices()})}}
6
8
 
7
- //# debugId=A4E0587AA743E84E64756E2164756E21
8
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/flightdeck.bin.ts", "../src/flightdeck.lib.ts", "../src/filesystem-storage.ts", "../src/flightdeck.env.ts"],
  "sourcesContent": [
    "#!/usr/bin/env node\n\nimport * as path from \"node:path\"\n\nimport type { OptionsGroup } from \"comline\"\nimport { cli, optional, parseBooleanOption, parseNumberOption } from \"comline\"\nimport { z } from \"zod\"\n\nimport type { FlightDeckOptions } from \"./flightdeck.lib\"\nimport { FlightDeck, FlightDeckLogger } from \"./flightdeck.lib\"\n\nconst CLI_LOGGER = new FlightDeckLogger(`comline`, process.pid, undefined, {\n\tjsonLogging: true,\n})\nObject.assign(console, {\n\tlog: CLI_LOGGER.info.bind(CLI_LOGGER),\n\tinfo: CLI_LOGGER.info.bind(CLI_LOGGER),\n\twarn: CLI_LOGGER.warn.bind(CLI_LOGGER),\n\terror: CLI_LOGGER.error.bind(CLI_LOGGER),\n})\n\nconst FLIGHTDECK_MANUAL = {\n\toptionsSchema: z.object({\n\t\tport: z.number().optional(),\n\t\tpackageName: z.string(),\n\t\tservices: z.record(z.object({ run: z.string(), waitFor: z.boolean() })),\n\t\tflightdeckRootDir: z.string(),\n\t\tscripts: z.object({\n\t\t\tdownload: z.string(),\n\t\t\tinstall: z.string(),\n\t\t\tcheckAvailability: z.string(),\n\t\t}),\n\t\tjsonLogging: z.boolean().optional(),\n\t}),\n\toptions: {\n\t\tport: {\n\t\t\tflag: `p`,\n\t\t\trequired: false,\n\t\t\tdescription: `Port to run the flightdeck server on.`,\n\t\t\texample: `--port=8080`,\n\t\t\tparse: parseNumberOption,\n\t\t},\n\t\tpackageName: {\n\t\t\tflag: `n`,\n\t\t\trequired: true,\n\t\t\tdescription: `Name of the package.`,\n\t\t\texample: `--packageName=\\\"my-app\\\"`,\n\t\t},\n\t\tservices: {\n\t\t\tflag: `s`,\n\t\t\trequired: true,\n\t\t\tdescription: `Map of service names to executables.`,\n\t\t\texample: `--services=\"{\\\\\"frontend\\\\\":{\\\\\"run\\\\\":\\\\\"./frontend\\\\\",\\\\\"waitFor\\\\\":false},\\\\\"backend\\\\\":{\\\\\"run\\\\\":\\\\\"./backend\\\\\",\\\\\"waitFor\\\\\":true}}\"`,\n\t\t\tparse: JSON.parse,\n\t\t},\n\t\tflightdeckRootDir: {\n\t\t\tflag: `d`,\n\t\t\trequired: true,\n\t\t\tdescription: `Directory where the service is stored.`,\n\t\t\texample: `--flightdeckRootDir=\\\"./services/sample/repo/my-app/current\\\"`,\n\t\t},\n\t\tscripts: {\n\t\t\tflag: `r`,\n\t\t\trequired: true,\n\t\t\tdescription: `Map of scripts to run.`,\n\t\t\texample: `--scripts=\"{\\\\\"download\\\\\":\\\\\"npm i\",\\\\\"install\\\\\":\\\\\"npm run build\\\\\"}\"`,\n\t\t\tparse: JSON.parse,\n\t\t},\n\t\tjsonLogging: {\n\t\t\tflag: `j`,\n\t\t\trequired: false,\n\t\t\tdescription: `Enable json logging.`,\n\t\t\texample: `--jsonLogging`,\n\t\t\tparse: parseBooleanOption,\n\t\t},\n\t},\n} satisfies OptionsGroup<FlightDeckOptions>\n\nconst SCHEMA_MANUAL = {\n\toptionsSchema: z.object({\n\t\toutdir: z.string().optional(),\n\t}),\n\toptions: {\n\t\toutdir: {\n\t\t\tflag: `o`,\n\t\t\trequired: false,\n\t\t\tdescription: `Directory to write the schema to.`,\n\t\t\texample: `--outdir=./dist`,\n\t\t},\n\t},\n} satisfies OptionsGroup<{ outdir?: string | undefined }>\n\nconst parse = cli(\n\t{\n\t\tcliName: `flightdeck`,\n\t\troutes: optional({ schema: null, $configPath: null }),\n\t\trouteOptions: {\n\t\t\t\"\": FLIGHTDECK_MANUAL,\n\t\t\t$configPath: FLIGHTDECK_MANUAL,\n\t\t\tschema: SCHEMA_MANUAL,\n\t\t},\n\t\tdebugOutput: true,\n\t\tdiscoverConfigPath: (args) => {\n\t\t\tif (args[0] === `schema`) {\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst configPath =\n\t\t\t\targs[0] ?? path.join(process.cwd(), `flightdeck.config.json`)\n\t\t\treturn configPath\n\t\t},\n\t},\n\tconsole,\n)\n\nconst { inputs, writeJsonSchema } = parse(process.argv)\n\nswitch (inputs.case) {\n\tcase `schema`:\n\t\t{\n\t\t\tconst { outdir } = inputs.opts\n\t\t\twriteJsonSchema(outdir ?? `.`)\n\t\t}\n\t\tbreak\n\tdefault: {\n\t\tconst flightDeck = new FlightDeck(inputs.opts)\n\t\tprocess.on(`close`, async () => {\n\t\t\tawait flightDeck.stopAllServices()\n\t\t})\n\t}\n}\n",
    "import { execSync, spawn } from \"node:child_process\"\nimport type { Server } from \"node:http\"\nimport { createServer } from \"node:http\"\nimport { homedir } from \"node:os\"\nimport { resolve } from \"node:path\"\nimport { inspect } from \"node:util\"\n\nimport { Future } from \"atom.io/internal\"\nimport { discoverType } from \"atom.io/introspection\"\nimport { fromEntries, toEntries } from \"atom.io/json\"\nimport { ChildSocket } from \"atom.io/realtime-server\"\nimport { CronJob } from \"cron\"\nimport { z } from \"zod\"\n\nimport type { LnavFormat } from \"../gen/lnav-format-schema.gen\"\nimport { FilesystemStorage } from \"./filesystem-storage\"\nimport { env } from \"./flightdeck.env\"\n\nexport const FLIGHTDECK_SETUP_PHASES = [`downloaded`, `installed`] as const\n\nexport type FlightDeckSetupPhase = (typeof FLIGHTDECK_SETUP_PHASES)[number]\n\nexport const FLIGHTDECK_UPDATE_PHASES = [`notified`, `confirmed`] as const\n\nexport type FlightDeckUpdatePhase = (typeof FLIGHTDECK_UPDATE_PHASES)[number]\n\nexport function isVersionNumber(version: string): boolean {\n\treturn (\n\t\t/^\\d+\\.\\d+\\.\\d+$/.test(version) || !Number.isNaN(Number.parseFloat(version))\n\t)\n}\n\nexport type FlightDeckOptions<S extends string = string> = {\n\tpackageName: string\n\tservices: { [service in S]: { run: string; waitFor: boolean } }\n\tscripts: {\n\t\tdownload: string\n\t\tinstall: string\n\t\tcheckAvailability?: string\n\t}\n\tport?: number | undefined\n\tflightdeckRootDir?: string | undefined\n\tjsonLogging?: boolean | undefined\n}\n\nexport class FlightDeck<S extends string = string> {\n\tprotected safety = 0\n\n\tprotected storage: FilesystemStorage<{\n\t\tsetupPhase: FlightDeckSetupPhase\n\t\tupdatePhase: FlightDeckUpdatePhase\n\t\tupdateAwaitedVersion: string\n\t}>\n\tprotected webhookServer: Server\n\tprotected services: {\n\t\t[service in S]: ChildSocket<\n\t\t\t{ timeToStop: []; updatesReady: [] },\n\t\t\t{ readyToUpdate: []; alive: [] }\n\t\t> | null\n\t}\n\tprotected serviceIdx: { readonly [service in S]: number }\n\tpublic defaultServicesReadyToUpdate: { readonly [service in S]: boolean }\n\tpublic servicesReadyToUpdate: { [service in S]: boolean }\n\tpublic autoRespawnDeadServices: boolean\n\n\tprotected logger: Pick<Console, `error` | `info` | `warn`>\n\tprotected serviceLoggers: {\n\t\treadonly [service in S]: FlightDeckLogger\n\t}\n\n\tprotected updateAvailabilityChecker: CronJob | null = null\n\n\tpublic servicesLive: Future<void>[]\n\tpublic servicesDead: Future<void>[]\n\tpublic live = new Future(() => {})\n\tpublic dead = new Future(() => {})\n\n\tprotected restartTimes: number[] = []\n\n\tpublic constructor(public readonly options: FlightDeckOptions<S>) {\n\t\tconst { FLIGHTDECK_SECRET } = env\n\t\tconst { flightdeckRootDir = resolve(homedir(), `.flightdeck`) } = options\n\t\tconst port = options.port ?? 8080\n\t\tconst origin = `http://localhost:${port}`\n\n\t\tconst servicesEntries = toEntries(options.services)\n\t\tthis.services = fromEntries(\n\t\t\tservicesEntries.map(([serviceName]) => [serviceName, null]),\n\t\t)\n\t\tthis.serviceIdx = fromEntries(\n\t\t\tservicesEntries.map(([serviceName], idx) => [serviceName, idx]),\n\t\t)\n\t\tthis.defaultServicesReadyToUpdate = fromEntries(\n\t\t\tservicesEntries.map(([serviceName, { waitFor }]) => [\n\t\t\t\tserviceName,\n\t\t\t\t!waitFor,\n\t\t\t]),\n\t\t)\n\t\tthis.servicesReadyToUpdate = { ...this.defaultServicesReadyToUpdate }\n\t\tthis.autoRespawnDeadServices = true\n\n\t\tthis.logger = new FlightDeckLogger(\n\t\t\tthis.options.packageName,\n\t\t\tprocess.pid,\n\t\t\tundefined,\n\t\t\t{ jsonLogging: this.options.jsonLogging ?? false },\n\t\t)\n\t\tthis.serviceLoggers = fromEntries(\n\t\t\tservicesEntries.map(([serviceName]) => [\n\t\t\t\tserviceName,\n\t\t\t\tnew FlightDeckLogger(\n\t\t\t\t\tthis.options.packageName,\n\t\t\t\t\tprocess.pid,\n\t\t\t\t\tserviceName,\n\t\t\t\t\t{ jsonLogging: this.options.jsonLogging ?? false },\n\t\t\t\t),\n\t\t\t]),\n\t\t)\n\n\t\tthis.servicesLive = servicesEntries.map(() => new Future(() => {}))\n\t\tthis.servicesDead = servicesEntries.map(() => new Future(() => {}))\n\t\tthis.live.use(Promise.all(this.servicesLive))\n\t\tthis.dead.use(Promise.all(this.servicesDead))\n\n\t\tthis.storage = new FilesystemStorage({\n\t\t\tpath: resolve(flightdeckRootDir, `storage`, options.packageName),\n\t\t})\n\n\t\tif (FLIGHTDECK_SECRET === undefined) {\n\t\t\tthis.logger.warn(\n\t\t\t\t`No FLIGHTDECK_SECRET environment variable found. FlightDeck will not run an update server.`,\n\t\t\t)\n\t\t} else {\n\t\t\tcreateServer((req, res) => {\n\t\t\t\tlet data: Uint8Array[] = []\n\t\t\t\treq\n\t\t\t\t\t.on(`data`, (chunk) => {\n\t\t\t\t\t\tdata.push(chunk instanceof Buffer ? chunk : Buffer.from(chunk))\n\t\t\t\t\t})\n\t\t\t\t\t.on(`end`, () => {\n\t\t\t\t\t\tconst authHeader = req.headers.authorization\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tif (typeof req.url === `undefined`) throw 400\n\t\t\t\t\t\t\tconst expectedAuthHeader = `Bearer ${FLIGHTDECK_SECRET}`\n\t\t\t\t\t\t\tif (authHeader !== `Bearer ${FLIGHTDECK_SECRET}`) {\n\t\t\t\t\t\t\t\tthis.logger.info(\n\t\t\t\t\t\t\t\t\t`Unauthorized: needed \\`${expectedAuthHeader}\\`, got \\`${authHeader}\\``,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\tthrow 401\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tconst url = new URL(req.url, origin)\n\t\t\t\t\t\t\tthis.logger.info(req.method, url.pathname)\n\n\t\t\t\t\t\t\tconst versionForeignInput = Buffer.concat(data).toString()\n\t\t\t\t\t\t\tif (!isVersionNumber(versionForeignInput)) {\n\t\t\t\t\t\t\t\tthrow 400\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tres.writeHead(200)\n\t\t\t\t\t\t\tres.end()\n\n\t\t\t\t\t\t\tthis.storage.setItem(`updatePhase`, `notified`)\n\t\t\t\t\t\t\tthis.storage.setItem(`updateAwaitedVersion`, versionForeignInput)\n\t\t\t\t\t\t\tconst { checkAvailability } = options.scripts\n\t\t\t\t\t\t\tif (checkAvailability) {\n\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker?.stop()\n\t\t\t\t\t\t\t\tthis.seekUpdate(versionForeignInput)\n\t\t\t\t\t\t\t\tconst updatePhase = this.storage.getItem(`updatePhase`)\n\t\t\t\t\t\t\t\tthis.logger.info(`> storage(\"updatePhase\") >`, updatePhase)\n\t\t\t\t\t\t\t\tif (updatePhase === `notified`) {\n\t\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker = new CronJob(\n\t\t\t\t\t\t\t\t\t\t`30 * * * * *`,\n\t\t\t\t\t\t\t\t\t\t() => {\n\t\t\t\t\t\t\t\t\t\t\tthis.seekUpdate(versionForeignInput)\n\t\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker.start()\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tthis.downloadPackage()\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} catch (thrown) {\n\t\t\t\t\t\t\tthis.logger.error(thrown, req.url)\n\t\t\t\t\t\t\tif (typeof thrown === `number`) {\n\t\t\t\t\t\t\t\tres.writeHead(thrown)\n\t\t\t\t\t\t\t\tres.end()\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} finally {\n\t\t\t\t\t\t\tdata = []\n\t\t\t\t\t\t}\n\t\t\t\t\t})\n\t\t\t}).listen(port, () => {\n\t\t\t\tthis.logger.info(`Server started on port ${port}`)\n\t\t\t})\n\t\t}\n\n\t\tthis.startAllServices()\n\t\t\t.then(() => {\n\t\t\t\tthis.logger.info(`All services started.`)\n\t\t\t})\n\t\t\t.catch((thrown) => {\n\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\tthis.logger.error(`Failed to start all services:`, thrown.message)\n\t\t\t\t}\n\t\t\t})\n\t}\n\n\tprotected seekUpdate(version: string): void {\n\t\tthis.logger.info(`Checking for updates...`)\n\t\tconst { checkAvailability } = this.options.scripts\n\t\tif (!checkAvailability) {\n\t\t\tthis.logger.info(`No checkAvailability script found.`)\n\t\t\treturn\n\t\t}\n\t\ttry {\n\t\t\tconst out = execSync(`${checkAvailability} ${version}`)\n\t\t\tthis.logger.info(`Check stdout:`, out.toString())\n\t\t\tthis.updateAvailabilityChecker?.stop()\n\t\t\tthis.storage.setItem(`updatePhase`, `confirmed`)\n\t\t\tthis.downloadPackage()\n\t\t\tthis.announceUpdate()\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Check failed:`, thrown.message)\n\t\t\t} else {\n\t\t\t\tconst thrownType = discoverType(thrown)\n\t\t\t\tthis.logger.error(`Check threw`, thrownType, thrown)\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected announceUpdate(): void {\n\t\tfor (const entry of toEntries(this.services)) {\n\t\t\tconst [serviceName, service] = entry\n\t\t\tif (service) {\n\t\t\t\tif (this.options.services[serviceName].waitFor) {\n\t\t\t\t\tservice.emit(`updatesReady`)\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tthis.startService(serviceName)\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected tryUpdate(): void {\n\t\tif (toEntries(this.servicesReadyToUpdate).every(([, isReady]) => isReady)) {\n\t\t\tthis.logger.info(`All services are ready to update.`)\n\t\t\tthis.stopAllServices()\n\t\t\t\t.then(() => {\n\t\t\t\t\tthis.logger.info(`All services stopped; starting up fresh...`)\n\t\t\t\t\tthis.startAllServices()\n\t\t\t\t\t\t.then(() => {\n\t\t\t\t\t\t\tthis.logger.info(`All services started; we're back online.`)\n\t\t\t\t\t\t})\n\t\t\t\t\t\t.catch((thrown) => {\n\t\t\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t\t\t`Failed to start all services:`,\n\t\t\t\t\t\t\t\t\tthrown.message,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t})\n\t\t\t\t})\n\t\t\t\t.catch((thrown) => {\n\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\tthis.logger.error(`Failed to stop all services:`, thrown.message)\n\t\t\t\t\t}\n\t\t\t\t})\n\t\t}\n\t}\n\n\tprotected startAllServices(): Future<unknown> {\n\t\tthis.logger.info(`Starting all services...`)\n\t\tthis.autoRespawnDeadServices = true\n\t\tconst setupPhase = this.storage.getItem(`setupPhase`)\n\t\tthis.logger.info(`> storage(\"setupPhase\") >`, setupPhase)\n\t\tswitch (setupPhase) {\n\t\t\tcase null:\n\t\t\t\tthis.logger.info(`Starting from scratch.`)\n\t\t\t\tthis.downloadPackage()\n\t\t\t\tthis.installPackage()\n\t\t\t\treturn this.startAllServices()\n\t\t\tcase `downloaded`:\n\t\t\t\tthis.logger.info(`Found package downloaded but not installed.`)\n\t\t\t\tthis.installPackage()\n\t\t\t\treturn this.startAllServices()\n\t\t\tcase `installed`: {\n\t\t\t\tfor (const [serviceName] of toEntries(this.services)) {\n\t\t\t\t\tthis.startService(serviceName)\n\t\t\t\t}\n\t\t\t\treturn this.live\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected startService(serviceName: S): void {\n\t\tthis.logger.info(\n\t\t\t`Starting service ${this.options.packageName}::${serviceName}, try ${this.safety}/2...`,\n\t\t)\n\t\tif (this.safety >= 2) {\n\t\t\tthrow new Error(`Out of tries...`)\n\t\t}\n\t\tthis.safety++\n\n\t\tconst [exe, ...args] = this.options.services[serviceName].run.split(` `)\n\t\tconst serviceProcess = spawn(exe, args, {\n\t\t\tcwd: this.options.flightdeckRootDir,\n\t\t\tenv: import.meta.env,\n\t\t})\n\t\tconst serviceLogger = this.serviceLoggers[serviceName]\n\t\tconst service = (this.services[serviceName] = new ChildSocket(\n\t\t\tserviceProcess,\n\t\t\t`${this.options.packageName}::${serviceName}`,\n\t\t\tserviceLogger,\n\t\t))\n\t\tserviceLogger.processCode = service.process.pid ?? -1\n\t\tthis.services[serviceName].onAny((...messages) => {\n\t\t\tserviceLogger.info(`💬`, ...messages)\n\t\t})\n\t\tthis.services[serviceName].on(`readyToUpdate`, () => {\n\t\t\tthis.logger.info(`Service \"${serviceName}\" is ready to update.`)\n\t\t\tthis.servicesReadyToUpdate[serviceName] = true\n\t\t\tthis.tryUpdate()\n\t\t})\n\t\tthis.services[serviceName].on(`alive`, () => {\n\t\t\tthis.servicesLive[this.serviceIdx[serviceName]].use(Promise.resolve())\n\t\t\tthis.servicesDead[this.serviceIdx[serviceName]] = new Future(() => {})\n\t\t\tif (this.dead.done) {\n\t\t\t\tthis.dead = new Future(() => {})\n\t\t\t}\n\t\t\tthis.dead.use(Promise.all(this.servicesDead))\n\t\t})\n\t\tthis.services[serviceName].process.once(`close`, (exitCode) => {\n\t\t\tthis.logger.info(\n\t\t\t\t`Auto-respawn saw \"${serviceName}\" exit with code ${exitCode}`,\n\t\t\t)\n\t\t\tthis.services[serviceName] = null\n\t\t\tif (!this.autoRespawnDeadServices) {\n\t\t\t\tthis.logger.info(`Auto-respawn is off; \"${serviceName}\" rests.`)\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst updatePhase = this.storage.getItem(`updatePhase`)\n\t\t\tthis.logger.info(`> storage(\"updatePhase\") >`, updatePhase)\n\t\t\tconst updatesAreReady = updatePhase === `confirmed`\n\t\t\tif (updatesAreReady) {\n\t\t\t\tthis.serviceLoggers[serviceName].info(`Updating before startup...`)\n\t\t\t\tthis.restartTimes = []\n\t\t\t\tthis.installPackage()\n\t\t\t\tthis.startService(serviceName)\n\t\t\t} else {\n\t\t\t\tconst now = Date.now()\n\t\t\t\tconst fiveMinutesAgo = now - 5 * 60 * 1000\n\t\t\t\tthis.restartTimes = this.restartTimes.filter(\n\t\t\t\t\t(time) => time > fiveMinutesAgo,\n\t\t\t\t)\n\t\t\t\tthis.restartTimes.push(now)\n\n\t\t\t\tif (this.restartTimes.length < 5) {\n\t\t\t\t\tthis.serviceLoggers[serviceName].info(`Crashed. Restarting...`)\n\t\t\t\t\tthis.startService(serviceName)\n\t\t\t\t} else {\n\t\t\t\t\tthis.serviceLoggers[serviceName].info(\n\t\t\t\t\t\t`Crashed 5 times in 5 minutes. Not restarting.`,\n\t\t\t\t\t)\n\t\t\t\t}\n\t\t\t}\n\t\t})\n\t\tthis.safety = 0\n\t}\n\n\tprotected downloadPackage(): void {\n\t\tthis.logger.info(`Downloading...`)\n\t\ttry {\n\t\t\tconst out = execSync(this.options.scripts.download)\n\t\t\tthis.logger.info(`Download stdout:`, out.toString())\n\t\t\tthis.storage.setItem(`setupPhase`, `downloaded`)\n\t\t\tthis.logger.info(`Downloaded!`)\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Failed to get the latest release: ${thrown.message}`)\n\t\t\t}\n\t\t\treturn\n\t\t}\n\t}\n\n\tprotected installPackage(): void {\n\t\tthis.logger.info(`Installing...`)\n\n\t\ttry {\n\t\t\tconst out = execSync(this.options.scripts.install)\n\t\t\tthis.logger.info(`Install stdout:`, out.toString())\n\t\t\tthis.storage.setItem(`setupPhase`, `installed`)\n\t\t\tthis.logger.info(`Installed!`)\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Failed to get the latest release: ${thrown.message}`)\n\t\t\t}\n\t\t\treturn\n\t\t}\n\t}\n\n\tpublic stopAllServices(): Future<unknown> {\n\t\tthis.logger.info(`Stopping all services... auto-respawn disabled.`)\n\t\tthis.autoRespawnDeadServices = false\n\t\tfor (const [serviceName] of toEntries(this.services)) {\n\t\t\tthis.stopService(serviceName)\n\t\t}\n\t\treturn this.dead\n\t}\n\n\tpublic stopService(serviceName: S): void {\n\t\tconst service = this.services[serviceName]\n\t\tif (service) {\n\t\t\tthis.logger.info(`Stopping service \"${serviceName}\"...`)\n\t\t\tthis.servicesDead[this.serviceIdx[serviceName]].use(\n\t\t\t\tnew Promise((pass) => {\n\t\t\t\t\tservice.emit(`timeToStop`)\n\t\t\t\t\tservice.process.once(`close`, (exitCode) => {\n\t\t\t\t\t\tthis.logger.info(\n\t\t\t\t\t\t\t`Stopped service \"${serviceName}\"; exited with code ${exitCode}`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tthis.services[serviceName] = null\n\t\t\t\t\t\tpass()\n\t\t\t\t\t})\n\t\t\t\t}),\n\t\t\t)\n\t\t\tthis.dead.use(Promise.all(this.servicesDead))\n\t\t\tthis.servicesLive[this.serviceIdx[serviceName]] = new Future(() => {})\n\t\t\tif (this.live.done) {\n\t\t\t\tthis.live = new Future(() => {})\n\t\t\t}\n\t\t\tthis.live.use(Promise.all(this.servicesLive))\n\t\t} else {\n\t\t\tthis.serviceLoggers[serviceName].error(\n\t\t\t\t`Tried to stop service, but it wasn't running.`,\n\t\t\t)\n\t\t}\n\t}\n}\n\nexport const flightDeckLogSchema = z.object({\n\tlevel: z.union([z.literal(`info`), z.literal(`warn`), z.literal(`ERR!`)]),\n\ttimestamp: z.number(),\n\tpackage: z.string(),\n\tservice: z.string().optional(),\n\tprocess: z.number(),\n\tbody: z.string(),\n})\nexport type FlightDeckLog = z.infer<typeof flightDeckLogSchema>\n\nconst LINE_FORMAT = `line-format` satisfies keyof LnavFormat\nconst VALUE = `value` satisfies keyof LnavFormat\n\nexport type LnavFormatVisualComponent = Exclude<\n\tExclude<LnavFormat[`line-format`], undefined>[number],\n\tstring\n>\n\nexport type LnavFormatBreakdown = Exclude<LnavFormat[`value`], undefined>\nexport type MemberOf<T> = T[keyof T]\nexport type LnavFormatValueDefinition = MemberOf<LnavFormatBreakdown>\n\nexport type FlightDeckFormat = {\n\t[LINE_FORMAT]: (\n\t\t| string\n\t\t| (LnavFormatVisualComponent & {\n\t\t\t\tfield: keyof FlightDeckLog | `__level__` | `__timestamp__`\n\t\t  })\n\t)[]\n\t[VALUE]: {\n\t\t[K in keyof FlightDeckLog]: LnavFormatValueDefinition & {\n\t\t\tkind: FlightDeckLog[K] extends number | undefined\n\t\t\t\t? `integer`\n\t\t\t\t: FlightDeckLog[K] extends string | undefined\n\t\t\t\t\t? `string`\n\t\t\t\t\t: never\n\t\t}\n\t}\n}\n\nexport const FLIGHTDECK_LNAV_FORMAT = {\n\ttitle: `FlightDeck Log`,\n\tdescription: `Format for events logged by the FlightDeck process manager.`,\n\t\"file-type\": `json`,\n\t\"timestamp-field\": `timestamp`,\n\t\"subsecond-field\": `subsecond`,\n\t\"subsecond-units\": `milli`,\n\t\"opid-field\": `service`,\n\t\"level-field\": `level`,\n\tlevel: {\n\t\tinfo: `info`,\n\t\twarning: `warn`,\n\t\terror: `err!`,\n\t},\n\t\"ordered-by-time\": true,\n\n\t[LINE_FORMAT]: [\n\t\t{\n\t\t\tprefix: ` `,\n\t\t\tfield: `__timestamp__`,\n\t\t\t\"timestamp-format\": `%Y-%m-%dT%H:%M:%S.%L%Z`,\n\t\t},\n\t\t{\n\t\t\tprefix: ` `,\n\t\t\tfield: `process`,\n\t\t\t\"min-width\": 5,\n\t\t},\n\t\t{\n\t\t\tprefix: `:`,\n\t\t\tfield: `package`,\n\t\t},\n\t\t{\n\t\t\tprefix: `:`,\n\t\t\tfield: `service`,\n\t\t\t\"default-value\": ``,\n\t\t},\n\t\t{\n\t\t\tprefix: `[`,\n\t\t\tfield: `__level__`,\n\t\t\tsuffix: `]`,\n\t\t},\n\t\t{\n\t\t\tprefix: `: `,\n\t\t\tfield: `body`,\n\t\t},\n\t],\n\n\t[VALUE]: {\n\t\ttimestamp: {\n\t\t\tkind: `integer`,\n\t\t},\n\t\tlevel: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tpackage: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tservice: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tprocess: {\n\t\t\tkind: `integer`,\n\t\t},\n\t\tbody: {\n\t\t\tkind: `string`,\n\t\t},\n\t},\n} as const satisfies FlightDeckFormat & LnavFormat\n\nexport class FlightDeckLogger\n\timplements Pick<Console, `error` | `info` | `warn`>\n{\n\tpublic readonly packageName: string\n\tpublic readonly serviceName?: string\n\tpublic readonly jsonLogging: boolean\n\tpublic processCode: number\n\tpublic constructor(\n\t\tpackageName: string,\n\t\tprocessCode: number,\n\t\tserviceName?: string,\n\t\toptions?: { jsonLogging: boolean },\n\t) {\n\t\tthis.packageName = packageName\n\t\tif (serviceName) {\n\t\t\tthis.serviceName = serviceName\n\t\t}\n\t\tthis.processCode = processCode\n\t\tthis.jsonLogging = options?.jsonLogging ?? false\n\t}\n\tpublic info(...messages: unknown[]): void {\n\t\tif (this.jsonLogging) {\n\t\t\tconst log: FlightDeckLog = {\n\t\t\t\ttimestamp: Date.now(),\n\t\t\t\tlevel: `info`,\n\t\t\t\tprocess: this.processCode,\n\t\t\t\tpackage: this.packageName,\n\t\t\t\tbody: messages\n\t\t\t\t\t.map((message) =>\n\t\t\t\t\t\ttypeof message === `string`\n\t\t\t\t\t\t\t? message\n\t\t\t\t\t\t\t: inspect(message, false, null, true),\n\t\t\t\t\t)\n\t\t\t\t\t.join(` `),\n\t\t\t}\n\t\t\tif (this.serviceName) {\n\t\t\t\tlog.service = this.serviceName\n\t\t\t}\n\t\t\tprocess.stdout.write(JSON.stringify(log) + `\\n`)\n\t\t} else {\n\t\t\tconst source = this.serviceName\n\t\t\t\t? `${this.packageName}::${this.serviceName}`\n\t\t\t\t: this.packageName\n\t\t\tconsole.log(`${source}:`, ...messages)\n\t\t}\n\t}\n\n\tpublic warn(...messages: unknown[]): void {\n\t\tif (this.jsonLogging) {\n\t\t\tconst log: FlightDeckLog = {\n\t\t\t\ttimestamp: Date.now(),\n\t\t\t\tlevel: `warn`,\n\t\t\t\tprocess: this.processCode,\n\t\t\t\tpackage: this.packageName,\n\t\t\t\tbody: messages\n\t\t\t\t\t.map((message) =>\n\t\t\t\t\t\ttypeof message === `string`\n\t\t\t\t\t\t\t? message\n\t\t\t\t\t\t\t: inspect(message, false, null, true),\n\t\t\t\t\t)\n\t\t\t\t\t.join(` `),\n\t\t\t}\n\t\t\tif (this.serviceName) {\n\t\t\t\tlog.service = this.serviceName\n\t\t\t}\n\t\t\tprocess.stdout.write(JSON.stringify(log) + `\\n`)\n\t\t} else {\n\t\t\tconst source = this.serviceName\n\t\t\t\t? `${this.packageName}::${this.serviceName}`\n\t\t\t\t: this.packageName\n\t\t\tconsole.warn(`${source}:`, ...messages)\n\t\t}\n\t}\n\n\tpublic error(...messages: unknown[]): void {\n\t\tif (this.jsonLogging) {\n\t\t\tconst log: FlightDeckLog = {\n\t\t\t\ttimestamp: Date.now() + Math.floor(Math.random() * 1000),\n\t\t\t\tlevel: `ERR!`,\n\t\t\t\tprocess: this.processCode,\n\t\t\t\tpackage: this.packageName,\n\t\t\t\tbody: messages\n\t\t\t\t\t.map((message) =>\n\t\t\t\t\t\ttypeof message === `string`\n\t\t\t\t\t\t\t? message\n\t\t\t\t\t\t\t: inspect(message, false, null, true),\n\t\t\t\t\t)\n\t\t\t\t\t.join(` `),\n\t\t\t}\n\t\t\tif (this.serviceName) {\n\t\t\t\tlog.service = this.serviceName\n\t\t\t}\n\t\t\tprocess.stdout.write(JSON.stringify(log) + `\\n`)\n\t\t} else {\n\t\t\tconst source = this.serviceName\n\t\t\t\t? `${this.packageName}::${this.serviceName}`\n\t\t\t\t: this.packageName\n\t\t\tconsole.error(`${source}:`, ...messages)\n\t\t}\n\t}\n}\n",
    "import {\n\texistsSync,\n\tmkdirSync,\n\treaddirSync,\n\treadFileSync,\n\trmSync,\n\tstatSync,\n\twriteFileSync,\n} from \"node:fs\"\nimport { resolve } from \"node:path\"\n\nexport type FilesystemStorageOptions = {\n\tpath: string\n}\n\nexport class FilesystemStorage<\n\tT extends Record<string, string> = Record<string, string>,\n> implements Storage\n{\n\tpublic rootDir: string\n\n\tpublic constructor(options: FilesystemStorageOptions) {\n\t\tthis.rootDir = options.path\n\t\tif (!existsSync(this.rootDir)) {\n\t\t\tmkdirSync(this.rootDir, { recursive: true })\n\t\t}\n\t}\n\n\tpublic getItem<K extends string & keyof T>(key: K): T[K] | null {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\tif (existsSync(filePath)) {\n\t\t\treturn readFileSync(filePath, `utf-8`) as T[K]\n\t\t}\n\t\treturn null\n\t}\n\n\tpublic setItem<K extends string & keyof T>(key: K, value: T[K]): void {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\twriteFileSync(filePath, value)\n\t}\n\n\tpublic removeItem<K extends string & keyof T>(key: K): void {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\tif (existsSync(filePath)) {\n\t\t\trmSync(filePath)\n\t\t}\n\t}\n\n\tpublic key(index: number): (string & keyof T) | null {\n\t\tconst filePaths = readdirSync(this.rootDir)\n\t\tconst filePathsByDateCreated = filePaths.sort((a, b) => {\n\t\t\tconst aStat = statSync(a)\n\t\t\tconst bStat = statSync(b)\n\t\t\treturn bStat.ctimeMs - aStat.ctimeMs\n\t\t})\n\t\treturn (filePathsByDateCreated[index] as string & keyof T) ?? null\n\t}\n\n\tpublic clear(): void {\n\t\trmSync(this.rootDir, { recursive: true })\n\t\tmkdirSync(this.rootDir, { recursive: true })\n\t}\n\n\tpublic get length(): number {\n\t\treturn readdirSync(this.rootDir).length\n\t}\n}\n",
    "import { createEnv } from \"@t3-oss/env-core\"\nimport { z } from \"zod\"\n\nexport const env = createEnv({\n\tserver: { FLIGHTDECK_SECRET: z.string().optional() },\n\tclientPrefix: `NEVER`,\n\tclient: {},\n\truntimeEnv: import.meta.env,\n\temptyStringAsUndefined: true,\n})\n"
  ],
  "mappings": "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",
  "debugId": "A4E0587AA743E84E64756E2164756E21",
  "names": []
}
9
+ //# debugId=8C257A7694D940D664756E2164756E21
10
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/flightdeck.bin.ts", "../src/flightdeck.lib.ts", "../src/filesystem-storage.ts", "../src/flightdeck.env.ts"],
  "sourcesContent": [
    "#!/usr/bin/env node\n\nimport * as path from \"node:path\"\n\nimport type { OptionsGroup } from \"comline\"\nimport { cli, optional, parseBooleanOption, parseNumberOption } from \"comline\"\nimport { z } from \"zod\"\n\nimport type { FlightDeckOptions } from \"./flightdeck.lib\"\nimport { FlightDeck, FlightDeckLogger } from \"./flightdeck.lib\"\n\nconst CLI_LOGGER = new FlightDeckLogger(`comline`, process.pid, undefined, {\n\tjsonLogging: true,\n})\nObject.assign(console, {\n\tlog: CLI_LOGGER.info.bind(CLI_LOGGER),\n\tinfo: CLI_LOGGER.info.bind(CLI_LOGGER),\n\twarn: CLI_LOGGER.warn.bind(CLI_LOGGER),\n\terror: CLI_LOGGER.error.bind(CLI_LOGGER),\n})\n\nconst FLIGHTDECK_MANUAL = {\n\toptionsSchema: z.object({\n\t\tport: z.number().optional(),\n\t\tpackageName: z.string(),\n\t\tservices: z.record(z.object({ run: z.string(), waitFor: z.boolean() })),\n\t\tflightdeckRootDir: z.string(),\n\t\tscripts: z.object({\n\t\t\tdownload: z.string(),\n\t\t\tinstall: z.string(),\n\t\t\tcheckAvailability: z.string(),\n\t\t}),\n\t\tjsonLogging: z.boolean().optional(),\n\t}),\n\toptions: {\n\t\tport: {\n\t\t\tflag: `p`,\n\t\t\trequired: false,\n\t\t\tdescription: `Port to run the flightdeck server on.`,\n\t\t\texample: `--port=8080`,\n\t\t\tparse: parseNumberOption,\n\t\t},\n\t\tpackageName: {\n\t\t\tflag: `n`,\n\t\t\trequired: true,\n\t\t\tdescription: `Name of the package.`,\n\t\t\texample: `--packageName=\\\"my-app\\\"`,\n\t\t},\n\t\tservices: {\n\t\t\tflag: `s`,\n\t\t\trequired: true,\n\t\t\tdescription: `Map of service names to executables.`,\n\t\t\texample: `--services=\"{\\\\\"frontend\\\\\":{\\\\\"run\\\\\":\\\\\"./frontend\\\\\",\\\\\"waitFor\\\\\":false},\\\\\"backend\\\\\":{\\\\\"run\\\\\":\\\\\"./backend\\\\\",\\\\\"waitFor\\\\\":true}}\"`,\n\t\t\tparse: JSON.parse,\n\t\t},\n\t\tflightdeckRootDir: {\n\t\t\tflag: `d`,\n\t\t\trequired: true,\n\t\t\tdescription: `Directory where the service is stored.`,\n\t\t\texample: `--flightdeckRootDir=\\\"./services/sample/repo/my-app/current\\\"`,\n\t\t},\n\t\tscripts: {\n\t\t\tflag: `r`,\n\t\t\trequired: true,\n\t\t\tdescription: `Map of scripts to run.`,\n\t\t\texample: `--scripts=\"{\\\\\"download\\\\\":\\\\\"npm i\",\\\\\"install\\\\\":\\\\\"npm run build\\\\\"}\"`,\n\t\t\tparse: JSON.parse,\n\t\t},\n\t\tjsonLogging: {\n\t\t\tflag: `j`,\n\t\t\trequired: false,\n\t\t\tdescription: `Enable json logging.`,\n\t\t\texample: `--jsonLogging`,\n\t\t\tparse: parseBooleanOption,\n\t\t},\n\t},\n} satisfies OptionsGroup<FlightDeckOptions>\n\nconst SCHEMA_MANUAL = {\n\toptionsSchema: z.object({\n\t\toutdir: z.string().optional(),\n\t}),\n\toptions: {\n\t\toutdir: {\n\t\t\tflag: `o`,\n\t\t\trequired: false,\n\t\t\tdescription: `Directory to write the schema to.`,\n\t\t\texample: `--outdir=./dist`,\n\t\t},\n\t},\n} satisfies OptionsGroup<{ outdir?: string | undefined }>\n\nconst parse = cli(\n\t{\n\t\tcliName: `flightdeck`,\n\t\troutes: optional({ schema: null, $configPath: null }),\n\t\trouteOptions: {\n\t\t\t\"\": FLIGHTDECK_MANUAL,\n\t\t\t$configPath: FLIGHTDECK_MANUAL,\n\t\t\tschema: SCHEMA_MANUAL,\n\t\t},\n\t\tdebugOutput: true,\n\t\tdiscoverConfigPath: (args) => {\n\t\t\tif (args[0] === `schema`) {\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst configPath =\n\t\t\t\targs[0] ?? path.join(process.cwd(), `flightdeck.config.json`)\n\t\t\treturn configPath\n\t\t},\n\t},\n\tconsole,\n)\n\nconst { inputs, writeJsonSchema } = parse(process.argv)\n\nswitch (inputs.case) {\n\tcase `schema`:\n\t\t{\n\t\t\tconst { outdir } = inputs.opts\n\t\t\twriteJsonSchema(outdir ?? `.`)\n\t\t}\n\t\tbreak\n\tdefault: {\n\t\tconst flightDeck = new FlightDeck(inputs.opts)\n\t\tprocess.on(`close`, async () => {\n\t\t\tawait flightDeck.stopAllServices()\n\t\t})\n\t}\n}\n",
    "import { execSync, spawn } from \"node:child_process\"\nimport type { Server } from \"node:http\"\nimport { createServer } from \"node:http\"\nimport { homedir } from \"node:os\"\nimport { resolve } from \"node:path\"\nimport { inspect } from \"node:util\"\n\nimport { Future } from \"atom.io/internal\"\nimport { discoverType } from \"atom.io/introspection\"\nimport { fromEntries, toEntries } from \"atom.io/json\"\nimport { ChildSocket } from \"atom.io/realtime-server\"\nimport { CronJob } from \"cron\"\nimport { z } from \"zod\"\n\nimport type { LnavFormat } from \"../gen/lnav-format-schema.gen\"\nimport { FilesystemStorage } from \"./filesystem-storage\"\nimport { env } from \"./flightdeck.env\"\n\nexport const FLIGHTDECK_SETUP_PHASES = [`downloaded`, `installed`] as const\n\nexport type FlightDeckSetupPhase = (typeof FLIGHTDECK_SETUP_PHASES)[number]\n\nexport const FLIGHTDECK_UPDATE_PHASES = [`notified`, `confirmed`] as const\n\nexport type FlightDeckUpdatePhase = (typeof FLIGHTDECK_UPDATE_PHASES)[number]\n\nexport function isVersionNumber(version: string): boolean {\n\treturn (\n\t\t/^\\d+\\.\\d+\\.\\d+$/.test(version) || !Number.isNaN(Number.parseFloat(version))\n\t)\n}\n\nexport type FlightDeckOptions<S extends string = string> = {\n\tpackageName: string\n\tservices: { [service in S]: { run: string; waitFor: boolean } }\n\tscripts: {\n\t\tdownload: string\n\t\tinstall: string\n\t\tcheckAvailability?: string\n\t}\n\tport?: number | undefined\n\tflightdeckRootDir?: string | undefined\n\tjsonLogging?: boolean | undefined\n}\n\nexport class FlightDeck<S extends string = string> {\n\tprotected safety = 0\n\n\tprotected storage: FilesystemStorage<{\n\t\tsetupPhase: FlightDeckSetupPhase\n\t\tupdatePhase: FlightDeckUpdatePhase\n\t\tupdateAwaitedVersion: string\n\t}>\n\tprotected webhookServer: Server\n\tprotected services: {\n\t\t[service in S]: ChildSocket<\n\t\t\t{ timeToStop: []; updatesReady: [] },\n\t\t\t{ readyToUpdate: []; alive: [] }\n\t\t> | null\n\t}\n\tprotected serviceIdx: { readonly [service in S]: number }\n\tpublic defaultServicesReadyToUpdate: { readonly [service in S]: boolean }\n\tpublic servicesReadyToUpdate: { [service in S]: boolean }\n\tpublic autoRespawnDeadServices: boolean\n\n\tprotected logger: Pick<Console, `error` | `info` | `warn`>\n\tprotected serviceLoggers: {\n\t\treadonly [service in S]: FlightDeckLogger\n\t}\n\n\tprotected updateAvailabilityChecker: CronJob | null = null\n\n\tpublic servicesLive: Future<void>[]\n\tpublic servicesDead: Future<void>[]\n\tpublic live = new Future(() => {})\n\tpublic dead = new Future(() => {})\n\n\tprotected restartTimes: number[] = []\n\n\tpublic constructor(public readonly options: FlightDeckOptions<S>) {\n\t\tconst { FLIGHTDECK_SECRET } = env\n\t\tconst { flightdeckRootDir = resolve(homedir(), `.flightdeck`) } = options\n\t\tconst port = options.port ?? 8080\n\t\tconst origin = `http://localhost:${port}`\n\n\t\tconst servicesEntries = toEntries(options.services)\n\t\tthis.services = fromEntries(\n\t\t\tservicesEntries.map(([serviceName]) => [serviceName, null]),\n\t\t)\n\t\tthis.serviceIdx = fromEntries(\n\t\t\tservicesEntries.map(([serviceName], idx) => [serviceName, idx]),\n\t\t)\n\t\tthis.defaultServicesReadyToUpdate = fromEntries(\n\t\t\tservicesEntries.map(([serviceName, { waitFor }]) => [\n\t\t\t\tserviceName,\n\t\t\t\t!waitFor,\n\t\t\t]),\n\t\t)\n\t\tthis.servicesReadyToUpdate = { ...this.defaultServicesReadyToUpdate }\n\t\tthis.autoRespawnDeadServices = true\n\n\t\tthis.logger = new FlightDeckLogger(\n\t\t\tthis.options.packageName,\n\t\t\tprocess.pid,\n\t\t\tundefined,\n\t\t\t{ jsonLogging: this.options.jsonLogging ?? false },\n\t\t)\n\t\tthis.serviceLoggers = fromEntries(\n\t\t\tservicesEntries.map(([serviceName]) => [\n\t\t\t\tserviceName,\n\t\t\t\tnew FlightDeckLogger(\n\t\t\t\t\tthis.options.packageName,\n\t\t\t\t\tprocess.pid,\n\t\t\t\t\tserviceName,\n\t\t\t\t\t{ jsonLogging: this.options.jsonLogging ?? false },\n\t\t\t\t),\n\t\t\t]),\n\t\t)\n\n\t\tthis.servicesLive = servicesEntries.map(() => new Future(() => {}))\n\t\tthis.servicesDead = servicesEntries.map(() => new Future(() => {}))\n\t\tthis.live.use(Promise.all(this.servicesLive))\n\t\tthis.dead.use(Promise.all(this.servicesDead))\n\n\t\tthis.storage = new FilesystemStorage({\n\t\t\tpath: resolve(flightdeckRootDir, `storage`, options.packageName),\n\t\t})\n\n\t\tif (FLIGHTDECK_SECRET === undefined) {\n\t\t\tthis.logger.warn(\n\t\t\t\t`No FLIGHTDECK_SECRET environment variable found. FlightDeck will not run an update server.`,\n\t\t\t)\n\t\t} else {\n\t\t\tcreateServer((req, res) => {\n\t\t\t\tlet data: Uint8Array[] = []\n\t\t\t\treq\n\t\t\t\t\t.on(`data`, (chunk) => {\n\t\t\t\t\t\tdata.push(chunk instanceof Buffer ? chunk : Buffer.from(chunk))\n\t\t\t\t\t})\n\t\t\t\t\t.on(`end`, () => {\n\t\t\t\t\t\tconst authHeader = req.headers.authorization\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tif (typeof req.url === `undefined`) throw 400\n\t\t\t\t\t\t\tconst expectedAuthHeader = `Bearer ${FLIGHTDECK_SECRET}`\n\t\t\t\t\t\t\tif (authHeader !== `Bearer ${FLIGHTDECK_SECRET}`) {\n\t\t\t\t\t\t\t\tthis.logger.info(\n\t\t\t\t\t\t\t\t\t`Unauthorized: needed \\`${expectedAuthHeader}\\`, got \\`${authHeader}\\``,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\tthrow 401\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tconst url = new URL(req.url, origin)\n\t\t\t\t\t\t\tthis.logger.info(req.method, url.pathname)\n\n\t\t\t\t\t\t\tconst versionForeignInput = Buffer.concat(data).toString()\n\t\t\t\t\t\t\tif (!isVersionNumber(versionForeignInput)) {\n\t\t\t\t\t\t\t\tthrow 400\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tres.writeHead(200)\n\t\t\t\t\t\t\tres.end()\n\n\t\t\t\t\t\t\tthis.storage.setItem(`updatePhase`, `notified`)\n\t\t\t\t\t\t\tthis.storage.setItem(`updateAwaitedVersion`, versionForeignInput)\n\t\t\t\t\t\t\tconst { checkAvailability } = options.scripts\n\t\t\t\t\t\t\tif (checkAvailability) {\n\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker?.stop()\n\t\t\t\t\t\t\t\tthis.seekUpdate(versionForeignInput)\n\t\t\t\t\t\t\t\tconst updatePhase = this.storage.getItem(`updatePhase`)\n\t\t\t\t\t\t\t\tthis.logger.info(`> storage(\"updatePhase\") >`, updatePhase)\n\t\t\t\t\t\t\t\tif (updatePhase === `notified`) {\n\t\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker = new CronJob(\n\t\t\t\t\t\t\t\t\t\t`30 * * * * *`,\n\t\t\t\t\t\t\t\t\t\t() => {\n\t\t\t\t\t\t\t\t\t\t\tthis.seekUpdate(versionForeignInput)\n\t\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker.start()\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tthis.downloadPackage()\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} catch (thrown) {\n\t\t\t\t\t\t\tthis.logger.error(thrown, req.url)\n\t\t\t\t\t\t\tif (typeof thrown === `number`) {\n\t\t\t\t\t\t\t\tres.writeHead(thrown)\n\t\t\t\t\t\t\t\tres.end()\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} finally {\n\t\t\t\t\t\t\tdata = []\n\t\t\t\t\t\t}\n\t\t\t\t\t})\n\t\t\t}).listen(port, () => {\n\t\t\t\tthis.logger.info(`Server started on port ${port}`)\n\t\t\t})\n\t\t}\n\n\t\tthis.startAllServices()\n\t\t\t.then(() => {\n\t\t\t\tthis.logger.info(`All services started.`)\n\t\t\t})\n\t\t\t.catch((thrown) => {\n\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\tthis.logger.error(`Failed to start all services:`, thrown.message)\n\t\t\t\t}\n\t\t\t})\n\t}\n\n\tprotected seekUpdate(version: string): void {\n\t\tthis.logger.info(`Checking for updates...`)\n\t\tconst { checkAvailability } = this.options.scripts\n\t\tif (!checkAvailability) {\n\t\t\tthis.logger.info(`No checkAvailability script found.`)\n\t\t\treturn\n\t\t}\n\t\ttry {\n\t\t\tconst out = execSync(`${checkAvailability} ${version}`)\n\t\t\tthis.logger.info(`Check stdout:`, out.toString())\n\t\t\tthis.updateAvailabilityChecker?.stop()\n\t\t\tthis.storage.setItem(`updatePhase`, `confirmed`)\n\t\t\tthis.downloadPackage()\n\t\t\tthis.announceUpdate()\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Check failed:`, thrown.message)\n\t\t\t} else {\n\t\t\t\tconst thrownType = discoverType(thrown)\n\t\t\t\tthis.logger.error(`Check threw`, thrownType, thrown)\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected announceUpdate(): void {\n\t\tfor (const entry of toEntries(this.services)) {\n\t\t\tconst [serviceName, service] = entry\n\t\t\tif (service) {\n\t\t\t\tif (this.options.services[serviceName].waitFor) {\n\t\t\t\t\tservice.emit(`updatesReady`)\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tthis.startService(serviceName)\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected tryUpdate(): void {\n\t\tif (toEntries(this.servicesReadyToUpdate).every(([, isReady]) => isReady)) {\n\t\t\tthis.logger.info(`All services are ready to update.`)\n\t\t\tthis.stopAllServices()\n\t\t\t\t.then(() => {\n\t\t\t\t\tthis.logger.info(`All services stopped; starting up fresh...`)\n\t\t\t\t\tthis.startAllServices()\n\t\t\t\t\t\t.then(() => {\n\t\t\t\t\t\t\tthis.logger.info(`All services started; we're back online.`)\n\t\t\t\t\t\t})\n\t\t\t\t\t\t.catch((thrown) => {\n\t\t\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t\t\t`Failed to start all services:`,\n\t\t\t\t\t\t\t\t\tthrown.message,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t})\n\t\t\t\t})\n\t\t\t\t.catch((thrown) => {\n\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\tthis.logger.error(`Failed to stop all services:`, thrown.message)\n\t\t\t\t\t}\n\t\t\t\t})\n\t\t}\n\t}\n\n\tprotected startAllServices(): Future<unknown> {\n\t\tthis.logger.info(`Starting all services...`)\n\t\tthis.autoRespawnDeadServices = true\n\t\tconst setupPhase = this.storage.getItem(`setupPhase`)\n\t\tthis.logger.info(`> storage(\"setupPhase\") >`, setupPhase)\n\t\tswitch (setupPhase) {\n\t\t\tcase null:\n\t\t\t\tthis.logger.info(`Starting from scratch.`)\n\t\t\t\tthis.downloadPackage()\n\t\t\t\tthis.installPackage()\n\t\t\t\treturn this.startAllServices()\n\t\t\tcase `downloaded`:\n\t\t\t\tthis.logger.info(`Found package downloaded but not installed.`)\n\t\t\t\tthis.installPackage()\n\t\t\t\treturn this.startAllServices()\n\t\t\tcase `installed`: {\n\t\t\t\tfor (const [serviceName] of toEntries(this.services)) {\n\t\t\t\t\tthis.startService(serviceName)\n\t\t\t\t}\n\t\t\t\treturn this.live\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected startService(serviceName: S): void {\n\t\tthis.logger.info(\n\t\t\t`Starting service ${this.options.packageName}::${serviceName}, try ${this.safety}/2...`,\n\t\t)\n\t\tif (this.safety >= 2) {\n\t\t\tthrow new Error(`Out of tries...`)\n\t\t}\n\t\tthis.safety++\n\n\t\tconst [exe, ...args] = this.options.services[serviceName].run.split(` `)\n\t\tconst serviceProcess = spawn(exe, args, {\n\t\t\tcwd: this.options.flightdeckRootDir,\n\t\t\tenv: import.meta.env,\n\t\t})\n\t\tconst serviceLogger = this.serviceLoggers[serviceName]\n\t\tconst service = (this.services[serviceName] = new ChildSocket(\n\t\t\tserviceProcess,\n\t\t\t`${this.options.packageName}::${serviceName}`,\n\t\t\tserviceLogger,\n\t\t))\n\t\tserviceLogger.processCode = service.process.pid ?? -1\n\t\tthis.services[serviceName].onAny((...messages) => {\n\t\t\tserviceLogger.info(`💬`, ...messages)\n\t\t})\n\t\tthis.services[serviceName].on(`readyToUpdate`, () => {\n\t\t\tthis.logger.info(`Service \"${serviceName}\" is ready to update.`)\n\t\t\tthis.servicesReadyToUpdate[serviceName] = true\n\t\t\tthis.tryUpdate()\n\t\t})\n\t\tthis.services[serviceName].on(`alive`, () => {\n\t\t\tthis.servicesLive[this.serviceIdx[serviceName]].use(Promise.resolve())\n\t\t\tthis.servicesDead[this.serviceIdx[serviceName]] = new Future(() => {})\n\t\t\tif (this.dead.done) {\n\t\t\t\tthis.dead = new Future(() => {})\n\t\t\t}\n\t\t\tthis.dead.use(Promise.all(this.servicesDead))\n\t\t})\n\t\tthis.services[serviceName].process.once(`close`, (exitCode) => {\n\t\t\tthis.logger.info(\n\t\t\t\t`Auto-respawn saw \"${serviceName}\" exit with code ${exitCode}`,\n\t\t\t)\n\t\t\tthis.services[serviceName] = null\n\t\t\tif (!this.autoRespawnDeadServices) {\n\t\t\t\tthis.logger.info(`Auto-respawn is off; \"${serviceName}\" rests.`)\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst updatePhase = this.storage.getItem(`updatePhase`)\n\t\t\tthis.logger.info(`> storage(\"updatePhase\") >`, updatePhase)\n\t\t\tconst updatesAreReady = updatePhase === `confirmed`\n\t\t\tif (updatesAreReady) {\n\t\t\t\tthis.serviceLoggers[serviceName].info(`Updating before startup...`)\n\t\t\t\tthis.restartTimes = []\n\t\t\t\tthis.installPackage()\n\t\t\t\tthis.startService(serviceName)\n\t\t\t} else {\n\t\t\t\tconst now = Date.now()\n\t\t\t\tconst fiveMinutesAgo = now - 5 * 60 * 1000\n\t\t\t\tthis.restartTimes = this.restartTimes.filter(\n\t\t\t\t\t(time) => time > fiveMinutesAgo,\n\t\t\t\t)\n\t\t\t\tthis.restartTimes.push(now)\n\n\t\t\t\tif (this.restartTimes.length < 5) {\n\t\t\t\t\tthis.serviceLoggers[serviceName].info(`Crashed. Restarting...`)\n\t\t\t\t\tthis.startService(serviceName)\n\t\t\t\t} else {\n\t\t\t\t\tthis.serviceLoggers[serviceName].info(\n\t\t\t\t\t\t`Crashed 5 times in 5 minutes. Not restarting.`,\n\t\t\t\t\t)\n\t\t\t\t}\n\t\t\t}\n\t\t})\n\t\tthis.safety = 0\n\t}\n\n\tprotected downloadPackage(): void {\n\t\tthis.logger.info(`Downloading...`)\n\t\ttry {\n\t\t\tconst out = execSync(this.options.scripts.download)\n\t\t\tthis.logger.info(`Download stdout:`, out.toString())\n\t\t\tthis.storage.setItem(`setupPhase`, `downloaded`)\n\t\t\tthis.logger.info(`Downloaded!`)\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Failed to get the latest release: ${thrown.message}`)\n\t\t\t}\n\t\t\treturn\n\t\t}\n\t}\n\n\tprotected installPackage(): void {\n\t\tthis.logger.info(`Installing...`)\n\n\t\ttry {\n\t\t\tconst out = execSync(this.options.scripts.install)\n\t\t\tthis.logger.info(`Install stdout:`, out.toString())\n\t\t\tthis.storage.setItem(`setupPhase`, `installed`)\n\t\t\tthis.logger.info(`Installed!`)\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Failed to get the latest release: ${thrown.message}`)\n\t\t\t}\n\t\t\treturn\n\t\t}\n\t}\n\n\tpublic stopAllServices(): Future<unknown> {\n\t\tthis.logger.info(`Stopping all services... auto-respawn disabled.`)\n\t\tthis.autoRespawnDeadServices = false\n\t\tfor (const [serviceName] of toEntries(this.services)) {\n\t\t\tthis.stopService(serviceName)\n\t\t}\n\t\treturn this.dead\n\t}\n\n\tpublic stopService(serviceName: S): void {\n\t\tconst service = this.services[serviceName]\n\t\tif (service) {\n\t\t\tthis.logger.info(`Stopping service \"${serviceName}\"...`)\n\t\t\tthis.servicesDead[this.serviceIdx[serviceName]].use(\n\t\t\t\tnew Promise((pass) => {\n\t\t\t\t\tservice.emit(`timeToStop`)\n\t\t\t\t\tservice.process.once(`close`, (exitCode) => {\n\t\t\t\t\t\tthis.logger.info(\n\t\t\t\t\t\t\t`Stopped service \"${serviceName}\"; exited with code ${exitCode}`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tthis.services[serviceName] = null\n\t\t\t\t\t\tpass()\n\t\t\t\t\t})\n\t\t\t\t}),\n\t\t\t)\n\t\t\tthis.dead.use(Promise.all(this.servicesDead))\n\t\t\tthis.servicesLive[this.serviceIdx[serviceName]] = new Future(() => {})\n\t\t\tif (this.live.done) {\n\t\t\t\tthis.live = new Future(() => {})\n\t\t\t}\n\t\t\tthis.live.use(Promise.all(this.servicesLive))\n\t\t} else {\n\t\t\tthis.serviceLoggers[serviceName].error(\n\t\t\t\t`Tried to stop service, but it wasn't running.`,\n\t\t\t)\n\t\t}\n\t}\n}\n\nexport const FLIGHTDECK_INFO = `info`\nexport const FLIGHTDECK_WARN = `warn`\nexport const FLIGHTDECK_ERROR = `ERR!`\n\nexport const flightDeckLogSchema = z.object({\n\tlevel: z.union([\n\t\tz.literal(FLIGHTDECK_INFO),\n\t\tz.literal(FLIGHTDECK_WARN),\n\t\tz.literal(FLIGHTDECK_ERROR),\n\t]),\n\ttimestamp: z.number(),\n\tpackage: z.string(),\n\tservice: z.string().optional(),\n\tprocess: z.number(),\n\tbody: z.string(),\n})\nexport type FlightDeckLog = z.infer<typeof flightDeckLogSchema>\n\nconst LINE_FORMAT = `line-format` satisfies keyof LnavFormat\nconst VALUE = `value` satisfies keyof LnavFormat\n\nexport type LnavFormatVisualComponent = Exclude<\n\tExclude<LnavFormat[`line-format`], undefined>[number],\n\tstring\n>\n\nexport type LnavFormatBreakdown = Exclude<LnavFormat[`value`], undefined>\nexport type MemberOf<T> = T[keyof T]\nexport type LnavFormatValueDefinition = MemberOf<LnavFormatBreakdown>\n\nexport type FlightDeckFormat = {\n\t[LINE_FORMAT]: (\n\t\t| string\n\t\t| (LnavFormatVisualComponent & {\n\t\t\t\tfield: keyof FlightDeckLog | `__level__` | `__timestamp__`\n\t\t  })\n\t)[]\n\t[VALUE]: {\n\t\t[K in keyof FlightDeckLog]: LnavFormatValueDefinition & {\n\t\t\tkind: FlightDeckLog[K] extends number | undefined\n\t\t\t\t? `integer`\n\t\t\t\t: FlightDeckLog[K] extends string | undefined\n\t\t\t\t\t? `string`\n\t\t\t\t\t: never\n\t\t}\n\t}\n}\n\nexport const FLIGHTDECK_LNAV_FORMAT = {\n\ttitle: `FlightDeck Log`,\n\tdescription: `Format for events logged by the FlightDeck process manager.`,\n\t\"file-type\": `json`,\n\t\"timestamp-field\": `timestamp`,\n\t\"timestamp-divisor\": 1000,\n\t\"module-field\": `package`,\n\t\"opid-field\": `service`,\n\t\"level-field\": `level`,\n\tlevel: {\n\t\tinfo: FLIGHTDECK_INFO,\n\t\twarning: FLIGHTDECK_WARN,\n\t\terror: FLIGHTDECK_ERROR,\n\t},\n\n\t[LINE_FORMAT]: [\n\t\t{\n\t\t\tfield: `level`,\n\t\t},\n\t\t{\n\t\t\tprefix: ` `,\n\t\t\tfield: `__timestamp__`,\n\t\t\t\"timestamp-format\": `%Y-%m-%dT%H:%M:%S.%L%Z`,\n\t\t},\n\t\t{\n\t\t\tprefix: ` `,\n\t\t\tfield: `process`,\n\t\t\t\"min-width\": 5,\n\t\t},\n\t\t{\n\t\t\tprefix: `:`,\n\t\t\tfield: `package`,\n\t\t},\n\t\t{\n\t\t\tprefix: `:`,\n\t\t\tfield: `service`,\n\t\t\t\"default-value\": ``,\n\t\t},\n\t\t{\n\t\t\tprefix: `: `,\n\t\t\tfield: `body`,\n\t\t},\n\t],\n\n\t[VALUE]: {\n\t\ttimestamp: {\n\t\t\tkind: `integer`,\n\t\t},\n\t\tlevel: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tpackage: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tservice: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tprocess: {\n\t\t\tkind: `integer`,\n\t\t},\n\t\tbody: {\n\t\t\tkind: `string`,\n\t\t},\n\t},\n} as const satisfies FlightDeckFormat & LnavFormat\n\nexport class FlightDeckLogger\n\timplements Pick<Console, `error` | `info` | `warn`>\n{\n\tpublic readonly packageName: string\n\tpublic readonly serviceName?: string\n\tpublic readonly jsonLogging: boolean\n\tpublic processCode: number\n\tpublic constructor(\n\t\tpackageName: string,\n\t\tprocessCode: number,\n\t\tserviceName?: string,\n\t\toptions?: { jsonLogging: boolean },\n\t) {\n\t\tthis.packageName = packageName\n\t\tif (serviceName) {\n\t\t\tthis.serviceName = serviceName\n\t\t}\n\t\tthis.processCode = processCode\n\t\tthis.jsonLogging = options?.jsonLogging ?? false\n\t}\n\tprotected log(\n\t\tlevel:\n\t\t\t| typeof FLIGHTDECK_ERROR\n\t\t\t| typeof FLIGHTDECK_INFO\n\t\t\t| typeof FLIGHTDECK_WARN,\n\t\t...messages: unknown[]\n\t): void {\n\t\tif (this.jsonLogging) {\n\t\t\tlet body = messages\n\t\t\t\t.map((message) =>\n\t\t\t\t\ttypeof message === `string`\n\t\t\t\t\t\t? message\n\t\t\t\t\t\t: inspect(message, false, null, true),\n\t\t\t\t)\n\t\t\t\t.join(` `)\n\t\t\tif (body.includes(`\\n`)) {\n\t\t\t\tbody = `\\n  ${body.split(`\\n`).join(`\\n  `)}`\n\t\t\t}\n\t\t\tconst log: FlightDeckLog = {\n\t\t\t\ttimestamp: Date.now(),\n\t\t\t\tlevel,\n\t\t\t\tprocess: this.processCode,\n\t\t\t\tpackage: this.packageName,\n\t\t\t\tbody,\n\t\t\t}\n\t\t\tif (this.serviceName) {\n\t\t\t\tlog.service = this.serviceName\n\t\t\t}\n\t\t\tprocess.stdout.write(JSON.stringify(log) + `\\n`)\n\t\t} else {\n\t\t\tconst source = this.serviceName\n\t\t\t\t? `${this.packageName}:${this.serviceName}`\n\t\t\t\t: this.packageName\n\t\t\tswitch (level) {\n\t\t\t\tcase FLIGHTDECK_INFO:\n\t\t\t\t\tconsole.log(`${source}:`, ...messages)\n\t\t\t\t\tbreak\n\t\t\t\tcase FLIGHTDECK_WARN:\n\t\t\t\t\tconsole.warn(`${source}:`, ...messages)\n\t\t\t\t\tbreak\n\t\t\t\tcase FLIGHTDECK_ERROR:\n\t\t\t\t\tconsole.error(`${source}:`, ...messages)\n\t\t\t\t\tbreak\n\t\t\t}\n\t\t}\n\t}\n\tpublic info(...messages: unknown[]): void {\n\t\tthis.log(FLIGHTDECK_INFO, ...messages)\n\t}\n\n\tpublic warn(...messages: unknown[]): void {\n\t\tthis.log(FLIGHTDECK_WARN, ...messages)\n\t}\n\n\tpublic error(...messages: unknown[]): void {\n\t\tthis.log(FLIGHTDECK_ERROR, ...messages)\n\t}\n}\n",
    "import {\n\texistsSync,\n\tmkdirSync,\n\treaddirSync,\n\treadFileSync,\n\trmSync,\n\tstatSync,\n\twriteFileSync,\n} from \"node:fs\"\nimport { resolve } from \"node:path\"\n\nexport type FilesystemStorageOptions = {\n\tpath: string\n}\n\nexport class FilesystemStorage<\n\tT extends Record<string, string> = Record<string, string>,\n> implements Storage\n{\n\tpublic rootDir: string\n\n\tpublic constructor(options: FilesystemStorageOptions) {\n\t\tthis.rootDir = options.path\n\t\tif (!existsSync(this.rootDir)) {\n\t\t\tmkdirSync(this.rootDir, { recursive: true })\n\t\t}\n\t}\n\n\tpublic getItem<K extends string & keyof T>(key: K): T[K] | null {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\tif (existsSync(filePath)) {\n\t\t\treturn readFileSync(filePath, `utf-8`) as T[K]\n\t\t}\n\t\treturn null\n\t}\n\n\tpublic setItem<K extends string & keyof T>(key: K, value: T[K]): void {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\twriteFileSync(filePath, value)\n\t}\n\n\tpublic removeItem<K extends string & keyof T>(key: K): void {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\tif (existsSync(filePath)) {\n\t\t\trmSync(filePath)\n\t\t}\n\t}\n\n\tpublic key(index: number): (string & keyof T) | null {\n\t\tconst filePaths = readdirSync(this.rootDir)\n\t\tconst filePathsByDateCreated = filePaths.sort((a, b) => {\n\t\t\tconst aStat = statSync(a)\n\t\t\tconst bStat = statSync(b)\n\t\t\treturn bStat.ctimeMs - aStat.ctimeMs\n\t\t})\n\t\treturn (filePathsByDateCreated[index] as string & keyof T) ?? null\n\t}\n\n\tpublic clear(): void {\n\t\trmSync(this.rootDir, { recursive: true })\n\t\tmkdirSync(this.rootDir, { recursive: true })\n\t}\n\n\tpublic get length(): number {\n\t\treturn readdirSync(this.rootDir).length\n\t}\n}\n",
    "import { createEnv } from \"@t3-oss/env-core\"\nimport { z } from \"zod\"\n\nexport const env = createEnv({\n\tserver: { FLIGHTDECK_SECRET: z.string().optional() },\n\tclientPrefix: `NEVER`,\n\tclient: {},\n\truntimeEnv: import.meta.env,\n\temptyStringAsUndefined: true,\n})\n"
  ],
  "mappings": "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",
  "debugId": "8C257A7694D940D664756E2164756E21",
  "names": []
}
@@ -5,17 +5,19 @@
5
5
  "description": "Format for events logged by the FlightDeck process manager.",
6
6
  "file-type": "json",
7
7
  "timestamp-field": "timestamp",
8
- "subsecond-field": "subsecond",
9
- "subsecond-units": "milli",
8
+ "timestamp-divisor": 1000,
9
+ "module-field": "package",
10
10
  "opid-field": "service",
11
11
  "level-field": "level",
12
12
  "level": {
13
13
  "info": "info",
14
14
  "warning": "warn",
15
- "error": "err!"
15
+ "error": "ERR!"
16
16
  },
17
- "ordered-by-time": true,
18
17
  "line-format": [
18
+ {
19
+ "field": "level"
20
+ },
19
21
  {
20
22
  "prefix": " ",
21
23
  "field": "__timestamp__",
@@ -35,11 +37,6 @@
35
37
  "field": "service",
36
38
  "default-value": ""
37
39
  },
38
- {
39
- "prefix": "[",
40
- "field": "__level__",
41
- "suffix": "]"
42
- },
43
40
  {
44
41
  "prefix": ": ",
45
42
  "field": "body"
package/dist/lib.d.ts CHANGED
@@ -867,6 +867,9 @@ declare class FlightDeck<S extends string = string> {
867
867
  stopAllServices(): Future<unknown>;
868
868
  stopService(serviceName: S): void;
869
869
  }
870
+ declare const FLIGHTDECK_INFO = "info";
871
+ declare const FLIGHTDECK_WARN = "warn";
872
+ declare const FLIGHTDECK_ERROR = "ERR!";
870
873
  declare const flightDeckLogSchema: z.ZodObject<{
871
874
  level: z.ZodUnion<[z.ZodLiteral<"info">, z.ZodLiteral<"warn">, z.ZodLiteral<"ERR!">]>;
872
875
  timestamp: z.ZodNumber;
@@ -911,17 +914,18 @@ declare const FLIGHTDECK_LNAV_FORMAT: {
911
914
  readonly description: "Format for events logged by the FlightDeck process manager.";
912
915
  readonly "file-type": "json";
913
916
  readonly "timestamp-field": "timestamp";
914
- readonly "subsecond-field": "subsecond";
915
- readonly "subsecond-units": "milli";
917
+ readonly "timestamp-divisor": 1000;
918
+ readonly "module-field": "package";
916
919
  readonly "opid-field": "service";
917
920
  readonly "level-field": "level";
918
921
  readonly level: {
919
922
  readonly info: "info";
920
923
  readonly warning: "warn";
921
- readonly error: "err!";
924
+ readonly error: "ERR!";
922
925
  };
923
- readonly "ordered-by-time": true;
924
926
  readonly "line-format": [{
927
+ readonly field: "level";
928
+ }, {
925
929
  readonly prefix: " ";
926
930
  readonly field: "__timestamp__";
927
931
  readonly "timestamp-format": "%Y-%m-%dT%H:%M:%S.%L%Z";
@@ -936,10 +940,6 @@ declare const FLIGHTDECK_LNAV_FORMAT: {
936
940
  readonly prefix: ":";
937
941
  readonly field: "service";
938
942
  readonly "default-value": "";
939
- }, {
940
- readonly prefix: "[";
941
- readonly field: "__level__";
942
- readonly suffix: "]";
943
943
  }, {
944
944
  readonly prefix: ": ";
945
945
  readonly field: "body";
@@ -973,6 +973,7 @@ declare class FlightDeckLogger implements Pick<Console, `error` | `info` | `warn
973
973
  constructor(packageName: string, processCode: number, serviceName?: string, options?: {
974
974
  jsonLogging: boolean;
975
975
  });
976
+ protected log(level: typeof FLIGHTDECK_ERROR | typeof FLIGHTDECK_INFO | typeof FLIGHTDECK_WARN, ...messages: unknown[]): void;
976
977
  info(...messages: unknown[]): void;
977
978
  warn(...messages: unknown[]): void;
978
979
  error(...messages: unknown[]): void;
@@ -1031,4 +1032,4 @@ declare namespace klaxon_lib {
1031
1032
  export { type klaxon_lib_AlertOptions as AlertOptions, type klaxon_lib_ChangesetsPublishResult as ChangesetsPublishResult, type klaxon_lib_ChangesetsPublishedPackage as ChangesetsPublishedPackage, type klaxon_lib_PackageConfig as PackageConfig, type klaxon_lib_ScrambleOptions as ScrambleOptions, type klaxon_lib_ScrambleResult as ScrambleResult, type klaxon_lib_SecretsConfig as SecretsConfig, klaxon_lib_alert as alert, klaxon_lib_scramble as scramble };
1032
1033
  }
1033
1034
 
1034
- export { FLIGHTDECK_LNAV_FORMAT, FLIGHTDECK_SETUP_PHASES, FLIGHTDECK_UPDATE_PHASES, FilesystemStorage, type FilesystemStorageOptions, FlightDeck, type FlightDeckFormat, type FlightDeckLog, FlightDeckLogger, type FlightDeckOptions, type FlightDeckSetupPhase, type FlightDeckUpdatePhase, klaxon_lib as Klaxon, type LnavFormatBreakdown, type LnavFormatValueDefinition, type LnavFormatVisualComponent, type MemberOf, flightDeckLogSchema, isVersionNumber };
1035
+ export { FLIGHTDECK_ERROR, FLIGHTDECK_INFO, FLIGHTDECK_LNAV_FORMAT, FLIGHTDECK_SETUP_PHASES, FLIGHTDECK_UPDATE_PHASES, FLIGHTDECK_WARN, FilesystemStorage, type FilesystemStorageOptions, FlightDeck, type FlightDeckFormat, type FlightDeckLog, FlightDeckLogger, type FlightDeckOptions, type FlightDeckSetupPhase, type FlightDeckUpdatePhase, klaxon_lib as Klaxon, type LnavFormatBreakdown, type LnavFormatValueDefinition, type LnavFormatVisualComponent, type MemberOf, flightDeckLogSchema, isVersionNumber };
package/dist/lib.js CHANGED
@@ -1,7 +1,9 @@
1
- var I=Object.defineProperty;var _=(e,t)=>{for(var i in t)I(e,i,{get:t[i],enumerable:!0,configurable:!0,set:(r)=>t[i]=()=>r})};import{existsSync as m,mkdirSync as F,readdirSync as L,readFileSync as U,rmSync as x,statSync as T,writeFileSync as R}from"node:fs";import{resolve as y}from"node:path";class k{rootDir;constructor(e){if(this.rootDir=e.path,!m(this.rootDir))F(this.rootDir,{recursive:!0})}getItem(e){let t=y(this.rootDir,e);if(m(t))return U(t,"utf-8");return null}setItem(e,t){let i=y(this.rootDir,e);R(i,t)}removeItem(e){let t=y(this.rootDir,e);if(m(t))x(t)}key(e){return L(this.rootDir).sort((r,c)=>{let n=T(r);return T(c).ctimeMs-n.ctimeMs})[e]??null}clear(){x(this.rootDir,{recursive:!0}),F(this.rootDir,{recursive:!0})}get length(){return L(this.rootDir).length}}import{execSync as S,spawn as j}from"node:child_process";import{createServer as H}from"node:http";import{homedir as O}from"node:os";import{resolve as E}from"node:path";import{inspect as w}from"node:util";import{Future as h}from"atom.io/internal";import{discoverType as M}from"atom.io/introspection";import{fromEntries as v,toEntries as u}from"atom.io/json";import{ChildSocket as V}from"atom.io/realtime-server";import{CronJob as B}from"cron";import{z as l}from"zod";import{createEnv as K}from"@t3-oss/env-core";import{z as N}from"zod";var P=K({server:{FLIGHTDECK_SECRET:N.string().optional()},clientPrefix:"NEVER",client:{},runtimeEnv:import.meta.env,emptyStringAsUndefined:!0});var ve=["downloaded","installed"],me=["notified","confirmed"];function G(e){return/^\d+\.\d+\.\d+$/.test(e)||!Number.isNaN(Number.parseFloat(e))}class z{options;safety=0;storage;webhookServer;services;serviceIdx;defaultServicesReadyToUpdate;servicesReadyToUpdate;autoRespawnDeadServices;logger;serviceLoggers;updateAvailabilityChecker=null;servicesLive;servicesDead;live=new h(()=>{});dead=new h(()=>{});restartTimes=[];constructor(e){this.options=e;let{FLIGHTDECK_SECRET:t}=P,{flightdeckRootDir:i=E(O(),".flightdeck")}=e,r=e.port??8080,c=`http://localhost:${r}`,n=u(e.services);if(this.services=v(n.map(([s])=>[s,null])),this.serviceIdx=v(n.map(([s],o)=>[s,o])),this.defaultServicesReadyToUpdate=v(n.map(([s,{waitFor:o}])=>[s,!o])),this.servicesReadyToUpdate={...this.defaultServicesReadyToUpdate},this.autoRespawnDeadServices=!0,this.logger=new b(this.options.packageName,process.pid,void 0,{jsonLogging:this.options.jsonLogging??!1}),this.serviceLoggers=v(n.map(([s])=>[s,new b(this.options.packageName,process.pid,s,{jsonLogging:this.options.jsonLogging??!1})])),this.servicesLive=n.map(()=>new h(()=>{})),this.servicesDead=n.map(()=>new h(()=>{})),this.live.use(Promise.all(this.servicesLive)),this.dead.use(Promise.all(this.servicesDead)),this.storage=new k({path:E(i,"storage",e.packageName)}),t===void 0)this.logger.warn("No FLIGHTDECK_SECRET environment variable found. FlightDeck will not run an update server.");else H((s,o)=>{let p=[];s.on("data",(a)=>{p.push(a instanceof Buffer?a:Buffer.from(a))}).on("end",()=>{let a=s.headers.authorization;try{if(typeof s.url==="undefined")throw 400;let d=`Bearer ${t}`;if(a!==`Bearer ${t}`)throw this.logger.info(`Unauthorized: needed \`${d}\`, got \`${a}\``),401;let f=new URL(s.url,c);this.logger.info(s.method,f.pathname);let g=Buffer.concat(p).toString();if(!G(g))throw 400;o.writeHead(200),o.end(),this.storage.setItem("updatePhase","notified"),this.storage.setItem("updateAwaitedVersion",g);let{checkAvailability:$}=e.scripts;if($){this.updateAvailabilityChecker?.stop(),this.seekUpdate(g);let D=this.storage.getItem("updatePhase");if(this.logger.info('> storage("updatePhase") >',D),D==="notified")this.updateAvailabilityChecker=new B("30 * * * * *",()=>{this.seekUpdate(g)}),this.updateAvailabilityChecker.start()}else this.downloadPackage()}catch(d){if(this.logger.error(d,s.url),typeof d==="number")o.writeHead(d),o.end()}finally{p=[]}})}).listen(r,()=>{this.logger.info(`Server started on port ${r}`)});this.startAllServices().then(()=>{this.logger.info("All services started.")}).catch((s)=>{if(s instanceof Error)this.logger.error("Failed to start all services:",s.message)})}seekUpdate(e){this.logger.info("Checking for updates...");let{checkAvailability:t}=this.options.scripts;if(!t){this.logger.info("No checkAvailability script found.");return}try{let i=S(`${t} ${e}`);this.logger.info("Check stdout:",i.toString()),this.updateAvailabilityChecker?.stop(),this.storage.setItem("updatePhase","confirmed"),this.downloadPackage(),this.announceUpdate()}catch(i){if(i instanceof Error)this.logger.error("Check failed:",i.message);else{let r=M(i);this.logger.error("Check threw",r,i)}}}announceUpdate(){for(let e of u(this.services)){let[t,i]=e;if(i){if(this.options.services[t].waitFor)i.emit("updatesReady")}else this.startService(t)}}tryUpdate(){if(u(this.servicesReadyToUpdate).every(([,e])=>e))this.logger.info("All services are ready to update."),this.stopAllServices().then(()=>{this.logger.info("All services stopped; starting up fresh..."),this.startAllServices().then(()=>{this.logger.info("All services started; we're back online.")}).catch((e)=>{if(e instanceof Error)this.logger.error("Failed to start all services:",e.message)})}).catch((e)=>{if(e instanceof Error)this.logger.error("Failed to stop all services:",e.message)})}startAllServices(){this.logger.info("Starting all services..."),this.autoRespawnDeadServices=!0;let e=this.storage.getItem("setupPhase");switch(this.logger.info('> storage("setupPhase") >',e),e){case null:return this.logger.info("Starting from scratch."),this.downloadPackage(),this.installPackage(),this.startAllServices();case"downloaded":return this.logger.info("Found package downloaded but not installed."),this.installPackage(),this.startAllServices();case"installed":{for(let[t]of u(this.services))this.startService(t);return this.live}}}startService(e){if(this.logger.info(`Starting service ${this.options.packageName}::${e}, try ${this.safety}/2...`),this.safety>=2)throw new Error("Out of tries...");this.safety++;let[t,...i]=this.options.services[e].run.split(" "),r=j(t,i,{cwd:this.options.flightdeckRootDir,env:import.meta.env}),c=this.serviceLoggers[e],n=this.services[e]=new V(r,`${this.options.packageName}::${e}`,c);c.processCode=n.process.pid??-1,this.services[e].onAny((...s)=>{c.info("\uD83D\uDCAC",...s)}),this.services[e].on("readyToUpdate",()=>{this.logger.info(`Service "${e}" is ready to update.`),this.servicesReadyToUpdate[e]=!0,this.tryUpdate()}),this.services[e].on("alive",()=>{if(this.servicesLive[this.serviceIdx[e]].use(Promise.resolve()),this.servicesDead[this.serviceIdx[e]]=new h(()=>{}),this.dead.done)this.dead=new h(()=>{});this.dead.use(Promise.all(this.servicesDead))}),this.services[e].process.once("close",(s)=>{if(this.logger.info(`Auto-respawn saw "${e}" exit with code ${s}`),this.services[e]=null,!this.autoRespawnDeadServices){this.logger.info(`Auto-respawn is off; "${e}" rests.`);return}let o=this.storage.getItem("updatePhase");if(this.logger.info('> storage("updatePhase") >',o),o==="confirmed")this.serviceLoggers[e].info("Updating before startup..."),this.restartTimes=[],this.installPackage(),this.startService(e);else{let a=Date.now(),d=a-300000;if(this.restartTimes=this.restartTimes.filter((f)=>f>d),this.restartTimes.push(a),this.restartTimes.length<5)this.serviceLoggers[e].info("Crashed. Restarting..."),this.startService(e);else this.serviceLoggers[e].info("Crashed 5 times in 5 minutes. Not restarting.")}}),this.safety=0}downloadPackage(){this.logger.info("Downloading...");try{let e=S(this.options.scripts.download);this.logger.info("Download stdout:",e.toString()),this.storage.setItem("setupPhase","downloaded"),this.logger.info("Downloaded!")}catch(e){if(e instanceof Error)this.logger.error(`Failed to get the latest release: ${e.message}`);return}}installPackage(){this.logger.info("Installing...");try{let e=S(this.options.scripts.install);this.logger.info("Install stdout:",e.toString()),this.storage.setItem("setupPhase","installed"),this.logger.info("Installed!")}catch(e){if(e instanceof Error)this.logger.error(`Failed to get the latest release: ${e.message}`);return}}stopAllServices(){this.logger.info("Stopping all services... auto-respawn disabled."),this.autoRespawnDeadServices=!1;for(let[e]of u(this.services))this.stopService(e);return this.dead}stopService(e){let t=this.services[e];if(t){if(this.logger.info(`Stopping service "${e}"...`),this.servicesDead[this.serviceIdx[e]].use(new Promise((i)=>{t.emit("timeToStop"),t.process.once("close",(r)=>{this.logger.info(`Stopped service "${e}"; exited with code ${r}`),this.services[e]=null,i()})})),this.dead.use(Promise.all(this.servicesDead)),this.servicesLive[this.serviceIdx[e]]=new h(()=>{}),this.live.done)this.live=new h(()=>{});this.live.use(Promise.all(this.servicesLive))}else this.serviceLoggers[e].error("Tried to stop service, but it wasn't running.")}}var ye=l.object({level:l.union([l.literal("info"),l.literal("warn"),l.literal("ERR!")]),timestamp:l.number(),package:l.string(),service:l.string().optional(),process:l.number(),body:l.string()}),J="line-format",Y="value",ke={title:"FlightDeck Log",description:"Format for events logged by the FlightDeck process manager.","file-type":"json","timestamp-field":"timestamp","subsecond-field":"subsecond","subsecond-units":"milli","opid-field":"service","level-field":"level",level:{info:"info",warning:"warn",error:"err!"},"ordered-by-time":!0,[J]:[{prefix:" ",field:"__timestamp__","timestamp-format":"%Y-%m-%dT%H:%M:%S.%L%Z"},{prefix:" ",field:"process","min-width":5},{prefix:":",field:"package"},{prefix:":",field:"service","default-value":""},{prefix:"[",field:"__level__",suffix:"]"},{prefix:": ",field:"body"}],[Y]:{timestamp:{kind:"integer"},level:{kind:"string"},package:{kind:"string"},service:{kind:"string"},process:{kind:"integer"},body:{kind:"string"}}};class b{packageName;serviceName;jsonLogging;processCode;constructor(e,t,i,r){if(this.packageName=e,i)this.serviceName=i;this.processCode=t,this.jsonLogging=r?.jsonLogging??!1}info(...e){if(this.jsonLogging){let t={timestamp:Date.now(),level:"info",process:this.processCode,package:this.packageName,body:e.map((i)=>typeof i==="string"?i:w(i,!1,null,!0)).join(" ")};if(this.serviceName)t.service=this.serviceName;process.stdout.write(JSON.stringify(t)+`
2
- `)}else{let t=this.serviceName?`${this.packageName}::${this.serviceName}`:this.packageName;console.log(`${t}:`,...e)}}warn(...e){if(this.jsonLogging){let t={timestamp:Date.now(),level:"warn",process:this.processCode,package:this.packageName,body:e.map((i)=>typeof i==="string"?i:w(i,!1,null,!0)).join(" ")};if(this.serviceName)t.service=this.serviceName;process.stdout.write(JSON.stringify(t)+`
3
- `)}else{let t=this.serviceName?`${this.packageName}::${this.serviceName}`:this.packageName;console.warn(`${t}:`,...e)}}error(...e){if(this.jsonLogging){let t={timestamp:Date.now()+Math.floor(Math.random()*1000),level:"ERR!",process:this.processCode,package:this.packageName,body:e.map((i)=>typeof i==="string"?i:w(i,!1,null,!0)).join(" ")};if(this.serviceName)t.service=this.serviceName;process.stdout.write(JSON.stringify(t)+`
4
- `)}else{let t=this.serviceName?`${this.packageName}::${this.serviceName}`:this.packageName;console.error(`${t}:`,...e)}}}var C={};_(C,{scramble:()=>Z,alert:()=>A});async function A({secret:e,endpoint:t,version:i}){return await fetch(t,{method:"POST",headers:{"Content-Type":"text/plain;charset=UTF-8",Authorization:`Bearer ${e}`},body:i})}async function Z({packageConfig:e,secretsConfig:t,publishedPackages:i}){let r=[];for(let s of i)if(s.name in e){let o=s.name,{endpoint:p}=e[o],a=t[o],d=s.version,f=A({secret:a,endpoint:p,version:d}).then((g)=>[o,g]);r.push(f)}let c=await Promise.all(r);return Object.fromEntries(c)}export{G as isVersionNumber,ye as flightDeckLogSchema,C as Klaxon,b as FlightDeckLogger,z as FlightDeck,k as FilesystemStorage,me as FLIGHTDECK_UPDATE_PHASES,ve as FLIGHTDECK_SETUP_PHASES,ke as FLIGHTDECK_LNAV_FORMAT};
1
+ var U=Object.defineProperty;var $=(e,t)=>{for(var i in t)U(e,i,{get:t[i],enumerable:!0,configurable:!0,set:(r)=>t[i]=()=>r})};import{existsSync as S,mkdirSync as T,readdirSync as x,readFileSync as K,rmSync as E,statSync as P,writeFileSync as H}from"node:fs";import{resolve as w}from"node:path";class b{rootDir;constructor(e){if(this.rootDir=e.path,!S(this.rootDir))T(this.rootDir,{recursive:!0})}getItem(e){let t=w(this.rootDir,e);if(S(t))return K(t,"utf-8");return null}setItem(e,t){let i=w(this.rootDir,e);H(i,t)}removeItem(e){let t=w(this.rootDir,e);if(S(t))E(t)}key(e){return x(this.rootDir).sort((r,n)=>{let a=P(r);return P(n).ctimeMs-a.ctimeMs})[e]??null}clear(){E(this.rootDir,{recursive:!0}),T(this.rootDir,{recursive:!0})}get length(){return x(this.rootDir).length}}import{execSync as F,spawn as G}from"node:child_process";import{createServer as V}from"node:http";import{homedir as B}from"node:os";import{resolve as I}from"node:path";import{inspect as M}from"node:util";import{Future as p}from"atom.io/internal";import{discoverType as N}from"atom.io/introspection";import{fromEntries as v,toEntries as u}from"atom.io/json";import{ChildSocket as z}from"atom.io/realtime-server";import{CronJob as J}from"cron";import{z as c}from"zod";import{createEnv as O}from"@t3-oss/env-core";import{z as j}from"zod";var A=O({server:{FLIGHTDECK_SECRET:j.string().optional()},clientPrefix:"NEVER",client:{},runtimeEnv:import.meta.env,emptyStringAsUndefined:!0});var ke=["downloaded","installed"],Se=["notified","confirmed"];function W(e){return/^\d+\.\d+\.\d+$/.test(e)||!Number.isNaN(Number.parseFloat(e))}class Y{options;safety=0;storage;webhookServer;services;serviceIdx;defaultServicesReadyToUpdate;servicesReadyToUpdate;autoRespawnDeadServices;logger;serviceLoggers;updateAvailabilityChecker=null;servicesLive;servicesDead;live=new p(()=>{});dead=new p(()=>{});restartTimes=[];constructor(e){this.options=e;let{FLIGHTDECK_SECRET:t}=A,{flightdeckRootDir:i=I(B(),".flightdeck")}=e,r=e.port??8080,n=`http://localhost:${r}`,a=u(e.services);if(this.services=v(a.map(([s])=>[s,null])),this.serviceIdx=v(a.map(([s],o)=>[s,o])),this.defaultServicesReadyToUpdate=v(a.map(([s,{waitFor:o}])=>[s,!o])),this.servicesReadyToUpdate={...this.defaultServicesReadyToUpdate},this.autoRespawnDeadServices=!0,this.logger=new D(this.options.packageName,process.pid,void 0,{jsonLogging:this.options.jsonLogging??!1}),this.serviceLoggers=v(a.map(([s])=>[s,new D(this.options.packageName,process.pid,s,{jsonLogging:this.options.jsonLogging??!1})])),this.servicesLive=a.map(()=>new p(()=>{})),this.servicesDead=a.map(()=>new p(()=>{})),this.live.use(Promise.all(this.servicesLive)),this.dead.use(Promise.all(this.servicesDead)),this.storage=new b({path:I(i,"storage",e.packageName)}),t===void 0)this.logger.warn("No FLIGHTDECK_SECRET environment variable found. FlightDeck will not run an update server.");else V((s,o)=>{let h=[];s.on("data",(l)=>{h.push(l instanceof Buffer?l:Buffer.from(l))}).on("end",()=>{let l=s.headers.authorization;try{if(typeof s.url==="undefined")throw 400;let d=`Bearer ${t}`;if(l!==`Bearer ${t}`)throw this.logger.info(`Unauthorized: needed \`${d}\`, got \`${l}\``),401;let f=new URL(s.url,n);this.logger.info(s.method,f.pathname);let g=Buffer.concat(h).toString();if(!W(g))throw 400;o.writeHead(200),o.end(),this.storage.setItem("updatePhase","notified"),this.storage.setItem("updateAwaitedVersion",g);let{checkAvailability:_}=e.scripts;if(_){this.updateAvailabilityChecker?.stop(),this.seekUpdate(g);let L=this.storage.getItem("updatePhase");if(this.logger.info('> storage("updatePhase") >',L),L==="notified")this.updateAvailabilityChecker=new J("30 * * * * *",()=>{this.seekUpdate(g)}),this.updateAvailabilityChecker.start()}else this.downloadPackage()}catch(d){if(this.logger.error(d,s.url),typeof d==="number")o.writeHead(d),o.end()}finally{h=[]}})}).listen(r,()=>{this.logger.info(`Server started on port ${r}`)});this.startAllServices().then(()=>{this.logger.info("All services started.")}).catch((s)=>{if(s instanceof Error)this.logger.error("Failed to start all services:",s.message)})}seekUpdate(e){this.logger.info("Checking for updates...");let{checkAvailability:t}=this.options.scripts;if(!t){this.logger.info("No checkAvailability script found.");return}try{let i=F(`${t} ${e}`);this.logger.info("Check stdout:",i.toString()),this.updateAvailabilityChecker?.stop(),this.storage.setItem("updatePhase","confirmed"),this.downloadPackage(),this.announceUpdate()}catch(i){if(i instanceof Error)this.logger.error("Check failed:",i.message);else{let r=N(i);this.logger.error("Check threw",r,i)}}}announceUpdate(){for(let e of u(this.services)){let[t,i]=e;if(i){if(this.options.services[t].waitFor)i.emit("updatesReady")}else this.startService(t)}}tryUpdate(){if(u(this.servicesReadyToUpdate).every(([,e])=>e))this.logger.info("All services are ready to update."),this.stopAllServices().then(()=>{this.logger.info("All services stopped; starting up fresh..."),this.startAllServices().then(()=>{this.logger.info("All services started; we're back online.")}).catch((e)=>{if(e instanceof Error)this.logger.error("Failed to start all services:",e.message)})}).catch((e)=>{if(e instanceof Error)this.logger.error("Failed to stop all services:",e.message)})}startAllServices(){this.logger.info("Starting all services..."),this.autoRespawnDeadServices=!0;let e=this.storage.getItem("setupPhase");switch(this.logger.info('> storage("setupPhase") >',e),e){case null:return this.logger.info("Starting from scratch."),this.downloadPackage(),this.installPackage(),this.startAllServices();case"downloaded":return this.logger.info("Found package downloaded but not installed."),this.installPackage(),this.startAllServices();case"installed":{for(let[t]of u(this.services))this.startService(t);return this.live}}}startService(e){if(this.logger.info(`Starting service ${this.options.packageName}::${e}, try ${this.safety}/2...`),this.safety>=2)throw new Error("Out of tries...");this.safety++;let[t,...i]=this.options.services[e].run.split(" "),r=G(t,i,{cwd:this.options.flightdeckRootDir,env:import.meta.env}),n=this.serviceLoggers[e],a=this.services[e]=new z(r,`${this.options.packageName}::${e}`,n);n.processCode=a.process.pid??-1,this.services[e].onAny((...s)=>{n.info("\uD83D\uDCAC",...s)}),this.services[e].on("readyToUpdate",()=>{this.logger.info(`Service "${e}" is ready to update.`),this.servicesReadyToUpdate[e]=!0,this.tryUpdate()}),this.services[e].on("alive",()=>{if(this.servicesLive[this.serviceIdx[e]].use(Promise.resolve()),this.servicesDead[this.serviceIdx[e]]=new p(()=>{}),this.dead.done)this.dead=new p(()=>{});this.dead.use(Promise.all(this.servicesDead))}),this.services[e].process.once("close",(s)=>{if(this.logger.info(`Auto-respawn saw "${e}" exit with code ${s}`),this.services[e]=null,!this.autoRespawnDeadServices){this.logger.info(`Auto-respawn is off; "${e}" rests.`);return}let o=this.storage.getItem("updatePhase");if(this.logger.info('> storage("updatePhase") >',o),o==="confirmed")this.serviceLoggers[e].info("Updating before startup..."),this.restartTimes=[],this.installPackage(),this.startService(e);else{let l=Date.now(),d=l-300000;if(this.restartTimes=this.restartTimes.filter((f)=>f>d),this.restartTimes.push(l),this.restartTimes.length<5)this.serviceLoggers[e].info("Crashed. Restarting..."),this.startService(e);else this.serviceLoggers[e].info("Crashed 5 times in 5 minutes. Not restarting.")}}),this.safety=0}downloadPackage(){this.logger.info("Downloading...");try{let e=F(this.options.scripts.download);this.logger.info("Download stdout:",e.toString()),this.storage.setItem("setupPhase","downloaded"),this.logger.info("Downloaded!")}catch(e){if(e instanceof Error)this.logger.error(`Failed to get the latest release: ${e.message}`);return}}installPackage(){this.logger.info("Installing...");try{let e=F(this.options.scripts.install);this.logger.info("Install stdout:",e.toString()),this.storage.setItem("setupPhase","installed"),this.logger.info("Installed!")}catch(e){if(e instanceof Error)this.logger.error(`Failed to get the latest release: ${e.message}`);return}}stopAllServices(){this.logger.info("Stopping all services... auto-respawn disabled."),this.autoRespawnDeadServices=!1;for(let[e]of u(this.services))this.stopService(e);return this.dead}stopService(e){let t=this.services[e];if(t){if(this.logger.info(`Stopping service "${e}"...`),this.servicesDead[this.serviceIdx[e]].use(new Promise((i)=>{t.emit("timeToStop"),t.process.once("close",(r)=>{this.logger.info(`Stopped service "${e}"; exited with code ${r}`),this.services[e]=null,i()})})),this.dead.use(Promise.all(this.servicesDead)),this.servicesLive[this.serviceIdx[e]]=new p(()=>{}),this.live.done)this.live=new p(()=>{});this.live.use(Promise.all(this.servicesLive))}else this.serviceLoggers[e].error("Tried to stop service, but it wasn't running.")}}var m="info",y="warn",k="ERR!",we=c.object({level:c.union([c.literal(m),c.literal(y),c.literal(k)]),timestamp:c.number(),package:c.string(),service:c.string().optional(),process:c.number(),body:c.string()}),Z="line-format",Q="value",be={title:"FlightDeck Log",description:"Format for events logged by the FlightDeck process manager.","file-type":"json","timestamp-field":"timestamp","timestamp-divisor":1000,"module-field":"package","opid-field":"service","level-field":"level",level:{info:m,warning:y,error:k},[Z]:[{field:"level"},{prefix:" ",field:"__timestamp__","timestamp-format":"%Y-%m-%dT%H:%M:%S.%L%Z"},{prefix:" ",field:"process","min-width":5},{prefix:":",field:"package"},{prefix:":",field:"service","default-value":""},{prefix:": ",field:"body"}],[Q]:{timestamp:{kind:"integer"},level:{kind:"string"},package:{kind:"string"},service:{kind:"string"},process:{kind:"integer"},body:{kind:"string"}}};class D{packageName;serviceName;jsonLogging;processCode;constructor(e,t,i,r){if(this.packageName=e,i)this.serviceName=i;this.processCode=t,this.jsonLogging=r?.jsonLogging??!1}log(e,...t){if(this.jsonLogging){let i=t.map((n)=>typeof n==="string"?n:M(n,!1,null,!0)).join(" ");if(i.includes(`
2
+ `))i=`
3
+ ${i.split(`
4
+ `).join(`
5
+ `)}`;let r={timestamp:Date.now(),level:e,process:this.processCode,package:this.packageName,body:i};if(this.serviceName)r.service=this.serviceName;process.stdout.write(JSON.stringify(r)+`
6
+ `)}else{let i=this.serviceName?`${this.packageName}:${this.serviceName}`:this.packageName;switch(e){case m:console.log(`${i}:`,...t);break;case y:console.warn(`${i}:`,...t);break;case k:console.error(`${i}:`,...t);break}}}info(...e){this.log(m,...e)}warn(...e){this.log(y,...e)}error(...e){this.log(k,...e)}}var R={};$(R,{scramble:()=>X,alert:()=>C});async function C({secret:e,endpoint:t,version:i}){return await fetch(t,{method:"POST",headers:{"Content-Type":"text/plain;charset=UTF-8",Authorization:`Bearer ${e}`},body:i})}async function X({packageConfig:e,secretsConfig:t,publishedPackages:i}){let r=[];for(let s of i)if(s.name in e){let o=s.name,{endpoint:h}=e[o],l=t[o],d=s.version,f=C({secret:l,endpoint:h,version:d}).then((g)=>[o,g]);r.push(f)}let n=await Promise.all(r);return Object.fromEntries(n)}export{W as isVersionNumber,we as flightDeckLogSchema,R as Klaxon,D as FlightDeckLogger,Y as FlightDeck,b as FilesystemStorage,y as FLIGHTDECK_WARN,Se as FLIGHTDECK_UPDATE_PHASES,ke as FLIGHTDECK_SETUP_PHASES,be as FLIGHTDECK_LNAV_FORMAT,m as FLIGHTDECK_INFO,k as FLIGHTDECK_ERROR};
5
7
 
6
- //# debugId=D614A1AC9E2EE18564756E2164756E21
7
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/filesystem-storage.ts", "../src/flightdeck.lib.ts", "../src/flightdeck.env.ts", "../src/klaxon.lib.ts"],
  "sourcesContent": [
    "import {\n\texistsSync,\n\tmkdirSync,\n\treaddirSync,\n\treadFileSync,\n\trmSync,\n\tstatSync,\n\twriteFileSync,\n} from \"node:fs\"\nimport { resolve } from \"node:path\"\n\nexport type FilesystemStorageOptions = {\n\tpath: string\n}\n\nexport class FilesystemStorage<\n\tT extends Record<string, string> = Record<string, string>,\n> implements Storage\n{\n\tpublic rootDir: string\n\n\tpublic constructor(options: FilesystemStorageOptions) {\n\t\tthis.rootDir = options.path\n\t\tif (!existsSync(this.rootDir)) {\n\t\t\tmkdirSync(this.rootDir, { recursive: true })\n\t\t}\n\t}\n\n\tpublic getItem<K extends string & keyof T>(key: K): T[K] | null {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\tif (existsSync(filePath)) {\n\t\t\treturn readFileSync(filePath, `utf-8`) as T[K]\n\t\t}\n\t\treturn null\n\t}\n\n\tpublic setItem<K extends string & keyof T>(key: K, value: T[K]): void {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\twriteFileSync(filePath, value)\n\t}\n\n\tpublic removeItem<K extends string & keyof T>(key: K): void {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\tif (existsSync(filePath)) {\n\t\t\trmSync(filePath)\n\t\t}\n\t}\n\n\tpublic key(index: number): (string & keyof T) | null {\n\t\tconst filePaths = readdirSync(this.rootDir)\n\t\tconst filePathsByDateCreated = filePaths.sort((a, b) => {\n\t\t\tconst aStat = statSync(a)\n\t\t\tconst bStat = statSync(b)\n\t\t\treturn bStat.ctimeMs - aStat.ctimeMs\n\t\t})\n\t\treturn (filePathsByDateCreated[index] as string & keyof T) ?? null\n\t}\n\n\tpublic clear(): void {\n\t\trmSync(this.rootDir, { recursive: true })\n\t\tmkdirSync(this.rootDir, { recursive: true })\n\t}\n\n\tpublic get length(): number {\n\t\treturn readdirSync(this.rootDir).length\n\t}\n}\n",
    "import { execSync, spawn } from \"node:child_process\"\nimport type { Server } from \"node:http\"\nimport { createServer } from \"node:http\"\nimport { homedir } from \"node:os\"\nimport { resolve } from \"node:path\"\nimport { inspect } from \"node:util\"\n\nimport { Future } from \"atom.io/internal\"\nimport { discoverType } from \"atom.io/introspection\"\nimport { fromEntries, toEntries } from \"atom.io/json\"\nimport { ChildSocket } from \"atom.io/realtime-server\"\nimport { CronJob } from \"cron\"\nimport { z } from \"zod\"\n\nimport type { LnavFormat } from \"../gen/lnav-format-schema.gen\"\nimport { FilesystemStorage } from \"./filesystem-storage\"\nimport { env } from \"./flightdeck.env\"\n\nexport const FLIGHTDECK_SETUP_PHASES = [`downloaded`, `installed`] as const\n\nexport type FlightDeckSetupPhase = (typeof FLIGHTDECK_SETUP_PHASES)[number]\n\nexport const FLIGHTDECK_UPDATE_PHASES = [`notified`, `confirmed`] as const\n\nexport type FlightDeckUpdatePhase = (typeof FLIGHTDECK_UPDATE_PHASES)[number]\n\nexport function isVersionNumber(version: string): boolean {\n\treturn (\n\t\t/^\\d+\\.\\d+\\.\\d+$/.test(version) || !Number.isNaN(Number.parseFloat(version))\n\t)\n}\n\nexport type FlightDeckOptions<S extends string = string> = {\n\tpackageName: string\n\tservices: { [service in S]: { run: string; waitFor: boolean } }\n\tscripts: {\n\t\tdownload: string\n\t\tinstall: string\n\t\tcheckAvailability?: string\n\t}\n\tport?: number | undefined\n\tflightdeckRootDir?: string | undefined\n\tjsonLogging?: boolean | undefined\n}\n\nexport class FlightDeck<S extends string = string> {\n\tprotected safety = 0\n\n\tprotected storage: FilesystemStorage<{\n\t\tsetupPhase: FlightDeckSetupPhase\n\t\tupdatePhase: FlightDeckUpdatePhase\n\t\tupdateAwaitedVersion: string\n\t}>\n\tprotected webhookServer: Server\n\tprotected services: {\n\t\t[service in S]: ChildSocket<\n\t\t\t{ timeToStop: []; updatesReady: [] },\n\t\t\t{ readyToUpdate: []; alive: [] }\n\t\t> | null\n\t}\n\tprotected serviceIdx: { readonly [service in S]: number }\n\tpublic defaultServicesReadyToUpdate: { readonly [service in S]: boolean }\n\tpublic servicesReadyToUpdate: { [service in S]: boolean }\n\tpublic autoRespawnDeadServices: boolean\n\n\tprotected logger: Pick<Console, `error` | `info` | `warn`>\n\tprotected serviceLoggers: {\n\t\treadonly [service in S]: FlightDeckLogger\n\t}\n\n\tprotected updateAvailabilityChecker: CronJob | null = null\n\n\tpublic servicesLive: Future<void>[]\n\tpublic servicesDead: Future<void>[]\n\tpublic live = new Future(() => {})\n\tpublic dead = new Future(() => {})\n\n\tprotected restartTimes: number[] = []\n\n\tpublic constructor(public readonly options: FlightDeckOptions<S>) {\n\t\tconst { FLIGHTDECK_SECRET } = env\n\t\tconst { flightdeckRootDir = resolve(homedir(), `.flightdeck`) } = options\n\t\tconst port = options.port ?? 8080\n\t\tconst origin = `http://localhost:${port}`\n\n\t\tconst servicesEntries = toEntries(options.services)\n\t\tthis.services = fromEntries(\n\t\t\tservicesEntries.map(([serviceName]) => [serviceName, null]),\n\t\t)\n\t\tthis.serviceIdx = fromEntries(\n\t\t\tservicesEntries.map(([serviceName], idx) => [serviceName, idx]),\n\t\t)\n\t\tthis.defaultServicesReadyToUpdate = fromEntries(\n\t\t\tservicesEntries.map(([serviceName, { waitFor }]) => [\n\t\t\t\tserviceName,\n\t\t\t\t!waitFor,\n\t\t\t]),\n\t\t)\n\t\tthis.servicesReadyToUpdate = { ...this.defaultServicesReadyToUpdate }\n\t\tthis.autoRespawnDeadServices = true\n\n\t\tthis.logger = new FlightDeckLogger(\n\t\t\tthis.options.packageName,\n\t\t\tprocess.pid,\n\t\t\tundefined,\n\t\t\t{ jsonLogging: this.options.jsonLogging ?? false },\n\t\t)\n\t\tthis.serviceLoggers = fromEntries(\n\t\t\tservicesEntries.map(([serviceName]) => [\n\t\t\t\tserviceName,\n\t\t\t\tnew FlightDeckLogger(\n\t\t\t\t\tthis.options.packageName,\n\t\t\t\t\tprocess.pid,\n\t\t\t\t\tserviceName,\n\t\t\t\t\t{ jsonLogging: this.options.jsonLogging ?? false },\n\t\t\t\t),\n\t\t\t]),\n\t\t)\n\n\t\tthis.servicesLive = servicesEntries.map(() => new Future(() => {}))\n\t\tthis.servicesDead = servicesEntries.map(() => new Future(() => {}))\n\t\tthis.live.use(Promise.all(this.servicesLive))\n\t\tthis.dead.use(Promise.all(this.servicesDead))\n\n\t\tthis.storage = new FilesystemStorage({\n\t\t\tpath: resolve(flightdeckRootDir, `storage`, options.packageName),\n\t\t})\n\n\t\tif (FLIGHTDECK_SECRET === undefined) {\n\t\t\tthis.logger.warn(\n\t\t\t\t`No FLIGHTDECK_SECRET environment variable found. FlightDeck will not run an update server.`,\n\t\t\t)\n\t\t} else {\n\t\t\tcreateServer((req, res) => {\n\t\t\t\tlet data: Uint8Array[] = []\n\t\t\t\treq\n\t\t\t\t\t.on(`data`, (chunk) => {\n\t\t\t\t\t\tdata.push(chunk instanceof Buffer ? chunk : Buffer.from(chunk))\n\t\t\t\t\t})\n\t\t\t\t\t.on(`end`, () => {\n\t\t\t\t\t\tconst authHeader = req.headers.authorization\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tif (typeof req.url === `undefined`) throw 400\n\t\t\t\t\t\t\tconst expectedAuthHeader = `Bearer ${FLIGHTDECK_SECRET}`\n\t\t\t\t\t\t\tif (authHeader !== `Bearer ${FLIGHTDECK_SECRET}`) {\n\t\t\t\t\t\t\t\tthis.logger.info(\n\t\t\t\t\t\t\t\t\t`Unauthorized: needed \\`${expectedAuthHeader}\\`, got \\`${authHeader}\\``,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\tthrow 401\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tconst url = new URL(req.url, origin)\n\t\t\t\t\t\t\tthis.logger.info(req.method, url.pathname)\n\n\t\t\t\t\t\t\tconst versionForeignInput = Buffer.concat(data).toString()\n\t\t\t\t\t\t\tif (!isVersionNumber(versionForeignInput)) {\n\t\t\t\t\t\t\t\tthrow 400\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tres.writeHead(200)\n\t\t\t\t\t\t\tres.end()\n\n\t\t\t\t\t\t\tthis.storage.setItem(`updatePhase`, `notified`)\n\t\t\t\t\t\t\tthis.storage.setItem(`updateAwaitedVersion`, versionForeignInput)\n\t\t\t\t\t\t\tconst { checkAvailability } = options.scripts\n\t\t\t\t\t\t\tif (checkAvailability) {\n\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker?.stop()\n\t\t\t\t\t\t\t\tthis.seekUpdate(versionForeignInput)\n\t\t\t\t\t\t\t\tconst updatePhase = this.storage.getItem(`updatePhase`)\n\t\t\t\t\t\t\t\tthis.logger.info(`> storage(\"updatePhase\") >`, updatePhase)\n\t\t\t\t\t\t\t\tif (updatePhase === `notified`) {\n\t\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker = new CronJob(\n\t\t\t\t\t\t\t\t\t\t`30 * * * * *`,\n\t\t\t\t\t\t\t\t\t\t() => {\n\t\t\t\t\t\t\t\t\t\t\tthis.seekUpdate(versionForeignInput)\n\t\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker.start()\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tthis.downloadPackage()\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} catch (thrown) {\n\t\t\t\t\t\t\tthis.logger.error(thrown, req.url)\n\t\t\t\t\t\t\tif (typeof thrown === `number`) {\n\t\t\t\t\t\t\t\tres.writeHead(thrown)\n\t\t\t\t\t\t\t\tres.end()\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} finally {\n\t\t\t\t\t\t\tdata = []\n\t\t\t\t\t\t}\n\t\t\t\t\t})\n\t\t\t}).listen(port, () => {\n\t\t\t\tthis.logger.info(`Server started on port ${port}`)\n\t\t\t})\n\t\t}\n\n\t\tthis.startAllServices()\n\t\t\t.then(() => {\n\t\t\t\tthis.logger.info(`All services started.`)\n\t\t\t})\n\t\t\t.catch((thrown) => {\n\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\tthis.logger.error(`Failed to start all services:`, thrown.message)\n\t\t\t\t}\n\t\t\t})\n\t}\n\n\tprotected seekUpdate(version: string): void {\n\t\tthis.logger.info(`Checking for updates...`)\n\t\tconst { checkAvailability } = this.options.scripts\n\t\tif (!checkAvailability) {\n\t\t\tthis.logger.info(`No checkAvailability script found.`)\n\t\t\treturn\n\t\t}\n\t\ttry {\n\t\t\tconst out = execSync(`${checkAvailability} ${version}`)\n\t\t\tthis.logger.info(`Check stdout:`, out.toString())\n\t\t\tthis.updateAvailabilityChecker?.stop()\n\t\t\tthis.storage.setItem(`updatePhase`, `confirmed`)\n\t\t\tthis.downloadPackage()\n\t\t\tthis.announceUpdate()\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Check failed:`, thrown.message)\n\t\t\t} else {\n\t\t\t\tconst thrownType = discoverType(thrown)\n\t\t\t\tthis.logger.error(`Check threw`, thrownType, thrown)\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected announceUpdate(): void {\n\t\tfor (const entry of toEntries(this.services)) {\n\t\t\tconst [serviceName, service] = entry\n\t\t\tif (service) {\n\t\t\t\tif (this.options.services[serviceName].waitFor) {\n\t\t\t\t\tservice.emit(`updatesReady`)\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tthis.startService(serviceName)\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected tryUpdate(): void {\n\t\tif (toEntries(this.servicesReadyToUpdate).every(([, isReady]) => isReady)) {\n\t\t\tthis.logger.info(`All services are ready to update.`)\n\t\t\tthis.stopAllServices()\n\t\t\t\t.then(() => {\n\t\t\t\t\tthis.logger.info(`All services stopped; starting up fresh...`)\n\t\t\t\t\tthis.startAllServices()\n\t\t\t\t\t\t.then(() => {\n\t\t\t\t\t\t\tthis.logger.info(`All services started; we're back online.`)\n\t\t\t\t\t\t})\n\t\t\t\t\t\t.catch((thrown) => {\n\t\t\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t\t\t`Failed to start all services:`,\n\t\t\t\t\t\t\t\t\tthrown.message,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t})\n\t\t\t\t})\n\t\t\t\t.catch((thrown) => {\n\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\tthis.logger.error(`Failed to stop all services:`, thrown.message)\n\t\t\t\t\t}\n\t\t\t\t})\n\t\t}\n\t}\n\n\tprotected startAllServices(): Future<unknown> {\n\t\tthis.logger.info(`Starting all services...`)\n\t\tthis.autoRespawnDeadServices = true\n\t\tconst setupPhase = this.storage.getItem(`setupPhase`)\n\t\tthis.logger.info(`> storage(\"setupPhase\") >`, setupPhase)\n\t\tswitch (setupPhase) {\n\t\t\tcase null:\n\t\t\t\tthis.logger.info(`Starting from scratch.`)\n\t\t\t\tthis.downloadPackage()\n\t\t\t\tthis.installPackage()\n\t\t\t\treturn this.startAllServices()\n\t\t\tcase `downloaded`:\n\t\t\t\tthis.logger.info(`Found package downloaded but not installed.`)\n\t\t\t\tthis.installPackage()\n\t\t\t\treturn this.startAllServices()\n\t\t\tcase `installed`: {\n\t\t\t\tfor (const [serviceName] of toEntries(this.services)) {\n\t\t\t\t\tthis.startService(serviceName)\n\t\t\t\t}\n\t\t\t\treturn this.live\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected startService(serviceName: S): void {\n\t\tthis.logger.info(\n\t\t\t`Starting service ${this.options.packageName}::${serviceName}, try ${this.safety}/2...`,\n\t\t)\n\t\tif (this.safety >= 2) {\n\t\t\tthrow new Error(`Out of tries...`)\n\t\t}\n\t\tthis.safety++\n\n\t\tconst [exe, ...args] = this.options.services[serviceName].run.split(` `)\n\t\tconst serviceProcess = spawn(exe, args, {\n\t\t\tcwd: this.options.flightdeckRootDir,\n\t\t\tenv: import.meta.env,\n\t\t})\n\t\tconst serviceLogger = this.serviceLoggers[serviceName]\n\t\tconst service = (this.services[serviceName] = new ChildSocket(\n\t\t\tserviceProcess,\n\t\t\t`${this.options.packageName}::${serviceName}`,\n\t\t\tserviceLogger,\n\t\t))\n\t\tserviceLogger.processCode = service.process.pid ?? -1\n\t\tthis.services[serviceName].onAny((...messages) => {\n\t\t\tserviceLogger.info(`💬`, ...messages)\n\t\t})\n\t\tthis.services[serviceName].on(`readyToUpdate`, () => {\n\t\t\tthis.logger.info(`Service \"${serviceName}\" is ready to update.`)\n\t\t\tthis.servicesReadyToUpdate[serviceName] = true\n\t\t\tthis.tryUpdate()\n\t\t})\n\t\tthis.services[serviceName].on(`alive`, () => {\n\t\t\tthis.servicesLive[this.serviceIdx[serviceName]].use(Promise.resolve())\n\t\t\tthis.servicesDead[this.serviceIdx[serviceName]] = new Future(() => {})\n\t\t\tif (this.dead.done) {\n\t\t\t\tthis.dead = new Future(() => {})\n\t\t\t}\n\t\t\tthis.dead.use(Promise.all(this.servicesDead))\n\t\t})\n\t\tthis.services[serviceName].process.once(`close`, (exitCode) => {\n\t\t\tthis.logger.info(\n\t\t\t\t`Auto-respawn saw \"${serviceName}\" exit with code ${exitCode}`,\n\t\t\t)\n\t\t\tthis.services[serviceName] = null\n\t\t\tif (!this.autoRespawnDeadServices) {\n\t\t\t\tthis.logger.info(`Auto-respawn is off; \"${serviceName}\" rests.`)\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst updatePhase = this.storage.getItem(`updatePhase`)\n\t\t\tthis.logger.info(`> storage(\"updatePhase\") >`, updatePhase)\n\t\t\tconst updatesAreReady = updatePhase === `confirmed`\n\t\t\tif (updatesAreReady) {\n\t\t\t\tthis.serviceLoggers[serviceName].info(`Updating before startup...`)\n\t\t\t\tthis.restartTimes = []\n\t\t\t\tthis.installPackage()\n\t\t\t\tthis.startService(serviceName)\n\t\t\t} else {\n\t\t\t\tconst now = Date.now()\n\t\t\t\tconst fiveMinutesAgo = now - 5 * 60 * 1000\n\t\t\t\tthis.restartTimes = this.restartTimes.filter(\n\t\t\t\t\t(time) => time > fiveMinutesAgo,\n\t\t\t\t)\n\t\t\t\tthis.restartTimes.push(now)\n\n\t\t\t\tif (this.restartTimes.length < 5) {\n\t\t\t\t\tthis.serviceLoggers[serviceName].info(`Crashed. Restarting...`)\n\t\t\t\t\tthis.startService(serviceName)\n\t\t\t\t} else {\n\t\t\t\t\tthis.serviceLoggers[serviceName].info(\n\t\t\t\t\t\t`Crashed 5 times in 5 minutes. Not restarting.`,\n\t\t\t\t\t)\n\t\t\t\t}\n\t\t\t}\n\t\t})\n\t\tthis.safety = 0\n\t}\n\n\tprotected downloadPackage(): void {\n\t\tthis.logger.info(`Downloading...`)\n\t\ttry {\n\t\t\tconst out = execSync(this.options.scripts.download)\n\t\t\tthis.logger.info(`Download stdout:`, out.toString())\n\t\t\tthis.storage.setItem(`setupPhase`, `downloaded`)\n\t\t\tthis.logger.info(`Downloaded!`)\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Failed to get the latest release: ${thrown.message}`)\n\t\t\t}\n\t\t\treturn\n\t\t}\n\t}\n\n\tprotected installPackage(): void {\n\t\tthis.logger.info(`Installing...`)\n\n\t\ttry {\n\t\t\tconst out = execSync(this.options.scripts.install)\n\t\t\tthis.logger.info(`Install stdout:`, out.toString())\n\t\t\tthis.storage.setItem(`setupPhase`, `installed`)\n\t\t\tthis.logger.info(`Installed!`)\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Failed to get the latest release: ${thrown.message}`)\n\t\t\t}\n\t\t\treturn\n\t\t}\n\t}\n\n\tpublic stopAllServices(): Future<unknown> {\n\t\tthis.logger.info(`Stopping all services... auto-respawn disabled.`)\n\t\tthis.autoRespawnDeadServices = false\n\t\tfor (const [serviceName] of toEntries(this.services)) {\n\t\t\tthis.stopService(serviceName)\n\t\t}\n\t\treturn this.dead\n\t}\n\n\tpublic stopService(serviceName: S): void {\n\t\tconst service = this.services[serviceName]\n\t\tif (service) {\n\t\t\tthis.logger.info(`Stopping service \"${serviceName}\"...`)\n\t\t\tthis.servicesDead[this.serviceIdx[serviceName]].use(\n\t\t\t\tnew Promise((pass) => {\n\t\t\t\t\tservice.emit(`timeToStop`)\n\t\t\t\t\tservice.process.once(`close`, (exitCode) => {\n\t\t\t\t\t\tthis.logger.info(\n\t\t\t\t\t\t\t`Stopped service \"${serviceName}\"; exited with code ${exitCode}`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tthis.services[serviceName] = null\n\t\t\t\t\t\tpass()\n\t\t\t\t\t})\n\t\t\t\t}),\n\t\t\t)\n\t\t\tthis.dead.use(Promise.all(this.servicesDead))\n\t\t\tthis.servicesLive[this.serviceIdx[serviceName]] = new Future(() => {})\n\t\t\tif (this.live.done) {\n\t\t\t\tthis.live = new Future(() => {})\n\t\t\t}\n\t\t\tthis.live.use(Promise.all(this.servicesLive))\n\t\t} else {\n\t\t\tthis.serviceLoggers[serviceName].error(\n\t\t\t\t`Tried to stop service, but it wasn't running.`,\n\t\t\t)\n\t\t}\n\t}\n}\n\nexport const flightDeckLogSchema = z.object({\n\tlevel: z.union([z.literal(`info`), z.literal(`warn`), z.literal(`ERR!`)]),\n\ttimestamp: z.number(),\n\tpackage: z.string(),\n\tservice: z.string().optional(),\n\tprocess: z.number(),\n\tbody: z.string(),\n})\nexport type FlightDeckLog = z.infer<typeof flightDeckLogSchema>\n\nconst LINE_FORMAT = `line-format` satisfies keyof LnavFormat\nconst VALUE = `value` satisfies keyof LnavFormat\n\nexport type LnavFormatVisualComponent = Exclude<\n\tExclude<LnavFormat[`line-format`], undefined>[number],\n\tstring\n>\n\nexport type LnavFormatBreakdown = Exclude<LnavFormat[`value`], undefined>\nexport type MemberOf<T> = T[keyof T]\nexport type LnavFormatValueDefinition = MemberOf<LnavFormatBreakdown>\n\nexport type FlightDeckFormat = {\n\t[LINE_FORMAT]: (\n\t\t| string\n\t\t| (LnavFormatVisualComponent & {\n\t\t\t\tfield: keyof FlightDeckLog | `__level__` | `__timestamp__`\n\t\t  })\n\t)[]\n\t[VALUE]: {\n\t\t[K in keyof FlightDeckLog]: LnavFormatValueDefinition & {\n\t\t\tkind: FlightDeckLog[K] extends number | undefined\n\t\t\t\t? `integer`\n\t\t\t\t: FlightDeckLog[K] extends string | undefined\n\t\t\t\t\t? `string`\n\t\t\t\t\t: never\n\t\t}\n\t}\n}\n\nexport const FLIGHTDECK_LNAV_FORMAT = {\n\ttitle: `FlightDeck Log`,\n\tdescription: `Format for events logged by the FlightDeck process manager.`,\n\t\"file-type\": `json`,\n\t\"timestamp-field\": `timestamp`,\n\t\"subsecond-field\": `subsecond`,\n\t\"subsecond-units\": `milli`,\n\t\"opid-field\": `service`,\n\t\"level-field\": `level`,\n\tlevel: {\n\t\tinfo: `info`,\n\t\twarning: `warn`,\n\t\terror: `err!`,\n\t},\n\t\"ordered-by-time\": true,\n\n\t[LINE_FORMAT]: [\n\t\t{\n\t\t\tprefix: ` `,\n\t\t\tfield: `__timestamp__`,\n\t\t\t\"timestamp-format\": `%Y-%m-%dT%H:%M:%S.%L%Z`,\n\t\t},\n\t\t{\n\t\t\tprefix: ` `,\n\t\t\tfield: `process`,\n\t\t\t\"min-width\": 5,\n\t\t},\n\t\t{\n\t\t\tprefix: `:`,\n\t\t\tfield: `package`,\n\t\t},\n\t\t{\n\t\t\tprefix: `:`,\n\t\t\tfield: `service`,\n\t\t\t\"default-value\": ``,\n\t\t},\n\t\t{\n\t\t\tprefix: `[`,\n\t\t\tfield: `__level__`,\n\t\t\tsuffix: `]`,\n\t\t},\n\t\t{\n\t\t\tprefix: `: `,\n\t\t\tfield: `body`,\n\t\t},\n\t],\n\n\t[VALUE]: {\n\t\ttimestamp: {\n\t\t\tkind: `integer`,\n\t\t},\n\t\tlevel: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tpackage: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tservice: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tprocess: {\n\t\t\tkind: `integer`,\n\t\t},\n\t\tbody: {\n\t\t\tkind: `string`,\n\t\t},\n\t},\n} as const satisfies FlightDeckFormat & LnavFormat\n\nexport class FlightDeckLogger\n\timplements Pick<Console, `error` | `info` | `warn`>\n{\n\tpublic readonly packageName: string\n\tpublic readonly serviceName?: string\n\tpublic readonly jsonLogging: boolean\n\tpublic processCode: number\n\tpublic constructor(\n\t\tpackageName: string,\n\t\tprocessCode: number,\n\t\tserviceName?: string,\n\t\toptions?: { jsonLogging: boolean },\n\t) {\n\t\tthis.packageName = packageName\n\t\tif (serviceName) {\n\t\t\tthis.serviceName = serviceName\n\t\t}\n\t\tthis.processCode = processCode\n\t\tthis.jsonLogging = options?.jsonLogging ?? false\n\t}\n\tpublic info(...messages: unknown[]): void {\n\t\tif (this.jsonLogging) {\n\t\t\tconst log: FlightDeckLog = {\n\t\t\t\ttimestamp: Date.now(),\n\t\t\t\tlevel: `info`,\n\t\t\t\tprocess: this.processCode,\n\t\t\t\tpackage: this.packageName,\n\t\t\t\tbody: messages\n\t\t\t\t\t.map((message) =>\n\t\t\t\t\t\ttypeof message === `string`\n\t\t\t\t\t\t\t? message\n\t\t\t\t\t\t\t: inspect(message, false, null, true),\n\t\t\t\t\t)\n\t\t\t\t\t.join(` `),\n\t\t\t}\n\t\t\tif (this.serviceName) {\n\t\t\t\tlog.service = this.serviceName\n\t\t\t}\n\t\t\tprocess.stdout.write(JSON.stringify(log) + `\\n`)\n\t\t} else {\n\t\t\tconst source = this.serviceName\n\t\t\t\t? `${this.packageName}::${this.serviceName}`\n\t\t\t\t: this.packageName\n\t\t\tconsole.log(`${source}:`, ...messages)\n\t\t}\n\t}\n\n\tpublic warn(...messages: unknown[]): void {\n\t\tif (this.jsonLogging) {\n\t\t\tconst log: FlightDeckLog = {\n\t\t\t\ttimestamp: Date.now(),\n\t\t\t\tlevel: `warn`,\n\t\t\t\tprocess: this.processCode,\n\t\t\t\tpackage: this.packageName,\n\t\t\t\tbody: messages\n\t\t\t\t\t.map((message) =>\n\t\t\t\t\t\ttypeof message === `string`\n\t\t\t\t\t\t\t? message\n\t\t\t\t\t\t\t: inspect(message, false, null, true),\n\t\t\t\t\t)\n\t\t\t\t\t.join(` `),\n\t\t\t}\n\t\t\tif (this.serviceName) {\n\t\t\t\tlog.service = this.serviceName\n\t\t\t}\n\t\t\tprocess.stdout.write(JSON.stringify(log) + `\\n`)\n\t\t} else {\n\t\t\tconst source = this.serviceName\n\t\t\t\t? `${this.packageName}::${this.serviceName}`\n\t\t\t\t: this.packageName\n\t\t\tconsole.warn(`${source}:`, ...messages)\n\t\t}\n\t}\n\n\tpublic error(...messages: unknown[]): void {\n\t\tif (this.jsonLogging) {\n\t\t\tconst log: FlightDeckLog = {\n\t\t\t\ttimestamp: Date.now() + Math.floor(Math.random() * 1000),\n\t\t\t\tlevel: `ERR!`,\n\t\t\t\tprocess: this.processCode,\n\t\t\t\tpackage: this.packageName,\n\t\t\t\tbody: messages\n\t\t\t\t\t.map((message) =>\n\t\t\t\t\t\ttypeof message === `string`\n\t\t\t\t\t\t\t? message\n\t\t\t\t\t\t\t: inspect(message, false, null, true),\n\t\t\t\t\t)\n\t\t\t\t\t.join(` `),\n\t\t\t}\n\t\t\tif (this.serviceName) {\n\t\t\t\tlog.service = this.serviceName\n\t\t\t}\n\t\t\tprocess.stdout.write(JSON.stringify(log) + `\\n`)\n\t\t} else {\n\t\t\tconst source = this.serviceName\n\t\t\t\t? `${this.packageName}::${this.serviceName}`\n\t\t\t\t: this.packageName\n\t\t\tconsole.error(`${source}:`, ...messages)\n\t\t}\n\t}\n}\n",
    "import { createEnv } from \"@t3-oss/env-core\"\nimport { z } from \"zod\"\n\nexport const env = createEnv({\n\tserver: { FLIGHTDECK_SECRET: z.string().optional() },\n\tclientPrefix: `NEVER`,\n\tclient: {},\n\truntimeEnv: import.meta.env,\n\temptyStringAsUndefined: true,\n})\n",
    "export type AlertOptions = {\n\tsecret: string\n\tendpoint: string\n\tversion: string\n}\n\nexport async function alert({\n\tsecret,\n\tendpoint,\n\tversion,\n}: AlertOptions): Promise<Response> {\n\tconst response = await fetch(endpoint, {\n\t\tmethod: `POST`,\n\t\theaders: {\n\t\t\t\"Content-Type\": `text/plain;charset=UTF-8`,\n\t\t\tAuthorization: `Bearer ${secret}`,\n\t\t},\n\t\tbody: version,\n\t})\n\n\treturn response\n}\n\n/**\n * @see https://github.com/changesets/action/blob/main/src/run.ts\n */\nexport type ChangesetsPublishedPackage = {\n\tname: string\n\tversion: string\n}\n\n/**\n * @see https://github.com/changesets/action/blob/main/src/run.ts\n */\nexport type ChangesetsPublishResult =\n\t| {\n\t\t\tpublished: true\n\t\t\tpublishedPackages: ChangesetsPublishedPackage[]\n\t  }\n\t| { published: false }\n\nexport type PackageConfig<K extends string> = {\n\t[key in K]: { endpoint: string }\n}\nexport type SecretsConfig<K extends string> = {\n\t[key in K]: string\n}\n\nexport type ScrambleOptions<K extends string = string> = {\n\tpackageConfig: PackageConfig<K>\n\tsecretsConfig: SecretsConfig<K>\n\tpublishedPackages: ChangesetsPublishedPackage[]\n}\n\nexport type ScrambleResult<K extends string = string> = {\n\t[key in K]: Response\n}\n\nexport async function scramble<K extends string = string>({\n\tpackageConfig,\n\tsecretsConfig,\n\tpublishedPackages,\n}: ScrambleOptions<K>): Promise<ScrambleResult<K>> {\n\tconst alertResults: Promise<readonly [K, Response]>[] = []\n\tfor (const publishedPackage of publishedPackages) {\n\t\tif (publishedPackage.name in packageConfig) {\n\t\t\tconst name = publishedPackage.name as K\n\t\t\tconst { endpoint } = packageConfig[name]\n\t\t\tconst secret = secretsConfig[name]\n\t\t\tconst version = publishedPackage.version\n\t\t\tconst alertResultPromise = alert({ secret, endpoint, version }).then(\n\t\t\t\t(alertResult) => [name, alertResult] as const,\n\t\t\t)\n\t\t\talertResults.push(alertResultPromise)\n\t\t}\n\t}\n\tconst alertResultsResolved = await Promise.all(alertResults)\n\tconst scrambleResult = Object.fromEntries(\n\t\talertResultsResolved,\n\t) as ScrambleResult<K>\n\treturn scrambleResult\n}\n"
  ],
  "mappings": "8HAAA,qBACC,eACA,iBACA,kBACA,YACA,cACA,mBACA,gBAED,kBAAS,kBAMF,MAAM,CAGb,CACQ,QAEA,WAAW,CAAC,EAAmC,CAErD,GADA,KAAK,QAAU,EAAQ,MAClB,EAAW,KAAK,OAAO,EAC3B,EAAU,KAAK,QAAS,CAAE,UAAW,EAAK,CAAC,EAItC,OAAmC,CAAC,EAAqB,CAC/D,IAAM,EAAW,EAAQ,KAAK,QAAS,CAAG,EAC1C,GAAI,EAAW,CAAQ,EACtB,OAAO,EAAa,EAAU,OAAO,EAEtC,OAAO,KAGD,OAAmC,CAAC,EAAQ,EAAmB,CACrE,IAAM,EAAW,EAAQ,KAAK,QAAS,CAAG,EAC1C,EAAc,EAAU,CAAK,EAGvB,UAAsC,CAAC,EAAc,CAC3D,IAAM,EAAW,EAAQ,KAAK,QAAS,CAAG,EAC1C,GAAI,EAAW,CAAQ,EACtB,EAAO,CAAQ,EAIV,GAAG,CAAC,EAA0C,CAOpD,OANkB,EAAY,KAAK,OAAO,EACD,KAAK,CAAC,EAAG,IAAM,CACvD,IAAM,EAAQ,EAAS,CAAC,EAExB,OADc,EAAS,CAAC,EACX,QAAU,EAAM,QAC7B,EAC8B,IAA+B,KAGxD,KAAK,EAAS,CACpB,EAAO,KAAK,QAAS,CAAE,UAAW,EAAK,CAAC,EACxC,EAAU,KAAK,QAAS,CAAE,UAAW,EAAK,CAAC,KAGjC,OAAM,EAAW,CAC3B,OAAO,EAAY,KAAK,OAAO,EAAE,OAEnC,CClEA,mBAAS,WAAU,2BAEnB,uBAAS,kBACT,kBAAS,gBACT,kBAAS,kBACT,kBAAS,kBAET,iBAAS,yBACT,uBAAS,8BACT,sBAAS,eAAa,qBACtB,sBAAS,gCACT,kBAAS,aACT,YAAS,YCZT,oBAAS,yBACT,YAAS,YAEF,IAAM,EAAM,EAAU,CAC5B,OAAQ,CAAE,kBAAmB,EAAE,OAAO,EAAE,SAAS,CAAE,EACnD,aAAc,QACd,OAAQ,CAAC,EACT,WAAY,YAAY,IACxB,uBAAwB,EACzB,CAAC,EDSM,IAAM,GAA0B,CAAC,aAAc,WAAW,EAIpD,GAA2B,CAAC,WAAY,WAAW,EAIzD,SAAS,CAAe,CAAC,EAA0B,CACzD,MACC,kBAAkB,KAAK,CAAO,IAAM,OAAO,MAAM,OAAO,WAAW,CAAO,CAAC,EAiBtE,MAAM,CAAsC,CAkCf,QAjCzB,OAAS,EAET,QAKA,cACA,SAMA,WACH,6BACA,sBACA,wBAEG,OACA,eAIA,0BAA4C,KAE/C,aACA,aACA,KAAO,IAAI,EAAO,IAAM,EAAE,EAC1B,KAAO,IAAI,EAAO,IAAM,EAAE,EAEvB,aAAyB,CAAC,EAE7B,WAAW,CAAiB,EAA+B,CAA/B,eAClC,IAAQ,qBAAsB,GACtB,oBAAoB,EAAQ,EAAQ,EAAG,aAAa,GAAM,EAC5D,EAAO,EAAQ,MAAQ,KACvB,EAAS,oBAAoB,IAE7B,EAAkB,EAAU,EAAQ,QAAQ,EA2ClD,GA1CA,KAAK,SAAW,EACf,EAAgB,IAAI,EAAE,KAAiB,CAAC,EAAa,IAAI,CAAC,CAC3D,EACA,KAAK,WAAa,EACjB,EAAgB,IAAI,EAAE,GAAc,IAAQ,CAAC,EAAa,CAAG,CAAC,CAC/D,EACA,KAAK,6BAA+B,EACnC,EAAgB,IAAI,EAAE,GAAe,cAAe,CACnD,GACC,CACF,CAAC,CACF,EACA,KAAK,sBAAwB,IAAK,KAAK,4BAA6B,EACpE,KAAK,wBAA0B,GAE/B,KAAK,OAAS,IAAI,EACjB,KAAK,QAAQ,YACb,QAAQ,IACR,OACA,CAAE,YAAa,KAAK,QAAQ,aAAe,EAAM,CAClD,EACA,KAAK,eAAiB,EACrB,EAAgB,IAAI,EAAE,KAAiB,CACtC,EACA,IAAI,EACH,KAAK,QAAQ,YACb,QAAQ,IACR,EACA,CAAE,YAAa,KAAK,QAAQ,aAAe,EAAM,CAClD,CACD,CAAC,CACF,EAEA,KAAK,aAAe,EAAgB,IAAI,IAAM,IAAI,EAAO,IAAM,EAAE,CAAC,EAClE,KAAK,aAAe,EAAgB,IAAI,IAAM,IAAI,EAAO,IAAM,EAAE,CAAC,EAClE,KAAK,KAAK,IAAI,QAAQ,IAAI,KAAK,YAAY,CAAC,EAC5C,KAAK,KAAK,IAAI,QAAQ,IAAI,KAAK,YAAY,CAAC,EAE5C,KAAK,QAAU,IAAI,EAAkB,CACpC,KAAM,EAAQ,EAAmB,UAAW,EAAQ,WAAW,CAChE,CAAC,EAEG,IAAsB,OACzB,KAAK,OAAO,KACX,4FACD,MAEA,GAAa,CAAC,EAAK,IAAQ,CAC1B,IAAI,EAAqB,CAAC,EAC1B,EACE,GAAG,OAAQ,CAAC,IAAU,CACtB,EAAK,KAAK,aAAiB,OAAS,EAAQ,OAAO,KAAK,CAAK,CAAC,EAC9D,EACA,GAAG,MAAO,IAAM,CAChB,IAAM,EAAa,EAAI,QAAQ,cAC/B,GAAI,CACH,UAAW,EAAI,MAAQ,YAAa,KAAM,KAC1C,IAAM,EAAqB,UAAU,IACrC,GAAI,IAAe,UAAU,IAI5B,MAHA,KAAK,OAAO,KACX,0BAA0B,cAA+B,KAC1D,EACM,IAEP,IAAM,EAAM,IAAI,IAAI,EAAI,IAAK,CAAM,EACnC,KAAK,OAAO,KAAK,EAAI,OAAQ,EAAI,QAAQ,EAEzC,IAAM,EAAsB,OAAO,OAAO,CAAI,EAAE,SAAS,EACzD,IAAK,EAAgB,CAAmB,EACvC,KAAM,KAGP,EAAI,UAAU,GAAG,EACjB,EAAI,IAAI,EAER,KAAK,QAAQ,QAAQ,cAAe,UAAU,EAC9C,KAAK,QAAQ,QAAQ,uBAAwB,CAAmB,EAChE,IAAQ,qBAAsB,EAAQ,QACtC,GAAI,EAAmB,CACtB,KAAK,2BAA2B,KAAK,EACrC,KAAK,WAAW,CAAmB,EACnC,IAAM,EAAc,KAAK,QAAQ,QAAQ,aAAa,EAEtD,GADA,KAAK,OAAO,KAAK,6BAA8B,CAAW,EACtD,IAAgB,WACnB,KAAK,0BAA4B,IAAI,EACpC,eACA,IAAM,CACL,KAAK,WAAW,CAAmB,EAErC,EACA,KAAK,0BAA0B,MAAM,MAGtC,MAAK,gBAAgB,QAEd,EAAP,CAED,GADA,KAAK,OAAO,MAAM,EAAQ,EAAI,GAAG,SACtB,IAAW,SACrB,EAAI,UAAU,CAAM,EACpB,EAAI,IAAI,SAER,CACD,EAAO,CAAC,GAET,EACF,EAAE,OAAO,EAAM,IAAM,CACrB,KAAK,OAAO,KAAK,0BAA0B,GAAM,EACjD,EAGF,KAAK,iBAAiB,EACpB,KAAK,IAAM,CACX,KAAK,OAAO,KAAK,uBAAuB,EACxC,EACA,MAAM,CAAC,IAAW,CAClB,GAAI,aAAkB,MACrB,KAAK,OAAO,MAAM,gCAAiC,EAAO,OAAO,EAElE,EAGO,UAAU,CAAC,EAAuB,CAC3C,KAAK,OAAO,KAAK,yBAAyB,EAC1C,IAAQ,qBAAsB,KAAK,QAAQ,QAC3C,IAAK,EAAmB,CACvB,KAAK,OAAO,KAAK,oCAAoC,EACrD,OAED,GAAI,CACH,IAAM,EAAM,EAAS,GAAG,KAAqB,GAAS,EACtD,KAAK,OAAO,KAAK,gBAAiB,EAAI,SAAS,CAAC,EAChD,KAAK,2BAA2B,KAAK,EACrC,KAAK,QAAQ,QAAQ,cAAe,WAAW,EAC/C,KAAK,gBAAgB,EACrB,KAAK,eAAe,QACZ,EAAP,CACD,GAAI,aAAkB,MACrB,KAAK,OAAO,MAAM,gBAAiB,EAAO,OAAO,MAC3C,CACN,IAAM,EAAa,EAAa,CAAM,EACtC,KAAK,OAAO,MAAM,cAAe,EAAY,CAAM,IAK5C,cAAc,EAAS,CAChC,QAAW,KAAS,EAAU,KAAK,QAAQ,EAAG,CAC7C,IAAO,EAAa,GAAW,EAC/B,GAAI,GACH,GAAI,KAAK,QAAQ,SAAS,GAAa,QACtC,EAAQ,KAAK,cAAc,MAG5B,MAAK,aAAa,CAAW,GAKtB,SAAS,EAAS,CAC3B,GAAI,EAAU,KAAK,qBAAqB,EAAE,MAAM,GAAI,KAAa,CAAO,EACvE,KAAK,OAAO,KAAK,mCAAmC,EACpD,KAAK,gBAAgB,EACnB,KAAK,IAAM,CACX,KAAK,OAAO,KAAK,4CAA4C,EAC7D,KAAK,iBAAiB,EACpB,KAAK,IAAM,CACX,KAAK,OAAO,KAAK,0CAA0C,EAC3D,EACA,MAAM,CAAC,IAAW,CAClB,GAAI,aAAkB,MACrB,KAAK,OAAO,MACX,gCACA,EAAO,OACR,EAED,EACF,EACA,MAAM,CAAC,IAAW,CAClB,GAAI,aAAkB,MACrB,KAAK,OAAO,MAAM,+BAAgC,EAAO,OAAO,EAEjE,EAIM,gBAAgB,EAAoB,CAC7C,KAAK,OAAO,KAAK,0BAA0B,EAC3C,KAAK,wBAA0B,GAC/B,IAAM,EAAa,KAAK,QAAQ,QAAQ,YAAY,EAEpD,OADA,KAAK,OAAO,KAAK,4BAA6B,CAAU,EAChD,QACF,KAIJ,OAHA,KAAK,OAAO,KAAK,wBAAwB,EACzC,KAAK,gBAAgB,EACrB,KAAK,eAAe,EACb,KAAK,iBAAiB,MACzB,aAGJ,OAFA,KAAK,OAAO,KAAK,6CAA6C,EAC9D,KAAK,eAAe,EACb,KAAK,iBAAiB,MACzB,YAAa,CACjB,QAAY,KAAgB,EAAU,KAAK,QAAQ,EAClD,KAAK,aAAa,CAAW,EAE9B,OAAO,KAAK,IACb,GAIQ,YAAY,CAAC,EAAsB,CAI5C,GAHA,KAAK,OAAO,KACX,oBAAoB,KAAK,QAAQ,gBAAgB,UAAoB,KAAK,aAC3E,EACI,KAAK,QAAU,EAClB,MAAM,IAAI,MAAM,iBAAiB,EAElC,KAAK,SAEL,IAAO,KAAQ,GAAQ,KAAK,QAAQ,SAAS,GAAa,IAAI,MAAM,GAAG,EACjE,EAAiB,EAAM,EAAK,EAAM,CACvC,IAAK,KAAK,QAAQ,kBAClB,IAAK,YAAY,GAClB,CAAC,EACK,EAAgB,KAAK,eAAe,GACpC,EAAW,KAAK,SAAS,GAAe,IAAI,EACjD,EACA,GAAG,KAAK,QAAQ,gBAAgB,IAChC,CACD,EACA,EAAc,YAAc,EAAQ,QAAQ,KAAO,GACnD,KAAK,SAAS,GAAa,MAAM,IAAI,IAAa,CACjD,EAAc,KAAK,eAAK,GAAG,CAAQ,EACnC,EACD,KAAK,SAAS,GAAa,GAAG,gBAAiB,IAAM,CACpD,KAAK,OAAO,KAAK,YAAY,wBAAkC,EAC/D,KAAK,sBAAsB,GAAe,GAC1C,KAAK,UAAU,EACf,EACD,KAAK,SAAS,GAAa,GAAG,QAAS,IAAM,CAG5C,GAFA,KAAK,aAAa,KAAK,WAAW,IAAc,IAAI,QAAQ,QAAQ,CAAC,EACrE,KAAK,aAAa,KAAK,WAAW,IAAgB,IAAI,EAAO,IAAM,EAAE,EACjE,KAAK,KAAK,KACb,KAAK,KAAO,IAAI,EAAO,IAAM,EAAE,EAEhC,KAAK,KAAK,IAAI,QAAQ,IAAI,KAAK,YAAY,CAAC,EAC5C,EACD,KAAK,SAAS,GAAa,QAAQ,KAAK,QAAS,CAAC,IAAa,CAK9D,GAJA,KAAK,OAAO,KACX,qBAAqB,qBAA+B,GACrD,EACA,KAAK,SAAS,GAAe,MACxB,KAAK,wBAAyB,CAClC,KAAK,OAAO,KAAK,yBAAyB,WAAqB,EAC/D,OAED,IAAM,EAAc,KAAK,QAAQ,QAAQ,aAAa,EAGtD,GAFA,KAAK,OAAO,KAAK,6BAA8B,CAAW,EAClC,IAAgB,YAEvC,KAAK,eAAe,GAAa,KAAK,4BAA4B,EAClE,KAAK,aAAe,CAAC,EACrB,KAAK,eAAe,EACpB,KAAK,aAAa,CAAW,MACvB,CACN,IAAM,EAAM,KAAK,IAAI,EACf,EAAiB,EAAM,OAM7B,GALA,KAAK,aAAe,KAAK,aAAa,OACrC,CAAC,IAAS,EAAO,CAClB,EACA,KAAK,aAAa,KAAK,CAAG,EAEtB,KAAK,aAAa,OAAS,EAC9B,KAAK,eAAe,GAAa,KAAK,wBAAwB,EAC9D,KAAK,aAAa,CAAW,MAE7B,MAAK,eAAe,GAAa,KAChC,+CACD,GAGF,EACD,KAAK,OAAS,EAGL,eAAe,EAAS,CACjC,KAAK,OAAO,KAAK,gBAAgB,EACjC,GAAI,CACH,IAAM,EAAM,EAAS,KAAK,QAAQ,QAAQ,QAAQ,EAClD,KAAK,OAAO,KAAK,mBAAoB,EAAI,SAAS,CAAC,EACnD,KAAK,QAAQ,QAAQ,aAAc,YAAY,EAC/C,KAAK,OAAO,KAAK,aAAa,QACtB,EAAP,CACD,GAAI,aAAkB,MACrB,KAAK,OAAO,MAAM,qCAAqC,EAAO,SAAS,EAExE,QAIQ,cAAc,EAAS,CAChC,KAAK,OAAO,KAAK,eAAe,EAEhC,GAAI,CACH,IAAM,EAAM,EAAS,KAAK,QAAQ,QAAQ,OAAO,EACjD,KAAK,OAAO,KAAK,kBAAmB,EAAI,SAAS,CAAC,EAClD,KAAK,QAAQ,QAAQ,aAAc,WAAW,EAC9C,KAAK,OAAO,KAAK,YAAY,QACrB,EAAP,CACD,GAAI,aAAkB,MACrB,KAAK,OAAO,MAAM,qCAAqC,EAAO,SAAS,EAExE,QAIK,eAAe,EAAoB,CACzC,KAAK,OAAO,KAAK,iDAAiD,EAClE,KAAK,wBAA0B,GAC/B,QAAY,KAAgB,EAAU,KAAK,QAAQ,EAClD,KAAK,YAAY,CAAW,EAE7B,OAAO,KAAK,KAGN,WAAW,CAAC,EAAsB,CACxC,IAAM,EAAU,KAAK,SAAS,GAC9B,GAAI,EAAS,CAgBZ,GAfA,KAAK,OAAO,KAAK,qBAAqB,OAAiB,EACvD,KAAK,aAAa,KAAK,WAAW,IAAc,IAC/C,IAAI,QAAQ,CAAC,IAAS,CACrB,EAAQ,KAAK,YAAY,EACzB,EAAQ,QAAQ,KAAK,QAAS,CAAC,IAAa,CAC3C,KAAK,OAAO,KACX,oBAAoB,wBAAkC,GACvD,EACA,KAAK,SAAS,GAAe,KAC7B,EAAK,EACL,EACD,CACF,EACA,KAAK,KAAK,IAAI,QAAQ,IAAI,KAAK,YAAY,CAAC,EAC5C,KAAK,aAAa,KAAK,WAAW,IAAgB,IAAI,EAAO,IAAM,EAAE,EACjE,KAAK,KAAK,KACb,KAAK,KAAO,IAAI,EAAO,IAAM,EAAE,EAEhC,KAAK,KAAK,IAAI,QAAQ,IAAI,KAAK,YAAY,CAAC,MAE5C,MAAK,eAAe,GAAa,MAChC,+CACD,EAGH,CAEO,IAAM,GAAsB,EAAE,OAAO,CAC3C,MAAO,EAAE,MAAM,CAAC,EAAE,QAAQ,MAAM,EAAG,EAAE,QAAQ,MAAM,EAAG,EAAE,QAAQ,MAAM,CAAC,CAAC,EACxE,UAAW,EAAE,OAAO,EACpB,QAAS,EAAE,OAAO,EAClB,QAAS,EAAE,OAAO,EAAE,SAAS,EAC7B,QAAS,EAAE,OAAO,EAClB,KAAM,EAAE,OAAO,CAChB,CAAC,EAGK,EAAc,cACd,EAAQ,QA6BD,GAAyB,CACrC,MAAO,iBACP,YAAa,8DACb,YAAa,OACb,kBAAmB,YACnB,kBAAmB,YACnB,kBAAmB,QACnB,aAAc,UACd,cAAe,QACf,MAAO,CACN,KAAM,OACN,QAAS,OACT,MAAO,MACR,EACA,kBAAmB,IAElB,GAAc,CACd,CACC,OAAQ,IACR,MAAO,gBACP,mBAAoB,wBACrB,EACA,CACC,OAAQ,IACR,MAAO,UACP,YAAa,CACd,EACA,CACC,OAAQ,IACR,MAAO,SACR,EACA,CACC,OAAQ,IACR,MAAO,UACP,gBAAiB,EAClB,EACA,CACC,OAAQ,IACR,MAAO,YACP,OAAQ,GACT,EACA,CACC,OAAQ,KACR,MAAO,MACR,CACD,GAEC,GAAQ,CACR,UAAW,CACV,KAAM,SACP,EACA,MAAO,CACN,KAAM,QACP,EACA,QAAS,CACR,KAAM,QACP,EACA,QAAS,CACR,KAAM,QACP,EACA,QAAS,CACR,KAAM,SACP,EACA,KAAM,CACL,KAAM,QACP,CACD,CACD,EAEO,MAAM,CAEb,CACiB,YACA,YACA,YACT,YACA,WAAW,CACjB,EACA,EACA,EACA,EACC,CAED,GADA,KAAK,YAAc,EACf,EACH,KAAK,YAAc,EAEpB,KAAK,YAAc,EACnB,KAAK,YAAc,GAAS,aAAe,GAErC,IAAI,IAAI,EAA2B,CACzC,GAAI,KAAK,YAAa,CACrB,IAAM,EAAqB,CAC1B,UAAW,KAAK,IAAI,EACpB,MAAO,OACP,QAAS,KAAK,YACd,QAAS,KAAK,YACd,KAAM,EACJ,IAAI,CAAC,WACE,IAAY,SAChB,EACA,EAAQ,EAAS,GAAO,KAAM,EAAI,CACtC,EACC,KAAK,GAAG,CACX,EACA,GAAI,KAAK,YACR,EAAI,QAAU,KAAK,YAEpB,QAAQ,OAAO,MAAM,KAAK,UAAU,CAAG,EAAI;AAAA,CAAI,MACzC,CACN,IAAM,EAAS,KAAK,YACjB,GAAG,KAAK,gBAAgB,KAAK,cAC7B,KAAK,YACR,QAAQ,IAAI,GAAG,KAAW,GAAG,CAAQ,GAIhC,IAAI,IAAI,EAA2B,CACzC,GAAI,KAAK,YAAa,CACrB,IAAM,EAAqB,CAC1B,UAAW,KAAK,IAAI,EACpB,MAAO,OACP,QAAS,KAAK,YACd,QAAS,KAAK,YACd,KAAM,EACJ,IAAI,CAAC,WACE,IAAY,SAChB,EACA,EAAQ,EAAS,GAAO,KAAM,EAAI,CACtC,EACC,KAAK,GAAG,CACX,EACA,GAAI,KAAK,YACR,EAAI,QAAU,KAAK,YAEpB,QAAQ,OAAO,MAAM,KAAK,UAAU,CAAG,EAAI;AAAA,CAAI,MACzC,CACN,IAAM,EAAS,KAAK,YACjB,GAAG,KAAK,gBAAgB,KAAK,cAC7B,KAAK,YACR,QAAQ,KAAK,GAAG,KAAW,GAAG,CAAQ,GAIjC,KAAK,IAAI,EAA2B,CAC1C,GAAI,KAAK,YAAa,CACrB,IAAM,EAAqB,CAC1B,UAAW,KAAK,IAAI,EAAI,KAAK,MAAM,KAAK,OAAO,EAAI,IAAI,EACvD,MAAO,OACP,QAAS,KAAK,YACd,QAAS,KAAK,YACd,KAAM,EACJ,IAAI,CAAC,WACE,IAAY,SAChB,EACA,EAAQ,EAAS,GAAO,KAAM,EAAI,CACtC,EACC,KAAK,GAAG,CACX,EACA,GAAI,KAAK,YACR,EAAI,QAAU,KAAK,YAEpB,QAAQ,OAAO,MAAM,KAAK,UAAU,CAAG,EAAI;AAAA,CAAI,MACzC,CACN,IAAM,EAAS,KAAK,YACjB,GAAG,KAAK,gBAAgB,KAAK,cAC7B,KAAK,YACR,QAAQ,MAAM,GAAG,KAAW,GAAG,CAAQ,GAG1C,4CEnoBA,eAAsB,CAAK,EAC1B,SACA,WACA,WACmC,CAUnC,OATiB,MAAM,MAAM,EAAU,CACtC,OAAQ,OACR,QAAS,CACR,eAAgB,2BAChB,cAAe,UAAU,GAC1B,EACA,KAAM,CACP,CAAC,EAwCF,eAAsB,CAAmC,EACxD,gBACA,gBACA,qBACkD,CAClD,IAAM,EAAkD,CAAC,EACzD,QAAW,KAAoB,EAC9B,GAAI,EAAiB,QAAQ,EAAe,CAC3C,IAAM,EAAO,EAAiB,MACtB,YAAa,EAAc,GAC7B,EAAS,EAAc,GACvB,EAAU,EAAiB,QAC3B,EAAqB,EAAM,CAAE,SAAQ,WAAU,SAAQ,CAAC,EAAE,KAC/D,CAAC,IAAgB,CAAC,EAAM,CAAW,CACpC,EACA,EAAa,KAAK,CAAkB,EAGtC,IAAM,EAAuB,MAAM,QAAQ,IAAI,CAAY,EAI3D,OAHuB,OAAO,YAC7B,CACD",
  "debugId": "D614A1AC9E2EE18564756E2164756E21",
  "names": []
}
8
+ //# debugId=16A02FFB0350BCFF64756E2164756E21
9
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/filesystem-storage.ts", "../src/flightdeck.lib.ts", "../src/flightdeck.env.ts", "../src/klaxon.lib.ts"],
  "sourcesContent": [
    "import {\n\texistsSync,\n\tmkdirSync,\n\treaddirSync,\n\treadFileSync,\n\trmSync,\n\tstatSync,\n\twriteFileSync,\n} from \"node:fs\"\nimport { resolve } from \"node:path\"\n\nexport type FilesystemStorageOptions = {\n\tpath: string\n}\n\nexport class FilesystemStorage<\n\tT extends Record<string, string> = Record<string, string>,\n> implements Storage\n{\n\tpublic rootDir: string\n\n\tpublic constructor(options: FilesystemStorageOptions) {\n\t\tthis.rootDir = options.path\n\t\tif (!existsSync(this.rootDir)) {\n\t\t\tmkdirSync(this.rootDir, { recursive: true })\n\t\t}\n\t}\n\n\tpublic getItem<K extends string & keyof T>(key: K): T[K] | null {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\tif (existsSync(filePath)) {\n\t\t\treturn readFileSync(filePath, `utf-8`) as T[K]\n\t\t}\n\t\treturn null\n\t}\n\n\tpublic setItem<K extends string & keyof T>(key: K, value: T[K]): void {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\twriteFileSync(filePath, value)\n\t}\n\n\tpublic removeItem<K extends string & keyof T>(key: K): void {\n\t\tconst filePath = resolve(this.rootDir, key)\n\t\tif (existsSync(filePath)) {\n\t\t\trmSync(filePath)\n\t\t}\n\t}\n\n\tpublic key(index: number): (string & keyof T) | null {\n\t\tconst filePaths = readdirSync(this.rootDir)\n\t\tconst filePathsByDateCreated = filePaths.sort((a, b) => {\n\t\t\tconst aStat = statSync(a)\n\t\t\tconst bStat = statSync(b)\n\t\t\treturn bStat.ctimeMs - aStat.ctimeMs\n\t\t})\n\t\treturn (filePathsByDateCreated[index] as string & keyof T) ?? null\n\t}\n\n\tpublic clear(): void {\n\t\trmSync(this.rootDir, { recursive: true })\n\t\tmkdirSync(this.rootDir, { recursive: true })\n\t}\n\n\tpublic get length(): number {\n\t\treturn readdirSync(this.rootDir).length\n\t}\n}\n",
    "import { execSync, spawn } from \"node:child_process\"\nimport type { Server } from \"node:http\"\nimport { createServer } from \"node:http\"\nimport { homedir } from \"node:os\"\nimport { resolve } from \"node:path\"\nimport { inspect } from \"node:util\"\n\nimport { Future } from \"atom.io/internal\"\nimport { discoverType } from \"atom.io/introspection\"\nimport { fromEntries, toEntries } from \"atom.io/json\"\nimport { ChildSocket } from \"atom.io/realtime-server\"\nimport { CronJob } from \"cron\"\nimport { z } from \"zod\"\n\nimport type { LnavFormat } from \"../gen/lnav-format-schema.gen\"\nimport { FilesystemStorage } from \"./filesystem-storage\"\nimport { env } from \"./flightdeck.env\"\n\nexport const FLIGHTDECK_SETUP_PHASES = [`downloaded`, `installed`] as const\n\nexport type FlightDeckSetupPhase = (typeof FLIGHTDECK_SETUP_PHASES)[number]\n\nexport const FLIGHTDECK_UPDATE_PHASES = [`notified`, `confirmed`] as const\n\nexport type FlightDeckUpdatePhase = (typeof FLIGHTDECK_UPDATE_PHASES)[number]\n\nexport function isVersionNumber(version: string): boolean {\n\treturn (\n\t\t/^\\d+\\.\\d+\\.\\d+$/.test(version) || !Number.isNaN(Number.parseFloat(version))\n\t)\n}\n\nexport type FlightDeckOptions<S extends string = string> = {\n\tpackageName: string\n\tservices: { [service in S]: { run: string; waitFor: boolean } }\n\tscripts: {\n\t\tdownload: string\n\t\tinstall: string\n\t\tcheckAvailability?: string\n\t}\n\tport?: number | undefined\n\tflightdeckRootDir?: string | undefined\n\tjsonLogging?: boolean | undefined\n}\n\nexport class FlightDeck<S extends string = string> {\n\tprotected safety = 0\n\n\tprotected storage: FilesystemStorage<{\n\t\tsetupPhase: FlightDeckSetupPhase\n\t\tupdatePhase: FlightDeckUpdatePhase\n\t\tupdateAwaitedVersion: string\n\t}>\n\tprotected webhookServer: Server\n\tprotected services: {\n\t\t[service in S]: ChildSocket<\n\t\t\t{ timeToStop: []; updatesReady: [] },\n\t\t\t{ readyToUpdate: []; alive: [] }\n\t\t> | null\n\t}\n\tprotected serviceIdx: { readonly [service in S]: number }\n\tpublic defaultServicesReadyToUpdate: { readonly [service in S]: boolean }\n\tpublic servicesReadyToUpdate: { [service in S]: boolean }\n\tpublic autoRespawnDeadServices: boolean\n\n\tprotected logger: Pick<Console, `error` | `info` | `warn`>\n\tprotected serviceLoggers: {\n\t\treadonly [service in S]: FlightDeckLogger\n\t}\n\n\tprotected updateAvailabilityChecker: CronJob | null = null\n\n\tpublic servicesLive: Future<void>[]\n\tpublic servicesDead: Future<void>[]\n\tpublic live = new Future(() => {})\n\tpublic dead = new Future(() => {})\n\n\tprotected restartTimes: number[] = []\n\n\tpublic constructor(public readonly options: FlightDeckOptions<S>) {\n\t\tconst { FLIGHTDECK_SECRET } = env\n\t\tconst { flightdeckRootDir = resolve(homedir(), `.flightdeck`) } = options\n\t\tconst port = options.port ?? 8080\n\t\tconst origin = `http://localhost:${port}`\n\n\t\tconst servicesEntries = toEntries(options.services)\n\t\tthis.services = fromEntries(\n\t\t\tservicesEntries.map(([serviceName]) => [serviceName, null]),\n\t\t)\n\t\tthis.serviceIdx = fromEntries(\n\t\t\tservicesEntries.map(([serviceName], idx) => [serviceName, idx]),\n\t\t)\n\t\tthis.defaultServicesReadyToUpdate = fromEntries(\n\t\t\tservicesEntries.map(([serviceName, { waitFor }]) => [\n\t\t\t\tserviceName,\n\t\t\t\t!waitFor,\n\t\t\t]),\n\t\t)\n\t\tthis.servicesReadyToUpdate = { ...this.defaultServicesReadyToUpdate }\n\t\tthis.autoRespawnDeadServices = true\n\n\t\tthis.logger = new FlightDeckLogger(\n\t\t\tthis.options.packageName,\n\t\t\tprocess.pid,\n\t\t\tundefined,\n\t\t\t{ jsonLogging: this.options.jsonLogging ?? false },\n\t\t)\n\t\tthis.serviceLoggers = fromEntries(\n\t\t\tservicesEntries.map(([serviceName]) => [\n\t\t\t\tserviceName,\n\t\t\t\tnew FlightDeckLogger(\n\t\t\t\t\tthis.options.packageName,\n\t\t\t\t\tprocess.pid,\n\t\t\t\t\tserviceName,\n\t\t\t\t\t{ jsonLogging: this.options.jsonLogging ?? false },\n\t\t\t\t),\n\t\t\t]),\n\t\t)\n\n\t\tthis.servicesLive = servicesEntries.map(() => new Future(() => {}))\n\t\tthis.servicesDead = servicesEntries.map(() => new Future(() => {}))\n\t\tthis.live.use(Promise.all(this.servicesLive))\n\t\tthis.dead.use(Promise.all(this.servicesDead))\n\n\t\tthis.storage = new FilesystemStorage({\n\t\t\tpath: resolve(flightdeckRootDir, `storage`, options.packageName),\n\t\t})\n\n\t\tif (FLIGHTDECK_SECRET === undefined) {\n\t\t\tthis.logger.warn(\n\t\t\t\t`No FLIGHTDECK_SECRET environment variable found. FlightDeck will not run an update server.`,\n\t\t\t)\n\t\t} else {\n\t\t\tcreateServer((req, res) => {\n\t\t\t\tlet data: Uint8Array[] = []\n\t\t\t\treq\n\t\t\t\t\t.on(`data`, (chunk) => {\n\t\t\t\t\t\tdata.push(chunk instanceof Buffer ? chunk : Buffer.from(chunk))\n\t\t\t\t\t})\n\t\t\t\t\t.on(`end`, () => {\n\t\t\t\t\t\tconst authHeader = req.headers.authorization\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tif (typeof req.url === `undefined`) throw 400\n\t\t\t\t\t\t\tconst expectedAuthHeader = `Bearer ${FLIGHTDECK_SECRET}`\n\t\t\t\t\t\t\tif (authHeader !== `Bearer ${FLIGHTDECK_SECRET}`) {\n\t\t\t\t\t\t\t\tthis.logger.info(\n\t\t\t\t\t\t\t\t\t`Unauthorized: needed \\`${expectedAuthHeader}\\`, got \\`${authHeader}\\``,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\tthrow 401\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tconst url = new URL(req.url, origin)\n\t\t\t\t\t\t\tthis.logger.info(req.method, url.pathname)\n\n\t\t\t\t\t\t\tconst versionForeignInput = Buffer.concat(data).toString()\n\t\t\t\t\t\t\tif (!isVersionNumber(versionForeignInput)) {\n\t\t\t\t\t\t\t\tthrow 400\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tres.writeHead(200)\n\t\t\t\t\t\t\tres.end()\n\n\t\t\t\t\t\t\tthis.storage.setItem(`updatePhase`, `notified`)\n\t\t\t\t\t\t\tthis.storage.setItem(`updateAwaitedVersion`, versionForeignInput)\n\t\t\t\t\t\t\tconst { checkAvailability } = options.scripts\n\t\t\t\t\t\t\tif (checkAvailability) {\n\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker?.stop()\n\t\t\t\t\t\t\t\tthis.seekUpdate(versionForeignInput)\n\t\t\t\t\t\t\t\tconst updatePhase = this.storage.getItem(`updatePhase`)\n\t\t\t\t\t\t\t\tthis.logger.info(`> storage(\"updatePhase\") >`, updatePhase)\n\t\t\t\t\t\t\t\tif (updatePhase === `notified`) {\n\t\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker = new CronJob(\n\t\t\t\t\t\t\t\t\t\t`30 * * * * *`,\n\t\t\t\t\t\t\t\t\t\t() => {\n\t\t\t\t\t\t\t\t\t\t\tthis.seekUpdate(versionForeignInput)\n\t\t\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\t\tthis.updateAvailabilityChecker.start()\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tthis.downloadPackage()\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} catch (thrown) {\n\t\t\t\t\t\t\tthis.logger.error(thrown, req.url)\n\t\t\t\t\t\t\tif (typeof thrown === `number`) {\n\t\t\t\t\t\t\t\tres.writeHead(thrown)\n\t\t\t\t\t\t\t\tres.end()\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} finally {\n\t\t\t\t\t\t\tdata = []\n\t\t\t\t\t\t}\n\t\t\t\t\t})\n\t\t\t}).listen(port, () => {\n\t\t\t\tthis.logger.info(`Server started on port ${port}`)\n\t\t\t})\n\t\t}\n\n\t\tthis.startAllServices()\n\t\t\t.then(() => {\n\t\t\t\tthis.logger.info(`All services started.`)\n\t\t\t})\n\t\t\t.catch((thrown) => {\n\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\tthis.logger.error(`Failed to start all services:`, thrown.message)\n\t\t\t\t}\n\t\t\t})\n\t}\n\n\tprotected seekUpdate(version: string): void {\n\t\tthis.logger.info(`Checking for updates...`)\n\t\tconst { checkAvailability } = this.options.scripts\n\t\tif (!checkAvailability) {\n\t\t\tthis.logger.info(`No checkAvailability script found.`)\n\t\t\treturn\n\t\t}\n\t\ttry {\n\t\t\tconst out = execSync(`${checkAvailability} ${version}`)\n\t\t\tthis.logger.info(`Check stdout:`, out.toString())\n\t\t\tthis.updateAvailabilityChecker?.stop()\n\t\t\tthis.storage.setItem(`updatePhase`, `confirmed`)\n\t\t\tthis.downloadPackage()\n\t\t\tthis.announceUpdate()\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Check failed:`, thrown.message)\n\t\t\t} else {\n\t\t\t\tconst thrownType = discoverType(thrown)\n\t\t\t\tthis.logger.error(`Check threw`, thrownType, thrown)\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected announceUpdate(): void {\n\t\tfor (const entry of toEntries(this.services)) {\n\t\t\tconst [serviceName, service] = entry\n\t\t\tif (service) {\n\t\t\t\tif (this.options.services[serviceName].waitFor) {\n\t\t\t\t\tservice.emit(`updatesReady`)\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tthis.startService(serviceName)\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected tryUpdate(): void {\n\t\tif (toEntries(this.servicesReadyToUpdate).every(([, isReady]) => isReady)) {\n\t\t\tthis.logger.info(`All services are ready to update.`)\n\t\t\tthis.stopAllServices()\n\t\t\t\t.then(() => {\n\t\t\t\t\tthis.logger.info(`All services stopped; starting up fresh...`)\n\t\t\t\t\tthis.startAllServices()\n\t\t\t\t\t\t.then(() => {\n\t\t\t\t\t\t\tthis.logger.info(`All services started; we're back online.`)\n\t\t\t\t\t\t})\n\t\t\t\t\t\t.catch((thrown) => {\n\t\t\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t\t\t`Failed to start all services:`,\n\t\t\t\t\t\t\t\t\tthrown.message,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t})\n\t\t\t\t})\n\t\t\t\t.catch((thrown) => {\n\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\tthis.logger.error(`Failed to stop all services:`, thrown.message)\n\t\t\t\t\t}\n\t\t\t\t})\n\t\t}\n\t}\n\n\tprotected startAllServices(): Future<unknown> {\n\t\tthis.logger.info(`Starting all services...`)\n\t\tthis.autoRespawnDeadServices = true\n\t\tconst setupPhase = this.storage.getItem(`setupPhase`)\n\t\tthis.logger.info(`> storage(\"setupPhase\") >`, setupPhase)\n\t\tswitch (setupPhase) {\n\t\t\tcase null:\n\t\t\t\tthis.logger.info(`Starting from scratch.`)\n\t\t\t\tthis.downloadPackage()\n\t\t\t\tthis.installPackage()\n\t\t\t\treturn this.startAllServices()\n\t\t\tcase `downloaded`:\n\t\t\t\tthis.logger.info(`Found package downloaded but not installed.`)\n\t\t\t\tthis.installPackage()\n\t\t\t\treturn this.startAllServices()\n\t\t\tcase `installed`: {\n\t\t\t\tfor (const [serviceName] of toEntries(this.services)) {\n\t\t\t\t\tthis.startService(serviceName)\n\t\t\t\t}\n\t\t\t\treturn this.live\n\t\t\t}\n\t\t}\n\t}\n\n\tprotected startService(serviceName: S): void {\n\t\tthis.logger.info(\n\t\t\t`Starting service ${this.options.packageName}::${serviceName}, try ${this.safety}/2...`,\n\t\t)\n\t\tif (this.safety >= 2) {\n\t\t\tthrow new Error(`Out of tries...`)\n\t\t}\n\t\tthis.safety++\n\n\t\tconst [exe, ...args] = this.options.services[serviceName].run.split(` `)\n\t\tconst serviceProcess = spawn(exe, args, {\n\t\t\tcwd: this.options.flightdeckRootDir,\n\t\t\tenv: import.meta.env,\n\t\t})\n\t\tconst serviceLogger = this.serviceLoggers[serviceName]\n\t\tconst service = (this.services[serviceName] = new ChildSocket(\n\t\t\tserviceProcess,\n\t\t\t`${this.options.packageName}::${serviceName}`,\n\t\t\tserviceLogger,\n\t\t))\n\t\tserviceLogger.processCode = service.process.pid ?? -1\n\t\tthis.services[serviceName].onAny((...messages) => {\n\t\t\tserviceLogger.info(`💬`, ...messages)\n\t\t})\n\t\tthis.services[serviceName].on(`readyToUpdate`, () => {\n\t\t\tthis.logger.info(`Service \"${serviceName}\" is ready to update.`)\n\t\t\tthis.servicesReadyToUpdate[serviceName] = true\n\t\t\tthis.tryUpdate()\n\t\t})\n\t\tthis.services[serviceName].on(`alive`, () => {\n\t\t\tthis.servicesLive[this.serviceIdx[serviceName]].use(Promise.resolve())\n\t\t\tthis.servicesDead[this.serviceIdx[serviceName]] = new Future(() => {})\n\t\t\tif (this.dead.done) {\n\t\t\t\tthis.dead = new Future(() => {})\n\t\t\t}\n\t\t\tthis.dead.use(Promise.all(this.servicesDead))\n\t\t})\n\t\tthis.services[serviceName].process.once(`close`, (exitCode) => {\n\t\t\tthis.logger.info(\n\t\t\t\t`Auto-respawn saw \"${serviceName}\" exit with code ${exitCode}`,\n\t\t\t)\n\t\t\tthis.services[serviceName] = null\n\t\t\tif (!this.autoRespawnDeadServices) {\n\t\t\t\tthis.logger.info(`Auto-respawn is off; \"${serviceName}\" rests.`)\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst updatePhase = this.storage.getItem(`updatePhase`)\n\t\t\tthis.logger.info(`> storage(\"updatePhase\") >`, updatePhase)\n\t\t\tconst updatesAreReady = updatePhase === `confirmed`\n\t\t\tif (updatesAreReady) {\n\t\t\t\tthis.serviceLoggers[serviceName].info(`Updating before startup...`)\n\t\t\t\tthis.restartTimes = []\n\t\t\t\tthis.installPackage()\n\t\t\t\tthis.startService(serviceName)\n\t\t\t} else {\n\t\t\t\tconst now = Date.now()\n\t\t\t\tconst fiveMinutesAgo = now - 5 * 60 * 1000\n\t\t\t\tthis.restartTimes = this.restartTimes.filter(\n\t\t\t\t\t(time) => time > fiveMinutesAgo,\n\t\t\t\t)\n\t\t\t\tthis.restartTimes.push(now)\n\n\t\t\t\tif (this.restartTimes.length < 5) {\n\t\t\t\t\tthis.serviceLoggers[serviceName].info(`Crashed. Restarting...`)\n\t\t\t\t\tthis.startService(serviceName)\n\t\t\t\t} else {\n\t\t\t\t\tthis.serviceLoggers[serviceName].info(\n\t\t\t\t\t\t`Crashed 5 times in 5 minutes. Not restarting.`,\n\t\t\t\t\t)\n\t\t\t\t}\n\t\t\t}\n\t\t})\n\t\tthis.safety = 0\n\t}\n\n\tprotected downloadPackage(): void {\n\t\tthis.logger.info(`Downloading...`)\n\t\ttry {\n\t\t\tconst out = execSync(this.options.scripts.download)\n\t\t\tthis.logger.info(`Download stdout:`, out.toString())\n\t\t\tthis.storage.setItem(`setupPhase`, `downloaded`)\n\t\t\tthis.logger.info(`Downloaded!`)\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Failed to get the latest release: ${thrown.message}`)\n\t\t\t}\n\t\t\treturn\n\t\t}\n\t}\n\n\tprotected installPackage(): void {\n\t\tthis.logger.info(`Installing...`)\n\n\t\ttry {\n\t\t\tconst out = execSync(this.options.scripts.install)\n\t\t\tthis.logger.info(`Install stdout:`, out.toString())\n\t\t\tthis.storage.setItem(`setupPhase`, `installed`)\n\t\t\tthis.logger.info(`Installed!`)\n\t\t} catch (thrown) {\n\t\t\tif (thrown instanceof Error) {\n\t\t\t\tthis.logger.error(`Failed to get the latest release: ${thrown.message}`)\n\t\t\t}\n\t\t\treturn\n\t\t}\n\t}\n\n\tpublic stopAllServices(): Future<unknown> {\n\t\tthis.logger.info(`Stopping all services... auto-respawn disabled.`)\n\t\tthis.autoRespawnDeadServices = false\n\t\tfor (const [serviceName] of toEntries(this.services)) {\n\t\t\tthis.stopService(serviceName)\n\t\t}\n\t\treturn this.dead\n\t}\n\n\tpublic stopService(serviceName: S): void {\n\t\tconst service = this.services[serviceName]\n\t\tif (service) {\n\t\t\tthis.logger.info(`Stopping service \"${serviceName}\"...`)\n\t\t\tthis.servicesDead[this.serviceIdx[serviceName]].use(\n\t\t\t\tnew Promise((pass) => {\n\t\t\t\t\tservice.emit(`timeToStop`)\n\t\t\t\t\tservice.process.once(`close`, (exitCode) => {\n\t\t\t\t\t\tthis.logger.info(\n\t\t\t\t\t\t\t`Stopped service \"${serviceName}\"; exited with code ${exitCode}`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tthis.services[serviceName] = null\n\t\t\t\t\t\tpass()\n\t\t\t\t\t})\n\t\t\t\t}),\n\t\t\t)\n\t\t\tthis.dead.use(Promise.all(this.servicesDead))\n\t\t\tthis.servicesLive[this.serviceIdx[serviceName]] = new Future(() => {})\n\t\t\tif (this.live.done) {\n\t\t\t\tthis.live = new Future(() => {})\n\t\t\t}\n\t\t\tthis.live.use(Promise.all(this.servicesLive))\n\t\t} else {\n\t\t\tthis.serviceLoggers[serviceName].error(\n\t\t\t\t`Tried to stop service, but it wasn't running.`,\n\t\t\t)\n\t\t}\n\t}\n}\n\nexport const FLIGHTDECK_INFO = `info`\nexport const FLIGHTDECK_WARN = `warn`\nexport const FLIGHTDECK_ERROR = `ERR!`\n\nexport const flightDeckLogSchema = z.object({\n\tlevel: z.union([\n\t\tz.literal(FLIGHTDECK_INFO),\n\t\tz.literal(FLIGHTDECK_WARN),\n\t\tz.literal(FLIGHTDECK_ERROR),\n\t]),\n\ttimestamp: z.number(),\n\tpackage: z.string(),\n\tservice: z.string().optional(),\n\tprocess: z.number(),\n\tbody: z.string(),\n})\nexport type FlightDeckLog = z.infer<typeof flightDeckLogSchema>\n\nconst LINE_FORMAT = `line-format` satisfies keyof LnavFormat\nconst VALUE = `value` satisfies keyof LnavFormat\n\nexport type LnavFormatVisualComponent = Exclude<\n\tExclude<LnavFormat[`line-format`], undefined>[number],\n\tstring\n>\n\nexport type LnavFormatBreakdown = Exclude<LnavFormat[`value`], undefined>\nexport type MemberOf<T> = T[keyof T]\nexport type LnavFormatValueDefinition = MemberOf<LnavFormatBreakdown>\n\nexport type FlightDeckFormat = {\n\t[LINE_FORMAT]: (\n\t\t| string\n\t\t| (LnavFormatVisualComponent & {\n\t\t\t\tfield: keyof FlightDeckLog | `__level__` | `__timestamp__`\n\t\t  })\n\t)[]\n\t[VALUE]: {\n\t\t[K in keyof FlightDeckLog]: LnavFormatValueDefinition & {\n\t\t\tkind: FlightDeckLog[K] extends number | undefined\n\t\t\t\t? `integer`\n\t\t\t\t: FlightDeckLog[K] extends string | undefined\n\t\t\t\t\t? `string`\n\t\t\t\t\t: never\n\t\t}\n\t}\n}\n\nexport const FLIGHTDECK_LNAV_FORMAT = {\n\ttitle: `FlightDeck Log`,\n\tdescription: `Format for events logged by the FlightDeck process manager.`,\n\t\"file-type\": `json`,\n\t\"timestamp-field\": `timestamp`,\n\t\"timestamp-divisor\": 1000,\n\t\"module-field\": `package`,\n\t\"opid-field\": `service`,\n\t\"level-field\": `level`,\n\tlevel: {\n\t\tinfo: FLIGHTDECK_INFO,\n\t\twarning: FLIGHTDECK_WARN,\n\t\terror: FLIGHTDECK_ERROR,\n\t},\n\n\t[LINE_FORMAT]: [\n\t\t{\n\t\t\tfield: `level`,\n\t\t},\n\t\t{\n\t\t\tprefix: ` `,\n\t\t\tfield: `__timestamp__`,\n\t\t\t\"timestamp-format\": `%Y-%m-%dT%H:%M:%S.%L%Z`,\n\t\t},\n\t\t{\n\t\t\tprefix: ` `,\n\t\t\tfield: `process`,\n\t\t\t\"min-width\": 5,\n\t\t},\n\t\t{\n\t\t\tprefix: `:`,\n\t\t\tfield: `package`,\n\t\t},\n\t\t{\n\t\t\tprefix: `:`,\n\t\t\tfield: `service`,\n\t\t\t\"default-value\": ``,\n\t\t},\n\t\t{\n\t\t\tprefix: `: `,\n\t\t\tfield: `body`,\n\t\t},\n\t],\n\n\t[VALUE]: {\n\t\ttimestamp: {\n\t\t\tkind: `integer`,\n\t\t},\n\t\tlevel: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tpackage: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tservice: {\n\t\t\tkind: `string`,\n\t\t},\n\t\tprocess: {\n\t\t\tkind: `integer`,\n\t\t},\n\t\tbody: {\n\t\t\tkind: `string`,\n\t\t},\n\t},\n} as const satisfies FlightDeckFormat & LnavFormat\n\nexport class FlightDeckLogger\n\timplements Pick<Console, `error` | `info` | `warn`>\n{\n\tpublic readonly packageName: string\n\tpublic readonly serviceName?: string\n\tpublic readonly jsonLogging: boolean\n\tpublic processCode: number\n\tpublic constructor(\n\t\tpackageName: string,\n\t\tprocessCode: number,\n\t\tserviceName?: string,\n\t\toptions?: { jsonLogging: boolean },\n\t) {\n\t\tthis.packageName = packageName\n\t\tif (serviceName) {\n\t\t\tthis.serviceName = serviceName\n\t\t}\n\t\tthis.processCode = processCode\n\t\tthis.jsonLogging = options?.jsonLogging ?? false\n\t}\n\tprotected log(\n\t\tlevel:\n\t\t\t| typeof FLIGHTDECK_ERROR\n\t\t\t| typeof FLIGHTDECK_INFO\n\t\t\t| typeof FLIGHTDECK_WARN,\n\t\t...messages: unknown[]\n\t): void {\n\t\tif (this.jsonLogging) {\n\t\t\tlet body = messages\n\t\t\t\t.map((message) =>\n\t\t\t\t\ttypeof message === `string`\n\t\t\t\t\t\t? message\n\t\t\t\t\t\t: inspect(message, false, null, true),\n\t\t\t\t)\n\t\t\t\t.join(` `)\n\t\t\tif (body.includes(`\\n`)) {\n\t\t\t\tbody = `\\n  ${body.split(`\\n`).join(`\\n  `)}`\n\t\t\t}\n\t\t\tconst log: FlightDeckLog = {\n\t\t\t\ttimestamp: Date.now(),\n\t\t\t\tlevel,\n\t\t\t\tprocess: this.processCode,\n\t\t\t\tpackage: this.packageName,\n\t\t\t\tbody,\n\t\t\t}\n\t\t\tif (this.serviceName) {\n\t\t\t\tlog.service = this.serviceName\n\t\t\t}\n\t\t\tprocess.stdout.write(JSON.stringify(log) + `\\n`)\n\t\t} else {\n\t\t\tconst source = this.serviceName\n\t\t\t\t? `${this.packageName}:${this.serviceName}`\n\t\t\t\t: this.packageName\n\t\t\tswitch (level) {\n\t\t\t\tcase FLIGHTDECK_INFO:\n\t\t\t\t\tconsole.log(`${source}:`, ...messages)\n\t\t\t\t\tbreak\n\t\t\t\tcase FLIGHTDECK_WARN:\n\t\t\t\t\tconsole.warn(`${source}:`, ...messages)\n\t\t\t\t\tbreak\n\t\t\t\tcase FLIGHTDECK_ERROR:\n\t\t\t\t\tconsole.error(`${source}:`, ...messages)\n\t\t\t\t\tbreak\n\t\t\t}\n\t\t}\n\t}\n\tpublic info(...messages: unknown[]): void {\n\t\tthis.log(FLIGHTDECK_INFO, ...messages)\n\t}\n\n\tpublic warn(...messages: unknown[]): void {\n\t\tthis.log(FLIGHTDECK_WARN, ...messages)\n\t}\n\n\tpublic error(...messages: unknown[]): void {\n\t\tthis.log(FLIGHTDECK_ERROR, ...messages)\n\t}\n}\n",
    "import { createEnv } from \"@t3-oss/env-core\"\nimport { z } from \"zod\"\n\nexport const env = createEnv({\n\tserver: { FLIGHTDECK_SECRET: z.string().optional() },\n\tclientPrefix: `NEVER`,\n\tclient: {},\n\truntimeEnv: import.meta.env,\n\temptyStringAsUndefined: true,\n})\n",
    "export type AlertOptions = {\n\tsecret: string\n\tendpoint: string\n\tversion: string\n}\n\nexport async function alert({\n\tsecret,\n\tendpoint,\n\tversion,\n}: AlertOptions): Promise<Response> {\n\tconst response = await fetch(endpoint, {\n\t\tmethod: `POST`,\n\t\theaders: {\n\t\t\t\"Content-Type\": `text/plain;charset=UTF-8`,\n\t\t\tAuthorization: `Bearer ${secret}`,\n\t\t},\n\t\tbody: version,\n\t})\n\n\treturn response\n}\n\n/**\n * @see https://github.com/changesets/action/blob/main/src/run.ts\n */\nexport type ChangesetsPublishedPackage = {\n\tname: string\n\tversion: string\n}\n\n/**\n * @see https://github.com/changesets/action/blob/main/src/run.ts\n */\nexport type ChangesetsPublishResult =\n\t| {\n\t\t\tpublished: true\n\t\t\tpublishedPackages: ChangesetsPublishedPackage[]\n\t  }\n\t| { published: false }\n\nexport type PackageConfig<K extends string> = {\n\t[key in K]: { endpoint: string }\n}\nexport type SecretsConfig<K extends string> = {\n\t[key in K]: string\n}\n\nexport type ScrambleOptions<K extends string = string> = {\n\tpackageConfig: PackageConfig<K>\n\tsecretsConfig: SecretsConfig<K>\n\tpublishedPackages: ChangesetsPublishedPackage[]\n}\n\nexport type ScrambleResult<K extends string = string> = {\n\t[key in K]: Response\n}\n\nexport async function scramble<K extends string = string>({\n\tpackageConfig,\n\tsecretsConfig,\n\tpublishedPackages,\n}: ScrambleOptions<K>): Promise<ScrambleResult<K>> {\n\tconst alertResults: Promise<readonly [K, Response]>[] = []\n\tfor (const publishedPackage of publishedPackages) {\n\t\tif (publishedPackage.name in packageConfig) {\n\t\t\tconst name = publishedPackage.name as K\n\t\t\tconst { endpoint } = packageConfig[name]\n\t\t\tconst secret = secretsConfig[name]\n\t\t\tconst version = publishedPackage.version\n\t\t\tconst alertResultPromise = alert({ secret, endpoint, version }).then(\n\t\t\t\t(alertResult) => [name, alertResult] as const,\n\t\t\t)\n\t\t\talertResults.push(alertResultPromise)\n\t\t}\n\t}\n\tconst alertResultsResolved = await Promise.all(alertResults)\n\tconst scrambleResult = Object.fromEntries(\n\t\talertResultsResolved,\n\t) as ScrambleResult<K>\n\treturn scrambleResult\n}\n"
  ],
  "mappings": "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",
  "debugId": "16A02FFB0350BCFF64756E2164756E21",
  "names": []
}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "flightdeck",
3
- "version": "0.2.6",
3
+ "version": "0.2.8",
4
4
  "license": "MIT",
5
5
  "author": {
6
6
  "name": "Jeremy Banka",
@@ -438,8 +438,16 @@ export class FlightDeck<S extends string = string> {
438
438
  }
439
439
  }
440
440
 
441
+ export const FLIGHTDECK_INFO = `info`
442
+ export const FLIGHTDECK_WARN = `warn`
443
+ export const FLIGHTDECK_ERROR = `ERR!`
444
+
441
445
  export const flightDeckLogSchema = z.object({
442
- level: z.union([z.literal(`info`), z.literal(`warn`), z.literal(`ERR!`)]),
446
+ level: z.union([
447
+ z.literal(FLIGHTDECK_INFO),
448
+ z.literal(FLIGHTDECK_WARN),
449
+ z.literal(FLIGHTDECK_ERROR),
450
+ ]),
443
451
  timestamp: z.number(),
444
452
  package: z.string(),
445
453
  service: z.string().optional(),
@@ -483,18 +491,20 @@ export const FLIGHTDECK_LNAV_FORMAT = {
483
491
  description: `Format for events logged by the FlightDeck process manager.`,
484
492
  "file-type": `json`,
485
493
  "timestamp-field": `timestamp`,
486
- "subsecond-field": `subsecond`,
487
- "subsecond-units": `milli`,
494
+ "timestamp-divisor": 1000,
495
+ "module-field": `package`,
488
496
  "opid-field": `service`,
489
497
  "level-field": `level`,
490
498
  level: {
491
- info: `info`,
492
- warning: `warn`,
493
- error: `err!`,
499
+ info: FLIGHTDECK_INFO,
500
+ warning: FLIGHTDECK_WARN,
501
+ error: FLIGHTDECK_ERROR,
494
502
  },
495
- "ordered-by-time": true,
496
503
 
497
504
  [LINE_FORMAT]: [
505
+ {
506
+ field: `level`,
507
+ },
498
508
  {
499
509
  prefix: ` `,
500
510
  field: `__timestamp__`,
@@ -514,11 +524,6 @@ export const FLIGHTDECK_LNAV_FORMAT = {
514
524
  field: `service`,
515
525
  "default-value": ``,
516
526
  },
517
- {
518
- prefix: `[`,
519
- field: `__level__`,
520
- suffix: `]`,
521
- },
522
527
  {
523
528
  prefix: `: `,
524
529
  field: `body`,
@@ -567,20 +572,30 @@ export class FlightDeckLogger
567
572
  this.processCode = processCode
568
573
  this.jsonLogging = options?.jsonLogging ?? false
569
574
  }
570
- public info(...messages: unknown[]): void {
575
+ protected log(
576
+ level:
577
+ | typeof FLIGHTDECK_ERROR
578
+ | typeof FLIGHTDECK_INFO
579
+ | typeof FLIGHTDECK_WARN,
580
+ ...messages: unknown[]
581
+ ): void {
571
582
  if (this.jsonLogging) {
583
+ let body = messages
584
+ .map((message) =>
585
+ typeof message === `string`
586
+ ? message
587
+ : inspect(message, false, null, true),
588
+ )
589
+ .join(` `)
590
+ if (body.includes(`\n`)) {
591
+ body = `\n ${body.split(`\n`).join(`\n `)}`
592
+ }
572
593
  const log: FlightDeckLog = {
573
594
  timestamp: Date.now(),
574
- level: `info`,
595
+ level,
575
596
  process: this.processCode,
576
597
  package: this.packageName,
577
- body: messages
578
- .map((message) =>
579
- typeof message === `string`
580
- ? message
581
- : inspect(message, false, null, true),
582
- )
583
- .join(` `),
598
+ body,
584
599
  }
585
600
  if (this.serviceName) {
586
601
  log.service = this.serviceName
@@ -588,63 +603,30 @@ export class FlightDeckLogger
588
603
  process.stdout.write(JSON.stringify(log) + `\n`)
589
604
  } else {
590
605
  const source = this.serviceName
591
- ? `${this.packageName}::${this.serviceName}`
606
+ ? `${this.packageName}:${this.serviceName}`
592
607
  : this.packageName
593
- console.log(`${source}:`, ...messages)
608
+ switch (level) {
609
+ case FLIGHTDECK_INFO:
610
+ console.log(`${source}:`, ...messages)
611
+ break
612
+ case FLIGHTDECK_WARN:
613
+ console.warn(`${source}:`, ...messages)
614
+ break
615
+ case FLIGHTDECK_ERROR:
616
+ console.error(`${source}:`, ...messages)
617
+ break
618
+ }
594
619
  }
595
620
  }
621
+ public info(...messages: unknown[]): void {
622
+ this.log(FLIGHTDECK_INFO, ...messages)
623
+ }
596
624
 
597
625
  public warn(...messages: unknown[]): void {
598
- if (this.jsonLogging) {
599
- const log: FlightDeckLog = {
600
- timestamp: Date.now(),
601
- level: `warn`,
602
- process: this.processCode,
603
- package: this.packageName,
604
- body: messages
605
- .map((message) =>
606
- typeof message === `string`
607
- ? message
608
- : inspect(message, false, null, true),
609
- )
610
- .join(` `),
611
- }
612
- if (this.serviceName) {
613
- log.service = this.serviceName
614
- }
615
- process.stdout.write(JSON.stringify(log) + `\n`)
616
- } else {
617
- const source = this.serviceName
618
- ? `${this.packageName}::${this.serviceName}`
619
- : this.packageName
620
- console.warn(`${source}:`, ...messages)
621
- }
626
+ this.log(FLIGHTDECK_WARN, ...messages)
622
627
  }
623
628
 
624
629
  public error(...messages: unknown[]): void {
625
- if (this.jsonLogging) {
626
- const log: FlightDeckLog = {
627
- timestamp: Date.now() + Math.floor(Math.random() * 1000),
628
- level: `ERR!`,
629
- process: this.processCode,
630
- package: this.packageName,
631
- body: messages
632
- .map((message) =>
633
- typeof message === `string`
634
- ? message
635
- : inspect(message, false, null, true),
636
- )
637
- .join(` `),
638
- }
639
- if (this.serviceName) {
640
- log.service = this.serviceName
641
- }
642
- process.stdout.write(JSON.stringify(log) + `\n`)
643
- } else {
644
- const source = this.serviceName
645
- ? `${this.packageName}::${this.serviceName}`
646
- : this.packageName
647
- console.error(`${source}:`, ...messages)
648
- }
630
+ this.log(FLIGHTDECK_ERROR, ...messages)
649
631
  }
650
632
  }