name: eforge-extend description: Author eforge TypeScript extensions from a natural-language request using the existing extension tooling and docs/examples
/eforge:extend
Author eforge TypeScript extensions from a natural-language request using the existing extension tooling, docs, and examples. The workflow is conversational: classify the requested behavior, scaffold with eforge_extension, edit the returned file with normal file tools, validate, optionally replay-test, reload, and summarize trust/security notes.
Workflow
Step 1: Argument intake
- If
$ARGUMENTSis missing or empty, ask what eforge behavior the user wants to add and stop until they answer. - Generate a kebab-case extension name from the request.
- Ask for a name only when the generated name is ambiguous or conflicts with an existing extension.
Step 2: Required context read
Before scaffolding or editing, read these required eforge docs. Prefer the repository-local paths below when they exist; in an installed consumer project where they are not present, use the same documents from https://eforge.build/docs/extensions, https://eforge.build/docs/extensions-api, and https://github.com/eforge-build/eforge/tree/main/examples/extensions.
docs/extensions.mddocs/extensions-api.mdexamples/extensions/README.md
Then read relevant examples based on the classified capability:
examples/extensions/minimal-event-logger.tsfor event hooks.examples/extensions/slack-webhook-notifier.tsfor webhook/notification hooks.examples/extensions/agent-context.tsfor prompt context.examples/extensions/agent-tools.tsfor custom agent tools.examples/extensions/profile-router.tsfor profile selection.examples/extensions/protected-paths.tsfor runtime-supported plan/final merge policy gates.examples/extensions/issue-tracker.tsfor runtime-supported input source adapters (GitHub, Linear, Jira).examples/extensions/reviewer-perspective.tsfor runtime-supported reviewer perspectives with declarative file-pattern applicability.
Step 3: Capability classification
Classify the requested behavior before choosing a template or editing code.
Runtime-supported capability families:
onEventevent subscriptions and event replay.onAgentRunprompt context, per-run tools, and per-run allowed/disallowed tuning.defineExtensionTool+registerTool+ returningtoolsfromonAgentRunfor runtime custom tool injection.registerToolalone is provenance capture; runtime tool injection requires returning the tool fromonAgentRun.registerProfileRouterpre-build profile selection.beforeQueueDispatch,beforePlanMerge, andbeforeFinalMergeblocking policy gates.require-approvalcurrently blocks because no approval workflow/UI/state exists, and policy gate contexts are read-only snapshots. These helpers do not sandbox extension code; extensions are trusted unsandboxed code.registerInputSourceinput source adapters — supply PRD/build-source artifacts from external systems viaeforge://input/<adapter>/<id>URIs. Adapter selection is bynamematch. Input-source failures (null return or throw) are fatal to enqueue; design adapters to return instructional content when credentials are absent.registerPrdEnricherPRD enrichers — mutate or augment PRD content before queue write. Enrichers run in registration order for every preprocessed source; failures are fail-open (extension:prd-enricher:failed). Gate behavior insideenrichusingctx.sourceKind,ctx.adapterId, orctx.sourcePath.registerReviewerPerspectivereviewer perspectives — inject custom prompt context into the reviewer agent during parallel review-cycle perspective dispatch (review.strategy: parallel, orautoonce the diff crosses the parallel-review thresholds). Applicability is declared viaappliesTo.fileGlobs,paths,extensions,categories, orminChanged*thresholds; a function-formappliesTo.fn(changedFiles, changedLines)escape hatch is also available. Applicability inputs are read-only snapshots; perspectives cannot mutate orchestration state. Failures are fail-open (extension:reviewer-perspective:skippedwith reasonapplicability-errororapplicability-timeout). Seeexamples/extensions/reviewer-perspective.tsfor an accessibility example.registerValidationProvidervalidation providers — run during the per-planvalidatebuild stage whenvalidateis included in the pipeline. Normal structured failures and command-form non-zero exits enter bounded recovery; thrown errors, non-empty string returns, timeouts, and unexpected shapes are hard failures.registerAction,registerAgentTask,registerConsoleContribution,registerConsoleWorkstation,registerIntegrationCommand, andregisterDeepLinkare platform seams for extension-authored actions, prompt-backed task contribution declarations, and host-facing contributions. Actions, integration commands, and action-backed deep links are discoverable, inspectable, and invokable by host integrations througheforge_extension_contribution; declarative Console contributions render in the Console System route (/console/system). Console workstations render under/console/workstationsas sandboxed iframes from either inlinesrcDocorframeBundleassets underworkstation-assets/.srcDocworkstations usewindow.eforge.invokeAction; bundle workstation code imports the browser-safe@eforge-build/extension-sdk/browserhelpers and runs inside the iframe/action bridge boundary, not the parent Console. Contribution registrations may declare dependency/capabilityrequirements; manifest entries includeavailability, and unavailable actions are rejected with error codeunavailable.registerAgentTaskrecords trusted prompt sources (assetorexport) and projects only safe metadata; task-start callers reference contributions and do not provide prompt paths, raw prompt text, or resolver functions. Contribution-reference task starts execute for registered planner-compatible task contributions; unsupported or non-compatible starts fail without prompt-path fallback. Action handlers receive immutablectx.dependencies/ctx.capabilitiesavailability lookups, daemon-ownedctx.agentTasksfor supported single-shot read-only planner tasks, andctx.buildQueue.enqueuefor trusted queue handoffs, including genericpostMergecommand arrays. Actions that enqueue should declare thebuild-queueside effect. Actions may declareoutputProfile(agent-compact,agent-paginated,markdown,ui-rich, ordebug-rich); id-shaped broad list/search/board read actions with array-shaped outputs produce warning diagnostics when they lack limit or cursor/page controls, or when array-shaped output schemas omit an explicit profile, but still register.agent-paginatedactions with limit and cursor/page controls are considered bounded for agent use without separate projection controls. Host tools render compact contribution lists by default, expose focused detail/show modes for schemas and diagnostics, render exact{ markdown }outputs as Markdown/plain text, and summarize oversized JSON with warnings, identity fields, counts, omitted counts, and cursor/offset hints;ui-richanddebug-richalso warn in coding-agent hosts even when they fit. Pi tool text uses a 12,000-character host-output budget, andeforge_extensionreturns compact extension-management projections by default instead of raw discovery objects. Use compact/paginated actions for agents and reserve raw HTTP/CLI--jsonstyle reads for intentional rich/debug inspection. Extension dependency/capability lookup reports availability only and does not invoke other extensions. Extensions do not import provider SDKs orAgentHarness, cannot send arbitrary raw prompt templates through task starts, and cannot implement multi-turn chat through that API. Raw extension-owned HTTP routes, parent-Console plugins, direct React loading into the parent Console, private Console imports/CSS/context, extension-owned AI planning/chat APIs outsidectx.agentTasks, and arbitrary frontend bundles outside the workstation frame/asset contract remain unsupported.- The shipped playbook action surface lives in the first-party
eforge-playbooksextension, and session-planning remains separate/built-in; neither is a user-authored native extension registration point.
Runtime-deferred capability families:
beforeEnqueueandbeforeValidationpolicy gates.- Approval workflow/UI/state and
modifypolicy decisions. - User-authored custom session-plan/playbook extraction, raw extension-owned HTTP routes, parent-Console plugins, direct parent Console React/private import loading, extension-owned AI planning/chat APIs outside
ctx.agentTasks, caller-supplied arbitrary raw prompt templates, multi-turn chat, and arbitrary frontend plugin bundles outside registered workstation iframes.
If user intent maps to the supported policy-gate, reviewer-perspective, validation-provider, extension-action, or agent-task-contribution declaration subset, explain that the capability executes or projects at runtime as described above, and note the parallel-review condition for reviewer perspectives, validate-stage/recovery behavior for validation providers, daemon invocation boundary for actions, or manifest-only status for task contributions. If user intent maps to deferred APIs, state whether the current SDK provides only a type contract or a captured registration, and that it will not execute at runtime yet. Ask whether to omit that portion or include it as a clearly labeled future-facing registration. Avoid promising that deferred capability families will enqueue, approve, mutate, provide raw HTTP/frontend surfaces, extract workflows, or run unsupported validation phases at runtime.
Step 4: Scope selection
- Default to
scope: "local"for experiments and project-local personal behavior; this targets.eforge/extensions/. - Use
scope: "user"only for explicit cross-project personal extensions. - Use
scope: "project"only for explicit team-shared extensions. Project/team extensions (eforge/extensions/) are unsandboxed arbitrary code committed to the repository; they require an explicit per-extension local trust record in.eforge/extension-trust.json— created byeforge extension trust <name>— before loading. Any subsequent code change invalidates the stored hash and blocks the extension until re-trusted. Warn the user of these implications before proceeding.
Step 4b: Package install workflow (when user requests installing a packaged extension)
If the user asks to install an extension from an npm package, local package directory, or tarball rather than authoring one from scratch, use the following workflow instead of Steps 5-6:
- Clarify the install source (npm package name, local package directory, or tarball path/URL) and desired scope.
- Warn the user that installed extensions are unsandboxed arbitrary code and installing from npm, tarballs, or local package directories introduces supply-chain risk.
- Call
eforge_extensionwithaction: "install",sourceset to the package name, local package directory, or tarball path/URL, and the selectedscope. Passtrust: true, and optionallytrustedBy, only when the user explicitly opts in after the warning. - On install success, call
eforge_extensionwithaction: "list"and confirm the installed entry appears. - If the entry has
trustState: "untrusted"or"changed"and the returned entry'sscopeisproject-team:- Read the installed extension files in full.
- Summarize all side effects (filesystem, network, environment, subprocess). Show the summary.
- Ask for explicit user confirmation before trusting. Do not proceed if they decline.
- After confirmation, call
eforge_extensionwithaction: "trust"and the extensionname.
- After trust, call validate and then reload (with the same confirmation warnings as Steps 7-9 below).
Promote and demote:
If the user asks to promote a project-local extension to project/team scope or demote it back:
- Call
eforge_extensionwithaction: "promote"or"demote"and the extensionname. - After promotion to project-team scope: warn that team members will be required to trust this extension, read the extension files, summarize side effects, and ask for explicit confirmation before trusting. Call
action: "trust"after confirmation. - After demotion to project-local scope: no trust step is required.
Update and remove:
For updating or removing an installed extension, call eforge_extension with action: "update" or action: "remove" and the extension name. After update, verify trust state (a changed source file invalidates the stored hash) and re-trust if needed. Pass force: true only when the user explicitly requests forced removal.
Step 5: Scaffold
- Optionally call
eforge_extensionwithaction: "list"first to inspect existing extensions and shadowing. - Call
eforge_extensionwithaction: "new", the generatedname, selectedscope, and selectedtemplate:event-loggerfor event-driven requests.blankfor non-event or complex mixed requests.
- Use the returned
pathfor file reads and edits; do not hard-code scope paths when the tool returns a path.
Step 6: Edit
- Read the scaffolded file at the returned
path. - Apply TypeScript edits using the required docs and examples as the source of truth.
- Use environment variables for secrets, tokens, webhook URLs, and other credentials; do not commit secrets into source.
- Preserve deferred-runtime caveats in code comments when including future-facing registrations.
- For policy gates, prefer
beforeQueueDispatch,beforePlanMerge, andbeforeFinalMerge; document thatrequire-approvalblocks in the current MVP and that extensions remain unsandboxed trusted code.
Step 7: Validate
Project/team scope: inspect and trust first. If the extension is in project/team scope (eforge/extensions/):
Read the extension file(s) in full before trust, validate, test, or reload.
Summarize all side effects (filesystem, network, environment, subprocess). Show the summary and ask for explicit user confirmation before proceeding. Do not proceed if they decline.
After confirmation, record the current content hash by calling
eforge_extensionwithaction: "trust"and the extensionnameorpath. This writes a trust record to.eforge/extension-trust.json. If the code changed since the last trust, the extension is blocked (statuschanged) until re-trusted — confirm with the user before overwriting an existing trust record.The hash covers the entrypoint for file-layout extensions and, for directory-layout extensions,
package.jsonplus.ts,.mts,.js, and.mjsfiles under the extension directory. Files imported from outside the extension directory and non-source/data files inside it are not hashed; keep implementation code inside the extension directory and in hashed source files to ensure relevant code changes are captured.Proceed to validate only after trust succeeds and the user has confirmed.
Before validation, warn that extension loading is unsandboxed in the daemon process; review the code first and do not validate code with unexpected top-level filesystem, network, or environment side effects.
Show the user a brief side-effect review and ask for explicit confirmation before calling validate; do not proceed if they decline.
After editing and confirmation, call
eforge_extensionwithaction: "validate"bynameor returnedpath.If validation returns
valid: false, summarize diagnostics, fix the file, and validate again.Do not call
reloadafter failed validation.
Step 8: Optional test
- Offer event replay testing after validation.
- Use
eforge_extensionwithaction: "test"bynameorpath, plusfixture,run, oreventwhen the user supplies an event source. - Warn that replay executes matching
onEventhandlers in the daemon process and can trigger filesystem, network, or environment side effects. - Ask for explicit confirmation before running the replay test, even after validation succeeds.
- If no safe event source exists, the user declines confirmation, or the extension has side effects, skip testing and record the reason in the final summary.
Step 9: Reload
- Before reload, warn that reload activates unsandboxed extension code in the daemon process.
- Ask for explicit confirmation before reload.
- Call
eforge_extensionwithaction: "reload"only after validation succeeds and after the user confirms that warning. - Summarize watcher fields from the response when present:
wasRunning,restarted,running, andmessage.
Step 10: Final summary
Report:
- Extension name, scope, returned path, selected template, and capability families used.
- Validation result, optional test result or skipped-test reason, and reload result.
- Any deferred-runtime caveats that apply to the generated extension.
- Security/trust notes: extensions are unsandboxed code execution, project/team scope requires explicit trust, and secrets belong in environment variables rather than committed source.
- Follow-up commands: validate, test with fixture/run, reload, and where to edit the file.