2021-11-21 17:11:48 +08:00
// Copyright 2021 The Gitea Authors. All rights reserved.
2019-07-30 09:59:10 +08:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
2021-11-21 17:11:48 +08:00
package issue
2019-07-30 09:59:10 +08:00
import (
"fmt"
2019-12-07 23:52:36 +08:00
"html"
2021-11-16 18:18:25 +00:00
"net/url"
2020-09-04 18:37:37 +03:00
"regexp"
"strconv"
2020-06-19 09:19:56 +02:00
"strings"
2020-09-04 18:37:37 +03:00
"time"
2019-07-30 09:59:10 +08:00
"code.gitea.io/gitea/models"
2022-04-28 13:48:48 +02:00
"code.gitea.io/gitea/models/db"
2022-05-11 18:09:36 +08:00
access_model "code.gitea.io/gitea/models/perm/access"
2021-12-10 09:27:50 +08:00
repo_model "code.gitea.io/gitea/models/repo"
2021-11-24 17:49:20 +08:00
user_model "code.gitea.io/gitea/models/user"
2019-12-07 23:52:36 +08:00
"code.gitea.io/gitea/modules/references"
2020-01-10 17:34:21 +08:00
"code.gitea.io/gitea/modules/repository"
2019-07-30 09:59:10 +08:00
)
2020-09-04 18:37:37 +03:00
const (
secondsByMinute = float64 ( time . Minute / time . Second ) // seconds in a minute
secondsByHour = 60 * secondsByMinute // seconds in an hour
secondsByDay = 8 * secondsByHour // seconds in a day
secondsByWeek = 5 * secondsByDay // seconds in a week
secondsByMonth = 4 * secondsByWeek // seconds in a month
)
var reDuration = regexp . MustCompile ( ` (?i)^(?:(\d+([\.,]\d+)?)(?:mo))?(?:(\d+([\.,]\d+)?)(?:w))?(?:(\d+([\.,]\d+)?)(?:d))?(?:(\d+([\.,]\d+)?)(?:h))?(?:(\d+([\.,]\d+)?)(?:m))?$ ` )
// timeLogToAmount parses time log string and returns amount in seconds
func timeLogToAmount ( str string ) int64 {
matches := reDuration . FindAllStringSubmatch ( str , - 1 )
if len ( matches ) == 0 {
return 0
}
match := matches [ 0 ]
var a int64
// months
if len ( match [ 1 ] ) > 0 {
mo , _ := strconv . ParseFloat ( strings . Replace ( match [ 1 ] , "," , "." , 1 ) , 64 )
a += int64 ( mo * secondsByMonth )
}
// weeks
if len ( match [ 3 ] ) > 0 {
w , _ := strconv . ParseFloat ( strings . Replace ( match [ 3 ] , "," , "." , 1 ) , 64 )
a += int64 ( w * secondsByWeek )
}
// days
if len ( match [ 5 ] ) > 0 {
d , _ := strconv . ParseFloat ( strings . Replace ( match [ 5 ] , "," , "." , 1 ) , 64 )
a += int64 ( d * secondsByDay )
}
// hours
if len ( match [ 7 ] ) > 0 {
h , _ := strconv . ParseFloat ( strings . Replace ( match [ 7 ] , "," , "." , 1 ) , 64 )
a += int64 ( h * secondsByHour )
}
// minutes
if len ( match [ 9 ] ) > 0 {
d , _ := strconv . ParseFloat ( strings . Replace ( match [ 9 ] , "," , "." , 1 ) , 64 )
a += int64 ( d * secondsByMinute )
}
return a
}
2021-11-24 17:49:20 +08:00
func issueAddTime ( issue * models . Issue , doer * user_model . User , time time . Time , timeLog string ) error {
2020-09-04 18:37:37 +03:00
amount := timeLogToAmount ( timeLog )
if amount == 0 {
return nil
}
_ , err := models . AddTime ( doer , issue , amount , time )
return err
}
2021-11-21 17:11:48 +08:00
// getIssueFromRef returns the issue referenced by a ref. Returns a nil *Issue
// if the provided ref references a non-existent issue.
2021-12-10 09:27:50 +08:00
func getIssueFromRef ( repo * repo_model . Repository , index int64 ) ( * models . Issue , error ) {
2021-11-21 17:11:48 +08:00
issue , err := models . GetIssueByIndex ( repo . ID , index )
2019-12-16 05:57:34 +08:00
if err != nil {
2021-11-21 17:11:48 +08:00
if models . IsErrIssueNotExist ( err ) {
return nil , nil
2019-12-07 23:52:36 +08:00
}
2021-11-21 17:11:48 +08:00
return nil , err
2019-12-07 23:52:36 +08:00
}
2021-11-21 17:11:48 +08:00
return issue , nil
2019-12-07 23:52:36 +08:00
}
// UpdateIssuesCommit checks if issues are manipulated by commit message.
2021-12-10 09:27:50 +08:00
func UpdateIssuesCommit ( doer * user_model . User , repo * repo_model . Repository , commits [ ] * repository . PushCommit , branchName string ) error {
2019-12-07 23:52:36 +08:00
// Commits are appended in the reverse order.
for i := len ( commits ) - 1 ; i >= 0 ; i -- {
c := commits [ i ]
type markKey struct {
ID int64
Action references . XRefAction
}
refMarked := make ( map [ markKey ] bool )
2021-12-10 09:27:50 +08:00
var refRepo * repo_model . Repository
2019-12-07 23:52:36 +08:00
var refIssue * models . Issue
var err error
for _ , ref := range references . FindAllIssueReferences ( c . Message ) {
// issue is from another repo
if len ( ref . Owner ) > 0 && len ( ref . Name ) > 0 {
refRepo , err = models . GetRepositoryFromMatch ( ref . Owner , ref . Name )
if err != nil {
continue
}
} else {
refRepo = repo
}
if refIssue , err = getIssueFromRef ( refRepo , ref . Index ) ; err != nil {
return err
}
if refIssue == nil {
continue
}
2022-05-11 18:09:36 +08:00
perm , err := access_model . GetUserRepoPermission ( db . DefaultContext , refRepo , doer )
2019-12-07 23:52:36 +08:00
if err != nil {
return err
}
key := markKey { ID : refIssue . ID , Action : ref . Action }
if refMarked [ key ] {
continue
}
refMarked [ key ] = true
// FIXME: this kind of condition is all over the code, it should be consolidated in a single place
2020-01-20 20:00:32 +08:00
canclose := perm . IsAdmin ( ) || perm . IsOwner ( ) || perm . CanWriteIssuesOrPulls ( refIssue . IsPull ) || refIssue . PosterID == doer . ID
cancomment := canclose || perm . CanReadIssuesOrPulls ( refIssue . IsPull )
2019-12-07 23:52:36 +08:00
// Don't proceed if the user can't comment
if ! cancomment {
continue
}
2021-11-16 18:18:25 +00:00
message := fmt . Sprintf ( ` <a href="%s/commit/%s">%s</a> ` , html . EscapeString ( repo . Link ( ) ) , html . EscapeString ( url . PathEscape ( c . Sha1 ) ) , html . EscapeString ( strings . SplitN ( c . Message , "\n" , 2 ) [ 0 ] ) )
2019-12-07 23:52:36 +08:00
if err = models . CreateRefComment ( doer , refRepo , refIssue , message , c . Sha1 ) ; err != nil {
return err
}
// Only issues can be closed/reopened this way, and user needs the correct permissions
if refIssue . IsPull || ! canclose {
continue
}
// Only process closing/reopening keywords
if ref . Action != references . XRefActionCloses && ref . Action != references . XRefActionReopens {
continue
}
if ! repo . CloseIssuesViaCommitInAnyBranch {
// If the issue was specified to be in a particular branch, don't allow commits in other branches to close it
if refIssue . Ref != "" {
if branchName != refIssue . Ref {
continue
}
// Otherwise, only process commits to the default branch
} else if branchName != repo . DefaultBranch {
continue
}
}
2021-04-09 09:40:34 +02:00
close := ref . Action == references . XRefActionCloses
2020-09-04 18:37:37 +03:00
if close && len ( ref . TimeLog ) > 0 {
if err := issueAddTime ( refIssue , doer , c . Timestamp , ref . TimeLog ) ; err != nil {
return err
}
}
2020-01-10 22:20:11 -03:00
if close != refIssue . IsClosed {
2021-11-21 17:11:48 +08:00
refIssue . Repo = refRepo
if err := ChangeStatus ( refIssue , doer , close ) ; err != nil {
2020-01-10 22:20:11 -03:00
return err
}
2019-12-07 23:52:36 +08:00
}
}
}
return nil
}