gsd-pi 2.71.0-dev.e17e0ce → 2.72.0-dev.593fa74
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/README.md +34 -1
- package/dist/cli.js +17 -0
- package/dist/mcp-server.js +37 -14
- package/dist/resources/agents/debugger.md +58 -0
- package/dist/resources/agents/doc-writer.md +43 -0
- package/dist/resources/agents/git-ops.md +56 -0
- package/dist/resources/agents/javascript-pro.md +46 -271
- package/dist/resources/agents/planner.md +55 -0
- package/dist/resources/agents/refactorer.md +47 -0
- package/dist/resources/agents/reviewer.md +48 -0
- package/dist/resources/agents/security.md +59 -0
- package/dist/resources/agents/tester.md +50 -0
- package/dist/resources/agents/typescript-pro.md +41 -235
- package/dist/resources/extensions/claude-code-cli/partial-builder.js +40 -12
- package/dist/resources/extensions/claude-code-cli/stream-adapter.js +103 -6
- package/dist/resources/extensions/gsd/auto/phases.js +4 -0
- package/dist/resources/extensions/gsd/auto-prompts.js +88 -33
- package/dist/resources/extensions/gsd/auto-start.js +24 -4
- package/dist/resources/extensions/gsd/auto.js +4 -0
- package/dist/resources/extensions/gsd/bootstrap/db-tools.js +3 -3
- package/dist/resources/extensions/gsd/bootstrap/register-shortcuts.js +2 -5
- package/dist/resources/extensions/gsd/doctor-providers.js +23 -0
- package/dist/resources/extensions/gsd/error-classifier.js +4 -1
- package/dist/resources/extensions/gsd/gate-registry.js +208 -0
- package/dist/resources/extensions/gsd/gsd-db.js +41 -0
- package/dist/resources/extensions/gsd/milestone-validation-gates.js +11 -12
- package/dist/resources/extensions/gsd/notification-overlay.js +26 -12
- package/dist/resources/extensions/gsd/notification-store.js +5 -4
- package/dist/resources/extensions/gsd/prompt-validation.js +126 -0
- package/dist/resources/extensions/gsd/prompts/complete-slice.md +3 -1
- package/dist/resources/extensions/gsd/prompts/execute-task.md +2 -0
- package/dist/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
- package/dist/resources/extensions/gsd/shortcut-defs.js +7 -1
- package/dist/resources/extensions/gsd/state.js +9 -2
- package/dist/resources/extensions/gsd/tools/complete-slice.js +52 -1
- package/dist/resources/extensions/gsd/tools/complete-task.js +51 -1
- package/dist/resources/extensions/gsd/tools/workflow-tool-executors.js +4 -1
- package/dist/resources/extensions/ollama/index.js +13 -5
- package/dist/resources/extensions/shared/gsd-phase-state.js +35 -0
- package/dist/resources/extensions/subagent/agents.js +8 -0
- package/dist/resources/extensions/subagent/index.js +17 -0
- package/dist/startup-model-validation.d.ts +0 -1
- package/dist/startup-model-validation.js +6 -2
- package/dist/web/standalone/.next/BUILD_ID +1 -1
- package/dist/web/standalone/.next/app-path-routes-manifest.json +13 -13
- package/dist/web/standalone/.next/build-manifest.json +2 -2
- package/dist/web/standalone/.next/prerender-manifest.json +3 -3
- package/dist/web/standalone/.next/server/app/_global-error.html +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.html +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/_not-found.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.html +1 -1
- package/dist/web/standalone/.next/server/app/index.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/__PAGE__.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_full.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_index.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app/index.segments/_tree.segment.rsc +1 -1
- package/dist/web/standalone/.next/server/app-paths-manifest.json +13 -13
- package/dist/web/standalone/.next/server/middleware-build-manifest.js +1 -1
- package/dist/web/standalone/.next/server/pages/404.html +1 -1
- package/dist/web/standalone/.next/server/pages/500.html +1 -1
- package/dist/web/standalone/.next/server/server-reference-manifest.json +1 -1
- package/package.json +1 -1
- package/packages/mcp-server/dist/server.d.ts +12 -1
- package/packages/mcp-server/dist/server.d.ts.map +1 -1
- package/packages/mcp-server/dist/server.js +90 -42
- package/packages/mcp-server/dist/server.js.map +1 -1
- package/packages/mcp-server/dist/workflow-tools.js +1 -1
- package/packages/mcp-server/dist/workflow-tools.js.map +1 -1
- package/packages/mcp-server/src/server.ts +110 -38
- package/packages/mcp-server/src/workflow-tools.ts +1 -1
- package/packages/pi-coding-agent/dist/core/model-resolver.test.d.ts +8 -0
- package/packages/pi-coding-agent/dist/core/model-resolver.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/model-resolver.test.js +75 -0
- package/packages/pi-coding-agent/dist/core/model-resolver.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/retry-handler.d.ts +5 -0
- package/packages/pi-coding-agent/dist/core/retry-handler.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.js +55 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/retry-handler.test.js +57 -0
- package/packages/pi-coding-agent/dist/core/retry-handler.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js +36 -0
- package/packages/pi-coding-agent/dist/modes/interactive/components/__tests__/tool-execution.test.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js +9 -2
- package/packages/pi-coding-agent/dist/modes/interactive/components/model-selector.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js +87 -12
- package/packages/pi-coding-agent/dist/modes/interactive/components/tool-execution.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.js +6 -1
- package/packages/pi-coding-agent/dist/modes/interactive/controllers/model-controller.js.map +1 -1
- package/packages/pi-coding-agent/package.json +1 -1
- package/packages/pi-coding-agent/src/core/model-resolver.test.ts +85 -0
- package/packages/pi-coding-agent/src/core/retry-handler.test.ts +83 -0
- package/packages/pi-coding-agent/src/core/retry-handler.ts +60 -1
- package/packages/pi-coding-agent/src/modes/interactive/components/__tests__/tool-execution.test.ts +72 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/model-selector.ts +15 -6
- package/packages/pi-coding-agent/src/modes/interactive/components/tool-execution.ts +84 -12
- package/packages/pi-coding-agent/src/modes/interactive/controllers/model-controller.ts +6 -1
- package/pkg/package.json +1 -1
- package/src/resources/agents/debugger.md +58 -0
- package/src/resources/agents/doc-writer.md +43 -0
- package/src/resources/agents/git-ops.md +56 -0
- package/src/resources/agents/javascript-pro.md +46 -271
- package/src/resources/agents/planner.md +55 -0
- package/src/resources/agents/refactorer.md +47 -0
- package/src/resources/agents/reviewer.md +48 -0
- package/src/resources/agents/security.md +59 -0
- package/src/resources/agents/tester.md +50 -0
- package/src/resources/agents/typescript-pro.md +41 -235
- package/src/resources/extensions/claude-code-cli/partial-builder.ts +45 -12
- package/src/resources/extensions/claude-code-cli/stream-adapter.ts +109 -3
- package/src/resources/extensions/claude-code-cli/tests/partial-builder.test.ts +91 -2
- package/src/resources/extensions/claude-code-cli/tests/stream-adapter.test.ts +133 -2
- package/src/resources/extensions/gsd/auto/phases.ts +4 -0
- package/src/resources/extensions/gsd/auto-prompts.ts +111 -33
- package/src/resources/extensions/gsd/auto-start.ts +31 -4
- package/src/resources/extensions/gsd/auto.ts +4 -0
- package/src/resources/extensions/gsd/bootstrap/db-tools.ts +3 -3
- package/src/resources/extensions/gsd/bootstrap/register-shortcuts.ts +2 -5
- package/src/resources/extensions/gsd/doctor-providers.ts +24 -0
- package/src/resources/extensions/gsd/error-classifier.ts +4 -1
- package/src/resources/extensions/gsd/gate-registry.ts +251 -0
- package/src/resources/extensions/gsd/gsd-db.ts +51 -0
- package/src/resources/extensions/gsd/milestone-validation-gates.ts +11 -13
- package/src/resources/extensions/gsd/notification-overlay.ts +27 -11
- package/src/resources/extensions/gsd/notification-store.ts +5 -4
- package/src/resources/extensions/gsd/prompt-validation.ts +157 -0
- package/src/resources/extensions/gsd/prompts/complete-slice.md +3 -1
- package/src/resources/extensions/gsd/prompts/execute-task.md +2 -0
- package/src/resources/extensions/gsd/prompts/validate-milestone.md +2 -0
- package/src/resources/extensions/gsd/shortcut-defs.ts +8 -1
- package/src/resources/extensions/gsd/state.ts +13 -2
- package/src/resources/extensions/gsd/tests/auto-start-model-capture.test.ts +14 -0
- package/src/resources/extensions/gsd/tests/complete-slice-gate-closure.test.ts +167 -0
- package/src/resources/extensions/gsd/tests/doctor-providers.test.ts +36 -0
- package/src/resources/extensions/gsd/tests/format-shortcut.test.ts +16 -0
- package/src/resources/extensions/gsd/tests/gate-dispatch.test.ts +27 -0
- package/src/resources/extensions/gsd/tests/gate-registry.test.ts +140 -0
- package/src/resources/extensions/gsd/tests/prompt-system-gate-coverage.test.ts +208 -0
- package/src/resources/extensions/gsd/tests/provider-errors.test.ts +9 -0
- package/src/resources/extensions/gsd/tests/register-shortcuts.test.ts +3 -2
- package/src/resources/extensions/gsd/tools/complete-slice.ts +63 -0
- package/src/resources/extensions/gsd/tools/complete-task.ts +63 -0
- package/src/resources/extensions/gsd/tools/workflow-tool-executors.ts +4 -1
- package/src/resources/extensions/gsd/types.ts +26 -0
- package/src/resources/extensions/ollama/index.ts +13 -3
- package/src/resources/extensions/ollama/ollama-status-indicator.test.ts +28 -0
- package/src/resources/extensions/shared/gsd-phase-state.ts +42 -0
- package/src/resources/extensions/shared/tests/gsd-phase-state.test.ts +48 -0
- package/src/resources/extensions/subagent/agents.ts +10 -0
- package/src/resources/extensions/subagent/index.ts +18 -0
- package/src/resources/extensions/subagent/tests/agents-conflicts.test.ts +33 -0
- /package/dist/web/standalone/.next/static/{cYPZv_bAhZk2ms-Pz6vsY → h8B07q4xc-ujHRD7esO6O}/_buildManifest.js +0 -0
- /package/dist/web/standalone/.next/static/{cYPZv_bAhZk2ms-Pz6vsY → h8B07q4xc-ujHRD7esO6O}/_ssgManifest.js +0 -0
|
@@ -2,254 +2,60 @@
|
|
|
2
2
|
name: typescript-pro
|
|
3
3
|
description: "TypeScript specialist for advanced type system patterns, complex generics, type-level programming, and end-to-end type safety across full-stack applications. Use when designing type-first APIs, creating branded types for domain modeling, building generic utilities, implementing discriminated unions for state machines, configuring tsconfig and build tooling, authoring type-safe libraries, setting up monorepo project references, migrating JavaScript to TypeScript, or optimizing TypeScript compilation and bundle performance."
|
|
4
4
|
model: sonnet
|
|
5
|
-
memory: project
|
|
6
5
|
---
|
|
7
6
|
|
|
8
|
-
You are a senior TypeScript developer with mastery of TypeScript 5.0+ and its ecosystem
|
|
7
|
+
You are a senior TypeScript developer with mastery of TypeScript 5.0+ and its ecosystem. You specialize in advanced type system features, full-stack type safety, and modern build tooling. Types are the specification — start there.
|
|
9
8
|
|
|
10
|
-
##
|
|
9
|
+
## Initialization
|
|
11
10
|
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
11
|
+
1. Read `tsconfig.json`, `package.json`, and build tool configs
|
|
12
|
+
2. Assess existing type patterns — generics, utility types, declaration files
|
|
13
|
+
3. Identify framework and runtime (React, Vue, Node.js, Deno)
|
|
14
|
+
4. Check lint/format config to align with project conventions
|
|
16
15
|
|
|
17
|
-
##
|
|
16
|
+
## Core Principles
|
|
18
17
|
|
|
19
|
-
|
|
18
|
+
- **Strict mode always**: `strict: true`, no `any` without documented justification
|
|
19
|
+
- **Type-first**: Define data shapes and API contracts before writing logic
|
|
20
|
+
- **Inference over annotation**: Let TypeScript infer where it produces correct, readable types
|
|
21
|
+
- **`satisfies` over type annotation**: Preserves literal types while validating
|
|
22
|
+
- **`as const`** for literal preservation in arrays and objects
|
|
23
|
+
- **`import type`** for type-only imports — reduces emit, improves tree shaking
|
|
24
|
+
- **Exhaustive checks** with `never` in switch/if-else — catch unhandled cases at compile time
|
|
20
25
|
|
|
21
|
-
|
|
22
|
-
2. **Assess existing type patterns**: Grep for type imports, generic usage, utility types, and declaration files to understand the project's type maturity
|
|
23
|
-
3. **Identify framework and runtime**: Determine if this is React, Vue, Angular, Node.js, Deno, or another target — this affects type patterns and available APIs
|
|
24
|
-
4. **Check existing lint/format config**: Look for .eslintrc, prettier config, biome config to align with project conventions
|
|
26
|
+
## Key Patterns
|
|
25
27
|
|
|
26
|
-
|
|
28
|
+
- Conditional types for flexible APIs: `T extends Array<infer U> ? { data: U[] } : { data: T }`
|
|
29
|
+
- Mapped types for transformations: `{ readonly [K in keyof T]: T[K] }`
|
|
30
|
+
- Template literal types for string manipulation: `` `on${Capitalize<T>}` ``
|
|
31
|
+
- Discriminated unions for state machines — each variant has a literal tag
|
|
32
|
+
- Branded types for domain modeling: `T & { readonly __brand: B }`
|
|
33
|
+
- Result types for error handling: `{ ok: true; value: T } | { ok: false; error: E }`
|
|
34
|
+
- Type guards at runtime boundaries — validate all external data (APIs, user input, files)
|
|
27
35
|
|
|
28
|
-
|
|
36
|
+
## Build & Tooling
|
|
29
37
|
|
|
30
|
-
-
|
|
31
|
-
-
|
|
32
|
-
-
|
|
33
|
-
-
|
|
34
|
-
-
|
|
35
|
-
-
|
|
36
|
-
- [ ] Generic constraints are as narrow as possible
|
|
37
|
-
- [ ] Discriminated unions preferred over optional fields for variant types
|
|
38
|
+
- `moduleResolution: "bundler"` for modern bundler projects
|
|
39
|
+
- `isolatedModules: true` for esbuild/SWC compatibility
|
|
40
|
+
- `incremental: true` with `.tsbuildinfo` for faster rebuilds
|
|
41
|
+
- `composite: true` + `declarationMap: true` for monorepo project references
|
|
42
|
+
- Type-only imports to reduce emit and improve tree shaking
|
|
43
|
+
- Monitor type instantiation counts with `--generateTrace` for slow compiles
|
|
38
44
|
|
|
39
|
-
##
|
|
45
|
+
## Testing
|
|
40
46
|
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
**Conditional types** for flexible APIs:
|
|
44
|
-
```typescript
|
|
45
|
-
type ApiResponse<T> = T extends Array<infer U>
|
|
46
|
-
? { data: U[]; total: number }
|
|
47
|
-
: { data: T };
|
|
48
|
-
```
|
|
49
|
-
|
|
50
|
-
**Mapped types** for transformations:
|
|
51
|
-
```typescript
|
|
52
|
-
type Readonly<T> = { readonly [K in keyof T]: T[K] };
|
|
53
|
-
type Optional<T, K extends keyof T> = Omit<T, K> & Partial<Pick<T, K>>;
|
|
54
|
-
```
|
|
55
|
-
|
|
56
|
-
**Template literal types** for string manipulation:
|
|
57
|
-
```typescript
|
|
58
|
-
type EventName<T extends string> = `on${Capitalize<T>}`;
|
|
59
|
-
type RouteParam<T extends string> = T extends `${infer _}:${infer Param}/${infer Rest}`
|
|
60
|
-
? Param | RouteParam<Rest>
|
|
61
|
-
: T extends `${infer _}:${infer Param}` ? Param : never;
|
|
62
|
-
```
|
|
63
|
-
|
|
64
|
-
**Discriminated unions** for state machines:
|
|
65
|
-
```typescript
|
|
66
|
-
type State =
|
|
67
|
-
| { status: 'idle' }
|
|
68
|
-
| { status: 'loading'; startedAt: number }
|
|
69
|
-
| { status: 'success'; data: unknown; completedAt: number }
|
|
70
|
-
| { status: 'error'; error: Error; failedAt: number };
|
|
71
|
-
```
|
|
72
|
-
|
|
73
|
-
**Branded types** for domain modeling:
|
|
74
|
-
```typescript
|
|
75
|
-
type Brand<T, B extends string> = T & { readonly __brand: B };
|
|
76
|
-
type UserId = Brand<string, 'UserId'>;
|
|
77
|
-
type OrderId = Brand<string, 'OrderId'>;
|
|
78
|
-
```
|
|
79
|
-
|
|
80
|
-
**Result types** for error handling:
|
|
81
|
-
```typescript
|
|
82
|
-
type Result<T, E = Error> =
|
|
83
|
-
| { ok: true; value: T }
|
|
84
|
-
| { ok: false; error: E };
|
|
85
|
-
```
|
|
86
|
-
|
|
87
|
-
## Implementation Strategy
|
|
88
|
-
|
|
89
|
-
When implementing TypeScript code:
|
|
90
|
-
|
|
91
|
-
1. **Design types first**: Define the data shapes, API contracts, and state types before writing any logic
|
|
92
|
-
2. **Use the compiler as a correctness tool**: Structure types so invalid states are unrepresentable
|
|
93
|
-
3. **Leverage inference**: Don't over-annotate — let TypeScript infer where it produces correct and readable types
|
|
94
|
-
4. **Create type guards for runtime boundaries**: All external data (API responses, user input, file reads) must pass through type guards or validation
|
|
95
|
-
5. **Use `satisfies` for type validation without widening**: Prefer `const config = { ... } satisfies Config` over `const config: Config = { ... }` when you want to preserve literal types
|
|
96
|
-
6. **Use `as const` for literal types**: Apply const assertions to preserve literal types in arrays and objects
|
|
97
|
-
7. **Exhaustive checking**: Use `never` type in switch/if-else chains to ensure all cases are handled
|
|
98
|
-
|
|
99
|
-
```typescript
|
|
100
|
-
function assertNever(x: never): never {
|
|
101
|
-
throw new Error(`Unexpected value: ${x}`);
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
function handleState(state: State): string {
|
|
105
|
-
switch (state.status) {
|
|
106
|
-
case 'idle': return 'Waiting';
|
|
107
|
-
case 'loading': return 'Loading...';
|
|
108
|
-
case 'success': return 'Done';
|
|
109
|
-
case 'error': return state.error.message;
|
|
110
|
-
default: return assertNever(state);
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
```
|
|
114
|
-
|
|
115
|
-
## Build and Tooling Optimization
|
|
116
|
-
|
|
117
|
-
**tsconfig.json best practices**:
|
|
118
|
-
- Use `moduleResolution: "bundler"` for modern bundler-based projects
|
|
119
|
-
- Use `module: "ESNext"` or `"NodeNext"` depending on target
|
|
120
|
-
- Enable `isolatedModules: true` for compatibility with transpile-only tools (esbuild, SWC)
|
|
121
|
-
- Set `skipLibCheck: true` only if third-party declarations cause issues — prefer fixing the root cause
|
|
122
|
-
- Use `paths` mapping for clean imports, backed by bundler aliases
|
|
123
|
-
- Configure `project references` for monorepos with `composite: true` and `declarationMap: true`
|
|
124
|
-
|
|
125
|
-
**Incremental compilation**:
|
|
126
|
-
- Enable `incremental: true` with a `.tsbuildinfo` output path
|
|
127
|
-
- Use `--build` mode for project references
|
|
128
|
-
- Configure `tsBuildInfoFile` to a persistent location in CI
|
|
129
|
-
|
|
130
|
-
**Performance tuning**:
|
|
131
|
-
- Use `type-only imports` to reduce emit and improve tree shaking
|
|
132
|
-
- Prefer `const enum` only when bundle size savings justify the trade-off (they don't work with `isolatedModules`)
|
|
133
|
-
- Avoid deeply recursive conditional types in hot paths — they slow the compiler
|
|
134
|
-
- Monitor type instantiation counts with `--generateTrace`
|
|
135
|
-
|
|
136
|
-
## Testing With Types
|
|
137
|
-
|
|
138
|
-
- Write type tests using `expectTypeOf` (from vitest) or `tsd` for declaration testing
|
|
139
|
-
- Create type-safe test utilities and fixtures
|
|
140
|
-
- Use generic factory functions for test data
|
|
141
|
-
- Ensure mock types match the real implementations
|
|
47
|
+
- Type tests with `expectTypeOf` (vitest) or `tsd` for declaration testing
|
|
48
|
+
- Type-safe test utilities and generic factory functions for test data
|
|
142
49
|
- Test type narrowing paths explicitly
|
|
50
|
+
- Ensure mock types match real implementations
|
|
143
51
|
|
|
144
|
-
|
|
145
|
-
import { expectTypeOf } from 'vitest';
|
|
146
|
-
|
|
147
|
-
test('type narrowing works', () => {
|
|
148
|
-
const result: Result<string> = { ok: true, value: 'hello' };
|
|
149
|
-
if (result.ok) {
|
|
150
|
-
expectTypeOf(result.value).toBeString();
|
|
151
|
-
} else {
|
|
152
|
-
expectTypeOf(result.error).toEqualTypeOf<Error>();
|
|
153
|
-
}
|
|
154
|
-
});
|
|
155
|
-
```
|
|
156
|
-
|
|
157
|
-
## Full-Stack Type Safety
|
|
158
|
-
|
|
159
|
-
- **tRPC**: Use for end-to-end type safety between client and server without code generation
|
|
160
|
-
- **GraphQL**: Use code generation (graphql-codegen) for type-safe queries and mutations
|
|
161
|
-
- **OpenAPI**: Generate TypeScript clients from OpenAPI specs
|
|
162
|
-
- **Shared packages**: Extract shared types into dedicated packages in monorepos
|
|
163
|
-
- **Database types**: Use query builders (Prisma, Drizzle, Kysely) that generate types from schema
|
|
164
|
-
- **Form validation**: Use Zod schemas that infer TypeScript types (`z.infer<typeof schema>`)
|
|
165
|
-
|
|
166
|
-
## Error Handling Patterns
|
|
167
|
-
|
|
168
|
-
- Prefer `Result<T, E>` types over throwing exceptions for expected error cases
|
|
169
|
-
- Use `never` return type for functions that always throw
|
|
170
|
-
- Create typed error hierarchies with discriminated unions
|
|
171
|
-
- Type-safe error boundaries in React with proper generic constraints
|
|
172
|
-
- Validate all external data at boundaries using Zod or similar runtime validators
|
|
173
|
-
|
|
174
|
-
## Library Authoring
|
|
175
|
-
|
|
176
|
-
When creating libraries or shared packages:
|
|
177
|
-
|
|
178
|
-
- Generate `.d.ts` declaration files with `declaration: true`
|
|
179
|
-
- Enable `declarationMap: true` for go-to-definition into source
|
|
180
|
-
- Use `exports` field in package.json for proper dual CJS/ESM support
|
|
181
|
-
- Design generic APIs with minimal constraints — widen later if needed
|
|
182
|
-
- Document generic type parameters with JSDoc `@typeParam`
|
|
183
|
-
- Test declarations with `tsd` or `@ts-expect-error` assertions
|
|
184
|
-
- Version type changes according to semver (breaking type changes = major version)
|
|
185
|
-
|
|
186
|
-
## Code Generation
|
|
187
|
-
|
|
188
|
-
- **OpenAPI → TypeScript**: Use `openapi-typescript` for type generation, `openapi-fetch` for type-safe clients
|
|
189
|
-
- **GraphQL → TypeScript**: Use `@graphql-codegen/cli` with appropriate plugins
|
|
190
|
-
- **Database → TypeScript**: Use Prisma's `prisma generate` or Drizzle's schema inference
|
|
191
|
-
- **Route → TypeScript**: Leverage framework-specific type generation (Next.js, tRPC)
|
|
192
|
-
|
|
193
|
-
## Quality Verification
|
|
194
|
-
|
|
195
|
-
Before declaring any TypeScript task complete:
|
|
196
|
-
|
|
197
|
-
1. **Compile check**: Run `npx tsc --noEmit` and resolve all errors
|
|
198
|
-
2. **Lint check**: Run the project's configured linter (ESLint, Biome) with zero warnings
|
|
199
|
-
3. **Type coverage**: Verify no untyped public APIs remain
|
|
200
|
-
4. **Test execution**: Run the test suite and verify passing
|
|
201
|
-
5. **Bundle analysis**: If applicable, verify bundle size impact
|
|
202
|
-
6. **Declaration quality**: If library code, verify generated `.d.ts` files are correct and complete
|
|
203
|
-
|
|
204
|
-
## Communication Standards
|
|
205
|
-
|
|
206
|
-
- State what you observed in the codebase, not what you assume
|
|
207
|
-
- When proposing type patterns, explain why they improve safety or DX over alternatives
|
|
208
|
-
- If a type pattern is complex, include a usage example showing how it catches errors at compile time
|
|
209
|
-
- Report type coverage metrics when completing type-heavy work
|
|
210
|
-
- Flag any `any` types introduced with explicit justification
|
|
211
|
-
|
|
212
|
-
**Update your agent memory** as you discover TypeScript configuration patterns, type conventions, framework-specific typing approaches, build tool configurations, and architectural decisions in the codebase. Write concise notes about what you found and where.
|
|
213
|
-
|
|
214
|
-
Examples of what to record:
|
|
215
|
-
- tsconfig.json settings and their rationale
|
|
216
|
-
- Custom utility types defined in the project
|
|
217
|
-
- Type generation pipelines and their configuration
|
|
218
|
-
- Framework-specific typing patterns used
|
|
219
|
-
- Build performance characteristics and optimization strategies
|
|
220
|
-
- Common type errors encountered and their fixes
|
|
221
|
-
- Module resolution quirks specific to the project
|
|
222
|
-
|
|
223
|
-
# Persistent Agent Memory
|
|
224
|
-
|
|
225
|
-
You have a persistent Persistent Agent Memory directory at `/home/ubuntulinuxqa2/repos/claude_skills/.claude/agent-memory/typescript-pro/`. Its contents persist across conversations.
|
|
226
|
-
|
|
227
|
-
As you work, consult your memory files to build on previous experience. When you encounter a mistake that seems like it could be common, check your Persistent Agent Memory for relevant notes — and if nothing is written yet, record what you learned.
|
|
228
|
-
|
|
229
|
-
Guidelines:
|
|
230
|
-
- `MEMORY.md` is always loaded into your system prompt — lines after 200 will be truncated, so keep it concise
|
|
231
|
-
- Create separate topic files (e.g., `debugging.md`, `patterns.md`) for detailed notes and link to them from MEMORY.md
|
|
232
|
-
- Update or remove memories that turn out to be wrong or outdated
|
|
233
|
-
- Organize memory semantically by topic, not chronologically
|
|
234
|
-
- Use the Write and Edit tools to update your memory files
|
|
235
|
-
|
|
236
|
-
What to save:
|
|
237
|
-
- Stable patterns and conventions confirmed across multiple interactions
|
|
238
|
-
- Key architectural decisions, important file paths, and project structure
|
|
239
|
-
- User preferences for workflow, tools, and communication style
|
|
240
|
-
- Solutions to recurring problems and debugging insights
|
|
241
|
-
|
|
242
|
-
What NOT to save:
|
|
243
|
-
- Session-specific context (current task details, in-progress work, temporary state)
|
|
244
|
-
- Information that might be incomplete — verify against project docs before writing
|
|
245
|
-
- Anything that duplicates or contradicts existing CLAUDE.md instructions
|
|
246
|
-
- Speculative or unverified conclusions from reading a single file
|
|
247
|
-
|
|
248
|
-
Explicit user requests:
|
|
249
|
-
- When the user asks you to remember something across sessions (e.g., "always use bun", "never auto-commit"), save it — no need to wait for multiple interactions
|
|
250
|
-
- When the user asks to forget or stop remembering something, find and remove the relevant entries from your memory files
|
|
251
|
-
- Since this memory is project-scope and shared with your team via version control, tailor your memories to this project
|
|
52
|
+
## Verification Checklist
|
|
252
53
|
|
|
253
|
-
|
|
54
|
+
1. `npx tsc --noEmit` — zero errors
|
|
55
|
+
2. Linter passes with zero warnings
|
|
56
|
+
3. No untyped public APIs remain
|
|
57
|
+
4. Tests passing, coverage target met
|
|
58
|
+
5. Declaration files correct for library code
|
|
59
|
+
6. No `any` without justification comment
|
|
254
60
|
|
|
255
|
-
|
|
61
|
+
Report concrete outcomes — files changed, type coverage, test results, trade-offs made.
|
|
@@ -19,6 +19,49 @@ import type {
|
|
|
19
19
|
import { hasXmlParameterTags, repairToolJson } from "@gsd/pi-ai";
|
|
20
20
|
import type { BetaContentBlock, BetaRawMessageStreamEvent, NonNullableUsage } from "./sdk-types.js";
|
|
21
21
|
|
|
22
|
+
// ---------------------------------------------------------------------------
|
|
23
|
+
// MCP tool name parsing
|
|
24
|
+
// ---------------------------------------------------------------------------
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Split a Claude Code MCP tool name (`mcp__<server>__<tool>`) into its parts.
|
|
28
|
+
* Returns null for non-prefixed names so callers can fall through unchanged.
|
|
29
|
+
*
|
|
30
|
+
* Server names may contain hyphens (`gsd-workflow`); the SDK uses the literal
|
|
31
|
+
* `__` delimiter between the server name and the tool name.
|
|
32
|
+
*/
|
|
33
|
+
export function parseMcpToolName(name: string): { server: string; tool: string } | null {
|
|
34
|
+
if (!name.startsWith("mcp__")) return null;
|
|
35
|
+
const rest = name.slice("mcp__".length);
|
|
36
|
+
const delim = rest.indexOf("__");
|
|
37
|
+
if (delim <= 0 || delim === rest.length - 2) return null;
|
|
38
|
+
return { server: rest.slice(0, delim), tool: rest.slice(delim + 2) };
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Build a GSD ToolCall block from a Claude Code SDK tool_use block, stripping
|
|
43
|
+
* the `mcp__<server>__` prefix from the name so registered extension renderers
|
|
44
|
+
* (which use the unprefixed canonical names) can match. The original server
|
|
45
|
+
* name is preserved on the block for diagnostics and rendering.
|
|
46
|
+
*/
|
|
47
|
+
function toolCallFromBlock(
|
|
48
|
+
id: string,
|
|
49
|
+
rawName: string,
|
|
50
|
+
input: Record<string, unknown>,
|
|
51
|
+
): ToolCall {
|
|
52
|
+
const parsed = parseMcpToolName(rawName);
|
|
53
|
+
const toolCall: ToolCall = {
|
|
54
|
+
type: "toolCall",
|
|
55
|
+
id,
|
|
56
|
+
name: parsed ? parsed.tool : rawName,
|
|
57
|
+
arguments: input,
|
|
58
|
+
};
|
|
59
|
+
if (parsed) {
|
|
60
|
+
(toolCall as ToolCall & { mcpServer?: string }).mcpServer = parsed.server;
|
|
61
|
+
}
|
|
62
|
+
return toolCall;
|
|
63
|
+
}
|
|
64
|
+
|
|
22
65
|
// ---------------------------------------------------------------------------
|
|
23
66
|
// Content-block mapping helpers
|
|
24
67
|
// ---------------------------------------------------------------------------
|
|
@@ -41,12 +84,7 @@ export function mapContentBlock(
|
|
|
41
84
|
} satisfies ThinkingContent;
|
|
42
85
|
|
|
43
86
|
case "tool_use":
|
|
44
|
-
return
|
|
45
|
-
type: "toolCall",
|
|
46
|
-
id: block.id,
|
|
47
|
-
name: block.name,
|
|
48
|
-
arguments: block.input,
|
|
49
|
-
} satisfies ToolCall;
|
|
87
|
+
return toolCallFromBlock(block.id, block.name, block.input);
|
|
50
88
|
|
|
51
89
|
case "server_tool_use":
|
|
52
90
|
return {
|
|
@@ -183,12 +221,7 @@ export class PartialMessageBuilder {
|
|
|
183
221
|
}
|
|
184
222
|
if (block.type === "tool_use") {
|
|
185
223
|
this.toolJsonAccum.set(streamIndex, "");
|
|
186
|
-
this.partial.content.push({
|
|
187
|
-
type: "toolCall",
|
|
188
|
-
id: block.id,
|
|
189
|
-
name: block.name,
|
|
190
|
-
arguments: {},
|
|
191
|
-
});
|
|
224
|
+
this.partial.content.push(toolCallFromBlock(block.id, block.name, {}));
|
|
192
225
|
return { type: "toolcall_start", contentIndex, partial: this.partial };
|
|
193
226
|
}
|
|
194
227
|
if (block.type === "server_tool_use") {
|
|
@@ -518,22 +518,83 @@ export function createClaudeCodeElicitationHandler(
|
|
|
518
518
|
};
|
|
519
519
|
}
|
|
520
520
|
|
|
521
|
+
/**
|
|
522
|
+
* Aborted by the caller's AbortSignal — distinct from exhaustion. GSD's
|
|
523
|
+
* agent loop keys off `stopReason === "aborted"` to treat this as a clean
|
|
524
|
+
* user cancel instead of a retry-eligible provider failure.
|
|
525
|
+
*/
|
|
526
|
+
export function makeAbortedMessage(model: string, lastTextContent: string): AssistantMessage {
|
|
527
|
+
const message: AssistantMessage = {
|
|
528
|
+
role: "assistant",
|
|
529
|
+
content: lastTextContent
|
|
530
|
+
? [{ type: "text", text: lastTextContent }]
|
|
531
|
+
: [{ type: "text", text: "Claude Code stream aborted by caller" }],
|
|
532
|
+
api: "anthropic-messages",
|
|
533
|
+
provider: "claude-code",
|
|
534
|
+
model,
|
|
535
|
+
usage: { ...ZERO_USAGE },
|
|
536
|
+
stopReason: "aborted",
|
|
537
|
+
timestamp: Date.now(),
|
|
538
|
+
};
|
|
539
|
+
return message;
|
|
540
|
+
}
|
|
541
|
+
|
|
521
542
|
// ---------------------------------------------------------------------------
|
|
522
543
|
// SDK options builder
|
|
523
544
|
// ---------------------------------------------------------------------------
|
|
524
545
|
|
|
546
|
+
/**
|
|
547
|
+
* Resolve the Claude Code permission mode for the current run.
|
|
548
|
+
*
|
|
549
|
+
* - Auto-mode / headless runs bypass permissions so tool calls don't block
|
|
550
|
+
* on prompts the user isn't watching.
|
|
551
|
+
* - Interactive runs default to `acceptEdits` so file/bash writes still
|
|
552
|
+
* land quickly but the SDK retains a permission gate.
|
|
553
|
+
* - `GSD_CLAUDE_CODE_PERMISSION_MODE` forces a specific mode when set.
|
|
554
|
+
*
|
|
555
|
+
* Cross-extension coupling is kept minimal by dynamically importing
|
|
556
|
+
* `isAutoActive` and falling back to the bypass default if the import
|
|
557
|
+
* fails (e.g. in unit tests that load stream-adapter in isolation).
|
|
558
|
+
*/
|
|
559
|
+
export async function resolveClaudePermissionMode(
|
|
560
|
+
env: NodeJS.ProcessEnv = process.env,
|
|
561
|
+
): Promise<"bypassPermissions" | "acceptEdits" | "default" | "plan"> {
|
|
562
|
+
const override = env.GSD_CLAUDE_CODE_PERMISSION_MODE?.trim();
|
|
563
|
+
if (override === "bypassPermissions" || override === "acceptEdits" || override === "default" || override === "plan") {
|
|
564
|
+
return override;
|
|
565
|
+
}
|
|
566
|
+
|
|
567
|
+
try {
|
|
568
|
+
const autoMod = (await import("../gsd/auto.js")) as { isAutoActive?: () => boolean };
|
|
569
|
+
if (typeof autoMod.isAutoActive === "function" && autoMod.isAutoActive()) {
|
|
570
|
+
return "bypassPermissions";
|
|
571
|
+
}
|
|
572
|
+
return "acceptEdits";
|
|
573
|
+
} catch {
|
|
574
|
+
// auto.ts unavailable (tests, non-GSD contexts) — stay permissive.
|
|
575
|
+
return "bypassPermissions";
|
|
576
|
+
}
|
|
577
|
+
}
|
|
578
|
+
|
|
525
579
|
/**
|
|
526
580
|
* Build the options object passed to the Claude Agent SDK's `query()` call.
|
|
527
581
|
*
|
|
528
582
|
* Extracted for testability — callers can verify session persistence,
|
|
529
583
|
* beta flags, and other configuration without mocking the full SDK.
|
|
584
|
+
*
|
|
585
|
+
* `permissionMode` / `allowDangerouslySkipPermissions` are resolved through
|
|
586
|
+
* {@link resolveClaudePermissionMode} so interactive runs don't silently
|
|
587
|
+
* bypass the SDK's permission gate. Callers that want the old always-bypass
|
|
588
|
+
* behaviour pass `permissionMode: "bypassPermissions"` explicitly.
|
|
530
589
|
*/
|
|
531
590
|
export function buildSdkOptions(
|
|
532
591
|
modelId: string,
|
|
533
592
|
prompt: string,
|
|
593
|
+
overrides?: { permissionMode?: "bypassPermissions" | "acceptEdits" | "default" | "plan" },
|
|
534
594
|
extraOptions: Record<string, unknown> = {},
|
|
535
595
|
): Record<string, unknown> {
|
|
536
596
|
const mcpServers = buildWorkflowMcpServers();
|
|
597
|
+
const permissionMode = overrides?.permissionMode ?? "bypassPermissions";
|
|
537
598
|
const disallowedTools = ["AskUserQuestion"];
|
|
538
599
|
return {
|
|
539
600
|
pathToClaudeCodeExecutable: getClaudePath(),
|
|
@@ -541,8 +602,8 @@ export function buildSdkOptions(
|
|
|
541
602
|
includePartialMessages: true,
|
|
542
603
|
persistSession: true,
|
|
543
604
|
cwd: process.cwd(),
|
|
544
|
-
permissionMode
|
|
545
|
-
allowDangerouslySkipPermissions:
|
|
605
|
+
permissionMode,
|
|
606
|
+
allowDangerouslySkipPermissions: permissionMode === "bypassPermissions",
|
|
546
607
|
settingSources: ["project"],
|
|
547
608
|
systemPrompt: { type: "preset", preset: "claude_code" },
|
|
548
609
|
disallowedTools,
|
|
@@ -656,6 +717,29 @@ function attachExternalResultsToToolBlocks(
|
|
|
656
717
|
}
|
|
657
718
|
}
|
|
658
719
|
|
|
720
|
+
/**
|
|
721
|
+
* Merge tool-call blocks from the active partial-message builder into the
|
|
722
|
+
* running list of intermediate tool calls, preserving order and de-duping
|
|
723
|
+
* by tool-call id. Exposed for testing the F3 fix (final-turn tool calls
|
|
724
|
+
* dropped when `result` arrives without a preceding synthetic `user`).
|
|
725
|
+
*/
|
|
726
|
+
export function mergePendingToolCalls(
|
|
727
|
+
intermediate: AssistantMessage["content"],
|
|
728
|
+
pending: AssistantMessage["content"],
|
|
729
|
+
): AssistantMessage["content"] {
|
|
730
|
+
const alreadyIncluded = new Set<string>();
|
|
731
|
+
for (const block of intermediate) {
|
|
732
|
+
if (block.type === "toolCall") alreadyIncluded.add(block.id);
|
|
733
|
+
}
|
|
734
|
+
for (const block of pending) {
|
|
735
|
+
if (block.type !== "toolCall") continue;
|
|
736
|
+
if (alreadyIncluded.has(block.id)) continue;
|
|
737
|
+
alreadyIncluded.add(block.id);
|
|
738
|
+
intermediate.push(block);
|
|
739
|
+
}
|
|
740
|
+
return intermediate;
|
|
741
|
+
}
|
|
742
|
+
|
|
659
743
|
// ---------------------------------------------------------------------------
|
|
660
744
|
// streamSimple implementation
|
|
661
745
|
// ---------------------------------------------------------------------------
|
|
@@ -712,9 +796,11 @@ async function pumpSdkMessages(
|
|
|
712
796
|
}
|
|
713
797
|
|
|
714
798
|
const prompt = buildPromptFromContext(context);
|
|
799
|
+
const permissionMode = await resolveClaudePermissionMode();
|
|
715
800
|
const sdkOpts = buildSdkOptions(
|
|
716
801
|
modelId,
|
|
717
802
|
prompt,
|
|
803
|
+
{ permissionMode },
|
|
718
804
|
typeof (options as ClaudeCodeStreamOptions | undefined)?.extensionUIContext === "object"
|
|
719
805
|
? {
|
|
720
806
|
onElicitation: createClaudeCodeElicitationHandler(
|
|
@@ -746,7 +832,17 @@ async function pumpSdkMessages(
|
|
|
746
832
|
stream.push({ type: "start", partial: initialPartial });
|
|
747
833
|
|
|
748
834
|
for await (const msg of queryResult as AsyncIterable<SDKMessage>) {
|
|
749
|
-
if (options?.signal?.aborted)
|
|
835
|
+
if (options?.signal?.aborted) {
|
|
836
|
+
// User-initiated cancel — emit an aborted error so the agent
|
|
837
|
+
// loop classifies this as a deliberate stop, not a transient
|
|
838
|
+
// provider failure that should be retried.
|
|
839
|
+
stream.push({
|
|
840
|
+
type: "error",
|
|
841
|
+
reason: "aborted",
|
|
842
|
+
error: makeAbortedMessage(modelId, lastTextContent),
|
|
843
|
+
});
|
|
844
|
+
return;
|
|
845
|
+
}
|
|
750
846
|
|
|
751
847
|
switch (msg.type) {
|
|
752
848
|
// -- Init --
|
|
@@ -857,6 +953,16 @@ async function pumpSdkMessages(
|
|
|
857
953
|
// events for proper TUI rendering, followed by the text response.
|
|
858
954
|
const finalContent: AssistantMessage["content"] = [];
|
|
859
955
|
|
|
956
|
+
// If the final turn ended without a synthetic user message
|
|
957
|
+
// (e.g. stop_reason: "tool_use" followed directly by result,
|
|
958
|
+
// or a turn with text but no tool execution), the `builder`
|
|
959
|
+
// still holds toolCall blocks that were never pushed into
|
|
960
|
+
// `intermediateToolBlocks`. Fold them in here so they aren't
|
|
961
|
+
// dropped from the final AssistantMessage.
|
|
962
|
+
if (builder) {
|
|
963
|
+
mergePendingToolCalls(intermediateToolBlocks, builder.message.content);
|
|
964
|
+
}
|
|
965
|
+
|
|
860
966
|
// Add tool calls from intermediate turns first (renders above text)
|
|
861
967
|
attachExternalResultsToToolBlocks(intermediateToolBlocks, toolResultsById);
|
|
862
968
|
finalContent.push(...intermediateToolBlocks);
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { describe, test } from "node:test";
|
|
2
2
|
import assert from "node:assert/strict";
|
|
3
|
-
import { PartialMessageBuilder } from "../partial-builder.ts";
|
|
4
|
-
import type { BetaRawMessageStreamEvent } from "../sdk-types.ts";
|
|
3
|
+
import { mapContentBlock, parseMcpToolName, PartialMessageBuilder } from "../partial-builder.ts";
|
|
4
|
+
import type { BetaContentBlock, BetaRawMessageStreamEvent } from "../sdk-types.ts";
|
|
5
5
|
|
|
6
6
|
describe("PartialMessageBuilder — malformed tool arguments (#2574)", () => {
|
|
7
7
|
/**
|
|
@@ -148,3 +148,92 @@ describe("PartialMessageBuilder — malformed tool arguments (#2574)", () => {
|
|
|
148
148
|
}
|
|
149
149
|
});
|
|
150
150
|
});
|
|
151
|
+
|
|
152
|
+
describe("parseMcpToolName", () => {
|
|
153
|
+
test("splits mcp__<server>__<tool> into parts", () => {
|
|
154
|
+
assert.deepEqual(
|
|
155
|
+
parseMcpToolName("mcp__gsd-workflow__gsd_plan_milestone"),
|
|
156
|
+
{ server: "gsd-workflow", tool: "gsd_plan_milestone" },
|
|
157
|
+
);
|
|
158
|
+
});
|
|
159
|
+
|
|
160
|
+
test("preserves server names containing hyphens", () => {
|
|
161
|
+
assert.deepEqual(
|
|
162
|
+
parseMcpToolName("mcp__my-cool-server__do_thing"),
|
|
163
|
+
{ server: "my-cool-server", tool: "do_thing" },
|
|
164
|
+
);
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
test("preserves tool names containing underscores", () => {
|
|
168
|
+
assert.deepEqual(
|
|
169
|
+
parseMcpToolName("mcp__srv__a_b_c_d"),
|
|
170
|
+
{ server: "srv", tool: "a_b_c_d" },
|
|
171
|
+
);
|
|
172
|
+
});
|
|
173
|
+
|
|
174
|
+
test("returns null for non-prefixed names", () => {
|
|
175
|
+
assert.equal(parseMcpToolName("Bash"), null);
|
|
176
|
+
assert.equal(parseMcpToolName("gsd_plan_milestone"), null);
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
test("returns null for malformed prefixes", () => {
|
|
180
|
+
assert.equal(parseMcpToolName("mcp__"), null);
|
|
181
|
+
assert.equal(parseMcpToolName("mcp__server"), null);
|
|
182
|
+
assert.equal(parseMcpToolName("mcp__server__"), null);
|
|
183
|
+
assert.equal(parseMcpToolName("mcp____tool"), null);
|
|
184
|
+
});
|
|
185
|
+
});
|
|
186
|
+
|
|
187
|
+
describe("PartialMessageBuilder — MCP tool name normalization", () => {
|
|
188
|
+
test("strips mcp__<server>__ prefix on content_block_start", () => {
|
|
189
|
+
const builder = new PartialMessageBuilder("claude-sonnet-4-20250514");
|
|
190
|
+
const event = builder.handleEvent({
|
|
191
|
+
type: "content_block_start",
|
|
192
|
+
index: 0,
|
|
193
|
+
content_block: {
|
|
194
|
+
type: "tool_use",
|
|
195
|
+
id: "tool_1",
|
|
196
|
+
name: "mcp__gsd-workflow__gsd_plan_milestone",
|
|
197
|
+
input: {},
|
|
198
|
+
},
|
|
199
|
+
} as BetaRawMessageStreamEvent);
|
|
200
|
+
|
|
201
|
+
assert.ok(event, "event should not be null");
|
|
202
|
+
assert.equal(event!.type, "toolcall_start");
|
|
203
|
+
if (event!.type === "toolcall_start") {
|
|
204
|
+
const toolCall = (event.partial.content[event.contentIndex] as any);
|
|
205
|
+
assert.equal(toolCall.name, "gsd_plan_milestone");
|
|
206
|
+
assert.equal(toolCall.mcpServer, "gsd-workflow");
|
|
207
|
+
}
|
|
208
|
+
});
|
|
209
|
+
|
|
210
|
+
test("leaves non-MCP tool names untouched", () => {
|
|
211
|
+
const builder = new PartialMessageBuilder("claude-sonnet-4-20250514");
|
|
212
|
+
const event = builder.handleEvent({
|
|
213
|
+
type: "content_block_start",
|
|
214
|
+
index: 0,
|
|
215
|
+
content_block: { type: "tool_use", id: "tool_1", name: "Bash", input: {} },
|
|
216
|
+
} as BetaRawMessageStreamEvent);
|
|
217
|
+
|
|
218
|
+
assert.ok(event);
|
|
219
|
+
if (event!.type === "toolcall_start") {
|
|
220
|
+
const toolCall = (event.partial.content[event.contentIndex] as any);
|
|
221
|
+
assert.equal(toolCall.name, "Bash");
|
|
222
|
+
assert.equal(toolCall.mcpServer, undefined);
|
|
223
|
+
}
|
|
224
|
+
});
|
|
225
|
+
|
|
226
|
+
test("mapContentBlock strips MCP prefix on full tool_use blocks", () => {
|
|
227
|
+
const block: BetaContentBlock = {
|
|
228
|
+
type: "tool_use",
|
|
229
|
+
id: "tool_2",
|
|
230
|
+
name: "mcp__gsd-workflow__gsd_task_complete",
|
|
231
|
+
input: { taskId: "T001" },
|
|
232
|
+
};
|
|
233
|
+
const mapped = mapContentBlock(block) as any;
|
|
234
|
+
assert.equal(mapped.type, "toolCall");
|
|
235
|
+
assert.equal(mapped.name, "gsd_task_complete");
|
|
236
|
+
assert.equal(mapped.mcpServer, "gsd-workflow");
|
|
237
|
+
assert.deepEqual(mapped.arguments, { taskId: "T001" });
|
|
238
|
+
});
|
|
239
|
+
});
|