forgeos 0.1.0-alpha.0
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/.npmignore +1 -0
- package/AGENTS.md +277 -0
- package/CHANGELOG.md +8 -0
- package/CONTRIBUTING.md +58 -0
- package/README.md +377 -0
- package/bin/forge-bun.mjs +110 -0
- package/bin/forge.mjs +19 -0
- package/package.json +96 -0
- package/packages/eslint-plugin-forge/index.ts +15 -0
- package/packages/eslint-plugin-forge/package.json +10 -0
- package/packages/eslint-plugin-forge/src/check-source.ts +95 -0
- package/packages/eslint-plugin-forge/src/load-artifacts.ts +24 -0
- package/packages/eslint-plugin-forge/src/rule-no-forge-guard-violation.ts +93 -0
- package/src/forge/_generated/actionSubscriptions.json +2 -0
- package/src/forge/_generated/actionSubscriptions.ts +10 -0
- package/src/forge/_generated/agentAdapterManifest.json +2 -0
- package/src/forge/_generated/agentAdapterManifest.ts +73 -0
- package/src/forge/_generated/agentContract.json +2 -0
- package/src/forge/_generated/agentContract.ts +912 -0
- package/src/forge/_generated/agentQuickstart.md +32 -0
- package/src/forge/_generated/aiContext.ts +59 -0
- package/src/forge/_generated/aiModels.json +2 -0
- package/src/forge/_generated/aiModels.ts +35 -0
- package/src/forge/_generated/aiProviders.json +2 -0
- package/src/forge/_generated/aiProviders.ts +23 -0
- package/src/forge/_generated/aiRegistry.json +2 -0
- package/src/forge/_generated/aiRegistry.ts +29 -0
- package/src/forge/_generated/api.json +2 -0
- package/src/forge/_generated/api.ts +8 -0
- package/src/forge/_generated/appGraph.json +2 -0
- package/src/forge/_generated/appGraph.ts +14511 -0
- package/src/forge/_generated/appMap.md +35 -0
- package/src/forge/_generated/artifactManifest.json +2 -0
- package/src/forge/_generated/artifactManifest.ts +7 -0
- package/src/forge/_generated/authClaims.json +2 -0
- package/src/forge/_generated/authClaims.ts +13 -0
- package/src/forge/_generated/authConfig.json +2 -0
- package/src/forge/_generated/authConfig.ts +17 -0
- package/src/forge/_generated/authContext.ts +23 -0
- package/src/forge/_generated/authRegistry.json +2 -0
- package/src/forge/_generated/authRegistry.ts +25 -0
- package/src/forge/_generated/buildInfo.json +2 -0
- package/src/forge/_generated/buildInfo.ts +9 -0
- package/src/forge/_generated/capabilityMap.json +2 -0
- package/src/forge/_generated/capabilityMap.md +15 -0
- package/src/forge/_generated/capabilityMap.ts +17 -0
- package/src/forge/_generated/client.ts +282 -0
- package/src/forge/_generated/clientApi.ts +9 -0
- package/src/forge/_generated/clientManifest.json +2 -0
- package/src/forge/_generated/clientManifest.ts +39 -0
- package/src/forge/_generated/clientTypes.ts +78 -0
- package/src/forge/_generated/configRegistry.json +2 -0
- package/src/forge/_generated/configRegistry.ts +4 -0
- package/src/forge/_generated/dataGraph.json +2 -0
- package/src/forge/_generated/dataGraph.ts +8 -0
- package/src/forge/_generated/db.json +2 -0
- package/src/forge/_generated/db.ts +2 -0
- package/src/forge/_generated/dbSecurityManifest.json +2 -0
- package/src/forge/_generated/dbSecurityManifest.ts +15 -0
- package/src/forge/_generated/dbSessionContext.json +2 -0
- package/src/forge/_generated/dbSessionContext.ts +39 -0
- package/src/forge/_generated/deployManifest.json +2 -0
- package/src/forge/_generated/deployManifest.ts +14 -0
- package/src/forge/_generated/devManifest.json +2 -0
- package/src/forge/_generated/devManifest.ts +47 -0
- package/src/forge/_generated/envSchema.json +2 -0
- package/src/forge/_generated/envSchema.ts +59 -0
- package/src/forge/_generated/frontendGraph.json +2 -0
- package/src/forge/_generated/frontendGraph.ts +27 -0
- package/src/forge/_generated/importGuards.json +2 -0
- package/src/forge/_generated/importGuards.ts +652 -0
- package/src/forge/_generated/index.ts +67 -0
- package/src/forge/_generated/liveProductionManifest.json +2 -0
- package/src/forge/_generated/liveProductionManifest.ts +23 -0
- package/src/forge/_generated/liveProtocol.json +2 -0
- package/src/forge/_generated/liveProtocol.ts +21 -0
- package/src/forge/_generated/liveQueryRegistry.json +2 -0
- package/src/forge/_generated/liveQueryRegistry.ts +9 -0
- package/src/forge/_generated/liveTransportConfig.json +2 -0
- package/src/forge/_generated/liveTransportConfig.ts +19 -0
- package/src/forge/_generated/makeRegistry.json +2 -0
- package/src/forge/_generated/makeRegistry.ts +163 -0
- package/src/forge/_generated/makeTemplates.json +2 -0
- package/src/forge/_generated/makeTemplates.ts +61 -0
- package/src/forge/_generated/mockMap.json +2 -0
- package/src/forge/_generated/mockMap.ts +7 -0
- package/src/forge/_generated/operationPlaybooks.md +145 -0
- package/src/forge/_generated/packageGraph.json +2 -0
- package/src/forge/_generated/packageGraph.ts +168569 -0
- package/src/forge/_generated/packageUpgradeRegistry.json +2 -0
- package/src/forge/_generated/packageUpgradeRegistry.ts +15 -0
- package/src/forge/_generated/permissionMatrix.json +2 -0
- package/src/forge/_generated/permissionMatrix.ts +7 -0
- package/src/forge/_generated/policyRegistry.json +2 -0
- package/src/forge/_generated/policyRegistry.ts +11 -0
- package/src/forge/_generated/queryRegistry.json +2 -0
- package/src/forge/_generated/queryRegistry.ts +9 -0
- package/src/forge/_generated/react.d.ts +22 -0
- package/src/forge/_generated/react.ts +29 -0
- package/src/forge/_generated/reactManifest.json +2 -0
- package/src/forge/_generated/reactManifest.ts +19 -0
- package/src/forge/_generated/releaseManifest.json +2 -0
- package/src/forge/_generated/releaseManifest.ts +25 -0
- package/src/forge/_generated/rlsPolicies.json +2 -0
- package/src/forge/_generated/rlsPolicies.sql +34 -0
- package/src/forge/_generated/rlsPolicies.ts +6 -0
- package/src/forge/_generated/runtimeGraph.json +2 -0
- package/src/forge/_generated/runtimeGraph.ts +8 -0
- package/src/forge/_generated/runtimeMatrix.json +2 -0
- package/src/forge/_generated/runtimeMatrix.ts +229125 -0
- package/src/forge/_generated/runtimeRegistry.ts +2 -0
- package/src/forge/_generated/runtimeRules.md +79 -0
- package/src/forge/_generated/secretRegistry.json +2 -0
- package/src/forge/_generated/secretRegistry.ts +50 -0
- package/src/forge/_generated/secretsContext.ts +11 -0
- package/src/forge/_generated/serverApi.ts +10 -0
- package/src/forge/_generated/sourceMapManifest.json +2 -0
- package/src/forge/_generated/sourceMapManifest.ts +7 -0
- package/src/forge/_generated/sqlPlan.json +2 -0
- package/src/forge/_generated/sqlPlan.ts +88 -0
- package/src/forge/_generated/subscriptionManifest.json +2 -0
- package/src/forge/_generated/subscriptionManifest.ts +7 -0
- package/src/forge/_generated/symbolicationManifest.json +2 -0
- package/src/forge/_generated/symbolicationManifest.ts +17 -0
- package/src/forge/_generated/telemetryRegistry.json +2 -0
- package/src/forge/_generated/telemetryRegistry.ts +9 -0
- package/src/forge/_generated/telemetrySinks.json +2 -0
- package/src/forge/_generated/telemetrySinks.ts +11 -0
- package/src/forge/_generated/tenantScope.json +2 -0
- package/src/forge/_generated/tenantScope.ts +8 -0
- package/src/forge/_generated/testGraph.json +2 -0
- package/src/forge/_generated/testGraph.ts +3054 -0
- package/src/forge/_generated/testPlanRegistry.json +2 -0
- package/src/forge/_generated/testPlanRegistry.ts +33 -0
- package/src/forge/_generated/uiRoutes.json +2 -0
- package/src/forge/_generated/uiRoutes.ts +16 -0
- package/src/forge/_generated/uiScenarios.json +2 -0
- package/src/forge/_generated/uiScenarios.ts +30 -0
- package/src/forge/_generated/uiTestManifest.json +2 -0
- package/src/forge/_generated/uiTestManifest.ts +27 -0
- package/src/forge/_generated/workflowRegistry.json +2 -0
- package/src/forge/_generated/workflowRegistry.ts +9 -0
- package/src/forge/_generated/workflowSubscriptions.json +2 -0
- package/src/forge/_generated/workflowSubscriptions.ts +10 -0
- package/src/forge/agent-adapters/index.ts +1002 -0
- package/src/forge/agent-adapters/types.ts +135 -0
- package/src/forge/cli/agent-contract.ts +50 -0
- package/src/forge/cli/ai.ts +148 -0
- package/src/forge/cli/auth.ts +198 -0
- package/src/forge/cli/build.ts +105 -0
- package/src/forge/cli/bun-exec.ts +4 -0
- package/src/forge/cli/commands.ts +1130 -0
- package/src/forge/cli/db.ts +316 -0
- package/src/forge/cli/deps.ts +277 -0
- package/src/forge/cli/dev.ts +529 -0
- package/src/forge/cli/doctor.ts +209 -0
- package/src/forge/cli/feature.ts +485 -0
- package/src/forge/cli/index.ts +25 -0
- package/src/forge/cli/lint-forge.ts +119 -0
- package/src/forge/cli/live.ts +179 -0
- package/src/forge/cli/main.ts +92 -0
- package/src/forge/cli/make.ts +133 -0
- package/src/forge/cli/new.ts +505 -0
- package/src/forge/cli/outbox.ts +297 -0
- package/src/forge/cli/output.ts +114 -0
- package/src/forge/cli/parse.ts +2211 -0
- package/src/forge/cli/policy.ts +204 -0
- package/src/forge/cli/query.ts +91 -0
- package/src/forge/cli/refactor.ts +221 -0
- package/src/forge/cli/release.ts +285 -0
- package/src/forge/cli/rls.ts +322 -0
- package/src/forge/cli/run.ts +76 -0
- package/src/forge/cli/secrets.ts +274 -0
- package/src/forge/cli/self-host.ts +468 -0
- package/src/forge/cli/serve.ts +93 -0
- package/src/forge/cli/telemetry.ts +219 -0
- package/src/forge/cli/verify.ts +587 -0
- package/src/forge/cli/version.ts +1 -0
- package/src/forge/cli/windows.ts +413 -0
- package/src/forge/cli/worker.ts +87 -0
- package/src/forge/cli/workflow.ts +424 -0
- package/src/forge/compiler/action-subscriptions/build.ts +116 -0
- package/src/forge/compiler/action-subscriptions/constants.ts +2 -0
- package/src/forge/compiler/action-subscriptions/index.ts +6 -0
- package/src/forge/compiler/action-subscriptions/parse.ts +6 -0
- package/src/forge/compiler/agent-contract/build.ts +1651 -0
- package/src/forge/compiler/agent-contract/types.ts +326 -0
- package/src/forge/compiler/ai-registry/build.ts +165 -0
- package/src/forge/compiler/ai-registry/constants.ts +2 -0
- package/src/forge/compiler/ai-registry/parse.ts +56 -0
- package/src/forge/compiler/api-surface/build.ts +107 -0
- package/src/forge/compiler/app-graph/build.ts +121 -0
- package/src/forge/compiler/app-graph/classify.ts +10 -0
- package/src/forge/compiler/app-graph/dup-symbol.ts +29 -0
- package/src/forge/compiler/app-graph/extract.ts +124 -0
- package/src/forge/compiler/app-graph/forge-apis.ts +29 -0
- package/src/forge/compiler/app-graph/index.ts +15 -0
- package/src/forge/compiler/app-graph/module-graph.ts +320 -0
- package/src/forge/compiler/app-graph/parser.ts +119 -0
- package/src/forge/compiler/app-graph/symbols.ts +48 -0
- package/src/forge/compiler/app-graph/tsconfig-hash.ts +62 -0
- package/src/forge/compiler/app-graph/types.ts +43 -0
- package/src/forge/compiler/app-graph/versions.ts +14 -0
- package/src/forge/compiler/cache/index.ts +17 -0
- package/src/forge/compiler/cache/key.ts +46 -0
- package/src/forge/compiler/cache/scheduler.ts +72 -0
- package/src/forge/compiler/cache/store.ts +78 -0
- package/src/forge/compiler/classifier/capabilities.ts +78 -0
- package/src/forge/compiler/classifier/classify.ts +113 -0
- package/src/forge/compiler/classifier/contexts.ts +188 -0
- package/src/forge/compiler/classifier/index.ts +18 -0
- package/src/forge/compiler/classifier/runtime-matrix.ts +45 -0
- package/src/forge/compiler/classifier/secrets.ts +41 -0
- package/src/forge/compiler/classifier/signals.ts +129 -0
- package/src/forge/compiler/client-sdk/build-manifest.ts +151 -0
- package/src/forge/compiler/client-sdk/render-client.ts +432 -0
- package/src/forge/compiler/data-graph/build.ts +131 -0
- package/src/forge/compiler/data-graph/constants.ts +5 -0
- package/src/forge/compiler/data-graph/index.ts +6 -0
- package/src/forge/compiler/data-graph/parse.ts +176 -0
- package/src/forge/compiler/data-graph/rls/build.ts +222 -0
- package/src/forge/compiler/data-graph/rls/types.ts +62 -0
- package/src/forge/compiler/data-graph/sql/ddl.ts +390 -0
- package/src/forge/compiler/data-graph/sql/naming.ts +10 -0
- package/src/forge/compiler/data-graph/sql/serialize.ts +85 -0
- package/src/forge/compiler/data-graph/sql/types.ts +37 -0
- package/src/forge/compiler/dev-manifest/build.ts +170 -0
- package/src/forge/compiler/dev-manifest/constants.ts +5 -0
- package/src/forge/compiler/diagnostics/codes.ts +611 -0
- package/src/forge/compiler/diagnostics/create.ts +245 -0
- package/src/forge/compiler/diagnostics/index.ts +55 -0
- package/src/forge/compiler/emitter/artifact-kind.ts +14 -0
- package/src/forge/compiler/emitter/barrel.ts +44 -0
- package/src/forge/compiler/emitter/constants.ts +7 -0
- package/src/forge/compiler/emitter/emit.ts +237 -0
- package/src/forge/compiler/emitter/index.ts +24 -0
- package/src/forge/compiler/emitter/lock.ts +62 -0
- package/src/forge/compiler/emitter/render.ts +73 -0
- package/src/forge/compiler/emitter/write.ts +35 -0
- package/src/forge/compiler/frontend-graph/build.ts +495 -0
- package/src/forge/compiler/fs/index.ts +23 -0
- package/src/forge/compiler/fs/memory.ts +233 -0
- package/src/forge/compiler/fs/node.ts +139 -0
- package/src/forge/compiler/fs/profile.ts +108 -0
- package/src/forge/compiler/fs/types.ts +52 -0
- package/src/forge/compiler/guards/artifacts.ts +96 -0
- package/src/forge/compiler/guards/check-ai-usage.ts +98 -0
- package/src/forge/compiler/guards/check-import-guards.ts +106 -0
- package/src/forge/compiler/guards/check-process-env.ts +98 -0
- package/src/forge/compiler/guards/check-query-usage.ts +76 -0
- package/src/forge/compiler/guards/index.ts +11 -0
- package/src/forge/compiler/guards/propagate-contexts.ts +57 -0
- package/src/forge/compiler/index.ts +17 -0
- package/src/forge/compiler/integration/add.ts +496 -0
- package/src/forge/compiler/integration/index.ts +17 -0
- package/src/forge/compiler/integration/plan.ts +283 -0
- package/src/forge/compiler/integration/render.ts +189 -0
- package/src/forge/compiler/integration/snapshot.ts +52 -0
- package/src/forge/compiler/integration/templates/ai.ts +131 -0
- package/src/forge/compiler/integration/templates/index.ts +8 -0
- package/src/forge/compiler/integration/templates/posthog.ts +145 -0
- package/src/forge/compiler/integration/templates/render.ts +113 -0
- package/src/forge/compiler/integration/templates/sentry.ts +151 -0
- package/src/forge/compiler/integration/templates/stripe.ts +109 -0
- package/src/forge/compiler/integration/templates/types.ts +14 -0
- package/src/forge/compiler/integration/templates/zod.ts +55 -0
- package/src/forge/compiler/live-production/types.ts +122 -0
- package/src/forge/compiler/live-query-registry/build.ts +150 -0
- package/src/forge/compiler/live-query-registry/constants.ts +2 -0
- package/src/forge/compiler/make-registry/build.ts +179 -0
- package/src/forge/compiler/orchestrator/discover.ts +214 -0
- package/src/forge/compiler/orchestrator/fast-check.ts +117 -0
- package/src/forge/compiler/orchestrator/generate-lock.ts +138 -0
- package/src/forge/compiler/orchestrator/guards.ts +5 -0
- package/src/forge/compiler/orchestrator/index.ts +27 -0
- package/src/forge/compiler/orchestrator/manifest-hashes.ts +21 -0
- package/src/forge/compiler/orchestrator/manifest.ts +92 -0
- package/src/forge/compiler/orchestrator/orphans.ts +51 -0
- package/src/forge/compiler/orchestrator/plan.ts +876 -0
- package/src/forge/compiler/orchestrator/profile.ts +36 -0
- package/src/forge/compiler/orchestrator/run.ts +277 -0
- package/src/forge/compiler/orchestrator/serialize.ts +886 -0
- package/src/forge/compiler/orchestrator/session.ts +96 -0
- package/src/forge/compiler/orchestrator/types.ts +31 -0
- package/src/forge/compiler/orchestrator/verify.ts +38 -0
- package/src/forge/compiler/orchestrator/workspace-index.ts +154 -0
- package/src/forge/compiler/package-graph/capabilities-stub.ts +33 -0
- package/src/forge/compiler/package-graph/checksum.ts +97 -0
- package/src/forge/compiler/package-graph/compiler.ts +392 -0
- package/src/forge/compiler/package-graph/constants.ts +4 -0
- package/src/forge/compiler/package-graph/dts-extractor.ts +142 -0
- package/src/forge/compiler/package-graph/exports-discovery.ts +84 -0
- package/src/forge/compiler/package-graph/extract-dts.ts +32 -0
- package/src/forge/compiler/package-graph/index.ts +33 -0
- package/src/forge/compiler/package-graph/jsdoc.ts +62 -0
- package/src/forge/compiler/package-graph/read-file.ts +21 -0
- package/src/forge/compiler/package-graph/resolve.ts +127 -0
- package/src/forge/compiler/package-manager/adapter.ts +237 -0
- package/src/forge/compiler/package-manager/bun-executable.ts +92 -0
- package/src/forge/compiler/package-manager/commands.ts +47 -0
- package/src/forge/compiler/package-manager/detect.ts +79 -0
- package/src/forge/compiler/package-manager/executor.ts +117 -0
- package/src/forge/compiler/package-manager/index.ts +22 -0
- package/src/forge/compiler/package-manager/parse-spec.ts +16 -0
- package/src/forge/compiler/package-manager/version.ts +27 -0
- package/src/forge/compiler/package-upgrades/apply.ts +195 -0
- package/src/forge/compiler/package-upgrades/comparator.ts +181 -0
- package/src/forge/compiler/package-upgrades/impact.ts +139 -0
- package/src/forge/compiler/package-upgrades/markdown.ts +97 -0
- package/src/forge/compiler/package-upgrades/planner.ts +532 -0
- package/src/forge/compiler/package-upgrades/risk.ts +208 -0
- package/src/forge/compiler/package-upgrades/types.ts +174 -0
- package/src/forge/compiler/policy-registry/build.ts +266 -0
- package/src/forge/compiler/policy-registry/constants.ts +2 -0
- package/src/forge/compiler/policy-registry/parse.ts +81 -0
- package/src/forge/compiler/primitives/compare.ts +26 -0
- package/src/forge/compiler/primitives/hash.ts +40 -0
- package/src/forge/compiler/primitives/header.ts +45 -0
- package/src/forge/compiler/primitives/index.ts +45 -0
- package/src/forge/compiler/primitives/paths.ts +24 -0
- package/src/forge/compiler/primitives/result.ts +164 -0
- package/src/forge/compiler/primitives/serialize.ts +66 -0
- package/src/forge/compiler/primitives/sort.ts +87 -0
- package/src/forge/compiler/query-registry/build.ts +114 -0
- package/src/forge/compiler/query-registry/constants.ts +2 -0
- package/src/forge/compiler/recipes/definitions.ts +289 -0
- package/src/forge/compiler/recipes/helpers.ts +37 -0
- package/src/forge/compiler/recipes/index.ts +21 -0
- package/src/forge/compiler/recipes/registry.ts +102 -0
- package/src/forge/compiler/release/build.ts +100 -0
- package/src/forge/compiler/release/types.ts +119 -0
- package/src/forge/compiler/runtime-graph/build.ts +137 -0
- package/src/forge/compiler/runtime-graph/constants.ts +5 -0
- package/src/forge/compiler/runtime-graph/index.ts +5 -0
- package/src/forge/compiler/sandbox/artifact-sanitize.ts +26 -0
- package/src/forge/compiler/sandbox/backends/child.ts +123 -0
- package/src/forge/compiler/sandbox/backends/docker.ts +173 -0
- package/src/forge/compiler/sandbox/index.ts +51 -0
- package/src/forge/compiler/sandbox/inspect.ts +143 -0
- package/src/forge/compiler/sandbox/inspector-entry.ts +115 -0
- package/src/forge/compiler/sandbox/limits.ts +31 -0
- package/src/forge/compiler/sandbox/scrub-env.ts +60 -0
- package/src/forge/compiler/sandbox/secret-scan.ts +54 -0
- package/src/forge/compiler/sandbox/serialize.ts +106 -0
- package/src/forge/compiler/sandbox/types.ts +7 -0
- package/src/forge/compiler/secret-registry/build.ts +123 -0
- package/src/forge/compiler/telemetry-registry/build.ts +89 -0
- package/src/forge/compiler/telemetry-registry/constants.ts +2 -0
- package/src/forge/compiler/telemetry-registry/parse.ts +13 -0
- package/src/forge/compiler/test-graph/build.ts +277 -0
- package/src/forge/compiler/types/action-subscriptions.ts +19 -0
- package/src/forge/compiler/types/ai-registry.ts +33 -0
- package/src/forge/compiler/types/app-graph.ts +80 -0
- package/src/forge/compiler/types/capability.ts +29 -0
- package/src/forge/compiler/types/classification.ts +9 -0
- package/src/forge/compiler/types/cli.ts +159 -0
- package/src/forge/compiler/types/data-graph.ts +24 -0
- package/src/forge/compiler/types/dev-manifest.ts +41 -0
- package/src/forge/compiler/types/diagnostic.ts +12 -0
- package/src/forge/compiler/types/emit.ts +25 -0
- package/src/forge/compiler/types/frontend-graph.ts +81 -0
- package/src/forge/compiler/types/import-guards.ts +19 -0
- package/src/forge/compiler/types/index.ts +98 -0
- package/src/forge/compiler/types/integration.ts +25 -0
- package/src/forge/compiler/types/json.ts +3 -0
- package/src/forge/compiler/types/live-query-registry.ts +32 -0
- package/src/forge/compiler/types/lock.ts +37 -0
- package/src/forge/compiler/types/package-graph.ts +84 -0
- package/src/forge/compiler/types/policy-registry.ts +69 -0
- package/src/forge/compiler/types/query-registry.ts +18 -0
- package/src/forge/compiler/types/runtime-graph.ts +30 -0
- package/src/forge/compiler/types/runtime-matrix.ts +16 -0
- package/src/forge/compiler/types/runtime.ts +30 -0
- package/src/forge/compiler/types/sandbox.ts +24 -0
- package/src/forge/compiler/types/secret-registry.ts +38 -0
- package/src/forge/compiler/types/telemetry-registry.ts +26 -0
- package/src/forge/compiler/types/test-graph.ts +45 -0
- package/src/forge/compiler/types/workflow-registry.ts +42 -0
- package/src/forge/compiler/workflow-registry/build.ts +180 -0
- package/src/forge/compiler/workflow-registry/constants.ts +2 -0
- package/src/forge/compiler/workflow-registry/index.ts +5 -0
- package/src/forge/compiler/workflow-registry/parse.ts +19 -0
- package/src/forge/dev/server.ts +1379 -0
- package/src/forge/dev/types.ts +49 -0
- package/src/forge/dev/watch.ts +109 -0
- package/src/forge/dev-console/cycle.ts +652 -0
- package/src/forge/dev-console/types.ts +99 -0
- package/src/forge/feature/compiler.ts +656 -0
- package/src/forge/feature/examples.ts +125 -0
- package/src/forge/feature/types.ts +177 -0
- package/src/forge/impact/index.ts +1160 -0
- package/src/forge/impact/types.ts +151 -0
- package/src/forge/intent/index.ts +490 -0
- package/src/forge/intent/types.ts +73 -0
- package/src/forge/make/fields.ts +146 -0
- package/src/forge/make/index.ts +1101 -0
- package/src/forge/make/naming.ts +42 -0
- package/src/forge/make/templates.ts +525 -0
- package/src/forge/make/types.ts +151 -0
- package/src/forge/platform/module.ts +20 -0
- package/src/forge/policy.ts +1 -0
- package/src/forge/react/index.ts +418 -0
- package/src/forge/refactor/index.ts +1936 -0
- package/src/forge/refactor/text-utils.ts +34 -0
- package/src/forge/refactor/types.ts +191 -0
- package/src/forge/refactor/workspace-fs.ts +171 -0
- package/src/forge/repair/index.ts +656 -0
- package/src/forge/repair/rules/index.ts +476 -0
- package/src/forge/repair/types.ts +175 -0
- package/src/forge/review/index.ts +992 -0
- package/src/forge/review/types.ts +196 -0
- package/src/forge/runtime/ai/check.ts +86 -0
- package/src/forge/runtime/ai/context.ts +394 -0
- package/src/forge/runtime/ai/cost-estimator.ts +41 -0
- package/src/forge/runtime/ai/mock.ts +49 -0
- package/src/forge/runtime/ai/providers.ts +78 -0
- package/src/forge/runtime/ai/state.ts +17 -0
- package/src/forge/runtime/ai/types.ts +67 -0
- package/src/forge/runtime/auth/authenticate.ts +58 -0
- package/src/forge/runtime/auth/claims.ts +119 -0
- package/src/forge/runtime/auth/config.ts +148 -0
- package/src/forge/runtime/auth/errors.ts +45 -0
- package/src/forge/runtime/auth/evaluate.ts +126 -0
- package/src/forge/runtime/auth/resolve.ts +74 -0
- package/src/forge/runtime/auth/types.ts +87 -0
- package/src/forge/runtime/auth/verifier.ts +138 -0
- package/src/forge/runtime/context/create-context.ts +204 -0
- package/src/forge/runtime/context/create-query-context.ts +34 -0
- package/src/forge/runtime/db/adapter.ts +31 -0
- package/src/forge/runtime/db/factory.ts +83 -0
- package/src/forge/runtime/db/generated-client.ts +294 -0
- package/src/forge/runtime/db/memory-adapter.ts +706 -0
- package/src/forge/runtime/db/migrate.ts +132 -0
- package/src/forge/runtime/db/outbox.ts +54 -0
- package/src/forge/runtime/db/pglite-adapter.ts +51 -0
- package/src/forge/runtime/db/postgres-adapter.ts +112 -0
- package/src/forge/runtime/db/read-only-client.ts +97 -0
- package/src/forge/runtime/db/session-context.ts +62 -0
- package/src/forge/runtime/executor.ts +446 -0
- package/src/forge/runtime/live/dependency-tracker.ts +57 -0
- package/src/forge/runtime/live/invalidation-log.ts +189 -0
- package/src/forge/runtime/live/live-query-runner.ts +267 -0
- package/src/forge/runtime/live/registry.ts +28 -0
- package/src/forge/runtime/live/sse.ts +75 -0
- package/src/forge/runtime/live/subscription-manager.ts +443 -0
- package/src/forge/runtime/live/types.ts +143 -0
- package/src/forge/runtime/outbox/claim.ts +153 -0
- package/src/forge/runtime/outbox/process.ts +298 -0
- package/src/forge/runtime/outbox/retry.ts +8 -0
- package/src/forge/runtime/outbox/subscriptions.ts +33 -0
- package/src/forge/runtime/outbox/types.ts +69 -0
- package/src/forge/runtime/policy/check.ts +157 -0
- package/src/forge/runtime/policy/load.ts +55 -0
- package/src/forge/runtime/query/registry.ts +19 -0
- package/src/forge/runtime/query/run-query.ts +347 -0
- package/src/forge/runtime/release/runtime.ts +322 -0
- package/src/forge/runtime/release/symbolicate.ts +175 -0
- package/src/forge/runtime/runner/command-transaction.ts +193 -0
- package/src/forge/runtime/runner/run-entry.ts +226 -0
- package/src/forge/runtime/secrets/check.ts +78 -0
- package/src/forge/runtime/secrets/create-context.ts +138 -0
- package/src/forge/runtime/secrets/env-loader.ts +94 -0
- package/src/forge/runtime/secrets/runtime-bundle.ts +47 -0
- package/src/forge/runtime/secrets/types.ts +31 -0
- package/src/forge/runtime/telemetry/buffer.ts +87 -0
- package/src/forge/runtime/telemetry/context.ts +192 -0
- package/src/forge/runtime/telemetry/correlation.ts +13 -0
- package/src/forge/runtime/telemetry/flush.ts +190 -0
- package/src/forge/runtime/telemetry/process.ts +20 -0
- package/src/forge/runtime/telemetry/scrubber.ts +115 -0
- package/src/forge/runtime/telemetry/sinks/local-jsonl.ts +39 -0
- package/src/forge/runtime/telemetry/sinks/posthog.ts +64 -0
- package/src/forge/runtime/telemetry/sinks/sentry.ts +60 -0
- package/src/forge/runtime/telemetry/spans.ts +58 -0
- package/src/forge/runtime/telemetry/types.ts +64 -0
- package/src/forge/runtime/workflows/cancel.ts +26 -0
- package/src/forge/runtime/workflows/create-run.ts +98 -0
- package/src/forge/runtime/workflows/process-run.ts +182 -0
- package/src/forge/runtime/workflows/process-step.ts +190 -0
- package/src/forge/runtime/workflows/process.ts +260 -0
- package/src/forge/runtime/workflows/registry.ts +51 -0
- package/src/forge/runtime/workflows/resolve-step.ts +46 -0
- package/src/forge/runtime/workflows/retry-run.ts +44 -0
- package/src/forge/runtime/workflows/retry.ts +8 -0
- package/src/forge/runtime/workflows/sanitize.ts +19 -0
- package/src/forge/runtime/workflows/start-from-outbox.ts +71 -0
- package/src/forge/runtime/workflows/types.ts +77 -0
- package/src/forge/server.ts +96 -0
- package/src/forge/ui/index.ts +770 -0
- package/src/forge/ui/types.ts +191 -0
- package/templates/b2b-support-web/.env.example +22 -0
- package/templates/b2b-support-web/.vscode/settings.json +14 -0
- package/templates/b2b-support-web/AGENTS.md +108 -0
- package/templates/b2b-support-web/README.md +48 -0
- package/templates/b2b-support-web/forge.config.ts +3 -0
- package/templates/b2b-support-web/package.json +34 -0
- package/templates/b2b-support-web/src/actions/captureTicketCreated.ts +14 -0
- package/templates/b2b-support-web/src/commands/closeTicket.ts +20 -0
- package/templates/b2b-support-web/src/commands/createTicket.ts +47 -0
- package/templates/b2b-support-web/src/commands/manageBilling.ts +9 -0
- package/templates/b2b-support-web/src/forge/schema.ts +35 -0
- package/templates/b2b-support-web/src/policies.ts +9 -0
- package/templates/b2b-support-web/src/queries/getTicket.ts +6 -0
- package/templates/b2b-support-web/src/queries/listTickets.ts +6 -0
- package/templates/b2b-support-web/src/queries/liveTickets.ts +9 -0
- package/templates/b2b-support-web/src/workflows/triageTicketWorkflow.ts +64 -0
- package/templates/b2b-support-web/tsconfig.json +14 -0
- package/templates/b2b-support-web/web/app/globals.css +77 -0
- package/templates/b2b-support-web/web/app/layout.tsx +13 -0
- package/templates/b2b-support-web/web/app/page.tsx +13 -0
- package/templates/b2b-support-web/web/app/providers.tsx +21 -0
- package/templates/b2b-support-web/web/app/tickets/page.tsx +21 -0
- package/templates/b2b-support-web/web/components/CreateTicketForm.tsx +43 -0
- package/templates/b2b-support-web/web/components/PolicyDeniedDemo.tsx +31 -0
- package/templates/b2b-support-web/web/components/TicketList.tsx +52 -0
- package/templates/b2b-support-web/web/components/TraceDetails.tsx +18 -0
- package/templates/b2b-support-web/web/components/TriageStatus.tsx +13 -0
- package/templates/b2b-support-web/web/lib/forge.ts +13 -0
- package/templates/b2b-support-web/web/next-env.d.ts +5 -0
- package/templates/b2b-support-web/web/next.config.ts +8 -0
- package/templates/b2b-support-web/web/package.json +21 -0
- package/templates/b2b-support-web/web/tsconfig.json +30 -0
- package/templates/minimal-web/.vscode/settings.json +14 -0
- package/templates/minimal-web/README.md +21 -0
- package/templates/minimal-web/forge.config.ts +3 -0
- package/templates/minimal-web/package.json +32 -0
- package/templates/minimal-web/src/actions/logNoteCreated.ts +11 -0
- package/templates/minimal-web/src/commands/createNote.ts +26 -0
- package/templates/minimal-web/src/forge/schema.ts +12 -0
- package/templates/minimal-web/src/policies.ts +6 -0
- package/templates/minimal-web/src/queries/listNotes.ts +8 -0
- package/templates/minimal-web/src/queries/liveNotes.ts +8 -0
- package/templates/minimal-web/tsconfig.json +15 -0
- package/templates/minimal-web/web/index.html +12 -0
- package/templates/minimal-web/web/package.json +21 -0
- package/templates/minimal-web/web/src/App.tsx +89 -0
- package/templates/minimal-web/web/src/lib/forge.ts +13 -0
- package/templates/minimal-web/web/src/main.tsx +13 -0
- package/templates/minimal-web/web/src/styles.css +156 -0
- package/templates/minimal-web/web/tsconfig.json +18 -0
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
import type { EmitFile } from "../types/emit.ts";
|
|
2
|
+
import {
|
|
3
|
+
hashStable,
|
|
4
|
+
prependDeterministicHeader,
|
|
5
|
+
normalizeNewlines,
|
|
6
|
+
serializeCanonical,
|
|
7
|
+
} from "../primitives/index.ts";
|
|
8
|
+
import { detectArtifactKind } from "./artifact-kind.ts";
|
|
9
|
+
|
|
10
|
+
export interface RenderContext {
|
|
11
|
+
generatorVersion: string;
|
|
12
|
+
inputHash: string;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
function renderJsonBody(content: string): string {
|
|
16
|
+
try {
|
|
17
|
+
const parsed: unknown = JSON.parse(content);
|
|
18
|
+
return serializeCanonical(parsed);
|
|
19
|
+
} catch {
|
|
20
|
+
return normalizeNewlines(content);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
function renderTypeScriptBody(content: string): string {
|
|
25
|
+
return normalizeNewlines(content);
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
function renderMarkdownBody(content: string): string {
|
|
29
|
+
return normalizeNewlines(content);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
function renderTextBody(content: string): string {
|
|
33
|
+
return normalizeNewlines(content);
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Normalize file body bytes by artifact kind (no deterministic header).
|
|
38
|
+
*/
|
|
39
|
+
export function renderBody(file: EmitFile): string {
|
|
40
|
+
const kind = detectArtifactKind(file.path);
|
|
41
|
+
|
|
42
|
+
switch (kind) {
|
|
43
|
+
case "json":
|
|
44
|
+
if (file.canonical) {
|
|
45
|
+
return normalizeNewlines(file.content);
|
|
46
|
+
}
|
|
47
|
+
return renderJsonBody(file.content);
|
|
48
|
+
case "typescript":
|
|
49
|
+
return renderTypeScriptBody(file.content);
|
|
50
|
+
case "markdown":
|
|
51
|
+
return renderMarkdownBody(file.content);
|
|
52
|
+
default:
|
|
53
|
+
return renderTextBody(file.content);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* Pure render: same EmitFile + context → same bytes (header included).
|
|
59
|
+
*/
|
|
60
|
+
export function render(file: EmitFile, context: RenderContext): string {
|
|
61
|
+
const body = renderBody(file);
|
|
62
|
+
|
|
63
|
+
if (file.contentHash !== hashStable(body)) {
|
|
64
|
+
throw new Error(
|
|
65
|
+
`EmitFile contentHash mismatch for ${file.path}: expected ${hashStable(body)}, got ${file.contentHash}`,
|
|
66
|
+
);
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
return prependDeterministicHeader(body, {
|
|
70
|
+
generatorVersion: context.generatorVersion,
|
|
71
|
+
inputHash: context.inputHash,
|
|
72
|
+
});
|
|
73
|
+
}
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { nodeFileSystem } from "../fs/index.ts";
|
|
2
|
+
import type { FileSystem } from "../fs/index.ts";
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Thin async I/O helpers for the emitter, delegating to an injectable
|
|
6
|
+
* {@link FileSystem}. The async signatures are preserved so callers in
|
|
7
|
+
* `emit.ts` are unchanged; tests can pass an `InMemoryFileSystem` to run the
|
|
8
|
+
* emitter without touching disk. Atomic write semantics live in
|
|
9
|
+
* {@link NodeFileSystem.writeText}.
|
|
10
|
+
*/
|
|
11
|
+
export async function readTextFileIfExists(
|
|
12
|
+
absolutePath: string,
|
|
13
|
+
fs: FileSystem = nodeFileSystem,
|
|
14
|
+
): Promise<string | null> {
|
|
15
|
+
return fs.readText(absolutePath);
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
export async function writeFileAtomic(
|
|
19
|
+
absolutePath: string,
|
|
20
|
+
content: string,
|
|
21
|
+
fs: FileSystem = nodeFileSystem,
|
|
22
|
+
): Promise<void> {
|
|
23
|
+
fs.writeText(absolutePath, content);
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export async function removeFileIfExists(
|
|
27
|
+
absolutePath: string,
|
|
28
|
+
fs: FileSystem = nodeFileSystem,
|
|
29
|
+
): Promise<boolean> {
|
|
30
|
+
const existed = fs.exists(absolutePath);
|
|
31
|
+
if (existed) {
|
|
32
|
+
fs.remove(absolutePath);
|
|
33
|
+
}
|
|
34
|
+
return existed;
|
|
35
|
+
}
|
|
@@ -0,0 +1,495 @@
|
|
|
1
|
+
import { basename, dirname, join, relative } from "node:path";
|
|
2
|
+
import { nodeFileSystem } from "../fs/index.ts";
|
|
3
|
+
import type { ClientManifest } from "../client-sdk/build-manifest.ts";
|
|
4
|
+
import type {
|
|
5
|
+
FrontendClientBindingInfo,
|
|
6
|
+
FrontendComponentInfo,
|
|
7
|
+
FrontendGraph,
|
|
8
|
+
FrontendProviderInfo,
|
|
9
|
+
FrontendRouteInfo,
|
|
10
|
+
} from "../types/frontend-graph.ts";
|
|
11
|
+
import { serializeCanonical } from "../primitives/serialize.ts";
|
|
12
|
+
import { createDiagnostic } from "../diagnostics/create.ts";
|
|
13
|
+
|
|
14
|
+
const WEB_ROOT = "web";
|
|
15
|
+
const API_URL_ENV = "NEXT_PUBLIC_FORGE_URL";
|
|
16
|
+
const VITE_API_URL_ENV = "VITE_FORGE_URL";
|
|
17
|
+
|
|
18
|
+
function toPosix(path: string): string {
|
|
19
|
+
return path.replace(/\\/g, "/");
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function uniqueSorted(items: string[]): string[] {
|
|
23
|
+
return [...new Set(items.filter(Boolean))].sort();
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
function readJson<T>(path: string): T | null {
|
|
27
|
+
if (!nodeFileSystem.exists(path)) {
|
|
28
|
+
return null;
|
|
29
|
+
}
|
|
30
|
+
try {
|
|
31
|
+
return JSON.parse(nodeFileSystem.readText(path) ?? "") as T;
|
|
32
|
+
} catch {
|
|
33
|
+
return null;
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
function walkFiles(root: string): string[] {
|
|
38
|
+
if (!nodeFileSystem.exists(root)) {
|
|
39
|
+
return [];
|
|
40
|
+
}
|
|
41
|
+
const files: string[] = [];
|
|
42
|
+
function walk(dir: string): void {
|
|
43
|
+
for (const entry of nodeFileSystem.readDir(dir)) {
|
|
44
|
+
const absolute = join(dir, entry.name);
|
|
45
|
+
if (entry.isDirectory) {
|
|
46
|
+
if (entry.name === "node_modules" || entry.name === ".next" || entry.name === "dist") {
|
|
47
|
+
continue;
|
|
48
|
+
}
|
|
49
|
+
walk(absolute);
|
|
50
|
+
} else if (entry.isFile) {
|
|
51
|
+
files.push(absolute);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
walk(root);
|
|
56
|
+
return files.sort((a, b) => toPosix(a).localeCompare(toPosix(b)));
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
type WebPackageJson = {
|
|
60
|
+
packageManager?: string;
|
|
61
|
+
scripts?: Record<string, string>;
|
|
62
|
+
dependencies?: Record<string, string>;
|
|
63
|
+
devDependencies?: Record<string, string>;
|
|
64
|
+
};
|
|
65
|
+
|
|
66
|
+
function detectFramework(webRoot: string): FrontendGraph["framework"] {
|
|
67
|
+
const pkg = readJson<WebPackageJson>(
|
|
68
|
+
join(webRoot, "package.json"),
|
|
69
|
+
);
|
|
70
|
+
const deps = { ...(pkg?.dependencies ?? {}), ...(pkg?.devDependencies ?? {}) };
|
|
71
|
+
if (deps.next || nodeFileSystem.exists(join(webRoot, "next.config.ts"))) {
|
|
72
|
+
return "next";
|
|
73
|
+
}
|
|
74
|
+
if (deps.vite || nodeFileSystem.exists(join(webRoot, "vite.config.ts"))) {
|
|
75
|
+
return "vite";
|
|
76
|
+
}
|
|
77
|
+
if (nodeFileSystem.exists(join(webRoot, "index.html"))) {
|
|
78
|
+
return "static";
|
|
79
|
+
}
|
|
80
|
+
return "unknown";
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
function detectPackageManager(pkg: WebPackageJson | null): FrontendGraph["webManifest"]["packageManager"] {
|
|
84
|
+
const raw = pkg?.packageManager;
|
|
85
|
+
if (raw?.startsWith("bun@")) return "bun";
|
|
86
|
+
if (raw?.startsWith("pnpm@")) return "pnpm";
|
|
87
|
+
if (raw?.startsWith("yarn@")) return "yarn";
|
|
88
|
+
if (raw?.startsWith("npm@")) return "npm";
|
|
89
|
+
return raw ? "unknown" : undefined;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
function routePathForFile(webRoot: string, file: string): string | null {
|
|
93
|
+
const rel = toPosix(relative(webRoot, file));
|
|
94
|
+
if (rel === "index.html") {
|
|
95
|
+
return "/";
|
|
96
|
+
}
|
|
97
|
+
if (
|
|
98
|
+
rel === "src/App.tsx" ||
|
|
99
|
+
rel === "src/App.ts" ||
|
|
100
|
+
rel === "src/App.jsx" ||
|
|
101
|
+
rel === "src/App.js"
|
|
102
|
+
) {
|
|
103
|
+
return "/";
|
|
104
|
+
}
|
|
105
|
+
if (rel === "app/page.tsx" || rel === "app/page.ts" || rel === "app/page.jsx" || rel === "app/page.js") {
|
|
106
|
+
return "/";
|
|
107
|
+
}
|
|
108
|
+
const match = rel.match(/^app\/(.+)\/page\.(tsx|ts|jsx|js)$/);
|
|
109
|
+
if (match) {
|
|
110
|
+
const route = match[1]
|
|
111
|
+
.split("/")
|
|
112
|
+
.filter((segment) => !segment.startsWith("(") && !segment.endsWith(")"))
|
|
113
|
+
.join("/");
|
|
114
|
+
return `/${route}`;
|
|
115
|
+
}
|
|
116
|
+
return null;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
function componentNameForFile(file: string): string {
|
|
120
|
+
return basename(file).replace(/\.(tsx|ts|jsx|js)$/, "");
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
function componentNameForText(file: string, text: string): string {
|
|
124
|
+
const exportMatch = text.match(/export\s+function\s+([A-Z][A-Za-z0-9_]*)\s*\(/);
|
|
125
|
+
if (exportMatch?.[1]) {
|
|
126
|
+
return exportMatch[1];
|
|
127
|
+
}
|
|
128
|
+
const fnMatch = text.match(/function\s+([A-Z][A-Za-z0-9_]*)\s*\(/);
|
|
129
|
+
if (fnMatch?.[1]) {
|
|
130
|
+
return fnMatch[1];
|
|
131
|
+
}
|
|
132
|
+
return componentNameForFile(file);
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
function isComponentFile(webRoot: string, file: string, text: string): boolean {
|
|
136
|
+
const rel = toPosix(relative(webRoot, file));
|
|
137
|
+
if (!/\.(tsx|jsx)$/.test(file)) {
|
|
138
|
+
return false;
|
|
139
|
+
}
|
|
140
|
+
if (rel === "src/main.tsx" || rel === "src/main.jsx") {
|
|
141
|
+
return false;
|
|
142
|
+
}
|
|
143
|
+
if (rel.startsWith("components/") || rel.startsWith("src/components/")) {
|
|
144
|
+
return true;
|
|
145
|
+
}
|
|
146
|
+
return /export\s+(default\s+)?function\s+[A-Z]/.test(text) || /<[A-Z][A-Za-z0-9_.]*[\s/>]/.test(text);
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
function detectUses(text: string, clientManifest: ClientManifest) {
|
|
150
|
+
const rawForgeFetches = [
|
|
151
|
+
...[...text.matchAll(/fetch\((["'`])([^"'`]*(?:\/commands\/|\/queries\/|\/live\/)[^"'`]*)\1/g)]
|
|
152
|
+
.map((match) => match[2] ?? ""),
|
|
153
|
+
...[...text.matchAll(/(["'`])(\/(?:commands|queries|live)\/[^"'`]+)\1/g)]
|
|
154
|
+
.map((match) => match[2] ?? ""),
|
|
155
|
+
];
|
|
156
|
+
return {
|
|
157
|
+
usesCommands: uniqueSorted(clientManifest.commands.filter((name) => text.includes(name))),
|
|
158
|
+
usesQueries: uniqueSorted(clientManifest.queries.filter((name) => text.includes(name))),
|
|
159
|
+
usesLiveQueries: uniqueSorted(clientManifest.liveQueries.filter((name) => text.includes(name))),
|
|
160
|
+
rawForgeFetches: uniqueSorted(rawForgeFetches),
|
|
161
|
+
};
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
function mergeUses(
|
|
165
|
+
left: ReturnType<typeof detectUses>,
|
|
166
|
+
right: ReturnType<typeof detectUses>,
|
|
167
|
+
): ReturnType<typeof detectUses> {
|
|
168
|
+
return {
|
|
169
|
+
usesCommands: uniqueSorted([...left.usesCommands, ...right.usesCommands]),
|
|
170
|
+
usesQueries: uniqueSorted([...left.usesQueries, ...right.usesQueries]),
|
|
171
|
+
usesLiveQueries: uniqueSorted([...left.usesLiveQueries, ...right.usesLiveQueries]),
|
|
172
|
+
rawForgeFetches: uniqueSorted([...left.rawForgeFetches, ...right.rawForgeFetches]),
|
|
173
|
+
};
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
function detectRouteUses(file: string, text: string, clientManifest: ClientManifest) {
|
|
177
|
+
let uses = detectUses(text, clientManifest);
|
|
178
|
+
if (!file.endsWith(".html")) {
|
|
179
|
+
return uses;
|
|
180
|
+
}
|
|
181
|
+
for (const match of text.matchAll(/<script[^>]+src=["']([^"']+)["']/g)) {
|
|
182
|
+
const scriptPath = match[1];
|
|
183
|
+
if (!scriptPath || /^https?:\/\//.test(scriptPath)) {
|
|
184
|
+
continue;
|
|
185
|
+
}
|
|
186
|
+
const absolute = join(dirname(file), scriptPath.replace(/^\//, ""));
|
|
187
|
+
if (nodeFileSystem.exists(absolute)) {
|
|
188
|
+
uses = mergeUses(uses, detectUses(nodeFileSystem.readText(absolute) ?? "", clientManifest));
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
return uses;
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
function devCommandFor(webRoot: string, framework: FrontendGraph["framework"]): string {
|
|
195
|
+
const pkg = readJson<{ scripts?: Record<string, string> }>(join(webRoot, "package.json"));
|
|
196
|
+
if (pkg?.scripts?.dev) {
|
|
197
|
+
return "cd web && bun run dev";
|
|
198
|
+
}
|
|
199
|
+
if (nodeFileSystem.exists(join(webRoot, "server.ts"))) {
|
|
200
|
+
return "bun web/server.ts";
|
|
201
|
+
}
|
|
202
|
+
if (framework === "static") {
|
|
203
|
+
return "forge dev --web";
|
|
204
|
+
}
|
|
205
|
+
return "none";
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
function apiEnvFor(framework: FrontendGraph["framework"]): string {
|
|
209
|
+
return framework === "vite" || framework === "static" ? VITE_API_URL_ENV : API_URL_ENV;
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
function stringProp(text: string, prop: "userId" | "tenantId" | "role"): string | undefined {
|
|
213
|
+
const literal = text.match(new RegExp(`${prop}\\s*:\\s*["'\`]([^"'\`]+)["'\`]`));
|
|
214
|
+
if (literal?.[1]) {
|
|
215
|
+
return literal[1];
|
|
216
|
+
}
|
|
217
|
+
const identifier = text.match(new RegExp(`${prop}\\s*:\\s*([A-Za-z_$][A-Za-z0-9_$]*)`));
|
|
218
|
+
const name = identifier?.[1];
|
|
219
|
+
if (!name) {
|
|
220
|
+
return undefined;
|
|
221
|
+
}
|
|
222
|
+
const constant = text.match(new RegExp(`(?:const|let|var)\\s+${name}\\s*=\\s*["'\`]([^"'\`]+)["'\`]`));
|
|
223
|
+
return constant?.[1];
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
function defaultWebPort(framework: FrontendGraph["framework"]): number {
|
|
227
|
+
return framework === "next" ? 3000 : 5173;
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
function componentNamesInText(text: string, components: FrontendComponentInfo[]): string[] {
|
|
231
|
+
return uniqueSorted(
|
|
232
|
+
components
|
|
233
|
+
.filter((component) => new RegExp(`<${component.name}(\\s|>|/)`).test(text))
|
|
234
|
+
.map((component) => component.name),
|
|
235
|
+
);
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
function bindingEntries(input: {
|
|
239
|
+
file: string;
|
|
240
|
+
route?: string;
|
|
241
|
+
component?: string;
|
|
242
|
+
usesCommands: string[];
|
|
243
|
+
usesQueries: string[];
|
|
244
|
+
usesLiveQueries: string[];
|
|
245
|
+
rawForgeFetches: string[];
|
|
246
|
+
}): FrontendClientBindingInfo[] {
|
|
247
|
+
return [
|
|
248
|
+
...input.usesCommands.map((name) => ({ kind: "command" as const, name })),
|
|
249
|
+
...input.usesQueries.map((name) => ({ kind: "query" as const, name })),
|
|
250
|
+
...input.usesLiveQueries.map((name) => ({ kind: "liveQuery" as const, name })),
|
|
251
|
+
...input.rawForgeFetches.map((name) => ({ kind: "rawFetch" as const, name })),
|
|
252
|
+
].map((binding) => ({
|
|
253
|
+
...binding,
|
|
254
|
+
file: input.file,
|
|
255
|
+
...(input.route ? { route: input.route } : {}),
|
|
256
|
+
...(input.component ? { component: input.component } : {}),
|
|
257
|
+
}));
|
|
258
|
+
}
|
|
259
|
+
|
|
260
|
+
export function buildFrontendGraph(input: {
|
|
261
|
+
workspaceRoot: string;
|
|
262
|
+
clientManifest: ClientManifest;
|
|
263
|
+
apiPort?: number;
|
|
264
|
+
webPort?: number;
|
|
265
|
+
}): FrontendGraph {
|
|
266
|
+
const webRoot = join(input.workspaceRoot, WEB_ROOT);
|
|
267
|
+
if (!nodeFileSystem.exists(webRoot)) {
|
|
268
|
+
return {
|
|
269
|
+
schemaVersion: "0.1.0",
|
|
270
|
+
present: false,
|
|
271
|
+
framework: "none",
|
|
272
|
+
routes: [],
|
|
273
|
+
components: [],
|
|
274
|
+
providers: [],
|
|
275
|
+
bridgeFiles: [],
|
|
276
|
+
webManifest: {
|
|
277
|
+
present: false,
|
|
278
|
+
framework: "none",
|
|
279
|
+
scripts: {},
|
|
280
|
+
urls: {
|
|
281
|
+
api: `http://127.0.0.1:${input.apiPort ?? 3765}`,
|
|
282
|
+
},
|
|
283
|
+
env: {
|
|
284
|
+
apiUrl: API_URL_ENV,
|
|
285
|
+
},
|
|
286
|
+
bridge: {
|
|
287
|
+
files: [],
|
|
288
|
+
valid: false,
|
|
289
|
+
},
|
|
290
|
+
},
|
|
291
|
+
clientBindings: [],
|
|
292
|
+
diagnostics: [],
|
|
293
|
+
};
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
const pkg = readJson<WebPackageJson>(join(webRoot, "package.json"));
|
|
297
|
+
const framework = detectFramework(webRoot);
|
|
298
|
+
const apiUrlEnv = apiEnvFor(framework);
|
|
299
|
+
const defaultApiUrl = `http://127.0.0.1:${input.apiPort ?? 3765}`;
|
|
300
|
+
const devUrl = `http://127.0.0.1:${input.webPort ?? defaultWebPort(framework)}`;
|
|
301
|
+
const files = walkFiles(webRoot);
|
|
302
|
+
const sourceFiles = files.filter((file) => /\.(tsx|ts|jsx|js|html)$/.test(file));
|
|
303
|
+
const routes: FrontendRouteInfo[] = [];
|
|
304
|
+
const components: FrontendComponentInfo[] = [];
|
|
305
|
+
const providers: FrontendProviderInfo[] = [];
|
|
306
|
+
const bridgeFiles: string[] = [];
|
|
307
|
+
const diagnostics: FrontendGraph["diagnostics"] = [];
|
|
308
|
+
const textByRel = new Map<string, string>();
|
|
309
|
+
|
|
310
|
+
for (const file of sourceFiles) {
|
|
311
|
+
const rel = toPosix(relative(input.workspaceRoot, file));
|
|
312
|
+
const text = nodeFileSystem.readText(file) ?? "";
|
|
313
|
+
const isBridgeFile =
|
|
314
|
+
rel === "web/lib/forge.ts" ||
|
|
315
|
+
rel === "web/lib/forge.tsx" ||
|
|
316
|
+
rel === "web/src/lib/forge.ts" ||
|
|
317
|
+
rel === "web/src/lib/forge.tsx";
|
|
318
|
+
textByRel.set(rel, text);
|
|
319
|
+
const uses = detectUses(text, input.clientManifest);
|
|
320
|
+
if (isComponentFile(webRoot, file, text)) {
|
|
321
|
+
components.push({ name: componentNameForText(file, text), file: rel, ...uses });
|
|
322
|
+
}
|
|
323
|
+
if (!isBridgeFile && text.includes("ForgeProvider")) {
|
|
324
|
+
const devAuth =
|
|
325
|
+
text.includes("devAuth") ||
|
|
326
|
+
(text.includes("userId") && text.includes("tenantId") && text.includes("role"));
|
|
327
|
+
providers.push({
|
|
328
|
+
name: "ForgeProvider",
|
|
329
|
+
file: rel,
|
|
330
|
+
...(text.includes(API_URL_ENV) ? { apiUrlEnv: API_URL_ENV } : {}),
|
|
331
|
+
...(text.includes(VITE_API_URL_ENV) ? { apiUrlEnv: VITE_API_URL_ENV } : {}),
|
|
332
|
+
devAuth,
|
|
333
|
+
...(devAuth && stringProp(text, "userId") ? { devAuthUserId: stringProp(text, "userId") } : {}),
|
|
334
|
+
...(devAuth && stringProp(text, "tenantId") ? { devAuthTenantId: stringProp(text, "tenantId") } : {}),
|
|
335
|
+
...(devAuth && stringProp(text, "role") ? { devAuthRole: stringProp(text, "role") } : {}),
|
|
336
|
+
});
|
|
337
|
+
}
|
|
338
|
+
if (
|
|
339
|
+
rel === "web/lib/forge.ts" ||
|
|
340
|
+
rel === "web/lib/forge.tsx" ||
|
|
341
|
+
rel === "web/src/lib/forge.ts" ||
|
|
342
|
+
rel === "web/src/lib/forge.tsx"
|
|
343
|
+
) {
|
|
344
|
+
bridgeFiles.push(rel);
|
|
345
|
+
}
|
|
346
|
+
if (!isBridgeFile && (/from\s+["']\.\.\/\.\.\/src\/forge\/_generated/.test(text) || /from\s+["'][^"']*\/src\/forge\/_generated/.test(text))) {
|
|
347
|
+
diagnostics.push(createDiagnostic({
|
|
348
|
+
severity: "warning",
|
|
349
|
+
code: "FORGE_FRONTEND_SERVER_IMPORT",
|
|
350
|
+
message: "frontend imports generated files directly; prefer the local web/lib/forge bridge",
|
|
351
|
+
file: rel,
|
|
352
|
+
fixHint: "Import generated client APIs through web/lib/forge.ts so agents and humans have one stable frontend bridge.",
|
|
353
|
+
suggestedCommands: ["forge inspect frontend --json", "forge make ui --framework vite --dry-run --json"],
|
|
354
|
+
docs: ["src/forge/_generated/frontendGraph.json", "src/forge/_generated/appMap.md"],
|
|
355
|
+
}));
|
|
356
|
+
}
|
|
357
|
+
if (uses.rawForgeFetches.length > 0) {
|
|
358
|
+
diagnostics.push(createDiagnostic({
|
|
359
|
+
severity: "warning",
|
|
360
|
+
code: "FORGE_FRONTEND_DIRECT_RUNTIME_FETCH",
|
|
361
|
+
message: "frontend calls Forge runtime endpoints directly; prefer generated hooks from the web bridge",
|
|
362
|
+
file: rel,
|
|
363
|
+
fixHint: "Replace raw /commands, /queries, or /live fetches with useCommand, useQuery, or useLiveQuery from the local Forge bridge.",
|
|
364
|
+
suggestedCommands: ["forge inspect frontend --json", "forge repair diagnose --diagnostic FORGE_FRONTEND_DIRECT_RUNTIME_FETCH --json"],
|
|
365
|
+
docs: ["src/forge/_generated/frontendGraph.json", "AGENTS.md"],
|
|
366
|
+
}));
|
|
367
|
+
}
|
|
368
|
+
}
|
|
369
|
+
|
|
370
|
+
for (const file of sourceFiles) {
|
|
371
|
+
const rel = toPosix(relative(input.workspaceRoot, file));
|
|
372
|
+
const text = textByRel.get(rel) ?? "";
|
|
373
|
+
const routePath = routePathForFile(webRoot, file);
|
|
374
|
+
if (routePath) {
|
|
375
|
+
if (framework === "vite" && rel === "web/index.html") {
|
|
376
|
+
continue;
|
|
377
|
+
}
|
|
378
|
+
const componentNames = componentNamesInText(text, components);
|
|
379
|
+
const componentUses = components
|
|
380
|
+
.filter((component) => componentNames.includes(component.name))
|
|
381
|
+
.reduce(
|
|
382
|
+
(acc, component) =>
|
|
383
|
+
mergeUses(acc, {
|
|
384
|
+
usesCommands: component.usesCommands,
|
|
385
|
+
usesQueries: component.usesQueries,
|
|
386
|
+
usesLiveQueries: component.usesLiveQueries,
|
|
387
|
+
rawForgeFetches: component.rawForgeFetches,
|
|
388
|
+
}),
|
|
389
|
+
detectRouteUses(file, text, input.clientManifest),
|
|
390
|
+
);
|
|
391
|
+
routes.push({
|
|
392
|
+
path: routePath,
|
|
393
|
+
file: rel,
|
|
394
|
+
components: componentNames,
|
|
395
|
+
...componentUses,
|
|
396
|
+
});
|
|
397
|
+
}
|
|
398
|
+
}
|
|
399
|
+
|
|
400
|
+
if (providers.length === 0 && framework !== "static") {
|
|
401
|
+
diagnostics.push(createDiagnostic({
|
|
402
|
+
severity: "warning",
|
|
403
|
+
code: "FORGE_FRONTEND_PROVIDER_MISSING",
|
|
404
|
+
message: "web app does not expose a ForgeProvider; generated hooks may not be wired",
|
|
405
|
+
fixHint: "Mount ForgeProvider once in the web app root/provider layer and pass devAuth for local development.",
|
|
406
|
+
suggestedCommands: ["forge inspect frontend --json", "forge make ui --framework vite --dry-run --json"],
|
|
407
|
+
docs: ["src/forge/_generated/frontendGraph.json", "AGENTS.md"],
|
|
408
|
+
}));
|
|
409
|
+
}
|
|
410
|
+
if (bridgeFiles.length === 0 && framework !== "static") {
|
|
411
|
+
diagnostics.push(createDiagnostic({
|
|
412
|
+
severity: "warning",
|
|
413
|
+
code: "FORGE_FRONTEND_BRIDGE_MISSING",
|
|
414
|
+
message: "web/**/lib/forge.ts bridge is missing; agents may use fragile generated import paths",
|
|
415
|
+
fixHint: "Create web/lib/forge.ts or web/src/lib/forge.ts and re-export api, client, ForgeProvider, and generated hooks from src/forge/_generated.",
|
|
416
|
+
suggestedCommands: ["forge make ui --framework vite --dry-run --json", "forge generate"],
|
|
417
|
+
docs: ["src/forge/_generated/frontendGraph.json", "AGENTS.md"],
|
|
418
|
+
}));
|
|
419
|
+
}
|
|
420
|
+
|
|
421
|
+
const clientBindings = uniqueSorted([
|
|
422
|
+
...routes.flatMap((route) =>
|
|
423
|
+
bindingEntries({
|
|
424
|
+
file: route.file,
|
|
425
|
+
route: route.path,
|
|
426
|
+
usesCommands: route.usesCommands,
|
|
427
|
+
usesQueries: route.usesQueries,
|
|
428
|
+
usesLiveQueries: route.usesLiveQueries,
|
|
429
|
+
rawForgeFetches: route.rawForgeFetches,
|
|
430
|
+
}).map((binding) => JSON.stringify(binding)),
|
|
431
|
+
),
|
|
432
|
+
...components.flatMap((component) =>
|
|
433
|
+
bindingEntries({
|
|
434
|
+
file: component.file,
|
|
435
|
+
component: component.name,
|
|
436
|
+
usesCommands: component.usesCommands,
|
|
437
|
+
usesQueries: component.usesQueries,
|
|
438
|
+
usesLiveQueries: component.usesLiveQueries,
|
|
439
|
+
rawForgeFetches: component.rawForgeFetches,
|
|
440
|
+
}).map((binding) => JSON.stringify(binding)),
|
|
441
|
+
),
|
|
442
|
+
]).map((binding) => JSON.parse(binding) as FrontendClientBindingInfo);
|
|
443
|
+
|
|
444
|
+
return {
|
|
445
|
+
schemaVersion: "0.1.0",
|
|
446
|
+
present: true,
|
|
447
|
+
framework,
|
|
448
|
+
root: WEB_ROOT,
|
|
449
|
+
dev: {
|
|
450
|
+
command: devCommandFor(webRoot, framework),
|
|
451
|
+
url: devUrl,
|
|
452
|
+
apiUrlEnv,
|
|
453
|
+
defaultApiUrl,
|
|
454
|
+
},
|
|
455
|
+
routes: routes.sort((a, b) => a.path.localeCompare(b.path)),
|
|
456
|
+
components: components.sort((a, b) => a.file.localeCompare(b.file)),
|
|
457
|
+
providers: providers.sort((a, b) => a.file.localeCompare(b.file)),
|
|
458
|
+
bridgeFiles: uniqueSorted(bridgeFiles),
|
|
459
|
+
webManifest: {
|
|
460
|
+
present: true,
|
|
461
|
+
framework,
|
|
462
|
+
root: WEB_ROOT,
|
|
463
|
+
...(detectPackageManager(pkg) ? { packageManager: detectPackageManager(pkg) } : {}),
|
|
464
|
+
scripts: {
|
|
465
|
+
...(pkg?.scripts?.dev ? { dev: pkg.scripts.dev } : {}),
|
|
466
|
+
...(pkg?.scripts?.build ? { build: pkg.scripts.build } : {}),
|
|
467
|
+
...(pkg?.scripts?.typecheck ? { typecheck: pkg.scripts.typecheck } : {}),
|
|
468
|
+
},
|
|
469
|
+
urls: {
|
|
470
|
+
dev: devUrl,
|
|
471
|
+
api: defaultApiUrl,
|
|
472
|
+
},
|
|
473
|
+
env: {
|
|
474
|
+
apiUrl: apiUrlEnv,
|
|
475
|
+
},
|
|
476
|
+
bridge: {
|
|
477
|
+
files: uniqueSorted(bridgeFiles),
|
|
478
|
+
valid: bridgeFiles.length > 0,
|
|
479
|
+
},
|
|
480
|
+
},
|
|
481
|
+
clientBindings: clientBindings.sort((a, b) =>
|
|
482
|
+
`${a.file}:${a.kind}:${a.name}`.localeCompare(`${b.file}:${b.kind}:${b.name}`),
|
|
483
|
+
),
|
|
484
|
+
diagnostics,
|
|
485
|
+
};
|
|
486
|
+
}
|
|
487
|
+
|
|
488
|
+
export function serializeFrontendGraphJson(graph: FrontendGraph): string {
|
|
489
|
+
return serializeCanonical(graph);
|
|
490
|
+
}
|
|
491
|
+
|
|
492
|
+
export function serializeFrontendGraphTs(graph: FrontendGraph): string {
|
|
493
|
+
const parsed = JSON.parse(serializeFrontendGraphJson(graph)) as unknown;
|
|
494
|
+
return `export const frontendGraph = ${JSON.stringify(parsed, null, 2)} as const;\n`;
|
|
495
|
+
}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
export type { FileSystem, DirEntry, FileStat } from "./types.ts";
|
|
2
|
+
export { NodeFileSystem } from "./node.ts";
|
|
3
|
+
export { InMemoryFileSystem } from "./memory.ts";
|
|
4
|
+
export {
|
|
5
|
+
createProfiledFileSystem,
|
|
6
|
+
formatFileSystemProfile,
|
|
7
|
+
getFileSystemProfile,
|
|
8
|
+
isForgeProfileEnabled,
|
|
9
|
+
resetFileSystemProfile,
|
|
10
|
+
} from "./profile.ts";
|
|
11
|
+
|
|
12
|
+
import { NodeFileSystem } from "./node.ts";
|
|
13
|
+
import {
|
|
14
|
+
createProfiledFileSystem,
|
|
15
|
+
isForgeProfileEnabled,
|
|
16
|
+
resetFileSystemProfile,
|
|
17
|
+
} from "./profile.ts";
|
|
18
|
+
|
|
19
|
+
resetFileSystemProfile();
|
|
20
|
+
const baseFileSystem = new NodeFileSystem();
|
|
21
|
+
export const nodeFileSystem = isForgeProfileEnabled()
|
|
22
|
+
? createProfiledFileSystem(baseFileSystem)
|
|
23
|
+
: baseFileSystem;
|