com.prowidesoftware.swift.model.BIC Maven / Gradle / Ivy
Show all versions of pw-swift-core Show documentation
/*
* 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.utils.IsoUtils;
import java.util.Objects;
import org.apache.commons.lang3.StringUtils;
/**
* Helper class to process BIC information.
*
* Bank Identifier Codes (also known as SWIFT-BIC, BIC, SWIFT ID or SWIFT code) is a unique identification code for
* both financial and non-financial institutions. When assigned to a non-financial institution, the code may also be
* known as a Business Entity Identifier or BEI.
*
*
It is composed by:
*
* - 4 letters: institution code or bank code.
*
- 2 letters: ISO 3166-1 alpha-2 country code
*
- 2 letters or digits: location code
*
- 3 letters: branch code
*
*
* @since 3.3
*/
public class BIC {
/**
* Fake "test & training" BIC with 8 chars for testing
*
* @since 7.6
*/
public static final transient String TEST8 = "TESTARZZ";
/**
* Fake Logical terminal address for testing,
* consisting of a fake "test & training" BIC of 12 chars
* (including the terminal identification)
*
* @see SwiftBlock1#getLogicalTerminal()
* @since 7.6
*/
public static final transient String TEST12 = "TESTARZZAXXX";
/**
* Constant value with which all partner bics start
*
* @since 7.8
*/
public static final String PARTNER_PREFIX = "PTS";
protected String branch = null;
private String institution = null;
private String country = null;
private String location = null;
private String subtype = null;
/**
* Constructor with a BIC8 or BIC11 code.
*
* For BIC codes with 12 characters (meaning it includes the logical terminal identifier) use
* {@link LogicalTerminalAddress} instead. This implementation will drop the LT identifier if a
* 12 characters full logical terminal addess is passed as parameter.
*
*
If the code is longer than 11 characters, the remainder will be store as part of the branch code.
*
* @param bic the BIC code to use in this BIC (8 or 11 chars)
*/
public BIC(final String bic) {
parse(bic);
}
/**
* Default constructor
*/
public BIC() {}
protected void parse(final String bic) {
if (bic != null) {
this.institution = StringUtils.trimToNull(StringUtils.substring(bic, 0, 4));
this.country = StringUtils.trimToNull(StringUtils.substring(bic, 4, 6));
this.location = StringUtils.trimToNull(StringUtils.substring(bic, 6, 8));
if (bic.length() >= 12) {
// drop LT identifier
this.branch = StringUtils.trimToNull(StringUtils.substring(bic, 9));
} else {
this.branch = StringUtils.trimToNull(StringUtils.substring(bic, 8));
}
}
}
/**
* Validates the BIC structure.
*
* @return true if the BIC is valid and false otherwise
* @see #validate() for details regarding the validation checks or if you need structured details of the validation
* problem found.
*/
public boolean isValid() {
return validate() == BicValidationResult.OK;
}
/**
* Validates the BIC structure.
*
*
Checks the syntax of the BIC, verifying: the total length is 8, 11 or 12 (LT identifier), the country is a
* valid ISO country code using {@link IsoUtils}, the institution is composed by upper case letters and the
* location and branch are composed by upper case letter or digits.
*
*
This method does not validate against any BIC directory.
*
* @return BicValidationResult with detailed information of the validation problem found
* @since 7.10.3
*/
public BicValidationResult validate() {
if (this.institution == null || this.country == null || this.location == null) {
return BicValidationResult.INVALID_LENGTH;
}
if (this.institution.length() != 4) {
return BicValidationResult.INVALID_INSTITUTION_LENGTH.setFound(this.institution);
}
if (this.country.length() != 2) {
return BicValidationResult.INVALID_COUNTRY_LENGTH.setFound(this.country);
}
if (this.location.length() != 2) {
return BicValidationResult.INVALID_LOCATION_LENGTH.setFound(this.location);
}
if (this.branch != null && this.branch.length() != 3) {
return BicValidationResult.INVALID_BRANCH_LENGTH.setFound(this.branch);
}
if (!isUpperCase(this.institution)) {
return BicValidationResult.INVALID_INSTITUTION_CHARSET.setFound(this.institution);
}
if (!IsoUtils.getInstance().isValidISOCountry(this.country)) {
return BicValidationResult.INVALID_COUNTRY.setFound(this.country);
}
if (!isUpperCaseOrDigit(this.location)) {
return BicValidationResult.INVALID_LOCATION_CHARSET.setFound(this.location);
}
if (this.branch != null && !isUpperCaseOrDigit(this.branch)) {
return BicValidationResult.INVALID_BRANCH_CHARSET.setFound(this.branch);
}
return BicValidationResult.OK;
}
private boolean isUpperCase(final String text) {
for (int i = 0; i < text.length(); i++) {
if (!Character.isUpperCase(text.charAt(i))) {
return false;
}
}
return true;
}
private boolean isUpperCaseOrDigit(final String text) {
for (int i = 0; i < text.length(); i++) {
if (!Character.isUpperCase(text.charAt(i)) && !Character.isDigit(text.charAt(i))) {
return false;
}
}
return true;
}
/**
* It returns the last three that conform the branch or null if branch is not present.
*
* @return the BIC's branch part or null if not found.
* @since 7.8.5
*/
public String getBranch() {
return this.branch;
}
/**
* @param branch the branch part of the BIC
* @since 7.10.3
*/
public void setBranch(String branch) {
this.branch = branch;
}
/**
* Returns the branch code or XXX as default
* @return branch code or default string
* @since 7.10.3
*/
public String getBranchOrDefault() {
return this.branch != null ? this.branch : "XXX";
}
/**
* Returns ths subtype code.
* Notice this information is not part of the BIC code, it must be explicitly set with {@link #setSubtype(String)}
* @return subtype code
* @since 7.4
*/
public String getSubtype() {
return subtype;
}
/**
* Sets a subtype code
* @param subtype subtype code to set
*
* @since 7.4
*/
public void setSubtype(final String subtype) {
this.subtype = subtype;
}
/**
* Returns true if the BIC is a Test & Training BIC code.
*
In SWIFT’s FIN messaging system, a BIC with a zero in the 8th position is a Test & Training BIC, and as
* such it cannot be used in production FIN messages.
*
* @return true if it is a Test & Training BIC, false if is not or if the condition cannot be determined
* @since 7.6
*/
public boolean isTestAndTraining() {
if (this.location != null) {
return this.location.charAt(1) == '0';
}
return false;
}
/**
* Returns true if the BIC is not live (not connected) on the network.
*
*
BICs can identify not only financial institutions but also non-financial ones
* either connected or not connected to the SWIFT network.
*
*
A BIC of an institution which is not connected to the SWIFT network
* still has a location code with the digit 1 at the end (for instance AFSEUS31).
* BICs like that are called non-SWIFT BICs (or BIC 1).
*
*
In SWIFT’s FIN messaging system, a BIC with a one in the 8th position is a Non-Live BIC.
*
*
Note this is not the opposite of {@link #isLive()}
*
* @return true if it is a Non-Live BIC, false if is not or if the condition cannot be determined
* @since 7.7
*/
public boolean isNonLive() {
if (this.location != null) {
return this.location.charAt(1) == '1';
}
return false;
}
/**
* Returns true if the BIC is live (connected and not test & training) on the network.
*
*
BICs can identify not only financial institutions but also non-financial ones
* either connected or not connected to the SWIFT network.
*
*
In SWIFT’s FIN messaging system, a BIC with a character different than zero (that would be Test &
* Training) or one (that would be non-connected) in the 8th position is a Live BIC.
*
*
Note this is not the opposite of {@link #isNonLive()}
*
* @return true if it is a Non-Live BIC, false if is not or if the condition cannot be determined
* @since 7.7
*/
public boolean isLive() {
if (this.location != null) {
return this.location.charAt(1) != '0' && this.location.charAt(1) != '1';
}
return false;
}
/**
* Returns the first 8 characters of the BIC code, composed by the institution code, country and location.
*
* @return the bic8 or null if the BIC has less than 8 characters
* @since 7.6
*/
public String getBic8() {
if (this.institution != null && this.country != null && this.location != null) {
return this.institution + this.country + this.location;
} else {
return null;
}
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
BIC bic = (BIC) o;
return Objects.equals(institution, bic.institution)
&& Objects.equals(country, bic.country)
&& Objects.equals(location, bic.location)
&& Objects.equals(branch, bic.branch)
&& Objects.equals(subtype, bic.subtype);
}
@Override
public int hashCode() {
return Objects.hash(institution, country, location, branch, subtype);
}
/**
* Returns the BIC code with 11 characters composed by institution code, country, location and branch.
*
If the branch is not present, then XXX will be used as default branch.
*
* @return the bic11 or null if the BIC has less than 8 characters
* @since 7.6
*/
public String getBic11() {
final String bic8 = getBic8();
if (bic8 != null) {
return bic8 + getBranchOrDefault();
}
return null;
}
/**
* Returns the Distinguished Name (DN) for this BIC.
*
*
The created DN always includes the BIC8 and "swift" and
* if the branch is present and not "XXX" it will also be included
* as organization unit (ou)
*
* @return ou=<branch>,o=<bic8>,o=swift
* @since 7.9.3
*/
public String distinguishedName() {
return distinguishedName(false);
}
/**
* Returns the Distinguished Name (DN) for this BIC.
*
*
The created DN always includes the BIC8 and "swift" and
* if the branch is present and not "XXX" it will also be included
* as organization unit (ou)
*
* @param includeDefaultBranch if true will return ou=<xxx even if the branch is not present
* @return ou=<branch>,o=<bic8>,o=swift
* @since 9.3.15
*/
public String distinguishedName(boolean includeDefaultBranch) {
StringBuilder result = new StringBuilder();
if (includeDefaultBranch || !Objects.equals(getBranchOrDefault(), "XXX")) {
result.append("ou=")
.append(StringUtils.lowerCase(getBranchOrDefault()))
.append(",");
}
result.append("o=").append(StringUtils.lowerCase(getBic8())).append(",o=swift");
return result.toString();
}
@Override
public String toString() {
return getBic11();
}
/**
* @return the institution identifier part of the BIC
* @since 7.10.3
*/
public String getInstitution() {
return institution;
}
/**
* @param institution the institution identifier part of the BIC
* @since 7.10.3
*/
public void setInstitution(String institution) {
this.institution = institution;
}
/**
* @return the country part of the BIC
* @since 7.10.3
*/
public String getCountry() {
return country;
}
/**
* @param country the country part of the BIC
* @since 7.10.3
*/
public void setCountry(String country) {
this.country = country;
}
/**
* @return the location part of the BIC
* @since 7.10.3
*/
public String getLocation() {
return location;
}
/**
* @param location the location part of the BIC
* @since 7.10.3
*/
public void setLocation(String location) {
this.location = location;
}
/**
* Returns a new BIC with second character of the location set to zero.
* If the location is invalid it is set with a default value. Also if it has more than 2 characters it is trimmed
* to two with a zero in the second position.
* @return test BIC
* @since 9.1.3
*/
public BIC asTestBic() {
BIC bic = new BIC(this.getBic11());
if (bic.getLocation() == null || bic.getLocation().isEmpty()) {
bic.setLocation("00");
} else {
bic.setLocation(bic.getLocation().charAt(0) + "0");
}
return bic;
}
}