2014-02-18 03:38:50 +04:00
// Copyright 2014 The Gogs Authors. All rights reserved.
2019-04-22 23:40:51 +03:00
// Copyright 2019 The Gitea Authors. All rights reserved.
2014-02-18 03:38:50 +04:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
2014-02-17 19:57:23 +04:00
package models
2019-04-22 23:40:51 +03:00
import (
"fmt"
"code.gitea.io/gitea/modules/log"
)
2015-02-24 08:27:22 +03:00
2016-11-26 03:07:57 +03:00
// AccessMode specifies the users access mode
2015-02-05 16:29:08 +03:00
type AccessMode int
2014-06-25 08:44:48 +04:00
2014-02-17 19:57:23 +04:00
const (
2016-11-26 03:07:57 +03:00
// AccessModeNone no access
AccessModeNone AccessMode = iota // 0
// AccessModeRead read access
AccessModeRead // 1
// AccessModeWrite write access
AccessModeWrite // 2
// AccessModeAdmin admin access
AccessModeAdmin // 3
// AccessModeOwner owner access
AccessModeOwner // 4
2014-02-17 19:57:23 +04:00
)
2016-03-21 19:47:54 +03:00
func ( mode AccessMode ) String ( ) string {
switch mode {
2016-11-07 19:20:37 +03:00
case AccessModeRead :
2016-03-21 19:47:54 +03:00
return "read"
2016-11-07 19:20:37 +03:00
case AccessModeWrite :
2016-03-21 19:47:54 +03:00
return "write"
2016-11-07 19:20:37 +03:00
case AccessModeAdmin :
2016-03-21 19:47:54 +03:00
return "admin"
2016-11-07 19:20:37 +03:00
case AccessModeOwner :
2016-03-21 19:47:54 +03:00
return "owner"
default :
return "none"
}
}
2019-04-22 23:40:51 +03:00
// ColorFormat provides a ColorFormatted version of this AccessMode
func ( mode AccessMode ) ColorFormat ( s fmt . State ) {
log . ColorFprintf ( s , "%d:%s" ,
log . NewColoredIDValue ( mode ) ,
mode )
}
2016-03-21 19:47:54 +03:00
// ParseAccessMode returns corresponding access mode to given permission string.
func ParseAccessMode ( permission string ) AccessMode {
switch permission {
case "write" :
2016-11-07 19:20:37 +03:00
return AccessModeWrite
2016-03-21 19:47:54 +03:00
case "admin" :
2016-11-07 19:20:37 +03:00
return AccessModeAdmin
2016-03-21 19:47:54 +03:00
default :
2016-11-07 19:20:37 +03:00
return AccessModeRead
2016-03-21 19:47:54 +03:00
}
}
2015-02-05 16:29:08 +03:00
// Access represents the highest access level of a user to the repository. The only access type
// that is not in this table is the real owner of a repository. In case of an organization
// repository, the members of the owners team are in this table.
type Access struct {
ID int64 ` xorm:"pk autoincr" `
UserID int64 ` xorm:"UNIQUE(s)" `
RepoID int64 ` xorm:"UNIQUE(s)" `
Mode AccessMode
2014-02-17 19:57:23 +04:00
}
2020-01-13 20:33:46 +03:00
func accessLevel ( e Engine , user * User , repo * Repository ) ( AccessMode , error ) {
2016-11-07 19:20:37 +03:00
mode := AccessModeNone
2020-01-13 20:33:46 +03:00
var userID int64
restricted := false
if user != nil {
userID = user . ID
restricted = user . IsRestricted
}
if ! restricted && ! repo . IsPrivate {
2016-11-07 19:20:37 +03:00
mode = AccessModeRead
2015-02-12 05:58:37 +03:00
}
2014-05-01 19:32:12 +04:00
2017-03-15 03:51:46 +03:00
if userID == 0 {
2015-11-19 19:40:00 +03:00
return mode , nil
}
2014-04-05 02:55:17 +04:00
2017-03-15 03:51:46 +03:00
if userID == repo . OwnerID {
2016-11-07 19:20:37 +03:00
return AccessModeOwner , nil
2014-04-12 05:47:39 +04:00
}
2015-02-05 16:29:08 +03:00
2017-03-15 03:51:46 +03:00
a := & Access { UserID : userID , RepoID : repo . ID }
2015-11-19 19:40:00 +03:00
if has , err := e . Get ( a ) ; ! has || err != nil {
return mode , err
}
return a . Mode , nil
2014-02-17 19:57:23 +04:00
}
2015-01-23 10:54:16 +03:00
2017-01-05 03:57:54 +03:00
type repoAccess struct {
Access ` xorm:"extends" `
Repository ` xorm:"extends" `
}
func ( repoAccess ) TableName ( ) string {
return "access"
}
2015-11-14 01:37:02 +03:00
// GetRepositoryAccesses finds all repositories with their access mode where a user has access but does not own.
2016-11-26 03:07:57 +03:00
func ( user * User ) GetRepositoryAccesses ( ) ( map [ * Repository ] AccessMode , error ) {
2016-12-28 04:34:35 +03:00
rows , err := x .
2017-01-05 03:57:54 +03:00
Join ( "INNER" , "repository" , "repository.id = access.repo_id" ) .
2016-12-28 04:34:35 +03:00
Where ( "access.user_id = ?" , user . ID ) .
And ( "repository.owner_id <> ?" , user . ID ) .
2017-01-05 03:57:54 +03:00
Rows ( new ( repoAccess ) )
2016-12-28 04:34:35 +03:00
if err != nil {
2015-01-23 10:54:16 +03:00
return nil , err
}
2016-12-28 04:34:35 +03:00
defer rows . Close ( )
2015-01-23 10:54:16 +03:00
2021-03-14 21:52:12 +03:00
repos := make ( map [ * Repository ] AccessMode , 10 )
ownerCache := make ( map [ int64 ] * User , 10 )
2016-12-28 04:34:35 +03:00
for rows . Next ( ) {
2017-01-05 03:57:54 +03:00
var repo repoAccess
2016-12-28 04:34:35 +03:00
err = rows . Scan ( & repo )
2015-01-23 10:54:16 +03:00
if err != nil {
return nil , err
}
2016-12-28 04:34:35 +03:00
var ok bool
if repo . Owner , ok = ownerCache [ repo . OwnerID ] ; ! ok {
if err = repo . GetOwner ( ) ; err != nil {
return nil , err
}
ownerCache [ repo . OwnerID ] = repo . Owner
2015-02-04 17:08:55 +03:00
}
2016-12-28 04:34:35 +03:00
repos [ & repo . Repository ] = repo . Access . Mode
2015-01-23 10:54:16 +03:00
}
return repos , nil
}
2015-02-05 16:29:08 +03:00
2016-07-24 09:32:46 +03:00
// GetAccessibleRepositories finds repositories which the user has access but does not own.
// If limit is smaller than 1 means returns all found results.
func ( user * User ) GetAccessibleRepositories ( limit int ) ( repos [ ] * Repository , _ error ) {
2016-11-10 18:16:32 +03:00
sess := x .
Where ( "owner_id !=? " , user . ID ) .
Desc ( "updated_unix" )
2016-07-24 09:32:46 +03:00
if limit > 0 {
sess . Limit ( limit )
repos = make ( [ ] * Repository , 0 , limit )
} else {
repos = make ( [ ] * Repository , 0 , 10 )
2015-11-14 01:37:02 +03:00
}
2016-11-10 18:16:32 +03:00
return repos , sess .
Join ( "INNER" , "access" , "access.user_id = ? AND access.repo_id = repository.id" , user . ID ) .
Find ( & repos )
2015-11-14 01:37:02 +03:00
}
2015-02-13 08:58:46 +03:00
func maxAccessMode ( modes ... AccessMode ) AccessMode {
2016-11-07 19:20:37 +03:00
max := AccessModeNone
2015-02-13 08:58:46 +03:00
for _ , mode := range modes {
if mode > max {
max = mode
}
}
return max
}
2020-01-13 20:33:46 +03:00
type userAccess struct {
User * User
Mode AccessMode
}
// updateUserAccess updates an access map so that user has at least mode
func updateUserAccess ( accessMap map [ int64 ] * userAccess , user * User , mode AccessMode ) {
if ua , ok := accessMap [ user . ID ] ; ok {
ua . Mode = maxAccessMode ( ua . Mode , mode )
} else {
accessMap [ user . ID ] = & userAccess { User : user , Mode : mode }
}
}
2017-01-05 03:50:34 +03:00
// FIXME: do cross-comparison so reduce deletions and additions to the minimum?
2020-01-13 20:33:46 +03:00
func ( repo * Repository ) refreshAccesses ( e Engine , accessMap map [ int64 ] * userAccess ) ( err error ) {
2016-11-07 19:20:37 +03:00
minMode := AccessModeRead
2015-03-01 05:44:09 +03:00
if ! repo . IsPrivate {
2016-11-07 19:20:37 +03:00
minMode = AccessModeWrite
2015-03-01 05:44:09 +03:00
}
newAccesses := make ( [ ] Access , 0 , len ( accessMap ) )
2020-01-13 20:33:46 +03:00
for userID , ua := range accessMap {
if ua . Mode < minMode && ! ua . User . IsRestricted {
2015-03-01 05:44:09 +03:00
continue
}
2020-01-13 20:33:46 +03:00
2015-03-01 05:44:09 +03:00
newAccesses = append ( newAccesses , Access {
UserID : userID ,
2015-08-08 17:43:14 +03:00
RepoID : repo . ID ,
2020-01-13 20:33:46 +03:00
Mode : ua . Mode ,
2015-03-01 05:44:09 +03:00
} )
}
2015-02-13 10:56:42 +03:00
2015-03-01 05:44:09 +03:00
// Delete old accesses and insert new ones for repository.
2015-08-08 17:43:14 +03:00
if _ , err = e . Delete ( & Access { RepoID : repo . ID } ) ; err != nil {
2015-03-01 05:44:09 +03:00
return fmt . Errorf ( "delete old accesses: %v" , err )
2020-03-22 18:12:55 +03:00
}
if len ( newAccesses ) == 0 {
return nil
}
if _ , err = e . Insert ( newAccesses ) ; err != nil {
2015-03-01 05:44:09 +03:00
return fmt . Errorf ( "insert new accesses: %v" , err )
}
2015-02-13 10:56:42 +03:00
return nil
}
2016-03-06 02:08:42 +03:00
// refreshCollaboratorAccesses retrieves repository collaborations with their access modes.
2020-01-13 20:33:46 +03:00
func ( repo * Repository ) refreshCollaboratorAccesses ( e Engine , accessMap map [ int64 ] * userAccess ) error {
2020-01-24 22:00:29 +03:00
collaborators , err := repo . getCollaborators ( e , ListOptions { } )
2015-02-05 16:29:08 +03:00
if err != nil {
2016-03-06 02:08:42 +03:00
return fmt . Errorf ( "getCollaborations: %v" , err )
2015-02-05 16:29:08 +03:00
}
2020-01-13 20:33:46 +03:00
for _ , c := range collaborators {
updateUserAccess ( accessMap , c . User , c . Collaboration . Mode )
2015-02-05 16:29:08 +03:00
}
2015-03-01 05:44:09 +03:00
return nil
}
// recalculateTeamAccesses recalculates new accesses for teams of an organization
// except the team whose ID is given. It is used to assign a team ID when
// remove repository from that team.
func ( repo * Repository ) recalculateTeamAccesses ( e Engine , ignTeamID int64 ) ( err error ) {
2020-01-13 20:33:46 +03:00
accessMap := make ( map [ int64 ] * userAccess , 20 )
2015-03-01 05:44:09 +03:00
if err = repo . getOwner ( e ) ; err != nil {
2015-02-05 16:29:08 +03:00
return err
2015-08-28 08:51:15 +03:00
} else if ! repo . Owner . IsOrganization ( ) {
return fmt . Errorf ( "owner is not an organization: %d" , repo . OwnerID )
2015-02-05 16:29:08 +03:00
}
2015-08-28 08:51:15 +03:00
2015-07-24 11:52:01 +03:00
if err = repo . refreshCollaboratorAccesses ( e , accessMap ) ; err != nil {
return fmt . Errorf ( "refreshCollaboratorAccesses: %v" , err )
}
2015-02-05 16:29:08 +03:00
2021-08-12 15:43:08 +03:00
if err = repo . Owner . loadTeams ( e ) ; err != nil {
2015-08-28 08:51:15 +03:00
return err
}
2015-03-25 01:14:04 +03:00
2015-08-28 08:51:15 +03:00
for _ , t := range repo . Owner . Teams {
if t . ID == ignTeamID {
continue
}
2015-02-05 16:29:08 +03:00
2015-08-28 08:51:15 +03:00
// Owner team gets owner access, and skip for teams that do not
// have relations with repository.
if t . IsOwnerTeam ( ) {
2016-11-07 19:20:37 +03:00
t . Authorize = AccessModeOwner
2015-08-28 08:51:15 +03:00
} else if ! t . hasRepository ( e , repo . ID ) {
continue
}
if err = t . getMembers ( e ) ; err != nil {
return fmt . Errorf ( "getMembers '%d': %v" , t . ID , err )
}
for _ , m := range t . Members {
2020-01-13 20:33:46 +03:00
updateUserAccess ( accessMap , m , t . Authorize )
2015-02-05 16:29:08 +03:00
}
}
2015-03-01 05:44:09 +03:00
return repo . refreshAccesses ( e , accessMap )
}
2015-02-05 16:29:08 +03:00
2019-10-15 03:55:21 +03:00
// recalculateUserAccess recalculates new access for a single user
// Usable if we know access only affected one user
func ( repo * Repository ) recalculateUserAccess ( e Engine , uid int64 ) ( err error ) {
minMode := AccessModeRead
if ! repo . IsPrivate {
minMode = AccessModeWrite
}
accessMode := AccessModeNone
collaborator , err := repo . getCollaboration ( e , uid )
if err != nil {
return err
} else if collaborator != nil {
accessMode = collaborator . Mode
}
if err = repo . getOwner ( e ) ; err != nil {
return err
} else if repo . Owner . IsOrganization ( ) {
var teams [ ] Team
if err := e . Join ( "INNER" , "team_repo" , "team_repo.team_id = team.id" ) .
Join ( "INNER" , "team_user" , "team_user.team_id = team.id" ) .
Where ( "team.org_id = ?" , repo . OwnerID ) .
And ( "team_repo.repo_id=?" , repo . ID ) .
And ( "team_user.uid=?" , uid ) .
Find ( & teams ) ; err != nil {
return err
}
for _ , t := range teams {
if t . IsOwnerTeam ( ) {
t . Authorize = AccessModeOwner
}
accessMode = maxAccessMode ( accessMode , t . Authorize )
}
}
// Delete old user accesses and insert new one for repository.
if _ , err = e . Delete ( & Access { RepoID : repo . ID , UserID : uid } ) ; err != nil {
return fmt . Errorf ( "delete old user accesses: %v" , err )
} else if accessMode >= minMode {
if _ , err = e . Insert ( & Access { RepoID : repo . ID , UserID : uid , Mode : accessMode } ) ; err != nil {
return fmt . Errorf ( "insert new user accesses: %v" , err )
}
}
return nil
}
2015-03-01 05:44:09 +03:00
func ( repo * Repository ) recalculateAccesses ( e Engine ) error {
2015-08-28 08:51:15 +03:00
if repo . Owner . IsOrganization ( ) {
return repo . recalculateTeamAccesses ( e , 0 )
}
2020-01-13 20:33:46 +03:00
accessMap := make ( map [ int64 ] * userAccess , 20 )
2015-03-01 05:44:09 +03:00
if err := repo . refreshCollaboratorAccesses ( e , accessMap ) ; err != nil {
return fmt . Errorf ( "refreshCollaboratorAccesses: %v" , err )
2015-02-05 16:29:08 +03:00
}
2015-03-01 05:44:09 +03:00
return repo . refreshAccesses ( e , accessMap )
2015-02-13 08:58:46 +03:00
}
2015-02-05 16:29:08 +03:00
2015-02-13 08:58:46 +03:00
// RecalculateAccesses recalculates all accesses for repository.
2016-11-26 03:07:57 +03:00
func ( repo * Repository ) RecalculateAccesses ( ) error {
return repo . recalculateAccesses ( x )
2015-02-05 16:29:08 +03:00
}