mirror of
https://github.com/Start9Labs/start-os.git
synced 2026-03-26 10:21:52 +00:00
* wip * initialize callbacks * wip * smtp * list_service_interfaces * wip * wip * fix domains * fix hostname handling in NetService * misc fixes * getInstalledPackages * misc fixes * publish v6 lib * refactor service effects * fix import * fix container runtime * fix tests * apply suggestions from review
302 lines
9.9 KiB
TypeScript
302 lines
9.9 KiB
TypeScript
import { types as T } from "@start9labs/start-sdk"
|
|
import * as net from "net"
|
|
import { object, string, number, literals, some, unknown } from "ts-matches"
|
|
import { Effects } from "../Models/Effects"
|
|
|
|
import { CallbackHolder } from "../Models/CallbackHolder"
|
|
import { MainEffects } from "@start9labs/start-sdk/cjs/lib/StartSdk"
|
|
const matchRpcError = object({
|
|
error: object(
|
|
{
|
|
code: number,
|
|
message: string,
|
|
data: some(
|
|
string,
|
|
object(
|
|
{
|
|
details: string,
|
|
debug: string,
|
|
},
|
|
["debug"],
|
|
),
|
|
),
|
|
},
|
|
["data"],
|
|
),
|
|
})
|
|
const testRpcError = matchRpcError.test
|
|
const testRpcResult = object({
|
|
result: unknown,
|
|
}).test
|
|
type RpcError = typeof matchRpcError._TYPE
|
|
|
|
const SOCKET_PATH = "/media/startos/rpc/host.sock"
|
|
let hostSystemId = 0
|
|
|
|
export type EffectContext = {
|
|
procedureId: string | null
|
|
callbacks: CallbackHolder | null
|
|
}
|
|
|
|
const rpcRoundFor =
|
|
(procedureId: string | null) =>
|
|
<K extends keyof Effects | "getStore" | "setStore" | "clearCallbacks">(
|
|
method: K,
|
|
params: Record<string, unknown>,
|
|
) => {
|
|
const id = hostSystemId++
|
|
const client = net.createConnection({ path: SOCKET_PATH }, () => {
|
|
client.write(
|
|
JSON.stringify({
|
|
id,
|
|
method,
|
|
params: { ...params, procedureId },
|
|
}) + "\n",
|
|
)
|
|
})
|
|
let bufs: Buffer[] = []
|
|
return new Promise((resolve, reject) => {
|
|
client.on("data", (data) => {
|
|
try {
|
|
bufs.push(data)
|
|
if (data.reduce((acc, x) => acc || x == 10, false)) {
|
|
const res: unknown = JSON.parse(
|
|
Buffer.concat(bufs).toString().split("\n")[0],
|
|
)
|
|
if (testRpcError(res)) {
|
|
let message = res.error.message
|
|
console.error("Error in host RPC:", { method, params })
|
|
if (string.test(res.error.data)) {
|
|
message += ": " + res.error.data
|
|
console.error(`Details: ${res.error.data}`)
|
|
} else {
|
|
if (res.error.data?.details) {
|
|
message += ": " + res.error.data.details
|
|
console.error(`Details: ${res.error.data.details}`)
|
|
}
|
|
if (res.error.data?.debug) {
|
|
message += "\n" + res.error.data.debug
|
|
console.error(`Debug: ${res.error.data.debug}`)
|
|
}
|
|
}
|
|
reject(new Error(`${message}@${method}`))
|
|
} else if (testRpcResult(res)) {
|
|
resolve(res.result)
|
|
} else {
|
|
reject(new Error(`malformed response ${JSON.stringify(res)}`))
|
|
}
|
|
}
|
|
} catch (error) {
|
|
reject(error)
|
|
}
|
|
client.end()
|
|
})
|
|
client.on("error", (error) => {
|
|
reject(error)
|
|
})
|
|
})
|
|
}
|
|
|
|
function makeEffects(context: EffectContext): Effects {
|
|
const rpcRound = rpcRoundFor(context.procedureId)
|
|
const self: Effects = {
|
|
bind(...[options]: Parameters<T.Effects["bind"]>) {
|
|
return rpcRound("bind", {
|
|
...options,
|
|
stack: new Error().stack,
|
|
}) as ReturnType<T.Effects["bind"]>
|
|
},
|
|
clearBindings(...[]: Parameters<T.Effects["clearBindings"]>) {
|
|
return rpcRound("clearBindings", {}) as ReturnType<
|
|
T.Effects["clearBindings"]
|
|
>
|
|
},
|
|
clearServiceInterfaces(
|
|
...[]: Parameters<T.Effects["clearServiceInterfaces"]>
|
|
) {
|
|
return rpcRound("clearServiceInterfaces", {}) as ReturnType<
|
|
T.Effects["clearServiceInterfaces"]
|
|
>
|
|
},
|
|
getInstalledPackages(...[]: Parameters<T.Effects["getInstalledPackages"]>) {
|
|
return rpcRound("getInstalledPackages", {}) as ReturnType<
|
|
T.Effects["getInstalledPackages"]
|
|
>
|
|
},
|
|
createOverlayedImage(options: {
|
|
imageId: string
|
|
}): Promise<[string, string]> {
|
|
return rpcRound("createOverlayedImage", options) as ReturnType<
|
|
T.Effects["createOverlayedImage"]
|
|
>
|
|
},
|
|
destroyOverlayedImage(options: { guid: string }): Promise<void> {
|
|
return rpcRound("destroyOverlayedImage", options) as ReturnType<
|
|
T.Effects["destroyOverlayedImage"]
|
|
>
|
|
},
|
|
executeAction(...[options]: Parameters<T.Effects["executeAction"]>) {
|
|
return rpcRound("executeAction", options) as ReturnType<
|
|
T.Effects["executeAction"]
|
|
>
|
|
},
|
|
exportAction(...[options]: Parameters<T.Effects["exportAction"]>) {
|
|
return rpcRound("exportAction", options) as ReturnType<
|
|
T.Effects["exportAction"]
|
|
>
|
|
},
|
|
exportServiceInterface: ((
|
|
...[options]: Parameters<Effects["exportServiceInterface"]>
|
|
) => {
|
|
return rpcRound("exportServiceInterface", options) as ReturnType<
|
|
T.Effects["exportServiceInterface"]
|
|
>
|
|
}) as Effects["exportServiceInterface"],
|
|
exposeForDependents(
|
|
...[options]: Parameters<T.Effects["exposeForDependents"]>
|
|
) {
|
|
return rpcRound("exposeForDependents", options) as ReturnType<
|
|
T.Effects["exposeForDependents"]
|
|
>
|
|
},
|
|
getConfigured(...[]: Parameters<T.Effects["getConfigured"]>) {
|
|
return rpcRound("getConfigured", {}) as ReturnType<
|
|
T.Effects["getConfigured"]
|
|
>
|
|
},
|
|
getContainerIp(...[]: Parameters<T.Effects["getContainerIp"]>) {
|
|
return rpcRound("getContainerIp", {}) as ReturnType<
|
|
T.Effects["getContainerIp"]
|
|
>
|
|
},
|
|
getHostInfo: ((...[allOptions]: Parameters<T.Effects["getHostInfo"]>) => {
|
|
const options = {
|
|
...allOptions,
|
|
callback: context.callbacks?.addCallback(allOptions.callback) || null,
|
|
}
|
|
return rpcRound("getHostInfo", options) as ReturnType<
|
|
T.Effects["getHostInfo"]
|
|
> as any
|
|
}) as Effects["getHostInfo"],
|
|
getServiceInterface(
|
|
...[options]: Parameters<T.Effects["getServiceInterface"]>
|
|
) {
|
|
return rpcRound("getServiceInterface", {
|
|
...options,
|
|
callback: context.callbacks?.addCallback(options.callback) || null,
|
|
}) as ReturnType<T.Effects["getServiceInterface"]>
|
|
},
|
|
|
|
getPrimaryUrl(...[options]: Parameters<T.Effects["getPrimaryUrl"]>) {
|
|
return rpcRound("getPrimaryUrl", {
|
|
...options,
|
|
callback: context.callbacks?.addCallback(options.callback) || null,
|
|
}) as ReturnType<T.Effects["getPrimaryUrl"]>
|
|
},
|
|
getServicePortForward(
|
|
...[options]: Parameters<T.Effects["getServicePortForward"]>
|
|
) {
|
|
return rpcRound("getServicePortForward", options) as ReturnType<
|
|
T.Effects["getServicePortForward"]
|
|
>
|
|
},
|
|
getSslCertificate(options: Parameters<T.Effects["getSslCertificate"]>[0]) {
|
|
return rpcRound("getSslCertificate", options) as ReturnType<
|
|
T.Effects["getSslCertificate"]
|
|
>
|
|
},
|
|
getSslKey(options: Parameters<T.Effects["getSslKey"]>[0]) {
|
|
return rpcRound("getSslKey", options) as ReturnType<
|
|
T.Effects["getSslKey"]
|
|
>
|
|
},
|
|
getSystemSmtp(...[options]: Parameters<T.Effects["getSystemSmtp"]>) {
|
|
return rpcRound("getSystemSmtp", {
|
|
...options,
|
|
callback: context.callbacks?.addCallback(options.callback) || null,
|
|
}) as ReturnType<T.Effects["getSystemSmtp"]>
|
|
},
|
|
listServiceInterfaces(
|
|
...[options]: Parameters<T.Effects["listServiceInterfaces"]>
|
|
) {
|
|
return rpcRound("listServiceInterfaces", {
|
|
...options,
|
|
callback: context.callbacks?.addCallback(options.callback) || null,
|
|
}) as ReturnType<T.Effects["listServiceInterfaces"]>
|
|
},
|
|
mount(...[options]: Parameters<T.Effects["mount"]>) {
|
|
return rpcRound("mount", options) as ReturnType<T.Effects["mount"]>
|
|
},
|
|
clearActions(...[]: Parameters<T.Effects["clearActions"]>) {
|
|
return rpcRound("clearActions", {}) as ReturnType<
|
|
T.Effects["clearActions"]
|
|
>
|
|
},
|
|
restart(...[]: Parameters<T.Effects["restart"]>) {
|
|
return rpcRound("restart", {}) as ReturnType<T.Effects["restart"]>
|
|
},
|
|
setConfigured(...[configured]: Parameters<T.Effects["setConfigured"]>) {
|
|
return rpcRound("setConfigured", { configured }) as ReturnType<
|
|
T.Effects["setConfigured"]
|
|
>
|
|
},
|
|
setDependencies(
|
|
dependencies: Parameters<T.Effects["setDependencies"]>[0],
|
|
): ReturnType<T.Effects["setDependencies"]> {
|
|
return rpcRound("setDependencies", dependencies) as ReturnType<
|
|
T.Effects["setDependencies"]
|
|
>
|
|
},
|
|
checkDependencies(
|
|
options: Parameters<T.Effects["checkDependencies"]>[0],
|
|
): ReturnType<T.Effects["checkDependencies"]> {
|
|
return rpcRound("checkDependencies", options) as ReturnType<
|
|
T.Effects["checkDependencies"]
|
|
>
|
|
},
|
|
getDependencies(): ReturnType<T.Effects["getDependencies"]> {
|
|
return rpcRound("getDependencies", {}) as ReturnType<
|
|
T.Effects["getDependencies"]
|
|
>
|
|
},
|
|
setHealth(...[options]: Parameters<T.Effects["setHealth"]>) {
|
|
return rpcRound("setHealth", options) as ReturnType<
|
|
T.Effects["setHealth"]
|
|
>
|
|
},
|
|
|
|
setMainStatus(o: { status: "running" | "stopped" }): Promise<void> {
|
|
return rpcRound("setMainStatus", o) as ReturnType<T.Effects["setHealth"]>
|
|
},
|
|
|
|
shutdown(...[]: Parameters<T.Effects["shutdown"]>) {
|
|
return rpcRound("shutdown", {}) as ReturnType<T.Effects["shutdown"]>
|
|
},
|
|
store: {
|
|
get: async (options: any) =>
|
|
rpcRound("getStore", {
|
|
...options,
|
|
callback: context.callbacks?.addCallback(options.callback) || null,
|
|
}) as any,
|
|
set: async (options: any) =>
|
|
rpcRound("setStore", options) as ReturnType<T.Effects["store"]["set"]>,
|
|
} as T.Effects["store"],
|
|
}
|
|
return self
|
|
}
|
|
|
|
export function makeProcedureEffects(procedureId: string): Effects {
|
|
return makeEffects({ procedureId, callbacks: null })
|
|
}
|
|
|
|
export function makeMainEffects(): MainEffects {
|
|
const rpcRound = rpcRoundFor(null)
|
|
return {
|
|
_type: "main",
|
|
clearCallbacks: () => {
|
|
return rpcRound("clearCallbacks", {}) as Promise<void>
|
|
},
|
|
...makeEffects({ procedureId: null, callbacks: new CallbackHolder() }),
|
|
}
|
|
}
|