From 931f1779de95bb27ca6259c7ec91263bfd0ca04f Mon Sep 17 00:00:00 2001 From: John Was Date: Thu, 19 Sep 2013 20:05:32 +0200 Subject: [PATCH] generating valid IBAN for every currently known locale that uses it; added an util to extract IBAN format from SWIFT site data --- src/Faker/Provider/Company.php | 140 ++++++++++++++++++++++ src/Faker/Provider/bg_BG/Company.php | 19 +++ src/Faker/Provider/cs_CZ/Company.php | 19 +++ src/Faker/Provider/da_DK/Company.php | 13 ++ src/Faker/Provider/de_AT/Company.php | 19 +++ src/Faker/Provider/de_DE/Company.php | 13 ++ src/Faker/Provider/el_GR/Company.php | 19 +++ src/Faker/Provider/en_GB/Company.php | 19 +++ src/Faker/Provider/es_ES/Company.php | 13 ++ src/Faker/Provider/fr_BE/Company.php | 13 ++ src/Faker/Provider/fr_FR/Company.php | 13 ++ src/Faker/Provider/is_IS/Company.php | 13 ++ src/Faker/Provider/it_IT/Company.php | 13 ++ src/Faker/Provider/lv_LV/Company.php | 19 +++ src/Faker/Provider/nl_BE/Company.php | 13 ++ src/Faker/Provider/nl_NL/Company.php | 13 ++ src/Faker/Provider/pl_PL/Company.php | 113 +++++++++++++++++ src/Faker/Provider/pt_BR/Company.php | 13 ++ src/Faker/Provider/sk_SK/Company.php | 13 ++ src/Faker/Provider/sr_Cyrl_RS/Company.php | 19 +++ src/Faker/Provider/sr_Latn_RS/Company.php | 19 +++ src/Faker/Provider/sr_RS/Company.php | 19 +++ src/Faker/Provider/tr_TR/Company.php | 19 +++ utils/IBAN_Registry.txt | 63 ++++++++++ utils/ibanext.php | 76 ++++++++++++ 25 files changed, 725 insertions(+) create mode 100644 src/Faker/Provider/bg_BG/Company.php create mode 100644 src/Faker/Provider/cs_CZ/Company.php create mode 100644 src/Faker/Provider/de_AT/Company.php create mode 100644 src/Faker/Provider/el_GR/Company.php create mode 100644 src/Faker/Provider/en_GB/Company.php create mode 100644 src/Faker/Provider/lv_LV/Company.php create mode 100644 src/Faker/Provider/sr_Cyrl_RS/Company.php create mode 100644 src/Faker/Provider/sr_Latn_RS/Company.php create mode 100644 src/Faker/Provider/sr_RS/Company.php create mode 100644 src/Faker/Provider/tr_TR/Company.php create mode 100644 utils/IBAN_Registry.txt create mode 100644 utils/ibanext.php diff --git a/src/Faker/Provider/Company.php b/src/Faker/Provider/Company.php index 208d9ce5..970b689a 100644 --- a/src/Faker/Provider/Company.php +++ b/src/Faker/Provider/Company.php @@ -10,6 +10,73 @@ class Company extends \Faker\Provider\Base protected static $companySuffix = array('Ltd'); + /** + * @var array list of IBAN formats, source: @link http://www.swift.com/dsp/resources/documents/IBAN_Registry.txt + */ + protected static $ibanFormats = array( + 'AD' => array(array('n', 4), array('n', 4), array('c', 12)), + 'AE' => array(array('n', 3), array('n', 16)), + 'AL' => array(array('n', 8), array('c', 16)), + 'AT' => array(array('n', 5), array('n', 11)), + 'AZ' => array(array('a', 4), array('c', 20)), + 'BA' => array(array('n', 3), array('n', 3), array('n', 8), array('n', 2)), + 'BE' => array(array('n', 3), array('n', 7), array('n', 2)), + 'BG' => array(array('a', 4), array('n', 4), array('n', 2), array('c', 8)), + 'BH' => array(array('a', 4), array('c', 14)), + 'BR' => array(array('n', 8), array('n', 5), array('n', 10), array('a', 1), array('c', 1)), + 'CH' => array(array('n', 5), array('c', 12)), + 'CR' => array(array('n', 3), array('n', 14)), + 'CY' => array(array('n', 3), array('n', 5), array('c', 16)), + 'CZ' => array(array('n', 4), array('n', 6), array('n', 10)), + 'DE' => array(array('n', 8), array('n', 10)), + 'DK' => array(array('n', 4), array('n', 9), array('n', 1)), + 'DO' => array(array('c', 4), array('n', 20)), + 'EE' => array(array('n', 2), array('n', 2), array('n', 11), array('n', 1)), + 'ES' => array(array('n', 4), array('n', 4), array('n', 1), array('n', 1), array('n', 10)), + 'FR' => array(array('n', 5), array('n', 5), array('c', 11), array('n', 2)), + 'GB' => array(array('a', 4), array('n', 6), array('n', 8)), + 'GE' => array(array('a', 2), array('n', 16)), + 'GI' => array(array('a', 4), array('c', 15)), + 'GR' => array(array('n', 3), array('n', 4), array('c', 16)), + 'GT' => array(array('c', 4), array('c', 20)), + 'HR' => array(array('n', 7), array('n', 10)), + 'HU' => array(array('n', 3), array('n', 4), array('n', 1), array('n', 15), array('n', 1)), + 'IE' => array(array('a', 4), array('n', 6), array('n', 8)), + 'IL' => array(array('n', 3), array('n', 3), array('n', 13)), + 'IS' => array(array('n', 4), array('n', 2), array('n', 6), array('n', 10)), + 'IT' => array(array('a', 1), array('n', 5), array('n', 5), array('c', 12)), + 'KW' => array(array('a', 4), array('c', 22)), + 'KZ' => array(array('n', 3), array('c', 13)), + 'LB' => array(array('n', 4), array('c', 20)), + 'LI' => array(array('n', 5), array('c', 12)), + 'LT' => array(array('n', 5), array('n', 11)), + 'LU' => array(array('n', 3), array('c', 13)), + 'LV' => array(array('a', 4), array('c', 13)), + 'MC' => array(array('n', 5), array('n', 5), array('c', 11), array('n', 2)), + 'MD' => array(array('c', 2), array('c', 18)), + 'ME' => array(array('n', 3), array('n', 13), array('n', 2)), + 'MK' => array(array('n', 3), array('c', 10), array('n', 2)), + 'MR' => array(array('n', 5), array('n', 5), array('n', 11), array('n', 2)), + 'MT' => array(array('a', 4), array('n', 5), array('c', 18)), + 'MU' => array(array('a', 4), array('n', 2), array('n', 2), array('n', 12), array('n', 3), array('a', 3)), + 'NL' => array(array('a', 4), array('n', 10)), + 'NO' => array(array('n', 4), array('n', 6), array('n', 1)), + 'PK' => array(array('a', 4), array('c', 16)), + 'PL' => array(array('n', 8), array('n', 16)), + 'PS' => array(array('a', 4), array('c', 21)), + 'PT' => array(array('n', 4), array('n', 4), array('n', 11), array('n', 2)), + 'RO' => array(array('a', 4), array('c', 16)), + 'RS' => array(array('n', 3), array('n', 13), array('n', 2)), + 'SA' => array(array('n', 2), array('c', 18)), + 'SE' => array(array('n', 3), array('n', 16), array('n', 1)), + 'SI' => array(array('n', 5), array('n', 8), array('n', 2)), + 'SK' => array(array('n', 4), array('n', 6), array('n', 10)), + 'SM' => array(array('a', 1), array('n', 5), array('n', 5), array('c', 12)), + 'TN' => array(array('n', 2), array('n', 3), array('n', 13), array('n', 2)), + 'TR' => array(array('n', 5), array('c', 1), array('c', 16)), + 'VG' => array(array('a', 4), array('n', 16)), + ); + /** * @example 'Acme Ltd' */ @@ -27,4 +94,77 @@ class Company extends \Faker\Provider\Base { return static::randomElement(static::$companySuffix); } + + /** + * International Bank Account Number (IBAN) + * @link http://en.wikipedia.org/wiki/International_Bank_Account_Number + * @param string $countryCode ISO 3166-1 alpha-2 country code + * @param string $prefix for generating bank account number of a specific bank + * @param integer $length total length without country code and 2 check digits + * @return string + */ + protected static function iban($countryCode, $prefix = '', $length = null) + { + $countryCode = strtoupper($countryCode); + $format = !isset(static::$ibanFormats[$countryCode]) ? array() : static::$ibanFormats[$countryCode]; + if ($length === null) { + if ($format === null) { + $length = 24; + } else { + $length = 0; + foreach($format as $part) { + list($class, $groupCount) = $part; + $length += $groupCount; + } + } + } + + $result = $prefix; + $length -= strlen($prefix); + $nextPart = array_shift($format); + if ($nextPart !== false) { + list($class, $groupCount) = $nextPart; + } else { + $class = 'n'; + $groupCount = 0; + } + $groupCount = $nextPart === false ? 0 : $nextPart[1]; + for ($i = 0; $i < $length; $i++) { + if ($nextPart !== false && $groupCount-- < 1) { + $nextPart = array_shift($format); + list($class, $groupCount) = $nextPart; + } + switch($class) { + default: + case 'c': $result .= mt_rand(0, 100) <= 50 ? static::randomDigit() : strtoupper(static::randomLetter()); break; + case 'a': $result .= strtoupper(static::randomLetter()); break; + case 'n': $result .= static::randomDigit(); break; + } + } + + $result = static::addBankCodeChecksum($result, $countryCode); + + $countryNumber = 100 * (ord($countryCode[0])-55) + (ord($countryCode[1])-55); + $tempResult = $result . $countryNumber . '00'; + // perform MOD97-10 checksum calculation + $checksum = (int)$tempResult[0]; + for ($i = 1; $i < strlen($tempResult); $i++) { + $checksum = (10 * $checksum + (int)$tempResult[$i]) % 97; + } + $checksum = 98 - $checksum; + if ($checksum < 10) + $checksum = '0'.$checksum; + return $countryCode . $checksum . $result; + } + + /** + * Calculates a checksum for the national bank and branch code part in the IBAN. + * @param string $iban randomly generated $iban + * @param string $countryCode ISO 3166-1 alpha-2 country code + * @return string IBAN with one character altered to a proper checksum + */ + protected static function addBankCodeChecksum($iban, $countryCode = '') + { + return $iban; + } } diff --git a/src/Faker/Provider/bg_BG/Company.php b/src/Faker/Provider/bg_BG/Company.php new file mode 100644 index 00000000..4184ac73 --- /dev/null +++ b/src/Faker/Provider/bg_BG/Company.php @@ -0,0 +1,19 @@ + 'Aareal Bank Aktiengesellschaft (Spółka Akcyjna) - Oddział w Polsce', + '249' => 'Alior Bank SA', + '247' => 'Banco Espirito Santo de Investimento, S.A. Spółka Akcyjna Oddział w Polsce', + '238' => 'Banco Mais S.A. (SA) Oddział w Polsce', + '106' => 'Bank BPH SA', + '219' => 'Bank DnB NORD Polska SA', + '203' => 'Bank Gospodarki Żywnościowej SA', + '113' => 'Bank Gospodarstwa Krajowego', + '122' => 'Bank Handlowo - Kredytowy SA (w likwidacji 31.03.92)', + '103' => 'Bank Handlowy w Warszawie SA', + '116' => 'Bank Millennium SA', + '154' => 'Bank Ochrony Środowiska SA', + '260' => 'Bank of China (Luxembourg)S.A. Spółka Akcyjna Oddział w Polsce', + '221' => 'Bank of Tokyo-Mitsubishi UFJ (Polska) SA', + '132' => 'Bank Pocztowy SA', + '124' => 'Bank Polska Kasa Opieki SA', + '193' => 'BANK POLSKIEJ SPÓŁDZIELCZOŚCI SA', + '109' => 'Bank Zachodni WBK SA', + '224' => 'Banque PSA Finance SA Oddział w Polsce', + '160' => 'BNP PARIBAS BANK POLSKA SA', + '235' => 'BNP PARIBAS SA Oddział w Polsce', + '243' => 'BNP Paribas Securities Services SKAOddział w Polsce', + '229' => 'BPI Bank Polskich Inwestycji SA', + '215' => 'BRE Bank Hipoteczny SA', + '114' => 'BRE Bank SA', + '239' => 'CAIXABANK, S.A. (SPÓŁKA AKCYJNA)ODDZIAŁ W POLSCE', + '254' => 'Citibank Europe plc (Publiczna Spółka Akcyjna) Oddział w Polsce', + '194' => 'Credit Agricole Bank Polska SA', + '252' => 'CREDIT SUISSE (LUXEMBOURG) S.A. Spółka Akcyjna, Oddział w Polsce', + '236' => 'Danske Bank A/S SA Oddział w Polsce', + '191' => 'Deutsche Bank PBC SA', + '188' => 'Deutsche Bank Polska SA', + '174' => 'DZ BANK Polska SA', + '241' => 'Elavon Financial Services Limited (Spółka z ograniczoną odpowiedzialnością) Oddział w Polsce', + '147' => 'Euro Bank SA', + '265' => 'EUROCLEAR Bank SA/NV (Spółka Akcyjna) - Oddział w Polsce', + '207' => 'FCE Bank Polska SA', + '214' => 'Fiat Bank Polska SA', + '253' => 'FM Bank SA', + '248' => 'Getin Noble Bank SA', + '128' => 'HSBC Bank Polska SA', + '195' => 'Idea Bank SA', + '255' => 'Ikano Bank GmbH (Sp. z o.o.) Oddział w Polsce', + '262' => 'Industrial and Commercial Bank of China (Europe) S.A. (Spółka Akcyjna) Oddział w Polsce', + '105' => 'ING Bank Śląski SA', + '266' => 'Intesa Sanpaolo S.p.A. Spółka Akcyjna Oddział w Polsce', + '168' => 'INVEST - BANK SA', + '258' => 'J.P. Morgan Europe Limited Sp. z o.o. Oddział w Polsce', + '158' => 'Mercedes-Benz Bank Polska SA', + '130' => 'Meritum Bank ICB SA', + '101' => 'Narodowy Bank Polski', + '256' => 'Nordea Bank AB SA Oddział w Polsce', + '144' => 'NORDEA BANK POLSKA SA', + '232' => 'Nykredit Realkredit A/S SA - Oddział w Polsce', + '189' => 'Pekao Bank Hipoteczny SA', + '187' => 'Polski Bank Przedsiębiorczości SA', + '102' => 'Powszechna Kasa Oszczędności Bank Polski SA', + '200' => 'Rabobank Polska SA', + '175' => 'Raiffeisen Bank Polska SA', + '167' => 'RBS Bank (Polska) SA', + '264' => 'RCI Banque Spółka Akcyjna Oddział w Polsce', + '212' => 'Santander Consumer Bank SA', + '263' => 'Saxo Bank A/S Spółka Akcyjna Oddział w Polsce', + '161' => 'SGB-Bank SA', + '237' => 'Skandinaviska Enskilda Banken AB (SA) - Oddział w Polsce', + '184' => 'Societe Generale SA Oddział w Polsce', + '225' => 'Svenska Handelsbanken AB SA Oddział w Polsce', + '227' => 'Sygma Banque Societe Anonyme (SA) Oddział w Polsce', + '216' => 'Toyota Bank Polska SA', + '257' => 'UBS Limited (spółka z ograniczoną odpowiedzialnością) Oddział w Polsce', + '261' => 'Vanquis Bank Limited (spółka z ograniczoną odpowiedzialnością) Oddział w Polsce', + '213' => 'VOLKSWAGEN BANK POLSKA SA', + ); + /* * Register of the National Economy * @link http://pl.wikipedia.org/wiki/REGON @@ -54,4 +132,39 @@ class Company extends \Faker\Provider\Company return implode('', $result); } + + /** + * @example 'Euro Bank SA' + */ + public static function bank() + { + return static::randomElement(static::$banks); + } + + /** + * International Bank Account Number (IBAN) + * @link http://en.wikipedia.org/wiki/International_Bank_Account_Number + * @param string $prefix for generating bank account number of a specific bank + * @param string $countryCode ISO 3166-1 alpha-2 country code + * @param integer $length total length without country code and 2 check digits + * @return string + */ + public static function bankAccountNumber($prefix = '', $countryCode = 'PL', $length = null) + { + return static::iban($countryCode, $prefix, $length); + } + + protected static function addBankCodeChecksum($iban, $countryCode = 'PL') + { + if ($countryCode != "PL" || strlen($iban) <= 8) { + return $iban; + } + $checksum = 0; + $weights = array(7, 1, 3, 9, 7, 1, 3); + for ($i = 0; $i < 7; $i++) { + $checksum += $weights[$i] * (int)$iban[$i]; + } + $checksum = $checksum % 10; + return substr($iban, 0, 7) . $checksum . substr($iban, 8); + } } diff --git a/src/Faker/Provider/pt_BR/Company.php b/src/Faker/Provider/pt_BR/Company.php index 2b9a98d1..d4168137 100644 --- a/src/Faker/Provider/pt_BR/Company.php +++ b/src/Faker/Provider/pt_BR/Company.php @@ -63,4 +63,17 @@ class Company extends \Faker\Provider\Company return join($result, ' '); } + + /** + * International Bank Account Number (IBAN) + * @link http://en.wikipedia.org/wiki/International_Bank_Account_Number + * @param string $prefix for generating bank account number of a specific bank + * @param string $countryCode ISO 3166-1 alpha-2 country code + * @param integer $length total length without country code and 2 check digits + * @return string + */ + public static function bankAccountNumber($prefix = '', $countryCode = 'BR', $length = null) + { + return static::iban($countryCode, $prefix, $length); + } } diff --git a/src/Faker/Provider/sk_SK/Company.php b/src/Faker/Provider/sk_SK/Company.php index 1f5fad2e..56bb2c7e 100644 --- a/src/Faker/Provider/sk_SK/Company.php +++ b/src/Faker/Provider/sk_SK/Company.php @@ -61,4 +61,17 @@ class Company extends \Faker\Provider\Company return join($result, ' '); } + + /** + * International Bank Account Number (IBAN) + * @link http://en.wikipedia.org/wiki/International_Bank_Account_Number + * @param string $prefix for generating bank account number of a specific bank + * @param string $countryCode ISO 3166-1 alpha-2 country code + * @param integer $length total length without country code and 2 check digits + * @return string + */ + public static function bankAccountNumber($prefix = '', $countryCode = 'SK', $length = null) + { + return static::iban($countryCode, $prefix, $length); + } } diff --git a/src/Faker/Provider/sr_Cyrl_RS/Company.php b/src/Faker/Provider/sr_Cyrl_RS/Company.php new file mode 100644 index 00000000..a727fa50 --- /dev/null +++ b/src/Faker/Provider/sr_Cyrl_RS/Company.php @@ -0,0 +1,19 @@ + 'Name of country', + 1 => 'Country code as defined in ISO 3166', + 2 => 'Domestic account number example', + 3 => 'BBAN', + 4 => 'BBAN structure ', + 5 => 'BBAN length', + 6 => 'Bank identifier position within the BBAN', + 7 => 'Bank identifier length', + 8 => 'Bank identifier example', + 9 => 'BBAN example', + 10 => 'IBAN', + 11 => 'IBAN structure', + 12 => 'IBAN length', + 13 => 'IBAN electronic format example', + 14 => 'IBAN print format example', + 15 => 'SEPA Country', + 16 => 'Contact details', +) +$example = array ( + 0 => 'Albania', + 1 => 'AL', + 2 => '0000000235698741', + 3 => '0', + 4 => '8!n16!c', + 5 => '24', + 6 => ' Bank Identifier 1-3, Branch Identifier:4-7, Check Digit 8 ', + 7 => '8!n', + 8 => '212-1100-9', + 9 => '212110090000000235698741', + 10 => '0', + 11 => 'AL2!n8!n16!c', + 12 => '28', + 13 => 'AL47212110090000000235698741', + 14 => 'AL47 2121 1009 0000 0002 3569 8741', + 15 => 'No', + 16 => '', + */ +$ibanFormats = array(); +$header = fgetcsv($handle, 0, "\t"); +while (($data = fgetcsv($handle, 0, "\t")) !== FALSE) { + foreach($data as $i=>$column) { + $data[$i] = trim($column); + } + list($countryName, $countryCode, $example1, $bban, $bbanStruct, $bbanLength, $bankPos, $bankStruct, $example2, $example3, $iban, $ibanStruct, $ibanLength, $example4, $example5, $sepa, $details) = $data; + $format = array(); + if (!preg_match_all('/\d+!./', $bbanStruct, $matches)) { + echo "$countryName - uknown BBAN structure field format: $bbanStruct.\n"; + continue; + } + foreach($matches[0] as $match) { + list($count, $class) = explode('!', $match); + $format[] = array($class, $count); + } + $ibanFormats[$countryCode] = $format; +} +ksort($ibanFormats); +//var_export($ibanFormats); +echo "array(\n"; +foreach($ibanFormats as $language=>$parts) { + echo "\t'$language' => array("; + $formats = array(); + foreach($parts as $format) { + $formats[] = "array('$format[0]', $format[1])"; + } + echo implode(",\t", $formats); + echo "),\n"; +} +echo ");";