@skaile/workspaces 0.22.0-beta.1 → 0.22.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 +380 -0
- package/dist/{asset-feeds-QXCSAJRN.js → asset-feeds-Y2CDCM3W.js} +13 -14
- package/dist/asset-feeds-Y2CDCM3W.js.map +1 -0
- package/dist/asset-manager/index.js +7 -7
- package/dist/asset-manager/installer.js +6 -6
- package/dist/asset-manager/src/index.d.ts.map +1 -1
- package/dist/base-assets/connectors/deploy.js +7 -7
- package/dist/base-assets/connectors/devserver.js +7 -7
- package/dist/base-assets/connectors/flow/adapter.js +7 -7
- package/dist/base-assets/connectors/flow/run-flow.js +8 -8
- package/dist/base-assets/connectors/flow.js +7 -7
- package/dist/base-assets/connectors/git/driver.d.ts.map +1 -1
- package/dist/base-assets/connectors/git.js +7 -7
- package/dist/base-assets/connectors/gmail.js +7 -7
- package/dist/base-assets/connectors/googledrive.js +7 -7
- package/dist/base-assets/connectors/local.js +7 -7
- package/dist/base-assets/connectors/mattermost.js +7 -7
- package/dist/base-assets/connectors/memory.js +7 -7
- package/dist/base-assets/connectors/minio.js +7 -7
- package/dist/base-assets/connectors/postgres.js +7 -7
- package/dist/base-assets/connectors/s3.js +7 -7
- package/dist/base-assets/connectors/sharepoint.js +7 -7
- package/dist/base-assets/connectors/sqlite.js +7 -7
- package/dist/base-assets/connectors/static-server.js +7 -7
- package/dist/base-assets/connectors/tunnel.js +7 -7
- package/dist/base-assets/connectors/webdav.js +7 -7
- package/dist/base-assets/connectors/xstate-store.js +7 -7
- package/dist/base-assets/connectors/xstate.js +7 -7
- package/dist/{chunk-PTIHB2TV.js → chunk-2RYQERIT.js} +4 -4
- package/dist/{chunk-PTIHB2TV.js.map → chunk-2RYQERIT.js.map} +1 -1
- package/dist/chunk-32NA4TVC.js +30 -0
- package/dist/chunk-32NA4TVC.js.map +1 -0
- package/dist/{chunk-DKGDOALM.js → chunk-53UNDY6K.js} +5 -5
- package/dist/{chunk-DKGDOALM.js.map → chunk-53UNDY6K.js.map} +1 -1
- package/dist/{chunk-VCYXVP2S.js → chunk-7HSXUKNB.js} +24 -13
- package/dist/chunk-7HSXUKNB.js.map +1 -0
- package/dist/chunk-7QBNJTTQ.js +3 -0
- package/dist/{chunk-W2O5LWYU.js.map → chunk-7QBNJTTQ.js.map} +1 -1
- package/dist/{chunk-UMOENHVH.js → chunk-ETMUGBHF.js} +3 -3
- package/dist/{chunk-UMOENHVH.js.map → chunk-ETMUGBHF.js.map} +1 -1
- package/dist/{chunk-7PTP3SQJ.js → chunk-GTS2FODO.js} +32 -7
- package/dist/chunk-GTS2FODO.js.map +1 -0
- package/dist/{chunk-D7K72XEY.js → chunk-JN2CUVSU.js} +3 -3
- package/dist/{chunk-D7K72XEY.js.map → chunk-JN2CUVSU.js.map} +1 -1
- package/dist/{chunk-3ECS5PFD.js → chunk-K2HDYSAM.js} +4 -4
- package/dist/{chunk-3ECS5PFD.js.map → chunk-K2HDYSAM.js.map} +1 -1
- package/dist/{chunk-4AZKT2BU.js → chunk-K7WPR77X.js} +33 -50
- package/dist/chunk-K7WPR77X.js.map +1 -0
- package/dist/{chunk-JHF66MCK.js → chunk-MNAHNDUI.js} +5 -3
- package/dist/chunk-MNAHNDUI.js.map +1 -0
- package/dist/{chunk-APAOQLPT.js → chunk-NBJ5TOEC.js} +3 -3
- package/dist/{chunk-APAOQLPT.js.map → chunk-NBJ5TOEC.js.map} +1 -1
- package/dist/{chunk-NJLHHZIW.js → chunk-NDD5VMN5.js} +2 -2
- package/dist/{chunk-NJLHHZIW.js.map → chunk-NDD5VMN5.js.map} +1 -1
- package/dist/{chunk-LT4DLEYE.js → chunk-OJN25VJO.js} +24 -8
- package/dist/chunk-OJN25VJO.js.map +1 -0
- package/dist/{chunk-GFNW72LW.js → chunk-PFOXL4SH.js} +4 -4
- package/dist/{chunk-GFNW72LW.js.map → chunk-PFOXL4SH.js.map} +1 -1
- package/dist/{chunk-V3QMSM5I.js → chunk-SKXCTV55.js} +13 -14
- package/dist/chunk-SKXCTV55.js.map +1 -0
- package/dist/{chunk-J3VKAEQP.js → chunk-V5TBKO5Q.js} +64 -14
- package/dist/chunk-V5TBKO5Q.js.map +1 -0
- package/dist/{chunk-I3UEM3FX.js → chunk-VUCPJBAG.js} +9 -4
- package/dist/chunk-VUCPJBAG.js.map +1 -0
- package/dist/{chunk-XIHFJVOD.js → chunk-WH2EB2SF.js} +3 -3
- package/dist/{chunk-XIHFJVOD.js.map → chunk-WH2EB2SF.js.map} +1 -1
- package/dist/{chunk-PBWMV5GM.js → chunk-WQ7DE5UC.js} +18 -4
- package/dist/chunk-WQ7DE5UC.js.map +1 -0
- package/dist/cli/index.js +199 -200
- package/dist/cli/index.js.map +1 -1
- package/dist/cli/src/commands/manage.d.ts +23 -32
- package/dist/cli/src/commands/manage.d.ts.map +1 -1
- package/dist/cli/src/commands/npx.d.ts +5 -3
- package/dist/cli/src/commands/npx.d.ts.map +1 -1
- package/dist/cli/src/commands/source.d.ts +7 -0
- package/dist/cli/src/commands/source.d.ts.map +1 -1
- package/dist/connectors/config.js +6 -6
- package/dist/connectors/index.js +7 -7
- package/dist/core/index.js +5 -5
- 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/manifest.d.ts +16 -0
- package/dist/core/src/manifest.d.ts.map +1 -1
- package/dist/core/src/models.d.ts +8 -2
- package/dist/core/src/models.d.ts.map +1 -1
- package/dist/core/src/repo-manager.d.ts +17 -2
- package/dist/core/src/repo-manager.d.ts.map +1 -1
- package/dist/core/src/walker.d.ts +4 -0
- package/dist/core/src/walker.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/workspace-config.js +3 -3
- package/dist/deploy/index.js +138 -42
- package/dist/deploy/index.js.map +1 -1
- package/dist/deploy/src/index.d.ts +4 -3
- package/dist/deploy/src/index.d.ts.map +1 -1
- package/dist/deploy/src/targets/container-runtime.d.ts.map +1 -1
- package/dist/deploy/src/targets/local.d.ts.map +1 -1
- package/dist/deploy/src/targets/nix.d.ts +36 -0
- package/dist/deploy/src/targets/nix.d.ts.map +1 -0
- package/dist/deploy/src/targets/process-handle.d.ts +34 -0
- package/dist/deploy/src/targets/process-handle.d.ts.map +1 -0
- package/dist/discovery/index.js +3 -3
- package/dist/{ensure-sources-SL2S4UEX.js → ensure-sources-REWWBH2K.js} +9 -9
- package/dist/{ensure-sources-SL2S4UEX.js.map → ensure-sources-REWWBH2K.js.map} +1 -1
- package/dist/library/index.js +4 -4
- package/dist/open-library-CT4VVESU.js +13 -0
- package/dist/{open-library-M4DB3D3J.js.map → open-library-CT4VVESU.js.map} +1 -1
- package/dist/plugin-registry/src/context.d.ts +30 -1
- package/dist/plugin-registry/src/context.d.ts.map +1 -1
- package/dist/plugin-registry/src/index.d.ts +1 -1
- package/dist/plugin-registry/src/index.d.ts.map +1 -1
- package/dist/{plugin-store-AJ3FGXIC.js → plugin-store-QS7TC5HY.js} +7 -7
- package/dist/{plugin-store-AJ3FGXIC.js.map → plugin-store-QS7TC5HY.js.map} +1 -1
- package/dist/plugins/src/catalog-source.d.ts +5 -0
- package/dist/plugins/src/catalog-source.d.ts.map +1 -1
- package/dist/runner/index.js +13 -12
- package/dist/runner/src/serve.d.ts +7 -0
- package/dist/runner/src/serve.d.ts.map +1 -1
- package/dist/sdk/asset-manager.js +7 -7
- package/dist/sdk/core.js +5 -5
- package/dist/sdk/index.js +13 -12
- package/dist/sdk/index.js.map +1 -1
- package/dist/sdk/runner.js +13 -12
- package/dist/sdk/transport/ws/client.js +2 -1
- package/dist/sdk/transport/ws/server.js +2 -1
- package/dist/sdk/transport/ws.js +4 -3
- package/dist/sdk/transport.js +4 -3
- package/dist/{setup-GBSQX7JF.js → setup-F6DGKL7J.js} +7 -7
- package/dist/{setup-GBSQX7JF.js.map → setup-F6DGKL7J.js.map} +1 -1
- package/dist/store-client-JP642EEI.js +14 -0
- package/dist/{store-client-5WBRUC5U.js.map → store-client-JP642EEI.js.map} +1 -1
- package/dist/transport/index.js +4 -3
- package/dist/transport/src/ws/auth.d.ts +34 -0
- package/dist/transport/src/ws/auth.d.ts.map +1 -0
- package/dist/transport/src/ws/client.d.ts +4 -0
- package/dist/transport/src/ws/client.d.ts.map +1 -1
- package/dist/transport/src/ws/index.d.ts +3 -2
- package/dist/transport/src/ws/index.d.ts.map +1 -1
- package/dist/transport/src/ws/server.d.ts +5 -0
- package/dist/transport/src/ws/server.d.ts.map +1 -1
- package/dist/transport/ws/client.js +2 -1
- package/dist/transport/ws/server.js +2 -1
- package/dist/transport/ws.js +4 -3
- package/dist/tui/index.js +13 -12
- package/dist/tui/index.js.map +1 -1
- package/dist/workspace-plugin/index.js +1 -1
- package/package.json +1 -1
- package/dist/asset-feeds-QXCSAJRN.js.map +0 -1
- package/dist/chunk-4AZKT2BU.js.map +0 -1
- package/dist/chunk-7PTP3SQJ.js.map +0 -1
- package/dist/chunk-I3UEM3FX.js.map +0 -1
- package/dist/chunk-J3VKAEQP.js.map +0 -1
- package/dist/chunk-JHF66MCK.js.map +0 -1
- package/dist/chunk-LT4DLEYE.js.map +0 -1
- package/dist/chunk-PBWMV5GM.js.map +0 -1
- package/dist/chunk-V3QMSM5I.js.map +0 -1
- package/dist/chunk-VCYXVP2S.js.map +0 -1
- package/dist/chunk-W2O5LWYU.js +0 -3
- package/dist/open-library-M4DB3D3J.js +0 -13
- package/dist/store-client-5WBRUC5U.js +0 -14
|
@@ -10,12 +10,16 @@ import type { AgentCommand, AgentEvent, ServerTransport } from "@skaile/workspac
|
|
|
10
10
|
*
|
|
11
11
|
* @param port - TCP port to bind to.
|
|
12
12
|
* @param onLog - Optional callback for internal log lines (client connect/disconnect events).
|
|
13
|
+
* @param authToken - When set, every upgrade must present this token via the
|
|
14
|
+
* `skaile-bearer.<base64url>` subprotocol or the handshake is rejected with
|
|
15
|
+
* HTTP 401. When unset, no auth is enforced (back-compatible default).
|
|
13
16
|
*
|
|
14
17
|
* @docLink packages/transport/dev-guide#websocket-server-transport
|
|
15
18
|
*/
|
|
16
19
|
export interface WebSocketServerOptions {
|
|
17
20
|
port: number;
|
|
18
21
|
onLog?: (line: string) => void;
|
|
22
|
+
authToken?: string;
|
|
19
23
|
}
|
|
20
24
|
/**
|
|
21
25
|
* `ServerTransport` implementation backed by the `ws` package on top of a
|
|
@@ -46,6 +50,7 @@ export declare class WebSocketServerTransport implements ServerTransport {
|
|
|
46
50
|
private activeClient;
|
|
47
51
|
private readonly port;
|
|
48
52
|
private readonly log;
|
|
53
|
+
private readonly authToken?;
|
|
49
54
|
private readonly commandHandlers;
|
|
50
55
|
private readonly connectHandlers;
|
|
51
56
|
private readonly disconnectHandlers;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../../../transport/src/ws/server.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAGH,OAAO,KAAK,EAAE,YAAY,EAAE,UAAU,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;
|
|
1
|
+
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../../../transport/src/ws/server.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAGH,OAAO,KAAK,EAAE,YAAY,EAAE,UAAU,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;AAI1F;;;;;;;;;;GAUG;AACH,MAAM,WAAW,sBAAsB;IACrC,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,CAAC,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;IAC/B,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAoDD;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,qBAAa,wBAAyB,YAAW,eAAe;IAC9D,OAAO,CAAC,UAAU,CAA2B;IAC7C,OAAO,CAAC,GAAG,CAAgC;IAC3C,OAAO,CAAC,YAAY,CAA0B;IAC9C,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAS;IAC9B,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAyB;IAC7C,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAS;IAEpC,OAAO,CAAC,QAAQ,CAAC,eAAe,CAA8C;IAC9E,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAyB;IACzD,OAAO,CAAC,QAAQ,CAAC,kBAAkB,CAAyB;gBAEhD,OAAO,EAAE,sBAAsB;IAM3C,+CAA+C;IACzC,MAAM,IAAI,OAAO,CAAC,IAAI,CAAC;IAyC7B,OAAO,CAAC,gBAAgB;IA6DxB;;;;;OAKG;IACG,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAiB5B,IAAI,CAAC,KAAK,EAAE,UAAU,GAAG,IAAI;IAI7B,SAAS,CAAC,OAAO,EAAE,CAAC,OAAO,EAAE,YAAY,KAAK,IAAI,GAAG,MAAM,IAAI;IAO/D,SAAS,CAAC,OAAO,EAAE,MAAM,IAAI,GAAG,MAAM,IAAI;IAO1C,YAAY,CAAC,OAAO,EAAE,MAAM,IAAI,GAAG,MAAM,IAAI;IAO7C,IAAI,SAAS,IAAI,OAAO,CAEvB;IAED,OAAO,CAAC,OAAO;CAUhB"}
|
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
export { WebSocketClientTransport } from '../../chunk-
|
|
1
|
+
export { WebSocketClientTransport } from '../../chunk-GTS2FODO.js';
|
|
2
|
+
import '../../chunk-32NA4TVC.js';
|
|
2
3
|
import '../../chunk-NSBPE2FW.js';
|
|
3
4
|
//# sourceMappingURL=client.js.map
|
|
4
5
|
//# sourceMappingURL=client.js.map
|
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
export { WebSocketServerTransport } from '../../chunk-
|
|
1
|
+
export { WebSocketServerTransport } from '../../chunk-WQ7DE5UC.js';
|
|
2
|
+
import '../../chunk-32NA4TVC.js';
|
|
2
3
|
import '../../chunk-NSBPE2FW.js';
|
|
3
4
|
//# sourceMappingURL=server.js.map
|
|
4
5
|
//# sourceMappingURL=server.js.map
|
package/dist/transport/ws.js
CHANGED
|
@@ -1,6 +1,7 @@
|
|
|
1
|
-
import '../chunk-
|
|
2
|
-
export { WebSocketClientTransport } from '../chunk-
|
|
3
|
-
export { WebSocketServerTransport } from '../chunk-
|
|
1
|
+
import '../chunk-7QBNJTTQ.js';
|
|
2
|
+
export { WebSocketClientTransport } from '../chunk-GTS2FODO.js';
|
|
3
|
+
export { WebSocketServerTransport } from '../chunk-WQ7DE5UC.js';
|
|
4
|
+
export { BEARER_SUBPROTOCOL_PREFIX, decodeBearerFromHeader, encodeBearerSubprotocol, tokensEqual } from '../chunk-32NA4TVC.js';
|
|
4
5
|
import '../chunk-NSBPE2FW.js';
|
|
5
6
|
//# sourceMappingURL=ws.js.map
|
|
6
7
|
//# sourceMappingURL=ws.js.map
|
package/dist/tui/index.js
CHANGED
|
@@ -1,10 +1,11 @@
|
|
|
1
|
-
import { createAgentSession } from '../chunk-
|
|
2
|
-
import '../chunk-
|
|
1
|
+
import { createAgentSession } from '../chunk-SKXCTV55.js';
|
|
2
|
+
import '../chunk-NDD5VMN5.js';
|
|
3
3
|
import '../chunk-X5YPJV4N.js';
|
|
4
4
|
import '../chunk-O7SG5PC2.js';
|
|
5
|
-
import '../chunk-
|
|
6
|
-
import '../chunk-
|
|
7
|
-
import '../chunk-
|
|
5
|
+
import '../chunk-7QBNJTTQ.js';
|
|
6
|
+
import '../chunk-GTS2FODO.js';
|
|
7
|
+
import '../chunk-WQ7DE5UC.js';
|
|
8
|
+
import '../chunk-32NA4TVC.js';
|
|
8
9
|
import '../chunk-W3UDISS2.js';
|
|
9
10
|
import '../chunk-TDSRLMDB.js';
|
|
10
11
|
import '../chunk-M5TE6YI5.js';
|
|
@@ -13,21 +14,21 @@ import '../chunk-KOVLSBXK.js';
|
|
|
13
14
|
import '../chunk-RRVQAE5D.js';
|
|
14
15
|
import '../chunk-6VTG73UY.js';
|
|
15
16
|
import '../chunk-LV2HPH3C.js';
|
|
16
|
-
import '../chunk-
|
|
17
|
-
import { loadConnectorDeclarations } from '../chunk-
|
|
17
|
+
import '../chunk-OJN25VJO.js';
|
|
18
|
+
import { loadConnectorDeclarations } from '../chunk-NBJ5TOEC.js';
|
|
18
19
|
import '../chunk-6MB7CRME.js';
|
|
19
20
|
import '../chunk-QAVZOJCV.js';
|
|
20
21
|
import '../chunk-6E6PKKAD.js';
|
|
21
22
|
import '../chunk-ICS76R4T.js';
|
|
22
23
|
import '../chunk-GZWJGNNN.js';
|
|
23
24
|
import '../chunk-FVTV7M76.js';
|
|
24
|
-
import { resolveSettings } from '../chunk-
|
|
25
|
+
import { resolveSettings } from '../chunk-ETMUGBHF.js';
|
|
25
26
|
import '../chunk-K5GBV4SA.js';
|
|
26
27
|
import '../chunk-KLNL7QHN.js';
|
|
27
|
-
import '../chunk-
|
|
28
|
-
import { resolveAgentDir } from '../chunk-
|
|
29
|
-
import '../chunk-
|
|
30
|
-
import '../chunk-
|
|
28
|
+
import '../chunk-PFOXL4SH.js';
|
|
29
|
+
import { resolveAgentDir } from '../chunk-V5TBKO5Q.js';
|
|
30
|
+
import '../chunk-K7WPR77X.js';
|
|
31
|
+
import '../chunk-VUCPJBAG.js';
|
|
31
32
|
import '../chunk-JKNWJ64A.js';
|
|
32
33
|
import '../chunk-O4JH3KUE.js';
|
|
33
34
|
import '../chunk-24UIWON4.js';
|
package/dist/tui/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../tui/src/telemetry.ts","../../tui/src/repl-renderer.ts","../../tui/src/repl-session.ts","../../tui/src/repl.ts"],"names":["text","pc","resolve"],"mappings":"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","file":"index.js","sourcesContent":["/**\n * Local telemetry sink that emits structured events to the TUI log panel.\n *\n * Implements `TelemetryProvider` so it can be injected into `createAgentSession`\n * without any HTTP or OpenTelemetry overhead. Instead of exporting spans over a\n * network, it converts each lifecycle event (trace start/end, span start/end,\n * generation, metric) into a {@link TelemetryLogEntry} and forwards it via the\n * `onEvent` callback. The REPL uses this to optionally display timing and token\n * cost inline when `--verbose` is active.\n *\n * @docLink packages/tui/concepts#tui-telemetry-provider\n */\n\nimport { randomUUID } from \"node:crypto\";\nimport type {\n Attributes,\n GenerationEvent,\n Span,\n SpanKind,\n SpanOptions,\n SpanResult,\n TelemetryProvider,\n Trace,\n TraceOptions,\n} from \"@skaile/workspaces/telemetry\";\n\n// ── Telemetry log entry types ────────────────────────────────────────────────\n\n/**\n * A structured telemetry event emitted by {@link TuiTelemetryProvider}.\n *\n * Discriminated union over `kind`. The TUI renders a subset of these (generation,\n * span_end for tool/turn, trace lifecycle) and suppresses the rest. Used as input\n * to {@link formatTelemetryEntry} and {@link renderLog}.\n *\n * @docLink packages/tui/concepts#tui-telemetry-provider\n */\nexport type TelemetryLogEntry =\n | { kind: \"trace_start\"; traceId: string; name: string; traceKind: string }\n | { kind: \"trace_end\"; traceId: string; status: string; durationMs: number }\n | { kind: \"span_start\"; spanId: string; name: string; spanKind: SpanKind }\n | {\n kind: \"span_end\";\n spanId: string;\n name: string;\n spanKind: SpanKind;\n status: string;\n durationMs: number;\n }\n | {\n kind: \"generation\";\n model: string;\n inputTokens: number;\n outputTokens: number;\n costUsd: number;\n durationMs: number;\n }\n | { kind: \"metric\"; name: string; value: number };\n\n// ── Internal span tracking ───────────────────────────────────────────────────\n\ninterface SpanRecord {\n spanId: string;\n traceId: string;\n name: string;\n spanKind: SpanKind;\n startTime: number;\n}\n\n// ── Provider ─────────────────────────────────────────────────────────────────\n\nexport class TuiTelemetryProvider implements TelemetryProvider {\n private readonly onEvent: (entry: TelemetryLogEntry) => void;\n private readonly spans = new Map<string, SpanRecord>();\n private readonly traces = new Map<string, { name: string; startTime: number }>();\n\n constructor(onEvent: (entry: TelemetryLogEntry) => void) {\n this.onEvent = onEvent;\n }\n\n startTrace(opts: TraceOptions): Trace {\n const traceId = randomUUID().replace(/-/g, \"\");\n const rootSpan: Span = { spanId: randomUUID().slice(0, 16), traceId };\n\n this.traces.set(traceId, { name: opts.name, startTime: Date.now() });\n this.onEvent({ kind: \"trace_start\", traceId, name: opts.name, traceKind: opts.kind });\n\n return { traceId, rootSpan };\n }\n\n endTrace(trace: Trace, result?: SpanResult): void {\n const record = this.traces.get(trace.traceId);\n if (record) {\n const durationMs = Date.now() - record.startTime;\n this.onEvent({\n kind: \"trace_end\",\n traceId: trace.traceId,\n status: result?.status ?? \"ok\",\n durationMs,\n });\n this.traces.delete(trace.traceId);\n }\n }\n\n startSpan(parent: Trace | Span, opts: SpanOptions): Span {\n const spanId = randomUUID().slice(0, 16);\n const traceId = parent.traceId;\n const span: Span = { spanId, traceId };\n\n this.spans.set(spanId, {\n spanId,\n traceId,\n name: opts.name,\n spanKind: opts.kind,\n startTime: Date.now(),\n });\n\n this.onEvent({ kind: \"span_start\", spanId, name: opts.name, spanKind: opts.kind });\n return span;\n }\n\n endSpan(span: Span, result?: SpanResult): void {\n const record = this.spans.get(span.spanId);\n if (record) {\n const durationMs = Date.now() - record.startTime;\n this.onEvent({\n kind: \"span_end\",\n spanId: span.spanId,\n name: record.name,\n spanKind: record.spanKind,\n status: result?.status ?? \"ok\",\n durationMs,\n });\n this.spans.delete(span.spanId);\n }\n }\n\n addEvent(_span: Span, _name: string, _attrs?: Attributes): void {\n // Silently ignored — events are not surfaced in TUI\n }\n\n logGeneration(_span: Span, generation: GenerationEvent): void {\n this.onEvent({\n kind: \"generation\",\n model: generation.model,\n inputTokens: generation.inputTokens ?? 0,\n outputTokens: generation.outputTokens ?? 0,\n costUsd: generation.costUsd ?? 0,\n durationMs: generation.durationMs,\n });\n }\n\n recordMetric(name: string, value: number, _tags?: Record<string, string>): void {\n this.onEvent({ kind: \"metric\", name, value });\n }\n\n async flush(): Promise<void> {}\n async shutdown(): Promise<void> {\n this.spans.clear();\n this.traces.clear();\n }\n}\n\n// ── Log formatting ───────────────────────────────────────────────────────────\n\n/**\n * Format a {@link TelemetryLogEntry} into a human-readable string for the log panel.\n *\n * Produces plain (no ANSI) strings suitable for structured logging or non-terminal\n * output. The format mirrors {@link renderLog} but without colour codes: generation\n * entries include model, token counts, cost, and duration; span_end entries for\n * `tool_call` and `agent_turn` kinds include name, status, and duration; trace\n * lifecycle entries include name and elapsed time. Returns `null` for entry kinds\n * that should not be displayed.\n *\n * @param entry - The structured telemetry event to format.\n * @returns Human-readable string, or `null` if the entry should be suppressed.\n *\n * @docLink packages/tui/concepts#tui-telemetry-provider\n */\nexport function formatTelemetryEntry(entry: TelemetryLogEntry): string | null {\n switch (entry.kind) {\n case \"generation\": {\n const tokens = `${entry.inputTokens}↑ ${entry.outputTokens}↓`;\n const cost = entry.costUsd > 0 ? ` $${entry.costUsd.toFixed(4)}` : \"\";\n const dur = entry.durationMs > 0 ? ` ${(entry.durationMs / 1000).toFixed(1)}s` : \"\";\n return `[gen] ${entry.model} ${tokens}${cost}${dur}`;\n }\n case \"span_end\":\n if (entry.spanKind === \"tool_call\") {\n return `[tool] ${entry.name} ${entry.status} ${entry.durationMs}ms`;\n }\n if (entry.spanKind === \"agent_turn\") {\n return `[turn] ${entry.name} ${(entry.durationMs / 1000).toFixed(1)}s`;\n }\n return null;\n case \"trace_start\":\n return `[trace] ${entry.name} started`;\n case \"trace_end\":\n return `[trace] done ${(entry.durationMs / 1000).toFixed(1)}s`;\n default:\n return null;\n }\n}\n","import pc from \"picocolors\";\nimport type { OutputLine } from \"@skaile/workspaces\";\nimport type { TelemetryLogEntry } from \"./telemetry.js\";\n\n/**\n * Render a single OutputLine to an ANSI-colored string for terminal display.\n *\n * Each `OutputLine` variant is mapped to a distinct colour: user input (cyan/bold),\n * agent text (plain), tool calls (blue italic), questions (yellow), errors (red),\n * status/system messages (dim italic), subagent lifecycle (blue/green italic), and\n * separator lines (dim dashes). Returns an empty string for unrecognised variants.\n *\n * @param line - The typed output line to render.\n * @returns ANSI-coloured string ready for terminal output.\n *\n * @docLink packages/tui/concepts#repl-renderer\n */\nexport function renderLine(line: OutputLine): string {\n switch (line.type) {\n case \"user\":\n return pc.cyan(pc.bold(`> ${line.content}`));\n case \"text\":\n return line.content;\n case \"tool\":\n return pc.blue(pc.italic(` Running: \\`${line.name}\\`...`));\n case \"question\":\n return pc.yellow(`?? ${line.text}\\n Reply below.`);\n case \"error\":\n return pc.red(`Error: ${line.message}`);\n case \"status\":\n return pc.dim(pc.italic(`${line.phase}...`));\n case \"system\":\n return pc.dim(pc.italic(line.message));\n case \"subagent\": {\n const icon = line.status === \"started\" ? \"▶\" : \"✓\";\n const color = line.status === \"started\" ? pc.blue : pc.green;\n return color(pc.italic(` ${icon} ${line.name}: ${line.description}`));\n }\n case \"separator\":\n return pc.dim(\"─\".repeat(60));\n default:\n return \"\";\n }\n}\n\n/**\n * Render a telemetry log entry to an ANSI string, or null to suppress.\n *\n * Formats `generation` entries with model name, token counts, cost, and duration;\n * `span_end` entries for `tool_call` and `agent_turn` kinds; and `trace_start` /\n * `trace_end` lifecycle entries. All output is styled dim for visual de-emphasis.\n * Returns `null` for entry kinds that should not appear in the log panel.\n *\n * @param entry - The structured telemetry event to render.\n * @returns Dim ANSI string, or `null` if the entry should be suppressed.\n *\n * @docLink packages/tui/concepts#repl-renderer\n */\nexport function renderLog(entry: TelemetryLogEntry): string | null {\n switch (entry.kind) {\n case \"generation\": {\n const tokens = `${entry.inputTokens}↑ ${entry.outputTokens}↓`;\n const cost = entry.costUsd > 0 ? ` $${entry.costUsd.toFixed(4)}` : \"\";\n const dur = entry.durationMs > 0 ? ` ${(entry.durationMs / 1000).toFixed(1)}s` : \"\";\n return pc.dim(`[gen] ${entry.model} ${tokens}${cost}${dur}`);\n }\n case \"span_end\":\n if (entry.spanKind === \"tool_call\") {\n return pc.dim(`[tool] ${entry.name} ${entry.status} ${entry.durationMs}ms`);\n }\n if (entry.spanKind === \"agent_turn\") {\n return pc.dim(`[turn] ${entry.name} ${(entry.durationMs / 1000).toFixed(1)}s`);\n }\n return null;\n case \"trace_start\":\n return pc.dim(`[trace] ${entry.name} started`);\n case \"trace_end\":\n return pc.dim(`[trace] done ${(entry.durationMs / 1000).toFixed(1)}s`);\n default:\n return null;\n }\n}\n","/**\n * Slim event bridge: maps AgentDriver events to rendered console lines.\n * No React, no ink, no UI state — just event → string → callback.\n */\n\nimport type { AgentEvent } from \"@skaile/workspaces/bridge\";\nimport type { AgentSession } from \"@skaile/workspaces/runner\";\nimport type { OutputLine, SessionState } from \"@skaile/workspaces\";\nimport { renderLine, renderLog } from \"./repl-renderer.js\";\nimport type { TelemetryLogEntry } from \"./telemetry.js\";\n\n/**\n * Handle returned by {@link createReplSession} for controlling an active REPL session.\n *\n * `state` reflects the current lifecycle state (`idle`, `active`, or `stopped`).\n * Call `send(text)` to forward a user message to the agent driver; the state\n * transitions to `active` until the driver emits `agent_end`. Call `stop()` to\n * terminate the driver and dispose of the underlying agent session.\n *\n * @docLink packages/tui/concepts#repl-session\n */\nexport interface ReplSessionHandle {\n readonly state: SessionState;\n send(text: string): void;\n stop(): void;\n}\n\n/**\n * Options for {@link createReplSession}.\n *\n * @param session - The agent session returned by `createAgentSession`.\n * @param onLine - Callback invoked with an ANSI-rendered string for each output line.\n * @param onLog - Callback invoked with a log string for each telemetry event (if verbose).\n * @param onStateChange - Callback invoked whenever the session state changes.\n *\n * @docLink packages/tui/concepts#repl-session\n */\nexport interface CreateReplSessionOpts {\n session: AgentSession;\n onLine?: (line: string) => void;\n onLog?: (line: string) => void;\n onStateChange?: (state: SessionState) => void;\n}\n\n/**\n * Create a slim event bridge that maps `AgentEvent` stream to rendered console lines.\n *\n * Subscribes to `agent-event` on the session driver and converts each event type\n * (`message_update`, `tool_call`, `error`, `agent_end`, `turn_end`, `subagent`) to\n * an `OutputLine`, renders it via {@link renderLine}, and forwards the string to\n * `opts.onLine`. Text content is buffered internally and flushed line-by-line as\n * newlines arrive. Returns a {@link ReplSessionHandle} for sending messages and\n * stopping the session.\n *\n * @param opts - Session, output callbacks, and state-change hook.\n * @returns Handle for sending messages and observing session state.\n *\n * @docLink packages/tui/concepts#repl-session\n */\nexport function createReplSession(opts: CreateReplSessionOpts): ReplSessionHandle {\n const { session, onLine, onLog, onStateChange } = opts;\n\n let state: SessionState = \"idle\";\n let textBuffer = \"\";\n\n function emit(outputLine: OutputLine) {\n const rendered = renderLine(outputLine);\n if (rendered) onLine?.(rendered);\n }\n\n function setState(s: SessionState) {\n state = s;\n onStateChange?.(s);\n }\n\n session.driver.on(\"agent-event\", (event: AgentEvent) => {\n switch (event.type) {\n case \"message_update\": {\n const delta = (event as any)._textDelta as string | undefined;\n if (delta) {\n textBuffer += delta;\n const lines = textBuffer.split(\"\\n\");\n if (lines.length > 1) {\n for (let i = 0; i < lines.length - 1; i++) {\n emit({ type: \"text\", content: lines[i]! });\n }\n textBuffer = lines[lines.length - 1]!;\n }\n }\n break;\n }\n\n case \"tool_call\": {\n const name = (event as any).tool?.name ?? (event as any).name ?? \"unknown\";\n emit({ type: \"tool\", name });\n break;\n }\n\n case \"error\": {\n const msg = (event as any).error ?? \"Unknown error\";\n emit({ type: \"error\", message: String(msg) });\n if ((event as any).fatal) setState(\"stopped\");\n break;\n }\n\n case \"agent_end\":\n case \"turn_end\": {\n if (textBuffer.trim()) {\n emit({ type: \"text\", content: textBuffer });\n }\n textBuffer = \"\";\n emit({ type: \"separator\" });\n if (event.type === \"agent_end\") setState(\"idle\");\n break;\n }\n\n case \"session_info\":\n break;\n\n default: {\n const raw = event as any;\n if (raw.type === \"subagent\") {\n const status = raw.status === \"finished\" ? \"finished\" : \"started\";\n emit({\n type: \"subagent\",\n name: raw.name ?? \"subagent\",\n status,\n description: raw.description ?? \"\",\n });\n }\n break;\n }\n }\n });\n\n return {\n get state() {\n return state;\n },\n\n send(text: string) {\n if (state === \"stopped\") return;\n setState(\"active\");\n textBuffer = \"\";\n session.driver.prompt(text).catch((err: Error) => {\n emit({ type: \"error\", message: err.message });\n setState(\"idle\");\n });\n },\n\n stop() {\n setState(\"stopped\");\n session.driver.kill();\n session.dispose();\n },\n };\n}\n\n/**\n * Create a telemetry sink that pipes {@link TelemetryLogEntry} events to the log callback.\n *\n * Wraps {@link renderLog} so that only displayable entries (generation, span_end for\n * tool/turn, trace lifecycle) are forwarded. Entries that `renderLog` returns `null`\n * for are silently suppressed. The returned function is used in `startRepl` to drain\n * buffered telemetry and to wire up live forwarding during the session.\n *\n * @param onLog - Callback invoked with an ANSI-rendered log string.\n * @returns A function that accepts a `TelemetryLogEntry` and routes it to `onLog`.\n *\n * @docLink packages/tui/concepts#repl-session\n */\nexport function createTelemetrySink(\n onLog: (line: string) => void,\n): (entry: TelemetryLogEntry) => void {\n return (entry) => {\n const text = renderLog(entry);\n if (text) onLog(text);\n };\n}\n","/**\n * Simple @clack-based agent REPL.\n * No React, no ink, no alt-screen — just a scrolling terminal.\n *\n * Design: clack's text() blocks until user submits, so agent output\n * flows naturally between prompts — no readline interleaving issues.\n * Loop: await text() → send → await idle → loop\n */\n\nimport * as p from \"@clack/prompts\";\nimport { resolve, join, basename, dirname, parse } from \"node:path\";\nimport { existsSync, readFileSync, statSync, readdirSync } from \"node:fs\";\nimport type { Dirent } from \"node:fs\";\nimport pc from \"picocolors\";\nimport { createAgentSession } from \"@skaile/workspaces/runner\";\nimport { resolveAgentDir, resolveSettings } from \"@skaile/workspaces/core\";\nimport { loadConnectorDeclarations } from \"@skaile/workspaces/connectors\";\nimport { TuiTelemetryProvider } from \"./telemetry.js\";\nimport type { TelemetryLogEntry } from \"./telemetry.js\";\nimport { createReplSession, createTelemetrySink } from \"./repl-session.js\";\n\nfunction findAiAssetsRoot(startDir: string): string | undefined {\n let current = resolve(startDir);\n const root = parse(current).root;\n while (current !== root) {\n const candidate = join(current, \"ai-assets\");\n try {\n if (statSync(candidate).isDirectory()) return candidate;\n } catch {}\n const parent = dirname(current);\n if (parent === current) break;\n current = parent;\n }\n return undefined;\n}\n\nfunction countSkills(projectDir: string): number {\n const aiAssetsRoot = findAiAssetsRoot(projectDir);\n if (!aiAssetsRoot) return 0;\n let count = 0;\n const walk = (dir: string) => {\n let entries: Dirent[];\n try {\n entries = readdirSync(dir, { withFileTypes: true });\n } catch {\n return;\n }\n if (entries.some((e) => e.isFile() && e.name === \"SKILL.md\")) {\n count++;\n return;\n }\n for (const e of entries) {\n if (e.isDirectory()) walk(join(dir, e.name));\n }\n };\n walk(aiAssetsRoot);\n return count;\n}\n\n/**\n * Options for {@link startRepl}.\n *\n * @param projectDir - Absolute path to the project root. Used to locate the agent dir, settings, and skill count.\n * @param cwd - Working directory passed to the agent process. Defaults to `projectDir`.\n * @param driver - Agent driver override (`\"omp\"` or `\"claude-sdk\"`). Falls back to settings resolution.\n * @param model - Model name passed to the agent driver.\n * @param provider - Provider name passed to the agent driver.\n * @param resume - When `true`, load the last session ID from `.skaile/last_session` and resume it.\n * @param resumeSessionId - Explicit session ID to resume. Takes precedence over `resume`.\n * @param verbose - When `true`, show telemetry log lines inline between prompts. Default: `false`.\n *\n * @docLink packages/tui/concepts#start-repl\n */\nexport interface ReplOptions {\n projectDir: string;\n cwd?: string;\n driver?: string;\n model?: string;\n provider?: string;\n resume?: boolean;\n resumeSessionId?: string;\n /** When true, show telemetry log lines inline. Default: false. */\n verbose?: boolean;\n}\n\n/**\n * Start the scrolling REPL for an interactive agent conversation.\n *\n * Initialises the agent session (including OMP extension injection when the driver is `omp`),\n * prints a banner with project metadata, then enters a `@clack/prompts` input loop.\n * Each iteration awaits user input, forwards it to the agent via {@link createReplSession},\n * and blocks until the agent signals `idle` or `stopped`. Slash commands (`/help`, `/log`,\n * `/quit`) are handled before the message reaches the agent.\n *\n * @param opts - Configuration for the REPL session.\n * @returns Resolves when the session ends (user quit or fatal error).\n *\n * @docLink packages/tui/concepts#start-repl\n */\nexport async function startRepl(opts: ReplOptions): Promise<void> {\n const projectDir = resolve(opts.projectDir);\n const agentCwd = opts.cwd ? resolve(opts.cwd) : projectDir;\n\n // ── OMP workspace extension injection ──────────────────────────────\n const settings = await resolveSettings(projectDir, { driver: opts.driver });\n const resolvedDriver = settings.driver ?? \"omp\";\n let ompCleanup: (() => void) | undefined;\n\n if (resolvedDriver === \"omp\") {\n const fsAsync = await import(\"node:fs/promises\");\n const pathMod = await import(\"node:path\");\n const { WorkspacePlugin } = await import(\"@skaile/workspaces/workspace-plugin\");\n\n const plugin = new WorkspacePlugin({ projectDir });\n const src = await plugin.buildOmpExtensionSource();\n const extPath = pathMod.default.join(agentCwd, \".omp\", \"extensions\", \"skaile.ts\");\n await fsAsync.default.mkdir(pathMod.default.dirname(extPath), { recursive: true });\n await fsAsync.default.writeFile(extPath, src);\n ompCleanup = () => {\n fsAsync.default.unlink(extPath).catch(() => {});\n };\n }\n\n // ── Session resume ─────────────────────────────────────────────────\n const SESSION_FILE = join(projectDir, \".skaile\", \"last_session\");\n let resumeSessionId = opts.resumeSessionId;\n if (!resumeSessionId && opts.resume) {\n try {\n if (existsSync(SESSION_FILE)) {\n resumeSessionId = readFileSync(SESSION_FILE, \"utf-8\").trim() || undefined;\n }\n } catch {}\n }\n\n // ── Telemetry ──────────────────────────────────────────────────────\n const telemetryBuffer: TelemetryLogEntry[] = [];\n const telemetry = new TuiTelemetryProvider((entry) => {\n telemetryBuffer.push(entry);\n });\n const trace = telemetry.startTrace({ name: \"skaile-repl\", kind: \"chat_session\" });\n\n // ── Create agent session ───────────────────────────────────────────\n const agentDir = resolveAgentDir(projectDir) ?? undefined;\n const session = await createAgentSession({\n projectDir,\n cwd: agentCwd !== projectDir ? agentCwd : undefined,\n agentDir,\n driver: opts.driver ?? resolvedDriver,\n model: opts.model,\n provider: opts.provider,\n resumeSessionId,\n telemetry,\n trace,\n onLog: () => {},\n });\n\n await session.driver.start();\n\n // ── Output + state tracking ────────────────────────────────────────\n let showLog = opts.verbose ?? false;\n let idleResolve: (() => void) | undefined;\n\n const write = (line: string) => {\n p.log.message(line);\n };\n\n const writeLog = (line: string) => {\n if (showLog) p.log.step(line);\n };\n\n // ── Create REPL session ────────────────────────────────────────────\n const replSession = createReplSession({\n session,\n onLine: write,\n onLog: writeLog,\n onStateChange: (state) => {\n if (state === \"idle\" || state === \"stopped\") {\n idleResolve?.();\n }\n },\n });\n\n // Drain buffered telemetry\n const sink = createTelemetrySink(writeLog);\n for (const entry of telemetryBuffer) sink(entry);\n telemetryBuffer.length = 0;\n\n // ── Banner ─────────────────────────────────────────────────────────\n const displayModel = session.model ?? opts.model ?? \"default\";\n const displayDriver = session.driverType ?? resolvedDriver;\n\n const skillCount = countSkills(projectDir);\n let connectorCount = 0;\n try {\n connectorCount = loadConnectorDeclarations(projectDir).length;\n } catch {}\n const agentName = agentDir ? basename(agentDir) : \"(none)\";\n\n p.intro(`${pc.bold(\"skaile repl\")} ${pc.dim(`— ${displayDriver} / ${displayModel}`)}`);\n p.log.info(pc.dim(`Project: ${projectDir}`));\n p.log.info(pc.dim(`Agent: ${agentName} · ${skillCount} skills · ${connectorCount} connectors`));\n if (resumeSessionId) p.log.info(pc.dim(`Resumed: ${resumeSessionId.slice(0, 12)}...`));\n p.log.info(pc.dim(\"Type /help for commands, Ctrl+C to quit.\"));\n\n // ── REPL loop ──────────────────────────────────────────────────────\n function shutdown() {\n replSession.stop();\n telemetry.endTrace(trace, { status: \"ok\" });\n void telemetry.shutdown();\n ompCleanup?.();\n p.outro(\"Bye.\");\n process.exit(0);\n }\n\n process.on(\"SIGINT\", shutdown);\n process.on(\"SIGTERM\", shutdown);\n\n while (replSession.state !== \"stopped\") {\n const input = await p.text({\n message: pc.gray(\">>>\"),\n placeholder: \"\",\n });\n\n if (p.isCancel(input)) {\n shutdown();\n return;\n }\n\n const trimmed = (input as string).trim();\n if (!trimmed) continue;\n\n if (trimmed === \"/quit\" || trimmed === \"/q\") {\n shutdown();\n return;\n }\n if (trimmed === \"/help\" || trimmed === \"/h\" || trimmed === \"/?\") {\n p.log.info(\n [\n \" /help, /h — this help\",\n \" /log — toggle verbose log output\",\n \" /quit, /q — exit\",\n \" Ctrl+C — exit\",\n ].join(\"\\n\"),\n );\n continue;\n }\n if (trimmed === \"/log\") {\n showLog = !showLog;\n p.log.info(`Log output: ${showLog ? \"on\" : \"off\"}`);\n continue;\n }\n\n const idlePromise = new Promise<void>((resolve) => {\n idleResolve = resolve;\n });\n replSession.send(trimmed);\n await idlePromise;\n }\n\n shutdown();\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["../../tui/src/telemetry.ts","../../tui/src/repl-renderer.ts","../../tui/src/repl-session.ts","../../tui/src/repl.ts"],"names":["text","pc","resolve"],"mappings":"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","file":"index.js","sourcesContent":["/**\n * Local telemetry sink that emits structured events to the TUI log panel.\n *\n * Implements `TelemetryProvider` so it can be injected into `createAgentSession`\n * without any HTTP or OpenTelemetry overhead. Instead of exporting spans over a\n * network, it converts each lifecycle event (trace start/end, span start/end,\n * generation, metric) into a {@link TelemetryLogEntry} and forwards it via the\n * `onEvent` callback. The REPL uses this to optionally display timing and token\n * cost inline when `--verbose` is active.\n *\n * @docLink packages/tui/concepts#tui-telemetry-provider\n */\n\nimport { randomUUID } from \"node:crypto\";\nimport type {\n Attributes,\n GenerationEvent,\n Span,\n SpanKind,\n SpanOptions,\n SpanResult,\n TelemetryProvider,\n Trace,\n TraceOptions,\n} from \"@skaile/workspaces/telemetry\";\n\n// ── Telemetry log entry types ────────────────────────────────────────────────\n\n/**\n * A structured telemetry event emitted by {@link TuiTelemetryProvider}.\n *\n * Discriminated union over `kind`. The TUI renders a subset of these (generation,\n * span_end for tool/turn, trace lifecycle) and suppresses the rest. Used as input\n * to {@link formatTelemetryEntry} and {@link renderLog}.\n *\n * @docLink packages/tui/concepts#tui-telemetry-provider\n */\nexport type TelemetryLogEntry =\n | { kind: \"trace_start\"; traceId: string; name: string; traceKind: string }\n | { kind: \"trace_end\"; traceId: string; status: string; durationMs: number }\n | { kind: \"span_start\"; spanId: string; name: string; spanKind: SpanKind }\n | {\n kind: \"span_end\";\n spanId: string;\n name: string;\n spanKind: SpanKind;\n status: string;\n durationMs: number;\n }\n | {\n kind: \"generation\";\n model: string;\n inputTokens: number;\n outputTokens: number;\n costUsd: number;\n durationMs: number;\n }\n | { kind: \"metric\"; name: string; value: number };\n\n// ── Internal span tracking ───────────────────────────────────────────────────\n\ninterface SpanRecord {\n spanId: string;\n traceId: string;\n name: string;\n spanKind: SpanKind;\n startTime: number;\n}\n\n// ── Provider ─────────────────────────────────────────────────────────────────\n\nexport class TuiTelemetryProvider implements TelemetryProvider {\n private readonly onEvent: (entry: TelemetryLogEntry) => void;\n private readonly spans = new Map<string, SpanRecord>();\n private readonly traces = new Map<string, { name: string; startTime: number }>();\n\n constructor(onEvent: (entry: TelemetryLogEntry) => void) {\n this.onEvent = onEvent;\n }\n\n startTrace(opts: TraceOptions): Trace {\n const traceId = randomUUID().replace(/-/g, \"\");\n const rootSpan: Span = { spanId: randomUUID().slice(0, 16), traceId };\n\n this.traces.set(traceId, { name: opts.name, startTime: Date.now() });\n this.onEvent({ kind: \"trace_start\", traceId, name: opts.name, traceKind: opts.kind });\n\n return { traceId, rootSpan };\n }\n\n endTrace(trace: Trace, result?: SpanResult): void {\n const record = this.traces.get(trace.traceId);\n if (record) {\n const durationMs = Date.now() - record.startTime;\n this.onEvent({\n kind: \"trace_end\",\n traceId: trace.traceId,\n status: result?.status ?? \"ok\",\n durationMs,\n });\n this.traces.delete(trace.traceId);\n }\n }\n\n startSpan(parent: Trace | Span, opts: SpanOptions): Span {\n const spanId = randomUUID().slice(0, 16);\n const traceId = parent.traceId;\n const span: Span = { spanId, traceId };\n\n this.spans.set(spanId, {\n spanId,\n traceId,\n name: opts.name,\n spanKind: opts.kind,\n startTime: Date.now(),\n });\n\n this.onEvent({ kind: \"span_start\", spanId, name: opts.name, spanKind: opts.kind });\n return span;\n }\n\n endSpan(span: Span, result?: SpanResult): void {\n const record = this.spans.get(span.spanId);\n if (record) {\n const durationMs = Date.now() - record.startTime;\n this.onEvent({\n kind: \"span_end\",\n spanId: span.spanId,\n name: record.name,\n spanKind: record.spanKind,\n status: result?.status ?? \"ok\",\n durationMs,\n });\n this.spans.delete(span.spanId);\n }\n }\n\n addEvent(_span: Span, _name: string, _attrs?: Attributes): void {\n // Silently ignored — events are not surfaced in TUI\n }\n\n logGeneration(_span: Span, generation: GenerationEvent): void {\n this.onEvent({\n kind: \"generation\",\n model: generation.model,\n inputTokens: generation.inputTokens ?? 0,\n outputTokens: generation.outputTokens ?? 0,\n costUsd: generation.costUsd ?? 0,\n durationMs: generation.durationMs,\n });\n }\n\n recordMetric(name: string, value: number, _tags?: Record<string, string>): void {\n this.onEvent({ kind: \"metric\", name, value });\n }\n\n async flush(): Promise<void> {}\n async shutdown(): Promise<void> {\n this.spans.clear();\n this.traces.clear();\n }\n}\n\n// ── Log formatting ───────────────────────────────────────────────────────────\n\n/**\n * Format a {@link TelemetryLogEntry} into a human-readable string for the log panel.\n *\n * Produces plain (no ANSI) strings suitable for structured logging or non-terminal\n * output. The format mirrors {@link renderLog} but without colour codes: generation\n * entries include model, token counts, cost, and duration; span_end entries for\n * `tool_call` and `agent_turn` kinds include name, status, and duration; trace\n * lifecycle entries include name and elapsed time. Returns `null` for entry kinds\n * that should not be displayed.\n *\n * @param entry - The structured telemetry event to format.\n * @returns Human-readable string, or `null` if the entry should be suppressed.\n *\n * @docLink packages/tui/concepts#tui-telemetry-provider\n */\nexport function formatTelemetryEntry(entry: TelemetryLogEntry): string | null {\n switch (entry.kind) {\n case \"generation\": {\n const tokens = `${entry.inputTokens}↑ ${entry.outputTokens}↓`;\n const cost = entry.costUsd > 0 ? ` $${entry.costUsd.toFixed(4)}` : \"\";\n const dur = entry.durationMs > 0 ? ` ${(entry.durationMs / 1000).toFixed(1)}s` : \"\";\n return `[gen] ${entry.model} ${tokens}${cost}${dur}`;\n }\n case \"span_end\":\n if (entry.spanKind === \"tool_call\") {\n return `[tool] ${entry.name} ${entry.status} ${entry.durationMs}ms`;\n }\n if (entry.spanKind === \"agent_turn\") {\n return `[turn] ${entry.name} ${(entry.durationMs / 1000).toFixed(1)}s`;\n }\n return null;\n case \"trace_start\":\n return `[trace] ${entry.name} started`;\n case \"trace_end\":\n return `[trace] done ${(entry.durationMs / 1000).toFixed(1)}s`;\n default:\n return null;\n }\n}\n","import pc from \"picocolors\";\nimport type { OutputLine } from \"@skaile/workspaces\";\nimport type { TelemetryLogEntry } from \"./telemetry.js\";\n\n/**\n * Render a single OutputLine to an ANSI-colored string for terminal display.\n *\n * Each `OutputLine` variant is mapped to a distinct colour: user input (cyan/bold),\n * agent text (plain), tool calls (blue italic), questions (yellow), errors (red),\n * status/system messages (dim italic), subagent lifecycle (blue/green italic), and\n * separator lines (dim dashes). Returns an empty string for unrecognised variants.\n *\n * @param line - The typed output line to render.\n * @returns ANSI-coloured string ready for terminal output.\n *\n * @docLink packages/tui/concepts#repl-renderer\n */\nexport function renderLine(line: OutputLine): string {\n switch (line.type) {\n case \"user\":\n return pc.cyan(pc.bold(`> ${line.content}`));\n case \"text\":\n return line.content;\n case \"tool\":\n return pc.blue(pc.italic(` Running: \\`${line.name}\\`...`));\n case \"question\":\n return pc.yellow(`?? ${line.text}\\n Reply below.`);\n case \"error\":\n return pc.red(`Error: ${line.message}`);\n case \"status\":\n return pc.dim(pc.italic(`${line.phase}...`));\n case \"system\":\n return pc.dim(pc.italic(line.message));\n case \"subagent\": {\n const icon = line.status === \"started\" ? \"▶\" : \"✓\";\n const color = line.status === \"started\" ? pc.blue : pc.green;\n return color(pc.italic(` ${icon} ${line.name}: ${line.description}`));\n }\n case \"separator\":\n return pc.dim(\"─\".repeat(60));\n default:\n return \"\";\n }\n}\n\n/**\n * Render a telemetry log entry to an ANSI string, or null to suppress.\n *\n * Formats `generation` entries with model name, token counts, cost, and duration;\n * `span_end` entries for `tool_call` and `agent_turn` kinds; and `trace_start` /\n * `trace_end` lifecycle entries. All output is styled dim for visual de-emphasis.\n * Returns `null` for entry kinds that should not appear in the log panel.\n *\n * @param entry - The structured telemetry event to render.\n * @returns Dim ANSI string, or `null` if the entry should be suppressed.\n *\n * @docLink packages/tui/concepts#repl-renderer\n */\nexport function renderLog(entry: TelemetryLogEntry): string | null {\n switch (entry.kind) {\n case \"generation\": {\n const tokens = `${entry.inputTokens}↑ ${entry.outputTokens}↓`;\n const cost = entry.costUsd > 0 ? ` $${entry.costUsd.toFixed(4)}` : \"\";\n const dur = entry.durationMs > 0 ? ` ${(entry.durationMs / 1000).toFixed(1)}s` : \"\";\n return pc.dim(`[gen] ${entry.model} ${tokens}${cost}${dur}`);\n }\n case \"span_end\":\n if (entry.spanKind === \"tool_call\") {\n return pc.dim(`[tool] ${entry.name} ${entry.status} ${entry.durationMs}ms`);\n }\n if (entry.spanKind === \"agent_turn\") {\n return pc.dim(`[turn] ${entry.name} ${(entry.durationMs / 1000).toFixed(1)}s`);\n }\n return null;\n case \"trace_start\":\n return pc.dim(`[trace] ${entry.name} started`);\n case \"trace_end\":\n return pc.dim(`[trace] done ${(entry.durationMs / 1000).toFixed(1)}s`);\n default:\n return null;\n }\n}\n","/**\n * Slim event bridge: maps AgentDriver events to rendered console lines.\n * No React, no ink, no UI state — just event → string → callback.\n */\n\nimport type { AgentEvent } from \"@skaile/workspaces/bridge\";\nimport type { AgentSession } from \"@skaile/workspaces/runner\";\nimport type { OutputLine, SessionState } from \"@skaile/workspaces\";\nimport { renderLine, renderLog } from \"./repl-renderer.js\";\nimport type { TelemetryLogEntry } from \"./telemetry.js\";\n\n/**\n * Handle returned by {@link createReplSession} for controlling an active REPL session.\n *\n * `state` reflects the current lifecycle state (`idle`, `active`, or `stopped`).\n * Call `send(text)` to forward a user message to the agent driver; the state\n * transitions to `active` until the driver emits `agent_end`. Call `stop()` to\n * terminate the driver and dispose of the underlying agent session.\n *\n * @docLink packages/tui/concepts#repl-session\n */\nexport interface ReplSessionHandle {\n readonly state: SessionState;\n send(text: string): void;\n stop(): void;\n}\n\n/**\n * Options for {@link createReplSession}.\n *\n * @param session - The agent session returned by `createAgentSession`.\n * @param onLine - Callback invoked with an ANSI-rendered string for each output line.\n * @param onLog - Callback invoked with a log string for each telemetry event (if verbose).\n * @param onStateChange - Callback invoked whenever the session state changes.\n *\n * @docLink packages/tui/concepts#repl-session\n */\nexport interface CreateReplSessionOpts {\n session: AgentSession;\n onLine?: (line: string) => void;\n onLog?: (line: string) => void;\n onStateChange?: (state: SessionState) => void;\n}\n\n/**\n * Create a slim event bridge that maps `AgentEvent` stream to rendered console lines.\n *\n * Subscribes to `agent-event` on the session driver and converts each event type\n * (`message_update`, `tool_call`, `error`, `agent_end`, `turn_end`, `subagent`) to\n * an `OutputLine`, renders it via {@link renderLine}, and forwards the string to\n * `opts.onLine`. Text content is buffered internally and flushed line-by-line as\n * newlines arrive. Returns a {@link ReplSessionHandle} for sending messages and\n * stopping the session.\n *\n * @param opts - Session, output callbacks, and state-change hook.\n * @returns Handle for sending messages and observing session state.\n *\n * @docLink packages/tui/concepts#repl-session\n */\nexport function createReplSession(opts: CreateReplSessionOpts): ReplSessionHandle {\n const { session, onLine, onLog, onStateChange } = opts;\n\n let state: SessionState = \"idle\";\n let textBuffer = \"\";\n\n function emit(outputLine: OutputLine) {\n const rendered = renderLine(outputLine);\n if (rendered) onLine?.(rendered);\n }\n\n function setState(s: SessionState) {\n state = s;\n onStateChange?.(s);\n }\n\n session.driver.on(\"agent-event\", (event: AgentEvent) => {\n switch (event.type) {\n case \"message_update\": {\n const delta = (event as any)._textDelta as string | undefined;\n if (delta) {\n textBuffer += delta;\n const lines = textBuffer.split(\"\\n\");\n if (lines.length > 1) {\n for (let i = 0; i < lines.length - 1; i++) {\n emit({ type: \"text\", content: lines[i]! });\n }\n textBuffer = lines[lines.length - 1]!;\n }\n }\n break;\n }\n\n case \"tool_call\": {\n const name = (event as any).tool?.name ?? (event as any).name ?? \"unknown\";\n emit({ type: \"tool\", name });\n break;\n }\n\n case \"error\": {\n const msg = (event as any).error ?? \"Unknown error\";\n emit({ type: \"error\", message: String(msg) });\n if ((event as any).fatal) setState(\"stopped\");\n break;\n }\n\n case \"agent_end\":\n case \"turn_end\": {\n if (textBuffer.trim()) {\n emit({ type: \"text\", content: textBuffer });\n }\n textBuffer = \"\";\n emit({ type: \"separator\" });\n if (event.type === \"agent_end\") setState(\"idle\");\n break;\n }\n\n case \"session_info\":\n break;\n\n default: {\n const raw = event as any;\n if (raw.type === \"subagent\") {\n const status = raw.status === \"finished\" ? \"finished\" : \"started\";\n emit({\n type: \"subagent\",\n name: raw.name ?? \"subagent\",\n status,\n description: raw.description ?? \"\",\n });\n }\n break;\n }\n }\n });\n\n return {\n get state() {\n return state;\n },\n\n send(text: string) {\n if (state === \"stopped\") return;\n setState(\"active\");\n textBuffer = \"\";\n session.driver.prompt(text).catch((err: Error) => {\n emit({ type: \"error\", message: err.message });\n setState(\"idle\");\n });\n },\n\n stop() {\n setState(\"stopped\");\n session.driver.kill();\n session.dispose();\n },\n };\n}\n\n/**\n * Create a telemetry sink that pipes {@link TelemetryLogEntry} events to the log callback.\n *\n * Wraps {@link renderLog} so that only displayable entries (generation, span_end for\n * tool/turn, trace lifecycle) are forwarded. Entries that `renderLog` returns `null`\n * for are silently suppressed. The returned function is used in `startRepl` to drain\n * buffered telemetry and to wire up live forwarding during the session.\n *\n * @param onLog - Callback invoked with an ANSI-rendered log string.\n * @returns A function that accepts a `TelemetryLogEntry` and routes it to `onLog`.\n *\n * @docLink packages/tui/concepts#repl-session\n */\nexport function createTelemetrySink(\n onLog: (line: string) => void,\n): (entry: TelemetryLogEntry) => void {\n return (entry) => {\n const text = renderLog(entry);\n if (text) onLog(text);\n };\n}\n","/**\n * Simple @clack-based agent REPL.\n * No React, no ink, no alt-screen — just a scrolling terminal.\n *\n * Design: clack's text() blocks until user submits, so agent output\n * flows naturally between prompts — no readline interleaving issues.\n * Loop: await text() → send → await idle → loop\n */\n\nimport * as p from \"@clack/prompts\";\nimport { resolve, join, basename, dirname, parse } from \"node:path\";\nimport { existsSync, readFileSync, statSync, readdirSync } from \"node:fs\";\nimport type { Dirent } from \"node:fs\";\nimport pc from \"picocolors\";\nimport { createAgentSession } from \"@skaile/workspaces/runner\";\nimport { resolveAgentDir, resolveSettings } from \"@skaile/workspaces/core\";\nimport { loadConnectorDeclarations } from \"@skaile/workspaces/connectors\";\nimport { TuiTelemetryProvider } from \"./telemetry.js\";\nimport type { TelemetryLogEntry } from \"./telemetry.js\";\nimport { createReplSession, createTelemetrySink } from \"./repl-session.js\";\n\nfunction findAiAssetsRoot(startDir: string): string | undefined {\n let current = resolve(startDir);\n const root = parse(current).root;\n while (current !== root) {\n const candidate = join(current, \"ai-assets\");\n try {\n if (statSync(candidate).isDirectory()) return candidate;\n } catch {}\n const parent = dirname(current);\n if (parent === current) break;\n current = parent;\n }\n return undefined;\n}\n\nfunction countSkills(projectDir: string): number {\n const aiAssetsRoot = findAiAssetsRoot(projectDir);\n if (!aiAssetsRoot) return 0;\n let count = 0;\n const walk = (dir: string) => {\n let entries: Dirent[];\n try {\n entries = readdirSync(dir, { withFileTypes: true });\n } catch {\n return;\n }\n if (entries.some((e) => e.isFile() && e.name === \"SKILL.md\")) {\n count++;\n return;\n }\n for (const e of entries) {\n if (e.isDirectory()) walk(join(dir, e.name));\n }\n };\n walk(aiAssetsRoot);\n return count;\n}\n\n/**\n * Options for {@link startRepl}.\n *\n * @param projectDir - Absolute path to the project root. Used to locate the agent dir, settings, and skill count.\n * @param cwd - Working directory passed to the agent process. Defaults to `projectDir`.\n * @param driver - Agent driver override (`\"omp\"` or `\"claude-sdk\"`). Falls back to settings resolution.\n * @param model - Model name passed to the agent driver.\n * @param provider - Provider name passed to the agent driver.\n * @param resume - When `true`, load the last session ID from `.skaile/last_session` and resume it.\n * @param resumeSessionId - Explicit session ID to resume. Takes precedence over `resume`.\n * @param verbose - When `true`, show telemetry log lines inline between prompts. Default: `false`.\n *\n * @docLink packages/tui/concepts#start-repl\n */\nexport interface ReplOptions {\n projectDir: string;\n cwd?: string;\n driver?: string;\n model?: string;\n provider?: string;\n resume?: boolean;\n resumeSessionId?: string;\n /** When true, show telemetry log lines inline. Default: false. */\n verbose?: boolean;\n}\n\n/**\n * Start the scrolling REPL for an interactive agent conversation.\n *\n * Initialises the agent session (including OMP extension injection when the driver is `omp`),\n * prints a banner with project metadata, then enters a `@clack/prompts` input loop.\n * Each iteration awaits user input, forwards it to the agent via {@link createReplSession},\n * and blocks until the agent signals `idle` or `stopped`. Slash commands (`/help`, `/log`,\n * `/quit`) are handled before the message reaches the agent.\n *\n * @param opts - Configuration for the REPL session.\n * @returns Resolves when the session ends (user quit or fatal error).\n *\n * @docLink packages/tui/concepts#start-repl\n */\nexport async function startRepl(opts: ReplOptions): Promise<void> {\n const projectDir = resolve(opts.projectDir);\n const agentCwd = opts.cwd ? resolve(opts.cwd) : projectDir;\n\n // ── OMP workspace extension injection ──────────────────────────────\n const settings = await resolveSettings(projectDir, { driver: opts.driver });\n const resolvedDriver = settings.driver ?? \"omp\";\n let ompCleanup: (() => void) | undefined;\n\n if (resolvedDriver === \"omp\") {\n const fsAsync = await import(\"node:fs/promises\");\n const pathMod = await import(\"node:path\");\n const { WorkspacePlugin } = await import(\"@skaile/workspaces/workspace-plugin\");\n\n const plugin = new WorkspacePlugin({ projectDir });\n const src = await plugin.buildOmpExtensionSource();\n const extPath = pathMod.default.join(agentCwd, \".omp\", \"extensions\", \"skaile.ts\");\n await fsAsync.default.mkdir(pathMod.default.dirname(extPath), { recursive: true });\n await fsAsync.default.writeFile(extPath, src);\n ompCleanup = () => {\n fsAsync.default.unlink(extPath).catch(() => {});\n };\n }\n\n // ── Session resume ─────────────────────────────────────────────────\n const SESSION_FILE = join(projectDir, \".skaile\", \"last_session\");\n let resumeSessionId = opts.resumeSessionId;\n if (!resumeSessionId && opts.resume) {\n try {\n if (existsSync(SESSION_FILE)) {\n resumeSessionId = readFileSync(SESSION_FILE, \"utf-8\").trim() || undefined;\n }\n } catch {}\n }\n\n // ── Telemetry ──────────────────────────────────────────────────────\n const telemetryBuffer: TelemetryLogEntry[] = [];\n const telemetry = new TuiTelemetryProvider((entry) => {\n telemetryBuffer.push(entry);\n });\n const trace = telemetry.startTrace({ name: \"skaile-repl\", kind: \"chat_session\" });\n\n // ── Create agent session ───────────────────────────────────────────\n const agentDir = resolveAgentDir(projectDir) ?? undefined;\n const session = await createAgentSession({\n projectDir,\n cwd: agentCwd !== projectDir ? agentCwd : undefined,\n agentDir,\n driver: opts.driver ?? resolvedDriver,\n model: opts.model,\n provider: opts.provider,\n resumeSessionId,\n telemetry,\n trace,\n onLog: () => {},\n });\n\n await session.driver.start();\n\n // ── Output + state tracking ────────────────────────────────────────\n let showLog = opts.verbose ?? false;\n let idleResolve: (() => void) | undefined;\n\n const write = (line: string) => {\n p.log.message(line);\n };\n\n const writeLog = (line: string) => {\n if (showLog) p.log.step(line);\n };\n\n // ── Create REPL session ────────────────────────────────────────────\n const replSession = createReplSession({\n session,\n onLine: write,\n onLog: writeLog,\n onStateChange: (state) => {\n if (state === \"idle\" || state === \"stopped\") {\n idleResolve?.();\n }\n },\n });\n\n // Drain buffered telemetry\n const sink = createTelemetrySink(writeLog);\n for (const entry of telemetryBuffer) sink(entry);\n telemetryBuffer.length = 0;\n\n // ── Banner ─────────────────────────────────────────────────────────\n const displayModel = session.model ?? opts.model ?? \"default\";\n const displayDriver = session.driverType ?? resolvedDriver;\n\n const skillCount = countSkills(projectDir);\n let connectorCount = 0;\n try {\n connectorCount = loadConnectorDeclarations(projectDir).length;\n } catch {}\n const agentName = agentDir ? basename(agentDir) : \"(none)\";\n\n p.intro(`${pc.bold(\"skaile repl\")} ${pc.dim(`— ${displayDriver} / ${displayModel}`)}`);\n p.log.info(pc.dim(`Project: ${projectDir}`));\n p.log.info(pc.dim(`Agent: ${agentName} · ${skillCount} skills · ${connectorCount} connectors`));\n if (resumeSessionId) p.log.info(pc.dim(`Resumed: ${resumeSessionId.slice(0, 12)}...`));\n p.log.info(pc.dim(\"Type /help for commands, Ctrl+C to quit.\"));\n\n // ── REPL loop ──────────────────────────────────────────────────────\n function shutdown() {\n replSession.stop();\n telemetry.endTrace(trace, { status: \"ok\" });\n void telemetry.shutdown();\n ompCleanup?.();\n p.outro(\"Bye.\");\n process.exit(0);\n }\n\n process.on(\"SIGINT\", shutdown);\n process.on(\"SIGTERM\", shutdown);\n\n while (replSession.state !== \"stopped\") {\n const input = await p.text({\n message: pc.gray(\">>>\"),\n placeholder: \"\",\n });\n\n if (p.isCancel(input)) {\n shutdown();\n return;\n }\n\n const trimmed = (input as string).trim();\n if (!trimmed) continue;\n\n if (trimmed === \"/quit\" || trimmed === \"/q\") {\n shutdown();\n return;\n }\n if (trimmed === \"/help\" || trimmed === \"/h\" || trimmed === \"/?\") {\n p.log.info(\n [\n \" /help, /h — this help\",\n \" /log — toggle verbose log output\",\n \" /quit, /q — exit\",\n \" Ctrl+C — exit\",\n ].join(\"\\n\"),\n );\n continue;\n }\n if (trimmed === \"/log\") {\n showLog = !showLog;\n p.log.info(`Log output: ${showLog ? \"on\" : \"off\"}`);\n continue;\n }\n\n const idlePromise = new Promise<void>((resolve) => {\n idleResolve = resolve;\n });\n replSession.send(trimmed);\n await idlePromise;\n }\n\n shutdown();\n}\n"]}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
export { PluginStore, WorkspacePlugin, buildClaudePluginFiles, resolveProjectDir } from '../chunk-
|
|
1
|
+
export { PluginStore, WorkspacePlugin, buildClaudePluginFiles, resolveProjectDir } from '../chunk-NDD5VMN5.js';
|
|
2
2
|
export { err, ok, okJson } from '../chunk-X5YPJV4N.js';
|
|
3
3
|
import '../chunk-24UIWON4.js';
|
|
4
4
|
import '../chunk-NSBPE2FW.js';
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@skaile/workspaces",
|
|
3
|
-
"version": "0.22.0
|
|
3
|
+
"version": "0.22.0",
|
|
4
4
|
"description": "Skaile workspaces runtime — types, core, bridge, runner, store, connectors, and supporting layers as one publishable package",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"packageManager": "bun@1.3.9",
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../cli/src/asset-feeds.ts"],"names":[],"mappings":";;;;;;;;;;;;;;AAoCA,eAAe,aAAA,GAAyE;AACtF,EAAA,MAAM,QAAoB,EAAC;AAC3B,EAAA,IAAI;AACF,IAAA,MAAM,CAAC,EAAE,kBAAA,EAAmB,EAAG,EAAE,eAAe,CAAA,GAAI,MAAM,OAAA,CAAQ,GAAA,CAAI;AAAA,MACpE,OAAO,4BAAmB,CAAA;AAAA,MAC1B,OAAO,oBAA4B;AAAA,KACpC,CAAA;AACD,IAAA,MAAM,EAAE,OAAA,EAAS,KAAA,EAAM,GAAI,MAAM,kBAAA,EAAmB;AACpD,IAAA,IAAI;AACF,MAAA,MAAM,UAAA,GAAkB,IAAA,CAAA,IAAA,CAAK,aAAA,EAAc,EAAG,SAAS,CAAA;AACvD,MAAA,MAAM,IAAA,GAAA,CAAQ,MAAM,OAAA,CAAQ,aAAA,EAAc,EAAG,MAAA;AAAA,QAC3C,CAAC,CAAA,KAAM,CAAC,CAAA,CAAE,IAAA,CAAK,WAAW,UAAU;AAAA,OACtC;AACA,MAAA,MAAM,UAA0B,EAAC;AACjC,MAAA,KAAA,MAAW,OAAO,IAAA,EAAM;AACtB,QAAA,IAAI,CAAC,UAAA,CAAW,GAAA,CAAI,IAAI,CAAA,EAAG;AAC3B,QAAA,KAAA,MAAW,CAAA,IAAK,SAAS,GAAA,CAAI,IAAA,EAAM,WAAW,GAAA,CAAI,IAAI,EAAE,CAAA,EAAG;AACzD,UAAA,OAAA,CAAQ,KAAK,CAAC,CAAA;AAAA,QAChB;AAAA,MACF;AACA,MAAA,OAAO,EAAE,SAAS,KAAA,EAAM;AAAA,IAC1B,CAAA,SAAE;AACA,MAAA,KAAA,EAAM;AAAA,IACR;AAAA,EACF,SAAS,GAAA,EAAK;AACZ,IAAA,KAAA,CAAM,IAAA,CAAK;AAAA,MACT,IAAA,EAAM,WAAA;AAAA,MACN,SAAS,GAAA,YAAe,KAAA,GAAQ,GAAA,CAAI,OAAA,GAAU,OAAO,GAAG;AAAA,KACzD,CAAA;AACD,IAAA,OAAO,EAAE,OAAA,EAAS,EAAC,EAAG,KAAA,EAAM;AAAA,EAC9B;AACF;AAWA,eAAe,YAAY,UAAA,EAGxB;AACD,EAAA,MAAM,QAAoB,EAAC;AAC3B,EAAA,IAAI;AACF,IAAA,MAAM,EAAE,oBAAA,EAAqB,GAAI,MAAM,OAAO,4BAAmB,CAAA;AACjE,IAAA,MAAM,EAAE,QAAQ,KAAA,EAAM,GAAI,MAAM,oBAAA,CAAqB,EAAE,YAAY,CAAA;AACnE,IAAA,IAAI;AACF,MAAA,MAAM,MAAA,GAAS,MAAM,MAAA,CAAO,UAAA,EAAW;AACvC,MAAA,MAAM,OAAA,GAA0B,MAAA,CAAO,GAAA,CAAI,CAAC,CAAA,MAAO;AAAA,QACjD,MAAM,CAAA,CAAE,IAAA;AAAA,QACR,IAAA,EAAO,EAAE,IAAA,IAAsB,OAAA;AAAA,QAC/B,WAAA,EAAa,EAAE,WAAA,IAAe,EAAA;AAAA,QAC9B,MAAA,EAAQ,CAAA,QAAA,EAAW,CAAA,CAAE,EAAE,CAAA,CAAA;AAAA,QACvB,UAAA,EAAY,OAAA;AAAA,QACZ,QAAQ,CAAA,CAAE,SAAA;AAAA,QACV,SAAS,CAAA,CAAE,OAAA;AAAA,QACX,UAAU,EAAC;AAAA,QACX,cAAc;AAAC,OACjB,CAAE,CAAA;AACF,MAAA,OAAO,EAAE,SAAS,KAAA,EAAM;AAAA,IAC1B,CAAA,SAAE;AACA,MAAA,KAAA,EAAM;AAAA,IACR;AAAA,EACF,SAAS,GAAA,EAAK;AACZ,IAAA,MAAM,MAAM,GAAA,YAAe,KAAA,GAAQ,GAAA,CAAI,OAAA,GAAU,OAAO,GAAG,CAAA;AAG3D,IAAA,IAAI,CAAC,QAAA,CAAS,IAAA,CAAK,GAAG,CAAA,EAAG;AACvB,MAAA,KAAA,CAAM,KAAK,EAAE,IAAA,EAAM,OAAA,EAAS,OAAA,EAAS,KAAK,CAAA;AAAA,IAC5C;AACA,IAAA,OAAO,EAAE,OAAA,EAAS,EAAC,EAAG,KAAA,EAAM;AAAA,EAC9B;AACF;AAQO,SAAS,aAAa,EAAA,EAAsD;AACjF,EAAA,OAAO,GAAG,MAAA,EAAO;AACnB;AAQA,eAAsB,gBAAA,CACpB,IACA,UAAA,EAC0B;AAC1B,EAAA,MAAM,CAAC,IAAA,EAAM,KAAK,CAAA,GAAI,MAAM,OAAA,CAAQ,GAAA,CAAI,CAAC,aAAA,EAAc,EAAG,WAAA,CAAY,UAAU,CAAC,CAAC,CAAA;AAClF,EAAA,OAAO;AAAA,IACL,OAAA,EAAS,CAAC,GAAG,YAAA,CAAa,EAAE,CAAA,EAAG,GAAG,IAAA,CAAK,OAAA,EAAS,GAAG,KAAA,CAAM,OAAO,CAAA;AAAA,IAChE,OAAO,CAAC,GAAG,KAAK,KAAA,EAAO,GAAG,MAAM,KAAK;AAAA,GACvC;AACF","file":"asset-feeds-QXCSAJRN.js","sourcesContent":["/**\n * asset-feeds.ts — gather installable assets from every feed the manage TUI\n * surfaces.\n *\n * Three feeds, normalized into `CatalogEntry`:\n * - **Project sources** — `skaile.yaml` `sources:` entries (scanned via\n * `AssetManager.search()`; the new `sources:` field is merged into the\n * legacy repositories map in `AssetManager.loadConfig`).\n * - **Local libraries** — user authoring places from `LocalIndex` (filesystem\n * scan of each library's `path`).\n * - **Remote store** — the configured catalog source when `catalog.url` is\n * not `local`. Failures are non-fatal — they surface as notes.\n */\n\nimport { existsSync } from \"node:fs\";\nimport * as path from \"node:path\";\nimport { type AssetKind, type CatalogEntry, scanRepo } from \"@skaile/workspaces/core\";\n\n/** One non-fatal note returned to the caller for surfacing in the UI. */\nexport interface FeedNote {\n feed: \"sources\" | \"libraries\" | \"store\";\n message: string;\n}\n\nexport interface AssetFeedResult {\n entries: CatalogEntry[];\n notes: FeedNote[];\n}\n\n/**\n * Pull assets from local libraries (the authoring places under\n * `~/.skaile/libraries/<name>/`, not the `~/.skaile/sources/` cache). Each\n * library's `path` is scanned with `scanRepo` and tagged with\n * `repository = \"library:<name>\"` so the manage TUI groups them into their\n * own headers.\n */\nasync function libraryAssets(): Promise<{ entries: CatalogEntry[]; notes: FeedNote[] }> {\n const notes: FeedNote[] = [];\n try {\n const [{ openLibraryManager }, { skaileHomeDir }] = await Promise.all([\n import(\"./open-library.ts\"),\n import(\"@skaile/workspaces/library\"),\n ]);\n const { manager, close } = await openLibraryManager();\n try {\n const sourcesDir = path.join(skaileHomeDir(), \"sources\");\n const libs = (await manager.listLibraries()).filter(\n (l) => !l.path.startsWith(sourcesDir),\n );\n const entries: CatalogEntry[] = [];\n for (const lib of libs) {\n if (!existsSync(lib.path)) continue;\n for (const e of scanRepo(lib.path, `library:${lib.name}`)) {\n entries.push(e);\n }\n }\n return { entries, notes };\n } finally {\n close();\n }\n } catch (err) {\n notes.push({\n feed: \"libraries\",\n message: err instanceof Error ? err.message : String(err),\n });\n return { entries: [], notes };\n }\n}\n\n/**\n * Pull a (best-effort) snapshot of the connected remote store catalog.\n *\n * - Returns no entries when the resolved catalog is local-only or the host\n * is unreachable (the failure becomes a note, not an exception).\n * - Maps `CatalogAsset` → `CatalogEntry` with `repository = \"store\"` and\n * `source = \"store://<id>\"`; per-file `source` paths are not available\n * for remote assets so we use the canonical ref as a stable placeholder.\n */\nasync function storeAssets(projectDir: string): Promise<{\n entries: CatalogEntry[];\n notes: FeedNote[];\n}> {\n const notes: FeedNote[] = [];\n try {\n const { resolveCatalogSource } = await import(\"./open-library.ts\");\n const { source, close } = await resolveCatalogSource({ projectDir });\n try {\n const assets = await source.listAssets();\n const entries: CatalogEntry[] = assets.map((a) => ({\n name: a.name,\n kind: (a.kind as AssetKind) ?? \"skill\",\n description: a.description ?? \"\",\n source: `store://${a.id}`,\n repository: \"store\",\n domain: a.publisher,\n version: a.version,\n requires: [],\n dependencies: [],\n }));\n return { entries, notes };\n } finally {\n close();\n }\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n // Local-only catalog throws a recognizable message — silence it; absence\n // is the user's choice, not an error.\n if (!/local/i.test(msg)) {\n notes.push({ feed: \"store\", message: msg });\n }\n return { entries: [], notes };\n }\n}\n\n/**\n * Project-source feed: leans on `AssetManager.search()`, which now merges\n * `skaile.yaml`'s `sources:` field into the repositories map. Provided as a\n * function rather than an inline call so the caller can supply a fresh\n * `AssetManager` instance.\n */\nexport function sourceAssets(am: { search: () => CatalogEntry[] }): CatalogEntry[] {\n return am.search();\n}\n\n/**\n * Run all three feeds and return a single normalized result.\n *\n * Feeds run in parallel where possible — the store fetch is the slow one\n * (one HTTP call) and is fired alongside the local library walk.\n */\nexport async function gatherAssetFeeds(\n am: { search: () => CatalogEntry[] },\n projectDir: string,\n): Promise<AssetFeedResult> {\n const [libs, store] = await Promise.all([libraryAssets(), storeAssets(projectDir)]);\n return {\n entries: [...sourceAssets(am), ...libs.entries, ...store.entries],\n notes: [...libs.notes, ...store.notes],\n };\n}\n"]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../core/src/manifest.ts"],"names":[],"mappings":"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","file":"chunk-4AZKT2BU.js","sourcesContent":["/**\n * Manifest parsing — reads SKILL.md, AGENT.md, agent.yaml, CONTRACT.md,\n * CONNECTOR.md, *.prompt.md, *.flow.yaml, *.bundle.yaml files and\n * produces CatalogEntry objects.\n * Used by @skaile/asset-manager for on-demand repo scanning.\n */\n\nimport { readdirSync, readFileSync, statSync } from \"node:fs\";\nimport { basename, dirname, join, relative } from \"node:path\";\nimport { parse } from \"yaml\";\nimport type { AssetKind, CatalogEntry, Dependency } from \"./models.js\";\nimport { parseDep } from \"./models.js\";\n\n// ── Frontmatter helpers ───────────────────────────────────────────────────────\n\n/**\n * Result of parsing YAML frontmatter from a markdown file.\n * @docLink packages/core/concepts#frontmatter-result\n */\nexport interface FrontmatterResult {\n /** Parsed YAML data. Empty object if no frontmatter block is present. */\n data: Record<string, unknown>;\n /** Markdown body below the frontmatter block (trimmed). */\n body: string;\n}\n\n/**\n * Parse YAML frontmatter from a markdown file (`---` delimited).\n * Returns `{ data: {}, body: text }` when no frontmatter block is present.\n *\n * @param text - Raw file contents\n * @returns Parsed frontmatter data and the remaining body text\n * @docLink packages/core/concepts#parse-frontmatter\n */\nexport function parseFrontmatter(text: string): FrontmatterResult {\n const m = text.match(/^---\\s*\\n([\\s\\S]*?)\\n---/);\n if (!m) return { data: {}, body: text.trim() };\n try {\n const data = (parse(m[1]!) as Record<string, unknown>) ?? {};\n const body = text.slice(m[0].length).trim();\n return { data, body };\n } catch {\n return { data: {}, body: text.trim() };\n }\n}\n\n/** Extract version from frontmatter, checking both top-level and metadata.version. */\nfunction extractVersion(meta: Record<string, unknown>): string {\n return String(\n meta.version ?? (meta.metadata as Record<string, unknown> | undefined)?.version ?? \"\",\n );\n}\n\n/**\n * Parse the `requires:` field from manifest frontmatter into typed `Dependency` objects.\n * Accepts a comma-separated string or an array of `\"kind:name\"` strings. Reads from either\n * the root or `metadata.requires` (matching `extractVersion`'s dual-location behavior) so\n * skill authors can follow the agentskills.io convention of placing skaile extensions under\n * `metadata:` while remaining backward-compatible with root-level declarations.\n *\n * @param meta - Parsed frontmatter data object\n * @returns Array of `Dependency` objects (empty when `requires` is absent)\n * @docLink packages/core/concepts#parse-requires\n */\nexport function parseRequires(meta: Record<string, unknown>): Dependency[] {\n const raw = meta.requires ?? (meta.metadata as Record<string, unknown> | undefined)?.requires;\n if (!raw) return [];\n const list =\n typeof raw === \"string\"\n ? raw.split(\",\").map((s) => s.trim())\n : Array.isArray(raw)\n ? (raw as string[])\n : [];\n return list.filter(Boolean).map(parseDep);\n}\n\n// ── Manifest parsers ──────────────────────────────────────────────────────────\n\n/**\n * Parse a `SKILL.md` file into a `CatalogEntry`.\n * The `kind` field defaults to `\"skill\"` but can be overridden to `\"agent\"` or `\"prompt\"`\n * via the `type:` frontmatter key. Falls back to the parent directory name when `name:` is absent.\n *\n * @param filePath - Absolute path to the `SKILL.md` file\n * @param repoName - Repository name used as the `repository` field of the entry\n * @returns Populated `CatalogEntry` with kind, name, description, version, and requires\n * @docLink packages/core/concepts#from-skill-md\n */\nexport function fromSkillMd(filePath: string, repoName: string): CatalogEntry {\n const { data: meta } = parseFrontmatter(readFileSync(filePath, \"utf8\"));\n let kind = String(meta.type ?? \"skill\") as AssetKind;\n if (![\"skill\", \"agent\", \"prompt\", \"persona\", \"ruleset\", \"knowledge\"].includes(kind))\n kind = \"skill\";\n return {\n name: String(meta.name ?? basename(dirname(filePath))),\n kind,\n description: String(meta.description ?? \"\"),\n source: filePath,\n publisher: repoName,\n version: extractVersion(meta),\n requires: parseRequires(meta),\n dependencies: [],\n };\n}\n\n/**\n * Parse an `AGENT.md` file into a `CatalogEntry` with `kind: \"agent\"`.\n * Falls back to the parent directory name when `name:` is absent.\n *\n * @param filePath - Absolute path to the `AGENT.md` file\n * @param repoName - Repository name used as the `repository` field of the entry\n * @returns Populated `CatalogEntry` with kind `\"agent\"`\n * @docLink packages/core/concepts#from-agent-md\n */\nexport function fromAgentMd(filePath: string, repoName: string): CatalogEntry {\n const { data: meta } = parseFrontmatter(readFileSync(filePath, \"utf8\"));\n return {\n name: String(meta.name ?? basename(dirname(filePath))),\n kind: \"agent\",\n description: String(meta.description ?? \"\"),\n source: filePath,\n publisher: repoName,\n version: extractVersion(meta),\n requires: parseRequires(meta),\n dependencies: [],\n };\n}\n\n/**\n * Parse an `agent.yaml` file into a `CatalogEntry` with `kind: \"agent\"`.\n * Collects sub-agent dependencies from `requires[]` (new format) and `dependencies[]`\n * (legacy format), skill abilities from `abilities[]`, and contracts from `contracts[]`.\n *\n * @param filePath - Absolute path to the `agent.yaml` file\n * @param repoName - Repository name used as the `repository` field of the entry\n * @returns Populated `CatalogEntry` with kind `\"agent\"` and full dependency graph\n * @docLink packages/core/concepts#from-agent-yaml\n */\nexport function fromAgentYaml(filePath: string, repoName: string): CatalogEntry {\n let meta: Record<string, unknown> = {};\n try {\n meta = (parse(readFileSync(filePath, \"utf8\")) as Record<string, unknown>) ?? {};\n } catch {}\n\n const requires: Dependency[] = [];\n\n // Sub-agent dependencies from requires[] (new format: {name, source, ...})\n const agentRequires = meta.requires;\n if (Array.isArray(agentRequires)) {\n for (const d of agentRequires) {\n if (typeof d === \"object\" && d !== null && \"name\" in d) {\n requires.push({\n kind: \"agent\",\n name: String((d as Record<string, unknown>).name),\n });\n }\n }\n }\n\n // Legacy sub-agent dependencies from dependencies[] (old format: {name, ...})\n const deps = meta.dependencies;\n if (Array.isArray(deps)) {\n for (const d of deps) {\n if (typeof d === \"object\" && d !== null && \"name\" in d) {\n requires.push({\n kind: \"agent\",\n name: String((d as Record<string, unknown>).name),\n });\n }\n }\n }\n\n // Skill abilities referenced in the system prompt\n const abilities = meta.abilities;\n if (Array.isArray(abilities)) {\n for (const a of abilities) {\n if (typeof a === \"string\" && a) {\n requires.push({ kind: \"skill\", name: a });\n }\n }\n }\n\n // Contracts referenced in the system prompt (co-located in skills dir)\n const contracts = meta.contracts;\n if (Array.isArray(contracts)) {\n for (const c of contracts) {\n if (typeof c === \"string\" && c) {\n requires.push({ kind: \"contract\", name: c });\n }\n }\n }\n\n // Mixin composition fields — track catalog refs as requires so `skaile install`\n // automatically pulls them. Local paths (. or /) are skipped — not catalog assets.\n for (const [field, defaultKind] of [\n [\"persona\", \"persona\"],\n [\"rules\", \"ruleset\"],\n [\"knowledge\", \"knowledge\"],\n ] as const) {\n const refs = meta[field];\n if (!Array.isArray(refs)) continue;\n for (const r of refs) {\n if (typeof r !== \"string\" || !r) continue;\n if (r.startsWith(\".\") || r.startsWith(\"/\")) continue; // local path — skip\n // Normalize: if no kind: prefix, inject the default kind\n const normalized = r.includes(\":\") ? r : `${defaultKind}:${r}`;\n // Strip @repo and #pin — only kind:name matters for requires\n const colonIdx = normalized.indexOf(\":\");\n const kind = normalized.slice(0, colonIdx).trim();\n let name = normalized.slice(colonIdx + 1).trim();\n const atIdx = name.indexOf(\"@\");\n if (atIdx !== -1) name = name.slice(0, atIdx).trim();\n const hashIdx = name.indexOf(\"#\");\n if (hashIdx !== -1) name = name.slice(0, hashIdx).trim();\n if (name) requires.push({ kind, name });\n }\n }\n\n return {\n name: String(meta.name ?? basename(dirname(filePath))),\n kind: \"agent\",\n description: String(meta.description ?? \"\"),\n source: filePath,\n publisher: repoName,\n version: String(meta.version ?? \"\"),\n requires,\n dependencies: [],\n };\n}\n\n/**\n * Parse a `*.prompt.md` file into a `CatalogEntry` with `kind: \"prompt\"`.\n * The name defaults to the filename stem (without the `.prompt.md` suffix)\n * when no `name:` frontmatter key is present.\n *\n * @param filePath - Absolute path to the `.prompt.md` file\n * @param repoName - Repository name used as the `repository` field of the entry\n * @returns Populated `CatalogEntry` with kind `\"prompt\"`\n * @docLink packages/core/concepts#from-prompt-md\n */\nexport function fromPromptMd(filePath: string, repoName: string): CatalogEntry {\n const { data: meta } = parseFrontmatter(readFileSync(filePath, \"utf8\"));\n const name = String(meta.name ?? basename(filePath).replace(/\\.prompt\\.md$/, \"\"));\n return {\n name,\n kind: \"prompt\",\n description: String(meta.description ?? \"\"),\n source: filePath,\n publisher: repoName,\n version: extractVersion(meta),\n requires: parseRequires(meta),\n dependencies: [],\n };\n}\n\n/**\n * Parse a `*.flow.yaml` or `*.flow.json` file into a `CatalogEntry` with `kind: \"flow\"`.\n * The name resolves from `name:` or `id:` frontmatter, falling back to the filename stem.\n *\n * @param filePath - Absolute path to the flow definition file\n * @param repoName - Repository name used as the `repository` field of the entry\n * @returns Populated `CatalogEntry` with kind `\"flow\"`\n * @docLink packages/core/concepts#from-flow-yaml\n */\nexport function fromFlowYaml(filePath: string, repoName: string): CatalogEntry {\n let meta: Record<string, unknown> = {};\n try {\n meta = (parse(readFileSync(filePath, \"utf8\")) as Record<string, unknown>) ?? {};\n } catch {}\n\n const stem = basename(filePath)\n .replace(/\\.flow\\.yaml$/, \"\")\n .replace(/\\.flow\\.json$/, \"\");\n return {\n name: String(meta.name ?? meta.id ?? stem),\n kind: \"flow\",\n description: String(meta.description ?? \"\"),\n source: filePath,\n publisher: repoName,\n version: String(meta.version ?? \"\"),\n requires: parseRequires(meta),\n dependencies: [],\n };\n}\n\n/**\n * Parse a `*.bundle.yaml` file into a `CatalogEntry` with `kind: \"bundle\"`.\n * Bundles declare a flat list of string `dependencies` (asset refs) rather than\n * typed `Dependency` objects; both `requires` and `dependencies` are populated.\n *\n * @param filePath - Absolute path to the bundle YAML file\n * @param repoName - Repository name used as the `repository` field of the entry\n * @returns Populated `CatalogEntry` with kind `\"bundle\"`\n * @docLink packages/core/concepts#from-bundle-yaml\n */\nexport function fromBundleYaml(filePath: string, repoName: string): CatalogEntry {\n let meta: Record<string, unknown> = {};\n try {\n meta = (parse(readFileSync(filePath, \"utf8\")) as Record<string, unknown>) ?? {};\n } catch {}\n\n const name = String(meta.name ?? basename(filePath).replace(/\\.bundle\\.yaml$/, \"\"));\n\n const deps = meta.dependencies;\n const dependencies = Array.isArray(deps) ? (deps as string[]).filter(Boolean) : [];\n\n return {\n name,\n kind: \"bundle\",\n description: String(meta.description ?? \"\"),\n source: filePath,\n publisher: repoName,\n version: String(meta.version ?? \"\"),\n requires: parseRequires(meta),\n dependencies,\n };\n}\n\n/**\n * Parse a `CONTRACT.md` file into a `CatalogEntry` with `kind: \"contract\"`.\n * Falls back to the parent directory name when `name:` is absent.\n *\n * @param filePath - Absolute path to the `CONTRACT.md` file\n * @param repoName - Repository name used as the `repository` field of the entry\n * @returns Populated `CatalogEntry` with kind `\"contract\"`\n * @docLink packages/core/concepts#from-contract-md\n */\nexport function fromContractMd(filePath: string, repoName: string): CatalogEntry {\n const { data: meta } = parseFrontmatter(readFileSync(filePath, \"utf8\"));\n return {\n name: String(meta.name ?? basename(dirname(filePath))),\n kind: \"contract\",\n description: String(meta.description ?? \"\"),\n source: filePath,\n publisher: repoName,\n version: extractVersion(meta),\n requires: parseRequires(meta),\n dependencies: [],\n };\n}\n\n/**\n * Parse an `MCP.md` file into a `CatalogEntry` with `kind: \"mcp-server\"`.\n * Falls back to the parent directory name when `name:` is absent.\n *\n * Frontmatter fields (catalog identity):\n * - `name` — server name (default: parent dir name)\n * - `description` — short description\n * - `version` — semver\n * - `keywords` — array of discovery tags\n *\n * Frontmatter fields (runtime defaults, stored in `metadata`):\n * - `transport` — \"stdio\" | \"sse\" | \"http\" (default: \"stdio\")\n * - `command` — command to execute (stdio)\n * - `args` — command arguments (stdio)\n * - `env` — environment variables (stdio)\n * - `url` — server URL (sse/http)\n * - `headers` — HTTP headers (sse/http)\n *\n * When `mcp:name` appears in `skaile.yaml` `dependencies`, the runner reads\n * these defaults from `metadata` to build a `McpServerDeclaration` without\n * requiring verbose inline config.\n *\n * @param filePath - Absolute path to the `MCP.md` file\n * @param repoName - Repository name used as the `repository` field of the entry\n * @returns Populated `CatalogEntry` with kind `\"mcp-server\"`\n * @docLink packages/core/concepts#from-mcp-server-md\n */\nexport function fromMcpServerMd(filePath: string, repoName: string): CatalogEntry {\n const { data: meta } = parseFrontmatter(readFileSync(filePath, \"utf8\"));\n\n // Extract runtime defaults into metadata\n const metadata: Record<string, unknown> = {};\n if (meta.transport) metadata.transport = String(meta.transport);\n if (meta.command) metadata.command = String(meta.command);\n if (Array.isArray(meta.args)) metadata.args = meta.args.map(String);\n if (meta.env && typeof meta.env === \"object\" && !Array.isArray(meta.env)) {\n metadata.env = meta.env as Record<string, string>;\n }\n if (meta.url) metadata.url = String(meta.url);\n if (meta.headers && typeof meta.headers === \"object\" && !Array.isArray(meta.headers)) {\n metadata.headers = meta.headers as Record<string, string>;\n }\n if (meta.recipe && typeof meta.recipe === \"object\" && !Array.isArray(meta.recipe)) {\n metadata.recipe = meta.recipe;\n }\n\n return {\n name: String(meta.name ?? basename(dirname(filePath))),\n kind: \"mcp-server\",\n description: String(meta.description ?? \"\"),\n source: filePath,\n publisher: repoName,\n version: extractVersion(meta),\n requires: parseRequires(meta),\n dependencies: [],\n metadata: Object.keys(metadata).length > 0 ? metadata : undefined,\n };\n}\n\n/**\n * Parse a `knowledge.yaml` file into a `CatalogEntry` with `kind: \"knowledge\"`.\n * Falls back to the parent directory name when `name:` is absent.\n *\n * @param filePath - Absolute path to the `knowledge.yaml` file\n * @param repoName - Repository name used as the `repository` field of the entry\n * @returns Populated `CatalogEntry` with kind `\"knowledge\"`\n * @docLink packages/core/concepts#from-knowledge-yaml\n */\nexport function fromKnowledgeYaml(filePath: string, repoName: string): CatalogEntry {\n let meta: Record<string, unknown> = {};\n try {\n meta = (parse(readFileSync(filePath, \"utf8\")) as Record<string, unknown>) ?? {};\n } catch {}\n return {\n name: String(meta.name ?? basename(dirname(filePath))),\n kind: \"knowledge\",\n description: String(meta.description ?? \"\"),\n source: filePath,\n publisher: repoName,\n version: String(meta.version ?? \"\"),\n requires: parseRequires(meta),\n dependencies: [],\n };\n}\n\n/**\n * Parse a `CONNECTOR.md` file into a `CatalogEntry` with `kind: \"connector\"`.\n * Falls back to the parent directory name when `name:` is absent.\n *\n * Frontmatter fields (catalog identity):\n * - `name` — connector name (default: parent dir name)\n * - `description` — short description\n * - `version` — semver\n *\n * Frontmatter fields (runtime defaults, stored in `metadata`):\n * - `entry` — adapter entry point (default: \"./adapter.ts\")\n * - `npm_deps` — npm dependencies required by the adapter\n * - `fields` — configuration fields the adapter expects\n * - `health_check` — whether to run a health check on connect\n *\n * The markdown body (below the frontmatter) is stored in `metadata.body`\n * so the runner can inject the connector's skill description into the\n * system prompt when `expose_as_skill` is enabled.\n *\n * @param filePath - Absolute path to the `CONNECTOR.md` file\n * @param repoName - Repository name used as the `repository` field of the entry\n * @returns Populated `CatalogEntry` with kind `\"connector\"`\n * @docLink packages/core/concepts#from-connector-md\n */\nexport function fromConnectorMd(filePath: string, repoName: string): CatalogEntry {\n const text = readFileSync(filePath, \"utf-8\");\n const { data: meta, body } = parseFrontmatter(text);\n const name = String(meta.name ?? basename(dirname(filePath)));\n return {\n name,\n kind: \"connector\" as AssetKind,\n description: String(meta.description ?? \"\"),\n source: filePath,\n publisher: repoName,\n version: extractVersion(meta),\n requires: parseRequires(meta),\n dependencies: [],\n metadata: {\n entry: meta.entry ?? \"./adapter.ts\",\n npm_deps: meta.npm_deps ?? {},\n fields: meta.fields ?? [],\n health_check: meta.health_check ?? null,\n body,\n },\n };\n}\n\n// ── Recursive walker ──────────────────────────────────────────────────────────\n\ninterface WalkEntry {\n path: string;\n name: string;\n}\n\n/**\n * Recursively yield every file entry under `dir`, sorted alphabetically at each level.\n * Skips `.git` and `node_modules` directories and follows symlinks (resolving to\n * the stat target). `node_modules` is excluded because Bun's `node_modules/.bun/...`\n * layout contains symlink farms that form cycles when packages reference each other\n * transitively — entering them causes unbounded recursion in this walker.\n *\n * @param dir - Absolute path to the directory to walk\n * @docLink packages/core/concepts#walk-dir\n */\nexport function* walkDir(dir: string): Generator<WalkEntry> {\n let dirEntries: import(\"node:fs\").Dirent<string>[];\n try {\n dirEntries = readdirSync(dir, { withFileTypes: true, encoding: \"utf8\" });\n } catch {\n return;\n }\n for (const e of dirEntries.sort((a, b) => (a.name < b.name ? -1 : 1))) {\n const full = join(dir, e.name);\n if (e.isSymbolicLink()) {\n try {\n if (statSync(full).isDirectory()) yield* walkDir(full);\n else yield { path: full, name: e.name };\n } catch {}\n } else if (e.isDirectory()) {\n if (e.name !== \".git\" && e.name !== \"node_modules\") yield* walkDir(full);\n } else if (e.isFile()) {\n yield { path: full, name: e.name };\n }\n }\n}\n\n// ── Public scanner ────────────────────────────────────────────────────────────\n\n/**\n * Directory names that hold assets of a given kind. The parent of one of these\n * is the asset's domain — the folder that groups skills/flows/agents/... that\n * belong together.\n */\nconst KIND_DIRS = new Set([\n \"skills\",\n \"agents\",\n \"agent\",\n \"flows\",\n \"connectors\",\n \"prompts\",\n \"bundles\",\n \"contracts\",\n \"knowledge\",\n \"mcp-servers\",\n \"mcp\",\n \"profiles\",\n]);\n\n/**\n * Manifest filenames that live INSIDE a folder named after the asset itself\n * (`<domain.../>/<asset-name>/SKILL.md`). For these, the asset's parent path\n * is two segments above the file.\n *\n * Bare-file manifests (`*.prompt.md`, `*.bundle.yaml`, `*.flow.yaml`,\n * `*.flow.json`) sit directly under the domain folder, so their parent path\n * is one segment above the file.\n */\nconst FOLDER_MANIFESTS = new Set([\n \"SKILL.md\",\n \"AGENT.md\",\n \"agent.yaml\",\n \"CONTRACT.md\",\n \"MCP.md\",\n \"CONNECTOR.md\",\n \"knowledge.yaml\",\n]);\n\n/**\n * Derive the domain for an asset.\n *\n * 1. If any segment in the path matches a known kind directory (`skills/`,\n * `flows/`, ...), return the **full path** from root to that kind dir's\n * parent (e.g. `skaileup/impl-quality/contracts/x/CONTRACT.md` →\n * `\"skaileup/impl-quality\"`). Returning only the immediate parent would\n * break hierarchy when the layout nests `<wrapper>/<domain>/<kind>/...`.\n * 2. Otherwise return the full path from root to the asset's parent folder,\n * `/`-joined. The number of segments stripped depends on whether the\n * manifest is folder-style (strip file + asset folder) or bare-file\n * (strip just the file).\n */\nfunction deriveDomain(root: string, filePath: string): string {\n const segs = relative(root, filePath).split(\"/\");\n for (let i = 1; i < segs.length; i++) {\n if (KIND_DIRS.has(segs[i]!)) return segs.slice(0, i).join(\"/\");\n }\n const fileName = segs[segs.length - 1] ?? \"\";\n const stripCount = FOLDER_MANIFESTS.has(fileName) ? 2 : 1;\n if (segs.length > stripCount) return segs.slice(0, segs.length - stripCount).join(\"/\");\n return \"\";\n}\n\n/**\n * Walk `root` recursively and return a `CatalogEntry` for every recognized asset manifest found.\n *\n * Recognized filenames: `SKILL.md`, `AGENT.md`, `agent.yaml`, `CONTRACT.md`, `MCP.md`,\n * `CONNECTOR.md`, `*.prompt.md`, `*.bundle.yaml`, `*.flow.yaml`, `*.flow.json`.\n *\n * When two entries share the same `kind:name` key, the one with the alphabetically\n * later source path wins (consistent with catalog precedence rules).\n *\n * @param root - Absolute path to the directory tree to scan\n * @param repoName - Repository name attached to every produced entry\n * @returns Array of deduplicated `CatalogEntry` objects, one per manifest file\n * @docLink packages/core/concepts#scan-directory\n */\nexport function scanDirectory(root: string, repoName: string): CatalogEntry[] {\n const entries = new Map<string, CatalogEntry>();\n\n function add(e: CatalogEntry): void {\n const key = `${e.kind}:${e.name}`;\n const existing = entries.get(key);\n if (!existing) {\n entries.set(key, e);\n return;\n }\n // Among duplicates, alphabetically later source path wins\n if (e.source > existing.source) entries.set(key, e);\n }\n\n for (const { path: filePath, name } of walkDir(root)) {\n const domain = deriveDomain(root, filePath);\n try {\n if (name === \"SKILL.md\") add({ ...fromSkillMd(filePath, repoName), domain });\n else if (name === \"AGENT.md\") add({ ...fromAgentMd(filePath, repoName), domain });\n else if (name === \"agent.yaml\") add({ ...fromAgentYaml(filePath, repoName), domain });\n else if (name === \"CONTRACT.md\") add({ ...fromContractMd(filePath, repoName), domain });\n else if (name === \"MCP.md\") add({ ...fromMcpServerMd(filePath, repoName), domain });\n else if (name === \"CONNECTOR.md\") add({ ...fromConnectorMd(filePath, repoName), domain });\n else if (name === \"knowledge.yaml\") add({ ...fromKnowledgeYaml(filePath, repoName), domain });\n else if (name.endsWith(\".prompt.md\")) add({ ...fromPromptMd(filePath, repoName), domain });\n else if (name.endsWith(\".bundle.yaml\"))\n add({ ...fromBundleYaml(filePath, repoName), domain });\n else if (name.endsWith(\".flow.yaml\") || name.endsWith(\".flow.json\"))\n add({ ...fromFlowYaml(filePath, repoName), domain });\n } catch {}\n }\n\n return [...entries.values()];\n}\n"]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../transport/src/ws/client.ts"],"names":[],"mappings":";AASA,IAAM,8BAAA,GAAiC,CAAA;AACvC,IAAM,yBAAA,GAA4B,GAAA;AAClC,IAAM,wBAAA,GAA2B,GAAA;AACjC,IAAM,oBAAA,GAAuB,GAAA;AAC7B,IAAM,0BAAA,GAA6B,GAAA;AA2C5B,IAAM,2BAAN,MAA0D;AAAA,EACvD,EAAA,GAAuB,IAAA;AAAA,EACd,GAAA;AAAA,EACA,eAAA;AAAA,EACA,WAAA;AAAA,EACA,gBAAA;AAAA,EACA,GAAA;AAAA,EAET,cAAA,GAAwD,IAAA;AAAA,EACxD,iBAAA,GAAoB,CAAA;AAAA,EACpB,gBAAA,GAAmB,KAAA;AAAA,EAEV,eAAA,uBAAsB,GAAA,EAAiC;AAAA,EACvD,kBAAA,uBAAyB,GAAA,EAAgB;AAAA,EAE1D,YAAY,OAAA,EAAiC;AAC3C,IAAA,IAAA,CAAK,MAAM,OAAA,CAAQ,GAAA;AACnB,IAAA,IAAA,CAAK,eAAA,GAAkB;AAAA,MACrB,WAAA,EAAa,OAAA,CAAQ,SAAA,EAAW,WAAA,IAAe,8BAAA;AAAA,MAC/C,MAAA,EAAQ,OAAA,CAAQ,SAAA,EAAW,MAAA,IAAU,yBAAA;AAAA,MACrC,KAAA,EAAO,OAAA,CAAQ,SAAA,EAAW,KAAA,IAAS;AAAA,KACrC;AACA,IAAA,IAAA,CAAK,WAAA,GAAc,QAAQ,WAAA,IAAe,oBAAA;AAC1C,IAAA,IAAA,CAAK,gBAAA,GAAmB,QAAQ,gBAAA,IAAoB,0BAAA;AACpD,IAAA,IAAA,CAAK,GAAA,GAAM,OAAA,CAAQ,KAAA,KAAU,MAAM;AAAA,IAAC,CAAA,CAAA;AAAA,EACtC;AAAA,EAEA,MAAM,OAAA,GAAyB;AAC7B,IAAA,IAAA,CAAK,gBAAA,GAAmB,KAAA;AACxB,IAAA,IAAA,CAAK,iBAAA,GAAoB,CAAA;AACzB,IAAA,OAAO,KAAK,SAAA,EAAU;AAAA,EACxB;AAAA,EAEA,MAAM,UAAA,GAA4B;AAChC,IAAA,IAAA,CAAK,gBAAA,GAAmB,IAAA;AACxB,IAAA,IAAA,CAAK,cAAA,EAAe;AACpB,IAAA,IAAI,KAAK,EAAA,EAAI;AACX,MAAA,IAAI,IAAA,CAAK,EAAA,CAAG,UAAA,KAAe,SAAA,CAAU,IAAA,EAAM;AACzC,QAAA,IAAA,CAAK,EAAA,CAAG,KAAA,CAAM,GAAA,EAAM,mBAAmB,CAAA;AAAA,MACzC;AACA,MAAA,IAAA,CAAK,EAAA,GAAK,IAAA;AAAA,IACZ;AAAA,EACF;AAAA,EAEA,KAAK,OAAA,EAA6B;AAChC,IAAA,IAAI,CAAC,IAAA,CAAK,EAAA,IAAM,KAAK,EAAA,CAAG,UAAA,KAAe,UAAU,IAAA,EAAM;AACrD,MAAA,IAAA,CAAK,IAAI,wCAAwC,CAAA;AACjD,MAAA;AAAA,IACF;AACA,IAAA,IAAA,CAAK,EAAA,CAAG,IAAA,CAAK,IAAA,CAAK,SAAA,CAAU,OAAO,CAAC,CAAA;AAAA,EACtC;AAAA,EAEA,UAAU,OAAA,EAAkD;AAC1D,IAAA,IAAA,CAAK,eAAA,CAAgB,IAAI,OAAO,CAAA;AAChC,IAAA,OAAO,MAAM;AACX,MAAA,IAAA,CAAK,eAAA,CAAgB,OAAO,OAAO,CAAA;AAAA,IACrC,CAAA;AAAA,EACF;AAAA,EAEA,aAAa,OAAA,EAAiC;AAC5C,IAAA,IAAA,CAAK,kBAAA,CAAmB,IAAI,OAAO,CAAA;AACnC,IAAA,OAAO,MAAM;AACX,MAAA,IAAA,CAAK,kBAAA,CAAmB,OAAO,OAAO,CAAA;AAAA,IACxC,CAAA;AAAA,EACF;AAAA,EAEA,IAAI,SAAA,GAAqB;AACvB,IAAA,OAAO,IAAA,CAAK,EAAA,EAAI,UAAA,KAAe,SAAA,CAAU,IAAA;AAAA,EAC3C;AAAA;AAAA;AAAA;AAAA,EAMQ,SAAA,GAA2B;AACjC,IAAA,OAAO,IAAI,OAAA,CAAc,CAAC,OAAA,EAAS,MAAA,KAAW;AAC5C,MAAA,IAAA,CAAK,GAAA,CAAI,CAAA,0BAAA,EAA6B,IAAA,CAAK,GAAG,CAAA,CAAE,CAAA;AAEhD,MAAA,MAAM,EAAA,GAAK,IAAI,SAAA,CAAU,IAAA,CAAK,GAAG,CAAA;AACjC,MAAA,IAAA,CAAK,EAAA,GAAK,EAAA;AAEV,MAAA,MAAM,OAAA,GAAU,WAAW,MAAM;AAC/B,QAAA,EAAA,CAAG,KAAA,EAAM;AACT,QAAA,MAAA,CAAO,IAAI,KAAA,CAAM,CAAA,sBAAA,EAAyB,IAAA,CAAK,GAAG,EAAE,CAAC,CAAA;AAAA,MACvD,CAAA,EAAG,KAAK,gBAAgB,CAAA;AAExB,MAAA,EAAA,CAAG,gBAAA,CAAiB,QAAQ,MAAM;AAChC,QAAA,YAAA,CAAa,OAAO,CAAA;AACpB,QAAA,IAAA,CAAK,IAAI,uBAAuB,CAAA;AAChC,QAAA,IAAA,CAAK,iBAAA,GAAoB,CAAA;AACzB,QAAA,IAAA,CAAK,cAAA,EAAe;AACpB,QAAA,OAAA,EAAQ;AAAA,MACV,CAAC,CAAA;AAED,MAAA,EAAA,CAAG,gBAAA,CAAiB,SAAA,EAAW,CAAC,KAAA,KAAU;AACxC,QAAA,IAAI;AACF,UAAA,MAAM,OAAO,IAAA,CAAK,KAAA,CAAM,MAAA,CAAO,KAAA,CAAM,IAAI,CAAC,CAAA;AAC1C,UAAA,KAAA,MAAW,OAAA,IAAW,KAAK,eAAA,EAAiB;AAC1C,YAAA,OAAA,CAAQ,IAAI,CAAA;AAAA,UACd;AAAA,QACF,CAAA,CAAA,MAAQ;AACN,UAAA,IAAA,CAAK,IAAI,qCAAqC,CAAA;AAAA,QAChD;AAAA,MACF,CAAC,CAAA;AAED,MAAA,EAAA,CAAG,gBAAA,CAAiB,OAAA,EAAS,CAAC,KAAA,KAAU;AACtC,QAAA,YAAA,CAAa,OAAO,CAAA;AACpB,QAAA,IAAA,CAAK,cAAA,EAAe;AACpB,QAAA,IAAA,CAAK,GAAA,CAAI,CAAA,+BAAA,EAAkC,KAAA,CAAM,IAAI,CAAA,CAAA,CAAG,CAAA;AAExD,QAAA,KAAA,MAAW,OAAA,IAAW,KAAK,kBAAA,EAAoB;AAC7C,UAAA,OAAA,EAAQ;AAAA,QACV;AAEA,QAAA,IAAI,CAAC,IAAA,CAAK,gBAAA,IAAoB,KAAA,CAAM,SAAS,GAAA,EAAM;AACjD,UAAA,IAAA,CAAK,iBAAA,EAAkB;AAAA,QACzB;AAAA,MACF,CAAC,CAAA;AAED,MAAA,EAAA,CAAG,gBAAA,CAAiB,SAAS,MAAM;AACjC,QAAA,YAAA,CAAa,OAAO,CAAA;AAAA,MAEtB,CAAC,CAAA;AAAA,IACH,CAAC,CAAA;AAAA,EACH;AAAA,EAEQ,iBAAA,GAA0B;AAChC,IAAA,IAAI,IAAA,CAAK,iBAAA,IAAqB,IAAA,CAAK,eAAA,CAAgB,WAAA,EAAa;AAC9D,MAAA,IAAA,CAAK,GAAA,CAAI,CAAA,sCAAA,EAAyC,IAAA,CAAK,iBAAiB,CAAA,SAAA,CAAW,CAAA;AACnF,MAAA;AAAA,IACF;AAEA,IAAA,IAAA,CAAK,iBAAA,EAAA;AACL,IAAA,MAAM,QAAQ,IAAA,CAAK,GAAA;AAAA,MACjB,IAAA,CAAK,eAAA,CAAgB,MAAA,GAAS,CAAA,KAAM,KAAK,iBAAA,GAAoB,CAAA,CAAA;AAAA,MAC7D,KAAK,eAAA,CAAgB;AAAA,KACvB;AACA,IAAA,IAAA,CAAK,IAAI,CAAA,4BAAA,EAA+B,KAAK,CAAA,YAAA,EAAe,IAAA,CAAK,iBAAiB,CAAA,CAAA,CAAG,CAAA;AAErF,IAAA,UAAA,CAAW,MAAM;AACf,MAAA,IAAI,KAAK,gBAAA,EAAkB;AAC3B,MAAA,IAAA,CAAK,SAAA,EAAU,CAAE,KAAA,CAAM,MAAM;AAAA,MAE7B,CAAC,CAAA;AAAA,IACH,GAAG,KAAK,CAAA;AAAA,EACV;AAAA,EAEQ,cAAA,GAAuB;AAC7B,IAAA,IAAA,CAAK,cAAA,EAAe;AACpB,IAAA,IAAA,CAAK,cAAA,GAAiB,YAAY,MAAM;AACtC,MAAA,IAAI,IAAA,CAAK,EAAA,EAAI,UAAA,KAAe,SAAA,CAAU,IAAA,EAAM;AAE1C,QAAA,IAAI;AACF,UAAC,IAAA,CAAK,GAAwC,IAAA,IAAO;AAAA,QACvD,CAAA,CAAA,MAAQ;AAAA,QAER;AAAA,MACF;AAAA,IACF,CAAA,EAAG,KAAK,WAAW,CAAA;AAAA,EACrB;AAAA,EAEQ,cAAA,GAAuB;AAC7B,IAAA,IAAI,KAAK,cAAA,EAAgB;AACvB,MAAA,aAAA,CAAc,KAAK,cAAc,CAAA;AACjC,MAAA,IAAA,CAAK,cAAA,GAAiB,IAAA;AAAA,IACxB;AAAA,EACF;AACF","file":"chunk-7PTP3SQJ.js","sourcesContent":["/**\n * WebSocketClientTransport — connects to a skaile serve WebSocket server.\n *\n * Handles connection, reconnection with exponential backoff, heartbeat,\n * and JSON serialization. Implements ClientTransport from @skaile/workspaces/types.\n */\n\nimport type { AgentCommand, AgentEvent, ClientTransport } from \"@skaile/workspaces/types\";\n\nconst DEFAULT_RECONNECT_MAX_ATTEMPTS = 5;\nconst DEFAULT_RECONNECT_BASE_MS = 1_000;\nconst DEFAULT_RECONNECT_MAX_MS = 30_000;\nconst DEFAULT_HEARTBEAT_MS = 30_000;\nconst DEFAULT_CONNECT_TIMEOUT_MS = 10_000;\n\n/**\n * Options for {@link WebSocketClientTransport}.\n *\n * @param url - WebSocket server URL to connect to (e.g. `ws://localhost:8080`).\n * @param reconnect - Exponential-backoff reconnect policy. Defaults: 5 attempts, 1s base, 30s max.\n * @param heartbeatMs - Interval in milliseconds between WebSocket ping frames. Default: 30 000.\n * @param connectTimeoutMs - Maximum time in milliseconds to wait for the initial handshake. Default: 10 000.\n * @param onLog - Optional callback for internal log lines (connection state, errors).\n *\n * @docLink packages/transport/dev-guide#websocket-client-transport\n */\nexport interface WebSocketClientOptions {\n url: string;\n reconnect?: {\n maxAttempts?: number;\n baseMs?: number;\n maxMs?: number;\n };\n heartbeatMs?: number;\n connectTimeoutMs?: number;\n onLog?: (line: string) => void;\n}\n\n/**\n * `ClientTransport` implementation that connects to a skaile agent server over WebSocket.\n *\n * Uses the browser-standard `WebSocket` API so it works in both Bun and browser\n * environments. On an unintentional disconnect it automatically retries with\n * exponential backoff (default: 5 attempts, 1s base, 30s max). A periodic ping\n * keeps the connection alive through NAT and load-balancer idle timeouts. All\n * `AgentCommand` / `AgentEvent` messages are JSON-serialised on the wire.\n *\n * @example\n * ```ts\n * const transport = new WebSocketClientTransport({ url: 'ws://localhost:8080' });\n * await transport.connect();\n * transport.send({ type: 'prompt', prompt: 'Hello' });\n * ```\n *\n * @docLink packages/transport/dev-guide#websocket-client-transport\n */\nexport class WebSocketClientTransport implements ClientTransport {\n private ws: WebSocket | null = null;\n private readonly url: string;\n private readonly reconnectConfig: Required<NonNullable<WebSocketClientOptions[\"reconnect\"]>>;\n private readonly heartbeatMs: number;\n private readonly connectTimeoutMs: number;\n private readonly log: (line: string) => void;\n\n private heartbeatTimer: ReturnType<typeof setInterval> | null = null;\n private reconnectAttempts = 0;\n private intentionalClose = false;\n\n private readonly messageHandlers = new Set<(event: AgentEvent) => void>();\n private readonly disconnectHandlers = new Set<() => void>();\n\n constructor(options: WebSocketClientOptions) {\n this.url = options.url;\n this.reconnectConfig = {\n maxAttempts: options.reconnect?.maxAttempts ?? DEFAULT_RECONNECT_MAX_ATTEMPTS,\n baseMs: options.reconnect?.baseMs ?? DEFAULT_RECONNECT_BASE_MS,\n maxMs: options.reconnect?.maxMs ?? DEFAULT_RECONNECT_MAX_MS,\n };\n this.heartbeatMs = options.heartbeatMs ?? DEFAULT_HEARTBEAT_MS;\n this.connectTimeoutMs = options.connectTimeoutMs ?? DEFAULT_CONNECT_TIMEOUT_MS;\n this.log = options.onLog ?? (() => {});\n }\n\n async connect(): Promise<void> {\n this.intentionalClose = false;\n this.reconnectAttempts = 0;\n return this.doConnect();\n }\n\n async disconnect(): Promise<void> {\n this.intentionalClose = true;\n this.clearHeartbeat();\n if (this.ws) {\n if (this.ws.readyState === WebSocket.OPEN) {\n this.ws.close(1000, \"Client disconnect\");\n }\n this.ws = null;\n }\n }\n\n send(command: AgentCommand): void {\n if (!this.ws || this.ws.readyState !== WebSocket.OPEN) {\n this.log(\"[ws-client] cannot send: not connected\");\n return;\n }\n this.ws.send(JSON.stringify(command));\n }\n\n onMessage(handler: (event: AgentEvent) => void): () => void {\n this.messageHandlers.add(handler);\n return () => {\n this.messageHandlers.delete(handler);\n };\n }\n\n onDisconnect(handler: () => void): () => void {\n this.disconnectHandlers.add(handler);\n return () => {\n this.disconnectHandlers.delete(handler);\n };\n }\n\n get connected(): boolean {\n return this.ws?.readyState === WebSocket.OPEN;\n }\n\n // ---------------------------------------------------------------------------\n // Internal\n // ---------------------------------------------------------------------------\n\n private doConnect(): Promise<void> {\n return new Promise<void>((resolve, reject) => {\n this.log(`[ws-client] connecting to ${this.url}`);\n\n const ws = new WebSocket(this.url);\n this.ws = ws;\n\n const timeout = setTimeout(() => {\n ws.close();\n reject(new Error(`Connection timeout to ${this.url}`));\n }, this.connectTimeoutMs);\n\n ws.addEventListener(\"open\", () => {\n clearTimeout(timeout);\n this.log(\"[ws-client] connected\");\n this.reconnectAttempts = 0;\n this.startHeartbeat();\n resolve();\n });\n\n ws.addEventListener(\"message\", (event) => {\n try {\n const data = JSON.parse(String(event.data)) as AgentEvent;\n for (const handler of this.messageHandlers) {\n handler(data);\n }\n } catch {\n this.log(\"[ws-client] failed to parse message\");\n }\n });\n\n ws.addEventListener(\"close\", (event) => {\n clearTimeout(timeout);\n this.clearHeartbeat();\n this.log(`[ws-client] disconnected (code=${event.code})`);\n\n for (const handler of this.disconnectHandlers) {\n handler();\n }\n\n if (!this.intentionalClose && event.code !== 1000) {\n this.scheduleReconnect();\n }\n });\n\n ws.addEventListener(\"error\", () => {\n clearTimeout(timeout);\n // The close event fires after error, handling reconnection\n });\n });\n }\n\n private scheduleReconnect(): void {\n if (this.reconnectAttempts >= this.reconnectConfig.maxAttempts) {\n this.log(`[ws-client] giving up reconnect after ${this.reconnectAttempts} attempts`);\n return;\n }\n\n this.reconnectAttempts++;\n const delay = Math.min(\n this.reconnectConfig.baseMs * 2 ** (this.reconnectAttempts - 1),\n this.reconnectConfig.maxMs,\n );\n this.log(`[ws-client] reconnecting in ${delay}ms (attempt ${this.reconnectAttempts})`);\n\n setTimeout(() => {\n if (this.intentionalClose) return;\n this.doConnect().catch(() => {\n // doConnect rejection triggers another close → scheduleReconnect cycle\n });\n }, delay);\n }\n\n private startHeartbeat(): void {\n this.clearHeartbeat();\n this.heartbeatTimer = setInterval(() => {\n if (this.ws?.readyState === WebSocket.OPEN) {\n // WebSocket ping — Bun's WebSocket supports ping natively\n try {\n (this.ws as unknown as { ping?: () => void }).ping?.();\n } catch {\n // Fallback: some environments don't support ping on the client\n }\n }\n }, this.heartbeatMs);\n }\n\n private clearHeartbeat(): void {\n if (this.heartbeatTimer) {\n clearInterval(this.heartbeatTimer);\n this.heartbeatTimer = null;\n }\n }\n}\n"]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../core/src/models.ts"],"names":[],"mappings":"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","file":"chunk-I3UEM3FX.js","sourcesContent":["/**\n * Pure data models — no I/O, no side effects.\n */\n\nimport semver from \"semver\";\n\n/**\n * Discriminator for catalog entries and asset references.\n * Used in asset ref strings (e.g. `'skill:my-skill'`, `'agent:my-agent'`).\n * @docLink packages/core/concepts#asset-kind\n */\nexport type AssetKind =\n | \"skill\"\n | \"agent\"\n | \"prompt\"\n | \"bundle\"\n | \"flow\"\n | \"contract\"\n | \"mcp-server\"\n | \"persona\" // identity fragment — replaces SOUL.md in composition\n | \"ruleset\" // behavioral constraints — replaces RULES.md in composition\n | \"knowledge\" // knowledge bundle directory\n | \"connector\"; // connector descriptor (CONNECTOR.md)\n\n/**\n * All valid AssetKind values as a readonly array.\n * @docLink packages/core/concepts#asset-kinds\n */\nexport const ASSET_KINDS: readonly AssetKind[] = [\n \"skill\",\n \"agent\",\n \"prompt\",\n \"bundle\",\n \"flow\",\n \"contract\",\n \"mcp-server\",\n \"persona\",\n \"ruleset\",\n \"knowledge\",\n \"connector\",\n];\n/**\n * Asset kinds that represent individually installable assets (excludes \"bundle\").\n * Note: persona, ruleset, and knowledge are valid AssetKind values but are not\n * individually installable — they are resolved at session creation time via the\n * mixin-resolver, not through the install/deploy pipeline.\n * @docLink packages/core/concepts#individual-kinds\n */\nexport const INDIVIDUAL_KINDS = [\n \"skill\",\n \"agent\",\n \"prompt\",\n \"flow\",\n \"contract\",\n \"mcp-server\",\n] as const satisfies AssetKind[];\n\n// ── Dependency ────────────────────────────────────────────────────────────────\n\n/**\n * Declared dependency on another catalog asset.\n * @docLink packages/core/concepts#dependency\n */\nexport interface Dependency {\n /** Asset kind (e.g. \"skill\", \"agent\", \"contract\"). */\n kind: string;\n /** Asset name within that kind. */\n name: string;\n /** Canonical publisher namespace (when known). */\n publisher?: string;\n}\n\n/**\n * Parse a `\"kind:name\"` dependency string, or a bare `\"name\"` (defaults kind to `\"skill\"`).\n *\n * @param s - Dependency string, e.g. `\"skill:use-exa\"` or `\"use-exa\"`\n * @returns Parsed `{ kind, name }` dependency object\n * @docLink packages/core/concepts#parse-dep\n */\nexport function parseDep(s: string): Dependency {\n const i = s.indexOf(\":\");\n if (i !== -1) {\n const kind = s.slice(0, i).trim();\n const name = s.slice(i + 1).trim();\n return { kind, name };\n }\n return { kind: \"skill\", name: s.trim() };\n}\n\n/**\n * Serialize a Dependency back to its `\"kind:name\"` string form.\n *\n * @param d - Dependency to serialize\n * @returns String in `\"kind:name\"` format\n * @docLink packages/core/concepts#dep-to-str\n */\nexport function depToStr(d: Dependency): string {\n return `${d.kind}:${d.name}`;\n}\n\n// ── CatalogEntry ──────────────────────────────────────────────────────────────\n\n/**\n * A resolved catalog entry describing an installable asset.\n * Produced by manifest parsers and stored in the catalog YAML.\n * @docLink packages/core/concepts#catalog-entry\n */\nexport interface CatalogEntry {\n /** Unique asset name within its kind. */\n name: string;\n /** Asset kind discriminator. */\n kind: AssetKind;\n /** Short description of what this asset does. */\n description: string;\n /** Absolute path to manifest file */\n source: string;\n /** Canonical publisher namespace this asset belongs to (formerly `repository`). */\n publisher?: string;\n /** Domain this asset belongs to (top-level directory in the repo root). */\n domain?: string;\n /** Semantic version string (may be empty if not declared in the manifest). */\n version: string;\n /** Skills, agents, or other assets this entry depends on. */\n requires: Dependency[];\n /** Bundle dependencies — asset refs to install (bundles only). */\n dependencies: string[];\n /**\n * Kind-specific metadata extracted from the manifest frontmatter.\n * Used by MCP servers to carry default config (transport, command, args, env, url, headers)\n * so that `mcp:name` dependency refs resolve to a runnable declaration without\n * requiring verbose inline config in `skaile.yaml`.\n */\n metadata?: Record<string, unknown>;\n}\n\n/**\n * Construct a CatalogEntry from a raw deserialized record (e.g. from catalog.yaml).\n *\n * @param d - Raw object with loosely typed fields\n * @returns Normalized CatalogEntry\n * @docLink packages/core/concepts#entry-from-raw\n */\nexport function entryFromRaw(d: Record<string, unknown>): CatalogEntry {\n const rawDeps = d.dependencies;\n const deps = Array.isArray(rawDeps) ? (rawDeps as string[]).filter(Boolean) : [];\n const entry: CatalogEntry = {\n name: String(d.name ?? \"\"),\n kind: String(d.kind ?? \"skill\") as AssetKind,\n description: String(d.description ?? \"\"),\n source: String(d.source ?? \"\"),\n publisher: d.publisher != null ? String(d.publisher) : undefined,\n domain: d.domain != null ? String(d.domain) : undefined,\n version: String(d.version ?? \"\"),\n requires: Array.isArray(d.requires)\n ? (d.requires as string[]).filter(Boolean).map(parseDep)\n : [],\n dependencies: deps,\n };\n if (d.metadata != null && typeof d.metadata === \"object\" && !Array.isArray(d.metadata)) {\n entry.metadata = d.metadata as Record<string, unknown>;\n }\n return entry;\n}\n\n/**\n * Serialize a CatalogEntry to a plain record suitable for YAML/JSON persistence.\n *\n * @param e - CatalogEntry to serialize\n * @returns Plain object with string-serialized requires and dependencies\n * @docLink packages/core/concepts#entry-to-raw\n */\nexport function entryToRaw(e: CatalogEntry): Record<string, unknown> {\n const raw: Record<string, unknown> = {\n name: e.name,\n kind: e.kind,\n description: e.description,\n source: e.source,\n publisher: e.publisher,\n domain: e.domain,\n version: e.version,\n requires: e.requires.map(depToStr),\n dependencies: e.dependencies,\n };\n if (e.metadata) raw.metadata = e.metadata;\n return raw;\n}\n\n// ── AssetRef ─────────────────────────────────────────────────────────────────\n\n/**\n * A parsed asset reference: `kind:name@<publisher>[#pin]`.\n *\n * `publisher` is the canonical, GitHub-shaped publisher namespace (≤39 chars,\n * alphanumeric+hyphen). `pin` is a SemVer constraint, exact SHA, or absent.\n * Floating refs (`main`, `latest`, `HEAD`) are rejected at parse time.\n *\n * Produced by `parseAssetRef`; consumed by `resolveAsset` and `resolveAll`.\n * @docLink packages/core/concepts#asset-ref\n */\nexport interface AssetRef {\n kind: string;\n name: string;\n /** Canonical publisher namespace. Optional only for ambiguity-resolution\n * during migration; new manifests require it. */\n publisher?: string;\n /** SemVer constraint (^1.4.0, ~1.4, 1.x, exact 1.4.0), 40-char SHA, or undefined. */\n pin?: string;\n}\n\n// GitHub-shaped: 1-39 chars, alphanumeric + single hyphens, no leading/trailing\n// hyphen, no double hyphens.\nconst PUBLISHER_RE =\n /^[A-Za-z0-9](?:[A-Za-z0-9]|-(?!-)){0,37}[A-Za-z0-9]$|^[A-Za-z0-9]$/;\nconst SHA_RE = /^[0-9a-f]{40}$/i;\n\n/** True when `pin` is a SemVer constraint, exact SemVer, or 40-char SHA. */\nfunction isValidPin(pin: string): boolean {\n if (SHA_RE.test(pin)) return true;\n if (semver.valid(pin)) return true;\n if (semver.validRange(pin)) return true;\n return false;\n}\n\n/**\n * Parse a canonical asset reference string into an `AssetRef` object.\n *\n * Grammar: `kind:name@<publisher>[#pin]`\n * \"skill:audit@skaile-ai\" → { kind: \"skill\", name: \"audit\", publisher: \"skaile-ai\" }\n * \"skill:audit@skaile-ai#^1.4.0\" → { …, pin: \"^1.4.0\" }\n * \"skill:audit@skaile-ai#<40-sha>\" → { …, pin: \"<40-sha>\" }\n *\n * `<publisher>` is required and GitHub-shaped (≤39 chars, alphanumeric+hyphen,\n * no leading/trailing or double hyphen). `pin` accepts a SemVer constraint\n * (`^`, `~`, `x`, exact), a 40-char SHA, or is absent. Floating refs\n * (`main`/`latest`/`HEAD`/anything non-canonical) are parse-time errors.\n *\n * The shorthand `mcp:` is normalized to `mcp-server:`.\n *\n * @param s - Asset reference string\n * @returns Parsed `AssetRef`\n * @throws When the publisher is missing/malformed or the pin is a floating ref.\n * @docLink packages/core/concepts#parse-asset-ref\n */\nexport function parseAssetRef(s: string): AssetRef {\n let rest = s.trim();\n let kind = \"skill\";\n\n const colonIdx = rest.indexOf(\":\");\n if (colonIdx !== -1) {\n kind = rest.slice(0, colonIdx).trim();\n rest = rest.slice(colonIdx + 1).trim();\n }\n if (kind === \"mcp\") kind = \"mcp-server\";\n\n let pin: string | undefined;\n const hashIdx = rest.indexOf(\"#\");\n if (hashIdx !== -1) {\n pin = rest.slice(hashIdx + 1).trim() || undefined;\n rest = rest.slice(0, hashIdx).trim();\n }\n\n let publisher: string | undefined;\n const atIdx = rest.indexOf(\"@\");\n if (atIdx !== -1) {\n publisher = rest.slice(atIdx + 1).trim();\n rest = rest.slice(0, atIdx).trim();\n if (publisher.length === 0) {\n throw new Error(`empty publisher in asset ref: \"${s}\"`);\n }\n if (!PUBLISHER_RE.test(publisher)) {\n if (publisher.length > 39) {\n throw new Error(`publisher length exceeds 39 chars: \"${publisher}\"`);\n }\n throw new Error(`publisher is not GitHub-shaped: \"${publisher}\"`);\n }\n } else {\n throw new Error(\n `publisher required in asset ref \"${s}\" — use \\`kind:name@<publisher>\\`. ` +\n \"Run the `migrate-skaile-manifest` skill on legacy manifests.\",\n );\n }\n\n if (pin !== undefined && !isValidPin(pin)) {\n throw new Error(\n `non-canonical pin \"${pin}\" in \"${s}\" — pin must be a SemVer ` +\n \"constraint (1.4.0, ^1.4, ~1.4, 1.x), a 40-char SHA, or absent. \" +\n \"Floating refs (main/latest/HEAD) are not allowed.\",\n );\n }\n\n return { kind, name: rest, publisher, pin };\n}\n\n/**\n * Serialize an AssetRef back to its canonical string form.\n *\n * @param ref - AssetRef to serialize\n * @returns String in `\"kind:name[@repository][#pin]\"` format\n * @docLink packages/core/concepts#asset-ref-to-str\n */\nexport function assetRefToStr(ref: AssetRef): string {\n let s = `${ref.kind}:${ref.name}`;\n if (ref.publisher) s += `@${ref.publisher}`;\n if (ref.pin) s += `#${ref.pin}`;\n return s;\n}\n\n/**\n * Convert an AssetRef to a simple Dependency, dropping the repository and pin qualifiers.\n *\n * @param ref - Full asset reference (may include repository and pin)\n * @returns Dependency with only kind and name\n * @docLink packages/core/concepts#asset-ref-to-dep\n */\nexport function assetRefToDep(ref: AssetRef): Dependency {\n return { kind: ref.kind, name: ref.name };\n}\n\n// ── Lock File ────────────────────────────────────────────────────────────────\n\n/** A path/sha256 pair recorded for one file inside a locked asset. */\nexport interface LockFileEntry {\n path: string;\n sha256: string;\n}\n\n/**\n * Per-asset entry in the lock file (v2 schema).\n * Keyed in `LockFile.assets` by canonical ref (`<publisher>/<kind>:<name>@<version>`).\n * @docLink packages/core/api-reference#lock-entry\n */\nexport interface LockEntry {\n /** Composite sha256 over sorted `<path>:<sha256>\\n` rows. */\n sha256: string;\n /** Resolved upstream source + commit. */\n source: { url: string; commit: string };\n /** Files that constitute the asset. */\n files: LockFileEntry[];\n /** True when an `overrides[]` entry pinned this resolution. */\n override_applied: boolean;\n}\n\n/** Per-source-URL entry in `LockFile.sources`. */\nexport interface LockSourceEntry {\n url: string;\n commit: string;\n}\n\n/**\n * Full skaile.lock.yaml structure (v2 schema).\n * @docLink packages/core/api-reference#lock-file\n */\nexport interface LockFile {\n /** Lock file schema version. v2 is canonical. */\n schema_version: 2;\n /** ISO timestamp of when the lock was generated. */\n locked_at: string;\n /** Resolved assets keyed by canonical ref `<publisher>/<kind>:<name>@<version>`. */\n assets: Record<string, LockEntry>;\n /** Every source URL that contributed at least one resolved asset. */\n sources: LockSourceEntry[];\n /**\n * Resolved plugin packages (from `skaile.yaml` `plugins:`), keyed by package\n * name. The plugin-store reconciler reads this slice into its reconcile-hash\n * so a lock change reinstalls even when the manifest list is unchanged.\n */\n plugins?: Record<string, LockPluginEntry>;\n}\n\n/** One pinned plugin package in the lock file's `plugins` slice. */\nexport interface LockPluginEntry {\n /** Resolved exact version (e.g. \"0.1.3\"). */\n version: string;\n /** Package integrity hash (e.g. \"sha512-...\"); may be empty until populated from bun's output. */\n integrity: string;\n}\n\n// ── AgentManifest ────────────────────────────────────────────────────────────\n\n/**\n * Agent package manifest — parsed from `agent.yaml` in an agent directory.\n * Defines the agent's identity, model preferences, capability requirements, and composition mixins.\n * @docLink packages/core/concepts#agent-manifest\n */\nexport interface AgentManifest {\n spec_version?: string;\n name?: string;\n version?: string;\n description?: string;\n model?: {\n preferred?: string;\n fallback?: string[];\n constraints?: { temperature?: number; max_tokens?: number };\n };\n tools?: {\n allowed?: string[];\n denied?: string[];\n };\n delegation?: { mode?: string };\n requires?: Array<{ name: string; source: string; version?: string; mount: string }>;\n tags?: string[];\n author?: string;\n metadata?: Record<string, unknown>;\n /**\n * @deprecated Use `persona`, `rules`, and `knowledge` mixin arrays instead.\n * This field is declared but was never resolved at runtime.\n */\n extends?: string;\n abilities?: string[];\n /**\n * Identity fragments to compose into the agent persona.\n * Each entry is a catalog ref (`persona:name@repo#pin`), a bare name\n * (defaults to kind \"persona\"), or a local path (`./my.persona.md`).\n * Resolved in order — concatenated with \\n\\n---\\n\\n separator.\n */\n persona?: string[];\n /**\n * Behavioral rule sets to compose into the agent's constraints.\n * Each entry is a catalog ref (`ruleset:name@repo#pin`), bare name,\n * or local path. All rules apply (union).\n */\n rules?: string[];\n /**\n * Knowledge bundles to compose into the agent's domain knowledge.\n * Each entry is a catalog ref (`knowledge:name@repo#pin`), bare name,\n * or local path to a knowledge directory. Loaded in declaration order.\n */\n knowledge?: string[];\n contracts?: string[];\n /**\n * v2 composition items. Each entry references an asset (skill, connector,\n * soul, ruleset, etc.) with binding semantics (discoverable, inline-live,\n * inline-snapshot). Resolved at session start via the composition module.\n */\n composes?: Array<{\n kind: string;\n ref: string;\n instance?: string;\n binding?: \"discoverable\" | \"inline-live\" | \"inline-snapshot\";\n }>;\n runtime?: {\n max_turns?: number;\n timeout?: number;\n [key: string]: unknown;\n };\n}\n\n// ── Repository ───────────────────────────────────────────────────────────────\n\ntype RepositoryKind = \"local\" | \"github\";\n\n/**\n * A registered asset repository (local path or cloned GitHub repo).\n * Stored in the catalog; consumed by `repo-manager` for cloning and scanning.\n * @docLink packages/core/concepts#repository\n */\nexport interface Repository {\n /** Logical repository name. */\n name: string;\n /** Storage kind: \"local\" for paths, \"github\" for remote URLs. */\n kind: RepositoryKind;\n /** Absolute path (local) or remote URL (github). */\n path: string;\n /** Active branch. */\n branch: string;\n /** Upstream fork URL (for fork-based workflows). */\n upstream?: string;\n}\n\n/**\n * Construct a `Repository` from a raw deserialized record.\n *\n * @param d - Raw object from YAML/JSON\n * @returns Normalized `Repository`\n * @docLink packages/core/concepts#repository-from-raw\n */\nexport function repositoryFromRaw(d: Record<string, unknown>): Repository {\n return {\n name: String(d.name ?? \"\"),\n kind: String(d.kind ?? \"local\") as RepositoryKind,\n path: String(d.path ?? \"\"),\n branch: String(d.branch ?? \"main\"),\n upstream: d.upstream != null ? String(d.upstream) : undefined,\n };\n}\n\n/**\n * Serialize a `Repository` to a plain record for YAML/JSON persistence.\n * Omits the `branch` field when it equals `\"main\"` to keep serialized output clean.\n *\n * @param r - Repository to serialize\n * @returns Plain object suitable for YAML/JSON\n * @docLink packages/core/concepts#repository-to-raw\n */\nexport function repositoryToRaw(r: Repository): Record<string, unknown> {\n const d: Record<string, unknown> = {\n name: r.name,\n kind: r.kind,\n path: r.path,\n };\n if (r.branch !== \"main\") d.branch = r.branch;\n if (r.upstream != null) d.upstream = r.upstream;\n return d;\n}\n"]}
|