From 4c2038367fede6ee37625451ce46d48a998c34fa Mon Sep 17 00:00:00 2001 From: 6543 <6543@obermui.de> Date: Sun, 30 Aug 2020 14:02:49 +0200 Subject: [PATCH 1/3] API: Milestone endpoints accept names too --- routers/api/v1/repo/milestone.go | 76 ++++++++++++++++++++------------ templates/swagger/v1_json.tmpl | 15 +++---- 2 files changed, 55 insertions(+), 36 deletions(-) diff --git a/routers/api/v1/repo/milestone.go b/routers/api/v1/repo/milestone.go index f6f6b29465bff..13b2413f055ca 100644 --- a/routers/api/v1/repo/milestone.go +++ b/routers/api/v1/repo/milestone.go @@ -7,6 +7,7 @@ package repo import ( "net/http" + "strconv" "time" "code.gitea.io/gitea/models" @@ -73,7 +74,7 @@ func ListMilestones(ctx *context.APIContext) { ctx.JSON(http.StatusOK, &apiMilestones) } -// GetMilestone get a milestone for a repository +// GetMilestone get a milestone for a repository by ID and if not available by name func GetMilestone(ctx *context.APIContext) { // swagger:operation GET /repos/{owner}/{repo}/milestones/{id} issue issueGetMilestone // --- @@ -93,23 +94,18 @@ func GetMilestone(ctx *context.APIContext) { // required: true // - name: id // in: path - // description: id of the milestone - // type: integer - // format: int64 + // description: the milestone to get, identified by ID and if not available by name + // type: string // required: true // responses: // "200": // "$ref": "#/responses/Milestone" - milestone, err := models.GetMilestoneByRepoID(ctx.Repo.Repository.ID, ctx.ParamsInt64(":id")) - if err != nil { - if models.IsErrMilestoneNotExist(err) { - ctx.NotFound() - } else { - ctx.Error(http.StatusInternalServerError, "GetMilestoneByRepoID", err) - } + milestone := getMile(ctx) + if ctx.Written() { return } + ctx.JSON(http.StatusOK, convert.ToAPIMilestone(milestone)) } @@ -165,7 +161,7 @@ func CreateMilestone(ctx *context.APIContext, form api.CreateMilestoneOption) { ctx.JSON(http.StatusCreated, convert.ToAPIMilestone(milestone)) } -// EditMilestone modify a milestone for a repository +// EditMilestone modify a milestone for a repository by ID and if not available by name func EditMilestone(ctx *context.APIContext, form api.EditMilestoneOption) { // swagger:operation PATCH /repos/{owner}/{repo}/milestones/{id} issue issueEditMilestone // --- @@ -187,9 +183,8 @@ func EditMilestone(ctx *context.APIContext, form api.EditMilestoneOption) { // required: true // - name: id // in: path - // description: id of the milestone - // type: integer - // format: int64 + // description: the milestone to edit, identified by ID and if not available by name + // type: string // required: true // - name: body // in: body @@ -199,13 +194,8 @@ func EditMilestone(ctx *context.APIContext, form api.EditMilestoneOption) { // "200": // "$ref": "#/responses/Milestone" - milestone, err := models.GetMilestoneByRepoID(ctx.Repo.Repository.ID, ctx.ParamsInt64(":id")) - if err != nil { - if models.IsErrMilestoneNotExist(err) { - ctx.NotFound() - } else { - ctx.Error(http.StatusInternalServerError, "GetMilestoneByRepoID", err) - } + milestone := getMile(ctx) + if ctx.Written() { return } @@ -231,7 +221,7 @@ func EditMilestone(ctx *context.APIContext, form api.EditMilestoneOption) { ctx.JSON(http.StatusOK, convert.ToAPIMilestone(milestone)) } -// DeleteMilestone delete a milestone for a repository +// DeleteMilestone delete a milestone for a repository by ID and if not available by name func DeleteMilestone(ctx *context.APIContext) { // swagger:operation DELETE /repos/{owner}/{repo}/milestones/{id} issue issueDeleteMilestone // --- @@ -249,17 +239,49 @@ func DeleteMilestone(ctx *context.APIContext) { // required: true // - name: id // in: path - // description: id of the milestone to delete - // type: integer - // format: int64 + // description: the milestone to delete, identified by ID and if not available by name + // type: string // required: true // responses: // "204": // "$ref": "#/responses/empty" - if err := models.DeleteMilestoneByRepoID(ctx.Repo.Repository.ID, ctx.ParamsInt64(":id")); err != nil { + m := getMile(ctx) + if ctx.Written() { + return + } + + if err := models.DeleteMilestoneByRepoID(ctx.Repo.Repository.ID, m.ID); err != nil { ctx.Error(http.StatusInternalServerError, "DeleteMilestoneByRepoID", err) return } ctx.Status(http.StatusNoContent) } + +// getMile get milestone by ID and if not available by name +func getMile(ctx *context.APIContext) *models.Milestone { + mile := ctx.Params(":id") + mileID, _ := strconv.ParseInt(mile, 0, 64) + + if mileID != 0 { + milestone, err := models.GetMilestoneByRepoID(ctx.Repo.Repository.ID, mileID) + if err == nil { + return milestone + } else if !models.IsErrMilestoneNotExist(err) { + ctx.Error(http.StatusInternalServerError, "GetMilestoneByRepoID", err) + return nil + } + } + + milestone, err := models.GetMilestoneByRepoIDANDName(ctx.Repo.Repository.ID, mile) + if err != nil { + if models.IsErrMilestoneNotExist(err) { + ctx.NotFound() + return nil + } + ctx.Error(http.StatusInternalServerError, "GetMilestoneByRepoID", err) + return nil + } + + return milestone +} diff --git a/templates/swagger/v1_json.tmpl b/templates/swagger/v1_json.tmpl index 4d6333ac4e8c7..3ba4912c80e19 100644 --- a/templates/swagger/v1_json.tmpl +++ b/templates/swagger/v1_json.tmpl @@ -6320,9 +6320,8 @@ "required": true }, { - "type": "integer", - "format": "int64", - "description": "id of the milestone", + "type": "string", + "description": "the milestone to get, identified by ID and if not available by name", "name": "id", "in": "path", "required": true @@ -6356,9 +6355,8 @@ "required": true }, { - "type": "integer", - "format": "int64", - "description": "id of the milestone to delete", + "type": "string", + "description": "the milestone to delete, identified by ID and if not available by name", "name": "id", "in": "path", "required": true @@ -6398,9 +6396,8 @@ "required": true }, { - "type": "integer", - "format": "int64", - "description": "id of the milestone", + "type": "string", + "description": "the milestone to edit, identified by ID and if not available by name", "name": "id", "in": "path", "required": true From 27c93f2c32c12ab614440d8a17e9bb36a32e284a Mon Sep 17 00:00:00 2001 From: 6543 <6543@obermui.de> Date: Sun, 30 Aug 2020 14:09:41 +0200 Subject: [PATCH 2/3] add test --- integrations/api_issue_milestone_test.go | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/integrations/api_issue_milestone_test.go b/integrations/api_issue_milestone_test.go index 4cc574b9ebc39..b9942d9a3c7b2 100644 --- a/integrations/api_issue_milestone_test.go +++ b/integrations/api_issue_milestone_test.go @@ -61,6 +61,11 @@ func TestAPIIssuesMilestone(t *testing.T) { DecodeJSON(t, resp, &apiMilestones) assert.Len(t, apiMilestones, 4) + req = NewRequest(t, "GET", fmt.Sprintf("/api/v1/repos/%s/%s/milestones/%s?token=%s", owner.Name, repo.Name, apiMilestones[2].Title, token)) + resp = session.MakeRequest(t, req, http.StatusOK) + DecodeJSON(t, resp, &apiMilestone) + assert.EqualValues(t, apiMilestones[2], apiMilestone) + req = NewRequest(t, "GET", fmt.Sprintf("/api/v1/repos/%s/%s/milestones?state=%s&name=%s&token=%s", owner.Name, repo.Name, "all", "milestone2", token)) resp = session.MakeRequest(t, req, http.StatusOK) DecodeJSON(t, resp, &apiMilestones) From 0cdc46dfdfd3f93514fa88ceca7dbfb119517cf0 Mon Sep 17 00:00:00 2001 From: 6543 <6543@obermui.de> Date: Wed, 2 Sep 2020 16:25:05 +0200 Subject: [PATCH 3/3] rename --- routers/api/v1/repo/milestone.go | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/routers/api/v1/repo/milestone.go b/routers/api/v1/repo/milestone.go index 13b2413f055ca..5a250dbd808e1 100644 --- a/routers/api/v1/repo/milestone.go +++ b/routers/api/v1/repo/milestone.go @@ -101,7 +101,7 @@ func GetMilestone(ctx *context.APIContext) { // "200": // "$ref": "#/responses/Milestone" - milestone := getMile(ctx) + milestone := getMilestoneByIDOrName(ctx) if ctx.Written() { return } @@ -194,7 +194,7 @@ func EditMilestone(ctx *context.APIContext, form api.EditMilestoneOption) { // "200": // "$ref": "#/responses/Milestone" - milestone := getMile(ctx) + milestone := getMilestoneByIDOrName(ctx) if ctx.Written() { return } @@ -246,7 +246,7 @@ func DeleteMilestone(ctx *context.APIContext) { // "204": // "$ref": "#/responses/empty" - m := getMile(ctx) + m := getMilestoneByIDOrName(ctx) if ctx.Written() { return } @@ -258,8 +258,8 @@ func DeleteMilestone(ctx *context.APIContext) { ctx.Status(http.StatusNoContent) } -// getMile get milestone by ID and if not available by name -func getMile(ctx *context.APIContext) *models.Milestone { +// getMilestoneByIDOrName get milestone by ID and if not available by name +func getMilestoneByIDOrName(ctx *context.APIContext) *models.Milestone { mile := ctx.Params(":id") mileID, _ := strconv.ParseInt(mile, 0, 64)