OkHttp Interceptor日志上报

最近为了做一些网络上的优化,所以就得提前埋点,为后续网络优化提供数据支持。

主要是对发起请求埋点,请求错误埋点,客户端请求耗时埋点。

事件上报到阿里云,接入的是阿里的应用实时监控服务。

网络请求使用的是OhHttp + Retrofit ,现在很多都是这么实现的了吧。

网络的埋点切入口,很容易就想到了 Interceptor,用起来了才觉得 Interceptor 是真的香,以前还只是觉得这个设计模式是真好。用到了才感觉是太香了。

上代码:

kotlin 复制代码
class ReportingInterceptor(val source: ACRSource) : Interceptor {

    private val gson = Gson()

    override fun intercept(chain: Interceptor.Chain): Response {
        val request = chain.request()
        val url = request.url.toString()

        AlibabaCloudRumUtil.requestStart(source, snapshots = url)

        val startTime = System.currentTimeMillis()
        val response = chain.proceed(request)

        AlibabaCloudRumUtil.requestCost(
            source,
            snapshots = url,
            System.currentTimeMillis() - startTime
        )

        // 判断是否是 HTTP 错误
        if (!response.isSuccessful) {
            AlibabaCloudRumUtil.requestFail(
                source,
                snapshots = url,
                params = AlibabaCloudRumUtil.errorMap(
                    response.code.toString(),
                    response.message
                )
            )
        } else {
            val source = response.body?.source()
            source?.request(Long.MAX_VALUE)
            try {
                val buffer = source?.buffer
                // 业务错误逻辑上报
                val result = gson.fromJson(
                    buffer?.clone()?.readString(Charset.forName("UTF-8")),
                    BaseModel::class.java
                )
                if (result?.code != 0) {
                    AlibabaCloudRumUtil.requestFail(
                        ACRSource.COROUTINE,
                        snapshots = url,
                        params = ResultException(
                            "${result.code}",
                            result?.message ?: "Unknown error"
                        ).toMap()
                    )
                }
            } catch (ex: Exception) {
                AlibabaCloudRumUtil.requestFail(
                    ACRSource.COROUTINE,
                    snapshots = url,
                    params = BusinessException.handlerException(ex).toMap()
                )
            } finally {
                source?.close()
            }
        }

        return response
    }
}

网络请求上报是带上了URL,现在的Retrofit 的URL 在其他地方是真的不好拿,这 Interceptor 确实方便的很。

而网络耗时也比较容易,在

val response = chain.proceed(request)

前后记录时间就可以了,虽然跟抓包数据显示的网络耗时有差距,但是我们毕竟只是粗略的统计而已。

错误上报的话,包括了 http 请求错误和 code != 0 的业务数据错误,都是需要上报的。

如果需要在其他地方拿到 URL ,需要在 Interceptor 里处理,重新 new 一个 Response 返回,然后把 URL 塞到 Header 里面,在其他地方把 Header 中的 URL 读取出来。但是会产生一点性能开销,毕竟每个请求都需要 new 一个新的 Response 。

相关推荐
Jeled21 小时前
Retrofit 与 OkHttp 全面解析与实战使用(含封装示例)
android·okhttp·android studio·retrofit
Jeled2 天前
Android 网络层最佳实践:Retrofit + OkHttp 封装与实战
android·okhttp·kotlin·android studio·retrofit
allk555 天前
OkHttp源码解析(一)
android·okhttp
allk555 天前
OkHttp源码解析(二)
android·okhttp
aFakeProgramer5 天前
拆分PDF.html 办公小工具
okhttp
一壶浊酒..6 天前
ajax局部更新
前端·ajax·okhttp
洛克大航海9 天前
Ajax基本使用
java·javascript·ajax·okhttp
whltaoin15 天前
Java 网络请求 Jar 包选型指南:从基础到实战
java·http·okhttp·网络请求·retrofit
华农第一蒟蒻16 天前
谈谈跨域问题
java·后端·nginx·安全·okhttp·c5全栈
一直向钱18 天前
android 基于okhttp的socket封装
android·okhttp