You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

home.go 12 kB

11 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
9 years ago
11 years ago
11 years ago
11 years ago
11 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509
  1. // Copyright 2014 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. "bytes"
  8. "fmt"
  9. "sort"
  10. "strings"
  11. "code.gitea.io/gitea/models"
  12. "code.gitea.io/gitea/modules/base"
  13. "code.gitea.io/gitea/modules/context"
  14. "code.gitea.io/gitea/modules/setting"
  15. "code.gitea.io/gitea/modules/util"
  16. "github.com/Unknwon/com"
  17. "github.com/Unknwon/paginater"
  18. )
  19. const (
  20. tplDashboard base.TplName = "user/dashboard/dashboard"
  21. tplIssues base.TplName = "user/dashboard/issues"
  22. tplProfile base.TplName = "user/profile"
  23. tplOrgHome base.TplName = "org/home"
  24. )
  25. // getDashboardContextUser finds out dashboard is viewing as which context user.
  26. func getDashboardContextUser(ctx *context.Context) *models.User {
  27. ctxUser := ctx.User
  28. orgName := ctx.Params(":org")
  29. if len(orgName) > 0 {
  30. // Organization.
  31. org, err := models.GetUserByName(orgName)
  32. if err != nil {
  33. if models.IsErrUserNotExist(err) {
  34. ctx.NotFound("GetUserByName", err)
  35. } else {
  36. ctx.ServerError("GetUserByName", err)
  37. }
  38. return nil
  39. }
  40. ctxUser = org
  41. }
  42. ctx.Data["ContextUser"] = ctxUser
  43. if err := ctx.User.GetOrganizations(true); err != nil {
  44. ctx.ServerError("GetOrganizations", err)
  45. return nil
  46. }
  47. ctx.Data["Orgs"] = ctx.User.Orgs
  48. return ctxUser
  49. }
  50. // retrieveFeeds loads feeds for the specified user
  51. func retrieveFeeds(ctx *context.Context, options models.GetFeedsOptions) {
  52. actions, err := models.GetFeeds(options)
  53. if err != nil {
  54. ctx.ServerError("GetFeeds", err)
  55. return
  56. }
  57. userCache := map[int64]*models.User{options.RequestedUser.ID: options.RequestedUser}
  58. if ctx.User != nil {
  59. userCache[ctx.User.ID] = ctx.User
  60. }
  61. for _, act := range actions {
  62. if act.ActUser != nil {
  63. userCache[act.ActUserID] = act.ActUser
  64. }
  65. repoOwner, ok := userCache[act.Repo.OwnerID]
  66. if !ok {
  67. repoOwner, err = models.GetUserByID(act.Repo.OwnerID)
  68. if err != nil {
  69. if models.IsErrUserNotExist(err) {
  70. continue
  71. }
  72. ctx.ServerError("GetUserByID", err)
  73. return
  74. }
  75. userCache[repoOwner.ID] = repoOwner
  76. }
  77. act.Repo.Owner = repoOwner
  78. }
  79. ctx.Data["Feeds"] = actions
  80. }
  81. // Dashboard render the dashborad page
  82. func Dashboard(ctx *context.Context) {
  83. ctxUser := getDashboardContextUser(ctx)
  84. if ctx.Written() {
  85. return
  86. }
  87. ctx.Data["Title"] = ctxUser.DisplayName() + " - " + ctx.Tr("dashboard")
  88. ctx.Data["PageIsDashboard"] = true
  89. ctx.Data["PageIsNews"] = true
  90. ctx.Data["SearchLimit"] = setting.UI.User.RepoPagingNum
  91. ctx.Data["EnableHeatmap"] = setting.Service.EnableUserHeatmap
  92. ctx.Data["HeatmapUser"] = ctxUser.Name
  93. var err error
  94. var mirrors []*models.Repository
  95. if ctxUser.IsOrganization() {
  96. env, err := ctxUser.AccessibleReposEnv(ctx.User.ID)
  97. if err != nil {
  98. ctx.ServerError("AccessibleReposEnv", err)
  99. return
  100. }
  101. mirrors, err = env.MirrorRepos()
  102. if err != nil {
  103. ctx.ServerError("env.MirrorRepos", err)
  104. return
  105. }
  106. } else {
  107. mirrors, err = ctxUser.GetMirrorRepositories()
  108. if err != nil {
  109. ctx.ServerError("GetMirrorRepositories", err)
  110. return
  111. }
  112. }
  113. ctx.Data["MaxShowRepoNum"] = setting.UI.User.RepoPagingNum
  114. if err := models.MirrorRepositoryList(mirrors).LoadAttributes(); err != nil {
  115. ctx.ServerError("MirrorRepositoryList.LoadAttributes", err)
  116. return
  117. }
  118. ctx.Data["MirrorCount"] = len(mirrors)
  119. ctx.Data["Mirrors"] = mirrors
  120. retrieveFeeds(ctx, models.GetFeedsOptions{
  121. RequestedUser: ctxUser,
  122. IncludePrivate: true,
  123. OnlyPerformedBy: false,
  124. IncludeDeleted: false,
  125. })
  126. if ctx.Written() {
  127. return
  128. }
  129. ctx.HTML(200, tplDashboard)
  130. }
  131. // Issues render the user issues page
  132. func Issues(ctx *context.Context) {
  133. isPullList := ctx.Params(":type") == "pulls"
  134. if isPullList {
  135. ctx.Data["Title"] = ctx.Tr("pull_requests")
  136. ctx.Data["PageIsPulls"] = true
  137. } else {
  138. ctx.Data["Title"] = ctx.Tr("issues")
  139. ctx.Data["PageIsIssues"] = true
  140. }
  141. ctxUser := getDashboardContextUser(ctx)
  142. if ctx.Written() {
  143. return
  144. }
  145. // Organization does not have view type and filter mode.
  146. var (
  147. viewType string
  148. sortType = ctx.Query("sort")
  149. filterMode = models.FilterModeAll
  150. )
  151. if ctxUser.IsOrganization() {
  152. viewType = "all"
  153. } else {
  154. viewType = ctx.Query("type")
  155. switch viewType {
  156. case "assigned":
  157. filterMode = models.FilterModeAssign
  158. case "created_by":
  159. filterMode = models.FilterModeCreate
  160. case "all": // filterMode already set to All
  161. default:
  162. viewType = "all"
  163. }
  164. }
  165. page := ctx.QueryInt("page")
  166. if page <= 1 {
  167. page = 1
  168. }
  169. repoID := ctx.QueryInt64("repo")
  170. isShowClosed := ctx.Query("state") == "closed"
  171. // Get repositories.
  172. var err error
  173. var userRepoIDs []int64
  174. if ctxUser.IsOrganization() {
  175. env, err := ctxUser.AccessibleReposEnv(ctx.User.ID)
  176. if err != nil {
  177. ctx.ServerError("AccessibleReposEnv", err)
  178. return
  179. }
  180. userRepoIDs, err = env.RepoIDs(1, ctxUser.NumRepos)
  181. if err != nil {
  182. ctx.ServerError("env.RepoIDs", err)
  183. return
  184. }
  185. } else {
  186. unitType := models.UnitTypeIssues
  187. if isPullList {
  188. unitType = models.UnitTypePullRequests
  189. }
  190. userRepoIDs, err = ctxUser.GetAccessRepoIDs(unitType)
  191. if err != nil {
  192. ctx.ServerError("ctxUser.GetAccessRepoIDs", err)
  193. return
  194. }
  195. }
  196. if len(userRepoIDs) == 0 {
  197. userRepoIDs = []int64{-1}
  198. }
  199. opts := &models.IssuesOptions{
  200. IsClosed: util.OptionalBoolOf(isShowClosed),
  201. IsPull: util.OptionalBoolOf(isPullList),
  202. SortType: sortType,
  203. }
  204. if repoID > 0 {
  205. opts.RepoIDs = []int64{repoID}
  206. }
  207. switch filterMode {
  208. case models.FilterModeAll:
  209. if repoID > 0 {
  210. if !com.IsSliceContainsInt64(userRepoIDs, repoID) {
  211. // force an empty result
  212. opts.RepoIDs = []int64{-1}
  213. }
  214. } else {
  215. opts.RepoIDs = userRepoIDs
  216. }
  217. case models.FilterModeAssign:
  218. opts.AssigneeID = ctxUser.ID
  219. case models.FilterModeCreate:
  220. opts.PosterID = ctxUser.ID
  221. case models.FilterModeMention:
  222. opts.MentionedID = ctxUser.ID
  223. }
  224. counts, err := models.CountIssuesByRepo(opts)
  225. if err != nil {
  226. ctx.ServerError("CountIssuesByRepo", err)
  227. return
  228. }
  229. opts.Page = page
  230. opts.PageSize = setting.UI.IssuePagingNum
  231. var labelIDs []int64
  232. selectLabels := ctx.Query("labels")
  233. if len(selectLabels) > 0 && selectLabels != "0" {
  234. labelIDs, err = base.StringsToInt64s(strings.Split(selectLabels, ","))
  235. if err != nil {
  236. ctx.ServerError("StringsToInt64s", err)
  237. return
  238. }
  239. }
  240. opts.LabelIDs = labelIDs
  241. issues, err := models.Issues(opts)
  242. if err != nil {
  243. ctx.ServerError("Issues", err)
  244. return
  245. }
  246. showReposMap := make(map[int64]*models.Repository, len(counts))
  247. for repoID := range counts {
  248. repo, err := models.GetRepositoryByID(repoID)
  249. if err != nil {
  250. ctx.ServerError("GetRepositoryByID", err)
  251. return
  252. }
  253. showReposMap[repoID] = repo
  254. }
  255. if repoID > 0 {
  256. if _, ok := showReposMap[repoID]; !ok {
  257. repo, err := models.GetRepositoryByID(repoID)
  258. if models.IsErrRepoNotExist(err) {
  259. ctx.NotFound("GetRepositoryByID", err)
  260. return
  261. } else if err != nil {
  262. ctx.ServerError("GetRepositoryByID", fmt.Errorf("[%d]%v", repoID, err))
  263. return
  264. }
  265. showReposMap[repoID] = repo
  266. }
  267. repo := showReposMap[repoID]
  268. // Check if user has access to given repository.
  269. perm, err := models.GetUserRepoPermission(repo, ctxUser)
  270. if err != nil {
  271. ctx.ServerError("GetUserRepoPermission", fmt.Errorf("[%d]%v", repoID, err))
  272. return
  273. }
  274. if !perm.CanRead(models.UnitTypeIssues) {
  275. ctx.Status(404)
  276. return
  277. }
  278. }
  279. showRepos := models.RepositoryListOfMap(showReposMap)
  280. sort.Sort(showRepos)
  281. if err = showRepos.LoadAttributes(); err != nil {
  282. ctx.ServerError("LoadAttributes", err)
  283. return
  284. }
  285. for _, issue := range issues {
  286. issue.Repo = showReposMap[issue.RepoID]
  287. }
  288. issueStats, err := models.GetUserIssueStats(models.UserIssueStatsOptions{
  289. UserID: ctxUser.ID,
  290. RepoID: repoID,
  291. UserRepoIDs: userRepoIDs,
  292. FilterMode: filterMode,
  293. IsPull: isPullList,
  294. IsClosed: isShowClosed,
  295. })
  296. if err != nil {
  297. ctx.ServerError("GetUserIssueStats", err)
  298. return
  299. }
  300. var total int
  301. if !isShowClosed {
  302. total = int(issueStats.OpenCount)
  303. } else {
  304. total = int(issueStats.ClosedCount)
  305. }
  306. ctx.Data["Issues"] = issues
  307. ctx.Data["Repos"] = showRepos
  308. ctx.Data["Counts"] = counts
  309. ctx.Data["Page"] = paginater.New(total, setting.UI.IssuePagingNum, page, 5)
  310. ctx.Data["IssueStats"] = issueStats
  311. ctx.Data["ViewType"] = viewType
  312. ctx.Data["SortType"] = sortType
  313. ctx.Data["RepoID"] = repoID
  314. ctx.Data["IsShowClosed"] = isShowClosed
  315. if isShowClosed {
  316. ctx.Data["State"] = "closed"
  317. } else {
  318. ctx.Data["State"] = "open"
  319. }
  320. ctx.HTML(200, tplIssues)
  321. }
  322. // ShowSSHKeys output all the ssh keys of user by uid
  323. func ShowSSHKeys(ctx *context.Context, uid int64) {
  324. keys, err := models.ListPublicKeys(uid)
  325. if err != nil {
  326. ctx.ServerError("ListPublicKeys", err)
  327. return
  328. }
  329. var buf bytes.Buffer
  330. for i := range keys {
  331. buf.WriteString(keys[i].OmitEmail())
  332. buf.WriteString("\n")
  333. }
  334. ctx.PlainText(200, buf.Bytes())
  335. }
  336. func showOrgProfile(ctx *context.Context) {
  337. ctx.SetParams(":org", ctx.Params(":username"))
  338. context.HandleOrgAssignment(ctx)
  339. if ctx.Written() {
  340. return
  341. }
  342. org := ctx.Org.Organization
  343. if !models.HasOrgVisible(org, ctx.User) {
  344. ctx.NotFound("HasOrgVisible", nil)
  345. return
  346. }
  347. ctx.Data["Title"] = org.DisplayName()
  348. var orderBy models.SearchOrderBy
  349. ctx.Data["SortType"] = ctx.Query("sort")
  350. switch ctx.Query("sort") {
  351. case "newest":
  352. orderBy = models.SearchOrderByNewest
  353. case "oldest":
  354. orderBy = models.SearchOrderByOldest
  355. case "recentupdate":
  356. orderBy = models.SearchOrderByRecentUpdated
  357. case "leastupdate":
  358. orderBy = models.SearchOrderByLeastUpdated
  359. case "reversealphabetically":
  360. orderBy = models.SearchOrderByAlphabeticallyReverse
  361. case "alphabetically":
  362. orderBy = models.SearchOrderByAlphabetically
  363. case "moststars":
  364. orderBy = models.SearchOrderByStarsReverse
  365. case "feweststars":
  366. orderBy = models.SearchOrderByStars
  367. case "mostforks":
  368. orderBy = models.SearchOrderByForksReverse
  369. case "fewestforks":
  370. orderBy = models.SearchOrderByForks
  371. default:
  372. ctx.Data["SortType"] = "recentupdate"
  373. orderBy = models.SearchOrderByRecentUpdated
  374. }
  375. keyword := strings.Trim(ctx.Query("q"), " ")
  376. ctx.Data["Keyword"] = keyword
  377. page := ctx.QueryInt("page")
  378. if page <= 0 {
  379. page = 1
  380. }
  381. var (
  382. repos []*models.Repository
  383. count int64
  384. err error
  385. )
  386. if ctx.IsSigned && !ctx.User.IsAdmin {
  387. env, err := org.AccessibleReposEnv(ctx.User.ID)
  388. if err != nil {
  389. ctx.ServerError("AccessibleReposEnv", err)
  390. return
  391. }
  392. if len(keyword) != 0 {
  393. env.AddKeyword(keyword)
  394. }
  395. repos, err = env.Repos(page, setting.UI.User.RepoPagingNum)
  396. if err != nil {
  397. ctx.ServerError("env.Repos", err)
  398. return
  399. }
  400. count, err = env.CountRepos()
  401. if err != nil {
  402. ctx.ServerError("env.CountRepos", err)
  403. return
  404. }
  405. } else {
  406. showPrivate := ctx.IsSigned && ctx.User.IsAdmin
  407. if len(keyword) == 0 {
  408. repos, err = models.GetUserRepositories(org.ID, showPrivate, page, setting.UI.User.RepoPagingNum, orderBy.String())
  409. if err != nil {
  410. ctx.ServerError("GetRepositories", err)
  411. return
  412. }
  413. count = models.CountUserRepositories(org.ID, showPrivate)
  414. } else {
  415. repos, count, err = models.SearchRepositoryByName(&models.SearchRepoOptions{
  416. Keyword: keyword,
  417. OwnerID: org.ID,
  418. OrderBy: orderBy,
  419. Private: showPrivate,
  420. Page: page,
  421. IsProfile: true,
  422. PageSize: setting.UI.User.RepoPagingNum,
  423. })
  424. if err != nil {
  425. ctx.ServerError("SearchRepositoryByName", err)
  426. return
  427. }
  428. }
  429. }
  430. if err := org.GetMembers(); err != nil {
  431. ctx.ServerError("GetMembers", err)
  432. return
  433. }
  434. ctx.Data["Repos"] = repos
  435. ctx.Data["Total"] = count
  436. ctx.Data["Page"] = paginater.New(int(count), setting.UI.User.RepoPagingNum, page, 5)
  437. ctx.Data["Members"] = org.Members
  438. ctx.Data["Teams"] = org.Teams
  439. ctx.HTML(200, tplOrgHome)
  440. }
  441. // Email2User show user page via email
  442. func Email2User(ctx *context.Context) {
  443. u, err := models.GetUserByEmail(ctx.Query("email"))
  444. if err != nil {
  445. if models.IsErrUserNotExist(err) {
  446. ctx.NotFound("GetUserByEmail", err)
  447. } else {
  448. ctx.ServerError("GetUserByEmail", err)
  449. }
  450. return
  451. }
  452. ctx.Redirect(setting.AppSubURL + "/user/" + u.Name)
  453. }