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