Android 关于 OkHttp 请求对参数进行加解密的封装

数据格式

不加密的情况下,数据一般是这样的(当然,data 也可能是一个列表):

请求:
{
    "id": 88
}

返回:
{
    "code": 200,
    "data": {
        "name": "Rose"
    },
    "message": "success"
}

加密的情况下:

请求:
{
    "encryptKey": "xxx",
    "encryptValue": "yyy"
}

返回:
{
    "code": 200,
    "data": {
        "encryptKey": "xxx",
        "encryptValue": "yyy"
    },
    "message": "success"
}

加解密流程:

  1. 发起请求(加密)
  • 获取一个含字符和数字的随机字符串(比如16位)key
  • 使用 AES 加密将 RequestBody 进行加密,秘钥为上面的 key,得到 encryptValue
  • 使用 秘钥对1 的公钥对 key 进行 RSA 加密,得到 encryptKey
  • 使用 encryptKeyencryptValue 生成新的 RequestBody 发起网络请求
  1. 接收返回(解密)
  • 常规解析得到 data.encryptKeydata.encryptValue
  • 使用 秘钥对2 的私钥对 data.encryptKey 进行 RSA 解密,得到 key
  • 使用 AES 解密将 data.encryptValue 进行解密,秘钥为上面的 key,得到解密后的实体
  1. 为什么不直接用 RSA 加解密,因为 AES 效率更高

封装 - 使用 interceptor 的方式

  • EncryptInterceptor.kt
class EncryptInterceptor : Interceptor {
    override fun intercept(chain: Interceptor.Chain): Response {
        val request = chain.call().request()
        var newRequest = request

        request.body()?.let {
            val buffer = Buffer()
            it.writeTo(buffer)
            val requestBodyStr = buffer.readString(Charsets.UTF_8)
            if (!TextUtils.isEmpty(requestBodyStr)) {
                // encrypt data here:
                // 1. get random alpha-numeric string, for example "1abcd234"
                // 2. use RSA with public key to encrypt the alpha-numeric string, this is the encryptKey
                // 3. use AES with the alpha-numeric string to encrypt the body, this is the encryptValue
                val randomString = MyUtil.getAlphaNumericString()
                val encryptKey = MyUtil.rsaEncrypt(randomString, "MIGXXX...XXX")
                val encryptValue = MyUtil.aesEncrypt(requestBodyStr, randomString)

                val encryptJson = GsonUtils.toJson(EncryptData(encryptKey, encryptValue))
                val encryptBody = RequestBody.create(
                    MediaType.parse("application/json; charset=utf-8"),
                    encryptJson
                )
                newRequest = request.newBuilder().post(encryptBody).build()
            }
        }

        // decrypt the response body vice versa.
        val response = chain.proceed(newRequest)
        response.body()?.let { responseBody ->
            val source: BufferedSource = responseBody.source()
            // Buffer the entire body(into source.buffer).
            source.request(Long.MAX_VALUE)
            // Must use clone, otherwise if we not create new
            // ResponseBody(for example result.data is null), the buffer will be empty,
            // after this interceptor returns, other places will get nothing with this buffer.
            val responseBodyStr = source.buffer.clone().readString(StandardCharsets.UTF_8)
            val typeToken = object : TypeToken<Result<EncryptData>>() {}.type
            val result = GsonUtils.fromJson<Result<EncryptData>>(responseBodyStr, typeToken)
            if (result.data != null) {
                val aesKey = MyUtil.rsaDecrypt(result.data.encryptKey, "MIGXXX...XXX")
                val decrypt = MyUtil.aesDecrypt(result.data.encryptValue, aesKey)
                val newResultJson =
                    """{"message":"${result.message}","code":${result.code},"data":$decrypt}""".trimIndent()
                // new instance of ResponseBody created here, should close the origin one.
                responseBody.use {
                    return response.newBuilder()
                        .body(
                            ResponseBody.create(
                                MediaType.parse("application/json"),
                                newResultJson
                            )
                        )
                        .build()
                }
            }
        }

        return response
    }
}
  • EncryptData.kt
data class EncryptData(val encryptKey: String, val encryptValue: String)
  • Result.kt
data class Result<T>(val message: String?, val code: Int, val data: T?)

封装 - 使用 converter 的方式

  • EncryptConverterFactory.kt
class EncryptConverterFactory : Converter.Factory() {

    companion object {
        fun create(): EncryptConverterFactory = EncryptConverterFactory()
    }

    override fun requestBodyConverter(
        type: Type,
        parameterAnnotations: Array<out Annotation>,
        methodAnnotations: Array<out Annotation>,
        retrofit: Retrofit
    ): Converter<*, RequestBody>? {
        return EncryptRequestBodyConverter()
    }

    override fun responseBodyConverter(
        type: Type,
        annotations: Array<out Annotation>,
        retrofit: Retrofit
    ): Converter<ResponseBody, *>? {
        if (type is ParameterizedType) {
            return EncryptResponseBodyConverter(type.actualTypeArguments[0])
        }

        return null
    }

    internal class EncryptRequestBodyConverter : Converter<Any, RequestBody> {
        override fun convert(value: Any): RequestBody? {
            val requestBodyStr = GsonUtils.toJson(value)

            val randomString = MyUtil.getAlphaNumericString()
            val encryptKey = MyUtil.rsaEncrypt(randomString, "MIGXXX...XXX")
            val encryptValue = MyUtil.aesEncrypt(requestBodyStr, randomString)

            val encryptJson = GsonUtils.toJson(EncryptData(encryptKey, encryptValue))
            return RequestBody.create(
                MediaType.parse("application/json; charset=utf-8"),
                encryptJson
            )
        }
    }

    internal class EncryptResponseBodyConverter(private val innerType: Type) :
        Converter<ResponseBody, Result<Any>> {
        @Throws(IOException::class)
        override fun convert(value: ResponseBody): Result<Any> {
            value.use {
                val originalStr = value.charStream()
                val result = GsonUtils.fromJson<Result<EncryptData>>(
                    originalStr,
                    object : TypeToken<Result<EncryptData>>() {}.type
                )
                return if (result.data != null) {
                    val aesKey = MyUtil.rsaDecrypt(result.data.encryptKey, "MIGXXX...XXX")
                    val decrypt = MyUtil.aesDecrypt(result.data.encryptValue, aesKey)
                    val newResult = GsonUtils.fromJson<Any>(decrypt, innerType)
                    Result(result.message, result.code, newResult)
                } else {
                    Result(result.message, result.code, null)
                }
            }
        }
    }

}

封装 - 改进 converter

  • EncryptConverterFactory.kt
class EncryptConverterFactory : Converter.Factory() {

    companion object {
        fun create(): EncryptConverterFactory = EncryptConverterFactory()
    }

    override fun requestBodyConverter(
        type: Type,
        parameterAnnotations: Array<out Annotation>,
        methodAnnotations: Array<out Annotation>,
        retrofit: Retrofit
    ): Converter<*, RequestBody>? {
        return EncryptRequestBodyConverter()
    }

    override fun responseBodyConverter(
        type: Type,
        annotations: Array<out Annotation>,
        retrofit: Retrofit
    ): Converter<ResponseBody, *>? {
        return EncryptResponseBodyConverter(type)
    }

    internal class EncryptRequestBodyConverter : Converter<Any, RequestBody> {
        override fun convert(value: Any): RequestBody? {
            val requestBodyStr = GsonUtils.toJson(value)

            val randomString = MyUtil.getAlphaNumericString()
            val encryptKey = MyUtil.rsaEncrypt(randomString, "MIGXXX...XXX")
            val encryptValue = MyUtil.aesEncrypt(requestBodyStr, randomString)

            val encryptJson = GsonUtils.toJson(EncryptData(encryptKey, encryptValue))
            return RequestBody.create(
                MediaType.parse("application/json; charset=utf-8"),
                encryptJson
            )
        }
    }

    internal class EncryptResponseBodyConverter(private val type: Type) :
        Converter<ResponseBody, Any> {
        @Throws(IOException::class)
        override fun convert(value: ResponseBody): Any {
            try {
                val originalStr = value.charStream()
                val originalMap = GsonUtils.fromJson<Any>(
                    originalStr,
                    Object::class.java
                ) as MutableMap<String, Any>
                originalMap["data"]?.let { data ->
                    val dataStr = GsonUtils.toJson(data)
                    val encryptData = GsonUtils.fromJson<EncryptData>(
                        dataStr, object : TypeToken<EncryptData>() {}.type
                    )
                    val aesKey = MyUtil.rsaDecrypt(encryptData.encryptKey, "MIGXXX...XXX")
                    val decrypt = MyUtil.aesDecrypt(encryptData.encryptValue, aesKey)
                    originalMap["data"] =
                        GsonUtils.fromJson<Any>(decrypt, Object::class.java)
                }

                val newStr = GsonUtils.toJson(originalMap)
                return GsonUtils.fromJson(newStr, type)
            } finally {
                value.close()
            }
        }
    }

}

这里不强制要求返回类型是 Result,只需要服务端返回的 json 数据的第一层级中有 "data" 字段即可,取出该字段进行解密,并重新赋值该字段,然后再进行解析

比较

interceptor 的方式效率要高点,但是 converter 的方式要更加灵活点。

补充

这里采用了把 encryptKey 也放在 body 里一起传输的方案,所以实现起来有点麻烦;其实也可以选择把 encryptKey 放在 header 里,然后对 body 整体加解密的方案,这样实现起来就会简单一些

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352

推荐阅读更多精彩内容