captcha.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. package service
  2. import (
  3. "bytes"
  4. "context"
  5. "encoding/hex"
  6. "fmt"
  7. "image/jpeg"
  8. "image/png"
  9. "math/rand"
  10. "strings"
  11. "sync"
  12. "time"
  13. "go-common/library/ecode"
  14. uuid "github.com/satori/go.uuid"
  15. )
  16. const (
  17. _captchaURL = "%s/x/v1/captcha/get?bid=%s&token=%s"
  18. )
  19. // Token use bid, get a token.
  20. func (s *Service) Token(c context.Context, bid string) (url string, token string, err error) {
  21. token = hex.EncodeToString(uuid.NewV4().Bytes())
  22. business := s.LookUp(bid)
  23. if err = s.dao.AddTokenCache(c, token, int32(time.Duration(business.TTL)/time.Second)); err != nil {
  24. return
  25. }
  26. url = fmt.Sprintf(_captchaURL, s.conf.Captcha.OuterHost, bid, token)
  27. return
  28. }
  29. // CaptchaImg get a captcha by token,bid.
  30. func (s *Service) CaptchaImg(c context.Context, token, bid string) (img []byte, err error) {
  31. code, img, ttl := s.randomCaptcha(bid)
  32. realCode, _, err := s.dao.CaptchaCache(c, token)
  33. if err != nil {
  34. return
  35. }
  36. if realCode == "" {
  37. err = ecode.CaptchaTokenExpired
  38. return
  39. }
  40. err = s.dao.UpdateTokenCache(c, token, code, ttl)
  41. return
  42. }
  43. // VerifyCaptcha verify captcha by token and code.
  44. func (s *Service) VerifyCaptcha(c context.Context, token, code string) (err error) {
  45. var (
  46. realCode string
  47. isInit bool
  48. )
  49. if realCode, isInit, err = s.dao.CaptchaCache(c, token); err != nil {
  50. return
  51. }
  52. if realCode == "" {
  53. err = ecode.CaptchaCodeNotFound
  54. return
  55. }
  56. if isInit {
  57. err = ecode.CaptchaNotCreate
  58. return
  59. }
  60. if ok := strings.ToLower(realCode) == strings.ToLower(code); ok {
  61. s.cacheCh.Save(func() {
  62. s.dao.DelCaptchaCache(context.Background(), token)
  63. })
  64. } else {
  65. err = ecode.CaptchaErr
  66. }
  67. return
  68. }
  69. func (s *Service) initGenerater(waiter *sync.WaitGroup, bid string, lenStart, lenEnd, width, length int) {
  70. s.generater(bid, lenStart, lenEnd, width, length)
  71. waiter.Done()
  72. }
  73. func (s *Service) generater(bid string, lenStart, lenEnd, width, length int) {
  74. images := make(map[string][]byte, s.conf.Captcha.Capacity)
  75. codes := make([]string, 0, s.conf.Captcha.Capacity)
  76. for i := 0; i < s.conf.Captcha.Capacity; i++ {
  77. img, code := s.captcha.createImage(lenStart, lenEnd, width, length, TypeALL)
  78. var b bytes.Buffer
  79. switch s.conf.Captcha.Ext {
  80. case "png":
  81. png.Encode(&b, img)
  82. case "jpeg":
  83. jpeg.Encode(&b, img, &jpeg.Options{Quality: 100})
  84. default:
  85. jpeg.Encode(&b, img, &jpeg.Options{Quality: 100})
  86. }
  87. images[code] = b.Bytes()
  88. codes = append(codes, code)
  89. }
  90. s.lock.Lock()
  91. s.mImage[bid] = images
  92. s.mCode[bid] = codes
  93. s.lock.Unlock()
  94. }
  95. func (s *Service) randomCaptcha(bid string) (code string, img []byte, ttl int32) {
  96. business := s.LookUp(bid)
  97. ttl = int32(time.Duration(business.TTL) / time.Second)
  98. rnd := rand.Intn(s.conf.Captcha.Capacity)
  99. code = s.mCode[business.BusinessID][rnd]
  100. img = s.mImage[business.BusinessID][code]
  101. return
  102. }