Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

profile.go 7.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  1. // Copyright 2015 The Gogs Authors. All rights reserved.
  2. // Copyright 2019 The Gitea Authors. All rights reserved.
  3. // Use of this source code is governed by a MIT-style
  4. // license that can be found in the LICENSE file.
  5. package user
  6. import (
  7. "fmt"
  8. "path"
  9. "strings"
  10. "code.gitea.io/gitea/models"
  11. "code.gitea.io/gitea/modules/context"
  12. "code.gitea.io/gitea/modules/markup/markdown"
  13. "code.gitea.io/gitea/modules/setting"
  14. "code.gitea.io/gitea/modules/util"
  15. "code.gitea.io/gitea/routers/org"
  16. )
  17. // GetUserByName get user by name
  18. func GetUserByName(ctx *context.Context, name string) *models.User {
  19. user, err := models.GetUserByName(name)
  20. if err != nil {
  21. if models.IsErrUserNotExist(err) {
  22. if redirectUserID, err := models.LookupUserRedirect(name); err == nil {
  23. context.RedirectToUser(ctx, name, redirectUserID)
  24. } else {
  25. ctx.NotFound("GetUserByName", err)
  26. }
  27. } else {
  28. ctx.ServerError("GetUserByName", err)
  29. }
  30. return nil
  31. }
  32. return user
  33. }
  34. // GetUserByParams returns user whose name is presented in URL paramenter.
  35. func GetUserByParams(ctx *context.Context) *models.User {
  36. return GetUserByName(ctx, ctx.Params(":username"))
  37. }
  38. // Profile render user's profile page
  39. func Profile(ctx *context.Context) {
  40. uname := ctx.Params(":username")
  41. // Special handle for FireFox requests favicon.ico.
  42. if uname == "favicon.ico" {
  43. ctx.ServeFile(path.Join(setting.StaticRootPath, "public/img/favicon.png"))
  44. return
  45. } else if strings.HasSuffix(uname, ".png") {
  46. ctx.Error(404)
  47. return
  48. }
  49. isShowKeys := false
  50. if strings.HasSuffix(uname, ".keys") {
  51. isShowKeys = true
  52. uname = strings.TrimSuffix(uname, ".keys")
  53. }
  54. isShowGPG := false
  55. if strings.HasSuffix(uname, ".gpg") {
  56. isShowGPG = true
  57. uname = strings.TrimSuffix(uname, ".gpg")
  58. }
  59. ctxUser := GetUserByName(ctx, uname)
  60. if ctx.Written() {
  61. return
  62. }
  63. // Show SSH keys.
  64. if isShowKeys {
  65. ShowSSHKeys(ctx, ctxUser.ID)
  66. return
  67. }
  68. // Show GPG keys.
  69. if isShowGPG {
  70. ShowGPGKeys(ctx, ctxUser.ID)
  71. return
  72. }
  73. if ctxUser.IsOrganization() {
  74. org.Home(ctx)
  75. return
  76. }
  77. // Show OpenID URIs
  78. openIDs, err := models.GetUserOpenIDs(ctxUser.ID)
  79. if err != nil {
  80. ctx.ServerError("GetUserOpenIDs", err)
  81. return
  82. }
  83. ctx.Data["Title"] = ctxUser.DisplayName()
  84. ctx.Data["PageIsUserProfile"] = true
  85. ctx.Data["Owner"] = ctxUser
  86. ctx.Data["OpenIDs"] = openIDs
  87. // no heatmap access for admins; GetUserHeatmapDataByUser ignores the calling user
  88. // so everyone would get the same empty heatmap
  89. if setting.Service.EnableUserHeatmap && !ctxUser.KeepActivityPrivate {
  90. data, err := models.GetUserHeatmapDataByUser(ctxUser, ctx.User)
  91. if err != nil {
  92. ctx.ServerError("GetUserHeatmapDataByUser", err)
  93. return
  94. }
  95. ctx.Data["HeatmapData"] = data
  96. }
  97. if len(ctxUser.Description) != 0 {
  98. ctx.Data["RenderedDescription"] = string(markdown.Render([]byte(ctxUser.Description), ctx.Repo.RepoLink, map[string]string{"mode": "document"}))
  99. }
  100. showPrivate := ctx.IsSigned && (ctx.User.IsAdmin || ctx.User.ID == ctxUser.ID)
  101. orgs, err := models.GetOrgsByUserID(ctxUser.ID, showPrivate)
  102. if err != nil {
  103. ctx.ServerError("GetOrgsByUserIDDesc", err)
  104. return
  105. }
  106. ctx.Data["Orgs"] = orgs
  107. ctx.Data["HasOrgsVisible"] = models.HasOrgsVisible(orgs, ctx.User)
  108. tab := ctx.Query("tab")
  109. ctx.Data["TabName"] = tab
  110. page := ctx.QueryInt("page")
  111. if page <= 0 {
  112. page = 1
  113. }
  114. topicOnly := ctx.QueryBool("topic")
  115. var (
  116. repos []*models.Repository
  117. count int64
  118. total int
  119. orderBy models.SearchOrderBy
  120. )
  121. ctx.Data["SortType"] = ctx.Query("sort")
  122. switch ctx.Query("sort") {
  123. case "newest":
  124. orderBy = models.SearchOrderByNewest
  125. case "oldest":
  126. orderBy = models.SearchOrderByOldest
  127. case "recentupdate":
  128. orderBy = models.SearchOrderByRecentUpdated
  129. case "leastupdate":
  130. orderBy = models.SearchOrderByLeastUpdated
  131. case "reversealphabetically":
  132. orderBy = models.SearchOrderByAlphabeticallyReverse
  133. case "alphabetically":
  134. orderBy = models.SearchOrderByAlphabetically
  135. case "moststars":
  136. orderBy = models.SearchOrderByStarsReverse
  137. case "feweststars":
  138. orderBy = models.SearchOrderByStars
  139. case "mostforks":
  140. orderBy = models.SearchOrderByForksReverse
  141. case "fewestforks":
  142. orderBy = models.SearchOrderByForks
  143. default:
  144. ctx.Data["SortType"] = "recentupdate"
  145. orderBy = models.SearchOrderByRecentUpdated
  146. }
  147. keyword := strings.Trim(ctx.Query("q"), " ")
  148. ctx.Data["Keyword"] = keyword
  149. switch tab {
  150. case "followers":
  151. items, err := ctxUser.GetFollowers(models.ListOptions{
  152. PageSize: setting.UI.User.RepoPagingNum,
  153. Page: page,
  154. })
  155. if err != nil {
  156. ctx.ServerError("GetFollowers", err)
  157. return
  158. }
  159. ctx.Data["Cards"] = items
  160. total = ctxUser.NumFollowers
  161. case "following":
  162. items, err := ctxUser.GetFollowing(models.ListOptions{
  163. PageSize: setting.UI.User.RepoPagingNum,
  164. Page: page,
  165. })
  166. if err != nil {
  167. ctx.ServerError("GetFollowing", err)
  168. return
  169. }
  170. ctx.Data["Cards"] = items
  171. total = ctxUser.NumFollowing
  172. case "activity":
  173. retrieveFeeds(ctx, models.GetFeedsOptions{RequestedUser: ctxUser,
  174. Actor: ctx.User,
  175. IncludePrivate: showPrivate,
  176. OnlyPerformedBy: true,
  177. IncludeDeleted: false,
  178. })
  179. if ctx.Written() {
  180. return
  181. }
  182. case "stars":
  183. ctx.Data["PageIsProfileStarList"] = true
  184. repos, count, err = models.SearchRepository(&models.SearchRepoOptions{
  185. ListOptions: models.ListOptions{
  186. PageSize: setting.UI.User.RepoPagingNum,
  187. Page: page,
  188. },
  189. Actor: ctx.User,
  190. Keyword: keyword,
  191. OrderBy: orderBy,
  192. Private: ctx.IsSigned,
  193. StarredByID: ctxUser.ID,
  194. Collaborate: util.OptionalBoolFalse,
  195. TopicOnly: topicOnly,
  196. IncludeDescription: setting.UI.SearchRepoDescription,
  197. })
  198. if err != nil {
  199. ctx.ServerError("SearchRepository", err)
  200. return
  201. }
  202. total = int(count)
  203. case "projects":
  204. ctx.Data["OpenProjects"], _, err = models.GetProjects(models.ProjectSearchOptions{
  205. Page: -1,
  206. IsClosed: util.OptionalBoolFalse,
  207. Type: models.ProjectTypeIndividual,
  208. })
  209. if err != nil {
  210. ctx.ServerError("GetProjects", err)
  211. return
  212. }
  213. default:
  214. repos, count, err = models.SearchRepository(&models.SearchRepoOptions{
  215. ListOptions: models.ListOptions{
  216. PageSize: setting.UI.User.RepoPagingNum,
  217. Page: page,
  218. },
  219. Actor: ctx.User,
  220. Keyword: keyword,
  221. OwnerID: ctxUser.ID,
  222. OrderBy: orderBy,
  223. Private: ctx.IsSigned,
  224. Collaborate: util.OptionalBoolFalse,
  225. TopicOnly: topicOnly,
  226. IncludeDescription: setting.UI.SearchRepoDescription,
  227. })
  228. if err != nil {
  229. ctx.ServerError("SearchRepository", err)
  230. return
  231. }
  232. total = int(count)
  233. }
  234. ctx.Data["Repos"] = repos
  235. ctx.Data["Total"] = total
  236. pager := context.NewPagination(total, setting.UI.User.RepoPagingNum, page, 5)
  237. pager.SetDefaultParams(ctx)
  238. ctx.Data["Page"] = pager
  239. ctx.Data["ShowUserEmail"] = len(ctxUser.Email) > 0 && ctx.IsSigned && (!ctxUser.KeepEmailPrivate || ctxUser.ID == ctx.User.ID)
  240. ctx.HTML(200, tplProfile)
  241. }
  242. // Action response for follow/unfollow user request
  243. func Action(ctx *context.Context) {
  244. u := GetUserByParams(ctx)
  245. if ctx.Written() {
  246. return
  247. }
  248. var err error
  249. switch ctx.Params(":action") {
  250. case "follow":
  251. err = models.FollowUser(ctx.User.ID, u.ID)
  252. case "unfollow":
  253. err = models.UnfollowUser(ctx.User.ID, u.ID)
  254. }
  255. if err != nil {
  256. ctx.ServerError(fmt.Sprintf("Action (%s)", ctx.Params(":action")), err)
  257. return
  258. }
  259. ctx.RedirectToFirst(ctx.Query("redirect_to"), u.HomeLink())
  260. }