@clinebot/core 0.0.5 → 0.0.7

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.
Files changed (57) hide show
  1. package/dist/agents/hooks-config-loader.d.ts +1 -0
  2. package/dist/index.d.ts +4 -1
  3. package/dist/index.node.d.ts +2 -0
  4. package/dist/index.node.js +134 -107
  5. package/dist/runtime/session-runtime.d.ts +3 -1
  6. package/dist/session/default-session-manager.d.ts +4 -0
  7. package/dist/session/rpc-spawn-lease.d.ts +7 -0
  8. package/dist/session/session-host.d.ts +2 -0
  9. package/dist/session/session-manager.d.ts +1 -0
  10. package/dist/storage/provider-settings-legacy-migration.d.ts +25 -0
  11. package/dist/telemetry/ITelemetryAdapter.d.ts +54 -0
  12. package/dist/telemetry/LoggerTelemetryAdapter.d.ts +21 -0
  13. package/dist/telemetry/OpenTelemetryAdapter.d.ts +43 -0
  14. package/dist/telemetry/OpenTelemetryProvider.d.ts +41 -0
  15. package/dist/telemetry/TelemetryService.d.ts +34 -0
  16. package/dist/telemetry/opentelemetry.d.ts +3 -0
  17. package/dist/telemetry/opentelemetry.js +27 -0
  18. package/dist/tools/schemas.d.ts +8 -8
  19. package/dist/types/config.d.ts +2 -1
  20. package/dist/types/events.d.ts +1 -1
  21. package/package.json +16 -3
  22. package/src/agents/hooks-config-loader.ts +21 -1
  23. package/src/index.node.ts +7 -0
  24. package/src/index.ts +16 -0
  25. package/src/input/file-indexer.test.ts +40 -0
  26. package/src/input/file-indexer.ts +21 -0
  27. package/src/runtime/hook-file-hooks.test.ts +98 -1
  28. package/src/runtime/hook-file-hooks.ts +93 -11
  29. package/src/runtime/runtime-builder.test.ts +20 -0
  30. package/src/runtime/runtime-builder.ts +1 -0
  31. package/src/runtime/session-runtime.ts +3 -1
  32. package/src/session/default-session-manager.test.ts +72 -0
  33. package/src/session/default-session-manager.ts +59 -1
  34. package/src/session/rpc-spawn-lease.test.ts +49 -0
  35. package/src/session/rpc-spawn-lease.ts +122 -0
  36. package/src/session/session-graph.ts +2 -0
  37. package/src/session/session-host.ts +14 -1
  38. package/src/session/session-manager.ts +1 -0
  39. package/src/storage/provider-settings-legacy-migration.test.ts +133 -1
  40. package/src/storage/provider-settings-legacy-migration.ts +60 -8
  41. package/src/telemetry/ITelemetryAdapter.ts +94 -0
  42. package/src/telemetry/LoggerTelemetryAdapter.test.ts +42 -0
  43. package/src/telemetry/LoggerTelemetryAdapter.ts +114 -0
  44. package/src/telemetry/OpenTelemetryAdapter.test.ts +157 -0
  45. package/src/telemetry/OpenTelemetryAdapter.ts +348 -0
  46. package/src/telemetry/OpenTelemetryProvider.test.ts +113 -0
  47. package/src/telemetry/OpenTelemetryProvider.ts +322 -0
  48. package/src/telemetry/TelemetryService.test.ts +134 -0
  49. package/src/telemetry/TelemetryService.ts +141 -0
  50. package/src/telemetry/opentelemetry.ts +20 -0
  51. package/src/tools/definitions.test.ts +82 -29
  52. package/src/tools/definitions.ts +41 -32
  53. package/src/tools/executors/editor.test.ts +35 -0
  54. package/src/tools/executors/editor.ts +33 -46
  55. package/src/tools/schemas.ts +34 -35
  56. package/src/types/config.ts +2 -0
  57. package/src/types/events.ts +6 -1
@@ -162,6 +162,53 @@ export interface MigrateLegacyProviderSettingsResult {
162
162
  lastUsedProvider?: string;
163
163
  }
164
164
 
165
+ export type LegacyClineUserInfo = {
166
+ idToken: string;
167
+ expiresAt: number;
168
+ refreshToken: string;
169
+ userInfo: {
170
+ id: string;
171
+ email: string;
172
+ displayName: string;
173
+ termsAcceptedAt: string;
174
+ clineBenchConsent: boolean;
175
+ createdAt: string;
176
+ updatedAt: string;
177
+ };
178
+ provider: string;
179
+ startedAt: number;
180
+ };
181
+
182
+ /**
183
+ * Resolves legacy Cline account auth data from the raw `cline:clineAccountId`
184
+ * secret string into the auth fields used by `ProviderSettings`.
185
+ *
186
+ * Returns `undefined` when the input is missing, empty, whitespace-only, or
187
+ * unparseable JSON.
188
+ */
189
+ export function resolveLegacyClineAuth(
190
+ rawAccountData: string | undefined,
191
+ ): ProviderSettings["auth"] | undefined {
192
+ const trimmed = rawAccountData?.trim();
193
+ if (!trimmed) {
194
+ return undefined;
195
+ }
196
+ try {
197
+ const data = JSON.parse(trimmed) as LegacyClineUserInfo;
198
+ if (!data) {
199
+ return undefined;
200
+ }
201
+ return {
202
+ accessToken: data.idToken,
203
+ refreshToken: data.refreshToken,
204
+ expiresAt: data.expiresAt,
205
+ accountId: data.userInfo?.id,
206
+ };
207
+ } catch {
208
+ return undefined;
209
+ }
210
+ }
211
+
165
212
  function trimNonEmpty(value: string | undefined): string | undefined {
166
213
  const trimmed = value?.trim();
167
214
  return trimmed ? trimmed : undefined;
@@ -400,14 +447,19 @@ function buildLegacyProviderSettings(
400
447
  Object.assign(providerSpecific, resolveLegacyCodexAuth(legacySecrets));
401
448
  }
402
449
  if (providerId === "cline") {
403
- const accountId = trimNonEmpty(
404
- legacySecrets["cline:clineAccountId"] ?? legacySecrets.clineAccountId,
405
- );
406
- if (accountId) {
407
- providerSpecific.auth = {
408
- ...(providerSpecific.auth ?? {}),
409
- accountId,
410
- };
450
+ try {
451
+ const legacyAuthString = trimNonEmpty(
452
+ legacySecrets["cline:clineAccountId"],
453
+ );
454
+
455
+ if (legacyAuthString) {
456
+ providerSpecific.auth = {
457
+ ...(providerSpecific.auth ?? {}),
458
+ ...resolveLegacyClineAuth(legacyAuthString),
459
+ };
460
+ }
461
+ } catch {
462
+ // Failed to parse stored cline auth data
411
463
  }
412
464
  }
413
465
  if (providerId === "openai" && legacyGlobalState.openAiHeaders) {
@@ -0,0 +1,94 @@
1
+ /**
2
+ * Telemetry adapter interface for the @clinebot/core SDK.
3
+ *
4
+ * This is the SDK-side counterpart to the extension's ITelemetryProvider.
5
+ * It is intentionally free of VS Code / host-provider dependencies so that
6
+ * any consumer (CLI, tests, third-party integrations) can plug in their own
7
+ * backend without pulling in the full extension runtime.
8
+ */
9
+
10
+ import type { TelemetryProperties } from "@clinebot/shared";
11
+
12
+ export type {
13
+ TelemetryArray,
14
+ TelemetryMetadata,
15
+ TelemetryObject,
16
+ TelemetryPrimitive,
17
+ TelemetryProperties,
18
+ TelemetryValue,
19
+ } from "@clinebot/shared";
20
+
21
+ // ---------------------------------------------------------------------------
22
+ // Adapter interface
23
+ // ---------------------------------------------------------------------------
24
+
25
+ /**
26
+ * Telemetry adapter that an SDK consumer implements (or uses via the
27
+ * provided {@link OpenTelemetryAdapter}) to receive Cline telemetry events.
28
+ *
29
+ * The interface intentionally mirrors ITelemetryProvider from the extension
30
+ * so that shared logic can be re-used or compared easily.
31
+ */
32
+ export interface ITelemetryAdapter {
33
+ /** Human-readable adapter name used for logging / diagnostics. */
34
+ readonly name: string;
35
+
36
+ /**
37
+ * Emit a standard telemetry event.
38
+ * Implementations may silently drop events when telemetry is disabled.
39
+ */
40
+ emit(event: string, properties?: TelemetryProperties): void;
41
+
42
+ /**
43
+ * Emit a *required* telemetry event that must not be suppressed by
44
+ * user opt-out settings (e.g. final opt-out confirmation events).
45
+ */
46
+ emitRequired(event: string, properties?: TelemetryProperties): void;
47
+
48
+ /**
49
+ * Record a monotonically-increasing counter metric.
50
+ * Implementations that do not support metrics may treat this as a no-op.
51
+ */
52
+ recordCounter(
53
+ name: string,
54
+ value: number,
55
+ attributes?: TelemetryProperties,
56
+ description?: string,
57
+ required?: boolean,
58
+ ): void;
59
+
60
+ /**
61
+ * Record a histogram (distribution) metric.
62
+ * Implementations that do not support metrics may treat this as a no-op.
63
+ */
64
+ recordHistogram(
65
+ name: string,
66
+ value: number,
67
+ attributes?: TelemetryProperties,
68
+ description?: string,
69
+ required?: boolean,
70
+ ): void;
71
+
72
+ /**
73
+ * Record a gauge (point-in-time) metric.
74
+ * Pass `null` as `value` to retire the series identified by
75
+ * `name + attributes` and prevent stale gauge entries.
76
+ * Implementations that do not support metrics may treat this as a no-op.
77
+ */
78
+ recordGauge(
79
+ name: string,
80
+ value: number | null,
81
+ attributes?: TelemetryProperties,
82
+ description?: string,
83
+ required?: boolean,
84
+ ): void;
85
+
86
+ /** Returns whether the adapter is currently accepting events. */
87
+ isEnabled(): boolean;
88
+
89
+ /** Flush any buffered events/metrics to the backend. */
90
+ flush(): Promise<void>;
91
+
92
+ /** Release all resources held by the adapter. */
93
+ dispose(): Promise<void>;
94
+ }
@@ -0,0 +1,42 @@
1
+ import { describe, expect, it, vi } from "vitest";
2
+ import { LoggerTelemetryAdapter } from "./LoggerTelemetryAdapter";
3
+
4
+ describe("LoggerTelemetryAdapter", () => {
5
+ it("logs events and metrics through the provided logger", async () => {
6
+ const logger = {
7
+ debug: vi.fn(),
8
+ info: vi.fn(),
9
+ warn: vi.fn(),
10
+ };
11
+ const adapter = new LoggerTelemetryAdapter({ logger });
12
+
13
+ adapter.emit("session.started", { sessionId: "s1" });
14
+ adapter.emitRequired("user.opt_out", { reason: "manual" });
15
+ adapter.recordCounter("cline.session.starts.total", 1, {
16
+ sessionId: "s1",
17
+ });
18
+
19
+ expect(logger.info).toHaveBeenCalledWith("telemetry.event", {
20
+ adapter: "LoggerTelemetryAdapter",
21
+ event: "session.started",
22
+ properties: { sessionId: "s1" },
23
+ });
24
+ expect(logger.warn).toHaveBeenCalledWith("telemetry.required_event", {
25
+ adapter: "LoggerTelemetryAdapter",
26
+ event: "user.opt_out",
27
+ properties: { reason: "manual" },
28
+ });
29
+ expect(logger.debug).toHaveBeenCalledWith("telemetry.metric", {
30
+ adapter: "LoggerTelemetryAdapter",
31
+ instrument: "counter",
32
+ name: "cline.session.starts.total",
33
+ value: 1,
34
+ attributes: { sessionId: "s1" },
35
+ description: undefined,
36
+ required: false,
37
+ });
38
+
39
+ await adapter.flush();
40
+ await adapter.dispose();
41
+ });
42
+ });
@@ -0,0 +1,114 @@
1
+ import type { BasicLogger } from "@clinebot/shared";
2
+ import type {
3
+ ITelemetryAdapter,
4
+ TelemetryProperties,
5
+ } from "./ITelemetryAdapter";
6
+
7
+ export interface LoggerTelemetryAdapterOptions {
8
+ logger?: BasicLogger;
9
+ name?: string;
10
+ enabled?: boolean | (() => boolean);
11
+ }
12
+
13
+ export class LoggerTelemetryAdapter implements ITelemetryAdapter {
14
+ readonly name: string;
15
+
16
+ private readonly logger?: BasicLogger;
17
+ private readonly enabled: boolean | (() => boolean);
18
+
19
+ constructor(options: LoggerTelemetryAdapterOptions = {}) {
20
+ this.name = options.name ?? "LoggerTelemetryAdapter";
21
+ this.logger = options.logger;
22
+ this.enabled = options.enabled ?? true;
23
+ }
24
+
25
+ emit(event: string, properties?: TelemetryProperties): void {
26
+ if (!this.isEnabled()) {
27
+ return;
28
+ }
29
+ this.logger?.info?.("telemetry.event", {
30
+ adapter: this.name,
31
+ event,
32
+ properties,
33
+ });
34
+ }
35
+
36
+ emitRequired(event: string, properties?: TelemetryProperties): void {
37
+ this.logger?.warn?.("telemetry.required_event", {
38
+ adapter: this.name,
39
+ event,
40
+ properties,
41
+ });
42
+ }
43
+
44
+ recordCounter(
45
+ name: string,
46
+ value: number,
47
+ attributes?: TelemetryProperties,
48
+ description?: string,
49
+ required?: boolean,
50
+ ): void {
51
+ if (!required && !this.isEnabled()) {
52
+ return;
53
+ }
54
+ this.logger?.debug?.("telemetry.metric", {
55
+ adapter: this.name,
56
+ instrument: "counter",
57
+ name,
58
+ value,
59
+ attributes,
60
+ description,
61
+ required: required === true,
62
+ });
63
+ }
64
+
65
+ recordHistogram(
66
+ name: string,
67
+ value: number,
68
+ attributes?: TelemetryProperties,
69
+ description?: string,
70
+ required?: boolean,
71
+ ): void {
72
+ if (!required && !this.isEnabled()) {
73
+ return;
74
+ }
75
+ this.logger?.debug?.("telemetry.metric", {
76
+ adapter: this.name,
77
+ instrument: "histogram",
78
+ name,
79
+ value,
80
+ attributes,
81
+ description,
82
+ required: required === true,
83
+ });
84
+ }
85
+
86
+ recordGauge(
87
+ name: string,
88
+ value: number | null,
89
+ attributes?: TelemetryProperties,
90
+ description?: string,
91
+ required?: boolean,
92
+ ): void {
93
+ if (!required && !this.isEnabled()) {
94
+ return;
95
+ }
96
+ this.logger?.debug?.("telemetry.metric", {
97
+ adapter: this.name,
98
+ instrument: "gauge",
99
+ name,
100
+ value,
101
+ attributes,
102
+ description,
103
+ required: required === true,
104
+ });
105
+ }
106
+
107
+ isEnabled(): boolean {
108
+ return typeof this.enabled === "function" ? this.enabled() : this.enabled;
109
+ }
110
+
111
+ async flush(): Promise<void> {}
112
+
113
+ async dispose(): Promise<void> {}
114
+ }
@@ -0,0 +1,157 @@
1
+ import type { LoggerProvider } from "@opentelemetry/sdk-logs";
2
+ import type { MeterProvider } from "@opentelemetry/sdk-metrics";
3
+ import { describe, expect, it, vi } from "vitest";
4
+ import { OpenTelemetryAdapter } from "./OpenTelemetryAdapter";
5
+
6
+ describe("OpenTelemetryAdapter", () => {
7
+ it("emits events in the telemetry service log format", () => {
8
+ const emit = vi.fn();
9
+ const adapter = new OpenTelemetryAdapter({
10
+ metadata: makeMetadata(),
11
+ distinctId: "user-123",
12
+ commonProperties: {
13
+ organization_id: "org-1",
14
+ },
15
+ loggerProvider: {
16
+ getLogger: () => ({ emit }),
17
+ } as unknown as LoggerProvider,
18
+ });
19
+
20
+ adapter.emit("task.created", {
21
+ ulid: "01HXYZ",
22
+ nested: {
23
+ mode: "act",
24
+ },
25
+ items: ["a", "b"],
26
+ nullable: null,
27
+ });
28
+
29
+ expect(emit).toHaveBeenCalledWith({
30
+ severityText: "INFO",
31
+ body: "task.created",
32
+ attributes: expect.objectContaining({
33
+ ulid: "01HXYZ",
34
+ "nested.mode": "act",
35
+ items: JSON.stringify(["a", "b"]),
36
+ nullable: "null",
37
+ distinct_id: "user-123",
38
+ organization_id: "org-1",
39
+ extension_version: "1.2.3",
40
+ cline_type: "cli",
41
+ platform: "terminal",
42
+ }),
43
+ });
44
+ });
45
+
46
+ it("marks required events with the expected flag", () => {
47
+ const emit = vi.fn();
48
+ const adapter = new OpenTelemetryAdapter({
49
+ metadata: makeMetadata(),
50
+ loggerProvider: {
51
+ getLogger: () => ({ emit }),
52
+ } as unknown as LoggerProvider,
53
+ enabled: false,
54
+ });
55
+
56
+ adapter.emitRequired("user.opt_out");
57
+
58
+ expect(emit).toHaveBeenCalledWith({
59
+ severityText: "INFO",
60
+ body: "user.opt_out",
61
+ attributes: expect.objectContaining({
62
+ _required: true,
63
+ }),
64
+ });
65
+ });
66
+
67
+ it("records metrics with merged telemetry attributes and retires gauge series", () => {
68
+ const counterAdd = vi.fn();
69
+ const histogramRecord = vi.fn();
70
+ let gaugeCallback:
71
+ | ((result: {
72
+ observe: (
73
+ value: number,
74
+ attributes?: Record<string, string | number | boolean>,
75
+ ) => void;
76
+ }) => void)
77
+ | undefined;
78
+
79
+ const forceFlush = vi.fn().mockResolvedValue(undefined);
80
+ const shutdown = vi.fn().mockResolvedValue(undefined);
81
+
82
+ const adapter = new OpenTelemetryAdapter({
83
+ metadata: makeMetadata(),
84
+ distinctId: "user-123",
85
+ meterProvider: {
86
+ getMeter: () =>
87
+ ({
88
+ createCounter: () => ({ add: counterAdd }),
89
+ createHistogram: () => ({ record: histogramRecord }),
90
+ createObservableGauge: () => ({
91
+ addCallback: (callback: typeof gaugeCallback) => {
92
+ gaugeCallback = callback;
93
+ },
94
+ }),
95
+ }) as never,
96
+ forceFlush,
97
+ shutdown,
98
+ } as unknown as MeterProvider,
99
+ });
100
+
101
+ adapter.recordCounter("cline.turns.total", 2, { ulid: "01HXYZ" });
102
+ adapter.recordHistogram("cline.api.duration.seconds", 1.5, {
103
+ ulid: "01HXYZ",
104
+ });
105
+ adapter.recordGauge("cline.workspace.active_roots", 3, { workspace: "a" });
106
+
107
+ expect(counterAdd).toHaveBeenCalledWith(
108
+ 2,
109
+ expect.objectContaining({
110
+ ulid: "01HXYZ",
111
+ distinct_id: "user-123",
112
+ extension_version: "1.2.3",
113
+ }),
114
+ );
115
+ expect(histogramRecord).toHaveBeenCalledWith(
116
+ 1.5,
117
+ expect.objectContaining({
118
+ ulid: "01HXYZ",
119
+ distinct_id: "user-123",
120
+ }),
121
+ );
122
+
123
+ const observe = vi.fn();
124
+ gaugeCallback?.({ observe });
125
+ expect(observe).toHaveBeenCalledWith(
126
+ 3,
127
+ expect.objectContaining({
128
+ workspace: "a",
129
+ distinct_id: "user-123",
130
+ }),
131
+ );
132
+
133
+ adapter.recordGauge("cline.workspace.active_roots", null, {
134
+ workspace: "a",
135
+ });
136
+ const observeAfterRetire = vi.fn();
137
+ gaugeCallback?.({ observe: observeAfterRetire });
138
+ expect(observeAfterRetire).not.toHaveBeenCalled();
139
+
140
+ return Promise.all([adapter.flush(), adapter.dispose()]).then(() => {
141
+ expect(forceFlush).toHaveBeenCalledTimes(1);
142
+ expect(shutdown).toHaveBeenCalledTimes(1);
143
+ });
144
+ });
145
+ });
146
+
147
+ function makeMetadata() {
148
+ return {
149
+ extension_version: "1.2.3",
150
+ cline_type: "cli",
151
+ platform: "terminal",
152
+ platform_version: "1.0.0",
153
+ os_type: "darwin",
154
+ os_version: "24.0.0",
155
+ is_dev: "true",
156
+ };
157
+ }