2016-03-14 06:20:22 +03:00
// Copyright 2016 The Gogs Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package repo
import (
"fmt"
"strings"
2016-11-11 12:39:44 +03:00
api "code.gitea.io/sdk/gitea"
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"
"code.gitea.io/gitea/modules/setting"
2017-01-25 05:43:02 +03:00
"code.gitea.io/gitea/modules/util"
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
// - name: page
// in: query
// description: page number of requested issues
// type: integer
// 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
}
2017-06-25 17:51:07 +03:00
issues , err := models . Issues ( & models . IssuesOptions {
2017-12-26 02:25:16 +03:00
RepoIDs : [ ] int64 { ctx . Repo . Repository . ID } ,
2017-06-25 17:51:07 +03:00
Page : ctx . QueryInt ( "page" ) ,
2017-08-03 08:09:16 +03:00
PageSize : setting . UI . IssuePagingNum ,
2017-06-25 17:51:07 +03:00
IsClosed : isClosed ,
} )
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
// required: true
// responses:
// "200":
// "$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
}
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
// ---
// summary: Create 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: body
// in: body
// schema:
// "$ref": "#/definitions/CreateIssueOption"
// responses:
// "201":
// "$ref": "#/responses/Issue"
2016-03-14 06:20:22 +03:00
issue := & models . Issue {
RepoID : ctx . Repo . Repository . ID ,
2016-08-14 14:17:26 +03:00
Title : form . Title ,
2016-07-23 20:08:22 +03:00
PosterID : ctx . User . ID ,
2016-03-14 06:20:22 +03:00
Poster : ctx . User ,
Content : form . Body ,
}
if ctx . Repo . IsWriter ( ) {
if len ( form . Assignee ) > 0 {
assignee , err := models . GetUserByName ( form . Assignee )
if err != nil {
if models . IsErrUserNotExist ( err ) {
ctx . Error ( 422 , "" , fmt . Sprintf ( "Assignee does not exist: [name: %s]" , form . Assignee ) )
} else {
ctx . Error ( 500 , "GetUserByName" , err )
}
return
}
2016-07-23 20:08:22 +03:00
issue . AssigneeID = assignee . ID
2016-03-14 06:20:22 +03:00
}
issue . MilestoneID = form . Milestone
} else {
form . Labels = nil
}
if err := models . NewIssue ( ctx . Repo . Repository , issue , form . Labels , nil ) ; err != nil {
ctx . Error ( 500 , "NewIssue" , err )
return
}
2016-05-28 04:23:39 +03:00
if form . Closed {
if err := issue . ChangeStatus ( ctx . User , ctx . Repo . Repository , true ) ; err != nil {
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
var err error
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
// ---
// summary: Edit 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
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
// 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
}
2016-07-23 20:08:22 +03:00
if ! issue . IsPoster ( ctx . User . ID ) && ! ctx . Repo . IsWriter ( ) {
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
}
if ctx . Repo . IsWriter ( ) && form . Assignee != nil &&
( issue . Assignee == nil || issue . Assignee . LowerName != strings . ToLower ( * form . Assignee ) ) {
if len ( * form . Assignee ) == 0 {
issue . AssigneeID = 0
} else {
assignee , err := models . GetUserByName ( * form . Assignee )
if err != nil {
if models . IsErrUserNotExist ( err ) {
2016-08-25 02:05:56 +03:00
ctx . Error ( 422 , "" , fmt . Sprintf ( "assignee does not exist: [name: %s]" , * form . Assignee ) )
2016-03-14 06:20:22 +03:00
} else {
ctx . Error ( 500 , "GetUserByName" , err )
}
return
}
2016-07-23 20:08:22 +03:00
issue . AssigneeID = assignee . ID
2016-03-14 06:20:22 +03:00
}
if err = models . UpdateIssueUserByAssignee ( issue ) ; err != nil {
ctx . Error ( 500 , "UpdateIssueUserByAssignee" , err )
return
}
}
if ctx . Repo . IsWriter ( ) && form . Milestone != nil &&
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 {
2016-11-29 11:25:47 +03:00
if err = issue . ChangeStatus ( ctx . User , ctx . Repo . Repository , api . StateClosed == api . StateType ( * form . State ) ) ; err != nil {
2016-08-23 19:09:32 +03:00
ctx . Error ( 500 , "ChangeStatus" , err )
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
}