Java—通過sign簽名認證實現(xiàn)安全的開放接口API

關注WC:CodingTechWork,一起學習進步。

@[toc]

引言

??在開發(fā)Java API時容诬,有可能需要一些安全認證來保證API的安全性。下面沿腰,我們通過timestamp以及sign簽名認證來實現(xiàn)基本的安全的開發(fā)接口API览徒。

API接口

curl {ip}:{port}/demo/test/app -X POST -H "Content-Type:application/json" -d '
{
    "appId":"app01",
    "key01": "value01",
    "key02": "value02,
    "key03": "value03",
    "sign": "77f4d9119d43c41491ab8a7c0530a618d6e0823560eb6dd028f62ae56b1ab7bb",
    "timestamp":165820977
}'

timestamp保證唯一性

通過timestamp校驗,保證約定時間內請求的唯一性

shell生成timestamp

  1. shell中生成秒級timestamp
echo $(date '+%s')
  1. shell中生成毫秒級timestamp
# 通過納秒除以1000000的方式實現(xiàn)
echo $[$(date '+%s%N')/1000000]

java生成timestamp模板

// 得到秒級
long currentTimeStamp = System.currentTimeMillis() / 1000;

sign簽名認證

實現(xiàn)方式

  1. app服務端配置對應appSecret, 例如app123456
  2. 所有請求參數按照字母排序, 配裝字符串, 采用key=value并用&鏈接
  3. 添加簽名appSecret, 字符末尾加上appSecret=app123456
  4. 采用SHA-256生產簽名
  5. 例:
    appId=app01&params=key01=value01;key02=value02;key03=value03&appSecret=app123456

shell生成sha-256值

echo -n "appId=app01&params=key01=value01;key02=value02;key03=value03&appSecret=app123456" | sha256sum | awk '{print $1}'

結果

77f4d9119d43c41491ab8a7c0530a618d6e0823560eb6dd028f62ae56b1ab7bb

java生成sha-256模板


package com.demo.test

import java.io.UnsupportedEncodingException;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Base64;

import org.apache.commons.lang.StringUtils;

import lombok.extern.slf4j.Slf4j;

@Slf4j
public class Encrypt {

    /**
     * 返回 SHA-256 串
     *
     * @param strText
     * @return
     */
    public static String SHA256(final String strText) {
        return SHA(strText, "SHA-256");
    }

    /**
     * 返回 SHA-512 串
     *
     * @param strText
     * @return
     */
    public static String SHA512(final String strText) {
        return SHA(strText, "SHA-512");
    }

    public static String MD5(final String strText) {
        return SHA(strText, "MD5");
    }

    /**
     * SHA 加密
     *
     * @param strText
     * @return
     */
    private static String SHA(final String strText, final String type) {
        String ret = null;

        if (strText != null && strText.length() > 0) {
            try {
                MessageDigest messageDigest = MessageDigest.getInstance(type);
                messageDigest.update(strText.getBytes());
                byte[] buffer = messageDigest.digest();

                StringBuffer sb = new StringBuffer();
                for (int i = 0; i < buffer.length; i++) {
                    String hex = Integer.toHexString(0xff & buffer[i]);
                    if (hex.length() == 1) {
                        sb.append('0');
                    }
                    sb.append(hex);
                }
                ret = sb.toString();
            } catch (NoSuchAlgorithmException e) {
                log.error("{} encrypt failed.{}", type, e);
            }
        }
        return ret;
    }

    /**
     * 手機號脫敏處理
     *
     * @param phone
     * @return
     */
    public static String desensitizedPhone(String phone) {
        if (StringUtils.isNotEmpty(phone)) {
            phone = phone.replaceAll("(\\w{3})\\w*(\\w{4})", "$1****$2");
        }
        return phone;
    }

    /**
     * base64加密
     *
     * @param content
     * @return
     */
    public static String base64Encoder(String content) {
        try {
            String base64encodedString = Base64.getEncoder().encodeToString(content.getBytes("utf-8"));
            return base64encodedString;
        } catch (UnsupportedEncodingException e) {
            log.error("base64Encoder failed", e);
        }
        return null;
    }
    /**
     * base64解密
     *
     * @param content
     * @return
     */
    public static String base64Decoder(String content) {
        try {
            // 解碼
            byte[] base64decodedBytes = Base64.getDecoder().decode(content);
            return new String(base64decodedBytes, "utf-8");
        } catch (UnsupportedEncodingException e) {
            log.error("base64Decoder failed", e);
        }
        return null;
    }

}

?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末颂龙,一起剝皮案震驚了整個濱河市习蓬,隨后出現(xiàn)的幾起案子纽什,更是在濱河造成了極大的恐慌,老刑警劉巖友雳,帶你破解...
    沈念sama閱讀 219,490評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件稿湿,死亡現(xiàn)場離奇詭異,居然都是意外死亡押赊,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評論 3 395
  • 文/潘曉璐 我一進店門包斑,熙熙樓的掌柜王于貴愁眉苦臉地迎上來流礁,“玉大人,你說我怎么就攤上這事罗丰∩袼В” “怎么了?”我有些...
    開封第一講書人閱讀 165,830評論 0 356
  • 文/不壞的土叔 我叫張陵萌抵,是天一觀的道長找御。 經常有香客問我,道長绍填,這世上最難降的妖魔是什么霎桅? 我笑而不...
    開封第一講書人閱讀 58,957評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮讨永,結果婚禮上滔驶,老公的妹妹穿的比我還像新娘。我一直安慰自己卿闹,他們只是感情好揭糕,可當我...
    茶點故事閱讀 67,974評論 6 393
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著锻霎,像睡著了一般著角。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上旋恼,一...
    開封第一講書人閱讀 51,754評論 1 307
  • 那天吏口,我揣著相機與錄音,去河邊找鬼蚌铜。 笑死锨侯,一個胖子當著我的面吹牛,可吹牛的內容都是我干的冬殃。 我是一名探鬼主播囚痴,決...
    沈念sama閱讀 40,464評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼审葬!你這毒婦竟也來了深滚?” 一聲冷哼從身側響起奕谭,我...
    開封第一講書人閱讀 39,357評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎痴荐,沒想到半個月后血柳,有當地人在樹林里發(fā)現(xiàn)了一具尸體,經...
    沈念sama閱讀 45,847評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡生兆,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,995評論 3 338
  • 正文 我和宋清朗相戀三年难捌,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片鸦难。...
    茶點故事閱讀 40,137評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡根吁,死狀恐怖,靈堂內的尸體忽然破棺而出合蔽,到底是詐尸還是另有隱情击敌,我是刑警寧澤,帶...
    沈念sama閱讀 35,819評論 5 346
  • 正文 年R本政府宣布拴事,位于F島的核電站沃斤,受9級特大地震影響,放射性物質發(fā)生泄漏刃宵。R本人自食惡果不足惜衡瓶,卻給世界環(huán)境...
    茶點故事閱讀 41,482評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望组去。 院中可真熱鬧鞍陨,春花似錦、人聲如沸从隆。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,023評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽键闺。三九已至寿烟,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間辛燥,已是汗流浹背筛武。 一陣腳步聲響...
    開封第一講書人閱讀 33,149評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留挎塌,地道東北人徘六。 一個月前我還...
    沈念sama閱讀 48,409評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像榴都,于是被迫代替她去往敵國和親待锈。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,086評論 2 355

推薦閱讀更多精彩內容