mirror of
				https://github.com/go-gitea/gitea
				synced 2025-09-28 03:28:13 +00:00 
			
		
		
		
	When transferring repositories that have issues linked to a project board to another organization, the issues remain associated with the original project board. This causes the columns in the project board to become bugged, making it difficult to move other issues in or out of the affected columns. As a solution, I removed the issue relations since the other organization does not have this project table. Fix for #31538 Co-authored-by: Jason Song <i@wolfogre.com>
		
			
				
	
	
		
			126 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			126 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2020 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package project
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"fmt"
 | |
| 
 | |
| 	"code.gitea.io/gitea/models/db"
 | |
| 	"code.gitea.io/gitea/modules/log"
 | |
| 	"code.gitea.io/gitea/modules/util"
 | |
| )
 | |
| 
 | |
| // ProjectIssue saves relation from issue to a project
 | |
| type ProjectIssue struct { //revive:disable-line:exported
 | |
| 	ID        int64 `xorm:"pk autoincr"`
 | |
| 	IssueID   int64 `xorm:"INDEX"`
 | |
| 	ProjectID int64 `xorm:"INDEX"`
 | |
| 
 | |
| 	// ProjectColumnID should not be zero since 1.22. If it's zero, the issue will not be displayed on UI and it might result in errors.
 | |
| 	ProjectColumnID int64 `xorm:"'project_board_id' INDEX"`
 | |
| 
 | |
| 	// the sorting order on the column
 | |
| 	Sorting int64 `xorm:"NOT NULL DEFAULT 0"`
 | |
| }
 | |
| 
 | |
| func init() {
 | |
| 	db.RegisterModel(new(ProjectIssue))
 | |
| }
 | |
| 
 | |
| func deleteProjectIssuesByProjectID(ctx context.Context, projectID int64) error {
 | |
| 	_, err := db.GetEngine(ctx).Where("project_id=?", projectID).Delete(&ProjectIssue{})
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| // NumIssues return counter of all issues assigned to a project
 | |
| func (p *Project) NumIssues(ctx context.Context) int {
 | |
| 	c, err := db.GetEngine(ctx).Table("project_issue").
 | |
| 		Where("project_id=?", p.ID).
 | |
| 		GroupBy("issue_id").
 | |
| 		Cols("issue_id").
 | |
| 		Count()
 | |
| 	if err != nil {
 | |
| 		log.Error("NumIssues: %v", err)
 | |
| 		return 0
 | |
| 	}
 | |
| 	return int(c)
 | |
| }
 | |
| 
 | |
| // NumClosedIssues return counter of closed issues assigned to a project
 | |
| func (p *Project) NumClosedIssues(ctx context.Context) int {
 | |
| 	c, err := db.GetEngine(ctx).Table("project_issue").
 | |
| 		Join("INNER", "issue", "project_issue.issue_id=issue.id").
 | |
| 		Where("project_issue.project_id=? AND issue.is_closed=?", p.ID, true).
 | |
| 		Cols("issue_id").
 | |
| 		Count()
 | |
| 	if err != nil {
 | |
| 		log.Error("NumClosedIssues: %v", err)
 | |
| 		return 0
 | |
| 	}
 | |
| 	return int(c)
 | |
| }
 | |
| 
 | |
| // NumOpenIssues return counter of open issues assigned to a project
 | |
| func (p *Project) NumOpenIssues(ctx context.Context) int {
 | |
| 	c, err := db.GetEngine(ctx).Table("project_issue").
 | |
| 		Join("INNER", "issue", "project_issue.issue_id=issue.id").
 | |
| 		Where("project_issue.project_id=? AND issue.is_closed=?", p.ID, false).
 | |
| 		Cols("issue_id").
 | |
| 		Count()
 | |
| 	if err != nil {
 | |
| 		log.Error("NumOpenIssues: %v", err)
 | |
| 		return 0
 | |
| 	}
 | |
| 	return int(c)
 | |
| }
 | |
| 
 | |
| func (c *Column) moveIssuesToAnotherColumn(ctx context.Context, newColumn *Column) error {
 | |
| 	if c.ProjectID != newColumn.ProjectID {
 | |
| 		return fmt.Errorf("columns have to be in the same project")
 | |
| 	}
 | |
| 
 | |
| 	if c.ID == newColumn.ID {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	res := struct {
 | |
| 		MaxSorting int64
 | |
| 		IssueCount int64
 | |
| 	}{}
 | |
| 	if _, err := db.GetEngine(ctx).Select("max(sorting) as max_sorting, count(*) as issue_count").
 | |
| 		Table("project_issue").
 | |
| 		Where("project_id=?", newColumn.ProjectID).
 | |
| 		And("project_board_id=?", newColumn.ID).
 | |
| 		Get(&res); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	issues, err := c.GetIssues(ctx)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	if len(issues) == 0 {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	nextSorting := util.Iif(res.IssueCount > 0, res.MaxSorting+1, 0)
 | |
| 	return db.WithTx(ctx, func(ctx context.Context) error {
 | |
| 		for i, issue := range issues {
 | |
| 			issue.ProjectColumnID = newColumn.ID
 | |
| 			issue.Sorting = nextSorting + int64(i)
 | |
| 			if _, err := db.GetEngine(ctx).ID(issue.ID).Cols("project_board_id", "sorting").Update(issue); err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 		}
 | |
| 		return nil
 | |
| 	})
 | |
| }
 | |
| 
 | |
| // DeleteAllProjectIssueByIssueIDsAndProjectIDs delete all project's issues by issue's and project's ids
 | |
| func DeleteAllProjectIssueByIssueIDsAndProjectIDs(ctx context.Context, issueIDs, projectIDs []int64) error {
 | |
| 	_, err := db.GetEngine(ctx).In("project_id", projectIDs).In("issue_id", issueIDs).Delete(&ProjectIssue{})
 | |
| 	return err
 | |
| }
 |