mirror of
				https://gitee.com/gitea/gitea
				synced 2025-11-04 16:40:24 +08:00 
			
		
		
		
	* Move access and repo permission to models/perm/access * fix test * fix git test * Move functions sequence * Some improvements per @KN4CK3R and @delvh * Move issues related code to models/issues * Move some issues related sub package * Merge * Fix test * Fix test * Fix test * Fix test * Rename some files
		
			
				
	
	
		
			211 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			211 lines
		
	
	
		
			5.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2018 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 issues
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"fmt"
 | 
						|
 | 
						|
	"code.gitea.io/gitea/models/db"
 | 
						|
	user_model "code.gitea.io/gitea/models/user"
 | 
						|
	"code.gitea.io/gitea/modules/timeutil"
 | 
						|
)
 | 
						|
 | 
						|
// ErrDependencyExists represents a "DependencyAlreadyExists" kind of error.
 | 
						|
type ErrDependencyExists struct {
 | 
						|
	IssueID      int64
 | 
						|
	DependencyID int64
 | 
						|
}
 | 
						|
 | 
						|
// IsErrDependencyExists checks if an error is a ErrDependencyExists.
 | 
						|
func IsErrDependencyExists(err error) bool {
 | 
						|
	_, ok := err.(ErrDependencyExists)
 | 
						|
	return ok
 | 
						|
}
 | 
						|
 | 
						|
func (err ErrDependencyExists) Error() string {
 | 
						|
	return fmt.Sprintf("issue dependency does already exist [issue id: %d, dependency id: %d]", err.IssueID, err.DependencyID)
 | 
						|
}
 | 
						|
 | 
						|
// ErrDependencyNotExists represents a "DependencyAlreadyExists" kind of error.
 | 
						|
type ErrDependencyNotExists struct {
 | 
						|
	IssueID      int64
 | 
						|
	DependencyID int64
 | 
						|
}
 | 
						|
 | 
						|
// IsErrDependencyNotExists checks if an error is a ErrDependencyExists.
 | 
						|
func IsErrDependencyNotExists(err error) bool {
 | 
						|
	_, ok := err.(ErrDependencyNotExists)
 | 
						|
	return ok
 | 
						|
}
 | 
						|
 | 
						|
func (err ErrDependencyNotExists) Error() string {
 | 
						|
	return fmt.Sprintf("issue dependency does not exist [issue id: %d, dependency id: %d]", err.IssueID, err.DependencyID)
 | 
						|
}
 | 
						|
 | 
						|
// ErrCircularDependency represents a "DependencyCircular" kind of error.
 | 
						|
type ErrCircularDependency struct {
 | 
						|
	IssueID      int64
 | 
						|
	DependencyID int64
 | 
						|
}
 | 
						|
 | 
						|
// IsErrCircularDependency checks if an error is a ErrCircularDependency.
 | 
						|
func IsErrCircularDependency(err error) bool {
 | 
						|
	_, ok := err.(ErrCircularDependency)
 | 
						|
	return ok
 | 
						|
}
 | 
						|
 | 
						|
func (err ErrCircularDependency) Error() string {
 | 
						|
	return fmt.Sprintf("circular dependencies exists (two issues blocking each other) [issue id: %d, dependency id: %d]", err.IssueID, err.DependencyID)
 | 
						|
}
 | 
						|
 | 
						|
// ErrDependenciesLeft represents an error where the issue you're trying to close still has dependencies left.
 | 
						|
type ErrDependenciesLeft struct {
 | 
						|
	IssueID int64
 | 
						|
}
 | 
						|
 | 
						|
// IsErrDependenciesLeft checks if an error is a ErrDependenciesLeft.
 | 
						|
func IsErrDependenciesLeft(err error) bool {
 | 
						|
	_, ok := err.(ErrDependenciesLeft)
 | 
						|
	return ok
 | 
						|
}
 | 
						|
 | 
						|
func (err ErrDependenciesLeft) Error() string {
 | 
						|
	return fmt.Sprintf("issue has open dependencies [issue id: %d]", err.IssueID)
 | 
						|
}
 | 
						|
 | 
						|
// ErrUnknownDependencyType represents an error where an unknown dependency type was passed
 | 
						|
type ErrUnknownDependencyType struct {
 | 
						|
	Type DependencyType
 | 
						|
}
 | 
						|
 | 
						|
// IsErrUnknownDependencyType checks if an error is ErrUnknownDependencyType
 | 
						|
func IsErrUnknownDependencyType(err error) bool {
 | 
						|
	_, ok := err.(ErrUnknownDependencyType)
 | 
						|
	return ok
 | 
						|
}
 | 
						|
 | 
						|
func (err ErrUnknownDependencyType) Error() string {
 | 
						|
	return fmt.Sprintf("unknown dependency type [type: %d]", err.Type)
 | 
						|
}
 | 
						|
 | 
						|
// IssueDependency represents an issue dependency
 | 
						|
type IssueDependency struct {
 | 
						|
	ID           int64              `xorm:"pk autoincr"`
 | 
						|
	UserID       int64              `xorm:"NOT NULL"`
 | 
						|
	IssueID      int64              `xorm:"UNIQUE(issue_dependency) NOT NULL"`
 | 
						|
	DependencyID int64              `xorm:"UNIQUE(issue_dependency) NOT NULL"`
 | 
						|
	CreatedUnix  timeutil.TimeStamp `xorm:"created"`
 | 
						|
	UpdatedUnix  timeutil.TimeStamp `xorm:"updated"`
 | 
						|
}
 | 
						|
 | 
						|
func init() {
 | 
						|
	db.RegisterModel(new(IssueDependency))
 | 
						|
}
 | 
						|
 | 
						|
// DependencyType Defines Dependency Type Constants
 | 
						|
type DependencyType int
 | 
						|
 | 
						|
// Define Dependency Types
 | 
						|
const (
 | 
						|
	DependencyTypeBlockedBy DependencyType = iota
 | 
						|
	DependencyTypeBlocking
 | 
						|
)
 | 
						|
 | 
						|
// CreateIssueDependency creates a new dependency for an issue
 | 
						|
func CreateIssueDependency(user *user_model.User, issue, dep *Issue) error {
 | 
						|
	ctx, committer, err := db.TxContext()
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	defer committer.Close()
 | 
						|
 | 
						|
	// Check if it aleready exists
 | 
						|
	exists, err := issueDepExists(ctx, issue.ID, dep.ID)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if exists {
 | 
						|
		return ErrDependencyExists{issue.ID, dep.ID}
 | 
						|
	}
 | 
						|
	// And if it would be circular
 | 
						|
	circular, err := issueDepExists(ctx, dep.ID, issue.ID)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if circular {
 | 
						|
		return ErrCircularDependency{issue.ID, dep.ID}
 | 
						|
	}
 | 
						|
 | 
						|
	if err := db.Insert(ctx, &IssueDependency{
 | 
						|
		UserID:       user.ID,
 | 
						|
		IssueID:      issue.ID,
 | 
						|
		DependencyID: dep.ID,
 | 
						|
	}); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	// Add comment referencing the new dependency
 | 
						|
	if err = createIssueDependencyComment(ctx, user, issue, dep, true); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	return committer.Commit()
 | 
						|
}
 | 
						|
 | 
						|
// RemoveIssueDependency removes a dependency from an issue
 | 
						|
func RemoveIssueDependency(user *user_model.User, issue, dep *Issue, depType DependencyType) (err error) {
 | 
						|
	ctx, committer, err := db.TxContext()
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	defer committer.Close()
 | 
						|
 | 
						|
	var issueDepToDelete IssueDependency
 | 
						|
 | 
						|
	switch depType {
 | 
						|
	case DependencyTypeBlockedBy:
 | 
						|
		issueDepToDelete = IssueDependency{IssueID: issue.ID, DependencyID: dep.ID}
 | 
						|
	case DependencyTypeBlocking:
 | 
						|
		issueDepToDelete = IssueDependency{IssueID: dep.ID, DependencyID: issue.ID}
 | 
						|
	default:
 | 
						|
		return ErrUnknownDependencyType{depType}
 | 
						|
	}
 | 
						|
 | 
						|
	affected, err := db.GetEngine(ctx).Delete(&issueDepToDelete)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	// If we deleted nothing, the dependency did not exist
 | 
						|
	if affected <= 0 {
 | 
						|
		return ErrDependencyNotExists{issue.ID, dep.ID}
 | 
						|
	}
 | 
						|
 | 
						|
	// Add comment referencing the removed dependency
 | 
						|
	if err = createIssueDependencyComment(ctx, user, issue, dep, false); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	return committer.Commit()
 | 
						|
}
 | 
						|
 | 
						|
// Check if the dependency already exists
 | 
						|
func issueDepExists(ctx context.Context, issueID, depID int64) (bool, error) {
 | 
						|
	return db.GetEngine(ctx).Where("(issue_id = ? AND dependency_id = ?)", issueID, depID).Exist(&IssueDependency{})
 | 
						|
}
 | 
						|
 | 
						|
// IssueNoDependenciesLeft checks if issue can be closed
 | 
						|
func IssueNoDependenciesLeft(ctx context.Context, issue *Issue) (bool, error) {
 | 
						|
	exists, err := db.GetEngine(ctx).
 | 
						|
		Table("issue_dependency").
 | 
						|
		Select("issue.*").
 | 
						|
		Join("INNER", "issue", "issue.id = issue_dependency.dependency_id").
 | 
						|
		Where("issue_dependency.issue_id = ?", issue.ID).
 | 
						|
		And("issue.is_closed = ?", "0").
 | 
						|
		Exist(&Issue{})
 | 
						|
 | 
						|
	return !exists, err
 | 
						|
}
 |