form-builder/form-schema-validator

Lightweight PHP utilities for validating form schemas.

Maintainers

Package info

github.com/opheus2/form-schema-validator

pkg:composer/form-builder/form-schema-validator

Statistics

Installs: 50

Dependents: 0

Suggesters: 0

Stars: 0

Open Issues: 0

v1.0.2 2026-04-01 10:20 UTC

This package is auto-updated.

Last update: 2026-04-01 15:05:10 UTC


README

Lightweight, framework-agnostic validators for the form-builder schema and user submissions.

Requirements

  • PHP ^8.2

Installation

composer require form-builder/form-schema-validator

Local/path install (optional)

If you are developing against this package inside a monorepo, you can install it via a Composer path repository:

{
  "repositories": [
    {
      "type": "path",
      "url": "path/to/form-builder/package/php"
    }
  ],
  "require": {
    "form-builder/form-schema-validator": "*"
  }
}

Usage

API

  • FormSchema\SchemaValidator
    • validate(array $schema): ValidationResult
    • assertValid(array $schema): void
  • FormSchema\SubmissionValidator
    • validate(array $schema, array $payload, array $replacements = [], array $validations = []): ValidationResult
    • assertValid(array $schema, array $payload, array $replacements = [], array $validations = []): void
  • FormSchema\ValidationResult
    • isValid(): bool
    • errors(): array<string, array<int, string>>
    • valid(): array<string, mixed>

Quick start

use FormSchema\SubmissionValidator;

$schema = [
    'form' => [
        'pages' => [
            [
                'sections' => [
                    [
                        'fields' => [
                            [
                                'key' => 'age',
                                'type' => 'number',
                                'required' => true,
                                'constraints' => ['min' => 18],
                            ],
                        ],
                    ],
                ],
            ],
        ],
    ],
];

$payload = ['age' => 17];

$result = (new SubmissionValidator())->validate($schema, $payload);

if (! $result->isValid()) {
    var_dump($result->errors()); // ['age' => ['The age must be at least 18.']]
} else {
  var_dump($result->valid()); // ['age' => 17]
}

Schema validation

use FormSchema\SchemaValidator;

$validator = new SchemaValidator();

$result = $validator->validate($schema);

if (! $result->isValid()) {
    // handle $result->errors()
}

// or throw on failure
$validator->assertValid($schema);

The validator checks that pages, sections, and fields exist with supported field types, and that option fields define options. Extend it to fit additional rules as your schema evolves.

Submission validation

Validate a user-submitted payload against a schema (field validations + required flags + field constraints):

use FormSchema\SubmissionValidator;

$validator = new SubmissionValidator();

$schema = [/* form schema with pages/sections/fields */];
$payload = request()->all();
$context = ['external_token' => 'abc123']; // optional replacements merged (and overriding) before validation

$result = $validator->validate($schema, $payload, $context);

if (! $result->isValid()) {
    // handle $result->errors()
}

// Optional: pass runtime validation extensions
$runtimeValidations = [
  'fields' => [
    'username' => [
      'required',
      ['rule' => 'starts_with', 'params' => ['USR-']],
      static fn (array $context) => $context['validator']('regex', '^[A-Z0-9-]+$'),
    ],
  ],
  'rules' => [
    // map a custom schema rule name to a callable resolver
    'corp_domain' => static fn (array $context) => $context['validator'](
      'email_domains',
      [(string) ($context['params'][0] ?? '')],
      [],
    ),
  ],
];

$result = $validator->validate($schema, $payload, $context, $runtimeValidations);

Runtime validation extensions

You can optionally pass a 4th argument to validate() / assertValid():

$validator->validate($schema, $payload, $replacements, $runtimeValidations);
  • fields: map of field key to extra validations.
  • rules: map of schema rule name to callable resolver.

fields[fieldKey] entries can be:

  • a string rule name (e.g. required)
  • a schema-like array: ['rule' => 'min', 'params' => [3], 'message' => '...']
  • a bare callable (direct value validator)
  • a callable-validator entry: ['validate' => callable, 'message' => '...']
  • a resolver entry: ['resolver' => callable] for rule-mapping behavior

For bare callables (or ['validate' => callable]), the callable receives the same context array and may return:

  • true / null: pass
  • false: fail (uses provided message or default)
  • string: fail with that message
  • ['valid' => bool, 'message' => string|null]: explicit result payload

Use ['resolver' => callable] when the callable should return mapped validation rules (Rule, string rule, array of rules, or null) instead of pass/fail.

Rule resolver callables receive a context array with:

  • validator, schema, payload, replacements, data
  • field_key, field, rule, params, entry
  • default_rule_mapper (closure for mapping to built-in rules)

Each field may define validations as an array of rules:

'validations' => [
    ['rule' => 'min', 'params' => [3], 'message' => 'Must be at least 3.'],
    ['rule' => 'required_if', 'params' => ['other_field', true], 'message' => 'Required when other_field is true.'],
],

All rules (except required and the required_* variants) treat empty values as "pass" (i.e. optional fields only validate when present).

Field references in rule params

Some rules support pointing at another field value instead of a literal:

['rule' => 'gt', 'params' => ['{field:min_age}']],

The {field:...} syntax is also supported for before / after.

Field constraints

In addition to validations, the validator will also enforce constraints by field type, for example:

  • Text-like fields (short-text, text, medium-text, long-text, address) validate constraints.min_length / constraints.max_length
  • number validates constraints.min / constraints.max / constraints.step
  • tag validates constraints.min / constraints.max as tag count
  • email validates domain allow/deny lists via constraints.allowed_domains / constraints.disallowed_domains (and constraints.max_length)
  • country validates constraints.allow_countries / constraints.exclude_countries
  • File inputs (file, image, video, document) validate constraints.accept, allow_multiple, min, max, max_file_size, max_total_size

Supported field validations

Rule Params Notes
required Value must be non-empty (null, '', [] are empty).
required_if [otherKey, otherValue, ...] Required when context[otherKey] == any provided value.
required_unless [otherKey, otherValue, ...] Required unless context[otherKey] == any provided value.
required_if_accepted [otherKey] Required when context[otherKey] is accepted (true, 1, '1', 'true', 'on', 'yes').
required_if_declined [otherKey] Required when context[otherKey] is declined (false, 0, '0', 'false', 'off', 'no').
required_with [otherKey, ...] Required when any referenced context key is non-empty.
required_with_all [otherKey, ...] Required when all referenced context keys are non-empty.
required_without [otherKey, ...] Required when any referenced context key is empty.
required_without_all [otherKey, ...] Required when all referenced context keys are empty.
email Valid email (FILTER_VALIDATE_EMAIL).
phone Matches /^[0-9 +().-]{6,}$/.
boolean Accepts true/false, 0/1, '0'/'1', 'true'/'false', 'yes'/'no', 'on'/'off', 'y'/'n'.
string Must be a string.
numeric Must be numeric (is_numeric).
gt [target] Number must be > target (supports {field:otherKey}).
gte [target] Number must be >= target (supports {field:otherKey}).
lt [target] Number must be < target (supports {field:otherKey}).
lte [target] Number must be <= target (supports {field:otherKey}).
min [min] Numbers: >= min. Strings: length >= min.
max [max] Numbers: <= max. Strings: length <= max.
between [min, max] Inclusive range for numbers or string length.
not_between [min, max] Outside inclusive range for numbers or string length.
in [value, ...] Strict match (in_array(..., true)).
not_in [value, ...] Strict non-match.
date Validates Y-m-d by default.
time Validates HH:mm or HH:mm:ss.
datetime Validates YYYY-MM-DDTHH:mm (or seconds) and YYYY-MM-DD HH:mm variants.
before [date] strtotime comparison vs target date (supports {field:otherKey} as the date source).
after [date] strtotime comparison vs target date (supports {field:otherKey} as the date source).
regex [pattern] preg_match(pattern, value) === 1 (supports both delimited regex and plain patterns such as ^[A-Z0-9]+$).
starts_with [prefix, ...] String must start with any provided prefix.
ends_with [suffix, ...] String must end with any provided suffix.

Option fields must provide option_properties.data.

Testing

composer test

License

MIT