com.prowidesoftware.swift.model.Money Maven / Gradle / Ivy
/*
* Copyright 2006-2023 Prowide
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.prowidesoftware.swift.model;
import com.prowidesoftware.swift.model.field.AmountContainer;
import com.prowidesoftware.swift.model.field.CurrencyContainer;
import com.prowidesoftware.swift.model.field.Field;
import java.io.Serializable;
import java.math.BigDecimal;
import java.util.Currency;
import java.util.Objects;
import java.util.logging.Logger;
import org.apache.commons.lang3.StringUtils;
/**
* A simple POJO to represent money, an amount associated with a currency.
*
* This might someday be replaced by an implementation of https://javamoney.github.io/api.html
*
* @since 8.0.1
*/
public final class Money implements Serializable {
private static final long serialVersionUID = -7552352742105490377L;
private static final transient Logger log = Logger.getLogger(Money.class.getName());
private final String currency;
private final BigDecimal amount;
/**
* @param currency a not null currency
* @param amount the value for the amount, may be null
*/
public Money(final Currency currency, final BigDecimal amount) {
this(currency.getCurrencyCode(), amount);
}
/**
* @param currency a not null currency code
* @param amount the value for the amount, may be null
*/
public Money(final String currency, final BigDecimal amount) {
Objects.requireNonNull(currency, "currency can not be null");
this.currency = currency;
if (amount == null) {
this.amount = BigDecimal.ZERO;
} else {
this.amount = amount;
}
}
/**
* @param currency a not null currency
* @param amount the value for the amount, may be null
*/
public Money(final Currency currency, Number amount) {
this(currency.getCurrencyCode(), amount);
}
/**
* @param currency a not null currency code
* @param amount the value for the amount, may be null
*/
public Money(final String currency, Number amount) {
this.currency = currency;
if (amount == null) {
this.amount = BigDecimal.ZERO;
} else {
if (amount instanceof BigDecimal) {
this.amount = (BigDecimal) amount;
} else if (amount instanceof Long) {
this.amount = new BigDecimal(amount.longValue());
} else if (amount instanceof Integer) {
this.amount = new BigDecimal(amount.intValue());
} else if (amount instanceof Short) {
this.amount = new BigDecimal(amount.intValue());
} else if (amount instanceof Double) {
/*
* we use valueOf instead of constructor because it uses a string under the covers to eliminate floating point rounding errors
*/
this.amount = BigDecimal.valueOf(amount.doubleValue());
} else {
throw new IllegalArgumentException("class " + amount.getClass().getName() + " is not supported");
}
}
}
/**
* Creates a currency amount from an MT field.
* The field must at least implement the {@link AmountContainer} interface and
* either have a currency component or implements the {@link CurrencyContainer}
* For some fields like the signed 19A or the 62[F,M] which have a debit/credit mark
* component, the amount will be positive or negative accordingly.
*
* @param field a field with currency and amount components
* @return the created currency amount object or null if field is null or invalid.
*/
/*
* Do not use API from MTs and Field classes here to avoid cyclic dependency in code generation
* Component numbers do not normally change, although keep code below in sync with fields 62FM, 19A and 33B.
*/
static Money of(Field field) {
if (field != null && field instanceof AmountContainer) {
/*
* amount from interface
*/
BigDecimal amount = ((AmountContainer) field).amount();
if (amount == null) {
log.warning("cannot extract amount component from field " + field.getName() + ":" + field.getValue());
return null;
}
/*
* special cases
*/
String currency = null;
boolean negative = false;
if ("62F".equals(field.getName()) || "62M".equals(field.getName())) {
currency = field.getComponent(3);
negative = StringUtils.equals("D", field.getComponent(1));
} else if ("19A".equals(field.getName())) {
negative = StringUtils.equals("N", field.getComponent(2));
} else if ("33B".equals(field.getName())) {
currency = field.getComponent(1);
}
/*
* currency from interface
*/
if (currency == null && field instanceof CurrencyContainer) {
currency = ((CurrencyContainer) field).currencyString();
}
if (currency == null) {
log.warning("cannot extract currency component from field " + field.getName() + ":" + field.getValue());
return null;
}
if (negative) {
amount = amount.negate();
}
return new Money(currency, amount);
}
return null;
}
/**
* Creates a currency amount from the first found field in the block
*
* @param fieldNames a list of field names to extract the money from
*/
static Money ofAny(SwiftTagListBlock block, String... fieldNames) {
for (String fieldName : fieldNames) {
Field field = block.getFieldByName(fieldName);
if (field != null) {
return of(field);
}
}
return null;
}
/**
* Creates a currency amount from the sum of all given fields.
* Will return null if currencies not match.
*
* @param fields fields to sum, currency must be the same for all
* @return total or null if cannot create amount from any field or if not all fields currency match
*/
static Money ofSum(Field... fields) {
if (fields == null || fields.length == 0) {
return null;
}
BigDecimal total = null;
String currency = null;
for (Field field : fields) {
Money money = of(field);
if (money == null) {
return null;
}
if (total == null) {
total = money.getAmount();
currency = money.getCurrency();
} else if (StringUtils.equals(currency, money.getCurrency())) {
total = total.add(money.getAmount());
} else {
log.warning("cannot sum amounts with different currencies, expected " + currency + " and found "
+ money.getCurrency() + " in field " + field.getName() + ":" + field.getValue());
return null;
}
}
if (total != null && currency != null) {
return new Money(currency, total);
} else {
return null;
}
}
public String getCurrency() {
return currency;
}
public BigDecimal getAmount() {
return amount;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy