mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-27 04:21:11 +00:00 
			
		
		
		
	* Add timestamps to Star, Label, LanguageStat, Follow, Watch and Collaboration * Star do not need updated * LanguageStat do not need update (they wont change) * fix unit-test
		
			
				
	
	
		
			317 lines
		
	
	
	
		
			8.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			317 lines
		
	
	
	
		
			8.6 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"
 | |
| 
 | |
| 	"code.gitea.io/gitea/modules/setting"
 | |
| 	"code.gitea.io/gitea/modules/timeutil"
 | |
| )
 | |
| 
 | |
| // RepoWatchMode specifies what kind of watch the user has on a repository
 | |
| type RepoWatchMode int8
 | |
| 
 | |
| const (
 | |
| 	// RepoWatchModeNone don't watch
 | |
| 	RepoWatchModeNone RepoWatchMode = iota // 0
 | |
| 	// RepoWatchModeNormal watch repository (from other sources)
 | |
| 	RepoWatchModeNormal // 1
 | |
| 	// RepoWatchModeDont explicit don't auto-watch
 | |
| 	RepoWatchModeDont // 2
 | |
| 	// RepoWatchModeAuto watch repository (from AutoWatchOnChanges)
 | |
| 	RepoWatchModeAuto // 3
 | |
| )
 | |
| 
 | |
| // 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)"`
 | |
| 	Mode        RepoWatchMode      `xorm:"SMALLINT NOT NULL DEFAULT 1"`
 | |
| 	CreatedUnix timeutil.TimeStamp `xorm:"INDEX created"`
 | |
| 	UpdatedUnix timeutil.TimeStamp `xorm:"INDEX updated"`
 | |
| }
 | |
| 
 | |
| // getWatch gets what kind of subscription a user has on a given repository; returns dummy record if none found
 | |
| func getWatch(e Engine, userID, repoID int64) (Watch, error) {
 | |
| 	watch := Watch{UserID: userID, RepoID: repoID}
 | |
| 	has, err := e.Get(&watch)
 | |
| 	if err != nil {
 | |
| 		return watch, err
 | |
| 	}
 | |
| 	if !has {
 | |
| 		watch.Mode = RepoWatchModeNone
 | |
| 	}
 | |
| 	return watch, nil
 | |
| }
 | |
| 
 | |
| // Decodes watchability of RepoWatchMode
 | |
| func isWatchMode(mode RepoWatchMode) bool {
 | |
| 	return mode != RepoWatchModeNone && mode != RepoWatchModeDont
 | |
| }
 | |
| 
 | |
| // IsWatching checks if user has watched given repository.
 | |
| func IsWatching(userID, repoID int64) bool {
 | |
| 	watch, err := getWatch(x, userID, repoID)
 | |
| 	return err == nil && isWatchMode(watch.Mode)
 | |
| }
 | |
| 
 | |
| func watchRepoMode(e Engine, watch Watch, mode RepoWatchMode) (err error) {
 | |
| 	if watch.Mode == mode {
 | |
| 		return nil
 | |
| 	}
 | |
| 	if mode == RepoWatchModeAuto && (watch.Mode == RepoWatchModeDont || isWatchMode(watch.Mode)) {
 | |
| 		// Don't auto watch if already watching or deliberately not watching
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	hadrec := watch.Mode != RepoWatchModeNone
 | |
| 	needsrec := mode != RepoWatchModeNone
 | |
| 	repodiff := 0
 | |
| 
 | |
| 	if isWatchMode(mode) && !isWatchMode(watch.Mode) {
 | |
| 		repodiff = 1
 | |
| 	} else if !isWatchMode(mode) && isWatchMode(watch.Mode) {
 | |
| 		repodiff = -1
 | |
| 	}
 | |
| 
 | |
| 	watch.Mode = mode
 | |
| 
 | |
| 	if !hadrec && needsrec {
 | |
| 		watch.Mode = mode
 | |
| 		if _, err = e.Insert(watch); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	} else if needsrec {
 | |
| 		watch.Mode = mode
 | |
| 		if _, err := e.ID(watch.ID).AllCols().Update(watch); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	} else if _, err = e.Delete(Watch{ID: watch.ID}); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	if repodiff != 0 {
 | |
| 		_, err = e.Exec("UPDATE `repository` SET num_watches = num_watches + ? WHERE id = ?", repodiff, watch.RepoID)
 | |
| 	}
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| // WatchRepoMode watch repository in specific mode.
 | |
| func WatchRepoMode(userID, repoID int64, mode RepoWatchMode) (err error) {
 | |
| 	var watch Watch
 | |
| 	if watch, err = getWatch(x, userID, repoID); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	return watchRepoMode(x, watch, mode)
 | |
| }
 | |
| 
 | |
| func watchRepo(e Engine, userID, repoID int64, doWatch bool) (err error) {
 | |
| 	var watch Watch
 | |
| 	if watch, err = getWatch(e, userID, repoID); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	if !doWatch && watch.Mode == RepoWatchModeAuto {
 | |
| 		err = watchRepoMode(e, watch, RepoWatchModeDont)
 | |
| 	} else if !doWatch {
 | |
| 		err = watchRepoMode(e, watch, RepoWatchModeNone)
 | |
| 	} else {
 | |
| 		err = watchRepoMode(e, watch, RepoWatchModeNormal)
 | |
| 	}
 | |
| 	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("`watch`.mode<>?", RepoWatchModeDont).
 | |
| 		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)
 | |
| }
 | |
| 
 | |
| // GetRepoWatchersIDs returns IDs of watchers for a given repo ID
 | |
| // but avoids joining with `user` for performance reasons
 | |
| // User permissions must be verified elsewhere if required
 | |
| func GetRepoWatchersIDs(repoID int64) ([]int64, error) {
 | |
| 	return getRepoWatchersIDs(x, repoID)
 | |
| }
 | |
| 
 | |
| func getRepoWatchersIDs(e Engine, repoID int64) ([]int64, error) {
 | |
| 	ids := make([]int64, 0, 64)
 | |
| 	return ids, e.Table("watch").
 | |
| 		Where("watch.repo_id=?", repoID).
 | |
| 		And("watch.mode<>?", RepoWatchModeDont).
 | |
| 		Select("user_id").
 | |
| 		Find(&ids)
 | |
| }
 | |
| 
 | |
| // GetWatchers returns range of users watching given repository.
 | |
| func (repo *Repository) GetWatchers(opts ListOptions) ([]*User, error) {
 | |
| 	sess := x.Where("watch.repo_id=?", repo.ID).
 | |
| 		Join("LEFT", "watch", "`user`.id=`watch`.user_id").
 | |
| 		And("`watch`.mode<>?", RepoWatchModeDont)
 | |
| 	if opts.Page > 0 {
 | |
| 		sess = opts.setSessionPagination(sess)
 | |
| 		users := make([]*User, 0, opts.PageSize)
 | |
| 
 | |
| 		return users, sess.Find(&users)
 | |
| 	}
 | |
| 
 | |
| 	users := make([]*User, 0, 8)
 | |
| 	return users, sess.Find(&users)
 | |
| }
 | |
| 
 | |
| func notifyWatchers(e Engine, actions ...*Action) error {
 | |
| 	var watchers []*Watch
 | |
| 	var repo *Repository
 | |
| 	var err error
 | |
| 	var permCode []bool
 | |
| 	var permIssue []bool
 | |
| 	var permPR []bool
 | |
| 
 | |
| 	for _, act := range actions {
 | |
| 		repoChanged := repo == nil || repo.ID != act.RepoID
 | |
| 
 | |
| 		if repoChanged {
 | |
| 			// Add feeds for user self and all watchers.
 | |
| 			watchers, 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)
 | |
| 		}
 | |
| 
 | |
| 		if repoChanged {
 | |
| 			act.loadRepo()
 | |
| 			repo = act.Repo
 | |
| 
 | |
| 			// check repo owner exist.
 | |
| 			if err := act.Repo.getOwner(e); err != nil {
 | |
| 				return fmt.Errorf("can't get repo owner: %v", err)
 | |
| 			}
 | |
| 		} else if act.Repo == nil {
 | |
| 			act.Repo = repo
 | |
| 		}
 | |
| 
 | |
| 		// 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)
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if repoChanged {
 | |
| 			permCode = make([]bool, len(watchers))
 | |
| 			permIssue = make([]bool, len(watchers))
 | |
| 			permPR = make([]bool, len(watchers))
 | |
| 			for i, watcher := range watchers {
 | |
| 				user, err := getUserByID(e, watcher.UserID)
 | |
| 				if err != nil {
 | |
| 					permCode[i] = false
 | |
| 					permIssue[i] = false
 | |
| 					permPR[i] = false
 | |
| 					continue
 | |
| 				}
 | |
| 				perm, err := getUserRepoPermission(e, repo, user)
 | |
| 				if err != nil {
 | |
| 					permCode[i] = false
 | |
| 					permIssue[i] = false
 | |
| 					permPR[i] = false
 | |
| 					continue
 | |
| 				}
 | |
| 				permCode[i] = perm.CanRead(UnitTypeCode)
 | |
| 				permIssue[i] = perm.CanRead(UnitTypeIssues)
 | |
| 				permPR[i] = perm.CanRead(UnitTypePullRequests)
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		for i, watcher := range watchers {
 | |
| 			if act.ActUserID == watcher.UserID {
 | |
| 				continue
 | |
| 			}
 | |
| 			act.ID = 0
 | |
| 			act.UserID = watcher.UserID
 | |
| 			act.Repo.Units = nil
 | |
| 
 | |
| 			switch act.OpType {
 | |
| 			case ActionCommitRepo, ActionPushTag, ActionDeleteTag, ActionPublishRelease, ActionDeleteBranch:
 | |
| 				if !permCode[i] {
 | |
| 					continue
 | |
| 				}
 | |
| 			case ActionCreateIssue, ActionCommentIssue, ActionCloseIssue, ActionReopenIssue:
 | |
| 				if !permIssue[i] {
 | |
| 					continue
 | |
| 				}
 | |
| 			case ActionCreatePullRequest, ActionCommentPull, ActionMergePullRequest, ActionClosePullRequest, ActionReopenPullRequest:
 | |
| 				if !permPR[i] {
 | |
| 					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(actions ...*Action) error {
 | |
| 	return notifyWatchers(x, actions...)
 | |
| }
 | |
| 
 | |
| // NotifyWatchersActions creates batch of actions for every watcher.
 | |
| func NotifyWatchersActions(acts []*Action) error {
 | |
| 	sess := x.NewSession()
 | |
| 	defer sess.Close()
 | |
| 	if err := sess.Begin(); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	for _, act := range acts {
 | |
| 		if err := notifyWatchers(sess, act); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 	return sess.Commit()
 | |
| }
 | |
| 
 | |
| func watchIfAuto(e Engine, userID, repoID int64, isWrite bool) error {
 | |
| 	if !isWrite || !setting.Service.AutoWatchOnChanges {
 | |
| 		return nil
 | |
| 	}
 | |
| 	watch, err := getWatch(e, userID, repoID)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	if watch.Mode != RepoWatchModeNone {
 | |
| 		return nil
 | |
| 	}
 | |
| 	return watchRepoMode(e, watch, RepoWatchModeAuto)
 | |
| }
 | |
| 
 | |
| // WatchIfAuto subscribes to repo if AutoWatchOnChanges is set
 | |
| func WatchIfAuto(userID int64, repoID int64, isWrite bool) error {
 | |
| 	return watchIfAuto(x, userID, repoID, isWrite)
 | |
| }
 |