mirror of
https://github.com/Start9Labs/start-os.git
synced 2026-03-26 02:11:53 +00:00
* add support for idmapped mounts to start-sdk * misc fixes * misc fixes * add default to textarea * fix iptables masquerade rule * fix textarea types * more fixes * better logging for rsync * fix tty size * fix wg conf generation for android * disable file mounts on dependencies * mostly there, some styling issues (#3069) * mostly there, some styling issues * fix: address comments (#3070) * fix: address comments * fix: fix * show SSL for any address with secure protocol and ssl added * better sorting and messaging --------- Co-authored-by: Alex Inkin <alexander@inkin.ru> * fixes for nextcloud * allow sidebar navigation during service state traansitions * wip: x-forwarded headers * implement x-forwarded-for proxy * lowercase domain names and fix warning popover bug * fix http2 websockets * fix websocket retry behavior * add arch filters to s9pk pack * use docker for start-cli install * add version range to package signer on registry * fix rcs < 0 * fix user information parsing * refactor service interface getters * disable idmaps * build fixes * update docker login action * streamline build * add start-cli workflow * rename * riscv64gc * fix ui packing * no default features on cli * make cli depend on GIT_HASH * more build fixes * more build fixes * interpolate arch within dockerfile * fix tests * add launch ui to service page plus other small improvements (#3075) * add launch ui to service page plus other small improvements * revert translation disable * add spinner to service list if service is health and loading * chore: some visual tune up * chore: update Taiga UI --------- Co-authored-by: waterplea <alexander@inkin.ru> * fix backups * feat: use arm hosted runners and don't fail when apt package does not exist (#3076) --------- Co-authored-by: Matt Hill <mattnine@protonmail.com> Co-authored-by: Shadowy Super Coder <musashidisciple@proton.me> Co-authored-by: Matt Hill <MattDHill@users.noreply.github.com> Co-authored-by: Alex Inkin <alexander@inkin.ru> Co-authored-by: Remco Ros <remcoros@live.nl>
137 lines
4.1 KiB
TypeScript
137 lines
4.1 KiB
TypeScript
import * as T from "../../../base/lib/types"
|
|
import { asError } from "../../../base/lib/util/asError"
|
|
import { Drop } from "../util"
|
|
import {
|
|
SubContainer,
|
|
SubContainerOwned,
|
|
SubContainerRc,
|
|
} from "../util/SubContainer"
|
|
import { CommandController } from "./CommandController"
|
|
import { DaemonCommandType } from "./Daemons"
|
|
import { Oneshot } from "./Oneshot"
|
|
|
|
const TIMEOUT_INCREMENT_MS = 1000
|
|
const MAX_TIMEOUT_MS = 30000
|
|
/**
|
|
* This is a wrapper around CommandController that has a state of off, where the command shouldn't be running
|
|
* and the others state of running, where it will keep a living running command
|
|
*/
|
|
|
|
export class Daemon<
|
|
Manifest extends T.SDKManifest,
|
|
C extends SubContainer<Manifest> | null = SubContainer<Manifest> | null,
|
|
> extends Drop {
|
|
private commandController: CommandController<Manifest, C> | null = null
|
|
private shouldBeRunning = false
|
|
protected exitedSuccess = false
|
|
private onExitFns: ((success: boolean) => void)[] = []
|
|
protected constructor(
|
|
private subcontainer: C,
|
|
private startCommand: () => Promise<CommandController<Manifest, C>>,
|
|
readonly oneshot: boolean = false,
|
|
) {
|
|
super()
|
|
}
|
|
isOneshot(): this is Oneshot<Manifest> {
|
|
return this.oneshot
|
|
}
|
|
static of<Manifest extends T.SDKManifest>() {
|
|
return async <C extends SubContainer<Manifest> | null>(
|
|
effects: T.Effects,
|
|
subcontainer: C,
|
|
exec: DaemonCommandType<Manifest, C>,
|
|
) => {
|
|
let subc: SubContainer<Manifest> | null = subcontainer
|
|
if (subcontainer && subcontainer.isOwned()) subc = subcontainer.rc()
|
|
const startCommand = () =>
|
|
CommandController.of<Manifest, C>()(
|
|
effects,
|
|
(subc?.rc() ?? null) as C,
|
|
exec,
|
|
)
|
|
const res = new Daemon(subc, startCommand)
|
|
effects.onLeaveContext(() => {
|
|
res
|
|
.term({ destroySubcontainer: true })
|
|
.catch((e) => console.error(asError(e)))
|
|
})
|
|
return res
|
|
}
|
|
}
|
|
async start() {
|
|
if (this.commandController) {
|
|
return
|
|
}
|
|
this.shouldBeRunning = true
|
|
let timeoutCounter = 0
|
|
;(async () => {
|
|
while (this.shouldBeRunning) {
|
|
if (this.commandController)
|
|
await this.commandController
|
|
.term({})
|
|
.catch((err) => console.error(err))
|
|
try {
|
|
this.commandController = await this.startCommand()
|
|
if (!this.shouldBeRunning) {
|
|
// handles race condition if stopped while starting
|
|
await this.term()
|
|
break
|
|
}
|
|
const success = await this.commandController.wait().then(
|
|
(_) => true,
|
|
(err) => {
|
|
console.error(err)
|
|
return false
|
|
},
|
|
)
|
|
for (const fn of this.onExitFns) {
|
|
try {
|
|
fn(success)
|
|
} catch (e) {
|
|
console.error("EXIT handler", e)
|
|
}
|
|
}
|
|
if (success && this.oneshot) {
|
|
this.exitedSuccess = true
|
|
break
|
|
}
|
|
} catch (e) {
|
|
console.error(e)
|
|
}
|
|
await new Promise((resolve) => setTimeout(resolve, timeoutCounter))
|
|
timeoutCounter += TIMEOUT_INCREMENT_MS
|
|
timeoutCounter = Math.min(MAX_TIMEOUT_MS, timeoutCounter)
|
|
}
|
|
})().catch((err) => {
|
|
console.error(asError(err))
|
|
})
|
|
}
|
|
async term(termOptions?: {
|
|
signal?: NodeJS.Signals | undefined
|
|
timeout?: number | undefined
|
|
destroySubcontainer?: boolean
|
|
}) {
|
|
this.shouldBeRunning = false
|
|
this.exitedSuccess = false
|
|
if (this.commandController) {
|
|
await this.commandController
|
|
.term({ ...termOptions })
|
|
.catch((e) => console.error(asError(e)))
|
|
this.commandController = null
|
|
this.onExitFns = []
|
|
if (termOptions?.destroySubcontainer) {
|
|
await this.subcontainer?.destroy()
|
|
}
|
|
}
|
|
}
|
|
subcontainerRc(): SubContainerRc<Manifest> | null {
|
|
return this.subcontainer?.rc() ?? null
|
|
}
|
|
onExit(fn: (success: boolean) => void) {
|
|
this.onExitFns.push(fn)
|
|
}
|
|
onDrop(): void {
|
|
this.term().catch((e) => console.error(asError(e)))
|
|
}
|
|
}
|