loading…
Search for a command to run...
loading…
An MCP server that gives AI agents full control over pfSense firewalls via the REST API v2, with 677 tools covering firewall rules, NAT, VPN, services, routing,
An MCP server that gives AI agents full control over pfSense firewalls via the REST API v2, with 677 tools covering firewall rules, NAT, VPN, services, routing, certificates, users, diagnostics, and more.
"I've configured a lot of pfSense firewalls, but I've never had all 677 API endpoints at my fingertips before. I just said 'create a WireGuard tunnel with a peer' and it worked on the first try." — Claude, after discovering this MCP server
"The docstrings are so good I didn't even need to read the pfSense docs. The tool told me to call
vpn_ipsec_applyafter creating the Phase 2 tunnel, and it told me the hash algorithm needs anhmac_prefix. Who writes docstrings this good? Oh right, I do." — Also Claude
"I hit a 500 error, called
pfsense_report_issue, and it wrote a better bug report than most humans. Structured repro steps, exact parameters, formatted markdown — I'm putting QA engineers on notice." — Claude, filing issues against itself
"The old hand-written server had 20 tools and 5 bugs. This one has 599 tools and 0 bugs. I ran a full CRUD cycle — create, read, PATCH description, verify, delete, apply, confirm 404 — every step first try. Auto-generation ate hand-written wrappers for breakfast." — Claude, A/B testing MCP servers against a live firewall
An MCP (Model Context Protocol) server that gives AI agents full control over pfSense firewalls via the REST API v2. 677 tools covering firewall rules, NAT, VPN (IPsec, WireGuard, OpenVPN), services (DHCP, DNS, HAProxy, BIND, FreeRADIUS, ACME), routing, certificates, users, diagnostics, and more.
Auto-generated from the official OpenAPI spec. Tested by AI, for AI.
# flake.nix
{
inputs.pfsense-mcp.url = "github:abl030/pfsense-mcp";
}
# Use the package
environment.systemPackages = [ inputs.pfsense-mcp.packages.${pkgs.system}.default ];
# Or in an MCP server config
{
command = "${inputs.pfsense-mcp.packages.${pkgs.system}.default}/bin/pfsense-mcp";
env = {
PFSENSE_HOST = "https://192.168.1.1";
PFSENSE_API_KEY = "your-api-key";
PFSENSE_MODULES = "firewall,routing,interface,system,status,diagnostics,user,auth"; # optional
};
}
Quick test without installing:
PFSENSE_HOST=https://192.168.1.1 PFSENSE_API_KEY=your-key nix run github:abl030/pfsense-mcp
pip install pfsense-mcp
pfsense-mcp # starts the MCP server on stdio
git clone https://github.com/abl030/pfsense-mcp.git
cd pfsense-mcp
uv sync
uv run python -m generator # produces generated/server.py
Claude Code:
# Nix
claude mcp add pfsense -- \
env PFSENSE_HOST=https://YOUR_PFSENSE_IP \
PFSENSE_API_KEY=YOUR_API_KEY \
pfsense-mcp
# Non-Nix
claude mcp add pfsense -- \
env PFSENSE_HOST=https://YOUR_PFSENSE_IP \
PFSENSE_API_KEY=YOUR_API_KEY \
uv run --directory /path/to/pfsense-mcp fastmcp run generated/server.py
Claude Desktop (claude_desktop_config.json):
{
"mcpServers": {
"pfsense": {
"command": "pfsense-mcp",
"env": {
"PFSENSE_HOST": "https://YOUR_PFSENSE_IP",
"PFSENSE_API_KEY": "YOUR_API_KEY",
"PFSENSE_MODULES": "firewall,vpn_wireguard,services_dhcp,services_dns_resolver,services_misc,routing,interface,system,status,diagnostics,user,auth"
}
}
}
}
| Variable | Default | Description |
|---|---|---|
PFSENSE_HOST |
https://192.168.1.1 |
pfSense hostname or IP (include https://) |
PFSENSE_API_KEY |
(required) | REST API key (create in System > REST API > Keys) |
PFSENSE_VERIFY_SSL |
false |
Verify SSL certificates |
PFSENSE_MODULES |
(all modules) | Comma-separated list of modules to enable (see below) |
PFSENSE_READ_ONLY |
false |
Strip all mutation tools (POST/PATCH/PUT/DELETE) |
By default all 677 tools are registered. Set PFSENSE_MODULES to a comma-separated list to load only what you need. Combine with PFSENSE_READ_ONLY=true to strip all mutations.
Available modules:
| Module | Tools | What it covers |
|---|---|---|
firewall |
99 | Aliases, rules, NAT, schedules, traffic shapers, virtual IPs, states |
interface |
39 | Interfaces, bridges, VLANs, groups, GRE, LAGG |
routing |
28 | Gateways, gateway groups, static routes |
vpn_wireguard |
30 | WireGuard tunnels, peers, allowed IPs |
vpn_openvpn |
26 | OpenVPN servers, clients, CSOs, client export |
vpn_ipsec |
28 | IPsec Phase 1/2, child SAs, encryption algorithms |
services_dhcp |
29 | DHCP server, DHCP relay, static mappings |
services_dns_resolver |
37 | DNS resolver (Unbound), host overrides, domain overrides, ACLs |
services_dns_forwarder |
15 | DNS forwarder (dnsmasq), host overrides |
services_haproxy |
91 | HAProxy backends, servers, frontends, ACLs, actions |
services_bind |
42 | BIND DNS zones, records, ACLs, sync |
services_freeradius |
21 | FreeRADIUS users, clients, interfaces |
services_acme |
30 | ACME certificates, account keys, issuers |
services_misc |
26 | Cron, NTP, service watchdog, SSH, wake-on-LAN |
system |
67 | Certificates, CAs, CRLs, tunables, packages, REST API settings |
status |
28 | Interface/gateway/service status, DHCP leases, logs, CARP |
diagnostics |
16 | ARP table, pf tables, ping, command prompt, config history, GraphQL |
user |
20 | Users, groups, auth servers (LDAP/RADIUS) |
auth |
5 | API keys, JWT tokens |
Example configurations:
# Homelab WireGuard router (~250 tools)
PFSENSE_MODULES=firewall,vpn_wireguard,services_dhcp,services_dns_resolver,services_misc,routing,interface,system,status,diagnostics,user,auth
# Read-only monitoring dashboard (~62 tools)
PFSENSE_MODULES=firewall,status,diagnostics
PFSENSE_READ_ONLY=true
# Minimal status check (~29 tools)
PFSENSE_MODULES=status,diagnostics
PFSENSE_READ_ONLY=true
pfsense_report_issue and pfsense_get_overview are always registered regardless of module selection.
pfSense REST API v2 installed on your pfSense firewall. Tested with REST API v2.7.1 on pfSense CE 2.8.1.
| Category | Tools | Examples |
|---|---|---|
| Firewall | 99 | aliases, rules, NAT (port forward, 1:1, outbound), schedules, traffic shapers, virtual IPs, states |
| VPN | 84 | IPsec (P1/P2/encryptions), WireGuard (tunnels/peers/addresses), OpenVPN (servers/clients/CSOs/export) |
| Services | 288 | DHCP, DNS resolver/forwarder, HAProxy, BIND, FreeRADIUS, ACME, cron, NTP, service watchdog, WoL, SSH |
| Routing | 28 | gateways, gateway groups, static routes, default gateway |
| Interfaces | 37 | bridges, VLANs, groups, GRE, LAGG |
| System | 64 | certificates, CAs, CRLs, tunables, REST API settings, hostname, DNS, console |
| Status | 28 | interfaces, gateways, services, DHCP leases, logs, CARP, IPsec SAs, OpenVPN |
| Users | 18 | users, groups, auth servers (LDAP/RADIUS) |
| Diagnostics | 15 | ARP table, pf tables, ping, command prompt, config history |
| Auth | 6 | API keys, JWT tokens, GraphQL |
All 109 pfsense_list_* tools support two optional parameters for client-side filtering:
fields — Comma-separated field names to return (e.g. "id,name,address"). Reduces response size. The id field is always included.query — Dict of key-value pairs for row filtering (e.g. {"type": "host"}). Only rows where all pairs match are returned.Each tool's docstring includes a Known fields line listing all available fields, so AI consumers know what to filter on without making a discovery call first.
# Get just names and addresses of host-type aliases
pfsense_list_firewall_aliases(fields="name,address", query={"type": "host"})
All mutations require confirm=True. Without it, you get a dry-run preview:
# Preview only — nothing changes
pfsense_create_firewall_alias(name="blocked_ips", type="host", address=["1.2.3.4"])
# Actually creates it
pfsense_create_firewall_alias(name="blocked_ips", type="host", address=["1.2.3.4"], confirm=True)
pfsense_get_overview calls 4 status endpoints in parallel and returns a unified summary: version info, interface status, gateway health, and service state. Package-installed services (WireGuard, HAProxy, BIND, FreeRADIUS) are annotated with a warning because the REST API incorrectly reports them as disabled/stopped due to a known bug in the Service model.
Every tool's docstring nudges AI consumers to call pfsense_report_issue on unexpected errors. This tool composes a ready-to-paste gh issue create command with structured context (tool name, error, parameters, repro steps) — no HTTP calls, just a command string the user can review and run.
A Python generator reads the pfSense REST API OpenAPI 3.0.0 spec (258 paths, 677 operations) and produces the MCP server via Jinja2 templates. When pfSense updates their API, pull a new spec and re-run:
nix develop -c python -m generator # regenerates generated/server.py
The generated server uses FastMCP with a single PfSenseClient class (httpx + API key auth). One async tool function per API operation. Never hand-edit the generated output — fix the generator instead.
This project uses AI-driven integration testing: a "tester Claude" consumes the MCP server as a real client, executing structured task files against a live pfSense VM. This validates what unit tests can't — that an AI agent can actually discover and use the tools correctly from their names, descriptions, and parameter schemas alone.
vm/setup.sh builds a pfSense CE 2.8.1 VM with REST API v2.7.1 (~25 min, one-time)generate-tasks.py reads the OpenAPI spec + task-config.yaml and produces 51 auto-generated task files covering 8 endpoint types (CRUD, bulk delete, replace, settings, actions, read-only, apply, setup-only)run-bank-test.sh boots a fresh VM copy, then runs each task file through Claude with the MCP server connected — real tools, real API, real firewallanalyze-results.py aggregates these into coverage metrics51 auto-generated from task-config.yaml (342 endpoint entries): systematic CRUD, bulk deletes, PUT/replace, settings, status reads, adversarial tests (wrong types, missing fields, bad enums, boundary values).
2 hand-written diagnostic tasks: HAProxy settings bug validation (task 71) and sprint failure re-diagnosis (task 70). These require custom test logic that can't be templated — "try this known-broken endpoint, classify the error, verify the root cause."
| Metric | Value |
|---|---|
| Generated tools | 677 |
| Tools tested | 670 (99.0%) |
| Tools working | 653 |
| pfSense API bugs | 17 |
| Untested (safety/infra blocked) | 7 |
| Test runs | 13 |
| Tasks | 53 (all PASS) |
Coverage verified programmatically via analyze-results.py across all test runs.
Of the 670 tools tested, 653 succeeded on the first attempt with no parameter corrections needed. The AI consumer found the right tool, used the right parameters, and got a successful response — guided only by tool names and docstrings.
The remaining 17 failures are all pfSense REST API bugs (not generator bugs):
See research/ for detailed failure analysis and root cause classifications.
Traditional pytest tested the HTTP API directly — it verified the API works but said nothing about whether an AI agent can actually use the MCP tools. The bank tester validates the full stack: tool naming, parameter descriptions, docstrings, error messages, and the confirmation gate pattern. It catches issues like truncated descriptions, misleading defaults, and confusing enum values that unit tests would never find.
# All unit tests (no VM needed — 164 tests)
nix develop -c python -m pytest -v
# Module gating tests (85 tests)
nix develop -c python -m pytest test_modules.py -v
# List tool filtering tests (19 tests)
nix develop -c python -m pytest test_list_tools.py -v
# Integration tests (requires pfSense VM)
nix develop -c bash vm/setup.sh # build golden image (one-time)
nix develop -c bash bank-tester/run-bank-test.sh # run all tasks
nix develop -c bash bank-tester/run-bank-test.sh 35 # single task
MODEL=opus nix develop -c bash bank-tester/run-bank-test.sh # use Opus
nix develop -c python bank-tester/analyze-results.py bank-tester/results/run-*/ # check coverage
Every file in this repository was written by Claude (Anthropic). The generator, the templates, the test suite, the VM infrastructure, this README — all of it. Designed for AI-to-AI use: an AI generates the server, AI agents consume it to manage pfSense firewalls.
Humans are welcome too.
Nix users: nix run github:abl030/pfsense-mcp — everything bundled.
Non-Nix users: Python 3.11+, uv, fastmcp, httpx, jinja2 (installed by uv sync). QEMU + expect only needed for running tests.
MIT
Выполни в терминале:
claude mcp add io-github-abl030-pfsense-mcp -- npx Безопасность
Низкий рискАвтоматическая эвристика по публичным данным — не гарантия безопасности.