analysis.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317
  1. package analysis
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "github.com/silenceper/wechat/v2/miniprogram/context"
  6. "github.com/silenceper/wechat/v2/util"
  7. )
  8. const (
  9. // 获取用户访问小程序日留存
  10. getAnalysisDailyRetainURL = "https://api.weixin.qq.com/datacube/getweanalysisappiddailyretaininfo?access_token=%s"
  11. // 获取用户访问小程序月留存
  12. getAnalysisMonthlyRetainURL = "https://api.weixin.qq.com/datacube/getweanalysisappidmonthlyretaininfo?access_token=%s"
  13. // 获取用户访问小程序周留存
  14. getAnalysisWeeklyRetainURL = "https://api.weixin.qq.com/datacube/getweanalysisappidweeklyretaininfo?access_token=%s"
  15. // 获取用户访问小程序数据概况
  16. getAnalysisDailySummaryURL = "https://api.weixin.qq.com/datacube/getweanalysisappiddailysummarytrend?access_token=%s"
  17. // 获取用户访问小程序数据日趋势
  18. getAnalysisDailyVisitTrendURL = "https://api.weixin.qq.com/datacube/getweanalysisappiddailyvisittrend?access_token=%s"
  19. // 获取用户访问小程序数据月趋势
  20. getAnalysisMonthlyVisitTrendURL = "https://api.weixin.qq.com/datacube/getweanalysisappidmonthlyvisittrend?access_token=%s"
  21. // 获取用户访问小程序数据周趋势
  22. getAnalysisWeeklyVisitTrendURL = "https://api.weixin.qq.com/datacube/getweanalysisappidweeklyvisittrend?access_token=%s"
  23. // 获取小程序新增或活跃用户的画像分布数据
  24. getAnalysisUserPortraitURL = "https://api.weixin.qq.com/datacube/getweanalysisappiduserportrait?access_token=%s"
  25. // 获取用户小程序访问分布数据
  26. getAnalysisVisitDistributionURL = "https://api.weixin.qq.com/datacube/getweanalysisappidvisitdistribution?access_token=%s"
  27. // 访问页面
  28. getAnalysisVisitPageURL = "https://api.weixin.qq.com/datacube/getweanalysisappidvisitpage?access_token=%s"
  29. )
  30. // Analysis analyis 数据分析
  31. type Analysis struct {
  32. *context.Context
  33. }
  34. // NewAnalysis new
  35. func NewAnalysis(ctx *context.Context) *Analysis {
  36. return &Analysis{ctx}
  37. }
  38. // fetchData 拉取统计数据
  39. func (analysis *Analysis) fetchData(urlStr string, body interface{}) (response []byte, err error) {
  40. var accessToken string
  41. accessToken, err = analysis.GetAccessToken()
  42. if err != nil {
  43. return
  44. }
  45. urlStr = fmt.Sprintf(urlStr, accessToken)
  46. response, err = util.PostJSON(urlStr, body)
  47. return
  48. }
  49. // RetainItem 留存项结构
  50. type RetainItem struct {
  51. Key int `json:"key"` // 标识,0开始表示当天,1表示1甜后,以此类推
  52. Value int `json:"value"` // key对应日期的新增用户数/活跃用户数(key=0时)或留存用户数(k>0时)
  53. }
  54. // ResAnalysisRetain 小程序留存数据返回
  55. type ResAnalysisRetain struct {
  56. util.CommonError
  57. RefDate string `json:"ref_date"` // 日期
  58. VisitUVNew []RetainItem `json:"visit_uv_new"` // 新增用户留存
  59. VisitUV []RetainItem `json:"visit_uv"` // 活跃用户留存
  60. }
  61. // getAnalysisRetain 获取用户访问小程序留存数据(日、月、周)
  62. func (analysis *Analysis) getAnalysisRetain(urlStr string, beginDate, endDate string) (result ResAnalysisRetain, err error) {
  63. body := map[string]string{
  64. "begin_date": beginDate,
  65. "end_date": endDate,
  66. }
  67. response, err := analysis.fetchData(urlStr, body)
  68. if err != nil {
  69. return
  70. }
  71. err = json.Unmarshal(response, &result)
  72. if err != nil {
  73. return
  74. }
  75. if result.ErrCode != 0 {
  76. err = fmt.Errorf("getAnalysisRetain error : errcode=%v , errmsg=%v", result.ErrCode, result.ErrMsg)
  77. return
  78. }
  79. return
  80. }
  81. // GetAnalysisDailyRetain 获取用户访问小程序日留存
  82. func (analysis *Analysis) GetAnalysisDailyRetain(beginDate, endDate string) (result ResAnalysisRetain, err error) {
  83. return analysis.getAnalysisRetain(getAnalysisDailyRetainURL, beginDate, endDate)
  84. }
  85. // GetAnalysisMonthlyRetain 获取用户访问小程序月留存
  86. func (analysis *Analysis) GetAnalysisMonthlyRetain(beginDate, endDate string) (result ResAnalysisRetain, err error) {
  87. return analysis.getAnalysisRetain(getAnalysisMonthlyRetainURL, beginDate, endDate)
  88. }
  89. // GetAnalysisWeeklyRetain 获取用户访问小程序周留存
  90. func (analysis *Analysis) GetAnalysisWeeklyRetain(beginDate, endDate string) (result ResAnalysisRetain, err error) {
  91. return analysis.getAnalysisRetain(getAnalysisWeeklyRetainURL, beginDate, endDate)
  92. }
  93. // ResAnalysisDailySummary 小程序访问数据概况
  94. type ResAnalysisDailySummary struct {
  95. util.CommonError
  96. List []struct {
  97. RefDate string `json:"ref_date"` // 日期
  98. VisitTotal int `json:"visit_total"` // 累计用户数
  99. SharePV int `json:"share_pv"` // 转发次数
  100. ShareUV int `json:"share_uv"` // 转发人数
  101. } `json:"list"`
  102. }
  103. // GetAnalysisDailySummary 获取用户访问小程序数据概况
  104. func (analysis *Analysis) GetAnalysisDailySummary(beginDate, endDate string) (result ResAnalysisDailySummary, err error) {
  105. body := map[string]string{
  106. "begin_date": beginDate,
  107. "end_date": endDate,
  108. }
  109. response, err := analysis.fetchData(getAnalysisDailySummaryURL, body)
  110. if err != nil {
  111. return
  112. }
  113. err = json.Unmarshal(response, &result)
  114. if err != nil {
  115. return
  116. }
  117. if result.ErrCode != 0 {
  118. err = fmt.Errorf("GetAnalysisDailySummary error : errcode=%v , errmsg=%v", result.ErrCode, result.ErrMsg)
  119. return
  120. }
  121. return
  122. }
  123. // ResAnalysisVisitTrend 小程序访问数据趋势(日、月、周)
  124. type ResAnalysisVisitTrend struct {
  125. util.CommonError
  126. List []struct {
  127. RefDate string `json:"ref_date"` // 日期
  128. SessionCnt int `json:"session_cnt"` // 打开次数
  129. VisitPV int `json:"visit_pv"` // 访问次数
  130. VisitUV int `json:"visit_uv"` // 访问人数
  131. VisitUVNew int `json:"visit_uv_new"` // 新用户数
  132. StayTimeUV float64 `json:"stay_time_uv"` // 人均停留时长
  133. StayTimeSession float64 `json:"stay_time_session"` // 次均停留时常
  134. VisitDepth float64 `json:"visit_depth"` // 平均访问深度
  135. } `json:"list"`
  136. }
  137. // getAnalysisRetain 获取小程序访问数据趋势(日、月、周)
  138. func (analysis *Analysis) getAnalysisVisitTrend(urlStr string, beginDate, endDate string) (result ResAnalysisVisitTrend, err error) {
  139. body := map[string]string{
  140. "begin_date": beginDate,
  141. "end_date": endDate,
  142. }
  143. response, err := analysis.fetchData(urlStr, body)
  144. if err != nil {
  145. return
  146. }
  147. err = json.Unmarshal(response, &result)
  148. if err != nil {
  149. return
  150. }
  151. if result.ErrCode != 0 {
  152. err = fmt.Errorf("getAnalysisVisitTrend error : errcode=%v , errmsg=%v", result.ErrCode, result.ErrMsg)
  153. return
  154. }
  155. return
  156. }
  157. // GetAnalysisDailyVisitTrend 获取用户访问小程序数据日趋势
  158. func (analysis *Analysis) GetAnalysisDailyVisitTrend(beginDate, endDate string) (result ResAnalysisVisitTrend, err error) {
  159. return analysis.getAnalysisVisitTrend(getAnalysisDailyVisitTrendURL, beginDate, endDate)
  160. }
  161. // GetAnalysisMonthlyVisitTrend 获取用户访问小程序数据月趋势
  162. func (analysis *Analysis) GetAnalysisMonthlyVisitTrend(beginDate, endDate string) (result ResAnalysisVisitTrend, err error) {
  163. return analysis.getAnalysisVisitTrend(getAnalysisMonthlyVisitTrendURL, beginDate, endDate)
  164. }
  165. // GetAnalysisWeeklyVisitTrend 获取用户访问小程序数据周趋势
  166. func (analysis *Analysis) GetAnalysisWeeklyVisitTrend(beginDate, endDate string) (result ResAnalysisVisitTrend, err error) {
  167. return analysis.getAnalysisVisitTrend(getAnalysisWeeklyVisitTrendURL, beginDate, endDate)
  168. }
  169. // UserPortraitItem 用户画像项目
  170. type UserPortraitItem struct {
  171. ID int `json:"id"` // 属性值id
  172. Name string `json:"name"` // 属性值名称
  173. Value int `json:"value"` // 该场景访问uv
  174. }
  175. // UserPortrait 用户画像
  176. type UserPortrait struct {
  177. Index int `json:"index"` // 分布类型
  178. Province []UserPortraitItem `json:"province"` // 省份,如北京、广东等
  179. City []UserPortraitItem `json:"city"` // 城市,如北京、广州等
  180. Genders []UserPortraitItem `json:"genders"` // 性别,包括男、女、未知
  181. Platforms []UserPortraitItem `json:"platforms"` // 终端类型,包括iPhone, android, 其他
  182. Devices []UserPortraitItem `json:"devices"` // 机型,如苹果iPhone 6, OPPO R9等
  183. Ages []UserPortraitItem `json:"ages"` // 年龄,包括17岁以下、18-24对等区间
  184. }
  185. // ResAnalysisUserPortrait 小程序新增或活跃用户的画像分布数据返回
  186. type ResAnalysisUserPortrait struct {
  187. util.CommonError
  188. RefDate string `json:"ref_date"` // 日期
  189. VisitUVNew UserPortrait `json:"visit_uv_new"` // 新用户画像
  190. VisitUV UserPortrait `json:"visit_uv"` // 活跃用户画像
  191. }
  192. // GetAnalysisUserPortrait 获取小程序新增或活跃用户的画像分布数据
  193. func (analysis *Analysis) GetAnalysisUserPortrait(beginDate, endDate string) (result ResAnalysisUserPortrait, err error) {
  194. body := map[string]string{
  195. "begin_date": beginDate,
  196. "end_date": endDate,
  197. }
  198. response, err := analysis.fetchData(getAnalysisUserPortraitURL, body)
  199. if err != nil {
  200. return
  201. }
  202. err = json.Unmarshal(response, &result)
  203. if err != nil {
  204. return
  205. }
  206. if result.ErrCode != 0 {
  207. err = fmt.Errorf("GetAnalysisUserPortrait error : errcode=%v , errmsg=%v", result.ErrCode, result.ErrMsg)
  208. return
  209. }
  210. return
  211. }
  212. // VisitDistributionIndexItem 访问分数数据结构
  213. type VisitDistributionIndexItem struct {
  214. Key int `json:"key"` // 场景id
  215. Value int `json:"value"` // 该场景id访问pv
  216. AccessSourceVisitUV int `json:"access_source_visit_uv"` // 该场景id访问uv
  217. }
  218. // VisitDistributionIndex 访问分布单分布类型数据
  219. type VisitDistributionIndex struct {
  220. Index string `json:"index"` // 分布类型
  221. ItemList []VisitDistributionIndexItem `json:"item_list"` // 分布数据列表
  222. }
  223. // ResAnalysisVisitDistribution 小程序访问分布数据返回
  224. type ResAnalysisVisitDistribution struct {
  225. util.CommonError
  226. RefDate string `json:"ref_date"` // 日期
  227. List []VisitDistributionIndex `json:"list"` // 数据列表
  228. }
  229. // GetAnalysisVisitDistribution 获取用户小程序访问分布数据
  230. func (analysis *Analysis) GetAnalysisVisitDistribution(beginDate, endDate string) (result ResAnalysisVisitDistribution, err error) {
  231. body := map[string]string{
  232. "begin_date": beginDate,
  233. "end_date": endDate,
  234. }
  235. response, err := analysis.fetchData(getAnalysisVisitDistributionURL, body)
  236. if err != nil {
  237. return
  238. }
  239. err = json.Unmarshal(response, &result)
  240. if err != nil {
  241. return
  242. }
  243. if result.ErrCode != 0 {
  244. err = fmt.Errorf("GetAnalysisVisitDistribution error : errcode=%v , errmsg=%v", result.ErrCode, result.ErrMsg)
  245. return
  246. }
  247. return
  248. }
  249. // VisitPageItem 访问单个页面的数据结构
  250. type VisitPageItem struct {
  251. PagePath string `json:"page_path"` // 页面路径
  252. PageVisitPV int `json:"page_visit_pv"` // 访问次数
  253. PageVisitUV int `json:"page_visit_uv"` // 访问人数
  254. PageStaytimePV float64 `json:"page_staytime_pv"` // 次均停留时常
  255. EntrypagePV int `json:"entrypage_pv"` // 进入页次数
  256. ExitpagePV int `json:"exitpage_pv"` // 退出页次数
  257. PageSharePV int `json:"page_share_pv"` // 转发次数
  258. PageShareUV int `json:"page_share_uv"` // 转发人数
  259. }
  260. // ResAnalysisVisitPage 访问小程序页面访问数据返回
  261. type ResAnalysisVisitPage struct {
  262. util.CommonError
  263. RefDate string `json:"ref_date"` // 日期
  264. List []VisitPageItem `json:"list"` // 数据列表
  265. }
  266. // GetAnalysisVisitPage 获取小程序页面访问数据
  267. func (analysis *Analysis) GetAnalysisVisitPage(beginDate, endDate string) (result ResAnalysisVisitPage, err error) {
  268. body := map[string]string{
  269. "begin_date": beginDate,
  270. "end_date": endDate,
  271. }
  272. response, err := analysis.fetchData(getAnalysisVisitPageURL, body)
  273. if err != nil {
  274. return
  275. }
  276. err = json.Unmarshal(response, &result)
  277. if err != nil {
  278. return
  279. }
  280. if result.ErrCode != 0 {
  281. err = fmt.Errorf("GetAnalysisVisitPage error : errcode=%v , errmsg=%v", result.ErrCode, result.ErrMsg)
  282. return
  283. }
  284. return
  285. }