counterfact 2.6.0 → 2.8.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 (98) hide show
  1. package/README.md +14 -207
  2. package/bin/README.md +24 -4
  3. package/bin/counterfact.js +54 -3
  4. package/dist/app.js +81 -28
  5. package/dist/counterfact-types/cookie-options.js +1 -0
  6. package/dist/counterfact-types/counterfact-response.js +7 -0
  7. package/dist/counterfact-types/example-names.js +1 -0
  8. package/dist/counterfact-types/example.js +1 -0
  9. package/dist/counterfact-types/generic-response-builder.js +1 -0
  10. package/dist/counterfact-types/http-status-code.js +1 -0
  11. package/dist/counterfact-types/if-has-key.js +1 -0
  12. package/dist/counterfact-types/index.js +0 -1
  13. package/dist/counterfact-types/maybe-promise.js +1 -0
  14. package/dist/counterfact-types/media-type.js +1 -0
  15. package/dist/counterfact-types/omit-all.js +1 -0
  16. package/dist/counterfact-types/omit-value-when-never.js +1 -0
  17. package/dist/counterfact-types/open-api-content.js +1 -0
  18. package/dist/counterfact-types/open-api-operation.js +1 -0
  19. package/dist/counterfact-types/open-api-parameters.js +1 -0
  20. package/dist/counterfact-types/open-api-response.js +1 -0
  21. package/dist/counterfact-types/random-function.js +1 -0
  22. package/dist/counterfact-types/response-builder-factory.js +1 -0
  23. package/dist/counterfact-types/response-builder.js +1 -0
  24. package/dist/counterfact-types/wide-operation-argument.js +1 -0
  25. package/dist/counterfact-types/wide-response-builder.js +1 -0
  26. package/dist/migrate/update-route-types.js +2 -3
  27. package/dist/repl/raw-http-client.js +19 -0
  28. package/dist/repl/repl.js +116 -4
  29. package/dist/repl/route-builder.js +68 -0
  30. package/dist/server/constants.js +8 -0
  31. package/dist/server/context-registry.js +70 -1
  32. package/dist/server/counterfact-types/cookie-options.ts +14 -0
  33. package/dist/server/counterfact-types/counterfact-response.ts +15 -0
  34. package/dist/server/counterfact-types/example-names.ts +13 -0
  35. package/dist/server/counterfact-types/example.ts +10 -0
  36. package/dist/server/counterfact-types/generic-response-builder.ts +164 -0
  37. package/dist/server/counterfact-types/http-status-code.ts +62 -0
  38. package/dist/server/counterfact-types/if-has-key.ts +19 -0
  39. package/dist/server/counterfact-types/index.ts +20 -338
  40. package/dist/server/counterfact-types/maybe-promise.ts +6 -0
  41. package/dist/server/counterfact-types/media-type.ts +6 -0
  42. package/dist/server/counterfact-types/omit-all.ts +11 -0
  43. package/dist/server/counterfact-types/omit-value-when-never.ts +11 -0
  44. package/dist/server/counterfact-types/open-api-content.ts +8 -0
  45. package/dist/server/counterfact-types/open-api-operation.ts +36 -0
  46. package/dist/server/counterfact-types/open-api-parameters.ts +16 -0
  47. package/dist/server/counterfact-types/open-api-response.ts +22 -0
  48. package/dist/server/counterfact-types/random-function.ts +9 -0
  49. package/dist/server/counterfact-types/response-builder-factory.ts +16 -0
  50. package/dist/server/counterfact-types/response-builder.ts +31 -0
  51. package/dist/server/counterfact-types/wide-operation-argument.ts +17 -0
  52. package/dist/server/counterfact-types/wide-response-builder.ts +26 -0
  53. package/dist/server/create-koa-app.js +28 -24
  54. package/dist/server/determine-module-kind.js +13 -0
  55. package/dist/server/dispatcher.js +64 -5
  56. package/dist/server/file-discovery.js +20 -9
  57. package/dist/server/is-proxy-enabled-for-path.js +12 -0
  58. package/dist/server/json-to-xml.js +11 -1
  59. package/dist/server/koa-middleware.js +25 -2
  60. package/dist/server/load-openapi-document.js +6 -0
  61. package/dist/server/module-dependency-graph.js +25 -0
  62. package/dist/server/module-loader.js +112 -17
  63. package/dist/server/module-tree.js +36 -0
  64. package/dist/server/openapi-document.js +69 -0
  65. package/dist/server/openapi-middleware.js +34 -5
  66. package/dist/server/openapi-watcher.js +35 -0
  67. package/dist/server/registry.js +89 -0
  68. package/dist/server/request-validator.js +3 -7
  69. package/dist/server/response-builder.js +18 -0
  70. package/dist/server/response-validator.js +58 -0
  71. package/dist/server/scenario-registry.js +55 -0
  72. package/dist/server/tools.js +29 -2
  73. package/dist/server/transpiler.js +23 -9
  74. package/dist/typescript-generator/code-generator.js +117 -4
  75. package/dist/typescript-generator/coder.js +80 -2
  76. package/dist/typescript-generator/operation-coder.js +13 -5
  77. package/dist/typescript-generator/operation-type-coder.js +40 -53
  78. package/dist/typescript-generator/parameters-type-coder.js +2 -4
  79. package/dist/typescript-generator/prune.js +2 -1
  80. package/dist/typescript-generator/read-only-comments.js +1 -1
  81. package/dist/typescript-generator/repository.js +76 -20
  82. package/dist/typescript-generator/requirement.js +77 -1
  83. package/dist/typescript-generator/reserved-words.js +50 -0
  84. package/dist/typescript-generator/scenario-file-generator.js +235 -0
  85. package/dist/typescript-generator/script.js +70 -7
  86. package/dist/typescript-generator/specification.js +27 -0
  87. package/dist/util/ensure-directory-exists.js +7 -0
  88. package/dist/util/forward-slash-path.js +63 -0
  89. package/dist/util/load-config-file.js +44 -0
  90. package/dist/util/read-file.js +11 -0
  91. package/dist/util/runtime-can-execute-erasable-ts.js +11 -0
  92. package/dist/util/windows-escape.js +18 -0
  93. package/package.json +9 -10
  94. package/dist/client/README.md +0 -14
  95. package/dist/client/index.html.hbs +0 -244
  96. package/dist/client/rapi-doc.html.hbs +0 -36
  97. package/dist/server/page-middleware.js +0 -23
  98. package/dist/typescript-generator/generate.js +0 -63
@@ -0,0 +1,62 @@
1
+ /**
2
+ * A union of all standard HTTP status codes.
3
+ * Used to constrain the status code argument in response builder calls and
4
+ * generated route handler types.
5
+ */
6
+ export type HttpStatusCode =
7
+ | 100
8
+ | 101
9
+ | 102
10
+ | 200
11
+ | 201
12
+ | 202
13
+ | 203
14
+ | 204
15
+ | 205
16
+ | 206
17
+ | 207
18
+ | 226
19
+ | 300
20
+ | 301
21
+ | 302
22
+ | 303
23
+ | 304
24
+ | 305
25
+ | 307
26
+ | 308
27
+ | 400
28
+ | 401
29
+ | 402
30
+ | 403
31
+ | 404
32
+ | 405
33
+ | 406
34
+ | 407
35
+ | 408
36
+ | 409
37
+ | 410
38
+ | 411
39
+ | 412
40
+ | 413
41
+ | 414
42
+ | 415
43
+ | 416
44
+ | 417
45
+ | 418
46
+ | 422
47
+ | 423
48
+ | 424
49
+ | 426
50
+ | 428
51
+ | 429
52
+ | 431
53
+ | 451
54
+ | 500
55
+ | 501
56
+ | 502
57
+ | 503
58
+ | 504
59
+ | 505
60
+ | 506
61
+ | 507
62
+ | 511;
@@ -0,0 +1,19 @@
1
+ /**
2
+ * Conditional type that resolves to `Yes` when `SomeObject` has at least one
3
+ * key that contains any string from `Keys` as a substring, and `No` otherwise.
4
+ * Used to determine whether a shortcut method (e.g. `.json()`, `.html()`)
5
+ * should be present on the response builder for a given response type.
6
+ */
7
+ export type IfHasKey<
8
+ SomeObject,
9
+ Keys extends readonly string[],
10
+ Yes,
11
+ No,
12
+ > = Keys extends [
13
+ infer FirstKey extends string,
14
+ ...infer RestKeys extends string[],
15
+ ]
16
+ ? Extract<keyof SomeObject, `${string}${FirstKey}${string}`> extends never
17
+ ? IfHasKey<SomeObject, RestKeys, Yes, No>
18
+ : Yes
19
+ : No;
@@ -1,339 +1,21 @@
1
- import { OpenApiHeader } from "./open-api-header";
2
-
3
- interface OpenApiContent {
4
- schema: unknown;
5
- }
6
-
7
- interface Example {
8
- description: string;
9
- summary: string;
10
- value: unknown;
11
- }
12
-
13
- interface CookieOptions {
14
- domain?: string;
15
- expires?: Date;
16
- httpOnly?: boolean;
17
- maxAge?: number;
18
- path?: string;
19
- sameSite?: "lax" | "none" | "strict";
20
- secure?: boolean;
21
- }
22
-
23
- const counterfactResponse = Symbol("Counterfact Response");
24
-
25
- type COUNTERFACT_RESPONSE = {
26
- [counterfactResponse]: typeof counterfactResponse;
27
- };
28
-
29
- type MediaType = `${string}/${string}`;
30
-
31
- type MaybePromise<T> = T | Promise<T>;
32
-
33
- type OmitAll<T, K extends readonly string[]> = {
34
- [P in keyof T as P extends `${string}${K[number]}${string}`
35
- ? never
36
- : P]: T[P];
37
- };
38
-
39
- type OmitValueWhenNever<Base> = Pick<
40
- Base,
41
- {
42
- [Key in keyof Base]: [Base[Key]] extends [never] ? never : Key;
43
- }[keyof Base]
44
- >;
45
-
46
- interface OpenApiResponse {
47
- content: { [key: MediaType]: OpenApiContent };
48
- examples?: { [key: string]: unknown };
49
- headers: { [key: string]: { schema: unknown } };
50
- requiredHeaders: string;
51
- }
52
-
53
- interface OpenApiResponses {
54
- [key: string]: OpenApiResponse;
55
- }
56
-
57
- type IfHasKey<
58
- SomeObject,
59
- Keys extends readonly string[],
60
- Yes,
61
- No,
62
- > = Keys extends [
63
- infer FirstKey extends string,
64
- ...infer RestKeys extends string[],
65
- ]
66
- ? Extract<keyof SomeObject, `${string}${FirstKey}${string}`> extends never
67
- ? IfHasKey<SomeObject, RestKeys, Yes, No>
68
- : Yes
69
- : No;
70
-
71
- type SchemasOf<T extends { [key: string]: { schema: unknown } }> = {
72
- [K in keyof T]: T[K]["schema"];
73
- }[keyof T];
74
-
75
- type MaybeShortcut<
76
- ContentTypes extends MediaType[],
77
- Response extends OpenApiResponse,
78
- > = IfHasKey<
79
- Response["content"],
80
- ContentTypes,
81
- (body: SchemasOf<Response["content"]>) => GenericResponseBuilder<{
82
- content: NeverIfEmpty<OmitAll<Response["content"], ContentTypes>>;
83
- headers: Response["headers"];
84
- requiredHeaders: Response["requiredHeaders"];
85
- }>,
86
- never
87
- >;
88
-
89
- type NeverIfEmpty<Record> = object extends Record ? never : Record;
90
-
91
- type MatchFunction<Response extends OpenApiResponse> = <
92
- ContentType extends MediaType & keyof Response["content"],
93
- >(
94
- contentType: ContentType,
95
- body: Response["content"][ContentType]["schema"],
96
- ) => GenericResponseBuilder<{
97
- content: NeverIfEmpty<Omit<Response["content"], ContentType>>;
98
- headers: Response["headers"];
99
- requiredHeaders: Response["requiredHeaders"];
100
- }>;
101
-
102
- type HeaderFunction<Response extends OpenApiResponse> = <
103
- Header extends string & keyof Response["headers"],
104
- >(
105
- header: Header,
106
- value: Response["headers"][Header]["schema"],
107
- ) => GenericResponseBuilder<{
108
- content: NeverIfEmpty<Response["content"]>;
109
- headers: NeverIfEmpty<Omit<Response["headers"], Header>>;
110
- requiredHeaders: Exclude<Response["requiredHeaders"], Header>;
111
- }>;
112
-
113
- type RandomFunction = () => MaybePromise<COUNTERFACT_RESPONSE>;
114
-
115
- type ExampleNames<Response extends OpenApiResponse> = Response extends {
116
- examples: infer E;
117
- }
118
- ? keyof E & string
119
- : never;
120
-
121
- interface ResponseBuilder {
122
- [status: number | `${number} ${string}`]: ResponseBuilder;
123
- binary: (body: Uint8Array | string) => ResponseBuilder;
124
- content?: { body: unknown; type: string }[];
125
- cookie: (
126
- name: string,
127
- value: string,
128
- options?: CookieOptions,
129
- ) => ResponseBuilder;
130
- example: (name: string) => ResponseBuilder;
131
- header: (name: string, value: string) => ResponseBuilder;
132
- headers: { [name: string]: string | string[] };
133
- html: (body: unknown) => ResponseBuilder;
134
- json: (body: unknown) => ResponseBuilder;
135
- match: (contentType: string, body: unknown) => ResponseBuilder;
136
- random: () => MaybePromise<ResponseBuilder>;
137
- randomLegacy: () => MaybePromise<ResponseBuilder>;
138
- status?: number;
139
- text: (body: unknown) => ResponseBuilder;
140
- xml: (body: unknown) => ResponseBuilder;
141
- }
142
-
143
- export type GenericResponseBuilderInner<
144
- Response extends OpenApiResponse = OpenApiResponse,
145
- > = OmitValueWhenNever<{
146
- binary: MaybeShortcut<["application/octet-stream"], Response>;
147
- cookie: (
148
- name: string,
149
- value: string,
150
- options?: CookieOptions,
151
- ) => GenericResponseBuilder<Response>;
152
- header: [keyof Response["headers"]] extends [never]
153
- ? never
154
- : HeaderFunction<Response>;
155
- html: MaybeShortcut<["text/html"], Response>;
156
- json: MaybeShortcut<
157
- [
158
- "application/json",
159
- "text/json",
160
- "text/x-json",
161
- "application/xml",
162
- "text/xml",
163
- ],
164
- Response
165
- >;
166
- match: [keyof Response["content"]] extends [never]
167
- ? never
168
- : MatchFunction<Response>;
169
- random: [keyof Response["content"]] extends [never] ? never : RandomFunction;
170
- example: [ExampleNames<Response>] extends [never]
171
- ? never
172
- : (name: ExampleNames<Response>) => COUNTERFACT_RESPONSE;
173
- text: MaybeShortcut<["text/plain"], Response>;
174
- xml: MaybeShortcut<["application/xml", "text/xml"], Response>;
175
- }>;
176
-
177
- type GenericResponseBuilder<
178
- Response extends OpenApiResponse = OpenApiResponse,
179
- > =
180
- object extends OmitValueWhenNever<Response>
181
- ? COUNTERFACT_RESPONSE
182
- : keyof OmitValueWhenNever<Response> extends "headers"
183
- ? {
184
- ALL_REMAINING_HEADERS_ARE_OPTIONAL: COUNTERFACT_RESPONSE;
185
- header: HeaderFunction<Response>;
186
- }
187
- : GenericResponseBuilderInner<Response>;
188
-
189
- type ResponseBuilderFactory<
190
- Responses extends OpenApiResponses = OpenApiResponses,
191
- > = {
192
- [StatusCode in keyof Responses]: GenericResponseBuilder<
193
- Responses[StatusCode]
194
- >;
195
- } & { [key: string]: GenericResponseBuilder<Responses["default"]> };
196
-
197
- type HttpStatusCode =
198
- | 100
199
- | 101
200
- | 102
201
- | 200
202
- | 201
203
- | 202
204
- | 203
205
- | 204
206
- | 205
207
- | 206
208
- | 207
209
- | 226
210
- | 300
211
- | 301
212
- | 302
213
- | 303
214
- | 304
215
- | 305
216
- | 307
217
- | 308
218
- | 400
219
- | 401
220
- | 402
221
- | 403
222
- | 404
223
- | 405
224
- | 406
225
- | 407
226
- | 408
227
- | 409
228
- | 410
229
- | 411
230
- | 412
231
- | 413
232
- | 414
233
- | 415
234
- | 416
235
- | 417
236
- | 418
237
- | 422
238
- | 423
239
- | 424
240
- | 426
241
- | 428
242
- | 429
243
- | 431
244
- | 451
245
- | 500
246
- | 501
247
- | 502
248
- | 503
249
- | 504
250
- | 505
251
- | 506
252
- | 507
253
- | 511;
254
-
255
- interface OpenApiParameters {
256
- in: "body" | "cookie" | "formData" | "header" | "path" | "query";
257
- name: string;
258
- required?: boolean;
259
- schema?: {
260
- [key: string]: unknown;
261
- type?: string;
262
- };
263
- type?: "string" | "number" | "integer" | "boolean";
264
- }
265
-
266
- interface OpenApiOperation {
267
- parameters?: OpenApiParameters[];
268
- produces?: string[];
269
- requestBody?: {
270
- content?: {
271
- [mediaType: string]: {
272
- schema: { [key: string]: unknown };
273
- };
274
- };
275
- required?: boolean;
276
- };
277
- responses: {
278
- [status: string]: {
279
- content?: {
280
- [type: number | string]: {
281
- examples?: { [key: string]: Example };
282
- schema: { [key: string]: unknown };
283
- };
284
- };
285
- examples?: { [key: string]: unknown };
286
- headers?: {
287
- [name: string]: OpenApiHeader;
288
- };
289
- schema?: { [key: string]: unknown };
290
- };
291
- };
292
- }
293
-
294
- interface WideResponseBuilder {
295
- binary: (body: Uint8Array | string) => WideResponseBuilder;
296
- example: (name: string) => WideResponseBuilder;
297
- cookie: (
298
- name: string,
299
- value: string,
300
- options?: CookieOptions,
301
- ) => WideResponseBuilder;
302
- header: (body: unknown) => WideResponseBuilder;
303
- html: (body: unknown) => WideResponseBuilder;
304
- json: (body: unknown) => WideResponseBuilder;
305
- match: (contentType: string, body: unknown) => WideResponseBuilder;
306
- random: () => MaybePromise<WideResponseBuilder>;
307
- text: (body: unknown) => WideResponseBuilder;
308
- xml: (body: unknown) => WideResponseBuilder;
309
- }
310
-
311
- interface WideOperationArgument {
312
- body: unknown;
313
- context: unknown;
314
- headers: { [key: string]: string };
315
- path: { [key: string]: string };
316
- proxy: (url: string) => { proxyUrl: string };
317
- query: { [key: string]: string };
318
- response: { [key: number]: WideResponseBuilder };
319
- }
320
-
321
- export type { COUNTERFACT_RESPONSE };
322
-
1
+ export type { CookieOptions } from "./cookie-options.js";
2
+ export type { COUNTERFACT_RESPONSE } from "./counterfact-response.js";
3
+ export type { ExampleNames } from "./example-names.js";
323
4
  export type {
324
- CookieOptions,
325
- ExampleNames,
326
- HttpStatusCode,
327
- MaybePromise,
328
- MediaType,
329
- OmitValueWhenNever,
330
- OpenApiOperation,
331
- OpenApiParameters,
332
- OpenApiResponse,
333
- ResponseBuilder,
334
- ResponseBuilderFactory,
335
- WideOperationArgument,
336
- WideResponseBuilder,
337
- OmitAll,
338
- IfHasKey,
339
- };
5
+ GenericResponseBuilder,
6
+ GenericResponseBuilderInner,
7
+ } from "./generic-response-builder.js";
8
+ export type { HttpStatusCode } from "./http-status-code.js";
9
+ export type { IfHasKey } from "./if-has-key.js";
10
+ export type { MaybePromise } from "./maybe-promise.js";
11
+ export type { MediaType } from "./media-type.js";
12
+ export type { OmitAll } from "./omit-all.js";
13
+ export type { OmitValueWhenNever } from "./omit-value-when-never.js";
14
+ export type { OpenApiHeader } from "./open-api-header.js";
15
+ export type { OpenApiOperation } from "./open-api-operation.js";
16
+ export type { OpenApiParameters } from "./open-api-parameters.js";
17
+ export type { OpenApiResponse } from "./open-api-response.js";
18
+ export type { ResponseBuilder } from "./response-builder.js";
19
+ export type { ResponseBuilderFactory } from "./response-builder-factory.js";
20
+ export type { WideOperationArgument } from "./wide-operation-argument.js";
21
+ export type { WideResponseBuilder } from "./wide-response-builder.js";
@@ -0,0 +1,6 @@
1
+ /**
2
+ * A value that is either `T` directly or a `Promise<T>`.
3
+ * Route handlers may return either synchronous values or promises, and
4
+ * Counterfact will await them transparently.
5
+ */
6
+ export type MaybePromise<T> = T | Promise<T>;
@@ -0,0 +1,6 @@
1
+ /**
2
+ * Represents an IANA media type string in the format `type/subtype`
3
+ * (e.g. `"application/json"`, `"text/plain"`, `"image/png"`).
4
+ * Used to identify the content type of an HTTP request or response body.
5
+ */
6
+ export type MediaType = `${string}/${string}`;
@@ -0,0 +1,11 @@
1
+ /**
2
+ * Removes all keys from `T` whose names contain any of the strings in `K`
3
+ * as a substring (prefix, suffix, or exact match).
4
+ * Used internally to narrow the set of available content-type methods on the
5
+ * response builder after one has already been called.
6
+ */
7
+ export type OmitAll<T, K extends readonly string[]> = {
8
+ [P in keyof T as P extends `${string}${K[number]}${string}`
9
+ ? never
10
+ : P]: T[P];
11
+ };
@@ -0,0 +1,11 @@
1
+ /**
2
+ * Creates a new type from `Base` that omits any keys whose value type is
3
+ * `never`. This is used to strip unavailable builder methods (those that
4
+ * don't apply to the current response shape) from the fluent response builder.
5
+ */
6
+ export type OmitValueWhenNever<Base> = Pick<
7
+ Base,
8
+ {
9
+ [Key in keyof Base]: [Base[Key]] extends [never] ? never : Key;
10
+ }[keyof Base]
11
+ >;
@@ -0,0 +1,8 @@
1
+ /**
2
+ * Represents a single content entry in an OpenAPI response object.
3
+ * The `schema` property holds the JSON Schema definition for the body of
4
+ * a response with this media type.
5
+ */
6
+ export interface OpenApiContent {
7
+ schema: unknown;
8
+ }
@@ -0,0 +1,36 @@
1
+ import type { Example } from "./example.js";
2
+ import type { OpenApiHeader } from "./open-api-header.js";
3
+ import type { OpenApiParameters } from "./open-api-parameters.js";
4
+
5
+ /**
6
+ * Describes a single HTTP operation (e.g. `GET /pets`) as defined in an
7
+ * OpenAPI document. Used internally to derive the strongly-typed argument
8
+ * and response builder types for generated route handler functions.
9
+ */
10
+ export interface OpenApiOperation {
11
+ parameters?: OpenApiParameters[];
12
+ produces?: string[];
13
+ requestBody?: {
14
+ content?: {
15
+ [mediaType: string]: {
16
+ schema: { [key: string]: unknown };
17
+ };
18
+ };
19
+ required?: boolean;
20
+ };
21
+ responses: {
22
+ [status: string]: {
23
+ content?: {
24
+ [type: number | string]: {
25
+ examples?: { [key: string]: Example };
26
+ schema: { [key: string]: unknown };
27
+ };
28
+ };
29
+ examples?: { [key: string]: unknown };
30
+ headers?: {
31
+ [name: string]: OpenApiHeader;
32
+ };
33
+ schema?: { [key: string]: unknown };
34
+ };
35
+ };
36
+ }
@@ -0,0 +1,16 @@
1
+ /**
2
+ * Describes a single parameter (path, query, header, cookie, body, or
3
+ * formData) as defined in an OpenAPI document. Used internally to type the
4
+ * `path`, `query`, `headers`, and `body` properties of a route handler's
5
+ * argument object.
6
+ */
7
+ export interface OpenApiParameters {
8
+ in: "body" | "cookie" | "formData" | "header" | "path" | "query";
9
+ name: string;
10
+ required?: boolean;
11
+ schema?: {
12
+ [key: string]: unknown;
13
+ type?: string;
14
+ };
15
+ type?: "string" | "number" | "integer" | "boolean";
16
+ }
@@ -0,0 +1,22 @@
1
+ import type { MediaType } from "./media-type.js";
2
+ import type { OpenApiContent } from "./open-api-content.js";
3
+
4
+ /**
5
+ * Describes a single HTTP response as modelled in an OpenAPI document.
6
+ * Contains the allowed content types, optional named examples, and the
7
+ * required/optional response headers for that response.
8
+ */
9
+ export interface OpenApiResponse {
10
+ content: { [key: MediaType]: OpenApiContent };
11
+ examples?: { [key: string]: unknown };
12
+ headers: { [key: string]: { schema: unknown } };
13
+ requiredHeaders: string;
14
+ }
15
+
16
+ /**
17
+ * A map of HTTP status codes (or `"default"`) to their corresponding
18
+ * `OpenApiResponse` definitions for a given operation.
19
+ */
20
+ export interface OpenApiResponses {
21
+ [key: string]: OpenApiResponse;
22
+ }
@@ -0,0 +1,9 @@
1
+ import type { COUNTERFACT_RESPONSE } from "./counterfact-response.js";
2
+ import type { MaybePromise } from "./maybe-promise.js";
3
+
4
+ /**
5
+ * The type of the `.random()` method on the response builder.
6
+ * When called, it randomly selects one of the available content-type examples
7
+ * and returns a completed `COUNTERFACT_RESPONSE`.
8
+ */
9
+ export type RandomFunction = () => MaybePromise<COUNTERFACT_RESPONSE>;
@@ -0,0 +1,16 @@
1
+ import type { GenericResponseBuilder } from "./generic-response-builder.js";
2
+ import type { OpenApiResponses } from "./open-api-response.js";
3
+
4
+ /**
5
+ * Maps each HTTP status code (or `"default"`) in an OpenAPI operation's
6
+ * response definitions to the corresponding `GenericResponseBuilder`.
7
+ * This is the type of the `response` property in a generated route handler's
8
+ * argument object, allowing handlers to call e.g. `response[200].json(body)`.
9
+ */
10
+ export type ResponseBuilderFactory<
11
+ Responses extends OpenApiResponses = OpenApiResponses,
12
+ > = {
13
+ [StatusCode in keyof Responses]: GenericResponseBuilder<
14
+ Responses[StatusCode]
15
+ >;
16
+ } & { [key: string]: GenericResponseBuilder<Responses["default"]> };
@@ -0,0 +1,31 @@
1
+ import type { CookieOptions } from "./cookie-options.js";
2
+ import type { MaybePromise } from "./maybe-promise.js";
3
+
4
+ /**
5
+ * A loosely-typed, chainable response builder used in non-generated contexts
6
+ * (e.g. middleware or wide/catch-all route handlers) where the exact response
7
+ * shape is not statically known. For generated route handlers, prefer the
8
+ * strongly-typed `GenericResponseBuilder`.
9
+ */
10
+ export interface ResponseBuilder {
11
+ [status: number | `${number} ${string}`]: ResponseBuilder;
12
+ binary: (body: Uint8Array | string) => ResponseBuilder;
13
+ content?: { body: unknown; type: string }[];
14
+ cookie: (
15
+ name: string,
16
+ value: string,
17
+ options?: CookieOptions,
18
+ ) => ResponseBuilder;
19
+ empty: () => ResponseBuilder;
20
+ example: (name: string) => ResponseBuilder;
21
+ header: (name: string, value: string) => ResponseBuilder;
22
+ headers: { [name: string]: string | string[] };
23
+ html: (body: unknown) => ResponseBuilder;
24
+ json: (body: unknown) => ResponseBuilder;
25
+ match: (contentType: string, body: unknown) => ResponseBuilder;
26
+ random: () => MaybePromise<ResponseBuilder>;
27
+ randomLegacy: () => MaybePromise<ResponseBuilder>;
28
+ status?: number;
29
+ text: (body: unknown) => ResponseBuilder;
30
+ xml: (body: unknown) => ResponseBuilder;
31
+ }
@@ -0,0 +1,17 @@
1
+ import type { WideResponseBuilder } from "./wide-response-builder.js";
2
+
3
+ /**
4
+ * The loosely-typed argument object passed to wide (catch-all) route handlers.
5
+ * Unlike the generated operation argument types, all fields are typed as
6
+ * `unknown` or broad index signatures. Use this when writing handlers that
7
+ * should accept any request without compile-time schema enforcement.
8
+ */
9
+ export interface WideOperationArgument {
10
+ body: unknown;
11
+ context: unknown;
12
+ headers: { [key: string]: string };
13
+ path: { [key: string]: string };
14
+ proxy: (url: string) => { proxyUrl: string };
15
+ query: { [key: string]: string };
16
+ response: { [key: number]: WideResponseBuilder };
17
+ }
@@ -0,0 +1,26 @@
1
+ import type { CookieOptions } from "./cookie-options.js";
2
+ import type { MaybePromise } from "./maybe-promise.js";
3
+
4
+ /**
5
+ * A loosely-typed response builder used in wide (catch-all) route handlers
6
+ * where the response shape is not known at compile time. Unlike the generated
7
+ * `GenericResponseBuilder`, this interface accepts `unknown` for all body
8
+ * arguments and does not enforce content-type constraints.
9
+ */
10
+ export interface WideResponseBuilder {
11
+ binary: (body: Uint8Array | string) => WideResponseBuilder;
12
+ empty: () => WideResponseBuilder;
13
+ example: (name: string) => WideResponseBuilder;
14
+ cookie: (
15
+ name: string,
16
+ value: string,
17
+ options?: CookieOptions,
18
+ ) => WideResponseBuilder;
19
+ header: (body: unknown) => WideResponseBuilder;
20
+ html: (body: unknown) => WideResponseBuilder;
21
+ json: (body: unknown) => WideResponseBuilder;
22
+ match: (contentType: string, body: unknown) => WideResponseBuilder;
23
+ random: () => MaybePromise<WideResponseBuilder>;
24
+ text: (body: unknown) => WideResponseBuilder;
25
+ xml: (body: unknown) => WideResponseBuilder;
26
+ }