2016-03-06 02:08:42 +03:00
// Copyright 2016 The Gogs Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package models
import (
"fmt"
)
// Collaboration represent the relation between an individual and a repository.
type Collaboration struct {
2016-03-10 03:53:30 +03:00
ID int64 ` xorm:"pk autoincr" `
RepoID int64 ` xorm:"UNIQUE(s) INDEX NOT NULL" `
UserID int64 ` xorm:"UNIQUE(s) INDEX NOT NULL" `
Mode AccessMode ` xorm:"DEFAULT 2 NOT NULL" `
2016-03-06 02:08:42 +03:00
}
2016-11-28 19:58:59 +03:00
// ModeI18nKey returns the collaboration mode I18n Key
2016-06-27 19:22:30 +03:00
func ( c * Collaboration ) ModeI18nKey ( ) string {
2016-03-06 02:08:42 +03:00
switch c . Mode {
2016-11-07 19:20:37 +03:00
case AccessModeRead :
2016-06-27 19:22:30 +03:00
return "repo.settings.collaboration.read"
2016-11-07 19:20:37 +03:00
case AccessModeWrite :
2016-06-27 19:22:30 +03:00
return "repo.settings.collaboration.write"
2016-11-07 19:20:37 +03:00
case AccessModeAdmin :
2016-06-27 19:22:30 +03:00
return "repo.settings.collaboration.admin"
default :
return "repo.settings.collaboration.undefined"
2016-03-06 02:08:42 +03:00
}
}
2016-08-11 21:35:46 +03:00
// AddCollaborator adds new collaboration to a repository with default access mode.
2016-03-06 02:08:42 +03:00
func ( repo * Repository ) AddCollaborator ( u * User ) error {
collaboration := & Collaboration {
RepoID : repo . ID ,
2016-07-23 20:08:22 +03:00
UserID : u . ID ,
2016-03-06 02:08:42 +03:00
}
has , err := x . Get ( collaboration )
if err != nil {
return err
} else if has {
return nil
}
2016-11-07 19:20:37 +03:00
collaboration . Mode = AccessModeWrite
2016-03-06 02:08:42 +03:00
sess := x . NewSession ( )
2017-06-21 03:57:05 +03:00
defer sess . Close ( )
2016-03-06 02:08:42 +03:00
if err = sess . Begin ( ) ; err != nil {
return err
}
if _ , err = sess . InsertOne ( collaboration ) ; err != nil {
return err
}
if repo . Owner . IsOrganization ( ) {
err = repo . recalculateTeamAccesses ( sess , 0 )
} else {
err = repo . recalculateAccesses ( sess )
}
if err != nil {
return fmt . Errorf ( "recalculateAccesses 'team=%v': %v" , repo . Owner . IsOrganization ( ) , err )
}
return sess . Commit ( )
}
func ( repo * Repository ) getCollaborations ( e Engine ) ( [ ] * Collaboration , error ) {
2016-11-29 16:05:26 +03:00
var collaborations [ ] * Collaboration
2016-03-06 02:08:42 +03:00
return collaborations , e . Find ( & collaborations , & Collaboration { RepoID : repo . ID } )
}
// Collaborator represents a user with collaboration details.
type Collaborator struct {
* User
Collaboration * Collaboration
}
func ( repo * Repository ) getCollaborators ( e Engine ) ( [ ] * Collaborator , error ) {
collaborations , err := repo . getCollaborations ( e )
if err != nil {
return nil , fmt . Errorf ( "getCollaborations: %v" , err )
}
collaborators := make ( [ ] * Collaborator , len ( collaborations ) )
for i , c := range collaborations {
user , err := getUserByID ( e , c . UserID )
if err != nil {
return nil , err
}
collaborators [ i ] = & Collaborator {
User : user ,
Collaboration : c ,
}
}
return collaborators , nil
}
// GetCollaborators returns the collaborators for a repository
func ( repo * Repository ) GetCollaborators ( ) ( [ ] * Collaborator , error ) {
return repo . getCollaborators ( x )
}
2016-12-26 10:37:01 +03:00
func ( repo * Repository ) isCollaborator ( e Engine , userID int64 ) ( bool , error ) {
return e . Get ( & Collaboration { RepoID : repo . ID , UserID : userID } )
}
// IsCollaborator check if a user is a collaborator of a repository
func ( repo * Repository ) IsCollaborator ( userID int64 ) ( bool , error ) {
return repo . isCollaborator ( x , userID )
}
2016-03-06 02:08:42 +03:00
// ChangeCollaborationAccessMode sets new access mode for the collaboration.
func ( repo * Repository ) ChangeCollaborationAccessMode ( uid int64 , mode AccessMode ) error {
// Discard invalid input
2016-11-07 19:20:37 +03:00
if mode <= AccessModeNone || mode > AccessModeOwner {
2016-03-06 02:08:42 +03:00
return nil
}
collaboration := & Collaboration {
RepoID : repo . ID ,
UserID : uid ,
}
has , err := x . Get ( collaboration )
if err != nil {
return fmt . Errorf ( "get collaboration: %v" , err )
} else if ! has {
return nil
}
2016-08-11 21:35:46 +03:00
if collaboration . Mode == mode {
return nil
}
2016-03-06 02:08:42 +03:00
collaboration . Mode = mode
sess := x . NewSession ( )
2017-06-21 03:57:05 +03:00
defer sess . Close ( )
2016-03-06 02:08:42 +03:00
if err = sess . Begin ( ) ; err != nil {
return err
}
2016-11-10 18:16:32 +03:00
if _ , err = sess .
Id ( collaboration . ID ) .
2017-09-25 07:59:27 +03:00
Cols ( "mode" ) .
2016-11-10 18:16:32 +03:00
Update ( collaboration ) ; err != nil {
2016-03-06 02:08:42 +03:00
return fmt . Errorf ( "update collaboration: %v" , err )
} else if _ , err = sess . Exec ( "UPDATE access SET mode = ? WHERE user_id = ? AND repo_id = ?" , mode , uid , repo . ID ) ; err != nil {
return fmt . Errorf ( "update access table: %v" , err )
}
return sess . Commit ( )
}
// DeleteCollaboration removes collaboration relation between the user and repository.
func ( repo * Repository ) DeleteCollaboration ( uid int64 ) ( err error ) {
collaboration := & Collaboration {
RepoID : repo . ID ,
UserID : uid ,
}
sess := x . NewSession ( )
2017-06-21 03:57:05 +03:00
defer sess . Close ( )
2016-03-06 02:08:42 +03:00
if err = sess . Begin ( ) ; err != nil {
return err
}
if has , err := sess . Delete ( collaboration ) ; err != nil || has == 0 {
return err
} else if err = repo . recalculateAccesses ( sess ) ; err != nil {
return err
}
return sess . Commit ( )
}