2014-06-25 00:44:48 -04:00
// Copyright 2014 The Gogs Authors. All rights reserved.
2019-02-18 17:00:27 +01:00
// Copyright 2019 The Gitea Authors. All rights reserved.
2014-06-25 00: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 03:37:01 -04:00
import (
2015-02-23 02:15:53 -05:00
"fmt"
2014-07-04 22:43:39 -04:00
"os"
2014-06-27 03:37:01 -04:00
"strings"
2015-09-07 13:58:23 -04:00
2018-01-07 23:48:37 -08:00
"code.gitea.io/gitea/modules/log"
2019-08-24 09:28:59 -03:00
"code.gitea.io/gitea/modules/setting"
2019-02-18 17:00:27 +01:00
"code.gitea.io/gitea/modules/structs"
2018-01-07 23:48:37 -08:00
2019-08-23 09:40:30 -07:00
"github.com/unknwon/com"
2019-06-23 23:22:43 +08:00
"xorm.io/builder"
2019-10-17 17:26:49 +08:00
"xorm.io/xorm"
2014-06-27 03:37:01 -04:00
)
2014-12-12 20:30:32 -05:00
// IsOwnedBy returns true if given user is in the owner team.
2017-12-20 23:43:26 -08:00
func ( org * User ) IsOwnedBy ( uid int64 ) ( bool , error ) {
2016-07-24 01:08:22 +08:00
return IsOrganizationOwner ( org . ID , uid )
2014-06-29 16:30:41 -04:00
}
// IsOrgMember returns true if given user is member of organization.
2017-12-20 23:43:26 -08:00
func ( org * User ) IsOrgMember ( uid int64 ) ( bool , error ) {
return IsOrganizationMember ( org . ID , uid )
2014-06-29 16:30:41 -04:00
}
2019-11-20 12:27:49 +01:00
// CanCreateOrgRepo returns true if given user can create repo in organization
func ( org * User ) CanCreateOrgRepo ( uid int64 ) ( bool , error ) {
return CanCreateOrgRepo ( org . ID , uid )
}
2015-02-13 00:58:46 -05:00
func ( org * User ) getTeam ( e Engine , name string ) ( * Team , error ) {
2016-07-24 01:08:22 +08:00
return getTeam ( e , org . ID , name )
2015-02-13 00:58:46 -05:00
}
2014-08-16 16:21:17 +08:00
// GetTeam returns named team of organization.
func ( org * User ) GetTeam ( name string ) ( * Team , error ) {
2015-02-13 00:58:46 -05:00
return org . getTeam ( x , name )
}
func ( org * User ) getOwnerTeam ( e Engine ) ( * Team , error ) {
2016-11-28 09:30:08 +08:00
return org . getTeam ( e , ownerTeamName )
2014-08-16 16:21:17 +08:00
}
2014-06-28 00:40:07 -04:00
// GetOwnerTeam returns owner team of organization.
func ( org * User ) GetOwnerTeam ( ) ( * Team , error ) {
2015-02-13 00:58:46 -05:00
return org . getOwnerTeam ( x )
}
func ( org * User ) getTeams ( e Engine ) error {
2019-11-06 10:37:14 +01:00
if org . Teams != nil {
return nil
}
2016-11-10 16:16:32 +01:00
return e .
Where ( "org_id=?" , org . ID ) .
2016-11-28 09:30:08 +08:00
OrderBy ( "CASE WHEN name LIKE '" + ownerTeamName + "' THEN '' ELSE name END" ) .
2016-11-10 16:16:32 +01:00
Find ( & org . Teams )
2014-06-28 00:40:07 -04:00
}
2020-01-24 19:00:29 +00:00
// GetTeams returns paginated teams that belong to organization.
func ( org * User ) GetTeams ( opts * SearchTeamOptions ) error {
if opts . Page != 0 {
return org . getTeams ( opts . getPaginatedSession ( ) )
}
2015-02-13 00:58:46 -05:00
return org . getTeams ( x )
2014-06-28 15:43:25 -04:00
}
// GetMembers returns all members of organization.
2019-12-06 13:34:54 +08:00
func ( org * User ) GetMembers ( ) ( err error ) {
2020-01-24 19:00:29 +00:00
org . Members , org . MembersIsPublic , err = FindOrgMembers ( & FindOrgMembersOpts {
2019-12-06 13:34:54 +08:00
OrgID : org . ID ,
} )
return
}
// FindOrgMembersOpts represensts find org members condtions
type FindOrgMembersOpts struct {
2020-01-24 19:00:29 +00:00
ListOptions
2019-12-06 13:34:54 +08:00
OrgID int64
PublicOnly bool
}
// CountOrgMembers counts the organization's members
func CountOrgMembers ( opts FindOrgMembersOpts ) ( int64 , error ) {
sess := x . Where ( "org_id=?" , opts . OrgID )
if opts . PublicOnly {
sess . And ( "is_public = ?" , true )
}
return sess . Count ( new ( OrgUser ) )
}
// FindOrgMembers loads organization members according conditions
2020-01-24 19:00:29 +00:00
func FindOrgMembers ( opts * FindOrgMembersOpts ) ( UserList , map [ int64 ] bool , error ) {
ous , err := GetOrgUsersByOrgID ( opts )
2014-06-28 15:43:25 -04:00
if err != nil {
2019-12-06 13:34:54 +08:00
return nil , nil , err
2014-06-28 15:43:25 -04:00
}
2016-11-10 09:10:35 +01:00
var ids = make ( [ ] int64 , len ( ous ) )
2019-08-02 18:06:28 +02:00
var idsIsPublic = make ( map [ int64 ] bool , len ( ous ) )
2014-06-28 15:43:25 -04:00
for i , ou := range ous {
2016-11-26 01:20:18 +01:00
ids [ i ] = ou . UID
2019-08-02 18:06:28 +02:00
idsIsPublic [ ou . UID ] = ou . IsPublic
2014-06-28 15:43:25 -04:00
}
2019-12-06 13:34:54 +08:00
users , err := GetUsersByIDs ( ids )
if err != nil {
return nil , nil , err
}
return users , idsIsPublic , nil
2014-06-28 15:43:25 -04:00
}
2014-08-15 18:29:41 +08:00
// AddMember adds new member to organization.
func ( org * User ) AddMember ( uid int64 ) error {
2016-07-24 01:08:22 +08:00
return AddOrgUser ( org . ID , uid )
2014-08-15 18:29:41 +08:00
}
// RemoveMember removes member from organization.
func ( org * User ) RemoveMember ( uid int64 ) error {
2016-07-24 01:08:22 +08:00
return RemoveOrgUser ( org . ID , uid )
2014-08-15 18:29:41 +08:00
}
2015-02-28 21:44:09 -05:00
func ( org * User ) removeOrgRepo ( e Engine , repoID int64 ) error {
2016-07-24 01:08:22 +08:00
return removeOrgRepo ( e , org . ID , repoID )
2015-02-28 21:44:09 -05:00
}
// RemoveOrgRepo removes all team-repository relations of organization.
func ( org * User ) RemoveOrgRepo ( repoID int64 ) error {
return org . removeOrgRepo ( x , repoID )
}
2014-06-27 03:37:01 -04:00
// CreateOrganization creates record of a new organization.
2015-03-26 17:11:47 -04:00
func CreateOrganization ( org , owner * User ) ( err error ) {
2016-12-31 03:33:30 +01:00
if ! owner . CanCreateOrganization ( ) {
return ErrUserNotAllowedCreateOrg { }
}
2016-07-23 18:58:18 +08:00
if err = IsUsableUsername ( org . Name ) ; err != nil {
2015-03-26 17:11:47 -04:00
return err
2014-06-27 03:37:01 -04:00
}
2015-02-22 18:24:49 -05:00
isExist , err := IsUserExist ( 0 , org . Name )
2014-06-27 03:37:01 -04:00
if err != nil {
2015-03-26 17:11:47 -04:00
return err
2014-06-27 03:37:01 -04:00
} else if isExist {
2015-03-26 17:11:47 -04:00
return ErrUserAlreadyExist { org . Name }
2014-06-27 03:37:01 -04:00
}
org . LowerName = strings . ToLower ( org . Name )
2016-12-20 14:32:02 +02:00
if org . Rands , err = GetUserSalt ( ) ; err != nil {
return err
}
if org . Salt , err = GetUserSalt ( ) ; err != nil {
return err
}
2015-09-06 10:08:14 -04:00
org . UseCustomAvatar = true
2015-12-11 15:48:02 -05:00
org . MaxRepoCreation = - 1
2014-06-27 03:37:01 -04:00
org . NumTeams = 1
org . NumMembers = 1
2017-01-24 11:16:36 -05:00
org . Type = UserTypeOrganization
2014-06-27 03:37:01 -04:00
sess := x . NewSession ( )
2017-06-21 03:57:05 +03:00
defer sess . Close ( )
2014-06-27 03:37:01 -04:00
if err = sess . Begin ( ) ; err != nil {
2015-03-26 17:11:47 -04:00
return err
2014-06-27 03:37:01 -04:00
}
if _ , err = sess . Insert ( org ) ; err != nil {
2015-03-26 17:11:47 -04:00
return fmt . Errorf ( "insert organization: %v" , err )
2014-07-04 22:43:39 -04:00
}
2017-03-08 23:05:15 +08:00
if err = org . generateRandomAvatar ( sess ) ; err != nil {
return fmt . Errorf ( "generate random avatar: %v" , err )
}
2015-09-06 10:08:14 -04:00
// Add initial creator to organization and owner team.
if _ , err = sess . Insert ( & OrgUser {
2018-01-07 23:48:37 -08:00
UID : owner . ID ,
OrgID : org . ID ,
2015-09-06 10:08:14 -04:00
} ) ; err != nil {
return fmt . Errorf ( "insert org-user relation: %v" , err )
}
2014-07-04 22:43:39 -04:00
2014-06-27 03:37:01 -04:00
// Create default owner team.
t := & Team {
2019-11-06 10:37:14 +01:00
OrgID : org . ID ,
LowerName : strings . ToLower ( ownerTeamName ) ,
Name : ownerTeamName ,
Authorize : AccessModeOwner ,
NumMembers : 1 ,
IncludesAllRepositories : true ,
2019-11-20 12:27:49 +01:00
CanCreateOrgRepo : true ,
2014-06-27 03:37:01 -04:00
}
if _ , err = sess . Insert ( t ) ; err != nil {
2015-03-26 17:11:47 -04:00
return fmt . Errorf ( "insert owner team: %v" , err )
2014-06-27 03:37:01 -04:00
}
2018-06-21 18:00:13 +02:00
// insert units for team
2019-05-30 11:09:05 -04:00
var units = make ( [ ] TeamUnit , 0 , len ( AllRepoUnitTypes ) )
for _ , tp := range AllRepoUnitTypes {
2018-06-21 18:00:13 +02:00
units = append ( units , TeamUnit {
OrgID : org . ID ,
TeamID : t . ID ,
Type : tp ,
} )
}
if _ , err = sess . Insert ( & units ) ; err != nil {
2019-06-12 21:41:28 +02:00
if err := sess . Rollback ( ) ; err != nil {
log . Error ( "CreateOrganization: sess.Rollback: %v" , err )
}
2018-06-21 18:00:13 +02:00
return err
}
2015-09-06 10:08:14 -04:00
if _ , err = sess . Insert ( & TeamUser {
2016-11-28 09:30:08 +08:00
UID : owner . ID ,
2016-07-24 01:08:22 +08:00
OrgID : org . ID ,
2015-02-23 02:15:53 -05:00
TeamID : t . ID ,
2015-09-06 10:08:14 -04:00
} ) ; err != nil {
2015-03-26 17:11:47 -04:00
return fmt . Errorf ( "insert team-user relation: %v" , err )
2015-02-23 02:15:53 -05:00
}
2015-03-26 17:11:47 -04:00
return sess . Commit ( )
2014-06-27 03:37:01 -04:00
}
2014-12-12 20:30:32 -05:00
// GetOrgByName returns organization by given name.
func GetOrgByName ( name string ) ( * User , error ) {
if len ( name ) == 0 {
2017-07-06 15:30:19 +02:00
return nil , ErrOrgNotExist { 0 , name }
2014-12-12 20:30:32 -05:00
}
u := & User {
LowerName : strings . ToLower ( name ) ,
2016-11-07 17:53:22 +01:00
Type : UserTypeOrganization ,
2014-12-12 20:30:32 -05:00
}
has , err := x . Get ( u )
if err != nil {
return nil , err
} else if ! has {
2017-07-06 15:30:19 +02:00
return nil , ErrOrgNotExist { 0 , name }
2014-12-12 20:30:32 -05:00
}
return u , nil
}
2014-08-28 22:29:00 +08:00
// CountOrganizations returns number of organizations.
func CountOrganizations ( ) int64 {
2016-11-10 16:16:32 +01:00
count , _ := x .
Where ( "type=1" ) .
Count ( new ( User ) )
2014-08-28 22:29:00 +08:00
return count
}
2014-06-28 00:40:07 -04:00
// DeleteOrganization completely and permanently deletes everything of organization.
func DeleteOrganization ( org * User ) ( err error ) {
sess := x . NewSession ( )
2016-12-27 10:02:14 +08:00
defer sess . Close ( )
2014-06-28 00:40:07 -04:00
if err = sess . Begin ( ) ; err != nil {
return err
}
2017-01-11 21:10:43 +08:00
if err = deleteOrg ( sess , org ) ; err != nil {
if IsErrUserOwnRepos ( err ) {
return err
} else if err != nil {
return fmt . Errorf ( "deleteOrg: %v" , err )
}
}
2017-09-19 01:08:30 -07:00
return sess . Commit ( )
2017-01-11 21:10:43 +08:00
}
func deleteOrg ( e * xorm . Session , u * User ) error {
if ! u . IsOrganization ( ) {
return fmt . Errorf ( "You can't delete none organization user: %s" , u . Name )
}
// Check ownership of repository.
count , err := getRepositoryCount ( e , u )
if err != nil {
return fmt . Errorf ( "GetRepositoryCount: %v" , err )
} else if count > 0 {
return ErrUserOwnRepos { UID : u . ID }
}
if err := deleteBeans ( e ,
& Team { OrgID : u . ID } ,
& OrgUser { OrgID : u . ID } ,
& TeamUser { OrgID : u . ID } ,
2018-06-21 18:00:13 +02:00
& TeamUnit { OrgID : u . ID } ,
2015-09-06 08:54:08 -04:00
) ; err != nil {
return fmt . Errorf ( "deleteBeans: %v" , err )
2014-06-28 00:40:07 -04:00
}
2015-09-06 08:54:08 -04:00
2017-10-04 21:43:04 -07:00
if _ , err = e . ID ( u . ID ) . Delete ( new ( User ) ) ; err != nil {
2017-01-11 21:10:43 +08:00
return fmt . Errorf ( "Delete: %v" , err )
2014-06-28 00:40:07 -04:00
}
2015-09-06 08:54:08 -04:00
2017-01-11 21:10:43 +08:00
// FIXME: system notice
// Note: There are something just cannot be roll back,
// so just keep error logs of those operations.
path := UserPath ( u . Name )
if err := os . RemoveAll ( path ) ; err != nil {
2017-01-29 12:13:57 -08:00
return fmt . Errorf ( "Failed to RemoveAll %s: %v" , path , err )
2017-01-11 21:10:43 +08:00
}
2017-03-06 16:07:18 +08:00
if len ( u . Avatar ) > 0 {
avatarPath := u . CustomAvatarPath ( )
if com . IsExist ( avatarPath ) {
if err := os . Remove ( avatarPath ) ; err != nil {
return fmt . Errorf ( "Failed to remove %s: %v" , avatarPath , err )
}
2017-01-11 21:10:43 +08:00
}
2016-12-27 10:02:14 +08:00
}
2017-01-11 21:10:43 +08:00
return nil
2014-06-28 00:40:07 -04:00
}
2014-06-25 00:44:48 -04:00
// ________ ____ ___
// \_____ \_______ ____ | | \______ ___________
// / | \_ __ \/ ___\| | / ___// __ \_ __ \
// / | \ | \/ /_/ > | /\___ \\ ___/| | \/
// \_______ /__| \___ /|______//____ >\___ >__|
// \/ /_____/ \/ \/
// OrgUser represents an organization-user relation.
type OrgUser struct {
2015-02-23 02:15:53 -05:00
ID int64 ` xorm:"pk autoincr" `
2016-11-26 01:20:18 +01:00
UID int64 ` xorm:"INDEX UNIQUE(s)" `
2015-02-23 02:15:53 -05:00
OrgID int64 ` xorm:"INDEX UNIQUE(s)" `
2017-01-06 13:14:33 -02:00
IsPublic bool ` xorm:"INDEX" `
2018-01-07 23:48:37 -08:00
}
func isOrganizationOwner ( e Engine , orgID , uid int64 ) ( bool , error ) {
2019-08-02 18:06:28 +02:00
ownerTeam , err := getOwnerTeam ( e , orgID )
if err != nil {
2019-09-23 22:08:03 +02:00
if IsErrTeamNotExist ( err ) {
2019-08-02 18:06:28 +02:00
log . Error ( "Organization does not have owner team: %d" , orgID )
return false , nil
}
2018-01-07 23:48:37 -08:00
return false , err
}
return isTeamMember ( e , orgID , ownerTeam . ID , uid )
2014-06-25 00:44:48 -04:00
}
2014-06-29 16:30:41 -04:00
// IsOrganizationOwner returns true if given user is in the owner team.
2017-12-20 23:43:26 -08:00
func IsOrganizationOwner ( orgID , uid int64 ) ( bool , error ) {
2018-01-07 23:48:37 -08:00
return isOrganizationOwner ( x , orgID , uid )
2014-06-29 16:30:41 -04:00
}
// IsOrganizationMember returns true if given user is member of organization.
2017-12-20 23:43:26 -08:00
func IsOrganizationMember ( orgID , uid int64 ) ( bool , error ) {
2019-01-14 02:29:58 +00:00
return isOrganizationMember ( x , orgID , uid )
}
func isOrganizationMember ( e Engine , orgID , uid int64 ) ( bool , error ) {
return e .
2016-11-10 16:16:32 +01:00
Where ( "uid=?" , uid ) .
2016-11-26 01:20:18 +01:00
And ( "org_id=?" , orgID ) .
2017-12-20 23:43:26 -08:00
Table ( "org_user" ) .
Exist ( )
2014-06-29 16:30:41 -04:00
}
2014-12-06 20:22:48 -05:00
// IsPublicMembership returns true if given user public his/her membership.
2017-12-20 23:43:26 -08:00
func IsPublicMembership ( orgID , uid int64 ) ( bool , error ) {
return x .
2016-11-10 16:16:32 +01:00
Where ( "uid=?" , uid ) .
2016-11-26 01:20:18 +01:00
And ( "org_id=?" , orgID ) .
2016-11-10 16:16:32 +01:00
And ( "is_public=?" , true ) .
2017-12-20 23:43:26 -08:00
Table ( "org_user" ) .
Exist ( )
2014-08-15 18:29:41 +08:00
}
2019-11-20 12:27:49 +01: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
}
return x .
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 ) )
}
2016-02-19 18:10:03 -05:00
func getOrgsByUserID ( sess * xorm . Session , userID int64 , showAll bool ) ( [ ] * User , error ) {
2016-01-30 21:51:11 +00:00
orgs := make ( [ ] * User , 0 , 10 )
2016-02-19 18:10:03 -05:00
if ! showAll {
sess . And ( "`org_user`.is_public=?" , true )
}
2016-11-10 09:10:35 +01:00
return orgs , sess .
And ( "`org_user`.uid=?" , userID ) .
Join ( "INNER" , "`org_user`" , "`org_user`.org_id=`user`.id" ) .
Asc ( "`user`.name" ) .
Find ( & orgs )
2016-01-30 21:51:11 +00:00
}
2016-02-14 20:36:03 -05:00
// GetOrgsByUserID returns a list of organizations that the given user ID
// has joined.
2016-02-19 18:10:03 -05:00
func GetOrgsByUserID ( userID int64 , showAll bool ) ( [ ] * User , error ) {
2017-01-28 00:11:41 +08:00
sess := x . NewSession ( )
defer sess . Close ( )
return getOrgsByUserID ( sess , userID , showAll )
2016-01-30 21:51:11 +00:00
}
2015-09-07 13:58:23 -04:00
func getOwnedOrgsByUserID ( sess * xorm . Session , userID int64 ) ( [ ] * User , error ) {
orgs := make ( [ ] * User , 0 , 10 )
2016-11-10 09:10:35 +01:00
return orgs , sess .
2018-01-07 23:48:37 -08: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 ) .
And ( "`team`.authorize=?" , AccessModeOwner ) .
2016-11-10 09:10:35 +01:00
Asc ( "`user`.name" ) .
Find ( & orgs )
2015-09-07 13:58:23 -04:00
}
2019-02-18 17:00:27 +01:00
// HasOrgVisible tells if the given user can see the given org
func HasOrgVisible ( org * User , user * User ) bool {
2019-04-26 02:59:10 +08:00
return hasOrgVisible ( x , org , user )
}
func hasOrgVisible ( e Engine , org * User , user * User ) bool {
2019-02-18 17:00:27 +01:00
// Not SignedUser
if user == nil {
2019-06-12 21:41:28 +02:00
return org . Visibility == structs . VisibleTypePublic
2019-02-18 17:00:27 +01:00
}
if user . IsAdmin {
return true
}
2020-01-13 19:33:46 +02:00
if ( org . Visibility == structs . VisibleTypePrivate || user . IsRestricted ) && ! org . isUserPartOfOrg ( e , user . ID ) {
2019-02-18 17:00:27 +01:00
return false
}
return true
}
// HasOrgsVisible tells if the given user can see at least one of the orgs provided
func HasOrgsVisible ( orgs [ ] * User , user * User ) bool {
if len ( orgs ) == 0 {
return false
}
for _ , org := range orgs {
if HasOrgVisible ( org , user ) {
return true
}
}
return false
}
2015-09-07 13:58:23 -04:00
// GetOwnedOrgsByUserID returns a list of organizations are owned by given user ID.
func GetOwnedOrgsByUserID ( userID int64 ) ( [ ] * User , error ) {
sess := x . NewSession ( )
2017-01-28 00:11:41 +08:00
defer sess . Close ( )
2015-09-07 13:58:23 -04:00
return getOwnedOrgsByUserID ( sess , userID )
}
2016-11-26 01:20:18 +01:00
// GetOwnedOrgsByUserIDDesc returns a list of organizations are owned by
2016-01-30 21:51:11 +00:00
// given user ID, ordered descending by the given condition.
2015-09-07 13:58:23 -04:00
func GetOwnedOrgsByUserIDDesc ( userID int64 , desc string ) ( [ ] * User , error ) {
2017-01-28 00:11:41 +08:00
return getOwnedOrgsByUserID ( x . Desc ( desc ) , userID )
2015-09-07 13:58:23 -04:00
}
2019-11-20 12:27:49 +01:00
// GetOrgsCanCreateRepoByUserID returns a list of organizations where given user ID
// are allowed to create repos.
func GetOrgsCanCreateRepoByUserID ( userID int64 ) ( [ ] * User , error ) {
orgs := make ( [ ] * User , 0 , 10 )
return orgs , x . Join ( "INNER" , "`team_user`" , "`team_user`.org_id=`user`.id" ) .
Join ( "INNER" , "`team`" , "`team`.id=`team_user`.team_id" ) .
Where ( "`team_user`.uid=?" , userID ) .
And ( builder . Eq { "`team`.authorize" : AccessModeOwner } . Or ( builder . Eq { "`team`.can_create_org_repo" : true } ) ) .
Desc ( "`user`.updated_unix" ) .
Find ( & orgs )
}
2015-12-17 02:28:47 -05:00
// GetOrgUsersByUserID returns all organization-user relations by user ID.
2020-01-24 19:00:29 +00:00
func GetOrgUsersByUserID ( uid int64 , opts * SearchOrganizationsOptions ) ( [ ] * OrgUser , error ) {
2014-06-25 00:44:48 -04:00
ous := make ( [ ] * OrgUser , 0 , 10 )
2016-11-10 09:10:35 +01:00
sess := x .
2018-07-20 05:10:17 +03:00
Join ( "LEFT" , "`user`" , "`org_user`.org_id=`user`.id" ) .
2016-11-12 00:55:06 +08:00
Where ( "`org_user`.uid=?" , uid )
2020-01-24 19:00:29 +00:00
if ! opts . All {
2015-12-17 02:28:47 -05:00
// Only show public organizations
sess . And ( "is_public=?" , true )
}
2020-01-24 19:00:29 +00:00
if opts . PageSize != 0 {
sess = opts . setSessionPagination ( sess )
}
2016-11-10 09:10:35 +01:00
err := sess .
Asc ( "`user`.name" ) .
Find ( & ous )
2014-06-25 00:44:48 -04:00
return ous , err
}
2016-07-16 00:36:39 +08:00
// GetOrgUsersByOrgID returns all organization-user relations by organization ID.
2020-01-24 19:00:29 +00:00
func GetOrgUsersByOrgID ( opts * FindOrgMembersOpts ) ( [ ] * OrgUser , error ) {
return getOrgUsersByOrgID ( x , opts )
2019-01-04 21:51:27 +00:00
}
2020-01-24 19:00:29 +00:00
func getOrgUsersByOrgID ( e Engine , opts * FindOrgMembersOpts ) ( [ ] * OrgUser , error ) {
sess := e . Where ( "org_id=?" , opts . OrgID )
if opts . PublicOnly {
2019-12-06 13:34:54 +08:00
sess . And ( "is_public = ?" , true )
}
2020-01-24 19:00:29 +00:00
if opts . ListOptions . PageSize > 0 {
sess = opts . setSessionPagination ( sess )
ous := make ( [ ] * OrgUser , 0 , opts . PageSize )
return ous , sess . Find ( & ous )
2019-12-06 13:34:54 +08:00
}
2020-01-24 19:00:29 +00:00
var ous [ ] * OrgUser
return ous , sess . Find ( & ous )
2014-06-25 05:14:36 -04:00
}
2014-08-15 18:29:41 +08:00
// ChangeOrgUserStatus changes public or private membership status.
2016-07-24 14:32:46 +08:00
func ChangeOrgUserStatus ( orgID , uid int64 , public bool ) error {
2014-08-15 18:29:41 +08:00
ou := new ( OrgUser )
2016-11-10 16:16:32 +01:00
has , err := x .
Where ( "uid=?" , uid ) .
And ( "org_id=?" , orgID ) .
Get ( ou )
2014-08-15 18:29:41 +08:00
if err != nil {
return err
} else if ! has {
return nil
}
ou . IsPublic = public
2017-10-04 21:43:04 -07:00
_ , err = x . ID ( ou . ID ) . Cols ( "is_public" ) . Update ( ou )
2014-08-15 18:29:41 +08:00
return err
}
// AddOrgUser adds new user to given organization.
2016-07-24 14:32:46 +08:00
func AddOrgUser ( orgID , uid int64 ) error {
2017-12-20 23:43:26 -08:00
isAlreadyMember , err := IsOrganizationMember ( orgID , uid )
if err != nil || isAlreadyMember {
return err
2014-08-15 18:29:41 +08:00
}
sess := x . NewSession ( )
defer sess . Close ( )
if err := sess . Begin ( ) ; err != nil {
return err
}
2014-08-24 21:09:05 +08:00
ou := & OrgUser {
2019-08-24 09:28:59 -03:00
UID : uid ,
OrgID : orgID ,
IsPublic : setting . Service . DefaultOrgMemberVisible ,
2014-08-24 21:09:05 +08:00
}
2014-08-15 18:29:41 +08:00
if _ , err := sess . Insert ( ou ) ; err != nil {
2019-06-12 21:41:28 +02:00
if err := sess . Rollback ( ) ; err != nil {
log . Error ( "AddOrgUser: sess.Rollback: %v" , err )
}
2014-08-15 18:29:41 +08:00
return err
2016-07-24 14:32:46 +08:00
} else if _ , err = sess . Exec ( "UPDATE `user` SET num_members = num_members + 1 WHERE id = ?" , orgID ) ; err != nil {
2019-06-12 21:41:28 +02:00
if err := sess . Rollback ( ) ; err != nil {
log . Error ( "AddOrgUser: sess.Rollback: %v" , err )
}
2014-08-15 18:29:41 +08:00
return err
}
return sess . Commit ( )
}
2018-02-23 16:42:02 +08:00
func removeOrgUser ( sess * xorm . Session , orgID , userID int64 ) error {
2014-08-15 18:29:41 +08:00
ou := new ( OrgUser )
2018-07-27 01:41:36 +02:00
has , err := sess .
2016-11-10 16:16:32 +01:00
Where ( "uid=?" , userID ) .
And ( "org_id=?" , orgID ) .
Get ( ou )
2014-08-15 18:29:41 +08:00
if err != nil {
2015-03-17 21:51:39 -04:00
return fmt . Errorf ( "get org-user: %v" , err )
2014-08-15 18:29:41 +08:00
} else if ! has {
return nil
}
2018-10-29 21:48:37 +08:00
org , err := getUserByID ( sess , orgID )
2014-08-24 21:09:05 +08:00
if err != nil {
2016-07-24 14:32:46 +08:00
return fmt . Errorf ( "GetUserByID [%d]: %v" , orgID , err )
2014-08-24 21:09:05 +08:00
}
2016-07-24 14:32:46 +08:00
2014-08-16 16:21:17 +08:00
// Check if the user to delete is the last member in owner team.
2018-10-29 21:48:37 +08:00
if isOwner , err := isOrganizationOwner ( sess , orgID , userID ) ; err != nil {
2017-12-20 23:43:26 -08:00
return err
} else if isOwner {
2018-10-29 21:48:37 +08:00
t , err := org . getOwnerTeam ( sess )
2014-08-16 16:21:17 +08:00
if err != nil {
return err
}
if t . NumMembers == 1 {
2018-10-29 21:48:37 +08:00
if err := t . getMembers ( sess ) ; err != nil {
2017-12-13 06:26:31 +08:00
return err
}
if t . Members [ 0 ] . ID == userID {
return ErrLastOrgOwner { UID : userID }
}
2014-08-16 16:21:17 +08:00
}
}
2017-10-04 21:43:04 -07:00
if _ , err := sess . ID ( ou . ID ) . Delete ( ou ) ; err != nil {
2014-08-15 18:29:41 +08:00
return err
2016-07-24 14:32:46 +08:00
} else if _ , err = sess . Exec ( "UPDATE `user` SET num_members=num_members-1 WHERE id=?" , orgID ) ; err != nil {
2014-08-15 18:29:41 +08:00
return err
}
2016-07-24 14:32:46 +08:00
// Delete all repository accesses and unwatch them.
2018-10-29 21:48:37 +08:00
env , err := org . accessibleReposEnv ( sess , userID )
2017-01-25 10:41:38 -05:00
if err != nil {
return fmt . Errorf ( "AccessibleReposEnv: %v" , err )
}
repoIDs , err := env . RepoIDs ( 1 , org . NumRepos )
if err != nil {
2017-01-24 11:16:36 -05:00
return fmt . Errorf ( "GetUserRepositories [%d]: %v" , userID , err )
2017-01-25 10:41:38 -05:00
}
for _ , repoID := range repoIDs {
2017-01-24 11:16:36 -05:00
if err = watchRepo ( sess , userID , repoID , false ) ; err != nil {
2014-08-27 16:39:36 +08:00
return err
2014-08-24 21:09:05 +08:00
}
}
2016-08-10 20:06:51 +02:00
if len ( repoIDs ) > 0 {
2016-11-10 16:16:32 +01:00
if _ , err = sess .
2017-01-24 11:16:36 -05:00
Where ( "user_id = ?" , userID ) .
2016-11-10 16:16:32 +01:00
In ( "repo_id" , repoIDs ) .
Delete ( new ( Access ) ) ; err != nil {
2016-08-10 20:06:51 +02:00
return err
}
2016-07-24 14:32:46 +08:00
}
2014-08-24 21:09:05 +08:00
// Delete member in his/her teams.
2019-01-17 06:09:50 +05:30
teams , err := getUserOrgTeams ( sess , org . ID , userID )
2014-08-24 21:09:05 +08:00
if err != nil {
return err
}
2015-03-17 21:51:39 -04:00
for _ , t := range teams {
2017-02-24 01:25:09 -05:00
if err = removeTeamMember ( sess , t , userID ) ; err != nil {
2014-08-24 21:09:05 +08:00
return err
}
}
2018-02-23 16:42:02 +08:00
return nil
}
// RemoveOrgUser removes user from given organization.
func RemoveOrgUser ( orgID , userID int64 ) error {
sess := x . NewSession ( )
defer sess . Close ( )
if err := sess . Begin ( ) ; err != nil {
return err
}
if err := removeOrgUser ( sess , orgID , userID ) ; err != nil {
return err
}
2014-08-15 18:29:41 +08:00
return sess . Commit ( )
}
2015-02-28 21:44:09 -05:00
func removeOrgRepo ( e Engine , orgID , repoID int64 ) error {
2017-02-04 11:00:07 -05: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-02-28 21:44:09 -05:00
OrgID : orgID ,
RepoID : repoID ,
2017-02-04 11:00:07 -05:00
} ) ; err != nil {
return err
}
teamIDs := make ( [ ] int64 , len ( teamRepos ) )
for i , teamRepo := range teamRepos {
2017-06-02 21:06:10 -04:00
teamIDs [ i ] = teamRepo . TeamID
2017-02-04 11:00:07 -05:00
}
2017-05-31 20:41:14 -04:00
_ , err := e . Decr ( "num_repos" ) . In ( "id" , teamIDs ) . Update ( new ( Team ) )
2015-02-28 21:44:09 -05:00
return err
}
2016-07-24 18:09:45 +08:00
func ( org * User ) getUserTeams ( e Engine , userID int64 , cols ... string ) ( [ ] * Team , error ) {
teams := make ( [ ] * Team , 0 , org . NumTeams )
2016-11-10 09:10:35 +01: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 09:10:35 +01:00
And ( "`team_user`.uid = ?" , userID ) .
Asc ( "`user`.name" ) .
Cols ( cols ... ) .
2016-11-10 16:16:32 +01:00
Find ( & teams )
2016-07-24 18:09:45 +08:00
}
2017-02-14 11:46:46 +08:00
func ( org * User ) getUserTeamIDs ( e Engine , userID int64 ) ( [ ] int64 , error ) {
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 )
}
2017-09-14 16:16:22 +08:00
// TeamsWithAccessToRepo returns all teamsthat have given access level to the repository.
func ( org * User ) TeamsWithAccessToRepo ( repoID int64 , mode AccessMode ) ( [ ] * Team , error ) {
return GetTeamsWithAccessToRepo ( org . ID , repoID , mode )
}
2016-11-27 12:59:12 +01:00
// GetUserTeamIDs returns of all team IDs of the organization that user is member of.
2016-07-24 14:32:46 +08:00
func ( org * User ) GetUserTeamIDs ( userID int64 ) ( [ ] int64 , error ) {
2017-02-14 11:46:46 +08:00
return org . getUserTeamIDs ( x , userID )
2016-07-24 14:32:46 +08:00
}
2016-11-26 01:20:18 +01:00
// GetUserTeams returns all teams that belong to user,
2016-07-24 18:09:45 +08:00
// and that the user has joined.
func ( org * User ) GetUserTeams ( userID int64 ) ( [ ] * Team , error ) {
return org . getUserTeams ( x , userID )
}
2017-01-25 10:41:38 -05: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 )
Repos ( page , pageSize int ) ( [ ] * Repository , error )
MirrorRepos ( ) ( [ ] * Repository , error )
2019-02-08 09:45:43 -07:00
AddKeyword ( keyword string )
2019-04-25 02:20:22 +08:00
SetSort ( SearchOrderBy )
2017-01-25 10:41:38 -05:00
}
2016-12-29 02:53:33 -06:00
2017-01-25 10:41:38 -05:00
type accessibleReposEnv struct {
org * User
2020-01-13 19:33:46 +02:00
user * User
2017-01-25 10:41:38 -05:00
teamIDs [ ] int64
2018-10-29 21:48:37 +08:00
e Engine
2019-02-08 09:45:43 -07:00
keyword string
2019-04-25 02:20:22 +08:00
orderBy SearchOrderBy
2017-01-25 10:41:38 -05:00
}
// AccessibleReposEnv an AccessibleReposEnvironment for the repositories in `org`
// that are accessible to the specified user.
func ( org * User ) AccessibleReposEnv ( userID int64 ) ( AccessibleReposEnvironment , error ) {
2018-10-29 21:48:37 +08:00
return org . accessibleReposEnv ( x , userID )
}
func ( org * User ) accessibleReposEnv ( e Engine , userID int64 ) ( AccessibleReposEnvironment , error ) {
2020-01-13 19:33:46 +02:00
var user * User
if userID > 0 {
u , err := getUserByID ( e , userID )
if err != nil {
return nil , err
}
user = u
}
2018-10-29 21:48:37 +08:00
teamIDs , err := org . getUserTeamIDs ( e , userID )
2016-07-24 14:32:46 +08:00
if err != nil {
2017-01-25 10:41:38 -05:00
return nil , err
2016-01-31 18:13:39 +00:00
}
2018-10-29 21:48:37 +08:00
return & accessibleReposEnv {
org : org ,
2020-01-13 19:33:46 +02:00
user : user ,
2018-10-29 21:48:37 +08:00
teamIDs : teamIDs ,
e : e ,
2019-04-25 02:20:22 +08:00
orderBy : SearchOrderByRecentUpdated ,
2018-10-29 21:48:37 +08:00
} , nil
2017-01-25 10:41:38 -05:00
}
2016-12-29 02:53:33 -06:00
2017-01-25 10:41:38 -05:00
func ( env * accessibleReposEnv ) cond ( ) builder . Cond {
2020-01-13 19:33:46 +02:00
var cond = builder . NewCond ( )
if env . user == nil || ! env . user . IsRestricted {
cond = cond . Or ( builder . Eq {
"`repository`.owner_id" : env . org . ID ,
"`repository`.is_private" : false ,
} )
2016-07-24 18:09:45 +08:00
}
2017-01-25 10:41:38 -05:00
if len ( env . teamIDs ) > 0 {
cond = cond . Or ( builder . In ( "team_repo.team_id" , env . teamIDs ) )
}
2019-02-08 09:45:43 -07:00
if env . keyword != "" {
cond = cond . And ( builder . Like { "`repository`.lower_name" , strings . ToLower ( env . keyword ) } )
}
2017-01-25 10:41:38 -05:00
return cond
}
2016-01-31 18:13:39 +00:00
2017-01-25 10:41:38 -05:00
func ( env * accessibleReposEnv ) CountRepos ( ) ( int64 , error ) {
2018-10-29 21:48:37 +08:00
repoCount , err := env . e .
2017-01-25 10:41:38 -05:00
Join ( "INNER" , "team_repo" , "`team_repo`.repo_id=`repository`.id" ) .
Where ( env . cond ( ) ) .
2017-01-24 11:16:36 -05:00
Distinct ( "`repository`.id" ) .
2017-01-25 10:41:38 -05:00
Count ( & Repository { } )
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 14:32:46 +08:00
if page <= 0 {
page = 1
}
2017-01-14 22:32:36 +08:00
2017-01-25 10:41:38 -05:00
repoIDs := make ( [ ] int64 , 0 , pageSize )
2018-10-29 21:48:37 +08:00
return repoIDs , env . e .
2017-01-25 10:41:38 -05:00
Table ( "repository" ) .
2016-11-10 16:16:32 +01:00
Join ( "INNER" , "team_repo" , "`team_repo`.repo_id=`repository`.id" ) .
2017-01-25 10:41:38 -05:00
Where ( env . cond ( ) ) .
2019-04-25 02:20:22 +08:00
GroupBy ( "`repository`.id,`repository`." + strings . Fields ( string ( env . orderBy ) ) [ 0 ] ) .
OrderBy ( string ( env . orderBy ) ) .
2016-11-10 16:16:32 +01:00
Limit ( pageSize , ( page - 1 ) * pageSize ) .
2017-01-25 10:41:38 -05:00
Cols ( "`repository`.id" ) .
Find ( & repoIDs )
}
2016-07-24 14:32:46 +08:00
2017-01-25 10:41:38 -05:00
func ( env * accessibleReposEnv ) Repos ( page , pageSize int ) ( [ ] * Repository , error ) {
repoIDs , err := env . RepoIDs ( page , pageSize )
2016-07-24 14:32:46 +08:00
if err != nil {
2017-01-25 10:41:38 -05:00
return nil , fmt . Errorf ( "GetUserRepositoryIDs: %v" , err )
2016-01-31 10:46:04 +00:00
}
2017-01-25 10:41:38 -05:00
repos := make ( [ ] * Repository , 0 , len ( repoIDs ) )
2018-10-20 23:25:14 +02:00
if len ( repoIDs ) == 0 {
2017-02-14 23:37:44 +08:00
return repos , nil
}
2018-10-29 21:48:37 +08:00
return repos , env . e .
2017-02-14 23:37:44 +08:00
In ( "`repository`.id" , repoIDs ) .
2019-04-25 02:20:22 +08:00
OrderBy ( string ( env . orderBy ) ) .
2017-01-25 10:41:38 -05:00
Find ( & repos )
2016-07-24 14:32:46 +08:00
}
2017-04-06 18:47:25 -07:00
func ( env * accessibleReposEnv ) MirrorRepoIDs ( ) ( [ ] int64 , error ) {
repoIDs := make ( [ ] int64 , 0 , 10 )
2018-10-29 21:48:37 +08:00
return repoIDs , env . e .
2017-04-06 18:47:25 -07:00
Table ( "repository" ) .
2016-11-10 16:16:32 +01:00
Join ( "INNER" , "team_repo" , "`team_repo`.repo_id=`repository`.id AND `repository`.is_mirror=?" , true ) .
2017-01-25 10:41:38 -05:00
Where ( env . cond ( ) ) .
2017-09-15 14:14:06 +08:00
GroupBy ( "`repository`.id, `repository`.updated_unix" ) .
2019-04-25 02:20:22 +08:00
OrderBy ( string ( env . orderBy ) ) .
2017-04-06 18:47:25 -07:00
Cols ( "`repository`.id" ) .
Find ( & repoIDs )
}
func ( env * accessibleReposEnv ) MirrorRepos ( ) ( [ ] * Repository , error ) {
repoIDs , err := env . MirrorRepoIDs ( )
if err != nil {
return nil , fmt . Errorf ( "MirrorRepoIDs: %v" , err )
}
repos := make ( [ ] * Repository , 0 , len ( repoIDs ) )
2018-10-20 23:25:14 +02:00
if len ( repoIDs ) == 0 {
2017-04-06 18:47:25 -07:00
return repos , nil
}
2018-10-29 21:48:37 +08:00
return repos , env . e .
2017-04-06 18:47:25 -07:00
In ( "`repository`.id" , repoIDs ) .
2016-11-10 16:16:32 +01:00
Find ( & repos )
2016-01-31 10:46:04 +00:00
}
2019-02-08 09:45:43 -07:00
func ( env * accessibleReposEnv ) AddKeyword ( keyword string ) {
env . keyword = keyword
}
2019-04-25 02:20:22 +08:00
func ( env * accessibleReposEnv ) SetSort ( orderBy SearchOrderBy ) {
env . orderBy = orderBy
}