實現fastdfs防盜鏈功能

2022-10-09 21:00:25

1、背景

我們可以通過fastdfs實現一個分散式檔案系統,如果我們的fastdfs部署在外網,那麼任何一個人知道了我們的上傳介面,那麼它就可以檔案的上傳和存取。那麼我們如何阻止他人存取我們fastdfs伺服器上的檔案呢?此處就需要使用fastdfs的防盜鏈功能。

2、實現原理

fastdfs的防盜鏈是通過token機制來實現的。當我們開啟防盜鏈功能後,需要在url後增加2個額外的引數tokentstokents的生成都是需要在伺服器端。

2.1 開啟防盜鏈

vim /etc/fdfs/http.conf

# true 表示開啟防盜鏈
http.anti_steal.check_token = true
# token的過期時間,單位為秒
http.anti_steal.token_ttl = 60
# 金鑰,不可洩漏,用於生成token
http.anti_steal.secret_key = thisisasecuritykey
# 當圖片拒絕存取後,顯示的圖片,此圖片需要可存取,不然可能會出現問題
http.anti_steal.token_check_fail = /data/fastdfs/401.jpg

http.anti_steal.token_check_fail 指定的圖片需要可存取,否則可能會出現問題

2.2 重啟 nginx

/usr/local/nginx/sbin/nginx -s reload

2.3 Java程式碼生成token

1、token生成規則

token = md5(檔案ID+私鑰+時間戳)

檔案ID:不能包含group

group1/M00/00/00/wKh5iWNBl7-AKvj1AAAwWD4VeAg577.jpg
`需要替換成`
M00/00/00/wKh5iWNBl7-AKvj1AAAwWD4VeAg577.jpg

私鑰:需要和 /etc/fdfs/http.conf 中的 http.anti_steal.secret_key 值一致
時間戳:單位秒

2、java生成token

/**
     * 生成token
     *
     * @param fileId          the filename return by FastDFS server,不能含有組
     * @param timestampSecond 時間戳 單位秒
     * @return token
     * @throws NoSuchAlgorithmException
     */
    private String generatorToken(String fileId, Long timestampSecond) throws NoSuchAlgorithmException {
        // 需要去掉 group
        fileId = fileId.substring(fileId.indexOf("/") + 1);
        byte[] bsFilename = fileId.getBytes(StandardCharsets.UTF_8);
        byte[] bsTimestamp = timestampSecond.toString().getBytes(StandardCharsets.UTF_8);
        // thisisasecuritykey 需要和 /etc/fdfs/http.conf 中的 http.anti_steal.secret_key 值一致
        byte[] bsKey = "thisisasecuritykey".getBytes(StandardCharsets.UTF_8);

        byte[] buff = new byte[bsFilename.length + bsKey.length + bsTimestamp.length];
        System.arraycopy(bsFilename, 0, buff, 0, bsFilename.length);
        System.arraycopy(bsKey, 0, buff, bsFilename.length, bsKey.length);
        System.arraycopy(bsTimestamp, 0, buff, bsFilename.length + bsKey.length, bsTimestamp.length);

        return md5(buff);
    }

    public static String md5(byte[] source) throws NoSuchAlgorithmException {
        char hexDigits[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};
        java.security.MessageDigest md = java.security.MessageDigest.getInstance("MD5");
        md.update(source);
        byte tmp[] = md.digest();
        char str[] = new char[32];
        int k = 0;
        for (int i = 0; i < 16; i++) {
            str[k++] = hexDigits[tmp[i] >>> 4 & 0xf];
            str[k++] = hexDigits[tmp[i] & 0xf];
        }

        return new String(str);
    }

3、測試

3.1 帶正確token存取

3.2 帶錯誤token存取

這個地方返回的圖片是 http.anti_steal.token_check_fail = /data/fastdfs/401.jpg 這個設定中設定的圖片。

4、專案程式碼

https://gitee.com/huan1993/spring-cloud-parent/tree/master/springboot/springboot-fastdfs

5、參考連結

  1. 使用FastDFS的內建防盜鏈功能官方文章 http://bbs.chinaunix.net/thread-1916999-1-1.html
  2. fastdfs faq http://bbs.chinaunix.net/thread-1920470-1-1.html