util.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579
  1. package util
  2. import (
  3. "fmt"
  4. "math"
  5. "math/rand"
  6. "regexp"
  7. "time"
  8. "gitee.com/xuyiping_admin/pkg/xerr"
  9. )
  10. const (
  11. LayoutTime = "2006-01-02 15:04:05"
  12. Layout = "2006-01-02"
  13. LayoutMonth = "2006-01"
  14. LetterBytes = "abcdefghijkmnpqrstuvwxyzABCDEFGHIJKLMNPQRSTUVWXYZ23456789"
  15. LetterIdxBits = 6 // 6 bits to represent a letter index
  16. LetterIdxMask = 1<<LetterIdxBits - 1 // All 1-bits, as many as letterIdxBits
  17. LetterIdxMax = 63 / LetterIdxBits // # of letter indices fitting in 63 bits
  18. NumberVBytes = "0123456789"
  19. )
  20. var (
  21. FrameIdMap = map[int32]int32{
  22. 1: 1, 2: 2, 3: 3, 4: 4, 5: 5, 6: 6, 8: 8, // 00-02
  23. 11: 11, 12: 12, 13: 13, 14: 14, 15: 15, 16: 16, 18: 18, // 02-04
  24. 21: 21, 22: 22, 23: 23, 24: 24, 25: 25, 26: 26, 28: 28, // 04-06
  25. 31: 31, 32: 32, 33: 33, 34: 34, 35: 35, 36: 36, 38: 38, // 06-08
  26. 41: 41, 42: 42, 43: 43, 44: 44, 45: 45, 46: 46, 48: 48, // 08-10
  27. 51: 51, 52: 52, 53: 53, 54: 54, 55: 55, 56: 56, 58: 58, // 10-12
  28. 61: 61, 62: 62, 63: 63, 64: 64, 65: 65, 66: 66, 68: 68, // 12-14
  29. 71: 71, 72: 72, 73: 73, 74: 74, 75: 75, 76: 76, 78: 78, // 14-16
  30. 81: 81, 82: 82, 83: 83, 84: 84, 85: 85, 86: 86, 88: 88, // 16-18
  31. 91: 91, 92: 92, 93: 93, 94: 94, 95: 95, 96: 96, 98: 98, // 18-20
  32. 101: 101, 102: 102, 103: 103, 104: 104, 105: 105, 106: 106, 108: 108, // 20-22
  33. 111: 111, 112: 112, 113: 113, 114: 114, 115: 115, 116: 116, 118: 118, // 22-24
  34. }
  35. SpecialHours = map[int]int{8: 2, 18: 4, 28: 6, 38: 8, 48: 10, 58: 12, 68: 14, 78: 16, 88: 18, 98: 20, 108: 22, 118: 0}
  36. FrameIdMapReverse = map[int32]int32{
  37. 0: 8,
  38. 1: 8,
  39. 2: 18,
  40. 3: 18,
  41. 4: 28,
  42. 5: 28,
  43. 6: 38,
  44. 7: 38,
  45. 8: 48,
  46. 9: 48,
  47. 10: 58,
  48. 11: 58,
  49. 12: 68,
  50. 13: 68,
  51. 14: 78,
  52. 15: 78,
  53. 16: 88,
  54. 17: 88,
  55. 18: 98,
  56. 19: 98,
  57. 20: 108,
  58. 21: 108,
  59. 22: 118, // 04-06
  60. 23: 118,
  61. }
  62. ExpectedFrameIDs = []int32{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11}
  63. FrameIdSlice = []int32{
  64. 1, 2, 3, 4, 5, 6, 8,
  65. 11, 12, 13, 14, 15, 16, 18,
  66. 21, 22, 23, 24, 25, 26, 28,
  67. 31, 32, 33, 34, 35, 36, 38,
  68. 41, 42, 43, 44, 45, 46, 48,
  69. 51, 52, 53, 54, 55, 56, 58,
  70. 61, 62, 63, 64, 65, 66, 68,
  71. 71, 72, 73, 74, 75, 76, 78,
  72. 81, 82, 83, 84, 85, 86, 88,
  73. 91, 92, 93, 94, 95, 96, 98,
  74. 101, 102, 103, 104, 105, 106, 108,
  75. 111, 112, 113, 114, 115, 116, 118,
  76. }
  77. )
  78. // GenerateRandomNumberString 生成指定长度的数字串
  79. func GenerateRandomNumberString(n int) string {
  80. result := make([]byte, n)
  81. // A rand.Int63() generates 63 random bits, enough for letterIdxMax characters!
  82. rand.Seed(time.Now().Local().UnixNano())
  83. for i, cache, remain := n-1, rand.Int63(), LetterIdxMax; i >= 0; {
  84. if remain == 0 {
  85. cache, remain = rand.Int63(), LetterIdxMax
  86. }
  87. if idx := int(cache & LetterIdxMask); idx < len(LetterBytes) {
  88. result[i] = LetterBytes[idx]
  89. i--
  90. }
  91. cache >>= LetterIdxBits
  92. remain--
  93. }
  94. return string(result)
  95. }
  96. func GenerateRandomNumber(n int) string {
  97. result := make([]byte, n)
  98. // A rand.Int63() generates 63 random bits, enough for letterIdxMax characters!
  99. rand.Seed(time.Now().Local().UnixNano())
  100. for i, cache, remain := n-1, rand.Int63(), LetterIdxMax; i >= 0; {
  101. if remain == 0 {
  102. cache, remain = rand.Int63(), LetterIdxMax
  103. }
  104. if idx := int(cache & LetterIdxMask); idx < len(NumberVBytes) {
  105. result[i] = NumberVBytes[idx]
  106. i--
  107. }
  108. cache >>= LetterIdxBits
  109. remain--
  110. }
  111. return string(result)
  112. }
  113. // TimeParseLocalUnix 获取当天零点的时间戳
  114. // eg 2023-02-22 => 1676995200
  115. func TimeParseLocalUnix(DayTime string) int64 {
  116. value := DayTime
  117. if len(DayTime) <= 11 {
  118. value = fmt.Sprintf("%s 00:00:00", DayTime)
  119. }
  120. theTime, _ := TimeParseLocal(LayoutTime, value)
  121. return theTime.Unix()
  122. }
  123. // TimeParseLocalEndUnix 获取当天24点的时间戳
  124. // eg 2023-02-22 => 1676995200
  125. func TimeParseLocalEndUnix(DayTime string) int64 {
  126. value := DayTime
  127. if len(DayTime) <= 11 {
  128. value = fmt.Sprintf("%s 23:59:59", DayTime)
  129. }
  130. theTime, _ := TimeParseLocal(LayoutTime, value)
  131. return theTime.Unix()
  132. }
  133. // ConvertParseLocalUnix 字符串转换当天时间戳
  134. // eg 15:04:05 => 1676998245
  135. func ConvertParseLocalUnix(timeParse string) (int64, error) {
  136. value := fmt.Sprintf("%s %s", time.Now().Local().Format(Layout), timeParse)
  137. theTime, err := TimeParseLocal(LayoutTime, value)
  138. if err != nil {
  139. return 0, err
  140. }
  141. return theTime.Unix(), nil
  142. }
  143. // DateTimeParseLocalUnix
  144. // eg 2025-10-13 15:04:05 => 1676998245
  145. func DateTimeParseLocalUnix(DayTime string) int64 {
  146. theTime, _ := TimeParseLocal(LayoutTime, DayTime)
  147. return theTime.Unix()
  148. }
  149. // GetMonthRemainDay 获取当前月还剩几天
  150. func GetMonthRemainDay() int {
  151. now := time.Now().Local()
  152. lastDayOfMonth := time.Date(now.Year(), now.Month()+1, 0, 23, 59, 59, 999999999, now.Location())
  153. return int(lastDayOfMonth.Sub(now).Hours()/24) + 1
  154. }
  155. // Ceil 向上取整函数
  156. func Ceil(x float64) float64 {
  157. // 使用 math.Floor 计算小于或等于 x 的最大整数
  158. // 然后检查 x 是否为整数,如果不是,则结果加 1
  159. // 注意:math.Floor 返回的是 float64 类型,所以我们需要进行比较
  160. // 来确定是否需要加 1
  161. intPart := math.Floor(x)
  162. if x-intPart > 0 {
  163. return intPart + 1
  164. }
  165. return intPart
  166. }
  167. // GetLastDayOfMonth
  168. // 接受一个字符串形式的月份(如 "2024-12"),
  169. // 并返回该月份的最后一天的日期(2024-12-31)
  170. func GetLastDayOfMonth(month string) (string, error) {
  171. t, err := TimeParseLocal(LayoutMonth, month)
  172. if err != nil {
  173. return "", err // 如果解析失败,返回错误
  174. }
  175. // 获取下个月的第一天
  176. nextMonth := t.AddDate(0, 1, 0)
  177. // 返回上个月的最后一天
  178. lastDay := nextMonth.AddDate(0, 0, -1)
  179. return lastDay.Format(Layout), nil
  180. }
  181. // GetMonthsInRange
  182. // 接受两个字符串形式的月份(如 "2024-01" 到 "2024-12"),
  183. // 并返回一个包含这两个月份之间所有月份的字符串切片。
  184. func GetMonthsInRange(startMonth, endMonth string) ([]string, error) {
  185. // 解析起始月份
  186. startTime, err := TimeParseLocal(LayoutMonth, startMonth)
  187. if err != nil {
  188. return nil, err
  189. }
  190. // 解析结束月份
  191. endTime, err := TimeParseLocal(LayoutMonth, endMonth)
  192. if err != nil {
  193. return nil, err
  194. }
  195. // 初始化结果切片
  196. var months []string
  197. // 循环添加每个月份直到达到或超过结束月份
  198. for curr := startTime; curr.Before(endTime) || curr.Equal(endTime); curr = curr.AddDate(0, 1, 0) {
  199. months = append(months, curr.Format(LayoutMonth))
  200. }
  201. return months, nil
  202. }
  203. // RoundToTwoDecimals 四舍五入并保留两位小数
  204. func RoundToTwoDecimals(num float64) float64 {
  205. // 使用 math.Round 函数进行四舍五入
  206. // 先乘以 100,然后四舍五入,最后除以 100
  207. return math.Round(num*100) / 100
  208. }
  209. // Get21DayPeriods 获取范围时间内21天周期的切片
  210. // 从结束时间开始往前推算,超过开始时间,继续向前推算,直至凑整21天
  211. func Get21DayPeriods(startDay, endDay string) ([][]string, error) {
  212. startDate, err := TimeParseLocal(Layout, startDay)
  213. if err != nil {
  214. return nil, err
  215. }
  216. endDate, err := TimeParseLocal(Layout, endDay)
  217. if err != nil {
  218. return nil, err
  219. }
  220. if startDate.After(endDate) {
  221. return nil, xerr.Custom("start date is after end date")
  222. }
  223. var periods [][]string
  224. for date := endDate; date.After(startDate); {
  225. periodEnd := date.AddDate(0, 0, -20)
  226. periods = append(periods, []string{periodEnd.Format(Layout), date.Format(Layout)})
  227. date = periodEnd.AddDate(0, 0, -1)
  228. }
  229. reverseRows(periods)
  230. return periods, nil
  231. }
  232. func reverseRows(matrix [][]string) {
  233. // 获取矩阵的行数
  234. rows := len(matrix)
  235. if rows == 0 {
  236. return // 如果矩阵为空,则直接返回
  237. }
  238. // 初始化两个指针,一个指向开始,一个指向末尾
  239. for i, j := 0, rows-1; i < j; i, j = i+1, j-1 {
  240. // 交换当前行和对应行
  241. matrix[i], matrix[j] = matrix[j], matrix[i]
  242. }
  243. }
  244. // GetRangeDayMiddleDay 获取指定日期范围中间的某一天
  245. func GetRangeDayMiddleDay(dateRange []string, middleDay int32) (string, error) {
  246. if len(dateRange) < 2 {
  247. return "", xerr.Custom("date range is not enough")
  248. }
  249. if middleDay < 0 {
  250. return "", xerr.Custom("middle day is not enough")
  251. }
  252. startDate, _ := TimeParseLocal(Layout, dateRange[0])
  253. return startDate.AddDate(0, 0, int(middleDay)-1).Format(Layout), nil
  254. }
  255. // GetRangeDayByDays 获取指定范围日期内按照指定天数来切割
  256. // (2024-10-01 ~ 2024-10-31,5)=> [[2024-10-01,2024-10-05], [2024-10-06,2024-10-10], [2024-10-11,2024-10-15], [2024-10-16,2024-10-20], [2024-10-21,2024-10-25], [2024-10-26,2024-10-30],[2024-10-31,2024-10-31]]
  257. func GetRangeDayByDays(startDay, endDay string, days int32) ([][]string, error) {
  258. var res [][]string
  259. if days <= 0 {
  260. return res, nil
  261. }
  262. startDate, err := TimeParseLocal(Layout, startDay)
  263. if err != nil {
  264. return nil, err
  265. }
  266. endDate, err := TimeParseLocal(Layout, endDay)
  267. if err != nil {
  268. return nil, err
  269. }
  270. if startDate.After(endDate) {
  271. return nil, xerr.Custom("start date is after end date")
  272. }
  273. if startDate == endDate {
  274. return [][]string{{startDay, endDay}}, nil
  275. }
  276. for date := startDate; date.Before(endDate) || date.Equal(endDate); date = date.AddDate(0, 0, int(days)) {
  277. if date.AddDate(0, 0, int(days)-1).After(endDate) {
  278. res = append(res, []string{date.Format(Layout), endDate.Format(Layout)})
  279. break
  280. }
  281. res = append(res, []string{date.Format(Layout), date.AddDate(0, 0, int(days)-1).Format(Layout)})
  282. }
  283. return res, nil
  284. }
  285. // 计算样本均值
  286. func mean(data []float64) float64 {
  287. sum := 0.0
  288. for _, v := range data {
  289. sum += v
  290. }
  291. return sum / float64(len(data))
  292. }
  293. // 计算样本标准差
  294. func stddev(data []float64, mean float64) float64 {
  295. sum := 0.0
  296. for _, v := range data {
  297. sum += (v - mean) * (v - mean)
  298. }
  299. variance := sum / float64(len(data)-1)
  300. return math.Sqrt(variance)
  301. }
  302. // ConfidenceInterval 计算95%置信区间
  303. func ConfidenceInterval(data []float64) (float64, float64) {
  304. n := float64(len(data))
  305. meanVal := mean(data)
  306. stdDev := stddev(data, meanVal)
  307. z := 1.96 // 95%置信水平对应的Z值
  308. marginOfError := z * (stdDev / math.Sqrt(n))
  309. lowerBound := meanVal - marginOfError
  310. upperBound := meanVal + marginOfError
  311. return lowerBound, upperBound
  312. }
  313. // ConfidenceInterval2 计算95%置信区间
  314. func ConfidenceInterval2(p float64, total float64) (min, max float64) {
  315. if p <= 0 || total <= 0 {
  316. return 0, 0
  317. }
  318. z := 1.96 // 95%置信水平对应的Z值
  319. marginOf := z * math.Sqrt(p*(1-p)/total) * 100
  320. return math.Max(0, p*100-math.Ceil(marginOf)), math.Max(1, p*100+math.Ceil(marginOf))
  321. }
  322. // RemoveDuplicates 去除切片中的重复元素
  323. func RemoveDuplicates(slice []string) []string {
  324. if len(slice) <= 0 {
  325. return slice
  326. }
  327. // 创建一个map来跟踪已经遇到的元素
  328. seen := make(map[string]struct{})
  329. // 创建一个新的切片来存储不重复的元素
  330. var result []string
  331. // 遍历原始切片
  332. for _, v := range slice {
  333. // 如果元素尚未在map中,则将其添加到结果切片和map中
  334. if _, exists := seen[v]; !exists {
  335. seen[v] = struct{}{}
  336. result = append(result, v)
  337. }
  338. }
  339. // 返回不重复的切片
  340. return result
  341. }
  342. // DaysBetween 计算两个日期(时间戳)之间的天数差
  343. func DaysBetween(startDayUnix int64, endDayUnix int64) int64 {
  344. time1 := time.Unix(startDayUnix, 0).Local()
  345. time2 := time.Unix(endDayUnix, 0).Local()
  346. // Truncate to the start of the day (00:00:00)
  347. startOfDay1 := time.Date(time1.Year(), time1.Month(), time1.Day(), 0, 0, 0, 0, time1.Location())
  348. startOfDay2 := time.Date(time2.Year(), time2.Month(), time2.Day(), 0, 0, 0, 0, time2.Location())
  349. // Calculate the difference in days
  350. daysDiff := int64(startOfDay2.Sub(startOfDay1).Hours() / 24)
  351. return daysDiff
  352. }
  353. // GetMonths 计算两个日期之间的月龄(近似值)
  354. func GetMonths(birthDate, now time.Time) int {
  355. years := now.Year() - birthDate.Year()
  356. months := int(now.Month()) - int(birthDate.Month())
  357. // 如果当前月份小于出生月份,年份减1,月份补12
  358. if months < 0 {
  359. years--
  360. months += 12
  361. }
  362. // 考虑天数差异(可选:如果当前日期小于出生日期,减少1个月)
  363. if now.Day() < birthDate.Day() {
  364. months--
  365. }
  366. totalMonths := years*12 + months
  367. if totalMonths < 0 {
  368. return 0
  369. }
  370. return totalMonths
  371. }
  372. // GetDaysBetween 获取两个日期之间的所有天数
  373. // 2024-10-01 ~ 2024-10-07 => [2024-10-01,2024-10-02,2024-10-03,2024-10-04,2024-10-05,2024-10-06,2024-10-07]
  374. func GetDaysBetween(startDate, endDate string) ([]string, error) {
  375. // 解析日期
  376. start, err := TimeParseLocal(Layout, startDate)
  377. if err != nil {
  378. return nil, fmt.Errorf("解析开始日期失败: %v", err)
  379. }
  380. end, err := TimeParseLocal(Layout, endDate)
  381. if err != nil {
  382. return nil, fmt.Errorf("解析结束日期失败: %v", err)
  383. }
  384. // 确保开始日期早于或等于结束日期
  385. if start.After(end) {
  386. return nil, fmt.Errorf("开始日期不能晚于结束日期")
  387. }
  388. // 存储日期的数组
  389. var days []string
  390. // 遍历日期
  391. for current := start; !current.After(end); current = current.AddDate(0, 0, 1) {
  392. days = append(days, current.Format(Layout))
  393. }
  394. return days, nil
  395. }
  396. // GetMonthsBetween 返回两个日期之间的所有月份
  397. // 参数格式为 "YYYY-MM",例如 "2025-01"
  398. func GetMonthsBetween(start, end string) ([]string, error) {
  399. // 检查参数合法性
  400. if err := validateDate(start); err != nil {
  401. return nil, fmt.Errorf("invalid start date: %v", err)
  402. }
  403. if err := validateDate(end); err != nil {
  404. return nil, fmt.Errorf("invalid end date: %v", err)
  405. }
  406. // 解析起始日期和结束日期
  407. startTime, err := TimeParseLocal(LayoutMonth, start)
  408. if err != nil {
  409. return nil, fmt.Errorf("failed to parse start date: %v", err)
  410. }
  411. endTime, err := TimeParseLocal(LayoutMonth, end)
  412. if err != nil {
  413. return nil, fmt.Errorf("failed to parse end date: %v", err)
  414. }
  415. // 检查起始日期是否早于结束日期
  416. if startTime.After(endTime) {
  417. return nil, xerr.Custom("start date must be before or equal to end date")
  418. }
  419. // 生成月份列表
  420. var months []string
  421. for current := startTime; !current.After(endTime); current = current.AddDate(0, 1, 0) {
  422. months = append(months, current.Format(LayoutMonth))
  423. }
  424. return months, nil
  425. }
  426. // validateDate 检查日期字符串的合法性
  427. func validateDate(date string) error {
  428. if len(date) != 7 || date[4] != '-' {
  429. return xerr.Custom("date format must be YYYY-MM")
  430. }
  431. _, err := TimeParseLocal(LayoutMonth, date)
  432. if err != nil {
  433. return xerr.Customf("invalid date: %v", err)
  434. }
  435. return nil
  436. }
  437. // MsgFormat 格式化消息字符串 字符串里面有多个冒号,仅删除冒号前后的空格(如果存在)
  438. func MsgFormat(input string) string {
  439. // 定义正则表达式,用于匹配冒号两边的空格
  440. re := regexp.MustCompile(`\s*:\s*`)
  441. // 使用正则表达式替换所有匹配的部分
  442. return re.ReplaceAllString(input, ":")
  443. }
  444. // IsValidFrameId 检查 FrameId 是否有效
  445. func IsValidFrameId(frameId int32) bool {
  446. _, ok := FrameIdMap[frameId]
  447. return ok
  448. }
  449. /*
  450. GetNeckRingActiveTimer
  451. 1. frameId值如果是:1到6代表每天的0点到2点,11-16 代表每天的2点到4点, 21-26 代表每天的4点到6点,31-36代表每天的6点到8点,
  452. 41-46 代表每天的8点到10点,51-56代表每天的10点到12点,61-66代表每天的12-14点,71-76代表每天的14-16点,81-86代表每天的16-18点,
  453. 91-96代表每天的18-20点,101-106代表每天的20-22点,111-116代表每天的22-24点。其中每个数字代表20分钟。
  454. 2. 如果frameId大于当前时间点,并往前推20个小时,如果在这个20小时范围内, 就代表frameId是昨天的。
  455. 3. 如果farmId的值出现8,18,28,3848,58,68,78,88,98,108,118数字分别代表2个小时,从0-2点开始,以此类推。
  456. 帮我根据frameId值,获取对应的时间(YYYY-MM-DD)和小时
  457. */
  458. func GetNeckRingActiveTimer(frameId int32) (dateTime string, hours int) {
  459. if frameId < 0 || frameId > 118 {
  460. return "", 0
  461. }
  462. if _, ok := FrameIdMap[frameId]; !ok {
  463. return "", 0
  464. }
  465. nowTime := time.Now().Local()
  466. currHour := nowTime.Hour()
  467. // 处理2小时的特殊 farmId
  468. hours, ok := SpecialHours[int(frameId)]
  469. day := 0
  470. if ok {
  471. if hours == 0 {
  472. hours = 24
  473. }
  474. if hours > currHour {
  475. day = -1
  476. }
  477. hours = 0
  478. dateTime = nowTime.AddDate(0, 0, day).Format(Layout)
  479. return
  480. }
  481. hours = int(math.Floor(float64(frameId)/10) * 2)
  482. units := int(frameId % 10)
  483. hours += units / 3
  484. if hours > currHour {
  485. for i := 0; i <= 20; i++ {
  486. twentyHoursAgo := nowTime.Add(-time.Duration(i) * time.Hour).Hour()
  487. if twentyHoursAgo == 0 {
  488. twentyHoursAgo = 24
  489. }
  490. if hours == twentyHoursAgo {
  491. day = -1
  492. break
  493. }
  494. }
  495. }
  496. dateTime = nowTime.AddDate(0, 0, day).Format(Layout)
  497. return
  498. }
  499. // XFrameId 获取XFrameId
  500. func XFrameId(frameid int32) int32 {
  501. return int32(math.Floor(float64(frameid / 10)))
  502. }
  503. // FrameIds 获取FrameIds
  504. func FrameIds(xFrameId int32) []int32 {
  505. frameIds := make([]int32, 0)
  506. for i := 1; i <= 8; i++ {
  507. if i == 7 {
  508. continue
  509. }
  510. frameIds = append(frameIds, xFrameId*10+int32(i))
  511. }
  512. return frameIds
  513. }