@executor-js/plugin-openapi 1.5.5 → 1.5.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/{AddOpenApiSource-7M52SRUX.js → AddOpenApiSource-7O4LSD7C.js} +79 -241
- package/dist/AddOpenApiSource-7O4LSD7C.js.map +1 -0
- package/dist/{EditOpenApiSource-WTAMRJUK.js → EditOpenApiSource-GIN5RQPL.js} +3 -3
- package/dist/{OpenApiAccountsPanel-3VJJXNQF.js → OpenApiAccountsPanel-7XT6ZMD5.js} +30 -28
- package/dist/OpenApiAccountsPanel-7XT6ZMD5.js.map +1 -0
- package/dist/api/group.d.ts +38 -37
- package/dist/api/index.d.ts +39 -38
- package/dist/{chunk-YVRI7KRC.js → chunk-C3IJX4AN.js} +257 -219
- package/dist/chunk-C3IJX4AN.js.map +1 -0
- package/dist/{chunk-OSIFYIQP.js → chunk-C6PH4R7Q.js} +94 -5
- package/dist/chunk-C6PH4R7Q.js.map +1 -0
- package/dist/{chunk-BSLE6HCE.js → chunk-IB36ED7Y.js} +13 -27
- package/dist/chunk-IB36ED7Y.js.map +1 -0
- package/dist/chunk-RCBR3QMJ.js +73 -0
- package/dist/chunk-RCBR3QMJ.js.map +1 -0
- package/dist/{chunk-V7VCHYOY.js → chunk-WJQIWTZF.js} +97 -172
- package/dist/chunk-WJQIWTZF.js.map +1 -0
- package/dist/client.js +3 -3
- package/dist/core.js +133 -7
- package/dist/core.js.map +1 -1
- package/dist/index.js +7 -5
- package/dist/index.js.map +1 -1
- package/dist/react/AddOpenApiSource.d.ts +0 -1
- package/dist/react/OpenApiSourceDetailsFields.d.ts +3 -0
- package/dist/react/atoms.d.ts +40 -52
- package/dist/react/auth-method-config.d.ts +7 -4
- package/dist/react/client.d.ts +38 -37
- package/dist/sdk/config.d.ts +27 -27
- package/dist/sdk/derive-auth.d.ts +9 -0
- package/dist/sdk/extract.d.ts +11 -1
- package/dist/sdk/index.d.ts +6 -2
- package/dist/sdk/invoke.d.ts +11 -1
- package/dist/sdk/migrate-config.d.ts +4 -0
- package/dist/sdk/migrate-config.test.d.ts +1 -0
- package/dist/sdk/openapi-utils.d.ts +4 -8
- package/dist/sdk/output-schema-migration.d.ts +21 -0
- package/dist/sdk/output-schema-migration.test.d.ts +1 -0
- package/dist/sdk/plugin.d.ts +6 -4
- package/dist/sdk/preview.d.ts +77 -1
- package/dist/sdk/server-url-resolution.test.d.ts +1 -0
- package/dist/sdk/spec-blob-migration.d.ts +7 -0
- package/dist/sdk/spec-blob.test.d.ts +1 -0
- package/dist/sdk/store.d.ts +10 -1
- package/dist/sdk/tool-row-projection.test.d.ts +1 -0
- package/dist/sdk/types.d.ts +61 -36
- package/dist/testing/index.d.ts +1 -1
- package/dist/testing.js +4 -20
- package/dist/testing.js.map +1 -1
- package/package.json +3 -3
- package/dist/AddOpenApiSource-7M52SRUX.js.map +0 -1
- package/dist/OpenApiAccountsPanel-3VJJXNQF.js.map +0 -1
- package/dist/chunk-BSLE6HCE.js.map +0 -1
- package/dist/chunk-OSIFYIQP.js.map +0 -1
- package/dist/chunk-QSSRVK6M.js +0 -139
- package/dist/chunk-QSSRVK6M.js.map +0 -1
- package/dist/chunk-V7VCHYOY.js.map +0 -1
- package/dist/chunk-YVRI7KRC.js.map +0 -1
- /package/dist/{EditOpenApiSource-WTAMRJUK.js.map → EditOpenApiSource-GIN5RQPL.js.map} +0 -0
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/react/AddOpenApiSource.tsx","../src/react/OpenApiSourceDetailsFields.tsx","../src/react/GoogleProductPicker.tsx","../src/sdk/google-oauth-batches.ts"],"sourcesContent":["import { useCallback, useEffect, useMemo, useRef, useState } from \"react\";\nimport { useAtomSet, useAtomValue } from \"@effect/atom-react\";\nimport { Link } from \"@tanstack/react-router\";\nimport * as Effect from \"effect/Effect\";\nimport * as Exit from \"effect/Exit\";\nimport * as Option from \"effect/Option\";\nimport * as Predicate from \"effect/Predicate\";\nimport * as Schema from \"effect/Schema\";\nimport * as AsyncResult from \"effect/unstable/reactivity/AsyncResult\";\n\nimport {\n AuthTemplateSlug,\n IntegrationSlug,\n type OAuthAuthentication,\n} from \"@executor-js/sdk/shared\";\nimport { integrationsOptimisticAtom } from \"@executor-js/react/api/atoms\";\nimport { integrationWriteKeys } from \"@executor-js/react/api/reactivity-keys\";\nimport {\n slugifyNamespace,\n useIntegrationIdentity,\n} from \"@executor-js/react/plugins/integration-identity\";\nimport { Button } from \"@executor-js/react/components/button\";\nimport {\n AuthTemplateEditor,\n type AuthTemplateEditorValue,\n} from \"@executor-js/react/components/auth-template-editor\";\nimport { CardStack, CardStackContent } from \"@executor-js/react/components/card-stack\";\nimport { FieldLabel } from \"@executor-js/react/components/field\";\nimport { FloatActions } from \"@executor-js/react/components/float-actions\";\nimport { Textarea } from \"@executor-js/react/components/textarea\";\nimport { IOSSpinner, Spinner } from \"@executor-js/react/components/spinner\";\nimport { PlusIcon, XIcon } from \"lucide-react\";\n\nimport { authenticationFromEditorValue, editorValueFromAuthentication } from \"./auth-method-config\";\nimport { addOpenApiSpec, previewOpenApiSpec } from \"./atoms\";\nimport { OpenApiSourceDetailsFields } from \"./OpenApiSourceDetailsFields\";\nimport { GoogleProductPicker } from \"./GoogleProductPicker\";\nimport { openApiPresets } from \"../sdk/presets\";\nimport {\n GOOGLE_BUNDLE_PRESET_ID,\n googleOpenApiPresets,\n type GoogleOpenApiPreset,\n} from \"../sdk/google-presets\";\nimport type { SpecPreview, HeaderPreset, OAuth2Preset } from \"../sdk/preview\";\nimport {\n type APIKeyAuthentication,\n type Authentication,\n type ServerInfo,\n TOKEN_VARIABLE,\n variable,\n} from \"../sdk/types\";\nimport { expandServerUrlOptions } from \"../sdk/openapi-utils\";\n\nconst GOOGLE_BUNDLE_BASE_URL = \"https://www.googleapis.com/\";\nconst GOOGLE_BUNDLE_FAVICON = \"https://fonts.gstatic.com/s/i/productlogos/googleg/v6/192px.svg\";\n\n// The bundle picker opens with the featured Google APIs pre-checked.\nconst googleBundleDefaultPresetIds: ReadonlySet<string> = new Set(\n googleOpenApiPresets\n .filter((preset: GoogleOpenApiPreset) => preset.featured)\n .map((preset: GoogleOpenApiPreset) => preset.id),\n);\n\nconst googleBundleUrls = (\n selectedPresetIds: ReadonlySet<string>,\n customUrls: readonly string[],\n): readonly string[] => {\n const fromPresets = googleOpenApiPresets.flatMap((preset: GoogleOpenApiPreset) =>\n preset.url && selectedPresetIds.has(preset.id) ? [preset.url] : [],\n );\n // Preset URLs first (stable order), then any custom Discovery URLs, de-duped.\n return [...new Set([...fromPresets, ...customUrls])];\n};\n\nconst ErrorMessage = Schema.Struct({ message: Schema.String });\nconst decodeErrorMessage = Schema.decodeUnknownOption(ErrorMessage);\n\nconst errorMessageFromExit = (exit: Exit.Exit<unknown, unknown>, fallback: string): string =>\n Option.match(Option.flatMap(Exit.findErrorOption(exit), decodeErrorMessage), {\n onNone: () => fallback,\n onSome: ({ message }) => message,\n });\n\nconst isIntegrationAlreadyExistsExit = (exit: Exit.Exit<unknown, unknown>): boolean =>\n Option.match(Exit.findErrorOption(exit), {\n onNone: () => false,\n onSome: Predicate.isTagged(\"IntegrationAlreadyExistsError\"),\n });\n\nconst integrationExistsMessage = (slug: string): string =>\n `An integration named \"${slug}\" already exists. To add more authentication, update your existing integration.`;\n\n// ---------------------------------------------------------------------------\n// OpenAPI url helpers — specs sometimes ship relative OAuth endpoints; resolve\n// them against the chosen base URL so the stored auth template is absolute.\n// ---------------------------------------------------------------------------\n\nexport function resolveOAuthUrl(url: string, baseUrl: string): string {\n if (!url) return url;\n // oxlint-disable-next-line executor/no-try-catch-or-throw -- boundary: URL constructor normalizes provider metadata URLs\n try {\n new URL(url);\n return url;\n } catch {\n if (!baseUrl) return url;\n // oxlint-disable-next-line executor/no-try-catch-or-throw -- boundary: URL constructor resolves relative provider metadata URLs\n try {\n return new URL(url, baseUrl).toString();\n } catch {\n return url;\n }\n }\n}\n\nconst standardOidcIdentityScopes = [\"openid\", \"email\", \"profile\"] as const;\n\nconst identityScopesForPreset = (\n identityScopes: OAuth2Preset[\"identityScopes\"],\n): readonly string[] => {\n if (identityScopes === false) return [];\n return identityScopes === \"auto\" ? standardOidcIdentityScopes : identityScopes;\n};\n\nconst resolvedOAuthScopes = (\n apiScopes: Iterable<string>,\n identityScopes: OAuth2Preset[\"identityScopes\"],\n): string[] => {\n const merged = new Set(apiScopes);\n for (const scope of identityScopesForPreset(identityScopes)) merged.add(scope);\n return [...merged];\n};\n\nconst isGoogleDiscoveryUrl = (url: string): boolean => {\n const trimmed = url.trim();\n if (!URL.canParse(trimmed)) return false;\n const parsed = new URL(trimmed);\n const host = parsed.hostname.toLowerCase();\n if (!host.endsWith(\"googleapis.com\")) return false;\n return parsed.pathname.includes(\"/discovery/\") || parsed.pathname.includes(\"$discovery\");\n};\n\nconst normalizePresetUrl = (url: string): string => {\n const trimmed = url.trim();\n if (!URL.canParse(trimmed)) return trimmed.replace(/\\/$/, \"\");\n const parsed = new URL(trimmed);\n parsed.hash = \"\";\n parsed.searchParams.sort();\n return parsed.toString().replace(/\\/$/, \"\");\n};\n\nconst specInputForAdd = (input: string) => {\n const value = input.trim();\n const parsed = Effect.runSyncExit(\n Effect.try({\n try: () => new URL(value),\n catch: () => null,\n }),\n );\n return Exit.isSuccess(parsed)\n ? isGoogleDiscoveryUrl(value)\n ? { kind: \"googleDiscovery\" as const, url: value }\n : { kind: \"url\" as const, url: value }\n : { kind: \"blob\" as const, value };\n};\n\n// ---------------------------------------------------------------------------\n// Auth-template builders — turn a preview preset into the integration's stored\n// `Authentication` template (v2). The header preset becomes an `apiKey` template\n// whose secret header value renders the resolved credential via `variable(token)`;\n// the oauth2 preset becomes an `oauth` template carrying the provider endpoints.\n//\n// Post-redesign the add flow no longer asks the user to pick ONE method: every\n// spec-detected method is registered so the integration's detail hub can list\n// them and Add-account can choose among them (P6: add without auth, connect\n// later).\n// ---------------------------------------------------------------------------\n\nconst headerPrefix = (preset: HeaderPreset, headerName: string): string | undefined => {\n const label = preset.label.toLowerCase();\n if (headerName.toLowerCase() === \"authorization\") {\n if (label.includes(\"bearer\")) return \"Bearer \";\n if (label.includes(\"basic\")) return \"Basic \";\n }\n return undefined;\n};\n\nconst apiKeyTemplateFromHeaderPreset = (\n preset: HeaderPreset,\n slug: AuthTemplateSlug,\n): APIKeyAuthentication => {\n const headers: Record<string, (string | ReturnType<typeof variable>)[]> = {};\n for (const headerName of preset.secretHeaders) {\n const prefix = headerPrefix(preset, headerName);\n headers[headerName] = prefix ? [prefix, variable(TOKEN_VARIABLE)] : [variable(TOKEN_VARIABLE)];\n }\n return { slug, type: \"apiKey\", headers };\n};\n\nconst oauthTemplateFromPreset = (\n preset: OAuth2Preset,\n baseUrl: string,\n slug: AuthTemplateSlug,\n scopes: readonly string[],\n): OAuthAuthentication => ({\n slug,\n type: \"oauth\",\n authorizationUrl: resolveOAuthUrl(\n Option.getOrElse(preset.authorizationUrl, () => \"\"),\n baseUrl,\n ),\n tokenUrl: resolveOAuthUrl(preset.tokenUrl, baseUrl),\n scopes: [...scopes],\n});\n\nconst expandServerOptions = (server: ServerInfo) =>\n expandServerUrlOptions(server).map((value) => ({ value, label: value }));\n\nconst firstBaseUrlForPreview = (preview: SpecPreview): string => {\n const firstServer = preview.servers[0];\n return firstServer ? (expandServerUrlOptions(firstServer)[0] ?? \"\") : \"\";\n};\n\n// ---------------------------------------------------------------------------\n// All spec-detected auth methods → the union of stored `Authentication`\n// templates. Header presets become apiKey templates; each oauth2 preset becomes\n// an oauth template (with its declared API scopes plus, for auth-code flows,\n// the standard identity scopes). Slugs stay deterministic per method so the\n// stored template is stable across previews of the same spec. Adding an\n// integration whose slug already exists is blocked (see the existing-slug\n// guard below); to add more auth, update the existing integration instead.\n// ---------------------------------------------------------------------------\n\nconst detectedAuthenticationTemplates = (\n headerPresets: readonly HeaderPreset[],\n oauth2Presets: readonly OAuth2Preset[],\n baseUrl: string,\n): readonly Authentication[] => {\n const templates: Authentication[] = [];\n headerPresets.forEach((preset, index) => {\n templates.push(\n apiKeyTemplateFromHeaderPreset(preset, AuthTemplateSlug.make(`apikey-${index}`)),\n );\n });\n for (const preset of oauth2Presets) {\n const scopes = resolvedOAuthScopes(Object.keys(preset.scopes), preset.identityScopes);\n templates.push(\n oauthTemplateFromPreset(\n preset,\n baseUrl,\n AuthTemplateSlug.make(`oauth-${preset.securitySchemeName}`),\n scopes,\n ),\n );\n }\n return templates;\n};\n\n// ---------------------------------------------------------------------------\n// Component — single progressive form. Post-redesign: preview → addSpec\n// (register the integration catalog entry with ALL detected auth methods) →\n// route to the integration's detail hub, where the user adds accounts. The add\n// flow no longer creates a connection.\n// ---------------------------------------------------------------------------\n\nexport default function AddOpenApiSource(props: {\n onComplete: (slug?: string) => void;\n onCancel: () => void;\n initialUrl?: string;\n initialPreset?: string;\n initialNamespace?: string;\n}) {\n const isGoogleBundlePreset = props.initialPreset === GOOGLE_BUNDLE_PRESET_ID;\n\n // Spec input. For the Google BUNDLE preset the input is a product picker (a set\n // of selected Discovery URLs), not a single spec URL/blob — the merge happens\n // server-side via `{ kind: \"googleDiscoveryBundle\", urls }`, so the textarea\n // preview path is bypassed entirely.\n const [specUrl, setSpecUrl] = useState(props.initialUrl ?? \"\");\n const [selectedPresetIds, setSelectedPresetIds] = useState<ReadonlySet<string>>(\n googleBundleDefaultPresetIds,\n );\n const [customDiscoveryUrls, setCustomDiscoveryUrls] = useState<readonly string[]>([]);\n const [analyzing, setAnalyzing] = useState(false);\n const [analyzeError, setAnalyzeError] = useState<string | null>(null);\n\n // After analysis\n const [preview, setPreview] = useState<SpecPreview | null>(null);\n const [baseUrl, setBaseUrl] = useState(isGoogleBundlePreset ? GOOGLE_BUNDLE_BASE_URL : \"\");\n const identityFallbackName = isGoogleBundlePreset\n ? \"Google\"\n : preview\n ? Option.getOrElse(preview.title, () => \"\")\n : \"\";\n const identity = useIntegrationIdentity({\n fallbackName: identityFallbackName,\n fallbackNamespace: props.initialNamespace ?? (isGoogleBundlePreset ? \"google\" : undefined),\n });\n\n const bundleDiscoveryUrls = useMemo(\n () => googleBundleUrls(selectedPresetIds, customDiscoveryUrls),\n [selectedPresetIds, customDiscoveryUrls],\n );\n\n const toggleBundlePreset = useCallback((presetId: string, checked: boolean) => {\n setSelectedPresetIds((current: ReadonlySet<string>) => {\n const next = new Set(current);\n if (checked) next.add(presetId);\n else next.delete(presetId);\n return next;\n });\n }, []);\n\n const addCustomDiscoveryUrl = useCallback((url: string) => {\n setCustomDiscoveryUrls((current: readonly string[]) =>\n current.includes(url) ? current : [...current, url],\n );\n }, []);\n\n const removeCustomDiscoveryUrl = useCallback((url: string) => {\n setCustomDiscoveryUrls((current: readonly string[]) =>\n current.filter((entry: string) => entry !== url),\n );\n }, []);\n\n // Submit\n const [adding, setAdding] = useState(false);\n const [addError, setAddError] = useState<string | null>(null);\n\n const doPreview = useAtomSet(previewOpenApiSpec, { mode: \"promiseExit\" });\n const doAdd = useAtomSet(addOpenApiSpec, { mode: \"promiseExit\" });\n\n // Keep the latest handleAnalyze in a ref so the debounced effect doesn't need\n // it as a dependency (it closes over fresh state).\n const handleAnalyzeRef = useRef<() => void>(() => {});\n\n useEffect(() => {\n // The bundle preset never analyzes a single spec — its input is the picker.\n if (isGoogleBundlePreset) return;\n const trimmed = specUrl.trim();\n if (!trimmed) return;\n if (preview) return;\n const handle = setTimeout(() => {\n handleAnalyzeRef.current();\n }, 400);\n return () => clearTimeout(handle);\n }, [specUrl, preview, isGoogleBundlePreset]);\n\n // ---- Derived state ----\n\n const servers: readonly ServerInfo[] = preview?.servers ?? [];\n const baseUrlOptions = Array.from(\n new Map(servers.flatMap(expandServerOptions).map((option) => [option.value, option])).values(),\n );\n const previewPresetIcon =\n openApiPresets.find(\n (preset) => preset.url && normalizePresetUrl(preset.url) === normalizePresetUrl(specUrl),\n )?.icon ?? null;\n\n const resolvedBaseUrl = baseUrl.trim();\n const resolvedSourceId =\n slugifyNamespace(identity.namespace) ||\n (preview ? Option.getOrElse(preview.title, () => \"openapi\") : \"openapi\");\n const resolvedDisplayName =\n identity.name.trim() ||\n (preview ? Option.getOrElse(preview.title, () => resolvedSourceId) : resolvedSourceId);\n\n // Register EVERY spec-detected auth method, not just a single selected one.\n // Keyed off `preview` (stable per analysis) so the memo doesn't re-run on the\n // freshly-allocated `?? []` fallback arrays.\n const authenticationTemplate: readonly Authentication[] = useMemo(\n () =>\n detectedAuthenticationTemplates(\n preview?.headerPresets ?? [],\n preview?.oauth2Presets ?? [],\n resolvedBaseUrl,\n ),\n [preview, resolvedBaseUrl],\n );\n\n const detectedMethodLabels: readonly string[] = useMemo(\n () => [\n ...(preview?.headerPresets ?? []).map((preset) => preset.label),\n ...(preview?.oauth2Presets ?? []).map((preset) => preset.label),\n ],\n [preview],\n );\n\n // Editable auth methods, seeded from the spec-detected templates. The add flow\n // registers EVERY method (P6) — so this is a LIST, preserving multi-method\n // specs (e.g. apiKey + OAuth). Each row carries its editor value plus the\n // detected template's original `seedSlug`, so an unedited detected method\n // submits with its EXACT original slug (preserving behavior); added methods\n // (no seed) get a deterministic fresh slug. The user can edit, add, or remove\n // a method before submitting; on submit the list converts back to\n // `Authentication[]`. Re-seeded whenever a fresh detection arrives (keyed on\n // the detected templates, which are stable per analysis + base URL).\n type AuthMethodRow = {\n readonly value: AuthTemplateEditorValue;\n readonly seedSlug?: string;\n };\n const [authMethods, setAuthMethods] = useState<readonly AuthMethodRow[]>([]);\n const seededFromRef = useRef<readonly Authentication[] | null>(null);\n useEffect(() => {\n if (seededFromRef.current === authenticationTemplate) return;\n seededFromRef.current = authenticationTemplate;\n setAuthMethods(\n authenticationTemplate.map((template: Authentication) => ({\n value: editorValueFromAuthentication(template),\n seedSlug: String(template.slug),\n })),\n );\n }, [authenticationTemplate]);\n\n const setAuthMethodAt = useCallback((index: number, next: AuthTemplateEditorValue) => {\n setAuthMethods((current: readonly AuthMethodRow[]) =>\n current.map((row: AuthMethodRow, i: number) => (i === index ? { ...row, value: next } : row)),\n );\n }, []);\n\n const removeAuthMethodAt = useCallback((index: number) => {\n setAuthMethods((current: readonly AuthMethodRow[]) =>\n current.filter((_row: AuthMethodRow, i: number) => i !== index),\n );\n }, []);\n\n const addAuthMethod = useCallback(() => {\n setAuthMethods((current: readonly AuthMethodRow[]) => [\n ...current,\n {\n value: {\n kind: \"apikey\",\n placements: [{ carrier: \"header\", name: \"Authorization\", prefix: \"\" }],\n },\n },\n ]);\n }, []);\n\n // The methods to register, mapped back to stored `Authentication[]`. Drops\n // `none` rows (nothing to register). An unedited detected method keeps its\n // original `seedSlug`; an added method gets a deterministic fresh slug.\n const editedAuthenticationTemplate: readonly Authentication[] = useMemo(() => {\n const templates: Authentication[] = [];\n authMethods.forEach((row: AuthMethodRow, index: number) => {\n const slug =\n row.seedSlug ?? (row.value.kind === \"oauth\" ? `oauth-${index}` : `apikey-${index}`);\n const template = authenticationFromEditorValue(row.value, slug);\n if (template !== null) templates.push(template);\n });\n return templates;\n }, [authMethods]);\n\n // Pre-empt the API's `IntegrationAlreadyExistsError`: adding an integration\n // whose slug already exists clobbers the existing one's connections/policies,\n // so the API blocks it. Surface that here from the tenant-scoped catalog list.\n const integrationsResult = useAtomValue(integrationsOptimisticAtom);\n const slugAlreadyExists = useMemo(\n () =>\n AsyncResult.isSuccess(integrationsResult) &&\n integrationsResult.value.some((integration) => integration.slug === resolvedSourceId),\n [integrationsResult, resolvedSourceId],\n );\n\n // The bundle path is ready once at least one Google API is selected (no\n // network preview gates it); the single/custom-spec path still requires a\n // successful preview. Both require a base URL and a free slug.\n const hasPreviewOrBundle = isGoogleBundlePreset\n ? bundleDiscoveryUrls.length > 0\n : preview !== null;\n const canAdd = hasPreviewOrBundle && resolvedBaseUrl.length > 0 && !slugAlreadyExists;\n\n // ---- Handlers ----\n\n const handleAnalyze = async () => {\n setAnalyzing(true);\n setAnalyzeError(null);\n setAddError(null);\n const exit = await doPreview({ payload: { spec: specUrl } });\n if (Exit.isFailure(exit)) {\n setAnalyzeError(errorMessageFromExit(exit, \"Failed to parse spec\"));\n setAnalyzing(false);\n return;\n }\n const result = exit.value;\n setPreview(result);\n setBaseUrl(firstBaseUrlForPreview(result));\n setAnalyzing(false);\n };\n\n handleAnalyzeRef.current = handleAnalyze;\n\n // Persist the integration and return its slug. Registers the catalog entry\n // with every detected auth method. Adding a slug that already exists is\n // rejected by the API (`IntegrationAlreadyExistsError`) — surfaced inline.\n const ensureIntegration = useCallback(async (): Promise<IntegrationSlug | null> => {\n // The Google BUNDLE preset emits the multi-service bundle input; the server\n // merges the selected Discovery documents into one `google` spec and stores\n // the unioned `googleOAuth2` auth template (so no client template is sent).\n // Every other preset/custom input keeps the single-spec url/blob/discovery\n // branch unchanged.\n const specForAdd = isGoogleBundlePreset\n ? ({ kind: \"googleDiscoveryBundle\" as const, urls: [...bundleDiscoveryUrls] } satisfies {\n readonly kind: \"googleDiscoveryBundle\";\n readonly urls: readonly string[];\n })\n : specInputForAdd(specUrl);\n const exit = await doAdd({\n payload: {\n spec: specForAdd,\n slug: resolvedSourceId,\n description: resolvedDisplayName,\n baseUrl: resolvedBaseUrl,\n ...(!isGoogleBundlePreset && editedAuthenticationTemplate.length > 0\n ? {\n authenticationTemplate: editedAuthenticationTemplate.map((entry) => ({\n ...entry,\n slug: String(entry.slug),\n })),\n }\n : {}),\n },\n reactivityKeys: integrationWriteKeys,\n });\n if (Exit.isFailure(exit)) {\n setAddError(\n isIntegrationAlreadyExistsExit(exit)\n ? integrationExistsMessage(resolvedSourceId)\n : errorMessageFromExit(exit, \"Failed to add integration\"),\n );\n return null;\n }\n return exit.value.slug;\n }, [\n isGoogleBundlePreset,\n bundleDiscoveryUrls,\n specUrl,\n doAdd,\n resolvedSourceId,\n resolvedDisplayName,\n resolvedBaseUrl,\n editedAuthenticationTemplate,\n ]);\n\n const handleAdd = async () => {\n setAdding(true);\n setAddError(null);\n\n const integration = await ensureIntegration();\n if (!integration) {\n setAdding(false);\n return;\n }\n\n props.onComplete(String(integration));\n };\n\n // ---- Render ----\n\n return (\n <div className=\"flex flex-1 flex-col gap-6\">\n <div>\n <h1 className=\"text-xl font-semibold text-foreground\">\n {isGoogleBundlePreset ? \"Add Google\" : \"Add OpenAPI Integration\"}\n </h1>\n {isGoogleBundlePreset ? (\n <p className=\"mt-1 text-[13px] text-muted-foreground\">\n Bundle Google APIs into one integration from their Discovery documents and register\n their methods as tools under a single shared OAuth consent.\n </p>\n ) : null}\n </div>\n\n {isGoogleBundlePreset ? (\n <GoogleProductPicker\n selectedPresetIds={selectedPresetIds}\n onToggle={toggleBundlePreset}\n customUrls={customDiscoveryUrls}\n onAddCustomUrl={addCustomDiscoveryUrl}\n onRemoveCustomUrl={removeCustomDiscoveryUrl}\n />\n ) : !preview ? (\n <CardStack>\n <CardStackContent className=\"border-t-0\">\n <div className=\"space-y-2 p-3\">\n <FieldLabel>OpenAPI Spec</FieldLabel>\n <div className=\"relative\">\n <Textarea\n value={specUrl}\n onChange={(e) => setSpecUrl((e.target as HTMLTextAreaElement).value)}\n placeholder=\"https://api.example.com/openapi.json\"\n rows={3}\n maxRows={10}\n className=\"font-mono text-sm\"\n />\n {analyzing && (\n <div className=\"pointer-events-none absolute right-2 top-2\">\n <IOSSpinner className=\"size-4\" />\n </div>\n )}\n </div>\n <p className=\"text-[11px] text-muted-foreground\">\n Paste a URL or raw JSON/YAML content.\n </p>\n </div>\n </CardStackContent>\n </CardStack>\n ) : null}\n\n {isGoogleBundlePreset ? (\n <OpenApiSourceDetailsFields\n title=\"Google\"\n description={`${bundleDiscoveryUrls.length} Google API${\n bundleDiscoveryUrls.length !== 1 ? \"s\" : \"\"\n } · one shared OAuth consent`}\n identity={identity}\n baseUrl={resolvedBaseUrl}\n onBaseUrlChange={setBaseUrl}\n faviconIcon={GOOGLE_BUNDLE_FAVICON}\n faviconUrl={resolvedBaseUrl}\n baseUrlMissingMessage=\"A base URL is required to make requests.\"\n />\n ) : preview ? (\n <OpenApiSourceDetailsFields\n title={Option.getOrElse(preview.title, () => \"API\")}\n description={`${Option.getOrElse(preview.version, () => \"\")}${\n Option.isSome(preview.version) ? \" · \" : \"\"\n }${preview.operationCount} operation${preview.operationCount !== 1 ? \"s\" : \"\"}${\n preview.tags.length > 0\n ? ` · ${preview.tags.length} tag${preview.tags.length !== 1 ? \"s\" : \"\"}`\n : \"\"\n }`}\n identity={identity}\n baseUrl={resolvedBaseUrl}\n onBaseUrlChange={setBaseUrl}\n baseUrlOptions={baseUrlOptions}\n specUrl={specUrl}\n onSpecUrlChange={(value) => {\n setSpecUrl(value);\n setPreview(null);\n setBaseUrl(\"\");\n }}\n faviconIcon={previewPresetIcon}\n faviconUrl={resolvedBaseUrl}\n baseUrlMissingMessage=\"A base URL is required to make requests.\"\n />\n ) : null}\n\n {analyzeError && (\n <div className=\"rounded-lg border border-destructive/30 bg-destructive/5 px-3 py-2\">\n <p className=\"text-[12px] text-destructive\">{analyzeError}</p>\n </div>\n )}\n\n {preview && !isGoogleBundlePreset && (\n <section className=\"space-y-3\">\n <div className=\"flex items-center justify-between gap-3\">\n <FieldLabel>How does this API authenticate?</FieldLabel>\n <Button type=\"button\" variant=\"outline\" size=\"sm\" onClick={addAuthMethod}>\n <PlusIcon />\n Add method\n </Button>\n </div>\n {authMethods.length === 0 ? (\n <p className=\"text-[11px] text-muted-foreground\">\n No authentication detected. Add a method, or add the integration without auth and\n connect an account from the integration page later.\n </p>\n ) : (\n <div className=\"flex flex-col gap-3\">\n {authMethods.map((row: AuthMethodRow, index: number) => (\n <div\n key={index}\n className=\"space-y-2 rounded-lg border border-border/60 bg-muted/20 p-3\"\n >\n <div className=\"flex items-center justify-between\">\n <span className=\"text-xs font-medium text-muted-foreground\">\n Method {index + 1}\n {detectedMethodLabels[index] ? ` · ${detectedMethodLabels[index]}` : \"\"}\n </span>\n <Button\n type=\"button\"\n variant=\"ghost\"\n size=\"icon-sm\"\n aria-label=\"Remove method\"\n className=\"text-muted-foreground hover:text-foreground\"\n onClick={() => removeAuthMethodAt(index)}\n >\n <XIcon />\n </Button>\n </div>\n <AuthTemplateEditor\n value={row.value}\n onChange={(next: AuthTemplateEditorValue) => setAuthMethodAt(index, next)}\n />\n </div>\n ))}\n </div>\n )}\n <p className=\"text-[11px] text-muted-foreground\">\n Every method here is registered with the integration. Connect an account from the\n integration page after adding.\n </p>\n </section>\n )}\n\n {hasPreviewOrBundle && slugAlreadyExists && !adding && (\n <div className=\"rounded-lg border border-destructive/30 bg-destructive/5 px-3 py-2\">\n <p className=\"text-[12px] text-destructive\">\n An integration named "{resolvedSourceId}" already exists. To add more\n authentication, update your existing integration.{\" \"}\n <Link\n to=\"/integrations/$namespace\"\n params={{ namespace: resolvedSourceId }}\n className=\"font-medium underline underline-offset-2\"\n >\n Open it\n </Link>\n </p>\n </div>\n )}\n\n {addError && (\n <div className=\"rounded-lg border border-destructive/30 bg-destructive/5 px-3 py-2\">\n <p className=\"text-[12px] text-destructive\">{addError}</p>\n </div>\n )}\n\n <FloatActions>\n <Button variant=\"ghost\" onClick={() => props.onCancel()} disabled={adding}>\n Cancel\n </Button>\n {(hasPreviewOrBundle || isGoogleBundlePreset) && (\n <Button onClick={() => void handleAdd()} disabled={!canAdd || adding}>\n {adding && <Spinner className=\"size-3.5\" />}\n {adding ? \"Adding…\" : isGoogleBundlePreset ? \"Connect Google\" : \"Add integration\"}\n </Button>\n )}\n </FloatActions>\n </div>\n );\n}\n","import {\n CardStack,\n CardStackContent,\n CardStackEntry,\n CardStackEntryContent,\n CardStackEntryDescription,\n CardStackEntryField,\n CardStackEntryTitle,\n} from \"@executor-js/react/components/card-stack\";\nimport {\n FreeformCombobox,\n type FreeformComboboxOption,\n} from \"@executor-js/react/components/combobox\";\nimport { Input } from \"@executor-js/react/components/input\";\nimport { IntegrationFavicon } from \"@executor-js/react/components/integration-favicon\";\nimport {\n IntegrationIdentityFieldRows,\n type IntegrationIdentity,\n} from \"@executor-js/react/plugins/integration-identity\";\n\nexport function OpenApiSourceDetailsFields(props: {\n readonly title: string;\n readonly description?: string;\n readonly identity: IntegrationIdentity;\n readonly baseUrl: string;\n readonly onBaseUrlChange: (value: string) => void;\n readonly baseUrlOptions?: readonly FreeformComboboxOption[];\n readonly specUrl?: string;\n readonly onSpecUrlChange?: (value: string) => void;\n readonly faviconIcon?: string | null;\n readonly faviconUrl?: string;\n readonly namespaceReadOnly?: boolean;\n readonly specUrlDisabled?: boolean;\n readonly saveState?: \"idle\" | \"saving\" | \"saved\";\n readonly baseUrlMissingMessage?: string;\n readonly footer?: string;\n}) {\n const baseUrlOptions = props.baseUrlOptions ?? [];\n\n return (\n <CardStack>\n <CardStackContent className=\"border-t-0\">\n <CardStackEntry>\n {(props.faviconIcon || props.faviconUrl) && (\n <IntegrationFavicon icon={props.faviconIcon} url={props.faviconUrl} size={16} />\n )}\n <CardStackEntryContent>\n <CardStackEntryTitle>{props.title}</CardStackEntryTitle>\n {props.description && (\n <CardStackEntryDescription>{props.description}</CardStackEntryDescription>\n )}\n </CardStackEntryContent>\n {props.saveState && props.saveState !== \"idle\" && (\n <span className=\"text-xs text-muted-foreground\">\n {props.saveState === \"saving\" ? \"Saving…\" : \"Saved\"}\n </span>\n )}\n </CardStackEntry>\n <IntegrationIdentityFieldRows\n identity={props.identity}\n namespaceReadOnly={props.namespaceReadOnly}\n />\n <div className=\"grid grid-cols-1 md:grid-cols-2\">\n <CardStackEntryField label=\"Base URL\">\n {baseUrlOptions.length > 0 ? (\n <FreeformCombobox\n value={props.baseUrl}\n onValueChange={props.onBaseUrlChange}\n options={baseUrlOptions}\n placeholder=\"https://api.example.com\"\n className=\"w-full\"\n inputClassName=\"font-mono text-sm\"\n />\n ) : (\n <Input\n value={props.baseUrl}\n onChange={(e) => props.onBaseUrlChange((e.target as HTMLInputElement).value)}\n placeholder=\"https://api.example.com\"\n className=\"font-mono text-sm\"\n />\n )}\n\n {props.baseUrlMissingMessage && !props.baseUrl && (\n <p className=\"text-[11px] text-amber-600 dark:text-amber-400\">\n {props.baseUrlMissingMessage}\n </p>\n )}\n </CardStackEntryField>\n {props.specUrl !== undefined && props.onSpecUrlChange && (\n <CardStackEntryField label=\"Spec URL\">\n <Input\n value={props.specUrl}\n onChange={(e) => props.onSpecUrlChange?.((e.target as HTMLInputElement).value)}\n placeholder=\"https://api.example.com/openapi.json\"\n className=\"font-mono text-sm\"\n disabled={props.specUrlDisabled}\n />\n </CardStackEntryField>\n )}\n </div>\n {props.footer && (\n <CardStackEntry>\n <CardStackEntryContent>\n <CardStackEntryTitle>{props.footer}</CardStackEntryTitle>\n </CardStackEntryContent>\n </CardStackEntry>\n )}\n </CardStackContent>\n </CardStack>\n );\n}\n","import { useMemo, useState } from \"react\";\nimport { ChevronDownIcon, PlusIcon, TriangleAlert, XIcon } from \"lucide-react\";\n\nimport { cn } from \"@executor-js/react/lib/utils\";\nimport { Badge } from \"@executor-js/react/components/badge\";\nimport { Button } from \"@executor-js/react/components/button\";\nimport { Checkbox } from \"@executor-js/react/components/checkbox\";\nimport {\n Collapsible,\n CollapsibleContent,\n CollapsibleTrigger,\n} from \"@executor-js/react/components/collapsible\";\nimport { FieldLabel } from \"@executor-js/react/components/field\";\nimport { Input } from \"@executor-js/react/components/input\";\nimport { IntegrationFavicon } from \"@executor-js/react/components/integration-favicon\";\n\nimport {\n googleOAuthConsentScopesForPreset,\n googleOpenApiPresets,\n type GoogleOpenApiOAuthAudience,\n type GoogleOpenApiPreset,\n} from \"../sdk/google-presets\";\nimport { googleOAuthConsentBatches } from \"../sdk/google-oauth-batches\";\nimport { isGoogleDiscoveryUrl } from \"../sdk/google-discovery\";\n\n// ---------------------------------------------------------------------------\n// GoogleProductPicker — the \"customize your Google connection\" surface.\n//\n// A checkable card grid over `googleOpenApiPresets`, grouped/annotated by\n// `oauthAudience`. The user picks which Google APIs to bundle into the single\n// `google` integration; the parent turns the selected discovery URLs into a\n// `{ kind: \"googleDiscoveryBundle\", urls }` add. A \"View scopes\" panel previews\n// the unioned OAuth consent (via `googleOAuthConsentBatches`) BEFORE connecting,\n// and a custom-URL escape hatch lets advanced users paste any Google Discovery\n// document the preset list doesn't cover.\n// ---------------------------------------------------------------------------\n\n// Audience groups, ordered from least- to most-privileged. The warning tiers\n// (`workspace-admin`, `unsupported-user`) carry a caution chip so the user sees\n// the consent risk before selecting.\nconst AUDIENCE_ORDER: readonly GoogleOpenApiOAuthAudience[] = [\n \"standard-user\",\n \"advanced-user\",\n \"workspace-admin\",\n \"unsupported-user\",\n];\n\nconst AUDIENCE_LABEL: Readonly<Record<GoogleOpenApiOAuthAudience, string>> = {\n \"standard-user\": \"Core Google services\",\n \"advanced-user\": \"Advanced services\",\n \"workspace-admin\": \"Workspace admin\",\n \"unsupported-user\": \"Limited user consent\",\n};\n\nconst AUDIENCE_DESCRIPTION: Readonly<Record<GoogleOpenApiOAuthAudience, string>> = {\n \"standard-user\": \"Connect with a normal Google account — one consent screen.\",\n \"advanced-user\": \"Broader scopes that may need an unverified-app warning to be accepted.\",\n \"workspace-admin\": \"Requires a Google Workspace admin account; not available on personal Gmail.\",\n \"unsupported-user\": \"Google does not grant these scopes through standard user OAuth consent.\",\n};\n\nconst audienceNeedsWarning = (audience: GoogleOpenApiOAuthAudience): boolean =>\n audience === \"workspace-admin\" || audience === \"unsupported-user\";\n\ntype GoogleProductPickerProps = {\n readonly selectedPresetIds: ReadonlySet<string>;\n readonly onToggle: (presetId: string, checked: boolean) => void;\n readonly customUrls: readonly string[];\n readonly onAddCustomUrl: (url: string) => void;\n readonly onRemoveCustomUrl: (url: string) => void;\n};\n\nconst AudienceWarningChip = ({ audience }: { audience: GoogleOpenApiOAuthAudience }) =>\n audience === \"workspace-admin\" ? (\n <Badge\n variant=\"outline\"\n className=\"shrink-0 border-amber-500/40 text-amber-700 dark:text-amber-400\"\n >\n <TriangleAlert className=\"size-3\" />\n Admin only\n </Badge>\n ) : audience === \"unsupported-user\" ? (\n <Badge variant=\"outline\" className=\"shrink-0 border-destructive/40 text-destructive\">\n <TriangleAlert className=\"size-3\" />\n Limited consent\n </Badge>\n ) : null;\n\n// A Google API row — borderless and single-line, leaning on hover/selected\n// fills instead of per-item card chrome (lightest separation that still reads).\n// Name + truncated summary share one baseline for a dense, scannable two-column\n// list; the audience warning chip trails on the right.\nconst ProductRow = ({\n preset,\n checked,\n onToggle,\n}: {\n readonly preset: GoogleOpenApiPreset;\n readonly checked: boolean;\n readonly onToggle: (checked: boolean) => void;\n}) => (\n <FieldLabel\n className={cn(\n // `w-full` overrides FieldLabel's base `w-fit` (which would size the row to\n // its content and overflow the column); `min-w-0` then lets the cell shrink\n // to its track so the name/summary truncates instead of spilling over.\n \"flex w-full min-w-0 cursor-pointer items-center gap-2.5 rounded-md px-2 py-1.5 transition-colors\",\n checked ? \"bg-primary/5\" : \"hover:bg-muted/40\",\n )}\n >\n <Checkbox checked={checked} onCheckedChange={(next) => onToggle(next === true)} />\n <div className=\"shrink-0\">\n <IntegrationFavicon icon={preset.icon} url={preset.url} size={16} />\n </div>\n {/* One truncating line — the name + summary clip to the cell with an\n ellipsis instead of overflowing into the neighbouring column. */}\n <div className=\"min-w-0 flex-1 truncate text-sm\">\n <span className=\"font-medium text-foreground\">{preset.name}</span>{\" \"}\n <span className=\"text-[11px] text-muted-foreground\">{preset.summary}</span>\n </div>\n <AudienceWarningChip audience={preset.oauthAudience} />\n </FieldLabel>\n);\n\nconst CustomUrlEscapeHatch = ({\n customUrls,\n onAddCustomUrl,\n onRemoveCustomUrl,\n}: {\n readonly customUrls: readonly string[];\n readonly onAddCustomUrl: (url: string) => void;\n readonly onRemoveCustomUrl: (url: string) => void;\n}) => {\n const [draft, setDraft] = useState(\"\");\n const trimmed = draft.trim();\n const isValid = isGoogleDiscoveryUrl(trimmed);\n const isDuplicate = customUrls.includes(trimmed);\n\n const commit = () => {\n if (!isValid || isDuplicate) return;\n onAddCustomUrl(trimmed);\n setDraft(\"\");\n };\n\n return (\n <div className=\"space-y-2\">\n <FieldLabel className=\"text-[11px] font-medium text-muted-foreground\">\n Add a custom Google Discovery URL\n </FieldLabel>\n <div className=\"flex items-center gap-2\">\n <Input\n value={draft}\n onChange={(event: React.ChangeEvent<HTMLInputElement>) => setDraft(event.target.value)}\n onKeyDown={(event: React.KeyboardEvent<HTMLInputElement>) => {\n if (event.key === \"Enter\") {\n event.preventDefault();\n commit();\n }\n }}\n placeholder=\"https://www.googleapis.com/discovery/v1/apis/<service>/<version>/rest\"\n className=\"font-mono text-[11px]\"\n />\n <Button\n type=\"button\"\n variant=\"outline\"\n size=\"sm\"\n disabled={!isValid || isDuplicate}\n onClick={commit}\n >\n <PlusIcon className=\"size-3.5\" />\n Add\n </Button>\n </div>\n {trimmed.length > 0 && !isValid ? (\n <p className=\"text-[11px] text-destructive\">\n Enter a Google Discovery document URL (a *.googleapis.com discovery/$discovery endpoint).\n </p>\n ) : null}\n {customUrls.length > 0 ? (\n <ul className=\"space-y-1\">\n {customUrls.map((url: string) => (\n <li\n key={url}\n className=\"flex items-center justify-between gap-2 rounded-md border border-border bg-muted/20 px-2.5 py-1.5\"\n >\n <span className=\"truncate font-mono text-[11px] text-foreground\">{url}</span>\n <Button\n type=\"button\"\n variant=\"ghost\"\n size=\"icon\"\n className=\"size-6 shrink-0\"\n onClick={() => onRemoveCustomUrl(url)}\n aria-label={`Remove ${url}`}\n >\n <XIcon className=\"size-3.5\" />\n </Button>\n </li>\n ))}\n </ul>\n ) : null}\n </div>\n );\n};\n\nexport function GoogleProductPicker({\n selectedPresetIds,\n onToggle,\n customUrls,\n onAddCustomUrl,\n onRemoveCustomUrl,\n}: GoogleProductPickerProps) {\n const [scopesOpen, setScopesOpen] = useState(false);\n\n const groups = useMemo(\n () =>\n AUDIENCE_ORDER.flatMap((audience: GoogleOpenApiOAuthAudience) => {\n const presets = googleOpenApiPresets.filter(\n (preset: GoogleOpenApiPreset) => preset.oauthAudience === audience,\n );\n return presets.length > 0 ? [{ audience, presets }] : [];\n }),\n [],\n );\n\n // The \"View scopes\" preview unions the selected presets' representative\n // consent scopes through `googleOAuthConsentBatches`, mirroring the unioned\n // scopes the bundle converter ultimately stores on the integration.\n const consentBatches = useMemo(\n () =>\n googleOAuthConsentBatches(\n googleOpenApiPresets\n .filter((preset: GoogleOpenApiPreset) => selectedPresetIds.has(preset.id))\n .map((preset: GoogleOpenApiPreset) => ({\n id: preset.id,\n name: preset.name,\n oauthAudience: preset.oauthAudience,\n scopes: googleOAuthConsentScopesForPreset(preset.id),\n })),\n ),\n [selectedPresetIds],\n );\n\n const selectedCount = selectedPresetIds.size + customUrls.length;\n\n return (\n <section className=\"space-y-4\">\n <div className=\"space-y-1\">\n <FieldLabel>Customize your Google connection</FieldLabel>\n <p className=\"text-[11px] text-muted-foreground\">\n Pick the Google APIs to bundle into one connection. They share a single OAuth consent and\n appear as merged tools under one Google integration.\n </p>\n </div>\n\n {groups.map(\n ({\n audience,\n presets,\n }: {\n readonly audience: GoogleOpenApiOAuthAudience;\n readonly presets: readonly GoogleOpenApiPreset[];\n }) => (\n <div key={audience} className=\"space-y-2\">\n <div className=\"flex items-center gap-2\">\n <span className=\"text-[11px] font-semibold tracking-wide text-foreground uppercase\">\n {AUDIENCE_LABEL[audience]}\n </span>\n {audienceNeedsWarning(audience) ? <AudienceWarningChip audience={audience} /> : null}\n <Button\n type=\"button\"\n variant=\"ghost\"\n size=\"sm\"\n className=\"ml-auto h-auto px-1.5 py-0.5 text-[11px] font-normal text-muted-foreground hover:bg-transparent hover:text-foreground\"\n onClick={() => {\n const allSelected = presets.every((preset: GoogleOpenApiPreset) =>\n selectedPresetIds.has(preset.id),\n );\n presets.forEach((preset: GoogleOpenApiPreset) =>\n onToggle(preset.id, !allSelected),\n );\n }}\n >\n {presets.every((preset: GoogleOpenApiPreset) => selectedPresetIds.has(preset.id))\n ? \"Clear\"\n : \"Select all\"}\n </Button>\n </div>\n <p className=\"text-[11px] text-muted-foreground\">{AUDIENCE_DESCRIPTION[audience]}</p>\n <div className=\"grid grid-cols-1 gap-x-4 gap-y-0.5 sm:grid-cols-2\">\n {presets.map((preset: GoogleOpenApiPreset) => (\n <ProductRow\n key={preset.id}\n preset={preset}\n checked={selectedPresetIds.has(preset.id)}\n onToggle={(checked: boolean) => onToggle(preset.id, checked)}\n />\n ))}\n </div>\n </div>\n ),\n )}\n\n <CustomUrlEscapeHatch\n customUrls={customUrls}\n onAddCustomUrl={onAddCustomUrl}\n onRemoveCustomUrl={onRemoveCustomUrl}\n />\n\n <div className=\"space-y-1.5 rounded-lg border border-border bg-muted/10 px-3 py-2.5\">\n <div className=\"flex items-center gap-2\">\n <span className=\"text-[11px] font-semibold tracking-wide text-foreground uppercase\">\n Authentication\n </span>\n <Badge variant=\"secondary\">OAuth</Badge>\n </div>\n <p className=\"text-[11px] text-muted-foreground\">\n The selected Google APIs share one OAuth consent. Review the scopes below, then connect a\n Google account from the integration page after adding.\n </p>\n </div>\n\n <Collapsible open={scopesOpen} onOpenChange={setScopesOpen}>\n <CollapsibleTrigger asChild>\n <Button type=\"button\" variant=\"outline\" size=\"sm\" disabled={consentBatches.length === 0}>\n <ChevronDownIcon\n className={cn(\"size-3.5 transition-transform\", scopesOpen ? \"rotate-180\" : \"\")}\n />\n View scopes\n {selectedCount > 0 ? (\n <Badge variant=\"secondary\" className=\"ml-1\">\n {selectedCount}\n </Badge>\n ) : null}\n </Button>\n </CollapsibleTrigger>\n <CollapsibleContent className=\"pt-3\">\n {consentBatches.length === 0 ? (\n <p className=\"text-[11px] text-muted-foreground\">\n Select at least one Google API to preview the OAuth consent.\n </p>\n ) : (\n <div className=\"space-y-3\">\n {consentBatches.map((batch) => (\n <div key={batch.id} className=\"space-y-1.5\">\n <span className=\"text-[11px] font-semibold text-foreground\">{batch.label}</span>\n <ul className=\"space-y-1\">\n {batch.apiScopes.map((scope: string) => (\n <li\n key={scope}\n className=\"rounded-md border border-border bg-muted/20 px-2.5 py-1 font-mono text-[11px] break-all text-muted-foreground\"\n >\n {scope}\n </li>\n ))}\n </ul>\n </div>\n ))}\n </div>\n )}\n </CollapsibleContent>\n </Collapsible>\n </section>\n );\n}\n\nexport default GoogleProductPicker;\n","import type { GoogleOpenApiOAuthAudience } from \"./google-presets\";\nimport { compactGoogleOAuthScopes } from \"./google-oauth-scopes\";\n\nexport type GoogleOAuthBatchInput = {\n readonly id: string;\n readonly name: string;\n readonly oauthAudience: GoogleOpenApiOAuthAudience;\n readonly scopes: readonly string[];\n};\n\nexport type GoogleOAuthConsentBatch = {\n readonly id: string;\n readonly label: string;\n readonly apiScopes: readonly string[];\n};\n\nconst GOOGLE_CLOUD_BATCH_IDS = new Set([\"google-bigquery\", \"google-cloud-resource-manager\"]);\n\nexport const googleOAuthConsentBatches = (\n items: readonly GoogleOAuthBatchInput[],\n): readonly GoogleOAuthConsentBatch[] => {\n const standardScopes: string[] = [];\n const cloudScopes: string[] = [];\n const batches: GoogleOAuthConsentBatch[] = [];\n\n for (const item of items) {\n if (item.scopes.length === 0) continue;\n if (item.oauthAudience === \"standard-user\") {\n standardScopes.push(...item.scopes);\n continue;\n }\n if (GOOGLE_CLOUD_BATCH_IDS.has(item.id)) {\n cloudScopes.push(...item.scopes);\n continue;\n }\n batches.push({\n id: item.id,\n label: item.name,\n apiScopes: item.scopes,\n });\n }\n\n const compactedStandardScopes = compactGoogleOAuthScopes(standardScopes);\n const compactedCloudScopes = compactGoogleOAuthScopes(cloudScopes);\n return [\n ...(compactedStandardScopes.length > 0\n ? [\n {\n id: \"google-core\",\n label: \"Core Google services\",\n apiScopes: compactedStandardScopes,\n },\n ]\n : []),\n ...batches.map((batch) => ({\n ...batch,\n apiScopes: compactGoogleOAuthScopes(batch.apiScopes),\n })),\n ...(compactedCloudScopes.length > 0\n ? [\n {\n id: \"google-cloud\",\n label: \"Google Cloud services\",\n apiScopes: compactedCloudScopes,\n },\n ]\n : []),\n ].filter((batch) => batch.apiScopes.length > 0);\n};\n"],"mappings":"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","names":["useMemo","useState","Button","CardStack","CardStackContent","FieldLabel","PlusIcon","XIcon","Input","IntegrationFavicon","jsx","jsxs","IntegrationFavicon","Input","jsx","jsxs","isGoogleDiscoveryUrl","useState","useMemo","CardStack","CardStackContent","FieldLabel","Button","PlusIcon","XIcon"]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/react/OpenApiAccountsPanel.tsx"],"sourcesContent":["import { useCallback, useMemo } from \"react\";\nimport { useAtomValue, useAtomSet } from \"@effect/atom-react\";\nimport * as Exit from \"effect/Exit\";\nimport * as AsyncResult from \"effect/unstable/reactivity/AsyncResult\";\nimport { AuthTemplateSlug, IntegrationSlug } from \"@executor-js/sdk/shared\";\nimport type { IntegrationAccountHandoff } from \"@executor-js/sdk/client\";\n\nimport { TriangleAlert } from \"lucide-react\";\n\nimport { AccountsSection } from \"@executor-js/react/components/accounts-section\";\nimport { Alert, AlertDescription, AlertTitle } from \"@executor-js/react/components/alert\";\nimport { integrationWriteKeys } from \"@executor-js/react/api/reactivity-keys\";\nimport type { CreateCustomMethod } from \"@executor-js/react/components/add-custom-method-modal\";\nimport type { AuthMethod, Placement } from \"@executor-js/react/lib/auth-placements\";\n\nimport { openApiConfigAtom, openapiConfigure } from \"./atoms\";\nimport { authMethodsFromConfig, templateFromPlacements } from \"./auth-method-config\";\nimport { googleAudienceWarningsForUrls } from \"../sdk/google-presets\";\nimport type { Authentication } from \"../sdk/types\";\n\nconst GOOGLE_AUDIENCE_WARNING: Readonly<Record<string, string>> = {\n \"workspace-admin\":\n \"This connection includes Google Workspace admin APIs (Chat, Admin Directory, Admin Reports). Connecting requires a Workspace admin account — personal Gmail accounts cannot grant these scopes.\",\n \"unsupported-user\":\n \"This connection includes APIs (e.g. Google Keep) that Google does not grant through standard user OAuth consent. Those tools may fail to authorize.\",\n};\n\nconst NO_AUTH_METHOD: AuthMethod = {\n id: \"none\",\n label: \"No authentication\",\n kind: \"none\",\n source: \"spec\",\n template: AuthTemplateSlug.make(\"none\"),\n placements: [],\n};\n\n// ---------------------------------------------------------------------------\n// OpenAPI Accounts hub — fills the generic detail page's `accounts` slot.\n//\n// Reads the integration's real `authenticationTemplate` (via `getConfig`),\n// converts it to generic `AuthMethod[]`, and composes the generic\n// `AccountsSection` — whose Add-account offers those methods plus a \"+ Custom\n// method\" row (apiKey-only). The custom-method create is INJECTED here\n// (`createCustomMethod`): generic placements → an `APIKeyAuthentication`\n// (`templateFromPlacements`, slug omitted → backend `custom_<id>`) merge-\n// appended onto the existing template and persisted via `configure`. Stays\n// plugin-side because it touches the OpenAPI sdk `Authentication` types.\n// ---------------------------------------------------------------------------\n\nexport default function OpenApiAccountsPanel(props: {\n readonly sourceId: string;\n readonly integrationName: string;\n readonly accountHandoff?: IntegrationAccountHandoff | null;\n}) {\n const { sourceId, integrationName, accountHandoff } = props;\n const slug = IntegrationSlug.make(sourceId);\n const configResult = useAtomValue(openApiConfigAtom(slug));\n const doConfigure = useAtomSet(openapiConfigure, { mode: \"promiseExit\" });\n\n // The wire `getConfig` template is structurally an `Authentication[]` (the\n // `slug` is an unbranded string on the wire); treat it as such for the\n // plugin-side converters that brand the slug back.\n const existingTemplate = useMemo<readonly Authentication[]>(() => {\n if (!AsyncResult.isSuccess(configResult) || configResult.value == null) return [];\n return (configResult.value.authenticationTemplate ?? []) as readonly Authentication[];\n }, [configResult]);\n\n const methods = useMemo<readonly AuthMethod[]>(() => {\n const declared = authMethodsFromConfig(existingTemplate);\n return declared.length > 0 ? declared : [NO_AUTH_METHOD];\n }, [existingTemplate]);\n\n // Add a custom apiKey method: build an `APIKeyAuthentication` from the generic\n // placements (slug omitted → backend backfills `custom_<id>`), merge-append it\n // onto the existing template, and persist. Returns the created `AuthMethod`\n // (derived from the same template) so Add-account can select it immediately.\n const createCustomMethod = useCallback<CreateCustomMethod>(\n async (input: { readonly label: string; readonly placements: readonly Placement[] }) => {\n const method = templateFromPlacements(input.placements);\n const exit = await doConfigure({\n params: { slug },\n payload: { authenticationTemplate: [...existingTemplate, method] },\n reactivityKeys: integrationWriteKeys,\n });\n if (Exit.isFailure(exit)) return null;\n const before = new Set(existingTemplate.map((template) => String(template.slug)));\n const created = authMethodsFromConfig(\n exit.value.authenticationTemplate as readonly Authentication[],\n ).find((candidate: AuthMethod) => !before.has(String(candidate.template)));\n return created ?? null;\n },\n [doConfigure, slug, existingTemplate],\n );\n\n const removeCustomMethod = useCallback(\n async (method: AuthMethod): Promise<boolean> => {\n if (method.source !== \"custom\") return false;\n const next = existingTemplate.filter(\n (template: Authentication) => String(template.slug) !== String(method.template),\n );\n const exit = await doConfigure({\n params: { slug },\n payload: { authenticationTemplate: next, mode: \"replace\" },\n reactivityKeys: integrationWriteKeys,\n });\n return Exit.isSuccess(exit);\n },\n [doConfigure, existingTemplate, slug],\n );\n\n // For a bundled `google` integration, surface a caution when any selected API\n // needs a privileged or unsupported OAuth consent the user should know about\n // BEFORE connecting an account. Derived from the stored Discovery URLs.\n const audienceWarnings = useMemo<readonly string[]>(() => {\n if (!AsyncResult.isSuccess(configResult) || configResult.value == null) return [];\n const urls = configResult.value.googleDiscoveryUrls ?? [];\n return googleAudienceWarningsForUrls(urls).flatMap((audience: string) => {\n const message = GOOGLE_AUDIENCE_WARNING[audience];\n return message ? [message] : [];\n });\n }, [configResult]);\n\n return (\n <div className=\"mx-auto max-w-3xl space-y-8 px-6 py-8\">\n {audienceWarnings.length > 0 && (\n <Alert variant=\"destructive\">\n <TriangleAlert />\n <AlertTitle>Some Google APIs need special consent</AlertTitle>\n <AlertDescription>\n {audienceWarnings.map((message: string) => (\n <p key={message}>{message}</p>\n ))}\n </AlertDescription>\n </Alert>\n )}\n <AccountsSection\n integration={slug}\n integrationName={integrationName}\n methods={methods}\n accountHandoff={accountHandoff}\n createCustomMethod={createCustomMethod}\n removeCustomMethod={removeCustomMethod}\n />\n </div>\n );\n}\n"],"mappings":"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","names":[]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/react/atoms.ts","../src/react/client.ts","../src/api/group.ts"],"sourcesContent":["import type { IntegrationSlug } from \"@executor-js/sdk/shared\";\nimport * as Atom from \"effect/unstable/reactivity/Atom\";\nimport { ReactivityKey } from \"@executor-js/react/api/reactivity-keys\";\nimport { OpenApiClient } from \"./client\";\n\n// ---------------------------------------------------------------------------\n// Query atoms — v2: the integration catalog is the unit of identity (the v1\n// per-scope `source` row is gone). `getIntegration` returns the catalog entry\n// (slug/description/kind/canRemove/canRefresh); credentials are owner-scoped\n// connections read from the shared `connections` API, not bound per source.\n// ---------------------------------------------------------------------------\n\nexport const openApiIntegrationAtom = (slug: IntegrationSlug) =>\n OpenApiClient.query(\"openapi\", \"getIntegration\", {\n params: { slug },\n timeToLive: \"15 seconds\",\n reactivityKeys: [ReactivityKey.integrations, ReactivityKey.tools],\n });\n\n// The full opaque config (including `authenticationTemplate`), used by the\n// configure UX to render existing auth methods and add custom ones.\nexport const openApiConfigAtom = (slug: IntegrationSlug) =>\n OpenApiClient.query(\"openapi\", \"getConfig\", {\n params: { slug },\n timeToLive: \"15 seconds\",\n reactivityKeys: [ReactivityKey.integrations, ReactivityKey.tools],\n });\n\n// ---------------------------------------------------------------------------\n// Mutation atoms\n// ---------------------------------------------------------------------------\n\nexport const previewOpenApiSpec = OpenApiClient.mutation(\"openapi\", \"previewSpec\");\n\nexport const addOpenApiSpec = OpenApiClient.mutation(\"openapi\", \"addSpec\");\n\nexport const removeOpenApiSpec = OpenApiClient.mutation(\"openapi\", \"removeSpec\");\n\n// Add / merge custom auth methods onto an integration's `authenticationTemplate`.\nexport const openapiConfigure = OpenApiClient.mutation(\"openapi\", \"configure\");\n\n// `getIntegration` is read-only; the atom family lets a caller pass a slug.\nexport const openApiIntegrationFamily = Atom.family(openApiIntegrationAtom);\n\n// `getConfig` is read-only; the atom family lets a caller pass a slug.\nexport const openApiConfigFamily = Atom.family(openApiConfigAtom);\n","import { createPluginAtomClient } from \"@executor-js/sdk/client\";\nimport {\n getExecutorApiBaseUrl,\n getExecutorServerAuthorizationHeader,\n} from \"@executor-js/react/api/server-connection\";\nimport { OpenApiGroup } from \"../api/group\";\n\nexport const OpenApiClient = createPluginAtomClient(OpenApiGroup, {\n baseUrl: getExecutorApiBaseUrl,\n authorizationHeader: getExecutorServerAuthorizationHeader,\n});\n","import { HttpApiEndpoint, HttpApiGroup } from \"effect/unstable/httpapi\";\nimport { Schema } from \"effect\";\nimport {\n InternalError,\n IntegrationAlreadyExistsError,\n IntegrationSlug,\n} from \"@executor-js/sdk/shared\";\n\nimport { OpenApiParseError, OpenApiExtractionError, OpenApiOAuthError } from \"../sdk/errors\";\nimport { SpecPreview } from \"../sdk/preview\";\n\n// ---------------------------------------------------------------------------\n// Errors — the plugin-domain tagged errors flow directly to clients\n// (4xx, each carrying its own `httpApiStatus`). `InternalError` is the shared\n// opaque 500 surface; `StorageError` → `InternalError` translation happens at\n// service wiring time. `IntegrationAlreadyExistsError` (409) blocks re-adding\n// an existing slug — see addSpec.\n// ---------------------------------------------------------------------------\n\nconst DomainErrors = [\n InternalError,\n OpenApiParseError,\n OpenApiExtractionError,\n OpenApiOAuthError,\n IntegrationAlreadyExistsError,\n] as const;\n\nconst SlugParams = {\n slug: Schema.String,\n};\n\n// ---------------------------------------------------------------------------\n// Payloads\n// ---------------------------------------------------------------------------\n\nconst OpenApiSpecInputPayload = Schema.Union([\n Schema.Struct({ kind: Schema.Literal(\"url\"), url: Schema.String }),\n Schema.Struct({ kind: Schema.Literal(\"blob\"), value: Schema.String }),\n Schema.Struct({\n kind: Schema.Literal(\"googleDiscovery\"),\n url: Schema.String,\n }),\n Schema.Struct({\n kind: Schema.Literal(\"googleDiscoveryBundle\"),\n urls: Schema.Array(Schema.String),\n }),\n]);\n\nconst AuthenticationVariablePayload = Schema.Struct({\n type: Schema.Literal(\"variable\"),\n name: Schema.String,\n});\nconst AuthenticationTemplateValuePayload = Schema.Union([\n Schema.String,\n Schema.Array(Schema.Union([Schema.String, AuthenticationVariablePayload])),\n]);\nconst AuthenticationPayload = Schema.Union([\n Schema.Struct({\n slug: Schema.String,\n type: Schema.Literal(\"apiKey\"),\n headers: Schema.optional(Schema.Record(Schema.String, AuthenticationTemplateValuePayload)),\n queryParams: Schema.optional(Schema.Record(Schema.String, AuthenticationTemplateValuePayload)),\n }),\n Schema.Struct({\n slug: Schema.String,\n type: Schema.Literal(\"oauth\"),\n authorizationUrl: Schema.String,\n tokenUrl: Schema.String,\n scopes: Schema.Array(Schema.String),\n }),\n]);\n\nconst AddSpecPayload = Schema.Struct({\n spec: OpenApiSpecInputPayload,\n slug: Schema.String,\n description: Schema.optional(Schema.String),\n baseUrl: Schema.optional(Schema.String),\n headers: Schema.optional(Schema.Record(Schema.String, Schema.String)),\n queryParams: Schema.optional(Schema.Record(Schema.String, Schema.String)),\n authenticationTemplate: Schema.optional(Schema.Array(AuthenticationPayload)),\n});\n\nconst PreviewSpecPayload = Schema.Struct({\n spec: Schema.String,\n});\n\n// The `configure` payload — the new/updated auth methods to merge onto the\n// integration's `authenticationTemplate`. Reuses the same `AuthenticationPayload`\n// schema as `addSpec` so a custom apiKey method round-trips identically.\nconst ConfigurePayload = Schema.Struct({\n authenticationTemplate: Schema.Array(AuthenticationPayload),\n mode: Schema.optional(Schema.Literals([\"merge\", \"replace\"])),\n});\n\n// ---------------------------------------------------------------------------\n// Responses\n// ---------------------------------------------------------------------------\n\nconst AddSpecResponse = Schema.Struct({\n slug: IntegrationSlug,\n toolCount: Schema.Number,\n});\n\nconst IntegrationView = Schema.Struct({\n slug: IntegrationSlug,\n description: Schema.String,\n kind: Schema.String,\n canRemove: Schema.Boolean,\n canRefresh: Schema.Boolean,\n});\n\n// The full opaque integration config, surfaced for the configure UX. Unlike\n// `IntegrationView` (catalog identity only), this carries the\n// `authenticationTemplate` the configure flow reads/writes.\nconst OpenApiConfigView = Schema.Struct({\n spec: Schema.String,\n sourceUrl: Schema.optional(Schema.String),\n googleDiscoveryUrls: Schema.optional(Schema.Array(Schema.String)),\n baseUrl: Schema.optional(Schema.String),\n headers: Schema.optional(Schema.Record(Schema.String, Schema.String)),\n queryParams: Schema.optional(Schema.Record(Schema.String, Schema.String)),\n authenticationTemplate: Schema.optional(Schema.Array(AuthenticationPayload)),\n});\n\n// The configure result — the merged `authenticationTemplate` after the new\n// custom methods were appended/replaced.\nconst ConfigureResponse = Schema.Struct({\n authenticationTemplate: Schema.Array(AuthenticationPayload),\n});\n\n// ---------------------------------------------------------------------------\n// Group — addSpec/preview/get/remove over the integration catalog.\n// ---------------------------------------------------------------------------\n\nexport const OpenApiGroup = HttpApiGroup.make(\"openapi\")\n .add(\n HttpApiEndpoint.post(\"previewSpec\", \"/openapi/preview\", {\n payload: PreviewSpecPayload,\n success: SpecPreview,\n error: DomainErrors,\n }),\n )\n .add(\n HttpApiEndpoint.post(\"addSpec\", \"/openapi/specs\", {\n payload: AddSpecPayload,\n success: AddSpecResponse,\n error: DomainErrors,\n }),\n )\n .add(\n HttpApiEndpoint.get(\"getIntegration\", \"/openapi/integrations/:slug\", {\n params: SlugParams,\n success: Schema.NullOr(IntegrationView),\n error: DomainErrors,\n }),\n )\n .add(\n HttpApiEndpoint.get(\"getConfig\", \"/openapi/integrations/:slug/config\", {\n params: SlugParams,\n success: Schema.NullOr(OpenApiConfigView),\n error: DomainErrors,\n }),\n )\n .add(\n HttpApiEndpoint.post(\"configure\", \"/openapi/integrations/:slug/config\", {\n params: SlugParams,\n payload: ConfigurePayload,\n success: ConfigureResponse,\n error: DomainErrors,\n }),\n )\n .add(\n HttpApiEndpoint.delete(\"removeSpec\", \"/openapi/integrations/:slug\", {\n params: SlugParams,\n success: Schema.Void,\n error: DomainErrors,\n }),\n );\n"],"mappings":"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","names":[]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/sdk/google-discovery.ts","../src/sdk/google-oauth-scopes.ts"],"sourcesContent":["// Converts Google Discovery documents directly into OpenAPI 3.x. Public\n// Discovery converters currently target Swagger 2.0 or a broad conversion\n// pipeline; this adapter emits the shape Executor parses while preserving\n// Executor-specific tool ids and query semantics.\nimport { Effect, Option, Predicate, Schema, SchemaGetter } from \"effect\";\nimport { HttpClient, HttpClientRequest } from \"effect/unstable/http\";\n\nimport { OpenApiParseError } from \"./errors\";\nimport { compactGoogleOAuthScopes } from \"./google-oauth-scopes\";\nimport type { SpecFetchCredentials } from \"./parse\";\nimport type { Authentication } from \"./types\";\nimport { AuthTemplateSlug } from \"@executor-js/sdk/shared\";\n\nconst DISCOVERY_SERVICE_HOST = \"https://www.googleapis.com/discovery/v1/apis\";\nconst GOOGLE_BUNDLE_BASE_URL = \"https://www.googleapis.com/\";\nconst GOOGLE_OAUTH_AUTHORIZATION_URL = \"https://accounts.google.com/o/oauth2/v2/auth\";\nconst GOOGLE_OAUTH_TOKEN_URL = \"https://oauth2.googleapis.com/token\";\nconst OPENAPI_SCHEMA_TYPES = new Set([\n \"array\",\n \"boolean\",\n \"integer\",\n \"null\",\n \"number\",\n \"object\",\n \"string\",\n]);\n\ntype JsonPrimitive = string | number | boolean | null;\ntype JsonValue = JsonPrimitive | readonly JsonValue[] | { readonly [key: string]: JsonValue };\n\ntype OpenApiSchemaObject = {\n readonly $ref?: string;\n readonly type?: \"array\" | \"boolean\" | \"integer\" | \"null\" | \"number\" | \"object\" | \"string\";\n readonly description?: string;\n readonly title?: string;\n readonly format?: string;\n readonly readOnly?: boolean;\n readonly default?: JsonValue;\n readonly enum?: readonly JsonValue[];\n readonly items?: OpenApiSchemaObject;\n readonly properties?: Record<string, OpenApiSchemaObject>;\n readonly required?: readonly string[];\n readonly additionalProperties?: boolean | OpenApiSchemaObject;\n};\n\ntype OpenApiParameterObject = {\n readonly name: string;\n readonly in: \"path\" | \"query\" | \"header\";\n readonly required: boolean;\n readonly description?: string;\n readonly schema: OpenApiSchemaObject;\n readonly style?: \"form\";\n readonly explode?: boolean;\n readonly allowReserved?: boolean;\n};\n\ntype OpenApiOperationObject = {\n readonly operationId: string;\n readonly \"x-executor-toolPath\": string;\n readonly \"x-executor-pathTemplate\"?: string;\n readonly tags?: readonly string[];\n readonly description?: string;\n readonly servers?: readonly { readonly url: string }[];\n readonly parameters: readonly OpenApiParameterObject[];\n readonly requestBody?: {\n readonly required: false;\n readonly content: {\n readonly \"application/json\": {\n readonly schema: OpenApiSchemaObject;\n };\n };\n };\n readonly responses: {\n readonly \"200\": {\n readonly description: \"Successful response\";\n readonly content: {\n readonly \"application/json\": {\n readonly schema: OpenApiSchemaObject;\n };\n };\n };\n };\n readonly security?: readonly Record<string, readonly string[]>[];\n readonly \"x-google-scopes\": readonly string[];\n};\n\ntype OpenApiDocument = {\n readonly openapi: \"3.1.0\";\n readonly info: {\n readonly title: string;\n readonly version: string;\n };\n readonly servers: readonly { readonly url: string }[];\n readonly paths: Record<string, Record<string, OpenApiOperationObject>>;\n readonly components: {\n readonly schemas: Record<string, OpenApiSchemaObject>;\n readonly securitySchemes?: Record<\n string,\n {\n readonly type: \"oauth2\";\n readonly flows: {\n readonly authorizationCode: {\n readonly authorizationUrl: string;\n readonly tokenUrl: string;\n readonly scopes: Record<string, string>;\n };\n };\n }\n >;\n };\n readonly security?: readonly Record<string, readonly string[]>[];\n readonly \"x-executor-origin\":\n | {\n readonly kind: \"googleDiscovery\";\n readonly discoveryUrl: string;\n readonly service: string;\n readonly version: string;\n }\n | {\n readonly kind: \"googleDiscoveryBundle\";\n readonly services: readonly {\n readonly discoveryUrl: string;\n readonly service: string;\n readonly version: string;\n }[];\n };\n};\n\nconst TextOption = Schema.OptionFromOptional(Schema.Trim).pipe(\n Schema.decode({\n decode: SchemaGetter.transform((value) => Option.filter(value, (text) => text.length > 0)),\n encode: SchemaGetter.transform((value) => value),\n }),\n Schema.withDecodingDefaultType(Effect.succeed(Option.none())),\n);\nconst TextArray = Schema.optional(Schema.Array(Schema.String)).pipe(\n Schema.withDecodingDefaultType(Effect.succeed([] as string[])),\n);\nconst UnknownRecord = Schema.Record(Schema.String, Schema.Unknown);\nconst UnknownRecordWithDefault = Schema.optional(UnknownRecord).pipe(\n Schema.withDecodingDefaultType(Effect.succeed({})),\n);\n\nconst DiscoveryParameter = Schema.Struct({\n type: Schema.optional(Schema.String),\n description: TextOption,\n properties: UnknownRecordWithDefault,\n items: Schema.optional(Schema.Unknown),\n additionalProperties: Schema.optional(Schema.Union([Schema.Boolean, Schema.Unknown])),\n enum: TextArray,\n format: Schema.optional(Schema.String),\n readOnly: Schema.optional(Schema.Boolean),\n default: Schema.optional(Schema.Union([Schema.String, Schema.Number, Schema.Boolean])),\n $ref: Schema.optional(Schema.String),\n location: Schema.optional(Schema.Literals([\"path\", \"query\", \"header\"])),\n required: Schema.optional(Schema.Boolean),\n repeated: Schema.optional(Schema.Boolean),\n});\ntype DiscoveryParameter = typeof DiscoveryParameter.Type;\n\nconst DiscoveryRef = Schema.Struct({\n $ref: Schema.optional(Schema.String),\n});\n\nconst DiscoveryMethod = Schema.Struct({\n id: TextOption,\n description: TextOption,\n httpMethod: Schema.optional(Schema.String),\n path: TextOption,\n parameters: UnknownRecordWithDefault,\n request: Schema.optional(DiscoveryRef),\n response: Schema.optional(DiscoveryRef),\n scopes: TextArray,\n});\ntype DiscoveryMethod = typeof DiscoveryMethod.Type;\n\nconst DiscoveryResource = Schema.Struct({\n methods: UnknownRecordWithDefault,\n resources: UnknownRecordWithDefault,\n});\n\nconst DiscoveryDocument = Schema.Struct({\n name: TextOption,\n version: TextOption,\n title: TextOption,\n rootUrl: TextOption,\n servicePath: Schema.optional(Schema.Trim).pipe(\n Schema.withDecodingDefaultType(Effect.succeed(\"\")),\n ),\n parameters: UnknownRecordWithDefault,\n methods: UnknownRecordWithDefault,\n resources: UnknownRecordWithDefault,\n schemas: UnknownRecordWithDefault,\n auth: Schema.optional(\n Schema.Struct({\n oauth2: Schema.optional(\n Schema.Struct({\n scopes: Schema.optional(\n Schema.Record(\n Schema.String,\n Schema.Struct({\n description: TextOption,\n }),\n ),\n ).pipe(Schema.withDecodingDefaultType(Effect.succeed({}))),\n }),\n ),\n }),\n ),\n});\ntype DiscoveryDocument = typeof DiscoveryDocument.Type;\n\nexport interface GoogleDiscoveryOpenApiConversion {\n readonly specText: string;\n readonly baseUrl: string;\n readonly title: string;\n readonly service: string;\n readonly version: string;\n readonly discoveryUrls?: readonly string[];\n /** The v2 oauth auth template the converted integration declares, when the\n * Discovery document advertises OAuth2 scopes. */\n readonly authenticationTemplate?: readonly Authentication[];\n}\n\nconst decodeDiscoveryDocument = Schema.decodeUnknownSync(DiscoveryDocument);\nconst decodeDiscoveryParameter = Schema.decodeUnknownSync(DiscoveryParameter);\nconst decodeDiscoveryMethod = Schema.decodeUnknownSync(DiscoveryMethod);\nconst decodeDiscoveryResource = Schema.decodeUnknownSync(DiscoveryResource);\nconst parseJson = Schema.decodeUnknownEffect(Schema.fromJsonString(Schema.Unknown));\n\nconst normalizeDiscoveryUrl = (discoveryUrl: string): string => {\n const trimmed = discoveryUrl.trim();\n if (!URL.canParse(trimmed)) return trimmed;\n const parsed = new URL(trimmed);\n if (parsed.pathname !== \"/$discovery/rest\") return trimmed;\n const version = parsed.searchParams.get(\"version\")?.trim();\n if (!version) return trimmed;\n const host = parsed.hostname.toLowerCase();\n if (!host.endsWith(\".googleapis.com\")) return trimmed;\n const rawService = host.slice(0, -\".googleapis.com\".length);\n const service =\n rawService === \"calendar-json\"\n ? \"calendar\"\n : rawService.endsWith(\"-json\")\n ? rawService.slice(0, -5)\n : rawService;\n return service ? `${DISCOVERY_SERVICE_HOST}/${service}/${version}/rest` : trimmed;\n};\n\nexport const isGoogleDiscoveryUrl = (url: string): boolean => {\n const trimmed = url.trim();\n if (!URL.canParse(trimmed)) return false;\n const parsed = new URL(trimmed);\n const host = parsed.hostname.toLowerCase();\n if (!host.endsWith(\"googleapis.com\")) return false;\n return parsed.pathname.includes(\"/discovery/\") || parsed.pathname.includes(\"$discovery\");\n};\n\nexport const fetchGoogleDiscoveryDocument = Effect.fn(\"OpenApi.fetchGoogleDiscoveryDocument\")(\n function* (discoveryUrl: string, credentials?: SpecFetchCredentials) {\n const client = yield* HttpClient.HttpClient;\n const requestUrl = new URL(discoveryUrl);\n for (const [name, value] of Object.entries(credentials?.queryParams ?? {})) {\n requestUrl.searchParams.set(name, value);\n }\n let request = HttpClientRequest.get(requestUrl.toString()).pipe(\n HttpClientRequest.setHeader(\"Accept\", \"application/json, */*\"),\n );\n for (const [name, value] of Object.entries(credentials?.headers ?? {})) {\n request = HttpClientRequest.setHeader(request, name, value);\n }\n const response = yield* client.execute(request).pipe(\n Effect.mapError(\n () =>\n new OpenApiParseError({\n message: \"Failed to fetch Google Discovery document\",\n }),\n ),\n );\n if (response.status < 200 || response.status >= 300) {\n return yield* new OpenApiParseError({\n message: `Failed to fetch Google Discovery document: HTTP ${response.status}`,\n });\n }\n return yield* response.text.pipe(\n Effect.mapError(\n () =>\n new OpenApiParseError({\n message: \"Failed to read Google Discovery document body\",\n }),\n ),\n );\n },\n);\n\nconst schemaRef = (name: string) => `#/components/schemas/${name}`;\n\nconst identitySchemaName = (name: string): string => name;\n\nconst schemaComponentPart = (value: string): string =>\n value\n .trim()\n .replace(/[^A-Za-z0-9._-]+/g, \"_\")\n .replace(/^_+|_+$/g, \"\") || \"schema\";\n\nconst normalizeDiscoveryPathTemplate = (pathTemplate: string): string =>\n pathTemplate.replaceAll(/\\{\\+([^{}]+)\\}/g, \"{$1}\");\n\nconst pathUsesReservedExpansion = (pathTemplate: string, parameterName: string): boolean =>\n pathTemplate.includes(`{+${parameterName}}`);\n\nconst uniquePathKey = (\n paths: Record<string, Record<string, OpenApiOperationObject>>,\n preferredPath: string,\n method: string,\n toolPath: string,\n): string => {\n if (!paths[preferredPath]?.[method]) return preferredPath;\n const disambiguated = `/${toolPath.replace(/[^A-Za-z0-9._~-]+/g, \"/\")}`;\n if (!paths[disambiguated]?.[method]) return disambiguated;\n let index = 2;\n while (paths[`${disambiguated}/${index}`]?.[method]) index += 1;\n return `${disambiguated}/${index}`;\n};\n\nconst discoveryDescription = (value: unknown): string | undefined =>\n typeof value === \"string\"\n ? value\n : Option.isOption(value) && Option.isSome(value)\n ? typeof value.value === \"string\"\n ? value.value\n : undefined\n : undefined;\n\nconst isRecord = (value: unknown): value is Record<string, unknown> =>\n typeof value === \"object\" && value !== null && !Array.isArray(value);\n\nconst jsonValue = (value: unknown): JsonValue | undefined => {\n if (\n value === null ||\n typeof value === \"string\" ||\n typeof value === \"number\" ||\n typeof value === \"boolean\"\n ) {\n return value;\n }\n if (Array.isArray(value)) {\n const values = value.map(jsonValue);\n return values.every(Predicate.isNotUndefined) ? values : undefined;\n }\n if (!isRecord(value)) return undefined;\n const entries = Object.entries(value).flatMap(([key, item]) => {\n const converted = jsonValue(item);\n return converted === undefined ? [] : [[key, converted] as const];\n });\n return Object.fromEntries(entries);\n};\n\nconst stringArray = (value: unknown): readonly string[] | undefined => {\n if (!Array.isArray(value)) return undefined;\n const strings = value.filter((item): item is string => typeof item === \"string\");\n return strings.length === value.length ? strings : undefined;\n};\n\nconst schemaType = (value: unknown): OpenApiSchemaObject[\"type\"] | undefined =>\n typeof value === \"string\" && OPENAPI_SCHEMA_TYPES.has(value)\n ? (value as OpenApiSchemaObject[\"type\"])\n : undefined;\n\nconst discoverySchemaToOpenApiSchema = (\n raw: unknown,\n schemaNameForRef: (name: string) => string = identitySchemaName,\n): OpenApiSchemaObject => {\n if (!isRecord(raw)) return {};\n const schema = raw;\n if (typeof schema.$ref === \"string\") return { $ref: schemaRef(schemaNameForRef(schema.$ref)) };\n\n const description = discoveryDescription(schema.description);\n const title = discoveryDescription(schema.title);\n const defaultValue = jsonValue(schema.default);\n const enumValues = Array.isArray(schema.enum)\n ? schema.enum.map(jsonValue).filter(Predicate.isNotUndefined)\n : [];\n const format = typeof schema.format === \"string\" ? schema.format : undefined;\n const readOnly = typeof schema.readOnly === \"boolean\" ? schema.readOnly : undefined;\n const type = schemaType(schema.type);\n\n const base = {\n ...(description !== undefined ? { description } : {}),\n ...(title !== undefined ? { title } : {}),\n ...(format !== undefined ? { format } : {}),\n ...(readOnly !== undefined ? { readOnly } : {}),\n ...(defaultValue !== undefined ? { default: defaultValue } : {}),\n ...(enumValues.length > 0 ? { enum: enumValues } : {}),\n } satisfies OpenApiSchemaObject;\n\n if (type === \"array\") {\n return {\n ...base,\n type: \"array\",\n items: discoverySchemaToOpenApiSchema(schema.items, schemaNameForRef),\n };\n }\n\n const properties = schema.properties;\n if (\n type === \"object\" ||\n (isRecord(properties) && Object.keys(properties).length > 0) ||\n schema.additionalProperties !== undefined\n ) {\n const convertedProperties = isRecord(properties)\n ? Object.fromEntries(\n Object.entries(properties).map(([name, value]) => [\n name,\n discoverySchemaToOpenApiSchema(value, schemaNameForRef),\n ]),\n )\n : undefined;\n const required = stringArray(schema.required);\n const additionalProperties =\n schema.additionalProperties === undefined\n ? undefined\n : typeof schema.additionalProperties === \"boolean\"\n ? schema.additionalProperties\n : discoverySchemaToOpenApiSchema(schema.additionalProperties, schemaNameForRef);\n return {\n ...base,\n type: \"object\",\n ...(convertedProperties && Object.keys(convertedProperties).length > 0\n ? { properties: convertedProperties }\n : {}),\n ...(required && required.length > 0 ? { required } : {}),\n ...(additionalProperties !== undefined ? { additionalProperties } : {}),\n };\n }\n\n return type !== undefined ? { ...base, type } : base;\n};\n\nconst parameterSchema = (\n parameter: DiscoveryParameter,\n schemaNameForRef: (name: string) => string = identitySchemaName,\n): OpenApiSchemaObject => {\n const base = discoverySchemaToOpenApiSchema(parameter, schemaNameForRef);\n return parameter.repeated\n ? {\n type: \"array\",\n items: base,\n }\n : base;\n};\n\nconst methodToolPath = (service: string, methodId: string): string =>\n methodId.startsWith(`${service}.`) ? methodId.slice(service.length + 1) : methodId;\n\nconst collectMethods = (resource: unknown): DiscoveryMethod[] => {\n const decoded = decodeDiscoveryResource(resource);\n const direct = Object.values(decoded.methods ?? {}).map((raw) => decodeDiscoveryMethod(raw));\n const nested = Object.values(decoded.resources ?? {}).flatMap(collectMethods);\n return [...direct, ...nested];\n};\n\nconst discoveryScopes = (document: DiscoveryDocument): Record<string, string> =>\n Object.fromEntries(\n Object.entries(document.auth?.oauth2?.scopes ?? {}).map(([scope, value]) => [\n scope,\n Option.getOrElse(value.description, () => \"\"),\n ]),\n );\n\n// The scope set the converted integration DECLARES (and that `oauth.start`\n// requests at connect) must match the consent the picker previews. Both run the\n// raw Discovery union through `compactGoogleOAuthScopes`, which drops scopes a\n// user OAuth consent screen can't show (`chat.bot`/`chat.app.*`/`keep`) and\n// collapses sub-scopes under their broad parent (`gmail.*` → `mail.google.com`,\n// `userinfo.email` → `email`). Descriptions are preserved where the raw map had\n// them; compaction-introduced identity scopes (`email`/`profile`) fall back to\n// the broad parent's description. Per-operation `x-google-scopes`/`security`\n// stay RAW — they describe which scope each method needs, not consent.\nconst compactDiscoveryScopeMap = (raw: Record<string, string>): Record<string, string> => {\n const descriptionFor = (scope: string): string => {\n if (raw[scope] !== undefined) return raw[scope];\n if (scope === \"email\") return raw[\"https://www.googleapis.com/auth/userinfo.email\"] ?? \"\";\n if (scope === \"profile\") return raw[\"https://www.googleapis.com/auth/userinfo.profile\"] ?? \"\";\n return \"\";\n };\n return Object.fromEntries(\n compactGoogleOAuthScopes(Object.keys(raw)).map((scope) => [scope, descriptionFor(scope)]),\n );\n};\n\nconst allDiscoveryMethods = (document: DiscoveryDocument): DiscoveryMethod[] => [\n ...Object.values(document.methods ?? {}).map((raw) => decodeDiscoveryMethod(raw)),\n ...Object.values(document.resources ?? {}).flatMap(collectMethods),\n];\n\ntype DiscoveryDocumentInfo = {\n readonly discoveryUrl: string;\n readonly document: DiscoveryDocument;\n readonly service: string;\n readonly version: string;\n readonly rootUrl: string;\n readonly baseUrl: string;\n readonly title: string;\n};\n\nconst discoveryDocumentInfo = (\n document: DiscoveryDocument,\n discoveryUrl: string,\n): Effect.Effect<DiscoveryDocumentInfo, OpenApiParseError> =>\n Effect.gen(function* () {\n const service = Option.getOrUndefined(document.name);\n const version = Option.getOrUndefined(document.version);\n const rootUrl = Option.getOrUndefined(document.rootUrl);\n if (!service || !version || !rootUrl) {\n return yield* new OpenApiParseError({\n message: \"Google Discovery document is missing one of: name, version, rootUrl\",\n });\n }\n\n return {\n discoveryUrl,\n document,\n service,\n version,\n rootUrl,\n baseUrl: new URL(document.servicePath || \"\", rootUrl).toString(),\n title: Option.getOrElse(document.title, () => `${service} ${version}`),\n };\n });\n\nconst buildDiscoveryOperation = (input: {\n readonly document: DiscoveryDocument;\n readonly method: DiscoveryMethod;\n readonly toolPath: string;\n readonly pathTemplate: string;\n readonly schemaNameForRef?: (name: string) => string;\n readonly serverUrl?: string;\n readonly tags?: readonly string[];\n}): OpenApiOperationObject => {\n const mergedParameters = new Map<string, DiscoveryParameter>();\n for (const [name, raw] of Object.entries(input.document.parameters ?? {})) {\n const parameter = decodeDiscoveryParameter(raw);\n if (parameter.location) mergedParameters.set(name, parameter);\n }\n for (const [name, raw] of Object.entries(input.method.parameters ?? {})) {\n const parameter = decodeDiscoveryParameter(raw);\n if (parameter.location) mergedParameters.set(name, parameter);\n }\n\n const methodScopes = input.method.scopes ?? [];\n const methodDescription = Option.getOrUndefined(input.method.description);\n const schemaNameForRef = input.schemaNameForRef ?? identitySchemaName;\n\n return {\n operationId: input.toolPath,\n \"x-executor-toolPath\": input.toolPath,\n \"x-executor-pathTemplate\": input.pathTemplate,\n ...(input.tags && input.tags.length > 0 ? { tags: input.tags } : {}),\n ...(methodDescription !== undefined ? { description: methodDescription } : {}),\n ...(input.serverUrl ? { servers: [{ url: input.serverUrl }] } : {}),\n parameters: [...mergedParameters.entries()].flatMap(([name, parameter]) => {\n const location = parameter.location;\n if (!location) return [];\n const description = Option.getOrUndefined(parameter.description);\n const allowReserved =\n location === \"path\" && pathUsesReservedExpansion(input.pathTemplate, name);\n return [\n {\n name,\n in: location,\n required: location === \"path\" ? true : parameter.required === true,\n ...(description !== undefined ? { description } : {}),\n schema: parameterSchema(parameter, schemaNameForRef),\n ...(location === \"query\"\n ? { style: \"form\" as const, explode: parameter.repeated === true }\n : {}),\n ...(allowReserved ? { allowReserved: true } : {}),\n },\n ];\n }),\n ...(input.method.request?.$ref\n ? {\n requestBody: {\n required: false,\n content: {\n \"application/json\": {\n schema: { $ref: schemaRef(schemaNameForRef(input.method.request.$ref)) },\n },\n },\n },\n }\n : {}),\n responses: {\n \"200\": {\n description: \"Successful response\",\n content: {\n \"application/json\": {\n schema: input.method.response?.$ref\n ? { $ref: schemaRef(schemaNameForRef(input.method.response.$ref)) }\n : {},\n },\n },\n },\n },\n ...(methodScopes.length > 0 ? { security: [{ googleOAuth2: methodScopes }] } : {}),\n \"x-google-scopes\": methodScopes,\n };\n};\n\nconst GOOGLE_OAUTH_SECURITY_SCHEME = \"googleOAuth2\";\n\n/** The v2 oauth auth template for a Google-discovery integration. The spec\n * itself carries the matching `securitySchemes.googleOAuth2` entry; this is the\n * catalog-level template a connection's access token renders through. */\nconst googleOauthTemplate = (scopes: Record<string, string>): readonly Authentication[] =>\n // A Google-discovery integration is ALWAYS OAuth, so it ALWAYS declares its\n // oauth method — even when the compacted scope set is empty (e.g. a bundle of\n // only limited-consent APIs like Google Keep, whose scopes Google won't grant\n // through standard consent). Without this the integration declares no auth\n // method and the \"Add account\" / \"Add a connection\" actions are disabled, so\n // you can't even start the flow. Empty `scopes` just requests no scope.\n [\n {\n slug: AuthTemplateSlug.make(GOOGLE_OAUTH_SECURITY_SCHEME),\n type: \"oauth\",\n authorizationUrl: GOOGLE_OAUTH_AUTHORIZATION_URL,\n tokenUrl: GOOGLE_OAUTH_TOKEN_URL,\n scopes: Object.keys(scopes),\n },\n ];\n\nexport const convertGoogleDiscoveryToOpenApi = Effect.fn(\"OpenApi.convertGoogleDiscovery\")(\n function* (input: { readonly discoveryUrl: string; readonly documentText: string }) {\n const parsed = yield* parseJson(input.documentText).pipe(\n Effect.mapError(\n () =>\n new OpenApiParseError({\n message: \"Failed to parse Google Discovery document\",\n }),\n ),\n );\n const document = yield* Effect.try({\n try: () => decodeDiscoveryDocument(parsed),\n catch: () =>\n new OpenApiParseError({\n message: \"Failed to decode Google Discovery document\",\n }),\n });\n\n const info = yield* discoveryDocumentInfo(document, input.discoveryUrl);\n const { service, version, baseUrl, title } = info;\n const paths: Record<string, Record<string, OpenApiOperationObject>> = {};\n\n for (const method of allDiscoveryMethods(document)) {\n const methodId = Option.getOrUndefined(method.id);\n const pathTemplate = Option.getOrUndefined(method.path);\n if (!methodId || !pathTemplate || !method.httpMethod) continue;\n\n const toolPath = methodToolPath(service, methodId);\n const path = normalizeDiscoveryPathTemplate(\n pathTemplate.startsWith(\"/\") ? pathTemplate : `/${pathTemplate}`,\n );\n const methodKey = method.httpMethod.toLowerCase();\n const pathKey = uniquePathKey(paths, path, methodKey, toolPath);\n\n paths[pathKey] ??= {};\n paths[pathKey]![methodKey] = buildDiscoveryOperation({\n document,\n method,\n toolPath,\n pathTemplate: pathTemplate.startsWith(\"/\") ? pathTemplate : `/${pathTemplate}`,\n });\n }\n\n const scopes = compactDiscoveryScopeMap(discoveryScopes(document));\n const authenticationTemplate = googleOauthTemplate(scopes);\n\n const spec: OpenApiDocument = {\n openapi: \"3.1.0\",\n info: {\n title,\n version,\n },\n servers: [{ url: baseUrl }],\n paths,\n components: {\n schemas: Object.fromEntries(\n Object.entries(document.schemas ?? {}).map(([name, schema]) => [\n name,\n discoverySchemaToOpenApiSchema(schema),\n ]),\n ),\n ...(authenticationTemplate\n ? {\n securitySchemes: {\n googleOAuth2: {\n type: \"oauth2\",\n flows: {\n authorizationCode: {\n authorizationUrl: GOOGLE_OAUTH_AUTHORIZATION_URL,\n tokenUrl: GOOGLE_OAUTH_TOKEN_URL,\n scopes,\n },\n },\n },\n },\n }\n : {}),\n },\n ...(authenticationTemplate ? { security: [{ googleOAuth2: Object.keys(scopes) }] } : {}),\n \"x-executor-origin\": {\n kind: \"googleDiscovery\",\n discoveryUrl: normalizeDiscoveryUrl(input.discoveryUrl),\n service,\n version,\n },\n };\n\n return {\n // @effect-diagnostics-next-line preferSchemaOverJson:off\n specText: JSON.stringify(spec),\n baseUrl,\n title,\n service,\n version,\n ...(authenticationTemplate ? { authenticationTemplate } : {}),\n };\n },\n);\n\nexport const convertGoogleDiscoveryBundleToOpenApi = Effect.fn(\n \"OpenApi.convertGoogleDiscoveryBundle\",\n)(function* (input: {\n readonly documents: readonly { readonly discoveryUrl: string; readonly documentText: string }[];\n}) {\n if (input.documents.length === 0) {\n return yield* new OpenApiParseError({\n message: \"Google Discovery bundle requires at least one document\",\n });\n }\n\n const infos = yield* Effect.forEach(input.documents, ({ discoveryUrl, documentText }) =>\n Effect.gen(function* () {\n const parsed = yield* parseJson(documentText).pipe(\n Effect.mapError(\n () =>\n new OpenApiParseError({\n message: \"Failed to parse Google Discovery document\",\n }),\n ),\n );\n const document = yield* Effect.try({\n try: () => decodeDiscoveryDocument(parsed),\n catch: () =>\n new OpenApiParseError({\n message: \"Failed to decode Google Discovery document\",\n }),\n });\n return yield* discoveryDocumentInfo(document, discoveryUrl);\n }),\n );\n\n const paths: Record<string, Record<string, OpenApiOperationObject>> = {};\n const schemas: Record<string, OpenApiSchemaObject> = {};\n const rawScopes: Record<string, string> = {};\n\n for (const info of infos) {\n const schemaPrefix = schemaComponentPart(`${info.service}_${info.version}`);\n const schemaNameForRef = (name: string) => `${schemaPrefix}_${schemaComponentPart(name)}`;\n\n for (const [scope, description] of Object.entries(discoveryScopes(info.document))) {\n rawScopes[scope] ??= description;\n }\n\n for (const [name, schema] of Object.entries(info.document.schemas ?? {})) {\n schemas[schemaNameForRef(name)] = discoverySchemaToOpenApiSchema(schema, schemaNameForRef);\n }\n\n for (const method of allDiscoveryMethods(info.document)) {\n const methodId = Option.getOrUndefined(method.id);\n const rawPathTemplate = Option.getOrUndefined(method.path);\n if (!methodId || !rawPathTemplate || !method.httpMethod) continue;\n\n const toolPath = methodId;\n const wirePath = rawPathTemplate.startsWith(\"/\") ? rawPathTemplate : `/${rawPathTemplate}`;\n const openApiPath = normalizeDiscoveryPathTemplate(wirePath);\n const methodKey = method.httpMethod.toLowerCase();\n const pathKey = uniquePathKey(paths, openApiPath, methodKey, toolPath);\n\n paths[pathKey] ??= {};\n paths[pathKey]![methodKey] = buildDiscoveryOperation({\n document: info.document,\n method,\n toolPath,\n pathTemplate: wirePath,\n schemaNameForRef,\n serverUrl: info.baseUrl,\n tags: [info.title],\n });\n }\n }\n\n const scopes = compactDiscoveryScopeMap(rawScopes);\n const authenticationTemplate = googleOauthTemplate(scopes);\n const spec: OpenApiDocument = {\n openapi: \"3.1.0\",\n info: {\n title: \"Google\",\n version: \"google-discovery-bundle\",\n },\n servers: [{ url: GOOGLE_BUNDLE_BASE_URL }],\n paths,\n components: {\n schemas,\n ...(authenticationTemplate\n ? {\n securitySchemes: {\n googleOAuth2: {\n type: \"oauth2\",\n flows: {\n authorizationCode: {\n authorizationUrl: GOOGLE_OAUTH_AUTHORIZATION_URL,\n tokenUrl: GOOGLE_OAUTH_TOKEN_URL,\n scopes,\n },\n },\n },\n },\n }\n : {}),\n },\n ...(authenticationTemplate ? { security: [{ googleOAuth2: Object.keys(scopes) }] } : {}),\n \"x-executor-origin\": {\n kind: \"googleDiscoveryBundle\",\n services: infos.map((info) => ({\n discoveryUrl: normalizeDiscoveryUrl(info.discoveryUrl),\n service: info.service,\n version: info.version,\n })),\n },\n };\n\n return {\n // @effect-diagnostics-next-line preferSchemaOverJson:off\n specText: JSON.stringify(spec),\n baseUrl: GOOGLE_BUNDLE_BASE_URL,\n title: \"Google\",\n service: \"google\",\n version: \"google-discovery-bundle\",\n discoveryUrls: infos.map((info) => normalizeDiscoveryUrl(info.discoveryUrl)),\n ...(authenticationTemplate ? { authenticationTemplate } : {}),\n };\n});\n","const googleUserConsentBlockedScopes = new Set([\n \"https://www.googleapis.com/auth/chat.bot\",\n \"https://www.googleapis.com/auth/chat.import\",\n \"https://www.googleapis.com/auth/keep\",\n \"https://www.googleapis.com/auth/keep.readonly\",\n]);\n\nconst googleUserConsentBlockedScopePrefixes = [\"https://www.googleapis.com/auth/chat.app.\"];\n\nconst googleBroadScopeGroups: readonly {\n readonly broad: string;\n readonly prefixes: readonly string[];\n}[] = [\n {\n broad: \"https://mail.google.com/\",\n prefixes: [\"https://www.googleapis.com/auth/gmail.\"],\n },\n {\n broad: \"https://www.googleapis.com/auth/calendar\",\n prefixes: [\"https://www.googleapis.com/auth/calendar.\"],\n },\n {\n broad: \"https://www.googleapis.com/auth/drive\",\n prefixes: [\"https://www.googleapis.com/auth/drive.\"],\n },\n];\n\nconst normalizeGoogleIdentityScope = (scope: string): string =>\n scope === \"https://www.googleapis.com/auth/userinfo.email\"\n ? \"email\"\n : scope === \"https://www.googleapis.com/auth/userinfo.profile\"\n ? \"profile\"\n : scope;\n\nconst orderedUniqueScopes = (scopes: Iterable<string>): string[] => {\n const ordered: string[] = [];\n const seen = new Set<string>();\n for (const scope of scopes) {\n const trimmed = scope.trim();\n if (!trimmed || seen.has(trimmed)) continue;\n seen.add(trimmed);\n ordered.push(trimmed);\n }\n return ordered;\n};\n\nexport const isGoogleUserConsentOAuthScope = (scope: string): boolean =>\n !googleUserConsentBlockedScopes.has(scope) &&\n !googleUserConsentBlockedScopePrefixes.some((prefix) => scope.startsWith(prefix));\n\nexport const filterGoogleUserConsentOAuthScopes = (scopes: Iterable<string>): string[] =>\n orderedUniqueScopes(scopes).filter(isGoogleUserConsentOAuthScope);\n\nexport const compactGoogleOAuthScopes = (scopes: Iterable<string>): string[] => {\n const ordered = filterGoogleUserConsentOAuthScopes([...scopes].map(normalizeGoogleIdentityScope));\n const present = new Set(ordered);\n return ordered.filter(\n (scope) =>\n !googleBroadScopeGroups.some(\n (group) =>\n scope !== group.broad &&\n present.has(group.broad) &&\n group.prefixes.some((prefix) => scope.startsWith(prefix)),\n ),\n );\n};\n"],"mappings":"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","names":[]}
|
package/dist/chunk-QSSRVK6M.js
DELETED
|
@@ -1,139 +0,0 @@
|
|
|
1
|
-
import {
|
|
2
|
-
TOKEN_VARIABLE,
|
|
3
|
-
variable
|
|
4
|
-
} from "./chunk-YVRI7KRC.js";
|
|
5
|
-
|
|
6
|
-
// src/react/auth-method-config.ts
|
|
7
|
-
import { AuthTemplateSlug } from "@executor-js/sdk/shared";
|
|
8
|
-
var isVariablePart = (part) => typeof part !== "string" && part.type === "variable";
|
|
9
|
-
var parseTemplateValue = (value) => {
|
|
10
|
-
if (typeof value === "string") return { prefix: "", variable: TOKEN_VARIABLE };
|
|
11
|
-
const parts = [];
|
|
12
|
-
for (const part of value) {
|
|
13
|
-
if (isVariablePart(part)) {
|
|
14
|
-
return { prefix: parts.join(""), variable: part.name };
|
|
15
|
-
}
|
|
16
|
-
if (typeof part === "string") parts.push(part);
|
|
17
|
-
}
|
|
18
|
-
return { prefix: parts.join(""), variable: TOKEN_VARIABLE };
|
|
19
|
-
};
|
|
20
|
-
var placementsFromApiKey = (template) => {
|
|
21
|
-
const placements = [];
|
|
22
|
-
for (const [name, value] of Object.entries(template.headers ?? {})) {
|
|
23
|
-
const { prefix, variable: variable2 } = parseTemplateValue(value);
|
|
24
|
-
placements.push({ carrier: "header", name, prefix, variable: variable2 });
|
|
25
|
-
}
|
|
26
|
-
for (const [name, value] of Object.entries(template.queryParams ?? {})) {
|
|
27
|
-
const { prefix, variable: variable2 } = parseTemplateValue(value);
|
|
28
|
-
placements.push({ carrier: "query", name, prefix, variable: variable2 });
|
|
29
|
-
}
|
|
30
|
-
return placements;
|
|
31
|
-
};
|
|
32
|
-
var labelForApiKey = (slug, placements) => {
|
|
33
|
-
const first = placements[0];
|
|
34
|
-
if (first) return `API key (${first.name || (first.carrier === "header" ? "header" : "query")})`;
|
|
35
|
-
return `API key (${slug})`;
|
|
36
|
-
};
|
|
37
|
-
function authMethodsFromConfig(templates) {
|
|
38
|
-
return templates.map((template) => {
|
|
39
|
-
const slug = String(template.slug);
|
|
40
|
-
const source = slug.startsWith("custom_") ? "custom" : "spec";
|
|
41
|
-
if (template.type === "oauth") {
|
|
42
|
-
return {
|
|
43
|
-
id: slug,
|
|
44
|
-
label: "OAuth2",
|
|
45
|
-
kind: "oauth",
|
|
46
|
-
source,
|
|
47
|
-
template: AuthTemplateSlug.make(slug),
|
|
48
|
-
placements: [],
|
|
49
|
-
// Carry the integration's declared endpoints/scopes so the
|
|
50
|
-
// client-registration form pre-fills them.
|
|
51
|
-
oauth: {
|
|
52
|
-
authorizationUrl: template.authorizationUrl,
|
|
53
|
-
tokenUrl: template.tokenUrl,
|
|
54
|
-
scopes: template.scopes
|
|
55
|
-
}
|
|
56
|
-
};
|
|
57
|
-
}
|
|
58
|
-
const placements = placementsFromApiKey(template);
|
|
59
|
-
return {
|
|
60
|
-
id: slug,
|
|
61
|
-
label: labelForApiKey(slug, placements),
|
|
62
|
-
kind: "apikey",
|
|
63
|
-
source,
|
|
64
|
-
template: AuthTemplateSlug.make(slug),
|
|
65
|
-
placements
|
|
66
|
-
};
|
|
67
|
-
});
|
|
68
|
-
}
|
|
69
|
-
var slugifyVariable = (name) => name.toLowerCase().replace(/[^a-z0-9]+/g, "_").replace(/^_+|_+$/g, "");
|
|
70
|
-
var assignVariables = (placements) => {
|
|
71
|
-
const named = placements.filter((p) => p.name);
|
|
72
|
-
const out = /* @__PURE__ */ new Map();
|
|
73
|
-
if (named.length <= 1) {
|
|
74
|
-
for (const p of named) out.set(p, p.variable ?? TOKEN_VARIABLE);
|
|
75
|
-
return out;
|
|
76
|
-
}
|
|
77
|
-
const taken = /* @__PURE__ */ new Set();
|
|
78
|
-
for (const p of named) {
|
|
79
|
-
const base = slugifyVariable(p.name) || "input";
|
|
80
|
-
let candidate = p.variable ?? base;
|
|
81
|
-
let n = 2;
|
|
82
|
-
while (taken.has(candidate)) candidate = `${base}_${n++}`;
|
|
83
|
-
taken.add(candidate);
|
|
84
|
-
out.set(p, candidate);
|
|
85
|
-
}
|
|
86
|
-
return out;
|
|
87
|
-
};
|
|
88
|
-
var valueFromPlacement = (placement, varName) => placement.prefix ? [placement.prefix, variable(varName)] : [variable(varName)];
|
|
89
|
-
function templateFromPlacements(placements, slug) {
|
|
90
|
-
const variables = assignVariables(placements);
|
|
91
|
-
const headers = {};
|
|
92
|
-
const queryParams = {};
|
|
93
|
-
for (const placement of placements) {
|
|
94
|
-
if (!placement.name) continue;
|
|
95
|
-
const varName = variables.get(placement) ?? TOKEN_VARIABLE;
|
|
96
|
-
if (placement.carrier === "header") {
|
|
97
|
-
headers[placement.name] = valueFromPlacement(placement, varName);
|
|
98
|
-
} else {
|
|
99
|
-
queryParams[placement.name] = valueFromPlacement(placement, varName);
|
|
100
|
-
}
|
|
101
|
-
}
|
|
102
|
-
return {
|
|
103
|
-
slug: AuthTemplateSlug.make(slug ?? ""),
|
|
104
|
-
type: "apiKey",
|
|
105
|
-
...Object.keys(headers).length > 0 ? { headers } : {},
|
|
106
|
-
...Object.keys(queryParams).length > 0 ? { queryParams } : {}
|
|
107
|
-
};
|
|
108
|
-
}
|
|
109
|
-
function editorValueFromAuthentication(template) {
|
|
110
|
-
if (template.type === "oauth") {
|
|
111
|
-
return {
|
|
112
|
-
kind: "oauth",
|
|
113
|
-
authorizationUrl: template.authorizationUrl ?? "",
|
|
114
|
-
tokenUrl: template.tokenUrl ?? "",
|
|
115
|
-
scopes: template.scopes ?? []
|
|
116
|
-
};
|
|
117
|
-
}
|
|
118
|
-
return { kind: "apikey", placements: placementsFromApiKey(template) };
|
|
119
|
-
}
|
|
120
|
-
var oauthTemplateFromEditorValue = (value, slug) => ({
|
|
121
|
-
slug: AuthTemplateSlug.make(slug ?? ""),
|
|
122
|
-
type: "oauth",
|
|
123
|
-
authorizationUrl: value.authorizationUrl,
|
|
124
|
-
tokenUrl: value.tokenUrl,
|
|
125
|
-
scopes: [...value.scopes]
|
|
126
|
-
});
|
|
127
|
-
function authenticationFromEditorValue(value, slug) {
|
|
128
|
-
if (value.kind === "none") return null;
|
|
129
|
-
if (value.kind === "oauth") return oauthTemplateFromEditorValue(value, slug);
|
|
130
|
-
return templateFromPlacements(value.placements, slug);
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
export {
|
|
134
|
-
authMethodsFromConfig,
|
|
135
|
-
templateFromPlacements,
|
|
136
|
-
editorValueFromAuthentication,
|
|
137
|
-
authenticationFromEditorValue
|
|
138
|
-
};
|
|
139
|
-
//# sourceMappingURL=chunk-QSSRVK6M.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/react/auth-method-config.ts"],"sourcesContent":["// ---------------------------------------------------------------------------\n// OpenAPI ↔ generic auth-method converters.\n//\n// The generic Accounts hub speaks in plugin-agnostic `AuthMethod` /\n// `Placement` values (`@executor-js/react/lib/auth-placements`). The OpenAPI\n// plugin stores auth as the HTTP-transport-specific `APIKeyAuthentication`\n// template (header/query slots templated with `variable(\"token\")`). These two\n// converters bridge the wire template to/from the generic placement model, so\n// they live with the OpenAPI plugin — they touch the openapi sdk\n// `Authentication` types and would pull transport specifics into core.\n// ---------------------------------------------------------------------------\n\nimport { AuthTemplateSlug } from \"@executor-js/sdk/shared\";\nimport type { AuthMethod, Placement } from \"@executor-js/react/lib/auth-placements\";\nimport type { AuthTemplateEditorValue } from \"@executor-js/react/components/auth-template-editor\";\n\nimport {\n TOKEN_VARIABLE,\n variable,\n type APIKeyAuthentication,\n type Authentication,\n type AuthenticationTemplateValue,\n} from \"../sdk/types\";\n\n// ---------------------------------------------------------------------------\n// Template value → placement prefix.\n//\n// A header/query slot serializes the credential as `name -> [prefix, token]`,\n// where `prefix` is the leading literal string before the `variable(\"token\")`\n// part. A bare `[token]` (or string-only) value has an empty prefix.\n// ---------------------------------------------------------------------------\n\nconst isVariablePart = (part: string | { readonly type: \"variable\"; readonly name: string }) =>\n typeof part !== \"string\" && part.type === \"variable\";\n\n/** Extract the literal prefix preceding the credential variable, plus the\n * variable name that placement renders from (`token` for single-input methods,\n * a distinct name per input for multi-input ones). */\nconst parseTemplateValue = (\n value: AuthenticationTemplateValue,\n): { readonly prefix: string; readonly variable: string } => {\n if (typeof value === \"string\") return { prefix: \"\", variable: TOKEN_VARIABLE };\n const parts: string[] = [];\n for (const part of value) {\n if (isVariablePart(part)) {\n return { prefix: parts.join(\"\"), variable: (part as { readonly name: string }).name };\n }\n if (typeof part === \"string\") parts.push(part);\n }\n return { prefix: parts.join(\"\"), variable: TOKEN_VARIABLE };\n};\n\nexport const placementsFromApiKey = (template: APIKeyAuthentication): Placement[] => {\n const placements: Placement[] = [];\n for (const [name, value] of Object.entries(template.headers ?? {})) {\n const { prefix, variable } = parseTemplateValue(value);\n placements.push({ carrier: \"header\", name, prefix, variable });\n }\n for (const [name, value] of Object.entries(template.queryParams ?? {})) {\n const { prefix, variable } = parseTemplateValue(value);\n placements.push({ carrier: \"query\", name, prefix, variable });\n }\n return placements;\n};\n\n// ---------------------------------------------------------------------------\n// Templates → generic methods.\n// ---------------------------------------------------------------------------\n\nconst labelForApiKey = (slug: string, placements: readonly Placement[]): string => {\n const first = placements[0];\n if (first) return `API key (${first.name || (first.carrier === \"header\" ? \"header\" : \"query\")})`;\n return `API key (${slug})`;\n};\n\n/** Map each stored auth template to a generic `AuthMethod`. */\nexport function authMethodsFromConfig(templates: readonly Authentication[]): AuthMethod[] {\n return templates.map((template: Authentication): AuthMethod => {\n const slug = String(template.slug);\n const source: \"spec\" | \"custom\" = slug.startsWith(\"custom_\") ? \"custom\" : \"spec\";\n if (template.type === \"oauth\") {\n return {\n id: slug,\n label: \"OAuth2\",\n kind: \"oauth\",\n source,\n template: AuthTemplateSlug.make(slug),\n placements: [],\n // Carry the integration's declared endpoints/scopes so the\n // client-registration form pre-fills them.\n oauth: {\n authorizationUrl: template.authorizationUrl,\n tokenUrl: template.tokenUrl,\n scopes: template.scopes,\n },\n };\n }\n const placements = placementsFromApiKey(template);\n return {\n id: slug,\n label: labelForApiKey(slug, placements),\n kind: \"apikey\",\n source,\n template: AuthTemplateSlug.make(slug),\n placements,\n };\n });\n}\n\n// ---------------------------------------------------------------------------\n// Generic placements → apiKey template (inverse).\n//\n// Each placement becomes a header/query slot whose value is\n// `prefix ? [prefix, variable(\"token\")] : [variable(\"token\")]`. A custom method\n// may omit `slug`; the backend backfills `custom_<id>`.\n// ---------------------------------------------------------------------------\n\n/** Slugify a placement name into a variable identifier: `DD-API-KEY` →\n * `dd_api_key`. */\nconst slugifyVariable = (name: string): string =>\n name\n .toLowerCase()\n .replace(/[^a-z0-9]+/g, \"_\")\n .replace(/^_+|_+$/g, \"\");\n\n/** Assign an input variable to each (named) placement. A lone input is the\n * canonical `token`; multiple inputs each get their own distinct variable so a\n * connection can carry a different value per location. An explicit\n * `placement.variable` (from a round-trip) is honored. */\nconst assignVariables = (placements: readonly Placement[]): Map<Placement, string> => {\n const named = placements.filter((p) => p.name);\n const out = new Map<Placement, string>();\n if (named.length <= 1) {\n for (const p of named) out.set(p, p.variable ?? TOKEN_VARIABLE);\n return out;\n }\n const taken = new Set<string>();\n for (const p of named) {\n const base = slugifyVariable(p.name) || \"input\";\n let candidate = p.variable ?? base;\n let n = 2;\n while (taken.has(candidate)) candidate = `${base}_${n++}`;\n taken.add(candidate);\n out.set(p, candidate);\n }\n return out;\n};\n\nconst valueFromPlacement = (placement: Placement, varName: string): AuthenticationTemplateValue =>\n placement.prefix ? [placement.prefix, variable(varName)] : [variable(varName)];\n\n/** Build an `APIKeyAuthentication` template from generic placements. When\n * `slug` is omitted the backend assigns a `custom_<id>` slug. */\nexport function templateFromPlacements(\n placements: readonly Placement[],\n slug?: string,\n): APIKeyAuthentication {\n const variables = assignVariables(placements);\n const headers: Record<string, AuthenticationTemplateValue> = {};\n const queryParams: Record<string, AuthenticationTemplateValue> = {};\n for (const placement of placements) {\n if (!placement.name) continue;\n const varName = variables.get(placement) ?? TOKEN_VARIABLE;\n if (placement.carrier === \"header\") {\n headers[placement.name] = valueFromPlacement(placement, varName);\n } else {\n queryParams[placement.name] = valueFromPlacement(placement, varName);\n }\n }\n return {\n slug: AuthTemplateSlug.make(slug ?? \"\"),\n type: \"apiKey\",\n ...(Object.keys(headers).length > 0 ? { headers } : {}),\n ...(Object.keys(queryParams).length > 0 ? { queryParams } : {}),\n };\n}\n\n// ---------------------------------------------------------------------------\n// Stored `Authentication` ⇆ generic `AuthTemplateEditorValue`.\n//\n// The shared add-time `AuthTemplateEditor` edits a plugin-agnostic value; these\n// bridge it to/from the OpenAPI wire template so a spec-detected method seeds an\n// editable default and the user's edits round-trip back to `Authentication[]`\n// on submit. apiKey → placements; oauth → endpoints + scopes.\n// ---------------------------------------------------------------------------\n\n/** Convert one stored `Authentication` template into a generic editor value. */\nexport function editorValueFromAuthentication(template: Authentication): AuthTemplateEditorValue {\n if (template.type === \"oauth\") {\n return {\n kind: \"oauth\",\n authorizationUrl: template.authorizationUrl ?? \"\",\n tokenUrl: template.tokenUrl ?? \"\",\n scopes: template.scopes ?? [],\n };\n }\n return { kind: \"apikey\", placements: placementsFromApiKey(template) };\n}\n\n/** Build an `OAuthAuthentication` template from a generic oauth editor value. */\nconst oauthTemplateFromEditorValue = (\n value: Extract<AuthTemplateEditorValue, { kind: \"oauth\" }>,\n slug?: string,\n): Authentication => ({\n slug: AuthTemplateSlug.make(slug ?? \"\"),\n type: \"oauth\",\n authorizationUrl: value.authorizationUrl,\n tokenUrl: value.tokenUrl,\n scopes: [...value.scopes],\n});\n\n/** Convert one generic editor value back into a stored `Authentication`, or\n * `null` for `none` (no method to register). The optional `slug` names the\n * template; when omitted the backend backfills `custom_<id>`. */\nexport function authenticationFromEditorValue(\n value: AuthTemplateEditorValue,\n slug?: string,\n): Authentication | null {\n if (value.kind === \"none\") return null;\n if (value.kind === \"oauth\") return oauthTemplateFromEditorValue(value, slug);\n return templateFromPlacements(value.placements, slug);\n}\n"],"mappings":"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","names":["variable"]}
|