ts-procedures 5.16.0 → 6.0.1

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 (147) hide show
  1. package/README.md +2 -0
  2. package/agent_config/claude-code/agents/ts-procedures-architect.md +13 -6
  3. package/agent_config/claude-code/skills/ts-procedures/SKILL.md +26 -4
  4. package/agent_config/claude-code/skills/ts-procedures/anti-patterns.md +87 -19
  5. package/agent_config/claude-code/skills/ts-procedures/api-reference.md +162 -16
  6. package/agent_config/claude-code/skills/ts-procedures/patterns.md +179 -16
  7. package/agent_config/claude-code/skills/ts-procedures-review/SKILL.md +1 -1
  8. package/agent_config/claude-code/skills/ts-procedures-review/checklist.md +20 -12
  9. package/agent_config/claude-code/skills/ts-procedures-scaffold/SKILL.md +2 -1
  10. package/agent_config/claude-code/skills/ts-procedures-scaffold/templates/client.md +22 -15
  11. package/agent_config/claude-code/skills/ts-procedures-scaffold/templates/express-rpc.md +20 -17
  12. package/agent_config/claude-code/skills/ts-procedures-scaffold/templates/hono-api.md +20 -16
  13. package/agent_config/claude-code/skills/ts-procedures-scaffold/templates/hono-rpc.md +20 -17
  14. package/agent_config/claude-code/skills/ts-procedures-scaffold/templates/hono-stream.md +16 -3
  15. package/agent_config/copilot/copilot-instructions.md +78 -12
  16. package/agent_config/cursor/cursorrules +78 -12
  17. package/build/client/call.d.ts +2 -1
  18. package/build/client/call.js +9 -1
  19. package/build/client/call.js.map +1 -1
  20. package/build/client/error-dispatch.d.ts +13 -0
  21. package/build/client/error-dispatch.js +26 -0
  22. package/build/client/error-dispatch.js.map +1 -0
  23. package/build/client/error-dispatch.test.d.ts +1 -0
  24. package/build/client/error-dispatch.test.js +56 -0
  25. package/build/client/error-dispatch.test.js.map +1 -0
  26. package/build/client/fetch-adapter.js +10 -4
  27. package/build/client/fetch-adapter.js.map +1 -1
  28. package/build/client/index.d.ts +2 -1
  29. package/build/client/index.js +5 -1
  30. package/build/client/index.js.map +1 -1
  31. package/build/client/stream.d.ts +2 -1
  32. package/build/client/stream.js +13 -3
  33. package/build/client/stream.js.map +1 -1
  34. package/build/client/typed-error-dispatch.test.d.ts +1 -0
  35. package/build/client/typed-error-dispatch.test.js +168 -0
  36. package/build/client/typed-error-dispatch.test.js.map +1 -0
  37. package/build/client/types.d.ts +37 -0
  38. package/build/codegen/e2e.test.js +9 -4
  39. package/build/codegen/e2e.test.js.map +1 -1
  40. package/build/codegen/emit-client-runtime.js +4 -0
  41. package/build/codegen/emit-client-runtime.js.map +1 -1
  42. package/build/codegen/emit-errors.d.ts +17 -6
  43. package/build/codegen/emit-errors.integration.test.d.ts +1 -0
  44. package/build/codegen/emit-errors.integration.test.js +162 -0
  45. package/build/codegen/emit-errors.integration.test.js.map +1 -0
  46. package/build/codegen/emit-errors.js +50 -39
  47. package/build/codegen/emit-errors.js.map +1 -1
  48. package/build/codegen/emit-errors.test.js +75 -78
  49. package/build/codegen/emit-errors.test.js.map +1 -1
  50. package/build/codegen/emit-index.d.ts +7 -0
  51. package/build/codegen/emit-index.js +26 -4
  52. package/build/codegen/emit-index.js.map +1 -1
  53. package/build/codegen/emit-index.test.js +55 -23
  54. package/build/codegen/emit-index.test.js.map +1 -1
  55. package/build/codegen/emit-scope.d.ts +8 -0
  56. package/build/codegen/emit-scope.js +82 -7
  57. package/build/codegen/emit-scope.js.map +1 -1
  58. package/build/codegen/pipeline.js +22 -2
  59. package/build/codegen/pipeline.js.map +1 -1
  60. package/build/implementations/http/doc-registry.d.ts +17 -1
  61. package/build/implementations/http/doc-registry.js +47 -79
  62. package/build/implementations/http/doc-registry.js.map +1 -1
  63. package/build/implementations/http/doc-registry.test.js +149 -16
  64. package/build/implementations/http/doc-registry.test.js.map +1 -1
  65. package/build/implementations/http/error-taxonomy.d.ts +249 -0
  66. package/build/implementations/http/error-taxonomy.js +252 -0
  67. package/build/implementations/http/error-taxonomy.js.map +1 -0
  68. package/build/implementations/http/error-taxonomy.test.d.ts +1 -0
  69. package/build/implementations/http/error-taxonomy.test.js +399 -0
  70. package/build/implementations/http/error-taxonomy.test.js.map +1 -0
  71. package/build/implementations/http/express-rpc/error-taxonomy.test.d.ts +1 -0
  72. package/build/implementations/http/express-rpc/error-taxonomy.test.js +83 -0
  73. package/build/implementations/http/express-rpc/error-taxonomy.test.js.map +1 -0
  74. package/build/implementations/http/express-rpc/index.d.ts +39 -8
  75. package/build/implementations/http/express-rpc/index.js +39 -8
  76. package/build/implementations/http/express-rpc/index.js.map +1 -1
  77. package/build/implementations/http/hono-api/error-taxonomy.test.d.ts +1 -0
  78. package/build/implementations/http/hono-api/error-taxonomy.test.js +137 -0
  79. package/build/implementations/http/hono-api/error-taxonomy.test.js.map +1 -0
  80. package/build/implementations/http/hono-api/index.d.ts +38 -1
  81. package/build/implementations/http/hono-api/index.js +32 -0
  82. package/build/implementations/http/hono-api/index.js.map +1 -1
  83. package/build/implementations/http/hono-rpc/error-taxonomy.test.d.ts +1 -0
  84. package/build/implementations/http/hono-rpc/error-taxonomy.test.js +64 -0
  85. package/build/implementations/http/hono-rpc/error-taxonomy.test.js.map +1 -0
  86. package/build/implementations/http/hono-rpc/index.d.ts +34 -7
  87. package/build/implementations/http/hono-rpc/index.js +31 -4
  88. package/build/implementations/http/hono-rpc/index.js.map +1 -1
  89. package/build/implementations/http/hono-stream/error-taxonomy.test.d.ts +1 -0
  90. package/build/implementations/http/hono-stream/error-taxonomy.test.js +87 -0
  91. package/build/implementations/http/hono-stream/error-taxonomy.test.js.map +1 -0
  92. package/build/implementations/http/hono-stream/index.d.ts +40 -3
  93. package/build/implementations/http/hono-stream/index.js +37 -10
  94. package/build/implementations/http/hono-stream/index.js.map +1 -1
  95. package/build/implementations/http/hono-stream/index.test.js +45 -18
  96. package/build/implementations/http/hono-stream/index.test.js.map +1 -1
  97. package/build/implementations/http/on-request-error.test.d.ts +1 -0
  98. package/build/implementations/http/on-request-error.test.js +173 -0
  99. package/build/implementations/http/on-request-error.test.js.map +1 -0
  100. package/build/implementations/http/route-errors.test.d.ts +1 -0
  101. package/build/implementations/http/route-errors.test.js +139 -0
  102. package/build/implementations/http/route-errors.test.js.map +1 -0
  103. package/build/implementations/types.d.ts +43 -3
  104. package/docs/client-and-codegen.md +105 -12
  105. package/docs/core.md +14 -5
  106. package/docs/http-integrations.md +138 -5
  107. package/docs/streaming.md +3 -1
  108. package/docs/superpowers/plans/2026-04-24-doc-registry-simplification.md +886 -0
  109. package/package.json +7 -2
  110. package/src/client/call.ts +10 -1
  111. package/src/client/error-dispatch.test.ts +72 -0
  112. package/src/client/error-dispatch.ts +27 -0
  113. package/src/client/fetch-adapter.ts +11 -5
  114. package/src/client/index.ts +9 -0
  115. package/src/client/stream.ts +14 -3
  116. package/src/client/typed-error-dispatch.test.ts +211 -0
  117. package/src/client/types.ts +42 -0
  118. package/src/codegen/e2e.test.ts +9 -4
  119. package/src/codegen/emit-client-runtime.ts +4 -0
  120. package/src/codegen/emit-errors.integration.test.ts +183 -0
  121. package/src/codegen/emit-errors.test.ts +91 -87
  122. package/src/codegen/emit-errors.ts +123 -41
  123. package/src/codegen/emit-index.test.ts +68 -24
  124. package/src/codegen/emit-index.ts +66 -4
  125. package/src/codegen/emit-scope.ts +124 -7
  126. package/src/codegen/pipeline.ts +25 -2
  127. package/src/implementations/http/README.md +21 -7
  128. package/src/implementations/http/doc-registry.test.ts +164 -16
  129. package/src/implementations/http/doc-registry.ts +58 -82
  130. package/src/implementations/http/error-taxonomy.test.ts +438 -0
  131. package/src/implementations/http/error-taxonomy.ts +361 -0
  132. package/src/implementations/http/express-rpc/README.md +23 -24
  133. package/src/implementations/http/express-rpc/error-taxonomy.test.ts +103 -0
  134. package/src/implementations/http/express-rpc/index.ts +75 -14
  135. package/src/implementations/http/hono-api/README.md +284 -0
  136. package/src/implementations/http/hono-api/error-taxonomy.test.ts +179 -0
  137. package/src/implementations/http/hono-api/index.ts +76 -1
  138. package/src/implementations/http/hono-rpc/README.md +20 -21
  139. package/src/implementations/http/hono-rpc/error-taxonomy.test.ts +82 -0
  140. package/src/implementations/http/hono-rpc/index.ts +65 -9
  141. package/src/implementations/http/hono-stream/README.md +44 -25
  142. package/src/implementations/http/hono-stream/error-taxonomy.test.ts +98 -0
  143. package/src/implementations/http/hono-stream/index.test.ts +54 -18
  144. package/src/implementations/http/hono-stream/index.ts +83 -13
  145. package/src/implementations/http/on-request-error.test.ts +201 -0
  146. package/src/implementations/http/route-errors.test.ts +176 -0
  147. package/src/implementations/types.ts +43 -3
@@ -0,0 +1,438 @@
1
+ import { describe, expect, test } from 'vitest'
2
+ import {
3
+ ProcedureError,
4
+ ProcedureValidationError,
5
+ ProcedureYieldValidationError,
6
+ } from '../../errors.js'
7
+ import type { TProcedureRegistration } from '../../index.js'
8
+ import {
9
+ defineErrorTaxonomy,
10
+ resolveErrorResponse,
11
+ defaultErrorTaxonomy,
12
+ } from './error-taxonomy.js'
13
+
14
+ class UseCaseError extends Error {
15
+ constructor(
16
+ readonly externalMsg: string,
17
+ readonly internalMsg: string
18
+ ) {
19
+ super(externalMsg)
20
+ this.name = 'UseCaseError'
21
+ Object.setPrototypeOf(this, UseCaseError.prototype)
22
+ }
23
+ }
24
+
25
+ class AuthError extends Error {
26
+ constructor(readonly reason: 'unauthenticated' | 'forbidden') {
27
+ super(reason)
28
+ this.name = 'AuthError'
29
+ Object.setPrototypeOf(this, AuthError.prototype)
30
+ }
31
+ }
32
+
33
+ const fakeProcedure = { name: 'Test', config: {}, handler: async () => {} } as unknown as TProcedureRegistration
34
+
35
+ describe('defineErrorTaxonomy', () => {
36
+ test('validates exactly one discriminator per entry', () => {
37
+ expect(() =>
38
+ defineErrorTaxonomy({
39
+ Bad: { statusCode: 400 } as any,
40
+ })
41
+ ).toThrow(/exactly one of/)
42
+
43
+ expect(() =>
44
+ defineErrorTaxonomy({
45
+ Bad: {
46
+ class: Error,
47
+ match: (e: unknown): e is Error => e instanceof Error,
48
+ statusCode: 400,
49
+ } as any,
50
+ })
51
+ ).toThrow(/exactly one of/)
52
+ })
53
+
54
+ test('accepts a valid entry', () => {
55
+ const t = defineErrorTaxonomy({
56
+ UseCaseError: { class: UseCaseError, statusCode: 422 },
57
+ })
58
+ expect(t.UseCaseError.statusCode).toBe(422)
59
+ })
60
+ })
61
+
62
+ describe('resolveErrorResponse', () => {
63
+ test('class match uses toResponse output', () => {
64
+ const taxonomy = defineErrorTaxonomy({
65
+ UseCaseError: {
66
+ class: UseCaseError,
67
+ statusCode: 422,
68
+ toResponse: (err) => ({ name: 'UseCaseError', message: err.externalMsg }),
69
+ },
70
+ })
71
+ const resolved = resolveErrorResponse({
72
+ err: new UseCaseError('external', 'internal'),
73
+ userTaxonomy: taxonomy,
74
+ procedure: fakeProcedure,
75
+ raw: {},
76
+ })
77
+ expect(resolved?.statusCode).toBe(422)
78
+ expect(resolved?.body).toEqual({ name: 'UseCaseError', message: 'external' })
79
+ })
80
+
81
+ test('match predicate catches 3rd-party errors', () => {
82
+ const mongoLike = Object.assign(new Error('dup'), { name: 'MongoServerError', code: 11000 })
83
+ const taxonomy = defineErrorTaxonomy({
84
+ MongoDuplicateKey: {
85
+ match: (e): e is Error => e instanceof Error && (e as any).code === 11000,
86
+ statusCode: 409,
87
+ toResponse: () => ({ name: 'Conflict' }),
88
+ },
89
+ })
90
+ const resolved = resolveErrorResponse({
91
+ err: mongoLike,
92
+ userTaxonomy: taxonomy,
93
+ procedure: fakeProcedure,
94
+ raw: {},
95
+ })
96
+ expect(resolved?.statusCode).toBe(409)
97
+ expect(resolved?.body).toEqual({ name: 'Conflict' })
98
+ })
99
+
100
+ test('default toResponse emits { name, message } from entry key', () => {
101
+ const taxonomy = defineErrorTaxonomy({
102
+ AuthError: { class: AuthError, statusCode: 401 },
103
+ })
104
+ const resolved = resolveErrorResponse({
105
+ err: new AuthError('unauthenticated'),
106
+ userTaxonomy: taxonomy,
107
+ procedure: fakeProcedure,
108
+ raw: {},
109
+ })
110
+ expect(resolved?.body).toEqual({ name: 'AuthError', message: 'unauthenticated' })
111
+ })
112
+
113
+ test('first matching entry wins — subclass declared before base', () => {
114
+ const taxonomy = defineErrorTaxonomy({
115
+ ProcedureValidationError: {
116
+ class: ProcedureValidationError,
117
+ statusCode: 400,
118
+ toResponse: () => ({ kind: 'validation' }),
119
+ },
120
+ ProcedureError: {
121
+ class: ProcedureError,
122
+ statusCode: 500,
123
+ toResponse: () => ({ kind: 'base' }),
124
+ },
125
+ })
126
+ const resolved = resolveErrorResponse({
127
+ err: new ProcedureValidationError('Test', 'bad', []),
128
+ userTaxonomy: taxonomy,
129
+ procedure: fakeProcedure,
130
+ includeDefaults: false,
131
+ raw: {},
132
+ })
133
+ expect(resolved?.statusCode).toBe(400)
134
+ expect(resolved?.body).toEqual({ name: 'ProcedureValidationError', kind: 'validation' })
135
+ })
136
+
137
+ test('topological sort fixes a subclass that was declared after its base', () => {
138
+ // Under first-match-wins without sorting, the base would catch the
139
+ // subclass. defineErrorTaxonomy reorders to keep the subclass entry first.
140
+ const taxonomy = defineErrorTaxonomy({
141
+ ProcedureError: {
142
+ class: ProcedureError,
143
+ statusCode: 500,
144
+ toResponse: () => ({ kind: 'base' }),
145
+ },
146
+ ProcedureValidationError: {
147
+ class: ProcedureValidationError,
148
+ statusCode: 400,
149
+ toResponse: () => ({ kind: 'validation' }),
150
+ },
151
+ })
152
+ const resolved = resolveErrorResponse({
153
+ err: new ProcedureValidationError('Test', 'bad', []),
154
+ userTaxonomy: taxonomy,
155
+ procedure: fakeProcedure,
156
+ includeDefaults: false,
157
+ raw: {},
158
+ })
159
+ expect(resolved?.statusCode).toBe(400)
160
+ expect(resolved?.body).toEqual({ name: 'ProcedureValidationError', kind: 'validation' })
161
+ })
162
+
163
+ test('falls through to unknownError when nothing matches', () => {
164
+ const resolved = resolveErrorResponse({
165
+ err: new Error('boom'),
166
+ userTaxonomy: defineErrorTaxonomy({
167
+ AuthError: { class: AuthError, statusCode: 401 },
168
+ }),
169
+ includeDefaults: false,
170
+ unknownError: {
171
+ statusCode: 500,
172
+ toResponse: () => ({ name: 'InternalServerError' }),
173
+ },
174
+ procedure: fakeProcedure,
175
+ raw: {},
176
+ })
177
+ expect(resolved?.statusCode).toBe(500)
178
+ expect(resolved?.body).toEqual({ name: 'InternalServerError' })
179
+ })
180
+
181
+ test('returns null when nothing matches and no unknownError', () => {
182
+ const resolved = resolveErrorResponse({
183
+ err: new Error('boom'),
184
+ userTaxonomy: defineErrorTaxonomy({
185
+ AuthError: { class: AuthError, statusCode: 401 },
186
+ }),
187
+ includeDefaults: false,
188
+ procedure: fakeProcedure,
189
+ raw: {},
190
+ })
191
+ expect(resolved).toBeNull()
192
+ })
193
+
194
+ test('default taxonomy catches ProcedureValidationError with status 400', () => {
195
+ const resolved = resolveErrorResponse({
196
+ err: new ProcedureValidationError('Test', 'bad', []),
197
+ procedure: fakeProcedure,
198
+ raw: {},
199
+ })
200
+ expect(resolved?.statusCode).toBe(400)
201
+ expect((resolved?.body as any).name).toBe('ProcedureValidationError')
202
+ })
203
+
204
+ test('default taxonomy catches ProcedureYieldValidationError with status 500', () => {
205
+ const resolved = resolveErrorResponse({
206
+ err: new ProcedureYieldValidationError('Test', 'bad yield', []),
207
+ procedure: fakeProcedure,
208
+ raw: {},
209
+ })
210
+ expect(resolved?.statusCode).toBe(500)
211
+ expect((resolved?.body as any).name).toBe('ProcedureYieldValidationError')
212
+ })
213
+
214
+ test('includeDefaults: false disables the default taxonomy', () => {
215
+ const resolved = resolveErrorResponse({
216
+ err: new ProcedureValidationError('Test', 'bad', []),
217
+ includeDefaults: false,
218
+ procedure: fakeProcedure,
219
+ raw: {},
220
+ })
221
+ expect(resolved).toBeNull()
222
+ })
223
+
224
+ test('user entry overrides default for the same class', () => {
225
+ const resolved = resolveErrorResponse({
226
+ err: new ProcedureValidationError('Test', 'bad', []),
227
+ userTaxonomy: defineErrorTaxonomy({
228
+ ProcedureValidationError: {
229
+ class: ProcedureValidationError,
230
+ statusCode: 418,
231
+ toResponse: () => ({ overridden: true }),
232
+ },
233
+ }),
234
+ procedure: fakeProcedure,
235
+ raw: {},
236
+ })
237
+ expect(resolved?.statusCode).toBe(418)
238
+ expect(resolved?.body).toEqual({ name: 'ProcedureValidationError', overridden: true })
239
+ })
240
+
241
+ test('onCatch is awaited via runOnCatch', async () => {
242
+ const calls: string[] = []
243
+ const taxonomy = defineErrorTaxonomy({
244
+ UseCaseError: {
245
+ class: UseCaseError,
246
+ statusCode: 422,
247
+ onCatch: async (err) => {
248
+ await new Promise((r) => setTimeout(r, 5))
249
+ calls.push(err.internalMsg)
250
+ },
251
+ },
252
+ })
253
+ const resolved = resolveErrorResponse({
254
+ err: new UseCaseError('ext', 'internal-log'),
255
+ userTaxonomy: taxonomy,
256
+ procedure: fakeProcedure,
257
+ raw: {},
258
+ })
259
+ expect(calls).toEqual([])
260
+ await resolved!.runOnCatch()
261
+ expect(calls).toEqual(['internal-log'])
262
+ })
263
+
264
+ test('unknownError onCatch is awaited', async () => {
265
+ const calls: unknown[] = []
266
+ const resolved = resolveErrorResponse({
267
+ err: new Error('boom'),
268
+ includeDefaults: false,
269
+ unknownError: {
270
+ toResponse: () => ({}),
271
+ onCatch: async (err) => {
272
+ await Promise.resolve()
273
+ calls.push(err)
274
+ },
275
+ },
276
+ procedure: fakeProcedure,
277
+ raw: {},
278
+ })
279
+ await resolved!.runOnCatch()
280
+ expect(calls).toHaveLength(1)
281
+ expect((calls[0] as Error).message).toBe('boom')
282
+ })
283
+
284
+ test('onCatch receives procedure, key and raw', async () => {
285
+ let received: any
286
+ const taxonomy = defineErrorTaxonomy({
287
+ AuthError: {
288
+ class: AuthError,
289
+ statusCode: 401,
290
+ onCatch: (_err, ctx) => {
291
+ received = ctx
292
+ },
293
+ },
294
+ })
295
+ const resolved = resolveErrorResponse({
296
+ err: new AuthError('forbidden'),
297
+ userTaxonomy: taxonomy,
298
+ procedure: fakeProcedure,
299
+ raw: { marker: 'hono-context' },
300
+ })
301
+ await resolved!.runOnCatch()
302
+ expect(received.procedure).toBe(fakeProcedure)
303
+ expect(received.key).toBe('AuthError')
304
+ expect(received.raw).toEqual({ marker: 'hono-context' })
305
+ })
306
+
307
+ test('defaultErrorTaxonomy exposes all four framework error mappings', () => {
308
+ expect(defaultErrorTaxonomy.ProcedureValidationError.statusCode).toBe(400)
309
+ expect(defaultErrorTaxonomy.ProcedureYieldValidationError.statusCode).toBe(500)
310
+ expect(defaultErrorTaxonomy.ProcedureError.statusCode).toBe(500)
311
+ })
312
+
313
+ test('user taxonomy matches cause inside a ProcedureError wrapper', () => {
314
+ // Simulates what the core does when a non-ProcedureError is thrown: wraps
315
+ // into ProcedureError with `cause` preserved.
316
+ const original = new UseCaseError('public', 'private')
317
+ const wrapped = new ProcedureError('Test', 'wrapped')
318
+ ;(wrapped as any).cause = original
319
+
320
+ const taxonomy = defineErrorTaxonomy({
321
+ UseCaseError: {
322
+ class: UseCaseError,
323
+ statusCode: 422,
324
+ toResponse: (err) => ({ name: 'UseCaseError', message: err.externalMsg }),
325
+ },
326
+ })
327
+
328
+ const resolved = resolveErrorResponse({
329
+ err: wrapped,
330
+ userTaxonomy: taxonomy,
331
+ procedure: fakeProcedure,
332
+ raw: {},
333
+ })
334
+ expect(resolved?.statusCode).toBe(422)
335
+ expect(resolved?.body).toEqual({ name: 'UseCaseError', message: 'public' })
336
+ })
337
+
338
+ test('wrapped ProcedureError falls through default taxonomy to unknownError', () => {
339
+ const original = new TypeError('db-broke')
340
+ const wrapped = new ProcedureError('Test', 'wrapped')
341
+ ;(wrapped as any).cause = original
342
+
343
+ const resolved = resolveErrorResponse({
344
+ err: wrapped,
345
+ unknownError: {
346
+ statusCode: 500,
347
+ toResponse: (err) => ({ name: 'InternalServerError', type: (err as Error).constructor.name }),
348
+ },
349
+ procedure: fakeProcedure,
350
+ raw: {},
351
+ })
352
+ expect(resolved?.statusCode).toBe(500)
353
+ expect(resolved?.body).toEqual({ name: 'InternalServerError', type: 'TypeError' })
354
+ })
355
+
356
+ test('direct ProcedureError (no cause) still caught by default entry', () => {
357
+ const direct = new ProcedureError('Test', 'from ctx.error')
358
+ const resolved = resolveErrorResponse({
359
+ err: direct,
360
+ procedure: fakeProcedure,
361
+ raw: {},
362
+ })
363
+ expect(resolved?.statusCode).toBe(500)
364
+ expect((resolved?.body as any).name).toBe('ProcedureError')
365
+ })
366
+
367
+ test('auto-injects name when toResponse omits it', () => {
368
+ const taxonomy = defineErrorTaxonomy({
369
+ UseCaseError: {
370
+ class: UseCaseError,
371
+ statusCode: 422,
372
+ // Returns a body without `name` — resolver should add one.
373
+ toResponse: (err) => ({ message: err.externalMsg, detail: err.internalMsg }),
374
+ },
375
+ })
376
+ const resolved = resolveErrorResponse({
377
+ err: new UseCaseError('ext', 'int'),
378
+ userTaxonomy: taxonomy,
379
+ procedure: fakeProcedure,
380
+ raw: {},
381
+ })
382
+ expect(resolved?.body).toEqual({ name: 'UseCaseError', message: 'ext', detail: 'int' })
383
+ })
384
+
385
+ test('preserves explicit name in toResponse output', () => {
386
+ const taxonomy = defineErrorTaxonomy({
387
+ UseCaseError: {
388
+ class: UseCaseError,
389
+ statusCode: 422,
390
+ toResponse: () => ({ name: 'CustomAlias', reason: 'x' }),
391
+ },
392
+ })
393
+ const resolved = resolveErrorResponse({
394
+ err: new UseCaseError('ext', 'int'),
395
+ userTaxonomy: taxonomy,
396
+ procedure: fakeProcedure,
397
+ raw: {},
398
+ })
399
+ expect((resolved?.body as any).name).toBe('CustomAlias')
400
+ })
401
+
402
+ test('defineErrorTaxonomy topologically sorts class entries (subclass before base)', () => {
403
+ // Declared base-before-subclass on purpose — sort must swap them.
404
+ const taxonomy = defineErrorTaxonomy({
405
+ ProcedureError: {
406
+ class: ProcedureError,
407
+ statusCode: 500,
408
+ toResponse: () => ({ kind: 'base' }),
409
+ },
410
+ ProcedureValidationError: {
411
+ class: ProcedureValidationError,
412
+ statusCode: 400,
413
+ toResponse: () => ({ kind: 'validation' }),
414
+ },
415
+ })
416
+ const keys = Object.keys(taxonomy)
417
+ expect(keys).toEqual(['ProcedureValidationError', 'ProcedureError'])
418
+
419
+ const resolved = resolveErrorResponse({
420
+ err: new ProcedureValidationError('Test', 'bad', []),
421
+ userTaxonomy: taxonomy,
422
+ includeDefaults: false,
423
+ procedure: fakeProcedure,
424
+ raw: {},
425
+ })
426
+ expect(resolved?.statusCode).toBe(400)
427
+ })
428
+
429
+ test('topological sort preserves declared order for unrelated classes', () => {
430
+ class A extends Error {}
431
+ class B extends Error {}
432
+ const taxonomy = defineErrorTaxonomy({
433
+ B: { class: B, statusCode: 400 },
434
+ A: { class: A, statusCode: 400 },
435
+ })
436
+ expect(Object.keys(taxonomy)).toEqual(['B', 'A'])
437
+ })
438
+ })