Building Extensions

This guide walks through creating an Pllan extension from scratch. Extensions can add channels, model providers, tools, or other capabilities.

Prerequisites

  • Pllan repository cloned and dependencies installed (pnpm install)
  • Familiarity with TypeScript (ESM)

Extension structure

Every extension lives under extensions/<name>/ and follows this layout:
extensions/my-channel/
├── package.json          # npm metadata + pllan config
├── index.ts              # Entry point (defineChannelPluginEntry)
├── setup-entry.ts        # Setup wizard (optional)
├── api.ts                # Public contract barrel (optional)
├── runtime-api.ts        # Internal runtime barrel (optional)
└── src/
    ├── channel.ts        # Channel adapter implementation
    ├── runtime.ts        # Runtime wiring
    └── *.test.ts         # Colocated tests

Step 1: Create the package

Create extensions/my-channel/package.json:
{
  "name": "@pllan/my-channel",
  "version": "2026.1.1",
  "description": "Pllan My Channel plugin",
  "type": "module",
  "dependencies": {},
  "pllan": {
    "extensions": ["./index.ts"],
    "setupEntry": "./setup-entry.ts",
    "channel": {
      "id": "my-channel",
      "label": "My Channel",
      "selectionLabel": "My Channel (plugin)",
      "docsPath": "/channels/my-channel",
      "docsLabel": "my-channel",
      "blurb": "Short description of the channel.",
      "order": 80
    },
    "install": {
      "npmSpec": "@pllan/my-channel",
      "localPath": "extensions/my-channel"
    }
  }
}
The pllan field tells the plugin system what your extension provides. For provider plugins, use providers instead of channel.

Step 2: Define the entry point

Create extensions/my-channel/index.ts:
import { defineChannelPluginEntry } from "pllan/plugin-sdk/core";

export default defineChannelPluginEntry({
  id: "my-channel",
  name: "My Channel",
  description: "Connects Pllan to My Channel",
  plugin: {
    // Channel adapter implementation
  },
});
For provider plugins, use definePluginEntry instead.

Step 3: Import from focused subpaths

The plugin SDK exposes many focused subpaths. Always import from specific subpaths rather than the monolithic root:
// Correct: focused subpaths
import { defineChannelPluginEntry } from "pllan/plugin-sdk/core";
import { createChannelReplyPipeline } from "pllan/plugin-sdk/channel-reply-pipeline";
import { createChannelPairingController } from "pllan/plugin-sdk/channel-pairing";
import { createPluginRuntimeStore } from "pllan/plugin-sdk/runtime-store";
import { createOptionalChannelSetupSurface } from "pllan/plugin-sdk/channel-setup";
import { resolveChannelGroupRequireMention } from "pllan/plugin-sdk/channel-policy";

// Wrong: monolithic root (lint will reject this)
import { ... } from "pllan/plugin-sdk";
Common subpaths:
SubpathPurpose
plugin-sdk/corePlugin entry definitions, base types
plugin-sdk/channel-setupOptional setup adapters/wizards
plugin-sdk/channel-pairingDM pairing primitives
plugin-sdk/channel-reply-pipelinePrefix + typing reply wiring
plugin-sdk/channel-config-schemaConfig schema builders
plugin-sdk/channel-policyGroup/DM policy helpers
plugin-sdk/secret-inputSecret input parsing/helpers
plugin-sdk/webhook-ingressWebhook request/target helpers
plugin-sdk/runtime-storePersistent plugin storage
plugin-sdk/allow-fromAllowlist resolution
plugin-sdk/reply-payloadMessage reply types
plugin-sdk/provider-onboardProvider onboarding config patches
plugin-sdk/testingTest utilities
Use the narrowest primitive that matches the job. Reach for channel-runtime or other larger helper barrels only when a dedicated subpath does not exist yet.

Step 4: Use local barrels for internal imports

Within your extension, create barrel files for internal code sharing instead of importing through the plugin SDK:
// api.ts — public contract for this extension
export { MyChannelConfig } from "./src/config.js";
export { MyChannelRuntime } from "./src/runtime.js";

// runtime-api.ts — internal-only exports (not for production consumers)
export { internalHelper } from "./src/helpers.js";
Self-import guardrail: never import your own extension back through its published SDK contract path from production files. Route internal imports through ./api.ts or ./runtime-api.ts instead. The SDK contract is for external consumers only.

Step 5: Add a plugin manifest

Create pllan.plugin.json in your extension root:
{
  "id": "my-channel",
  "kind": "channel",
  "channels": ["my-channel"],
  "name": "My Channel Plugin",
  "description": "Connects Pllan to My Channel"
}
See Plugin manifest for the full schema.

Step 6: Test with contract tests

Pllan runs contract tests against all registered plugins. After adding your extension, run:
pnpm test:contracts:channels   # channel plugins
pnpm test:contracts:plugins    # provider plugins
Contract tests verify your plugin conforms to the expected interface (setup wizard, session binding, message handling, group policy, etc.). For unit tests, import test helpers from the public testing surface:
import { createTestRuntime } from "pllan/plugin-sdk/testing";

Lint enforcement

Three scripts enforce SDK boundaries:
  1. No monolithic root importspllan/plugin-sdk root is rejected
  2. No direct src/ imports — extensions cannot import ../../src/ directly
  3. No self-imports — extensions cannot import their own plugin-sdk/<name> subpath
Run pnpm check to verify all boundaries before committing.

Checklist

Before submitting your extension:
  • package.json has correct pllan metadata
  • Entry point uses defineChannelPluginEntry or definePluginEntry
  • All imports use focused plugin-sdk/<subpath> paths
  • Internal imports use local barrels, not SDK self-imports
  • pllan.plugin.json manifest is present and valid
  • Contract tests pass (pnpm test:contracts)
  • Unit tests colocated as *.test.ts
  • pnpm check passes (lint + format)
  • Doc page created under docs/channels/ or docs/plugins/