A small, hand-wired PHP application kernel built on Symfony components, Doctrine ORM, Firebase JWT, and a strict-typed PSR-7 fork. Designed for security-conscious SaaS applications that want Symfony-grade components without Symfony's full kernel, bundle system, and compile step.
- Slim is too thin. No Doctrine, no validation, no security primitives — the consumer wires everything.
- Symfony is too heavy. A compiled DI container, an event dispatcher, bundles, MakerBundle, recipes. Excellent for large apps; more than most SaaS workloads need.
- Laravel is opinionated and non-Symfony. Facades, ActiveRecord, and a separate ecosystem.
- Appkit sits in between. Symfony components plus Doctrine plus a thin abstract kernel, with a hand-compiled container so the file you read is the resolution path that runs.
- Fast boot. No DI compile step, no cache invalidation. Config files are
loaded with
require; OPcache handles the rest. - Transparent control flow. No event dispatcher by design. Reading
handleAuthentication()top-to-bottom shows exactly what runs. - RoadRunner-aware. Every stateful service implements
ResetInterface; the kernel rebuildsApplicationStateper request. - Security hardening already wired. CSRF rotation on login, session-fixation defence, token unserialize allowlist, password timing-parity dummy, brute-force protection, generic 401 responses.
- Strict typing. PHP 8.2+,
declare(strict_types=1)throughout. The bundled PSR-7 implementation is a strict-typed fork ofnyholm/psr7.
composer create-project modufolio/appkit-skeleton my-app
cd my-app
composer startThe skeleton lives in its own repository: modufolio/appkit-skeleton.
<?php
declare(strict_types=1);
namespace App\Controller;
use Modufolio\Appkit\Core\AbstractController;
use Modufolio\Psr7\Http\Response;
use Psr\Http\Message\ResponseInterface;
use Symfony\Component\Routing\Attribute\Route;
final class HelloController extends AbstractController
{
#[Route('/hello/{name}', methods: ['GET'])]
public function show(string $name): ResponseInterface
{
return Response::json(['message' => "Hello, {$name}"]);
}
}Full guides under docs/:
- Getting started — install, configure, and run your first app
- Kernel — request lifecycle, service container, boot
- Routing — routes, parameters, access control
- Controllers — controllers and parameter attributes
- Dependency injection — wiring services with config files
- Templates — layouts, snippets, sections, asset helpers
- Security — firewalls, access control, CSRF, roles
- Authenticators — form login, JWT, OAuth 2.1, 2FA, brute-force
- Database — Doctrine ORM, QueryBuilder, pagination, soft delete
- Forms — validation,
ValidationResult, payload mapping - Exception handling — turning exceptions into HTTP responses
- File uploads — validating and storing uploaded files
- Image processing — Darkroom, Dimensions, DiskManager
- Console — built-in commands,
make:entity, writing your own - Toolkit — array, file, string, and directory utilities
- Testing — PHPUnit, EntityFactory, static analysis
- Deployment — Nginx/Caddy, permissions, RoadRunner, databases
- Configuration — environment variables and config reference
Start with the introduction for the architecture overview and the design philosophy the rest of the documentation assumes.
- PHP 8.2 or later
- Composer
- Extensions:
curl,dom,exif,fileinfo,gd,intl,libxml,pdo,simplexml,sqlite3,zip
See composer.json for the canonical dependency list.
MIT. See LICENSE.