secretspec.toml Reference
secretspec.toml Reference
Section titled “secretspec.toml Reference”The secretspec.toml file defines project-specific secret requirements. This file should be checked into version control.
[project] Section
Section titled “[project] Section”[project]name = "my-app" # Project name (required)revision = "1.0" # Format version (required, must be "1.0")extends = ["../shared"] # Paths to parent configs for inheritance (optional)require_reason = "agents" # When to require a reason for secret access (optional)| Field | Type | Required | Description |
|---|---|---|---|
name | string | Yes | Project identifier |
revision | string | Yes | Format version (must be “1.0”) |
extends | array[string] | No | Paths to parent configuration files |
require_reason | "agents" | boolean | No | When secret access must supply a reason (via --reason, SECRETSPEC_REASON, or the SDK’s with_reason()). Defaults to "agents". |
Requiring a reason for secret access
Section titled “Requiring a reason for secret access”require_reason controls when secretspec demands a reason for accessing secrets.
It accepts three values:
| Value | Behavior |
|---|---|
"agents" (default) | Require a reason only when an AI agent is detected. Humans running interactively are unaffected. |
true | Require a reason from every caller (humans, CI, agents). |
false | Never require a reason. |
Because the rule is enforced inside secretspec and checked into secretspec.toml,
every clone, CI runner, and AI agent is held to it — there is no per-tool opt-out:
# Under an AI agent, with the default "agents" policy:$ secretspec run -- ./deploy.shError: Accessing secrets requires a reason. Provide one with --reason "<why...>" ...
$ secretspec run --reason "Deploy web frontend" -- ./deploy.sh # okAgent detection. secretspec delegates detection of known agents to the
detect-coding-agent crate, which
maintains the per-tool signal list (Claude Code, Cursor, Codex, Gemini CLI,
Copilot, and more). It treats autonomous and hybrid environments as agents but
not human-driven interactive editors. In addition, secretspec checks its own
SECRETSPEC_AGENT environment variable as an explicit opt-in:
# Mark any harness the detector does not recognize as an agent:$ export SECRETSPEC_AGENT=1If your agent isn’t auto-detected, set SECRETSPEC_AGENT=1 (or use
require_reason = true to require a reason from everyone).
The reason is recorded in secretspec’s own audit log and is also forwarded to providers that support auditing (e.g. the Proton Pass provider records it in the agent audit log).
[profiles.*] Section
Section titled “[profiles.*] Section”Defines secret variables for different environments. At least a [profiles.default] section is required.
[profiles.default] # Default profile (required)DATABASE_URL = { description = "PostgreSQL connection", required = true }API_KEY = { description = "External API key", required = true }REDIS_URL = { description = "Redis cache", required = false, default = "redis://localhost:6379" }
[profiles.production] # Additional profile (optional)DATABASE_URL = { description = "Production database", required = true }Secret Variable Options
Section titled “Secret Variable Options”Each secret variable is defined as a table with the following fields:
| Field | Type | Required | Description |
|---|---|---|---|
description | string | Yes | Human-readable description of the secret |
required | boolean | No* | Whether the value must be provided (default: true) |
default | string | No** | Default value if not provided |
providers | array[string] | No | List of provider aliases to use in fallback order |
as_path | boolean | No | Write secret to temp file and return file path (default: false) |
type | string | No*** | Secret type for generation: password, hex, base64, uuid, command, rsa_private_key |
generate | boolean or table | No*** | Enable auto-generation when secret is missing |
*If default is provided, required defaults to false
**Only valid when required = false
***type is required when generate is enabled; generate and default cannot both be set
Complete Example
Section titled “Complete Example”[project]name = "web-api"revision = "1.0"extends = ["../shared/secretspec.toml"] # Optional inheritance
# Provider aliases used by profile provider chains[providers]prod_vault = "onepassword://vault/Production"shared_vault = "onepassword://vault/Shared"keyring = "keyring://"env = "env://"
# Default profile - always loaded first[profiles.default]APP_NAME = { description = "Application name", required = false, default = "MyApp" }LOG_LEVEL = { description = "Log verbosity", required = false, default = "info" }GITHUB_TOKEN = { description = "GitHub token", required = true, providers = ["env"] }
# Development profile - extends default[profiles.development]DATABASE_URL = { description = "Database connection", required = false, default = "sqlite://./dev.db" }API_URL = { description = "API endpoint", required = false, default = "http://localhost:3000" }DEBUG = { description = "Debug mode", required = false, default = "true" }
# Production profile - extends default[profiles.production]DATABASE_URL = { description = "PostgreSQL cluster connection", required = true, providers = ["prod_vault", "keyring"] }API_URL = { description = "Production API endpoint", required = true }SENTRY_DSN = { description = "Error tracking service", required = true, providers = ["shared_vault"] }REDIS_URL = { description = "Redis cache connection", required = true }Provider Aliases
Section titled “Provider Aliases”Provider aliases may be declared in two places:
- In
secretspec.toml— a top-level[providers]table. Check this into version control so every team member and CI runner sees the same mapping out of the box. - In
~/.config/secretspec/config.toml— a per-user[defaults.providers]table for personal overrides.
On conflict the project-level alias wins, so a stale local config cannot silently shadow the team’s mapping.
[providers]prod_vault = "onepassword://vault/Production"shared_vault = "onepassword://vault/Shared"keyring = "keyring://"env = "env://"
[profiles.production]DATABASE_URL = { description = "Production DB", providers = ["prod_vault", "keyring"] }[defaults]provider = "keyring"
[defaults.providers]prod_vault = "onepassword://vault/Production"shared_vault = "onepassword://vault/Shared"keyring = "keyring://"env = "env://"Manage user-level aliases via CLI:
# Add a provider alias to your user config$ secretspec config provider add prod_vault "onepassword://vault/Production"
# List all aliases known to your user config$ secretspec config provider list
# Remove an alias from your user config$ secretspec config provider remove prod_vaultThe CLI commands operate on the user-global config only — edit secretspec.toml by hand to change project-level aliases.
Audit Logging
Section titled “Audit Logging”secretspec records every secret access to a local audit log.
Auditing is a per-machine/operator concern — where the log lives and whether it is
on — so it is configured in the user-global config, not the project’s
secretspec.toml. A cloned repository therefore cannot redirect or silence your
audit log. Auditing is on by default; configure it under the top-level
[audit] table:
[audit]enabled = true # set false to turn auditing offpath = "~/.local/state/secretspec/audit.log" # default: per-user XDG state dirmax_size_bytes = 1048576 # default: 1 MiB| Field | Type | Default | Description |
|---|---|---|---|
enabled | boolean | true | Whether to record secret access. |
path | string | per-user state dir | Where to write the JSON Lines log. Must be absolute (a leading ~ is expanded); a relative path is rejected and auditing is disabled. |
max_size_bytes | integer | 1048576 (1 MiB) | Hard size cap. At the cap the file is truncated and restarted; no rotated backups are kept. |
Secret values are never written to the log, and credentials embedded in provider URIs are redacted. Audit failures never block secret access. See Audit Logging for the record format and full details.
as_path Option
Section titled “as_path Option”When as_path = true, the secret value is written to a temporary file and the file path is returned instead of the value:
[profiles.default]TLS_CERT = { description = "TLS certificate", as_path = true }GOOGLE_APPLICATION_CREDENTIALS = { description = "GCP service account", as_path = true }| Context | Behavior |
|---|---|
CLI (get, check, run) | Files are persisted (not deleted after command exits) |
| Rust SDK | Files cleaned up when ValidatedSecrets is dropped; use keep_temp_files() to persist |
| Rust SDK types | PathBuf or Option<PathBuf> instead of String |
Secret Generation
Section titled “Secret Generation”When type and generate are set, missing secrets are automatically generated during check or run and stored via the configured provider:
[profiles.default]# Simple: generate with type defaultsDB_PASSWORD = { description = "Database password", type = "password", generate = true }REQUEST_ID = { description = "Request ID prefix", type = "uuid", generate = true }
# Custom optionsAPI_TOKEN = { description = "API token", type = "hex", generate = { bytes = 32 } }SESSION_KEY = { description = "Session key", type = "base64", generate = { bytes = 64 } }
# Shell commandMONGO_KEY = { description = "MongoDB keyfile", type = "command", generate = { command = "openssl rand -base64 765" } }
# RSA private key (PKCS1 PEM)JWT_SIGNING_KEY = { description = "JWT signing key", type = "rsa_private_key", generate = true }
# Type without generate: informational only, no auto-generationMANUAL_SECRET = { description = "Manually managed", type = "password" }Generation Types
Section titled “Generation Types”| Type | Default Output | Options |
|---|---|---|
password | 32 alphanumeric chars | length (int), charset ("alphanumeric" or "ascii") |
hex | 64 hex chars (32 bytes) | bytes (int) |
base64 | 44 chars (32 bytes) | bytes (int) |
uuid | UUID v4 (36 chars) | none |
command | stdout of command | command (string, required) |
rsa_private_key | 2048-bit RSA private key (PKCS1 PEM) | bits (int) |
Behavior
Section titled “Behavior”- Generation only triggers when a secret is missing — existing secrets are never overwritten
- Generated values are stored via the secret’s configured provider (or the default provider)
- Subsequent runs find the stored value and skip generation (idempotent)
generateanddefaultcannot both be set on the same secrettype = "command"requiresgenerate = { command = "..." }(not justgenerate = true)
Profile Inheritance
Section titled “Profile Inheritance”- All profiles automatically inherit from
[profiles.default] - Profile-specific values override default values
- Use the
extendsfield in[project]to inherit from other secretspec.toml files