gitea/routers/web/org/home.go
Admin 1af82412c0
Some checks failed
Build and Release / Create Release (push) Has been skipped
Build and Release / Lint (push) Failing after 3s
Build and Release / Build Binaries (amd64, darwin) (push) Has been skipped
Build and Release / Build Binaries (amd64, linux) (push) Has been skipped
Build and Release / Build Binaries (amd64, windows) (push) Has been skipped
Build and Release / Build Binaries (arm64, darwin) (push) Has been skipped
Build and Release / Build Binaries (arm64, linux) (push) Has been skipped
Build and Release / Integration Tests (PostgreSQL) (push) Successful in 48s
Build and Release / Unit Tests (push) Successful in 2m4s
feat: auto-create .profile repo with README and redirect to edit
2026-01-11 03:57:14 +00:00

332 lines
9.8 KiB
Go

// Copyright 2019 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
package org
import (
"net/http"
"path"
"strings"
"code.gitea.io/gitea/models/db"
"code.gitea.io/gitea/models/organization"
"code.gitea.io/gitea/models/renderhelper"
repo_model "code.gitea.io/gitea/models/repo"
"code.gitea.io/gitea/modules/git"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/markup/markdown"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/templates"
"code.gitea.io/gitea/modules/timeutil"
"code.gitea.io/gitea/modules/util"
shared_user "code.gitea.io/gitea/routers/web/shared/user"
"code.gitea.io/gitea/services/context"
org_service "code.gitea.io/gitea/services/org"
repo_service "code.gitea.io/gitea/services/repository"
)
// RecentRepoActivity holds repo and its latest commit info
type RecentRepoActivity struct {
Repo *repo_model.Repository
CommitMessage string
CommitTime timeutil.TimeStamp
}
const tplOrgHome templates.TplName = "org/home"
// Home show organization home page
func Home(ctx *context.Context) {
uname := ctx.PathParam("username")
if strings.HasSuffix(uname, ".keys") || strings.HasSuffix(uname, ".gpg") {
ctx.NotFound(nil)
return
}
ctx.SetPathParam("org", uname)
context.OrgAssignment(context.OrgAssignmentOptions{})(ctx)
if ctx.Written() {
return
}
home(ctx, false)
}
func Repositories(ctx *context.Context) {
home(ctx, true)
}
func home(ctx *context.Context, viewRepositories bool) {
org := ctx.Org.Organization
ctx.Data["PageIsUserProfile"] = true
ctx.Data["Title"] = org.DisplayName()
var orderBy db.SearchOrderBy
sortOrder := ctx.FormString("sort")
if _, ok := repo_model.OrderByFlatMap[sortOrder]; !ok {
sortOrder = setting.UI.ExploreDefaultSort // TODO: add new default sort order for org home?
}
ctx.Data["SortType"] = sortOrder
orderBy = repo_model.OrderByFlatMap[sortOrder]
keyword := ctx.FormTrim("q")
ctx.Data["Keyword"] = keyword
language := ctx.FormTrim("language")
ctx.Data["Language"] = language
page := ctx.FormInt("page")
if page <= 0 {
page = 1
}
archived := ctx.FormOptionalBool("archived")
ctx.Data["IsArchived"] = archived
fork := ctx.FormOptionalBool("fork")
ctx.Data["IsFork"] = fork
mirror := ctx.FormOptionalBool("mirror")
ctx.Data["IsMirror"] = mirror
template := ctx.FormOptionalBool("template")
ctx.Data["IsTemplate"] = template
private := ctx.FormOptionalBool("private")
ctx.Data["IsPrivate"] = private
opts := &organization.FindOrgMembersOpts{
Doer: ctx.Doer,
OrgID: org.ID,
IsDoerMember: ctx.Org.IsMember,
ListOptions: db.ListOptions{Page: 1, PageSize: 25},
}
members, _, err := organization.FindOrgMembers(ctx, opts)
if err != nil {
ctx.ServerError("FindOrgMembers", err)
return
}
ctx.Data["Members"] = members
ctx.Data["Teams"] = ctx.Org.Teams
ctx.Data["DisableNewPullMirrors"] = setting.Mirror.DisableNewPull
ctx.Data["ShowMemberAndTeamTab"] = ctx.Org.IsMember || len(members) > 0
ctx.Data["IsOrganizationMember"] = ctx.Org.IsMember
// Load recently updated repositories for activity section
// Only show private repos if user is signed in and is org member
showPrivate := ctx.IsSigned && ctx.Org.IsMember
recentRepos, _, err := repo_model.SearchRepository(ctx, repo_model.SearchRepoOptions{
ListOptions: db.ListOptions{
PageSize: 10,
Page: 1,
},
OwnerID: org.ID,
OrderBy: db.SearchOrderByRecentUpdated,
Private: showPrivate,
Actor: ctx.Doer,
})
if err != nil {
log.Error("SearchRepository for recent repos: %v", err)
} else {
// Load commit info for each repo
var recentActivity []*RecentRepoActivity
for _, repo := range recentRepos {
activity := &RecentRepoActivity{Repo: repo}
gitRepo, err := git.OpenRepository(ctx, repo.RepoPath())
if err == nil {
commit, err := gitRepo.GetBranchCommit(repo.DefaultBranch)
if err == nil {
activity.CommitMessage = commit.Summary()
activity.CommitTime = timeutil.TimeStamp(commit.Author.When.Unix())
}
gitRepo.Close()
}
recentActivity = append(recentActivity, activity)
}
ctx.Data["RecentActivity"] = recentActivity
}
prepareResult, err := shared_user.RenderUserOrgHeader(ctx)
if err != nil {
ctx.ServerError("RenderUserOrgHeader", err)
return
}
// Load pinned repositories with details
pinnedRepos, err := org_service.GetOrgPinnedReposWithDetails(ctx, org.ID)
if err != nil {
log.Error("GetOrgPinnedReposWithDetails: %v", err)
}
ctx.Data["PinnedRepos"] = pinnedRepos
// Load pinned groups
pinnedGroups, err := organization.GetOrgPinnedGroups(ctx, org.ID)
if err != nil {
log.Error("GetOrgPinnedGroups: %v", err)
}
ctx.Data["PinnedGroups"] = pinnedGroups
// Organize pinned repos by group for template
pinnedByGroup := make(map[int64][]*organization.OrgPinnedRepo)
var ungroupedPinned []*organization.OrgPinnedRepo
for _, p := range pinnedRepos {
if p.Repo == nil {
continue
}
if p.GroupID == 0 {
ungroupedPinned = append(ungroupedPinned, p)
} else {
pinnedByGroup[p.GroupID] = append(pinnedByGroup[p.GroupID], p)
}
}
ctx.Data["PinnedByGroup"] = pinnedByGroup
ctx.Data["UngroupedPinned"] = ungroupedPinned
ctx.Data["HasPinnedRepos"] = len(pinnedRepos) > 0
// Load public members (limit to 12 for overview display)
publicMembers, totalPublicMembers, err := organization.GetPublicOrgMembers(ctx, org.ID, 12)
if err != nil {
log.Error("GetPublicOrgMembers: %v", err)
}
ctx.Data["PublicMembers"] = publicMembers
ctx.Data["TotalPublicMembers"] = totalPublicMembers
ctx.Data["HasMorePublicMembers"] = totalPublicMembers > 12
// Load organization stats
orgStats, err := org_service.GetOrgOverviewStats(ctx, org.ID)
if err != nil {
log.Error("GetOrgOverviewStats: %v", err)
}
ctx.Data["OrgStats"] = orgStats
// Always show overview by default for organizations
isViewOverview := !viewRepositories
// Load profile readme if available
prepareOrgProfileReadme(ctx, prepareResult)
ctx.Data["PageIsViewRepositories"] = !isViewOverview
ctx.Data["PageIsViewOverview"] = isViewOverview
ctx.Data["ShowOrgProfileReadmeSelector"] = isViewOverview && prepareResult.ProfilePublicReadmeBlob != nil && prepareResult.ProfilePrivateReadmeBlob != nil
repos, count, err := repo_model.SearchRepository(ctx, repo_model.SearchRepoOptions{
ListOptions: db.ListOptions{
PageSize: setting.UI.User.RepoPagingNum,
Page: page,
},
Keyword: keyword,
OwnerID: org.ID,
OrderBy: orderBy,
Private: ctx.IsSigned,
Actor: ctx.Doer,
Language: language,
IncludeDescription: setting.UI.SearchRepoDescription,
Archived: archived,
Fork: fork,
Mirror: mirror,
Template: template,
IsPrivate: private,
})
if err != nil {
ctx.ServerError("SearchRepository", err)
return
}
ctx.Data["Repos"] = repos
ctx.Data["Total"] = count
pager := context.NewPagination(int(count), setting.UI.User.RepoPagingNum, page, 5)
pager.AddParamFromRequest(ctx.Req)
ctx.Data["Page"] = pager
ctx.HTML(http.StatusOK, tplOrgHome)
}
func prepareOrgProfileReadme(ctx *context.Context, prepareResult *shared_user.PrepareOwnerHeaderResult) bool {
viewAs := ctx.FormString("view_as", util.Iif(ctx.Org.IsMember, "member", "public"))
viewAsMember := viewAs == "member"
var profileRepo *repo_model.Repository
var readmeBlob *git.Blob
if viewAsMember {
if prepareResult.ProfilePrivateReadmeBlob != nil {
profileRepo, readmeBlob = prepareResult.ProfilePrivateRepo, prepareResult.ProfilePrivateReadmeBlob
} else {
profileRepo, readmeBlob = prepareResult.ProfilePublicRepo, prepareResult.ProfilePublicReadmeBlob
viewAsMember = false
}
} else {
if prepareResult.ProfilePublicReadmeBlob != nil {
profileRepo, readmeBlob = prepareResult.ProfilePublicRepo, prepareResult.ProfilePublicReadmeBlob
} else {
profileRepo, readmeBlob = prepareResult.ProfilePrivateRepo, prepareResult.ProfilePrivateReadmeBlob
viewAsMember = true
}
}
if readmeBlob == nil {
return false
}
readmeBytes, err := readmeBlob.GetBlobContent(setting.UI.MaxDisplayFileSize)
if err != nil {
log.Error("failed to GetBlobContent for profile %q (view as %q) readme: %v", profileRepo.FullName(), viewAs, err)
return false
}
rctx := renderhelper.NewRenderContextRepoFile(ctx, profileRepo, renderhelper.RepoFileOptions{
CurrentRefPath: path.Join("branch", util.PathEscapeSegments(profileRepo.DefaultBranch)),
})
ctx.Data["ProfileReadmeContent"], err = markdown.RenderString(rctx, readmeBytes)
if err != nil {
log.Error("failed to GetBlobContent for profile %q (view as %q) readme: %v", profileRepo.FullName(), viewAs, err)
return false
}
ctx.Data["IsViewingOrgAsMember"] = viewAsMember
return true
}
// CreateProfileRepo creates a .profile repository with README for the organization
func CreateProfileRepo(ctx *context.Context) {
org := ctx.Org.Organization
// Check if user can create repos in this org
if !ctx.Org.CanCreateOrgRepo {
ctx.Flash.Error(ctx.Tr("org.profile_repo_no_permission"))
ctx.Redirect(org.AsUser().HomeLink())
return
}
// Check if .profile repo already exists
exists, err := repo_model.IsRepositoryModelExist(ctx, org.AsUser(), ".profile")
if err != nil {
ctx.ServerError("IsRepositoryExist", err)
return
}
if exists {
ctx.Redirect(org.AsUser().HomeLink() + "/.profile")
return
}
// Create the .profile repository
repo, err := repo_service.CreateRepository(ctx, ctx.Doer, org.AsUser(), repo_service.CreateRepoOptions{
Name: ".profile",
Description: "Organization profile",
AutoInit: true,
Readme: "Default",
DefaultBranch: "main",
IsPrivate: false,
})
if err != nil {
log.Error("CreateProfileRepo: %v", err)
ctx.Flash.Error(ctx.Tr("org.profile_repo_create_failed"))
ctx.Redirect(org.AsUser().HomeLink())
return
}
// Redirect to edit the README
ctx.Redirect(repo.Link() + "/_edit/main/README.md")
}