gitea源码

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