Java生成银联PIN BLOCK

import org.apache.commons.codec.binary.Hex;
import org.apache.commons.lang3.StringUtils;
import org.bouncycastle.pqc.math.linearalgebra.ByteUtils;

/**
 * 银联工具包
 */
public class UnionPayUtil {
    private static final int PIN_MIN_LENGTH = 4;
    private static final int PIN_MAX_LENGTH = 12;
    private static final int HEXADECIMAL = 16;
    /**
     * DES/3SDE算法PIN BLOCK大小为8
     * SM4算法PIN BLOCK大小为16
     */
    private static final int PIN_BLOCK_SIZE = 16;

    /**
     * 生成 PIN Block
     * @param pin
     * @return
     */
    private static byte[] pinBlock(String pin) {
        int pinLength = pin.length();
        if (pinLength < PIN_MIN_LENGTH || pinLength > PIN_MAX_LENGTH) {
            throw new IllegalArgumentException("Pin should be 4 to 12 digits long");
        }

        /**
         * PIN 格式 16BYTE 大小
         * 1BYTE PIN长度 + 15BYTE Pin
         * Pin不足部分右补F
         */
        byte[] pinBlock = new byte[PIN_BLOCK_SIZE];
        //如果是奇数右补F转换成如果不足12位右补F,补到12位长度
        pin = StringUtils.rightPad(pin,PIN_MAX_LENGTH,'F');
        //先处理PIN BLOCK 长度
        pinBlock[0] = (byte) pinLength;
        //处理PIN
        for (int i = 0; i <= pin.length() - 1; i = i + 2) {
            int pinBlockIndex = (i / 2) + 1;
            pinBlock[pinBlockIndex] = (byte) Integer.parseInt(pin.substring(i, i + 2), HEXADECIMAL);
        }

        /**
         * 不足32位补,需要补的位长度为PinBlock减去PIN长度域的长度和PIN数据长度
         */

        int blockIndex = (pin.length() / 2) + 1;
        for (int i = blockIndex; i <= PIN_BLOCK_SIZE - 1; i++) {
            pinBlock[i] = (byte) 0xFF;
        }
        return pinBlock;
    }

    /**
     * 格式
     * 2BYTE %H0000
     * 14BYTE 取主账号的右12位(不包括最右边的校验位),主账号不足12位左补0
     *
     * @param mAccNo 主账号
     * @return
     */
    private static byte[] pan(String mAccNo) {
        /**
         * 不足12位左补0,后面要取右12位,不包括校验位,全长是不足13位的就需要补
         */
        StringUtils.leftPad(mAccNo, 13, '0');
        byte[] panByte = new byte[PIN_BLOCK_SIZE];
        //截取右12位,包括校验位是13位
        String pan = StringUtils.left(StringUtils.right(mAccNo, 12 + 1), 12);
        /**
         * 2BYTE %H0000
         * 14BYTE 取主账号的右12位(不包括最右边的校验位),主账号不足12位左补0
         */
        //初始化左边为0区域
        int leftSize = PIN_BLOCK_SIZE - 12 / 2;
        for (int i = 0; i <= leftSize - 1; i++) {
            panByte[i] = (byte) 0x00;
        }
        int panIndex = 0;
        for (int i = leftSize; i <= PIN_BLOCK_SIZE - 1; i++, panIndex+=2) {
            String a = pan.substring(panIndex, panIndex + 2);
            panByte[i] = (byte) Integer.parseInt(a, HEXADECIMAL);
        }
        return panByte;
    }

    /**
     * SM4加密,并进行转16进制计算
     *
     * @return
     */
    public static String encryptPin(String accNo, String pin, String key) {
        byte[] pinBlock = null;
        try {
            pinBlock = pinBlock(pin, accNo);
            byte[] keyBytes = ByteUtils.fromHexString(key);
            pinBlock = SM4Helper.encrypt(keyBytes,pinBlock);
            return ByteUtils.toHexString(pinBlock);
        } catch (Exception e) {
            return "";
        }
    }


    /**
     * @param pin    Personal Identification Number
     * @param mAccNo Main account number
     * @return
     */
    private static byte[] pinBlock(String pin, String mAccNo) {
        byte[] pingBlock = pinBlock(pin);
        byte[] pan = pan(mAccNo);
        byte[] tByte = new byte[PIN_BLOCK_SIZE];
        for (int i = 0; i < PIN_BLOCK_SIZE; i++) {
            tByte[i] = (byte) (pingBlock[i] ^ pan[i]);
        }
        return tByte;
    }

    public static void main(String[] args) {
        String key = "11111111111111111111111111111111";
        String pin = "123456";
        String accNo = "123456789012345678";
        String pinBlock = Hex.encodeHexString(pinBlock(pin, accNo));
        System.out.println(pinBlock);
    }
}
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

友情链接更多精彩内容