scafera / form
Form handling and validation for the Scafera framework
Requires
- php: >=8.4
- scafera/kernel: ^1.0
- symfony/form: ^8.0
- symfony/security-csrf: ^8.0
- symfony/validator: ^8.0
Requires (Dev)
- phpunit/phpunit: ^12.5
README
Form handling and validation for the Scafera framework. DTO-based forms with attribute validation, CSRF protection, and typed value handling.
Internally adopts symfony/form, symfony/validator, and symfony/security-csrf. Userland code never imports Symfony Form or Validator types — boundary enforcement blocks it at compile time.
Provides: Form handling and validation for Scafera — DTO classes with
Rule\*attributes, handled byFormHandler(web path, CSRF + old values) orValidator(standalone / API path). CSRF tokens scoped per DTO class (ADR-061). No Symfony FormType in userland (ADR-059).Depends on: A Scafera host project. For file uploads, compose with
scafera/filein the controller (ADR-062) — form and file are intentionally separate capabilities.Extension points:
- Attributes —
Rule\NotBlank,MaxLength,MinLength,Min,Max,Positive,OneOf,FileExtensionon DTO properties- Controlled zone —
src/Form/allows SymfonyAbstractTypeclasses for complex forms that exceed DTO capabilities (same pattern assrc/Repository/for Doctrine)- Supported DTO property types —
string,int,float,bool,DateTimeImmutable,BackedEnum, nullable variantsNot responsible for: File uploads (compose with
scafera/file, ADR-062) · form themes orform_widget()(plain HTML only) · persistence (form is input-only) · direct use ofSymfony\Component\FormorSymfony\Component\Validatoroutsidesrc/Form/(blocked byFormBoundaryPassandFormBoundaryValidator).
This is a capability package. It adds optional form handling and input validation to a Scafera project. It does not define folder structure or architectural rules — those belong to architecture packages.
What it provides
Validator— standalone DTO validation (API path)FormHandler— form creation, submission, CSRF, and validation in one flow (web path)Form— result object with errors, old values, and CSRF tokenRule\*attributes — declarative validation on DTO propertiesDtoHydrator— internal DTO mapping with type transformation (nullable, enum, DateTimeImmutable)src/Form/controlled zone — Symfony Form types for complex forms
Design decisions
- DTOs with attributes, not Symfony FormType — the common case uses plain PHP classes with
Rule\*attributes. No Symfony types in userland (ADR-059). - CSRF scoped per form identity — each DTO class gets its own CSRF token. A token from one form cannot submit another (ADR-061).
- File uploads handled separately — use
scafera/filealongsideFormHandlerin your controller (ADR-062). src/Form/is a controlled zone, not an escape hatch — Symfony Form types are allowed in this directory only, following the same pattern assrc/Repository/for Doctrine.- Type transformation on old values —
old()returns typed values (int, enum, null), not raw strings. Templates render correctly without manual casting.
Installation
composer require scafera/form
Requirements
- PHP >= 8.4
- scafera/kernel
Input DTO
use Scafera\Form\Rule; final class CreatePostInput { #[Rule\NotBlank] #[Rule\MaxLength(200)] public string $title = ''; #[Rule\MaxLength(5000)] public ?string $body = null; #[Rule\Positive] public int $priority = 1; }
Supported property types: string, int, float, bool, DateTimeImmutable, nullable variants, and BackedEnum.
Available rules
| Rule | Parameters |
|---|---|
#[NotBlank] |
— |
#[MaxLength(int)] |
max characters |
#[MinLength(int)] |
min characters |
#[Min(int|float)] |
minimum value |
#[Max(int|float)] |
maximum value |
#[Positive] |
must be > 0 |
#[OneOf(array)] |
allowed values |
#[Email] |
email format |
#[FileExtension(array)] |
allowed file extensions |
Standalone validation (API path)
use Scafera\Form\Validator; $input = new CreatePostInput(); $input->title = ''; $result = $validator->validate($input); $result->hasErrors(); // true $result->firstError(); // ValidationError{field: 'title', message: '...'} $result->toArray(); // ['title' => ['This value should not be blank.']]
Form handling (web path)
use Scafera\Form\FormHandler; #[Route('/posts/new', methods: ['GET', 'POST'])] final class CreatePost { public function __construct( private readonly FormHandler $form, private readonly ViewInterface $view, ) {} public function __invoke(Request $request): ResponseInterface { $form = $this->form->handle($request, CreatePostInput::class); if ($form->isValid()) { $input = $form->getData(); // typed CreatePostInput // persist... return new RedirectResponse('/posts'); } return new Response($this->view->render('posts/new.html.twig', [ 'form' => $form, ])); } }
Template
Plain HTML. No form themes, no form_widget():
<form method="POST"> <input type="hidden" name="_csrf" value="{{ form.csrfToken() }}"> <label>Title</label> <input name="title" value="{{ form.old('title') }}"> {% if form.error('title') %} <span class="error">{{ form.error('title') }}</span> {% endif %} <button type="submit">Create</button> </form>
Form API
| Method | Returns |
|---|---|
isSubmitted() |
bool |
isValid() |
bool (true only when submitted + valid) |
getData() |
typed input DTO |
errors() |
array<string, list<string>> |
error(string $field) |
?string (first error for field) |
old(string $field) |
mixed (submitted value, type-transformed) |
csrfToken() |
string |
Controlled zone: src/Form/
For forms that exceed DTO capabilities (nested forms, dynamic fields, entity-backed choices), place Symfony AbstractType classes in src/Form/:
src/
Form/ ← Symfony Form imports allowed here
OrderForm.php
This is a controlled zone — the same pattern as src/Repository/ for Doctrine. Symfony Form and Validator imports are allowed in this directory and blocked everywhere else.
Rules for src/Form/:
- May reference
App\Entityfor type hints and structure — must NOT perform persistence or lifecycle operations - Must NOT import from
App\Service,App\Repository,App\Controller,App\Command, orApp\Integration - Only controllers may import from
App\Form— services must NOT depend on forms - You are responsible for wiring Form classes into your controller via Symfony's
FormFactorydirectly
Boundary enforcement
| Blocked | Use instead |
|---|---|
Symfony\Component\Form\* |
Scafera\Form\FormHandler |
Symfony\Component\Validator\* |
Scafera\Form\Validator + Rule\* |
Allowed inside src/Form/ only (controlled zone). Enforced via compiler pass (build time) and validator (scafera validate). Detects use, new, and extends patterns.
License
MIT