. */ declare(strict_types=1); namespace App\Services\CSV\Converter; /** * Class Amount. */ class Amount implements ConverterInterface { public static function negative(string $amount): string { if (1 === bccomp($amount, '0')) { $amount = bcmul($amount, '-1'); } return $amount; } public static function positive(string $amount): string { if (-1 === bccomp($amount, '0')) { $amount = bcmul($amount, '-1'); } return $amount; } /** * Some people, when confronted with a problem, think "I know, I'll use regular expressions." Now they have two problems. * - Jamie Zawinski. * * @param mixed $value */ public function convert($value): string { if (null === $value || '' === $value) { return '0'; } app('log')->debug(sprintf('Start with amount "%s"', $value)); $original = $value; $value = $this->stripAmount((string)$value); $decimal = null; if ($this->decimalIsDot($value)) { $decimal = '.'; app('log')->debug(sprintf('Decimal character in "%s" seems to be a dot.', $value)); } if ($this->decimalIsComma($value)) { $decimal = ','; app('log')->debug(sprintf('Decimal character in "%s" seems to be a comma.', $value)); } // decimal character is null? find out if "0.1" or ".1" or "0,1" or ",1" if ($this->alternativeDecimalSign($value)) { $decimal = $this->getAlternativeDecimalSign($value); } // decimal character still null? Search from the left for '.',',' or ' '. if (null === $decimal) { // See issue #8404 // $decimal = $this->findFromLeft($value); app('log')->debug('Disabled "findFromLeft" because it happens more often that "1.000" is thousand than "1.000" is 1 with three zeroes.'); } // if decimal is dot, replace all comma's and spaces with nothing if (null !== $decimal) { $value = $this->replaceDecimal($decimal, $value); app('log')->debug(sprintf('Converted amount from "%s" to "%s".', $original, $value)); } if (null === $decimal) { // replace all: $search = ['.', ' ', ',']; $value = str_replace($search, '', $value); app('log')->debug(sprintf('No decimal character found. Converted amount from "%s" to "%s".', $original, $value)); } if (str_starts_with($value, '.')) { $value = '0'.$value; } if (is_numeric($value)) { app('log')->debug(sprintf('Final NUMERIC value is: "%s"', $value)); return $value; } // @codeCoverageIgnoreStart app('log')->debug(sprintf('Final value is: "%s"', $value)); $formatted = sprintf('%01.12f', $value); app('log')->debug(sprintf('Is formatted to : "%s"', $formatted)); return $formatted; // @codeCoverageIgnoreEnd } /** * Add extra configuration parameters. */ public function setConfiguration(string $configuration): void {} /** * Check if the value has a dot or comma on an alternative place, * catching strings like ",1" or ".5". */ private function alternativeDecimalSign(string $value): bool { $length = strlen($value); $altPosition = $length - 2; return $length > 1 && ('.' === $value[$altPosition] || ',' === $value[$altPosition]); } /** * Helper function to see if the decimal separator is a comma. */ private function decimalIsComma(string $value): bool { $length = strlen($value); $decimalPosition = $length - 3; $result = $length > 2 && ',' === $value[$decimalPosition]; if (true === $result) { return true; } // if false, try to see if this number happens to be formatted like: // 0,xxxxxxxxxx if (1 === substr_count($value, ',') && str_starts_with($value, '0,')) { return true; } return false; } /** * Helper function to see if the decimal separator is a dot. */ private function decimalIsDot(string $value): bool { $length = strlen($value); $decimalPosition = $length - 3; return ($length > 2 && '.' === $value[$decimalPosition]) || ($length > 2 && strpos($value, '.') > $decimalPosition); } /** * Search from the left for decimal sign. */ private function findFromLeft(string $value): ?string { $decimal = null; app('log')->debug('Decimal is still NULL, probably number with >2 decimals. Search for a dot.'); $res = strrpos($value, '.'); if (false !== $res) { // blandly assume this is the one. app('log')->debug(sprintf('Searched from the left for "." in amount "%s", assume this is the decimal sign.', $value)); $decimal = '.'; } return $decimal; } /** * Returns the alternative decimal point used, such as a dot or a comma, * from strings like ",1" or "0.5". */ private function getAlternativeDecimalSign(string $value): string { $length = strlen($value); $altPosition = $length - 2; return $value[$altPosition]; } /** * Replaces other characters like thousand separators with nothing to make the decimal separator the only special * character in the string. */ private function replaceDecimal(string $decimal, string $value): string { $search = [',', ' ']; // default when decimal sign is a dot. if (',' === $decimal) { $search = ['.', ' ']; } $value = str_replace($search, '', $value); // @noinspection CascadeStringReplacementInspection return str_replace(',', '.', $value); } /** * Strip amount from weird characters. */ private function stripAmount(string $value): string { if (str_starts_with($value, '--')) { $value = substr($value, 2); } // have to strip the € because apparently the Postbank (DE) thinks "1.000,00 €" is a normal way to format a number. // 2020-12-01 added "EUR" because another German bank doesn't know what a data format is. // This way of stripping exceptions is unsustainable. $value = trim((string)str_replace(['€', 'EUR'], '', $value)); $str = preg_replace('/[^\-().,0-9 ]/', '', $value); $len = strlen($str); if (str_starts_with($str, '(') && ')' === $str[$len - 1]) { $str = '-'.substr($str, 1, $len - 2); } $str = trim($str); app('log')->debug(sprintf('Stripped "%s" to "%s"', $value, $str)); return $str; } }