@skaile/workspaces 0.21.0 → 0.22.0-beta.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +40 -0
- package/dist/{asset-feeds-CI76R7FI.js → asset-feeds-PJDJ3QYI.js} +11 -11
- package/dist/{asset-feeds-CI76R7FI.js.map → asset-feeds-PJDJ3QYI.js.map} +1 -1
- package/dist/asset-manager/index.js +9 -9
- package/dist/asset-manager/installer.js +8 -8
- package/dist/base-assets/connectors/deploy.js +10 -9
- package/dist/base-assets/connectors/devserver.js +10 -9
- package/dist/base-assets/connectors/flow/adapter.js +10 -9
- package/dist/base-assets/connectors/flow/run-flow.js +11 -10
- package/dist/base-assets/connectors/flow.js +10 -9
- package/dist/base-assets/connectors/git.js +10 -9
- package/dist/base-assets/connectors/gmail.js +10 -9
- package/dist/base-assets/connectors/googledrive.js +10 -9
- package/dist/base-assets/connectors/local.js +10 -9
- package/dist/base-assets/connectors/mattermost.js +10 -9
- package/dist/base-assets/connectors/memory.js +10 -9
- package/dist/base-assets/connectors/minio.js +10 -9
- package/dist/base-assets/connectors/postgres.js +10 -9
- package/dist/base-assets/connectors/s3.js +10 -9
- package/dist/base-assets/connectors/sharepoint.js +10 -9
- package/dist/base-assets/connectors/sqlite.js +10 -9
- package/dist/base-assets/connectors/static-server.js +10 -9
- package/dist/base-assets/connectors/tunnel.js +10 -9
- package/dist/base-assets/connectors/webdav.js +10 -9
- package/dist/base-assets/connectors/xstate-store.js +10 -9
- package/dist/base-assets/connectors/xstate.js +10 -9
- package/dist/bridge/drivers/claude-sdk.js +13 -3
- package/dist/bridge/drivers/claude-sdk.js.map +1 -1
- package/dist/bridge/drivers/codex.js +13 -3
- package/dist/bridge/drivers/codex.js.map +1 -1
- package/dist/bridge/drivers/echo.js +13 -4
- package/dist/bridge/drivers/echo.js.map +1 -1
- package/dist/bridge/drivers/omp.js +13 -3
- package/dist/bridge/drivers/omp.js.map +1 -1
- package/dist/bridge/index.js +3 -2
- package/dist/bridge/src/drivers/claude-sdk.d.ts +7 -0
- package/dist/bridge/src/drivers/claude-sdk.d.ts.map +1 -1
- package/dist/bridge/src/drivers/codex.d.ts +7 -0
- package/dist/bridge/src/drivers/codex.d.ts.map +1 -1
- package/dist/bridge/src/drivers/echo.d.ts +6 -0
- package/dist/bridge/src/drivers/echo.d.ts.map +1 -1
- package/dist/bridge/src/drivers/omp.d.ts +6 -0
- package/dist/bridge/src/drivers/omp.d.ts.map +1 -1
- package/dist/bridge/src/registry.d.ts +32 -34
- package/dist/bridge/src/registry.d.ts.map +1 -1
- package/dist/{chunk-DEQ3OOTU.js → chunk-2DNSSQ22.js} +7 -7
- package/dist/{chunk-DEQ3OOTU.js.map → chunk-2DNSSQ22.js.map} +1 -1
- package/dist/{chunk-ZWIG55ZX.js → chunk-2XY6732A.js} +3 -3
- package/dist/{chunk-ZWIG55ZX.js.map → chunk-2XY6732A.js.map} +1 -1
- package/dist/{chunk-2WVQMRIE.js → chunk-CSDQBWE6.js} +5 -5
- package/dist/{chunk-2WVQMRIE.js.map → chunk-CSDQBWE6.js.map} +1 -1
- package/dist/{chunk-KFDTS7RX.js → chunk-F3MGZ5E6.js} +3 -3
- package/dist/{chunk-KFDTS7RX.js.map → chunk-F3MGZ5E6.js.map} +1 -1
- package/dist/{chunk-XAVM2BAJ.js → chunk-G6GKWGOW.js} +114 -619
- package/dist/chunk-G6GKWGOW.js.map +1 -0
- package/dist/{chunk-H45ANMIU.js → chunk-GKM6MDUC.js} +3 -3
- package/dist/{chunk-H45ANMIU.js.map → chunk-GKM6MDUC.js.map} +1 -1
- package/dist/{chunk-4ACWI5YT.js → chunk-IGQEXBBG.js} +44 -36
- package/dist/chunk-IGQEXBBG.js.map +1 -0
- package/dist/{chunk-RDH4SSMH.js → chunk-J2FCO6TM.js} +2 -2
- package/dist/{chunk-RDH4SSMH.js.map → chunk-J2FCO6TM.js.map} +1 -1
- package/dist/{chunk-BSY56QS7.js → chunk-KA46DUM4.js} +3 -3
- package/dist/{chunk-BSY56QS7.js.map → chunk-KA46DUM4.js.map} +1 -1
- package/dist/{chunk-XGWGLIHZ.js → chunk-MO4JPTRD.js} +4 -4
- package/dist/{chunk-XGWGLIHZ.js.map → chunk-MO4JPTRD.js.map} +1 -1
- package/dist/{chunk-G4BR355S.js → chunk-NGC7ZQI4.js} +10 -10
- package/dist/{chunk-G4BR355S.js.map → chunk-NGC7ZQI4.js.map} +1 -1
- package/dist/{chunk-W5DFC35Z.js → chunk-RENHNO4J.js} +81 -4
- package/dist/chunk-RENHNO4J.js.map +1 -0
- package/dist/{chunk-4S4TZDCD.js → chunk-SL6JVGRD.js} +3 -3
- package/dist/{chunk-4S4TZDCD.js.map → chunk-SL6JVGRD.js.map} +1 -1
- package/dist/{chunk-5QNQLSBW.js → chunk-TKOLD2O7.js} +22 -5
- package/dist/chunk-TKOLD2O7.js.map +1 -0
- package/dist/{chunk-DFUXWNTS.js → chunk-TTY56FQQ.js} +4 -4
- package/dist/{chunk-DFUXWNTS.js.map → chunk-TTY56FQQ.js.map} +1 -1
- package/dist/chunk-UZRY5UI2.js +96 -0
- package/dist/chunk-UZRY5UI2.js.map +1 -0
- package/dist/{chunk-NCUTHLRV.js → chunk-UZVHJ7LX.js} +4 -4
- package/dist/{chunk-NCUTHLRV.js.map → chunk-UZVHJ7LX.js.map} +1 -1
- package/dist/{chunk-FRPKLIEZ.js → chunk-WIR34WMU.js} +3 -3
- package/dist/{chunk-FRPKLIEZ.js.map → chunk-WIR34WMU.js.map} +1 -1
- package/dist/{chunk-37JKX6D7.js → chunk-X5Y4EGZB.js} +2 -2
- package/dist/{chunk-37JKX6D7.js.map → chunk-X5Y4EGZB.js.map} +1 -1
- package/dist/{chunk-S2OVTCAL.js → chunk-XHFMUGDD.js} +3 -3
- package/dist/{chunk-S2OVTCAL.js.map → chunk-XHFMUGDD.js.map} +1 -1
- package/dist/cli/index.js +701 -413
- package/dist/cli/index.js.map +1 -1
- package/dist/cli/src/commands/deploy.d.ts +24 -0
- package/dist/cli/src/commands/deploy.d.ts.map +1 -0
- package/dist/cli/src/commands/integration.d.ts +19 -0
- package/dist/cli/src/commands/integration.d.ts.map +1 -0
- package/dist/cli/src/commands/plugin-registry-cmd.d.ts +22 -0
- package/dist/cli/src/commands/plugin-registry-cmd.d.ts.map +1 -0
- package/dist/cli/src/commands/serve.d.ts.map +1 -1
- package/dist/cli/src/plugin-store/index.d.ts +37 -0
- package/dist/cli/src/plugin-store/index.d.ts.map +1 -0
- package/dist/cli/src/plugin-store/load.d.ts +35 -0
- package/dist/cli/src/plugin-store/load.d.ts.map +1 -0
- package/dist/cli/src/plugin-store/paths.d.ts +23 -0
- package/dist/cli/src/plugin-store/paths.d.ts.map +1 -0
- package/dist/cli/src/plugin-store/reconcile.d.ts +50 -0
- package/dist/cli/src/plugin-store/reconcile.d.ts.map +1 -0
- package/dist/cli/src/plugin-store/spec.d.ts +25 -0
- package/dist/cli/src/plugin-store/spec.d.ts.map +1 -0
- package/dist/connectors/config.js +8 -8
- package/dist/connectors/index.js +10 -9
- package/dist/connectors/src/connector-manager.d.ts.map +1 -1
- package/dist/connectors/src/connector-registry.d.ts +42 -22
- package/dist/connectors/src/connector-registry.d.ts.map +1 -1
- package/dist/connectors/src/index.d.ts +2 -3
- package/dist/connectors/src/index.d.ts.map +1 -1
- package/dist/connectors-shared/index.js +8 -0
- package/dist/connectors-shared/index.js.map +1 -0
- package/dist/connectors-shared/src/index.d.ts +11 -0
- package/dist/connectors-shared/src/index.d.ts.map +1 -0
- package/dist/connectors-shared/src/schemas.d.ts +10 -0
- package/dist/connectors-shared/src/schemas.d.ts.map +1 -0
- package/dist/connectors-shared/src/types.d.ts +11 -0
- package/dist/connectors-shared/src/types.d.ts.map +1 -0
- package/dist/core/index.js +7 -7
- package/dist/core/manifest.js +2 -2
- package/dist/core/models.js +1 -1
- package/dist/core/runtime-assets.js +4 -4
- package/dist/core/src/index.d.ts +2 -2
- package/dist/core/src/index.d.ts.map +1 -1
- package/dist/core/src/lock.d.ts +20 -1
- package/dist/core/src/lock.d.ts.map +1 -1
- package/dist/core/src/models.d.ts +13 -0
- package/dist/core/src/models.d.ts.map +1 -1
- package/dist/core/src/workspace-config.d.ts +14 -0
- package/dist/core/src/workspace-config.d.ts.map +1 -1
- package/dist/core/src/workspace-yaml-editor.d.ts +20 -0
- package/dist/core/src/workspace-yaml-editor.d.ts.map +1 -1
- package/dist/core/workspace-config.js +3 -3
- package/dist/deploy/index.js +454 -0
- package/dist/deploy/index.js.map +1 -0
- package/dist/deploy/src/handle-store.d.ts +22 -0
- package/dist/deploy/src/handle-store.d.ts.map +1 -0
- package/dist/deploy/src/index.d.ts +21 -0
- package/dist/deploy/src/index.d.ts.map +1 -0
- package/dist/deploy/src/targets/container-runtime.d.ts +38 -0
- package/dist/deploy/src/targets/container-runtime.d.ts.map +1 -0
- package/dist/deploy/src/targets/docker.d.ts +18 -0
- package/dist/deploy/src/targets/docker.d.ts.map +1 -0
- package/dist/deploy/src/targets/local.d.ts +30 -0
- package/dist/deploy/src/targets/local.d.ts.map +1 -0
- package/dist/deploy/src/targets/podman.d.ts +18 -0
- package/dist/deploy/src/targets/podman.d.ts.map +1 -0
- package/dist/deploy/src/targets/port.d.ts +10 -0
- package/dist/deploy/src/targets/port.d.ts.map +1 -0
- package/dist/deploy/src/targets/stream-lines.d.ts +44 -0
- package/dist/deploy/src/targets/stream-lines.d.ts.map +1 -0
- package/dist/discovery/index.js +3 -3
- package/dist/{ensure-sources-IDVQ77NJ.js → ensure-sources-COGVKY44.js} +11 -11
- package/dist/{ensure-sources-IDVQ77NJ.js.map → ensure-sources-COGVKY44.js.map} +1 -1
- package/dist/{flows-6BNO4GKK.js → flows-DYFTPCPM.js} +4 -4
- package/dist/{flows-6BNO4GKK.js.map → flows-DYFTPCPM.js.map} +1 -1
- package/dist/library/index.js +4 -4
- package/dist/open-library-DWAQFUSQ.js +13 -0
- package/dist/{open-library-IOYWFK7M.js.map → open-library-DWAQFUSQ.js.map} +1 -1
- package/dist/plugin-registry/index.js +4 -0
- package/dist/plugin-registry/index.js.map +1 -0
- package/dist/plugin-registry/src/context.d.ts +29 -0
- package/dist/plugin-registry/src/context.d.ts.map +1 -0
- package/dist/plugin-registry/src/deploy-handle.d.ts +60 -0
- package/dist/plugin-registry/src/deploy-handle.d.ts.map +1 -0
- package/dist/plugin-registry/src/errors.d.ts +23 -0
- package/dist/plugin-registry/src/errors.d.ts.map +1 -0
- package/dist/plugin-registry/src/index.d.ts +16 -0
- package/dist/plugin-registry/src/index.d.ts.map +1 -0
- package/dist/plugin-registry/src/internal.d.ts +13 -0
- package/dist/plugin-registry/src/internal.d.ts.map +1 -0
- package/dist/plugin-registry/src/registry.d.ts +25 -0
- package/dist/plugin-registry/src/registry.d.ts.map +1 -0
- package/dist/plugin-registry/src/targets.d.ts +42 -0
- package/dist/plugin-registry/src/targets.d.ts.map +1 -0
- package/dist/plugin-store-6OENKNFW.js +144 -0
- package/dist/plugin-store-6OENKNFW.js.map +1 -0
- package/dist/runner/index.js +14 -13
- package/dist/sdk/asset-manager.js +9 -9
- package/dist/sdk/bridge.js +3 -2
- package/dist/sdk/core.js +7 -7
- package/dist/sdk/index.js +15 -14
- package/dist/sdk/index.js.map +1 -1
- package/dist/sdk/runner.js +14 -13
- package/dist/{setup-AIOLUTKV.js → setup-ACMP3QZC.js} +12 -11
- package/dist/setup-ACMP3QZC.js.map +1 -0
- package/dist/store-client-ZSLNOOQG.js +14 -0
- package/dist/{store-client-CYEH2GKC.js.map → store-client-ZSLNOOQG.js.map} +1 -1
- package/dist/tui/index.js +14 -13
- package/dist/tui/index.js.map +1 -1
- package/dist/workspace-plugin/adapters/mcp.js +2 -2
- package/dist/workspace-plugin/adapters/omp.js +3 -3
- package/dist/workspace-plugin/index.js +1 -1
- package/package.json +20 -23
- package/dist/base-assets/connectors/redis/adapter.d.ts +0 -39
- package/dist/base-assets/connectors/redis/adapter.d.ts.map +0 -1
- package/dist/base-assets/connectors/redis.js +0 -20
- package/dist/base-assets/connectors/redis.js.map +0 -1
- package/dist/base-assets/connectors/yjs/adapter.d.ts +0 -50
- package/dist/base-assets/connectors/yjs/adapter.d.ts.map +0 -1
- package/dist/base-assets/connectors/yjs.js +0 -20
- package/dist/base-assets/connectors/yjs.js.map +0 -1
- package/dist/chunk-4ACWI5YT.js.map +0 -1
- package/dist/chunk-5QNQLSBW.js.map +0 -1
- package/dist/chunk-W5DFC35Z.js.map +0 -1
- package/dist/chunk-XAVM2BAJ.js.map +0 -1
- package/dist/cli/src/commands/plugin.d.ts +0 -14
- package/dist/cli/src/commands/plugin.d.ts.map +0 -1
- package/dist/open-library-IOYWFK7M.js +0 -13
- package/dist/setup-AIOLUTKV.js.map +0 -1
- package/dist/store-client-CYEH2GKC.js +0 -14
|
@@ -0,0 +1,454 @@
|
|
|
1
|
+
import { pluginRegistry } from '../chunk-UZRY5UI2.js';
|
|
2
|
+
import { portableSpawnSync, portableSpawn } from '../chunk-RENHNO4J.js';
|
|
3
|
+
import '../chunk-K5GBV4SA.js';
|
|
4
|
+
import '../chunk-KLNL7QHN.js';
|
|
5
|
+
import '../chunk-TTY56FQQ.js';
|
|
6
|
+
import '../chunk-TKOLD2O7.js';
|
|
7
|
+
import '../chunk-WIR34WMU.js';
|
|
8
|
+
import '../chunk-X5Y4EGZB.js';
|
|
9
|
+
import '../chunk-JKNWJ64A.js';
|
|
10
|
+
import '../chunk-O4JH3KUE.js';
|
|
11
|
+
import '../chunk-24UIWON4.js';
|
|
12
|
+
import '../chunk-NSBPE2FW.js';
|
|
13
|
+
import * as z from 'zod';
|
|
14
|
+
import { createServer } from 'net';
|
|
15
|
+
import { createRequire } from 'module';
|
|
16
|
+
import { join, dirname, resolve } from 'path';
|
|
17
|
+
import { mkdir, writeFile, readFile, rm } from 'fs/promises';
|
|
18
|
+
|
|
19
|
+
function allocateFreePort(host = "127.0.0.1") {
|
|
20
|
+
return new Promise((resolve, reject) => {
|
|
21
|
+
const server = createServer();
|
|
22
|
+
server.once("error", reject);
|
|
23
|
+
server.listen(0, host, () => {
|
|
24
|
+
const addr = server.address();
|
|
25
|
+
if (addr === null || typeof addr === "string") {
|
|
26
|
+
server.close();
|
|
27
|
+
reject(new Error("could not determine allocated port"));
|
|
28
|
+
return;
|
|
29
|
+
}
|
|
30
|
+
const { port } = addr;
|
|
31
|
+
server.close(() => resolve(port));
|
|
32
|
+
});
|
|
33
|
+
});
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
// deploy/src/targets/stream-lines.ts
|
|
37
|
+
var LineBuffer = class {
|
|
38
|
+
constructor(max = 1e3) {
|
|
39
|
+
this.max = max;
|
|
40
|
+
}
|
|
41
|
+
max;
|
|
42
|
+
lines = [];
|
|
43
|
+
waiters = /* @__PURE__ */ new Set();
|
|
44
|
+
push(line) {
|
|
45
|
+
const entry = { ts: Date.now(), line };
|
|
46
|
+
this.lines.push(entry);
|
|
47
|
+
if (this.lines.length > this.max) this.lines.shift();
|
|
48
|
+
for (const w of this.waiters) w(entry);
|
|
49
|
+
}
|
|
50
|
+
/** Snapshot of buffered lines at or after `sinceMs` (epoch millis). */
|
|
51
|
+
since(sinceMs = 0) {
|
|
52
|
+
return this.lines.filter((l) => l.ts >= sinceMs);
|
|
53
|
+
}
|
|
54
|
+
/** Subscribe to future pushes; returns an unsubscribe fn. */
|
|
55
|
+
subscribe(fn) {
|
|
56
|
+
this.waiters.add(fn);
|
|
57
|
+
return () => this.waiters.delete(fn);
|
|
58
|
+
}
|
|
59
|
+
};
|
|
60
|
+
async function pumpLines(stream, buffer) {
|
|
61
|
+
const reader = stream.getReader();
|
|
62
|
+
const decoder = new TextDecoder();
|
|
63
|
+
let pending = "";
|
|
64
|
+
try {
|
|
65
|
+
while (true) {
|
|
66
|
+
const { done, value } = await reader.read();
|
|
67
|
+
if (done) break;
|
|
68
|
+
pending += decoder.decode(value, { stream: true });
|
|
69
|
+
let idx = pending.indexOf("\n");
|
|
70
|
+
while (idx !== -1) {
|
|
71
|
+
buffer.push(pending.slice(0, idx));
|
|
72
|
+
pending = pending.slice(idx + 1);
|
|
73
|
+
idx = pending.indexOf("\n");
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
} catch {
|
|
77
|
+
} finally {
|
|
78
|
+
if (pending.length > 0) buffer.push(pending);
|
|
79
|
+
reader.releaseLock();
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
function waitForLine(buffer, pattern, opts) {
|
|
83
|
+
for (const { line } of buffer.since(0)) {
|
|
84
|
+
if (pattern.test(line)) return Promise.resolve();
|
|
85
|
+
}
|
|
86
|
+
return new Promise((resolve, reject) => {
|
|
87
|
+
let settled = false;
|
|
88
|
+
const finish = (fn) => {
|
|
89
|
+
if (settled) return;
|
|
90
|
+
settled = true;
|
|
91
|
+
clearTimeout(timer);
|
|
92
|
+
unsub();
|
|
93
|
+
opts.signal.removeEventListener("abort", onAbort);
|
|
94
|
+
fn();
|
|
95
|
+
};
|
|
96
|
+
const unsub = buffer.subscribe(({ line }) => {
|
|
97
|
+
if (pattern.test(line)) finish(resolve);
|
|
98
|
+
});
|
|
99
|
+
const onAbort = () => finish(() => reject(new Error("aborted while waiting for ready line")));
|
|
100
|
+
const timer = setTimeout(
|
|
101
|
+
() => finish(
|
|
102
|
+
() => reject(new Error(`timed out after ${opts.timeoutMs}ms waiting for ready line`))
|
|
103
|
+
),
|
|
104
|
+
opts.timeoutMs
|
|
105
|
+
);
|
|
106
|
+
if (opts.signal.aborted) onAbort();
|
|
107
|
+
else opts.signal.addEventListener("abort", onAbort);
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
// deploy/src/targets/container-runtime.ts
|
|
112
|
+
var LOOPBACK_BINDS = /* @__PURE__ */ new Set(["127.0.0.1", "localhost", "::1"]);
|
|
113
|
+
var DEFAULT_READY_PATTERN = "ws://";
|
|
114
|
+
var DEFAULT_READY_TIMEOUT_MS = 6e4;
|
|
115
|
+
var containerConfigSchema = z.object({
|
|
116
|
+
image: z.string(),
|
|
117
|
+
port: z.number().int().positive().optional(),
|
|
118
|
+
bind: z.string().default("127.0.0.1"),
|
|
119
|
+
mounts: z.array(z.string()).default([]),
|
|
120
|
+
env: z.record(z.string(), z.string()).default({}),
|
|
121
|
+
network: z.string().optional(),
|
|
122
|
+
pullPolicy: z.enum(["always", "if-missing", "never"]).default("if-missing"),
|
|
123
|
+
// v1 defers in-provider image builds; "local" means "run the given image".
|
|
124
|
+
buildStrategy: z.literal("local").default("local"),
|
|
125
|
+
containerPort: z.number().int().positive().optional(),
|
|
126
|
+
readyPattern: z.string().optional()
|
|
127
|
+
}).strict().refine((c) => LOOPBACK_BINDS.has(c.bind), {
|
|
128
|
+
message: "non-loopback bind requires tls config (not yet supported); use a loopback bind",
|
|
129
|
+
path: ["bind"]
|
|
130
|
+
});
|
|
131
|
+
function payloadSchemaFor(bin) {
|
|
132
|
+
return z.object({
|
|
133
|
+
containerId: z.string(),
|
|
134
|
+
hostPort: z.number(),
|
|
135
|
+
runtime: z.enum(["docker", "podman"]).refine((r) => r === bin, {
|
|
136
|
+
message: `runtime mismatch: handle belongs to ${bin}`
|
|
137
|
+
})
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
function pullFlag(policy) {
|
|
141
|
+
const mapped = policy === "if-missing" ? "missing" : policy;
|
|
142
|
+
return ["--pull", mapped];
|
|
143
|
+
}
|
|
144
|
+
function isRunning(bin, containerId) {
|
|
145
|
+
const res = portableSpawnSync([bin, "inspect", "-f", "{{.State.Running}}", containerId]);
|
|
146
|
+
return res.exitCode === 0 && res.stdout.trim() === "true";
|
|
147
|
+
}
|
|
148
|
+
function buildHandle(bin, payload, config, ctx, ready) {
|
|
149
|
+
let state = "starting";
|
|
150
|
+
let followProc = null;
|
|
151
|
+
return {
|
|
152
|
+
wsUrl: `ws://${config.bind}:${payload.hostPort}`,
|
|
153
|
+
get state() {
|
|
154
|
+
return state;
|
|
155
|
+
},
|
|
156
|
+
payload,
|
|
157
|
+
async waitReady(timeoutMs) {
|
|
158
|
+
try {
|
|
159
|
+
await ready();
|
|
160
|
+
state = "ready";
|
|
161
|
+
} catch (err) {
|
|
162
|
+
state = "errored";
|
|
163
|
+
throw err;
|
|
164
|
+
}
|
|
165
|
+
},
|
|
166
|
+
async health() {
|
|
167
|
+
const healthy = isRunning(bin, payload.containerId);
|
|
168
|
+
return healthy ? { healthy } : { healthy, detail: `container ${payload.containerId} not running` };
|
|
169
|
+
},
|
|
170
|
+
async *logs(opts) {
|
|
171
|
+
const buffer = new LineBuffer();
|
|
172
|
+
const args = ["logs"];
|
|
173
|
+
if (opts?.follow) args.push("--follow");
|
|
174
|
+
followProc = portableSpawn([bin, ...args, payload.containerId]);
|
|
175
|
+
void pumpLines(followProc.stdout, buffer);
|
|
176
|
+
void pumpLines(followProc.stderr, buffer);
|
|
177
|
+
const since = opts?.sinceMs ?? 0;
|
|
178
|
+
if (!opts?.follow) await followProc.exited;
|
|
179
|
+
for (const l of buffer.since(since)) yield l;
|
|
180
|
+
if (!opts?.follow) return;
|
|
181
|
+
const queue = [];
|
|
182
|
+
let notify = null;
|
|
183
|
+
const unsub = buffer.subscribe((l) => {
|
|
184
|
+
queue.push(l);
|
|
185
|
+
notify?.();
|
|
186
|
+
});
|
|
187
|
+
try {
|
|
188
|
+
while (!ctx.signal.aborted) {
|
|
189
|
+
if (queue.length === 0) {
|
|
190
|
+
await new Promise((r) => {
|
|
191
|
+
notify = r;
|
|
192
|
+
});
|
|
193
|
+
notify = null;
|
|
194
|
+
}
|
|
195
|
+
while (queue.length > 0) yield queue.shift();
|
|
196
|
+
}
|
|
197
|
+
} finally {
|
|
198
|
+
unsub();
|
|
199
|
+
followProc?.kill("SIGTERM");
|
|
200
|
+
}
|
|
201
|
+
},
|
|
202
|
+
async stop() {
|
|
203
|
+
state = "stopping";
|
|
204
|
+
followProc?.kill("SIGTERM");
|
|
205
|
+
portableSpawnSync([bin, "rm", "-f", payload.containerId]);
|
|
206
|
+
state = "stopped";
|
|
207
|
+
}
|
|
208
|
+
};
|
|
209
|
+
}
|
|
210
|
+
function makeContainerDeployTarget(bin) {
|
|
211
|
+
const payloadSchema = payloadSchemaFor(bin);
|
|
212
|
+
return {
|
|
213
|
+
id: bin,
|
|
214
|
+
displayName: bin === "docker" ? "Docker container" : "Podman container",
|
|
215
|
+
apiVersion: 1,
|
|
216
|
+
tlsTermination: "none",
|
|
217
|
+
configSchema: containerConfigSchema,
|
|
218
|
+
payloadSchema,
|
|
219
|
+
async create(config, ctx) {
|
|
220
|
+
const log = ctx.log;
|
|
221
|
+
const hostPort = config.port ?? await allocateFreePort(config.bind);
|
|
222
|
+
const containerPort = config.containerPort ?? hostPort;
|
|
223
|
+
const name = `skaile-ws-${ctx.workspaceId}`;
|
|
224
|
+
const pattern = new RegExp(config.readyPattern ?? DEFAULT_READY_PATTERN);
|
|
225
|
+
const args = [
|
|
226
|
+
bin,
|
|
227
|
+
"run",
|
|
228
|
+
"-d",
|
|
229
|
+
"--name",
|
|
230
|
+
name,
|
|
231
|
+
"-p",
|
|
232
|
+
`${config.bind}:${hostPort}:${containerPort}`,
|
|
233
|
+
...pullFlag(config.pullPolicy)
|
|
234
|
+
];
|
|
235
|
+
if (config.network) args.push("--network", config.network);
|
|
236
|
+
for (const [k, v] of Object.entries(config.env)) args.push("-e", `${k}=${v}`);
|
|
237
|
+
for (const m of config.mounts) args.push("-v", m);
|
|
238
|
+
args.push(config.image);
|
|
239
|
+
const runRes = portableSpawnSync(args);
|
|
240
|
+
if (runRes.exitCode !== 0) {
|
|
241
|
+
throw new Error(`${bin} run failed: ${runRes.stderr.trim() || runRes.stdout.trim()}`);
|
|
242
|
+
}
|
|
243
|
+
const containerId = runRes.stdout.trim();
|
|
244
|
+
log.info("container started", { runtime: bin, containerId, hostPort });
|
|
245
|
+
const payload = { containerId, hostPort, runtime: bin };
|
|
246
|
+
const ready = async () => {
|
|
247
|
+
const buffer = new LineBuffer();
|
|
248
|
+
const logProc = portableSpawn([bin, "logs", "--follow", containerId]);
|
|
249
|
+
void pumpLines(logProc.stdout, buffer);
|
|
250
|
+
void pumpLines(logProc.stderr, buffer);
|
|
251
|
+
try {
|
|
252
|
+
await waitForLine(buffer, pattern, {
|
|
253
|
+
timeoutMs: DEFAULT_READY_TIMEOUT_MS,
|
|
254
|
+
signal: ctx.signal
|
|
255
|
+
});
|
|
256
|
+
} finally {
|
|
257
|
+
logProc.kill("SIGTERM");
|
|
258
|
+
}
|
|
259
|
+
};
|
|
260
|
+
return buildHandle(bin, payload, config, ctx, ready);
|
|
261
|
+
},
|
|
262
|
+
async restore(payload, ctx) {
|
|
263
|
+
const log = ctx.log;
|
|
264
|
+
if (!isRunning(bin, payload.containerId)) {
|
|
265
|
+
throw new Error(`${bin} container ${payload.containerId} is not running`);
|
|
266
|
+
}
|
|
267
|
+
log.info("re-attached to container", { runtime: bin, containerId: payload.containerId });
|
|
268
|
+
return buildHandle(
|
|
269
|
+
bin,
|
|
270
|
+
payload,
|
|
271
|
+
{ bind: "127.0.0.1"},
|
|
272
|
+
ctx,
|
|
273
|
+
async () => {
|
|
274
|
+
}
|
|
275
|
+
);
|
|
276
|
+
}
|
|
277
|
+
};
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
// deploy/src/targets/docker.ts
|
|
281
|
+
var dockerDeployTarget = makeContainerDeployTarget("docker");
|
|
282
|
+
var localConfigSchema = z.object({
|
|
283
|
+
port: z.number().int().positive().optional(),
|
|
284
|
+
env: z.record(z.string(), z.string()).optional(),
|
|
285
|
+
cwd: z.string().optional(),
|
|
286
|
+
command: z.string().optional(),
|
|
287
|
+
args: z.array(z.string()).optional(),
|
|
288
|
+
// `skaile serve` emits structured logs rather than a single banner; the
|
|
289
|
+
// ws:// URL line is the most stable readiness marker.
|
|
290
|
+
readyPattern: z.string().optional()
|
|
291
|
+
}).strict().default({});
|
|
292
|
+
var localPayloadSchema = z.object({
|
|
293
|
+
pid: z.number(),
|
|
294
|
+
port: z.number(),
|
|
295
|
+
cwd: z.string()
|
|
296
|
+
});
|
|
297
|
+
var DEFAULT_READY_PATTERN2 = "ws://";
|
|
298
|
+
var DEFAULT_READY_TIMEOUT_MS2 = 3e4;
|
|
299
|
+
function resolveSkaileBin() {
|
|
300
|
+
const require2 = createRequire(import.meta.url);
|
|
301
|
+
const pkgPath = require2.resolve("@skaile/workspaces/package.json");
|
|
302
|
+
const pkg = require2("@skaile/workspaces/package.json");
|
|
303
|
+
const rel = pkg.bin?.skaile ?? "dist/cli/index.js";
|
|
304
|
+
return resolve(dirname(pkgPath), rel);
|
|
305
|
+
}
|
|
306
|
+
function pidAlive(pid) {
|
|
307
|
+
try {
|
|
308
|
+
process.kill(pid, 0);
|
|
309
|
+
return true;
|
|
310
|
+
} catch {
|
|
311
|
+
return false;
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
function buildHandle2(payload, ctx, ready, proc) {
|
|
315
|
+
let state = proc ? "starting" : "ready";
|
|
316
|
+
return {
|
|
317
|
+
wsUrl: `ws://127.0.0.1:${payload.port}`,
|
|
318
|
+
get state() {
|
|
319
|
+
return state;
|
|
320
|
+
},
|
|
321
|
+
payload,
|
|
322
|
+
async waitReady(timeoutMs) {
|
|
323
|
+
try {
|
|
324
|
+
await ready();
|
|
325
|
+
state = "ready";
|
|
326
|
+
} catch (err) {
|
|
327
|
+
state = "errored";
|
|
328
|
+
throw err;
|
|
329
|
+
}
|
|
330
|
+
},
|
|
331
|
+
async health() {
|
|
332
|
+
const healthy = pidAlive(payload.pid);
|
|
333
|
+
return healthy ? { healthy } : { healthy, detail: `pid ${payload.pid} not alive` };
|
|
334
|
+
},
|
|
335
|
+
async *logs(opts) {
|
|
336
|
+
if (!proc) return;
|
|
337
|
+
const since = opts?.sinceMs ?? 0;
|
|
338
|
+
for (const l of proc.buffer.since(since)) yield l;
|
|
339
|
+
if (!opts?.follow) return;
|
|
340
|
+
const queue = [];
|
|
341
|
+
let notify = null;
|
|
342
|
+
const unsub = proc.buffer.subscribe((l) => {
|
|
343
|
+
queue.push(l);
|
|
344
|
+
notify?.();
|
|
345
|
+
});
|
|
346
|
+
try {
|
|
347
|
+
while (!ctx.signal.aborted) {
|
|
348
|
+
if (queue.length === 0) {
|
|
349
|
+
await new Promise((r) => {
|
|
350
|
+
notify = r;
|
|
351
|
+
});
|
|
352
|
+
notify = null;
|
|
353
|
+
}
|
|
354
|
+
while (queue.length > 0) yield queue.shift();
|
|
355
|
+
}
|
|
356
|
+
} finally {
|
|
357
|
+
unsub();
|
|
358
|
+
}
|
|
359
|
+
},
|
|
360
|
+
async stop() {
|
|
361
|
+
state = "stopping";
|
|
362
|
+
if (pidAlive(payload.pid)) {
|
|
363
|
+
try {
|
|
364
|
+
process.kill(payload.pid, "SIGTERM");
|
|
365
|
+
} catch {
|
|
366
|
+
}
|
|
367
|
+
}
|
|
368
|
+
state = "stopped";
|
|
369
|
+
}
|
|
370
|
+
};
|
|
371
|
+
}
|
|
372
|
+
var localDeployTarget = {
|
|
373
|
+
id: "local",
|
|
374
|
+
displayName: "Local process",
|
|
375
|
+
apiVersion: 1,
|
|
376
|
+
tlsTermination: "none",
|
|
377
|
+
configSchema: localConfigSchema,
|
|
378
|
+
payloadSchema: localPayloadSchema,
|
|
379
|
+
async create(config, ctx) {
|
|
380
|
+
const log = ctx.log;
|
|
381
|
+
const port = config.port ?? await allocateFreePort();
|
|
382
|
+
const cwd = config.cwd ?? process.cwd();
|
|
383
|
+
const command = config.command ?? resolveSkaileBin();
|
|
384
|
+
const args = config.args ?? ["serve", "--port", String(port)];
|
|
385
|
+
const pattern = new RegExp(config.readyPattern ?? DEFAULT_READY_PATTERN2);
|
|
386
|
+
const proc = portableSpawn([command, ...args], {
|
|
387
|
+
cwd,
|
|
388
|
+
env: { ...process.env, ...config.env }
|
|
389
|
+
});
|
|
390
|
+
const buffer = new LineBuffer();
|
|
391
|
+
void pumpLines(proc.stdout, buffer);
|
|
392
|
+
void pumpLines(proc.stderr, buffer);
|
|
393
|
+
log.info("local process spawned", { pid: proc.pid, port, command });
|
|
394
|
+
const payload = { pid: proc.pid, port, cwd };
|
|
395
|
+
const ready = () => waitForLine(buffer, pattern, { timeoutMs: DEFAULT_READY_TIMEOUT_MS2, signal: ctx.signal });
|
|
396
|
+
return buildHandle2(payload, ctx, ready, { buffer });
|
|
397
|
+
},
|
|
398
|
+
async restore(payload, ctx) {
|
|
399
|
+
const log = ctx.log;
|
|
400
|
+
if (!pidAlive(payload.pid)) {
|
|
401
|
+
throw new Error(`local deploy pid ${payload.pid} is no longer alive`);
|
|
402
|
+
}
|
|
403
|
+
log.info("re-attached to local process", { pid: payload.pid, port: payload.port });
|
|
404
|
+
return buildHandle2(payload, ctx, async () => {
|
|
405
|
+
}, null);
|
|
406
|
+
}
|
|
407
|
+
};
|
|
408
|
+
|
|
409
|
+
// deploy/src/targets/podman.ts
|
|
410
|
+
var podmanDeployTarget = makeContainerDeployTarget("podman");
|
|
411
|
+
var storedHandleSchema = z.object({
|
|
412
|
+
targetId: z.string().min(1),
|
|
413
|
+
payload: z.unknown()
|
|
414
|
+
});
|
|
415
|
+
function handlePath(projectDir) {
|
|
416
|
+
return join(projectDir, ".skaile", "deploy", "handle.json");
|
|
417
|
+
}
|
|
418
|
+
async function writeHandle(projectDir, targetId, payload) {
|
|
419
|
+
const path = handlePath(projectDir);
|
|
420
|
+
await mkdir(dirname(path), { recursive: true });
|
|
421
|
+
const envelope = { targetId, payload };
|
|
422
|
+
await writeFile(path, `${JSON.stringify(envelope, null, 2)}
|
|
423
|
+
`, "utf8");
|
|
424
|
+
}
|
|
425
|
+
async function readHandle(projectDir) {
|
|
426
|
+
let raw;
|
|
427
|
+
try {
|
|
428
|
+
raw = await readFile(handlePath(projectDir), "utf8");
|
|
429
|
+
} catch {
|
|
430
|
+
return null;
|
|
431
|
+
}
|
|
432
|
+
let parsed;
|
|
433
|
+
try {
|
|
434
|
+
parsed = JSON.parse(raw);
|
|
435
|
+
} catch {
|
|
436
|
+
return null;
|
|
437
|
+
}
|
|
438
|
+
const result = storedHandleSchema.safeParse(parsed);
|
|
439
|
+
return result.success ? result.data : null;
|
|
440
|
+
}
|
|
441
|
+
async function clearHandle(projectDir) {
|
|
442
|
+
await rm(handlePath(projectDir), { force: true });
|
|
443
|
+
}
|
|
444
|
+
|
|
445
|
+
// deploy/src/index.ts
|
|
446
|
+
function registerBuiltinDeployTargets(registry = pluginRegistry) {
|
|
447
|
+
for (const t of [localDeployTarget, dockerDeployTarget, podmanDeployTarget]) {
|
|
448
|
+
if (!registry.get("deployTarget", t.id)) registry.register("deployTarget", t);
|
|
449
|
+
}
|
|
450
|
+
}
|
|
451
|
+
|
|
452
|
+
export { clearHandle, dockerDeployTarget, handlePath, localDeployTarget, makeContainerDeployTarget, podmanDeployTarget, readHandle, registerBuiltinDeployTargets, writeHandle };
|
|
453
|
+
//# sourceMappingURL=index.js.map
|
|
454
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../deploy/src/targets/port.ts","../../deploy/src/targets/stream-lines.ts","../../deploy/src/targets/container-runtime.ts","../../deploy/src/targets/docker.ts","../../deploy/src/targets/local.ts","../../deploy/src/targets/podman.ts","../../deploy/src/handle-store.ts","../../deploy/src/index.ts"],"names":["z2","DEFAULT_READY_PATTERN","DEFAULT_READY_TIMEOUT_MS","require","resolvePath","buildHandle","z3","dirname"],"mappings":"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","file":"index.js","sourcesContent":["/**\n * Free-port allocation for deploy targets.\n *\n * `connectors/src/port-pool.ts` exists but its `PortPool` is a stateful\n * range-allocator scoped to the connectors package; deploy targets just need a\n * single OS-assigned ephemeral port, so a small inline finder is the right fit.\n */\n\nimport { createServer } from \"node:net\";\n\n/** Bind to port 0, read the OS-assigned port, release it, and return it. */\nexport function allocateFreePort(host = \"127.0.0.1\"): Promise<number> {\n return new Promise((resolve, reject) => {\n const server = createServer();\n server.once(\"error\", reject);\n server.listen(0, host, () => {\n const addr = server.address();\n if (addr === null || typeof addr === \"string\") {\n server.close();\n reject(new Error(\"could not determine allocated port\"));\n return;\n }\n const { port } = addr;\n server.close(() => resolve(port));\n });\n });\n}\n","/**\n * Shared line-buffering helpers for deploy targets.\n *\n * Both the local target (subprocess stdout) and the container runtimes\n * (`<bin> logs`) read text off a web `ReadableStream<Uint8Array>` and need the\n * same UTF-8 decode + newline-split logic, plus a timestamped ring buffer so\n * `DeployHandle.logs()` can replay what was already seen.\n */\n\n/** One buffered log line with the wall-clock time it was observed. */\nexport interface BufferedLine {\n ts: number;\n line: string;\n}\n\n/**\n * Bounded FIFO of decoded lines. Drives both the readiness watcher and\n * `logs()` replay; the cap keeps a long-running process from growing unbounded.\n */\nexport class LineBuffer {\n private readonly lines: BufferedLine[] = [];\n private readonly waiters = new Set<(line: BufferedLine) => void>();\n\n constructor(private readonly max = 1000) {}\n\n push(line: string): void {\n const entry: BufferedLine = { ts: Date.now(), line };\n this.lines.push(entry);\n if (this.lines.length > this.max) this.lines.shift();\n for (const w of this.waiters) w(entry);\n }\n\n /** Snapshot of buffered lines at or after `sinceMs` (epoch millis). */\n since(sinceMs = 0): BufferedLine[] {\n return this.lines.filter((l) => l.ts >= sinceMs);\n }\n\n /** Subscribe to future pushes; returns an unsubscribe fn. */\n subscribe(fn: (line: BufferedLine) => void): () => void {\n this.waiters.add(fn);\n return () => this.waiters.delete(fn);\n }\n}\n\n/**\n * Pump a byte stream into a {@link LineBuffer}, splitting on `\\n`. Resolves when\n * the stream ends. Errors (e.g. the process being killed mid-read) are\n * swallowed — a dead stream just means no more lines.\n */\nexport async function pumpLines(\n stream: ReadableStream<Uint8Array>,\n buffer: LineBuffer,\n): Promise<void> {\n const reader = stream.getReader();\n const decoder = new TextDecoder();\n let pending = \"\";\n try {\n while (true) {\n const { done, value } = await reader.read();\n if (done) break;\n pending += decoder.decode(value, { stream: true });\n let idx = pending.indexOf(\"\\n\");\n while (idx !== -1) {\n buffer.push(pending.slice(0, idx));\n pending = pending.slice(idx + 1);\n idx = pending.indexOf(\"\\n\");\n }\n }\n } catch {\n // Stream torn down (process killed) — nothing more to read.\n } finally {\n if (pending.length > 0) buffer.push(pending);\n reader.releaseLock();\n }\n}\n\n/**\n * Resolve once a buffered line matches `pattern`, reject on timeout or abort.\n * Checks already-buffered lines first so a fast-printing process never races\n * the subscription.\n */\nexport function waitForLine(\n buffer: LineBuffer,\n pattern: RegExp,\n opts: { timeoutMs: number; signal: AbortSignal },\n): Promise<void> {\n for (const { line } of buffer.since(0)) {\n if (pattern.test(line)) return Promise.resolve();\n }\n return new Promise<void>((resolve, reject) => {\n let settled = false;\n const finish = (fn: () => void) => {\n if (settled) return;\n settled = true;\n clearTimeout(timer);\n unsub();\n opts.signal.removeEventListener(\"abort\", onAbort);\n fn();\n };\n const unsub = buffer.subscribe(({ line }) => {\n if (pattern.test(line)) finish(resolve);\n });\n const onAbort = () => finish(() => reject(new Error(\"aborted while waiting for ready line\")));\n const timer = setTimeout(\n () =>\n finish(() =>\n reject(new Error(`timed out after ${opts.timeoutMs}ms waiting for ready line`)),\n ),\n opts.timeoutMs,\n );\n if (opts.signal.aborted) onAbort();\n else opts.signal.addEventListener(\"abort\", onAbort);\n });\n}\n","/**\n * Shared docker/podman deploy target, parameterized by binary name.\n *\n * docker and podman expose the same CLI surface for the run/inspect/logs/rm\n * verbs this target needs, so one implementation backs both\n * (`makeContainerDeployTarget(\"docker\" | \"podman\")`). `tlsTermination: \"none\"`\n * means loopback-only: a zod refinement rejects a non-loopback `bind` until tls\n * config lands (locked decision #12).\n */\n\nimport { portableSpawn, portableSpawnSync, type PortableSubprocess } from \"@skaile/workspaces/core\";\nimport type { DeployContext, DeployHandle, DeployTarget } from \"@skaile/workspaces/plugin-registry\";\nimport * as z from \"zod\";\nimport { allocateFreePort } from \"./port.js\";\nimport { LineBuffer, pumpLines, waitForLine } from \"./stream-lines.js\";\n\ntype ContainerBin = \"docker\" | \"podman\";\n\nconst LOOPBACK_BINDS = new Set([\"127.0.0.1\", \"localhost\", \"::1\"]);\nconst DEFAULT_READY_PATTERN = \"ws://\";\nconst DEFAULT_READY_TIMEOUT_MS = 60_000;\n\nconst containerConfigSchema = z\n .object({\n image: z.string(),\n port: z.number().int().positive().optional(),\n bind: z.string().default(\"127.0.0.1\"),\n mounts: z.array(z.string()).default([]),\n env: z.record(z.string(), z.string()).default({}),\n network: z.string().optional(),\n pullPolicy: z.enum([\"always\", \"if-missing\", \"never\"]).default(\"if-missing\"),\n // v1 defers in-provider image builds; \"local\" means \"run the given image\".\n buildStrategy: z.literal(\"local\").default(\"local\"),\n containerPort: z.number().int().positive().optional(),\n readyPattern: z.string().optional(),\n })\n .strict()\n .refine((c) => LOOPBACK_BINDS.has(c.bind), {\n message: \"non-loopback bind requires tls config (not yet supported); use a loopback bind\",\n path: [\"bind\"],\n });\n\ntype ContainerConfig = z.infer<typeof containerConfigSchema>;\n\nfunction payloadSchemaFor(bin: ContainerBin) {\n return z.object({\n containerId: z.string(),\n hostPort: z.number(),\n runtime: z.enum([\"docker\", \"podman\"]).refine((r) => r === bin, {\n message: `runtime mismatch: handle belongs to ${bin}`,\n }),\n });\n}\n\ntype ContainerPayload = { containerId: string; hostPort: number; runtime: ContainerBin };\n\nfunction pullFlag(policy: ContainerConfig[\"pullPolicy\"]): string[] {\n // `docker run --pull` takes always|missing|never; map our \"if-missing\".\n const mapped = policy === \"if-missing\" ? \"missing\" : policy;\n return [\"--pull\", mapped];\n}\n\n/** True when `<bin> inspect` reports the container in a running state. */\nfunction isRunning(bin: ContainerBin, containerId: string): boolean {\n const res = portableSpawnSync([bin, \"inspect\", \"-f\", \"{{.State.Running}}\", containerId]);\n return res.exitCode === 0 && res.stdout.trim() === \"true\";\n}\n\nfunction buildHandle(\n bin: ContainerBin,\n payload: ContainerPayload,\n config: Pick<ContainerConfig, \"bind\" | \"readyPattern\">,\n ctx: DeployContext,\n ready: () => Promise<void>,\n): DeployHandle<ContainerPayload> {\n let state: DeployHandle[\"state\"] = \"starting\";\n let followProc: PortableSubprocess | null = null;\n return {\n wsUrl: `ws://${config.bind}:${payload.hostPort}`,\n get state() {\n return state;\n },\n payload,\n async waitReady(timeoutMs?: number) {\n try {\n await ready();\n state = \"ready\";\n } catch (err) {\n state = \"errored\";\n throw err;\n }\n void timeoutMs;\n },\n async health() {\n const healthy = isRunning(bin, payload.containerId);\n return healthy\n ? { healthy }\n : { healthy, detail: `container ${payload.containerId} not running` };\n },\n async *logs(opts) {\n const buffer = new LineBuffer();\n const args = [\"logs\"];\n if (opts?.follow) args.push(\"--follow\");\n followProc = portableSpawn([bin, ...args, payload.containerId]);\n void pumpLines(followProc.stdout, buffer);\n void pumpLines(followProc.stderr, buffer);\n const since = opts?.sinceMs ?? 0;\n // Give the snapshot read a tick to drain non-follow output.\n if (!opts?.follow) await followProc.exited;\n for (const l of buffer.since(since)) yield l;\n if (!opts?.follow) return;\n const queue: { ts: number; line: string }[] = [];\n let notify: (() => void) | null = null;\n const unsub = buffer.subscribe((l) => {\n queue.push(l);\n notify?.();\n });\n try {\n while (!ctx.signal.aborted) {\n if (queue.length === 0) {\n await new Promise<void>((r) => {\n notify = r;\n });\n notify = null;\n }\n while (queue.length > 0) yield queue.shift() as { ts: number; line: string };\n }\n } finally {\n unsub();\n followProc?.kill(\"SIGTERM\");\n }\n },\n async stop() {\n state = \"stopping\";\n followProc?.kill(\"SIGTERM\");\n // `rm -f` removes a running or stopped container and is a no-op (nonzero\n // exit, ignored) when the container is already gone — idempotent.\n portableSpawnSync([bin, \"rm\", \"-f\", payload.containerId]);\n state = \"stopped\";\n },\n };\n}\n\n/** Build a docker/podman deploy target backed by `bin`. */\nexport function makeContainerDeployTarget(\n bin: ContainerBin,\n): DeployTarget<ContainerConfig, ContainerPayload> {\n const payloadSchema = payloadSchemaFor(bin) as unknown as z.ZodType<ContainerPayload>;\n return {\n id: bin,\n displayName: bin === \"docker\" ? \"Docker container\" : \"Podman container\",\n apiVersion: 1,\n tlsTermination: \"none\",\n configSchema: containerConfigSchema as unknown as z.ZodType<ContainerConfig>,\n payloadSchema,\n\n async create(config, ctx) {\n const log = ctx.log;\n const hostPort = config.port ?? (await allocateFreePort(config.bind));\n const containerPort = config.containerPort ?? hostPort;\n const name = `skaile-ws-${ctx.workspaceId}`;\n const pattern = new RegExp(config.readyPattern ?? DEFAULT_READY_PATTERN);\n\n const args = [\n bin,\n \"run\",\n \"-d\",\n \"--name\",\n name,\n \"-p\",\n `${config.bind}:${hostPort}:${containerPort}`,\n ...pullFlag(config.pullPolicy),\n ];\n if (config.network) args.push(\"--network\", config.network);\n for (const [k, v] of Object.entries(config.env)) args.push(\"-e\", `${k}=${v}`);\n for (const m of config.mounts) args.push(\"-v\", m);\n args.push(config.image);\n\n const runRes = portableSpawnSync(args);\n if (runRes.exitCode !== 0) {\n throw new Error(`${bin} run failed: ${runRes.stderr.trim() || runRes.stdout.trim()}`);\n }\n const containerId = runRes.stdout.trim();\n log.info(\"container started\", { runtime: bin, containerId, hostPort });\n\n const payload: ContainerPayload = { containerId, hostPort, runtime: bin };\n const ready = async () => {\n // Poll logs for the ready line; fall back to a running-state check so a\n // ready image that never prints the marker still resolves.\n const buffer = new LineBuffer();\n const logProc = portableSpawn([bin, \"logs\", \"--follow\", containerId]);\n void pumpLines(logProc.stdout, buffer);\n void pumpLines(logProc.stderr, buffer);\n try {\n await waitForLine(buffer, pattern, {\n timeoutMs: DEFAULT_READY_TIMEOUT_MS,\n signal: ctx.signal,\n });\n } finally {\n logProc.kill(\"SIGTERM\");\n }\n };\n return buildHandle(bin, payload, config, ctx, ready);\n },\n\n async restore(payload, ctx) {\n const log = ctx.log;\n if (!isRunning(bin, payload.containerId)) {\n throw new Error(`${bin} container ${payload.containerId} is not running`);\n }\n log.info(\"re-attached to container\", { runtime: bin, containerId: payload.containerId });\n return buildHandle(\n bin,\n payload,\n { bind: \"127.0.0.1\", readyPattern: undefined },\n ctx,\n async () => {},\n );\n },\n };\n}\n","/** `dockerDeployTarget` — the shared container runtime bound to the `docker` bin. */\n\nimport { makeContainerDeployTarget } from \"./container-runtime.js\";\n\nexport const dockerDeployTarget = makeContainerDeployTarget(\"docker\");\n","/**\n * `localDeployTarget` — stand a workspace up as a local child process and reach\n * it over `ws://127.0.0.1:<port>`. `tlsTermination: \"none\"` (loopback only).\n *\n * The `command` / `args` / `readyPattern` config fields exist so the target is\n * testable hermetically: a test points them at a mock server instead of a real\n * `skaile serve`. Defaults resolve the `skaile` bin from this package and run\n * `skaile serve --port <port>`.\n */\n\nimport { createRequire } from \"node:module\";\nimport { dirname, resolve as resolvePath } from \"node:path\";\nimport { portableSpawn, type PortableSubprocess } from \"@skaile/workspaces/core\";\nimport type { DeployContext, DeployHandle, DeployTarget } from \"@skaile/workspaces/plugin-registry\";\nimport * as z from \"zod\";\nimport { allocateFreePort } from \"./port.js\";\nimport { LineBuffer, pumpLines, waitForLine } from \"./stream-lines.js\";\n\nconst localConfigSchema = z\n .object({\n port: z.number().int().positive().optional(),\n env: z.record(z.string(), z.string()).optional(),\n cwd: z.string().optional(),\n command: z.string().optional(),\n args: z.array(z.string()).optional(),\n // `skaile serve` emits structured logs rather than a single banner; the\n // ws:// URL line is the most stable readiness marker.\n readyPattern: z.string().optional(),\n })\n .strict()\n .default({});\n\ntype LocalConfig = z.infer<typeof localConfigSchema>;\n\nconst localPayloadSchema = z.object({\n pid: z.number(),\n port: z.number(),\n cwd: z.string(),\n});\n\ntype LocalPayload = z.infer<typeof localPayloadSchema>;\n\nconst DEFAULT_READY_PATTERN = \"ws://\";\nconst DEFAULT_READY_TIMEOUT_MS = 30_000;\n\n/** Resolve the `skaile` bin shipped by this package, for the default command. */\nfunction resolveSkaileBin(): string {\n const require = createRequire(import.meta.url);\n const pkgPath = require.resolve(\"@skaile/workspaces/package.json\");\n const pkg = require(\"@skaile/workspaces/package.json\") as { bin?: Record<string, string> };\n const rel = pkg.bin?.skaile ?? \"dist/cli/index.js\";\n return resolvePath(dirname(pkgPath), rel);\n}\n\n/** True when a pid is alive (signal 0 probes without delivering a signal). */\nfunction pidAlive(pid: number): boolean {\n try {\n process.kill(pid, 0);\n return true;\n } catch {\n return false;\n }\n}\n\ninterface LocalProcess {\n proc: PortableSubprocess;\n buffer: LineBuffer;\n}\n\nfunction buildHandle(\n payload: LocalPayload,\n ctx: DeployContext,\n ready: () => Promise<void>,\n proc: LocalProcess | null,\n): DeployHandle<LocalPayload> {\n let state: DeployHandle[\"state\"] = proc ? \"starting\" : \"ready\";\n return {\n wsUrl: `ws://127.0.0.1:${payload.port}`,\n get state() {\n return state;\n },\n payload,\n async waitReady(timeoutMs?: number) {\n try {\n await ready();\n state = \"ready\";\n } catch (err) {\n state = \"errored\";\n throw err;\n }\n // `timeoutMs` is honoured inside `ready()` via the config-default timeout;\n // the param stays in the signature for contract parity.\n void timeoutMs;\n },\n async health() {\n const healthy = pidAlive(payload.pid);\n return healthy ? { healthy } : { healthy, detail: `pid ${payload.pid} not alive` };\n },\n async *logs(opts) {\n if (!proc) return; // a restored handle re-attaches by pid; it has no stream.\n const since = opts?.sinceMs ?? 0;\n for (const l of proc.buffer.since(since)) yield l;\n if (!opts?.follow) return;\n const queue: { ts: number; line: string }[] = [];\n let notify: (() => void) | null = null;\n const unsub = proc.buffer.subscribe((l) => {\n queue.push(l);\n notify?.();\n });\n try {\n while (!ctx.signal.aborted) {\n if (queue.length === 0) {\n await new Promise<void>((r) => {\n notify = r;\n });\n notify = null;\n }\n while (queue.length > 0) yield queue.shift() as { ts: number; line: string };\n }\n } finally {\n unsub();\n }\n },\n async stop() {\n state = \"stopping\";\n if (pidAlive(payload.pid)) {\n try {\n process.kill(payload.pid, \"SIGTERM\");\n } catch {\n // Already gone between the check and the kill — idempotent.\n }\n }\n state = \"stopped\";\n },\n };\n}\n\n/** Built-in target that runs the workspace as a local child process. */\nexport const localDeployTarget: DeployTarget<LocalConfig, LocalPayload> = {\n id: \"local\",\n displayName: \"Local process\",\n apiVersion: 1,\n tlsTermination: \"none\",\n configSchema: localConfigSchema as unknown as z.ZodType<LocalConfig>,\n payloadSchema: localPayloadSchema,\n\n async create(config, ctx) {\n const log = ctx.log;\n const port = config.port ?? (await allocateFreePort());\n const cwd = config.cwd ?? process.cwd();\n const command = config.command ?? resolveSkaileBin();\n const args = config.args ?? [\"serve\", \"--port\", String(port)];\n const pattern = new RegExp(config.readyPattern ?? DEFAULT_READY_PATTERN);\n\n const proc = portableSpawn([command, ...args], {\n cwd,\n env: { ...process.env, ...config.env },\n });\n const buffer = new LineBuffer();\n void pumpLines(proc.stdout, buffer);\n void pumpLines(proc.stderr, buffer);\n log.info(\"local process spawned\", { pid: proc.pid, port, command });\n\n const payload: LocalPayload = { pid: proc.pid, port, cwd };\n const ready = () =>\n waitForLine(buffer, pattern, { timeoutMs: DEFAULT_READY_TIMEOUT_MS, signal: ctx.signal });\n return buildHandle(payload, ctx, ready, { proc, buffer });\n },\n\n async restore(payload, ctx) {\n const log = ctx.log;\n if (!pidAlive(payload.pid)) {\n throw new Error(`local deploy pid ${payload.pid} is no longer alive`);\n }\n log.info(\"re-attached to local process\", { pid: payload.pid, port: payload.port });\n // Re-attached handle is already running; readiness is a no-op.\n return buildHandle(payload, ctx, async () => {}, null);\n },\n};\n","/** `podmanDeployTarget` — the shared container runtime bound to the `podman` bin. */\n\nimport { makeContainerDeployTarget } from \"./container-runtime.js\";\n\nexport const podmanDeployTarget = makeContainerDeployTarget(\"podman\");\n","/**\n * Registry-owned persistence for the detach handle at\n * `<project>/.skaile/deploy/handle.json` (locked decision #13).\n *\n * The file is a `{ targetId, payload }` envelope: `targetId` selects which\n * `DeployTarget` to look up, `payload` is that target's opaque restore slice.\n * The store never interprets `payload` — it only validates the envelope shape.\n */\n\nimport { mkdir, readFile, rm, writeFile } from \"node:fs/promises\";\nimport { dirname, join } from \"node:path\";\nimport * as z from \"zod\";\n\n/** Parsed detach-handle envelope. `payload` is the target's opaque restore slice. */\nexport interface StoredHandle {\n targetId: string;\n payload: unknown;\n}\n\nconst storedHandleSchema = z.object({\n targetId: z.string().min(1),\n payload: z.unknown(),\n});\n\n/** Absolute path to the detach-handle file for a project. */\nexport function handlePath(projectDir: string): string {\n return join(projectDir, \".skaile\", \"deploy\", \"handle.json\");\n}\n\n/** Write the detach handle, creating `.skaile/deploy/` if missing. */\nexport async function writeHandle(\n projectDir: string,\n targetId: string,\n payload: unknown,\n): Promise<void> {\n const path = handlePath(projectDir);\n await mkdir(dirname(path), { recursive: true });\n const envelope: StoredHandle = { targetId, payload };\n await writeFile(path, `${JSON.stringify(envelope, null, 2)}\\n`, \"utf8\");\n}\n\n/** Read + validate the detach handle. Returns null when absent or malformed. */\nexport async function readHandle(projectDir: string): Promise<StoredHandle | null> {\n let raw: string;\n try {\n raw = await readFile(handlePath(projectDir), \"utf8\");\n } catch {\n return null;\n }\n let parsed: unknown;\n try {\n parsed = JSON.parse(raw);\n } catch {\n return null;\n }\n const result = storedHandleSchema.safeParse(parsed);\n return result.success ? result.data : null;\n}\n\n/** Remove the detach handle. Idempotent — a missing file is not an error. */\nexport async function clearHandle(projectDir: string): Promise<void> {\n await rm(handlePath(projectDir), { force: true });\n}\n","/**\n * `@skaile/workspaces/deploy` — built-in deploy targets + the detach-handle store.\n *\n * Ships the three loopback-only built-ins (local, docker, podman),\n * `registerBuiltinDeployTargets()` to load them into a `PluginRegistry`, and the\n * `.skaile/deploy/handle.json` read/write/restore API. Cloud targets (fly,\n * vercel-sandbox, k8s) ship as separate packages that self-register.\n */\n\nimport { pluginRegistry, type PluginRegistry } from \"@skaile/workspaces/plugin-registry\";\nimport { dockerDeployTarget } from \"./targets/docker.js\";\nimport { localDeployTarget } from \"./targets/local.js\";\nimport { podmanDeployTarget } from \"./targets/podman.js\";\n\nexport { localDeployTarget } from \"./targets/local.js\";\nexport { dockerDeployTarget } from \"./targets/docker.js\";\nexport { podmanDeployTarget } from \"./targets/podman.js\";\nexport { makeContainerDeployTarget } from \"./targets/container-runtime.js\";\nexport {\n clearHandle,\n handlePath,\n readHandle,\n writeHandle,\n type StoredHandle,\n} from \"./handle-store.js\";\n\n/**\n * Register the built-in deploy targets into `registry` (the process-wide\n * `pluginRegistry` by default). Idempotent — an id already present is skipped,\n * so calling this from multiple CLI entrypoints is safe.\n */\nexport function registerBuiltinDeployTargets(registry: PluginRegistry = pluginRegistry): void {\n for (const t of [localDeployTarget, dockerDeployTarget, podmanDeployTarget]) {\n if (!registry.get(\"deployTarget\", t.id)) registry.register(\"deployTarget\", t);\n }\n}\n"]}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Registry-owned persistence for the detach handle at
|
|
3
|
+
* `<project>/.skaile/deploy/handle.json` (locked decision #13).
|
|
4
|
+
*
|
|
5
|
+
* The file is a `{ targetId, payload }` envelope: `targetId` selects which
|
|
6
|
+
* `DeployTarget` to look up, `payload` is that target's opaque restore slice.
|
|
7
|
+
* The store never interprets `payload` — it only validates the envelope shape.
|
|
8
|
+
*/
|
|
9
|
+
/** Parsed detach-handle envelope. `payload` is the target's opaque restore slice. */
|
|
10
|
+
export interface StoredHandle {
|
|
11
|
+
targetId: string;
|
|
12
|
+
payload: unknown;
|
|
13
|
+
}
|
|
14
|
+
/** Absolute path to the detach-handle file for a project. */
|
|
15
|
+
export declare function handlePath(projectDir: string): string;
|
|
16
|
+
/** Write the detach handle, creating `.skaile/deploy/` if missing. */
|
|
17
|
+
export declare function writeHandle(projectDir: string, targetId: string, payload: unknown): Promise<void>;
|
|
18
|
+
/** Read + validate the detach handle. Returns null when absent or malformed. */
|
|
19
|
+
export declare function readHandle(projectDir: string): Promise<StoredHandle | null>;
|
|
20
|
+
/** Remove the detach handle. Idempotent — a missing file is not an error. */
|
|
21
|
+
export declare function clearHandle(projectDir: string): Promise<void>;
|
|
22
|
+
//# sourceMappingURL=handle-store.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"handle-store.d.ts","sourceRoot":"","sources":["../../../deploy/src/handle-store.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAMH,qFAAqF;AACrF,MAAM,WAAW,YAAY;IAC3B,QAAQ,EAAE,MAAM,CAAC;IACjB,OAAO,EAAE,OAAO,CAAC;CAClB;AAOD,6DAA6D;AAC7D,wBAAgB,UAAU,CAAC,UAAU,EAAE,MAAM,GAAG,MAAM,CAErD;AAED,sEAAsE;AACtE,wBAAsB,WAAW,CAC/B,UAAU,EAAE,MAAM,EAClB,QAAQ,EAAE,MAAM,EAChB,OAAO,EAAE,OAAO,GACf,OAAO,CAAC,IAAI,CAAC,CAKf;AAED,gFAAgF;AAChF,wBAAsB,UAAU,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC,CAejF;AAED,6EAA6E;AAC7E,wBAAsB,WAAW,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAEnE"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* `@skaile/workspaces/deploy` — built-in deploy targets + the detach-handle store.
|
|
3
|
+
*
|
|
4
|
+
* Ships the three loopback-only built-ins (local, docker, podman),
|
|
5
|
+
* `registerBuiltinDeployTargets()` to load them into a `PluginRegistry`, and the
|
|
6
|
+
* `.skaile/deploy/handle.json` read/write/restore API. Cloud targets (fly,
|
|
7
|
+
* vercel-sandbox, k8s) ship as separate packages that self-register.
|
|
8
|
+
*/
|
|
9
|
+
import { type PluginRegistry } from "@skaile/workspaces/plugin-registry";
|
|
10
|
+
export { localDeployTarget } from "./targets/local.js";
|
|
11
|
+
export { dockerDeployTarget } from "./targets/docker.js";
|
|
12
|
+
export { podmanDeployTarget } from "./targets/podman.js";
|
|
13
|
+
export { makeContainerDeployTarget } from "./targets/container-runtime.js";
|
|
14
|
+
export { clearHandle, handlePath, readHandle, writeHandle, type StoredHandle, } from "./handle-store.js";
|
|
15
|
+
/**
|
|
16
|
+
* Register the built-in deploy targets into `registry` (the process-wide
|
|
17
|
+
* `pluginRegistry` by default). Idempotent — an id already present is skipped,
|
|
18
|
+
* so calling this from multiple CLI entrypoints is safe.
|
|
19
|
+
*/
|
|
20
|
+
export declare function registerBuiltinDeployTargets(registry?: PluginRegistry): void;
|
|
21
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../deploy/src/index.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,EAAkB,KAAK,cAAc,EAAE,MAAM,oCAAoC,CAAC;AAKzF,OAAO,EAAE,iBAAiB,EAAE,MAAM,oBAAoB,CAAC;AACvD,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AACzD,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AACzD,OAAO,EAAE,yBAAyB,EAAE,MAAM,gCAAgC,CAAC;AAC3E,OAAO,EACL,WAAW,EACX,UAAU,EACV,UAAU,EACV,WAAW,EACX,KAAK,YAAY,GAClB,MAAM,mBAAmB,CAAC;AAE3B;;;;GAIG;AACH,wBAAgB,4BAA4B,CAAC,QAAQ,GAAE,cAA+B,GAAG,IAAI,CAI5F"}
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared docker/podman deploy target, parameterized by binary name.
|
|
3
|
+
*
|
|
4
|
+
* docker and podman expose the same CLI surface for the run/inspect/logs/rm
|
|
5
|
+
* verbs this target needs, so one implementation backs both
|
|
6
|
+
* (`makeContainerDeployTarget("docker" | "podman")`). `tlsTermination: "none"`
|
|
7
|
+
* means loopback-only: a zod refinement rejects a non-loopback `bind` until tls
|
|
8
|
+
* config lands (locked decision #12).
|
|
9
|
+
*/
|
|
10
|
+
import type { DeployTarget } from "@skaile/workspaces/plugin-registry";
|
|
11
|
+
import * as z from "zod";
|
|
12
|
+
type ContainerBin = "docker" | "podman";
|
|
13
|
+
declare const containerConfigSchema: z.ZodObject<{
|
|
14
|
+
image: z.ZodString;
|
|
15
|
+
port: z.ZodOptional<z.ZodNumber>;
|
|
16
|
+
bind: z.ZodDefault<z.ZodString>;
|
|
17
|
+
mounts: z.ZodDefault<z.ZodArray<z.ZodString>>;
|
|
18
|
+
env: z.ZodDefault<z.ZodRecord<z.ZodString, z.ZodString>>;
|
|
19
|
+
network: z.ZodOptional<z.ZodString>;
|
|
20
|
+
pullPolicy: z.ZodDefault<z.ZodEnum<{
|
|
21
|
+
never: "never";
|
|
22
|
+
always: "always";
|
|
23
|
+
"if-missing": "if-missing";
|
|
24
|
+
}>>;
|
|
25
|
+
buildStrategy: z.ZodDefault<z.ZodLiteral<"local">>;
|
|
26
|
+
containerPort: z.ZodOptional<z.ZodNumber>;
|
|
27
|
+
readyPattern: z.ZodOptional<z.ZodString>;
|
|
28
|
+
}, z.core.$strict>;
|
|
29
|
+
type ContainerConfig = z.infer<typeof containerConfigSchema>;
|
|
30
|
+
type ContainerPayload = {
|
|
31
|
+
containerId: string;
|
|
32
|
+
hostPort: number;
|
|
33
|
+
runtime: ContainerBin;
|
|
34
|
+
};
|
|
35
|
+
/** Build a docker/podman deploy target backed by `bin`. */
|
|
36
|
+
export declare function makeContainerDeployTarget(bin: ContainerBin): DeployTarget<ContainerConfig, ContainerPayload>;
|
|
37
|
+
export {};
|
|
38
|
+
//# sourceMappingURL=container-runtime.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"container-runtime.d.ts","sourceRoot":"","sources":["../../../../deploy/src/targets/container-runtime.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAGH,OAAO,KAAK,EAA+B,YAAY,EAAE,MAAM,oCAAoC,CAAC;AACpG,OAAO,KAAK,CAAC,MAAM,KAAK,CAAC;AAIzB,KAAK,YAAY,GAAG,QAAQ,GAAG,QAAQ,CAAC;AAMxC,QAAA,MAAM,qBAAqB;;;;;;;;;;;;;;;kBAkBvB,CAAC;AAEL,KAAK,eAAe,GAAG,CAAC,CAAC,KAAK,CAAC,OAAO,qBAAqB,CAAC,CAAC;AAY7D,KAAK,gBAAgB,GAAG;IAAE,WAAW,EAAE,MAAM,CAAC;IAAC,QAAQ,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,YAAY,CAAA;CAAE,CAAC;AAyFzF,2DAA2D;AAC3D,wBAAgB,yBAAyB,CACvC,GAAG,EAAE,YAAY,GAChB,YAAY,CAAC,eAAe,EAAE,gBAAgB,CAAC,CA0EjD"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
/** `dockerDeployTarget` — the shared container runtime bound to the `docker` bin. */
|
|
2
|
+
export declare const dockerDeployTarget: import("../../../plugin-registry/src/targets.ts").DeployTarget<{
|
|
3
|
+
image: string;
|
|
4
|
+
bind: string;
|
|
5
|
+
mounts: string[];
|
|
6
|
+
env: Record<string, string>;
|
|
7
|
+
pullPolicy: "never" | "always" | "if-missing";
|
|
8
|
+
buildStrategy: "local";
|
|
9
|
+
port?: number | undefined;
|
|
10
|
+
network?: string | undefined;
|
|
11
|
+
containerPort?: number | undefined;
|
|
12
|
+
readyPattern?: string | undefined;
|
|
13
|
+
}, {
|
|
14
|
+
containerId: string;
|
|
15
|
+
hostPort: number;
|
|
16
|
+
runtime: "docker" | "podman";
|
|
17
|
+
}>;
|
|
18
|
+
//# sourceMappingURL=docker.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"docker.d.ts","sourceRoot":"","sources":["../../../../deploy/src/targets/docker.ts"],"names":[],"mappings":"AAAA,qFAAqF;AAIrF,eAAO,MAAM,kBAAkB;;;;;;;;;;;;;;;EAAsC,CAAC"}
|