common.go 33 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313
  1. package utils
  2. import (
  3. "bufio"
  4. "crypto/hmac"
  5. "crypto/md5"
  6. cryRand "crypto/rand"
  7. "crypto/sha1"
  8. "crypto/sha256"
  9. "encoding/base64"
  10. "encoding/hex"
  11. "encoding/json"
  12. "errors"
  13. "fmt"
  14. "gorm.io/gorm"
  15. "image"
  16. "image/png"
  17. "io"
  18. "math"
  19. "math/big"
  20. "math/rand"
  21. "net"
  22. "net/http"
  23. "os"
  24. "os/exec"
  25. "path"
  26. "regexp"
  27. "sort"
  28. "strconv"
  29. "strings"
  30. "time"
  31. )
  32. // 随机数种子
  33. var rnd = rand.New(rand.NewSource(time.Now().UnixNano()))
  34. func GetRandString(size int) string {
  35. allLetterDigit := []string{"0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z", "a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z", "!", "@", "#", "$", "%", "^", "&", "*"}
  36. randomSb := ""
  37. digitSize := len(allLetterDigit)
  38. for i := 0; i < size; i++ {
  39. randomSb += allLetterDigit[rnd.Intn(digitSize)]
  40. }
  41. return randomSb
  42. }
  43. func GetRandStringNoSpecialChar(size int) string {
  44. allLetterDigit := []string{"0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z", "a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l", "m", "n", "o", "p", "q", "r", "s", "t", "u", "v", "w", "x", "y", "z"}
  45. randomSb := ""
  46. digitSize := len(allLetterDigit)
  47. for i := 0; i < size; i++ {
  48. randomSb += allLetterDigit[rnd.Intn(digitSize)]
  49. }
  50. return randomSb
  51. }
  52. func StringsToJSON(str string) string {
  53. rs := []rune(str)
  54. jsons := ""
  55. for _, r := range rs {
  56. rint := int(r)
  57. if rint < 128 {
  58. jsons += string(r)
  59. } else {
  60. jsons += "\\u" + strconv.FormatInt(int64(rint), 16) // json
  61. }
  62. }
  63. return jsons
  64. }
  65. // 序列化
  66. func ToString(v interface{}) string {
  67. data, _ := json.Marshal(v)
  68. return string(data)
  69. }
  70. // md5加密
  71. func MD5(data string) string {
  72. m := md5.Sum([]byte(data))
  73. return hex.EncodeToString(m[:])
  74. }
  75. // 获取数字随机字符
  76. func GetRandDigit(n int) string {
  77. return fmt.Sprintf("%0"+strconv.Itoa(n)+"d", rnd.Intn(int(math.Pow10(n))))
  78. }
  79. // 获取随机数
  80. func GetRandNumber(n int) int {
  81. return rnd.Intn(n)
  82. }
  83. func GetRandInt(min, max int) int {
  84. if min >= max || min == 0 || max == 0 {
  85. return max
  86. }
  87. return rand.Intn(max-min) + min
  88. }
  89. func GetToday(format string) string {
  90. today := time.Now().Format(format)
  91. return today
  92. }
  93. // 获取今天剩余秒数
  94. func GetTodayLastSecond() time.Duration {
  95. today := GetToday(FormatDate) + " 23:59:59"
  96. end, _ := time.ParseInLocation(FormatDateTime, today, time.Local)
  97. return time.Duration(end.Unix()-time.Now().Local().Unix()) * time.Second
  98. }
  99. // 处理出生日期函数
  100. func GetBrithDate(idcard string) string {
  101. l := len(idcard)
  102. var s string
  103. if l == 15 {
  104. s = "19" + idcard[6:8] + "-" + idcard[8:10] + "-" + idcard[10:12]
  105. return s
  106. }
  107. if l == 18 {
  108. s = idcard[6:10] + "-" + idcard[10:12] + "-" + idcard[12:14]
  109. return s
  110. }
  111. return GetToday(FormatDate)
  112. }
  113. // 处理性别
  114. func WhichSexByIdcard(idcard string) string {
  115. var sexs = [2]string{"女", "男"}
  116. length := len(idcard)
  117. if length == 18 {
  118. sex, _ := strconv.Atoi(string(idcard[16]))
  119. return sexs[sex%2]
  120. } else if length == 15 {
  121. sex, _ := strconv.Atoi(string(idcard[14]))
  122. return sexs[sex%2]
  123. }
  124. return "男"
  125. }
  126. // 截取小数点后几位
  127. func SubFloatToString(f float64, m int) string {
  128. n := strconv.FormatFloat(f, 'f', -1, 64)
  129. if n == "" {
  130. return ""
  131. }
  132. if m >= len(n) {
  133. return n
  134. }
  135. newn := strings.Split(n, ".")
  136. if m == 0 {
  137. return newn[0]
  138. }
  139. if len(newn) < 2 || m >= len(newn[1]) {
  140. return n
  141. }
  142. return newn[0] + "." + newn[1][:m]
  143. }
  144. // 截取小数点后几位
  145. func SubFloatToFloat(f float64, m int) float64 {
  146. newn := SubFloatToString(f, m)
  147. newf, _ := strconv.ParseFloat(newn, 64)
  148. return newf
  149. }
  150. // 截取小数点后几位
  151. func SubFloatToFloatStr(f float64, m int) string {
  152. newn := SubFloatToString(f, m)
  153. return newn
  154. }
  155. // 获取相差时间-年
  156. func GetYearDiffer(start_time, end_time string) int {
  157. t1, _ := time.ParseInLocation("2006-01-02", start_time, time.Local)
  158. t2, _ := time.ParseInLocation("2006-01-02", end_time, time.Local)
  159. age := t2.Year() - t1.Year()
  160. if t2.Month() < t1.Month() || (t2.Month() == t1.Month() && t2.Day() < t1.Day()) {
  161. age--
  162. }
  163. return age
  164. }
  165. // 获取相差时间-秒
  166. func GetSecondDifferByTime(start_time, end_time time.Time) int64 {
  167. diff := end_time.Unix() - start_time.Unix()
  168. return diff
  169. }
  170. func FixFloat(f float64, m int) float64 {
  171. newn := SubFloatToString(f+0.00000001, m)
  172. newf, _ := strconv.ParseFloat(newn, 64)
  173. return newf
  174. }
  175. // 将字符串数组转化为逗号分割的字符串形式 ["str1","str2","str3"] >>> "str1,str2,str3"
  176. func StrListToString(strList []string) (str string) {
  177. if len(strList) > 0 {
  178. for k, v := range strList {
  179. if k == 0 {
  180. str = v
  181. } else {
  182. str = str + "," + v
  183. }
  184. }
  185. return
  186. }
  187. return ""
  188. }
  189. // Token
  190. func GetToken() string {
  191. randStr := GetRandString(64)
  192. token := MD5(randStr + Md5Key)
  193. tokenLen := 64 - len(token)
  194. return strings.ToUpper(token + GetRandString(tokenLen))
  195. }
  196. // 数据没有记录
  197. func ErrNoRow() string {
  198. return "<QuerySeter> no row found"
  199. }
  200. // 判断文件是否存在
  201. func FileIsExist(filePath string) bool {
  202. _, err := os.Stat(filePath)
  203. return err == nil || os.IsExist(err)
  204. }
  205. // 获取图片扩展名
  206. func GetImgExt(file string) (ext string, err error) {
  207. var headerByte []byte
  208. headerByte = make([]byte, 8)
  209. fd, err := os.Open(file)
  210. if err != nil {
  211. return "", err
  212. }
  213. defer fd.Close()
  214. _, err = fd.Read(headerByte)
  215. if err != nil {
  216. return "", err
  217. }
  218. xStr := fmt.Sprintf("%x", headerByte)
  219. switch {
  220. case xStr == "89504e470d0a1a0a":
  221. ext = ".png"
  222. case xStr == "0000010001002020":
  223. ext = ".ico"
  224. case xStr == "0000020001002020":
  225. ext = ".cur"
  226. case xStr[:12] == "474946383961" || xStr[:12] == "474946383761":
  227. ext = ".gif"
  228. case xStr[:10] == "0000020000" || xStr[:10] == "0000100000":
  229. ext = ".tga"
  230. case xStr[:8] == "464f524d":
  231. ext = ".iff"
  232. case xStr[:8] == "52494646":
  233. ext = ".ani"
  234. case xStr[:4] == "4d4d" || xStr[:4] == "4949":
  235. ext = ".tiff"
  236. case xStr[:4] == "424d":
  237. ext = ".bmp"
  238. case xStr[:4] == "ffd8":
  239. ext = ".jpg"
  240. case xStr[:2] == "0a":
  241. ext = ".pcx"
  242. default:
  243. ext = ""
  244. }
  245. return ext, nil
  246. }
  247. // 保存图片
  248. func SaveImage(path string, img image.Image) (err error) {
  249. //需要保持的文件
  250. imgfile, err := os.Create(path)
  251. defer imgfile.Close()
  252. // 以PNG格式保存文件
  253. err = png.Encode(imgfile, img)
  254. return err
  255. }
  256. // 下载图片
  257. func DownloadImage(imgUrl string) (filePath string, err error) {
  258. imgPath := "./static/imgs/"
  259. fileName := path.Base(imgUrl)
  260. res, err := http.Get(imgUrl)
  261. if err != nil {
  262. fmt.Println("A error occurred!")
  263. return
  264. }
  265. defer res.Body.Close()
  266. // 获得get请求响应的reader对象
  267. reader := bufio.NewReaderSize(res.Body, 32*1024)
  268. filePath = imgPath + fileName
  269. file, err := os.Create(filePath)
  270. if err != nil {
  271. return
  272. }
  273. // 获得文件的writer对象
  274. writer := bufio.NewWriter(file)
  275. written, _ := io.Copy(writer, reader)
  276. fmt.Printf("Total length: %d \n", written)
  277. return
  278. }
  279. // 保存base64数据为文件
  280. func SaveBase64ToFile(content, path string) error {
  281. data, err := base64.StdEncoding.DecodeString(content)
  282. if err != nil {
  283. return err
  284. }
  285. f, err := os.Create(path)
  286. defer f.Close()
  287. if err != nil {
  288. return err
  289. }
  290. f.Write(data)
  291. return nil
  292. }
  293. func SaveBase64ToFileBySeek(content, path string) (err error) {
  294. data, err := base64.StdEncoding.DecodeString(content)
  295. exist, err := PathExists(path)
  296. if err != nil {
  297. return
  298. }
  299. if !exist {
  300. f, err := os.Create(path)
  301. if err != nil {
  302. return err
  303. }
  304. n, _ := f.Seek(0, 2)
  305. // 从末尾的偏移量开始写入内容
  306. _, err = f.WriteAt([]byte(data), n)
  307. defer f.Close()
  308. } else {
  309. f, err := os.OpenFile(path, os.O_WRONLY, 0644)
  310. if err != nil {
  311. return err
  312. }
  313. n, _ := f.Seek(0, 2)
  314. // 从末尾的偏移量开始写入内容
  315. _, err = f.WriteAt([]byte(data), n)
  316. defer f.Close()
  317. }
  318. return nil
  319. }
  320. func PathExists(path string) (bool, error) {
  321. _, err := os.Stat(path)
  322. if err == nil {
  323. return true, nil
  324. }
  325. if os.IsNotExist(err) {
  326. return false, nil
  327. }
  328. return false, err
  329. }
  330. func StartIndex(page, pagesize int) int {
  331. if page > 1 {
  332. return (page - 1) * pagesize
  333. }
  334. return 0
  335. }
  336. func PageCount(count, pagesize int) int {
  337. if count%pagesize > 0 {
  338. return count/pagesize + 1
  339. } else {
  340. return count / pagesize
  341. }
  342. }
  343. func TrimHtml(src string) string {
  344. //将HTML标签全转换成小写
  345. re, _ := regexp.Compile("\\<[\\S\\s]+?\\>")
  346. src = re.ReplaceAllStringFunc(src, strings.ToLower)
  347. re, _ = regexp.Compile("\\<img[\\S\\s]+?\\>")
  348. src = re.ReplaceAllString(src, "[图片]")
  349. re, _ = regexp.Compile("class[\\S\\s]+?>")
  350. src = re.ReplaceAllString(src, "")
  351. re, _ = regexp.Compile("\\<[\\S\\s]+?\\>")
  352. src = re.ReplaceAllString(src, "")
  353. return strings.TrimSpace(src)
  354. }
  355. //1556164246 -> 2019-04-25 03:50:46 +0000
  356. //timestamp
  357. func TimeToTimestamp() {
  358. fmt.Println(time.Unix(1556164246, 0).Format("2006-01-02 15:04:05"))
  359. }
  360. func ToUnicode(text string) string {
  361. textQuoted := strconv.QuoteToASCII(text)
  362. textUnquoted := textQuoted[1 : len(textQuoted)-1]
  363. return textUnquoted
  364. }
  365. func VersionToInt(version string) int {
  366. version = strings.Replace(version, ".", "", -1)
  367. n, _ := strconv.Atoi(version)
  368. return n
  369. }
  370. func IsCheckInList(list []int, s int) bool {
  371. for _, v := range list {
  372. if v == s {
  373. return true
  374. }
  375. }
  376. return false
  377. }
  378. func round(num float64) int {
  379. return int(num + math.Copysign(0.5, num))
  380. }
  381. func toFixed(num float64, precision int) float64 {
  382. output := math.Pow(10, float64(precision))
  383. return float64(round(num*output)) / output
  384. }
  385. // GetWilsonScore returns Wilson Score
  386. func GetWilsonScore(p, n float64) float64 {
  387. if p == 0 && n == 0 {
  388. return 0
  389. }
  390. return toFixed(((p+1.9208)/(p+n)-1.96*math.Sqrt(p*n/(p+n)+0.9604)/(p+n))/(1+3.8416/(p+n)), 2)
  391. }
  392. // 将中文数字转化成数字,比如 第三百四十五章,返回第345章 不支持一亿及以上
  393. func ChangeWordsToNum(str string) (numStr string) {
  394. words := ([]rune)(str)
  395. num := 0
  396. n := 0
  397. for i := 0; i < len(words); i++ {
  398. word := string(words[i : i+1])
  399. switch word {
  400. case "万":
  401. if n == 0 {
  402. n = 1
  403. }
  404. n = n * 10000
  405. num = num*10000 + n
  406. n = 0
  407. case "千":
  408. if n == 0 {
  409. n = 1
  410. }
  411. n = n * 1000
  412. num += n
  413. n = 0
  414. case "百":
  415. if n == 0 {
  416. n = 1
  417. }
  418. n = n * 100
  419. num += n
  420. n = 0
  421. case "十":
  422. if n == 0 {
  423. n = 1
  424. }
  425. n = n * 10
  426. num += n
  427. n = 0
  428. case "一":
  429. n += 1
  430. case "二":
  431. n += 2
  432. case "三":
  433. n += 3
  434. case "四":
  435. n += 4
  436. case "五":
  437. n += 5
  438. case "六":
  439. n += 6
  440. case "七":
  441. n += 7
  442. case "八":
  443. n += 8
  444. case "九":
  445. n += 9
  446. case "零":
  447. default:
  448. if n > 0 {
  449. num += n
  450. n = 0
  451. }
  452. if num == 0 {
  453. numStr += word
  454. } else {
  455. numStr += strconv.Itoa(num) + word
  456. num = 0
  457. }
  458. }
  459. }
  460. if n > 0 {
  461. num += n
  462. n = 0
  463. }
  464. if num != 0 {
  465. numStr += strconv.Itoa(num)
  466. }
  467. return
  468. }
  469. func Sha1(data string) string {
  470. sha1 := sha1.New()
  471. sha1.Write([]byte(data))
  472. return hex.EncodeToString(sha1.Sum([]byte("")))
  473. }
  474. func GetVideoPlaySeconds(videoPath string) (playSeconds float64, err error) {
  475. cmd := `ffmpeg -i ` + videoPath + ` 2>&1 | grep 'Duration' | cut -d ' ' -f 4 | sed s/,//`
  476. out, err := exec.Command("bash", "-c", cmd).Output()
  477. if err != nil {
  478. return
  479. }
  480. outTimes := string(out)
  481. fmt.Println("outTimes:", outTimes)
  482. if outTimes != "" {
  483. timeArr := strings.Split(outTimes, ":")
  484. h := timeArr[0]
  485. m := timeArr[1]
  486. s := timeArr[2]
  487. hInt, err := strconv.Atoi(h)
  488. if err != nil {
  489. return playSeconds, err
  490. }
  491. mInt, err := strconv.Atoi(m)
  492. if err != nil {
  493. return playSeconds, err
  494. }
  495. s = strings.Trim(s, " ")
  496. s = strings.Trim(s, "\n")
  497. sInt, err := strconv.ParseFloat(s, 64)
  498. if err != nil {
  499. return playSeconds, err
  500. }
  501. playSeconds = float64(hInt)*3600 + float64(mInt)*60 + float64(sInt)
  502. }
  503. return
  504. }
  505. func GetMaxTradeCode(tradeCode string) (maxTradeCode string, err error) {
  506. tradeCode = strings.Replace(tradeCode, "W", "", -1)
  507. tradeCode = strings.Trim(tradeCode, " ")
  508. tradeCodeInt, err := strconv.Atoi(tradeCode)
  509. if err != nil {
  510. return
  511. }
  512. tradeCodeInt = tradeCodeInt + 1
  513. maxTradeCode = fmt.Sprintf("W%06d", tradeCodeInt)
  514. return
  515. }
  516. // excel日期字段格式化 yyyy-mm-dd
  517. func ConvertToFormatDay(excelDaysString string) string {
  518. // 2006-01-02 距离 1900-01-01的天数
  519. baseDiffDay := 38719 //在网上工具计算的天数需要加2天,什么原因没弄清楚
  520. curDiffDay := excelDaysString
  521. b, _ := strconv.Atoi(curDiffDay)
  522. // 获取excel的日期距离2006-01-02的天数
  523. realDiffDay := b - baseDiffDay
  524. //fmt.Println("realDiffDay:",realDiffDay)
  525. // 距离2006-01-02 秒数
  526. realDiffSecond := realDiffDay * 24 * 3600
  527. //fmt.Println("realDiffSecond:",realDiffSecond)
  528. // 2006-01-02 15:04:05距离1970-01-01 08:00:00的秒数 网上工具可查出
  529. baseOriginSecond := 1136185445
  530. resultTime := time.Unix(int64(baseOriginSecond+realDiffSecond), 0).Format("2006-01-02")
  531. return resultTime
  532. }
  533. func CheckPwd(pwd string) bool {
  534. compile := `([0-9a-z]+){6,12}|(a-z0-9]+){6,12}`
  535. reg := regexp.MustCompile(compile)
  536. flag := reg.MatchString(pwd)
  537. return flag
  538. }
  539. func GetMonthStartAndEnd(myYear string, myMonth string) (startDate, endDate string) {
  540. // 数字月份必须前置补零
  541. if len(myMonth) == 1 {
  542. myMonth = "0" + myMonth
  543. }
  544. yInt, _ := strconv.Atoi(myYear)
  545. timeLayout := "2006-01-02 15:04:05"
  546. loc, _ := time.LoadLocation("Local")
  547. theTime, _ := time.ParseInLocation(timeLayout, myYear+"-"+myMonth+"-01 00:00:00", loc)
  548. newMonth := theTime.Month()
  549. t1 := time.Date(yInt, newMonth, 1, 0, 0, 0, 0, time.Local).Format("2006-01-02")
  550. t2 := time.Date(yInt, newMonth+1, 0, 0, 0, 0, 0, time.Local).Format("2006-01-02")
  551. return t1, t2
  552. }
  553. // 移除字符串中的空格
  554. func TrimStr(str string) (str2 string) {
  555. return strings.Replace(str, " ", "", -1)
  556. }
  557. // 字符串转换为time
  558. func StrTimeToTime(strTime string) time.Time {
  559. timeLayout := "2006-01-02 15:04:05" //转化所需模板
  560. loc, _ := time.LoadLocation("Local") //重要:获取时区
  561. resultTime, _ := time.ParseInLocation(timeLayout, strTime, loc)
  562. return resultTime
  563. }
  564. // 字符串类型时间转周几
  565. func StrDateTimeToWeek(strTime string) string {
  566. var WeekDayMap = map[string]string{
  567. "Monday": "周一",
  568. "Tuesday": "周二",
  569. "Wednesday": "周三",
  570. "Thursday": "周四",
  571. "Friday": "周五",
  572. "Saturday": "周六",
  573. "Sunday": "周日",
  574. }
  575. var ctime = StrTimeToTime(strTime).Format("2006-01-02")
  576. startday, _ := time.Parse("2006-01-02", ctime)
  577. staweek_int := startday.Weekday().String()
  578. return WeekDayMap[staweek_int]
  579. }
  580. // 时间格式转年月日字符串
  581. func TimeToStrYmd(time2 time.Time) string {
  582. var Ymd string
  583. year := time2.Year()
  584. month := time2.Format("1")
  585. day1 := time.Now().Day()
  586. Ymd = strconv.Itoa(year) + "年" + month + "月" + strconv.Itoa(day1) + "日"
  587. return Ymd
  588. }
  589. // 时间格式去掉时分秒
  590. func TimeRemoveHms(strTime string) string {
  591. var Ymd string
  592. var resultTime = StrTimeToTime(strTime)
  593. year := resultTime.Year()
  594. month := resultTime.Format("01")
  595. day1 := resultTime.Day()
  596. Ymd = strconv.Itoa(year) + "." + month + "." + strconv.Itoa(day1)
  597. return Ymd
  598. }
  599. // 文章上一次编辑时间
  600. func ArticleLastTime(strTime string) string {
  601. var newTime string
  602. stamp, _ := time.ParseInLocation("2006-01-02 15:04:05", strTime, time.Local)
  603. diffTime := time.Now().Unix() - stamp.Unix()
  604. if diffTime <= 60 {
  605. newTime = "当前"
  606. } else if diffTime < 60*60 {
  607. newTime = strconv.FormatInt(diffTime/60, 10) + "分钟前"
  608. } else if diffTime < 24*60*60 {
  609. newTime = strconv.FormatInt(diffTime/(60*60), 10) + "小时前"
  610. } else if diffTime < 30*24*60*60 {
  611. newTime = strconv.FormatInt(diffTime/(24*60*60), 10) + "天前"
  612. } else if diffTime < 12*30*24*60*60 {
  613. newTime = strconv.FormatInt(diffTime/(30*24*60*60), 10) + "月前"
  614. } else {
  615. newTime = "1年前"
  616. }
  617. return newTime
  618. }
  619. // 人民币小写转大写
  620. func ConvertNumToCny(num float64) (str string, err error) {
  621. strNum := strconv.FormatFloat(num*100, 'f', 0, 64)
  622. sliceUnit := []string{"仟", "佰", "拾", "亿", "仟", "佰", "拾", "万", "仟", "佰", "拾", "元", "角", "分"}
  623. // log.Println(sliceUnit[:len(sliceUnit)-2])
  624. s := sliceUnit[len(sliceUnit)-len(strNum):]
  625. upperDigitUnit := map[string]string{"0": "零", "1": "壹", "2": "贰", "3": "叁", "4": "肆", "5": "伍", "6": "陆", "7": "柒", "8": "捌", "9": "玖"}
  626. for k, v := range strNum[:] {
  627. str = str + upperDigitUnit[string(v)] + s[k]
  628. }
  629. reg, err := regexp.Compile(`零角零分$`)
  630. str = reg.ReplaceAllString(str, "整")
  631. reg, err = regexp.Compile(`零角`)
  632. str = reg.ReplaceAllString(str, "零")
  633. reg, err = regexp.Compile(`零分$`)
  634. str = reg.ReplaceAllString(str, "整")
  635. reg, err = regexp.Compile(`零[仟佰拾]`)
  636. str = reg.ReplaceAllString(str, "零")
  637. reg, err = regexp.Compile(`零{2,}`)
  638. str = reg.ReplaceAllString(str, "零")
  639. reg, err = regexp.Compile(`零亿`)
  640. str = reg.ReplaceAllString(str, "亿")
  641. reg, err = regexp.Compile(`零万`)
  642. str = reg.ReplaceAllString(str, "万")
  643. reg, err = regexp.Compile(`零*元`)
  644. str = reg.ReplaceAllString(str, "元")
  645. reg, err = regexp.Compile(`亿零{0, 3}万`)
  646. str = reg.ReplaceAllString(str, "^元")
  647. reg, err = regexp.Compile(`零元`)
  648. str = reg.ReplaceAllString(str, "零")
  649. return
  650. }
  651. // GetNowWeekMonday 获取本周周一的时间
  652. func GetNowWeekMonday() time.Time {
  653. offset := int(time.Monday - time.Now().Weekday())
  654. mondayTime := time.Now().AddDate(0, 0, offset)
  655. mondayTime = time.Date(mondayTime.Year(), mondayTime.Month(), mondayTime.Day(), 0, 0, 0, 0, mondayTime.Location())
  656. return mondayTime
  657. }
  658. // GetNowWeekLastDay 获取本周最后一天的时间
  659. func GetNowWeekLastDay() time.Time {
  660. offset := int(time.Monday - time.Now().Weekday())
  661. firstDayTime := time.Now().AddDate(0, 0, offset)
  662. firstDayTime = time.Date(firstDayTime.Year(), firstDayTime.Month(), firstDayTime.Day(), 0, 0, 0, 0, firstDayTime.Location()).AddDate(0, 0, 6)
  663. lastDayTime := time.Date(firstDayTime.Year(), firstDayTime.Month(), firstDayTime.Day(), 23, 59, 59, 0, firstDayTime.Location())
  664. return lastDayTime
  665. }
  666. // GetNowMonthFirstDay 获取本月第一天的时间
  667. func GetNowMonthFirstDay() time.Time {
  668. nowMonthFirstDay := time.Date(time.Now().Year(), time.Now().Month(), 1, 0, 0, 0, 0, time.Now().Location())
  669. return nowMonthFirstDay
  670. }
  671. // GetNowMonthLastDay 获取本月最后一天的时间
  672. func GetNowMonthLastDay() time.Time {
  673. nowMonthLastDay := time.Date(time.Now().Year(), time.Now().Month(), 1, 0, 0, 0, 0, time.Now().Location()).AddDate(0, 1, -1)
  674. nowMonthLastDay = time.Date(nowMonthLastDay.Year(), nowMonthLastDay.Month(), nowMonthLastDay.Day(), 23, 59, 59, 0, nowMonthLastDay.Location())
  675. return nowMonthLastDay
  676. }
  677. // GetNowQuarterFirstDay 获取本季度第一天的时间
  678. func GetNowQuarterFirstDay() time.Time {
  679. month := int(time.Now().Month())
  680. var nowQuarterFirstDay time.Time
  681. if month >= 1 && month <= 3 {
  682. //1月1号
  683. nowQuarterFirstDay = time.Date(time.Now().Year(), 1, 1, 0, 0, 0, 0, time.Now().Location())
  684. } else if month >= 4 && month <= 6 {
  685. //4月1号
  686. nowQuarterFirstDay = time.Date(time.Now().Year(), 4, 1, 0, 0, 0, 0, time.Now().Location())
  687. } else if month >= 7 && month <= 9 {
  688. nowQuarterFirstDay = time.Date(time.Now().Year(), 7, 1, 0, 0, 0, 0, time.Now().Location())
  689. } else {
  690. nowQuarterFirstDay = time.Date(time.Now().Year(), 10, 1, 0, 0, 0, 0, time.Now().Location())
  691. }
  692. return nowQuarterFirstDay
  693. }
  694. // GetNowQuarterLastDay 获取本季度最后一天的时间
  695. func GetNowQuarterLastDay() time.Time {
  696. month := int(time.Now().Month())
  697. var nowQuarterLastDay time.Time
  698. if month >= 1 && month <= 3 {
  699. //03-31 23:59:59
  700. nowQuarterLastDay = time.Date(time.Now().Year(), 3, 31, 23, 59, 59, 0, time.Now().Location())
  701. } else if month >= 4 && month <= 6 {
  702. //06-30 23:59:59
  703. nowQuarterLastDay = time.Date(time.Now().Year(), 6, 30, 23, 59, 59, 0, time.Now().Location())
  704. } else if month >= 7 && month <= 9 {
  705. //09-30 23:59:59
  706. nowQuarterLastDay = time.Date(time.Now().Year(), 9, 30, 23, 59, 59, 0, time.Now().Location())
  707. } else {
  708. //12-31 23:59:59
  709. nowQuarterLastDay = time.Date(time.Now().Year(), 12, 31, 23, 59, 59, 0, time.Now().Location())
  710. }
  711. return nowQuarterLastDay
  712. }
  713. // GetNowHalfYearFirstDay 获取当前半年的第一天的时间
  714. func GetNowHalfYearFirstDay() time.Time {
  715. month := int(time.Now().Month())
  716. var nowHalfYearLastDay time.Time
  717. if month >= 1 && month <= 6 {
  718. //03-31 23:59:59
  719. nowHalfYearLastDay = time.Date(time.Now().Year(), 1, 1, 0, 0, 0, 0, time.Now().Location())
  720. } else {
  721. //12-31 23:59:59
  722. nowHalfYearLastDay = time.Date(time.Now().Year(), 7, 1, 0, 0, 0, 0, time.Now().Location())
  723. }
  724. return nowHalfYearLastDay
  725. }
  726. // GetNowHalfYearLastDay 获取当前半年的最后一天的时间
  727. func GetNowHalfYearLastDay() time.Time {
  728. month := int(time.Now().Month())
  729. var nowHalfYearLastDay time.Time
  730. if month >= 1 && month <= 6 {
  731. //03-31 23:59:59
  732. nowHalfYearLastDay = time.Date(time.Now().Year(), 6, 30, 23, 59, 59, 0, time.Now().Location())
  733. } else {
  734. //12-31 23:59:59
  735. nowHalfYearLastDay = time.Date(time.Now().Year(), 12, 31, 23, 59, 59, 0, time.Now().Location())
  736. }
  737. return nowHalfYearLastDay
  738. }
  739. // GetNowYearFirstDay 获取当前年的最后一天的时间
  740. func GetNowYearFirstDay() time.Time {
  741. //12-31 23:59:59
  742. nowYearFirstDay := time.Date(time.Now().Year(), 1, 1, 0, 0, 0, 0, time.Now().Location())
  743. return nowYearFirstDay
  744. }
  745. // GetNowYearLastDay 获取当前年的最后一天的时间
  746. func GetNowYearLastDay() time.Time {
  747. //12-31 23:59:59
  748. nowYearLastDay := time.Date(time.Now().Year(), 12, 31, 23, 59, 59, 0, time.Now().Location())
  749. return nowYearLastDay
  750. }
  751. // CalculationDate 计算两个日期之间相差n年m月y天
  752. func CalculationDate(startDate, endDate time.Time) (beetweenDay string, err error) {
  753. //startDate := time.Date(2021, 3, 28, 0, 0, 0, 0, time.Now().Location())
  754. //endDate := time.Date(2022, 3, 31, 0, 0, 0, 0, time.Now().Location())
  755. numYear := endDate.Year() - startDate.Year()
  756. numMonth := int(endDate.Month()) - int(startDate.Month())
  757. numDay := 0
  758. //获取截止月的总天数
  759. endDateDays := getMonthDay(endDate.Year(), int(endDate.Month()))
  760. //获取截止月的前一个月
  761. endDatePrevMonthDate := endDate.AddDate(0, -1, 0)
  762. //获取截止日期的上一个月的总天数
  763. endDatePrevMonthDays := getMonthDay(endDatePrevMonthDate.Year(), int(endDatePrevMonthDate.Month()))
  764. //获取开始日期的的月份总天数
  765. startDateMonthDays := getMonthDay(startDate.Year(), int(startDate.Month()))
  766. //判断,截止月是否完全被选中,如果相等,那么代表截止月份全部天数被选择
  767. if endDate.Day() == endDateDays {
  768. numDay = startDateMonthDays - startDate.Day() + 1
  769. //如果剩余天数正好与开始日期的天数是一致的,那么月份加1
  770. if numDay == startDateMonthDays {
  771. numMonth++
  772. numDay = 0
  773. //超过月份了,那么年份加1
  774. if numMonth == 12 {
  775. numYear++
  776. numMonth = 0
  777. }
  778. }
  779. } else {
  780. numDay = endDate.Day() - startDate.Day() + 1
  781. }
  782. //天数小于0,那么向月份借一位
  783. if numDay < 0 {
  784. //向上一个月借一个月的天数
  785. numDay += endDatePrevMonthDays
  786. //总月份减去一个月
  787. numMonth = numMonth - 1
  788. }
  789. //月份小于0,那么向年份借一位
  790. if numMonth < 0 {
  791. //向上一个年借12个月
  792. numMonth += 12
  793. //总年份减去一年
  794. numYear = numYear - 1
  795. }
  796. if numYear < 0 {
  797. err = errors.New("日期异常")
  798. return
  799. }
  800. if numYear > 0 {
  801. beetweenDay += fmt.Sprint(numYear, "年")
  802. }
  803. if numMonth > 0 {
  804. beetweenDay += fmt.Sprint(numMonth, "个月")
  805. }
  806. if numDay > 0 {
  807. beetweenDay += fmt.Sprint(numDay, "天")
  808. }
  809. return
  810. }
  811. // getMonthDay 获取某年某月有多少天
  812. func getMonthDay(year, month int) (days int) {
  813. if month != 2 {
  814. if month == 4 || month == 6 || month == 9 || month == 11 {
  815. days = 30
  816. } else {
  817. days = 31
  818. }
  819. } else {
  820. if ((year%4) == 0 && (year%100) != 0) || (year%400) == 0 {
  821. days = 29
  822. } else {
  823. days = 28
  824. }
  825. }
  826. return
  827. }
  828. // GetOrmInReplace 获取orm的in查询替换?的方法
  829. func GetOrmInReplace(num int) string {
  830. template := make([]string, num)
  831. for i := 0; i < num; i++ {
  832. template[i] = "?"
  833. }
  834. return strings.Join(template, ",")
  835. }
  836. // InArrayByInt php中的in_array(判断Int类型的切片中是否存在该int值)
  837. func InArrayByInt(idIntList []int, searchId int) (has bool) {
  838. for _, id := range idIntList {
  839. if id == searchId {
  840. has = true
  841. return
  842. }
  843. }
  844. return
  845. }
  846. // InArrayByStr php中的in_array(判断String类型的切片中是否存在该string值)
  847. func InArrayByStr(idStrList []string, searchId string) (has bool) {
  848. for _, id := range idStrList {
  849. if id == searchId {
  850. has = true
  851. return
  852. }
  853. }
  854. return
  855. }
  856. // RangeRand 取区间随机数
  857. func RangeRand(min, max int64) int64 {
  858. if min > max {
  859. return max
  860. }
  861. if min < 0 {
  862. f64Min := math.Abs(float64(min))
  863. i64Min := int64(f64Min)
  864. result, _ := cryRand.Int(cryRand.Reader, big.NewInt(max+1+i64Min))
  865. return result.Int64() - i64Min
  866. } else {
  867. result, _ := cryRand.Int(cryRand.Reader, big.NewInt(max-min+1))
  868. return min + result.Int64()
  869. }
  870. }
  871. func GetLocalIP() (ip string, err error) {
  872. addrs, err := net.InterfaceAddrs()
  873. if err != nil {
  874. return
  875. }
  876. for _, addr := range addrs {
  877. ipAddr, ok := addr.(*net.IPNet)
  878. if !ok {
  879. continue
  880. }
  881. if ipAddr.IP.IsLoopback() {
  882. continue
  883. }
  884. if !ipAddr.IP.IsGlobalUnicast() {
  885. continue
  886. }
  887. return ipAddr.IP.String(), nil
  888. }
  889. return
  890. }
  891. // HmacSha256 计算HmacSha256
  892. // key 是加密所使用的key
  893. // data 是加密的内容
  894. func HmacSha256(key string, data string) []byte {
  895. mac := hmac.New(sha256.New, []byte(key))
  896. _, _ = mac.Write([]byte(data))
  897. return mac.Sum(nil)
  898. }
  899. // HmacSha256ToBase64 将加密后的二进制转Base64字符串
  900. func HmacSha256ToBase64(key string, data string) string {
  901. return base64.URLEncoding.EncodeToString(HmacSha256(key, data))
  902. }
  903. func GetSign(nonce, timestamp string) (sign string) {
  904. signStrMap := map[string]string{
  905. "nonce": nonce,
  906. "timestamp": timestamp,
  907. "appid": AppId,
  908. }
  909. keys := make([]string, 0, len(signStrMap))
  910. for k := range signStrMap {
  911. keys = append(keys, k)
  912. }
  913. sort.Strings(keys)
  914. var signStr string
  915. for _, k := range keys {
  916. signStr += k + "=" + signStrMap[k] + "&"
  917. }
  918. signStr = strings.Trim(signStr, "&")
  919. fmt.Println("signStr:" + signStr)
  920. sign = HmacSha256ToBase64(Secret, signStr)
  921. fmt.Println("sign:" + sign)
  922. return
  923. }
  924. // GetSignV2
  925. // @Description: 获取签名
  926. // @author: Roc
  927. // @datetime 2025-07-03 16:43:03
  928. // @param nonce string
  929. // @param timestamp string
  930. // @param appid string
  931. // @param secret string
  932. // @return sign string
  933. func GetSignV2(nonce, timestamp, appid, secret string) (sign string) {
  934. signStrMap := map[string]string{
  935. "nonce": nonce,
  936. "timestamp": timestamp,
  937. "appid": appid,
  938. }
  939. keys := make([]string, 0, len(signStrMap))
  940. for k := range signStrMap {
  941. keys = append(keys, k)
  942. }
  943. sort.Strings(keys)
  944. var signStr string
  945. for _, k := range keys {
  946. signStr += k + "=" + signStrMap[k] + "&"
  947. }
  948. signStr = strings.Trim(signStr, "&")
  949. fmt.Println("signStr:" + signStr)
  950. sign = HmacSha256ToBase64(secret, signStr)
  951. fmt.Println("sign:" + sign)
  952. return
  953. }
  954. // GetLikeKeywordPars
  955. //
  956. // @Description: 获取sql查询中的参数切片
  957. // @author: Roc
  958. // @datetime2023-10-23 14:50:18
  959. // @param pars []interface{}
  960. // @param keyword string
  961. // @param num int
  962. // @return newPars []interface{}
  963. func GetLikeKeywordPars(pars []interface{}, keyword string, num int) (newPars []interface{}) {
  964. newPars = pars
  965. if newPars == nil {
  966. newPars = make([]interface{}, 0)
  967. }
  968. for i := 1; i <= num; i++ {
  969. newPars = append(newPars, `%`+keyword+`%`)
  970. }
  971. return
  972. }
  973. func TimeTransferString(format string, t time.Time) string {
  974. str := t.Format(format)
  975. if t.IsZero() {
  976. return ""
  977. }
  978. return str
  979. }
  980. func GetDateByDateTypeV2(dateType int, tmpStartDate, tmpEndDate string, startYear, yearMax int) (startDate, endDate string) {
  981. startDate = tmpStartDate
  982. endDate = tmpEndDate
  983. switch dateType {
  984. case 1:
  985. startDate = "2000-01-01"
  986. endDate = ""
  987. case 2:
  988. startDate = "2010-01-01"
  989. endDate = ""
  990. case 3:
  991. startDate = "2015-01-01"
  992. endDate = ""
  993. case 4:
  994. //startDate = strconv.Itoa(time.Now().Year()) + "-01-01"
  995. startDate = "2021-01-01"
  996. endDate = ""
  997. case 5:
  998. //startDate = startDate + "-01"
  999. //endDate = endDate + "-01"
  1000. case 6:
  1001. //startDate = startDate + "-01"
  1002. endDate = ""
  1003. case 7:
  1004. startDate = "2018-01-01"
  1005. endDate = ""
  1006. case 8:
  1007. startDate = "2019-01-01"
  1008. endDate = ""
  1009. case 9:
  1010. startDate = "2020-01-01"
  1011. endDate = ""
  1012. case 11:
  1013. startDate = "2022-01-01"
  1014. endDate = ""
  1015. case DateTypeNYears:
  1016. if startYear == 0 { //默认取最近5年
  1017. startYear = 5
  1018. }
  1019. if yearMax == 0 {
  1020. return
  1021. }
  1022. startYear = startYear - 1
  1023. baseDate, _ := time.Parse(FormatDate, fmt.Sprintf("%d-01-01", yearMax))
  1024. startDate = baseDate.AddDate(-startYear, 0, 0).Format(FormatDate)
  1025. endDate = ""
  1026. }
  1027. // 兼容日期错误
  1028. {
  1029. if strings.Count(startDate, "-") == 1 {
  1030. startDate = startDate + "-01"
  1031. }
  1032. if strings.Count(endDate, "-") == 1 {
  1033. endDate = endDate + "-01"
  1034. }
  1035. }
  1036. return
  1037. }
  1038. // MapSorter 对于map 排序
  1039. type MapSorter []Item
  1040. type Item struct {
  1041. Key int
  1042. Val float64
  1043. }
  1044. func NewMapSorter(m map[int]float64) MapSorter {
  1045. ms := make(MapSorter, 0, len(m))
  1046. for k, v := range m {
  1047. ms = append(ms, Item{k, v})
  1048. }
  1049. return ms
  1050. }
  1051. func (ms MapSorter) Len() int {
  1052. return len(ms)
  1053. }
  1054. func (ms MapSorter) Less(i, j int) bool {
  1055. return ms[i].Val > ms[j].Val // 按值排序
  1056. //return ms[i].Key < ms[j].Key // 按键排序
  1057. }
  1058. func (ms MapSorter) Swap(i, j int) {
  1059. ms[i], ms[j] = ms[j], ms[i]
  1060. }
  1061. // GetLikeKeyword
  1062. //
  1063. // @Description: 获取sql查询中的like查询字段
  1064. // @author: Roc
  1065. // @datetime2023-10-23 14:46:32
  1066. // @param keyword string
  1067. // @return string
  1068. func GetLikeKeyword(keyword string) string {
  1069. return `%` + keyword + `%`
  1070. }
  1071. // VerifyFrequency
  1072. // @Description: 校验频度是否合规
  1073. // @author: Roc
  1074. // @datetime 2024-04-26 13:30:22
  1075. // @param frequency string 待校验的频度
  1076. // @return bool
  1077. func VerifyFrequency(frequency string) bool {
  1078. return InArrayByStr([]string{"年度", "半年度", "季度", "月度", "旬度", "周度", "日度"}, frequency)
  1079. }
  1080. // DateConvMysqlConvMongo
  1081. // @Description: 将mysql中的日期比较符转换成mongo中的日期比较符
  1082. // @author: Roc
  1083. // @datetime 2024-05-08 11:03:26
  1084. // @param dateCon string
  1085. func DateConvMysqlConvMongo(dateCon string) string {
  1086. cond := ""
  1087. switch dateCon {
  1088. case "=":
  1089. cond = "$eq"
  1090. case "<":
  1091. cond = "$lt"
  1092. case "<=":
  1093. cond = "$lte"
  1094. case ">":
  1095. cond = "$gt"
  1096. case ">=":
  1097. cond = "$gte"
  1098. }
  1099. return cond
  1100. }
  1101. // GetCurrentTime 获取当前时间 格式为 2024-08-07 15:29:58
  1102. func GetCurrentTime() string {
  1103. return time.Now().Format("2006-01-02 15:04:05")
  1104. }
  1105. // GetAssessmentWeekAndFriday 获取给定日期的周数,以及所在周的周五
  1106. // 规则:某年的第一个周五所在周为该年的第一周
  1107. // 返回:格式化字符串(如"202504")、周数(如4)和所在周的周五日期
  1108. func GetAssessmentWeekAndFriday(t time.Time) (string, int, time.Time) {
  1109. year := t.Year()
  1110. // 找到该年第一个周五的日期
  1111. firstFriday := findFirstFriday(year)
  1112. // 计算当前日期与该年第一个周五所在周的第一天(周一)的天数差
  1113. daysSinceFirstWeek := t.YearDay() - firstFriday.YearDay() + int(firstFriday.Weekday()) - int(time.Monday)
  1114. weekNum := 1
  1115. if daysSinceFirstWeek > 0 {
  1116. weekNum += daysSinceFirstWeek / 7
  1117. if daysSinceFirstWeek%7 != 0 {
  1118. weekNum++
  1119. }
  1120. } else {
  1121. // 如果当前日期在第一个周五所在周之前,则属于上一年的最后一周
  1122. prevYear := year - 1
  1123. prevFirstFriday := findFirstFriday(prevYear)
  1124. daysInPrevYear := daysInYear(prevYear)
  1125. daysSincePrevFirstWeek := t.YearDay() + (daysInPrevYear - prevFirstFriday.YearDay()) + int(prevFirstFriday.Weekday()) - int(time.Monday)
  1126. weekNum = daysSincePrevFirstWeek / 7
  1127. if daysSincePrevFirstWeek%7 != 0 {
  1128. weekNum++
  1129. }
  1130. year = prevYear
  1131. }
  1132. // 计算当前日期所在周的周五
  1133. currentWeekFriday := findFridayOfWeek(t)
  1134. // 格式化输出
  1135. formatted := fmt.Sprintf("%04d%02d", year, weekNum)
  1136. return formatted, weekNum, currentWeekFriday
  1137. }
  1138. // findFirstFriday 找到某年的第一个周五
  1139. func findFirstFriday(year int) time.Time {
  1140. // 从1月1日开始找
  1141. date := time.Date(year, time.January, 1, 0, 0, 0, 0, time.UTC)
  1142. // 找到第一个周五
  1143. for date.Weekday() != time.Friday {
  1144. date = date.AddDate(0, 0, 1)
  1145. }
  1146. return date
  1147. }
  1148. // findFridayOfWeek 找到给定日期所在周的周五
  1149. func findFridayOfWeek(t time.Time) time.Time {
  1150. // 获取当前日期是周几 (0=周日, 1=周一, ..., 6=周六)
  1151. weekday := int(t.Weekday())
  1152. // 计算到周五的天数差 (周五是5)
  1153. daysToFriday := (5 - weekday + 7) % 7
  1154. // 如果当前就是周五,daysToFriday会是0
  1155. if daysToFriday < 0 {
  1156. daysToFriday += 7
  1157. }
  1158. return t.AddDate(0, 0, daysToFriday)
  1159. }
  1160. // daysInYear 计算某年有多少天
  1161. func daysInYear(year int) int {
  1162. first := time.Date(year, time.January, 1, 0, 0, 0, 0, time.UTC)
  1163. last := time.Date(year, time.December, 31, 0, 0, 0, 0, time.UTC)
  1164. return last.YearDay() - first.YearDay() + 1
  1165. }
  1166. // IsErrNoRow
  1167. // @Description: 判断是否是gorm的查询不到数据的报错
  1168. // @param err
  1169. // @return bool
  1170. func IsErrNoRow(err error) bool {
  1171. if err == nil {
  1172. return false
  1173. }
  1174. return errors.Is(err, gorm.ErrRecordNotFound)
  1175. }