Android Okhttp/Retrofit網(wǎng)絡(luò)請求加解密實現(xiàn)方案

一败潦、加密方案

比較安全的方案應(yīng)該是AES+RSA的加密方式。具體如下圖所示准脂。


AES+RSA流程

為什么要這樣做呢劫扒?
1、RSA是非對稱加密狸膏,公鑰和私鑰分開粟关,且公鑰可以公開,很適合網(wǎng)絡(luò)數(shù)據(jù)傳輸場景。但RSA加密比較慢闷板,據(jù)說比AES慢100倍澎灸,且對加密的數(shù)據(jù)長度也有限制。
2遮晚、AES是對稱加密性昭,加密速度快,安全性高县遣,但密鑰的保存是個問題糜颠,在網(wǎng)絡(luò)數(shù)據(jù)傳輸?shù)膱鼍熬秃苋菀子捎诿荑€泄露造成安全隱患
3、所以萧求,AES+RSA結(jié)合才更好其兴,AES加密數(shù)據(jù),且密鑰隨機生成夸政,RSA用對方(服務(wù)器)的公鑰加密隨機生成的AES密鑰元旬。傳輸時要把密文,加密的AES密鑰和自己的公鑰傳給對方(服務(wù)器)守问。對方(服務(wù)器)接到數(shù)據(jù)后匀归,用自己的私鑰解密AES密鑰,再拿AES密鑰解密數(shù)據(jù)得到明文耗帕。這樣就綜合了兩種加密體系的優(yōu)點穆端。
4、除上面說的外仿便,還可以加簽名体啰,即對傳輸?shù)臄?shù)據(jù)(加密前)先做個哈希,然后用自己的RSA私鑰對哈希簽名(對方拿到自己的公鑰可以驗簽)嗽仪,這樣可以驗證傳輸內(nèi)容有沒有被修改過荒勇。

二、加密相關(guān)的一些坑

1钦幔、數(shù)據(jù)類型

就java來說,加密的輸入和輸出都是字節(jié)數(shù)組類型的常柄,也就是二進制數(shù)據(jù)鲤氢,網(wǎng)絡(luò)傳輸或本地保存都需要重新編碼為字符串。推薦使用Base64西潘。Android 有自帶的Base64實現(xiàn)卷玉,flag要選Base64.NO_WRAP,不然末尾會有換行影響服務(wù)端解碼喷市。
Android中Base64加密

//字節(jié)數(shù)組轉(zhuǎn)字符串
String str = Base64.encodeToString(byte_data, Base64.NO_WRAP);
//字符串轉(zhuǎn)字節(jié)數(shù)組
byte[] bytes = Base64.decode(keyStr, Base64.NO_WRAP);

2相种、加密參數(shù)

總而言之,這些不同語言都有實現(xiàn)庫品姓,調(diào)用即可寝并,關(guān)鍵是參數(shù)要一致箫措,具體還需要和后臺聯(lián)調(diào)一下。
rsa加解密的內(nèi)容超長的問題解決

AES算法:
Android端--->"AES/CFB/NOPADDING"
密鑰長度一般128衬潦,256安全性更高
ECB模式不安全斤蔓,使用會有黃色警告。

RSA算法:
密鑰長度=1024已經(jīng)被認為不安全了(RSA 768已于2009年被破解)镀岛,推薦>=2048弦牡。加密的明文長度和密鑰長度是相關(guān)的。
Android端-->"RSA/ECB/PKCS1Padding"

RSA簽名算法:
Android端-->"SHA1withRSA"
試過MD5withRSA漂羊,但是和后臺無法兼容

三驾锰、OkHttp/Retrofit的實現(xiàn)

現(xiàn)在說到網(wǎng)絡(luò)框架,應(yīng)該毫無疑問是Retrofit了走越。上面說的加密方案說到底還是要在網(wǎng)絡(luò)請求框架內(nèi)加上椭豫,怎么做入侵最小,怎么做最方便才是重點买喧。
1捻悯、坑定不能直接在接口調(diào)用層做加密嘹黔,加參數(shù)巨缘,這樣每個接口都要修改,這是不可能的抛蚤。
2低淡、ConverterFactory處理姓言,這也是網(wǎng)上可以搜到的很多文章的寫法,但我覺得還是有入侵蔗蹋。而且有點麻煩何荚。
3、OkHttp添加攔截器猪杭,這種方法入侵最胁吞痢(可以說沒有),實現(xiàn)呢也非常優(yōu)雅皂吮。
下面的實現(xiàn)戒傻,網(wǎng)上也找不到多少可以參考的文章,但不得不說蜂筹,OkHttp的封裝和設(shè)計真的很好用需纳,所見即所得∫张玻看下源碼不翩,就知道該怎么用了,連文檔都不用查。

-----------------------------------------------------------------------------------------------------------------------------------------------
----->先定義一個攔截器的實現(xiàn):
-----------------------------------------------------------------------------------------------------------------------------------------------

public class DataEncryptInterceptor implements Interceptor {
        @Override
        public Response intercept(Chain chain) throws IOException {
            //請求
            Request request = chain.request();
            RequestBody oldRequestBody = request.body();
            Buffer requestBuffer = new Buffer();
            oldRequestBody.writeTo(requestBuffer);
            String oldBodyStr = requestBuffer.readUtf8();
            requestBuffer.close();
            MediaType mediaType = MediaType.parse("text/plain; charset=utf-8");
            //生成隨機AES密鑰并用serverPublicKey進行RSA加密
            SecretKeySpec appAESKeySpec = EncryptUtils.generateAESKey(256);
            String appAESKeyStr = EncryptUtils.covertAESKey2String(appAESKeySpec);
            String appEncryptedKey = RSAUtils.encryptDataString(appAESKeyStr, serverPublicKey);
            //計算body 哈希 并使用app私鑰RSA簽名
            String appSignature = RSAUtils.signature(oldBodyStr, appPrivateKey);
            //隨機AES密鑰加密oldBodyStr
            String newBodyStr = EncryptUtils.encryptAES(appAESKeySpec, oldBodyStr);
            RequestBody newBody = RequestBody.create(mediaType, newBodyStr);
            //構(gòu)造新的request
            request = request.newBuilder()
                    .header("Content-Type", newBody.contentType().toString())
                    .header("Content-Length", String.valueOf(newBody.contentLength()))
                    .method(request.method(), newBody)
                    .header("appEncryptedKey", appEncryptedKey)
                    .header("appSignature", appSignature)
                    .header("appPublicKey", appPublicKeyStr)
                    .build();
            //響應(yīng)
            Response response = chain.proceed(request);
            if (response.code() == 200) {//只有約定的返回碼才經(jīng)過加密口蝠,才需要走解密的邏輯
                //獲取響應(yīng)頭
                String serverEncryptedKey = response.header("serverEncryptedKey");
                //用app的RSA私鑰解密AES加密密鑰
                String serverDecryptedKey = RSAUtils.decryptDataString(serverEncryptedKey, appPrivateKey);
                SecretKeySpec serverAESKeySpec = EncryptUtils.covertString2AESKey(serverDecryptedKey);
                //用AES密鑰解密oldResponseBodyStr
                ResponseBody oldResponseBody = response.body();
                String oldResponseBodyStr = oldResponseBody.string();
                String newResponseBodyStr = EncryptUtils.decryptAES(serverAESKeySpec, oldResponseBodyStr);
                oldResponseBody.close();
                //構(gòu)造新的response
                ResponseBody newResponseBody = ResponseBody.create(mediaType, newResponseBodyStr);
                response = response.newBuilder().body(newResponseBody).build();
            }
            response.close();
            //返回
            return response;
        }
    }

-----------------------------------------------------------------------------------------------------------------------------------------------
----->然后OkHttp加入該攔截器:
-----------------------------------------------------------------------------------------------------------------------------------------------

new OkHttpClient.Builder()
                .addInterceptor(new DataEncryptInterceptor())
                ....
                .build();

-----------------------------------------------------------------------------------------------------------------------------------------------
----->這樣就搞定了器钟。
-----------------------------------------------------------------------------------------------------------------------------------------------

主要注意點:
0、和接口無關(guān)的新加的數(shù)據(jù)放在請求頭里亚皂。
1俱箱、該close的要close,不然會內(nèi)存泄漏灭必。
2狞谱、新舊Request和Response要區(qū)分好,新的要替換舊的去傳遞或返回禁漓。
3跟衅、要對response.code()做處理,只有在和后臺約定好的返回碼下才走解密的邏輯播歼,具體看自己的需求伶跷,不一定都是200。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末秘狞,一起剝皮案震驚了整個濱河市叭莫,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌烁试,老刑警劉巖雇初,帶你破解...
    沈念sama閱讀 217,509評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異减响,居然都是意外死亡靖诗,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評論 3 394
  • 文/潘曉璐 我一進店門支示,熙熙樓的掌柜王于貴愁眉苦臉地迎上來刊橘,“玉大人,你說我怎么就攤上這事颂鸿〈倜啵” “怎么了?”我有些...
    開封第一講書人閱讀 163,875評論 0 354
  • 文/不壞的土叔 我叫張陵嘴纺,是天一觀的道長败晴。 經(jīng)常有香客問我,道長颖医,這世上最難降的妖魔是什么位衩? 我笑而不...
    開封第一講書人閱讀 58,441評論 1 293
  • 正文 為了忘掉前任裆蒸,我火速辦了婚禮熔萧,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己佛致,他們只是感情好贮缕,可當我...
    茶點故事閱讀 67,488評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著俺榆,像睡著了一般感昼。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上罐脊,一...
    開封第一講書人閱讀 51,365評論 1 302
  • 那天定嗓,我揣著相機與錄音,去河邊找鬼萍桌。 笑死宵溅,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的上炎。 我是一名探鬼主播恃逻,決...
    沈念sama閱讀 40,190評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼藕施!你這毒婦竟也來了寇损?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,062評論 0 276
  • 序言:老撾萬榮一對情侶失蹤裳食,失蹤者是張志新(化名)和其女友劉穎矛市,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體胞谈,經(jīng)...
    沈念sama閱讀 45,500評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡尘盼,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,706評論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了烦绳。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片卿捎。...
    茶點故事閱讀 39,834評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖径密,靈堂內(nèi)的尸體忽然破棺而出午阵,到底是詐尸還是另有隱情,我是刑警寧澤享扔,帶...
    沈念sama閱讀 35,559評論 5 345
  • 正文 年R本政府宣布底桂,位于F島的核電站,受9級特大地震影響惧眠,放射性物質(zhì)發(fā)生泄漏籽懦。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,167評論 3 328
  • 文/蒙蒙 一氛魁、第九天 我趴在偏房一處隱蔽的房頂上張望暮顺。 院中可真熱鬧厅篓,春花似錦、人聲如沸捶码。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,779評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽惫恼。三九已至档押,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間祈纯,已是汗流浹背令宿。 一陣腳步聲響...
    開封第一講書人閱讀 32,912評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留腕窥,地道東北人掀淘。 一個月前我還...
    沈念sama閱讀 47,958評論 2 370
  • 正文 我出身青樓,卻偏偏與公主長得像油昂,于是被迫代替她去往敵國和親革娄。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,779評論 2 354

推薦閱讀更多精彩內(nèi)容

  • 本文主要介紹移動端的加解密算法的分類冕碟、其優(yōu)缺點特性及應(yīng)用拦惋,幫助讀者由淺入深地了解和選擇加解密算法。文中會包含算法的...
    蘋果粉閱讀 11,504評論 5 29
  • 隨著對于安全度的不斷要求言秸,對于數(shù)據(jù)加解密與破解之間的斗爭,加解密的方式也在不斷發(fā)生著變化迎捺,來看看現(xiàn)在流行的一些加解...
    zhouhao_180閱讀 2,083評論 1 12
  • 最近公司用到RSA數(shù)據(jù)加密傳輸举畸,本人也只會使用,并不知其原理凳枝,剛好今天在csdn看到一位大牛的博客寫得很到位抄沮,遂搬...
    爸比好酷閱讀 1,418評論 0 1
  • 馬上暑假啦是否還在愁青銅白銀的小學(xué)生坑到你呢?[em]e400837[/em] 這里在線接LOL代練岖瑰。安全效率叛买。價...
    沒有故事的淡忘閱讀 252評論 1 1
  • 君君喜歡吃重慶辣子雞,今天說起要做這個菜蹋订,朵朵說她是高手率挣。嗯哼,跟高手學(xué)習(xí)學(xué)習(xí)露戒。認真聽清楚朵朵指點辣子雞椒功,再...
    七七行記閱讀 265評論 0 0