2016-02-27 11:31:24 -05:00
// Copyright 2016 The Gogs 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"
"html/template"
2016-08-29 20:00:06 -07:00
"regexp"
2016-02-27 11:31:24 -05:00
"strconv"
"strings"
"github.com/go-xorm/xorm"
2016-08-03 11:51:22 -07:00
2016-11-11 10:39:44 +01:00
api "code.gitea.io/sdk/gitea"
2016-02-27 11:31:24 -05:00
)
2016-08-29 20:00:06 -07:00
var labelColorPattern = regexp . MustCompile ( "#([a-fA-F0-9]{6})" )
// GetLabelTemplateFile loads the label template file by given name,
2018-03-13 04:03:55 +02:00
// then parses and returns a list of name-color pairs and optionally description.
func GetLabelTemplateFile ( name string ) ( [ ] [ 3 ] string , error ) {
2016-08-29 20:00:06 -07:00
data , err := getRepoInitFile ( "label" , name )
if err != nil {
return nil , fmt . Errorf ( "getRepoInitFile: %v" , err )
}
lines := strings . Split ( string ( data ) , "\n" )
2018-03-13 04:03:55 +02:00
list := make ( [ ] [ 3 ] string , 0 , len ( lines ) )
2016-08-29 20:00:06 -07:00
for i := 0 ; i < len ( lines ) ; i ++ {
line := strings . TrimSpace ( lines [ i ] )
if len ( line ) == 0 {
continue
}
2018-03-13 04:03:55 +02:00
parts := strings . SplitN ( line , ";" , 2 )
fields := strings . SplitN ( parts [ 0 ] , " " , 2 )
2016-08-29 20:00:06 -07:00
if len ( fields ) != 2 {
return nil , fmt . Errorf ( "line is malformed: %s" , line )
}
if ! labelColorPattern . MatchString ( fields [ 0 ] ) {
return nil , fmt . Errorf ( "bad HTML color code in line: %s" , line )
}
2018-03-13 04:03:55 +02:00
var description string
if len ( parts ) > 1 {
description = strings . TrimSpace ( parts [ 1 ] )
}
2016-08-29 20:00:06 -07:00
fields [ 1 ] = strings . TrimSpace ( fields [ 1 ] )
2018-03-13 04:03:55 +02:00
list = append ( list , [ 3 ] string { fields [ 1 ] , fields [ 0 ] , description } )
2016-08-29 20:00:06 -07:00
}
return list , nil
}
2016-02-27 11:31:24 -05:00
// Label represents a label of repository for issues.
type Label struct {
ID int64 ` xorm:"pk autoincr" `
RepoID int64 ` xorm:"INDEX" `
Name string
2018-03-13 04:03:55 +02:00
Description string
2016-02-27 11:31:24 -05:00
Color string ` xorm:"VARCHAR(7)" `
NumIssues int
NumClosedIssues int
NumOpenIssues int ` xorm:"-" `
IsChecked bool ` xorm:"-" `
2019-01-23 06:10:38 +02:00
QueryString string
IsSelected bool
2016-02-27 11:31:24 -05:00
}
2016-11-25 09:11:12 +01:00
// APIFormat converts a Label to the api.Label format
2016-08-14 03:32:24 -07:00
func ( label * Label ) APIFormat ( ) * api . Label {
return & api . Label {
ID : label . ID ,
Name : label . Name ,
2016-10-07 19:17:27 +02:00
Color : strings . TrimLeft ( label . Color , "#" ) ,
2016-08-14 03:32:24 -07:00
}
}
2016-02-27 11:31:24 -05:00
// CalOpenIssues calculates the open issues of label.
2016-08-14 03:32:24 -07:00
func ( label * Label ) CalOpenIssues ( ) {
label . NumOpenIssues = label . NumIssues - label . NumClosedIssues
2016-02-27 11:31:24 -05:00
}
2019-01-23 06:10:38 +02:00
// LoadSelectedLabelsAfterClick calculates the set of selected labels when a label is clicked
func ( label * Label ) LoadSelectedLabelsAfterClick ( currentSelectedLabels [ ] int64 ) {
var labelQuerySlice [ ] string
labelSelected := false
labelID := strconv . FormatInt ( label . ID , 10 )
for _ , s := range currentSelectedLabels {
if s == label . ID {
labelSelected = true
} else if s > 0 {
labelQuerySlice = append ( labelQuerySlice , strconv . FormatInt ( s , 10 ) )
}
}
if ! labelSelected {
labelQuerySlice = append ( labelQuerySlice , labelID )
}
label . IsSelected = labelSelected
label . QueryString = strings . Join ( labelQuerySlice , "," )
}
2016-02-27 11:31:24 -05:00
// ForegroundColor calculates the text color for labels based
// on their background color.
2016-11-25 09:11:12 +01:00
func ( label * Label ) ForegroundColor ( ) template . CSS {
if strings . HasPrefix ( label . Color , "#" ) {
if color , err := strconv . ParseUint ( label . Color [ 1 : ] , 16 , 64 ) ; err == nil {
2016-02-27 11:31:24 -05:00
r := float32 ( 0xFF & ( color >> 16 ) )
g := float32 ( 0xFF & ( color >> 8 ) )
b := float32 ( 0xFF & color )
luminance := ( 0.2126 * r + 0.7152 * g + 0.0722 * b ) / 255
2016-08-27 12:44:39 -07:00
if luminance < 0.66 {
2016-02-27 11:31:24 -05:00
return template . CSS ( "#fff" )
}
}
}
// default to black
return template . CSS ( "#000" )
}
2017-06-25 02:15:09 -04:00
func newLabel ( e Engine , label * Label ) error {
_ , err := e . Insert ( label )
2016-02-27 11:31:24 -05:00
return err
}
2017-06-25 02:15:09 -04:00
// NewLabel creates a new label for a repository
func NewLabel ( label * Label ) error {
return newLabel ( x , label )
}
// NewLabels creates new labels for a repository.
func NewLabels ( labels ... * Label ) error {
sess := x . NewSession ( )
defer sess . Close ( )
if err := sess . Begin ( ) ; err != nil {
return err
}
for _ , label := range labels {
if err := newLabel ( sess , label ) ; err != nil {
return err
}
}
return sess . Commit ( )
}
2016-10-07 19:17:27 +02:00
// getLabelInRepoByName returns a label by Name in given repository.
// If pass repoID as 0, then ORM will ignore limitation of repository
// and can return arbitrary label with any valid ID.
func getLabelInRepoByName ( e Engine , repoID int64 , labelName string ) ( * Label , error ) {
2018-10-20 23:25:14 +02:00
if len ( labelName ) == 0 {
2016-10-07 19:17:27 +02:00
return nil , ErrLabelNotExist { 0 , repoID }
}
l := & Label {
Name : labelName ,
RepoID : repoID ,
}
2017-01-31 20:31:35 -05:00
has , err := e . Get ( l )
2016-10-07 19:17:27 +02:00
if err != nil {
return nil , err
} else if ! has {
return nil , ErrLabelNotExist { 0 , l . RepoID }
}
return l , nil
}
2016-08-03 11:51:22 -07:00
// getLabelInRepoByID returns a label by ID in given repository.
// If pass repoID as 0, then ORM will ignore limitation of repository
// and can return arbitrary label with any valid ID.
func getLabelInRepoByID ( e Engine , repoID , labelID int64 ) ( * Label , error ) {
if labelID <= 0 {
return nil , ErrLabelNotExist { labelID , repoID }
2016-02-27 11:31:24 -05:00
}
2016-08-03 11:51:22 -07:00
l := & Label {
ID : labelID ,
RepoID : repoID ,
}
2017-01-31 20:31:35 -05:00
has , err := e . Get ( l )
2016-02-27 11:31:24 -05:00
if err != nil {
return nil , err
} else if ! has {
2016-08-03 11:51:22 -07:00
return nil , ErrLabelNotExist { l . ID , l . RepoID }
2016-02-27 11:31:24 -05:00
}
return l , nil
}
// GetLabelByID returns a label by given ID.
func GetLabelByID ( id int64 ) ( * Label , error ) {
2016-08-03 11:51:22 -07:00
return getLabelInRepoByID ( x , 0 , id )
}
2016-12-02 09:28:45 +01:00
// GetLabelInRepoByName returns a label by name in given repository.
2016-10-07 19:17:27 +02:00
func GetLabelInRepoByName ( repoID int64 , labelName string ) ( * Label , error ) {
return getLabelInRepoByName ( x , repoID , labelName )
}
2016-08-03 11:51:22 -07:00
// GetLabelInRepoByID returns a label by ID in given repository.
func GetLabelInRepoByID ( repoID , labelID int64 ) ( * Label , error ) {
return getLabelInRepoByID ( x , repoID , labelID )
}
// GetLabelsInRepoByIDs returns a list of labels by IDs in given repository,
// it silently ignores label IDs that are not belong to the repository.
func GetLabelsInRepoByIDs ( repoID int64 , labelIDs [ ] int64 ) ( [ ] * Label , error ) {
labels := make ( [ ] * Label , 0 , len ( labelIDs ) )
2016-11-10 16:16:32 +01:00
return labels , x .
Where ( "repo_id = ?" , repoID ) .
2016-11-12 16:29:18 +08:00
In ( "id" , labelIDs ) .
2016-11-10 16:16:32 +01:00
Asc ( "name" ) .
Find ( & labels )
2016-02-27 11:31:24 -05:00
}
// GetLabelsByRepoID returns all labels that belong to given repository by ID.
2016-12-24 15:41:09 +01:00
func GetLabelsByRepoID ( repoID int64 , sortType string ) ( [ ] * Label , error ) {
2016-02-27 11:31:24 -05:00
labels := make ( [ ] * Label , 0 , 10 )
2016-12-24 15:41:09 +01:00
sess := x . Where ( "repo_id = ?" , repoID )
switch sortType {
case "reversealphabetically" :
sess . Desc ( "name" )
case "leastissues" :
sess . Asc ( "num_issues" )
case "mostissues" :
sess . Desc ( "num_issues" )
default :
sess . Asc ( "name" )
}
return labels , sess . Find ( & labels )
2016-02-27 11:31:24 -05:00
}
func getLabelsByIssueID ( e Engine , issueID int64 ) ( [ ] * Label , error ) {
2017-02-11 20:01:33 +08:00
var labels [ ] * Label
return labels , e . Where ( "issue_label.issue_id = ?" , issueID ) .
Join ( "LEFT" , "issue_label" , "issue_label.label_id = label.id" ) .
Asc ( "label.name" ) .
2016-11-10 16:16:32 +01:00
Find ( & labels )
2016-02-27 11:31:24 -05:00
}
// GetLabelsByIssueID returns all labels that belong to given issue by ID.
func GetLabelsByIssueID ( issueID int64 ) ( [ ] * Label , error ) {
return getLabelsByIssueID ( x , issueID )
}
func updateLabel ( e Engine , l * Label ) error {
2017-10-04 21:43:04 -07:00
_ , err := e . ID ( l . ID ) . AllCols ( ) . Update ( l )
2016-02-27 11:31:24 -05:00
return err
}
// UpdateLabel updates label information.
func UpdateLabel ( l * Label ) error {
return updateLabel ( x , l )
}
// DeleteLabel delete a label of given repository.
func DeleteLabel ( repoID , labelID int64 ) error {
2016-08-03 11:51:22 -07:00
_ , err := GetLabelInRepoByID ( repoID , labelID )
2016-02-27 11:31:24 -05:00
if err != nil {
if IsErrLabelNotExist ( err ) {
return nil
}
return err
}
sess := x . NewSession ( )
2017-06-21 03:57:05 +03:00
defer sess . Close ( )
2016-02-27 11:31:24 -05:00
if err = sess . Begin ( ) ; err != nil {
return err
}
2017-10-04 21:43:04 -07:00
if _ , err = sess . ID ( labelID ) . Delete ( new ( Label ) ) ; err != nil {
2016-02-27 11:31:24 -05:00
return err
2016-11-10 16:16:32 +01:00
} else if _ , err = sess .
Where ( "label_id = ?" , labelID ) .
Delete ( new ( IssueLabel ) ) ; err != nil {
2016-02-27 11:31:24 -05:00
return err
}
2016-08-03 11:51:22 -07:00
2017-02-11 20:56:57 +08:00
// Clear label id in comment table
if _ , err = sess . Where ( "label_id = ?" , labelID ) . Cols ( "label_id" ) . Update ( & Comment { } ) ; err != nil {
return err
}
2016-02-27 11:31:24 -05:00
return sess . Commit ( )
}
// .___ .____ ___. .__
// | | ______ ________ __ ____ | | _____ \_ |__ ____ | |
// | |/ ___// ___/ | \_/ __ \| | \__ \ | __ \_/ __ \| |
// | |\___ \ \___ \| | /\ ___/| |___ / __ \| \_\ \ ___/| |__
// |___/____ >____ >____/ \___ >_______ (____ /___ /\___ >____/
// \/ \/ \/ \/ \/ \/ \/
2017-01-04 19:50:34 -05:00
// IssueLabel represents an issue-label relation.
2016-02-27 11:31:24 -05:00
type IssueLabel struct {
ID int64 ` xorm:"pk autoincr" `
IssueID int64 ` xorm:"UNIQUE(s)" `
LabelID int64 ` xorm:"UNIQUE(s)" `
}
func hasIssueLabel ( e Engine , issueID , labelID int64 ) bool {
2016-08-03 11:51:22 -07:00
has , _ := e . Where ( "issue_id = ? AND label_id = ?" , issueID , labelID ) . Get ( new ( IssueLabel ) )
2016-02-27 11:31:24 -05:00
return has
}
// HasIssueLabel returns true if issue has been labeled.
func HasIssueLabel ( issueID , labelID int64 ) bool {
return hasIssueLabel ( x , issueID , labelID )
}
2017-01-30 20:46:45 +08:00
func newIssueLabel ( e * xorm . Session , issue * Issue , label * Label , doer * User ) ( err error ) {
2016-02-27 11:31:24 -05:00
if _ , err = e . Insert ( & IssueLabel {
IssueID : issue . ID ,
LabelID : label . ID ,
} ) ; err != nil {
return err
}
2017-01-30 20:46:45 +08:00
if err = issue . loadRepo ( e ) ; err != nil {
return
}
if _ , err = createLabelComment ( e , doer , issue . Repo , issue , label , true ) ; err != nil {
return err
}
2016-02-27 11:31:24 -05:00
label . NumIssues ++
if issue . IsClosed {
label . NumClosedIssues ++
}
return updateLabel ( e , label )
}
// NewIssueLabel creates a new issue-label relation.
2017-01-30 20:46:45 +08:00
func NewIssueLabel ( issue * Issue , label * Label , doer * User ) ( err error ) {
2016-08-03 11:51:22 -07:00
if HasIssueLabel ( issue . ID , label . ID ) {
return nil
}
2016-02-27 11:31:24 -05:00
sess := x . NewSession ( )
2017-06-21 03:57:05 +03:00
defer sess . Close ( )
2016-02-27 11:31:24 -05:00
if err = sess . Begin ( ) ; err != nil {
return err
}
2017-01-30 20:46:45 +08:00
if err = newIssueLabel ( sess , issue , label , doer ) ; err != nil {
2016-02-27 11:31:24 -05:00
return err
2016-08-03 11:51:22 -07:00
}
return sess . Commit ( )
}
2017-01-30 20:46:45 +08:00
func newIssueLabels ( e * xorm . Session , issue * Issue , labels [ ] * Label , doer * User ) ( err error ) {
2016-08-03 11:51:22 -07:00
for i := range labels {
if hasIssueLabel ( e , issue . ID , labels [ i ] . ID ) {
continue
}
2017-01-30 20:46:45 +08:00
if err = newIssueLabel ( e , issue , labels [ i ] , doer ) ; err != nil {
2016-08-03 11:51:22 -07:00
return fmt . Errorf ( "newIssueLabel: %v" , err )
}
}
return nil
}
// NewIssueLabels creates a list of issue-label relations.
2017-01-30 20:46:45 +08:00
func NewIssueLabels ( issue * Issue , labels [ ] * Label , doer * User ) ( err error ) {
2016-08-03 11:51:22 -07:00
sess := x . NewSession ( )
2017-06-21 03:57:05 +03:00
defer sess . Close ( )
2016-08-03 11:51:22 -07:00
if err = sess . Begin ( ) ; err != nil {
return err
}
2017-01-30 20:46:45 +08:00
if err = newIssueLabels ( sess , issue , labels , doer ) ; err != nil {
2016-08-03 11:51:22 -07:00
return err
2016-02-27 11:31:24 -05:00
}
return sess . Commit ( )
}
func getIssueLabels ( e Engine , issueID int64 ) ( [ ] * IssueLabel , error ) {
issueLabels := make ( [ ] * IssueLabel , 0 , 10 )
2016-11-10 16:16:32 +01:00
return issueLabels , e .
Where ( "issue_id=?" , issueID ) .
Asc ( "label_id" ) .
Find ( & issueLabels )
2016-02-27 11:31:24 -05:00
}
2017-01-31 20:31:35 -05:00
func deleteIssueLabel ( e * xorm . Session , issue * Issue , label * Label , doer * User ) ( err error ) {
if count , err := e . Delete ( & IssueLabel {
2016-02-27 11:31:24 -05:00
IssueID : issue . ID ,
LabelID : label . ID ,
} ) ; err != nil {
return err
2017-01-31 20:31:35 -05:00
} else if count == 0 {
return nil
2016-02-27 11:31:24 -05:00
}
2017-01-30 20:46:45 +08:00
if err = issue . loadRepo ( e ) ; err != nil {
return
}
if _ , err = createLabelComment ( e , doer , issue . Repo , issue , label , false ) ; err != nil {
return err
}
2016-02-27 11:31:24 -05:00
label . NumIssues --
if issue . IsClosed {
label . NumClosedIssues --
}
return updateLabel ( e , label )
}
// DeleteIssueLabel deletes issue-label relation.
2017-01-31 20:31:35 -05:00
func DeleteIssueLabel ( issue * Issue , label * Label , doer * User ) ( err error ) {
2016-02-27 11:31:24 -05:00
sess := x . NewSession ( )
2017-06-21 03:57:05 +03:00
defer sess . Close ( )
2016-02-27 11:31:24 -05:00
if err = sess . Begin ( ) ; err != nil {
return err
}
2017-01-31 20:31:35 -05:00
if err = deleteIssueLabel ( sess , issue , label , doer ) ; err != nil {
2016-02-27 11:31:24 -05:00
return err
}
return sess . Commit ( )
}