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

com.credibledoc.iso8583packer.ifa.IfaBitmapPacker Maven / Gradle / Ivy

package com.credibledoc.iso8583packer.ifa;

import com.credibledoc.iso8583packer.bitmap.BitmapPacker;
import com.credibledoc.iso8583packer.bitmap.BitmapService;
import com.credibledoc.iso8583packer.exception.PackerRuntimeException;
import com.credibledoc.iso8583packer.hex.HexService;
import com.credibledoc.iso8583packer.ifb.IfbBitmapPacker;
import com.credibledoc.iso8583packer.message.MsgField;
import com.credibledoc.iso8583packer.message.MsgValue;

import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.util.Arrays;
import java.util.BitSet;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * The {@link BitmapPacker} implementation for IFA format. It uses {@link #ISO_88591} charset.
 * 

* Actual documentation and examples * ifa-bitmap-packer.md. * * @author Kyrylo Semenko */ public class IfaBitmapPacker implements BitmapPacker { private static final Charset ISO_88591 = StandardCharsets.ISO_8859_1; /** * Contains created instances. Each instance is a Singleton. Key is the {@link #bitsetBytesLength} value. */ private static final Map instances = new ConcurrentHashMap<>(); /** * Number of bytes in a packed state. If the value is -1, it means that the number of bytes * depends on a maximal child's {@link MsgField#getFieldNum()} value, see the {@link #getInstance()} description. */ private final int bitsetBytesLength; /** * Helps to prepare {@link BitSet} data. */ private final IfbBitmapPacker ifbBitmapPacker; private IfaBitmapPacker(int bitsetBytesLength) { this.bitsetBytesLength = bitsetBytesLength; ifbBitmapPacker = IfbBitmapPacker.getInstance(bitsetBytesLength); } /** * Static factory. Creates and returns singletons stored in the {@link #instances} map. * @param bitsetBytesLength number of bytes in {@link BitSet}. For example:

BitsetBytesLength   Bitset HEX   Packed HEX  
24 C000000000000000C0000000000000004000000000000000   433030303030303030303030303030304330303030303030303030303030303034303030303030303030303030303030
16 D0000000000000004000000000000000 4430303030303030303030303030303034303030303030303030303030303030
8 5000000000000000 35303030303030303030303030303030
* See https://neapay.com/online-tools/bitmap-fields-decoder.html * @return Existing instance from {@link #instances} or a new created instance. */ public static IfaBitmapPacker getInstance(int bitsetBytesLength) { if (bitsetBytesLength != 8 && bitsetBytesLength != 16 && bitsetBytesLength != 24) { throw new PackerRuntimeException("Expected value is 8, 16 or 24."); } instances.computeIfAbsent(bitsetBytesLength, k -> new IfaBitmapPacker(bitsetBytesLength)); return instances.get(bitsetBytesLength); } /** * Static factory that creates an adaptable bit map, where its length depends on a maximal * child {@link MsgField#getFieldNum()} value. The packed * bitMap bytes may have 8*2, 16*2 or 24*2 bytes. * See examples in the {@link #getInstance(int)} method description. * * @return Existing instance from {@link #instances} or a new created instance. */ public static IfaBitmapPacker getInstance() { int bitsetBytesLength = -1; instances.computeIfAbsent(bitsetBytesLength, k -> new IfaBitmapPacker(bitsetBytesLength)); return instances.get(bitsetBytesLength); } /** * @param bitSet for packing * @return Packed bytes */ @Override public byte[] pack(BitSet bitSet) { byte[] ifb = ifbBitmapPacker.pack(bitSet); return HexService.bytesToHex(ifb).getBytes(ISO_88591); } /** * @param msgValue the target container for storing the unpacked {@link BitSet} * @param bytes the source bytes * @param offset starting offset within the bytes * @return Consumed bytes number */ @Override public int unpack(MsgValue msgValue, byte[] bytes, int offset) { int resolvedLen = resolvePackedLen(bytes, offset); byte[] ifaBytes = Arrays.copyOfRange(bytes, offset, offset + resolvedLen); String ifbString = new String(ifaBytes, ISO_88591); byte[] ifbBytes = HexService.hex2byte(ifbString); int unpackedLen = ifbBitmapPacker.unpack(msgValue, ifbBytes, 0); if (bitsetBytesLength != -1 && unpackedLen != bitsetBytesLength) { throw new PackerRuntimeException("Result bytes length '" + unpackedLen + "' not equals with required packedBytesLength '" + resolvedLen + "'."); } return resolvedLen; } private int resolvePackedLen(byte[] bytes, int offset) { int result = 8; if (getPackedBytesLength() != -1) { return getPackedBytesLength(); } if (firstBitHasFlag(bytes, offset, 0)) { result = 16; // existing secondary bitmap if (firstBitHasFlag(bytes, offset, 8 * 2)) { result = 24; // existing tertiary bitmap } } return result * 2; } private boolean firstBitHasFlag(byte[] bytes, int firstOffset, int secondOffset) { int offset = firstOffset + secondOffset; if (bytes.length < offset + 2) { return false; } byte[] ifaBytes = Arrays.copyOfRange(bytes, offset, offset + 2); String ifbString = new String(ifaBytes, ISO_88591); byte[] ifbBytes = HexService.hex2byte(ifbString); return BitmapService.hasFlag(ifbBytes[0]); } @Override public int getPackedBytesLength() { if (bitsetBytesLength > 0) { return bitsetBytesLength * 2; } return bitsetBytesLength; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy