All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.prowidesoftware.swift.model.BIC Maven / Gradle / Ivy

There is a newer version: SRU2023-10.1.16
Show newest version
/*
 * 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) { DistinguishedName.Builder dnBuilder = new DistinguishedName.Builder(getBic8()); if (includeDefaultBranch || !Objects.equals(getBranchOrDefault(), "XXX")) { dnBuilder.withBranch(getBranchOrDefault()); } return dnBuilder.build().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; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy