2015-12-21 20:24:11 +08:00
|
|
|
// Copyright 2015 The Gogs Authors. All rights reserved.
|
2019-02-19 00:00:27 +08:00
|
|
|
// Copyright 2019 The Gitea Authors. All rights reserved.
|
2015-12-21 20:24:11 +08:00
|
|
|
// Use of this source code is governed by a MIT-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package user
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2021-04-05 23:30:52 +08:00
|
|
|
"net/http"
|
2015-12-25 18:25:47 +08:00
|
|
|
"path"
|
2015-12-21 20:24:11 +08:00
|
|
|
"strings"
|
|
|
|
|
2016-11-11 00:24:48 +08:00
|
|
|
"code.gitea.io/gitea/models"
|
2021-09-24 19:32:56 +08:00
|
|
|
"code.gitea.io/gitea/models/db"
|
2021-11-11 15:03:30 +08:00
|
|
|
user_model "code.gitea.io/gitea/models/user"
|
2016-11-11 00:24:48 +08:00
|
|
|
"code.gitea.io/gitea/modules/context"
|
2021-04-20 06:25:08 +08:00
|
|
|
"code.gitea.io/gitea/modules/markup"
|
2020-08-05 15:48:37 +08:00
|
|
|
"code.gitea.io/gitea/modules/markup/markdown"
|
2016-11-11 00:24:48 +08:00
|
|
|
"code.gitea.io/gitea/modules/setting"
|
2017-10-27 05:16:13 +08:00
|
|
|
"code.gitea.io/gitea/modules/util"
|
2021-10-16 22:21:16 +08:00
|
|
|
"code.gitea.io/gitea/routers/web/feed"
|
2021-06-09 07:33:54 +08:00
|
|
|
"code.gitea.io/gitea/routers/web/org"
|
2015-12-21 20:24:11 +08:00
|
|
|
)
|
|
|
|
|
2016-11-18 11:03:03 +08:00
|
|
|
// GetUserByName get user by name
|
2016-03-12 00:56:52 +08:00
|
|
|
func GetUserByName(ctx *context.Context, name string) *models.User {
|
2016-01-09 13:28:05 +08:00
|
|
|
user, err := models.GetUserByName(name)
|
2015-12-21 20:24:11 +08:00
|
|
|
if err != nil {
|
|
|
|
if models.IsErrUserNotExist(err) {
|
2021-11-11 15:03:30 +08:00
|
|
|
if redirectUserID, err := user_model.LookupUserRedirect(name); err == nil {
|
2021-01-24 23:23:05 +08:00
|
|
|
context.RedirectToUser(ctx, name, redirectUserID)
|
|
|
|
} else {
|
|
|
|
ctx.NotFound("GetUserByName", err)
|
|
|
|
}
|
2015-12-21 20:24:11 +08:00
|
|
|
} else {
|
2018-01-11 05:34:17 +08:00
|
|
|
ctx.ServerError("GetUserByName", err)
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return user
|
|
|
|
}
|
|
|
|
|
2016-01-09 13:28:05 +08:00
|
|
|
// GetUserByParams returns user whose name is presented in URL paramenter.
|
2016-03-12 00:56:52 +08:00
|
|
|
func GetUserByParams(ctx *context.Context) *models.User {
|
2016-01-09 13:28:05 +08:00
|
|
|
return GetUserByName(ctx, ctx.Params(":username"))
|
|
|
|
}
|
|
|
|
|
2016-11-18 11:03:03 +08:00
|
|
|
// Profile render user's profile page
|
2016-03-12 00:56:52 +08:00
|
|
|
func Profile(ctx *context.Context) {
|
2015-12-21 20:24:11 +08:00
|
|
|
uname := ctx.Params(":username")
|
2021-04-28 20:35:06 +08:00
|
|
|
|
2015-12-21 20:24:11 +08:00
|
|
|
// Special handle for FireFox requests favicon.ico.
|
|
|
|
if uname == "favicon.ico" {
|
2015-12-25 18:25:47 +08:00
|
|
|
ctx.ServeFile(path.Join(setting.StaticRootPath, "public/img/favicon.png"))
|
2015-12-21 20:24:11 +08:00
|
|
|
return
|
2021-04-28 20:35:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if strings.HasSuffix(uname, ".png") {
|
2021-04-05 23:30:52 +08:00
|
|
|
ctx.Error(http.StatusNotFound)
|
2015-12-21 20:24:11 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
isShowKeys := false
|
|
|
|
if strings.HasSuffix(uname, ".keys") {
|
|
|
|
isShowKeys = true
|
2019-04-15 00:43:56 +08:00
|
|
|
uname = strings.TrimSuffix(uname, ".keys")
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|
|
|
|
|
2019-04-15 00:43:56 +08:00
|
|
|
isShowGPG := false
|
|
|
|
if strings.HasSuffix(uname, ".gpg") {
|
|
|
|
isShowGPG = true
|
|
|
|
uname = strings.TrimSuffix(uname, ".gpg")
|
|
|
|
}
|
|
|
|
|
2021-10-16 22:21:16 +08:00
|
|
|
showFeedType := ""
|
|
|
|
if strings.HasSuffix(uname, ".rss") {
|
|
|
|
showFeedType = "rss"
|
|
|
|
uname = strings.TrimSuffix(uname, ".rss")
|
|
|
|
} else if strings.Contains(ctx.Req.Header.Get("Accept"), "application/rss+xml") {
|
|
|
|
showFeedType = "rss"
|
|
|
|
}
|
|
|
|
if strings.HasSuffix(uname, ".atom") {
|
|
|
|
showFeedType = "atom"
|
|
|
|
uname = strings.TrimSuffix(uname, ".atom")
|
|
|
|
} else if strings.Contains(ctx.Req.Header.Get("Accept"), "application/atom+xml") {
|
|
|
|
showFeedType = "atom"
|
|
|
|
}
|
|
|
|
|
2019-04-15 00:43:56 +08:00
|
|
|
ctxUser := GetUserByName(ctx, uname)
|
2015-12-21 20:24:11 +08:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-06-27 03:53:14 +08:00
|
|
|
if ctxUser.IsOrganization() {
|
2021-10-16 22:21:16 +08:00
|
|
|
/*
|
|
|
|
// TODO: enable after rss.RetrieveFeeds() do handle org correctly
|
|
|
|
// Show Org RSS feed
|
|
|
|
if len(showFeedType) != 0 {
|
|
|
|
rss.ShowUserFeed(ctx, ctxUser, showFeedType)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
2021-06-27 03:53:14 +08:00
|
|
|
org.Home(ctx)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// check view permissions
|
|
|
|
if !ctxUser.IsVisibleToUser(ctx.User) {
|
|
|
|
ctx.NotFound("user", fmt.Errorf(uname))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-12-21 20:24:11 +08:00
|
|
|
// Show SSH keys.
|
|
|
|
if isShowKeys {
|
2016-07-24 14:32:46 +08:00
|
|
|
ShowSSHKeys(ctx, ctxUser.ID)
|
2015-12-21 20:24:11 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-04-15 00:43:56 +08:00
|
|
|
// Show GPG keys.
|
|
|
|
if isShowGPG {
|
|
|
|
ShowGPGKeys(ctx, ctxUser.ID)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-10-16 22:21:16 +08:00
|
|
|
// Show User RSS feed
|
|
|
|
if len(showFeedType) != 0 {
|
|
|
|
feed.ShowUserFeed(ctx, ctxUser, showFeedType)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-03-20 16:31:08 +08:00
|
|
|
// Show OpenID URIs
|
2021-11-17 17:58:31 +08:00
|
|
|
openIDs, err := user_model.GetUserOpenIDs(ctxUser.ID)
|
2017-03-20 16:31:08 +08:00
|
|
|
if err != nil {
|
2018-01-11 05:34:17 +08:00
|
|
|
ctx.ServerError("GetUserOpenIDs", err)
|
2017-03-20 16:31:08 +08:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-07-24 14:32:46 +08:00
|
|
|
ctx.Data["Title"] = ctxUser.DisplayName()
|
2015-12-21 20:24:11 +08:00
|
|
|
ctx.Data["PageIsUserProfile"] = true
|
2016-07-24 14:32:46 +08:00
|
|
|
ctx.Data["Owner"] = ctxUser
|
2017-03-20 16:31:08 +08:00
|
|
|
ctx.Data["OpenIDs"] = openIDs
|
2020-11-19 06:00:16 +08:00
|
|
|
|
2021-03-05 06:59:13 +08:00
|
|
|
if setting.Service.EnableUserHeatmap {
|
2020-12-22 10:53:37 +08:00
|
|
|
data, err := models.GetUserHeatmapDataByUser(ctxUser, ctx.User)
|
2020-11-19 06:00:16 +08:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetUserHeatmapDataByUser", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Data["HeatmapData"] = data
|
|
|
|
}
|
|
|
|
|
2020-08-05 15:48:37 +08:00
|
|
|
if len(ctxUser.Description) != 0 {
|
2021-04-20 06:25:08 +08:00
|
|
|
content, err := markdown.RenderString(&markup.RenderContext{
|
|
|
|
URLPrefix: ctx.Repo.RepoLink,
|
|
|
|
Metas: map[string]string{"mode": "document"},
|
2021-06-21 06:39:12 +08:00
|
|
|
GitRepo: ctx.Repo.GitRepo,
|
2021-08-29 04:15:56 +08:00
|
|
|
Ctx: ctx,
|
2021-04-20 06:25:08 +08:00
|
|
|
}, ctxUser.Description)
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("RenderString", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Data["RenderedDescription"] = content
|
2020-08-05 15:48:37 +08:00
|
|
|
}
|
|
|
|
|
2017-02-02 20:32:40 +08:00
|
|
|
showPrivate := ctx.IsSigned && (ctx.User.IsAdmin || ctx.User.ID == ctxUser.ID)
|
2016-01-31 05:51:11 +08:00
|
|
|
|
2021-11-22 21:51:45 +08:00
|
|
|
orgs, err := models.FindOrgs(models.FindOrgOptions{
|
|
|
|
UserID: ctxUser.ID,
|
|
|
|
IncludePrivate: showPrivate,
|
|
|
|
})
|
2016-01-12 10:09:59 +08:00
|
|
|
if err != nil {
|
2021-11-22 21:51:45 +08:00
|
|
|
ctx.ServerError("FindOrgs", err)
|
2016-01-12 10:09:59 +08:00
|
|
|
return
|
|
|
|
}
|
2016-02-07 17:20:58 +08:00
|
|
|
|
2016-01-12 10:09:59 +08:00
|
|
|
ctx.Data["Orgs"] = orgs
|
2019-02-19 00:00:27 +08:00
|
|
|
ctx.Data["HasOrgsVisible"] = models.HasOrgsVisible(orgs, ctx.User)
|
2015-12-21 20:24:11 +08:00
|
|
|
|
2021-08-11 08:31:13 +08:00
|
|
|
tab := ctx.FormString("tab")
|
2015-12-21 20:24:11 +08:00
|
|
|
ctx.Data["TabName"] = tab
|
2017-02-14 15:28:22 +08:00
|
|
|
|
2021-07-29 09:42:15 +08:00
|
|
|
page := ctx.FormInt("page")
|
2017-02-14 15:28:22 +08:00
|
|
|
if page <= 0 {
|
|
|
|
page = 1
|
|
|
|
}
|
|
|
|
|
2021-07-29 09:42:15 +08:00
|
|
|
topicOnly := ctx.FormBool("topic")
|
2018-09-13 10:33:48 +08:00
|
|
|
|
2017-02-14 15:28:22 +08:00
|
|
|
var (
|
|
|
|
repos []*models.Repository
|
|
|
|
count int64
|
2019-04-20 12:15:19 +08:00
|
|
|
total int
|
2017-09-22 20:53:21 +08:00
|
|
|
orderBy models.SearchOrderBy
|
2017-02-14 15:28:22 +08:00
|
|
|
)
|
|
|
|
|
2021-08-11 08:31:13 +08:00
|
|
|
ctx.Data["SortType"] = ctx.FormString("sort")
|
|
|
|
switch ctx.FormString("sort") {
|
2017-02-14 15:28:22 +08:00
|
|
|
case "newest":
|
2017-09-22 20:53:21 +08:00
|
|
|
orderBy = models.SearchOrderByNewest
|
2017-02-14 15:28:22 +08:00
|
|
|
case "oldest":
|
2017-09-22 20:53:21 +08:00
|
|
|
orderBy = models.SearchOrderByOldest
|
2017-02-14 15:28:22 +08:00
|
|
|
case "recentupdate":
|
2017-09-22 20:53:21 +08:00
|
|
|
orderBy = models.SearchOrderByRecentUpdated
|
2017-02-14 15:28:22 +08:00
|
|
|
case "leastupdate":
|
2017-09-22 20:53:21 +08:00
|
|
|
orderBy = models.SearchOrderByLeastUpdated
|
2017-02-14 15:28:22 +08:00
|
|
|
case "reversealphabetically":
|
2017-09-22 20:53:21 +08:00
|
|
|
orderBy = models.SearchOrderByAlphabeticallyReverse
|
2017-02-14 15:28:22 +08:00
|
|
|
case "alphabetically":
|
2017-09-22 20:53:21 +08:00
|
|
|
orderBy = models.SearchOrderByAlphabetically
|
2018-05-24 09:03:42 +08:00
|
|
|
case "moststars":
|
|
|
|
orderBy = models.SearchOrderByStarsReverse
|
|
|
|
case "feweststars":
|
|
|
|
orderBy = models.SearchOrderByStars
|
|
|
|
case "mostforks":
|
|
|
|
orderBy = models.SearchOrderByForksReverse
|
|
|
|
case "fewestforks":
|
|
|
|
orderBy = models.SearchOrderByForks
|
2017-02-14 15:28:22 +08:00
|
|
|
default:
|
|
|
|
ctx.Data["SortType"] = "recentupdate"
|
2017-10-05 13:02:43 +08:00
|
|
|
orderBy = models.SearchOrderByRecentUpdated
|
2017-02-14 15:28:22 +08:00
|
|
|
}
|
|
|
|
|
2021-08-11 23:08:52 +08:00
|
|
|
keyword := ctx.FormTrim("q")
|
2017-02-14 15:28:22 +08:00
|
|
|
ctx.Data["Keyword"] = keyword
|
2015-12-21 20:24:11 +08:00
|
|
|
switch tab {
|
2020-02-10 04:18:01 +08:00
|
|
|
case "followers":
|
2021-09-24 19:32:56 +08:00
|
|
|
items, err := ctxUser.GetFollowers(db.ListOptions{
|
2020-02-10 04:18:01 +08:00
|
|
|
PageSize: setting.UI.User.RepoPagingNum,
|
|
|
|
Page: page,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetFollowers", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Data["Cards"] = items
|
|
|
|
|
|
|
|
total = ctxUser.NumFollowers
|
|
|
|
case "following":
|
2021-09-24 19:32:56 +08:00
|
|
|
items, err := ctxUser.GetFollowing(db.ListOptions{
|
2020-02-10 04:18:01 +08:00
|
|
|
PageSize: setting.UI.User.RepoPagingNum,
|
|
|
|
Page: page,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetFollowing", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Data["Cards"] = items
|
|
|
|
|
|
|
|
total = ctxUser.NumFollowing
|
2015-12-21 20:24:11 +08:00
|
|
|
case "activity":
|
2021-10-16 22:21:16 +08:00
|
|
|
ctx.Data["Feeds"] = feed.RetrieveFeeds(ctx, models.GetFeedsOptions{RequestedUser: ctxUser,
|
2020-01-14 01:33:46 +08:00
|
|
|
Actor: ctx.User,
|
2017-08-23 09:30:54 +08:00
|
|
|
IncludePrivate: showPrivate,
|
|
|
|
OnlyPerformedBy: true,
|
|
|
|
IncludeDeleted: false,
|
2021-08-11 08:31:13 +08:00
|
|
|
Date: ctx.FormString("date"),
|
2017-08-23 09:30:54 +08:00
|
|
|
})
|
2015-12-21 20:24:11 +08:00
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
2016-12-29 22:58:24 +08:00
|
|
|
case "stars":
|
2017-02-14 15:28:22 +08:00
|
|
|
ctx.Data["PageIsProfileStarList"] = true
|
2019-08-26 01:06:36 +08:00
|
|
|
repos, count, err = models.SearchRepository(&models.SearchRepoOptions{
|
2021-09-24 19:32:56 +08:00
|
|
|
ListOptions: db.ListOptions{
|
2020-01-25 03:00:29 +08:00
|
|
|
PageSize: setting.UI.User.RepoPagingNum,
|
|
|
|
Page: page,
|
|
|
|
},
|
2020-01-14 01:33:46 +08:00
|
|
|
Actor: ctx.User,
|
2019-08-26 01:06:36 +08:00
|
|
|
Keyword: keyword,
|
|
|
|
OrderBy: orderBy,
|
|
|
|
Private: ctx.IsSigned,
|
|
|
|
StarredByID: ctxUser.ID,
|
|
|
|
Collaborate: util.OptionalBoolFalse,
|
|
|
|
TopicOnly: topicOnly,
|
|
|
|
IncludeDescription: setting.UI.SearchRepoDescription,
|
2019-05-15 23:24:39 +08:00
|
|
|
})
|
|
|
|
if err != nil {
|
2019-08-26 01:06:36 +08:00
|
|
|
ctx.ServerError("SearchRepository", err)
|
2019-05-15 23:24:39 +08:00
|
|
|
return
|
2017-02-07 19:54:16 +08:00
|
|
|
}
|
|
|
|
|
2019-04-20 12:15:19 +08:00
|
|
|
total = int(count)
|
2020-08-17 11:07:38 +08:00
|
|
|
case "projects":
|
|
|
|
ctx.Data["OpenProjects"], _, err = models.GetProjects(models.ProjectSearchOptions{
|
|
|
|
Page: -1,
|
|
|
|
IsClosed: util.OptionalBoolFalse,
|
|
|
|
Type: models.ProjectTypeIndividual,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetProjects", err)
|
|
|
|
return
|
|
|
|
}
|
2021-04-16 00:53:57 +08:00
|
|
|
case "watching":
|
|
|
|
repos, count, err = models.SearchRepository(&models.SearchRepoOptions{
|
2021-09-24 19:32:56 +08:00
|
|
|
ListOptions: db.ListOptions{
|
2021-04-16 00:53:57 +08:00
|
|
|
PageSize: setting.UI.User.RepoPagingNum,
|
|
|
|
Page: page,
|
|
|
|
},
|
|
|
|
Actor: ctx.User,
|
|
|
|
Keyword: keyword,
|
|
|
|
OrderBy: orderBy,
|
|
|
|
Private: ctx.IsSigned,
|
|
|
|
WatchedByID: ctxUser.ID,
|
|
|
|
Collaborate: util.OptionalBoolFalse,
|
|
|
|
TopicOnly: topicOnly,
|
|
|
|
IncludeDescription: setting.UI.SearchRepoDescription,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("SearchRepository", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
total = int(count)
|
2015-12-21 20:24:11 +08:00
|
|
|
default:
|
2019-08-26 01:06:36 +08:00
|
|
|
repos, count, err = models.SearchRepository(&models.SearchRepoOptions{
|
2021-09-24 19:32:56 +08:00
|
|
|
ListOptions: db.ListOptions{
|
2020-01-25 03:00:29 +08:00
|
|
|
PageSize: setting.UI.User.RepoPagingNum,
|
|
|
|
Page: page,
|
|
|
|
},
|
2020-01-14 01:33:46 +08:00
|
|
|
Actor: ctx.User,
|
2019-08-26 01:06:36 +08:00
|
|
|
Keyword: keyword,
|
|
|
|
OwnerID: ctxUser.ID,
|
|
|
|
OrderBy: orderBy,
|
|
|
|
Private: ctx.IsSigned,
|
|
|
|
Collaborate: util.OptionalBoolFalse,
|
|
|
|
TopicOnly: topicOnly,
|
|
|
|
IncludeDescription: setting.UI.SearchRepoDescription,
|
2019-05-15 23:24:39 +08:00
|
|
|
})
|
|
|
|
if err != nil {
|
2019-08-26 01:06:36 +08:00
|
|
|
ctx.ServerError("SearchRepository", err)
|
2019-05-15 23:24:39 +08:00
|
|
|
return
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|
2019-05-15 23:24:39 +08:00
|
|
|
|
|
|
|
total = int(count)
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|
2019-04-20 12:15:19 +08:00
|
|
|
ctx.Data["Repos"] = repos
|
|
|
|
ctx.Data["Total"] = total
|
|
|
|
|
|
|
|
pager := context.NewPagination(total, setting.UI.User.RepoPagingNum, page, 5)
|
|
|
|
pager.SetDefaultParams(ctx)
|
|
|
|
ctx.Data["Page"] = pager
|
2015-12-21 20:24:11 +08:00
|
|
|
|
2019-02-19 22:11:50 +08:00
|
|
|
ctx.Data["ShowUserEmail"] = len(ctxUser.Email) > 0 && ctx.IsSigned && (!ctxUser.KeepEmailPrivate || ctxUser.ID == ctx.User.ID)
|
2017-08-17 17:08:03 +08:00
|
|
|
|
2021-04-05 23:30:52 +08:00
|
|
|
ctx.HTML(http.StatusOK, tplProfile)
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|
|
|
|
|
2016-11-18 11:03:03 +08:00
|
|
|
// Action response for follow/unfollow user request
|
2016-03-12 00:56:52 +08:00
|
|
|
func Action(ctx *context.Context) {
|
2015-12-21 20:24:11 +08:00
|
|
|
u := GetUserByParams(ctx)
|
|
|
|
if ctx.Written() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var err error
|
|
|
|
switch ctx.Params(":action") {
|
|
|
|
case "follow":
|
2021-11-17 17:58:31 +08:00
|
|
|
err = user_model.FollowUser(ctx.User.ID, u.ID)
|
2015-12-21 20:24:11 +08:00
|
|
|
case "unfollow":
|
2021-11-17 17:58:31 +08:00
|
|
|
err = user_model.UnfollowUser(ctx.User.ID, u.ID)
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2018-01-11 05:34:17 +08:00
|
|
|
ctx.ServerError(fmt.Sprintf("Action (%s)", ctx.Params(":action")), err)
|
2015-12-21 20:24:11 +08:00
|
|
|
return
|
|
|
|
}
|
2021-11-17 02:18:25 +08:00
|
|
|
// FIXME: We should check this URL and make sure that it's a valid Gitea URL
|
2021-08-11 08:31:13 +08:00
|
|
|
ctx.RedirectToFirst(ctx.FormString("redirect_to"), u.HomeLink())
|
2015-12-21 20:24:11 +08:00
|
|
|
}
|