2014-04-10 22:20:58 +04:00
// Copyright 2014 The Gogs Authors. All rights reserved.
2019-01-24 01:30:19 +03:00
// Copyright 2019 The Gitea Authors. All rights reserved.
2014-04-10 22:20:58 +04:00
// 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 12:51:28 +03:00
"context"
2014-04-10 22:20:58 +04:00
"fmt"
"time"
2021-11-17 15:34:35 +03:00
_ "image/jpeg" // Needed for jpeg support
2021-09-19 14:49:59 +03:00
"code.gitea.io/gitea/models/db"
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"
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/modules/setting"
2019-02-18 19:00:27 +03:00
"code.gitea.io/gitea/modules/structs"
2019-06-23 18:22:43 +03:00
"xorm.io/builder"
2014-04-10 22:20:58 +04:00
)
2021-11-18 20:42:27 +03:00
// GetOrganizationCount returns count of membership of organization of the user.
2021-11-24 12:49:20 +03:00
func GetOrganizationCount ( ctx context . Context , u * user_model . User ) ( int64 , error ) {
2021-11-18 20:42:27 +03:00
return db . GetEngine ( ctx ) .
2016-11-10 18:16:32 +03:00
Where ( "uid=?" , u . ID ) .
Count ( new ( OrgUser ) )
2015-09-06 15:54:08 +03:00
}
2019-10-08 20:55:16 +03:00
// GetRepositoryIDs returns repositories IDs where user owned and has unittypes
2020-01-17 10:34:37 +03:00
// Caller shall check that units is not globally disabled
2021-11-24 12:49:20 +03:00
func GetRepositoryIDs ( u * user_model . User , units ... unit . Type ) ( [ ] int64 , error ) {
2019-10-08 20:55:16 +03:00
var ids [ ] int64
2021-09-23 18:45:36 +03:00
sess := db . GetEngine ( db . DefaultContext ) . Table ( "repository" ) . Cols ( "repository.id" )
2018-06-21 19:00:13 +03:00
if len ( units ) > 0 {
2019-10-08 20:55:16 +03:00
sess = sess . Join ( "INNER" , "repo_unit" , "repository.id = repo_unit.repo_id" )
sess = sess . In ( "repo_unit.type" , units )
2018-06-21 19:00:13 +03:00
}
2019-10-08 20:55:16 +03:00
return ids , sess . Where ( "owner_id = ?" , u . ID ) . Find ( & ids )
2017-02-17 03:58:19 +03:00
}
2021-01-13 07:19:17 +03:00
// GetActiveRepositoryIDs returns non-archived repositories IDs where user owned and has unittypes
2021-11-24 12:49:20 +03:00
// 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 00:45:03 +03:00
2021-11-24 12:49:20 +03:00
sess := db . GetEngine ( db . DefaultContext ) . Table ( "repository" ) . Cols ( "repository.id" )
2018-09-07 03:40:58 +03:00
2021-11-24 12:49:20 +03:00
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 18:23:05 +03:00
}
2021-11-24 12:49:20 +03:00
sess . Where ( builder . Eq { "is_archived" : false } )
2021-01-24 18:23:05 +03:00
2021-11-24 12:49:20 +03:00
return ids , sess . Where ( "owner_id = ?" , u . ID ) . GroupBy ( "repository.id" ) . Find ( & ids )
2014-04-10 22:20:58 +04:00
}
2021-11-24 12:49:20 +03:00
// 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 17:53:57 +03:00
2021-11-24 12:49:20 +03:00
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-27 21:47:35 +03:00
}
2021-11-24 12:49:20 +03:00
if len ( units ) > 0 {
return FilterOutRepoIdsWithoutUnitAccess ( u , ids , units ... )
2020-11-14 19:53:43 +03:00
}
2021-06-27 21:47:35 +03:00
2021-11-24 12:49:20 +03:00
return ids , nil
2014-04-10 22:20:58 +04:00
}
2021-11-24 12:49:20 +03:00
// 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 17:53:57 +03:00
2021-11-24 12:49:20 +03:00
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 07:59:27 +03:00
2021-11-24 12:49:20 +03:00
if len ( units ) > 0 {
return FilterOutRepoIdsWithoutUnitAccess ( u , ids , units ... )
2021-06-27 21:47:35 +03:00
}
2021-11-24 12:49:20 +03:00
return ids , nil
2017-08-12 17:18:44 +03:00
}
2021-11-24 12:49:20 +03:00
// 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 18:41:00 +03:00
if err != nil {
2021-11-24 12:49:20 +03:00
return nil , err
2020-11-21 00:45:55 +03:00
}
2021-11-24 12:49:20 +03:00
return append ( ids , ids2 ... ) , nil
}
2021-11-21 18:41:00 +03:00
2021-11-24 12:49:20 +03:00
// 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 17:53:57 +03:00
}
2021-11-24 12:49:20 +03:00
ids2 , err := GetActiveOrgRepositoryIDs ( u , units ... )
if err != nil {
return nil , err
2020-11-21 00:45:55 +03:00
}
2021-11-24 12:49:20 +03:00
return append ( ids , ids2 ... ) , nil
2015-08-29 20:13:24 +03:00
}
2015-09-06 15:54:08 +03:00
// deleteBeans deletes all given beans, beans should contain delete conditions.
2021-09-19 14:49:59 +03:00
func deleteBeans ( e db . Engine , beans ... interface { } ) ( err error ) {
2015-03-18 04:51:39 +03:00
for i := range beans {
if _ , err = e . Delete ( beans [ i ] ) ; err != nil {
return err
}
}
return nil
}
2021-11-18 20:42:27 +03:00
// DeleteUser deletes models associated to an user.
2021-11-24 12:49:20 +03:00
func DeleteUser ( ctx context . Context , u * user_model . User ) ( err error ) {
2021-11-18 08:58:42 +03:00
e := db . GetEngine ( ctx )
2014-06-27 11:37:01 +04:00
2015-08-17 12:05:37 +03:00
// ***** START: Watch *****
2017-05-20 11:48:22 +03:00
watchedRepoIDs := make ( [ ] int64 , 0 , 10 )
if err = e . Table ( "watch" ) . Cols ( "watch.repo_id" ) .
2019-11-10 12:22:19 +03:00
Where ( "watch.user_id = ?" , u . ID ) . And ( "watch.mode <>?" , RepoWatchModeDont ) . Find ( & watchedRepoIDs ) ; err != nil {
2015-03-18 04:51:39 +03:00
return fmt . Errorf ( "get all watches: %v" , err )
2014-04-10 22:20:58 +04:00
}
2018-07-05 00:47:05 +03:00
if _ , err = e . Decr ( "num_watches" ) . In ( "id" , watchedRepoIDs ) . NoAutoTime ( ) . Update ( new ( Repository ) ) ; err != nil {
2017-05-20 11:48:22 +03:00
return fmt . Errorf ( "decrease repository num_watches: %v" , err )
2014-04-12 05:47:39 +04:00
}
2015-08-17 12:05:37 +03:00
// ***** END: Watch *****
2015-03-18 04:51:39 +03:00
2015-08-17 12:05:37 +03:00
// ***** START: Star *****
2017-05-20 11:48:22 +03:00
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 12:05:37 +03:00
return fmt . Errorf ( "get all stars: %v" , err )
2018-07-05 00:47:05 +03:00
} else if _ , err = e . Decr ( "num_stars" ) . In ( "id" , starredRepoIDs ) . NoAutoTime ( ) . Update ( new ( Repository ) ) ; err != nil {
2017-05-20 11:48:22 +03:00
return fmt . Errorf ( "decrease repository num_stars: %v" , err )
2015-08-17 12:05:37 +03:00
}
// ***** END: Star *****
2015-03-18 04:51:39 +03:00
2015-08-17 12:05:37 +03:00
// ***** START: Follow *****
2017-05-20 11:48:22 +03:00
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 12:49:20 +03:00
} else if _ , err = e . Decr ( "num_followers" ) . In ( "id" , followeeIDs ) . Update ( new ( user_model . User ) ) ; err != nil {
2017-05-20 11:48:22 +03:00
return fmt . Errorf ( "decrease user num_followers: %v" , err )
2015-08-17 12:05:37 +03:00
}
2017-05-20 11:48:22 +03:00
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 12:49:20 +03:00
} else if _ , err = e . Decr ( "num_following" ) . In ( "id" , followerIDs ) . Update ( new ( user_model . User ) ) ; err != nil {
2017-05-20 11:48:22 +03:00
return fmt . Errorf ( "decrease user num_following: %v" , err )
2014-04-10 22:20:58 +04:00
}
2015-08-17 12:05:37 +03:00
// ***** END: Follow *****
2015-03-18 04:51:39 +03:00
2015-09-06 15:54:08 +03:00
if err = deleteBeans ( e ,
2016-07-23 20:08:22 +03:00
& AccessToken { UID : u . ID } ,
& Collaboration { UserID : u . ID } ,
& Access { UserID : u . ID } ,
& Watch { UserID : u . ID } ,
& Star { UID : u . ID } ,
2021-11-17 12:58:31 +03:00
& user_model . Follow { UserID : u . ID } ,
& user_model . Follow { FollowID : u . ID } ,
2016-07-23 20:08:22 +03:00
& Action { UserID : u . ID } ,
& IssueUser { UID : u . ID } ,
2021-11-11 10:03:30 +03:00
& user_model . EmailAddress { UID : u . ID } ,
2021-11-17 12:58:31 +03:00
& user_model . UserOpenID { UID : u . ID } ,
2017-12-04 02:14:26 +03:00
& Reaction { UserID : u . ID } ,
2018-12-18 19:26:26 +03:00
& TeamUser { UID : u . ID } ,
& Collaboration { UserID : u . ID } ,
& Stopwatch { UserID : u . ID } ,
2021-11-22 12:47:23 +03:00
& user_model . Setting { UserID : u . ID } ,
2015-03-18 04:51:39 +03:00
) ; err != nil {
2015-12-01 04:45:55 +03:00
return fmt . Errorf ( "deleteBeans: %v" , err )
2014-04-10 22:20:58 +04:00
}
2015-03-18 04:51:39 +03:00
2021-01-22 05:56:19 +03:00
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-17 23:48:38 +03:00
}
}
2015-08-17 12:05:37 +03:00
// ***** START: PublicKey *****
2018-12-18 19:26:26 +03:00
if _ , err = e . Delete ( & PublicKey { OwnerID : u . ID } ) ; err != nil {
2016-07-26 12:26:48 +03:00
return fmt . Errorf ( "deletePublicKeys: %v" , err )
2014-04-10 22:20:58 +04:00
}
2015-08-17 12:05:37 +03:00
// ***** END: PublicKey *****
2014-04-10 22:20:58 +04:00
2018-12-18 19:26:26 +03:00
// ***** START: GPGPublicKey *****
2021-09-24 14:32:56 +03:00
keys , err := listGPGKeys ( e , u . ID , db . ListOptions { } )
2021-02-04 12:16:21 +03:00
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 19:26:26 +03:00
if _ , err = e . Delete ( & GPGKey { OwnerID : u . ID } ) ; err != nil {
return fmt . Errorf ( "deleteGPGKeys: %v" , err )
}
// ***** END: GPGPublicKey *****
2015-08-14 21:48:05 +03:00
// Clear assignee.
2018-05-09 19:29:04 +03:00
if err = clearAssigneeByUserID ( e , u . ID ) ; err != nil {
2015-08-17 12:05:37 +03:00
return fmt . Errorf ( "clear assignee: %v" , err )
2015-08-14 21:48:05 +03:00
}
2017-02-22 10:14:37 +03:00
// ***** START: ExternalLoginUser *****
2017-03-20 17:13:52 +03:00
if err = removeAllAccountLinks ( e , u ) ; err != nil {
2017-02-22 10:14:37 +03:00
return fmt . Errorf ( "ExternalLoginUser: %v" , err )
}
// ***** END: ExternalLoginUser *****
2021-11-24 12:49:20 +03:00
if _ , err = e . ID ( u . ID ) . Delete ( new ( user_model . User ) ) ; err != nil {
2015-08-17 12:05:37 +03:00
return fmt . Errorf ( "Delete: %v" , err )
2015-03-18 04:51:39 +03:00
}
2015-08-17 12:05:37 +03:00
return nil
2014-06-21 08:51:41 +04:00
}
2016-11-15 01:33:58 +03:00
// GetStarredRepos returns the repos starred by a particular user
2021-09-24 14:32:56 +03:00
func GetStarredRepos ( userID int64 , private bool , listOptions db . ListOptions ) ( [ ] * Repository , error ) {
2021-09-23 18:45:36 +03:00
sess := db . GetEngine ( db . DefaultContext ) . Where ( "star.uid=?" , userID ) .
2016-11-15 01:33:58 +03:00
Join ( "LEFT" , "star" , "`repository`.id=`star`.repo_id" )
if ! private {
sess = sess . And ( "is_private=?" , false )
}
2020-01-24 22:00:29 +03:00
if listOptions . Page != 0 {
2021-09-24 14:32:56 +03:00
sess = db . SetSessionPagination ( sess , & listOptions )
2020-01-24 22:00:29 +03:00
repos := make ( [ ] * Repository , 0 , listOptions . PageSize )
return repos , sess . Find ( & repos )
2016-11-15 01:33:58 +03:00
}
2020-01-24 22:00:29 +03:00
repos := make ( [ ] * Repository , 0 , 10 )
return repos , sess . Find ( & repos )
2016-11-15 01:33:58 +03:00
}
2016-12-24 04:53:11 +03:00
// GetWatchedRepos returns the repos watched by a particular user
2021-09-24 14:32:56 +03:00
func GetWatchedRepos ( userID int64 , private bool , listOptions db . ListOptions ) ( [ ] * Repository , int64 , error ) {
2021-09-23 18:45:36 +03:00
sess := db . GetEngine ( db . DefaultContext ) . Where ( "watch.user_id=?" , userID ) .
2019-11-10 12:22:19 +03:00
And ( "`watch`.mode<>?" , RepoWatchModeDont ) .
2016-12-24 04:53:11 +03:00
Join ( "LEFT" , "watch" , "`repository`.id=`watch`.repo_id" )
if ! private {
sess = sess . And ( "is_private=?" , false )
}
2020-01-24 22:00:29 +03:00
if listOptions . Page != 0 {
2021-09-24 14:32:56 +03:00
sess = db . SetSessionPagination ( sess , & listOptions )
2020-01-24 22:00:29 +03:00
repos := make ( [ ] * Repository , 0 , listOptions . PageSize )
2021-08-12 15:43:08 +03:00
total , err := sess . FindAndCount ( & repos )
return repos , total , err
2016-12-24 04:53:11 +03:00
}
2020-01-24 22:00:29 +03:00
repos := make ( [ ] * Repository , 0 , 10 )
2021-08-12 15:43:08 +03:00
total , err := sess . FindAndCount ( & repos )
return repos , total , err
2016-12-24 04:53:11 +03:00
}
2017-05-10 16:10:18 +03:00
2021-11-24 12:49:20 +03:00
// 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 16:07:51 +03:00
}
2021-11-17 12:58:31 +03:00
2021-11-24 12:49:20 +03:00
func isUserVisibleToViewer ( e db . Engine , u * user_model . User , viewer * user_model . User ) bool {
if viewer != nil && viewer . IsAdmin {
return true
2021-11-17 12:58:31 +03:00
}
2021-11-24 12:49:20 +03:00
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 12:58:31 +03:00
2021-11-24 12:49:20 +03:00
// If they follow - they see each over
follower := user_model . IsFollowing ( u . ID , viewer . ID )
if follower {
return true
}
2021-11-17 12:58:31 +03:00
2021-11-24 12:49:20 +03:00
// 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 12:58:31 +03:00
2021-11-24 12:49:20 +03:00
if count < 0 {
// No common organization
return false
}
2021-11-18 08:58:42 +03:00
2021-11-24 12:49:20 +03:00
// they are in an organization together
return true
2021-11-18 08:58:42 +03:00
}
2021-11-24 12:49:20 +03:00
return false
2021-11-18 08:58:42 +03:00
}