2014-04-10 23:50:58 +05:30
// Copyright 2014 The Gogs Authors. All rights reserved.
2019-01-24 04:00:19 +05:30
// Copyright 2019 The Gitea Authors. All rights reserved.
2014-04-10 23:50:58 +05:30
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package models
import (
2019-12-15 15:21:28 +05:30
"context"
2014-04-10 23:50:58 +05:30
"fmt"
"time"
2021-11-17 18:04:35 +05:30
_ "image/jpeg" // Needed for jpeg support
2021-09-19 17:19:59 +05:30
"code.gitea.io/gitea/models/db"
2021-11-10 01:27:58 +05:30
"code.gitea.io/gitea/models/unit"
2021-11-11 12:33:30 +05:30
user_model "code.gitea.io/gitea/models/user"
2016-11-10 21:54:48 +05:30
"code.gitea.io/gitea/modules/setting"
2019-02-18 21:30:27 +05:30
"code.gitea.io/gitea/modules/structs"
2019-06-23 20:52:43 +05:30
"xorm.io/builder"
2014-04-10 23:50:58 +05:30
)
2021-11-18 23:12:27 +05:30
// GetOrganizationCount returns count of membership of organization of the user.
2021-11-24 15:19:20 +05:30
func GetOrganizationCount ( ctx context . Context , u * user_model . User ) ( int64 , error ) {
2021-11-18 23:12:27 +05:30
return db . GetEngine ( ctx ) .
2016-11-10 20:46:32 +05:30
Where ( "uid=?" , u . ID ) .
Count ( new ( OrgUser ) )
2015-09-06 18:24:08 +05:30
}
2019-10-08 23:25:16 +05:30
// GetRepositoryIDs returns repositories IDs where user owned and has unittypes
2020-01-17 13:04:37 +05:30
// Caller shall check that units is not globally disabled
2021-11-24 15:19:20 +05:30
func GetRepositoryIDs ( u * user_model . User , units ... unit . Type ) ( [ ] int64 , error ) {
2019-10-08 23:25:16 +05:30
var ids [ ] int64
2021-09-23 21:15:36 +05:30
sess := db . GetEngine ( db . DefaultContext ) . Table ( "repository" ) . Cols ( "repository.id" )
2018-06-21 21:30:13 +05:30
if len ( units ) > 0 {
2019-10-08 23:25:16 +05:30
sess = sess . Join ( "INNER" , "repo_unit" , "repository.id = repo_unit.repo_id" )
sess = sess . In ( "repo_unit.type" , units )
2018-06-21 21:30:13 +05:30
}
2019-10-08 23:25:16 +05:30
return ids , sess . Where ( "owner_id = ?" , u . ID ) . Find ( & ids )
2017-02-17 06:28:19 +05:30
}
2021-01-13 09:49:17 +05:30
// GetActiveRepositoryIDs returns non-archived repositories IDs where user owned and has unittypes
2021-11-24 15:19:20 +05:30
// Caller shall check that units is not globally disabled
func GetActiveRepositoryIDs ( u * user_model . User , units ... unit . Type ) ( [ ] int64 , error ) {
var ids [ ] int64
2016-01-28 03:15:03 +05:30
2021-11-24 15:19:20 +05:30
sess := db . GetEngine ( db . DefaultContext ) . Table ( "repository" ) . Cols ( "repository.id" )
2018-09-07 06:10:58 +05:30
2021-11-24 15:19:20 +05:30
if len ( units ) > 0 {
sess = sess . Join ( "INNER" , "repo_unit" , "repository.id = repo_unit.repo_id" )
sess = sess . In ( "repo_unit.type" , units )
2021-01-24 20:53:05 +05:30
}
2021-11-24 15:19:20 +05:30
sess . Where ( builder . Eq { "is_archived" : false } )
2021-01-24 20:53:05 +05:30
2021-11-24 15:19:20 +05:30
return ids , sess . Where ( "owner_id = ?" , u . ID ) . GroupBy ( "repository.id" ) . Find ( & ids )
2014-04-10 23:50:58 +05:30
}
2021-11-24 15:19:20 +05:30
// GetOrgRepositoryIDs returns repositories IDs where user's team owned and has unittypes
// Caller shall check that units is not globally disabled
func GetOrgRepositoryIDs ( u * user_model . User , units ... unit . Type ) ( [ ] int64 , error ) {
var ids [ ] int64
2017-02-25 20:23:57 +05:30
2021-11-24 15:19:20 +05:30
if err := db . GetEngine ( db . DefaultContext ) . Table ( "repository" ) .
Cols ( "repository.id" ) .
Join ( "INNER" , "team_user" , "repository.owner_id = team_user.org_id" ) .
Join ( "INNER" , "team_repo" , "(? != ? and repository.is_private != ?) OR (team_user.team_id = team_repo.team_id AND repository.id = team_repo.repo_id)" , true , u . IsRestricted , true ) .
Where ( "team_user.uid = ?" , u . ID ) .
GroupBy ( "repository.id" ) . Find ( & ids ) ; err != nil {
return nil , err
2021-06-28 00:17:35 +05:30
}
2021-11-24 15:19:20 +05:30
if len ( units ) > 0 {
return FilterOutRepoIdsWithoutUnitAccess ( u , ids , units ... )
2020-11-14 22:23:43 +05:30
}
2021-06-28 00:17:35 +05:30
2021-11-24 15:19:20 +05:30
return ids , nil
2014-04-10 23:50:58 +05:30
}
2021-11-24 15:19:20 +05:30
// GetActiveOrgRepositoryIDs returns non-archived repositories IDs where user's team owned and has unittypes
// Caller shall check that units is not globally disabled
func GetActiveOrgRepositoryIDs ( u * user_model . User , units ... unit . Type ) ( [ ] int64 , error ) {
var ids [ ] int64
2017-02-25 20:23:57 +05:30
2021-11-24 15:19:20 +05:30
if err := db . GetEngine ( db . DefaultContext ) . Table ( "repository" ) .
Cols ( "repository.id" ) .
Join ( "INNER" , "team_user" , "repository.owner_id = team_user.org_id" ) .
Join ( "INNER" , "team_repo" , "(? != ? and repository.is_private != ?) OR (team_user.team_id = team_repo.team_id AND repository.id = team_repo.repo_id)" , true , u . IsRestricted , true ) .
Where ( "team_user.uid = ?" , u . ID ) .
Where ( builder . Eq { "is_archived" : false } ) .
GroupBy ( "repository.id" ) . Find ( & ids ) ; err != nil {
return nil , err
}
2017-09-25 10:29:27 +05:30
2021-11-24 15:19:20 +05:30
if len ( units ) > 0 {
return FilterOutRepoIdsWithoutUnitAccess ( u , ids , units ... )
2021-06-28 00:17:35 +05:30
}
2021-11-24 15:19:20 +05:30
return ids , nil
2017-08-12 19:48:44 +05:30
}
2021-11-24 15:19:20 +05:30
// GetAccessRepoIDs returns all repositories IDs where user's or user is a team member organizations
// Caller shall check that units is not globally disabled
func GetAccessRepoIDs ( u * user_model . User , units ... unit . Type ) ( [ ] int64 , error ) {
ids , err := GetRepositoryIDs ( u , units ... )
if err != nil {
return nil , err
}
ids2 , err := GetOrgRepositoryIDs ( u , units ... )
2021-11-21 21:11:00 +05:30
if err != nil {
2021-11-24 15:19:20 +05:30
return nil , err
2020-11-21 03:15:55 +05:30
}
2021-11-24 15:19:20 +05:30
return append ( ids , ids2 ... ) , nil
}
2021-11-21 21:11:00 +05:30
2021-11-24 15:19:20 +05:30
// GetActiveAccessRepoIDs returns all non-archived repositories IDs where user's or user is a team member organizations
// Caller shall check that units is not globally disabled
func GetActiveAccessRepoIDs ( u * user_model . User , units ... unit . Type ) ( [ ] int64 , error ) {
ids , err := GetActiveRepositoryIDs ( u , units ... )
if err != nil {
return nil , err
2017-02-25 20:23:57 +05:30
}
2021-11-24 15:19:20 +05:30
ids2 , err := GetActiveOrgRepositoryIDs ( u , units ... )
if err != nil {
return nil , err
2020-11-21 03:15:55 +05:30
}
2021-11-24 15:19:20 +05:30
return append ( ids , ids2 ... ) , nil
2015-08-29 22:43:24 +05:30
}
2015-09-06 18:24:08 +05:30
// deleteBeans deletes all given beans, beans should contain delete conditions.
2021-09-19 17:19:59 +05:30
func deleteBeans ( e db . Engine , beans ... interface { } ) ( err error ) {
2015-03-18 07:21:39 +05:30
for i := range beans {
if _ , err = e . Delete ( beans [ i ] ) ; err != nil {
return err
}
}
return nil
}
2021-11-18 23:12:27 +05:30
// DeleteUser deletes models associated to an user.
2021-11-24 15:19:20 +05:30
func DeleteUser ( ctx context . Context , u * user_model . User ) ( err error ) {
2021-11-18 11:28:42 +05:30
e := db . GetEngine ( ctx )
2014-06-27 13:07:01 +05:30
2015-08-17 14:35:37 +05:30
// ***** START: Watch *****
2017-05-20 14:18:22 +05:30
watchedRepoIDs := make ( [ ] int64 , 0 , 10 )
if err = e . Table ( "watch" ) . Cols ( "watch.repo_id" ) .
2019-11-10 14:52:19 +05:30
Where ( "watch.user_id = ?" , u . ID ) . And ( "watch.mode <>?" , RepoWatchModeDont ) . Find ( & watchedRepoIDs ) ; err != nil {
2015-03-18 07:21:39 +05:30
return fmt . Errorf ( "get all watches: %v" , err )
2014-04-10 23:50:58 +05:30
}
2018-07-05 03:17:05 +05:30
if _ , err = e . Decr ( "num_watches" ) . In ( "id" , watchedRepoIDs ) . NoAutoTime ( ) . Update ( new ( Repository ) ) ; err != nil {
2017-05-20 14:18:22 +05:30
return fmt . Errorf ( "decrease repository num_watches: %v" , err )
2014-04-12 07:17:39 +05:30
}
2015-08-17 14:35:37 +05:30
// ***** END: Watch *****
2015-03-18 07:21:39 +05:30
2015-08-17 14:35:37 +05:30
// ***** START: Star *****
2017-05-20 14:18:22 +05:30
starredRepoIDs := make ( [ ] int64 , 0 , 10 )
if err = e . Table ( "star" ) . Cols ( "star.repo_id" ) .
Where ( "star.uid = ?" , u . ID ) . Find ( & starredRepoIDs ) ; err != nil {
2015-08-17 14:35:37 +05:30
return fmt . Errorf ( "get all stars: %v" , err )
2018-07-05 03:17:05 +05:30
} else if _ , err = e . Decr ( "num_stars" ) . In ( "id" , starredRepoIDs ) . NoAutoTime ( ) . Update ( new ( Repository ) ) ; err != nil {
2017-05-20 14:18:22 +05:30
return fmt . Errorf ( "decrease repository num_stars: %v" , err )
2015-08-17 14:35:37 +05:30
}
// ***** END: Star *****
2015-03-18 07:21:39 +05:30
2015-08-17 14:35:37 +05:30
// ***** START: Follow *****
2017-05-20 14:18:22 +05:30
followeeIDs := make ( [ ] int64 , 0 , 10 )
if err = e . Table ( "follow" ) . Cols ( "follow.follow_id" ) .
Where ( "follow.user_id = ?" , u . ID ) . Find ( & followeeIDs ) ; err != nil {
return fmt . Errorf ( "get all followees: %v" , err )
2021-11-24 15:19:20 +05:30
} else if _ , err = e . Decr ( "num_followers" ) . In ( "id" , followeeIDs ) . Update ( new ( user_model . User ) ) ; err != nil {
2017-05-20 14:18:22 +05:30
return fmt . Errorf ( "decrease user num_followers: %v" , err )
2015-08-17 14:35:37 +05:30
}
2017-05-20 14:18:22 +05:30
followerIDs := make ( [ ] int64 , 0 , 10 )
if err = e . Table ( "follow" ) . Cols ( "follow.user_id" ) .
Where ( "follow.follow_id = ?" , u . ID ) . Find ( & followerIDs ) ; err != nil {
return fmt . Errorf ( "get all followers: %v" , err )
2021-11-24 15:19:20 +05:30
} else if _ , err = e . Decr ( "num_following" ) . In ( "id" , followerIDs ) . Update ( new ( user_model . User ) ) ; err != nil {
2017-05-20 14:18:22 +05:30
return fmt . Errorf ( "decrease user num_following: %v" , err )
2014-04-10 23:50:58 +05:30
}
2015-08-17 14:35:37 +05:30
// ***** END: Follow *****
2015-03-18 07:21:39 +05:30
2015-09-06 18:24:08 +05:30
if err = deleteBeans ( e ,
2016-07-23 22:38:22 +05:30
& AccessToken { UID : u . ID } ,
& Collaboration { UserID : u . ID } ,
& Access { UserID : u . ID } ,
& Watch { UserID : u . ID } ,
& Star { UID : u . ID } ,
2021-11-17 15:28:31 +05:30
& user_model . Follow { UserID : u . ID } ,
& user_model . Follow { FollowID : u . ID } ,
2016-07-23 22:38:22 +05:30
& Action { UserID : u . ID } ,
& IssueUser { UID : u . ID } ,
2021-11-11 12:33:30 +05:30
& user_model . EmailAddress { UID : u . ID } ,
2021-11-17 15:28:31 +05:30
& user_model . UserOpenID { UID : u . ID } ,
2017-12-04 04:44:26 +05:30
& Reaction { UserID : u . ID } ,
2018-12-18 21:56:26 +05:30
& TeamUser { UID : u . ID } ,
& Collaboration { UserID : u . ID } ,
& Stopwatch { UserID : u . ID } ,
2021-11-22 15:17:23 +05:30
& user_model . Setting { UserID : u . ID } ,
2015-03-18 07:21:39 +05:30
) ; err != nil {
2015-12-01 07:15:55 +05:30
return fmt . Errorf ( "deleteBeans: %v" , err )
2014-04-10 23:50:58 +05:30
}
2015-03-18 07:21:39 +05:30
2021-01-22 08:26:19 +05:30
if setting . Service . UserDeleteWithCommentsMaxTime != 0 &&
u . CreatedUnix . AsTime ( ) . Add ( setting . Service . UserDeleteWithCommentsMaxTime ) . After ( time . Now ( ) ) {
// Delete Comments
const batchSize = 50
for start := 0 ; ; start += batchSize {
comments := make ( [ ] * Comment , 0 , batchSize )
if err = e . Where ( "type=? AND poster_id=?" , CommentTypeComment , u . ID ) . Limit ( batchSize , start ) . Find ( & comments ) ; err != nil {
return err
}
if len ( comments ) == 0 {
break
}
for _ , comment := range comments {
if err = deleteComment ( e , comment ) ; err != nil {
return err
}
}
}
// Delete Reactions
if err = deleteReaction ( e , & ReactionOptions { Doer : u } ) ; err != nil {
return err
2021-01-18 02:18:38 +05:30
}
}
2015-08-17 14:35:37 +05:30
// ***** START: PublicKey *****
2018-12-18 21:56:26 +05:30
if _ , err = e . Delete ( & PublicKey { OwnerID : u . ID } ) ; err != nil {
2016-07-26 14:56:48 +05:30
return fmt . Errorf ( "deletePublicKeys: %v" , err )
2014-04-10 23:50:58 +05:30
}
2015-08-17 14:35:37 +05:30
// ***** END: PublicKey *****
2014-04-10 23:50:58 +05:30
2018-12-18 21:56:26 +05:30
// ***** START: GPGPublicKey *****
2021-09-24 17:02:56 +05:30
keys , err := listGPGKeys ( e , u . ID , db . ListOptions { } )
2021-02-04 14:46:21 +05:30
if err != nil {
return fmt . Errorf ( "ListGPGKeys: %v" , err )
}
// Delete GPGKeyImport(s).
for _ , key := range keys {
if _ , err = e . Delete ( & GPGKeyImport { KeyID : key . KeyID } ) ; err != nil {
return fmt . Errorf ( "deleteGPGKeyImports: %v" , err )
}
}
2018-12-18 21:56:26 +05:30
if _ , err = e . Delete ( & GPGKey { OwnerID : u . ID } ) ; err != nil {
return fmt . Errorf ( "deleteGPGKeys: %v" , err )
}
// ***** END: GPGPublicKey *****
2015-08-15 00:18:05 +05:30
// Clear assignee.
2018-05-09 21:59:04 +05:30
if err = clearAssigneeByUserID ( e , u . ID ) ; err != nil {
2015-08-17 14:35:37 +05:30
return fmt . Errorf ( "clear assignee: %v" , err )
2015-08-15 00:18:05 +05:30
}
2017-02-22 12:44:37 +05:30
// ***** START: ExternalLoginUser *****
2021-11-28 19:41:58 +05:30
if err = user_model . RemoveAllAccountLinks ( ctx , u ) ; err != nil {
2017-02-22 12:44:37 +05:30
return fmt . Errorf ( "ExternalLoginUser: %v" , err )
}
// ***** END: ExternalLoginUser *****
2021-11-24 15:19:20 +05:30
if _ , err = e . ID ( u . ID ) . Delete ( new ( user_model . User ) ) ; err != nil {
2015-08-17 14:35:37 +05:30
return fmt . Errorf ( "Delete: %v" , err )
2015-03-18 07:21:39 +05:30
}
2015-08-17 14:35:37 +05:30
return nil
2014-06-21 10:21:41 +05:30
}
2016-11-15 04:03:58 +05:30
// GetStarredRepos returns the repos starred by a particular user
2021-09-24 17:02:56 +05:30
func GetStarredRepos ( userID int64 , private bool , listOptions db . ListOptions ) ( [ ] * Repository , error ) {
2021-09-23 21:15:36 +05:30
sess := db . GetEngine ( db . DefaultContext ) . Where ( "star.uid=?" , userID ) .
2016-11-15 04:03:58 +05:30
Join ( "LEFT" , "star" , "`repository`.id=`star`.repo_id" )
if ! private {
sess = sess . And ( "is_private=?" , false )
}
2020-01-25 00:30:29 +05:30
if listOptions . Page != 0 {
2021-09-24 17:02:56 +05:30
sess = db . SetSessionPagination ( sess , & listOptions )
2020-01-25 00:30:29 +05:30
repos := make ( [ ] * Repository , 0 , listOptions . PageSize )
return repos , sess . Find ( & repos )
2016-11-15 04:03:58 +05:30
}
2020-01-25 00:30:29 +05:30
repos := make ( [ ] * Repository , 0 , 10 )
return repos , sess . Find ( & repos )
2016-11-15 04:03:58 +05:30
}
2016-12-24 07:23:11 +05:30
// GetWatchedRepos returns the repos watched by a particular user
2021-09-24 17:02:56 +05:30
func GetWatchedRepos ( userID int64 , private bool , listOptions db . ListOptions ) ( [ ] * Repository , int64 , error ) {
2021-09-23 21:15:36 +05:30
sess := db . GetEngine ( db . DefaultContext ) . Where ( "watch.user_id=?" , userID ) .
2019-11-10 14:52:19 +05:30
And ( "`watch`.mode<>?" , RepoWatchModeDont ) .
2016-12-24 07:23:11 +05:30
Join ( "LEFT" , "watch" , "`repository`.id=`watch`.repo_id" )
if ! private {
sess = sess . And ( "is_private=?" , false )
}
2020-01-25 00:30:29 +05:30
if listOptions . Page != 0 {
2021-09-24 17:02:56 +05:30
sess = db . SetSessionPagination ( sess , & listOptions )
2020-01-25 00:30:29 +05:30
repos := make ( [ ] * Repository , 0 , listOptions . PageSize )
2021-08-12 18:13:08 +05:30
total , err := sess . FindAndCount ( & repos )
return repos , total , err
2016-12-24 07:23:11 +05:30
}
2020-01-25 00:30:29 +05:30
repos := make ( [ ] * Repository , 0 , 10 )
2021-08-12 18:13:08 +05:30
total , err := sess . FindAndCount ( & repos )
return repos , total , err
2016-12-24 07:23:11 +05:30
}
2017-05-10 18:40:18 +05:30
2021-11-24 15:19:20 +05:30
// IsUserVisibleToViewer check if viewer is able to see user profile
func IsUserVisibleToViewer ( u * user_model . User , viewer * user_model . User ) bool {
return isUserVisibleToViewer ( db . GetEngine ( db . DefaultContext ) , u , viewer )
2020-10-14 18:37:51 +05:30
}
2021-11-17 15:28:31 +05:30
2021-11-24 15:19:20 +05:30
func isUserVisibleToViewer ( e db . Engine , u * user_model . User , viewer * user_model . User ) bool {
if viewer != nil && viewer . IsAdmin {
return true
2021-11-17 15:28:31 +05:30
}
2021-11-24 15:19:20 +05:30
switch u . Visibility {
case structs . VisibleTypePublic :
return true
case structs . VisibleTypeLimited :
if viewer == nil || viewer . IsRestricted {
return false
}
return true
case structs . VisibleTypePrivate :
if viewer == nil || viewer . IsRestricted {
return false
}
2021-11-17 15:28:31 +05:30
2021-11-24 15:19:20 +05:30
// If they follow - they see each over
follower := user_model . IsFollowing ( u . ID , viewer . ID )
if follower {
return true
}
2021-11-17 15:28:31 +05:30
2021-11-24 15:19:20 +05:30
// Now we need to check if they in some organization together
count , err := e . Table ( "team_user" ) .
Where (
builder . And (
builder . Eq { "uid" : viewer . ID } ,
builder . Or (
builder . Eq { "org_id" : u . ID } ,
builder . In ( "org_id" ,
builder . Select ( "org_id" ) .
From ( "team_user" , "t2" ) .
Where ( builder . Eq { "uid" : u . ID } ) ) ) ) ) .
Count ( new ( TeamUser ) )
if err != nil {
return false
}
2021-11-17 15:28:31 +05:30
2021-11-24 15:19:20 +05:30
if count < 0 {
// No common organization
return false
}
2021-11-18 11:28:42 +05:30
2021-11-24 15:19:20 +05:30
// they are in an organization together
return true
2021-11-18 11:28:42 +05:30
}
2021-11-24 15:19:20 +05:30
return false
2021-11-18 11:28:42 +05:30
}