蛋蛋星球 后台端
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

1487 lines
59 KiB

  1. package egg_energy
  2. import (
  3. md2 "applet/app/md/institutional_management/egg_energy"
  4. "applet/app/utils"
  5. "code.fnuoos.com/EggPlanet/egg_system_rules.git/md"
  6. "code.fnuoos.com/go_rely_warehouse/zyos_go_es.git/es"
  7. "context"
  8. "encoding/json"
  9. "errors"
  10. "fmt"
  11. "github.com/olivere/elastic/v7"
  12. "strconv"
  13. "strings"
  14. )
  15. // 查询Elasticsearch并返回结果
  16. func QueryElasticsearch(req md2.UserEggFlowReq, indexName string) (resp []md2.UserEggFlowReqRespList, total int64, err error) {
  17. // 构建查询条件
  18. boolQuery := elastic.NewBoolQuery()
  19. if req.ScoreValueStart != "" && req.ScoreValueEnd != "" {
  20. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("score_value").Gte(req.ScoreValueStart).Lte(req.ScoreValueEnd))
  21. }
  22. if req.EcpmStart != "" && req.EcpmEnd != "" {
  23. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("ecpm").Gte(req.EcpmStart).Lte(req.EcpmEnd))
  24. }
  25. if req.InviteUserNumsStart != "" && req.InviteUserNumsEnd != "" {
  26. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("invite_user_nums").Gte(req.InviteUserNumsStart).Lte(req.InviteUserNumsEnd))
  27. }
  28. if req.TeamActivityNumsStart != "" && req.TeamActivityNumsEnd != "" {
  29. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("team_activity_nums").Gte(req.TeamActivityNumsStart).Lte(req.TeamActivityNumsEnd))
  30. }
  31. if req.SignInNumsStart != "" && req.SignInNumsEnd != "" {
  32. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("sign_in_nums").Gte(req.SignInNumsStart).Lte(req.SignInNumsEnd))
  33. }
  34. if req.ImActivityNumsStart != "" && req.ImActivityNumsEnd != "" {
  35. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("im_activity_nums").Gte(req.ImActivityNumsStart).Lte(req.ImActivityNumsEnd))
  36. }
  37. if req.SendRedPackageNumsStart != "" && req.SendRedPackageNumsEnd != "" {
  38. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("send_red_package_nums").Gte(req.SendRedPackageNumsStart).Lte(req.SendRedPackageNumsEnd))
  39. }
  40. if req.EggEnergyExchangeAccountBalanceStart != "" && req.EggEnergyExchangeAccountBalanceEnd != "" {
  41. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("egg_energy_exchange_account_balance").Gte(req.EggEnergyExchangeAccountBalanceStart).Lte(req.EggEnergyExchangeAccountBalanceEnd))
  42. }
  43. if req.AccountBalanceExchangeEggEnergyNumsStart != "" && req.AccountBalanceExchangeEggEnergyNumsEnd != "" {
  44. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("account_balance_exchange_egg_energy_nums").Gte(req.AccountBalanceExchangeEggEnergyNumsStart).Lte(req.AccountBalanceExchangeEggEnergyNumsEnd))
  45. }
  46. if req.SendCircleOfFriendNumsStart != "" && req.SendCircleOfFriendNumsEnd != "" {
  47. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("send_circle_of_friend_nums").Gte(req.SendCircleOfFriendNumsStart).Lte(req.SendCircleOfFriendNumsEnd))
  48. }
  49. if req.ForumCommentsNumsStart != "" && req.ForumCommentsNumsEnd != "" {
  50. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("forum_comments_nums").Gte(req.ForumCommentsNumsStart).Lte(req.ForumCommentsNumsEnd))
  51. }
  52. if req.CollegeLearningNumsStart != "" && req.CollegeLearningNumsEnd != "" {
  53. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("college_learning_nums").Gte(req.CollegeLearningNumsStart).Lte(req.CollegeLearningNumsEnd))
  54. }
  55. if req.ViolateNumsStart != "" && req.ViolateNumsEnd != "" {
  56. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("violate_nums").Gte(req.ViolateNumsStart).Lte(req.ViolateNumsEnd))
  57. }
  58. if req.BrowseInterfaceNumsStart != "" && req.BrowseInterfaceNumsEnd != "" {
  59. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("browse_interface_nums").Gte(req.BrowseInterfaceNumsStart).Lte(req.BrowseInterfaceNumsEnd))
  60. }
  61. if req.PersonAddActivityValueStart != "" && req.PersonAddActivityValueEnd != "" {
  62. boolQuery = boolQuery.Filter(elastic.NewRangeQuery("person_add_activity_value").Gte(req.PersonAddActivityValueStart).Lte(req.PersonAddActivityValueEnd))
  63. }
  64. // 执行查询
  65. searchResult, err := es.EsClient.Search().
  66. Index(indexName).
  67. Query(boolQuery).
  68. From((req.Page - 1) * req.PageSize).Size(req.PageSize).
  69. Pretty(true).
  70. Do(context.Background())
  71. if err != nil {
  72. if strings.Contains(err.Error(), "no such index") {
  73. return nil, 0, nil
  74. }
  75. return
  76. }
  77. // 检查是否有结果
  78. if searchResult.Hits.TotalHits.Value == 0 {
  79. return
  80. }
  81. // 解析结果
  82. for _, hit := range searchResult.Hits.Hits {
  83. var doc md2.UserEggFlowReqRespList
  84. err = json.Unmarshal(hit.Source, &doc)
  85. if err != nil {
  86. return
  87. }
  88. resp = append(resp, doc)
  89. }
  90. total = searchResult.Hits.TotalHits.Value
  91. return
  92. }
  93. // StatisticsUserEggScoreValueRange 统计用户蛋蛋分范围
  94. func StatisticsUserEggScoreValueRange(esIndexName string) (resp []map[string]string, err error) {
  95. resp = []map[string]string{}
  96. var result *elastic.SearchResult
  97. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  98. // 1、 创建 extended_stats 高级统计
  99. aggs := elastic.NewExtendedStatsAggregation().Field("score_value")
  100. result, err = es.EsClient.Search().
  101. Index(esIndexName).
  102. TrackTotalHits(true).
  103. Query(boolQuery). // 设置查询条件
  104. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  105. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  106. Pretty(true). // 返回可读的json格式
  107. Do(context.Background())
  108. if err != nil {
  109. return resp, err
  110. }
  111. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  112. agg, found := result.Aggregations.ExtendedStats("result")
  113. if !found {
  114. // 打印结果,注意:这里使用的是取值运算符
  115. return resp, errors.New("未聚合出数据")
  116. }
  117. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  118. //2、histogram 直方图聚合
  119. stdDeviation := *agg.StdDeviation //标准平方差
  120. min := *agg.Min //最小值
  121. max := *agg.Max //最大值
  122. avg := *agg.Avg //平均数
  123. totalCount := agg.Count //总数
  124. discreteCoefficient := stdDeviation / avg //离散系数
  125. newAggs := elastic.NewHistogramAggregation().Field("score_value").Interval(discreteCoefficient).ExtendedBounds(min, max)
  126. searchResult, err := es.EsClient.Search().
  127. Index(esIndexName).
  128. TrackTotalHits(true).
  129. Query(boolQuery). // 设置查询条件
  130. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  131. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  132. Pretty(true). // 返回可读的json格式
  133. Do(context.Background())
  134. if err != nil {
  135. return resp, err
  136. }
  137. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  138. newAgg, found := searchResult.Aggregations.Histogram("result")
  139. if !found {
  140. return resp, errors.New("未聚合出数据")
  141. }
  142. // 3、组装数据
  143. var keys []string
  144. var values []int64
  145. var tempTotalDocCount int64
  146. var endStart, endEnd string
  147. for _, bucket := range newAgg.Buckets {
  148. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  149. if bucket.Key <= stdDeviation {
  150. bucketValue := bucket.Key
  151. keys = append(keys, utils.Float64ToStr(bucketValue))
  152. values = append(values, bucket.DocCount)
  153. } else {
  154. bucketValue := bucket.Key
  155. if tempTotalDocCount == 0 {
  156. endStart = utils.Float64ToStr(bucketValue)
  157. }
  158. tempTotalDocCount += bucket.DocCount
  159. endEnd = utils.Float64ToStr(bucketValue)
  160. }
  161. }
  162. for k, v := range keys {
  163. value, _ := Operate(values[k], totalCount)
  164. if k+1 == len(keys) {
  165. resp = append(resp, map[string]string{
  166. "key": v + " ~ " + utils.AnyToString(max) + " 元",
  167. "value": utils.Float64ToStrPrec10(value),
  168. })
  169. } else {
  170. resp = append(resp, map[string]string{
  171. "key": v + " ~ " + keys[k+1] + " 元",
  172. "value": utils.Float64ToStrPrec10(value),
  173. })
  174. }
  175. }
  176. if tempTotalDocCount > 0 {
  177. temp, _ := Operate(tempTotalDocCount, totalCount)
  178. resp = append(resp, map[string]string{
  179. "key": endStart + " ~ " + endEnd + " 元",
  180. "value": utils.Float64ToStrPrec10(temp),
  181. })
  182. }
  183. }
  184. return resp, nil
  185. }
  186. // StatisticsUserEggKindProportion 统计用户蛋蛋分"评比类型"占比
  187. func StatisticsUserEggKindProportion(esIndexName string) (result []map[string]interface{}, err error) {
  188. var res *elastic.SearchResult
  189. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  190. aggs := elastic.NewTermsAggregation().Field("score_value_kind")
  191. res, err = es.EsClient.Search().
  192. Index(esIndexName).
  193. TrackTotalHits(true).
  194. Query(boolQuery). // 设置查询条件
  195. Aggregation("terms", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  196. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  197. Pretty(true). // 返回可读的json格式
  198. Do(context.Background())
  199. if err != nil {
  200. return result, err
  201. }
  202. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  203. agg, _ := res.Aggregations.Terms("terms")
  204. var total float64
  205. for _, bucket := range agg.Buckets {
  206. var tmp = map[string]interface{}{}
  207. bucketValue := bucket.Key
  208. tmp["key"] = bucketValue
  209. tmp["count"] = bucket.DocCount
  210. switch bucketValue {
  211. case 1:
  212. tmp["name"] = "人工"
  213. break
  214. case 2:
  215. tmp["name"] = "自动"
  216. break
  217. default:
  218. tmp["name"] = "未知"
  219. }
  220. result = append(result, tmp)
  221. total += utils.AnyToFloat64(bucket.DocCount)
  222. }
  223. for _, value := range result {
  224. value["proportion"] = utils.AnyToString(utils.AnyToFloat64(value["count"]) / total)
  225. }
  226. return result, nil
  227. }
  228. // StatisticsUserEggEcpmRange 统计用户"ecpm"范围
  229. func StatisticsUserEggEcpmRange(esIndexName string) (resp []map[string]string, err error) {
  230. resp = []map[string]string{}
  231. var result *elastic.SearchResult
  232. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  233. // 1、 创建 extended_stats 高级统计
  234. aggs := elastic.NewExtendedStatsAggregation().Field("ecpm")
  235. result, err = es.EsClient.Search().
  236. Index(esIndexName).
  237. TrackTotalHits(true).
  238. Query(boolQuery). // 设置查询条件
  239. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  240. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  241. Pretty(true). // 返回可读的json格式
  242. Do(context.Background())
  243. if err != nil {
  244. return resp, err
  245. }
  246. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  247. agg, found := result.Aggregations.ExtendedStats("result")
  248. if !found {
  249. // 打印结果,注意:这里使用的是取值运算符
  250. return resp, errors.New("未聚合出数据")
  251. }
  252. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  253. //2、histogram 直方图聚合
  254. stdDeviation := *agg.StdDeviation //标准平方差
  255. min := *agg.Min //最小值
  256. max := *agg.Max //最大值
  257. avg := *agg.Avg //平均数
  258. totalCount := agg.Count //总数
  259. discreteCoefficient := stdDeviation / avg //离散系数
  260. if stdDeviation < avg {
  261. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  262. discreteCoefficient = stdDeviation
  263. }
  264. newAggs := elastic.NewHistogramAggregation().Field("ecpm").Interval(discreteCoefficient).ExtendedBounds(min, max)
  265. searchResult, err := es.EsClient.Search().
  266. Index(esIndexName).
  267. TrackTotalHits(true).
  268. Query(boolQuery). // 设置查询条件
  269. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  270. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  271. Pretty(true). // 返回可读的json格式
  272. Do(context.Background())
  273. if err != nil {
  274. return resp, err
  275. }
  276. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  277. newAgg, found := searchResult.Aggregations.Histogram("result")
  278. if !found {
  279. return resp, errors.New("未聚合出数据")
  280. }
  281. // 3、组装数据
  282. var keys []string
  283. var values []int64
  284. for _, bucket := range newAgg.Buckets {
  285. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  286. bucketValue := bucket.Key
  287. keys = append(keys, utils.Float64ToStr(bucketValue))
  288. values = append(values, bucket.DocCount)
  289. }
  290. for k, v := range keys {
  291. value, _ := Operate(values[k], totalCount)
  292. if k+1 == len(keys) {
  293. resp = append(resp, map[string]string{
  294. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  295. "value": utils.Float64ToStrPrec10(value),
  296. })
  297. } else {
  298. resp = append(resp, map[string]string{
  299. "key": v + " ~ " + keys[k+1] + " 元",
  300. "value": utils.Float64ToStrPrec10(value),
  301. })
  302. }
  303. }
  304. }
  305. return resp, nil
  306. }
  307. func StatisticsUserEggEcpmRange1(esIndexName string) (resp []map[string]string, err error) {
  308. resp = []map[string]string{}
  309. var result *elastic.SearchResult
  310. // 创建百分位数聚合
  311. percentBoolQuery := elastic.NewRangeQuery("ecpm").Gt(0)
  312. percentilesAgg := elastic.NewPercentilesAggregation().Field("ecpm").Percentiles([]float64{5, 90}...)
  313. // 执行查询
  314. response, err := es.EsClient.Search().
  315. Index(esIndexName).
  316. Query(percentBoolQuery).
  317. Aggregation("percentiles_agg", percentilesAgg).
  318. Size(0).
  319. Do(context.Background())
  320. if err != nil {
  321. // 处理错误
  322. return resp, err
  323. }
  324. // 获取百分位数结果
  325. percentiles, found := response.Aggregations.Percentiles("percentiles_agg")
  326. if !found {
  327. return resp, err
  328. }
  329. // 获取第10和第90百分位数
  330. p5 := percentiles.Values["5.0"]
  331. p90 := percentiles.Values["90.0"]
  332. boolQuery := elastic.NewRangeQuery("ecpm").Gte(p5).Lte(p90)
  333. // 1、 创建 extended_stats 高级统计
  334. aggs := elastic.NewExtendedStatsAggregation().Field("ecpm")
  335. result, err = es.EsClient.Search().
  336. Index(esIndexName).
  337. TrackTotalHits(true).
  338. Query(boolQuery). // 设置查询条件
  339. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  340. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  341. Pretty(true). // 返回可读的json格式
  342. Do(context.Background())
  343. if err != nil {
  344. return resp, err
  345. }
  346. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  347. agg, found := result.Aggregations.ExtendedStats("result")
  348. if !found {
  349. // 打印结果,注意:这里使用的是取值运算符
  350. return resp, errors.New("未聚合出数据")
  351. }
  352. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  353. //2、histogram 直方图聚合
  354. stdDeviation := *agg.StdDeviation //标准平方差
  355. min := *agg.Min //最小值
  356. max := *agg.Max //最大值
  357. avg := *agg.Avg //平均数
  358. totalCount := agg.Count //总数
  359. discreteCoefficient := stdDeviation / avg //离散系数
  360. if stdDeviation < avg {
  361. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  362. discreteCoefficient = stdDeviation
  363. }
  364. newAggs := elastic.NewHistogramAggregation().Field("ecpm").Interval(discreteCoefficient).MinDocCount(1).ExtendedBounds(min, max)
  365. searchResult, err := es.EsClient.Search().
  366. Index(esIndexName).
  367. TrackTotalHits(true).
  368. Query(boolQuery). // 设置查询条件
  369. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  370. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  371. Pretty(true). // 返回可读的json格式
  372. Do(context.Background())
  373. if err != nil {
  374. return resp, err
  375. }
  376. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  377. newAgg, found := searchResult.Aggregations.Histogram("result")
  378. if !found {
  379. return resp, errors.New("未聚合出数据")
  380. }
  381. // 3、组装数据
  382. var keys []string
  383. var values []int64
  384. for _, bucket := range newAgg.Buckets {
  385. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  386. bucketValue := bucket.Key
  387. keys = append(keys, utils.Float64ToStr(bucketValue))
  388. values = append(values, bucket.DocCount)
  389. }
  390. for k, v := range keys {
  391. value, _ := Operate(values[k], totalCount)
  392. if k+1 == len(keys) {
  393. resp = append(resp, map[string]string{
  394. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  395. "value": utils.Float64ToStrPrec10(value),
  396. })
  397. } else {
  398. resp = append(resp, map[string]string{
  399. "key": v + " ~ " + keys[k+1] + " 元",
  400. "value": utils.Float64ToStrPrec10(value),
  401. })
  402. }
  403. }
  404. }
  405. fmt.Println(resp)
  406. return resp, nil
  407. }
  408. // StatisticsUserEggInviteUserNumsRange 统计用户"拉新人数"范围
  409. func StatisticsUserEggInviteUserNumsRange(esIndexName string) (resp []map[string]string, err error) {
  410. resp = []map[string]string{}
  411. var result *elastic.SearchResult
  412. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  413. // 1、 创建 extended_stats 高级统计
  414. aggs := elastic.NewExtendedStatsAggregation().Field("invite_user_nums")
  415. result, err = es.EsClient.Search().
  416. Index(esIndexName).
  417. TrackTotalHits(true).
  418. Query(boolQuery). // 设置查询条件
  419. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  420. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  421. Pretty(true). // 返回可读的json格式
  422. Do(context.Background())
  423. if err != nil {
  424. return resp, err
  425. }
  426. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  427. agg, found := result.Aggregations.ExtendedStats("result")
  428. if !found {
  429. // 打印结果,注意:这里使用的是取值运算符
  430. return resp, errors.New("未聚合出数据")
  431. }
  432. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  433. //2、histogram 直方图聚合
  434. stdDeviation := *agg.StdDeviation //标准平方差
  435. min := *agg.Min //最小值
  436. max := *agg.Max //最大值
  437. avg := *agg.Avg //平均数
  438. totalCount := agg.Count //总数
  439. discreteCoefficient := stdDeviation / avg //离散系数
  440. if stdDeviation < avg {
  441. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  442. discreteCoefficient = stdDeviation
  443. }
  444. newAggs := elastic.NewHistogramAggregation().Field("invite_user_nums").Interval(discreteCoefficient).ExtendedBounds(min, max)
  445. searchResult, err := es.EsClient.Search().
  446. Index(esIndexName).
  447. TrackTotalHits(true).
  448. Query(boolQuery). // 设置查询条件
  449. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  450. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  451. Pretty(true). // 返回可读的json格式
  452. Do(context.Background())
  453. if err != nil {
  454. return resp, err
  455. }
  456. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  457. newAgg, found := searchResult.Aggregations.Histogram("result")
  458. if !found {
  459. return resp, errors.New("未聚合出数据")
  460. }
  461. // 3、组装数据
  462. var keys []string
  463. var values []int64
  464. for _, bucket := range newAgg.Buckets {
  465. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  466. bucketValue := bucket.Key
  467. keys = append(keys, utils.Float64ToStr(bucketValue))
  468. values = append(values, bucket.DocCount)
  469. }
  470. for k, v := range keys {
  471. value, _ := Operate(values[k], totalCount)
  472. if k+1 == len(keys) {
  473. resp = append(resp, map[string]string{
  474. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  475. "value": utils.Float64ToStrPrec10(value),
  476. })
  477. } else {
  478. resp = append(resp, map[string]string{
  479. "key": v + " ~ " + keys[k+1] + " 元",
  480. "value": utils.Float64ToStrPrec10(value),
  481. })
  482. }
  483. }
  484. }
  485. return resp, nil
  486. }
  487. // StatisticsUserEggTeamActivityNumsRange 统计用户"团队活跃次数"范围
  488. func StatisticsUserEggTeamActivityNumsRange(esIndexName string) (resp []map[string]string, err error) {
  489. resp = []map[string]string{}
  490. var result *elastic.SearchResult
  491. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  492. // 1、 创建 extended_stats 高级统计
  493. aggs := elastic.NewExtendedStatsAggregation().Field("team_activity_nums")
  494. result, err = es.EsClient.Search().
  495. Index(esIndexName).
  496. TrackTotalHits(true).
  497. Query(boolQuery). // 设置查询条件
  498. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  499. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  500. Pretty(true). // 返回可读的json格式
  501. Do(context.Background())
  502. if err != nil {
  503. return resp, err
  504. }
  505. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  506. agg, found := result.Aggregations.ExtendedStats("result")
  507. if !found {
  508. // 打印结果,注意:这里使用的是取值运算符
  509. return resp, errors.New("未聚合出数据")
  510. }
  511. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  512. //2、histogram 直方图聚合
  513. stdDeviation := *agg.StdDeviation //标准平方差
  514. min := *agg.Min //最小值
  515. max := *agg.Max //最大值
  516. avg := *agg.Avg //平均数
  517. totalCount := agg.Count //总数
  518. discreteCoefficient := stdDeviation / avg //离散系数
  519. if stdDeviation < avg {
  520. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  521. discreteCoefficient = stdDeviation
  522. }
  523. newAggs := elastic.NewHistogramAggregation().Field("team_activity_nums").Interval(discreteCoefficient).ExtendedBounds(min, max)
  524. searchResult, err := es.EsClient.Search().
  525. Index(esIndexName).
  526. TrackTotalHits(true).
  527. Query(boolQuery). // 设置查询条件
  528. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  529. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  530. Pretty(true). // 返回可读的json格式
  531. Do(context.Background())
  532. if err != nil {
  533. return resp, err
  534. }
  535. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  536. newAgg, found := searchResult.Aggregations.Histogram("result")
  537. if !found {
  538. return resp, errors.New("未聚合出数据")
  539. }
  540. // 3、组装数据
  541. var keys []string
  542. var values []int64
  543. for _, bucket := range newAgg.Buckets {
  544. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  545. bucketValue := bucket.Key
  546. keys = append(keys, utils.Float64ToStr(bucketValue))
  547. values = append(values, bucket.DocCount)
  548. }
  549. for k, v := range keys {
  550. value, _ := Operate(values[k], totalCount)
  551. if k+1 == len(keys) {
  552. resp = append(resp, map[string]string{
  553. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  554. "value": utils.Float64ToStrPrec10(value),
  555. })
  556. } else {
  557. resp = append(resp, map[string]string{
  558. "key": v + " ~ " + keys[k+1] + " 元",
  559. "value": utils.Float64ToStrPrec10(value),
  560. })
  561. }
  562. }
  563. }
  564. return resp, nil
  565. }
  566. // StatisticsUserEggSignInNumsRange 统计用户"签到次数"范围
  567. func StatisticsUserEggSignInNumsRange(esIndexName string) (resp []map[string]string, err error) {
  568. resp = []map[string]string{}
  569. var result *elastic.SearchResult
  570. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  571. // 1、 创建 extended_stats 高级统计
  572. aggs := elastic.NewExtendedStatsAggregation().Field("sign_in_nums")
  573. result, err = es.EsClient.Search().
  574. Index(esIndexName).
  575. TrackTotalHits(true).
  576. Query(boolQuery). // 设置查询条件
  577. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  578. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  579. Pretty(true). // 返回可读的json格式
  580. Do(context.Background())
  581. if err != nil {
  582. return resp, err
  583. }
  584. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  585. agg, found := result.Aggregations.ExtendedStats("result")
  586. if !found {
  587. // 打印结果,注意:这里使用的是取值运算符
  588. return resp, errors.New("未聚合出数据")
  589. }
  590. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  591. //2、histogram 直方图聚合
  592. stdDeviation := *agg.StdDeviation //标准平方差
  593. min := *agg.Min //最小值
  594. max := *agg.Max //最大值
  595. avg := *agg.Avg //平均数
  596. totalCount := agg.Count //总数
  597. discreteCoefficient := stdDeviation / avg //离散系数
  598. if stdDeviation < avg {
  599. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  600. discreteCoefficient = stdDeviation
  601. }
  602. newAggs := elastic.NewHistogramAggregation().Field("sign_in_nums").Interval(discreteCoefficient).ExtendedBounds(min, max)
  603. searchResult, err := es.EsClient.Search().
  604. Index(esIndexName).
  605. TrackTotalHits(true).
  606. Query(boolQuery). // 设置查询条件
  607. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  608. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  609. Pretty(true). // 返回可读的json格式
  610. Do(context.Background())
  611. if err != nil {
  612. return resp, err
  613. }
  614. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  615. newAgg, found := searchResult.Aggregations.Histogram("result")
  616. if !found {
  617. return resp, errors.New("未聚合出数据")
  618. }
  619. // 3、组装数据
  620. var keys []string
  621. var values []int64
  622. for _, bucket := range newAgg.Buckets {
  623. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  624. bucketValue := bucket.Key
  625. keys = append(keys, utils.Float64ToStr(bucketValue))
  626. values = append(values, bucket.DocCount)
  627. }
  628. for k, v := range keys {
  629. value, _ := Operate(values[k], totalCount)
  630. if k+1 == len(keys) {
  631. resp = append(resp, map[string]string{
  632. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  633. "value": utils.Float64ToStrPrec10(value),
  634. })
  635. } else {
  636. resp = append(resp, map[string]string{
  637. "key": v + " ~ " + keys[k+1] + " 元",
  638. "value": utils.Float64ToStrPrec10(value),
  639. })
  640. }
  641. }
  642. }
  643. return resp, nil
  644. }
  645. // StatisticsUserEggSendRedPackageNumsRange 统计用户"发红包次数"范围
  646. func StatisticsUserEggSendRedPackageNumsRange(esIndexName string) (resp []map[string]string, err error) {
  647. resp = []map[string]string{}
  648. var result *elastic.SearchResult
  649. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  650. // 1、 创建 extended_stats 高级统计
  651. aggs := elastic.NewExtendedStatsAggregation().Field("send_red_package_nums")
  652. result, err = es.EsClient.Search().
  653. Index(esIndexName).
  654. TrackTotalHits(true).
  655. Query(boolQuery). // 设置查询条件
  656. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  657. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  658. Pretty(true). // 返回可读的json格式
  659. Do(context.Background())
  660. if err != nil {
  661. return resp, err
  662. }
  663. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  664. agg, found := result.Aggregations.ExtendedStats("result")
  665. if !found {
  666. // 打印结果,注意:这里使用的是取值运算符
  667. return resp, errors.New("未聚合出数据")
  668. }
  669. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  670. //2、histogram 直方图聚合
  671. stdDeviation := *agg.StdDeviation //标准平方差
  672. min := *agg.Min //最小值
  673. max := *agg.Max //最大值
  674. avg := *agg.Avg //平均数
  675. totalCount := agg.Count //总数
  676. discreteCoefficient := stdDeviation / avg //离散系数
  677. if stdDeviation < avg {
  678. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  679. discreteCoefficient = stdDeviation
  680. }
  681. newAggs := elastic.NewHistogramAggregation().Field("send_red_package_nums").Interval(discreteCoefficient).ExtendedBounds(min, max)
  682. searchResult, err := es.EsClient.Search().
  683. Index(esIndexName).
  684. TrackTotalHits(true).
  685. Query(boolQuery). // 设置查询条件
  686. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  687. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  688. Pretty(true). // 返回可读的json格式
  689. Do(context.Background())
  690. if err != nil {
  691. return resp, err
  692. }
  693. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  694. newAgg, found := searchResult.Aggregations.Histogram("result")
  695. if !found {
  696. return resp, errors.New("未聚合出数据")
  697. }
  698. // 3、组装数据
  699. var keys []string
  700. var values []int64
  701. for _, bucket := range newAgg.Buckets {
  702. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  703. bucketValue := bucket.Key
  704. keys = append(keys, utils.Float64ToStr(bucketValue))
  705. values = append(values, bucket.DocCount)
  706. }
  707. for k, v := range keys {
  708. value, _ := Operate(values[k], totalCount)
  709. if k+1 == len(keys) {
  710. resp = append(resp, map[string]string{
  711. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  712. "value": utils.Float64ToStrPrec10(value),
  713. })
  714. } else {
  715. resp = append(resp, map[string]string{
  716. "key": v + " ~ " + keys[k+1] + " 元",
  717. "value": utils.Float64ToStrPrec10(value),
  718. })
  719. }
  720. }
  721. }
  722. return resp, nil
  723. }
  724. // StatisticsUserEggEggEnergyExchangeAccountBalanceRange 统计用户"蛋蛋能量兑换余额数量"范围
  725. func StatisticsUserEggEggEnergyExchangeAccountBalanceRange(esIndexName string) (resp []map[string]string, err error) {
  726. resp = []map[string]string{}
  727. var result *elastic.SearchResult
  728. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  729. // 1、 创建 extended_stats 高级统计
  730. aggs := elastic.NewExtendedStatsAggregation().Field("account_balance_exchange_egg_energy_nums")
  731. result, err = es.EsClient.Search().
  732. Index(esIndexName).
  733. TrackTotalHits(true).
  734. Query(boolQuery). // 设置查询条件
  735. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  736. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  737. Pretty(true). // 返回可读的json格式
  738. Do(context.Background())
  739. if err != nil {
  740. return resp, err
  741. }
  742. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  743. agg, found := result.Aggregations.ExtendedStats("result")
  744. if !found {
  745. // 打印结果,注意:这里使用的是取值运算符
  746. return resp, errors.New("未聚合出数据")
  747. }
  748. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  749. //2、histogram 直方图聚合
  750. stdDeviation := *agg.StdDeviation //标准平方差
  751. min := *agg.Min //最小值
  752. max := *agg.Max //最大值
  753. avg := *agg.Avg //平均数
  754. totalCount := agg.Count //总数
  755. discreteCoefficient := stdDeviation / avg //离散系数
  756. if stdDeviation < avg {
  757. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  758. discreteCoefficient = stdDeviation
  759. }
  760. newAggs := elastic.NewHistogramAggregation().Field("account_balance_exchange_egg_energy_nums").Interval(discreteCoefficient).ExtendedBounds(min, max)
  761. searchResult, err := es.EsClient.Search().
  762. Index(esIndexName).
  763. TrackTotalHits(true).
  764. Query(boolQuery). // 设置查询条件
  765. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  766. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  767. Pretty(true). // 返回可读的json格式
  768. Do(context.Background())
  769. if err != nil {
  770. return resp, err
  771. }
  772. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  773. newAgg, found := searchResult.Aggregations.Histogram("result")
  774. if !found {
  775. return resp, errors.New("未聚合出数据")
  776. }
  777. // 3、组装数据
  778. var keys []string
  779. var values []int64
  780. for _, bucket := range newAgg.Buckets {
  781. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  782. bucketValue := bucket.Key
  783. keys = append(keys, utils.Float64ToStr(bucketValue))
  784. values = append(values, bucket.DocCount)
  785. }
  786. for k, v := range keys {
  787. value, _ := Operate(values[k], totalCount)
  788. if k+1 == len(keys) {
  789. resp = append(resp, map[string]string{
  790. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  791. "value": utils.Float64ToStrPrec10(value),
  792. })
  793. } else {
  794. resp = append(resp, map[string]string{
  795. "key": v + " ~ " + keys[k+1] + " 元",
  796. "value": utils.Float64ToStrPrec10(value),
  797. })
  798. }
  799. }
  800. }
  801. return resp, nil
  802. }
  803. // StatisticsUserEggAccountBalanceExchangeEggEnergyNumsRange 统计用户"余额兑换蛋蛋能量数量"范围
  804. func StatisticsUserEggAccountBalanceExchangeEggEnergyNumsRange(esIndexName string) (resp []map[string]string, err error) {
  805. resp = []map[string]string{}
  806. var result *elastic.SearchResult
  807. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  808. // 1、 创建 extended_stats 高级统计
  809. aggs := elastic.NewExtendedStatsAggregation().Field("egg_energy_exchange_account_balance")
  810. result, err = es.EsClient.Search().
  811. Index(esIndexName).
  812. TrackTotalHits(true).
  813. Query(boolQuery). // 设置查询条件
  814. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  815. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  816. Pretty(true). // 返回可读的json格式
  817. Do(context.Background())
  818. if err != nil {
  819. return resp, err
  820. }
  821. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  822. agg, found := result.Aggregations.ExtendedStats("result")
  823. if !found {
  824. // 打印结果,注意:这里使用的是取值运算符
  825. return resp, errors.New("未聚合出数据")
  826. }
  827. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  828. //2、histogram 直方图聚合
  829. stdDeviation := *agg.StdDeviation //标准平方差
  830. min := *agg.Min //最小值
  831. max := *agg.Max //最大值
  832. avg := *agg.Avg //平均数
  833. totalCount := agg.Count //总数
  834. discreteCoefficient := stdDeviation / avg //离散系数
  835. if stdDeviation < avg {
  836. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  837. discreteCoefficient = stdDeviation
  838. }
  839. newAggs := elastic.NewHistogramAggregation().Field("egg_energy_exchange_account_balance").Interval(discreteCoefficient).ExtendedBounds(min, max)
  840. searchResult, err := es.EsClient.Search().
  841. Index(esIndexName).
  842. TrackTotalHits(true).
  843. Query(boolQuery). // 设置查询条件
  844. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  845. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  846. Pretty(true). // 返回可读的json格式
  847. Do(context.Background())
  848. if err != nil {
  849. return resp, err
  850. }
  851. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  852. newAgg, found := searchResult.Aggregations.Histogram("result")
  853. if !found {
  854. return resp, errors.New("未聚合出数据")
  855. }
  856. // 3、组装数据
  857. var keys []string
  858. var values []int64
  859. for _, bucket := range newAgg.Buckets {
  860. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  861. bucketValue := bucket.Key
  862. keys = append(keys, utils.Float64ToStr(bucketValue))
  863. values = append(values, bucket.DocCount)
  864. }
  865. for k, v := range keys {
  866. value, _ := Operate(values[k], totalCount)
  867. if k+1 == len(keys) {
  868. resp = append(resp, map[string]string{
  869. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  870. "value": utils.Float64ToStrPrec10(value),
  871. })
  872. } else {
  873. resp = append(resp, map[string]string{
  874. "key": v + " ~ " + keys[k+1] + " 元",
  875. "value": utils.Float64ToStrPrec10(value),
  876. })
  877. }
  878. }
  879. }
  880. return resp, nil
  881. }
  882. // StatisticsUserEggSendCircleOfFriendNumsRange 统计用户"发朋友圈次数"范围
  883. func StatisticsUserEggSendCircleOfFriendNumsRange(esIndexName string) (resp []map[string]string, err error) {
  884. resp = []map[string]string{}
  885. var result *elastic.SearchResult
  886. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  887. // 1、 创建 extended_stats 高级统计
  888. aggs := elastic.NewExtendedStatsAggregation().Field("send_circle_of_friend_nums")
  889. result, err = es.EsClient.Search().
  890. Index(esIndexName).
  891. TrackTotalHits(true).
  892. Query(boolQuery). // 设置查询条件
  893. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  894. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  895. Pretty(true). // 返回可读的json格式
  896. Do(context.Background())
  897. if err != nil {
  898. return resp, err
  899. }
  900. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  901. agg, found := result.Aggregations.ExtendedStats("result")
  902. if !found {
  903. // 打印结果,注意:这里使用的是取值运算符
  904. return resp, errors.New("未聚合出数据")
  905. }
  906. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  907. //2、histogram 直方图聚合
  908. stdDeviation := *agg.StdDeviation //标准平方差
  909. min := *agg.Min //最小值
  910. max := *agg.Max //最大值
  911. avg := *agg.Avg //平均数
  912. totalCount := agg.Count //总数
  913. discreteCoefficient := stdDeviation / avg //离散系数
  914. if stdDeviation < avg {
  915. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  916. discreteCoefficient = stdDeviation
  917. }
  918. newAggs := elastic.NewHistogramAggregation().Field("send_circle_of_friend_nums").Interval(discreteCoefficient).ExtendedBounds(min, max)
  919. searchResult, err := es.EsClient.Search().
  920. Index(esIndexName).
  921. TrackTotalHits(true).
  922. Query(boolQuery). // 设置查询条件
  923. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  924. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  925. Pretty(true). // 返回可读的json格式
  926. Do(context.Background())
  927. if err != nil {
  928. return resp, err
  929. }
  930. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  931. newAgg, found := searchResult.Aggregations.Histogram("result")
  932. if !found {
  933. return resp, errors.New("未聚合出数据")
  934. }
  935. // 3、组装数据
  936. var keys []string
  937. var values []int64
  938. for _, bucket := range newAgg.Buckets {
  939. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  940. bucketValue := bucket.Key
  941. keys = append(keys, utils.Float64ToStr(bucketValue))
  942. values = append(values, bucket.DocCount)
  943. }
  944. for k, v := range keys {
  945. value, _ := Operate(values[k], totalCount)
  946. if k+1 == len(keys) {
  947. resp = append(resp, map[string]string{
  948. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  949. "value": utils.Float64ToStrPrec10(value),
  950. })
  951. } else {
  952. resp = append(resp, map[string]string{
  953. "key": v + " ~ " + keys[k+1] + " 元",
  954. "value": utils.Float64ToStrPrec10(value),
  955. })
  956. }
  957. }
  958. }
  959. return resp, nil
  960. }
  961. // StatisticsUserEggForumCommentsNumsRange 统计用户"论坛评论次数"范围
  962. func StatisticsUserEggForumCommentsNumsRange(esIndexName string) (resp []map[string]string, err error) {
  963. resp = []map[string]string{}
  964. var result *elastic.SearchResult
  965. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  966. // 1、 创建 extended_stats 高级统计
  967. aggs := elastic.NewExtendedStatsAggregation().Field("forum_comments_nums")
  968. result, err = es.EsClient.Search().
  969. Index(esIndexName).
  970. TrackTotalHits(true).
  971. Query(boolQuery). // 设置查询条件
  972. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  973. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  974. Pretty(true). // 返回可读的json格式
  975. Do(context.Background())
  976. if err != nil {
  977. return resp, err
  978. }
  979. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  980. agg, found := result.Aggregations.ExtendedStats("result")
  981. if !found {
  982. // 打印结果,注意:这里使用的是取值运算符
  983. return resp, errors.New("未聚合出数据")
  984. }
  985. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  986. //2、histogram 直方图聚合
  987. stdDeviation := *agg.StdDeviation //标准平方差
  988. min := *agg.Min //最小值
  989. max := *agg.Max //最大值
  990. avg := *agg.Avg //平均数
  991. totalCount := agg.Count //总数
  992. discreteCoefficient := stdDeviation / avg //离散系数
  993. if stdDeviation < avg {
  994. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  995. discreteCoefficient = stdDeviation
  996. }
  997. newAggs := elastic.NewHistogramAggregation().Field("forum_comments_nums").Interval(discreteCoefficient).ExtendedBounds(min, max)
  998. searchResult, err := es.EsClient.Search().
  999. Index(esIndexName).
  1000. TrackTotalHits(true).
  1001. Query(boolQuery). // 设置查询条件
  1002. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  1003. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  1004. Pretty(true). // 返回可读的json格式
  1005. Do(context.Background())
  1006. if err != nil {
  1007. return resp, err
  1008. }
  1009. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  1010. newAgg, found := searchResult.Aggregations.Histogram("result")
  1011. if !found {
  1012. return resp, errors.New("未聚合出数据")
  1013. }
  1014. // 3、组装数据
  1015. var keys []string
  1016. var values []int64
  1017. for _, bucket := range newAgg.Buckets {
  1018. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  1019. bucketValue := bucket.Key
  1020. keys = append(keys, utils.Float64ToStr(bucketValue))
  1021. values = append(values, bucket.DocCount)
  1022. }
  1023. for k, v := range keys {
  1024. value, _ := Operate(values[k], totalCount)
  1025. if k+1 == len(keys) {
  1026. resp = append(resp, map[string]string{
  1027. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  1028. "value": utils.Float64ToStrPrec10(value),
  1029. })
  1030. } else {
  1031. resp = append(resp, map[string]string{
  1032. "key": v + " ~ " + keys[k+1] + " 元",
  1033. "value": utils.Float64ToStrPrec10(value),
  1034. })
  1035. }
  1036. }
  1037. }
  1038. return resp, nil
  1039. }
  1040. // StatisticsUserEggCollegeLearningNumsRange 统计用户"学院学习次数"范围
  1041. func StatisticsUserEggCollegeLearningNumsRange(esIndexName string) (resp []map[string]string, err error) {
  1042. resp = []map[string]string{}
  1043. var result *elastic.SearchResult
  1044. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  1045. // 1、 创建 extended_stats 高级统计
  1046. aggs := elastic.NewExtendedStatsAggregation().Field("college_learning_nums")
  1047. result, err = es.EsClient.Search().
  1048. Index(esIndexName).
  1049. TrackTotalHits(true).
  1050. Query(boolQuery). // 设置查询条件
  1051. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  1052. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  1053. Pretty(true). // 返回可读的json格式
  1054. Do(context.Background())
  1055. if err != nil {
  1056. return resp, err
  1057. }
  1058. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  1059. agg, found := result.Aggregations.ExtendedStats("result")
  1060. if !found {
  1061. // 打印结果,注意:这里使用的是取值运算符
  1062. return resp, errors.New("未聚合出数据")
  1063. }
  1064. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  1065. //2、histogram 直方图聚合
  1066. stdDeviation := *agg.StdDeviation //标准平方差
  1067. min := *agg.Min //最小值
  1068. max := *agg.Max //最大值
  1069. avg := *agg.Avg //平均数
  1070. totalCount := agg.Count //总数
  1071. discreteCoefficient := stdDeviation / avg //离散系数
  1072. if stdDeviation < avg {
  1073. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  1074. discreteCoefficient = stdDeviation
  1075. }
  1076. newAggs := elastic.NewHistogramAggregation().Field("college_learning_nums").Interval(discreteCoefficient).ExtendedBounds(min, max)
  1077. searchResult, err := es.EsClient.Search().
  1078. Index(esIndexName).
  1079. TrackTotalHits(true).
  1080. Query(boolQuery). // 设置查询条件
  1081. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  1082. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  1083. Pretty(true). // 返回可读的json格式
  1084. Do(context.Background())
  1085. if err != nil {
  1086. return resp, err
  1087. }
  1088. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  1089. newAgg, found := searchResult.Aggregations.Histogram("result")
  1090. if !found {
  1091. return resp, errors.New("未聚合出数据")
  1092. }
  1093. // 3、组装数据
  1094. var keys []string
  1095. var values []int64
  1096. for _, bucket := range newAgg.Buckets {
  1097. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  1098. bucketValue := bucket.Key
  1099. keys = append(keys, utils.Float64ToStr(bucketValue))
  1100. values = append(values, bucket.DocCount)
  1101. }
  1102. for k, v := range keys {
  1103. value, _ := Operate(values[k], totalCount)
  1104. if k+1 == len(keys) {
  1105. resp = append(resp, map[string]string{
  1106. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  1107. "value": utils.Float64ToStrPrec10(value),
  1108. })
  1109. } else {
  1110. resp = append(resp, map[string]string{
  1111. "key": v + " ~ " + keys[k+1] + " 元",
  1112. "value": utils.Float64ToStrPrec10(value),
  1113. })
  1114. }
  1115. }
  1116. }
  1117. return resp, nil
  1118. }
  1119. // StatisticsUserEggViolateNumsRange 统计用户"违规次数"范围
  1120. func StatisticsUserEggViolateNumsRange(esIndexName string) (resp []map[string]string, err error) {
  1121. resp = []map[string]string{}
  1122. var result *elastic.SearchResult
  1123. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  1124. // 1、 创建 extended_stats 高级统计
  1125. aggs := elastic.NewExtendedStatsAggregation().Field("violate_nums")
  1126. result, err = es.EsClient.Search().
  1127. Index(esIndexName).
  1128. TrackTotalHits(true).
  1129. Query(boolQuery). // 设置查询条件
  1130. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  1131. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  1132. Pretty(true). // 返回可读的json格式
  1133. Do(context.Background())
  1134. if err != nil {
  1135. return resp, err
  1136. }
  1137. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  1138. agg, found := result.Aggregations.ExtendedStats("result")
  1139. if !found {
  1140. // 打印结果,注意:这里使用的是取值运算符
  1141. return resp, errors.New("未聚合出数据")
  1142. }
  1143. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  1144. //2、histogram 直方图聚合
  1145. stdDeviation := *agg.StdDeviation //标准平方差
  1146. min := *agg.Min //最小值
  1147. max := *agg.Max //最大值
  1148. avg := *agg.Avg //平均数
  1149. totalCount := agg.Count //总数
  1150. discreteCoefficient := stdDeviation / avg //离散系数
  1151. if stdDeviation < avg {
  1152. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  1153. discreteCoefficient = stdDeviation
  1154. }
  1155. newAggs := elastic.NewHistogramAggregation().Field("violate_nums").Interval(discreteCoefficient).ExtendedBounds(min, max)
  1156. searchResult, err := es.EsClient.Search().
  1157. Index(esIndexName).
  1158. TrackTotalHits(true).
  1159. Query(boolQuery). // 设置查询条件
  1160. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  1161. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  1162. Pretty(true). // 返回可读的json格式
  1163. Do(context.Background())
  1164. if err != nil {
  1165. return resp, err
  1166. }
  1167. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  1168. newAgg, found := searchResult.Aggregations.Histogram("result")
  1169. if !found {
  1170. return resp, errors.New("未聚合出数据")
  1171. }
  1172. // 3、组装数据
  1173. var keys []string
  1174. var values []int64
  1175. for _, bucket := range newAgg.Buckets {
  1176. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  1177. bucketValue := bucket.Key
  1178. keys = append(keys, utils.Float64ToStr(bucketValue))
  1179. values = append(values, bucket.DocCount)
  1180. }
  1181. for k, v := range keys {
  1182. value, _ := Operate(values[k], totalCount)
  1183. if k+1 == len(keys) {
  1184. resp = append(resp, map[string]string{
  1185. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  1186. "value": utils.Float64ToStrPrec10(value),
  1187. })
  1188. } else {
  1189. resp = append(resp, map[string]string{
  1190. "key": v + " ~ " + keys[k+1] + " 元",
  1191. "value": utils.Float64ToStrPrec10(value),
  1192. })
  1193. }
  1194. }
  1195. }
  1196. return resp, nil
  1197. }
  1198. // StatisticsUserEggBrowseInterfaceNumsRange 统计用户"浏览界面次数"范围
  1199. func StatisticsUserEggBrowseInterfaceNumsRange(esIndexName string) (resp []map[string]string, err error) {
  1200. resp = []map[string]string{}
  1201. var result *elastic.SearchResult
  1202. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  1203. // 1、 创建 extended_stats 高级统计
  1204. aggs := elastic.NewExtendedStatsAggregation().Field("browse_interface_nums")
  1205. result, err = es.EsClient.Search().
  1206. Index(esIndexName).
  1207. TrackTotalHits(true).
  1208. Query(boolQuery). // 设置查询条件
  1209. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  1210. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  1211. Pretty(true). // 返回可读的json格式
  1212. Do(context.Background())
  1213. if err != nil {
  1214. return resp, err
  1215. }
  1216. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  1217. agg, found := result.Aggregations.ExtendedStats("result")
  1218. if !found {
  1219. // 打印结果,注意:这里使用的是取值运算符
  1220. return resp, errors.New("未聚合出数据")
  1221. }
  1222. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  1223. //2、histogram 直方图聚合
  1224. stdDeviation := *agg.StdDeviation //标准平方差
  1225. min := *agg.Min //最小值
  1226. max := *agg.Max //最大值
  1227. avg := *agg.Avg //平均数
  1228. totalCount := agg.Count //总数
  1229. discreteCoefficient := stdDeviation / avg //离散系数
  1230. if stdDeviation < avg {
  1231. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  1232. discreteCoefficient = stdDeviation
  1233. }
  1234. newAggs := elastic.NewHistogramAggregation().Field("browse_interface_nums").Interval(discreteCoefficient).ExtendedBounds(min, max)
  1235. searchResult, err := es.EsClient.Search().
  1236. Index(esIndexName).
  1237. TrackTotalHits(true).
  1238. Query(boolQuery). // 设置查询条件
  1239. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  1240. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  1241. Pretty(true). // 返回可读的json格式
  1242. Do(context.Background())
  1243. if err != nil {
  1244. return resp, err
  1245. }
  1246. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  1247. newAgg, found := searchResult.Aggregations.Histogram("result")
  1248. if !found {
  1249. return resp, errors.New("未聚合出数据")
  1250. }
  1251. // 3、组装数据
  1252. var keys []string
  1253. var values []int64
  1254. for _, bucket := range newAgg.Buckets {
  1255. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  1256. bucketValue := bucket.Key
  1257. keys = append(keys, utils.Float64ToStr(bucketValue))
  1258. values = append(values, bucket.DocCount)
  1259. }
  1260. for k, v := range keys {
  1261. value, _ := Operate(values[k], totalCount)
  1262. if k+1 == len(keys) {
  1263. resp = append(resp, map[string]string{
  1264. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  1265. "value": utils.Float64ToStrPrec10(value),
  1266. })
  1267. } else {
  1268. resp = append(resp, map[string]string{
  1269. "key": v + " ~ " + keys[k+1] + " 元",
  1270. "value": utils.Float64ToStrPrec10(value),
  1271. })
  1272. }
  1273. }
  1274. }
  1275. return resp, nil
  1276. }
  1277. // StatisticsUserEggPersonAddActivityValueRange 统计用户"个人活跃积分增量值"范围
  1278. func StatisticsUserEggPersonAddActivityValueRange(esIndexName string) (resp []map[string]string, err error) {
  1279. resp = []map[string]string{}
  1280. var result *elastic.SearchResult
  1281. boolQuery := elastic.NewBoolQuery().Must() // 创建bool查询
  1282. // 1、 创建 extended_stats 高级统计
  1283. aggs := elastic.NewExtendedStatsAggregation().Field("person_add_activity_value")
  1284. result, err = es.EsClient.Search().
  1285. Index(esIndexName).
  1286. TrackTotalHits(true).
  1287. Query(boolQuery). // 设置查询条件
  1288. Aggregation("result", aggs). // 设置聚合条件,并为聚合条件设置一个名字
  1289. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  1290. Pretty(true). // 返回可读的json格式
  1291. Do(context.Background())
  1292. if err != nil {
  1293. return resp, err
  1294. }
  1295. // 使用Cardinality函数和前面定义的聚合条件名称,查询结果
  1296. agg, found := result.Aggregations.ExtendedStats("result")
  1297. if !found {
  1298. // 打印结果,注意:这里使用的是取值运算符
  1299. return resp, errors.New("未聚合出数据")
  1300. }
  1301. if agg.StdDeviation != nil && *agg.StdDeviation != 0 {
  1302. //2、histogram 直方图聚合
  1303. stdDeviation := *agg.StdDeviation //标准平方差
  1304. min := *agg.Min //最小值
  1305. max := *agg.Max //最大值
  1306. avg := *agg.Avg //平均数
  1307. totalCount := agg.Count //总数
  1308. discreteCoefficient := stdDeviation / avg //离散系数
  1309. if stdDeviation < avg {
  1310. //TODO::如果标准差相对较小,而平均值较大,离散系数可能会导致较大的间隔,直接使用标准差或其他更合适的值作为间隔。
  1311. discreteCoefficient = stdDeviation
  1312. }
  1313. newAggs := elastic.NewHistogramAggregation().Field("person_add_activity_value").Interval(discreteCoefficient).ExtendedBounds(min, max)
  1314. searchResult, err := es.EsClient.Search().
  1315. Index(esIndexName).
  1316. TrackTotalHits(true).
  1317. Query(boolQuery). // 设置查询条件
  1318. Aggregation("result", newAggs). // 设置聚合条件,并为聚合条件设置一个名字
  1319. Size(0). // 设置分页参数 - 每页大小,设置为0,代表不返回搜索结果,仅返回聚合分析结果
  1320. Pretty(true). // 返回可读的json格式
  1321. Do(context.Background())
  1322. if err != nil {
  1323. return resp, err
  1324. }
  1325. // 使用Terms函数和前面定义的聚合条件名称,查询结果
  1326. newAgg, found := searchResult.Aggregations.Histogram("result")
  1327. if !found {
  1328. return resp, errors.New("未聚合出数据")
  1329. }
  1330. // 3、组装数据
  1331. var keys []string
  1332. var values []int64
  1333. for _, bucket := range newAgg.Buckets {
  1334. // 每一个桶都有一个key值,其实就是分组的值,可以理解为SQL的group by值
  1335. bucketValue := bucket.Key
  1336. keys = append(keys, utils.Float64ToStr(bucketValue))
  1337. values = append(values, bucket.DocCount)
  1338. }
  1339. for k, v := range keys {
  1340. value, _ := Operate(values[k], totalCount)
  1341. if k+1 == len(keys) {
  1342. resp = append(resp, map[string]string{
  1343. "key": v + " ~ " + utils.Float64ToStr(max) + " 元",
  1344. "value": utils.Float64ToStrPrec10(value),
  1345. })
  1346. } else {
  1347. resp = append(resp, map[string]string{
  1348. "key": v + " ~ " + keys[k+1] + " 元",
  1349. "value": utils.Float64ToStrPrec10(value),
  1350. })
  1351. }
  1352. }
  1353. }
  1354. return resp, nil
  1355. }
  1356. func GetYearsAndWeeks() (resp map[string][]string, err error) {
  1357. // 1、 获取到所有的索引
  1358. indices, err := es.GetIndexFromAlias(md.EggEnergyUserEggScoreEsAlias)
  1359. if err != nil {
  1360. return nil, err
  1361. }
  1362. //2、组装数据
  1363. resp = map[string][]string{}
  1364. for _, index := range indices {
  1365. // 分割字符串
  1366. parts := strings.Split(index, "_")
  1367. // 提取 2024 和 46
  1368. resp[parts[len(parts)-1][:4]] = append(resp[parts[len(parts)-1][:4]], parts[len(parts)-1][4:])
  1369. }
  1370. return
  1371. }
  1372. func GetYearsAndWeekStr(esIndexName string) (string, string) {
  1373. parts := strings.Split(esIndexName, "_")
  1374. return parts[len(parts)-1][:4], parts[len(parts)-1][4:]
  1375. }
  1376. // Operate 除运算保留10位小数
  1377. func Operate(num1, num2 int64) (float64, error) {
  1378. value, err := strconv.ParseFloat(fmt.Sprintf("%.10f", float64(num1)/float64(num2)), 64)
  1379. if err != nil {
  1380. return 0, err
  1381. }
  1382. return value, nil
  1383. }