From b820154f419b1351e5ccbd8d5576849e67d7a650 Mon Sep 17 00:00:00 2001 From: someniatko Date: Mon, 22 May 2023 22:39:04 +0300 Subject: [PATCH 1/3] Add @psalm-immutable and @psalm-pure annotations some shortcuts were taken - for example, for the places where \NumberFormatter is used, @psalm-suppress annotations was required also, unfortunately, Currency::of() is using ISOCurrencyProvider which is heavily impure internally, therefore all static methods using Currency::of() were not marked as pure. --- src/AbstractMoney.php | 2 ++ src/Context.php | 2 ++ src/Context/AutoContext.php | 2 ++ src/Context/CashContext.php | 2 ++ src/Context/CustomContext.php | 2 ++ src/Context/DefaultContext.php | 2 ++ src/Currency.php | 2 ++ src/Money.php | 22 ++++++++++++++++++++++ src/RationalMoney.php | 2 ++ 9 files changed, 38 insertions(+) diff --git a/src/AbstractMoney.php b/src/AbstractMoney.php index cade2ad..80ae1b1 100644 --- a/src/AbstractMoney.php +++ b/src/AbstractMoney.php @@ -18,6 +18,8 @@ * * Please consider this class sealed: extending this class yourself is not supported, and breaking changes (such as * adding new abstract methods) can happen at any time, even in a minor version. + * + * @psalm-immutable */ abstract class AbstractMoney implements MoneyContainer, Stringable, JsonSerializable { diff --git a/src/Context.php b/src/Context.php index dca84c7..7a428bf 100644 --- a/src/Context.php +++ b/src/Context.php @@ -11,6 +11,8 @@ /** * Adjusts a rational number to a decimal amount. + * + * @psalm-immutable */ interface Context { diff --git a/src/Context/AutoContext.php b/src/Context/AutoContext.php index 4e6bc56..91d7c6e 100644 --- a/src/Context/AutoContext.php +++ b/src/Context/AutoContext.php @@ -13,6 +13,8 @@ /** * Automatically adjusts the scale of a number to the strict minimum. + * + * @psalm-immutable */ final class AutoContext implements Context { diff --git a/src/Context/CashContext.php b/src/Context/CashContext.php index 1a73716..d252e4f 100644 --- a/src/Context/CashContext.php +++ b/src/Context/CashContext.php @@ -12,6 +12,8 @@ /** * Adjusts a number to the default scale for the currency, respecting a cash rounding. + * + * @psalm-immutable */ final class CashContext implements Context { diff --git a/src/Context/CustomContext.php b/src/Context/CustomContext.php index 299b650..789cb31 100644 --- a/src/Context/CustomContext.php +++ b/src/Context/CustomContext.php @@ -12,6 +12,8 @@ /** * Adjusts a number to a custom scale, and optionally step. + * + * @psalm-immutable */ final class CustomContext implements Context { diff --git a/src/Context/DefaultContext.php b/src/Context/DefaultContext.php index 51b93e4..0999758 100644 --- a/src/Context/DefaultContext.php +++ b/src/Context/DefaultContext.php @@ -12,6 +12,8 @@ /** * Adjusts a number to the default scale for the currency. + * + * @psalm-immutable */ final class DefaultContext implements Context { diff --git a/src/Currency.php b/src/Currency.php index c2a014e..6a13725 100644 --- a/src/Currency.php +++ b/src/Currency.php @@ -9,6 +9,8 @@ /** * A currency. This class is immutable. + * + * @psalm-immutable */ final class Currency implements Stringable { diff --git a/src/Money.php b/src/Money.php index 6a8ad25..5744262 100644 --- a/src/Money.php +++ b/src/Money.php @@ -30,6 +30,8 @@ * - CashContext is similar to DefaultContext, but supports a cash rounding step. * - CustomContext handles monies with a custom scale, and optionally step. * - AutoContext automatically adjusts the scale of the money to the minimum required. + * + * @psalm-immutable */ final class Money extends AbstractMoney { @@ -71,6 +73,8 @@ private function __construct(BigDecimal $amount, Currency $currency, Context $co * @return Money * * @throws MoneyMismatchException If all the monies are not in the same currency. + * + * @psalm-pure */ public static function min(Money $money, Money ...$monies) : Money { @@ -96,6 +100,8 @@ public static function min(Money $money, Money ...$monies) : Money * @return Money * * @throws MoneyMismatchException If all the monies are not in the same currency. + * + * @psalm-pure */ public static function max(Money $money, Money ...$monies) : Money { @@ -121,6 +127,8 @@ public static function max(Money $money, Money ...$monies) : Money * @return Money * * @throws MoneyMismatchException If all the monies are not in the same currency and context. + * + * @psalm-pure */ public static function total(Money $money, Money ...$monies) : Money { @@ -146,6 +154,8 @@ public static function total(Money $money, Money ...$monies) : Money * @return Money * * @throws RoundingNecessaryException If RoundingMode::UNNECESSARY is used but rounding is necessary. + * + * @psalm-pure */ public static function create(BigNumber $amount, Currency $currency, Context $context, int $roundingMode = RoundingMode::UNNECESSARY) : Money { @@ -641,6 +651,10 @@ private function gcdOfMultipleInt(array $values): int { $values = array_map(fn (int $value) => BigInteger::of($value), $values); + /** + * @psalm-suppress ImpureMethodCall + * FIXME: remove the suppression after BigInteger::gcdMultiple() is marked @psalm-pure + */ return BigInteger::gcdMultiple(...$values)->toInt(); } @@ -745,6 +759,10 @@ public function convertedTo( int $roundingMode = RoundingMode::UNNECESSARY, ) : Money { if (! $currency instanceof Currency) { + /** + * FIXME: dunno what to do here, Currency::of() uses ISOCurrencyProvider which is heavily "impure". + * @psalm-suppress ImpureMethodCall + */ $currency = Currency::of($currency); } @@ -769,6 +787,7 @@ public function convertedTo( */ public function formatWith(\NumberFormatter $formatter) : string { + /** @psalm-suppress ImpureMethodCall */ return $formatter->formatCurrency( $this->amount->toFloat(), $this->currency->getCurrencyCode() @@ -785,6 +804,9 @@ public function formatWith(\NumberFormatter $formatter) : string * @param bool $allowWholeNumber Whether to allow formatting as a whole number if the amount has no fraction. * * @return string + * + * @psalm-suppress ImpureMethodCall - \NumberFormatter is impure, but we use it here in a side effect free way + * @psalm-suppress ImpureStaticVariable - static variables are used for optimization reasons */ public function formatTo(string $locale, bool $allowWholeNumber = false) : string { diff --git a/src/RationalMoney.php b/src/RationalMoney.php index fa98d39..d71e3b9 100644 --- a/src/RationalMoney.php +++ b/src/RationalMoney.php @@ -15,6 +15,8 @@ * * This is used to represent intermediate calculation results, and may not be exactly convertible to a decimal amount * with a finite number of digits. The final conversion to a Money may require rounding. + * + * @psalm-immutable */ final class RationalMoney extends AbstractMoney { From 93256b099035ada6f3dabd71a8a4a4cae0ed95bc Mon Sep 17 00:00:00 2001 From: someniatko Date: Mon, 25 Sep 2023 10:23:07 +0300 Subject: [PATCH 2/3] Add @psalm-pure annotations to `Currency` static constructors as discussed in https://github.com/brick/money/pull/75 --- src/Currency.php | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/src/Currency.php b/src/Currency.php index 6a13725..ccb3259 100644 --- a/src/Currency.php +++ b/src/Currency.php @@ -78,9 +78,15 @@ public function __construct(string $currencyCode, int $numericCode, string $name * @param string|int $currencyCode The 3-letter or numeric ISO 4217 currency code. * * @throws UnknownCurrencyException If an unknown currency code is given. + * + * @psalm-pure */ public static function of(string|int $currencyCode) : Currency { + /** + * @psalm-suppress ImpureMethodCall + * @see https://github.com/brick/money/pull/75 + */ return ISOCurrencyProvider::getInstance()->getCurrency($currencyCode); } @@ -92,9 +98,15 @@ public static function of(string|int $currencyCode) : Currency * @return Currency * * @throws UnknownCurrencyException If the country code is unknown, or there is no single currency for the country. + * + * @psalm-pure */ public static function ofCountry(string $countryCode) : Currency { + /** + * @psalm-suppress ImpureMethodCall + * @see https://github.com/brick/money/pull/75 + */ return ISOCurrencyProvider::getInstance()->getCurrencyForCountry($countryCode); } From 6943ffe79f432d993ce39a05d86e05cb5dc0d580 Mon Sep 17 00:00:00 2001 From: someniatko Date: Mon, 25 Sep 2023 10:26:40 +0300 Subject: [PATCH 3/3] Add @psalm-pure annotations to the static methods which use `Currency::of()` --- src/Money.php | 10 ++++++---- src/RationalMoney.php | 2 ++ 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/src/Money.php b/src/Money.php index 5744262..b687dfd 100644 --- a/src/Money.php +++ b/src/Money.php @@ -187,6 +187,8 @@ public static function create(BigNumber $amount, Currency $currency, Context $co * @throws UnknownCurrencyException If the currency is an unknown currency code. * @throws RoundingNecessaryException If the rounding mode is RoundingMode::UNNECESSARY, and rounding is necessary * to represent the amount at the requested scale. + * + * @psalm-pure */ public static function of( BigNumber|int|float|string $amount, @@ -227,6 +229,8 @@ public static function of( * @throws UnknownCurrencyException If the currency is an unknown currency code. * @throws RoundingNecessaryException If the rounding mode is RoundingMode::UNNECESSARY, and rounding is necessary * to represent the amount at the requested scale. + * + * @psalm-pure */ public static function ofMinor( BigNumber|int|float|string $minorAmount, @@ -257,6 +261,8 @@ public static function ofMinor( * @param Context|null $context An optional context. * * @return Money + * + * @psalm-pure */ public static function zero(Currency|string|int $currency, ?Context $context = null) : Money { @@ -759,10 +765,6 @@ public function convertedTo( int $roundingMode = RoundingMode::UNNECESSARY, ) : Money { if (! $currency instanceof Currency) { - /** - * FIXME: dunno what to do here, Currency::of() uses ISOCurrencyProvider which is heavily "impure". - * @psalm-suppress ImpureMethodCall - */ $currency = Currency::of($currency); } diff --git a/src/RationalMoney.php b/src/RationalMoney.php index d71e3b9..97c9e87 100644 --- a/src/RationalMoney.php +++ b/src/RationalMoney.php @@ -43,6 +43,8 @@ public function __construct(BigRational $amount, Currency $currency) * @param Currency|string|int $currency The Currency instance, ISO currency code or ISO numeric currency code. * * @return RationalMoney + * + * @psalm-pure */ public static function of(BigNumber|int|float|string $amount, Currency|string|int $currency) : RationalMoney {