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:
Fabien Potencier 2020-08-02 10:06:13 +02:00
commit f76ac74b20
6 changed files with 274 additions and 0 deletions

View File

@ -29,6 +29,7 @@ CHANGELOG
* })
*/
```
* added the `Isin` constraint and validator
5.1.0
-----

View 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).';
}

View File

@ -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();
}
}

View File

@ -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>

View File

@ -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>

View File

@ -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();
}
}