2014-08-14 00:12:21 -06:00
|
|
|
// Copyright 2014 The Gogs Authors. All rights reserved.
|
2020-01-24 12:00:29 -07:00
|
|
|
// Copyright 2020 The Gitea Authors.
|
2022-11-27 11:20:29 -07:00
|
|
|
// SPDX-License-Identifier: MIT
|
2014-08-14 00:12:21 -06:00
|
|
|
|
2016-03-11 09:56:52 -07:00
|
|
|
package context
|
2014-08-14 00:12:21 -06:00
|
|
|
|
|
|
|
import (
|
2016-01-31 06:28:42 -07:00
|
|
|
"strings"
|
|
|
|
|
2022-03-29 00:29:02 -06:00
|
|
|
"code.gitea.io/gitea/models/organization"
|
2021-11-28 04:58:28 -07:00
|
|
|
"code.gitea.io/gitea/models/perm"
|
2023-01-20 04:42:33 -07:00
|
|
|
"code.gitea.io/gitea/models/unit"
|
2021-11-11 00:03:30 -07:00
|
|
|
user_model "code.gitea.io/gitea/models/user"
|
2022-03-31 11:31:53 -06:00
|
|
|
"code.gitea.io/gitea/modules/setting"
|
2022-07-15 08:21:54 -06:00
|
|
|
"code.gitea.io/gitea/modules/structs"
|
2014-08-14 00:12:21 -06:00
|
|
|
)
|
|
|
|
|
2016-11-24 23:51:01 -07:00
|
|
|
// Organization contains organization context
|
2016-03-13 15:37:44 -06:00
|
|
|
type Organization struct {
|
2019-11-20 04:27:49 -07:00
|
|
|
IsOwner bool
|
|
|
|
IsMember bool
|
|
|
|
IsTeamMember bool // Is member of team.
|
|
|
|
IsTeamAdmin bool // In owner team or team that has admin permission level.
|
2022-03-29 00:29:02 -06:00
|
|
|
Organization *organization.Organization
|
2019-11-20 04:27:49 -07:00
|
|
|
OrgLink string
|
|
|
|
CanCreateOrgRepo bool
|
2023-08-15 08:00:35 -06:00
|
|
|
PublicMemberOnly bool // Only display public members
|
2016-03-13 15:37:44 -06:00
|
|
|
|
2022-03-29 00:29:02 -06:00
|
|
|
Team *organization.Team
|
|
|
|
Teams []*organization.Team
|
2016-03-13 15:37:44 -06:00
|
|
|
}
|
|
|
|
|
2023-01-20 04:42:33 -07:00
|
|
|
func (org *Organization) CanWriteUnit(ctx *Context, unitType unit.Type) bool {
|
2023-03-10 08:18:20 -07:00
|
|
|
return org.Organization.UnitPermission(ctx, ctx.Doer, unitType) >= perm.AccessModeWrite
|
2023-01-20 04:42:33 -07:00
|
|
|
}
|
|
|
|
|
2023-03-10 08:18:20 -07:00
|
|
|
func (org *Organization) CanReadUnit(ctx *Context, unitType unit.Type) bool {
|
|
|
|
return org.Organization.UnitPermission(ctx, ctx.Doer, unitType) >= perm.AccessModeRead
|
|
|
|
}
|
2023-01-20 04:42:33 -07:00
|
|
|
|
2023-03-10 08:18:20 -07:00
|
|
|
func GetOrganizationByParams(ctx *Context) {
|
|
|
|
orgName := ctx.Params(":org")
|
|
|
|
|
|
|
|
var err error
|
2023-01-20 04:42:33 -07:00
|
|
|
|
2023-03-10 08:18:20 -07:00
|
|
|
ctx.Org.Organization, err = organization.GetOrgByName(ctx, orgName)
|
|
|
|
if err != nil {
|
|
|
|
if organization.IsErrOrgNotExist(err) {
|
|
|
|
redirectUserID, err := user_model.LookupUserRedirect(orgName)
|
|
|
|
if err == nil {
|
2023-05-20 19:50:53 -06:00
|
|
|
RedirectToUser(ctx.Base, orgName, redirectUserID)
|
2023-03-10 08:18:20 -07:00
|
|
|
} else if user_model.IsErrUserRedirectNotExist(err) {
|
|
|
|
ctx.NotFound("GetUserByName", err)
|
|
|
|
} else {
|
|
|
|
ctx.ServerError("LookupUserRedirect", err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ctx.ServerError("GetUserByName", err)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
2023-01-20 04:42:33 -07:00
|
|
|
}
|
|
|
|
|
2016-11-24 23:51:01 -07:00
|
|
|
// HandleOrgAssignment handles organization assignment
|
2015-11-24 17:14:00 -07:00
|
|
|
func HandleOrgAssignment(ctx *Context, args ...bool) {
|
|
|
|
var (
|
2016-01-31 06:28:42 -07:00
|
|
|
requireMember bool
|
|
|
|
requireOwner bool
|
|
|
|
requireTeamMember bool
|
2016-01-31 08:18:28 -07:00
|
|
|
requireTeamAdmin bool
|
2015-11-24 17:14:00 -07:00
|
|
|
)
|
|
|
|
if len(args) >= 1 {
|
|
|
|
requireMember = args[0]
|
|
|
|
}
|
|
|
|
if len(args) >= 2 {
|
|
|
|
requireOwner = args[1]
|
|
|
|
}
|
|
|
|
if len(args) >= 3 {
|
2016-01-31 06:28:42 -07:00
|
|
|
requireTeamMember = args[2]
|
|
|
|
}
|
|
|
|
if len(args) >= 4 {
|
2016-01-31 08:18:28 -07:00
|
|
|
requireTeamAdmin = args[3]
|
2015-11-24 17:14:00 -07:00
|
|
|
}
|
2014-08-14 00:12:21 -06:00
|
|
|
|
2015-11-24 17:14:00 -07:00
|
|
|
var err error
|
2023-03-10 08:18:20 -07:00
|
|
|
|
|
|
|
if ctx.ContextUser == nil {
|
|
|
|
// if Organization is not defined, get it from params
|
|
|
|
if ctx.Org.Organization == nil {
|
|
|
|
GetOrganizationByParams(ctx)
|
|
|
|
if ctx.Written() {
|
|
|
|
return
|
2021-01-24 08:23:05 -07:00
|
|
|
}
|
2014-08-14 00:12:21 -06:00
|
|
|
}
|
2023-03-10 08:18:20 -07:00
|
|
|
} else if ctx.ContextUser.IsOrganization() {
|
|
|
|
if ctx.Org == nil {
|
|
|
|
ctx.Org = &Organization{}
|
|
|
|
}
|
|
|
|
ctx.Org.Organization = (*organization.Organization)(ctx.ContextUser)
|
|
|
|
} else {
|
|
|
|
// ContextUser is an individual User
|
2015-11-24 17:14:00 -07:00
|
|
|
return
|
|
|
|
}
|
2023-03-10 08:18:20 -07:00
|
|
|
|
2015-11-24 17:14:00 -07:00
|
|
|
org := ctx.Org.Organization
|
2022-07-15 08:21:54 -06:00
|
|
|
|
|
|
|
// Handle Visibility
|
|
|
|
if org.Visibility != structs.VisibleTypePublic && !ctx.IsSigned {
|
|
|
|
// We must be signed in to see limited or private organizations
|
|
|
|
ctx.NotFound("OrgAssignment", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if org.Visibility == structs.VisibleTypePrivate {
|
|
|
|
requireMember = true
|
|
|
|
} else if ctx.IsSigned && ctx.Doer.IsRestricted {
|
|
|
|
requireMember = true
|
|
|
|
}
|
|
|
|
|
2022-03-26 03:04:22 -06:00
|
|
|
ctx.ContextUser = org.AsUser()
|
2015-11-24 17:14:00 -07:00
|
|
|
ctx.Data["Org"] = org
|
2014-08-14 00:12:21 -06:00
|
|
|
|
2015-11-28 14:31:06 -07:00
|
|
|
// Admin has super access.
|
2022-03-22 01:03:22 -06:00
|
|
|
if ctx.IsSigned && ctx.Doer.IsAdmin {
|
2015-11-28 14:31:06 -07:00
|
|
|
ctx.Org.IsOwner = true
|
|
|
|
ctx.Org.IsMember = true
|
2016-01-31 06:28:42 -07:00
|
|
|
ctx.Org.IsTeamMember = true
|
2016-01-31 08:18:28 -07:00
|
|
|
ctx.Org.IsTeamAdmin = true
|
2019-11-20 04:27:49 -07:00
|
|
|
ctx.Org.CanCreateOrgRepo = true
|
2015-11-28 14:31:06 -07:00
|
|
|
} else if ctx.IsSigned {
|
2022-03-22 01:03:22 -06:00
|
|
|
ctx.Org.IsOwner, err = org.IsOwnedBy(ctx.Doer.ID)
|
2017-12-21 00:43:26 -07:00
|
|
|
if err != nil {
|
2018-01-10 14:34:17 -07:00
|
|
|
ctx.ServerError("IsOwnedBy", err)
|
2017-12-21 00:43:26 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-11-24 17:14:00 -07:00
|
|
|
if ctx.Org.IsOwner {
|
|
|
|
ctx.Org.IsMember = true
|
2016-01-31 06:28:42 -07:00
|
|
|
ctx.Org.IsTeamMember = true
|
2016-01-31 08:18:28 -07:00
|
|
|
ctx.Org.IsTeamAdmin = true
|
2019-11-20 04:27:49 -07:00
|
|
|
ctx.Org.CanCreateOrgRepo = true
|
2015-11-24 17:14:00 -07:00
|
|
|
} else {
|
2022-03-22 01:03:22 -06:00
|
|
|
ctx.Org.IsMember, err = org.IsOrgMember(ctx.Doer.ID)
|
2017-12-21 00:43:26 -07:00
|
|
|
if err != nil {
|
2018-01-10 14:34:17 -07:00
|
|
|
ctx.ServerError("IsOrgMember", err)
|
2017-12-21 00:43:26 -07:00
|
|
|
return
|
2014-08-14 00:12:21 -06:00
|
|
|
}
|
2022-03-22 01:03:22 -06:00
|
|
|
ctx.Org.CanCreateOrgRepo, err = org.CanCreateOrgRepo(ctx.Doer.ID)
|
2019-11-20 04:27:49 -07:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("CanCreateOrgRepo", err)
|
|
|
|
return
|
|
|
|
}
|
2014-08-14 00:12:21 -06:00
|
|
|
}
|
2015-11-24 17:14:00 -07:00
|
|
|
} else {
|
|
|
|
// Fake data.
|
2021-11-24 02:49:20 -07:00
|
|
|
ctx.Data["SignedUser"] = &user_model.User{}
|
2015-11-24 17:14:00 -07:00
|
|
|
}
|
|
|
|
if (requireMember && !ctx.Org.IsMember) ||
|
|
|
|
(requireOwner && !ctx.Org.IsOwner) {
|
2018-01-10 14:34:17 -07:00
|
|
|
ctx.NotFound("OrgAssignment", err)
|
2015-11-24 17:14:00 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
ctx.Data["IsOrganizationOwner"] = ctx.Org.IsOwner
|
2015-11-28 14:31:06 -07:00
|
|
|
ctx.Data["IsOrganizationMember"] = ctx.Org.IsMember
|
2022-03-31 11:31:53 -06:00
|
|
|
ctx.Data["IsPackageEnabled"] = setting.Packages.Enabled
|
2022-10-10 17:12:03 -06:00
|
|
|
ctx.Data["IsRepoIndexerEnabled"] = setting.Indexer.RepoIndexerEnabled
|
2021-11-22 08:21:55 -07:00
|
|
|
ctx.Data["IsPublicMember"] = func(uid int64) bool {
|
2022-03-29 00:29:02 -06:00
|
|
|
is, _ := organization.IsPublicMembership(ctx.Org.Organization.ID, uid)
|
2021-11-22 08:21:55 -07:00
|
|
|
return is
|
|
|
|
}
|
2019-11-20 04:27:49 -07:00
|
|
|
ctx.Data["CanCreateOrgRepo"] = ctx.Org.CanCreateOrgRepo
|
2014-08-15 04:29:41 -06:00
|
|
|
|
2021-11-19 04:41:40 -07:00
|
|
|
ctx.Org.OrgLink = org.AsUser().OrganisationLink()
|
2015-11-24 17:14:00 -07:00
|
|
|
ctx.Data["OrgLink"] = ctx.Org.OrgLink
|
2014-08-16 02:21:17 -06:00
|
|
|
|
2023-08-15 08:00:35 -06:00
|
|
|
// Member
|
|
|
|
ctx.Org.PublicMemberOnly = ctx.Doer == nil || !ctx.Org.IsMember && !ctx.Doer.IsAdmin
|
|
|
|
opts := &organization.FindOrgMembersOpts{
|
|
|
|
OrgID: org.ID,
|
|
|
|
PublicOnly: ctx.Org.PublicMemberOnly,
|
|
|
|
}
|
|
|
|
ctx.Data["NumMembers"], err = organization.CountOrgMembers(opts)
|
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("CountOrgMembers", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-11-24 17:14:00 -07:00
|
|
|
// Team.
|
2016-01-31 06:28:42 -07:00
|
|
|
if ctx.Org.IsMember {
|
2022-02-26 13:12:01 -07:00
|
|
|
shouldSeeAllTeams := false
|
2016-01-31 09:13:31 -07:00
|
|
|
if ctx.Org.IsOwner {
|
2022-02-26 13:12:01 -07:00
|
|
|
shouldSeeAllTeams = true
|
|
|
|
} else {
|
2022-03-22 01:03:22 -06:00
|
|
|
teams, err := org.GetUserTeams(ctx.Doer.ID)
|
2022-02-26 13:12:01 -07:00
|
|
|
if err != nil {
|
|
|
|
ctx.ServerError("GetUserTeams", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for _, team := range teams {
|
|
|
|
if team.IncludesAllRepositories && team.AccessMode >= perm.AccessModeAdmin {
|
|
|
|
shouldSeeAllTeams = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if shouldSeeAllTeams {
|
2021-11-19 04:41:40 -07:00
|
|
|
ctx.Org.Teams, err = org.LoadTeams()
|
|
|
|
if err != nil {
|
2021-08-12 06:43:08 -06:00
|
|
|
ctx.ServerError("LoadTeams", err)
|
2016-01-31 09:13:31 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
} else {
|
2022-03-22 01:03:22 -06:00
|
|
|
ctx.Org.Teams, err = org.GetUserTeams(ctx.Doer.ID)
|
2016-07-24 04:09:45 -06:00
|
|
|
if err != nil {
|
2018-01-10 14:34:17 -07:00
|
|
|
ctx.ServerError("GetUserTeams", err)
|
2016-01-31 09:13:31 -07:00
|
|
|
return
|
|
|
|
}
|
2016-01-31 06:28:42 -07:00
|
|
|
}
|
2023-08-15 08:00:35 -06:00
|
|
|
ctx.Data["NumTeams"] = len(ctx.Org.Teams)
|
2016-01-31 06:28:42 -07:00
|
|
|
}
|
|
|
|
|
2015-11-24 17:14:00 -07:00
|
|
|
teamName := ctx.Params(":team")
|
|
|
|
if len(teamName) > 0 {
|
2016-01-31 06:28:42 -07:00
|
|
|
teamExists := false
|
2021-11-19 04:41:40 -07:00
|
|
|
for _, team := range ctx.Org.Teams {
|
2016-02-04 11:03:34 -07:00
|
|
|
if team.LowerName == strings.ToLower(teamName) {
|
2016-01-31 06:28:42 -07:00
|
|
|
teamExists = true
|
|
|
|
ctx.Org.Team = team
|
|
|
|
ctx.Org.IsTeamMember = true
|
|
|
|
ctx.Data["Team"] = ctx.Org.Team
|
|
|
|
break
|
2014-08-16 02:21:17 -06:00
|
|
|
}
|
2016-01-31 06:28:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if !teamExists {
|
2018-01-10 14:34:17 -07:00
|
|
|
ctx.NotFound("OrgAssignment", err)
|
2016-01-31 06:28:42 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx.Data["IsTeamMember"] = ctx.Org.IsTeamMember
|
|
|
|
if requireTeamMember && !ctx.Org.IsTeamMember {
|
2018-01-10 14:34:17 -07:00
|
|
|
ctx.NotFound("OrgAssignment", err)
|
2014-08-24 07:09:05 -06:00
|
|
|
return
|
2014-08-16 02:21:17 -06:00
|
|
|
}
|
2016-01-31 06:28:42 -07:00
|
|
|
|
2022-01-04 20:37:00 -07:00
|
|
|
ctx.Org.IsTeamAdmin = ctx.Org.Team.IsOwnerTeam() || ctx.Org.Team.AccessMode >= perm.AccessModeAdmin
|
2016-01-31 08:18:28 -07:00
|
|
|
ctx.Data["IsTeamAdmin"] = ctx.Org.IsTeamAdmin
|
|
|
|
if requireTeamAdmin && !ctx.Org.IsTeamAdmin {
|
2018-01-10 14:34:17 -07:00
|
|
|
ctx.NotFound("OrgAssignment", err)
|
2016-01-31 06:28:42 -07:00
|
|
|
return
|
|
|
|
}
|
2015-11-24 17:14:00 -07:00
|
|
|
}
|
2023-09-06 00:38:14 -06:00
|
|
|
ctx.Data["ContextUser"] = ctx.ContextUser
|
2023-03-10 08:18:20 -07:00
|
|
|
|
|
|
|
ctx.Data["CanReadProjects"] = ctx.Org.CanReadUnit(ctx, unit.TypeProjects)
|
|
|
|
ctx.Data["CanReadPackages"] = ctx.Org.CanReadUnit(ctx, unit.TypePackages)
|
|
|
|
ctx.Data["CanReadCode"] = ctx.Org.CanReadUnit(ctx, unit.TypeCode)
|
2015-11-24 17:14:00 -07:00
|
|
|
}
|
|
|
|
|
2021-01-29 08:35:30 -07:00
|
|
|
// OrgAssignment returns a middleware to handle organization assignment
|
2021-01-26 08:36:53 -07:00
|
|
|
func OrgAssignment(args ...bool) func(ctx *Context) {
|
2015-11-24 17:14:00 -07:00
|
|
|
return func(ctx *Context) {
|
|
|
|
HandleOrgAssignment(ctx, args...)
|
2014-08-14 00:12:21 -06:00
|
|
|
}
|
|
|
|
}
|