2018-05-09 18:29:04 +02:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2022-11-27 13:20:29 -05:00
// SPDX-License-Identifier: MIT
2018-05-09 18:29:04 +02:00
2022-06-13 17:37:59 +08:00
package issues
2018-05-09 18:29:04 +02:00
import (
2021-11-19 21:39:57 +08:00
"context"
2018-05-09 18:29:04 +02:00
"fmt"
2021-09-19 19:49:59 +08:00
"code.gitea.io/gitea/models/db"
2021-11-24 17:49:20 +08:00
user_model "code.gitea.io/gitea/models/user"
2019-11-10 09:07:21 +01:00
"code.gitea.io/gitea/modules/util"
2023-12-07 15:27:36 +08:00
"xorm.io/builder"
2018-05-09 18:29:04 +02:00
)
// IssueAssignees saves all issue assignees
type IssueAssignees struct {
ID int64 ` xorm:"pk autoincr" `
AssigneeID int64 ` xorm:"INDEX" `
IssueID int64 ` xorm:"INDEX" `
}
2021-09-19 19:49:59 +08:00
func init ( ) {
db . RegisterModel ( new ( IssueAssignees ) )
}
2020-02-29 03:49:50 +01:00
// LoadAssignees load assignees of this issue.
2022-05-20 22:08:52 +08:00
func ( issue * Issue ) LoadAssignees ( ctx context . Context ) ( err error ) {
2024-05-31 20:10:11 +08:00
if issue . isAssigneeLoaded || len ( issue . Assignees ) > 0 {
return nil
}
2018-05-09 18:29:04 +02:00
// Reset maybe preexisting assignees
2021-11-24 17:49:20 +08:00
issue . Assignees = [ ] * user_model . User { }
2022-05-20 22:08:52 +08:00
issue . Assignee = nil
2018-05-09 18:29:04 +02:00
2024-05-31 20:10:11 +08:00
if err = db . GetEngine ( ctx ) . Table ( "`user`" ) .
2018-05-09 18:29:04 +02:00
Join ( "INNER" , "issue_assignees" , "assignee_id = `user`.id" ) .
Where ( "issue_assignees.issue_id = ?" , issue . ID ) .
2024-05-31 20:10:11 +08:00
Find ( & issue . Assignees ) ; err != nil {
2018-05-09 18:29:04 +02:00
return err
}
2024-05-31 20:10:11 +08:00
issue . isAssigneeLoaded = true
2018-05-09 18:29:04 +02:00
// Check if we have at least one assignee and if yes put it in as `Assignee`
if len ( issue . Assignees ) > 0 {
issue . Assignee = issue . Assignees [ 0 ]
}
2024-05-31 20:10:11 +08:00
return nil
2018-05-09 18:29:04 +02:00
}
2019-11-18 05:08:20 -03:00
// GetAssigneeIDsByIssue returns the IDs of users assigned to an issue
// but skips joining with `user` for performance reasons.
// User permissions must be verified elsewhere if required.
2022-11-19 09:12:33 +01:00
func GetAssigneeIDsByIssue ( ctx context . Context , issueID int64 ) ( [ ] int64 , error ) {
2019-11-18 05:08:20 -03:00
userIDs := make ( [ ] int64 , 0 , 5 )
2022-11-19 09:12:33 +01:00
return userIDs , db . GetEngine ( ctx ) .
Table ( "issue_assignees" ) .
2019-11-18 05:08:20 -03:00
Cols ( "assignee_id" ) .
Where ( "issue_id = ?" , issueID ) .
Distinct ( "assignee_id" ) .
Find ( & userIDs )
}
2018-05-09 18:29:04 +02:00
// IsUserAssignedToIssue returns true when the user is assigned to the issue
2022-05-20 22:08:52 +08:00
func IsUserAssignedToIssue ( ctx context . Context , issue * Issue , user * user_model . User ) ( isAssigned bool , err error ) {
2023-12-07 15:27:36 +08:00
return db . Exist [ IssueAssignees ] ( ctx , builder . Eq { "assignee_id" : user . ID , "issue_id" : issue . ID } )
2018-05-09 18:29:04 +02:00
}
2024-03-04 09:16:03 +01:00
type AssignedIssuesOptions struct {
db . ListOptions
AssigneeID int64
RepoOwnerID int64
}
func ( opts * AssignedIssuesOptions ) ToConds ( ) builder . Cond {
cond := builder . NewCond ( )
if opts . AssigneeID != 0 {
cond = cond . And ( builder . In ( "issue.id" , builder . Select ( "issue_id" ) . From ( "issue_assignees" ) . Where ( builder . Eq { "assignee_id" : opts . AssigneeID } ) ) )
}
if opts . RepoOwnerID != 0 {
cond = cond . And ( builder . In ( "issue.repo_id" , builder . Select ( "id" ) . From ( "repository" ) . Where ( builder . Eq { "owner_id" : opts . RepoOwnerID } ) ) )
}
return cond
}
func GetAssignedIssues ( ctx context . Context , opts * AssignedIssuesOptions ) ( [ ] * Issue , int64 , error ) {
return db . FindAndCount [ Issue ] ( ctx , opts )
}
2022-03-29 22:57:33 +08:00
// ToggleIssueAssignee changes a user between assigned and not assigned for this issue, and make issue comment for it.
2023-04-15 02:18:28 +08:00
func ToggleIssueAssignee ( ctx context . Context , issue * Issue , doer * user_model . User , assigneeID int64 ) ( removed bool , comment * Comment , err error ) {
ctx , committer , err := db . TxContext ( ctx )
2021-11-19 21:39:57 +08:00
if err != nil {
2019-10-25 16:46:37 +02:00
return false , nil , err
2018-05-09 18:29:04 +02:00
}
2021-11-19 21:39:57 +08:00
defer committer . Close ( )
2018-05-09 18:29:04 +02:00
2022-03-29 22:57:33 +08:00
removed , comment , err = toggleIssueAssignee ( ctx , issue , doer , assigneeID , false )
2019-10-25 16:46:37 +02:00
if err != nil {
return false , nil , err
2018-05-09 18:29:04 +02:00
}
2021-11-19 21:39:57 +08:00
if err := committer . Commit ( ) ; err != nil {
2019-10-25 16:46:37 +02:00
return false , nil , err
2019-07-31 17:24:38 +08:00
}
2019-10-25 16:46:37 +02:00
return removed , comment , nil
2018-05-09 18:29:04 +02:00
}
2022-03-29 22:57:33 +08:00
func toggleIssueAssignee ( ctx context . Context , issue * Issue , doer * user_model . User , assigneeID int64 , isCreate bool ) ( removed bool , comment * Comment , err error ) {
2022-05-20 22:08:52 +08:00
removed , err = toggleUserAssignee ( ctx , issue , assigneeID )
2018-05-09 18:29:04 +02:00
if err != nil {
2022-10-24 21:29:17 +02:00
return false , nil , fmt . Errorf ( "UpdateIssueUserByAssignee: %w" , err )
2018-05-09 18:29:04 +02:00
}
// Repo infos
2022-04-08 17:11:15 +08:00
if err = issue . LoadRepo ( ctx ) ; err != nil {
2022-10-24 21:29:17 +02:00
return false , nil , fmt . Errorf ( "loadRepo: %w" , err )
2018-05-09 18:29:04 +02:00
}
2021-03-15 02:52:12 +08:00
opts := & CreateCommentOptions {
2019-11-16 02:18:09 +08:00
Type : CommentTypeAssignees ,
Doer : doer ,
Repo : issue . Repo ,
Issue : issue ,
RemovedAssignee : removed ,
AssigneeID : assigneeID ,
2019-12-01 10:44:39 +08:00
}
// Comment
2022-12-10 10:46:31 +08:00
comment , err = CreateComment ( ctx , opts )
2019-10-25 16:46:37 +02:00
if err != nil {
2022-10-24 21:29:17 +02:00
return false , nil , fmt . Errorf ( "createComment: %w" , err )
2018-05-09 18:29:04 +02:00
}
2018-11-28 19:26:14 +08:00
// if pull request is in the middle of creation - don't call webhook
2018-11-21 01:10:18 +08:00
if isCreate {
2019-10-25 16:46:37 +02:00
return removed , comment , err
2018-11-21 01:10:18 +08:00
}
2019-10-25 16:46:37 +02:00
return removed , comment , nil
2018-05-09 18:29:04 +02:00
}
2019-10-25 16:46:37 +02:00
// toggles user assignee state in database
2022-05-20 22:08:52 +08:00
func toggleUserAssignee ( ctx context . Context , issue * Issue , assigneeID int64 ) ( removed bool , err error ) {
2019-10-25 16:46:37 +02:00
// Check if the user exists
2022-12-03 10:48:26 +08:00
assignee , err := user_model . GetUserByID ( ctx , assigneeID )
2019-10-25 16:46:37 +02:00
if err != nil {
return false , err
2018-05-09 18:29:04 +02:00
}
2019-10-25 16:46:37 +02:00
// Check if the submitted user is already assigned, if yes delete him otherwise add him
2022-05-20 22:08:52 +08:00
found := false
i := 0
for ; i < len ( issue . Assignees ) ; i ++ {
2019-10-25 16:46:37 +02:00
if issue . Assignees [ i ] . ID == assigneeID {
2022-05-20 22:08:52 +08:00
found = true
2019-10-25 16:46:37 +02:00
break
2018-05-09 18:29:04 +02:00
}
}
2019-10-25 16:46:37 +02:00
assigneeIn := IssueAssignees { AssigneeID : assigneeID , IssueID : issue . ID }
2022-05-20 22:08:52 +08:00
if found {
issue . Assignees = append ( issue . Assignees [ : i ] , issue . Assignees [ i + 1 : ] ... )
_ , err = db . DeleteByBean ( ctx , & assigneeIn )
2018-05-09 18:29:04 +02:00
if err != nil {
2022-05-20 22:08:52 +08:00
return found , err
2019-10-25 16:46:37 +02:00
}
} else {
issue . Assignees = append ( issue . Assignees , assignee )
2022-05-20 22:08:52 +08:00
if err = db . Insert ( ctx , & assigneeIn ) ; err != nil {
return found , err
2018-05-09 18:29:04 +02:00
}
}
2022-05-20 22:08:52 +08:00
return found , nil
2018-05-09 18:29:04 +02:00
}
// MakeIDsFromAPIAssigneesToAdd returns an array with all assignee IDs
2022-11-19 09:12:33 +01:00
func MakeIDsFromAPIAssigneesToAdd ( ctx context . Context , oneAssignee string , multipleAssignees [ ] string ) ( assigneeIDs [ ] int64 , err error ) {
2019-11-10 09:07:21 +01:00
var requestAssignees [ ] string
2018-05-09 18:29:04 +02:00
// Keeping the old assigning method for compatibility reasons
Improve utils of slices (#22379)
- Move the file `compare.go` and `slice.go` to `slice.go`.
- Fix `ExistsInSlice`, it's buggy
- It uses `sort.Search`, so it assumes that the input slice is sorted.
- It passes `func(i int) bool { return slice[i] == target })` to
`sort.Search`, that's incorrect, check the doc of `sort.Search`.
- Conbine `IsInt64InSlice(int64, []int64)` and `ExistsInSlice(string,
[]string)` to `SliceContains[T]([]T, T)`.
- Conbine `IsSliceInt64Eq([]int64, []int64)` and `IsEqualSlice([]string,
[]string)` to `SliceSortedEqual[T]([]T, T)`.
- Add `SliceEqual[T]([]T, T)` as a distinction from
`SliceSortedEqual[T]([]T, T)`.
- Redesign `RemoveIDFromList([]int64, int64) ([]int64, bool)` to
`SliceRemoveAll[T]([]T, T) []T`.
- Add `SliceContainsFunc[T]([]T, func(T) bool)` and
`SliceRemoveAllFunc[T]([]T, func(T) bool)` for general use.
- Add comments to explain why not `golang.org/x/exp/slices`.
- Add unit tests.
2023-01-11 13:31:16 +08:00
if oneAssignee != "" && ! util . SliceContainsString ( multipleAssignees , oneAssignee ) {
2019-11-10 09:07:21 +01:00
requestAssignees = append ( requestAssignees , oneAssignee )
}
2018-05-09 18:29:04 +02:00
2021-03-15 02:52:12 +08:00
// Prevent empty assignees
2019-11-10 09:07:21 +01:00
if len ( multipleAssignees ) > 0 && multipleAssignees [ 0 ] != "" {
requestAssignees = append ( requestAssignees , multipleAssignees ... )
2018-05-09 18:29:04 +02:00
}
// Get the IDs of all assignees
2022-11-19 09:12:33 +01:00
assigneeIDs , err = user_model . GetUserIDsByNames ( ctx , requestAssignees , false )
2018-05-09 18:29:04 +02:00
2022-06-20 12:02:49 +02:00
return assigneeIDs , err
2018-05-09 18:29:04 +02:00
}