aboutsummaryrefslogtreecommitdiff
path: root/internal/route/user/home.go
diff options
context:
space:
mode:
authorUnknwon <u@gogs.io>2019-10-24 01:51:46 -0700
committerGitHub <noreply@github.com>2019-10-24 01:51:46 -0700
commit01c8df01ec0608f1f25b2f1444adabb98fa5ee8a (patch)
treef8a7e5dd8d2a8c51e1ce2cabb9d33571a93314dd /internal/route/user/home.go
parent613139e7bef81d3573e7988a47eb6765f3de347a (diff)
internal: move packages under this directory (#5836)
* Rename pkg -> internal * Rename routes -> route * Move route -> internal/route * Rename models -> db * Move db -> internal/db * Fix route2 -> route * Move cmd -> internal/cmd * Bump version
Diffstat (limited to 'internal/route/user/home.go')
-rw-r--r--internal/route/user/home.go424
1 files changed, 424 insertions, 0 deletions
diff --git a/internal/route/user/home.go b/internal/route/user/home.go
new file mode 100644
index 00000000..c411fae0
--- /dev/null
+++ b/internal/route/user/home.go
@@ -0,0 +1,424 @@
+// Copyright 2014 The Gogs Authors. All rights reserved.
+// Use of this source code is governed by a MIT-style
+// license that can be found in the LICENSE file.
+
+package user
+
+import (
+ "bytes"
+ "fmt"
+
+ "github.com/unknwon/com"
+ "github.com/unknwon/paginater"
+
+ "gogs.io/gogs/internal/context"
+ "gogs.io/gogs/internal/db"
+ "gogs.io/gogs/internal/db/errors"
+ "gogs.io/gogs/internal/setting"
+)
+
+const (
+ DASHBOARD = "user/dashboard/dashboard"
+ NEWS_FEED = "user/dashboard/feeds"
+ ISSUES = "user/dashboard/issues"
+ PROFILE = "user/profile"
+ ORG_HOME = "org/home"
+)
+
+// getDashboardContextUser finds out dashboard is viewing as which context user.
+func getDashboardContextUser(c *context.Context) *db.User {
+ ctxUser := c.User
+ orgName := c.Params(":org")
+ if len(orgName) > 0 {
+ // Organization.
+ org, err := db.GetUserByName(orgName)
+ if err != nil {
+ c.NotFoundOrServerError("GetUserByName", errors.IsUserNotExist, err)
+ return nil
+ }
+ ctxUser = org
+ }
+ c.Data["ContextUser"] = ctxUser
+
+ if err := c.User.GetOrganizations(true); err != nil {
+ c.Handle(500, "GetOrganizations", err)
+ return nil
+ }
+ c.Data["Orgs"] = c.User.Orgs
+
+ return ctxUser
+}
+
+// retrieveFeeds loads feeds from database by given context user.
+// The user could be organization so it is not always the logged in user,
+// which is why we have to explicitly pass the context user ID.
+func retrieveFeeds(c *context.Context, ctxUser *db.User, userID int64, isProfile bool) {
+ actions, err := db.GetFeeds(ctxUser, userID, c.QueryInt64("after_id"), isProfile)
+ if err != nil {
+ c.Handle(500, "GetFeeds", err)
+ return
+ }
+
+ // Check access of private repositories.
+ feeds := make([]*db.Action, 0, len(actions))
+ unameAvatars := make(map[string]string)
+ for _, act := range actions {
+ // Cache results to reduce queries.
+ _, ok := unameAvatars[act.ActUserName]
+ if !ok {
+ u, err := db.GetUserByName(act.ActUserName)
+ if err != nil {
+ if errors.IsUserNotExist(err) {
+ continue
+ }
+ c.Handle(500, "GetUserByName", err)
+ return
+ }
+ unameAvatars[act.ActUserName] = u.RelAvatarLink()
+ }
+
+ act.ActAvatar = unameAvatars[act.ActUserName]
+ feeds = append(feeds, act)
+ }
+ c.Data["Feeds"] = feeds
+ if len(feeds) > 0 {
+ afterID := feeds[len(feeds)-1].ID
+ c.Data["AfterID"] = afterID
+ c.Header().Set("X-AJAX-URL", fmt.Sprintf("%s?after_id=%d", c.Data["Link"], afterID))
+ }
+}
+
+func Dashboard(c *context.Context) {
+ ctxUser := getDashboardContextUser(c)
+ if c.Written() {
+ return
+ }
+
+ retrieveFeeds(c, ctxUser, c.User.ID, false)
+ if c.Written() {
+ return
+ }
+
+ if c.Req.Header.Get("X-AJAX") == "true" {
+ c.HTML(200, NEWS_FEED)
+ return
+ }
+
+ c.Data["Title"] = ctxUser.DisplayName() + " - " + c.Tr("dashboard")
+ c.Data["PageIsDashboard"] = true
+ c.Data["PageIsNews"] = true
+
+ // Only user can have collaborative repositories.
+ if !ctxUser.IsOrganization() {
+ collaborateRepos, err := c.User.GetAccessibleRepositories(setting.UI.User.RepoPagingNum)
+ if err != nil {
+ c.Handle(500, "GetAccessibleRepositories", err)
+ return
+ } else if err = db.RepositoryList(collaborateRepos).LoadAttributes(); err != nil {
+ c.Handle(500, "RepositoryList.LoadAttributes", err)
+ return
+ }
+ c.Data["CollaborativeRepos"] = collaborateRepos
+ }
+
+ var err error
+ var repos, mirrors []*db.Repository
+ var repoCount int64
+ if ctxUser.IsOrganization() {
+ repos, repoCount, err = ctxUser.GetUserRepositories(c.User.ID, 1, setting.UI.User.RepoPagingNum)
+ if err != nil {
+ c.Handle(500, "GetUserRepositories", err)
+ return
+ }
+
+ mirrors, err = ctxUser.GetUserMirrorRepositories(c.User.ID)
+ if err != nil {
+ c.Handle(500, "GetUserMirrorRepositories", err)
+ return
+ }
+ } else {
+ if err = ctxUser.GetRepositories(1, setting.UI.User.RepoPagingNum); err != nil {
+ c.Handle(500, "GetRepositories", err)
+ return
+ }
+ repos = ctxUser.Repos
+ repoCount = int64(ctxUser.NumRepos)
+
+ mirrors, err = ctxUser.GetMirrorRepositories()
+ if err != nil {
+ c.Handle(500, "GetMirrorRepositories", err)
+ return
+ }
+ }
+ c.Data["Repos"] = repos
+ c.Data["RepoCount"] = repoCount
+ c.Data["MaxShowRepoNum"] = setting.UI.User.RepoPagingNum
+
+ if err := db.MirrorRepositoryList(mirrors).LoadAttributes(); err != nil {
+ c.Handle(500, "MirrorRepositoryList.LoadAttributes", err)
+ return
+ }
+ c.Data["MirrorCount"] = len(mirrors)
+ c.Data["Mirrors"] = mirrors
+
+ c.HTML(200, DASHBOARD)
+}
+
+func Issues(c *context.Context) {
+ isPullList := c.Params(":type") == "pulls"
+ if isPullList {
+ c.Data["Title"] = c.Tr("pull_requests")
+ c.Data["PageIsPulls"] = true
+ } else {
+ c.Data["Title"] = c.Tr("issues")
+ c.Data["PageIsIssues"] = true
+ }
+
+ ctxUser := getDashboardContextUser(c)
+ if c.Written() {
+ return
+ }
+
+ var (
+ sortType = c.Query("sort")
+ filterMode = db.FILTER_MODE_YOUR_REPOS
+ )
+
+ // Note: Organization does not have view type and filter mode.
+ if !ctxUser.IsOrganization() {
+ viewType := c.Query("type")
+ types := []string{
+ string(db.FILTER_MODE_YOUR_REPOS),
+ string(db.FILTER_MODE_ASSIGN),
+ string(db.FILTER_MODE_CREATE),
+ }
+ if !com.IsSliceContainsStr(types, viewType) {
+ viewType = string(db.FILTER_MODE_YOUR_REPOS)
+ }
+ filterMode = db.FilterMode(viewType)
+ }
+
+ page := c.QueryInt("page")
+ if page <= 1 {
+ page = 1
+ }
+
+ repoID := c.QueryInt64("repo")
+ isShowClosed := c.Query("state") == "closed"
+
+ // Get repositories.
+ var (
+ err error
+ repos []*db.Repository
+ userRepoIDs []int64
+ showRepos = make([]*db.Repository, 0, 10)
+ )
+ if ctxUser.IsOrganization() {
+ repos, _, err = ctxUser.GetUserRepositories(c.User.ID, 1, ctxUser.NumRepos)
+ if err != nil {
+ c.Handle(500, "GetRepositories", err)
+ return
+ }
+ } else {
+ if err := ctxUser.GetRepositories(1, c.User.NumRepos); err != nil {
+ c.Handle(500, "GetRepositories", err)
+ return
+ }
+ repos = ctxUser.Repos
+ }
+
+ userRepoIDs = make([]int64, 0, len(repos))
+ for _, repo := range repos {
+ userRepoIDs = append(userRepoIDs, repo.ID)
+
+ if filterMode != db.FILTER_MODE_YOUR_REPOS {
+ continue
+ }
+
+ if isPullList {
+ if isShowClosed && repo.NumClosedPulls == 0 ||
+ !isShowClosed && repo.NumOpenPulls == 0 {
+ continue
+ }
+ } else {
+ if !repo.EnableIssues || repo.EnableExternalTracker ||
+ isShowClosed && repo.NumClosedIssues == 0 ||
+ !isShowClosed && repo.NumOpenIssues == 0 {
+ continue
+ }
+ }
+
+ showRepos = append(showRepos, repo)
+ }
+
+ // Filter repositories if the page shows issues.
+ if !isPullList {
+ userRepoIDs, err = db.FilterRepositoryWithIssues(userRepoIDs)
+ if err != nil {
+ c.Handle(500, "FilterRepositoryWithIssues", err)
+ return
+ }
+ }
+
+ issueOptions := &db.IssuesOptions{
+ RepoID: repoID,
+ Page: page,
+ IsClosed: isShowClosed,
+ IsPull: isPullList,
+ SortType: sortType,
+ }
+ switch filterMode {
+ case db.FILTER_MODE_YOUR_REPOS:
+ // Get all issues from repositories from this user.
+ if userRepoIDs == nil {
+ issueOptions.RepoIDs = []int64{-1}
+ } else {
+ issueOptions.RepoIDs = userRepoIDs
+ }
+
+ case db.FILTER_MODE_ASSIGN:
+ // Get all issues assigned to this user.
+ issueOptions.AssigneeID = ctxUser.ID
+
+ case db.FILTER_MODE_CREATE:
+ // Get all issues created by this user.
+ issueOptions.PosterID = ctxUser.ID
+ }
+
+ issues, err := db.Issues(issueOptions)
+ if err != nil {
+ c.Handle(500, "Issues", err)
+ return
+ }
+
+ if repoID > 0 {
+ repo, err := db.GetRepositoryByID(repoID)
+ if err != nil {
+ c.Handle(500, "GetRepositoryByID", fmt.Errorf("[#%d] %v", repoID, err))
+ return
+ }
+
+ if err = repo.GetOwner(); err != nil {
+ c.Handle(500, "GetOwner", fmt.Errorf("[#%d] %v", repoID, err))
+ return
+ }
+
+ // Check if user has access to given repository.
+ if !repo.IsOwnedBy(ctxUser.ID) && !repo.HasAccess(ctxUser.ID) {
+ c.Handle(404, "Issues", fmt.Errorf("#%d", repoID))
+ return
+ }
+ }
+
+ for _, issue := range issues {
+ if err = issue.Repo.GetOwner(); err != nil {
+ c.Handle(500, "GetOwner", fmt.Errorf("[#%d] %v", issue.RepoID, err))
+ return
+ }
+ }
+
+ issueStats := db.GetUserIssueStats(repoID, ctxUser.ID, userRepoIDs, filterMode, isPullList)
+
+ var total int
+ if !isShowClosed {
+ total = int(issueStats.OpenCount)
+ } else {
+ total = int(issueStats.ClosedCount)
+ }
+
+ c.Data["Issues"] = issues
+ c.Data["Repos"] = showRepos
+ c.Data["Page"] = paginater.New(total, setting.UI.IssuePagingNum, page, 5)
+ c.Data["IssueStats"] = issueStats
+ c.Data["ViewType"] = string(filterMode)
+ c.Data["SortType"] = sortType
+ c.Data["RepoID"] = repoID
+ c.Data["IsShowClosed"] = isShowClosed
+
+ if isShowClosed {
+ c.Data["State"] = "closed"
+ } else {
+ c.Data["State"] = "open"
+ }
+
+ c.HTML(200, ISSUES)
+}
+
+func ShowSSHKeys(c *context.Context, uid int64) {
+ keys, err := db.ListPublicKeys(uid)
+ if err != nil {
+ c.Handle(500, "ListPublicKeys", err)
+ return
+ }
+
+ var buf bytes.Buffer
+ for i := range keys {
+ buf.WriteString(keys[i].OmitEmail())
+ buf.WriteString("\n")
+ }
+ c.PlainText(200, buf.Bytes())
+}
+
+func showOrgProfile(c *context.Context) {
+ c.SetParams(":org", c.Params(":username"))
+ context.HandleOrgAssignment(c)
+ if c.Written() {
+ return
+ }
+
+ org := c.Org.Organization
+ c.Data["Title"] = org.FullName
+
+ page := c.QueryInt("page")
+ if page <= 0 {
+ page = 1
+ }
+
+ var (
+ repos []*db.Repository
+ count int64
+ err error
+ )
+ if c.IsLogged && !c.User.IsAdmin {
+ repos, count, err = org.GetUserRepositories(c.User.ID, page, setting.UI.User.RepoPagingNum)
+ if err != nil {
+ c.Handle(500, "GetUserRepositories", err)
+ return
+ }
+ c.Data["Repos"] = repos
+ } else {
+ showPrivate := c.IsLogged && c.User.IsAdmin
+ repos, err = db.GetUserRepositories(&db.UserRepoOptions{
+ UserID: org.ID,
+ Private: showPrivate,
+ Page: page,
+ PageSize: setting.UI.User.RepoPagingNum,
+ })
+ if err != nil {
+ c.Handle(500, "GetRepositories", err)
+ return
+ }
+ c.Data["Repos"] = repos
+ count = db.CountUserRepositories(org.ID, showPrivate)
+ }
+ c.Data["Page"] = paginater.New(int(count), setting.UI.User.RepoPagingNum, page, 5)
+
+ if err := org.GetMembers(); err != nil {
+ c.Handle(500, "GetMembers", err)
+ return
+ }
+ c.Data["Members"] = org.Members
+
+ c.Data["Teams"] = org.Teams
+
+ c.HTML(200, ORG_HOME)
+}
+
+func Email2User(c *context.Context) {
+ u, err := db.GetUserByEmail(c.Query("email"))
+ if err != nil {
+ c.NotFoundOrServerError("GetUserByEmail", errors.IsUserNotExist, err)
+ return
+ }
+ c.Redirect(setting.AppSubURL + "/user/" + u.Name)
+}