2014-06-25 08:44:48 +04:00
// Copyright 2014 The Gogs Authors. All rights reserved.
2019-02-18 19:00:27 +03:00
// Copyright 2019 The Gitea Authors. All rights reserved.
2014-06-25 08:44:48 +04:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package models
2014-06-27 11:37:01 +04:00
import (
2021-11-18 20:42:27 +03:00
"context"
2015-02-23 10:15:53 +03:00
"fmt"
2014-06-27 11:37:01 +04:00
"strings"
2015-09-07 20:58:23 +03:00
2021-09-19 14:49:59 +03:00
"code.gitea.io/gitea/models/db"
2021-11-28 14:58:28 +03:00
"code.gitea.io/gitea/models/perm"
2021-12-10 04:27:50 +03:00
repo_model "code.gitea.io/gitea/models/repo"
2021-11-09 22:57:58 +03:00
"code.gitea.io/gitea/models/unit"
2021-11-11 10:03:30 +03:00
user_model "code.gitea.io/gitea/models/user"
2018-01-08 10:48:37 +03:00
"code.gitea.io/gitea/modules/log"
2019-08-24 15:28:59 +03:00
"code.gitea.io/gitea/modules/setting"
2019-02-18 19:00:27 +03:00
"code.gitea.io/gitea/modules/structs"
2018-01-08 10:48:37 +03:00
2019-06-23 18:22:43 +03:00
"xorm.io/builder"
2014-06-27 11:37:01 +04:00
)
2021-11-19 14:41:40 +03:00
// Organization represents an organization
2021-11-24 12:49:20 +03:00
type Organization user_model . User
2021-11-19 14:41:40 +03:00
// OrgFromUser converts user to organization
2021-11-24 12:49:20 +03:00
func OrgFromUser ( user * user_model . User ) * Organization {
2021-11-19 14:41:40 +03:00
return ( * Organization ) ( user )
}
// TableName represents the real table name of Organization
func ( Organization ) TableName ( ) string {
return "user"
}
2014-12-13 04:30:32 +03:00
// IsOwnedBy returns true if given user is in the owner team.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) IsOwnedBy ( uid int64 ) ( bool , error ) {
2016-07-23 20:08:22 +03:00
return IsOrganizationOwner ( org . ID , uid )
2014-06-30 00:30:41 +04:00
}
// IsOrgMember returns true if given user is member of organization.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) IsOrgMember ( uid int64 ) ( bool , error ) {
2017-12-21 10:43:26 +03:00
return IsOrganizationMember ( org . ID , uid )
2014-06-30 00:30:41 +04:00
}
2019-11-20 14:27:49 +03:00
// CanCreateOrgRepo returns true if given user can create repo in organization
2021-11-19 14:41:40 +03:00
func ( org * Organization ) CanCreateOrgRepo ( uid int64 ) ( bool , error ) {
2019-11-20 14:27:49 +03:00
return CanCreateOrgRepo ( org . ID , uid )
}
2021-11-19 14:41:40 +03:00
func ( org * Organization ) getTeam ( e db . Engine , name string ) ( * Team , error ) {
2016-07-23 20:08:22 +03:00
return getTeam ( e , org . ID , name )
2015-02-13 08:58:46 +03:00
}
2014-08-16 12:21:17 +04:00
// GetTeam returns named team of organization.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) GetTeam ( name string ) ( * Team , error ) {
2021-09-23 18:45:36 +03:00
return org . getTeam ( db . GetEngine ( db . DefaultContext ) , name )
2015-02-13 08:58:46 +03:00
}
2021-11-19 14:41:40 +03:00
func ( org * Organization ) getOwnerTeam ( e db . Engine ) ( * Team , error ) {
2016-11-28 04:30:08 +03:00
return org . getTeam ( e , ownerTeamName )
2014-08-16 12:21:17 +04:00
}
2014-06-28 08:40:07 +04:00
// GetOwnerTeam returns owner team of organization.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) GetOwnerTeam ( ) ( * Team , error ) {
2021-09-23 18:45:36 +03:00
return org . getOwnerTeam ( db . GetEngine ( db . DefaultContext ) )
2015-02-13 08:58:46 +03:00
}
2021-11-19 14:41:40 +03:00
func ( org * Organization ) loadTeams ( e db . Engine ) ( [ ] * Team , error ) {
var teams [ ] * Team
return teams , e .
2016-11-10 18:16:32 +03:00
Where ( "org_id=?" , org . ID ) .
2016-11-28 04:30:08 +03:00
OrderBy ( "CASE WHEN name LIKE '" + ownerTeamName + "' THEN '' ELSE name END" ) .
2021-11-19 14:41:40 +03:00
Find ( & teams )
2014-06-28 08:40:07 +04:00
}
2021-08-12 15:43:08 +03:00
// LoadTeams load teams if not loaded.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) LoadTeams ( ) ( [ ] * Team , error ) {
2021-09-23 18:45:36 +03:00
return org . loadTeams ( db . GetEngine ( db . DefaultContext ) )
2014-06-28 23:43:25 +04:00
}
// GetMembers returns all members of organization.
2021-11-28 17:11:58 +03:00
func ( org * Organization ) GetMembers ( ) ( user_model . UserList , map [ int64 ] bool , error ) {
2021-11-19 14:41:40 +03:00
return FindOrgMembers ( & FindOrgMembersOpts {
2019-12-06 08:34:54 +03:00
OrgID : org . ID ,
} )
2021-11-19 14:41:40 +03:00
}
// HasMemberWithUserID returns true if user with userID is part of the u organisation.
func ( org * Organization ) HasMemberWithUserID ( userID int64 ) bool {
return org . hasMemberWithUserID ( db . GetEngine ( db . DefaultContext ) , userID )
}
func ( org * Organization ) hasMemberWithUserID ( e db . Engine , userID int64 ) bool {
isMember , err := isOrganizationMember ( e , org . ID , userID )
if err != nil {
log . Error ( "IsOrganizationMember: %v" , err )
return false
}
return isMember
}
// AvatarLink returns the full avatar link with http host
func ( org * Organization ) AvatarLink ( ) string {
return org . AsUser ( ) . AvatarLink ( )
}
// HTMLURL returns the organization's full link.
func ( org * Organization ) HTMLURL ( ) string {
return org . AsUser ( ) . HTMLURL ( )
}
// OrganisationLink returns the organization sub page link.
func ( org * Organization ) OrganisationLink ( ) string {
return org . AsUser ( ) . OrganisationLink ( )
}
// ShortName ellipses username to length
func ( org * Organization ) ShortName ( length int ) string {
return org . AsUser ( ) . ShortName ( length )
}
// HomeLink returns the user or organization home page link.
func ( org * Organization ) HomeLink ( ) string {
return org . AsUser ( ) . HomeLink ( )
2019-12-06 08:34:54 +03:00
}
2021-11-22 18:21:55 +03:00
// CanCreateRepo returns if user login can create a repository
// NOTE: functions calling this assume a failure due to repository count limit; if new checks are added, those functions should be revised
func ( org * Organization ) CanCreateRepo ( ) bool {
return org . AsUser ( ) . CanCreateRepo ( )
}
2021-07-08 14:38:13 +03:00
// FindOrgMembersOpts represensts find org members conditions
2019-12-06 08:34:54 +03:00
type FindOrgMembersOpts struct {
2021-09-24 14:32:56 +03:00
db . ListOptions
2019-12-06 08:34:54 +03:00
OrgID int64
PublicOnly bool
}
// CountOrgMembers counts the organization's members
2021-08-12 15:43:08 +03:00
func CountOrgMembers ( opts * FindOrgMembersOpts ) ( int64 , error ) {
2021-09-23 18:45:36 +03:00
sess := db . GetEngine ( db . DefaultContext ) . Where ( "org_id=?" , opts . OrgID )
2019-12-06 08:34:54 +03:00
if opts . PublicOnly {
sess . And ( "is_public = ?" , true )
}
return sess . Count ( new ( OrgUser ) )
}
// FindOrgMembers loads organization members according conditions
2021-11-28 17:11:58 +03:00
func FindOrgMembers ( opts * FindOrgMembersOpts ) ( user_model . UserList , map [ int64 ] bool , error ) {
2020-01-24 22:00:29 +03:00
ous , err := GetOrgUsersByOrgID ( opts )
2014-06-28 23:43:25 +04:00
if err != nil {
2019-12-06 08:34:54 +03:00
return nil , nil , err
2014-06-28 23:43:25 +04:00
}
2021-03-14 21:52:12 +03:00
ids := make ( [ ] int64 , len ( ous ) )
idsIsPublic := make ( map [ int64 ] bool , len ( ous ) )
2014-06-28 23:43:25 +04:00
for i , ou := range ous {
2016-11-26 03:20:18 +03:00
ids [ i ] = ou . UID
2019-08-02 19:06:28 +03:00
idsIsPublic [ ou . UID ] = ou . IsPublic
2014-06-28 23:43:25 +04:00
}
2019-12-06 08:34:54 +03:00
2021-11-28 17:11:58 +03:00
users , err := user_model . GetUsersByIDs ( ids )
2019-12-06 08:34:54 +03:00
if err != nil {
return nil , nil , err
}
return users , idsIsPublic , nil
2014-06-28 23:43:25 +04:00
}
2014-08-15 14:29:41 +04:00
// AddMember adds new member to organization.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) AddMember ( uid int64 ) error {
2016-07-23 20:08:22 +03:00
return AddOrgUser ( org . ID , uid )
2014-08-15 14:29:41 +04:00
}
// RemoveMember removes member from organization.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) RemoveMember ( uid int64 ) error {
2016-07-23 20:08:22 +03:00
return RemoveOrgUser ( org . ID , uid )
2014-08-15 14:29:41 +04:00
}
2021-11-19 14:41:40 +03:00
func ( org * Organization ) removeOrgRepo ( e db . Engine , repoID int64 ) error {
2016-07-23 20:08:22 +03:00
return removeOrgRepo ( e , org . ID , repoID )
2015-03-01 05:44:09 +03:00
}
// RemoveOrgRepo removes all team-repository relations of organization.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) RemoveOrgRepo ( repoID int64 ) error {
2021-09-23 18:45:36 +03:00
return org . removeOrgRepo ( db . GetEngine ( db . DefaultContext ) , repoID )
2015-03-01 05:44:09 +03:00
}
2021-11-19 14:41:40 +03:00
// AsUser returns the org as user object
2021-11-24 12:49:20 +03:00
func ( org * Organization ) AsUser ( ) * user_model . User {
return ( * user_model . User ) ( org )
2021-11-19 14:41:40 +03:00
}
// DisplayName returns full name if it's not empty,
// returns username otherwise.
func ( org * Organization ) DisplayName ( ) string {
return org . AsUser ( ) . DisplayName ( )
}
// CustomAvatarRelativePath returns user custom avatar relative path.
func ( org * Organization ) CustomAvatarRelativePath ( ) string {
return org . Avatar
}
2014-06-27 11:37:01 +04:00
// CreateOrganization creates record of a new organization.
2021-11-24 12:49:20 +03:00
func CreateOrganization ( org * Organization , owner * user_model . User ) ( err error ) {
2016-12-31 05:33:30 +03:00
if ! owner . CanCreateOrganization ( ) {
return ErrUserNotAllowedCreateOrg { }
}
2021-11-24 12:49:20 +03:00
if err = user_model . IsUsableUsername ( org . Name ) ; err != nil {
2015-03-27 00:11:47 +03:00
return err
2014-06-27 11:37:01 +04:00
}
2021-11-24 12:49:20 +03:00
isExist , err := user_model . IsUserExist ( 0 , org . Name )
2014-06-27 11:37:01 +04:00
if err != nil {
2015-03-27 00:11:47 +03:00
return err
2014-06-27 11:37:01 +04:00
} else if isExist {
2021-11-24 12:49:20 +03:00
return user_model . ErrUserAlreadyExist { Name : org . Name }
2014-06-27 11:37:01 +04:00
}
org . LowerName = strings . ToLower ( org . Name )
2021-11-24 12:49:20 +03:00
if org . Rands , err = user_model . GetUserSalt ( ) ; err != nil {
2016-12-20 15:32:02 +03:00
return err
}
2021-11-24 12:49:20 +03:00
if org . Salt , err = user_model . GetUserSalt ( ) ; err != nil {
2016-12-20 15:32:02 +03:00
return err
}
2015-09-06 17:08:14 +03:00
org . UseCustomAvatar = true
2015-12-11 23:48:02 +03:00
org . MaxRepoCreation = - 1
2014-06-27 11:37:01 +04:00
org . NumTeams = 1
org . NumMembers = 1
2021-11-24 12:49:20 +03:00
org . Type = user_model . UserTypeOrganization
2014-06-27 11:37:01 +04:00
2021-11-11 10:03:30 +03:00
ctx , committer , err := db . TxContext ( )
if err != nil {
2015-03-27 00:11:47 +03:00
return err
2014-06-27 11:37:01 +04:00
}
2021-11-11 10:03:30 +03:00
defer committer . Close ( )
2014-06-27 11:37:01 +04:00
2021-11-11 10:03:30 +03:00
if err = user_model . DeleteUserRedirect ( ctx , org . Name ) ; err != nil {
2021-01-24 18:23:05 +03:00
return err
}
2021-11-11 10:03:30 +03:00
if err = db . Insert ( ctx , org ) ; err != nil {
2015-03-27 00:11:47 +03:00
return fmt . Errorf ( "insert organization: %v" , err )
2014-07-05 06:43:39 +04:00
}
2021-11-24 12:49:20 +03:00
if err = user_model . GenerateRandomAvatarCtx ( ctx , org . AsUser ( ) ) ; err != nil {
2017-03-08 18:05:15 +03:00
return fmt . Errorf ( "generate random avatar: %v" , err )
}
2015-09-06 17:08:14 +03:00
// Add initial creator to organization and owner team.
2021-11-11 10:03:30 +03:00
if err = db . Insert ( ctx , & OrgUser {
2018-01-08 10:48:37 +03:00
UID : owner . ID ,
OrgID : org . ID ,
2015-09-06 17:08:14 +03:00
} ) ; err != nil {
return fmt . Errorf ( "insert org-user relation: %v" , err )
}
2014-07-05 06:43:39 +04:00
2014-06-27 11:37:01 +04:00
// Create default owner team.
t := & Team {
2019-11-06 12:37:14 +03:00
OrgID : org . ID ,
LowerName : strings . ToLower ( ownerTeamName ) ,
Name : ownerTeamName ,
2022-01-05 06:37:00 +03:00
AccessMode : perm . AccessModeOwner ,
2019-11-06 12:37:14 +03:00
NumMembers : 1 ,
IncludesAllRepositories : true ,
2019-11-20 14:27:49 +03:00
CanCreateOrgRepo : true ,
2014-06-27 11:37:01 +04:00
}
2021-11-11 10:03:30 +03:00
if err = db . Insert ( ctx , t ) ; err != nil {
2015-03-27 00:11:47 +03:00
return fmt . Errorf ( "insert owner team: %v" , err )
2014-06-27 11:37:01 +04:00
}
2018-06-21 19:00:13 +03:00
// insert units for team
2021-11-09 22:57:58 +03:00
units := make ( [ ] TeamUnit , 0 , len ( unit . AllRepoUnitTypes ) )
for _ , tp := range unit . AllRepoUnitTypes {
2018-06-21 19:00:13 +03:00
units = append ( units , TeamUnit {
OrgID : org . ID ,
TeamID : t . ID ,
Type : tp ,
} )
}
2021-11-11 10:03:30 +03:00
if err = db . Insert ( ctx , & units ) ; err != nil {
2018-06-21 19:00:13 +03:00
return err
}
2021-11-11 10:03:30 +03:00
if err = db . Insert ( ctx , & TeamUser {
2016-11-28 04:30:08 +03:00
UID : owner . ID ,
2016-07-23 20:08:22 +03:00
OrgID : org . ID ,
2015-02-23 10:15:53 +03:00
TeamID : t . ID ,
2015-09-06 17:08:14 +03:00
} ) ; err != nil {
2015-03-27 00:11:47 +03:00
return fmt . Errorf ( "insert team-user relation: %v" , err )
2015-02-23 10:15:53 +03:00
}
2021-11-11 10:03:30 +03:00
return committer . Commit ( )
2014-06-27 11:37:01 +04:00
}
2014-12-13 04:30:32 +03:00
// GetOrgByName returns organization by given name.
2021-11-19 14:41:40 +03:00
func GetOrgByName ( name string ) ( * Organization , error ) {
2014-12-13 04:30:32 +03:00
if len ( name ) == 0 {
2017-07-06 16:30:19 +03:00
return nil , ErrOrgNotExist { 0 , name }
2014-12-13 04:30:32 +03:00
}
2021-11-19 14:41:40 +03:00
u := & Organization {
2014-12-13 04:30:32 +03:00
LowerName : strings . ToLower ( name ) ,
2021-11-24 12:49:20 +03:00
Type : user_model . UserTypeOrganization ,
2014-12-13 04:30:32 +03:00
}
2021-09-23 18:45:36 +03:00
has , err := db . GetEngine ( db . DefaultContext ) . Get ( u )
2014-12-13 04:30:32 +03:00
if err != nil {
return nil , err
} else if ! has {
2017-07-06 16:30:19 +03:00
return nil , ErrOrgNotExist { 0 , name }
2014-12-13 04:30:32 +03:00
}
return u , nil
}
2014-08-28 18:29:00 +04:00
// CountOrganizations returns number of organizations.
func CountOrganizations ( ) int64 {
2021-09-23 18:45:36 +03:00
count , _ := db . GetEngine ( db . DefaultContext ) .
2016-11-10 18:16:32 +03:00
Where ( "type=1" ) .
2021-11-19 14:41:40 +03:00
Count ( new ( Organization ) )
2014-08-28 18:29:00 +04:00
return count
}
2021-11-18 20:42:27 +03:00
// DeleteOrganization deletes models associated to an organization.
2021-11-19 14:41:40 +03:00
func DeleteOrganization ( ctx context . Context , org * Organization ) error {
2021-11-24 12:49:20 +03:00
if org . Type != user_model . UserTypeOrganization {
2021-11-19 14:41:40 +03:00
return fmt . Errorf ( "%s is a user not an organization" , org . Name )
}
2022-02-17 11:37:48 +03:00
if err := db . DeleteBeans ( ctx ,
2021-11-18 20:42:27 +03:00
& Team { OrgID : org . ID } ,
& OrgUser { OrgID : org . ID } ,
& TeamUser { OrgID : org . ID } ,
& TeamUnit { OrgID : org . ID } ,
2015-09-06 15:54:08 +03:00
) ; err != nil {
return fmt . Errorf ( "deleteBeans: %v" , err )
2014-06-28 08:40:07 +04:00
}
2015-09-06 15:54:08 +03:00
2022-02-17 11:37:48 +03:00
if _ , err := db . GetEngine ( ctx ) . ID ( org . ID ) . Delete ( new ( user_model . User ) ) ; err != nil {
2017-01-11 16:10:43 +03:00
return fmt . Errorf ( "Delete: %v" , err )
2014-06-28 08:40:07 +04:00
}
2015-09-06 15:54:08 +03:00
2017-01-11 16:10:43 +03:00
return nil
2014-06-28 08:40:07 +04:00
}
2014-06-25 08:44:48 +04:00
// ________ ____ ___
// \_____ \_______ ____ | | \______ ___________
// / | \_ __ \/ ___\| | / ___// __ \_ __ \
// / | \ | \/ /_/ > | /\___ \\ ___/| | \/
// \_______ /__| \___ /|______//____ >\___ >__|
// \/ /_____/ \/ \/
// OrgUser represents an organization-user relation.
type OrgUser struct {
2015-02-23 10:15:53 +03:00
ID int64 ` xorm:"pk autoincr" `
2016-11-26 03:20:18 +03:00
UID int64 ` xorm:"INDEX UNIQUE(s)" `
2015-02-23 10:15:53 +03:00
OrgID int64 ` xorm:"INDEX UNIQUE(s)" `
2017-01-06 18:14:33 +03:00
IsPublic bool ` xorm:"INDEX" `
2018-01-08 10:48:37 +03:00
}
2021-09-19 14:49:59 +03:00
func init ( ) {
db . RegisterModel ( new ( OrgUser ) )
}
func isOrganizationOwner ( e db . Engine , orgID , uid int64 ) ( bool , error ) {
2019-08-02 19:06:28 +03:00
ownerTeam , err := getOwnerTeam ( e , orgID )
if err != nil {
2019-09-23 23:08:03 +03:00
if IsErrTeamNotExist ( err ) {
2019-08-02 19:06:28 +03:00
log . Error ( "Organization does not have owner team: %d" , orgID )
return false , nil
}
2018-01-08 10:48:37 +03:00
return false , err
}
return isTeamMember ( e , orgID , ownerTeam . ID , uid )
2014-06-25 08:44:48 +04:00
}
2014-06-30 00:30:41 +04:00
// IsOrganizationOwner returns true if given user is in the owner team.
2017-12-21 10:43:26 +03:00
func IsOrganizationOwner ( orgID , uid int64 ) ( bool , error ) {
2021-09-23 18:45:36 +03:00
return isOrganizationOwner ( db . GetEngine ( db . DefaultContext ) , orgID , uid )
2014-06-30 00:30:41 +04:00
}
// IsOrganizationMember returns true if given user is member of organization.
2017-12-21 10:43:26 +03:00
func IsOrganizationMember ( orgID , uid int64 ) ( bool , error ) {
2021-09-23 18:45:36 +03:00
return isOrganizationMember ( db . GetEngine ( db . DefaultContext ) , orgID , uid )
2019-01-14 05:29:58 +03:00
}
2021-09-19 14:49:59 +03:00
func isOrganizationMember ( e db . Engine , orgID , uid int64 ) ( bool , error ) {
2019-01-14 05:29:58 +03:00
return e .
2016-11-10 18:16:32 +03:00
Where ( "uid=?" , uid ) .
2016-11-26 03:20:18 +03:00
And ( "org_id=?" , orgID ) .
2017-12-21 10:43:26 +03:00
Table ( "org_user" ) .
Exist ( )
2014-06-30 00:30:41 +04:00
}
2014-12-07 04:22:48 +03:00
// IsPublicMembership returns true if given user public his/her membership.
2017-12-21 10:43:26 +03:00
func IsPublicMembership ( orgID , uid int64 ) ( bool , error ) {
2021-09-23 18:45:36 +03:00
return db . GetEngine ( db . DefaultContext ) .
2016-11-10 18:16:32 +03:00
Where ( "uid=?" , uid ) .
2016-11-26 03:20:18 +03:00
And ( "org_id=?" , orgID ) .
2016-11-10 18:16:32 +03:00
And ( "is_public=?" , true ) .
2017-12-21 10:43:26 +03:00
Table ( "org_user" ) .
Exist ( )
2014-08-15 14:29:41 +04:00
}
2019-11-20 14:27:49 +03:00
// CanCreateOrgRepo returns true if user can create repo in organization
func CanCreateOrgRepo ( orgID , uid int64 ) ( bool , error ) {
if owner , err := IsOrganizationOwner ( orgID , uid ) ; owner || err != nil {
return owner , err
}
2021-09-23 18:45:36 +03:00
return db . GetEngine ( db . DefaultContext ) .
2019-11-20 14:27:49 +03:00
Where ( builder . Eq { "team.can_create_org_repo" : true } ) .
Join ( "INNER" , "team_user" , "team_user.team_id = team.id" ) .
And ( "team_user.uid = ?" , uid ) .
And ( "team_user.org_id = ?" , orgID ) .
Exist ( new ( Team ) )
}
2021-10-12 13:47:19 +03:00
// GetOrgUserMaxAuthorizeLevel returns highest authorize level of user in an organization
2021-11-28 14:58:28 +03:00
func ( org * Organization ) GetOrgUserMaxAuthorizeLevel ( uid int64 ) ( perm . AccessMode , error ) {
var authorize perm . AccessMode
2021-10-12 13:47:19 +03:00
_ , err := db . GetEngine ( db . DefaultContext ) .
Select ( "max(team.authorize)" ) .
Table ( "team" ) .
Join ( "INNER" , "team_user" , "team_user.team_id = team.id" ) .
Where ( "team_user.uid = ?" , uid ) .
And ( "team_user.org_id = ?" , org . ID ) .
Get ( & authorize )
return authorize , err
}
2021-03-01 03:47:30 +03:00
// GetUsersWhoCanCreateOrgRepo returns users which are able to create repo in organization
2021-11-24 12:49:20 +03:00
func GetUsersWhoCanCreateOrgRepo ( orgID int64 ) ( [ ] * user_model . User , error ) {
2021-09-23 18:45:36 +03:00
return getUsersWhoCanCreateOrgRepo ( db . GetEngine ( db . DefaultContext ) , orgID )
2021-03-01 03:47:30 +03:00
}
2021-11-24 12:49:20 +03:00
func getUsersWhoCanCreateOrgRepo ( e db . Engine , orgID int64 ) ( [ ] * user_model . User , error ) {
users := make ( [ ] * user_model . User , 0 , 10 )
2021-11-09 11:54:46 +03:00
return users , e .
2021-03-01 03:47:30 +03:00
Join ( "INNER" , "`team_user`" , "`team_user`.uid=`user`.id" ) .
Join ( "INNER" , "`team`" , "`team`.id=`team_user`.team_id" ) .
2021-11-28 14:58:28 +03:00
Where ( builder . Eq { "team.can_create_org_repo" : true } . Or ( builder . Eq { "team.authorize" : perm . AccessModeOwner } ) ) .
2021-03-01 03:47:30 +03:00
And ( "team_user.org_id = ?" , orgID ) . Asc ( "`user`.name" ) . Find ( & users )
}
2021-06-14 15:18:09 +03:00
// MinimalOrg represents a simple orgnization with only needed columns
2021-11-19 14:41:40 +03:00
type MinimalOrg = Organization
2021-06-14 15:18:09 +03:00
// GetUserOrgsList returns one user's all orgs list
2021-11-24 12:49:20 +03:00
func GetUserOrgsList ( user * user_model . User ) ( [ ] * MinimalOrg , error ) {
schema , err := db . TableInfo ( new ( user_model . User ) )
2021-09-01 08:31:42 +03:00
if err != nil {
return nil , err
}
outputCols := [ ] string {
"id" ,
"name" ,
"full_name" ,
"visibility" ,
"avatar" ,
"avatar_email" ,
"use_custom_avatar" ,
}
groupByCols := & strings . Builder { }
for _ , col := range outputCols {
fmt . Fprintf ( groupByCols , "`%s`.%s," , schema . Name , col )
}
groupByStr := groupByCols . String ( )
groupByStr = groupByStr [ 0 : len ( groupByStr ) - 1 ]
2021-11-21 18:41:00 +03:00
sess := db . GetEngine ( db . DefaultContext )
sess = sess . Select ( groupByStr + ", count(distinct repo_id) as org_count" ) .
2021-06-14 15:18:09 +03:00
Table ( "user" ) .
2021-09-01 08:31:42 +03:00
Join ( "INNER" , "team" , "`team`.org_id = `user`.id" ) .
Join ( "INNER" , "team_user" , "`team`.id = `team_user`.team_id" ) .
Join ( "LEFT" , builder .
Select ( "id as repo_id, owner_id as repo_owner_id" ) .
From ( "repository" ) .
Where ( accessibleRepositoryCondition ( user ) ) , "`repository`.repo_owner_id = `team`.org_id" ) .
Where ( "`team_user`.uid = ?" , user . ID ) .
GroupBy ( groupByStr )
type OrgCount struct {
2021-11-19 14:41:40 +03:00
Organization ` xorm:"extends" `
OrgCount int
2021-09-01 08:31:42 +03:00
}
orgCounts := make ( [ ] * OrgCount , 0 , 10 )
if err := sess .
Asc ( "`user`.name" ) .
Find ( & orgCounts ) ; err != nil {
return nil , err
}
orgs := make ( [ ] * MinimalOrg , len ( orgCounts ) )
for i , orgCount := range orgCounts {
2021-11-19 14:41:40 +03:00
orgCount . Organization . NumRepos = orgCount . OrgCount
orgs [ i ] = & orgCount . Organization
2021-09-01 08:31:42 +03:00
}
return orgs , nil
2021-06-14 15:18:09 +03:00
}
2021-11-24 12:49:20 +03:00
// SearchOrganizationsOptions options to filter organizations
type SearchOrganizationsOptions struct {
db . ListOptions
All bool
}
2021-11-22 16:51:45 +03:00
// FindOrgOptions finds orgs options
type FindOrgOptions struct {
db . ListOptions
UserID int64
IncludePrivate bool
}
func queryUserOrgIDs ( userID int64 , includePrivate bool ) * builder . Builder {
2022-01-05 06:37:00 +03:00
cond := builder . Eq { "uid" : userID }
2021-11-22 16:51:45 +03:00
if ! includePrivate {
cond [ "is_public" ] = true
}
return builder . Select ( "org_id" ) . From ( "org_user" ) . Where ( cond )
}
func ( opts FindOrgOptions ) toConds ( ) builder . Cond {
2022-01-05 06:37:00 +03:00
cond := builder . NewCond ( )
2021-11-22 16:51:45 +03:00
if opts . UserID > 0 {
cond = cond . And ( builder . In ( "`user`.`id`" , queryUserOrgIDs ( opts . UserID , opts . IncludePrivate ) ) )
}
if ! opts . IncludePrivate {
cond = cond . And ( builder . Eq { "`user`.visibility" : structs . VisibleTypePublic } )
}
return cond
}
// FindOrgs returns a list of organizations according given conditions
func FindOrgs ( opts FindOrgOptions ) ( [ ] * Organization , error ) {
orgs := make ( [ ] * Organization , 0 , 10 )
sess := db . GetEngine ( db . DefaultContext ) .
Where ( opts . toConds ( ) ) .
Asc ( "`user`.name" )
if opts . Page > 0 && opts . PageSize > 0 {
sess . Limit ( opts . PageSize , opts . PageSize * ( opts . Page - 1 ) )
}
return orgs , sess . Find ( & orgs )
}
// CountOrgs returns total count organizations according options
func CountOrgs ( opts FindOrgOptions ) ( int64 , error ) {
return db . GetEngine ( db . DefaultContext ) .
Where ( opts . toConds ( ) ) .
2021-11-24 12:49:20 +03:00
Count ( new ( user_model . User ) )
2021-11-22 16:51:45 +03:00
}
2021-11-22 18:21:55 +03:00
func getOwnedOrgsByUserID ( sess db . Engine , userID int64 ) ( [ ] * Organization , error ) {
orgs := make ( [ ] * Organization , 0 , 10 )
2016-11-10 11:10:35 +03:00
return orgs , sess .
2018-01-08 10:48:37 +03:00
Join ( "INNER" , "`team_user`" , "`team_user`.org_id=`user`.id" ) .
Join ( "INNER" , "`team`" , "`team`.id=`team_user`.team_id" ) .
Where ( "`team_user`.uid=?" , userID ) .
2021-11-28 14:58:28 +03:00
And ( "`team`.authorize=?" , perm . AccessModeOwner ) .
2016-11-10 11:10:35 +03:00
Asc ( "`user`.name" ) .
Find ( & orgs )
2015-09-07 20:58:23 +03:00
}
2021-06-26 22:53:14 +03:00
// HasOrgOrUserVisible tells if the given user can see the given org or user
2021-11-24 12:49:20 +03:00
func HasOrgOrUserVisible ( org , user * user_model . User ) bool {
2021-09-23 18:45:36 +03:00
return hasOrgOrUserVisible ( db . GetEngine ( db . DefaultContext ) , org , user )
2019-04-25 21:59:10 +03:00
}
2021-11-24 12:49:20 +03:00
func hasOrgOrUserVisible ( e db . Engine , orgOrUser , user * user_model . User ) bool {
2019-02-18 19:00:27 +03:00
// Not SignedUser
if user == nil {
2021-06-26 22:53:14 +03:00
return orgOrUser . Visibility == structs . VisibleTypePublic
2019-02-18 19:00:27 +03:00
}
2021-06-26 22:53:14 +03:00
if user . IsAdmin || orgOrUser . ID == user . ID {
2019-02-18 19:00:27 +03:00
return true
}
2021-11-19 14:41:40 +03:00
if ( orgOrUser . Visibility == structs . VisibleTypePrivate || user . IsRestricted ) && ! OrgFromUser ( orgOrUser ) . hasMemberWithUserID ( e , user . ID ) {
2019-02-18 19:00:27 +03:00
return false
}
return true
}
// HasOrgsVisible tells if the given user can see at least one of the orgs provided
2021-11-24 12:49:20 +03:00
func HasOrgsVisible ( orgs [ ] * Organization , user * user_model . User ) bool {
2019-02-18 19:00:27 +03:00
if len ( orgs ) == 0 {
return false
}
for _ , org := range orgs {
2021-11-19 14:41:40 +03:00
if HasOrgOrUserVisible ( org . AsUser ( ) , user ) {
2019-02-18 19:00:27 +03:00
return true
}
}
return false
}
2015-09-07 20:58:23 +03:00
// GetOwnedOrgsByUserID returns a list of organizations are owned by given user ID.
2021-11-22 18:21:55 +03:00
func GetOwnedOrgsByUserID ( userID int64 ) ( [ ] * Organization , error ) {
2021-11-21 18:41:00 +03:00
return getOwnedOrgsByUserID ( db . GetEngine ( db . DefaultContext ) , userID )
2015-09-07 20:58:23 +03:00
}
2016-11-26 03:20:18 +03:00
// GetOwnedOrgsByUserIDDesc returns a list of organizations are owned by
2016-01-31 00:51:11 +03:00
// given user ID, ordered descending by the given condition.
2021-11-22 18:21:55 +03:00
func GetOwnedOrgsByUserIDDesc ( userID int64 , desc string ) ( [ ] * Organization , error ) {
2021-09-23 18:45:36 +03:00
return getOwnedOrgsByUserID ( db . GetEngine ( db . DefaultContext ) . Desc ( desc ) , userID )
2015-09-07 20:58:23 +03:00
}
2019-11-20 14:27:49 +03:00
// GetOrgsCanCreateRepoByUserID returns a list of organizations where given user ID
// are allowed to create repos.
2021-11-22 18:21:55 +03:00
func GetOrgsCanCreateRepoByUserID ( userID int64 ) ( [ ] * Organization , error ) {
orgs := make ( [ ] * Organization , 0 , 10 )
2019-11-20 14:27:49 +03:00
2021-09-23 18:45:36 +03:00
return orgs , db . GetEngine ( db . DefaultContext ) . Where ( builder . In ( "id" , builder . Select ( "`user`.id" ) . From ( "`user`" ) .
2020-05-07 16:24:59 +03:00
Join ( "INNER" , "`team_user`" , "`team_user`.org_id = `user`.id" ) .
Join ( "INNER" , "`team`" , "`team`.id = `team_user`.team_id" ) .
Where ( builder . Eq { "`team_user`.uid" : userID } ) .
2021-11-28 14:58:28 +03:00
And ( builder . Eq { "`team`.authorize" : perm . AccessModeOwner } . Or ( builder . Eq { "`team`.can_create_org_repo" : true } ) ) ) ) .
2020-10-16 01:15:55 +03:00
Asc ( "`user`.name" ) .
Find ( & orgs )
2019-11-20 14:27:49 +03:00
}
2015-12-17 10:28:47 +03:00
// GetOrgUsersByUserID returns all organization-user relations by user ID.
2020-01-24 22:00:29 +03:00
func GetOrgUsersByUserID ( uid int64 , opts * SearchOrganizationsOptions ) ( [ ] * OrgUser , error ) {
2014-06-25 08:44:48 +04:00
ous := make ( [ ] * OrgUser , 0 , 10 )
2021-09-23 18:45:36 +03:00
sess := db . GetEngine ( db . DefaultContext ) .
2018-07-20 05:10:17 +03:00
Join ( "LEFT" , "`user`" , "`org_user`.org_id=`user`.id" ) .
2016-11-11 19:55:06 +03:00
Where ( "`org_user`.uid=?" , uid )
2020-01-24 22:00:29 +03:00
if ! opts . All {
2015-12-17 10:28:47 +03:00
// Only show public organizations
sess . And ( "is_public=?" , true )
}
2020-01-24 22:00:29 +03:00
if opts . PageSize != 0 {
2021-09-24 14:32:56 +03:00
sess = db . SetSessionPagination ( sess , opts )
2020-01-24 22:00:29 +03:00
}
2016-11-10 11:10:35 +03:00
err := sess .
Asc ( "`user`.name" ) .
Find ( & ous )
2014-06-25 08:44:48 +04:00
return ous , err
}
2016-07-15 19:36:39 +03:00
// GetOrgUsersByOrgID returns all organization-user relations by organization ID.
2020-01-24 22:00:29 +03:00
func GetOrgUsersByOrgID ( opts * FindOrgMembersOpts ) ( [ ] * OrgUser , error ) {
2021-09-23 18:45:36 +03:00
return getOrgUsersByOrgID ( db . GetEngine ( db . DefaultContext ) , opts )
2019-01-05 00:51:27 +03:00
}
2021-09-19 14:49:59 +03:00
func getOrgUsersByOrgID ( e db . Engine , opts * FindOrgMembersOpts ) ( [ ] * OrgUser , error ) {
2020-01-24 22:00:29 +03:00
sess := e . Where ( "org_id=?" , opts . OrgID )
if opts . PublicOnly {
2019-12-06 08:34:54 +03:00
sess . And ( "is_public = ?" , true )
}
2020-01-24 22:00:29 +03:00
if opts . ListOptions . PageSize > 0 {
2021-09-24 14:32:56 +03:00
sess = db . SetSessionPagination ( sess , opts )
2020-01-24 22:00:29 +03:00
ous := make ( [ ] * OrgUser , 0 , opts . PageSize )
return ous , sess . Find ( & ous )
2019-12-06 08:34:54 +03:00
}
2020-01-24 22:00:29 +03:00
var ous [ ] * OrgUser
return ous , sess . Find ( & ous )
2014-06-25 13:14:36 +04:00
}
2014-08-15 14:29:41 +04:00
// ChangeOrgUserStatus changes public or private membership status.
2016-07-24 09:32:46 +03:00
func ChangeOrgUserStatus ( orgID , uid int64 , public bool ) error {
2014-08-15 14:29:41 +04:00
ou := new ( OrgUser )
2021-09-23 18:45:36 +03:00
has , err := db . GetEngine ( db . DefaultContext ) .
2016-11-10 18:16:32 +03:00
Where ( "uid=?" , uid ) .
And ( "org_id=?" , orgID ) .
Get ( ou )
2014-08-15 14:29:41 +04:00
if err != nil {
return err
} else if ! has {
return nil
}
ou . IsPublic = public
2021-09-23 18:45:36 +03:00
_ , err = db . GetEngine ( db . DefaultContext ) . ID ( ou . ID ) . Cols ( "is_public" ) . Update ( ou )
2014-08-15 14:29:41 +04:00
return err
}
// AddOrgUser adds new user to given organization.
2016-07-24 09:32:46 +03:00
func AddOrgUser ( orgID , uid int64 ) error {
2017-12-21 10:43:26 +03:00
isAlreadyMember , err := IsOrganizationMember ( orgID , uid )
if err != nil || isAlreadyMember {
return err
2014-08-15 14:29:41 +04:00
}
2021-11-21 18:41:00 +03:00
ctx , committer , err := db . TxContext ( )
if err != nil {
2014-08-15 14:29:41 +04:00
return err
}
2021-11-21 18:41:00 +03:00
defer committer . Close ( )
2014-08-15 14:29:41 +04:00
2014-08-24 17:09:05 +04:00
ou := & OrgUser {
2019-08-24 15:28:59 +03:00
UID : uid ,
OrgID : orgID ,
IsPublic : setting . Service . DefaultOrgMemberVisible ,
2014-08-24 17:09:05 +04:00
}
2021-11-21 18:41:00 +03:00
if err := db . Insert ( ctx , ou ) ; err != nil {
2014-08-15 14:29:41 +04:00
return err
2021-11-21 18:41:00 +03:00
} else if _ , err = db . Exec ( ctx , "UPDATE `user` SET num_members = num_members + 1 WHERE id = ?" , orgID ) ; err != nil {
2014-08-15 14:29:41 +04:00
return err
}
2021-11-21 18:41:00 +03:00
return committer . Commit ( )
2014-08-15 14:29:41 +04:00
}
2021-11-19 14:41:40 +03:00
// GetOrgByIDCtx returns the user object by given ID if exists.
func GetOrgByIDCtx ( ctx context . Context , id int64 ) ( * Organization , error ) {
u := new ( Organization )
has , err := db . GetEngine ( ctx ) . ID ( id ) . Get ( u )
if err != nil {
return nil , err
} else if ! has {
2021-11-24 12:49:20 +03:00
return nil , user_model . ErrUserNotExist {
UID : id ,
Name : "" ,
KeyID : 0 ,
}
2021-11-19 14:41:40 +03:00
}
return u , nil
}
// GetOrgByID returns the user object by given ID if exists.
func GetOrgByID ( id int64 ) ( * Organization , error ) {
return GetOrgByIDCtx ( db . DefaultContext , id )
}
func removeOrgUser ( ctx context . Context , orgID , userID int64 ) error {
2014-08-15 14:29:41 +04:00
ou := new ( OrgUser )
2021-11-19 14:41:40 +03:00
sess := db . GetEngine ( ctx )
2018-07-27 02:41:36 +03:00
has , err := sess .
2016-11-10 18:16:32 +03:00
Where ( "uid=?" , userID ) .
And ( "org_id=?" , orgID ) .
Get ( ou )
2014-08-15 14:29:41 +04:00
if err != nil {
2015-03-18 04:51:39 +03:00
return fmt . Errorf ( "get org-user: %v" , err )
2014-08-15 14:29:41 +04:00
} else if ! has {
return nil
}
2021-11-19 14:41:40 +03:00
org , err := GetOrgByIDCtx ( ctx , orgID )
2014-08-24 17:09:05 +04:00
if err != nil {
2016-07-24 09:32:46 +03:00
return fmt . Errorf ( "GetUserByID [%d]: %v" , orgID , err )
2014-08-24 17:09:05 +04:00
}
2016-07-24 09:32:46 +03:00
2014-08-16 12:21:17 +04:00
// Check if the user to delete is the last member in owner team.
2018-10-29 16:48:37 +03:00
if isOwner , err := isOrganizationOwner ( sess , orgID , userID ) ; err != nil {
2017-12-21 10:43:26 +03:00
return err
} else if isOwner {
2018-10-29 16:48:37 +03:00
t , err := org . getOwnerTeam ( sess )
2014-08-16 12:21:17 +04:00
if err != nil {
return err
}
if t . NumMembers == 1 {
2018-10-29 16:48:37 +03:00
if err := t . getMembers ( sess ) ; err != nil {
2017-12-13 01:26:31 +03:00
return err
}
if t . Members [ 0 ] . ID == userID {
return ErrLastOrgOwner { UID : userID }
}
2014-08-16 12:21:17 +04:00
}
}
2017-10-05 07:43:04 +03:00
if _ , err := sess . ID ( ou . ID ) . Delete ( ou ) ; err != nil {
2014-08-15 14:29:41 +04:00
return err
2016-07-24 09:32:46 +03:00
} else if _ , err = sess . Exec ( "UPDATE `user` SET num_members=num_members-1 WHERE id=?" , orgID ) ; err != nil {
2014-08-15 14:29:41 +04:00
return err
}
2016-07-24 09:32:46 +03:00
// Delete all repository accesses and unwatch them.
2018-10-29 16:48:37 +03:00
env , err := org . accessibleReposEnv ( sess , userID )
2017-01-25 18:41:38 +03:00
if err != nil {
return fmt . Errorf ( "AccessibleReposEnv: %v" , err )
}
repoIDs , err := env . RepoIDs ( 1 , org . NumRepos )
if err != nil {
2017-01-24 19:16:36 +03:00
return fmt . Errorf ( "GetUserRepositories [%d]: %v" , userID , err )
2017-01-25 18:41:38 +03:00
}
for _ , repoID := range repoIDs {
2021-12-12 18:48:20 +03:00
if err = repo_model . WatchRepoCtx ( ctx , userID , repoID , false ) ; err != nil {
2014-08-27 12:39:36 +04:00
return err
2014-08-24 17:09:05 +04:00
}
}
2016-08-10 21:06:51 +03:00
if len ( repoIDs ) > 0 {
2016-11-10 18:16:32 +03:00
if _ , err = sess .
2017-01-24 19:16:36 +03:00
Where ( "user_id = ?" , userID ) .
2016-11-10 18:16:32 +03:00
In ( "repo_id" , repoIDs ) .
Delete ( new ( Access ) ) ; err != nil {
2016-08-10 21:06:51 +03:00
return err
}
2016-07-24 09:32:46 +03:00
}
2014-08-24 17:09:05 +04:00
// Delete member in his/her teams.
2019-01-17 03:39:50 +03:00
teams , err := getUserOrgTeams ( sess , org . ID , userID )
2014-08-24 17:09:05 +04:00
if err != nil {
return err
}
2015-03-18 04:51:39 +03:00
for _ , t := range teams {
2021-11-19 14:41:40 +03:00
if err = removeTeamMember ( ctx , t , userID ) ; err != nil {
2014-08-24 17:09:05 +04:00
return err
}
}
2018-02-23 11:42:02 +03:00
return nil
}
// RemoveOrgUser removes user from given organization.
func RemoveOrgUser ( orgID , userID int64 ) error {
2021-11-19 14:41:40 +03:00
ctx , committer , err := db . TxContext ( )
if err != nil {
2018-02-23 11:42:02 +03:00
return err
}
2021-11-19 14:41:40 +03:00
defer committer . Close ( )
if err := removeOrgUser ( ctx , orgID , userID ) ; err != nil {
2018-02-23 11:42:02 +03:00
return err
}
2021-11-19 14:41:40 +03:00
return committer . Commit ( )
2014-08-15 14:29:41 +04:00
}
2021-09-19 14:49:59 +03:00
func removeOrgRepo ( e db . Engine , orgID , repoID int64 ) error {
2017-02-04 19:00:07 +03:00
teamRepos := make ( [ ] * TeamRepo , 0 , 10 )
if err := e . Find ( & teamRepos , & TeamRepo { OrgID : orgID , RepoID : repoID } ) ; err != nil {
return err
}
if len ( teamRepos ) == 0 {
return nil
}
if _ , err := e . Delete ( & TeamRepo {
2015-03-01 05:44:09 +03:00
OrgID : orgID ,
RepoID : repoID ,
2017-02-04 19:00:07 +03:00
} ) ; err != nil {
return err
}
teamIDs := make ( [ ] int64 , len ( teamRepos ) )
for i , teamRepo := range teamRepos {
2017-06-03 04:06:10 +03:00
teamIDs [ i ] = teamRepo . TeamID
2017-02-04 19:00:07 +03:00
}
2017-06-01 03:41:14 +03:00
_ , err := e . Decr ( "num_repos" ) . In ( "id" , teamIDs ) . Update ( new ( Team ) )
2015-03-01 05:44:09 +03:00
return err
}
2021-11-19 14:41:40 +03:00
func ( org * Organization ) getUserTeams ( e db . Engine , userID int64 , cols ... string ) ( [ ] * Team , error ) {
2016-07-24 13:09:45 +03:00
teams := make ( [ ] * Team , 0 , org . NumTeams )
2016-11-10 11:10:35 +03:00
return teams , e .
Where ( "`team_user`.org_id = ?" , org . ID ) .
Join ( "INNER" , "team_user" , "`team_user`.team_id = team.id" ) .
2018-07-20 05:10:17 +03:00
Join ( "INNER" , "`user`" , "`user`.id=team_user.uid" ) .
2016-11-10 11:10:35 +03:00
And ( "`team_user`.uid = ?" , userID ) .
Asc ( "`user`.name" ) .
Cols ( cols ... ) .
2016-11-10 18:16:32 +03:00
Find ( & teams )
2016-07-24 13:09:45 +03:00
}
2021-11-19 14:41:40 +03:00
func ( org * Organization ) getUserTeamIDs ( e db . Engine , userID int64 ) ( [ ] int64 , error ) {
2017-02-14 06:46:46 +03:00
teamIDs := make ( [ ] int64 , 0 , org . NumTeams )
return teamIDs , e .
Table ( "team" ) .
Cols ( "team.id" ) .
Where ( "`team_user`.org_id = ?" , org . ID ) .
Join ( "INNER" , "team_user" , "`team_user`.team_id = team.id" ) .
And ( "`team_user`.uid = ?" , userID ) .
Find ( & teamIDs )
}
2020-06-22 18:21:11 +03:00
// TeamsWithAccessToRepo returns all teams that have given access level to the repository.
2021-11-28 14:58:28 +03:00
func ( org * Organization ) TeamsWithAccessToRepo ( repoID int64 , mode perm . AccessMode ) ( [ ] * Team , error ) {
2017-09-14 11:16:22 +03:00
return GetTeamsWithAccessToRepo ( org . ID , repoID , mode )
}
2016-11-27 14:59:12 +03:00
// GetUserTeamIDs returns of all team IDs of the organization that user is member of.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) GetUserTeamIDs ( userID int64 ) ( [ ] int64 , error ) {
2021-09-23 18:45:36 +03:00
return org . getUserTeamIDs ( db . GetEngine ( db . DefaultContext ) , userID )
2016-07-24 09:32:46 +03:00
}
2016-11-26 03:20:18 +03:00
// GetUserTeams returns all teams that belong to user,
2016-07-24 13:09:45 +03:00
// and that the user has joined.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) GetUserTeams ( userID int64 ) ( [ ] * Team , error ) {
2021-09-23 18:45:36 +03:00
return org . getUserTeams ( db . GetEngine ( db . DefaultContext ) , userID )
2016-07-24 13:09:45 +03:00
}
2017-01-25 18:41:38 +03:00
// AccessibleReposEnvironment operations involving the repositories that are
// accessible to a particular user
type AccessibleReposEnvironment interface {
CountRepos ( ) ( int64 , error )
RepoIDs ( page , pageSize int ) ( [ ] int64 , error )
2021-12-10 04:27:50 +03:00
Repos ( page , pageSize int ) ( [ ] * repo_model . Repository , error )
MirrorRepos ( ) ( [ ] * repo_model . Repository , error )
2019-02-08 19:45:43 +03:00
AddKeyword ( keyword string )
2021-11-24 12:49:20 +03:00
SetSort ( db . SearchOrderBy )
2017-01-25 18:41:38 +03:00
}
2016-12-29 11:53:33 +03:00
2017-01-25 18:41:38 +03:00
type accessibleReposEnv struct {
2021-11-19 14:41:40 +03:00
org * Organization
2021-11-24 12:49:20 +03:00
user * user_model . User
2020-12-27 22:58:03 +03:00
team * Team
2017-01-25 18:41:38 +03:00
teamIDs [ ] int64
2021-09-19 14:49:59 +03:00
e db . Engine
2019-02-08 19:45:43 +03:00
keyword string
2021-11-24 12:49:20 +03:00
orderBy db . SearchOrderBy
2017-01-25 18:41:38 +03:00
}
2021-01-13 07:19:17 +03:00
// AccessibleReposEnv builds an AccessibleReposEnvironment for the repositories in `org`
2017-01-25 18:41:38 +03:00
// that are accessible to the specified user.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) AccessibleReposEnv ( userID int64 ) ( AccessibleReposEnvironment , error ) {
2021-09-23 18:45:36 +03:00
return org . accessibleReposEnv ( db . GetEngine ( db . DefaultContext ) , userID )
2018-10-29 16:48:37 +03:00
}
2021-11-19 14:41:40 +03:00
func ( org * Organization ) accessibleReposEnv ( e db . Engine , userID int64 ) ( AccessibleReposEnvironment , error ) {
2021-11-24 12:49:20 +03:00
var user * user_model . User
2020-01-13 20:33:46 +03:00
if userID > 0 {
2021-11-24 12:49:20 +03:00
u , err := user_model . GetUserByIDEngine ( e , userID )
2020-01-13 20:33:46 +03:00
if err != nil {
return nil , err
}
user = u
}
2018-10-29 16:48:37 +03:00
teamIDs , err := org . getUserTeamIDs ( e , userID )
2016-07-24 09:32:46 +03:00
if err != nil {
2017-01-25 18:41:38 +03:00
return nil , err
2016-01-31 21:13:39 +03:00
}
2018-10-29 16:48:37 +03:00
return & accessibleReposEnv {
org : org ,
2020-01-13 20:33:46 +03:00
user : user ,
2018-10-29 16:48:37 +03:00
teamIDs : teamIDs ,
e : e ,
2021-11-24 12:49:20 +03:00
orderBy : db . SearchOrderByRecentUpdated ,
2018-10-29 16:48:37 +03:00
} , nil
2017-01-25 18:41:38 +03:00
}
2016-12-29 11:53:33 +03:00
2020-12-27 22:58:03 +03:00
// AccessibleTeamReposEnv an AccessibleReposEnvironment for the repositories in `org`
// that are accessible to the specified team.
2021-11-19 14:41:40 +03:00
func ( org * Organization ) AccessibleTeamReposEnv ( team * Team ) AccessibleReposEnvironment {
2020-12-27 22:58:03 +03:00
return & accessibleReposEnv {
org : org ,
team : team ,
2021-09-23 18:45:36 +03:00
e : db . GetEngine ( db . DefaultContext ) ,
2021-11-24 12:49:20 +03:00
orderBy : db . SearchOrderByRecentUpdated ,
2020-12-27 22:58:03 +03:00
}
}
2017-01-25 18:41:38 +03:00
func ( env * accessibleReposEnv ) cond ( ) builder . Cond {
2021-03-14 21:52:12 +03:00
cond := builder . NewCond ( )
2020-12-27 22:58:03 +03:00
if env . team != nil {
cond = cond . And ( builder . Eq { "team_repo.team_id" : env . team . ID } )
} else {
if env . user == nil || ! env . user . IsRestricted {
cond = cond . Or ( builder . Eq {
"`repository`.owner_id" : env . org . ID ,
"`repository`.is_private" : false ,
} )
}
if len ( env . teamIDs ) > 0 {
cond = cond . Or ( builder . In ( "team_repo.team_id" , env . teamIDs ) )
}
2017-01-25 18:41:38 +03:00
}
2019-02-08 19:45:43 +03:00
if env . keyword != "" {
cond = cond . And ( builder . Like { "`repository`.lower_name" , strings . ToLower ( env . keyword ) } )
}
2017-01-25 18:41:38 +03:00
return cond
}
2016-01-31 21:13:39 +03:00
2017-01-25 18:41:38 +03:00
func ( env * accessibleReposEnv ) CountRepos ( ) ( int64 , error ) {
2018-10-29 16:48:37 +03:00
repoCount , err := env . e .
2017-01-25 18:41:38 +03:00
Join ( "INNER" , "team_repo" , "`team_repo`.repo_id=`repository`.id" ) .
Where ( env . cond ( ) ) .
2017-01-24 19:16:36 +03:00
Distinct ( "`repository`.id" ) .
2021-12-10 04:27:50 +03:00
Count ( & repo_model . Repository { } )
2017-01-25 18:41:38 +03:00
if err != nil {
return 0 , fmt . Errorf ( "count user repositories in organization: %v" , err )
}
return repoCount , nil
}
func ( env * accessibleReposEnv ) RepoIDs ( page , pageSize int ) ( [ ] int64 , error ) {
2016-07-24 09:32:46 +03:00
if page <= 0 {
page = 1
}
2017-01-14 17:32:36 +03:00
2017-01-25 18:41:38 +03:00
repoIDs := make ( [ ] int64 , 0 , pageSize )
2018-10-29 16:48:37 +03:00
return repoIDs , env . e .
2017-01-25 18:41:38 +03:00
Table ( "repository" ) .
2016-11-10 18:16:32 +03:00
Join ( "INNER" , "team_repo" , "`team_repo`.repo_id=`repository`.id" ) .
2017-01-25 18:41:38 +03:00
Where ( env . cond ( ) ) .
2019-04-24 21:20:22 +03:00
GroupBy ( "`repository`.id,`repository`." + strings . Fields ( string ( env . orderBy ) ) [ 0 ] ) .
OrderBy ( string ( env . orderBy ) ) .
2016-11-10 18:16:32 +03:00
Limit ( pageSize , ( page - 1 ) * pageSize ) .
2017-01-25 18:41:38 +03:00
Cols ( "`repository`.id" ) .
Find ( & repoIDs )
}
2016-07-24 09:32:46 +03:00
2021-12-10 04:27:50 +03:00
func ( env * accessibleReposEnv ) Repos ( page , pageSize int ) ( [ ] * repo_model . Repository , error ) {
2017-01-25 18:41:38 +03:00
repoIDs , err := env . RepoIDs ( page , pageSize )
2016-07-24 09:32:46 +03:00
if err != nil {
2017-01-25 18:41:38 +03:00
return nil , fmt . Errorf ( "GetUserRepositoryIDs: %v" , err )
2016-01-31 13:46:04 +03:00
}
2021-12-10 04:27:50 +03:00
repos := make ( [ ] * repo_model . Repository , 0 , len ( repoIDs ) )
2018-10-21 00:25:14 +03:00
if len ( repoIDs ) == 0 {
2017-02-14 18:37:44 +03:00
return repos , nil
}
2018-10-29 16:48:37 +03:00
return repos , env . e .
2017-02-14 18:37:44 +03:00
In ( "`repository`.id" , repoIDs ) .
2019-04-24 21:20:22 +03:00
OrderBy ( string ( env . orderBy ) ) .
2017-01-25 18:41:38 +03:00
Find ( & repos )
2016-07-24 09:32:46 +03:00
}
2017-04-07 04:47:25 +03:00
func ( env * accessibleReposEnv ) MirrorRepoIDs ( ) ( [ ] int64 , error ) {
repoIDs := make ( [ ] int64 , 0 , 10 )
2018-10-29 16:48:37 +03:00
return repoIDs , env . e .
2017-04-07 04:47:25 +03:00
Table ( "repository" ) .
2016-11-10 18:16:32 +03:00
Join ( "INNER" , "team_repo" , "`team_repo`.repo_id=`repository`.id AND `repository`.is_mirror=?" , true ) .
2017-01-25 18:41:38 +03:00
Where ( env . cond ( ) ) .
2017-09-15 09:14:06 +03:00
GroupBy ( "`repository`.id, `repository`.updated_unix" ) .
2019-04-24 21:20:22 +03:00
OrderBy ( string ( env . orderBy ) ) .
2017-04-07 04:47:25 +03:00
Cols ( "`repository`.id" ) .
Find ( & repoIDs )
}
2021-12-10 04:27:50 +03:00
func ( env * accessibleReposEnv ) MirrorRepos ( ) ( [ ] * repo_model . Repository , error ) {
2017-04-07 04:47:25 +03:00
repoIDs , err := env . MirrorRepoIDs ( )
if err != nil {
return nil , fmt . Errorf ( "MirrorRepoIDs: %v" , err )
}
2021-12-10 04:27:50 +03:00
repos := make ( [ ] * repo_model . Repository , 0 , len ( repoIDs ) )
2018-10-21 00:25:14 +03:00
if len ( repoIDs ) == 0 {
2017-04-07 04:47:25 +03:00
return repos , nil
}
2018-10-29 16:48:37 +03:00
return repos , env . e .
2017-04-07 04:47:25 +03:00
In ( "`repository`.id" , repoIDs ) .
2016-11-10 18:16:32 +03:00
Find ( & repos )
2016-01-31 13:46:04 +03:00
}
2019-02-08 19:45:43 +03:00
func ( env * accessibleReposEnv ) AddKeyword ( keyword string ) {
env . keyword = keyword
}
2019-04-24 21:20:22 +03:00
2021-11-24 12:49:20 +03:00
func ( env * accessibleReposEnv ) SetSort ( orderBy db . SearchOrderBy ) {
2019-04-24 21:20:22 +03:00
env . orderBy = orderBy
}