feature #37565 [Validator] Add Isin validator constraint (lmasforne)
This PR was merged into the 5.2-dev branch.
Discussion
----------
[Validator] Add Isin validator constraint
Co-Authored-By: Yannis Foucher <33806646+YaFou@users.noreply.github.com>
| Q | A
| ------------- | ---
| Branch? | master
| Bug fix? | no
| New feature? | yes
| Deprecations? | no
| Tickets | Fix #36362
| License | MIT
| Doc PR | symfony/symfony-docs#13960
Rebase of https://github.com/symfony/symfony/pull/36368
I asked him by mail and he didn't have time to finish the PR and allowed me to do it.
Commits
-------
8e1ffc8b99
Feature #36362 add Isin validator constraint
This commit is contained in:
commit
f76ac74b20
@ -29,6 +29,7 @@ CHANGELOG
|
||||
* })
|
||||
*/
|
||||
```
|
||||
* added the `Isin` constraint and validator
|
||||
|
||||
5.1.0
|
||||
-----
|
||||
|
38
src/Symfony/Component/Validator/Constraints/Isin.php
Normal file
38
src/Symfony/Component/Validator/Constraints/Isin.php
Normal file
@ -0,0 +1,38 @@
|
||||
<?php
|
||||
|
||||
/*
|
||||
* This file is part of the Symfony package.
|
||||
*
|
||||
* (c) Fabien Potencier <fabien@symfony.com>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
|
||||
namespace Symfony\Component\Validator\Constraints;
|
||||
|
||||
use Symfony\Component\Validator\Constraint;
|
||||
|
||||
/**
|
||||
* @Annotation
|
||||
* @Target({"PROPERTY", "METHOD", "ANNOTATION"})
|
||||
*
|
||||
* @author Laurent Masforné <l.masforne@gmail.com>
|
||||
*/
|
||||
class Isin extends Constraint
|
||||
{
|
||||
const VALIDATION_LENGTH = 12;
|
||||
const VALIDATION_PATTERN = '/[A-Z]{2}[A-Z0-9]{9}[0-9]{1}/';
|
||||
|
||||
const INVALID_LENGTH_ERROR = '88738dfc-9ed5-ba1e-aebe-402a2a9bf58e';
|
||||
const INVALID_PATTERN_ERROR = '3d08ce0-ded9-a93d-9216-17ac21265b65e';
|
||||
const INVALID_CHECKSUM_ERROR = '32089b-0ee1-93ba-399e-aa232e62f2d29d';
|
||||
|
||||
protected static $errorNames = [
|
||||
self::INVALID_LENGTH_ERROR => 'INVALID_LENGTH_ERROR',
|
||||
self::INVALID_PATTERN_ERROR => 'INVALID_PATTERN_ERROR',
|
||||
self::INVALID_CHECKSUM_ERROR => 'INVALID_CHECKSUM_ERROR',
|
||||
];
|
||||
|
||||
public $message = 'This is not a valid International Securities Identification Number (ISIN).';
|
||||
}
|
@ -0,0 +1,92 @@
|
||||
<?php
|
||||
|
||||
/*
|
||||
* This file is part of the Symfony package.
|
||||
*
|
||||
* (c) Fabien Potencier <fabien@symfony.com>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
|
||||
namespace Symfony\Component\Validator\Constraints;
|
||||
|
||||
use Symfony\Component\Validator\Constraint;
|
||||
use Symfony\Component\Validator\ConstraintValidator;
|
||||
use Symfony\Component\Validator\Exception\UnexpectedTypeException;
|
||||
use Symfony\Component\Validator\Exception\UnexpectedValueException;
|
||||
use Symfony\Component\Validator\Validator\ValidatorInterface;
|
||||
|
||||
/**
|
||||
* @author Laurent Masforné <l.masforne@gmail.com>
|
||||
*
|
||||
* @see https://en.wikipedia.org/wiki/International_Securities_Identification_Number
|
||||
*/
|
||||
class IsinValidator extends ConstraintValidator
|
||||
{
|
||||
/**
|
||||
* @var ValidatorInterface
|
||||
*/
|
||||
private $validator;
|
||||
|
||||
public function __construct(ValidatorInterface $validator)
|
||||
{
|
||||
$this->validator = $validator;
|
||||
}
|
||||
|
||||
/**
|
||||
* {@inheritdoc}
|
||||
*/
|
||||
public function validate($value, Constraint $constraint)
|
||||
{
|
||||
if (!$constraint instanceof Isin) {
|
||||
throw new UnexpectedTypeException($constraint, Isin::class);
|
||||
}
|
||||
|
||||
if (null === $value || '' === $value) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (!is_scalar($value) && !(\is_object($value) && method_exists($value, '__toString'))) {
|
||||
throw new UnexpectedValueException($value, 'string');
|
||||
}
|
||||
|
||||
$value = strtoupper($value);
|
||||
|
||||
if (Isin::VALIDATION_LENGTH !== \strlen($value)) {
|
||||
$this->context->buildViolation($constraint->message)
|
||||
->setParameter('{{ value }}', $this->formatValue($value))
|
||||
->setCode(Isin::INVALID_LENGTH_ERROR)
|
||||
->addViolation();
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
if (!preg_match(Isin::VALIDATION_PATTERN, $value)) {
|
||||
$this->context->buildViolation($constraint->message)
|
||||
->setParameter('{{ value }}', $this->formatValue($value))
|
||||
->setCode(Isin::INVALID_PATTERN_ERROR)
|
||||
->addViolation();
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
if (!$this->isCorrectChecksum($value)) {
|
||||
$this->context->buildViolation($constraint->message)
|
||||
->setParameter('{{ value }}', $this->formatValue($value))
|
||||
->setCode(Isin::INVALID_CHECKSUM_ERROR)
|
||||
->addViolation();
|
||||
}
|
||||
}
|
||||
|
||||
private function isCorrectChecksum(string $input): bool
|
||||
{
|
||||
$characters = str_split($input);
|
||||
foreach ($characters as $i => $char) {
|
||||
$characters[$i] = \intval($char, 36);
|
||||
}
|
||||
$number = implode('', $characters);
|
||||
|
||||
return 0 === $this->validator->validate($number, new Luhn())->count();
|
||||
}
|
||||
}
|
@ -382,6 +382,10 @@
|
||||
<source>Each element of this collection should satisfy its own set of constraints.</source>
|
||||
<target>Each element of this collection should satisfy its own set of constraints.</target>
|
||||
</trans-unit>
|
||||
<trans-unit id="99">
|
||||
<source>This value is not a valid International Securities Identification Number (ISIN).</source>
|
||||
<target>This value is not a valid International Securities Identification Number (ISIN).</target>
|
||||
</trans-unit>
|
||||
</body>
|
||||
</file>
|
||||
</xliff>
|
||||
|
@ -382,6 +382,10 @@
|
||||
<source>Each element of this collection should satisfy its own set of constraints.</source>
|
||||
<target>Chaque élément de cette collection doit satisfaire à son propre jeu de contraintes.</target>
|
||||
</trans-unit>
|
||||
<trans-unit id="99">
|
||||
<source>This value is not a valid International Securities Identification Number (ISIN).</source>
|
||||
<target>Cette valeur n'est pas un code international de sécurité valide (ISIN).</target>
|
||||
</trans-unit>
|
||||
</body>
|
||||
</file>
|
||||
</xliff>
|
||||
|
@ -0,0 +1,135 @@
|
||||
<?php
|
||||
|
||||
namespace Symfony\Component\Validator\Tests\Constraints;
|
||||
|
||||
use Symfony\Component\Validator\Constraints\Isin;
|
||||
use Symfony\Component\Validator\Constraints\IsinValidator;
|
||||
use Symfony\Component\Validator\Test\ConstraintValidatorTestCase;
|
||||
use Symfony\Component\Validator\ValidatorBuilder;
|
||||
|
||||
class IsinValidatorTest extends ConstraintValidatorTestCase
|
||||
{
|
||||
protected function createValidator()
|
||||
{
|
||||
$validatorBuilder = new ValidatorBuilder();
|
||||
|
||||
return new IsinValidator($validatorBuilder->getValidator());
|
||||
}
|
||||
|
||||
public function testNullIsValid()
|
||||
{
|
||||
$this->validator->validate(null, new Isin());
|
||||
|
||||
$this->assertNoViolation();
|
||||
}
|
||||
|
||||
public function testEmptyStringIsValid()
|
||||
{
|
||||
$this->validator->validate('', new Isin());
|
||||
|
||||
$this->assertNoViolation();
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider getValidIsin
|
||||
*/
|
||||
public function testValidIsin($isin)
|
||||
{
|
||||
$this->validator->validate($isin, new Isin());
|
||||
$this->assertNoViolation();
|
||||
}
|
||||
|
||||
public function getValidIsin()
|
||||
{
|
||||
return [
|
||||
['XS2125535901'], // Goldman Sachs International
|
||||
['DE000HZ8VA77'], // UniCredit Bank AG
|
||||
['CH0528261156'], // Leonteq Securities AG [Guernsey]
|
||||
['US0378331005'], // Apple, Inc.
|
||||
['AU0000XVGZA3'], // TREASURY CORP VICTORIA 5 3/4% 2005-2016
|
||||
['GB0002634946'], // BAE Systems
|
||||
['CH0528261099'], // Leonteq Securities AG [Guernsey]
|
||||
['XS2155672814'], // OP Corporate Bank plc
|
||||
['XS2155687259'], // Orbian Financial Services III, LLC
|
||||
['XS2155696672'], // Sheffield Receivables Company LLC
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider getIsinWithInvalidLenghFormat
|
||||
*/
|
||||
public function testIsinWithInvalidFormat($isin)
|
||||
{
|
||||
$this->assertViolationRaised($isin, Isin::INVALID_LENGTH_ERROR);
|
||||
}
|
||||
|
||||
public function getIsinWithInvalidLenghFormat()
|
||||
{
|
||||
return [
|
||||
['X'],
|
||||
['XS'],
|
||||
['XS2'],
|
||||
['XS21'],
|
||||
['XS215'],
|
||||
['XS2155'],
|
||||
['XS21556'],
|
||||
['XS215569'],
|
||||
['XS2155696'],
|
||||
['XS21556966'],
|
||||
['XS215569667'],
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider getIsinWithInvalidPattern
|
||||
*/
|
||||
public function testIsinWithInvalidPattern($isin)
|
||||
{
|
||||
$this->assertViolationRaised($isin, Isin::INVALID_PATTERN_ERROR);
|
||||
}
|
||||
|
||||
public function getIsinWithInvalidPattern()
|
||||
{
|
||||
return [
|
||||
['X12155696679'],
|
||||
['123456789101'],
|
||||
['XS215569667E'],
|
||||
['XS215E69667A'],
|
||||
];
|
||||
}
|
||||
|
||||
/**
|
||||
* @dataProvider getIsinWithValidFormatButIncorrectChecksum
|
||||
*/
|
||||
public function testIsinWithValidFormatButIncorrectChecksum($isin)
|
||||
{
|
||||
$this->assertViolationRaised($isin, Isin::INVALID_CHECKSUM_ERROR);
|
||||
}
|
||||
|
||||
public function getIsinWithValidFormatButIncorrectChecksum()
|
||||
{
|
||||
return [
|
||||
['XS2112212144'],
|
||||
['DE013228VA77'],
|
||||
['CH0512361156'],
|
||||
['XS2125660123'],
|
||||
['XS2012587408'],
|
||||
['XS2012380102'],
|
||||
['XS2012239364'],
|
||||
];
|
||||
}
|
||||
|
||||
private function assertViolationRaised($isin, $code)
|
||||
{
|
||||
$constraint = new Isin([
|
||||
'message' => 'myMessage',
|
||||
]);
|
||||
|
||||
$this->validator->validate($isin, $constraint);
|
||||
|
||||
$this->buildViolation('myMessage')
|
||||
->setParameter('{{ value }}', '"'.$isin.'"')
|
||||
->setCode($code)
|
||||
->assertRaised();
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user