From d4e065c315376dac56c010423291082a91b44343 Mon Sep 17 00:00:00 2001 From: octane0411 Date: Mon, 19 Jan 2026 19:03:11 +0800 Subject: [PATCH 1/7] feat: implement openspec plugin core logic --- DESIGN.md | 95 ++++++++++++++++++++++++++++++++++++++++++ README.md | 12 +++++- package.json | 13 ++++++ src/config.ts | 40 ++++++++++++++++++ src/index.ts | 17 ++++++++ src/prompts.ts | 24 +++++++++++ src/utils/detection.ts | 17 ++++++++ tsconfig.json | 22 ++++++++++ tsconfig.tsbuildinfo | 1 + 9 files changed, 240 insertions(+), 1 deletion(-) create mode 100644 DESIGN.md create mode 100644 package.json create mode 100644 src/config.ts create mode 100644 src/index.ts create mode 100644 src/prompts.ts create mode 100644 src/utils/detection.ts create mode 100644 tsconfig.json create mode 100644 tsconfig.tsbuildinfo diff --git a/DESIGN.md b/DESIGN.md new file mode 100644 index 0000000..4b73e94 --- /dev/null +++ b/DESIGN.md @@ -0,0 +1,95 @@ +# OpenCode Plugin OpenSpec Design + +## 1. Overview +This plugin integrates OpenSpec into OpenCode. It is **context-aware**: it activates only when it detects an OpenSpec-initialized project. When active, it dynamically registers an `openspec-plan` agent via the `config` hook, tailored for architectural planning. + +## 2. Activation Logic +The plugin checks for OpenSpec presence at startup. + +- **Condition**: Existence of `openspec/AGENTS.md` (or `AGENTS.md`). +- **Action**: If detected, the plugin hooks into the configuration loading process to inject the OpenSpec agent. + +## 3. Features (Active Mode) + +### 3.1. Agent Injection (`config` hook) +Instead of just relying on runtime hooks, we will use the `config` hook to modify the OpenCode configuration directly. This is how `oh-my-opencode` replaces the default plan agent. + +**Implementation Strategy:** +1. **Hook**: `config` +2. **Logic**: + - Check if `openspec/AGENTS.md` exists. + - If yes, inject a new agent `openspec-plan` into the `agent` configuration object. + - **Agent Configuration**: + - `name`: "openspec-plan" + - `mode`: "primary" + - `description`: "OpenSpec Architect - Plan and specify software architecture." + - `prompt`: (Custom System Prompt for OpenSpec) + - `permission`: + - Explicitly `allow` editing `**/*.spec.md`, `project.md`, `AGENTS.md`. + - This avoids reliance on the global `permission.ask` hook for basic operations, though we can keep `permission.ask` as a fallback or for finer control. + - **Optional**: Hide/Demote default `plan` or `sisyphus-plan` to reduce confusion. + +### 3.2. Auto-Permission (Fallback) +While the agent config handles most permissions, we can still use `permission.ask` for edge cases or to ensure a smooth experience if the static permission config isn't enough. + +- **Scope**: `openspec/**/*.md`, `specs/**/*.md`. +- **Logic**: Intercept `permission.ask` and return `allow` for these patterns. + +## 4. Dependencies +- `@opencode-ai/plugin`: ^1.1.1 +- `@opencode-ai/sdk`: ^1.1.1 + +## 5. Project Structure +``` +opencode-plugin-openspec/ +├── package.json +├── tsconfig.json +├── README.md +├── DESIGN.md +└── src/ + ├── index.ts # Plugin entry point + ├── config.ts # Config hook implementation (Agent injection) + ├── prompts.ts # System prompts + └── utils/ + └── detection.ts # Detection logic +``` + +## 6. Implementation Details + +### 6.1. Config Hook (`src/config.ts`) +```typescript +export const configHook: Hooks["config"] = async (config, ctx) => { + if (!await isOpenSpecProject(ctx)) return config; + + // Define OpenSpec Plan Agent + const openSpecAgent = { + name: "openspec-plan", + mode: "primary", + description: "OpenSpec Architect", + prompt: OPENSPEC_SYSTEM_PROMPT, + permission: { + edit: { + "**/*.spec.md": "allow", + "**/project.md": "allow", + "**/AGENTS.md": "allow" + } + } + }; + + // Inject into config + return { + ...config, + agent: { + ...(config.agent || {}), + "openspec-plan": openSpecAgent + } + }; +}; +``` + +### 6.2. System Prompt +The prompt will instruct the model to: +- Act as an Architect. +- Read `project.md` and `AGENTS.md` for context. +- Create/Update `specs/*.spec.md` files. +- **NOT** write implementation code. diff --git a/README.md b/README.md index c7f24a4..4d90c2b 100644 --- a/README.md +++ b/README.md @@ -1,2 +1,12 @@ # opencode-plugin-openspec -An OpenCode plugin that integrates OpenSpec, allowing Plan Mode to create and edit spec files. + +An OpenCode plugin that integrates OpenSpec. + +## Features + +- **New Mode: `openspec-plan`**: A dedicated planning mode that allows creating and editing OpenSpec files (`*.spec.md`), while keeping the rest of the codebase read-only. +- **Auto-Permission**: Automatically grants write permissions for openSpec files in `openspec-plan` mode. + +## Installation + +(Instructions to be added) diff --git a/package.json b/package.json new file mode 100644 index 0000000..7bc66fe --- /dev/null +++ b/package.json @@ -0,0 +1,13 @@ +{ + "name": "opencode-plugin-openspec", + "version": "0.1.0", + "main": "src/index.ts", + "peerDependencies": { + "@opencode-ai/plugin": "^1.1.1", + "@opencode-ai/sdk": "^1.1.1" + }, + "devDependencies": { + "bun-types": "latest", + "typescript": "^5.0.0" + } +} diff --git a/src/config.ts b/src/config.ts new file mode 100644 index 0000000..b570162 --- /dev/null +++ b/src/config.ts @@ -0,0 +1,40 @@ +import type { Hooks } from "@opencode-ai/plugin"; +import { isOpenSpecProject } from "./utils/detection"; +import { OPENSPEC_SYSTEM_PROMPT } from "./prompts"; + +export function createConfigHook(ctx: { directory: string }): Hooks["config"] { + return async (config) => { + // 1. Check if this is an OpenSpec project + const mockCtx = { directory: ctx.directory } as any; + + if (!await isOpenSpecProject(mockCtx)) { + return; + } + + // 2. Define the OpenSpec Plan Agent + const openSpecAgent = { + name: "openspec-plan", + mode: "primary", + description: "OpenSpec Architect - Plan and specify software architecture.", + prompt: OPENSPEC_SYSTEM_PROMPT, + permission: { + edit: { + "**/*.spec.md": "allow", + "**/project.md": "allow", + "**/AGENTS.md": "allow", + // Allow creating new spec directories + "specs/**": "allow", + "openspec/**": "allow" + } + }, + color: "#FF6B6B" // Distinctive color for the agent + }; + + // 3. Inject into configuration + // We use 'any' cast here to bypass strict type checking on the config object structure + // because we are dynamically extending it. + const agentConfig = (config.agent || {}) as any; + agentConfig["openspec-plan"] = openSpecAgent; + config.agent = agentConfig; + }; +} diff --git a/src/index.ts b/src/index.ts new file mode 100644 index 0000000..5d42f3a --- /dev/null +++ b/src/index.ts @@ -0,0 +1,17 @@ +import type { Plugin } from "@opencode-ai/plugin"; +import { createConfigHook } from "./config"; +import { isOpenSpecProject } from "./utils/detection"; + +const OpenSpecPlugin: Plugin = async (ctx) => { + const isActive = await isOpenSpecProject(ctx); + + if (!isActive) { + return {}; + } + + return { + config: createConfigHook(ctx), + }; +}; + +export default OpenSpecPlugin; diff --git a/src/prompts.ts b/src/prompts.ts new file mode 100644 index 0000000..9255ceb --- /dev/null +++ b/src/prompts.ts @@ -0,0 +1,24 @@ +export const OPENSPEC_SYSTEM_PROMPT = ` + +You are the **OpenSpec Architect**. +Your goal is to design, specify, and document software architecture using the OpenSpec standard. +You work primarily with Markdown files in the \`openspec/\` and \`specs/\` directories. + + + +This project follows the OpenSpec standard for documentation-driven development. +Key files: +- \`project.md\`: High-level project vision, goals, and scope. +- \`openspec/AGENTS.md\` (or \`AGENTS.md\`): Definitions of agents and their roles. +- \`specs/**/*.spec.md\`: Detailed specifications for components or features. + + + +1. **Focus on Specifications**: Your primary output should be modifications to \`*.spec.md\` files. +2. **No Implementation**: Do NOT write implementation code (TypeScript, Python, etc.) unless explicitly requested to "implement" or "prototype". Your job is to *plan*, not to *build*. +3. **Structure**: + - When designing a new feature, create a new spec file in \`specs//spec.md\`. + - Link back to \`project.md\` to ensure alignment with high-level goals. +4. **Format**: Follow the existing Markdown structure found in the project. Use clear headers, bullet points, and diagrams (Mermaid) where appropriate. + +`.trim(); diff --git a/src/utils/detection.ts b/src/utils/detection.ts new file mode 100644 index 0000000..7f22ae9 --- /dev/null +++ b/src/utils/detection.ts @@ -0,0 +1,17 @@ +import type { PluginInput } from "@opencode-ai/plugin"; +import { existsSync } from "node:fs"; +import { join } from "node:path"; + +/** + * Checks if the current workspace is an OpenSpec project. + * + * Detection logic: + * 1. Checks for `openspec/AGENTS.md` (Primary indicator) + * 2. Checks for `AGENTS.md` in root (Secondary indicator) + */ +export async function isOpenSpecProject(ctx: PluginInput): Promise { + const openspecAgentsPath = join(ctx.directory, "openspec", "AGENTS.md"); + const rootAgentsPath = join(ctx.directory, "AGENTS.md"); + + return existsSync(openspecAgentsPath) || existsSync(rootAgentsPath); +} diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..7556e1d --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,22 @@ +{ + "compilerOptions": { + "lib": ["ESNext"], + "module": "esnext", + "target": "esnext", + "moduleResolution": "bundler", + "moduleDetection": "force", + "allowImportingTsExtensions": true, + "noEmit": true, + "composite": true, + "strict": true, + "downlevelIteration": true, + "skipLibCheck": true, + "jsx": "react-jsx", + "allowSyntheticDefaultImports": true, + "forceConsistentCasingInFileNames": true, + "allowJs": true, + "types": [ + "bun-types" // add Bun global + ] + } +} diff --git a/tsconfig.tsbuildinfo b/tsconfig.tsbuildinfo new file mode 100644 index 0000000..851fc22 --- /dev/null +++ b/tsconfig.tsbuildinfo @@ -0,0 +1 @@ +{"fileNames":["./node_modules/typescript/lib/lib.es5.d.ts","./node_modules/typescript/lib/lib.es2015.d.ts","./node_modules/typescript/lib/lib.es2016.d.ts","./node_modules/typescript/lib/lib.es2017.d.ts","./node_modules/typescript/lib/lib.es2018.d.ts","./node_modules/typescript/lib/lib.es2019.d.ts","./node_modules/typescript/lib/lib.es2020.d.ts","./node_modules/typescript/lib/lib.es2021.d.ts","./node_modules/typescript/lib/lib.es2022.d.ts","./node_modules/typescript/lib/lib.es2023.d.ts","./node_modules/typescript/lib/lib.es2024.d.ts","./node_modules/typescript/lib/lib.esnext.d.ts","./node_modules/typescript/lib/lib.es2015.core.d.ts","./node_modules/typescript/lib/lib.es2015.collection.d.ts","./node_modules/typescript/lib/lib.es2015.generator.d.ts","./node_modules/typescript/lib/lib.es2015.iterable.d.ts","./node_modules/typescript/lib/lib.es2015.promise.d.ts","./node_modules/typescript/lib/lib.es2015.proxy.d.ts","./node_modules/typescript/lib/lib.es2015.reflect.d.ts","./node_modules/typescript/lib/lib.es2015.symbol.d.ts","./node_modules/typescript/lib/lib.es2015.symbol.wellknown.d.ts","./node_modules/typescript/lib/lib.es2016.array.include.d.ts","./node_modules/typescript/lib/lib.es2016.intl.d.ts","./node_modules/typescript/lib/lib.es2017.arraybuffer.d.ts","./node_modules/typescript/lib/lib.es2017.date.d.ts","./node_modules/typescript/lib/lib.es2017.object.d.ts","./node_modules/typescript/lib/lib.es2017.sharedmemory.d.ts","./node_modules/typescript/lib/lib.es2017.string.d.ts","./node_modules/typescript/lib/lib.es2017.intl.d.ts","./node_modules/typescript/lib/lib.es2017.typedarrays.d.ts","./node_modules/typescript/lib/lib.es2018.asyncgenerator.d.ts","./node_modules/typescript/lib/lib.es2018.asynciterable.d.ts","./node_modules/typescript/lib/lib.es2018.intl.d.ts","./node_modules/typescript/lib/lib.es2018.promise.d.ts","./node_modules/typescript/lib/lib.es2018.regexp.d.ts","./node_modules/typescript/lib/lib.es2019.array.d.ts","./node_modules/typescript/lib/lib.es2019.object.d.ts","./node_modules/typescript/lib/lib.es2019.string.d.ts","./node_modules/typescript/lib/lib.es2019.symbol.d.ts","./node_modules/typescript/lib/lib.es2019.intl.d.ts","./node_modules/typescript/lib/lib.es2020.bigint.d.ts","./node_modules/typescript/lib/lib.es2020.date.d.ts","./node_modules/typescript/lib/lib.es2020.promise.d.ts","./node_modules/typescript/lib/lib.es2020.sharedmemory.d.ts","./node_modules/typescript/lib/lib.es2020.string.d.ts","./node_modules/typescript/lib/lib.es2020.symbol.wellknown.d.ts","./node_modules/typescript/lib/lib.es2020.intl.d.ts","./node_modules/typescript/lib/lib.es2020.number.d.ts","./node_modules/typescript/lib/lib.es2021.promise.d.ts","./node_modules/typescript/lib/lib.es2021.string.d.ts","./node_modules/typescript/lib/lib.es2021.weakref.d.ts","./node_modules/typescript/lib/lib.es2021.intl.d.ts","./node_modules/typescript/lib/lib.es2022.array.d.ts","./node_modules/typescript/lib/lib.es2022.error.d.ts","./node_modules/typescript/lib/lib.es2022.intl.d.ts","./node_modules/typescript/lib/lib.es2022.object.d.ts","./node_modules/typescript/lib/lib.es2022.string.d.ts","./node_modules/typescript/lib/lib.es2022.regexp.d.ts","./node_modules/typescript/lib/lib.es2023.array.d.ts","./node_modules/typescript/lib/lib.es2023.collection.d.ts","./node_modules/typescript/lib/lib.es2023.intl.d.ts","./node_modules/typescript/lib/lib.es2024.arraybuffer.d.ts","./node_modules/typescript/lib/lib.es2024.collection.d.ts","./node_modules/typescript/lib/lib.es2024.object.d.ts","./node_modules/typescript/lib/lib.es2024.promise.d.ts","./node_modules/typescript/lib/lib.es2024.regexp.d.ts","./node_modules/typescript/lib/lib.es2024.sharedmemory.d.ts","./node_modules/typescript/lib/lib.es2024.string.d.ts","./node_modules/typescript/lib/lib.esnext.array.d.ts","./node_modules/typescript/lib/lib.esnext.collection.d.ts","./node_modules/typescript/lib/lib.esnext.intl.d.ts","./node_modules/typescript/lib/lib.esnext.disposable.d.ts","./node_modules/typescript/lib/lib.esnext.promise.d.ts","./node_modules/typescript/lib/lib.esnext.decorators.d.ts","./node_modules/typescript/lib/lib.esnext.iterator.d.ts","./node_modules/typescript/lib/lib.esnext.float16.d.ts","./node_modules/typescript/lib/lib.esnext.error.d.ts","./node_modules/typescript/lib/lib.esnext.sharedmemory.d.ts","./node_modules/typescript/lib/lib.decorators.d.ts","./node_modules/typescript/lib/lib.decorators.legacy.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/types.gen.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/core/auth.gen.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/core/pathserializer.gen.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/core/bodyserializer.gen.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/core/types.gen.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/core/serversentevents.gen.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/client/utils.gen.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/client/types.gen.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/core/params.gen.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/client/client.gen.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/client/index.d.ts","./node_modules/@opencode-ai/sdk/dist/gen/sdk.gen.d.ts","./node_modules/@opencode-ai/sdk/dist/client.d.ts","./node_modules/@opencode-ai/sdk/dist/server.d.ts","./node_modules/@opencode-ai/sdk/dist/index.d.ts","./node_modules/@opencode-ai/plugin/dist/shell.d.ts","./node_modules/zod/v4/core/standard-schema.d.cts","./node_modules/zod/v4/core/util.d.cts","./node_modules/zod/v4/core/versions.d.cts","./node_modules/zod/v4/core/schemas.d.cts","./node_modules/zod/v4/core/checks.d.cts","./node_modules/zod/v4/core/errors.d.cts","./node_modules/zod/v4/core/core.d.cts","./node_modules/zod/v4/core/parse.d.cts","./node_modules/zod/v4/core/regexes.d.cts","./node_modules/zod/v4/locales/ar.d.cts","./node_modules/zod/v4/locales/az.d.cts","./node_modules/zod/v4/locales/be.d.cts","./node_modules/zod/v4/locales/ca.d.cts","./node_modules/zod/v4/locales/cs.d.cts","./node_modules/zod/v4/locales/da.d.cts","./node_modules/zod/v4/locales/de.d.cts","./node_modules/zod/v4/locales/en.d.cts","./node_modules/zod/v4/locales/eo.d.cts","./node_modules/zod/v4/locales/es.d.cts","./node_modules/zod/v4/locales/fa.d.cts","./node_modules/zod/v4/locales/fi.d.cts","./node_modules/zod/v4/locales/fr.d.cts","./node_modules/zod/v4/locales/fr-ca.d.cts","./node_modules/zod/v4/locales/he.d.cts","./node_modules/zod/v4/locales/hu.d.cts","./node_modules/zod/v4/locales/id.d.cts","./node_modules/zod/v4/locales/is.d.cts","./node_modules/zod/v4/locales/it.d.cts","./node_modules/zod/v4/locales/ja.d.cts","./node_modules/zod/v4/locales/ka.d.cts","./node_modules/zod/v4/locales/kh.d.cts","./node_modules/zod/v4/locales/km.d.cts","./node_modules/zod/v4/locales/ko.d.cts","./node_modules/zod/v4/locales/lt.d.cts","./node_modules/zod/v4/locales/mk.d.cts","./node_modules/zod/v4/locales/ms.d.cts","./node_modules/zod/v4/locales/nl.d.cts","./node_modules/zod/v4/locales/no.d.cts","./node_modules/zod/v4/locales/ota.d.cts","./node_modules/zod/v4/locales/ps.d.cts","./node_modules/zod/v4/locales/pl.d.cts","./node_modules/zod/v4/locales/pt.d.cts","./node_modules/zod/v4/locales/ru.d.cts","./node_modules/zod/v4/locales/sl.d.cts","./node_modules/zod/v4/locales/sv.d.cts","./node_modules/zod/v4/locales/ta.d.cts","./node_modules/zod/v4/locales/th.d.cts","./node_modules/zod/v4/locales/tr.d.cts","./node_modules/zod/v4/locales/ua.d.cts","./node_modules/zod/v4/locales/uk.d.cts","./node_modules/zod/v4/locales/ur.d.cts","./node_modules/zod/v4/locales/vi.d.cts","./node_modules/zod/v4/locales/zh-cn.d.cts","./node_modules/zod/v4/locales/zh-tw.d.cts","./node_modules/zod/v4/locales/yo.d.cts","./node_modules/zod/v4/locales/index.d.cts","./node_modules/zod/v4/core/registries.d.cts","./node_modules/zod/v4/core/doc.d.cts","./node_modules/zod/v4/core/api.d.cts","./node_modules/zod/v4/core/json-schema.d.cts","./node_modules/zod/v4/core/to-json-schema.d.cts","./node_modules/zod/v4/core/index.d.cts","./node_modules/zod/v4/classic/errors.d.cts","./node_modules/zod/v4/classic/parse.d.cts","./node_modules/zod/v4/classic/schemas.d.cts","./node_modules/zod/v4/classic/checks.d.cts","./node_modules/zod/v4/classic/compat.d.cts","./node_modules/zod/v4/classic/iso.d.cts","./node_modules/zod/v4/classic/coerce.d.cts","./node_modules/zod/v4/classic/external.d.cts","./node_modules/zod/index.d.cts","./node_modules/@opencode-ai/plugin/dist/tool.d.ts","./node_modules/@opencode-ai/plugin/dist/index.d.ts","./src/utils/detection.ts","./src/prompts.ts","./src/config.ts","./src/index.ts","./node_modules/undici-types/header.d.ts","./node_modules/undici-types/readable.d.ts","./node_modules/@types/node/compatibility/disposable.d.ts","./node_modules/@types/node/compatibility/indexable.d.ts","./node_modules/@types/node/compatibility/iterators.d.ts","./node_modules/@types/node/compatibility/index.d.ts","./node_modules/@types/node/globals.typedarray.d.ts","./node_modules/@types/node/buffer.buffer.d.ts","./node_modules/@types/node/globals.d.ts","./node_modules/@types/node/web-globals/abortcontroller.d.ts","./node_modules/@types/node/web-globals/domexception.d.ts","./node_modules/@types/node/web-globals/events.d.ts","./node_modules/@types/node/web-globals/fetch.d.ts","./node_modules/@types/node/assert.d.ts","./node_modules/@types/node/assert/strict.d.ts","./node_modules/@types/node/async_hooks.d.ts","./node_modules/@types/node/buffer.d.ts","./node_modules/@types/node/child_process.d.ts","./node_modules/@types/node/cluster.d.ts","./node_modules/@types/node/console.d.ts","./node_modules/@types/node/constants.d.ts","./node_modules/@types/node/crypto.d.ts","./node_modules/@types/node/dgram.d.ts","./node_modules/@types/node/diagnostics_channel.d.ts","./node_modules/@types/node/dns.d.ts","./node_modules/@types/node/dns/promises.d.ts","./node_modules/@types/node/domain.d.ts","./node_modules/@types/node/events.d.ts","./node_modules/@types/node/fs.d.ts","./node_modules/@types/node/fs/promises.d.ts","./node_modules/@types/node/http.d.ts","./node_modules/@types/node/http2.d.ts","./node_modules/@types/node/https.d.ts","./node_modules/@types/node/inspector.generated.d.ts","./node_modules/@types/node/module.d.ts","./node_modules/@types/node/net.d.ts","./node_modules/@types/node/os.d.ts","./node_modules/@types/node/path.d.ts","./node_modules/@types/node/perf_hooks.d.ts","./node_modules/@types/node/process.d.ts","./node_modules/@types/node/punycode.d.ts","./node_modules/@types/node/querystring.d.ts","./node_modules/@types/node/readline.d.ts","./node_modules/@types/node/readline/promises.d.ts","./node_modules/@types/node/repl.d.ts","./node_modules/@types/node/sea.d.ts","./node_modules/@types/node/stream.d.ts","./node_modules/@types/node/stream/promises.d.ts","./node_modules/@types/node/stream/consumers.d.ts","./node_modules/@types/node/stream/web.d.ts","./node_modules/@types/node/string_decoder.d.ts","./node_modules/@types/node/test.d.ts","./node_modules/@types/node/timers.d.ts","./node_modules/@types/node/timers/promises.d.ts","./node_modules/@types/node/tls.d.ts","./node_modules/@types/node/trace_events.d.ts","./node_modules/@types/node/tty.d.ts","./node_modules/@types/node/url.d.ts","./node_modules/@types/node/util.d.ts","./node_modules/@types/node/v8.d.ts","./node_modules/@types/node/vm.d.ts","./node_modules/@types/node/wasi.d.ts","./node_modules/@types/node/worker_threads.d.ts","./node_modules/@types/node/zlib.d.ts","./node_modules/@types/node/index.d.ts","./node_modules/undici-types/file.d.ts","./node_modules/undici-types/fetch.d.ts","./node_modules/undici-types/formdata.d.ts","./node_modules/undici-types/connector.d.ts","./node_modules/undici-types/client.d.ts","./node_modules/undici-types/errors.d.ts","./node_modules/undici-types/dispatcher.d.ts","./node_modules/undici-types/global-dispatcher.d.ts","./node_modules/undici-types/global-origin.d.ts","./node_modules/undici-types/pool-stats.d.ts","./node_modules/undici-types/pool.d.ts","./node_modules/undici-types/handlers.d.ts","./node_modules/undici-types/balanced-pool.d.ts","./node_modules/undici-types/agent.d.ts","./node_modules/undici-types/mock-interceptor.d.ts","./node_modules/undici-types/mock-agent.d.ts","./node_modules/undici-types/mock-client.d.ts","./node_modules/undici-types/mock-pool.d.ts","./node_modules/undici-types/mock-errors.d.ts","./node_modules/undici-types/proxy-agent.d.ts","./node_modules/undici-types/env-http-proxy-agent.d.ts","./node_modules/undici-types/retry-handler.d.ts","./node_modules/undici-types/retry-agent.d.ts","./node_modules/undici-types/api.d.ts","./node_modules/undici-types/interceptors.d.ts","./node_modules/undici-types/util.d.ts","./node_modules/undici-types/cookies.d.ts","./node_modules/undici-types/patch.d.ts","./node_modules/undici-types/websocket.d.ts","./node_modules/undici-types/eventsource.d.ts","./node_modules/undici-types/filereader.d.ts","./node_modules/undici-types/diagnostics-channel.d.ts","./node_modules/undici-types/content-type.d.ts","./node_modules/undici-types/cache.d.ts","./node_modules/undici-types/index.d.ts","./node_modules/bun-types/globals.d.ts","./node_modules/bun-types/s3.d.ts","./node_modules/bun-types/fetch.d.ts","./node_modules/bun-types/bun.d.ts","./node_modules/bun-types/extensions.d.ts","./node_modules/bun-types/devserver.d.ts","./node_modules/bun-types/ffi.d.ts","./node_modules/bun-types/html-rewriter.d.ts","./node_modules/bun-types/jsc.d.ts","./node_modules/bun-types/sqlite.d.ts","./node_modules/bun-types/vendor/expect-type/utils.d.ts","./node_modules/bun-types/vendor/expect-type/overloads.d.ts","./node_modules/bun-types/vendor/expect-type/branding.d.ts","./node_modules/bun-types/vendor/expect-type/messages.d.ts","./node_modules/bun-types/vendor/expect-type/index.d.ts","./node_modules/bun-types/test.d.ts","./node_modules/bun-types/wasm.d.ts","./node_modules/bun-types/overrides.d.ts","./node_modules/bun-types/deprecated.d.ts","./node_modules/bun-types/redis.d.ts","./node_modules/bun-types/shell.d.ts","./node_modules/bun-types/serve.d.ts","./node_modules/bun-types/sql.d.ts","./node_modules/bun-types/security.d.ts","./node_modules/bun-types/bundle.d.ts","./node_modules/bun-types/bun.ns.d.ts","./node_modules/bun-types/index.d.ts"],"fileIdsList":[[95,96,168,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[167,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[81,88,92,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[88,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[82,84,87,88,89,90,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[82,85,86,87,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[84,88,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[83,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[85,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[82,84,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[81,86,88,91,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[93,94,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[81,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,187,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,189,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[190,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,195,223,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,191,196,201,209,220,231,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,191,192,201,209,274,275,276,277,279,290,291,292,293,294,295,296,297],[176,177,178,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,193,232,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,194,195,202,210,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,195,220,228,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,196,198,201,209,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,189,190,197,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,198,199,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,200,201,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,189,190,201,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,201,202,203,220,231,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,201,202,203,216,220,223,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,198,201,204,209,220,231,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,201,202,204,205,209,220,228,231,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,204,206,220,228,231,274,275,276,277,279,290,291,292,293,294,295,296,297],[179,180,181,182,183,184,185,186,187,188,189,190,191,192,193,194,195,196,197,198,199,200,201,202,203,204,205,206,207,208,209,210,211,212,213,214,215,216,217,218,219,220,221,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,237,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,201,207,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,208,231,236,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,198,201,209,220,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,210,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,211,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,189,190,212,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,187,188,189,190,191,192,193,194,195,196,197,198,199,200,201,202,203,204,205,206,207,208,209,210,211,212,213,214,215,216,217,218,219,220,221,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,237,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,214,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,215,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,201,216,217,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,216,218,232,234,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,201,220,221,223,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,222,223,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,220,221,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,223,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,224,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,187,190,220,225,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,201,226,227,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,226,227,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,195,209,220,228,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,229,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,209,230,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,204,215,231,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,195,232,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,220,233,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,208,234,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,235,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,273,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,201,203,212,220,223,231,234,236,273,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,220,237,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,195,202,204,228,232,236,273,274,275,276,279,280,290,291,292,293,294,295,296,297],[181,190,274,275,276,277,279,290,291,293,294,295,296,297],[181,190,274,275,276,277,290,291,292,293,294,295,296,297],[181,190,273,274,275,277,279,290,291,292,293,294,295,296,297],[181,190,195,212,220,223,228,232,236,273,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,238,274,275,276,277,278,279,280,281,282,283,289,290,291,292,293,294,295,296,297,298,299],[181,182,190,193,195,202,203,210,223,228,231,237,274,275,276,277,279,290,292,293,294,295,296,297],[181,190,274,275,276,277,279,290,291,292,294,295,296,297],[181,190,202,274,276,277,279,290,291,292,293,294,295,296,297],[181,190,274,275,276,277,279,290,291,292,293,294,295,296],[181,190,274,275,276,277,279,290,291,292,293,294,296,297],[181,190,274,275,276,277,279,290,291,292,293,295,296,297],[181,190,274,275,276,277,279,283,290,291,292,293,294,295,297],[181,190,274,275,276,277,279,288,290,291,292,293,294,295,296,297],[181,190,274,275,276,277,279,284,285,290,291,292,293,294,295,296,297],[181,190,274,275,276,277,279,284,285,286,287,290,291,292,293,294,295,296,297],[181,190,274,275,276,277,279,284,286,290,291,292,293,294,295,296,297],[181,190,274,275,276,277,279,284,290,291,292,293,294,295,296,297],[181,190,274,275,276,277,279,291,292,293,294,295,296,297],[181,190,231,245,249,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,220,231,245,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,240,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,228,231,242,245,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,209,228,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,238,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,238,240,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,209,231,242,245,274,275,276,277,279,290,291,292,293,294,295,296,297],[174,175,181,190,201,220,231,241,244,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,245,252,274,275,276,277,279,290,291,292,293,294,295,296,297],[174,181,190,243,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,245,266,267,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,223,231,238,241,245,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,238,266,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,238,239,240,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,245,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,239,240,241,242,243,244,245,246,247,249,250,251,252,253,254,255,256,257,258,259,260,261,262,263,264,265,267,268,269,270,271,272,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,245,260,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,245,252,253,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,243,245,253,254,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,244,274,275,276,277,279,290,291,292,293,294,295,296,297],[174,181,190,240,245,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,245,249,253,254,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,249,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,231,243,245,248,274,275,276,277,279,290,291,292,293,294,295,296,297],[174,181,190,242,245,252,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,220,274,275,276,277,279,290,291,292,293,294,295,296,297],[181,190,236,238,240,245,266,274,275,276,277,279,290,291,292,293,294,295,296,297],[166,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[158,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[158,161,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[152,158,159,160,161,162,163,164,165,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[158,159,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[158,160,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[98,100,101,102,103,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[98,100,102,103,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[98,100,102,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[97,98,100,101,103,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[98,99,100,101,102,103,104,105,152,153,154,155,156,157,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[100,103,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[97,98,99,101,102,103,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[100,153,156,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[100,101,102,103,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[102,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[106,107,108,109,110,111,112,113,114,115,116,117,118,119,120,121,122,123,124,125,126,127,128,129,130,131,132,133,134,135,136,137,138,139,140,141,142,143,144,145,146,147,148,149,150,151,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[169,170,171,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[169,170,172,181,190,274,275,276,277,279,290,291,292,293,294,295,296,297],[169,181,190,202,211,274,275,276,277,279,290,291,292,293,294,295,296,297]],"fileInfos":[{"version":"c430d44666289dae81f30fa7b2edebf186ecc91a2d4c71266ea6ae76388792e1","affectsGlobalScope":true,"impliedFormat":1},{"version":"45b7ab580deca34ae9729e97c13cfd999df04416a79116c3bfb483804f85ded4","impliedFormat":1},{"version":"3facaf05f0c5fc569c5649dd359892c98a85557e3e0c847964caeb67076f4d75","impliedFormat":1},{"version":"e44bb8bbac7f10ecc786703fe0a6a4b952189f908707980ba8f3c8975a760962","impliedFormat":1},{"version":"5e1c4c362065a6b95ff952c0eab010f04dcd2c3494e813b493ecfd4fcb9fc0d8","impliedFormat":1},{"version":"68d73b4a11549f9c0b7d352d10e91e5dca8faa3322bfb77b661839c42b1ddec7","impliedFormat":1},{"version":"5efce4fc3c29ea84e8928f97adec086e3dc876365e0982cc8479a07954a3efd4","impliedFormat":1},{"version":"feecb1be483ed332fad555aff858affd90a48ab19ba7272ee084704eb7167569","impliedFormat":1},{"version":"ee7bad0c15b58988daa84371e0b89d313b762ab83cb5b31b8a2d1162e8eb41c2","impliedFormat":1},{"version":"27bdc30a0e32783366a5abeda841bc22757c1797de8681bbe81fbc735eeb1c10","impliedFormat":1},{"version":"8fd575e12870e9944c7e1d62e1f5a73fcf23dd8d3a321f2a2c74c20d022283fe","impliedFormat":1},{"version":"2ab096661c711e4a81cc464fa1e6feb929a54f5340b46b0a07ac6bbf857471f0","impliedFormat":1},{"version":"c57796738e7f83dbc4b8e65132f11a377649c00dd3eee333f672b8f0a6bea671","affectsGlobalScope":true,"impliedFormat":1},{"version":"dc2df20b1bcdc8c2d34af4926e2c3ab15ffe1160a63e58b7e09833f616efff44","affectsGlobalScope":true,"impliedFormat":1},{"version":"515d0b7b9bea2e31ea4ec968e9edd2c39d3eebf4a2d5cbd04e88639819ae3b71","affectsGlobalScope":true,"impliedFormat":1},{"version":"0559b1f683ac7505ae451f9a96ce4c3c92bdc71411651ca6ddb0e88baaaad6a3","affectsGlobalScope":true,"impliedFormat":1},{"version":"0dc1e7ceda9b8b9b455c3a2d67b0412feab00bd2f66656cd8850e8831b08b537","affectsGlobalScope":true,"impliedFormat":1},{"version":"ce691fb9e5c64efb9547083e4a34091bcbe5bdb41027e310ebba8f7d96a98671","affectsGlobalScope":true,"impliedFormat":1},{"version":"8d697a2a929a5fcb38b7a65594020fcef05ec1630804a33748829c5ff53640d0","affectsGlobalScope":true,"impliedFormat":1},{"version":"4ff2a353abf8a80ee399af572debb8faab2d33ad38c4b4474cff7f26e7653b8d","affectsGlobalScope":true,"impliedFormat":1},{"version":"fb0f136d372979348d59b3f5020b4cdb81b5504192b1cacff5d1fbba29378aa1","affectsGlobalScope":true,"impliedFormat":1},{"version":"d15bea3d62cbbdb9797079416b8ac375ae99162a7fba5de2c6c505446486ac0a","affectsGlobalScope":true,"impliedFormat":1},{"version":"68d18b664c9d32a7336a70235958b8997ebc1c3b8505f4f1ae2b7e7753b87618","affectsGlobalScope":true,"impliedFormat":1},{"version":"eb3d66c8327153d8fa7dd03f9c58d351107fe824c79e9b56b462935176cdf12a","affectsGlobalScope":true,"impliedFormat":1},{"version":"38f0219c9e23c915ef9790ab1d680440d95419ad264816fa15009a8851e79119","affectsGlobalScope":true,"impliedFormat":1},{"version":"69ab18c3b76cd9b1be3d188eaf8bba06112ebbe2f47f6c322b5105a6fbc45a2e","affectsGlobalScope":true,"impliedFormat":1},{"version":"a680117f487a4d2f30ea46f1b4b7f58bef1480456e18ba53ee85c2746eeca012","affectsGlobalScope":true,"impliedFormat":1},{"version":"2f11ff796926e0832f9ae148008138ad583bd181899ab7dd768a2666700b1893","affectsGlobalScope":true,"impliedFormat":1},{"version":"4de680d5bb41c17f7f68e0419412ca23c98d5749dcaaea1896172f06435891fc","affectsGlobalScope":true,"impliedFormat":1},{"version":"954296b30da6d508a104a3a0b5d96b76495c709785c1d11610908e63481ee667","affectsGlobalScope":true,"impliedFormat":1},{"version":"ac9538681b19688c8eae65811b329d3744af679e0bdfa5d842d0e32524c73e1c","affectsGlobalScope":true,"impliedFormat":1},{"version":"0a969edff4bd52585473d24995c5ef223f6652d6ef46193309b3921d65dd4376","affectsGlobalScope":true,"impliedFormat":1},{"version":"9e9fbd7030c440b33d021da145d3232984c8bb7916f277e8ffd3dc2e3eae2bdb","affectsGlobalScope":true,"impliedFormat":1},{"version":"811ec78f7fefcabbda4bfa93b3eb67d9ae166ef95f9bff989d964061cbf81a0c","affectsGlobalScope":true,"impliedFormat":1},{"version":"717937616a17072082152a2ef351cb51f98802fb4b2fdabd32399843875974ca","affectsGlobalScope":true,"impliedFormat":1},{"version":"d7e7d9b7b50e5f22c915b525acc5a49a7a6584cf8f62d0569e557c5cfc4b2ac2","affectsGlobalScope":true,"impliedFormat":1},{"version":"71c37f4c9543f31dfced6c7840e068c5a5aacb7b89111a4364b1d5276b852557","affectsGlobalScope":true,"impliedFormat":1},{"version":"576711e016cf4f1804676043e6a0a5414252560eb57de9faceee34d79798c850","affectsGlobalScope":true,"impliedFormat":1},{"version":"89c1b1281ba7b8a96efc676b11b264de7a8374c5ea1e6617f11880a13fc56dc6","affectsGlobalScope":true,"impliedFormat":1},{"version":"74f7fa2d027d5b33eb0471c8e82a6c87216223181ec31247c357a3e8e2fddc5b","affectsGlobalScope":true,"impliedFormat":1},{"version":"d6d7ae4d1f1f3772e2a3cde568ed08991a8ae34a080ff1151af28b7f798e22ca","affectsGlobalScope":true,"impliedFormat":1},{"version":"063600664504610fe3e99b717a1223f8b1900087fab0b4cad1496a114744f8df","affectsGlobalScope":true,"impliedFormat":1},{"version":"934019d7e3c81950f9a8426d093458b65d5aff2c7c1511233c0fd5b941e608ab","affectsGlobalScope":true,"impliedFormat":1},{"version":"52ada8e0b6e0482b728070b7639ee42e83a9b1c22d205992756fe020fd9f4a47","affectsGlobalScope":true,"impliedFormat":1},{"version":"3bdefe1bfd4d6dee0e26f928f93ccc128f1b64d5d501ff4a8cf3c6371200e5e6","affectsGlobalScope":true,"impliedFormat":1},{"version":"59fb2c069260b4ba00b5643b907ef5d5341b167e7d1dbf58dfd895658bda2867","affectsGlobalScope":true,"impliedFormat":1},{"version":"639e512c0dfc3fad96a84caad71b8834d66329a1f28dc95e3946c9b58176c73a","affectsGlobalScope":true,"impliedFormat":1},{"version":"368af93f74c9c932edd84c58883e736c9e3d53cec1fe24c0b0ff451f529ceab1","affectsGlobalScope":true,"impliedFormat":1},{"version":"af3dd424cf267428f30ccfc376f47a2c0114546b55c44d8c0f1d57d841e28d74","affectsGlobalScope":true,"impliedFormat":1},{"version":"995c005ab91a498455ea8dfb63aa9f83fa2ea793c3d8aa344be4a1678d06d399","affectsGlobalScope":true,"impliedFormat":1},{"version":"959d36cddf5e7d572a65045b876f2956c973a586da58e5d26cde519184fd9b8a","affectsGlobalScope":true,"impliedFormat":1},{"version":"965f36eae237dd74e6cca203a43e9ca801ce38824ead814728a2807b1910117d","affectsGlobalScope":true,"impliedFormat":1},{"version":"3925a6c820dcb1a06506c90b1577db1fdbf7705d65b62b99dce4be75c637e26b","affectsGlobalScope":true,"impliedFormat":1},{"version":"0a3d63ef2b853447ec4f749d3f368ce642264246e02911fcb1590d8c161b8005","affectsGlobalScope":true,"impliedFormat":1},{"version":"8cdf8847677ac7d20486e54dd3fcf09eda95812ac8ace44b4418da1bbbab6eb8","affectsGlobalScope":true,"impliedFormat":1},{"version":"8444af78980e3b20b49324f4a16ba35024fef3ee069a0eb67616ea6ca821c47a","affectsGlobalScope":true,"impliedFormat":1},{"version":"3287d9d085fbd618c3971944b65b4be57859f5415f495b33a6adc994edd2f004","affectsGlobalScope":true,"impliedFormat":1},{"version":"b4b67b1a91182421f5df999988c690f14d813b9850b40acd06ed44691f6727ad","affectsGlobalScope":true,"impliedFormat":1},{"version":"df83c2a6c73228b625b0beb6669c7ee2a09c914637e2d35170723ad49c0f5cd4","affectsGlobalScope":true,"impliedFormat":1},{"version":"436aaf437562f276ec2ddbee2f2cdedac7664c1e4c1d2c36839ddd582eeb3d0a","affectsGlobalScope":true,"impliedFormat":1},{"version":"8e3c06ea092138bf9fa5e874a1fdbc9d54805d074bee1de31b99a11e2fec239d","affectsGlobalScope":true,"impliedFormat":1},{"version":"87dc0f382502f5bbce5129bdc0aea21e19a3abbc19259e0b43ae038a9fc4e326","affectsGlobalScope":true,"impliedFormat":1},{"version":"b1cb28af0c891c8c96b2d6b7be76bd394fddcfdb4709a20ba05a7c1605eea0f9","affectsGlobalScope":true,"impliedFormat":1},{"version":"2fef54945a13095fdb9b84f705f2b5994597640c46afeb2ce78352fab4cb3279","affectsGlobalScope":true,"impliedFormat":1},{"version":"ac77cb3e8c6d3565793eb90a8373ee8033146315a3dbead3bde8db5eaf5e5ec6","affectsGlobalScope":true,"impliedFormat":1},{"version":"56e4ed5aab5f5920980066a9409bfaf53e6d21d3f8d020c17e4de584d29600ad","affectsGlobalScope":true,"impliedFormat":1},{"version":"4ece9f17b3866cc077099c73f4983bddbcb1dc7ddb943227f1ec070f529dedd1","affectsGlobalScope":true,"impliedFormat":1},{"version":"0a6282c8827e4b9a95f4bf4f5c205673ada31b982f50572d27103df8ceb8013c","affectsGlobalScope":true,"impliedFormat":1},{"version":"1c9319a09485199c1f7b0498f2988d6d2249793ef67edda49d1e584746be9032","affectsGlobalScope":true,"impliedFormat":1},{"version":"e3a2a0cee0f03ffdde24d89660eba2685bfbdeae955a6c67e8c4c9fd28928eeb","affectsGlobalScope":true,"impliedFormat":1},{"version":"811c71eee4aa0ac5f7adf713323a5c41b0cf6c4e17367a34fbce379e12bbf0a4","affectsGlobalScope":true,"impliedFormat":1},{"version":"51ad4c928303041605b4d7ae32e0c1ee387d43a24cd6f1ebf4a2699e1076d4fa","affectsGlobalScope":true,"impliedFormat":1},{"version":"60037901da1a425516449b9a20073aa03386cce92f7a1fd902d7602be3a7c2e9","affectsGlobalScope":true,"impliedFormat":1},{"version":"d4b1d2c51d058fc21ec2629fff7a76249dec2e36e12960ea056e3ef89174080f","affectsGlobalScope":true,"impliedFormat":1},{"version":"22adec94ef7047a6c9d1af3cb96be87a335908bf9ef386ae9fd50eeb37f44c47","affectsGlobalScope":true,"impliedFormat":1},{"version":"196cb558a13d4533a5163286f30b0509ce0210e4b316c56c38d4c0fd2fb38405","affectsGlobalScope":true,"impliedFormat":1},{"version":"73f78680d4c08509933daf80947902f6ff41b6230f94dd002ae372620adb0f60","affectsGlobalScope":true,"impliedFormat":1},{"version":"c5239f5c01bcfa9cd32f37c496cf19c61d69d37e48be9de612b541aac915805b","affectsGlobalScope":true,"impliedFormat":1},{"version":"8e7f8264d0fb4c5339605a15daadb037bf238c10b654bb3eee14208f860a32ea","affectsGlobalScope":true,"impliedFormat":1},{"version":"782dec38049b92d4e85c1585fbea5474a219c6984a35b004963b00beb1aab538","affectsGlobalScope":true,"impliedFormat":1},{"version":"e533f9d5102c7a047291eba18ef68c132e883a9101826aee664d3a1d22d68b0e","impliedFormat":99},{"version":"96135fbf1dab9f857c9df88a191fdc435517f7a5b021d09cad0c290a1dcec436","impliedFormat":99},{"version":"c36ce4ffa39eae9d8d5f3262bbff170f55b375156bc4eeadadfeede2820aa4c2","impliedFormat":99},{"version":"44fd7a3b2fac384971e638548c227835a2f8c3c2232edff7e0f951a0e136e562","impliedFormat":99},{"version":"80498d31235c22ee711a9fb6f0ca2bcb8fbf5c9a787f3d3a76962b440df70013","impliedFormat":99},{"version":"c41ccf7ec4fecea9d6e5b72ae7f20c91bbe85a8a82caada5ae8a3bc5a56c8926","impliedFormat":99},{"version":"77fed0312d2ec2b08afb890b7760b5f2328f6e09065ea05b5fc35d5294fbb434","impliedFormat":99},{"version":"11f6e61b31e0c89eae79ce790895bf85b918572c32c2de45998f7cbacbb3b963","impliedFormat":99},{"version":"d23fac93be3afb1c83269864f13b5e5fc246b05cb21b7e019e38403e4ef1aba5","impliedFormat":99},{"version":"358599da1d31542e1d3ebf35a4a98665395ea8eaf0063826b8838974f7e826e5","impliedFormat":99},{"version":"06240cf9eca32a7ea5436ee67c8420bc1ddacfaf0d28cd17d09dd56a65c8fb9a","impliedFormat":99},{"version":"5f71dae22207eb5119b2e005e046c0404a29454292e07c68668e39c9931414cd","impliedFormat":99},{"version":"84edb938f2b673df750cf83e9516218404e9df7e32e66f2f063c813d7f4d0e62","impliedFormat":99},{"version":"579f61b51c0e5dc0119ab49fc536130749280c7a23e489e753fcce08c229ce2c","impliedFormat":99},{"version":"2a7a35e3f1ead49569fe0423ae65055184f9384b97999cd90bb99878f727e585","impliedFormat":99},{"version":"340d8057526217987a3a797d93a03b6e6c9a9e1cd8e7c46a676ea008a710afe1","impliedFormat":99},{"version":"309ebd217636d68cf8784cbc3272c16fb94fb8e969e18b6fe88c35200340aef1","impliedFormat":1},{"version":"f987c74a4b4baf361afbf22a16d230ee490d662f9aa2066853bb7ebbb8611355","impliedFormat":1},{"version":"1ff91526fcdd634148c655ef86e912a273ce6a0239e2505701561f086678262b","impliedFormat":1},{"version":"d9faf4a343833207c6c5cd2322fb6771b56dc1c8ece975072e85227c2d326bc2","impliedFormat":1},{"version":"8d67b13da77316a8a2fabc21d340866ddf8a4b99e76a6c951cc45189142df652","impliedFormat":1},{"version":"7952419455ca298776db0005b9b5b75571d484d526a29bfbdf041652213bce6f","impliedFormat":1},{"version":"21360500b20e0ec570f26f1cbb388c155ede043698970f316969840da4f16465","impliedFormat":1},{"version":"3a819c2928ee06bbcc84e2797fd3558ae2ebb7e0ed8d87f71732fb2e2acc87b4","impliedFormat":1},{"version":"1765e61249cb44bf5064d42bfa06956455bbc74dc05f074d5727e8962592c920","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"26384fb401f582cae1234213c3dc75fdc80e3d728a0a1c55b405be8a0c6dddbe","impliedFormat":1},{"version":"26384fb401f582cae1234213c3dc75fdc80e3d728a0a1c55b405be8a0c6dddbe","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"26384fb401f582cae1234213c3dc75fdc80e3d728a0a1c55b405be8a0c6dddbe","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"26384fb401f582cae1234213c3dc75fdc80e3d728a0a1c55b405be8a0c6dddbe","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"26384fb401f582cae1234213c3dc75fdc80e3d728a0a1c55b405be8a0c6dddbe","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"26384fb401f582cae1234213c3dc75fdc80e3d728a0a1c55b405be8a0c6dddbe","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"e0bfe601a9fdf6defe94ed62dc60ac71597566001a1f86e705c95e431a9c816d","impliedFormat":1},{"version":"7b9e6b3c726d47935bdc9ebc78fe5398e28e751ba7d70e9e011f01fbd5b618be","impliedFormat":1},{"version":"6e5857f38aa297a859cab4ec891408659218a5a2610cd317b6dcbef9979459cc","impliedFormat":1},{"version":"add0ce7b77ba5b308492fa68f77f24d1ed1d9148534bdf05ac17c30763fc1a79","impliedFormat":1},{"version":"56ccc6238510b913f5e6c21afdc447632873f76748d0b30a87cb313b42f1c196","impliedFormat":1},{"version":"c1a2e05eb6d7ca8d7e4a7f4c93ccf0c2857e842a64c98eaee4d85841ee9855e6","impliedFormat":1},{"version":"85021a58f728318a9c83977a8a3a09196dcfc61345e0b8bbbb39422c1594f36b","impliedFormat":1},{"version":"d91805544905a40fbd639ba1b85f65dc13d6996a07034848d634aa9edb63479e","impliedFormat":1},{"version":"6042774c61ece4ba77b3bf375f15942eb054675b7957882a00c22c0e4fe5865c","impliedFormat":1},{"version":"5a3bd57ed7a9d9afef74c75f77fce79ba3c786401af9810cdf45907c4e93f30e","impliedFormat":1},{"version":"8610f5dc475d74c4b095aafa0c191548bfd43f65802e6da54b5e526202b8cfe0","impliedFormat":1},{"version":"7b9496d2e1664155c3c293e1fbbe2aba288614163c88cb81ed6061905924b8f9","impliedFormat":1},{"version":"e27451b24234dfed45f6cf22112a04955183a99c42a2691fb4936d63cfe42761","impliedFormat":1},{"version":"58d65a2803c3b6629b0e18c8bf1bc883a686fcf0333230dd0151ab6e85b74307","impliedFormat":1},{"version":"e818471014c77c103330aee11f00a7a00b37b35500b53ea6f337aefacd6174c9","impliedFormat":1},{"version":"2fbc91ba70096f93f57e22d1f0af22b707dbb3f9f5692cc4f1200861d3b75d88","impliedFormat":1},{"version":"29f823cbe0166e10e7176a94afe609a24b9e5af3858628c541ff8ce1727023cd","impliedFormat":1},{"version":"6894a380b126ba4670f991282a7e26d556a6da1e70cb1fc25068b8da45b4c0d7","impliedFormat":99},{"version":"9c2e0b7b184613de664f70e6685e6031c0cb08cbac93757ca63d3b0979116188","impliedFormat":99},"986cc1e72a96ba3f2c51f410b231502f59c6ba832af0aade0e642226f188189e","fd8126980980abfb00c0f4ada333358c7b403141ad127103de5dab76fd55d37b",{"version":"306bde0ca3235a14bc325d741742dde785f3749f3b6fb59ef22ccd137615bbef","signature":"779e89e031c2d2444bd3c8f88cc133eb3cb8619aad2b1d5340aca7fbd3f7b795"},{"version":"27417454ce3b588c11edd84b76b2b59a7999a35d1adda9ab090bd15065f7fe6d","signature":"78e177db97107b210f5715d71cd0b36f16af085c0a446a1cfbc561752da207e7"},{"version":"5929864ce17fba74232584d90cb721a89b7ad277220627cc97054ba15a98ea8f","impliedFormat":1},{"version":"763fe0f42b3d79b440a9b6e51e9ba3f3f91352469c1e4b3b67bfa4ff6352f3f4","impliedFormat":1},{"version":"70521b6ab0dcba37539e5303104f29b721bfb2940b2776da4cc818c07e1fefc1","affectsGlobalScope":true,"impliedFormat":1},{"version":"ab41ef1f2cdafb8df48be20cd969d875602483859dc194e9c97c8a576892c052","affectsGlobalScope":true,"impliedFormat":1},{"version":"d153a11543fd884b596587ccd97aebbeed950b26933ee000f94009f1ab142848","affectsGlobalScope":true,"impliedFormat":1},{"version":"21d819c173c0cf7cc3ce57c3276e77fd9a8a01d35a06ad87158781515c9a438a","impliedFormat":1},{"version":"98cffbf06d6bab333473c70a893770dbe990783904002c4f1a960447b4b53dca","affectsGlobalScope":true,"impliedFormat":1},{"version":"ba481bca06f37d3f2c137ce343c7d5937029b2468f8e26111f3c9d9963d6568d","affectsGlobalScope":true,"impliedFormat":1},{"version":"6d9ef24f9a22a88e3e9b3b3d8c40ab1ddb0853f1bfbd5c843c37800138437b61","affectsGlobalScope":true,"impliedFormat":1},{"version":"1db0b7dca579049ca4193d034d835f6bfe73096c73663e5ef9a0b5779939f3d0","affectsGlobalScope":true,"impliedFormat":1},{"version":"9798340ffb0d067d69b1ae5b32faa17ab31b82466a3fc00d8f2f2df0c8554aaa","affectsGlobalScope":true,"impliedFormat":1},{"version":"f26b11d8d8e4b8028f1c7d618b22274c892e4b0ef5b3678a8ccbad85419aef43","affectsGlobalScope":true,"impliedFormat":1},{"version":"2cbe0621042e2a68c7cbce5dfed3906a1862a16a7d496010636cdbdb91341c0f","affectsGlobalScope":true,"impliedFormat":1},{"version":"e2677634fe27e87348825bb041651e22d50a613e2fdf6a4a3ade971d71bac37e","impliedFormat":1},{"version":"7394959e5a741b185456e1ef5d64599c36c60a323207450991e7a42e08911419","impliedFormat":1},{"version":"8c0bcd6c6b67b4b503c11e91a1fb91522ed585900eab2ab1f61bba7d7caa9d6f","impliedFormat":1},{"version":"8cd19276b6590b3ebbeeb030ac271871b9ed0afc3074ac88a94ed2449174b776","affectsGlobalScope":true,"impliedFormat":1},{"version":"696eb8d28f5949b87d894b26dc97318ef944c794a9a4e4f62360cd1d1958014b","impliedFormat":1},{"version":"3f8fa3061bd7402970b399300880d55257953ee6d3cd408722cb9ac20126460c","impliedFormat":1},{"version":"35ec8b6760fd7138bbf5809b84551e31028fb2ba7b6dc91d95d098bf212ca8b4","affectsGlobalScope":true,"impliedFormat":1},{"version":"5524481e56c48ff486f42926778c0a3cce1cc85dc46683b92b1271865bcf015a","impliedFormat":1},{"version":"68bd56c92c2bd7d2339457eb84d63e7de3bd56a69b25f3576e1568d21a162398","affectsGlobalScope":true,"impliedFormat":1},{"version":"3e93b123f7c2944969d291b35fed2af79a6e9e27fdd5faa99748a51c07c02d28","impliedFormat":1},{"version":"9d19808c8c291a9010a6c788e8532a2da70f811adb431c97520803e0ec649991","impliedFormat":1},{"version":"87aad3dd9752067dc875cfaa466fc44246451c0c560b820796bdd528e29bef40","impliedFormat":1},{"version":"4aacb0dd020eeaef65426153686cc639a78ec2885dc72ad220be1d25f1a439df","impliedFormat":1},{"version":"f0bd7e6d931657b59605c44112eaf8b980ba7f957a5051ed21cb93d978cf2f45","impliedFormat":1},{"version":"8db0ae9cb14d9955b14c214f34dae1b9ef2baee2fe4ce794a4cd3ac2531e3255","affectsGlobalScope":true,"impliedFormat":1},{"version":"15fc6f7512c86810273af28f224251a5a879e4261b4d4c7e532abfbfc3983134","impliedFormat":1},{"version":"58adba1a8ab2d10b54dc1dced4e41f4e7c9772cbbac40939c0dc8ce2cdb1d442","impliedFormat":1},{"version":"2fd4c143eff88dabb57701e6a40e02a4dbc36d5eb1362e7964d32028056a782b","impliedFormat":1},{"version":"714435130b9015fae551788df2a88038471a5a11eb471f27c4ede86552842bc9","impliedFormat":1},{"version":"855cd5f7eb396f5f1ab1bc0f8580339bff77b68a770f84c6b254e319bbfd1ac7","impliedFormat":1},{"version":"5650cf3dace09e7c25d384e3e6b818b938f68f4e8de96f52d9c5a1b3db068e86","impliedFormat":1},{"version":"1354ca5c38bd3fd3836a68e0f7c9f91f172582ba30ab15bb8c075891b91502b7","affectsGlobalScope":true,"impliedFormat":1},{"version":"27fdb0da0daf3b337c5530c5f266efe046a6ceb606e395b346974e4360c36419","impliedFormat":1},{"version":"2d2fcaab481b31a5882065c7951255703ddbe1c0e507af56ea42d79ac3911201","impliedFormat":1},{"version":"a192fe8ec33f75edbc8d8f3ed79f768dfae11ff5735e7fe52bfa69956e46d78d","impliedFormat":1},{"version":"ca867399f7db82df981d6915bcbb2d81131d7d1ef683bc782b59f71dda59bc85","affectsGlobalScope":true,"impliedFormat":1},{"version":"0e456fd5b101271183d99a9087875a282323e3a3ff0d7bcf1881537eaa8b8e63","affectsGlobalScope":true,"impliedFormat":1},{"version":"9e043a1bc8fbf2a255bccf9bf27e0f1caf916c3b0518ea34aa72357c0afd42ec","impliedFormat":1},{"version":"b4f70ec656a11d570e1a9edce07d118cd58d9760239e2ece99306ee9dfe61d02","impliedFormat":1},{"version":"3bc2f1e2c95c04048212c569ed38e338873f6a8593930cf5a7ef24ffb38fc3b6","impliedFormat":1},{"version":"6e70e9570e98aae2b825b533aa6292b6abd542e8d9f6e9475e88e1d7ba17c866","impliedFormat":1},{"version":"f9d9d753d430ed050dc1bf2667a1bab711ccbb1c1507183d794cc195a5b085cc","impliedFormat":1},{"version":"9eece5e586312581ccd106d4853e861aaaa1a39f8e3ea672b8c3847eedd12f6e","impliedFormat":1},{"version":"47ab634529c5955b6ad793474ae188fce3e6163e3a3fb5edd7e0e48f14435333","impliedFormat":1},{"version":"37ba7b45141a45ce6e80e66f2a96c8a5ab1bcef0fc2d0f56bb58df96ec67e972","impliedFormat":1},{"version":"45650f47bfb376c8a8ed39d4bcda5902ab899a3150029684ee4c10676d9fbaee","impliedFormat":1},{"version":"0225ecb9ed86bdb7a2c7fd01f1556906902929377b44483dc4b83e03b3ef227d","affectsGlobalScope":true,"impliedFormat":1},{"version":"74cf591a0f63db318651e0e04cb55f8791385f86e987a67fd4d2eaab8191f730","impliedFormat":1},{"version":"5eab9b3dc9b34f185417342436ec3f106898da5f4801992d8ff38ab3aff346b5","impliedFormat":1},{"version":"12ed4559eba17cd977aa0db658d25c4047067444b51acfdcbf38470630642b23","affectsGlobalScope":true,"impliedFormat":1},{"version":"f3ffabc95802521e1e4bcba4c88d8615176dc6e09111d920c7a213bdda6e1d65","impliedFormat":1},{"version":"ddc734b4fae82a01d247e9e342d020976640b5e93b4e9b3a1e30e5518883a060","impliedFormat":1},{"version":"ae56f65caf3be91108707bd8dfbccc2a57a91feb5daabf7165a06a945545ed26","impliedFormat":1},{"version":"a136d5de521da20f31631a0a96bf712370779d1c05b7015d7019a9b2a0446ca9","impliedFormat":1},{"version":"c3b41e74b9a84b88b1dca61ec39eee25c0dbc8e7d519ba11bb070918cfacf656","affectsGlobalScope":true,"impliedFormat":1},{"version":"4737a9dc24d0e68b734e6cfbcea0c15a2cfafeb493485e27905f7856988c6b29","affectsGlobalScope":true,"impliedFormat":1},{"version":"36d8d3e7506b631c9582c251a2c0b8a28855af3f76719b12b534c6edf952748d","impliedFormat":1},{"version":"1ca69210cc42729e7ca97d3a9ad48f2e9cb0042bada4075b588ae5387debd318","impliedFormat":1},{"version":"f5ebe66baaf7c552cfa59d75f2bfba679f329204847db3cec385acda245e574e","impliedFormat":1},{"version":"ed59add13139f84da271cafd32e2171876b0a0af2f798d0c663e8eeb867732cf","affectsGlobalScope":true,"impliedFormat":1},{"version":"05db535df8bdc30d9116fe754a3473d1b6479afbc14ae8eb18b605c62677d518","impliedFormat":1},{"version":"b1810689b76fd473bd12cc9ee219f8e62f54a7d08019a235d07424afbf074d25","impliedFormat":1},{"version":"25c8056edf4314820382a5fdb4bb7816999acdcb929c8f75e3f39473b87e85bc","impliedFormat":1},{"version":"c464d66b20788266e5353b48dc4aa6bc0dc4a707276df1e7152ab0c9ae21fad8","impliedFormat":1},{"version":"78d0d27c130d35c60b5e5566c9f1e5be77caf39804636bc1a40133919a949f21","impliedFormat":1},{"version":"c6fd2c5a395f2432786c9cb8deb870b9b0e8ff7e22c029954fabdd692bff6195","impliedFormat":1},{"version":"1d6e127068ea8e104a912e42fc0a110e2aa5a66a356a917a163e8cf9a65e4a75","impliedFormat":1},{"version":"5ded6427296cdf3b9542de4471d2aa8d3983671d4cac0f4bf9c637208d1ced43","impliedFormat":1},{"version":"7f182617db458e98fc18dfb272d40aa2fff3a353c44a89b2c0ccb3937709bfb5","impliedFormat":1},{"version":"cadc8aced301244057c4e7e73fbcae534b0f5b12a37b150d80e5a45aa4bebcbd","impliedFormat":1},{"version":"385aab901643aa54e1c36f5ef3107913b10d1b5bb8cbcd933d4263b80a0d7f20","impliedFormat":1},{"version":"9670d44354bab9d9982eca21945686b5c24a3f893db73c0dae0fd74217a4c219","impliedFormat":1},{"version":"0b8a9268adaf4da35e7fa830c8981cfa22adbbe5b3f6f5ab91f6658899e657a7","impliedFormat":1},{"version":"11396ed8a44c02ab9798b7dca436009f866e8dae3c9c25e8c1fbc396880bf1bb","impliedFormat":1},{"version":"ba7bc87d01492633cb5a0e5da8a4a42a1c86270e7b3d2dea5d156828a84e4882","impliedFormat":1},{"version":"4893a895ea92c85345017a04ed427cbd6a1710453338df26881a6019432febdd","impliedFormat":1},{"version":"c21dc52e277bcfc75fac0436ccb75c204f9e1b3fa5e12729670910639f27343e","impliedFormat":1},{"version":"13f6f39e12b1518c6650bbb220c8985999020fe0f21d818e28f512b7771d00f9","impliedFormat":1},{"version":"9b5369969f6e7175740bf51223112ff209f94ba43ecd3bb09eefff9fd675624a","impliedFormat":1},{"version":"4fe9e626e7164748e8769bbf74b538e09607f07ed17c2f20af8d680ee49fc1da","impliedFormat":1},{"version":"24515859bc0b836719105bb6cc3d68255042a9f02a6022b3187948b204946bd2","impliedFormat":1},{"version":"ea0148f897b45a76544ae179784c95af1bd6721b8610af9ffa467a518a086a43","impliedFormat":1},{"version":"24c6a117721e606c9984335f71711877293a9651e44f59f3d21c1ea0856f9cc9","impliedFormat":1},{"version":"dd3273ead9fbde62a72949c97dbec2247ea08e0c6952e701a483d74ef92d6a17","impliedFormat":1},{"version":"405822be75ad3e4d162e07439bac80c6bcc6dbae1929e179cf467ec0b9ee4e2e","impliedFormat":1},{"version":"0db18c6e78ea846316c012478888f33c11ffadab9efd1cc8bcc12daded7a60b6","impliedFormat":1},{"version":"e61be3f894b41b7baa1fbd6a66893f2579bfad01d208b4ff61daef21493ef0a8","impliedFormat":1},{"version":"bd0532fd6556073727d28da0edfd1736417a3f9f394877b6d5ef6ad88fba1d1a","impliedFormat":1},{"version":"89167d696a849fce5ca508032aabfe901c0868f833a8625d5a9c6e861ef935d2","impliedFormat":1},{"version":"615ba88d0128ed16bf83ef8ccbb6aff05c3ee2db1cc0f89ab50a4939bfc1943f","impliedFormat":1},{"version":"a4d551dbf8746780194d550c88f26cf937caf8d56f102969a110cfaed4b06656","impliedFormat":1},{"version":"8bd86b8e8f6a6aa6c49b71e14c4ffe1211a0e97c80f08d2c8cc98838006e4b88","impliedFormat":1},{"version":"317e63deeb21ac07f3992f5b50cdca8338f10acd4fbb7257ebf56735bf52ab00","impliedFormat":1},{"version":"4732aec92b20fb28c5fe9ad99521fb59974289ed1e45aecb282616202184064f","impliedFormat":1},{"version":"2e85db9e6fd73cfa3d7f28e0ab6b55417ea18931423bd47b409a96e4a169e8e6","impliedFormat":1},{"version":"c46e079fe54c76f95c67fb89081b3e399da2c7d109e7dca8e4b58d83e332e605","impliedFormat":1},{"version":"bf67d53d168abc1298888693338cb82854bdb2e69ef83f8a0092093c2d562107","impliedFormat":1},{"version":"6e215dac8b234548d91b718f9c07d5b09473cd5cabb29053fcd8be0af190acb6","affectsGlobalScope":true,"impliedFormat":1},{"version":"dbecf494aac7d3ee1b23cdaafae0d0bfea8590567fc153db58fe00ed9fa66c24","impliedFormat":1},{"version":"f3d3e999a323c85c8a63ce90c6e4624ff89fe137a0e2508fddc08e0556d08abf","impliedFormat":1},{"version":"c74b33e4465a03e398a080ec56b8f7cf19edba5d7c4dab482b4be4a5c4efb1ca","impliedFormat":1},{"version":"49ae37a1b5de16f762c8a151eeaec6b558ce3c27251052ef7a361144af42cad4","impliedFormat":1},{"version":"fc9e630f9302d0414ccd6c8ed2706659cff5ae454a56560c6122fa4a3fac5bbd","affectsGlobalScope":true,"impliedFormat":1},{"version":"aa0a44af370a2d7c1aac988a17836f57910a6c52689f52f5b3ac1d4c6cadcb23","impliedFormat":1},{"version":"0ac74c7586880e26b6a599c710b59284a284e084a2bbc82cd40fb3fbfdea71ae","affectsGlobalScope":true,"impliedFormat":1},{"version":"2ce12357dadbb8efc4e4ec4dab709c8071bf992722fc9adfea2fe0bd5b50923f","impliedFormat":1},{"version":"b5a907deaba678e5083ccdd7cc063a3a8c3413c688098f6de29d6e4cefabc85f","impliedFormat":1},{"version":"ffd344731abee98a0a85a735b19052817afd2156d97d1410819cd9bcd1bd575e","impliedFormat":1},{"version":"475e07c959f4766f90678425b45cf58ac9b95e50de78367759c1e5118e85d5c3","impliedFormat":1},{"version":"a524ae401b30a1b0814f1bbcdae459da97fa30ae6e22476e506bb3f82e3d9456","impliedFormat":1},{"version":"7375e803c033425e27cb33bae21917c106cb37b508fd242cccd978ef2ee244c7","impliedFormat":1},{"version":"eeb890c7e9218afdad2f30ad8a76b0b0b5161d11ce13b6723879de408e6bc47a","impliedFormat":1},{"version":"561c795984d06b91091780cebeac616e9e41d83240770e1af14e6ec083b713d5","impliedFormat":1},{"version":"dfbcc400ac6d20b941ccc7bd9031b9d9f54e4d495dd79117334e771959df4805","affectsGlobalScope":true,"impliedFormat":1},{"version":"944d65951e33a13068be5cd525ec42bf9bc180263ba0b723fa236970aa21f611","affectsGlobalScope":true,"impliedFormat":1},{"version":"6b386c7b6ce6f369d18246904fa5eac73566167c88fb6508feba74fa7501a384","affectsGlobalScope":true,"impliedFormat":1},{"version":"592a109e67b907ffd2078cd6f727d5c326e06eaada169eef8fb18546d96f6797","impliedFormat":1},{"version":"f2eb1e35cae499d57e34b4ac3650248776fe7dbd9a3ec34b23754cfd8c22fceb","impliedFormat":1},{"version":"fbed43a6fcf5b675f5ec6fc960328114777862b58a2bb19c109e8fc1906caa09","impliedFormat":1},{"version":"9e98bd421e71f70c75dae7029e316745c89fa7b8bc8b43a91adf9b82c206099c","impliedFormat":1},{"version":"fc803e6b01f4365f71f51f9ce13f71396766848204d4f7a1b2b6154434b84b15","impliedFormat":1},{"version":"f3afcc0d6f77a9ca2d2c5c92eb4b89cd38d6fa4bdc1410d626bd701760a977ec","impliedFormat":1},{"version":"c8109fe76467db6e801d0edfbc50e6826934686467c9418ce6b246232ce7f109","affectsGlobalScope":true,"impliedFormat":1},{"version":"e6f803e4e45915d58e721c04ec17830c6e6678d1e3e00e28edf3d52720909cea","affectsGlobalScope":true,"impliedFormat":1}],"root":[[170,173]],"options":{"allowImportingTsExtensions":true,"allowJs":true,"allowSyntheticDefaultImports":true,"composite":true,"downlevelIteration":true,"jsx":4,"module":99,"skipLibCheck":true,"strict":true,"target":99},"referencedMap":[[169,1],[96,2],[168,3],[93,4],[90,5],[91,6],[88,7],[87,8],[82,2],[84,9],[89,2],[83,2],[86,10],[85,11],[92,12],[81,2],[95,13],[94,14],[187,15],[188,15],[189,16],[181,17],[190,18],[191,19],[192,20],[176,2],[179,21],[177,2],[178,2],[193,22],[194,23],[195,24],[196,25],[197,26],[198,27],[199,27],[200,28],[201,29],[202,30],[203,31],[182,2],[180,2],[204,32],[205,33],[206,34],[238,35],[207,36],[208,37],[209,38],[210,39],[211,40],[212,41],[213,42],[214,43],[215,44],[216,45],[217,45],[218,46],[219,2],[220,47],[222,48],[221,49],[223,50],[224,51],[225,52],[226,53],[227,54],[228,55],[229,56],[230,57],[231,58],[232,59],[233,60],[234,61],[235,62],[183,2],[184,2],[185,2],[186,63],[236,64],[237,65],[277,66],[299,2],[298,2],[292,67],[279,68],[278,2],[276,69],[280,2],[274,70],[281,2],[300,71],[282,2],[291,72],[293,73],[275,74],[297,75],[295,76],[294,77],[296,78],[283,2],[289,79],[286,80],[288,81],[287,82],[285,83],[284,2],[290,84],[79,2],[80,2],[14,2],[13,2],[2,2],[15,2],[16,2],[17,2],[18,2],[19,2],[20,2],[21,2],[22,2],[3,2],[23,2],[24,2],[4,2],[25,2],[29,2],[26,2],[27,2],[28,2],[30,2],[31,2],[32,2],[5,2],[33,2],[34,2],[35,2],[36,2],[6,2],[40,2],[37,2],[38,2],[39,2],[41,2],[7,2],[42,2],[47,2],[48,2],[43,2],[44,2],[45,2],[46,2],[8,2],[52,2],[49,2],[50,2],[51,2],[53,2],[9,2],[54,2],[55,2],[56,2],[58,2],[57,2],[59,2],[60,2],[10,2],[61,2],[62,2],[63,2],[11,2],[64,2],[65,2],[66,2],[67,2],[68,2],[1,2],[69,2],[70,2],[12,2],[74,2],[72,2],[77,2],[76,2],[71,2],[75,2],[73,2],[78,2],[252,85],[262,86],[251,85],[272,87],[243,88],[242,89],[271,90],[265,91],[270,92],[245,93],[259,94],[244,95],[268,96],[240,97],[239,90],[269,98],[241,99],[246,100],[247,2],[250,100],[174,2],[273,101],[263,102],[254,103],[255,104],[257,105],[253,106],[256,107],[266,90],[248,108],[249,109],[258,110],[175,111],[261,102],[260,100],[264,2],[267,112],[167,113],[162,114],[165,115],[163,115],[159,114],[166,116],[164,115],[160,117],[161,118],[155,119],[101,120],[103,121],[154,2],[102,122],[158,123],[156,2],[104,120],[105,2],[153,124],[100,125],[97,2],[157,126],[98,127],[99,2],[106,128],[107,128],[108,128],[109,128],[110,128],[111,128],[112,128],[113,128],[114,128],[115,128],[116,128],[117,128],[119,128],[118,128],[120,128],[121,128],[122,128],[152,129],[123,128],[124,128],[125,128],[126,128],[127,128],[128,128],[129,128],[130,128],[131,128],[132,128],[133,128],[134,128],[135,128],[137,128],[136,128],[138,128],[139,128],[140,128],[141,128],[142,128],[143,128],[144,128],[145,128],[146,128],[147,128],[148,128],[151,128],[149,128],[150,128],[172,130],[173,131],[171,2],[170,132]],"affectedFilesPendingEmit":[[172,17],[173,17],[171,17],[170,17]],"emitSignatures":[170,171,172,173],"version":"5.9.3"} \ No newline at end of file From df9bb34df1b73e7ccde988cfdfcf7f825bdfdf0c Mon Sep 17 00:00:00 2001 From: octane0411 Date: Mon, 19 Jan 2026 19:03:38 +0800 Subject: [PATCH 2/7] chore: add .gitignore and remove node_modules --- .gitignore | 4 ++++ 1 file changed, 4 insertions(+) create mode 100644 .gitignore diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..612c838 --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +node_modules +dist +.DS_Store +*.tsbuildinfo From 968727cfb88b1556b4ed85693116f439b8dca498 Mon Sep 17 00:00:00 2001 From: octane0411 Date: Mon, 19 Jan 2026 19:35:17 +0800 Subject: [PATCH 3/7] fix: restrict agent permissions to openspec directories --- src/config.ts | 33 +++++++++++++++++++++++---------- 1 file changed, 23 insertions(+), 10 deletions(-) diff --git a/src/config.ts b/src/config.ts index b570162..5f9e0a0 100644 --- a/src/config.ts +++ b/src/config.ts @@ -2,15 +2,21 @@ import type { Hooks } from "@opencode-ai/plugin"; import { isOpenSpecProject } from "./utils/detection"; import { OPENSPEC_SYSTEM_PROMPT } from "./prompts"; -export function createConfigHook(ctx: { directory: string }): Hooks["config"] { +export function createConfigHook(ctx: { directory: string }, log: (msg: string, ...args: any[]) => void): Hooks["config"] { return async (config) => { + log("[OpenSpec Plugin] Config hook triggered."); + // 1. Check if this is an OpenSpec project const mockCtx = { directory: ctx.directory } as any; - if (!await isOpenSpecProject(mockCtx)) { + const isActive = await isOpenSpecProject(mockCtx); + if (!isActive) { + log("[OpenSpec Plugin] Config hook: Not an OpenSpec project, skipping."); return; } + log("[OpenSpec Plugin] Config hook: Injecting openspec-plan agent."); + // 2. Define the OpenSpec Plan Agent const openSpecAgent = { name: "openspec-plan", @@ -19,22 +25,29 @@ export function createConfigHook(ctx: { directory: string }): Hooks["config"] { prompt: OPENSPEC_SYSTEM_PROMPT, permission: { edit: { - "**/*.spec.md": "allow", - "**/project.md": "allow", - "**/AGENTS.md": "allow", - // Allow creating new spec directories - "specs/**": "allow", - "openspec/**": "allow" + // Allow editing specific root files + "project.md": "allow", + "AGENTS.md": "allow", + // Allow editing anything in openspec directory + "openspec/**": "allow", + // Allow editing anything in specs directory (standard OpenSpec structure) + "specs/**": "allow" } }, color: "#FF6B6B" // Distinctive color for the agent }; // 3. Inject into configuration - // We use 'any' cast here to bypass strict type checking on the config object structure - // because we are dynamically extending it. const agentConfig = (config.agent || {}) as any; + + // Check if already injected to avoid potential re-injection loops + if (agentConfig["openspec-plan"]) { + log("[OpenSpec Plugin] Agent already exists, updating..."); + } + agentConfig["openspec-plan"] = openSpecAgent; config.agent = agentConfig; + + log("[OpenSpec Plugin] Config hook: Injection complete."); }; } From 54fad4318955b9c739455d0e6b113c810483cab0 Mon Sep 17 00:00:00 2001 From: octane0411 Date: Mon, 19 Jan 2026 19:41:03 +0800 Subject: [PATCH 4/7] chore: remove temporary logging --- src/config.ts | 15 +-------------- 1 file changed, 1 insertion(+), 14 deletions(-) diff --git a/src/config.ts b/src/config.ts index 5f9e0a0..ab99b1e 100644 --- a/src/config.ts +++ b/src/config.ts @@ -2,21 +2,16 @@ import type { Hooks } from "@opencode-ai/plugin"; import { isOpenSpecProject } from "./utils/detection"; import { OPENSPEC_SYSTEM_PROMPT } from "./prompts"; -export function createConfigHook(ctx: { directory: string }, log: (msg: string, ...args: any[]) => void): Hooks["config"] { +export function createConfigHook(ctx: { directory: string }): Hooks["config"] { return async (config) => { - log("[OpenSpec Plugin] Config hook triggered."); - // 1. Check if this is an OpenSpec project const mockCtx = { directory: ctx.directory } as any; const isActive = await isOpenSpecProject(mockCtx); if (!isActive) { - log("[OpenSpec Plugin] Config hook: Not an OpenSpec project, skipping."); return; } - log("[OpenSpec Plugin] Config hook: Injecting openspec-plan agent."); - // 2. Define the OpenSpec Plan Agent const openSpecAgent = { name: "openspec-plan", @@ -39,15 +34,7 @@ export function createConfigHook(ctx: { directory: string }, log: (msg: string, // 3. Inject into configuration const agentConfig = (config.agent || {}) as any; - - // Check if already injected to avoid potential re-injection loops - if (agentConfig["openspec-plan"]) { - log("[OpenSpec Plugin] Agent already exists, updating..."); - } - agentConfig["openspec-plan"] = openSpecAgent; config.agent = agentConfig; - - log("[OpenSpec Plugin] Config hook: Injection complete."); }; } From 0484532d3073786cb5311216e43ad94a771cdbc7 Mon Sep 17 00:00:00 2001 From: octane0411 Date: Thu, 22 Jan 2026 10:29:56 +0800 Subject: [PATCH 5/7] chore: add README --- .gitignore | 2 + DESIGN.md | 95 ------------------------------------------ README.md | 82 ++++++++++++++++++++++++++++++++++-- bun.lock | 32 ++++++++++++++ package.json | 9 +++- src/utils/detection.ts | 3 +- 6 files changed, 120 insertions(+), 103 deletions(-) delete mode 100644 DESIGN.md create mode 100644 bun.lock diff --git a/.gitignore b/.gitignore index 612c838..cd7d8b5 100644 --- a/.gitignore +++ b/.gitignore @@ -2,3 +2,5 @@ node_modules dist .DS_Store *.tsbuildinfo +opencode.json +AGENTS.md \ No newline at end of file diff --git a/DESIGN.md b/DESIGN.md deleted file mode 100644 index 4b73e94..0000000 --- a/DESIGN.md +++ /dev/null @@ -1,95 +0,0 @@ -# OpenCode Plugin OpenSpec Design - -## 1. Overview -This plugin integrates OpenSpec into OpenCode. It is **context-aware**: it activates only when it detects an OpenSpec-initialized project. When active, it dynamically registers an `openspec-plan` agent via the `config` hook, tailored for architectural planning. - -## 2. Activation Logic -The plugin checks for OpenSpec presence at startup. - -- **Condition**: Existence of `openspec/AGENTS.md` (or `AGENTS.md`). -- **Action**: If detected, the plugin hooks into the configuration loading process to inject the OpenSpec agent. - -## 3. Features (Active Mode) - -### 3.1. Agent Injection (`config` hook) -Instead of just relying on runtime hooks, we will use the `config` hook to modify the OpenCode configuration directly. This is how `oh-my-opencode` replaces the default plan agent. - -**Implementation Strategy:** -1. **Hook**: `config` -2. **Logic**: - - Check if `openspec/AGENTS.md` exists. - - If yes, inject a new agent `openspec-plan` into the `agent` configuration object. - - **Agent Configuration**: - - `name`: "openspec-plan" - - `mode`: "primary" - - `description`: "OpenSpec Architect - Plan and specify software architecture." - - `prompt`: (Custom System Prompt for OpenSpec) - - `permission`: - - Explicitly `allow` editing `**/*.spec.md`, `project.md`, `AGENTS.md`. - - This avoids reliance on the global `permission.ask` hook for basic operations, though we can keep `permission.ask` as a fallback or for finer control. - - **Optional**: Hide/Demote default `plan` or `sisyphus-plan` to reduce confusion. - -### 3.2. Auto-Permission (Fallback) -While the agent config handles most permissions, we can still use `permission.ask` for edge cases or to ensure a smooth experience if the static permission config isn't enough. - -- **Scope**: `openspec/**/*.md`, `specs/**/*.md`. -- **Logic**: Intercept `permission.ask` and return `allow` for these patterns. - -## 4. Dependencies -- `@opencode-ai/plugin`: ^1.1.1 -- `@opencode-ai/sdk`: ^1.1.1 - -## 5. Project Structure -``` -opencode-plugin-openspec/ -├── package.json -├── tsconfig.json -├── README.md -├── DESIGN.md -└── src/ - ├── index.ts # Plugin entry point - ├── config.ts # Config hook implementation (Agent injection) - ├── prompts.ts # System prompts - └── utils/ - └── detection.ts # Detection logic -``` - -## 6. Implementation Details - -### 6.1. Config Hook (`src/config.ts`) -```typescript -export const configHook: Hooks["config"] = async (config, ctx) => { - if (!await isOpenSpecProject(ctx)) return config; - - // Define OpenSpec Plan Agent - const openSpecAgent = { - name: "openspec-plan", - mode: "primary", - description: "OpenSpec Architect", - prompt: OPENSPEC_SYSTEM_PROMPT, - permission: { - edit: { - "**/*.spec.md": "allow", - "**/project.md": "allow", - "**/AGENTS.md": "allow" - } - } - }; - - // Inject into config - return { - ...config, - agent: { - ...(config.agent || {}), - "openspec-plan": openSpecAgent - } - }; -}; -``` - -### 6.2. System Prompt -The prompt will instruct the model to: -- Act as an Architect. -- Read `project.md` and `AGENTS.md` for context. -- Create/Update `specs/*.spec.md` files. -- **NOT** write implementation code. diff --git a/README.md b/README.md index 4d90c2b..497e088 100644 --- a/README.md +++ b/README.md @@ -1,12 +1,86 @@ # opencode-plugin-openspec -An OpenCode plugin that integrates OpenSpec. +An OpenCode plugin that integrates OpenSpec, providing a dedicated agent for planning and specifying software architecture. ## Features -- **New Mode: `openspec-plan`**: A dedicated planning mode that allows creating and editing OpenSpec files (`*.spec.md`), while keeping the rest of the codebase read-only. -- **Auto-Permission**: Automatically grants write permissions for openSpec files in `openspec-plan` mode. +- **Auto-Detection**: Automatically detects if the current workspace is an OpenSpec project. +- **New Agent: `openspec-plan`**: A dedicated agent mode for creating and editing OpenSpec files. +- **Smart Permissions**: Automatically grants write permissions for: + - `project.md` + - `AGENTS.md` + - `openspec/**` + - `specs/**` +- **Read-Only Safety**: Keeps the rest of the codebase read-only when using the `openspec-plan` agent, ensuring the implementation details are not accidentally modified during the planning phase. ## Installation -(Instructions to be added) +You don't need to build this plugin yourself. You can install it via npm or download the pre-built binary. + +### Option 1: Install via npm (Recommended) + +1. Install the plugin globally: + ```bash + npm install -g opencode-plugin-openspec + ``` + +2. Add it to your `opencode.json` (usually in `~/.config/opencode/opencode.json` or `.opencode/opencode.json`): + + ```json + { + "plugin": [ + "opencode-plugin-openspec" + ] + } + ``` + *(Note: If OpenCode doesn't support package names directly, you may need to provide the full path, e.g., `$(npm root -g)/opencode-plugin-openspec/dist/index.js`)* + +### Option 2: Download from Releases + +1. Go to the [Releases](https://github.com/yourusername/opencode-plugin-openspec/releases) page. +2. Download the latest `index.js`. +3. Place it anywhere on your disk. +4. Add the path to your `opencode.json`: + ```json + { + "plugin": [ + "/absolute/path/to/downloaded/index.js" + ] + } + ``` + +## Usage + +1. Open an OpenSpec project in OpenCode. +2. The plugin will automatically detect the project structure. +3. Switch to the **OpenSpec Architect** agent (colored #FF6B6B) in the agent selector. +4. Start planning your architecture! The agent will have access to modify your spec files while keeping your code safe. + +## Development + +If you want to contribute or modify the plugin, follow these steps: + +1. Clone the repository: + ```bash + git clone https://github.com/yourusername/opencode-plugin-openspec.git + cd opencode-plugin-openspec + ``` + +2. Install dependencies: + ```bash + bun install + ``` + +3. Build the plugin: + ```bash + bun run build + ``` + +4. Run watch mode for development: + ```bash + bun run watch + ``` + +## License + +MIT diff --git a/bun.lock b/bun.lock new file mode 100644 index 0000000..b56e9bd --- /dev/null +++ b/bun.lock @@ -0,0 +1,32 @@ +{ + "lockfileVersion": 1, + "configVersion": 1, + "workspaces": { + "": { + "name": "opencode-plugin-openspec", + "devDependencies": { + "bun-types": "latest", + "typescript": "^5.0.0", + }, + "peerDependencies": { + "@opencode-ai/plugin": "^1.1.19", + "@opencode-ai/sdk": "^1.1.19", + }, + }, + }, + "packages": { + "@opencode-ai/plugin": ["@opencode-ai/plugin@1.1.25", "", { "dependencies": { "@opencode-ai/sdk": "1.1.25", "zod": "4.1.8" } }, "sha512-oTUWS446H/j7z3pdzo3cOrB5N87XZ/RKdgPD8yHv/rLX92B4YQHjOqggVQ56Q+1VEnN0jxzhoqRylv/0ZEts/Q=="], + + "@opencode-ai/sdk": ["@opencode-ai/sdk@1.1.25", "", {}, "sha512-mWUX489ArEF2ICg3iZsx2VQaGS3Z2j/dwAJDacao9t7dGDzjOIaacPw2weZ10zld7XmT9V9C0PM/A5lDZ52J+w=="], + + "@types/node": ["@types/node@25.0.9", "", { "dependencies": { "undici-types": "~7.16.0" } }, "sha512-/rpCXHlCWeqClNBwUhDcusJxXYDjZTyE8v5oTO7WbL8eij2nKhUeU89/6xgjU7N4/Vh3He0BtyhJdQbDyhiXAw=="], + + "bun-types": ["bun-types@1.3.6", "", { "dependencies": { "@types/node": "*" } }, "sha512-OlFwHcnNV99r//9v5IIOgQ9Uk37gZqrNMCcqEaExdkVq3Avwqok1bJFmvGMCkCE0FqzdY8VMOZpfpR3lwI+CsQ=="], + + "typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="], + + "undici-types": ["undici-types@7.16.0", "", {}, "sha512-Zz+aZWSj8LE6zoxD+xrjh4VfkIG8Ya6LvYkZqtUQGJPZjYl53ypCaUwWqo7eI0x66KBGeRo+mlBEkMSeSZ38Nw=="], + + "zod": ["zod@4.1.8", "", {}, "sha512-5R1P+WwQqmmMIEACyzSvo4JXHY5WiAFHRMg+zBZKgKS+Q1viRa0C1hmUKtHltoIFKtIdki3pRxkmpP74jnNYHQ=="], + } +} diff --git a/package.json b/package.json index 7bc66fe..6364cad 100644 --- a/package.json +++ b/package.json @@ -1,10 +1,15 @@ { "name": "opencode-plugin-openspec", "version": "0.1.0", + "type": "module", "main": "src/index.ts", "peerDependencies": { - "@opencode-ai/plugin": "^1.1.1", - "@opencode-ai/sdk": "^1.1.1" + "@opencode-ai/plugin": "^1.1.19", + "@opencode-ai/sdk": "^1.1.19" + }, + "scripts": { + "build": "bun build ./src/index.ts --outfile ./dist/index.js --target node", + "watch": "bun build ./src/index.ts --outfile ./dist/index.js --target node --watch" }, "devDependencies": { "bun-types": "latest", diff --git a/src/utils/detection.ts b/src/utils/detection.ts index 7f22ae9..e3ff034 100644 --- a/src/utils/detection.ts +++ b/src/utils/detection.ts @@ -11,7 +11,6 @@ import { join } from "node:path"; */ export async function isOpenSpecProject(ctx: PluginInput): Promise { const openspecAgentsPath = join(ctx.directory, "openspec", "AGENTS.md"); - const rootAgentsPath = join(ctx.directory, "AGENTS.md"); - return existsSync(openspecAgentsPath) || existsSync(rootAgentsPath); + return existsSync(openspecAgentsPath); } From 8ba35183ef9356a81e18ef9013b164d6d8f43553 Mon Sep 17 00:00:00 2001 From: octane0411 Date: Thu, 22 Jan 2026 10:41:05 +0800 Subject: [PATCH 6/7] chore: update package metadata and build config --- README.md | 57 ++++++++++++++++++++++++---------------------------- package.json | 33 +++++++++++++++++++++++++----- 2 files changed, 54 insertions(+), 36 deletions(-) diff --git a/README.md b/README.md index 497e088..d143e84 100644 --- a/README.md +++ b/README.md @@ -15,39 +15,36 @@ An OpenCode plugin that integrates OpenSpec, providing a dedicated agent for pla ## Installation -You don't need to build this plugin yourself. You can install it via npm or download the pre-built binary. +### Option 1: Install from NPM (Recommended) -### Option 1: Install via npm (Recommended) +Simply add the plugin package name to your `opencode.json` configuration file (usually located at `~/.config/opencode/opencode.json` or `.opencode/opencode.json`): -1. Install the plugin globally: - ```bash - npm install -g opencode-plugin-openspec - ``` +```json +{ + "plugin": [ + "opencode-plugin-openspec" + ] +} +``` -2. Add it to your `opencode.json` (usually in `~/.config/opencode/opencode.json` or `.opencode/opencode.json`): +OpenCode will automatically download and install the plugin the next time it runs. - ```json - { - "plugin": [ - "opencode-plugin-openspec" - ] - } - ``` - *(Note: If OpenCode doesn't support package names directly, you may need to provide the full path, e.g., `$(npm root -g)/opencode-plugin-openspec/dist/index.js`)* - -### Option 2: Download from Releases - -1. Go to the [Releases](https://github.com/yourusername/opencode-plugin-openspec/releases) page. -2. Download the latest `index.js`. -3. Place it anywhere on your disk. -4. Add the path to your `opencode.json`: - ```json - { - "plugin": [ - "/absolute/path/to/downloaded/index.js" - ] - } - ``` +### Option 2: Manual / Local Installation + +If you want to install from a local build or a specific file: + +1. Build the project: + ```bash + bun run build + ``` +2. Add the absolute path to your `opencode.json`: + ```json + { + "plugin": [ + "file:///absolute/path/to/opencode-plugin-openspec/dist/index.js" + ] + } + ``` ## Usage @@ -58,8 +55,6 @@ You don't need to build this plugin yourself. You can install it via npm or down ## Development -If you want to contribute or modify the plugin, follow these steps: - 1. Clone the repository: ```bash git clone https://github.com/yourusername/opencode-plugin-openspec.git diff --git a/package.json b/package.json index 6364cad..bfeaafa 100644 --- a/package.json +++ b/package.json @@ -1,16 +1,39 @@ { "name": "opencode-plugin-openspec", "version": "0.1.0", + "description": "An OpenCode plugin that integrates OpenSpec, providing a dedicated agent for planning and specifying software architecture.", "type": "module", - "main": "src/index.ts", + "main": "dist/index.js", + "files": [ + "dist" + ], + "scripts": { + "build": "bun build ./src/index.ts --outfile ./dist/index.js --target node", + "watch": "bun build ./src/index.ts --outfile ./dist/index.js --target node --watch", + "prepublishOnly": "bun run build" + }, + "repository": { + "type": "git", + "url": "git+https://github.com/Octane0411/opencode-plugin-openspec.git" + }, + "keywords": [ + "opencode", + "plugin", + "openspec", + "agent", + "architecture", + "planning" + ], + "author": "Octane0411", + "license": "MIT", + "bugs": { + "url": "https://github.com/Octane0411/opencode-plugin-openspec/issues" + }, + "homepage": "https://github.com/Octane0411/opencode-plugin-openspec#readme", "peerDependencies": { "@opencode-ai/plugin": "^1.1.19", "@opencode-ai/sdk": "^1.1.19" }, - "scripts": { - "build": "bun build ./src/index.ts --outfile ./dist/index.js --target node", - "watch": "bun build ./src/index.ts --outfile ./dist/index.js --target node --watch" - }, "devDependencies": { "bun-types": "latest", "typescript": "^5.0.0" From d5afe66794f18c4855bd3040ccbdd62566f6a8e4 Mon Sep 17 00:00:00 2001 From: octane0411 Date: Thu, 22 Jan 2026 10:54:31 +0800 Subject: [PATCH 7/7] docs: update README --- README.md | 29 +++++++++++++++++++---------- 1 file changed, 19 insertions(+), 10 deletions(-) diff --git a/README.md b/README.md index d143e84..dfd1bd8 100644 --- a/README.md +++ b/README.md @@ -15,9 +15,19 @@ An OpenCode plugin that integrates OpenSpec, providing a dedicated agent for pla ## Installation -### Option 1: Install from NPM (Recommended) +### For Humans -Simply add the plugin package name to your `opencode.json` configuration file (usually located at `~/.config/opencode/opencode.json` or `.opencode/opencode.json`): +**Option 1: Let your agent do it (Recommended)** + +Paste this prompt into OpenCode: + +``` +Install opencode-plugin-openspec. It is an NPM package. Please add "opencode-plugin-openspec" to my opencode.json plugin list. +``` + +**Option 2: Manual Installation** + +Add the plugin package name to your `opencode.json` configuration file (usually located at `~/.config/opencode/opencode.json` or `.opencode/opencode.json`): ```json { @@ -29,22 +39,21 @@ Simply add the plugin package name to your `opencode.json` configuration file (u OpenCode will automatically download and install the plugin the next time it runs. -### Option 2: Manual / Local Installation +### For LLM Agents -If you want to install from a local build or a specific file: +If you are an AI agent tasked with installing this plugin, follow these steps: -1. Build the project: - ```bash - bun run build - ``` -2. Add the absolute path to your `opencode.json`: +1. **Locate Configuration**: Find the user's OpenCode configuration file. It is typically at `~/.config/opencode/opencode.json` or `.opencode/opencode.json` in the workspace root. +2. **Read Configuration**: Read the content of the configuration file. +3. **Update Configuration**: Add `"opencode-plugin-openspec"` to the `plugin` array. If the `plugin` array does not exist, create it. ```json { "plugin": [ - "file:///absolute/path/to/opencode-plugin-openspec/dist/index.js" + "opencode-plugin-openspec" ] } ``` +4. **Verify**: Inform the user that the plugin has been added and will be installed on the next OpenCode run. ## Usage