2016-03-13 23:20:22 -04:00
// Copyright 2016 The Gogs Authors. All rights reserved.
2018-06-19 11:15:11 -04:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2016-03-13 23:20:22 -04: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-17 17:23:58 -04:00
"net/http"
2016-03-13 23:20:22 -04:00
"strings"
2019-01-01 12:56:47 -05:00
"time"
2016-03-13 23:20:22 -04:00
2016-11-10 11:24:48 -05:00
"code.gitea.io/gitea/models"
"code.gitea.io/gitea/modules/context"
2019-02-20 19:54:05 -05:00
issue_indexer "code.gitea.io/gitea/modules/indexer/issues"
2016-11-10 11:24:48 -05:00
"code.gitea.io/gitea/modules/setting"
2019-08-23 12:40:30 -04:00
api "code.gitea.io/gitea/modules/structs"
2019-08-15 10:46:21 -04:00
"code.gitea.io/gitea/modules/timeutil"
2017-01-24 21:43:02 -05:00
"code.gitea.io/gitea/modules/util"
2019-09-30 09:50:44 -04:00
issue_service "code.gitea.io/gitea/services/issue"
2019-09-17 20:17:12 -04:00
milestone_service "code.gitea.io/gitea/services/milestone"
2016-03-13 23:20:22 -04:00
)
2016-11-24 02:04:31 -05:00
// ListIssues list the issues of a repository
2016-03-13 23:20:22 -04:00
func ListIssues ( ctx * context . APIContext ) {
2017-11-13 02:02:25 -05: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 10:20:44 -05: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 02:02:25 -05:00
// - name: page
// in: query
// description: page number of requested issues
// type: integer
2018-03-07 05:00:56 -05:00
// - name: q
// in: query
// description: search string
// type: string
2017-11-13 02:02:25 -05:00
// responses:
// "200":
// "$ref": "#/responses/IssueList"
2017-06-25 10:51:07 -04: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 13:17:27 -04:00
}
2018-03-07 05:00:56 -05:00
var issues [ ] * models . Issue
keyword := strings . Trim ( ctx . Query ( "q" ) , " " )
if strings . IndexByte ( keyword , 0 ) >= 0 {
keyword = ""
}
var issueIDs [ ] int64
2019-02-04 10:20:44 -05:00
var labelIDs [ ] int64
2018-03-07 05:00:56 -05:00
var err error
if len ( keyword ) > 0 {
2019-02-20 19:54:05 -05:00
issueIDs , err = issue_indexer . SearchIssuesByKeyword ( ctx . Repo . Repository . ID , keyword )
2018-03-07 05:00:56 -05:00
}
2019-02-04 10:20:44 -05: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 05:00:56 -05: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 10:20:44 -05:00
if len ( keyword ) == 0 || len ( issueIDs ) > 0 || len ( labelIDs ) > 0 {
2018-03-07 05:00:56 -05: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 10:20:44 -05:00
LabelIDs : labelIDs ,
2018-03-07 05:00:56 -05:00
} )
}
2016-03-13 23:20:22 -04:00
if err != nil {
ctx . Error ( 500 , "Issues" , err )
return
}
apiIssues := make ( [ ] * api . Issue , len ( issues ) )
for i := range issues {
2016-08-14 07:17:26 -04:00
apiIssues [ i ] = issues [ i ] . APIFormat ( )
2016-03-13 23:20:22 -04:00
}
2016-07-23 12:23:54 -04:00
ctx . SetLinkHeader ( ctx . Repo . Repository . NumIssues , setting . UI . IssuePagingNum )
2016-03-13 23:20:22 -04:00
ctx . JSON ( 200 , & apiIssues )
}
2016-11-24 02:04:31 -05:00
// GetIssue get an issue of a repository
2016-03-13 23:20:22 -04:00
func GetIssue ( ctx * context . APIContext ) {
2018-01-04 01:31:40 -05:00
// swagger:operation GET /repos/{owner}/{repo}/issues/{index} issue issueGetIssue
2017-11-13 02:02:25 -05:00
// ---
2018-01-04 01:31:40 -05:00
// summary: Get an issue
2017-11-13 02:02:25 -05: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 01:31:40 -05:00
// - name: index
2017-11-13 02:02:25 -05:00
// in: path
2018-01-04 01:31:40 -05:00
// description: index of the issue to get
2017-11-13 02:02:25 -05:00
// type: integer
2018-10-20 23:40:42 -04:00
// format: int64
2017-11-13 02:02:25 -05:00
// required: true
// responses:
// "200":
// "$ref": "#/responses/Issue"
2019-02-19 12:07:19 -05:00
issue , err := models . GetIssueWithAttrsByIndex ( ctx . Repo . Repository . ID , ctx . ParamsInt64 ( ":index" ) )
2016-03-13 23:20:22 -04:00
if err != nil {
if models . IsErrIssueNotExist ( err ) {
2019-03-18 22:29:43 -04:00
ctx . NotFound ( )
2016-03-13 23:20:22 -04:00
} else {
ctx . Error ( 500 , "GetIssueByIndex" , err )
}
return
}
2016-08-14 07:17:26 -04:00
ctx . JSON ( 200 , issue . APIFormat ( ) )
2016-03-13 23:20:22 -04:00
}
2016-11-24 02:04:31 -05:00
// CreateIssue create an issue of a repository
2016-03-13 23:20:22 -04:00
func CreateIssue ( ctx * context . APIContext , form api . CreateIssueOption ) {
2017-11-13 02:02:25 -05:00
// swagger:operation POST /repos/{owner}/{repo}/issues issue issueCreateIssue
// ---
2019-01-01 12:56:47 -05:00
// summary: Create an issue. If using deadline only the date will be taken into account, and time of day ignored.
2017-11-13 02:02:25 -05: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 15:05:28 -04:00
2019-08-15 10:46:21 -04:00
var deadlineUnix timeutil . TimeStamp
2018-11-28 06:26:14 -05:00
if form . Deadline != nil && ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
2019-08-15 10:46:21 -04:00
deadlineUnix = timeutil . TimeStamp ( form . Deadline . Unix ( ) )
2018-05-01 15:05:28 -04:00
}
2016-03-13 23:20:22 -04:00
issue := & models . Issue {
2018-05-01 15:05:28 -04:00
RepoID : ctx . Repo . Repository . ID ,
2018-12-13 10:55:43 -05:00
Repo : ctx . Repo . Repository ,
2018-05-01 15:05:28 -04:00
Title : form . Title ,
PosterID : ctx . User . ID ,
Poster : ctx . User ,
Content : form . Body ,
DeadlineUnix : deadlineUnix ,
2016-03-13 23:20:22 -04:00
}
2018-06-19 11:15:11 -04:00
var assigneeIDs = make ( [ ] int64 , 0 )
var err error
2018-11-28 06:26:14 -05:00
if ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
2018-06-19 11:15:11 -04: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-13 23:20:22 -04:00
}
2019-10-25 10:46:37 -04:00
// Check if the passed assignees is assignable
for _ , aID := range assigneeIDs {
assignee , err := models . GetUserByID ( aID )
if err != nil {
ctx . Error ( 500 , "GetUserByID" , err )
return
}
valid , err := models . CanBeAssigned ( assignee , ctx . Repo . Repository , false )
if err != nil {
ctx . Error ( 500 , "canBeAssigned" , err )
return
}
if ! valid {
ctx . Error ( 422 , "canBeAssigned" , models . ErrUserDoesNotHaveAccessToRepo { UserID : aID , RepoName : ctx . Repo . Repository . Name } )
return
}
}
2018-06-19 11:15:11 -04:00
} else {
// setting labels is not allowed if user is not a writer
form . Labels = make ( [ ] int64 , 0 )
2016-03-13 23:20:22 -04:00
}
2019-10-28 12:45:43 -04:00
if err := issue_service . NewIssue ( ctx . Repo . Repository , issue , form . Labels , nil , assigneeIDs ) ; err != nil {
2018-05-09 12:29:04 -04:00
if models . IsErrUserDoesNotHaveAccessToRepo ( err ) {
ctx . Error ( 400 , "UserDoesNotHaveAccessToRepo" , err )
return
}
2016-03-13 23:20:22 -04:00
ctx . Error ( 500 , "NewIssue" , err )
return
}
2016-05-27 21:23:39 -04:00
if form . Closed {
2019-10-28 01:26:46 -04:00
if err := issue_service . ChangeStatus ( issue , ctx . User , true ) ; err != nil {
2018-07-17 17:23:58 -04:00
if models . IsErrDependenciesLeft ( err ) {
ctx . Error ( http . StatusPreconditionFailed , "DependenciesLeft" , "cannot close this issue because it still has open dependencies" )
return
}
2016-08-14 07:17:26 -04:00
ctx . Error ( 500 , "ChangeStatus" , err )
2016-05-27 21:23:39 -04:00
return
}
}
2016-03-13 23:20:22 -04: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 07:17:26 -04:00
ctx . JSON ( 201 , issue . APIFormat ( ) )
2016-03-13 23:20:22 -04:00
}
2016-11-24 02:04:31 -05:00
// EditIssue modify an issue of a repository
2016-03-13 23:20:22 -04:00
func EditIssue ( ctx * context . APIContext , form api . EditIssueOption ) {
2018-01-04 01:31:40 -05:00
// swagger:operation PATCH /repos/{owner}/{repo}/issues/{index} issue issueEditIssue
2017-11-13 02:02:25 -05:00
// ---
2019-01-01 12:56:47 -05:00
// summary: Edit an issue. If using deadline only the date will be taken into account, and time of day ignored.
2017-11-13 02:02:25 -05: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 01:31:40 -05:00
// - name: index
2017-11-13 02:02:25 -05:00
// in: path
2018-01-04 01:31:40 -05:00
// description: index of the issue to edit
2017-11-13 02:02:25 -05:00
// type: integer
2018-10-20 23:40:42 -04:00
// format: int64
2017-11-13 02:02:25 -05:00
// required: true
// - name: body
// in: body
// schema:
// "$ref": "#/definitions/EditIssueOption"
// responses:
// "201":
// "$ref": "#/responses/Issue"
2016-03-13 23:20:22 -04:00
issue , err := models . GetIssueByIndex ( ctx . Repo . Repository . ID , ctx . ParamsInt64 ( ":index" ) )
if err != nil {
if models . IsErrIssueNotExist ( err ) {
2019-03-18 22:29:43 -04:00
ctx . NotFound ( )
2016-03-13 23:20:22 -04:00
} else {
ctx . Error ( 500 , "GetIssueByIndex" , err )
}
return
}
2018-12-13 10:55:43 -05:00
issue . Repo = ctx . Repo . Repository
2016-03-13 23:20:22 -04:00
2019-04-23 13:07:12 -04:00
err = issue . LoadAttributes ( )
if err != nil {
ctx . Error ( 500 , "LoadAttributes" , err )
return
}
2018-11-28 06:26:14 -05:00
if ! issue . IsPoster ( ctx . User . ID ) && ! ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
2016-03-13 23:20:22 -04:00
ctx . Status ( 403 )
return
}
if len ( form . Title ) > 0 {
2016-08-14 06:32:24 -04:00
issue . Title = form . Title
2016-03-13 23:20:22 -04:00
}
if form . Body != nil {
issue . Content = * form . Body
}
2018-05-09 12:29:04 -04:00
// Update the deadline
2019-10-27 19:35:20 -04:00
if form . Deadline != nil && ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
deadlineUnix := timeutil . TimeStamp ( form . Deadline . Unix ( ) )
if err := models . UpdateIssueDeadline ( issue , deadlineUnix , ctx . User ) ; err != nil {
ctx . Error ( 500 , "UpdateIssueDeadline" , err )
return
}
issue . DeadlineUnix = deadlineUnix
2018-05-01 15:05:28 -04:00
}
2018-05-09 12:29:04 -04:00
// Add/delete assignees
2019-03-09 16:15:45 -05:00
// Deleting is done the GitHub way (quote from their api documentation):
2018-05-09 12:29:04 -04: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 06:26:14 -05:00
if ctx . Repo . CanWrite ( models . UnitTypeIssues ) && ( form . Assignees != nil || form . Assignee != nil ) {
2018-05-09 12:29:04 -04:00
oneAssignee := ""
if form . Assignee != nil {
oneAssignee = * form . Assignee
2016-03-13 23:20:22 -04:00
}
2019-10-25 10:46:37 -04:00
err = issue_service . UpdateAssignees ( issue , oneAssignee , form . Assignees , ctx . User )
2018-05-09 12:29:04 -04:00
if err != nil {
2019-10-25 10:46:37 -04:00
ctx . Error ( 500 , "UpdateAssignees" , err )
2016-03-13 23:20:22 -04:00
return
}
}
2018-05-09 12:29:04 -04:00
2018-11-28 06:26:14 -05:00
if ctx . Repo . CanWrite ( models . UnitTypeIssues ) && form . Milestone != nil &&
2016-03-13 23:20:22 -04:00
issue . MilestoneID != * form . Milestone {
2016-08-15 21:40:32 -04:00
oldMilestoneID := issue . MilestoneID
2016-03-13 23:20:22 -04:00
issue . MilestoneID = * form . Milestone
2019-09-17 20:17:12 -04:00
if err = milestone_service . ChangeMilestoneAssign ( issue , ctx . User , oldMilestoneID ) ; err != nil {
2016-03-13 23:20:22 -04:00
ctx . Error ( 500 , "ChangeMilestoneAssign" , err )
return
}
}
if err = models . UpdateIssue ( issue ) ; err != nil {
ctx . Error ( 500 , "UpdateIssue" , err )
return
}
2016-08-23 12:09:32 -04:00
if form . State != nil {
2019-10-28 01:26:46 -04:00
if err = issue_service . ChangeStatus ( issue , ctx . User , api . StateClosed == api . StateType ( * form . State ) ) ; err != nil {
2018-07-17 17:23:58 -04:00
if models . IsErrDependenciesLeft ( err ) {
ctx . Error ( http . StatusPreconditionFailed , "DependenciesLeft" , "cannot close this issue because it still has open dependencies" )
return
}
2016-08-23 12:09:32 -04:00
ctx . Error ( 500 , "ChangeStatus" , err )
return
}
}
2016-03-13 23:20:22 -04: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 07:17:26 -04:00
ctx . JSON ( 201 , issue . APIFormat ( ) )
2016-03-13 23:20:22 -04:00
}
2018-07-16 08:43:00 -04: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 12:56:47 -05: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 08:43:00 -04: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-20 23:40:42 -04:00
// format: int64
2018-07-16 08:43:00 -04: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 ) {
2019-03-18 22:29:43 -04:00
ctx . NotFound ( )
2018-07-16 08:43:00 -04:00
} else {
ctx . Error ( 500 , "GetIssueByIndex" , err )
}
return
}
2018-11-28 06:26:14 -05:00
if ! ctx . Repo . CanWrite ( models . UnitTypeIssues ) {
2018-07-16 08:43:00 -04:00
ctx . Status ( 403 )
return
}
2019-08-15 10:46:21 -04:00
var deadlineUnix timeutil . TimeStamp
2019-01-01 12:56:47 -05:00
var deadline time . Time
2018-07-16 08:43:00 -04:00
if form . Deadline != nil && ! form . Deadline . IsZero ( ) {
2019-01-01 12:56:47 -05:00
deadline = time . Date ( form . Deadline . Year ( ) , form . Deadline . Month ( ) , form . Deadline . Day ( ) ,
23 , 59 , 59 , 0 , form . Deadline . Location ( ) )
2019-08-15 10:46:21 -04:00
deadlineUnix = timeutil . TimeStamp ( deadline . Unix ( ) )
2018-07-16 08:43:00 -04:00
}
if err := models . UpdateIssueDeadline ( issue , deadlineUnix , ctx . User ) ; err != nil {
ctx . Error ( 500 , "UpdateIssueDeadline" , err )
return
}
2019-01-01 12:56:47 -05:00
ctx . JSON ( 201 , api . IssueDeadline { Deadline : & deadline } )
2018-07-16 08:43:00 -04:00
}
2019-02-06 21:57:25 -05: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 ) {
2019-03-18 22:29:43 -04:00
ctx . NotFound ( )
2019-02-06 21:57:25 -05:00
} 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 ) {
2019-03-18 22:29:43 -04:00
ctx . NotFound ( )
2019-02-06 21:57:25 -05:00
} 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 )
}