ae-tracking 0.1.0 → 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +24 -0
- package/dist/src/cli/init.js +1 -1
- package/package.json +1 -1
- package/skills/data-integration-helper/SKILL.md +198 -0
- package/skills/data-integration-helper/references/android_sdk_faq.md +179 -0
- package/skills/data-integration-helper/references/c_sdk_compilation.md +83 -0
- package/skills/data-integration-helper/references/c_sdk_faq.md +416 -0
- package/skills/data-integration-helper/references/cocoscreator_sdk_faq.md +123 -0
- package/skills/data-integration-helper/references/cpp_server_sdk_faq.md +365 -0
- package/skills/data-integration-helper/references/index.md +56 -0
- package/skills/data-integration-helper/references/ios_sdk_faq.md +118 -0
- package/skills/data-integration-helper/references/java_sdk_faq.md +406 -0
- package/skills/data-integration-helper/references/javascript_sdk_faq.md +129 -0
- package/skills/data-integration-helper/references/logbus2_guide.md +307 -0
- package/skills/data-integration-helper/references/logbus2_parser_plugin.md +346 -0
- package/skills/data-integration-helper/references/minigame_sdk_faq.md +115 -0
- package/skills/data-integration-helper/references/miniprogram_sdk_faq.md +118 -0
- package/skills/data-integration-helper/references/python_sdk_faq.md +232 -0
- package/skills/data-integration-helper/references/restful_api_notes.md +72 -0
- package/skills/data-integration-helper/references/sdk_log_guide.md +314 -0
- package/skills/data-integration-helper/references/sdk_usage_notes.md +105 -0
- package/skills/data-integration-helper/references/unity_sdk_faq.md +115 -0
- package/skills/generate-tracking-code/SKILL.md +33 -10
- package/skills/generate-tracking-code/references/client-sdk-insert.md +33 -0
- package/skills/generate-tracking-code/references/server-sdk-insert.md +29 -0
- package/skills/generate-tracking-code/references/snippet-delivery.md +14 -7
- package/skills/generate-tracking-plan/SKILL.md +2 -2
- package/wiki/te-docs/raw//346/216/245/345/205/245/346/214/207/345/215/227//345/256/242/346/210/267/347/253/257-sdk/ios.md +2 -2
- package/wiki/te-docs/raw//346/216/245/345/205/245/346/214/207/345/215/227//345/256/242/346/210/267/347/253/257-sdk/macos.md +1 -1
|
@@ -0,0 +1,365 @@
|
|
|
1
|
+
# 一、环境配置
|
|
2
|
+
|
|
3
|
+
SDK(当前版本1.2.7)下载路径:https://github.com/ThinkingDataAnalytics/cpp-server-sdk
|
|
4
|
+
|
|
5
|
+
测试在不同运行环境中的兼容性问题:
|
|
6
|
+
|
|
7
|
+
| 系统环境 | IDE | 编译器 | 是否可以正常运行 |
|
|
8
|
+
|----------|-----|--------|------------------|
|
|
9
|
+
| Windows | Visual Studio | msvc | ✅ |
|
|
10
|
+
| Windows | Clion | MinGW | ✅ |
|
|
11
|
+
| Windows | VSCode | MinGW | ✅ |
|
|
12
|
+
| Mac | Clion | Clang(g++) | ✅ |
|
|
13
|
+
| Mac | VSCode | Clang(g++) | ✅ |
|
|
14
|
+
| Linux | - | g++ | ✅ |
|
|
15
|
+
|
|
16
|
+
修改 `CMakeLists.txt` 文件,编译loggerConsumer。(原始CMakeLists文件中包含三种模式的Consumer,项目需要使用哪一种Consumer就将哪部分的注释打开,实际生产环境只建议使用loggerConsumer)
|
|
17
|
+
|
|
18
|
+
C++SDK的标准为c++11、SDK版本为1.2.7(如果直接使用数数编译好的头文件和库文件时不兼容报错。可以使用自己的编译器版本自行编译)。
|
|
19
|
+
|
|
20
|
+
#### 初始化
|
|
21
|
+
|
|
22
|
+
导入头文件:
|
|
23
|
+
|
|
24
|
+
```cpp
|
|
25
|
+
#include "../include/ThinkingAnalyticsAPI.h"
|
|
26
|
+
#include "../include/TALoggerConsumer.h"
|
|
27
|
+
#include "../include/TADebugConsumer.h"
|
|
28
|
+
#include "../include/TABatchConsumer.h"
|
|
29
|
+
```
|
|
30
|
+
|
|
31
|
+
#### LoggerConsumer初始化样例代码
|
|
32
|
+
|
|
33
|
+
```cpp
|
|
34
|
+
//定义getLoggerConsumer() 方法,返回LoggerConsumer对象
|
|
35
|
+
unique_ptr<TAConsumer> getLoggerConsumer() {
|
|
36
|
+
LoggerConsumer::Config config = LoggerConsumer::Config("H:/log", 20, 10,LoggerConsumer::HOURLY);
|
|
37
|
+
config.fileNamePrefix = "te";
|
|
38
|
+
config.rotateMode = LoggerConsumer::HOURLY;
|
|
39
|
+
unique_ptr<TAConsumer> ptr(new LoggerConsumer(config));
|
|
40
|
+
return ptr;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
// 实际项目main方法中创建te对象
|
|
44
|
+
//unique_ptr<TAConsumer> consumer = getDebugConsumer();
|
|
45
|
+
unique_ptr<TAConsumer> consumer = getLoggerConsumer();
|
|
46
|
+
//unique_ptr<TAConsumer> consumer = getBatchConsumer();
|
|
47
|
+
ThinkingDataAnalytics te(*consumer, false);
|
|
48
|
+
```
|
|
49
|
+
|
|
50
|
+
`LOG_DIRECTORY`为写入本地的文件夹地址。
|
|
51
|
+
|
|
52
|
+
#### BatchConsumer初始化样例代码
|
|
53
|
+
|
|
54
|
+
```cpp
|
|
55
|
+
unique_ptr<TAConsumer> getBatchConsumer() {
|
|
56
|
+
unique_ptr<TAConsumer> ptr(new TABatchConsumer("appId", "serverUrl", 20, true, "/test/cert.pem"));
|
|
57
|
+
return ptr;
|
|
58
|
+
}
|
|
59
|
+
```
|
|
60
|
+
|
|
61
|
+
#### DebugConsumer初始化样例代码
|
|
62
|
+
|
|
63
|
+
```cpp
|
|
64
|
+
unique_ptr<TAConsumer> getDebugConsumer() {
|
|
65
|
+
unique_ptr<TAConsumer> ptr(new TADebugConsumer("appId", "serverUrl", "", "123456789"));
|
|
66
|
+
return ptr;
|
|
67
|
+
}
|
|
68
|
+
```
|
|
69
|
+
|
|
70
|
+
#### 初始化te对象后上报各个类型数据
|
|
71
|
+
|
|
72
|
+
```cpp
|
|
73
|
+
//上传事件
|
|
74
|
+
TaSDK::PropertiesNode event_properties;
|
|
75
|
+
event_properties.SetString("name1", "XZ_debug");//上报字符串类型属性
|
|
76
|
+
event_properties.SetNumber("test_number_int", 3);//上报数值类型属性
|
|
77
|
+
event_properties.SetBool("test_bool", true);//上报布尔类型属性
|
|
78
|
+
//上报列表型属性SetList
|
|
79
|
+
std::vector<std::string> list;
|
|
80
|
+
list.emplace_back("item11");
|
|
81
|
+
list.emplace_back("item21");
|
|
82
|
+
event_properties.SetList("test_list1", list);
|
|
83
|
+
//上报对象类型属性SetObject
|
|
84
|
+
PropertiesNode properties;
|
|
85
|
+
properties.SetString("name2", "logBugs");
|
|
86
|
+
properties.SetString("#uuid", "1234567890");
|
|
87
|
+
properties.SetNumber("test_number_int", 3);
|
|
88
|
+
properties.SetNumber("test_number_double", 3.14);
|
|
89
|
+
event_properties.SetObject("obj", properties);
|
|
90
|
+
te.track(accountId, distincId, eventName, event_properties);
|
|
91
|
+
//上传用户属性
|
|
92
|
+
TaSDK::PropertiesNode userSet_properties;
|
|
93
|
+
userSet_properties.SetString("userName", "test");
|
|
94
|
+
te.user_set(accountId, distincId, userSet_properties);
|
|
95
|
+
// 调用flush接口数据会立即写入文件,生产环境注意避免频繁调用flush引发IO或网络开销问题
|
|
96
|
+
// 一般情况不需要手动调用
|
|
97
|
+
te.flush();
|
|
98
|
+
te.close();
|
|
99
|
+
return 0;
|
|
100
|
+
```
|
|
101
|
+
|
|
102
|
+
# 二、工作原理
|
|
103
|
+
|
|
104
|
+
#### **C++ SDK支持几种工作模式?分别适用于什么场景?**
|
|
105
|
+
|
|
106
|
+
C++ SDK 支持以下3种工作模式:
|
|
107
|
+
|
|
108
|
+
1. **LoggerConsumer** 批量实时将数据写入本地文件,文件可以按每天、每小时或指定文件大小分割,需要搭配 LogBus 上报数据。优点在于数据的储存与上报解耦,数据持久化存储不容易丢失;缺点在于需要另外部署 LogBus 进行上报,LogBus会占用一部分系统资源。
|
|
109
|
+
2. **BatchConsumer** 批量实时地向TA服务器传输数据,不需要搭配上报工具。优点在于使用简单无需搭配上报工具;缺点在于数据没有持久化存储,仅在内存中做缓存,如果网络不稳定缓存数据超过缓存区上限后会丢失数据。
|
|
110
|
+
3. **DebugConsumer** 逐条发送数据,服务端会对数据进行严格校验,当某个属性不符合规范时整条数据都不会入库,当数据格式错误时会打印详细错误信息。DebugConsumer 推荐在开发调试阶段使用,禁止生产环境使用。
|
|
111
|
+
|
|
112
|
+
#### **如何获取上报地址和APP_ID?**
|
|
113
|
+
|
|
114
|
+
项目管理者可以在数数WEB界面,选择具体项目后,进入项目管理 - 项目配置 - 接入配置界面获取APP_ID及数据上报地址。如果WEB页面上没有数据上报地址,需要咨询项目管理者。上报地址分为公网地址和私网地址:
|
|
115
|
+
|
|
116
|
+
- 公网地址:适用于客户端数据上报,以及公网环境下的服务端数据接入
|
|
117
|
+
- 私网地址:适用于内网环境下的数据接入和测试,内网上报地址为集群每个节点的8991端口
|
|
118
|
+
|
|
119
|
+
#### **LoggerConsumer的工作原理是什么?有哪些配置参数?**
|
|
120
|
+
|
|
121
|
+
上报操作会写入缓存,超过bufferSize(默认20条数据)才会写入磁盘,不满足写入条件的数据需要自行调用flush方法。写入文件时会加文件锁所以不能多进程写入同一个文件。
|
|
122
|
+
|
|
123
|
+
LoggerConsumer 常用配置参数如下表:
|
|
124
|
+
|
|
125
|
+
| 参数 | 描述 | 默认值 | 取值范围 | 备注 |
|
|
126
|
+
|------|------|--------|----------|------|
|
|
127
|
+
| logDir | 日志文件写入的路径 | 无 | 字符串 | 多级目录会自动创建 |
|
|
128
|
+
| rotateMode | 日志切分模式 | DAILY | HOURLY、DAILY | |
|
|
129
|
+
| fileNamePrefix | 日志文件名前缀 | 无 | 字符串 | |
|
|
130
|
+
| fileSize | 日志切分大小 | 0(不切分) | INT | 单位为MB |
|
|
131
|
+
| bufferSize | 缓冲区大小,达到阈值进行内存刷写到日志 | 20 | INT | 单位为数据条数 |
|
|
132
|
+
|
|
133
|
+
#### **BatchConsumer的工作原理是什么?有哪些配置参数?**
|
|
134
|
+
|
|
135
|
+
BatchConsumer 工作原理详细介绍:上报操作会写入缓存,当上报数据数量大于batchSize(默认20)或因网络通信失败等问题未上报数据导致cacheBuffer不为空时调用flush,不满足条件需要自行调用flush方法。flush时如果通信失败会重试3次,失败后存入cacheBuffer,长时间通信失败导致 未成功发送总条数 / batchSize大于maxCacheSize时会丢弃最早的batchSize条数据。
|
|
136
|
+
|
|
137
|
+
BatchConsumer 常用配置参数如下表:
|
|
138
|
+
|
|
139
|
+
| 参数 | 描述 | 默认值 | 取值范围 | 备注 |
|
|
140
|
+
|------|------|--------|----------|------|
|
|
141
|
+
| batchSize | 批次大小,达到阈值触发数据上报 | 20 | INT | 单位为数据条数 |
|
|
142
|
+
|
|
143
|
+
#### **DebugConsumer的工作原理是什么?有哪些配置参数?**
|
|
144
|
+
|
|
145
|
+
原理:每条数据都直接走http请求上报数据,不用调flush。数据格式会进行严格校验。
|
|
146
|
+
|
|
147
|
+
# 三、常见问题
|
|
148
|
+
|
|
149
|
+
#### **使用 LoggerConsumer 有哪些注意事项?**
|
|
150
|
+
|
|
151
|
+
- **搭配Logbus上报**
|
|
152
|
+
- LoggerConsumer + LogBus为数数标准的数据上报方案,LoggerConsumer使得数据持久化,数据得到不丢失的保障;LogBus为数据传输作保证,同时将数据持久化和上报解耦。
|
|
153
|
+
- **文件写权限**
|
|
154
|
+
- 写入日志目录需要有写入和读取的权限,通常Windows环境会有写入权限问题。
|
|
155
|
+
- **磁盘空间**
|
|
156
|
+
- 磁盘空间保证充裕,并可以合理在LogBus上配置删除策略。
|
|
157
|
+
- **磁盘性能NFS情况**
|
|
158
|
+
- NFS磁盘就是通过网络连接到本地计算机的一种远程文件系统,它可以让用户在本地计算机上像访问本地磁盘一样访问NFS磁盘上的文件和目录。在使用此类磁盘时,需要关注写入速率及网络波动导致的数据异常写入的问题。
|
|
159
|
+
- **UUID**
|
|
160
|
+
- 建议添加UUID,防止网络抖动及极端情况造成数据重复,但会稍微消耗效率,也可在LogBus侧打开。如不添加则默认在集群receiver组件处添加。
|
|
161
|
+
- **多进程写不同文件**
|
|
162
|
+
- 支持多进程写不同文件,但要保证不同进程的处理逻辑没有依赖关系(如不同服务器的用户行为写到不同日志)
|
|
163
|
+
- **容器环境**
|
|
164
|
+
- 将数据写入路径映射到外部磁盘,防止容器关闭数据文件丢失。
|
|
165
|
+
|
|
166
|
+
#### **LoggerConsumer 是否支持多线程?**
|
|
167
|
+
|
|
168
|
+
不支持多线程写同一个文件。TALoggerConsumer的 add()方法中用m_flush_mutex.lock()排他锁对文件加锁。可以多线程写入不同文件或写入不同目录,避免异常。
|
|
169
|
+
|
|
170
|
+
#### **LoggerConsumer 性能指标如何?**
|
|
171
|
+
|
|
172
|
+
**测试环境**
|
|
173
|
+
|
|
174
|
+
```cpp
|
|
175
|
+
Kernel Version: 3.10.0-1160.42.2.el7.x86_64
|
|
176
|
+
Operating System: CentOS Linux 7 (Core)
|
|
177
|
+
OSType: linux
|
|
178
|
+
Architecture: x86_64
|
|
179
|
+
CPUs: 4
|
|
180
|
+
Total Memory: 15.25GiB
|
|
181
|
+
```
|
|
182
|
+
|
|
183
|
+
**测试场景**
|
|
184
|
+
|
|
185
|
+
启动服务端代码,开启10个子线程,使用**log_consumer每个线程持续**track5分钟,**track接口调用速率为每秒钟10万次**,获取当前进程的cpu占用率,内存占用率。
|
|
186
|
+
|
|
187
|
+
**测试结果**
|
|
188
|
+
|
|
189
|
+
5分钟实际写入数据量为16066600条,平均每秒钟写入53555条数据。track期间:cpu均值在67%左右,4核总均值在268%左右浮动;memory均值稳定在4.1M左右,无内存泄漏的产生。
|
|
190
|
+
|
|
191
|
+
#### **LoggerConsumer 是否存在丢数风险?如何避免?**
|
|
192
|
+
|
|
193
|
+
如果磁盘写满或者服务器宕机可能导致数据丢失,建议:
|
|
194
|
+
|
|
195
|
+
1. 定期检查写入日志路径磁盘剩余容量
|
|
196
|
+
2. 降低batchSize参数值,增加内存刷写频率,但会导致频繁IO,需结合具体场景综合考虑
|
|
197
|
+
|
|
198
|
+
#### **BatchConsumer 为什么会存在丢数风险?如何避免?**
|
|
199
|
+
|
|
200
|
+
`BatchConsumer`在内存中维护了一个batchSize 大小的队列负责存放单批次数据。因为`BatchConsumer`基于内存存储,所以当发生内存溢出或者服务器宕机时,内存中未发送的数据会全部丢失,建议:
|
|
201
|
+
|
|
202
|
+
1. 使用LoggerConsumer + LogBus搭配进行数据上报
|
|
203
|
+
2. batchSize参数不宜设置过大,可能会导致单次发送数据时间增加,会增加发生网络错误的概率
|
|
204
|
+
|
|
205
|
+
批量实时地向 TE 服务器传输数据,不需要搭配传输工具。**在长时间网络中断情况下,有数据丢失的风险,不建议在正式环境中使用。优点在于使用简单无需搭配上报工具;缺点在于数据没有持久化存储,仅在内存中做缓存,如果网络不稳定缓存数据超过缓存区上限后会丢失数据。**
|
|
206
|
+
|
|
207
|
+
#### **BatchConsumer 性能指标如何?适合在什么场景下使用?**
|
|
208
|
+
|
|
209
|
+
BatchConsumer适合中小数据量,且日志上报服务器和TE集群内网打通时使用。
|
|
210
|
+
|
|
211
|
+
#### **DebugConsumer 为什么在生产环境禁用?**
|
|
212
|
+
|
|
213
|
+
逐条发送数据,服务端会对数据进行严格校验,当某个属性不符合规范时整条数据都不会入库,当数据格式错误时会打印详细错误信息。DebugConsumer 仅适用开发调试阶段,不适合在正式环境使用。
|
|
214
|
+
|
|
215
|
+
#### **什么时候需要调用 `close()` 方法?**
|
|
216
|
+
|
|
217
|
+
程序需要正常结束时调用,close()方法会将内存中的数据进行写入文件或发送。close() 函数会调用 flush() 上报或写入缓存的数据,并关闭和释放相应资源。一种常见的错误用法是在每次调用 flush() 后调用 close(),这样会导致下一次调用 flush() 时报错,除非每次对SDK重新初始化。
|
|
218
|
+
|
|
219
|
+
#### **在程序中调用了 `track()` 或者 `user_set()` 方法, 为什么在 TE 后台没有看到数据?**
|
|
220
|
+
|
|
221
|
+
请依次检查以下情况:
|
|
222
|
+
|
|
223
|
+
- **检查上报地址和appid**
|
|
224
|
+
- curl https://push_url/health-check,上报地址正常会返回`ok`
|
|
225
|
+
- curl https://push_url/check_appid?appid=目标APPID,上报地址和appid匹配会返回`{"code":0}`
|
|
226
|
+
- **数据太少,未触发上报**
|
|
227
|
+
- 对于BatchConsumer和LoggerConsumer需要达到batchSize才会真正触发数据写入或数据上报
|
|
228
|
+
- 也可以手动调用flush方法触发数据上报,但不建议频繁主动调用flush
|
|
229
|
+
- **错误数据**
|
|
230
|
+
- 可以在WEB界面查看错误数据原因
|
|
231
|
+
- **数据时间**
|
|
232
|
+
- 服务端数据接收事件上限:相对服务器时间的前三年至后3天
|
|
233
|
+
- 客户端数据接收事件上限:相对服务器时间的前10天至后3天
|
|
234
|
+
- **历史通道**
|
|
235
|
+
- 当在项目管理中开启历史通道后,上报10天前的历史数据时会被ETL直接搬运到HDFS历史通道,才能入库可供查询
|
|
236
|
+
- **埋点方案**
|
|
237
|
+
- 在项目管理中设置了埋点方案且在数据处理规则中选择了`不在埋点方案中的事件:不允许入库`后,将无法上报不再埋点方案中的数据,可在错误数据中查看
|
|
238
|
+
- **白名单**
|
|
239
|
+
- 在项目管理中设置了IP白名单后,将只有白名单内的IP才能上报数据,白名单生效存在10分钟的延迟
|
|
240
|
+
|
|
241
|
+
#### **首次上报属性业务为空(对象组为例该如果上报 如果是{}会识别成该字段为列表)如何正确上报?**
|
|
242
|
+
|
|
243
|
+
上报为各个语言的Null类型。
|
|
244
|
+
|
|
245
|
+
#### **事件时间默认值?如何将自定义的时间赋值给#time?**
|
|
246
|
+
|
|
247
|
+
默认为ThinkingDataAnalytics::track() 时的服务器当前时间。自定义的时间赋值给#time参考前面上报样例代码。
|
|
248
|
+
|
|
249
|
+
#### **SDK的线程模型?是异步的还是阻塞?**
|
|
250
|
+
|
|
251
|
+
SDK 主要作用是对于数据格式的组织,以及缓冲区落盘。**线程模型是阻塞式的**,在 flush 函数调用时触发。
|
|
252
|
+
|
|
253
|
+
#### **SDK中有 GBK 转 UTF8 的 C++ 标准函数么?对比javaSDK的timer,C++SDK中有支持吗?**
|
|
254
|
+
|
|
255
|
+
目前都没有集成这些功能,后续新版本SDK会新增。
|
|
256
|
+
|
|
257
|
+
#### **ubuntu 20.04 默认 cmake 版本是 3.16.1,C++SDK的编译选项是 cmake_minimum_required(VERSION 3.22)。必须指定 cmake 到 3.22 以上版本么?**
|
|
258
|
+
|
|
259
|
+
理论上是可以的,可先用低版本的编译器尝试。或者自行用低版本的编译器编译SDK源码。
|
|
260
|
+
|
|
261
|
+
#### **客户项目代码和 C++SDK的兼容性问题**
|
|
262
|
+
|
|
263
|
+
**数数C++SDK的c++标准为c++11。**SDK库引用了 set(CMAKE_CXX_STANDARD 11),使用该SDK必须引用 set(CMAKE_CXX_STANDARD 11)。若客户项目引入代码的c++标准较低,引入c++11标准的SDK会出现一堆报错,即项目有些代码是不支持 C++11。需要保证代码的c++标准高于11。
|
|
264
|
+
|
|
265
|
+
#### **使用debug/batch consumer的方式上报数据至https协议的receiver url时返回-1,报错Unsupported protocol**
|
|
266
|
+
|
|
267
|
+
`SERVER_URL`: 数据上传的 URL
|
|
268
|
+
|
|
269
|
+
- 如果您对接的是云服务,填入: https://global-receiver-ta.thinkingdata.cn
|
|
270
|
+
- 如果您使用私有化部署版本,请为数据采集地址绑定域名,**并配置 HTTPS 证书**:https://数据采集地址绑定域名
|
|
271
|
+
|
|
272
|
+
# 四、属性类型、预置属性
|
|
273
|
+
|
|
274
|
+
#### **C++ SDK 如何上报对象和对象组类型?**
|
|
275
|
+
|
|
276
|
+
C++ SDK支持上报对象和对象组类型。可参考上面logger consumer上报各个数据类型的样例代码。代码示例请参考服务端SDK复杂类型上报。
|
|
277
|
+
|
|
278
|
+
#### **某属性首次上报为空值,应该如何上报?**
|
|
279
|
+
|
|
280
|
+
对象组为例,如果是[]会识别成该字段为列表,需要设为[{}]。
|
|
281
|
+
|
|
282
|
+
#### **上报数据中为什么没有 "#ip"?**
|
|
283
|
+
|
|
284
|
+
服务端的#ip需要单独上报,层级与#distinct_id、#event_name、#time、properties等同级。
|
|
285
|
+
|
|
286
|
+
#### **公共属性**
|
|
287
|
+
|
|
288
|
+
服务端的公共属性无法精确到用户级,在多线程情况下报了用户级属性数据可能会导致用户数据对不上。仅建议在公共属性内放区服id等不会产生大变化的字段,其余均放入普通属性。
|
|
289
|
+
|
|
290
|
+
#### **时区**
|
|
291
|
+
|
|
292
|
+
如果数据事件时间不是utc8又对时区偏移有需求,可以在普通属性内增加#zone_offset这个时区字段,假设事件时间是utc0就给时区字段赋值数值类型的0即可。
|
|
293
|
+
|
|
294
|
+
#### **用户属性**
|
|
295
|
+
|
|
296
|
+
1. 对于一般的用户属性,您可以调用 `user_set` 来进行设置。使用该接口上传的属性将会覆盖原有的属性值,如果之前不存在该用户属性,则会新建该用户属性,类型与传入属性的类型一致:
|
|
297
|
+
|
|
298
|
+
```cpp
|
|
299
|
+
TaSDK::PropertiesNode userSet_properties;
|
|
300
|
+
userSet_properties.SetString("userName", "A");
|
|
301
|
+
te.user_set(accountId, distincId, userSet_properties);
|
|
302
|
+
|
|
303
|
+
userSet_properties.SetString("userName", "B");
|
|
304
|
+
te.user_set(accountId, distincId, userSet_properties); // userName 会变为 "B"
|
|
305
|
+
```
|
|
306
|
+
|
|
307
|
+
2. 如果您要上传的用户属性只要设置一次,则可以调用`user_setOnce`来进行设置,当该属性之前已经有值的时候,将会忽略这条信息:
|
|
308
|
+
|
|
309
|
+
```cpp
|
|
310
|
+
TaSDK::PropertiesNode userSetOnce_properties;
|
|
311
|
+
userSetOnce_properties.SetString("user_one_name", "A");
|
|
312
|
+
te.user_setOnce(accountId, distincId, userSetOnce_properties);
|
|
313
|
+
|
|
314
|
+
userSetOnce_properties.SetString("user_one_name", "B");
|
|
315
|
+
te.user_setOnce(accountId, distincId, userSetOnce_properties); // user_one_name 仍为 "A"
|
|
316
|
+
```
|
|
317
|
+
|
|
318
|
+
3. 当您要上传数值型的属性时,您可以调用`user_add`来对该属性进行累加操作,如果该属性还未被设置,则会赋值 0 后再进行计算:
|
|
319
|
+
|
|
320
|
+
```cpp
|
|
321
|
+
TaSDK::PropertiesNode userAdd_properties;
|
|
322
|
+
userAdd_properties.SetNumber("cash", 30);
|
|
323
|
+
te.user_add(accountId, distincId, userAdd_properties);
|
|
324
|
+
|
|
325
|
+
userAdd_properties.SetNumber("cash", 60);
|
|
326
|
+
te.user_add(accountId, distincId, userAdd_properties); // cash 会变为 90
|
|
327
|
+
```
|
|
328
|
+
|
|
329
|
+
4. 您可以调用 `user_append` 对数组类型的用户属性进行追加操作:
|
|
330
|
+
|
|
331
|
+
```cpp
|
|
332
|
+
TaSDK::PropertiesNode userAppend_properties;
|
|
333
|
+
vector<string> userAppenListValue;
|
|
334
|
+
userAppenListValue.push_back("11");
|
|
335
|
+
userAppenListValue.push_back("33");
|
|
336
|
+
userAppend_properties.SetList("arr1", userAppenListValue);
|
|
337
|
+
te.user_append(accountId, distincId, userAppend_properties); // arr1 为 ["11", "33"]
|
|
338
|
+
|
|
339
|
+
TaSDK::PropertiesNode userAppend_properties_new;
|
|
340
|
+
vector<string> userAppenListValueNew;
|
|
341
|
+
userAppenListValueNew.push_back("22");
|
|
342
|
+
userAppenListValueNew.push_back("33");
|
|
343
|
+
userAppend_properties_new.SetList("arr1", userAppenListValueNew);
|
|
344
|
+
te.user_append(accountId, distincId, userAppend_properties_new); // arr1 为 ["11", "33", "22", "33"]
|
|
345
|
+
```
|
|
346
|
+
|
|
347
|
+
#### **可更新事件**
|
|
348
|
+
|
|
349
|
+
在服务器为集群模式下,毫秒级别的可更新事件可能会乱序消费导致最终更新结果不达到预期。遇到这种情况时可以在可更新事件中properties内加一个普通属性指定这条数据的序号,比如属性 order为数值类型的1,然后在properties外增加预置属性#transaction_property,值设置为 "order",入库时会根据 order的大小来判断是否要入库,order大于1更新,小于等于1则丢弃。
|
|
350
|
+
|
|
351
|
+
```cpp
|
|
352
|
+
// updatable event
|
|
353
|
+
eventName = "update_event";
|
|
354
|
+
string updateEventId = "update_001";
|
|
355
|
+
TaSDK::PropertiesNode update_event_properties;
|
|
356
|
+
update_event_properties.SetString("price", "100");
|
|
357
|
+
update_event_properties.SetString("status", "3");
|
|
358
|
+
te.track_update(accountId, distincId, eventName, updateEventId, update_event_properties);
|
|
359
|
+
|
|
360
|
+
TaSDK::PropertiesNode update_event_new_properties;
|
|
361
|
+
update_event_new_properties.SetString("status", "5");
|
|
362
|
+
te.track_update(accountId, distincId, eventName, updateEventId, update_event_new_properties); // status 变为 5,price 仍为 100
|
|
363
|
+
```
|
|
364
|
+
|
|
365
|
+
# 五、异常报错
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
# 飞书文档补充
|
|
2
|
+
|
|
3
|
+
存放从飞书拉取的数据集成相关补充文档。
|
|
4
|
+
|
|
5
|
+
## 文档列表
|
|
6
|
+
|
|
7
|
+
| 文档 | 说明 |
|
|
8
|
+
|------|------|
|
|
9
|
+
| [android_sdk_faq.md](./android_sdk_faq.md) | Android SDK 常见问题 |
|
|
10
|
+
| [ios_sdk_faq.md](./ios_sdk_faq.md) | iOS SDK 常见问题 |
|
|
11
|
+
| [javascript_sdk_faq.md](./javascript_sdk_faq.md) | JavaScript SDK 常见问题 |
|
|
12
|
+
| [miniprogram_sdk_faq.md](./miniprogram_sdk_faq.md) | 小程序 SDK 常见问题 |
|
|
13
|
+
| [minigame_sdk_faq.md](./minigame_sdk_faq.md) | 小游戏 SDK 常见问题 |
|
|
14
|
+
| [unity_sdk_faq.md](./unity_sdk_faq.md) | Unity SDK 常见问题 |
|
|
15
|
+
| [cocoscreator_sdk_faq.md](./cocoscreator_sdk_faq.md) | Cocos Creator SDK 常见问题 |
|
|
16
|
+
| [java_sdk_faq.md](./java_sdk_faq.md) | Java SDK 常见问题 |
|
|
17
|
+
| [python_sdk_faq.md](./python_sdk_faq.md) | Python SDK 常见问题 |
|
|
18
|
+
| [cpp_server_sdk_faq.md](./cpp_server_sdk_faq.md) | C++ (服务端) SDK 常见问题 |
|
|
19
|
+
| [c_sdk_faq.md](./c_sdk_faq.md) | C SDK 常见问题 |
|
|
20
|
+
| [c_sdk_compilation.md](./c_sdk_compilation.md) | C SDK 编译流程 |
|
|
21
|
+
| [logbus2_guide.md](./logbus2_guide.md) | LogBus2 使用指南 |
|
|
22
|
+
| [logbus2_parser_plugin.md](./logbus2_parser_plugin.md) | LogBus2 数据转译插件 |
|
|
23
|
+
| [restful_api_notes.md](./restful_api_notes.md) | Restful API 使用注意事项 |
|
|
24
|
+
| [sdk_usage_notes.md](./sdk_usage_notes.md) | SDK 使用注意事项(客户端/服务端/多端上报) |
|
|
25
|
+
| [sdk_log_guide.md](./sdk_log_guide.md) | 查看客户端 SDK 日志(iOS/Android/Unity/OpenHarmony) |
|
|
26
|
+
|
|
27
|
+
---
|
|
28
|
+
|
|
29
|
+
## 分类索引
|
|
30
|
+
|
|
31
|
+
### 客户端 SDK FAQ
|
|
32
|
+
|
|
33
|
+
- **原生移动端**: Android, iOS
|
|
34
|
+
- **Web/H5**: JavaScript
|
|
35
|
+
- **小程序/小游戏**: 小程序, 小游戏
|
|
36
|
+
- **游戏引擎**: Unity, Cocos Creator
|
|
37
|
+
|
|
38
|
+
### 服务端 SDK FAQ
|
|
39
|
+
|
|
40
|
+
- **Java**: java_sdk_faq.md
|
|
41
|
+
- **Python**: python_sdk_faq.md
|
|
42
|
+
- **C++**: cpp_server_sdk_faq.md
|
|
43
|
+
- **C**: c_sdk_faq.md, c_sdk_compilation.md
|
|
44
|
+
|
|
45
|
+
### 数据集成工具
|
|
46
|
+
|
|
47
|
+
- **LogBus2**: logbus2_guide.md, logbus2_parser_plugin.md
|
|
48
|
+
- **Restful API**: restful_api_notes.md
|
|
49
|
+
|
|
50
|
+
### 最佳实践
|
|
51
|
+
|
|
52
|
+
- **SDK 使用注意事项**: sdk_usage_notes.md(客户端/服务端/多端上报)
|
|
53
|
+
|
|
54
|
+
### 常见问题与最佳实践
|
|
55
|
+
|
|
56
|
+
以上 FAQ 文档涵盖各平台 SDK 使用中的常见问题与解决方案。
|
|
@@ -0,0 +1,118 @@
|
|
|
1
|
+
---
|
|
2
|
+
title: "iOS SDK FAQ"
|
|
3
|
+
code: "ios_sdk_faq"
|
|
4
|
+
source: "Feishu MCP"
|
|
5
|
+
doc_id: "wikcn4nM313WpHDriYEdxvUmzqe"
|
|
6
|
+
fetched_at: "2026-04-20T17:29:37Z"
|
|
7
|
+
---
|
|
8
|
+
|
|
9
|
+
# 集成 SDK
|
|
10
|
+
## SDK 集成方式说明
|
|
11
|
+
- 目前支持手动集成和自动集成(CocoaPods)两种方式,不支持 Carthage 方式集成
|
|
12
|
+
|
|
13
|
+
# 初始化 SDK
|
|
14
|
+
## 推荐初始化 SDK 初始化位置
|
|
15
|
+
- 建议在用户同意隐私协议后再进行 SDK 初始化
|
|
16
|
+
```objectivec
|
|
17
|
+
if (授权隐私政策) {
|
|
18
|
+
NSString *appid = @"APPID";
|
|
19
|
+
NSString *url = @"SERVER_URL";
|
|
20
|
+
[TDAnalytics startAnalyticsWithAppId:appid serverUrl:url];
|
|
21
|
+
}
|
|
22
|
+
```
|
|
23
|
+
|
|
24
|
+
# SDK 缓存机制 & 限制
|
|
25
|
+
## SDK 哪些数据会在本地存储?
|
|
26
|
+
- 基础数据:访客 ID、设备 ID、静态通用事件属性等,存储在 Library 目录
|
|
27
|
+
- 事件数据:缓存到本地数据库 Library/TDData-data.plist
|
|
28
|
+
|
|
29
|
+
## 事件数据缓存数量限制
|
|
30
|
+
- SDK 默认缓存限制为 10000 条数据
|
|
31
|
+
- 可自定义缓存数量,最小值为 5000 条
|
|
32
|
+
|
|
33
|
+
## 缓存过期时间
|
|
34
|
+
- SDK 默认过期时间为 10 天
|
|
35
|
+
- 可在 info.plist 中配置
|
|
36
|
+
|
|
37
|
+
# SDK 数据上报策略
|
|
38
|
+
## 立即上报
|
|
39
|
+
- APP 切换到后台会立即上报缓存数据
|
|
40
|
+
- debug / debug_only 模式会立即上报
|
|
41
|
+
- 自动采集事件会立刻上报
|
|
42
|
+
- 手动调用 flush() 接口会立即上报
|
|
43
|
+
- 每个请求上限为 50 条数据
|
|
44
|
+
|
|
45
|
+
## 批量上报
|
|
46
|
+
- 时间间隔超过配置时长上报(默认 30 秒)
|
|
47
|
+
- 缓存事件数量达到阈值(默认 30 条)
|
|
48
|
+
|
|
49
|
+
# 访客 ID(#distinct_id)
|
|
50
|
+
## 访客 ID 如何设置
|
|
51
|
+
```objectivec
|
|
52
|
+
[TDAnalytics setDistinctId:@"Thinker"];
|
|
53
|
+
```
|
|
54
|
+
|
|
55
|
+
## 访客 ID 何时会变
|
|
56
|
+
- 用户清除了应用数据
|
|
57
|
+
- 用户更换了设备或卸载重装
|
|
58
|
+
- 调用 identify 接口手动设置
|
|
59
|
+
|
|
60
|
+
# 账号 ID(#account_id)
|
|
61
|
+
## 账号 ID 如何设置
|
|
62
|
+
```objectivec
|
|
63
|
+
[TDAnalytics login:@"TD"];
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
# Debug 模式
|
|
67
|
+
## Normal,Debug,DebugOnly 模式区别
|
|
68
|
+
|
|
69
|
+
| 模式 | 上报模式 | 本地缓存 | 数据入库 | 数据严格检查 | 数据加密 |
|
|
70
|
+
|------|----------|----------|----------|--------------|----------|
|
|
71
|
+
| Normal | 批量上报 | 是 | 是 | 否 | 支持 |
|
|
72
|
+
| Debug | 逐条上报 | 否 | 是 | 是 | 不支持 |
|
|
73
|
+
| DebugOnly | 逐条上报 | 否 | 否 | 是 | 不支持 |
|
|
74
|
+
|
|
75
|
+
> 请勿在生产环境使用 Debug 和 DebugOnly 模式
|
|
76
|
+
|
|
77
|
+
# 自动采集事件
|
|
78
|
+
## ta_app_install
|
|
79
|
+
- 新安装 APP 或卸载重装 APP 时触发
|
|
80
|
+
- #install_time 和 #time 不相等是正常的
|
|
81
|
+
|
|
82
|
+
## ta_app_start
|
|
83
|
+
- 第一次开启自动采集时
|
|
84
|
+
- app 后台切换到前台后触发
|
|
85
|
+
|
|
86
|
+
## ta_app_end
|
|
87
|
+
- app 从前台切换到后台、杀掉 app 进程后
|
|
88
|
+
|
|
89
|
+
## ta_app_crash
|
|
90
|
+
- 监听 NSSetUncaughtExceptionHandler 捕获的异常
|
|
91
|
+
- 在 Debug 模式下 ta_app_crash 可能丢失
|
|
92
|
+
|
|
93
|
+
# 公共事件属性
|
|
94
|
+
## 静态公共事件属性
|
|
95
|
+
- `[TDAnalytics setSuperProperties:@{@"vip_level": @(2)}];`
|
|
96
|
+
- 存在本地,每次 track 都会带上
|
|
97
|
+
|
|
98
|
+
## 动态公共事件属性
|
|
99
|
+
```objectivec
|
|
100
|
+
[TDAnalytics setDynamicSuperProperties:^NSDictionary * _Nonnull{
|
|
101
|
+
return @{@"now": [NSDate date]};
|
|
102
|
+
}];
|
|
103
|
+
```
|
|
104
|
+
|
|
105
|
+
## 属性优先级
|
|
106
|
+
- 用户自定义事件属性 > 动态公共事件属性 > 静态公共事件属性
|
|
107
|
+
|
|
108
|
+
# 预置属性
|
|
109
|
+
## #device_id
|
|
110
|
+
- 获取的是 idfv,获取不到就获取 UUID
|
|
111
|
+
- Keychain 持久化存储,卸载重装后不会改变
|
|
112
|
+
|
|
113
|
+
## #ip
|
|
114
|
+
- TE 服务器获取 http 请求 header 中的 ip 信息
|
|
115
|
+
- 使用第三方 IP 库解析地理位置
|
|
116
|
+
|
|
117
|
+
# 已知问题
|
|
118
|
+
详见文档完整内容。
|