Skip to content

LAA-Software-Engineering/agentic-control-plane

Agentic Control Plane

CI Release Go 1.22+ License: Apache-2.0 Go Reference

image

Declarative YAML for agents, tools, workflows, and policies — with a Terraform-style plan/apply loop, local SQLite state, and execution traces.

This is not another opaque agent framework. It is a control plane: you describe the desired shape of your agent system in versioned resources, then validate, plan, apply, run, and inspect logs the same way you would operate real infrastructure.


Why this exists

Most agent stacks today bury prompts, tool wiring, and permissions in application code. That makes it hard to answer basic operational questions: Is this config valid? What changed? What are we about to deploy? What actually ran? Did policy allow it?

Agentic Control Plane pushes those concerns into explicit YAML (Kubernetes-like resources) and a small Go CLI (agentctl), so teams can:

  • Review diffs and plans before changes land
  • Track deployment state separately from runtime traces
  • Enforce policies (budgets, approvals, tool rules) at execution time
  • Stay local-first while the architecture leaves room for a future remote control plane

The full product vision, YAML spec v0, and architecture are documented in docs/DESIGN_DOC.md.

Featured walkthrough: declarative PR review with a policy-blocked (simulated) GitHub comment — no API keys required — in examples/pr-review-demo/README.md. For the live GitHub read/write path with a mock reviewer (CI-friendly, no API keys), see examples/pr-review-github/README.md. For the same flow with OpenAI gpt-4o-mini plus GitHub Actions (PR workflow posts a review comment), see examples/pr-review-github-actions/README.md ( .github/workflows/agentctl-pr-review.yml; optional manual owner/repo/number: .github/workflows/agentctl-pr-review-publish.yml).


Mental model

Idea Analogy
Desired resources in Git GitOps
plan / apply / drift Terraform
Typed resources (Project, Workflow, Policy, …) Kubernetes-style API
Tool and IO contracts OpenAPI-style explicitness

Features (MVP today)

  • agentctl init — scaffold project.yaml, policies, tools, and a sample workflow
  • agentctl validate — load project, apply project defaults (spec.defaults), then environment overlays (-e / --env, Environment resources §7.6), then validate graph, schemas, and references; runs policy lint (ungated sensitive tools, invalid HITL config, etc.) as advisory output — use --strict to exit 2 on high-severity lint findings (fail-closed safety metadata still gates at run even when lint passes)
  • agentctl plan — diff desired graph vs SQLite deployment state; risk hints including policy lint; JSON/YAML output includes policyLint and a deploymentBaseline digest for the store snapshot
  • agentctl apply — persist plan (TTY confirm or --auto-approve / AGENTCTL_AUTO_APPROVE); optimistic concurrency — if the deployment store changed after the plan snapshot (e.g. another process applied the same --state file while this run waited at the prompt), apply fails with exit code 3; re-run plan then apply
  • agentctl run — execute a workflow locally; JSON Schema for inputs where configured; policy gates pause for human-in-the-loop (HITL) approval when a tool call requires it
  • agentctl logs — read trace events from SQLite (--run, --workflow, or recent runs)
  • agentctl audit verify — re-walk hash-linked trace chains and detect tampering (see docs/AUDIT_CHAIN.md)
  • Toolsnative, http, mock, and mcp — MCP supports stdio (subprocess) or streamable HTTP (spec.mcp.transport: http, url, optional headers with env: tokens)
  • Project defaults — besides model and policy, optional runtime flows to spec.runtime on agents/workflows when omitted (MVP: local or unset; see spec validation)
  • Output — table, JSON, or YAML (-o / --output)
  • State — single SQLite file (default .agentic/state.db under the project root; override with --state)
  • Tests — unit/integration coverage, golden CLI output tests, end-to-end init → … → logs in test/integration

See section 18 (MVP) and section 19 (End Goal) in docs/DESIGN_DOC.md for the full included/excluded list.


Quick start

Prerequisites

  • From source: Go 1.22+
  • From a release binary: no Go toolchain; put agentctl (or agentctl.exe) on your PATH after extracting the archive

Build

git clone https://github.com/LAA-Software-Engineering/agentic-control-plane.git
cd agentic-control-plane
make build   # writes bin/agentctl

Or: make install / go install ./cmd/agentctl (honours GOBIN / GOPATH/bin; ensure that directory is on PATH).

Prebuilt binaries

GitHub Releases ship agentctl for common platforms (.tar.gz on Linux/macOS, .zip on Windows) plus SHA256SUMS.txt. Pick the archive that matches your machine, for example:

Platform Asset suffix
Linux x86_64 linux-amd64.tar.gz
Linux arm64 linux-arm64.tar.gz
macOS Intel darwin-amd64.tar.gz
macOS Apple Silicon darwin-arm64.tar.gz
Windows x86_64 windows-amd64.zip (contains agentctl.exe)

agentctl version reports the release tag (e.g. v0.1.4).

Releases are created automatically when changes land on main, using a patch semver bump over the latest vMAJOR.MINOR.PATCH tag (merges that only touch Markdown or the root Makefile do not trigger a release). To cut minor or major bumps on demand, run the Release workflow manually (Actions → Release → Run workflow) and choose the bump type.

Create a project and run the loop

From the repo root (or anywhere):

agentctl init my-agent-system
agentctl validate --project my-agent-system
agentctl plan   --project my-agent-system
agentctl apply  --project my-agent-system --auto-approve
agentctl run    workflow/hello --project my-agent-system
agentctl logs   --project my-agent-system --workflow hello
agentctl audit verify --project my-agent-system --run <run-id>
agentctl inspect --web --project my-agent-system   # read-only local UI on http://127.0.0.1:8787

inspect --web binds to localhost only and opens the state DB read-only. Avoid running it while agentctl run is writing the same SQLite file (you may see database is locked without WAL); use it when runs are idle or on a copy of the DB.

Example project.yaml

The project root is a Project resource: apiVersion, kind, metadata.name, and spec.imports listing other YAML files (policies, tools, workflows). After agentctl init my-agent-system, my-agent-system/project.yaml looks like this:

apiVersion: agentic.dev/v0
kind: Project
metadata:
  name: my-agent-system
spec:
  imports:
    - ./policies/default.yaml
    - ./tools/helper.yaml
    - ./workflows/hello.yaml
  defaults:
    policy: default
    model: openai/gpt-4o-mini
    runtime: local
  providers:
    models:
      openai:
        type: openai
        apiKeyFrom: env:OPENAI_API_KEY
      # Optional: Claude via Messages API (set ANTHROPIC_API_KEY and use e.g. defaults.model: anthropic/claude-sonnet-4-20250514)
      # anthropic:
      #   type: anthropic
      #   apiKeyFrom: env:ANTHROPIC_API_KEY

Field-by-field rules, extra kinds, env overlays, MCP HTTP tools, and defaults.runtime are in docs/DESIGN_DOC.md. See docs/EXAMPLES.md for Anthropic fragments, MCP over HTTP, and structured-output notes.

Notes:

  • init creates my-agent-system/ with apiVersion: agentic.dev/v0 resources and a hello workflow (native echo tool only — no network).
  • apply in non-interactive environments needs --auto-approve or AGENTCTL_AUTO_APPROVE=1.
  • run HITL: gated tool calls exit with Status: interrupted (exit 0). Resume with --resume <run-id> --decision approve|reject|edit|switch (use --decision-edit-json / --decision-switch-target when needed), or skip prompts with --auto-approve / AGENTCTL_AUTO_APPROVE=1. Pre-approve a specific call with repeated --approve <uses>. Set AGENTCTL_HITL_ACTOR to attribute decisions in trace logs.
  • Policy.spec.hitl.interruptOn keys are Tool metadata.name values; they configure review options (edit rules, switch targets) for calls already gated by approvals.requiredFor or safety metadata — they do not gate tools on their own.
  • run stores traces in the same SQLite file used for plan/apply (default .agentic/state.db under --project). Optional OTLP export (spec.telemetry, off by default) is additive only — see docs/OTEL.md. When enabled you need serviceName plus either consoleExport: true or an endpoint (https://… or env:VAR, e.g. env:OTEL_EXPORTER_OTLP_ENDPOINT). Export that variable before run if you use env:; if it is missing or the collector is unreachable, agentctl logs a warning, skips OTLP, and the workflow still completes (SQLite traces unchanged).
  • If spec.traces.retentionDays is a positive integer, runs older than that many UTC calendar days (by runs.started_at) are deleted lazily on run and logs (child trace rows cascade). Unset or non-positive means no pruning.
  • Trace payload redaction (issue #110): before SQLite storage, event JSON is sanitized, key-redacted, and size-capped. Defaults mask common secret key names (substring match on map keys). Optional project knobs:
    • spec.traces.redactKeys / maxPayloadBytes — merged with defaults; also available under spec.traces.redaction together with maxDepth, maxStringChars, and maxBytes (max bytes for binary previews in sanitized values, not the overall JSON cap).
    • Stored events may show [REDACTED], payload_truncated / preview, or depth/binary placeholders in logs / inspect --web.
  • Use logs --run <id> after a run if you want a single run’s trace (IDs are printed by run).

Global flags (common)

Flag Purpose
--project <path> Project root (default .)
--state <path> SQLite file override
-e / --env Environment overlay name
-o / --output table, json, or yaml
--no-color ASCII-friendly validate output

Exit codes are summarized in section 11.2 of docs/DESIGN_DOC.md (0 success, 2 validation, 3 plan/apply conflict when deployment state changed after plan or resolved config drifted before run, 4 execution, 5 policy denial, …).

User-local config (per-developer overrides)

Config is resolved in this order (highest wins): CLI flagsenvironment overlay (-e) → project YAMLuser-localbuilt-in defaults.

Optional user-local files (git-ignored, strict YAML — typos fail validate):

Path Scope
$XDG_CONFIG_HOME/agentctl/config.yaml or ~/.config/agentctl/config.yaml Global per-user defaults (defaults, state, providers, traces, telemetry)
.agentic/local.yaml under --project Project-scoped overrides (same fields; wins over the global file)

validate, plan, and apply write .agentic/resolved-config.json (digest of the resolved graph + env + state path). run rejects drift from that snapshot with exit 3 — re-run validate or plan after changing config.


Repository layout (high level)

Path Role
cmd/agentctl CLI entrypoint
internal/cli Cobra commands, flags, golden tests
internal/spec YAML types, normalize, validate
internal/config Layered config resolution, immutable snapshot
internal/project Load project + imports
internal/plan Planner and risk summary
internal/apply Apply plan to deployment store
internal/engine Workflow execution
internal/policy Policy evaluation
internal/state/sqlite SQLite deployment + runtime/trace tables
internal/audit Tamper-evident hash chain for trace events (issue #116)
test/integration End-to-end CLI flow tests
docs/DESIGN_DOC.md Spec, CLI UX, architecture, roadmap
docs/GITHUB_ACTIONS.md Running agentctl from GitHub Actions (tokens, exit code 5, template path)
examples/pr-review-github-actions/ Full gpt-4o-mini project; PR workflow .github/workflows/agentctl-pr-review.yml; optional publish .github/workflows/agentctl-pr-review-publish.yml

Development

make defaults to help, which lists targets; the table below mirrors the Makefile (## comments and recipes).

Target What it does
help Show usage and target list (default goal)
all fmtvettestbuild (handy before a push)
build go buildbin/agentctl
install go install ./cmd/agentctl (-trimpath; uses GOBIN / GOPATH/bin)
clean Remove bin/ and coverage.out
fmt go fmt ./...
verify-fmt Fail if gofmt -l would list files (matches CI-style formatting check)
vet go vet ./...
test go test ./... -race
test-coverage Tests with -coverprofile=coverage.out and a one-line go tool cover -func summary
check vet + test only (no formatting writes)
ci verify-fmt + vet + test (no build)

CI (.github/workflows/ci.yml) runs Linux, macOS, and Windows on Go 1.22.x, plus Go 1.23.x on Linux, with race and shuffle enabled (workflow steps are defined in YAML, not via make ci).

Updating CLI golden files

When table output is intentionally changed:

GO_UPDATE_GOLDEN=1 go test ./internal/cli/... -run TestGolden_

Roadmap

Near term (MVP hardening)

Recent landings already cover much of “hardening”: plan/apply optimistic concurrency (exit 3 when deployment state drifts), MCP over streamable HTTP as well as stdio, trace retention (spec.traces.retentionDays), defaults.runtime / spec.runtime (MVP local), and clearer defaults vs environment overlay documentation. What is still open for near-term polish:

  • More diff / drift UX where the design doc calls for it (beyond today’s resource-level diff)
  • Richer logs filtering (see sections 10.2 and 17.3 in docs/DESIGN_DOC.md); inspect --web covers read-only run/state browsing (#109)
  • agentctl test-style workflow fixtures (stretch per design doc)

Post-MVP (from design doc section 19)

  • Modules/registry, remote shared state, reconciliation controllers
  • Parallel steps, subworkflows, schedules/events
  • Stronger drift semantics and multi-runtime targets
  • Deeper approval workflows and multi-tenant controls

The recommended implementation phases are outlined in section 20 of docs/DESIGN_DOC.md.


Documentation

  • docs/DESIGN_DOC.md — design document v0 (problem statement, spec, CLI, engine, state model, testing strategy, MVP vs end state, section 23 recommendation).
  • docs/AUDIT_CHAIN.md — hash-linked trace audit chain and agentctl audit verify (issue #116).
  • docs/ATTRIBUTION.md — tenant, thread, and actor fields on runs and traces (issue #111).
  • docs/OTEL.md — optional OTLP trace export alongside SQLite (issue #108).
  • examples/pr-review-demo/README.md — end-to-end demo: structured review output, traceable run, approval-gated write (validateplanapplyrunlogs).
  • docs/EXAMPLES.md — copy-paste YAML and CLI examples (init, mock vs OpenAI, workflows, environment overlays).
  • CODE_OF_CONDUCT.md — Contributor Covenant 2.1; participation expectations and reporting.
  • License: MIT

Contributing

Issues and pull requests are welcome. See CONTRIBUTING.md for local setup, tests, golden updates, and pull request expectations.


Local declarative agent systems with validate, plan, apply, run, and logs.
(Closing recommendation in docs/DESIGN_DOC.md, section 23.)

About

Terraform-style plan/apply for agent systems: versioned YAML for agents, tools, workflows, and policies; local-first SQLite state; MCP & HTTP tools; structured traces.

Topics

Resources

License

Code of conduct

Contributing

Stars

Watchers

Forks

Packages

 
 
 

Contributors