APIGateway中加密驗(yàn)簽介紹

需要提供給接口調(diào)用方一個(gè)用來加密的key归露,調(diào)用方根據(jù)key雕什、一些其他參數(shù)以及業(yè)務(wù)參數(shù)進(jìn)行加密裕照,還需要對報(bào)文進(jìn)行簽名,使用加密的參數(shù)請求接口在刺。

APIGateway接收到請求后進(jìn)行驗(yàn)簽逆害,再進(jìn)行解密头镊,得到參數(shù)后進(jìn)行處理。

處理完成需要按照跟請求接口一樣的方式將結(jié)果進(jìn)行加密加簽魄幕,然后將結(jié)果返回給調(diào)用方相艇,調(diào)用方按照同樣的方式進(jìn)行驗(yàn)簽解密拿到結(jié)果。

方案:摘要簽名纯陨、對稱加密

  1. kv參數(shù)排序坛芽,按照key自然排序,ASCII升序翼抠,并將參數(shù)值URLEncode一下靡馁。
  2. 在參數(shù)中加入兩個(gè)額外參數(shù)timestamp和nonce。timestamp表示請求發(fā)送時(shí)間机久,APIGateway可以對比當(dāng)前時(shí)間戳,來判斷請求是否正常赔嚎;nonce表示一個(gè)隨機(jī)數(shù)膘盖,也就是常說的鹽值,APIGateway可以通過一定時(shí)間段內(nèi)判斷nonce是否重復(fù)來判斷請求是否正常尤误。
  3. 使用給定的key侠畔,將參數(shù)進(jìn)行加密,使用AES或者DES等损晤。
  4. 再將所有參數(shù)進(jìn)行簽名软棺,MD5或者SHA1等

示例代碼:

public class CodecClient {

    public static final String HMAC_SHA1 = "HmacSHA1";

    public static final String AES = "AES";

    public static final String AES_CBC = "AES/CBC/PKCS5Padding";


    public static String encrypt(String key, String content) {
        try {
            SecretKeySpec secretKeySpec = new SecretKeySpec(key.getBytes(StandardCharsets.UTF_8), AES);
            Cipher cipher = Cipher.getInstance(AES_CBC);
            IvParameterSpec iv = new IvParameterSpec(new byte[16]);
            cipher.init(Cipher.ENCRYPT_MODE, secretKeySpec, iv);
            return bytesToHexString(cipher.doFinal(content.getBytes(StandardCharsets.UTF_8)));
        } catch (NoSuchPaddingException | NoSuchAlgorithmException | InvalidKeyException | BadPaddingException | IllegalBlockSizeException | InvalidAlgorithmParameterException e) {
            throw new RuntimeException("Encrypt data error!");
        }
    }

    public static String decrypt(String key, String content) {
        try {
            SecretKeySpec secretKeySpec = new SecretKeySpec(key.getBytes(StandardCharsets.UTF_8), AES);
            Cipher cipher = Cipher.getInstance(AES_CBC);
            IvParameterSpec iv = new IvParameterSpec(new byte[16]);
            cipher.init(Cipher.DECRYPT_MODE, secretKeySpec, iv);
            return new String(cipher.doFinal(hexStringToBytes(content)));
        } catch (NoSuchAlgorithmException | InvalidKeyException | InvalidAlgorithmParameterException | NoSuchPaddingException | BadPaddingException | IllegalBlockSizeException e) {
            e.printStackTrace();
            throw new RuntimeException("Decrypt data error!");
        }
    }

    public static String sign(String key, String data, String nonce) {
        try {
            SecretKeySpec secretKeySpec = new SecretKeySpec(key.getBytes(StandardCharsets.UTF_8), HMAC_SHA1);
            Mac mac = Mac.getInstance(HMAC_SHA1);
            mac.init(secretKeySpec);
            mac.update(data.getBytes(StandardCharsets.UTF_8));
            return bytesToHexString(mac.doFinal(nonce.getBytes(StandardCharsets.UTF_8)));
        } catch (NoSuchAlgorithmException | InvalidKeyException e) {
            throw new RuntimeException("Sign error!");
        }
    }

    private static String bytesToHexString(byte[] bytes) {
        StringBuilder builder = new StringBuilder(bytes.length * 2);
        for (byte data : bytes) {
            builder.append(String.format("%02x", data & 0xff));
        }
        return builder.toString();
    }

    private static byte[] hexStringToBytes(String str) {
        byte[] bytes = new byte[str.length() / 2];
        for(int i = 0; i < str.length() / 2; i++) {
            String subStr = str.substring(i * 2, i * 2 + 2);
            bytes[i] = (byte) Integer.parseInt(subStr, 16);
        }

        return bytes;
    }

    public static void main(String[] args) throws UnsupportedEncodingException {
        String key = "xxxhhhsshkjkkddd";
        JSONObject jsonObject = new JSONObject();
        jsonObject.put("id", 123);
        jsonObject.put("name", "測試姓名");
        jsonObject.put("alias", "tom");

        String param = URLEncoder.encode(jsonObject.toJSONString(), "utf-8");
        System.out.println("param: " + param);

        int nonce = new SecureRandom().nextInt();
        String data = String.format("apiCode=%s&data=%s&nonce=%d", "createUser#XDFFDDD", param, nonce);
        System.out.println("data: " + data);

        data = encrypt(key, data);
        System.out.println("encrypt data: " + data);

        String sign = sign(key, data, nonce + "");
        System.out.println("sign: " + sign);

        System.out.println("key: " + key);
        String decryptData = decrypt(key, data);
        System.out.println("DecryptData: " + decryptData);
    }
}

源碼:https://github.com/dachengxi/APIGateway
原文鏈接:https://cxis.me/2020/04/08/APIGateway%E4%B8%AD%E5%8A%A0%E5%AF%86%E9%AA%8C%E7%AD%BE%E4%BB%8B%E7%BB%8D/

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市尤勋,隨后出現(xiàn)的幾起案子喘落,更是在濱河造成了極大的恐慌,老刑警劉巖最冰,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件瘦棋,死亡現(xiàn)場離奇詭異,居然都是意外死亡暖哨,警方通過查閱死者的電腦和手機(jī)赌朋,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來篇裁,“玉大人沛慢,你說我怎么就攤上這事〈锊迹” “怎么了团甲?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵,是天一觀的道長往枣。 經(jīng)常有香客問我伐庭,道長粉渠,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任圾另,我火速辦了婚禮霸株,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘集乔。我一直安慰自己去件,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般铅辞。 火紅的嫁衣襯著肌膚如雪酸休。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天投慈,我揣著相機(jī)與錄音,去河邊找鬼。 笑死授霸,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的际插。 我是一名探鬼主播碘耳,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼框弛!你這毒婦竟也來了辛辨?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤瑟枫,失蹤者是張志新(化名)和其女友劉穎斗搞,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體力奋,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡榜旦,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了景殷。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片溅呢。...
    茶點(diǎn)故事閱讀 37,997評論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖猿挚,靈堂內(nèi)的尸體忽然破棺而出咐旧,到底是詐尸還是另有隱情,我是刑警寧澤绩蜻,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布铣墨,位于F島的核電站,受9級特大地震影響办绝,放射性物質(zhì)發(fā)生泄漏伊约。R本人自食惡果不足惜姚淆,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望屡律。 院中可真熱鬧腌逢,春花似錦、人聲如沸超埋。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽霍殴。三九已至媒惕,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間来庭,已是汗流浹背妒蔚。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留月弛,地道東北人面睛。 一個(gè)月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓,卻偏偏與公主長得像尊搬,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子土涝,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,722評論 2 345