mirror of
				https://github.com/go-gitea/gitea
				synced 2025-11-03 04:48:25 +00:00 
			
		
		
		
	* Fix sqlite deadlock when assigning to a PR Fix 5639 Signed-off-by: Andrew Thornton <art27@cantab.net> * More possible deadlocks found and fixed Signed-off-by: Andrew Thornton <art27@cantab.net>
		
			
				
	
	
		
			140 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			140 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2017 The Gitea 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"
 | 
						|
 | 
						|
// Watch is connection request for receiving repository notification.
 | 
						|
type Watch struct {
 | 
						|
	ID     int64 `xorm:"pk autoincr"`
 | 
						|
	UserID int64 `xorm:"UNIQUE(watch)"`
 | 
						|
	RepoID int64 `xorm:"UNIQUE(watch)"`
 | 
						|
}
 | 
						|
 | 
						|
func isWatching(e Engine, userID, repoID int64) bool {
 | 
						|
	has, _ := e.Get(&Watch{UserID: userID, RepoID: repoID})
 | 
						|
	return has
 | 
						|
}
 | 
						|
 | 
						|
// IsWatching checks if user has watched given repository.
 | 
						|
func IsWatching(userID, repoID int64) bool {
 | 
						|
	return isWatching(x, userID, repoID)
 | 
						|
}
 | 
						|
 | 
						|
func watchRepo(e Engine, userID, repoID int64, watch bool) (err error) {
 | 
						|
	if watch {
 | 
						|
		if isWatching(e, userID, repoID) {
 | 
						|
			return nil
 | 
						|
		}
 | 
						|
		if _, err = e.Insert(&Watch{RepoID: repoID, UserID: userID}); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		_, err = e.Exec("UPDATE `repository` SET num_watches = num_watches + 1 WHERE id = ?", repoID)
 | 
						|
	} else {
 | 
						|
		if !isWatching(e, userID, repoID) {
 | 
						|
			return nil
 | 
						|
		}
 | 
						|
		if _, err = e.Delete(&Watch{0, userID, repoID}); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
		_, err = e.Exec("UPDATE `repository` SET num_watches = num_watches - 1 WHERE id = ?", repoID)
 | 
						|
	}
 | 
						|
	return err
 | 
						|
}
 | 
						|
 | 
						|
// WatchRepo watch or unwatch repository.
 | 
						|
func WatchRepo(userID, repoID int64, watch bool) (err error) {
 | 
						|
	return watchRepo(x, userID, repoID, watch)
 | 
						|
}
 | 
						|
 | 
						|
func getWatchers(e Engine, repoID int64) ([]*Watch, error) {
 | 
						|
	watches := make([]*Watch, 0, 10)
 | 
						|
	return watches, e.Where("`watch`.repo_id=?", repoID).
 | 
						|
		And("`user`.is_active=?", true).
 | 
						|
		And("`user`.prohibit_login=?", false).
 | 
						|
		Join("INNER", "`user`", "`user`.id = `watch`.user_id").
 | 
						|
		Find(&watches)
 | 
						|
}
 | 
						|
 | 
						|
// GetWatchers returns all watchers of given repository.
 | 
						|
func GetWatchers(repoID int64) ([]*Watch, error) {
 | 
						|
	return getWatchers(x, repoID)
 | 
						|
}
 | 
						|
 | 
						|
// GetWatchers returns range of users watching given repository.
 | 
						|
func (repo *Repository) GetWatchers(page int) ([]*User, error) {
 | 
						|
	users := make([]*User, 0, ItemsPerPage)
 | 
						|
	sess := x.Where("watch.repo_id=?", repo.ID).
 | 
						|
		Join("LEFT", "watch", "`user`.id=`watch`.user_id")
 | 
						|
	if page > 0 {
 | 
						|
		sess = sess.Limit(ItemsPerPage, (page-1)*ItemsPerPage)
 | 
						|
	}
 | 
						|
	return users, sess.Find(&users)
 | 
						|
}
 | 
						|
 | 
						|
func notifyWatchers(e Engine, act *Action) error {
 | 
						|
	// Add feeds for user self and all watchers.
 | 
						|
	watches, err := getWatchers(e, act.RepoID)
 | 
						|
	if err != nil {
 | 
						|
		return fmt.Errorf("get watchers: %v", err)
 | 
						|
	}
 | 
						|
 | 
						|
	// Add feed for actioner.
 | 
						|
	act.UserID = act.ActUserID
 | 
						|
	if _, err = e.InsertOne(act); err != nil {
 | 
						|
		return fmt.Errorf("insert new actioner: %v", err)
 | 
						|
	}
 | 
						|
 | 
						|
	act.loadRepo()
 | 
						|
	// check repo owner exist.
 | 
						|
	if err := act.Repo.getOwner(e); err != nil {
 | 
						|
		return fmt.Errorf("can't get repo owner: %v", err)
 | 
						|
	}
 | 
						|
 | 
						|
	// Add feed for organization
 | 
						|
	if act.Repo.Owner.IsOrganization() && act.ActUserID != act.Repo.Owner.ID {
 | 
						|
		act.ID = 0
 | 
						|
		act.UserID = act.Repo.Owner.ID
 | 
						|
		if _, err = e.InsertOne(act); err != nil {
 | 
						|
			return fmt.Errorf("insert new actioner: %v", err)
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	for i := range watches {
 | 
						|
		if act.ActUserID == watches[i].UserID {
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		act.ID = 0
 | 
						|
		act.UserID = watches[i].UserID
 | 
						|
		act.Repo.Units = nil
 | 
						|
 | 
						|
		switch act.OpType {
 | 
						|
		case ActionCommitRepo, ActionPushTag, ActionDeleteTag, ActionDeleteBranch:
 | 
						|
			if !act.Repo.checkUnitUser(e, act.UserID, false, UnitTypeCode) {
 | 
						|
				continue
 | 
						|
			}
 | 
						|
		case ActionCreateIssue, ActionCommentIssue, ActionCloseIssue, ActionReopenIssue:
 | 
						|
			if !act.Repo.checkUnitUser(e, act.UserID, false, UnitTypeIssues) {
 | 
						|
				continue
 | 
						|
			}
 | 
						|
		case ActionCreatePullRequest, ActionMergePullRequest, ActionClosePullRequest, ActionReopenPullRequest:
 | 
						|
			if !act.Repo.checkUnitUser(e, act.UserID, false, UnitTypePullRequests) {
 | 
						|
				continue
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if _, err = e.InsertOne(act); err != nil {
 | 
						|
			return fmt.Errorf("insert new action: %v", err)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// NotifyWatchers creates batch of actions for every watcher.
 | 
						|
func NotifyWatchers(act *Action) error {
 | 
						|
	return notifyWatchers(x, act)
 | 
						|
}
 |