2016-03-14 06:20:22 +03:00
// Copyright 2016 The Gogs Authors. All rights reserved.
2018-06-19 18:15:11 +03:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2016-03-14 06:20:22 +03:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package repo
import (
"fmt"
2018-07-18 00:23:58 +03:00
"net/http"
2016-03-14 06:20:22 +03:00
"strings"
2019-01-01 20:56:47 +03:00
"time"
2016-03-14 06:20:22 +03:00
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/models"
"code.gitea.io/gitea/modules/context"
2019-02-21 03:54:05 +03:00
issue_indexer "code.gitea.io/gitea/modules/indexer/issues"
2018-10-18 14:23:05 +03:00
"code.gitea.io/gitea/modules/notification"
2016-11-10 19:24:48 +03:00
"code.gitea.io/gitea/modules/setting"
2017-01-25 05:43:02 +03:00
"code.gitea.io/gitea/modules/util"
2018-03-29 16:32:40 +03:00
api "code.gitea.io/sdk/gitea"
2016-03-14 06:20:22 +03:00
)
2016-11-24 10:04:31 +03:00
// ListIssues list the issues of a repository
2016-03-14 06:20:22 +03:00
func ListIssues ( ctx * context . APIContext ) {
2017-11-13 10:02:25 +03:00
// swagger:operation GET /repos/{owner}/{repo}/issues issue issueListIssues
// ---
// summary: List a repository's issues
// produces:
// - application/json
// parameters:
// - name: owner
// in: path
// description: owner of the repo
// type: string
// required: true
// - name: repo
// in: path
// description: name of the repo
// type: string
// required: true
// - name: state
// in: query
// description: whether issue is open or closed
// type: string
2019-02-04 18:20:44 +03:00
// - name: labels
// in: query
// description: comma separated list of labels. Fetch only issues that have any of this labels. Non existent labels are discarded
// type: string
2017-11-13 10:02:25 +03:00
// - name: page
// in: query
// description: page number of requested issues
// type: integer
2018-03-07 13:00:56 +03:00
// - name: q
// in: query
// description: search string
// type: string
2017-11-13 10:02:25 +03:00
// responses:
// "200":
// "$ref": "#/responses/IssueList"
2017-06-25 17:51:07 +03:00
var isClosed util . OptionalBool
switch ctx . Query ( "state" ) {
case "closed" :
isClosed = util . OptionalBoolTrue
case "all" :
isClosed = util . OptionalBoolNone
default :
isClosed = util . OptionalBoolFalse
2016-10-07 20:17:27 +03:00
}
2018-03-07 13:00:56 +03:00
var issues [ ] * models . Issue
keyword := strings . Trim ( ctx . Query ( "q" ) , " " )
if strings . IndexByte ( keyword , 0 ) >= 0 {
keyword = ""
}
var issueIDs [ ] int64
2019-02-04 18:20:44 +03:00
var labelIDs [ ] int64
2018-03-07 13:00:56 +03:00
var err error
if len ( keyword ) > 0 {
2019-02-21 03:54:05 +03:00
issueIDs , err = issue_indexer . SearchIssuesByKeyword ( ctx . Repo . Repository . ID , keyword )
2018-03-07 13:00:56 +03:00
}
2019-02-04 18:20:44 +03:00
if splitted := strings . Split ( ctx . Query ( "labels" ) , "," ) ; len ( splitted ) > 0 {
labelIDs , err = models . GetLabelIDsInRepoByNames ( ctx . Repo . Repository . ID , splitted )
if err != nil {
ctx . Error ( 500 , "GetLabelIDsInRepoByNames" , err )
return
}
}
2018-03-07 13:00:56 +03:00
// Only fetch the issues if we either don't have a keyword or the search returned issues
// This would otherwise return all issues if no issues were found by the search.
2019-02-04 18:20:44 +03:00
if len ( keyword ) == 0 || len ( issueIDs ) > 0 || len ( labelIDs ) > 0 {
2018-03-07 13:00:56 +03:00
issues , err = models . Issues ( & models . IssuesOptions {
RepoIDs : [ ] int64 { ctx . Repo . Repository . ID } ,
Page : ctx . QueryInt ( "page" ) ,
PageSize : setting . UI . IssuePagingNum ,
IsClosed : isClosed ,
IssueIDs : issueIDs ,
2019-02-04 18:20:44 +03:00
LabelIDs : labelIDs ,
2018-03-07 13:00:56 +03:00
} )
}
2016-03-14 06:20:22 +03:00
if err != nil {
ctx . Error ( 500 , "Issues" , err )
return
}
apiIssues := make ( [ ] * api . Issue , len ( issues ) )
for i := range issues {
2016-08-14 14:17:26 +03:00
apiIssues [ i ] = issues [ i ] . APIFormat ( )
2016-03-14 06:20:22 +03:00
}
2016-07-23 19:23:54 +03:00
ctx . SetLinkHeader ( ctx . Repo . Repository . NumIssues , setting . UI . IssuePagingNum )
2016-03-14 06:20:22 +03:00
ctx . JSON ( 200 , & apiIssues )
}
2016-11-24 10:04:31 +03:00
// GetIssue get an issue of a repository
2016-03-14 06:20:22 +03:00
func GetIssue ( ctx * context . APIContext ) {
2018-01-04 09:31:40 +03:00
// swagger:operation GET /repos/{owner}/{repo}/issues/{index} issue issueGetIssue
2017-11-13 10:02:25 +03:00
// ---
2018-01-04 09:31:40 +03:00
// summary: Get an issue
2017-11-13 10:02:25 +03:00
// produces:
// - application/json
// parameters:
// - name: owner
// in: path
// description: owner of the repo
// type: string
// required: true
// - name: repo
// in: path
// description: name of the repo
// type: string
// required: true
2018-01-04 09:31:40 +03:00
// - name: index
2017-11-13 10:02:25 +03:00
// in: path
2018-01-04 09:31:40 +03:00
// description: index of the issue to get
2017-11-13 10:02:25 +03:00
// type: integer
2018-10-21 06:40:42 +03:00
// format: int64
2017-11-13 10:02:25 +03:00
// required: true
// responses:
// "200":
// "$ref": "#/responses/Issue"
2019-02-19 20:07:19 +03:00
issue , err := models . GetIssueWithAttrsByIndex ( ctx . Repo . Repository . ID , ctx . ParamsInt64 ( ":index" ) )
2016-03-14 06:20:22 +03:00
if err != nil {
if models . IsErrIssueNotExist ( err ) {
ctx . Status ( 404 )
} else {
ctx . Error ( 500 , "GetIssueByIndex" , err )
}
return
}
2016-08-14 14:17:26 +03:00
ctx . JSON ( 200 , issue . APIFormat ( ) )
2016-03-14 06:20:22 +03:00
}
2016-11-24 10:04:31 +03:00
// CreateIssue create an issue of a repository
2016-03-14 06:20:22 +03:00
func CreateIssue ( ctx * context . APIContext , form api . CreateIssueOption ) {
2017-11-13 10:02:25 +03:00
// swagger:operation POST /repos/{owner}/{repo}/issues issue issueCreateIssue
// ---
2019-01-01 20:56:47 +03:00
// summary: Create an issue. If using deadline only the date will be taken into account, and time of day ignored.
2017-11-13 10:02:25 +03:00
// consumes:
// - application/json
// produces:
// - application/json
// parameters:
// - name: owner
// in: path
// description: owner of the repo
// type: string
// required: true
// - name: repo
// in: path
// description: name of the repo
// type: string
// required: true
// - name: body
// in: body
// schema:
// "$ref": "#/definitions/CreateIssueOption"
// responses:
// "201":
// "$ref": "#/responses/Issue"
2018-05-01 22:05:28 +03:00
var deadlineUnix util . TimeStamp
2018-11-28 14:26:14 +03:00
if form . Deadline != nil && ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
2018-05-01 22:05:28 +03:00
deadlineUnix = util . TimeStamp ( form . Deadline . Unix ( ) )
}
2016-03-14 06:20:22 +03:00
issue := & models . Issue {
2018-05-01 22:05:28 +03:00
RepoID : ctx . Repo . Repository . ID ,
2018-12-13 18:55:43 +03:00
Repo : ctx . Repo . Repository ,
2018-05-01 22:05:28 +03:00
Title : form . Title ,
PosterID : ctx . User . ID ,
Poster : ctx . User ,
Content : form . Body ,
DeadlineUnix : deadlineUnix ,
2016-03-14 06:20:22 +03:00
}
2018-06-19 18:15:11 +03:00
var assigneeIDs = make ( [ ] int64 , 0 )
var err error
2018-11-28 14:26:14 +03:00
if ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
2018-06-19 18:15:11 +03:00
issue . MilestoneID = form . Milestone
assigneeIDs , err = models . MakeIDsFromAPIAssigneesToAdd ( form . Assignee , form . Assignees )
if err != nil {
if models . IsErrUserNotExist ( err ) {
ctx . Error ( 422 , "" , fmt . Sprintf ( "Assignee does not exist: [name: %s]" , err ) )
} else {
ctx . Error ( 500 , "AddAssigneeByName" , err )
}
return
2016-03-14 06:20:22 +03:00
}
2018-06-19 18:15:11 +03:00
} else {
// setting labels is not allowed if user is not a writer
form . Labels = make ( [ ] int64 , 0 )
2016-03-14 06:20:22 +03:00
}
2018-05-09 19:29:04 +03:00
if err := models . NewIssue ( ctx . Repo . Repository , issue , form . Labels , assigneeIDs , nil ) ; err != nil {
if models . IsErrUserDoesNotHaveAccessToRepo ( err ) {
ctx . Error ( 400 , "UserDoesNotHaveAccessToRepo" , err )
return
}
2016-03-14 06:20:22 +03:00
ctx . Error ( 500 , "NewIssue" , err )
return
}
2018-10-18 14:23:05 +03:00
notification . NotifyNewIssue ( issue )
2016-05-28 04:23:39 +03:00
if form . Closed {
2018-12-13 18:55:43 +03:00
if err := issue . ChangeStatus ( ctx . User , true ) ; err != nil {
2018-07-18 00:23:58 +03:00
if models . IsErrDependenciesLeft ( err ) {
ctx . Error ( http . StatusPreconditionFailed , "DependenciesLeft" , "cannot close this issue because it still has open dependencies" )
return
}
2016-08-14 14:17:26 +03:00
ctx . Error ( 500 , "ChangeStatus" , err )
2016-05-28 04:23:39 +03:00
return
}
}
2016-03-14 06:20:22 +03:00
// Refetch from database to assign some automatic values
issue , err = models . GetIssueByID ( issue . ID )
if err != nil {
ctx . Error ( 500 , "GetIssueByID" , err )
return
}
2016-08-14 14:17:26 +03:00
ctx . JSON ( 201 , issue . APIFormat ( ) )
2016-03-14 06:20:22 +03:00
}
2016-11-24 10:04:31 +03:00
// EditIssue modify an issue of a repository
2016-03-14 06:20:22 +03:00
func EditIssue ( ctx * context . APIContext , form api . EditIssueOption ) {
2018-01-04 09:31:40 +03:00
// swagger:operation PATCH /repos/{owner}/{repo}/issues/{index} issue issueEditIssue
2017-11-13 10:02:25 +03:00
// ---
2019-01-01 20:56:47 +03:00
// summary: Edit an issue. If using deadline only the date will be taken into account, and time of day ignored.
2017-11-13 10:02:25 +03:00
// consumes:
// - application/json
// produces:
// - application/json
// parameters:
// - name: owner
// in: path
// description: owner of the repo
// type: string
// required: true
// - name: repo
// in: path
// description: name of the repo
// type: string
// required: true
2018-01-04 09:31:40 +03:00
// - name: index
2017-11-13 10:02:25 +03:00
// in: path
2018-01-04 09:31:40 +03:00
// description: index of the issue to edit
2017-11-13 10:02:25 +03:00
// type: integer
2018-10-21 06:40:42 +03:00
// format: int64
2017-11-13 10:02:25 +03:00
// required: true
// - name: body
// in: body
// schema:
// "$ref": "#/definitions/EditIssueOption"
// responses:
// "201":
// "$ref": "#/responses/Issue"
2016-03-14 06:20:22 +03:00
issue , err := models . GetIssueByIndex ( ctx . Repo . Repository . ID , ctx . ParamsInt64 ( ":index" ) )
if err != nil {
if models . IsErrIssueNotExist ( err ) {
ctx . Status ( 404 )
} else {
ctx . Error ( 500 , "GetIssueByIndex" , err )
}
return
}
2018-12-13 18:55:43 +03:00
issue . Repo = ctx . Repo . Repository
2016-03-14 06:20:22 +03:00
2018-11-28 14:26:14 +03:00
if ! issue . IsPoster ( ctx . User . ID ) && ! ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
2016-03-14 06:20:22 +03:00
ctx . Status ( 403 )
return
}
if len ( form . Title ) > 0 {
2016-08-14 13:32:24 +03:00
issue . Title = form . Title
2016-03-14 06:20:22 +03:00
}
if form . Body != nil {
issue . Content = * form . Body
}
2018-05-09 19:29:04 +03:00
// Update the deadline
2018-05-01 22:05:28 +03:00
var deadlineUnix util . TimeStamp
2018-11-28 14:26:14 +03:00
if form . Deadline != nil && ! form . Deadline . IsZero ( ) && ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
2018-05-01 22:05:28 +03:00
deadlineUnix = util . TimeStamp ( form . Deadline . Unix ( ) )
}
if err := models . UpdateIssueDeadline ( issue , deadlineUnix , ctx . User ) ; err != nil {
ctx . Error ( 500 , "UpdateIssueDeadline" , err )
return
}
2018-05-09 19:29:04 +03:00
// Add/delete assignees
2019-03-10 00:15:45 +03:00
// Deleting is done the GitHub way (quote from their api documentation):
2018-05-09 19:29:04 +03:00
// https://developer.github.com/v3/issues/#edit-an-issue
// "assignees" (array): Logins for Users to assign to this issue.
// Pass one or more user logins to replace the set of assignees on this Issue.
// Send an empty array ([]) to clear all assignees from the Issue.
2018-11-28 14:26:14 +03:00
if ctx . Repo . CanWrite ( models . UnitTypeIssues ) && ( form . Assignees != nil || form . Assignee != nil ) {
2018-05-09 19:29:04 +03:00
oneAssignee := ""
if form . Assignee != nil {
oneAssignee = * form . Assignee
2016-03-14 06:20:22 +03:00
}
2018-05-09 19:29:04 +03:00
err = models . UpdateAPIAssignee ( issue , oneAssignee , form . Assignees , ctx . User )
if err != nil {
ctx . Error ( 500 , "UpdateAPIAssignee" , err )
2016-03-14 06:20:22 +03:00
return
}
}
2018-05-09 19:29:04 +03:00
2018-11-28 14:26:14 +03:00
if ctx . Repo . CanWrite ( models . UnitTypeIssues ) && form . Milestone != nil &&
2016-03-14 06:20:22 +03:00
issue . MilestoneID != * form . Milestone {
2016-08-16 04:40:32 +03:00
oldMilestoneID := issue . MilestoneID
2016-03-14 06:20:22 +03:00
issue . MilestoneID = * form . Milestone
2017-02-01 05:36:08 +03:00
if err = models . ChangeMilestoneAssign ( issue , ctx . User , oldMilestoneID ) ; err != nil {
2016-03-14 06:20:22 +03:00
ctx . Error ( 500 , "ChangeMilestoneAssign" , err )
return
}
}
if err = models . UpdateIssue ( issue ) ; err != nil {
ctx . Error ( 500 , "UpdateIssue" , err )
return
}
2016-08-23 19:09:32 +03:00
if form . State != nil {
2018-12-13 18:55:43 +03:00
if err = issue . ChangeStatus ( ctx . User , api . StateClosed == api . StateType ( * form . State ) ) ; err != nil {
2018-07-18 00:23:58 +03:00
if models . IsErrDependenciesLeft ( err ) {
ctx . Error ( http . StatusPreconditionFailed , "DependenciesLeft" , "cannot close this issue because it still has open dependencies" )
return
}
2016-08-23 19:09:32 +03:00
ctx . Error ( 500 , "ChangeStatus" , err )
return
}
2018-10-18 14:23:05 +03:00
notification . NotifyIssueChangeStatus ( ctx . User , issue , api . StateClosed == api . StateType ( * form . State ) )
2016-08-23 19:09:32 +03:00
}
2016-03-14 06:20:22 +03:00
// Refetch from database to assign some automatic values
issue , err = models . GetIssueByID ( issue . ID )
if err != nil {
ctx . Error ( 500 , "GetIssueByID" , err )
return
}
2016-08-14 14:17:26 +03:00
ctx . JSON ( 201 , issue . APIFormat ( ) )
2016-03-14 06:20:22 +03:00
}
2018-07-16 15:43:00 +03:00
// UpdateIssueDeadline updates an issue deadline
func UpdateIssueDeadline ( ctx * context . APIContext , form api . EditDeadlineOption ) {
// swagger:operation POST /repos/{owner}/{repo}/issues/{index}/deadline issue issueEditIssueDeadline
// ---
2019-01-01 20:56:47 +03:00
// summary: Set an issue deadline. If set to null, the deadline is deleted. If using deadline only the date will be taken into account, and time of day ignored.
2018-07-16 15:43:00 +03:00
// consumes:
// - application/json
// produces:
// - application/json
// parameters:
// - name: owner
// in: path
// description: owner of the repo
// type: string
// required: true
// - name: repo
// in: path
// description: name of the repo
// type: string
// required: true
// - name: index
// in: path
// description: index of the issue to create or update a deadline on
// type: integer
2018-10-21 06:40:42 +03:00
// format: int64
2018-07-16 15:43:00 +03:00
// required: true
// - name: body
// in: body
// schema:
// "$ref": "#/definitions/EditDeadlineOption"
// responses:
// "201":
// "$ref": "#/responses/IssueDeadline"
// "403":
// description: Not repo writer
// "404":
// description: Issue not found
issue , err := models . GetIssueByIndex ( ctx . Repo . Repository . ID , ctx . ParamsInt64 ( ":index" ) )
if err != nil {
if models . IsErrIssueNotExist ( err ) {
ctx . Status ( 404 )
} else {
ctx . Error ( 500 , "GetIssueByIndex" , err )
}
return
}
2018-11-28 14:26:14 +03:00
if ! ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
2018-07-16 15:43:00 +03:00
ctx . Status ( 403 )
return
}
var deadlineUnix util . TimeStamp
2019-01-01 20:56:47 +03:00
var deadline time . Time
2018-07-16 15:43:00 +03:00
if form . Deadline != nil && ! form . Deadline . IsZero ( ) {
2019-01-01 20:56:47 +03:00
deadline = time . Date ( form . Deadline . Year ( ) , form . Deadline . Month ( ) , form . Deadline . Day ( ) ,
23 , 59 , 59 , 0 , form . Deadline . Location ( ) )
deadlineUnix = util . TimeStamp ( deadline . Unix ( ) )
2018-07-16 15:43:00 +03:00
}
if err := models . UpdateIssueDeadline ( issue , deadlineUnix , ctx . User ) ; err != nil {
ctx . Error ( 500 , "UpdateIssueDeadline" , err )
return
}
2019-01-01 20:56:47 +03:00
ctx . JSON ( 201 , api . IssueDeadline { Deadline : & deadline } )
2018-07-16 15:43:00 +03:00
}
2019-02-07 05:57:25 +03:00
// StartIssueStopwatch creates a stopwatch for the given issue.
func StartIssueStopwatch ( ctx * context . APIContext ) {
// swagger:operation POST /repos/{owner}/{repo}/issues/{index}/stopwatch/start issue issueStartStopWatch
// ---
// summary: Start stopwatch on an issue.
// consumes:
// - application/json
// produces:
// - application/json
// parameters:
// - name: owner
// in: path
// description: owner of the repo
// type: string
// required: true
// - name: repo
// in: path
// description: name of the repo
// type: string
// required: true
// - name: index
// in: path
// description: index of the issue to create the stopwatch on
// type: integer
// format: int64
// required: true
// responses:
// "201":
// "$ref": "#/responses/empty"
// "403":
// description: Not repo writer, user does not have rights to toggle stopwatch
// "404":
// description: Issue not found
// "409":
// description: Cannot start a stopwatch again if it already exists
issue , err := models . GetIssueByIndex ( ctx . Repo . Repository . ID , ctx . ParamsInt64 ( ":index" ) )
if err != nil {
if models . IsErrIssueNotExist ( err ) {
ctx . Status ( 404 )
} else {
ctx . Error ( 500 , "GetIssueByIndex" , err )
}
return
}
if ! ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
ctx . Status ( 403 )
return
}
if ! ctx . Repo . CanUseTimetracker ( issue , ctx . User ) {
ctx . Status ( 403 )
return
}
if models . StopwatchExists ( ctx . User . ID , issue . ID ) {
ctx . Error ( 409 , "StopwatchExists" , "a stopwatch has already been started for this issue" )
return
}
if err := models . CreateOrStopIssueStopwatch ( ctx . User , issue ) ; err != nil {
ctx . Error ( 500 , "CreateOrStopIssueStopwatch" , err )
return
}
ctx . Status ( 201 )
}
// StopIssueStopwatch stops a stopwatch for the given issue.
func StopIssueStopwatch ( ctx * context . APIContext ) {
// swagger:operation POST /repos/{owner}/{repo}/issues/{index}/stopwatch/stop issue issueStopWatch
// ---
// summary: Stop an issue's existing stopwatch.
// consumes:
// - application/json
// produces:
// - application/json
// parameters:
// - name: owner
// in: path
// description: owner of the repo
// type: string
// required: true
// - name: repo
// in: path
// description: name of the repo
// type: string
// required: true
// - name: index
// in: path
// description: index of the issue to stop the stopwatch on
// type: integer
// format: int64
// required: true
// responses:
// "201":
// "$ref": "#/responses/empty"
// "403":
// description: Not repo writer, user does not have rights to toggle stopwatch
// "404":
// description: Issue not found
// "409":
// description: Cannot stop a non existent stopwatch
issue , err := models . GetIssueByIndex ( ctx . Repo . Repository . ID , ctx . ParamsInt64 ( ":index" ) )
if err != nil {
if models . IsErrIssueNotExist ( err ) {
ctx . Status ( 404 )
} else {
ctx . Error ( 500 , "GetIssueByIndex" , err )
}
return
}
if ! ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
ctx . Status ( 403 )
return
}
if ! ctx . Repo . CanUseTimetracker ( issue , ctx . User ) {
ctx . Status ( 403 )
return
}
if ! models . StopwatchExists ( ctx . User . ID , issue . ID ) {
ctx . Error ( 409 , "StopwatchExists" , "cannot stop a non existent stopwatch" )
return
}
if err := models . CreateOrStopIssueStopwatch ( ctx . User , issue ) ; err != nil {
ctx . Error ( 500 , "CreateOrStopIssueStopwatch" , err )
return
}
ctx . Status ( 201 )
}