2020-08-17 06:07:38 +03:00
// Copyright 2020 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 (
2021-09-29 23:53:12 +03:00
"fmt"
"regexp"
2021-09-19 14:49:59 +03:00
"code.gitea.io/gitea/models/db"
2020-08-17 06:07:38 +03:00
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/timeutil"
2021-01-15 23:29:32 +03:00
"xorm.io/builder"
2020-08-17 06:07:38 +03:00
)
type (
// ProjectBoardType is used to represent a project board type
ProjectBoardType uint8
// ProjectBoardList is a list of all project boards in a repository
ProjectBoardList [ ] * ProjectBoard
)
const (
// ProjectBoardTypeNone is a project board type that has no predefined columns
ProjectBoardTypeNone ProjectBoardType = iota
// ProjectBoardTypeBasicKanban is a project board type that has basic predefined columns
ProjectBoardTypeBasicKanban
// ProjectBoardTypeBugTriage is a project board type that has predefined columns suited to hunting down bugs
ProjectBoardTypeBugTriage
)
2021-09-29 23:53:12 +03:00
// BoardColorPattern is a regexp witch can validate BoardColor
var BoardColorPattern = regexp . MustCompile ( "^#[0-9a-fA-F]{6}$" )
2020-08-17 06:07:38 +03:00
// ProjectBoard is used to represent boards on a project
type ProjectBoard struct {
ID int64 ` xorm:"pk autoincr" `
Title string
2021-09-29 23:53:12 +03:00
Default bool ` xorm:"NOT NULL DEFAULT false" ` // issues not assigned to a specific board will be assigned to this board
Sorting int8 ` xorm:"NOT NULL DEFAULT 0" `
Color string ` xorm:"VARCHAR(7)" `
2020-08-17 06:07:38 +03:00
ProjectID int64 ` xorm:"INDEX NOT NULL" `
CreatorID int64 ` xorm:"NOT NULL" `
CreatedUnix timeutil . TimeStamp ` xorm:"INDEX created" `
UpdatedUnix timeutil . TimeStamp ` xorm:"INDEX updated" `
Issues [ ] * Issue ` xorm:"-" `
}
2021-09-19 14:49:59 +03:00
func init ( ) {
db . RegisterModel ( new ( ProjectBoard ) )
}
2020-08-17 06:07:38 +03:00
// IsProjectBoardTypeValid checks if the project board type is valid
func IsProjectBoardTypeValid ( p ProjectBoardType ) bool {
switch p {
case ProjectBoardTypeNone , ProjectBoardTypeBasicKanban , ProjectBoardTypeBugTriage :
return true
default :
return false
}
}
2021-11-21 18:41:00 +03:00
func createBoardsForProjectsType ( sess db . Engine , project * Project ) error {
2020-08-17 06:07:38 +03:00
var items [ ] string
switch project . BoardType {
case ProjectBoardTypeBugTriage :
items = setting . Project . ProjectBoardBugTriageType
case ProjectBoardTypeBasicKanban :
items = setting . Project . ProjectBoardBasicKanbanType
case ProjectBoardTypeNone :
fallthrough
default :
return nil
}
if len ( items ) == 0 {
return nil
}
2021-03-14 21:52:12 +03:00
boards := make ( [ ] ProjectBoard , 0 , len ( items ) )
2020-08-17 06:07:38 +03:00
for _ , v := range items {
boards = append ( boards , ProjectBoard {
CreatedUnix : timeutil . TimeStampNow ( ) ,
CreatorID : project . CreatorID ,
Title : v ,
ProjectID : project . ID ,
} )
}
_ , err := sess . Insert ( boards )
return err
}
// NewProjectBoard adds a new project board to a given project
func NewProjectBoard ( board * ProjectBoard ) error {
2021-09-29 23:53:12 +03:00
if len ( board . Color ) != 0 && ! BoardColorPattern . MatchString ( board . Color ) {
return fmt . Errorf ( "bad color code: %s" , board . Color )
}
2021-09-23 18:45:36 +03:00
_ , err := db . GetEngine ( db . DefaultContext ) . Insert ( board )
2020-08-17 06:07:38 +03:00
return err
}
// DeleteProjectBoardByID removes all issues references to the project board.
func DeleteProjectBoardByID ( boardID int64 ) error {
2021-11-21 18:41:00 +03:00
ctx , committer , err := db . TxContext ( )
if err != nil {
2020-08-17 06:07:38 +03:00
return err
}
2021-11-21 18:41:00 +03:00
defer committer . Close ( )
2020-08-17 06:07:38 +03:00
2021-11-21 18:41:00 +03:00
if err := deleteProjectBoardByID ( db . GetEngine ( ctx ) , boardID ) ; err != nil {
2020-08-17 06:07:38 +03:00
return err
}
2021-11-21 18:41:00 +03:00
return committer . Commit ( )
2020-08-17 06:07:38 +03:00
}
2021-09-19 14:49:59 +03:00
func deleteProjectBoardByID ( e db . Engine , boardID int64 ) error {
2020-08-17 06:07:38 +03:00
board , err := getProjectBoard ( e , boardID )
if err != nil {
if IsErrProjectBoardNotExist ( err ) {
return nil
}
return err
}
if err = board . removeIssues ( e ) ; err != nil {
return err
}
if _ , err := e . ID ( board . ID ) . Delete ( board ) ; err != nil {
return err
}
return nil
}
2021-09-19 14:49:59 +03:00
func deleteProjectBoardByProjectID ( e db . Engine , projectID int64 ) error {
2020-08-17 06:07:38 +03:00
_ , err := e . Where ( "project_id=?" , projectID ) . Delete ( & ProjectBoard { } )
return err
}
// GetProjectBoard fetches the current board of a project
func GetProjectBoard ( boardID int64 ) ( * ProjectBoard , error ) {
2021-09-23 18:45:36 +03:00
return getProjectBoard ( db . GetEngine ( db . DefaultContext ) , boardID )
2020-08-17 06:07:38 +03:00
}
2021-09-19 14:49:59 +03:00
func getProjectBoard ( e db . Engine , boardID int64 ) ( * ProjectBoard , error ) {
2020-08-17 06:07:38 +03:00
board := new ( ProjectBoard )
has , err := e . ID ( boardID ) . Get ( board )
if err != nil {
return nil , err
} else if ! has {
return nil , ErrProjectBoardNotExist { BoardID : boardID }
}
return board , nil
}
2021-02-11 19:32:27 +03:00
// UpdateProjectBoard updates a project board
2020-08-17 06:07:38 +03:00
func UpdateProjectBoard ( board * ProjectBoard ) error {
2021-09-23 18:45:36 +03:00
return updateProjectBoard ( db . GetEngine ( db . DefaultContext ) , board )
2020-08-17 06:07:38 +03:00
}
2021-09-19 14:49:59 +03:00
func updateProjectBoard ( e db . Engine , board * ProjectBoard ) error {
2021-02-11 19:32:27 +03:00
var fieldToUpdate [ ] string
if board . Sorting != 0 {
fieldToUpdate = append ( fieldToUpdate , "sorting" )
}
if board . Title != "" {
fieldToUpdate = append ( fieldToUpdate , "title" )
}
2021-09-29 23:53:12 +03:00
if len ( board . Color ) != 0 && ! BoardColorPattern . MatchString ( board . Color ) {
return fmt . Errorf ( "bad color code: %s" , board . Color )
}
fieldToUpdate = append ( fieldToUpdate , "color" )
2021-02-11 19:32:27 +03:00
_ , err := e . ID ( board . ID ) . Cols ( fieldToUpdate ... ) . Update ( board )
2020-08-17 06:07:38 +03:00
return err
}
// GetProjectBoards fetches all boards related to a project
2021-01-15 23:29:32 +03:00
// if no default board set, first board is a temporary "Uncategorized" board
func GetProjectBoards ( projectID int64 ) ( ProjectBoardList , error ) {
2021-09-23 18:45:36 +03:00
return getProjectBoards ( db . GetEngine ( db . DefaultContext ) , projectID )
2021-01-15 23:29:32 +03:00
}
2020-08-17 06:07:38 +03:00
2021-09-19 14:49:59 +03:00
func getProjectBoards ( e db . Engine , projectID int64 ) ( [ ] * ProjectBoard , error ) {
2021-03-14 21:52:12 +03:00
boards := make ( [ ] * ProjectBoard , 0 , 5 )
2020-08-17 06:07:38 +03:00
2021-02-11 19:32:27 +03:00
if err := e . Where ( "project_id=? AND `default`=?" , projectID , false ) . OrderBy ( "Sorting" ) . Find ( & boards ) ; err != nil {
2021-01-15 23:29:32 +03:00
return nil , err
}
defaultB , err := getDefaultBoard ( e , projectID )
if err != nil {
return nil , err
}
return append ( [ ] * ProjectBoard { defaultB } , boards ... ) , nil
2020-08-17 06:07:38 +03:00
}
2021-01-15 23:29:32 +03:00
// getDefaultBoard return default board and create a dummy if none exist
2021-09-19 14:49:59 +03:00
func getDefaultBoard ( e db . Engine , projectID int64 ) ( * ProjectBoard , error ) {
2021-01-15 23:29:32 +03:00
var board ProjectBoard
exist , err := e . Where ( "project_id=? AND `default`=?" , projectID , true ) . Get ( & board )
if err != nil {
return nil , err
}
if exist {
return & board , nil
}
// represents a board for issues not assigned to one
2020-08-17 06:07:38 +03:00
return & ProjectBoard {
ProjectID : projectID ,
Title : "Uncategorized" ,
Default : true ,
} , nil
}
2021-01-15 23:29:32 +03:00
// SetDefaultBoard represents a board for issues not assigned to one
// if boardID is 0 unset default
func SetDefaultBoard ( projectID , boardID int64 ) error {
2021-09-23 18:45:36 +03:00
_ , err := db . GetEngine ( db . DefaultContext ) . Where ( builder . Eq {
2021-01-15 23:29:32 +03:00
"project_id" : projectID ,
"`default`" : true ,
} ) . Cols ( "`default`" ) . Update ( & ProjectBoard { Default : false } )
if err != nil {
return err
}
if boardID > 0 {
2021-09-23 18:45:36 +03:00
_ , err = db . GetEngine ( db . DefaultContext ) . ID ( boardID ) . Where ( builder . Eq { "project_id" : projectID } ) .
2021-01-15 23:29:32 +03:00
Cols ( "`default`" ) . Update ( & ProjectBoard { Default : true } )
}
return err
}
2020-08-17 06:07:38 +03:00
// LoadIssues load issues assigned to this board
func ( b * ProjectBoard ) LoadIssues ( ) ( IssueList , error ) {
2021-01-15 23:29:32 +03:00
issueList := make ( [ ] * Issue , 0 , 10 )
if b . ID != 0 {
issues , err := Issues ( & IssuesOptions {
ProjectBoardID : b . ID ,
ProjectID : b . ProjectID ,
2021-12-08 09:57:18 +03:00
SortType : "project-column-sorting" ,
2021-01-15 23:29:32 +03:00
} )
if err != nil {
return nil , err
}
issueList = issues
}
if b . Default {
issues , err := Issues ( & IssuesOptions {
ProjectBoardID : - 1 , // Issues without ProjectBoardID
ProjectID : b . ProjectID ,
2021-12-08 09:57:18 +03:00
SortType : "project-column-sorting" ,
2021-01-15 23:29:32 +03:00
} )
if err != nil {
return nil , err
}
issueList = append ( issueList , issues ... )
}
2021-01-20 22:53:48 +03:00
if err := IssueList ( issueList ) . LoadComments ( ) ; err != nil {
return nil , err
}
2021-01-15 23:29:32 +03:00
b . Issues = issueList
return issueList , nil
2020-08-17 06:07:38 +03:00
}
// LoadIssues load issues assigned to the boards
func ( bs ProjectBoardList ) LoadIssues ( ) ( IssueList , error ) {
issues := make ( IssueList , 0 , len ( bs ) * 10 )
for i := range bs {
il , err := bs [ i ] . LoadIssues ( )
if err != nil {
return nil , err
}
bs [ i ] . Issues = il
issues = append ( issues , il ... )
}
return issues , nil
}
2021-02-11 19:32:27 +03:00
// UpdateProjectBoardSorting update project board sorting
func UpdateProjectBoardSorting ( bs ProjectBoardList ) error {
for i := range bs {
2021-09-23 18:45:36 +03:00
_ , err := db . GetEngine ( db . DefaultContext ) . ID ( bs [ i ] . ID ) . Cols (
2021-02-11 19:32:27 +03:00
"sorting" ,
) . Update ( bs [ i ] )
if err != nil {
return err
}
}
return nil
}