API
网络请求
Axios
SDK支持@ohos/axios
库网络请求采集, 可以使用下面2种方式监控axios:
-
tingyun.init
时传入axios实例import tingyun from '@tingyun/sdk-core'
import axios from '@ohos/axios'
tingyun.init({
// 传入axios实例对象
axios: axios
}) -
使用
tingyun.registerAxiosInstance
注册axios实例, 可以注册多个实例API定义:
export type Tingyun = {
registerAxiosInstance: (axiosInstance: AxiosInstance) => void
}示例:
import tingyun from '@tingyun/sdk-core'
import { axiosInstance1, axiosInstance2 } from './utils/axios'
tingyun.registerAxiosInstance(axiosInstance1)
tingyun.registerAxiosInstance(axiosInstance2)
// ...
说明:
- 两种方式可以同时使用, 相同axios实例只会被注册一次
- 首次启动时, SDK默认关闭网络请求开关, 如果需要监控首次启动时的网络请求, 需要在
tingyun.init
时配置网络请求开关默认为truetingyun.init({
network: {
enabled: true
}
})
RCP
SDK通过提供拦截器的方式支持对RCP(remote communication platform)的监控。用户需要在调用createSession
时加入SDK提供的拦截器
import tingyun from '@tingyun/sdk-core'
const session = rcp.createSession({
// ...
// 加入拦截器
interceptors: [new tingyun.RCPInterceptor()]
})
SDK网络请求接口
SDK对系统@ohos.net.http
包HttpRequest.request
接口进行了封装,对外暴露了request
接口。调用这个接口发送的网络请求会被SDK采集
API定义:
import http from '@ohos.net.http';
export type HttpRequestWrapperOptions = {
url: string
options?: http.HttpRequestOptions
// http.createHttp() 返回的request对象,如果不传SDK内部会创建。不传httpRequest时,SDK内部只会在用户传入callback时在用户callback执行完后同步调用httpRequest.destroy。传入httpRequest时, SDK除了调用httpRequest.request之外不会调用其他方法。如果需要调用httpRequest其他方法、自行控制destroy时机以及使用promise方式时,需要传入httpRequest
httpRequest?: http.HttpRequest
}
export interface HttpRequestWrapper {
(options: HttpRequestWrapperOptions, callback: AsyncCallback<http.HttpResponse>): void
(options: HttpRequestWrapperOptions): Promise<http.HttpResponse>
}
export type Tingyun = {
// ...
// http request封装
request: HttpRequestWrapper
}
使用示例:
示例1: 传递URL (promise)
tingyun.request({url: 'xxx'})
.then((res) => {
// ...
})
.catch((err) => {
// ...
})
示例2: 传递URL (callback)
tingyun.request({url: 'xxx'}, (err, res) => {
// ...
})
示例3: 传递URL和options (promise)
tingyun.request({url: 'xxx', options: {
method: http.RequestMethod.POST,
header: {
'Content-Type': 'application/json'
},
extraData: {
// ...
}
}})
.then((res) => {
// ...
})
.catch((err) => {
// ...
})
示例4: 传递URL和options (callback)
tingyun.request({url: 'xxx', options: {
method: http.RequestMethod.POST,
header: {
'Content-Type': 'application/json'
},
extraData: {
// ...
}
}}, (err, res) => {
// ...
})
示例5: 使用自定义请求对象
const httpRequest = http.createHttp()
tingyun.request({
url: 'xxx',
httpRequest: httpRequest
}, (err, res) => {
// ...
})
自定义请求
为了支持用户使用的一些自定义网络请求库,SDK对外提供一套可以收集用户自身采集的网络库信息的ArkTS层API。为了满足多种使用场景, 提供两种使用模式:
-
模式1:请求发起时和请求结束后分别调用SDK API,SDK在请求结束API被调用时,收集这条网络请求
第一种模式又分为2种使用方式:
- 请求开始前调用
tingyun.startRequest
,结束后调用tingyun.endRequest
, 两次调用都传递一个自行维护的请求ID关联同一个请求。这种方式调用比较灵活,但需要自行维护一个请求ID标识一个请求,如果现有的请求库本身提供了请求ID,或者请求开始和结束的代码结构比较分散时,适合用这种方式 - 请求开始前调用
tingyun.startRequest
获取到SDK返回的请求实例,请求结束后调用请求实例的end方法结束请求。这种方式不需要自行维护请求ID,但代码结构较为复杂时,可能需要自行传递请求实例对象, 如果发送请求代码逻辑比较简单,在单一函数内完成时,适合用这种方式
- 请求开始前调用
-
模式2:请求结束后或任意时刻, 一次性调用
tingyun.createRequest
, 上报一条完整的网络请求数据
模式1: startRequest / endRequest
API定义:
export type Tingyun = {
// 开始自定义请求
startRequest: (options: RequestStartOptions) => IRequestInstance
// 结束自定义请求
endRequest: (options: RequestEndOptions) => void
}
export interface IRequestInstance {
end: (options?: RequestEndOptions) => void
clear: () => void
readonly id: string
}
export type RequestMethod = 'OPTIONS' | 'GET' | 'HEAD' | 'POST' | 'PUT' | 'DELETE' | 'TRACE' | 'CONNECT'
export type KeyValueStore<T> = Record<string, T> | Map<string, T>
export const LibNames = {
Flutter: 10,
OhosHttp: 13,
OhosAxios: 14,
OhosRCP: 15
} as const
export type LibName = typeof LibNames[keyof typeof LibNames]
export type RequestStartOptions = {
// 请求URL
url: string
// 请求ID: 请求唯一标识, 不传SDK将自动生成一个UUID格式字符串
id?: string
// 请求开始时间: 13位毫秒时间戳, 如果不传SDK将接口调用时间点作为开始点
startTime?: number
// method: 默认为GET
method?: RequestMethod
// 网络库名称
libName?: LibName
// 请求头
reqHeaders?: KeyValueStore<string>
// 请求体: 仅支持传文本类型, 如果是JSON格式需要序列化为字符串传入
reqBody?: string
// 发送字节数
bytesSent?: number
// 请求对象超时清理时间, 单位毫秒: 开始请求一段时间后如果不调用endRequest, SDK内会自动清理当前请求释放资源, 默认为90000ms
timeout?: number
}
export type RequestEndOptions = {
// 请求ID: 请求唯一标识, 当使用Tingyun.endRequest时必传,当使用请求实例对象结束时不传
id?: string
// http状态码
statusCode?: number
// 错误码: 网络错误或其他错误的错误码, 请求成功或发生http错误(4xx, 5xx), 不需要传
errorCode?: number
// 请求结束时间: 13位毫秒时间戳, 如果不传SDK将接口调用的时间点作为结束点
endTime?: number
// 请求总耗时, 单位毫秒: 如果不传使用endTime - startTime计算
duration?: number
// dns耗时, 单位毫秒: dns结束时间 - dns开始时间
dns?: number
// 建立连接时间, 单位毫秒: 建立连接结束时间 - 建立连接开始时间
connect?: number
// ssl握手时间, 单位毫秒: ssl握手结束时间 - ssl握手开始时间
ssl?: number
// 首包, 单位毫秒: 首包结束时间 - 首包开始时间
firstPacket?: number
// 剩余包, 单位毫秒: 剩余包结束时间 - 剩余包开始时间
remainPacket?: number
// 本地排队时间, 单位毫秒
localQueue?: number
// 发送字节数: 会覆盖startRequest时传递的bytesSent
bytesSent?: number
// 接收字节数
bytesReceived?: number
// 请求目标地址服务端IP
ip?: string
// 请求头: 会覆盖startRequest时传递的reqHeaders
reqHeaders?: KeyValueStore<string>
// 请求体: 仅支持传文本类型, 如果是JSON格式需要序列化为字符串传入. 会覆盖startRequest时传递的reqBody
reqBody?: string
// 返回头
resHeaders?: KeyValueStore<string>
// 返回体: 仅支持传文本类型的返回体, 如果是JSON格式需要序列化为字符串传入
resBody?: string
// 错误信息
errorMessage?: string
// 错误堆栈
errorStack?: string
// 链路追踪第三方apm请求头
apms?: KeyValueStore<string>
// 包含Tingyun APM的原始返回头和值
tyApm?: KeyValueStore<string>
}
示例1: 使用startRequest/endRequest,自定义请求ID
import tingyun from '@tingyun/sdk-core'
async function testRequest1() {
// 自定义请求ID
const reqId = genRequestId()
const url = 'https://example.com'
const method = 'POST'
const reqBody = {
key1: 1
}
const reqHeaders = {
'Content-Type': 'application/json;charset=UTF-8'
}
// 请求开始前, 调用startRequest开始请求
tingyun.startRequest({
id: reqId,
url: url,
method: method,
reqHeaders: reqHeaders,
reqBody: JSON.stringify(reqBody)
})
try {
// 调用自己的请求库
const response = await callCustomHttpLib(url, {
method: method,
reqBody: reqBody,
reqHeaders: reqHeaders
})
// 请求成功, 调用endRequest结束请求, 传递标识请求的请求ID
tingyun.endRequest({
id: reqId,
statusCode: response.statusCode,
resHeaders: response.headers,
resBody: response.body
})
} catch(e) {
// 请求失败, 调用endRequest结束请求, 传递标识请求的请求ID
tingyun.endRequest({
id: reqId,
errorCode: e.code
})
}
}
示例2: 使用startRequest返回的请求实例结束请求
import tingyun from '@tingyun/sdk-core'
async function testRequest2() {
const url = 'https://example.com'
const method = 'POST'
const reqBody = {
key1: 1
}
const reqHeaders = {
'Content-Type': 'application/json;charset=UTF-8'
}
// 请求开始前, 调用startRequest开始请求, 并获取请 求实例
const requestInstance = tingyun.startRequest({
url: url,
method: method,
reqHeaders: reqHeaders,
reqBody: JSON.stringify(reqBody)
})
try {
// 调用自己的请求库
const response = await callCustomHttpLib(url, {
method: method,
reqBody: reqBody,
reqHeaders: reqHeaders
})
// 请求成功, 使用请求实例的end方法结束请求
requestInstance.end({
statusCode: response.statusCode,
resHeaders: response.headers,
resBody: response.body
})
} catch(e) {
// 请求失败, 使用请求实例的end方法结束请求
requestInstance.end({
errorCode: e.code
})
}
}
模式2: createRequest
API定义:
export type Tingyun = {
createRequest: (options: RequestOptions) => void
}
// RequestStartOptions RequestEndOptions 见上方
export type RequestOptions = Omit<RequestStartOptions & RequestEndOptions, 'id' | 'timeout'>
示例:
async function testRequest3() {
const url = 'https://example.com'
const method = 'POST'
const reqBody = {
key1: 1
}
const reqHeaders = {
'Content-Type': 'application/json;charset=UTF-8'
}
// 设置开始时间
const startTime = Date.now()
let endTime
try {
// 调用自己的请求库
const response = await callCustomHttpLib(url, {
method: method,
reqBody: reqBody,
reqHeaders: reqHeaders
})
// 设置结束时间
endTime = Date.now()
// 请求成功, 调用createRequest创建自定义请求
tingyun.createRequest({
url: url,
startTime: startTime,
endTime: endTime,
statusCode: response.statusCode,
reqHeaders: reqHeaders,
resHeaders: response.headers,
resBody: response.body
})
} catch(e) {
// 设置结束时间
endTime = Date.now()
// 请求失败, 调用createRequest创建自定义请求
tingyun.createRequest({
url: url,
startTime: startTime,
endTime: endTime,
errorCode: e.code
})
}
}
用户体验监控
注册UIContext
页面路由和用户操作监控需要注册UIContext
API定义:
export Tingyun = {
registerUIContext(uiContext: UIContext)
}
使用方式:
import tingyun from '@tingyun/sdk-core'
export default class EntryAbility extends UIAbility {
onWindowStageCreate(windowStage: window.WindowStage): void {
windowStage.loadContent('pages/Index', (err, data) => {
// 获取当前UIAbility的主窗口
windowStage.getMainWindow((err: BusinessError, data) => {
const uiContext = data.getUIContext()
// 注册当前窗口的UIContext
tingyun.registerUIContext(uiContext)
})
});
}
}
操作监控
SDK支持采集用户点击事件
使用方式:
-
注册UIContext
-
在需要采集的组件上设置
id
属性或添加自定义属性_ty_name
来设置操作名称, 自定义属性的优先级高于id
示例:
Button('clickTest')
// 设置组件ID
.id('<自定义操作名称>')
// 自定义属性
.customProperty('_ty_name', '<自定义操作名称>')
.onClick(() => {
})
自定义操作
API定义:
export type ActionStartOptions = {
// 父级操作的key, 不传视为顶级操作
parent?: string
// 事件唯一标识, 不传则使用name作为唯一标识
key?: string
// 操作开始时间, 毫秒时间戳, 不传取调用时间
startTime?: number
}
export type ActionEndOptions = {
// 操作名称
name?: string
// 事件唯一标识, 不传则使用name作为唯一标识
key?: string
// 操作结束时间, 毫秒时间戳, 不传取调用时间
endTime?: number
// 自定义tag
tag?: string
// 自定义信息
metaData?: KeyValueStore<ValueType>
}
export interface IActionInstance {
end: (options?: ActionEndOptions) => void
readonly key: string
readonly name: string
}
export type Tingyun = {
// 开始用户操作
startAction: (name: string, options?: ActionStartOptions) => IActionInstance
// 结束用户操作
endAction: (options: ActionEndOptions) => void
}
使用方式:
示例1: 基本使用
// 开始操作
const action = tingyun.startAction('action1')
// ...
// 结束操作
action.end()
示例2: 传递自定义tag和附加信息
// 开始操作
const action = tingyun.startAction('action1')
// ...
// 结束操作
action.end({
// tag
tag: 'tag1',
// 附加信息
metaData: {
prop1: ''
}
})
示例3: 自定义开始时间和结束时间
// 开始操作
const action = tingyun.startAction('action1', {
startTime: <13位毫秒时间戳>
})
// ...
// 结束操作
action.end({
endTime: <13位毫秒时间戳>
})
自定义冷启动耗时结束点
API定义:
export type CustomLaunchOptions = {
className?: string
}
export type Tingyun = {
// 自定义启动结束时间
setCustomLaunchTime: (options?: CustomLaunchOptions) => void
}
使用方式:
-
SDK启动时设置
customLaunchEnd
为trueimport tingyun from '@tingyun/sdk-core'
tingyun.init({
// ...
ue: {
customLaunchEnd: true
}
}) -
在冷启动结束时调用
tingyun.setCustomLaunchTime
import tingyun from '@tingyun/sdk-core'
tingyun.setCustomLaunchTime()
Webview数据采集
API定义:
export type Tingyun = {
// 注册webController
registerWebviewController: (controller: webview.WebviewController) => void
// 获取需要注入的JS代码
getWebScriptItem: (scriptRules?: string[]) => any
}
使用方式:
import tingyun from '@tingyun/sdk-core'
Web({ src: 'xxxx', controller: this.controller })
// 确保javaScriptAccess和domStorageAccess都是开启状态
.javaScriptAccess(true)
.domStorageAccess(true)
.onControllerAttached(() => {
// onControllerAttached回调中调用tingyun.registerWebviewController
tingyun.registerWebviewController(this.controller)
})
// 注入webview监控所需的JS代码
.javaScriptOnDocumentStart([tingyun.getWebScriptItem()])
自定义错误
API:
export const ErrorTypes = {
// 自定义错误
CUSTOM: 1,
// SDK捕获的错误
CAPTURED: 2
} as const
export type ErrorType = typeof ErrorTypes[keyof typeof ErrorTypes]
export type ErrorInfo = {
// 堆栈
stack?: string
// 线程id
threadId?: number
// 线程名称
threadName?: string
}
export type KeyValueStore<T> = Record<string, T> | Map<string, T>
export type ValueType = string | number | boolean | object
export type ErrorOptions = {
// 错误类型, 默认为自定义错误
type?: ErrorType
// 错误补充信息
error?: ErrorInfo
// 附加信息
metaData?: KeyValueStore<ValueType>
}
export type Tingyun = {
// 自定义错误
reportError: (message: string, options?: ErrorOptions) => void
}
使用方式:
import tingyun from '@tingyun/sdk-core'
tingyun.reportError('<customError>')