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.

avatar.go 6.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. // Copyright 2021 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package avatars
  4. import (
  5. "context"
  6. "net/url"
  7. "path"
  8. "strconv"
  9. "strings"
  10. "sync"
  11. "code.gitea.io/gitea/models/db"
  12. system_model "code.gitea.io/gitea/models/system"
  13. "code.gitea.io/gitea/modules/base"
  14. "code.gitea.io/gitea/modules/cache"
  15. "code.gitea.io/gitea/modules/log"
  16. "code.gitea.io/gitea/modules/setting"
  17. )
  18. const (
  19. // DefaultAvatarClass is the default class of a rendered avatar
  20. DefaultAvatarClass = "ui avatar vm"
  21. // DefaultAvatarPixelSize is the default size in pixels of a rendered avatar
  22. DefaultAvatarPixelSize = 28
  23. )
  24. // EmailHash represents a pre-generated hash map (mainly used by LibravatarURL, it queries email server's DNS records)
  25. type EmailHash struct {
  26. Hash string `xorm:"pk varchar(32)"`
  27. Email string `xorm:"UNIQUE NOT NULL"`
  28. }
  29. func init() {
  30. db.RegisterModel(new(EmailHash))
  31. }
  32. var (
  33. defaultAvatarLink string
  34. once sync.Once
  35. )
  36. // DefaultAvatarLink the default avatar link
  37. func DefaultAvatarLink() string {
  38. once.Do(func() {
  39. u, err := url.Parse(setting.AppSubURL)
  40. if err != nil {
  41. log.Error("Can not parse AppSubURL: %v", err)
  42. return
  43. }
  44. u.Path = path.Join(u.Path, "/assets/img/avatar_default.png")
  45. defaultAvatarLink = u.String()
  46. })
  47. return defaultAvatarLink
  48. }
  49. // HashEmail hashes email address to MD5 string. https://en.gravatar.com/site/implement/hash/
  50. func HashEmail(email string) string {
  51. return base.EncodeMD5(strings.ToLower(strings.TrimSpace(email)))
  52. }
  53. // GetEmailForHash converts a provided md5sum to the email
  54. func GetEmailForHash(md5Sum string) (string, error) {
  55. return cache.GetString("Avatar:"+md5Sum, func() (string, error) {
  56. emailHash := EmailHash{
  57. Hash: strings.ToLower(strings.TrimSpace(md5Sum)),
  58. }
  59. _, err := db.GetEngine(db.DefaultContext).Get(&emailHash)
  60. return emailHash.Email, err
  61. })
  62. }
  63. // LibravatarURL returns the URL for the given email. Slow due to the DNS lookup.
  64. // This function should only be called if a federated avatar service is enabled.
  65. func LibravatarURL(email string) (*url.URL, error) {
  66. urlStr, err := system_model.LibravatarService.FromEmail(email)
  67. if err != nil {
  68. log.Error("LibravatarService.FromEmail(email=%s): error %v", email, err)
  69. return nil, err
  70. }
  71. u, err := url.Parse(urlStr)
  72. if err != nil {
  73. log.Error("Failed to parse libravatar url(%s): error %v", urlStr, err)
  74. return nil, err
  75. }
  76. return u, nil
  77. }
  78. // saveEmailHash returns an avatar link for a provided email,
  79. // the email and hash are saved into database, which will be used by GetEmailForHash later
  80. func saveEmailHash(email string) string {
  81. lowerEmail := strings.ToLower(strings.TrimSpace(email))
  82. emailHash := HashEmail(lowerEmail)
  83. _, _ = cache.GetString("Avatar:"+emailHash, func() (string, error) {
  84. emailHash := &EmailHash{
  85. Email: lowerEmail,
  86. Hash: emailHash,
  87. }
  88. // OK we're going to open a session just because I think that that might hide away any problems with postgres reporting errors
  89. if err := db.WithTx(db.DefaultContext, func(ctx context.Context) error {
  90. has, err := db.GetEngine(ctx).Where("email = ? AND hash = ?", emailHash.Email, emailHash.Hash).Get(new(EmailHash))
  91. if has || err != nil {
  92. // Seriously we don't care about any DB problems just return the lowerEmail - we expect the transaction to fail most of the time
  93. return nil
  94. }
  95. _, _ = db.GetEngine(ctx).Insert(emailHash)
  96. return nil
  97. }); err != nil {
  98. // Seriously we don't care about any DB problems just return the lowerEmail - we expect the transaction to fail most of the time
  99. return lowerEmail, nil
  100. }
  101. return lowerEmail, nil
  102. })
  103. return emailHash
  104. }
  105. // GenerateUserAvatarFastLink returns a fast link (302) to the user's avatar: "/user/avatar/${User.Name}/${size}"
  106. func GenerateUserAvatarFastLink(userName string, size int) string {
  107. if size < 0 {
  108. size = 0
  109. }
  110. return setting.AppSubURL + "/user/avatar/" + url.PathEscape(userName) + "/" + strconv.Itoa(size)
  111. }
  112. // GenerateUserAvatarImageLink returns a link for `User.Avatar` image file: "/avatars/${User.Avatar}"
  113. func GenerateUserAvatarImageLink(userAvatar string, size int) string {
  114. if size > 0 {
  115. return setting.AppSubURL + "/avatars/" + url.PathEscape(userAvatar) + "?size=" + strconv.Itoa(size)
  116. }
  117. return setting.AppSubURL + "/avatars/" + url.PathEscape(userAvatar)
  118. }
  119. // generateRecognizedAvatarURL generate a recognized avatar (Gravatar/Libravatar) URL, it modifies the URL so the parameter is passed by a copy
  120. func generateRecognizedAvatarURL(u url.URL, size int) string {
  121. urlQuery := u.Query()
  122. urlQuery.Set("d", "identicon")
  123. if size > 0 {
  124. urlQuery.Set("s", strconv.Itoa(size))
  125. }
  126. u.RawQuery = urlQuery.Encode()
  127. return u.String()
  128. }
  129. // generateEmailAvatarLink returns a email avatar link.
  130. // if final is true, it may use a slow path (eg: query DNS).
  131. // if final is false, it always uses a fast path.
  132. func generateEmailAvatarLink(email string, size int, final bool) string {
  133. email = strings.TrimSpace(email)
  134. if email == "" {
  135. return DefaultAvatarLink()
  136. }
  137. enableFederatedAvatarSetting, _ := system_model.GetSetting(system_model.KeyPictureEnableFederatedAvatar)
  138. enableFederatedAvatar := enableFederatedAvatarSetting.GetValueBool()
  139. var err error
  140. if enableFederatedAvatar && system_model.LibravatarService != nil {
  141. emailHash := saveEmailHash(email)
  142. if final {
  143. // for final link, we can spend more time on slow external query
  144. var avatarURL *url.URL
  145. if avatarURL, err = LibravatarURL(email); err != nil {
  146. return DefaultAvatarLink()
  147. }
  148. return generateRecognizedAvatarURL(*avatarURL, size)
  149. }
  150. // for non-final link, we should return fast (use a 302 redirection link)
  151. urlStr := setting.AppSubURL + "/avatar/" + url.PathEscape(emailHash)
  152. if size > 0 {
  153. urlStr += "?size=" + strconv.Itoa(size)
  154. }
  155. return urlStr
  156. }
  157. disableGravatarSetting, _ := system_model.GetSetting(system_model.KeyPictureDisableGravatar)
  158. disableGravatar := disableGravatarSetting.GetValueBool()
  159. if !disableGravatar {
  160. // copy GravatarSourceURL, because we will modify its Path.
  161. avatarURLCopy := *system_model.GravatarSourceURL
  162. avatarURLCopy.Path = path.Join(avatarURLCopy.Path, HashEmail(email))
  163. return generateRecognizedAvatarURL(avatarURLCopy, size)
  164. }
  165. return DefaultAvatarLink()
  166. }
  167. // GenerateEmailAvatarFastLink returns a avatar link (fast, the link may be a delegated one: "/avatar/${hash}")
  168. func GenerateEmailAvatarFastLink(email string, size int) string {
  169. return generateEmailAvatarLink(email, size, false)
  170. }
  171. // GenerateEmailAvatarFinalLink returns a avatar final link (maybe slow)
  172. func GenerateEmailAvatarFinalLink(email string, size int) string {
  173. return generateEmailAvatarLink(email, size, true)
  174. }