| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586 | // Copyright 2020 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 dbimport (	"github.com/jinzhu/gorm"	"github.com/t-tiger/gorm-bulk-insert"	log "unknwon.dev/clog/v2")// PermsStore is the persistent interface for permissions.//// NOTE: All methods are sorted in alphabetical order.type PermsStore interface {	// AccessMode returns the access mode of given user has to the repository.	AccessMode(userID int64, repo *Repository) AccessMode	// Authorize returns true if the user has as good as desired access mode to the repository.	Authorize(userID int64, repo *Repository, desired AccessMode) bool	// SetRepoPerms does a full update to which users have which level of access to given repository.	// Keys of the "accessMap" are user IDs.	SetRepoPerms(repoID int64, accessMap map[int64]AccessMode) error}var Perms PermsStorevar _ PermsStore = (*perms)(nil)type perms struct {	*gorm.DB}func (db *perms) AccessMode(userID int64, repo *Repository) (mode AccessMode) {	if repo == nil {		return AccessModeNone	}	// Everyone has read access to public repository.	if !repo.IsPrivate {		mode = AccessModeRead	}	// Anonymous user gets the default access.	if userID <= 0 {		return mode	}	if userID == repo.OwnerID {		return AccessModeOwner	}	access := new(Access)	err := db.Where("user_id = ? AND repo_id = ?", userID, repo.ID).First(access).Error	if err != nil {		if !gorm.IsRecordNotFoundError(err) {			log.Error("Failed to get access [user_id: %d, repo_id: %d]: %v", userID, repo.ID, err)		}		return mode	}	return access.Mode}func (db *perms) Authorize(userID int64, repo *Repository, desired AccessMode) bool {	return desired <= db.AccessMode(userID, repo)}func (db *perms) SetRepoPerms(repoID int64, accessMap map[int64]AccessMode) error {	records := make([]interface{}, 0, len(accessMap))	for userID, mode := range accessMap {		records = append(records, &Access{			UserID: userID,			RepoID: repoID,			Mode:   mode,		})	}	return db.Transaction(func(tx *gorm.DB) error {		err := tx.Where("repo_id = ?", repoID).Delete(new(Access)).Error		if err != nil {			return err		}		return gormbulk.BulkInsert(tx, records, 3000)	})}
 |