From fe5136faa9a3073936545c6cd2066c9b0c983d00 Mon Sep 17 00:00:00 2001 From: Pol Dellaiera Date: Wed, 17 Jul 2024 11:08:13 +0200 Subject: [PATCH] Refactor spain country handler (#42) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Refactor Spain country handler to support more Spanish TIN * Join PATTERN_2 and PATTERN_3 * Improves readability * Fix PATTERN_2 * Add reference of Spanish TINs * Added more cases for testing * Added reference of Junta de Andalucía * static method getChecksum * Added TOC on documentation * Documentation: remove DNI and NIE link notes * getChacksum as private method * Refactor Spain country handler to support more Spanish TIN * Join PATTERN_2 and PATTERN_3 --------- Co-authored-by: Fernando Herrero --- docs/TIN-Country_Sheet_ES.md | 175 +++++++++++++++++++ spec/loophp/Tin/CountryHandler/SpainSpec.php | 6 +- src/CountryHandler/CountryHandler.php | 2 +- src/CountryHandler/Spain.php | 120 ++++++++----- 4 files changed, 256 insertions(+), 47 deletions(-) create mode 100644 docs/TIN-Country_Sheet_ES.md diff --git a/docs/TIN-Country_Sheet_ES.md b/docs/TIN-Country_Sheet_ES.md new file mode 100644 index 0000000..8e98206 --- /dev/null +++ b/docs/TIN-Country_Sheet_ES.md @@ -0,0 +1,175 @@ +# TAX IDENTIFICATION NUMBERS (TINs) - Country Sheet: Spain (ES) + +**References:** +- [Spain-TIN.pdf on OECD](https://www.oecd.org/tax/automatic-exchange/crs-implementation-and-assistance/tax-identification-numbers/Spain-TIN.pdf) +- [BOE-A-2008-3580](https://www.boe.es/eli/es/o/2008/02/20/eha451/con#a3) +- [Calculation of the DNI/NIE check digit](https://www.interior.gob.es/opencms/es/servicios-al-ciudadano/tramites-y-gestiones/dni/calculo-del-digito-de-control-del-nif-nie/) + +# Table of contents +- [Section I – TIN Description](#section-i--tin-description) +- [Section II – TIN Structure](#section-ii--tin-structure) + - [Natural Persons with DNI¹ or NIE²](#natural-persons-with-dni-or-nie) + - [Natural Persons without DNI¹ or NIE²](#natural-persons-without-dni-or-nie) + - [Non-Natural Persons](#non-natural-persons) +- [Section III – Calculation of the TIN check digit](#section-iii--calculation-of-the-tin-check-digit) + - [Natural Persons with DNI or NIE](#natural-persons-with-dni-or-nie-1) + - [Natural Persons without DNI or NIE and Non-Natural Persons](#natural-persons-without-dni-or-nie-and-non-natural-persons) + +## Section I – TIN Description + +Spain issues TINs, which are reported on official documents of identification. + +TIN in Spain is **unique** for tax and customs purposes and contains **nine characters, the last of them is a letter for control (Natural persons) or a control character (Non - natural persons)**. + +**Natural persons of Spanish nationality:** Generally, the TIN is the number on your National Identity Card, issued by the Ministry of Internal Affairs (General Directorate of Police). The Tax Administration will provide Spanish natural persons who are not obliged to possess a National Identity Card (DNI) with a Tax Identification Number (TIN) starting with an L (non-resident Spaniards) or a K (resident Spaniards under the age of 14 years), upon request. + +**Natural persons without Spanish nationality:** Generally, their Tax Identification Number (TIN) is the Foreigners’ Identification Number (NIE), likewise issued by the Ministry of Internal Affairs. Natural persons without Spanish nationality who do not possess a Foreigners’ Identification Number (NIE) but need a Tax Identification Number (TIN) because they are going to engage in transactions involving Spanish taxation can obtain a Tax Identification Number starting with the letter M, that will have a +transitory nature, until they obtain a Foreigners’ Identification Number (NIE), where appropriate, also issued by the Tax Administration. + +Concerning the **entities**, they are obliged to obtain a TIN, which is issued by the Tax Administration + +[TOC](#table-of-contents) + +## Section II – TIN Structure + +### Natural Persons with DNI¹ or NIE² +1. **DNI** = Documento Nacional de Identidad (National Identity Card) +2. **NIE** = Número de Identificación de Extranjero (Foreigners’ Identification Number) + +| Format | Explanation | Comments | +|---------- |----------------------------------------|------------------------------| +| 99999999C | 8 digits + 1 Control letter | Spanish Natural Persons: DNI | +| X9999999C | Letter X + 7 digits + 1 Control letter | Foreigners with NIE | +| Y9999999C | Letter Y + 7 digits + 1 Control letter | Foreigners with NIE | +| Z9999999C | Letter Z + 7 digits + 1 Control letter | Foreigners with NIE | + +[TOC](#table-of-contents) + +### Natural Persons without DNI¹ or NIE² +1. **DNI** = Documento Nacional de Identidad (National Identity Card) +2. **NIE** = Número de Identificación de Extranjero (Foreigners’ Identification Number) + +| Format | Explanation | Comments | +|---------- |----------------------------------------|-----------------------------------------| +| K9999999C | Letter K + 7 digits + 1 Control letter | Resident Spaniards under 14 without DNI | +| L9999999C | Letter L + 7 digits + 1 Control letter | Non-resident Spaniards without DNI | +| M9999999C | Letter M + 7 digits + 1 Control letter | Foreigners without NIE | + +[TOC](#table-of-contents) + +### Non-Natural Persons + +| Format | Explanation | Comments | +|-----------|-------------|----------| +| L9999999C | Initial Letter + 7 digits + 1 Control character | The first letter reports on legal form ([BOE-A-2008-3580](https://www.boe.es/eli/es/o/2008/02/20/eha451/con#a3)) | + +For entities, the tax identification number will begin with a letter, which will include information about its legal form according to the following keys: + +| Letter | Spanish | English | +|---|-|-| +| A | Sociedades anónimas | Public limited companies | +| B | Sociedades de responsabilidad limitada | Limited liability companies | +| C | Sociedades colectivas | Collective societies | +| D | Sociedades comanditarias | Limited partnerships | +| E | Comunidades de bienes, herencias yacentes y demás entidades carentes de personalidad jurídica no incluidas expresamente en otras claves | Communities of property, existing inheritances and other entities lacking legal personality not expressly included in other keys | +| F | Sociedades cooperativas | Cooperative societies | +| G | Asociaciones | Associations | +| H | Comunidades de propietarios en régimen de propiedad horizontal | Communities of owners under horizontal property regime | +| J | Sociedades civiles | Civil societies | +| P | Corporaciones Locales | Local Corporations | +| Q | Organismos públicos | Public organizations | +| R | Congregaciones e instituciones religiosas | Congregations and religious institutions | +| S | Órganos de la Administración del Estado y de las Comunidades Autónomas | Bodies of the Administration of the State and the Autonomous Communities | +| U | Uniones Temporales de Empresas | Temporary Business Unions | +| V | Otros tipos no definidos en el resto de claves | Other types not defined in the rest of the keys | +| N | Entidad extranjera | Foreign entity | +| W | Establecimiento permanente de entidad no residente en territorio español | Permanent establishment of a non-resident in Spain | + +[TOC](#table-of-contents) + +## Section III – Calculation of the TIN check digit + +### Natural Persons with DNI or NIE +[Reference](https://www.interior.gob.es/opencms/es/servicios-al-ciudadano/tramites-y-gestiones/dni/calculo-del-digito-de-control-del-nif-nie/) + +**Cases:** + +| Format | Explanation | Type | Standardization | +|---------- |----------------------------------------|------|-----------------| +| 99999999C | 8 digits + 1 Control letter | DNI | 99999999 + C | +| X9999999C | Letter X + 7 digits + 1 Control letter | NIE | 09999999 + C | +| Y9999999C | Letter Y + 7 digits + 1 Control letter | NIE | 19999999 + C | +| Z9999999C | Letter Z + 7 digits + 1 Control letter | NIE | 29999999 + C | + +**For NIEs the first letter is replaced as follow:** +``` +X => 0 +Y => 1 +Z => 2 +``` + +**Examples of standardization of NIEs:** +``` +X1234567L => 01234567L +Y1234567X => 11234567X +Z1234567R => 21234567R +``` + +The number is divided by 23 and the remainder is replaced by a letter that is determined by inspection using the following table: + +| Remainder | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | +|-----------|:-:|:-:|:-:|:-:|:-:|:-:|:-:|:-:|:-:|:-:|:--:|:--:| +| **Letter**| T | R | W | A | G | M | Y | F | P | D | X | B | + +| Remainder | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | 22 | +|-----------|:--:|:--:|:--:|:--:|:--:|:--:|:--:|:--:|:--:|:--:|:--:| +| **Letter**| N | J | Z | S | Q | V | H | L | C | K | E | + +**Example of calculus:** +1. TIN = **X1234567L** +2. Standardization of NIE: TIN = **01234567L** +3. TinNumber = **01234567**, TinChecksum = **L** +4. TinNumber MODULUS 23 = **01234567 % 23** = **19** +5. **19** is the **L** letter + +[TOC](#table-of-contents) + +### Natural Persons without DNI or NIE and Non-Natural Persons +[Reference](https://www.juntadeandalucia.es/servicios/madeja/sites/default/files/historico/1.4.0/contenido-libro-pautas-196.html#Validacion_de_NIF_con_tipo_distinto_a_DNI) + +**Cases:** +| Format | Explanation | +|---------- |-------------------------------------------| +| K9999999C | Letter K + 7 digits + 1 Control character | +| L9999999C | Letter L + 7 digits + 1 Control character | +| M9999999C | Letter M + 7 digits + 1 Control character | +| L9999999C | 1 Letter + 7 digits + 1 Control character | + +**Method:** +In the case of NIF that are not obtained from the DNI or NIE, the control code is obtained using the 7-digit number, excluding the initial letter and the final letter or digit. +1. The even positions of the 7 central digits are added, that is, the initial letter or the control code are not taken into account. (Sum = A) +2. For each of the digits in the odd positions, the digit is multiplied by 2 and the figures in the result are added, but if the result has a single digit, this figure is simply added. (e.g. if the digit is 6, the result would be 6 x 2 = 12 -> 1 + 2 = 3, but if the digit is 2, the result would be 2 x 2 = 4). (Sum = B) +3. Add the result of the 2 previous steps. (A + B = C) +4. We subtract the last digit of the previous sum (C) from 10, the result of which would be the control code (e.g. if C = 14, the last digit is 4, so we would have 10 - 4 = 6). If the last digit of the sum from the previous step is 0 (e.g. C = 30), no subtraction is performed and 0 is taken as the control code. + +If the control code is a number, this would be the result of the last operation. If it is a letter, the following relationship would be used: + +| Result | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 0 | +|---------|---|---|---|---|---|---|---|---|---|---| +| Control | A | B | C | D | E | F | G | H | I | J | + +**Example of calculus:** +TIN = **M2812345C** => Digits: **2812345** +1. Even positions digits (2**8**1**2**3**4**5): 8 + 2 + 4 = **14** +2. Odd positions digits (**2**8**1**2**3**4**5**): + **2** * 2 = **4** + **1** * 2 = **2** + **3** * 2 = **6** + **5** * 2 = 10 => 1 + 0 = **1** + **4** + **2** + **6** + **1** = **13** +3. 14 + 13 = 2**7** +4. 10 - **7** = **3** + +Result = 3 => Letter **C** + +[TOC](#table-of-contents) \ No newline at end of file diff --git a/spec/loophp/Tin/CountryHandler/SpainSpec.php b/spec/loophp/Tin/CountryHandler/SpainSpec.php index bdf28ca..a715835 100644 --- a/spec/loophp/Tin/CountryHandler/SpainSpec.php +++ b/spec/loophp/Tin/CountryHandler/SpainSpec.php @@ -13,11 +13,11 @@ class SpainSpec extends AbstractAlgorithmSpec { - public const INVALID_NUMBER_CHECK = 'X1234567Z'; + public const INVALID_NUMBER_CHECK = ['X1234567Z', 'P2009300B', 'K0867756J']; public const INVALID_NUMBER_LENGTH = '542372254545445A'; - public const INVALID_NUMBER_PATTERN = 'wwwwwwwww'; + public const INVALID_NUMBER_PATTERN = ['wwwwwwwww', 'K0867756N']; - public const VALID_NUMBER = ['54237A', 'X1234567L', 'Z1234567R', 'M2812345C']; + public const VALID_NUMBER = ['54237A', 'X1234567L', 'Y1234567X', 'Z1234567R', 'M2812345C', 'B05327986', 'P2009300A', 'K0867756I']; } diff --git a/src/CountryHandler/CountryHandler.php b/src/CountryHandler/CountryHandler.php index affd210..f46e188 100644 --- a/src/CountryHandler/CountryHandler.php +++ b/src/CountryHandler/CountryHandler.php @@ -146,7 +146,7 @@ protected function matchLength(string $tin, int $length): bool protected function matchPattern(string $subject, string $pattern): bool { - return 1 === preg_match(sprintf('/%s/', $pattern), $subject); + return 1 === preg_match(sprintf('/%s/i', $pattern), $subject); } protected function normalizeTin(string $tin): string diff --git a/src/CountryHandler/Spain.php b/src/CountryHandler/Spain.php index ad67acc..69b92b2 100644 --- a/src/CountryHandler/Spain.php +++ b/src/CountryHandler/Spain.php @@ -9,8 +9,6 @@ namespace loophp\Tin\CountryHandler; -use function strlen; - use const STR_PAD_LEFT; /** @@ -18,6 +16,21 @@ */ final class Spain extends CountryHandler { + /** + * @var string + */ + public const CHECKSUM_LETTER = 'KLMNPQRSW'; + + /** + * @var string + */ + public const CONTROL_1 = 'TRWAGMYFPDXBNJZSQVHLCKE'; + + /** + * @var string + */ + public const CONTROL_2 = 'JABCDEFGHI'; + /** * @var string */ @@ -29,25 +42,27 @@ final class Spain extends CountryHandler public const LENGTH = 9; /** - * @var string + * @var array */ - public const PATTERN_1 = '\\d{8}[a-zA-Z]'; + public const NIE = ['X', 'Y', 'Z']; /** + * Spanish Natural Persons: DNI + * Foreigners with NIE. + * * @var string */ - public const PATTERN_2 = '[XYZKLMxyzklm]\\d{7}[a-zA-Z]'; + public const PATTERN_1 = '(^[XYZ\d]\d{7})([' . self::CONTROL_1 . ']$)'; /** - * @var array + * Non-resident Spaniards without DNI + * Resident Spaniards under 14 without DNI + * Foreigners without NIE + * Legal entities (companies, organizations, public entities, ...). + * + * @var string */ - private static $tabConvertToChar = [ - 'T', 'R', 'W', 'A', 'G', - 'M', 'Y', 'F', 'P', 'D', - 'X', 'B', 'N', 'J', 'Z', - 'S', 'Q', 'V', 'H', 'L', - 'C', 'K', 'E', - ]; + public const PATTERN_2 = '(^[ABCDEFGHJKLMNPQRSUVW])(\d{7})([' . self::CONTROL_2 . '\d]$)'; public function getTIN(): string { @@ -61,33 +76,48 @@ protected function hasValidPattern(string $tin): bool protected function hasValidRule(string $tin): bool { - return ($this->isFollowPattern1($tin) && $this->isFollowRule1($tin)) - || ($this->isFollowPattern2($tin) && $this->isFollowRule2($tin)); + return $this->isFollowRule1($tin) || $this->isFollowRule2($tin); } - private function getCharFromNumber(int $sum): string + /** + * Return checksum char for Spanish TIN. + * + * @param string $tin + * The TIN without Country indicative ('ES') + * @param null|bool $digit + * Optional: for Non-Natural Persons TIN forces return checksum char as digit 0-9 + * + * @return null|string + * Return checksum char or null on failure + */ + private function getChecksum(string $tin, ?bool $digit = null): ? string { - return self::$tabConvertToChar[$sum - 1]; - } + // Natural Persons with DNI or NIE + if (1 === preg_match('~' . self::PATTERN_1 . '?~', strtoupper($tin), $tinParts)) { + $tinNumber = (int) str_replace(self::NIE, array_keys(self::NIE), $tinParts[1]); - private function getNumberFromChar(string $m): int - { - switch ($m) { - case 'K': - case 'L': - case 'M': - case 'X': - return 0; + return substr(self::CONTROL_1, $tinNumber % 23, 1); + } + + // Natural Persons without DNI or NIE and Non-Natural Persons + if (1 === preg_match('~' . self::PATTERN_2 . '?~', strtoupper($tin), $tinParts)) { + $checksum = 0; - case 'Y': - return 1; + foreach (str_split($tinParts[2]) as $pos => $val) { + $checksum += array_sum(str_split((string) ((int) $val * (2 - ($pos % 2))))); + } - case 'Z': - return 2; + $checksum1 = (string) ((10 - ($checksum % 10)) % 10); + $checksum2 = substr(self::CONTROL_2, (int) $checksum1, 1); - default: - return -1; + if (null === $digit) { + $digit = false === strpos(self::CHECKSUM_LETTER, $tinParts[1]); + } + + return $digit ? $checksum1 : $checksum2; } + + return null; } private function isFollowPattern1(string $tin): bool @@ -102,22 +132,26 @@ private function isFollowPattern2(string $tin): bool private function isFollowRule1(string $tin): bool { - $number = (int) (substr($tin, 0, strlen($tin) - 1)); - $checkDigit = $tin[strlen($tin) - 1]; - $remainderBy23 = $number % 23; - $sum = $remainderBy23 + 1; + if (1 !== preg_match('~' . self::PATTERN_1 . '~', strtoupper($tin), $tinParts)) { + return false; + } + + [, $tinNumber, $tinChecksum] = $tinParts; - return $this->getCharFromNumber($sum) === $checkDigit; + return $this->getChecksum($tinNumber) === $tinChecksum; } private function isFollowRule2(string $tin): bool { - $c1 = (string) $this->getNumberFromChar($tin[0]); - $number = (int) ($c1 . substr($tin, 1, strlen($tin))); - $checkDigit = $tin[strlen($tin) - 1]; - $remainderBy23 = $number % 23; - $sum = $remainderBy23 + 1; + if (1 !== preg_match('~' . self::PATTERN_2 . '~', strtoupper($tin), $tinParts)) { + return false; + } + + [,$tinFirstLetter , $tinNumber, $tinChecksum] = $tinParts; + + $tinNumber = $tinFirstLetter . $tinNumber; + $digit = (false === strpos(self::CONTROL_2, $tinChecksum)); - return $this->getCharFromNumber($sum) === $checkDigit; + return $this->getChecksum($tinNumber, $digit) === $tinChecksum; } }