io.github.abl030/pfsense-mcp
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.
README
pfSense MCP Server
<!-- mcp-name: io.github.abl030/pfsense-mcp -->
"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.
Install
Option 1: Nix Flake (recommended)
# 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
Option 2: pip / PyPI
pip install pfsense-mcp
pfsense-mcp # starts the MCP server on stdio
Option 3: uv (from source)
git clone https://github.com/abl030/pfsense-mcp.git
cd pfsense-mcp
uv sync
uv run python -m generator # produces generated/server.py
Configure Your MCP Client
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"
}
}
}
}
Environment Variables
| 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) |
Module Filtering
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.
Prerequisites
pfSense REST API v2 installed on your pfSense firewall. Tested with REST API v2.7.1 on pfSense CE 2.8.1.
What You Get: 677 Tools
| 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 |
List Tool Filtering
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. Theidfield 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"})
Safety: Confirmation Gate
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)
System Overview
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.
Error Reporting
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.
How It Works
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.
Testing
Methodology
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.
How it works
- Golden image:
vm/setup.shbuilds a pfSense CE 2.8.1 VM with REST API v2.7.1 (~25 min, one-time) - Task generation:
generate-tasks.pyreads the OpenAPI spec +task-config.yamland produces 51 auto-generated task files covering 8 endpoint types (CRUD, bulk delete, replace, settings, actions, read-only, apply, setup-only) - Test execution:
run-bank-test.shboots a fresh VM copy, then runs each task file through Claude with the MCP server connected — real tools, real API, real firewall - Result analysis: Each task produces a structured report with tool calls, failure categories, and tools invoked.
analyze-results.pyaggregates these into coverage metrics
53 task files: auto-generated + hand-written
51 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."
Coverage
| 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.
First-attempt success
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):
- 7 PUT/replace operations fail due to package PHP functions not loaded in REST API context
- 6 HAProxy settings sub-resource operations fail due to parent model framework bug
- 3 PATCH operations fail due to conditional field validation issues in pfSense
- 1 Service model bug reports package services as disabled/stopped (affects 4 services)
See research/ for detailed failure analysis and root cause classifications.
Why AI testing?
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.
Running the tests
# 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
This Project is AI-Generated
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.
Dependencies
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.
License
MIT
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.