mirror of
				https://codeberg.org/forgejo/forgejo.git
				synced 2025-10-22 01:52:24 +00:00 
			
		
		
		
	Resolve #25233 <img width="1315" alt="图片" src="https://github.com/go-gitea/gitea/assets/81045/3ba59b58-471a-4e1b-985c-87edac2268c0"> <img width="1297" alt="图片" src="https://github.com/go-gitea/gitea/assets/81045/b6caa12f-323b-4f70-9c44-ef91cb71a26c">
		
			
				
	
	
		
			149 lines
		
	
	
	
		
			4.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			149 lines
		
	
	
	
		
			4.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2023 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package git
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 
 | |
| 	"code.gitea.io/gitea/models/db"
 | |
| 	user_model "code.gitea.io/gitea/models/user"
 | |
| 	"code.gitea.io/gitea/modules/container"
 | |
| 	"code.gitea.io/gitea/modules/util"
 | |
| 
 | |
| 	"xorm.io/builder"
 | |
| 	"xorm.io/xorm"
 | |
| )
 | |
| 
 | |
| type BranchList []*Branch
 | |
| 
 | |
| func (branches BranchList) LoadDeletedBy(ctx context.Context) error {
 | |
| 	ids := container.Set[int64]{}
 | |
| 	for _, branch := range branches {
 | |
| 		if !branch.IsDeleted {
 | |
| 			continue
 | |
| 		}
 | |
| 		ids.Add(branch.DeletedByID)
 | |
| 	}
 | |
| 	usersMap := make(map[int64]*user_model.User, len(ids))
 | |
| 	if err := db.GetEngine(ctx).In("id", ids.Values()).Find(&usersMap); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	for _, branch := range branches {
 | |
| 		if !branch.IsDeleted {
 | |
| 			continue
 | |
| 		}
 | |
| 		branch.DeletedBy = usersMap[branch.DeletedByID]
 | |
| 		if branch.DeletedBy == nil {
 | |
| 			branch.DeletedBy = user_model.NewGhostUser()
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (branches BranchList) LoadPusher(ctx context.Context) error {
 | |
| 	ids := container.Set[int64]{}
 | |
| 	for _, branch := range branches {
 | |
| 		if branch.PusherID > 0 { // pusher_id maybe zero because some branches are sync by backend with no pusher
 | |
| 			ids.Add(branch.PusherID)
 | |
| 		}
 | |
| 	}
 | |
| 	usersMap := make(map[int64]*user_model.User, len(ids))
 | |
| 	if err := db.GetEngine(ctx).In("id", ids.Values()).Find(&usersMap); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	for _, branch := range branches {
 | |
| 		if branch.PusherID <= 0 {
 | |
| 			continue
 | |
| 		}
 | |
| 		branch.Pusher = usersMap[branch.PusherID]
 | |
| 		if branch.Pusher == nil {
 | |
| 			branch.Pusher = user_model.NewGhostUser()
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| type FindBranchOptions struct {
 | |
| 	db.ListOptions
 | |
| 	RepoID             int64
 | |
| 	ExcludeBranchNames []string
 | |
| 	IsDeletedBranch    util.OptionalBool
 | |
| 	OrderBy            string
 | |
| 	Keyword            string
 | |
| }
 | |
| 
 | |
| func (opts *FindBranchOptions) Cond() builder.Cond {
 | |
| 	cond := builder.NewCond()
 | |
| 	if opts.RepoID > 0 {
 | |
| 		cond = cond.And(builder.Eq{"repo_id": opts.RepoID})
 | |
| 	}
 | |
| 
 | |
| 	if len(opts.ExcludeBranchNames) > 0 {
 | |
| 		cond = cond.And(builder.NotIn("name", opts.ExcludeBranchNames))
 | |
| 	}
 | |
| 	if !opts.IsDeletedBranch.IsNone() {
 | |
| 		cond = cond.And(builder.Eq{"is_deleted": opts.IsDeletedBranch.IsTrue()})
 | |
| 	}
 | |
| 	if opts.Keyword != "" {
 | |
| 		cond = cond.And(builder.Like{"name", opts.Keyword})
 | |
| 	}
 | |
| 	return cond
 | |
| }
 | |
| 
 | |
| func CountBranches(ctx context.Context, opts FindBranchOptions) (int64, error) {
 | |
| 	return db.GetEngine(ctx).Where(opts.Cond()).Count(&Branch{})
 | |
| }
 | |
| 
 | |
| func orderByBranches(sess *xorm.Session, opts FindBranchOptions) *xorm.Session {
 | |
| 	if !opts.IsDeletedBranch.IsFalse() { // if deleted branch included, put them at the end
 | |
| 		sess = sess.OrderBy("is_deleted ASC")
 | |
| 	}
 | |
| 
 | |
| 	if opts.OrderBy == "" {
 | |
| 		// the commit_time might be the same, so add the "name" to make sure the order is stable
 | |
| 		opts.OrderBy = "commit_time DESC, name ASC"
 | |
| 	}
 | |
| 	return sess.OrderBy(opts.OrderBy)
 | |
| }
 | |
| 
 | |
| func FindBranches(ctx context.Context, opts FindBranchOptions) (BranchList, error) {
 | |
| 	sess := db.GetEngine(ctx).Where(opts.Cond())
 | |
| 	if opts.PageSize > 0 && !opts.IsListAll() {
 | |
| 		sess = db.SetSessionPagination(sess, &opts.ListOptions)
 | |
| 	}
 | |
| 	sess = orderByBranches(sess, opts)
 | |
| 
 | |
| 	var branches []*Branch
 | |
| 	return branches, sess.Find(&branches)
 | |
| }
 | |
| 
 | |
| func FindBranchNames(ctx context.Context, opts FindBranchOptions) ([]string, error) {
 | |
| 	sess := db.GetEngine(ctx).Select("name").Where(opts.Cond())
 | |
| 	if opts.PageSize > 0 && !opts.IsListAll() {
 | |
| 		sess = db.SetSessionPagination(sess, &opts.ListOptions)
 | |
| 	}
 | |
| 	sess = orderByBranches(sess, opts)
 | |
| 	var branches []string
 | |
| 	if err := sess.Table("branch").Find(&branches); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return branches, nil
 | |
| }
 | |
| 
 | |
| func FindBranchesByRepoAndBranchName(ctx context.Context, repoBranches map[int64]string) (map[int64]string, error) {
 | |
| 	cond := builder.NewCond()
 | |
| 	for repoID, branchName := range repoBranches {
 | |
| 		cond = cond.Or(builder.And(builder.Eq{"repo_id": repoID}, builder.Eq{"name": branchName}))
 | |
| 	}
 | |
| 	var branches []*Branch
 | |
| 	if err := db.GetEngine(ctx).
 | |
| 		Where(cond).Find(&branches); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	branchMap := make(map[int64]string, len(branches))
 | |
| 	for _, branch := range branches {
 | |
| 		branchMap[branch.RepoID] = branch.CommitID
 | |
| 	}
 | |
| 	return branchMap, nil
 | |
| }
 |