@apifuse/provider-sdk 2.1.0-beta.5 → 2.1.0-beta.6

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/package.json CHANGED
@@ -1,94 +1,109 @@
1
1
  {
2
- "name": "@apifuse/provider-sdk",
3
- "version": "2.1.0-beta.5",
4
- "private": false,
5
- "type": "module",
6
- "description": "APIFuse Provider SDK Build providers with zero architectural constraints",
7
- "license": "MIT",
8
- "main": "./src/index.ts",
9
- "types": "./src/index.ts",
10
- "publishConfig": {
11
- "access": "public"
12
- },
13
- "files": [
14
- "src",
15
- "!src/__tests__",
16
- "!src/__tests__/**",
17
- "!src/**/*.test.ts",
18
- "!src/index.test.ts",
19
- "bin",
20
- "README.md",
21
- "AUTHORING.md",
22
- "CHANGELOG.md",
23
- "SUBMISSION.md"
24
- ],
25
- "keywords": [
26
- "apifuse",
27
- "provider",
28
- "sdk",
29
- "api",
30
- "zod",
31
- "hono"
32
- ],
33
- "bin": {
34
- "apifuse": "./bin/apifuse.ts"
35
- },
36
- "exports": {
37
- ".": {
38
- "default": "./src/index.ts",
39
- "import": "./src/index.ts",
40
- "types": "./src/index.ts"
41
- },
42
- "./provider": {
43
- "default": "./src/provider.ts",
44
- "import": "./src/provider.ts",
45
- "types": "./src/provider.ts"
46
- },
47
- "./server": {
48
- "default": "./src/server/index.ts",
49
- "import": "./src/server/index.ts",
50
- "types": "./src/server/index.ts"
51
- },
52
- "./testing": {
53
- "default": "./src/testing/index.ts",
54
- "import": "./src/testing/index.ts",
55
- "types": "./src/testing/index.ts"
56
- },
57
- "./create": {
58
- "default": "./src/cli/create.ts",
59
- "import": "./src/cli/create.ts",
60
- "types": "./src/cli/create.ts"
61
- }
62
- },
63
- "scripts": {
64
- "lint": "biome check",
65
- "lint:fix": "biome lint --write",
66
- "format": "biome format --write",
67
- "type-check": "tsgo --noEmit",
68
- "test": "bun test",
69
- "check": "bun run lint && bun run type-check",
70
- "pack:check": "bun bin/apifuse-pack-check.ts",
71
- "pack:smoke": "bun bin/apifuse-pack-smoke.ts"
72
- },
73
- "devDependencies": {
74
- "@biomejs/biome": "^2.5.0",
75
- "@types/bun": "latest",
76
- "@types/node": "^25.9.3",
77
- "typescript": "^6.0.3"
78
- },
79
- "dependencies": {
80
- "@clack/prompts": "^1.5.1",
81
- "@types/ms": "^2.1.0",
82
- "ajv": "^8.17",
83
- "hono": "^4.12.25",
84
- "impit": "0.14.1",
85
- "ioredis": "^5.11.1",
86
- "ms": "^2.1.3",
87
- "playwright": "^1.55.1",
88
- "playwright-extra": "^4.3.6",
89
- "puppeteer-extra-plugin-stealth": "^2.11.2",
90
- "re2-wasm": "^1.0",
91
- "safe-regex": "^2.1",
92
- "zod": "^4.4.3"
93
- }
2
+ "name": "@apifuse/provider-sdk",
3
+ "version": "2.1.0-beta.6",
4
+ "private": false,
5
+ "type": "module",
6
+ "description": "APIFuse Provider SDK \u2014 Build providers with zero architectural constraints",
7
+ "license": "MIT",
8
+ "main": "./src/index.ts",
9
+ "types": "./src/index.ts",
10
+ "publishConfig": {
11
+ "access": "public"
12
+ },
13
+ "files": [
14
+ "src",
15
+ "!src/__tests__",
16
+ "!src/__tests__/**",
17
+ "!src/**/*.test.ts",
18
+ "!src/index.test.ts",
19
+ "bin",
20
+ "README.md",
21
+ "AUTHORING.md",
22
+ "CHANGELOG.md",
23
+ "SUBMISSION.md"
24
+ ],
25
+ "keywords": [
26
+ "apifuse",
27
+ "provider",
28
+ "sdk",
29
+ "api",
30
+ "zod",
31
+ "hono"
32
+ ],
33
+ "bin": {
34
+ "apifuse": "./bin/apifuse.ts"
35
+ },
36
+ "exports": {
37
+ ".": {
38
+ "default": "./src/index.ts",
39
+ "import": "./src/index.ts",
40
+ "types": "./src/index.ts"
41
+ },
42
+ "./provider": {
43
+ "default": "./src/provider.ts",
44
+ "import": "./src/provider.ts",
45
+ "types": "./src/provider.ts"
46
+ },
47
+ "./contract": {
48
+ "default": "./src/contract.ts",
49
+ "import": "./src/contract.ts",
50
+ "types": "./src/contract.ts"
51
+ },
52
+ "./server": {
53
+ "default": "./src/server/index.ts",
54
+ "import": "./src/server/index.ts",
55
+ "types": "./src/server/index.ts"
56
+ },
57
+ "./testing": {
58
+ "default": "./src/testing/index.ts",
59
+ "import": "./src/testing/index.ts",
60
+ "types": "./src/testing/index.ts"
61
+ },
62
+ "./create": {
63
+ "default": "./src/cli/create.ts",
64
+ "import": "./src/cli/create.ts",
65
+ "types": "./src/cli/create.ts"
66
+ }
67
+ },
68
+ "scripts": {
69
+ "lint": "biome lint .",
70
+ "lint:fix": "biome lint --write",
71
+ "format": "biome format --write",
72
+ "type-check": "tsgo --noEmit",
73
+ "test": "bun test",
74
+ "check": "bun run lint && bun run type-check",
75
+ "pack:check": "bun bin/apifuse-pack-check.ts",
76
+ "pack:smoke": "bun bin/apifuse-pack-smoke.ts",
77
+ "release:guard": "bun scripts/guard-release-pr.ts",
78
+ "format:check": "biome format ."
79
+ },
80
+ "devDependencies": {
81
+ "@biomejs/biome": "^2.5.0",
82
+ "@types/bun": "latest",
83
+ "@types/node": "^25.9.3",
84
+ "@typescript/native-preview": "7.0.0-dev.20260419.1"
85
+ },
86
+ "dependencies": {
87
+ "@clack/prompts": "^1.5.1",
88
+ "@types/ms": "^2.1.0",
89
+ "ajv": "^8.17",
90
+ "hono": "^4.12.25",
91
+ "impit": "0.14.1",
92
+ "ioredis": "^5.11.1",
93
+ "ms": "^2.1.3",
94
+ "playwright": "^1.55.1",
95
+ "playwright-extra": "^4.3.6",
96
+ "puppeteer-extra-plugin-stealth": "^2.11.2",
97
+ "re2-wasm": "^1.0",
98
+ "safe-regex": "^2.1",
99
+ "zod": "^4.4.3"
100
+ },
101
+ "repository": {
102
+ "type": "git",
103
+ "url": "git+https://github.com/APIFuseHQ/provider-sdk.git"
104
+ },
105
+ "bugs": {
106
+ "url": "https://github.com/APIFuseHQ/provider-sdk/issues"
107
+ },
108
+ "homepage": "https://github.com/APIFuseHQ/provider-sdk#readme"
94
109
  }
@@ -8,7 +8,12 @@ import {
8
8
  TurnValidationError,
9
9
  ValidationError,
10
10
  } from "../errors";
11
- import type { AuthFlowDefinition, AuthTurn, FlowContext } from "../types";
11
+ import type {
12
+ AuthFlowDefinition,
13
+ AuthFlowInputHandler,
14
+ AuthTurn,
15
+ FlowContext,
16
+ } from "../types";
12
17
 
13
18
  type TurnKind =
14
19
  | "abort"
@@ -21,7 +26,7 @@ type TurnKind =
21
26
  | "redirect"
22
27
  | "retry";
23
28
 
24
- type CeremonyHandler = AuthFlowDefinition["start"];
29
+ type CeremonyHandler = AuthFlowInputHandler;
25
30
 
26
31
  type JsonObject = Record<string, unknown>;
27
32
 
@@ -44,6 +49,7 @@ const authTurnSchema = {
44
49
  additionalProperties: true,
45
50
  },
46
51
  hint: { type: "string" },
52
+ hintKey: { type: "string" },
47
53
  timing: {
48
54
  type: "object",
49
55
  additionalProperties: false,
@@ -13,6 +13,7 @@ export type ProviderChoiceTokenErrorReason =
13
13
  | "invalid_shape"
14
14
  | "invalid_signature"
15
15
  | "invalid_payload"
16
+ | "invalid_binding"
16
17
  | "stale";
17
18
 
18
19
  export class ProviderChoiceTokenError extends Error {
@@ -36,14 +36,14 @@ export const COMMAND_MANIFEST: Record<
36
36
  summary:
37
37
  "Start the local provider dev server with the standard provider server contract.",
38
38
  usage: "apifuse dev [path]",
39
- examples: ["apifuse dev .", "apifuse dev providers/airkorea"],
39
+ examples: ["apifuse dev .", "apifuse dev providers/korea-air-quality"],
40
40
  modulePath: "./apifuse-dev",
41
41
  },
42
42
  check: {
43
43
  name: "check",
44
44
  summary: "Validate provider structure, metadata, fixtures, and schemas.",
45
45
  usage: "apifuse check [path]",
46
- examples: ["apifuse check .", "apifuse check providers/airkorea"],
46
+ examples: ["apifuse check .", "apifuse check providers/korea-air-quality"],
47
47
  modulePath: "./apifuse-check",
48
48
  },
49
49
  "submit-check": {
@@ -72,7 +72,7 @@ export const COMMAND_MANIFEST: Record<
72
72
  usage:
73
73
  'apifuse record [path] --operation <operation> --params \'{"value":"hello"}\'',
74
74
  examples: [
75
- 'apifuse record providers/airkorea --operation realtime --params \'{"stationName":"종로구"}\'',
75
+ 'apifuse record providers/korea-air-quality --operation realtime --params \'{"stationName":"종로구"}\'',
76
76
  ],
77
77
  modulePath: "./apifuse-record",
78
78
  },
@@ -80,7 +80,10 @@ export const COMMAND_MANIFEST: Record<
80
80
  name: "test",
81
81
  summary: "Run provider-focused tests and surface actionable failures.",
82
82
  usage: "apifuse test [path] [--json] [--verbose]",
83
- examples: ["apifuse test .", "apifuse test providers/airkorea --json"],
83
+ examples: [
84
+ "apifuse test .",
85
+ "apifuse test providers/korea-air-quality --json",
86
+ ],
84
87
  modulePath: "./apifuse-test",
85
88
  },
86
89
  perf: {
@@ -90,7 +93,7 @@ export const COMMAND_MANIFEST: Record<
90
93
  usage:
91
94
  "apifuse perf <path> --operation <operation> [--params '<json>'] [options]",
92
95
  examples: [
93
- 'apifuse perf providers/airkorea --operation realtime --params \'{"stationName":"종로구"}\' --runs 5',
96
+ 'apifuse perf providers/korea-air-quality --operation realtime --params \'{"stationName":"종로구"}\' --runs 5',
94
97
  ],
95
98
  modulePath: "./apifuse-perf",
96
99
  },
package/src/cli/create.ts CHANGED
@@ -502,6 +502,15 @@ export async function buildProviderCreatePlan(
502
502
  path: resolve(providerRoot, "operations", "ping.ts"),
503
503
  content: await renderTemplate("operations/ping.ts.tpl", {
504
504
  DISPLAY_NAME: escapeTemplate(options.displayName),
505
+ HANDLER_CTX: options.runtime === "browser" ? "ctx" : "_ctx",
506
+ BROWSER_HANDLER_BLOCK:
507
+ options.runtime === "browser"
508
+ ? '\n const page = await ctx.browser.newPage();\n await page.goto("https://example.com");\n const title = await page.title();\n const frames = await page.frames();\n await page.close();\n'
509
+ : "",
510
+ BROWSER_RESPONSE_FIELDS:
511
+ options.runtime === "browser"
512
+ ? ",\n pageTitle: title,\n frameCount: frames.length"
513
+ : "",
505
514
  }),
506
515
  },
507
516
  {
@@ -701,6 +710,14 @@ function renderStarterLocaleCatalog(
701
710
  locale === "ko"
702
711
  ? "생성된 provider가 샘플 payload를 round-trip했음을 보여주는 사람이 읽을 수 있는 확인 메시지"
703
712
  : "Human-readable confirmation that the generated provider round-tripped the sample payload.",
713
+ pageTitle:
714
+ locale === "ko"
715
+ ? "browser 런타임 provider일 때 로드된 페이지의 제목 (해당되지 않으면 생략)"
716
+ : "Title of the loaded page when the provider uses the browser runtime; omitted otherwise.",
717
+ frameCount:
718
+ locale === "ko"
719
+ ? "browser 런타임 provider일 때 로드된 페이지의 frame 개수 (해당되지 않으면 생략)"
720
+ : "Number of frames in the loaded page when the provider uses the browser runtime; omitted otherwise.",
704
721
  },
705
722
  },
706
723
  };
@@ -744,6 +761,17 @@ function renderAuthBlock(authMode: CreateAuthMode): string {
744
761
  },
745
762
  hint: "Generated placeholder credential flow completed. Replace this with real auth logic.",
746
763
  }),
764
+ refresh: async () => ({
765
+ kind: "complete",
766
+ turnId: crypto.randomUUID(),
767
+ data: {
768
+ credential: {
769
+ username: "replace-with-refreshed-username",
770
+ password: "replace-with-refreshed-password",
771
+ },
772
+ },
773
+ hint: "Return refreshed credential data here, or throw AuthError with code AUTH_REQUIRED when silent refresh is not possible.",
774
+ }),
747
775
  },
748
776
  }`;
749
777
  case "oauth2":
@@ -6,10 +6,11 @@ export const pingOperation = defineOperation({
6
6
  descriptionKey: "operations.ping.description",
7
7
  input: pingInputSchema,
8
8
  output: pingOutputSchema,
9
- handler: async (_ctx, input) => {
9
+ handler: async ({{HANDLER_CTX}}, input) => {
10
+ {{BROWSER_HANDLER_BLOCK}}
10
11
  return {
11
12
  ok: true,
12
- message: "{{DISPLAY_NAME}} received: " + input.value,
13
+ message: "{{DISPLAY_NAME}} received: " + input.value{{BROWSER_RESPONSE_FIELDS}},
13
14
  };
14
15
  },
15
16
  fixtures: {
@@ -11,6 +11,14 @@ export const pingOutputSchema = describeKey(
11
11
  z.object({
12
12
  ok: describeKey(z.boolean(), "schemaDescriptions.output.ok"),
13
13
  message: describeKey(z.string(), "schemaDescriptions.output.message"),
14
+ pageTitle: describeKey(
15
+ z.string().optional(),
16
+ "schemaDescriptions.output.pageTitle",
17
+ ),
18
+ frameCount: describeKey(
19
+ z.number().int().nonnegative().optional(),
20
+ "schemaDescriptions.output.frameCount",
21
+ ),
14
22
  }),
15
23
  "schemaDescriptions.output.root",
16
24
  );
@@ -16,6 +16,8 @@ export const DEFAULT_PROXY_LIFETIME_ENV =
16
16
  "APIFUSE__PROXY__DEFAULT_LIFETIME_MINUTES";
17
17
  export const PROVIDER_CACHE_REDIS_URL_ENV =
18
18
  "APIFUSE__PROVIDER__CACHE_REDIS_URL";
19
+ export const PROVIDER_STATE_REDIS_URL_ENV =
20
+ "APIFUSE__PROVIDER__STATE_REDIS_URL";
19
21
  export const REDIS_URL_ENV = "APIFUSE__REDIS__URL";
20
22
 
21
23
  export type ProxyOptions = {
@@ -175,6 +177,19 @@ const SMARTPROXY_EXTRACTION_SOFT_REFRESH_MS = 10_000;
175
177
 
176
178
  function redisUrlFromEnv(): string | undefined {
177
179
  return (
180
+ process.env.APIFUSE__PROVIDER__CACHE_REDIS_URL?.trim() ||
181
+ process.env[REDIS_URL_ENV]?.trim() ||
182
+ undefined
183
+ );
184
+ }
185
+
186
+ export function providerCacheRedisUrlFromEnv(): string | undefined {
187
+ return redisUrlFromEnv();
188
+ }
189
+
190
+ export function providerStateRedisUrlFromEnv(): string | undefined {
191
+ return (
192
+ process.env[PROVIDER_STATE_REDIS_URL_ENV]?.trim() ||
178
193
  process.env[PROVIDER_CACHE_REDIS_URL_ENV]?.trim() ||
179
194
  process.env[REDIS_URL_ENV]?.trim() ||
180
195
  undefined
@@ -620,7 +635,10 @@ function selectProxyPoolIndex(poolSize: number, attempt = 0): number {
620
635
  if (poolSize <= 1) {
621
636
  return 0;
622
637
  }
623
- return Math.max(0, Math.floor(attempt)) % poolSize;
638
+ const normalizedAttempt = Number.isFinite(attempt)
639
+ ? Math.max(0, Math.floor(attempt))
640
+ : 0;
641
+ return normalizedAttempt % poolSize;
624
642
  }
625
643
 
626
644
  function buildSmartproxyCacheKey(
@@ -0,0 +1,75 @@
1
+ export type JsonPrimitive = string | number | boolean | null;
2
+ export type JsonValue =
3
+ | JsonPrimitive
4
+ | readonly JsonValue[]
5
+ | { readonly [key: string]: JsonValue };
6
+
7
+ export function canonicalJson(value: unknown): string {
8
+ return JSON.stringify(canonicalize(toJsonValue(value) ?? null));
9
+ }
10
+
11
+ export function toJsonValue(value: unknown): JsonValue | undefined {
12
+ if (
13
+ value === null ||
14
+ typeof value === "string" ||
15
+ typeof value === "boolean"
16
+ ) {
17
+ return value;
18
+ }
19
+ if (typeof value === "number") {
20
+ return Number.isFinite(value) ? value : undefined;
21
+ }
22
+ if (Array.isArray(value)) {
23
+ return value.flatMap((item) => {
24
+ const json = toJsonValue(item);
25
+ return json === undefined ? [] : [json];
26
+ });
27
+ }
28
+ if (!isRecord(value)) return undefined;
29
+ return compactObject(
30
+ Object.fromEntries(
31
+ Object.entries(value).flatMap(([key, item]) => {
32
+ const json = toJsonValue(item);
33
+ return json === undefined ? [] : [[key, json]];
34
+ }),
35
+ ),
36
+ );
37
+ }
38
+
39
+ export function compactObject(
40
+ value: Record<string, JsonValue | undefined>,
41
+ ): JsonValue {
42
+ return Object.fromEntries(
43
+ Object.entries(value).filter((entry): entry is [string, JsonValue] => {
44
+ const [, item] = entry;
45
+ return item !== undefined;
46
+ }),
47
+ );
48
+ }
49
+
50
+ export function copyRecordWithout(
51
+ value: unknown,
52
+ ignoredKeys: ReadonlySet<string>,
53
+ ): Record<string, unknown> {
54
+ if (!isRecord(value)) return {};
55
+ return Object.fromEntries(
56
+ Object.entries(value).filter(([key]) => !ignoredKeys.has(key)),
57
+ );
58
+ }
59
+
60
+ export function isRecord(value: unknown): value is Record<string, unknown> {
61
+ return value !== null && typeof value === "object" && !Array.isArray(value);
62
+ }
63
+
64
+ function canonicalize(value: JsonValue): JsonValue {
65
+ if (Array.isArray(value)) return value.map(canonicalize);
66
+ if (!isRecord(value)) return value;
67
+ return Object.fromEntries(
68
+ Object.entries(value)
69
+ .sort(([leftKey], [rightKey]) => leftKey.localeCompare(rightKey))
70
+ .flatMap(([key, item]) => {
71
+ const json = toJsonValue(item);
72
+ return json === undefined ? [] : [[key, canonicalize(json)]];
73
+ }),
74
+ );
75
+ }
@@ -0,0 +1,89 @@
1
+ import { createHash } from "node:crypto";
2
+ import { type ZodType, z } from "zod";
3
+ import {
4
+ canonicalJson,
5
+ compactObject,
6
+ isRecord,
7
+ type JsonValue,
8
+ toJsonValue,
9
+ } from "./contract-json";
10
+ import type { SchemaLike } from "./types";
11
+
12
+ export function describeSchema(schema: SchemaLike): JsonValue {
13
+ if (isZodSchema(schema)) {
14
+ const jsonSchema = zodJsonSchema(schema);
15
+ return compactObject({
16
+ kind: "schema",
17
+ vendor: "zod",
18
+ typeName: getSchemaTypeName(schema),
19
+ jsonSchema,
20
+ jsonSchemaHash:
21
+ jsonSchema === undefined
22
+ ? undefined
23
+ : digest(canonicalJson(jsonSchema)),
24
+ });
25
+ }
26
+ const standard = isRecord(schema) ? schema["~standard"] : undefined;
27
+ if (isRecord(standard)) {
28
+ return compactObject({
29
+ kind: "schema",
30
+ standard: "standard-schema-v1",
31
+ vendor: typeof standard.vendor === "string" ? standard.vendor : "unknown",
32
+ version:
33
+ typeof standard.version === "number" ||
34
+ typeof standard.version === "string"
35
+ ? standard.version
36
+ : undefined,
37
+ });
38
+ }
39
+ return compactObject({
40
+ kind: "schema",
41
+ vendor: "zod",
42
+ typeName: getSchemaTypeName(schema),
43
+ });
44
+ }
45
+
46
+ export function serializeSmsMatcher(
47
+ value: Record<string, unknown>,
48
+ ): Record<string, unknown> {
49
+ const code = value.code;
50
+ if (!isRecord(code)) return value;
51
+ const pattern = code.pattern;
52
+ if (!(pattern instanceof RegExp)) return value;
53
+ return {
54
+ ...value,
55
+ code: {
56
+ ...code,
57
+ pattern: {
58
+ source: pattern.source,
59
+ flags: pattern.flags,
60
+ },
61
+ },
62
+ };
63
+ }
64
+
65
+ function digest(value: string): string {
66
+ return createHash("sha256").update(value).digest("hex");
67
+ }
68
+
69
+ function isZodSchema(schema: SchemaLike): schema is ZodType {
70
+ return schema instanceof z.ZodType;
71
+ }
72
+
73
+ function zodJsonSchema(schema: ZodType): JsonValue | undefined {
74
+ try {
75
+ const jsonSchema = z.toJSONSchema(schema);
76
+ return toJsonValue(jsonSchema);
77
+ } catch (error) {
78
+ if (error instanceof Error) return undefined;
79
+ throw error;
80
+ }
81
+ }
82
+
83
+ function getSchemaTypeName(schema: SchemaLike): string | undefined {
84
+ if (!isRecord(schema)) return undefined;
85
+ const def = schema._def;
86
+ if (!isRecord(def)) return undefined;
87
+ const typeName = def.typeName ?? def.type;
88
+ return typeof typeName === "string" ? typeName : undefined;
89
+ }
@@ -0,0 +1,52 @@
1
+ import type { JsonValue } from "./contract-json";
2
+ import type { ProviderDefinition } from "./types";
3
+
4
+ export const PROVIDER_CONTRACT_SCHEMA_VERSION = "2026-06-23";
5
+
6
+ export interface ProviderContractSnapshot {
7
+ readonly schemaVersion: typeof PROVIDER_CONTRACT_SCHEMA_VERSION;
8
+ readonly provider: {
9
+ readonly id: string;
10
+ readonly version: string;
11
+ readonly runtime: ProviderDefinition["runtime"];
12
+ };
13
+ readonly allowedHosts?: readonly string[];
14
+ readonly stealth?: JsonValue;
15
+ readonly proxy?: JsonValue;
16
+ readonly stt?: JsonValue;
17
+ readonly browser?: JsonValue;
18
+ readonly auth?: JsonValue;
19
+ readonly reviewed?: JsonValue;
20
+ readonly access?: JsonValue;
21
+ readonly secrets?: JsonValue;
22
+ readonly credential?: JsonValue;
23
+ readonly context?: JsonValue;
24
+ readonly meta: JsonValue;
25
+ readonly healthMonitor?: JsonValue;
26
+ readonly healthJourneys?: readonly JsonValue[];
27
+ readonly operations: readonly ProviderContractOperation[];
28
+ }
29
+
30
+ export interface ProviderContractOperation {
31
+ readonly id: string;
32
+ readonly descriptionKey?: JsonValue;
33
+ readonly docs?: JsonValue;
34
+ readonly whenToUseKeys?: JsonValue;
35
+ readonly whenNotToUseKeys?: JsonValue;
36
+ readonly derivations?: JsonValue;
37
+ readonly inputExamples?: JsonValue;
38
+ readonly annotations?: JsonValue;
39
+ readonly contract?: JsonValue;
40
+ readonly tags?: JsonValue;
41
+ readonly relatedOperations?: JsonValue;
42
+ readonly toolRouter?: JsonValue;
43
+ readonly observability?: JsonValue;
44
+ readonly transport?: JsonValue;
45
+ readonly inputSchema: JsonValue;
46
+ readonly outputSchema: JsonValue;
47
+ readonly fixtures?: JsonValue;
48
+ readonly upstream?: JsonValue;
49
+ readonly hints?: JsonValue;
50
+ readonly healthCheck?: JsonValue;
51
+ readonly healthCheckUnsupported?: JsonValue;
52
+ }