Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Introduce Form Template to improve inference for processed payloads #248

Merged
merged 10 commits into from
Nov 28, 2023
2 changes: 1 addition & 1 deletion src/Element.php
Original file line number Diff line number Diff line change
Expand Up @@ -311,7 +311,7 @@ public function getLabelAttributes(): array
*
* Implementation will decide if this will overwrite or merge.
*
* @return $this
* @return self
gsteel marked this conversation as resolved.
Show resolved Hide resolved
* @throws Exception\InvalidArgumentException
*/
public function setLabelOptions(iterable $arrayOrTraversable)
Expand Down
2 changes: 1 addition & 1 deletion src/Fieldset.php
Original file line number Diff line number Diff line change
Expand Up @@ -585,7 +585,7 @@ public function bindValues(array $values = [], ?array $validationGroup = null)
/**
* Set if this fieldset is used as a base fieldset
*
* @return $this
* @return self
*/
public function setUseAsBaseFieldset(bool $useAsBaseFieldset)
{
Expand Down
24 changes: 22 additions & 2 deletions src/Form.php
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,10 @@
use function is_object;
use function sprintf;

/**
* @template TFilteredValues
* @implements FormInterface<TFilteredValues>
*/
class Form extends Fieldset implements FormInterface
{
/** @var array<string, scalar|null> */
Expand Down Expand Up @@ -487,7 +491,7 @@ public function isValid(): bool
* By default, retrieves normalized values; pass one of the
* FormInterface::VALUES_* constants to shape the behavior.
*
* @return array|object
* @inheritDoc
* @throws Exception\DomainException
*/
public function getData(int $flag = FormInterface::VALUES_NORMALIZED)
Expand All @@ -512,6 +516,19 @@ public function getData(int $flag = FormInterface::VALUES_NORMALIZED)
return $filter->getValues();
}

/** @return TFilteredValues */
public function getValidatedPayload(): array
{
if (! $this->hasValidated) {
throw new Exception\DomainException(sprintf(
'%s cannot return data as validation has not yet occurred',
__METHOD__
));
}

return $this->getInputFilter()->getValues();
}

/**
* Set the validation group (set of values to validate)
*
Expand Down Expand Up @@ -582,6 +599,7 @@ protected function prepareValidationGroup(Fieldset $formOrFieldset, array $data,
/**
* Set the input filter used by this form
*
* @param InputFilterInterface<TFilteredValues> $inputFilter
* @return $this
*/
public function setInputFilter(InputFilterInterface $inputFilter)
Expand All @@ -599,6 +617,8 @@ public function setInputFilter(InputFilterInterface $inputFilter)

/**
* Retrieve input filter used by this form
*
* @return InputFilterInterface<TFilteredValues>
*/
public function getInputFilter(): InputFilterInterface
{
Expand Down Expand Up @@ -655,7 +675,7 @@ public function useInputFilterDefaults(): bool
/**
* Set flag indicating whether or not to prefer the form input filter over element and fieldset defaults
*
* @return $this
* @return self<TFilteredValues>
*/
public function setPreferFormInputFilter(bool $preferFormInputFilter)
{
Expand Down
23 changes: 22 additions & 1 deletion src/FormInterface.php
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,9 @@

use Laminas\InputFilter\InputFilterInterface;

/**
* @template TFilteredValues
*/
interface FormInterface extends FieldsetInterface
{
public const BIND_ON_VALIDATE = 0x00;
Expand Down Expand Up @@ -42,12 +45,15 @@ public function setBindOnValidate(int $bindOnValidateFlag);
/**
* Set input filter
*
* @param InputFilterInterface<TFilteredValues> $inputFilter
* @return $this
*/
public function setInputFilter(InputFilterInterface $inputFilter);

/**
* Retrieve input filter
*
* @return InputFilterInterface<TFilteredValues>
*/
public function getInputFilter(): InputFilterInterface;

Expand All @@ -64,10 +70,25 @@ public function isValid(): bool;
* By default, retrieves normalized values; pass one of the VALUES_*
* constants to shape the behavior.
*
* @return array|object
* @param self::VALUES_* $flag
* @return TFilteredValues|object|array<string, mixed>
* @psalm-return (
* $flag is self::VALUES_NORMALIZED
* ? TFilteredValues|object
* : ($flag is self::VALUES_RAW ? array<string, mixed> : TFilteredValues)
* )
*/
public function getData(int $flag = FormInterface::VALUES_NORMALIZED);

/**
* Return the normalized, validated data
*
* This method returns the payload as filtered by the composed Input Filter
*
* @return TFilteredValues
*/
public function getValidatedPayload(): array;

/**
* Set the validation group (set of values to validate)
*
Expand Down
4 changes: 2 additions & 2 deletions src/LabelAwareInterface.php
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ public function getLabel(): ?string;
* Set the attributes to use with the label
*
* @param array<string, scalar|null> $labelAttributes
* @return $this
* @return self
*/
public function setLabelAttributes(array $labelAttributes);

Expand All @@ -38,7 +38,7 @@ public function getLabelAttributes(): array;
*
* Implementation will decide if this will overwrite or merge.
*
* @return $this
* @return self
*/
public function setLabelOptions(iterable $arrayOrTraversable);

Expand Down
1 change: 1 addition & 0 deletions test/Integration/TestAsset/Form.php
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
use Laminas\Form\Form as BaseForm;
use Laminas\Form\FormElementManager;

/** @extends BaseForm<array<string, mixed>> */
final class Form extends BaseForm
{
/** @var null|FormElementManager */
Expand Down
28 changes: 28 additions & 0 deletions test/StaticAnalysis/Asset/ExampleForm.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
<?php

declare(strict_types=1);

namespace LaminasTest\Form\StaticAnalysis\Asset;

use Laminas\Form\Element\Number;
use Laminas\Form\Element\Text;
use Laminas\Form\Form;

/**
* @psalm-import-type ValidPayload from ExampleInputFilter
* @extends Form<ValidPayload>
*/
final class ExampleForm extends Form
{
public function init(): void
{
$this->add([
'name' => 'string',
'type' => Text::class,
]);
$this->add([
'name' => 'number',
'type' => Number::class,
]);
}
}
34 changes: 34 additions & 0 deletions test/StaticAnalysis/Asset/ExampleInputFilter.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
<?php

declare(strict_types=1);

namespace LaminasTest\Form\StaticAnalysis\Asset;

use Laminas\Filter\ToInt;
use Laminas\InputFilter\InputFilter;

/**
* @psalm-type ValidPayload = array{
* string: non-empty-string,
* number: int,
* }
* @extends InputFilter<ValidPayload>
*/
final class ExampleInputFilter extends InputFilter
{
public function init(): void
{
$this->add([
'name' => 'string',
'required' => true,
]);

$this->add([
'name' => 'number',
'required' => true,
'filters' => [
['name' => ToInt::class],
],
]);
}
}
74 changes: 74 additions & 0 deletions test/StaticAnalysis/FormTemplates.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
<?php

declare(strict_types=1);

namespace LaminasTest\Form\StaticAnalysis;

use Laminas\Form\FormInterface;
use LaminasTest\Form\StaticAnalysis\Asset\ExampleForm;

use function assert;
use function is_array;

final class FormTemplates
{
/** @return non-empty-string */
public function getStringTypeFromValidPayload(): string
{
$form = new ExampleForm();

$validatedPayload = $form->getValidatedPayload();

return $validatedPayload['string'];
}

public function getIntTypeFromValidPayload(): int
{
$form = new ExampleForm();

return $form->getValidatedPayload()['number'];
}

/** @return non-empty-string */
public function getSingleValueFromComposedInputFilter(): string
{
$form = new ExampleForm();

return $form->getInputFilter()->getValues()['string'];
}

/** @return non-empty-string */
public function getDataWithExplicitArray(): string
{
$form = new ExampleForm();
$data = $form->getData(FormInterface::VALUES_AS_ARRAY);

return $data['string'];
}

/** @return non-empty-string */
public function getDataWithValuesNormalized(): string
{
$form = new ExampleForm();
$data = $form->getData(FormInterface::VALUES_NORMALIZED);
assert(is_array($data));

return $data['string'];
}

/** @return non-empty-string */
public function testThatFluidReturnTypesPreserveTemplatesForSetPreferFormInputFilter(): string
{
$form = new ExampleForm();
return $form->setPreferFormInputFilter(true)
->getData(FormInterface::VALUES_AS_ARRAY)['string'];
}

/** @return non-empty-string */
public function testThatFluidReturnTypesPreserveTemplatesForSetWrapElements(): string
{
$form = new ExampleForm();
return $form->setWrapElements(true)
->getData(FormInterface::VALUES_AS_ARRAY)['string'];
}
}
1 change: 1 addition & 0 deletions test/TestAsset/Annotation/Form.php
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@

use Laminas\Form\Form as BaseForm;

/** @extends BaseForm<array<string, mixed>> */
class Form extends BaseForm
{
}
18 changes: 16 additions & 2 deletions test/TestAsset/CreateAddressForm.php
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,21 @@
use Laminas\Form\Form;
use Laminas\Hydrator\ClassMethodsHydrator;

/**
* @psalm-type Payload = array{
* street: non-empty-string,
* phones: list<array{id: mixed, number: non-empty-string}>,
* city: array{
* name: non-empty-string,
* zipCode: non-empty-string,
* country: array{
* name: non-empty-string,
* continent: non-empty-string,
* },
* },
* }
* @extends Form<Payload>
*/
class CreateAddressForm extends Form
{
public function __construct()
Expand All @@ -15,8 +30,7 @@ public function __construct()

$this
->setAttribute('method', 'post')
->setHydrator(new ClassMethodsHydrator(false))
->setInputFilter(new InputFilter());
->setHydrator(new ClassMethodsHydrator(false));

$address = new AddressFieldset();
$address->setUseAsBaseFieldset(true);
Expand Down
1 change: 1 addition & 0 deletions test/TestAsset/CustomCreatedForm.php
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
use DateTime;
use Laminas\Form\Form;

/** @extends Form<array<string, mixed>> */
class CustomCreatedForm extends Form
{
public function __construct(private DateTime $created, ?string $name = null, array $options = [])
Expand Down
Loading