Development and Contribution Guide
To get started with development on the pfSense REST API package, you will need to have a good understanding of the package's structure, object-oriented PHP, and how to contribute to the project. This guide will help you get started with the REST API package and provide you with the information you need to develop effectively.
Contributing
For those looking to contribute to the project, you can contribute to the project in a variety of ways, including but not limited to:
- Bug fixes
- New features
- Documentation improvements
- Reporting and collaborating to open issues
- Opening and contributing to discussions
To make a code contribution, please follow these steps:
- Fork the repository on GitHub.
- Clone your fork locally and make your changes.
- Commit your changes and push them to your fork.
- Open a pull request on the main repository to merge your changes into the project.
- Wait for your pull request to be reviewed and merged.
Important
If your contribution involves a security vulnerability, please do not open a public issue or pull request. Instead, please report the vulnerability to a project maintainer directly.
Project Structure
The majority of the pfSense REST API package code can be found at pfSense-pkg-RESTAPI/files/usr/local/pkg/RESTAPI
.
The project utilizes PHP namespaces to organize its codebase.
Below is a brief overview of the project's structure:
RESTAPI/
The main namespace for the REST API package. All classes and interfaces are organized under this namespace or a child
namespace within the RESTAPI/
directory.
.resources/
While not a PHP namespace, this directory contains common resources and tools used by the REST API package. This includes:
cache/
Contains the cache files for the REST API package. Cache files are typically JSON datasets populated by \RESTAPI\Core\Cache child classes and are refreshed on the schedule defined in the class.
vendor/
Contains additional PHP libraries and classes required by the REST API package. Because pfSense does not include a PHP package manager, these libraries are installed to this directory via composer when the package is built.
scripts/
Contains helper scripts for the REST API package. These scripts are used to automate tasks such as generating OpenAPI documentation, running tests, and more.
Auth/
Contains the \RESTAPI\Core\Auth child classes. Auth classes are used to define and manage authentication methods for the REST API package.
Caches/
Contains the \RESTAPI\Core\Cache child classes. Cache classes are used to define and manage cache data and files for the REST API package.
ContentHandlers/
Contains the \RESTAPI\Core\ContentHandler child classes. ContentHandler classes are used to define and manage the supported content-type
and accept
header options for the REST API package.
Core/
Contains the core classes and interfaces for the REST API package framework.
Dispatchers/
Contains the \RESTAPI\Core\Dispatcher child classes. Dispatcher classes are used to define and manage long-running processes that run in the background.
Endpoints/
Contains the \RESTAPI\Core\Endpoint child classes for the REST API package. Endpoint classes are used to define, customize, and manage the REST API endpoints. Each endpoint class corresponds to a specific REST API endpoint in the pfSense web root and is responsible for dynamically generating the endpoint's PHP file in the web root.
Fields/
Contains the \RESTAPI\Core\Field child classes. Field classes are used to define and manage various data types supported by the REST API and are intended to be assigned to Model classes.
Forms/
Contains the \RESTAPI\Core\Form child classes. Forms are used to dynamically generate pfSense webConfigurator pages for a specific Model class.
Models/
Contains the \RESTAPI\Core\Model child classes. Model classes are used to define and manage the data structure of the object's the REST API package interacts with (i.e. firewall rules, static routes, etc.).
ModelTraits/
Contains PHP Traits that can be used to extend the functionality of Model classes. Traits are used to define and manage common functionality that can be shared across multiple Model classes. For example, Models that
interact with log files can use the LogFileTrait
to add log file specific functionality to the Model.
QueryFilters/
Contains the \RESTAPI\Core\QueryFilter child classes. QueryFilter classes are used to define and manage the query filters that are used to filter data returned by the REST API. For information about writing query filters, see the Building Custom Query Filter Classes documentation.
Responses/
Contains the \RESTAPI\Core\Response child classes. Response classes are throwable classes that are used to generate and return a specific HTTP response code and message. Endpoints will automatically catch these exceptions if thrown and return the appropriate, JSON formatted response to the client along with the appropriate HTTP response code and headers.
Tests/
Contains the \RESTAPI\Core\TestCase child classes. TestCase classes are used to define and manage the tests for the REST API package. See the Testing section for more information.
Validators/
Contains the \RESTAPI\Core\Validator child classes. Validator classes are reusable classes that are used to validate data for a specific Field object. They are intended to be assigned to a Field object's validators
property to perform the specific validation against that field.
autoloader.inc
The autoloader script for the REST API package. This file can be included in your .php or .inc file's require_once
statements to automatically load all RESTAPI classes.
Tip
The full PHP API reference for this package including documentation for all applicable classes can be found here.
Style Guidelines
This projects uses opinionated code formatters to ensure a consistent code style across the project, this also ensures that contributions are easier to review and maintain. The project uses the following code formatters:
- Prettier with the PHP plugin for PHP files
- From the project root, run
npm install
to install Prettier and the Prettier-PHP plugin. - From the project root, run
./node_modules/.bin/prettier --write .
to format all files in the project.
- From the project root, run
- Black for Python files
- From the project root, run
pip install -r requirements.txt
to install Black. - From the project root, run
black .
to format all Python files in the project.
- From the project root, run
Building the Package From Source
Building this package requires access to a suitable FreeBSD build environment. The package must be built using the FreeBSD version that corresponds to the pfSense version you are building for. Luckily, there are several tools available to help you build the package:
Scripted Build Process
Step 1: Install Virtualbox and Vagrant
On your development machine, install Virtualbox and Vagrant. These tools will allow you to create a FreeBSD virtual machine for building the package with minimal effort.
Step 2: Clone the Repository
Clone the repository to your development machine:
git clone git@github.com:jaredhendrickson13/pfsense-api.git
Step 3: Run the vagrant-build.sh script
From the project root, run the vagrant-build.sh
script to create a FreeBSD virtual machine and build the package:
sh vagrant-build.sh
After the script completes, the package will be built and the resulting .pkg
file will be located in your current
working directory.
Important
- You will need to copy this
.pkg
file to your pfSense instance and install it usingpkg
. The package will not operate correctly if installed on a non-pfSense FreeBSD system. - The script will not automatically destroy the virtual machine after the build process completes. You will need to manually destroy the virtual machine using
vagrant destroy
when you are finished.
Supported Build Environment Variables
Below are the supported environment variables to customize the build process:
FREEBSD_VERSION
: The version of FreeBSD to use for the build. This must be an existing Vagrant box name including thefreebsd/
prefix. The default value isfreebsd/FreeBSD-14.0-CURRENT
. For a list of available boxes, see the Vagrant Cloud page.BUILD_VERSION
: The version tag to give the build, this must be in a FreeBSD package version format (e.g.0.0_0
). The default value is0.0_0-dev
.
Testing
Testing this package poses a number of challenges. Because pfSense is a hardened, security-focused operating system, it is difficult to install traditional testing tools like PHPUnit. Instead, the package's framework includes a custom TestCase class that resembles PHPUnit, but is designed to work within the pfSense environment without requiring additional dependencies. This project uses a mix of unit tests, integration tests, and end-to-end tests to ensure the package is stable and reliable. Common test cases include:
- Ensuring the package's framework functions as intended.
- Ensuring Model classes correctly configure and interact with the pfSense system. This includes testing that the corresponding backend services are correctly configured after a Model object is created, updated, deleted, etc.
- Ensuring that the package's endpoints correctly handle requests and return the expected responses.
It is not important to differentiate between unit, integration and end-to-end tests when writing tests for this package. The package's testing suite is designed to run all tests in a single command and will automatically categorize tests. The most important thing is to ensure that tests are written to sufficiently cover the expected behavior of the package.
Writing Tests
Writing tests for the package is similar to writing tests for any other PHP project. The package's testing framework
will automatically discover and run tests that extend the package's TestCase class
and are located in the RESTAPI/Tests/
directory. To write a test, you can create a new PHP file in the RESTAPI/Tests/
directory and extend the TestCase
class. Any method within your TestCase class that begins with test
will be run
as a test case.
Below is a simple example of a TestCase class:
<?php
namespace RESTAPI\Tests;
require_once('RESTAPI/autoloader.inc');
use RESTAPI\Core\TestCase;
class MyTestCase extends TestCase
{
public function test_something()
{
# Add your test logic here
$this->assert_is_true(true);
}
}
Important
The testing framework will make a copy of the pfSense configuration before the tests are run and will restore the configuration after the tests complete. This ensures that the tests do not modify the pfSense configuration permanently which could cause issues with the firewall or remaining tests. It is still recommended that any changes made to the configuration during a test are removed or reverted at the end of the test.
Running Tests
To run all tests for the package, you can use the following command:
pfsense-restapi runtests
To run tests that contain a specific keyword, you can use the following command:
pfsense-restapi runtests <keyword>
Danger
Running tests is a disruptive process and should not be done on a production pfSense instance. It is recommended to only run tests on a development or testing instance of pfSense. A virtual machine with snapshots is ideal for this.
Documentation
There are three different sets of documentation for this project, each serves a different purpose and is managed by different tools:
Mkdocs
The pfSense REST API Guide site is built using Mkdocs. These docs are mostly intended to help users
get started using the package, as well as providing an introduction to development of the package for more advanced users.
The source files for the site are written in Markdown and located in the docs/
directory and the configuration file
is located at mkdocs.yml
. To start a local development server for the documentation site while you work on documentation,
you can run the following commands:
python3 -m pip install -r requirement.txt
mkdocs serve
Note
You must have Python 3 installed on your system to run the Mkdocs development server. It is recommended to setup a virtual environment for this project to avoid conflicts with other Python projects.
PHPDoc
The PHP reference documentation for this project is generated using
PHPDocumentor. The PHP reference provides detailed documentation for all PHP classes, functions,
script, etc. included in the package and is intended for those wishing to aide in development of the package. PHPDoc
looks at the PHPDoc comments in the project's PHP files and generates a static HTML site with the documentation.
The PHPDoc configuration file is located at phpdoc.dist.xml
. There are two ways to generate the PHP documentation
and start a local development server to view the documentation:
Important
This should be done on your local development machine and not on a pfSense instance.
Via Phar
# Download the phpdoc phar
wget https://phpdoc.org/phpDocumentor.phar
chmod +x phpDocumentor.phar
# Build the docs
./phpDocumentor.phar -c phpdoc.dist.xml
# Start a local development server
cd .phpdoc/build
php -S localhost:8000
Via Docker
# Build the docs using Docker
docker run --rm -v $(pwd):/data phpdoc/phpdoc run -c phpdoc.dist.xml
# Start a local development server
cd .phpdoc/build
php -S localhost:8000
Note
This method requires you to have Docker installed and running on your system. If you do not have Docker installed, you can use the Phar method to generate the documentation.
OpenAPI (Swagger UI)
The OpenAPI documentation (Swagger UI) provides accurate documentation for the various API endpoints and their available attributes. The OpenAPI specification is automatically generated by the package using properties defined in Endpoint, Model, Fields, Auth, ContentHandler and Response classes. Each of these classes have properties that map to OpenAPI properties and are used to generate the OpenAPI documentation. The OpenAPI documentation is automatically generated when the package is initially installed and the Swagger UI is accessible under 'System > REST API > Documentation'. If you make changes to the package's codebase, you can regenerate the OpenAPI documentation to include the changes by running the following command on your pfSense instance:
pfsense-restapi buildschemas
GraphQL
The GraphQL schema provides accurate documentation for the various GraphQL queries and mutations available in the package. The GraphQL schema is automatically generated by the package using properties defined in Endpoint, Models, and Fields. Each of these classes have properties that map to GraphQL properties and are used to generate the GraphQL schema. The GraphQL schema is automatically generated when the package is initially installed and the GraphQL schema is accessible at /api/v2/graphql/schema. If you make changes to the package's codebase, you can regenerate the GraphQL schema to include the changes by running the following command on your pfSense instance:
pfsense-restapi buildschemas