You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
235 lines
7.6 KiB
235 lines
7.6 KiB
<?php
|
|
|
|
/*
|
|
* Amount.php
|
|
* Copyright (c) 2021 james@firefly-iii.org
|
|
*
|
|
* This file is part of the Firefly III Data Importer
|
|
* (https://github.com/firefly-iii/data-importer).
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
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;
|
|
}
|
|
}
|