123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425 |
- // 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"
- "net/http"
- "github.com/unknwon/com"
- "github.com/unknwon/paginater"
- "gogs.io/gogs/internal/conf"
- "gogs.io/gogs/internal/context"
- "gogs.io/gogs/internal/db"
- )
- 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.NotFoundOrError(err, "get user by name")
- return nil
- }
- ctxUser = org
- }
- c.Data["ContextUser"] = ctxUser
- if err := c.User.GetOrganizations(true); err != nil {
- c.Error(err, "get organizations")
- 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.Error(err, "get feeds")
- 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 db.IsErrUserNotExist(err) {
- continue
- }
- c.Error(err, "get user by name")
- 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.Success(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(conf.UI.User.RepoPagingNum)
- if err != nil {
- c.Error(err, "get accessible repositories")
- return
- } else if err = db.RepositoryList(collaborateRepos).LoadAttributes(); err != nil {
- c.Error(err, "load attributes")
- 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, conf.UI.User.RepoPagingNum)
- if err != nil {
- c.Error(err, "get user repositories")
- return
- }
- mirrors, err = ctxUser.GetUserMirrorRepositories(c.User.ID)
- if err != nil {
- c.Error(err, "get user mirror repositories")
- return
- }
- } else {
- if err = ctxUser.GetRepositories(1, conf.UI.User.RepoPagingNum); err != nil {
- c.Error(err, "get repositories")
- return
- }
- repos = ctxUser.Repos
- repoCount = int64(ctxUser.NumRepos)
- mirrors, err = ctxUser.GetMirrorRepositories()
- if err != nil {
- c.Error(err, "get mirror repositories")
- return
- }
- }
- c.Data["Repos"] = repos
- c.Data["RepoCount"] = repoCount
- c.Data["MaxShowRepoNum"] = conf.UI.User.RepoPagingNum
- if err := db.MirrorRepositoryList(mirrors).LoadAttributes(); err != nil {
- c.Error(err, "load attributes")
- return
- }
- c.Data["MirrorCount"] = len(mirrors)
- c.Data["Mirrors"] = mirrors
- c.Success(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.Error(err, "get repositories")
- return
- }
- } else {
- if err := ctxUser.GetRepositories(1, c.User.NumRepos); err != nil {
- c.Error(err, "get repositories")
- 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.Error(err, "filter repositories with issues")
- 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.Error(err, "list issues")
- return
- }
- if repoID > 0 {
- repo, err := db.GetRepositoryByID(repoID)
- if err != nil {
- c.Error(err, "get repository by ID")
- return
- }
- if err = repo.GetOwner(); err != nil {
- c.Error(err, "get owner")
- return
- }
- // Check if user has access to given repository.
- if !repo.IsOwnedBy(ctxUser.ID) && !repo.HasAccess(ctxUser.ID) {
- c.NotFound()
- return
- }
- }
- for _, issue := range issues {
- if err = issue.Repo.GetOwner(); err != nil {
- c.Error(err, "get owner")
- 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, conf.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.Success(ISSUES)
- }
- func ShowSSHKeys(c *context.Context, uid int64) {
- keys, err := db.ListPublicKeys(uid)
- if err != nil {
- c.Error(err, "list public keys")
- return
- }
- var buf bytes.Buffer
- for i := range keys {
- buf.WriteString(keys[i].OmitEmail())
- buf.WriteString("\n")
- }
- c.PlainText(http.StatusOK, buf.String())
- }
- 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, conf.UI.User.RepoPagingNum)
- if err != nil {
- c.Error(err, "get user repositories")
- 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: conf.UI.User.RepoPagingNum,
- })
- if err != nil {
- c.Error(err, "get user repositories")
- return
- }
- c.Data["Repos"] = repos
- count = db.CountUserRepositories(org.ID, showPrivate)
- }
- c.Data["Page"] = paginater.New(int(count), conf.UI.User.RepoPagingNum, page, 5)
- if err := org.GetMembers(12); err != nil {
- c.Error(err, "get members")
- return
- }
- c.Data["Members"] = org.Members
- c.Data["Teams"] = org.Teams
- c.Success(ORG_HOME)
- }
- func Email2User(c *context.Context) {
- u, err := db.GetUserByEmail(c.Query("email"))
- if err != nil {
- c.NotFoundOrError(err, "get user by email")
- return
- }
- c.Redirect(conf.Server.Subpath + "/user/" + u.Name)
- }
|