gcgov / framework
Open source framework for PHP applications. Includes MongoDB modelling system.
Requires
- php: >=8.3
- ext-fileinfo: *
- ext-mongodb: *
- ext-pdo: *
- andrewsauder/json-deserialize: ^3.0
- google/cloud-kms: ^2.1
- guzzlehttp/guzzle: ^7.0
- hybridauth/hybridauth: ^3.8
- lcobucci/clock: ^2.0
- lcobucci/jwt: ^4.0
- microsoft/microsoft-graph: ^1.25
- mongodb/mongodb: ^2.1
- monolog/monolog: ^3.4
- nikic/fast-route: ^1.3
- phpmailer/phpmailer: ^6.2
- spatie/typescript-transformer: ^2.4
- swaggest/json-diff: ^3.11
- symfony/expression-language: ^7.1
- symfony/property-access: ^7.1
- symfony/validator: ^7.1
- thenetworg/oauth2-azure: ^2.1
- zircote/swagger-php: ^4.7.9
- dev-main
- v6.0.10
- v6.0.9
- v6.0.8
- v6.0.7
- v6.0.6
- v6.0.5
- v6.0.4
- v6.0.3
- v6.0.2
- v6.0.1
- v6.0.0
- v6.0.0-rc.2
- v6.0.0-rc.1
- v5.0.1
- v5.0.0
- v4.3.5
- v4.3.4
- v4.3.3
- v4.3.2
- v4.3.1
- v4.3.0
- v4.2.2
- v4.2.1
- v4.2.0
- v4.1.5
- v4.1.4
- v4.1.3
- v4.1.2
- v4.1.1
- v4.1.0
- v4.0.6
- v4.0.5
- v4.0.4
- v4.0.3
- v4.0.2
- v4.0.1
- v4.0.0
- v3.1.5
- v3.1.4
- v3.1.3
- v3.1.2
- v3.1.1
- v3.1.0
- v3.0.1
- v3.0.0
- v2.4.2
- v2.4.1
- v2.3.3
- v2.3.2
- v2.3.1
- v2.3.0
- v2.2.2
- v2.2.1
- v2.2.0
- v2.1.1
- v2.1.0
- v2.0.2
- v2.0.1
- v2.0.0
- v2.0.0-alpha
- v1.14.1
- v1.14.0
- v1.13.0
- v1.12.3
- v1.12.2
- v1.12.1
- v1.12.0
- v1.11.2
- v1.11.1
- v1.11.0
- v1.10.2
- v1.10.1
- v1.10.0
- v1.9.0
- v1.8.7
- v1.8.6
- v1.8.5
- v1.8.4
- v1.8.3
- v1.8.2
- v1.8.1
- v1.8.0
- v1.7.0
- v1.6.4
- v1.6.3
- v1.6.2
- v1.6.1
- v1.6.0
- v1.5.4
- v1.5.3
- v1.5.2
- v1.5.1
- v1.5.0
- v1.4.1
- v1.4.0
- v1.3.10
- v1.3.9
- v1.3.8
- v1.3.7
- v1.3.6
- v1.3.5
- v1.3.4
- v1.3.3
- v1.3.2
- v1.3.1
- v1.3.0
- v1.2.2
- v1.2.1
- v1.2.0
- v1.1.2
- v1.1.1
- v1.1.0
- v1.0.0
- dev-production
- dev-feat-efficiency
- dev-newaudit
- dev-feature-mdb-simplification
- dev-migrate-mdb
This package is auto-updated.
Last update: 2026-04-23 19:03:06 UTC
README
A PHP framework for Garrett County Government, Maryland, USA applications.
The framework can be used to generate a full SSR app or as a rest API. It is primarily used internally to generate APIs. Using the available extensions, a full-fledged API with Microsoft Oauth authentication can be created with no custom code.
Getting Started
The easiest way to start is to use the framework scaffolding project to start a new api and the frontend app template to start a corresponding front end application.
Requirements
Framework package requirements from composer.json:
- PHP
>=8.3 - PHP extensions:
ext-mongodb,ext-fileinfo,ext-pdo
Install dependencies with Composer:
composer install
Minimum Application Contract
The framework expects these app classes/files to exist in your /app directory:
\app\appimplementing\gcgov\framework\interfaces\app\app\routerimplementing\gcgov\framework\interfaces\router\app\rendererimplementing\gcgov\framework\interfaces\render
Controllers should implement \gcgov\framework\interfaces\controller.
Required configuration files:
/app/config/app.json/app/config/environment.json
If either file is missing, the framework throws a config exception during request handling.
System Architecture
Application File System
All apps utilizing the framework for an entire lifecycle should use this file structure.
/api
├── app
│ ├── app.php
│ ├── constants.php
│ ├── renderer.php
│ ├── router.php
│ ├── cli
│ │ ├── index.php
│ │ ├── local.bat
│ │ ├── local-debug.bat
│ │ └── prod.bat
│ ├── config
│ │ ├── app.json
│ │ └── environment.json
│ ├── controllers
│ │ └── {controller.php}
│ └── models
│ └── {model.php}
└── www
└── index.php
When you start with the framework scaffolding project, you'll automatically start with some extra folders and tools.
/api
│...
├── www
│ │...
│ ├── web.config
│ ├── web-local.config
│ └── web-prod.config
├── app
│ │...
│ └── config
│ └── environment-local.json
│ └── environment-prod.json
├── scripts
│ ├── create-jwt-keys.ps1
│ └── setup.ps1
├── srv
│ ├── {env}
│ │ └── php.ini
│ ├── tmp
│ │ ├── files
│ │ ├── opcache
│ │ ├── sessions
│ │ ├── soaptmp
│ │ └── tmp
│ └── jwtCertificates
├── db
│ ├── backup
│ ├── restore-live-to-local.ps1
│ └── local-createuser.js
├── logs
└── update-production.ps1
Core Files and Application Namespacing
The webserver should point requests to /www/index.php. URL rewriting should route application paths to this file.
gcgov\framework\router routes using $_SERVER['REQUEST_URI'] and $_SERVER['REQUEST_METHOD']. Rewrite rules should preserve the original request path in REQUEST_URI.
CLI requests should point to /app/cli/index.php.
The framework will register namespace \app to the /app directory and requires three core files in the root
of /app:
Components
Controllers
\app\controllers
A controller method called by the router must return one of the following supported types. It should always provide a response and never end code execution manually to ensure that the entire application lifecycle is executed.
New controller response types may be added to the framework to support new scenarios by adding the type and setting up
rendering methods in \gcgov\framework\renderer
- \gcgov\framework\models\controllerDataResponse
- \gcgov\framework\models\controllerPagedDataResponse
- \gcgov\framework\models\controllerFileResponse
- \gcgov\framework\models\controllerFileBase64EncodedContentResponse
- \gcgov\framework\models\controllerViewResponse
Models
\app\models
Interfaces
\app\interfaces
Exceptions
\app\exceptions
Traits
\app\traits
Services
\app\services
Routing
\app\router method getRoutes() must return an array of \gcgov\framework\models\route that maps the URL path to the
controller and defines authentication requirements.
\gcgov\framework\models\route( string|array $httpMethod = '', string $route = '', string $class = '', string $method = '', bool $authentication = false, array $requiredRoles = [], bool $allowShortLivedUrlTokens=false )
The following route will map incoming GET requests to /structure to controller \app\controllers\structure
method getAll. The route requires authentication and the user must have the role Structure.Read to execute the
request.
new route( 'GET', 'structure', '\app\controllers\structure', 'getAll', true, [ 'Structure.Read' ] );
When loading the app via CLI, the method will be CLI instead of a normal HTTP method. CLI routes do not support authentication.
new route( 'CLI', 'structure/cleanup', '\app\controllers\structure', 'cleanup', false );
Request Lifecycle
\www\index.php\app\app::_before()\app\app::__construct()\app\router::_before()\app\router::__construct()\app\router::route()\app\router::_after()\app\renderer::_before()\app\controllers\{route-controller}::_before()\app\controllers\{route-controller}::__construct()\app\controllers\{route-controller}::{route-method}()\app\controllers\{route-controller}::_after()\app\renderer::_after()\app\app::_after()
CLI
Using the framework scaffolding project, you can run the app from CLI with > app/cli/{env}.bat {url-path} Ex: > app/cli/local.bat /structure/cleanup
To enable XDebug on the CLI execution, run > app/cli/local-debug.bat {url-path}
Ex: > app/cli/local-debug.bat /structure/cleanup
Framework Services
Formatting
- Sanitize file name:
\gcgov\framework\services\formatting::fileName( string $fileName, string $replacementForIllegalChars = '-', bool $forceLowerCase = true ): string - Sanitize Excel tab name:
\gcgov\framework\services\formatting::xlsxTabName( string $tabName, string $replacementForIllegalChars = ' ', bool $forceLowerCase = false ) : string - Format DateInterval to human readable string:
\gcgov\framework\services\formatting::getDateIntervalHumanText( \DateInterval $interval ) : string
GUID
Create a GUID \gcgov\framework\services\guid::create()
HTTP
Get status text for HTTP code \gcgov\framework\services\http::statusText( int $code )
Logging
\gcgov\framework\services\log will automatically create and append a log in /logs with a filename equal to the channel
- Debug
\gcgov\framework\services\log::debug( string $channel, string $message, array $context = [] ) - Info
\gcgov\framework\services\log::info( string $channel, string $message, array $context = [] ) - Notice
\gcgov\framework\services\log::notice( string $channel, string $message, array $context = [] ) - Warning
\gcgov\framework\services\log::warning( string $channel, string $message, array $context = [] ) - Error
\gcgov\framework\services\log::error( string $channel, string $message, array $context = [] ) - Critical
\gcgov\framework\services\log::critical( string $channel, string $message, array $context = [] ) - Alert
\gcgov\framework\services\log::alert( string $channel, string $message, array $context = [] ) - Emergency
\gcgov\framework\services\log::emergency( string $channel, string $message, array $context = [] )
JWT Auth & Certificates
\gcgov\framework\services\jwtAuth\jwtAuth() provides all JWT authentication mechanisms. Explore the Oauth Server
Service and Microsoft Auth Token Exchange extensions before rolling a new solution for authentication.
Microsoft Services
Deprecated - use https://github.com/andrewsauder/microsoftServices instead
MongoDB
Comprehensive database modeling system \gcgov\framework\services\mongodb.
Use this service by defining classes in \app\models that extend \gcgov\framework\services\mongodb\model (top-level collection documents) or \gcgov\framework\services\mongodb\embeddable (nested documents that are not stored as their own collection).
How Models Work
The Mongo model stack is layered like this:
embeddablehandles BSON and JSON serialization/deserialization, typemaps,_meta, and validation helpers.dispatcherhandles embedded-model propagation (insert/update/delete in parent collections) and cascade deletion behavior.factoryprovides static data access and persistence APIs.modelis the base class your top-level collection models extend.
Every class extending \gcgov\framework\services\mongodb\model must define a public $_id field of type \MongoDB\BSON\ObjectId.
By default, a model's collection name is the class name. You can override this and user-facing names with constants:
final class inspection extends \gcgov\framework\services\mongodb\model { const _COLLECTION = 'inspection'; const _HUMAN = 'inspection'; const _HUMAN_PLURAL = 'inspections'; public \MongoDB\BSON\ObjectId $_id; }
Core Model APIs (Static)
Available through any model class (for example \app\models\inspection):
- Read:
countDocuments,getAll,getPagedResponse,getOne,getOneBy - Write:
save,saveMany - Delete:
delete,deleteMany,deleteManyBy - Analytics:
aggregation
Save and delete operations are transaction-aware. If you do not pass a \MongoDB\Driver\Session, the service opens and manages one for the operation.
Serialization, Typemaps, and _meta
embeddable provides the serialization pipeline used by all models and embedded documents:
- BSON typemaps are generated from typed properties so Mongo results hydrate into your model classes.
- Date handling maps Mongo
UTCDateTimevalues to PHPDateTimeImmutableduring read. _metais managed automatically and can include field labels, UI state, validation state, and DB operation results.- Property and class attributes control behavior such as
#[includeMeta],#[excludeBsonSerialize],#[excludeBsonUnserialize],#[excludeJsonDeserialize], and#[redact(...)].
Embedded Model Dispatch and Cascading
When saving a model, dispatcher can automatically propagate changes to embedded copies in other collections:
_insertEmbeddedpushes newly saved models into configured parent arrays._updateEmbeddedupdates embedded copies by matching_id._deleteEmbeddedremoves embedded copies when the source model is deleted._deleteCascaderecursively deletes related models marked with cascade attributes.
This behavior is driven by typemap + attribute metadata, enabling denormalized Mongo document patterns while keeping embedded data synchronized.
Additional Features Provided by the Mongo Service
#[autoIncrement]support for generated counters (including grouped/formatting scenarios)._beforeSaveand_afterSavelifecycle hooks on models.- Optional auditing/diff logging when enabled in environment config.
- Validation integration via
updateValidationState()using Symfony validation attributes.
For full reference, configuration options, attributes, and detailed examples, see:
PDODB
Initiate PDO connections using SQL connection details in app/config/environment.json. It is only a small wrapper around the native PDO class.
Read user connection: new gcgov\framework\services\pdodb\pdodb(true, $databaseName)
Write user connection: new gcgov\framework\services\pdodb\pdodb(false, $databaseName)
Extensions
Extensions add service or app level functionality to the app that registers them. Extensions may expose new endpoints.
- Open API Documentation
gcgov/framework-service-documentation- https://github.com/gcgov/framework-service-documentation
- Add namespace
\gcgov\framework\services\documentationto\app\app->registerFrameworkServiceNamespaces()
- Microsoft Auth Token Exchange
gcgov/framework-service-auth-ms- https://github.com/gcgov/framework-service-auth-ms-front
- Add namespace
\gcgov\framework\services\authmsfrontto\app\app->registerFrameworkServiceNamespaces()
- Oauth Server Service
gcgov/framework-service-auth-oauth-server- https://github.com/gcgov/framework-service-auth-oauth-server
- Add namespace
\gcgov\framework\services\authoauthto\app\app->registerFrameworkServiceNamespaces()
- User CRUD
gcgov/framework-service-user-crud- https://github.com/gcgov/framework-service-user-crud
- Add namespace
\gcgov\framework\services\usercrudto\app\app->registerFrameworkServiceNamespaces()
- Cron Monitor
gcgov/framework-service-gcgov-cron-monitor- https://github.com/gcgov/framework-service-gcgov-cron-monitor/
- Add namespace
gcgov\framework\services\cronMonitorto\app\app->registerFrameworkServiceNamespaces()