refactor(API): refactor secret creation and update functionality (#26751)
According to the GitHub API Spec: https://docs.github.com/en/rest/actions/secrets?apiVersion=2022-11-28#create-or-update-an-organization-secret Merge the Create and Update secret into a single API. - Remove the `CreateSecretOption` struct and replace it with `CreateOrUpdateSecretOption` in `modules/structs/secret.go` - Update the `CreateOrUpdateOrgSecret` function in `routers/api/v1/org/action.go` to use `CreateOrUpdateSecretOption` instead of `UpdateSecretOption` - Remove the `CreateOrgSecret` function in `routers/api/v1/org/action.go` and replace it with `CreateOrUpdateOrgSecret` - Update the Swagger documentation in `routers/api/v1/swagger/options.go` and `templates/swagger/v1_json.tmpl` to reflect the changes in the struct names and function names Signed-off-by: Bo-Yi Wu <appleboy.tw@gmail.com>
This commit is contained in:
parent
6945918d34
commit
8cd46024fd
5 changed files with 51 additions and 155 deletions
|
@ -14,21 +14,9 @@ type Secret struct {
|
||||||
Created time.Time `json:"created_at"`
|
Created time.Time `json:"created_at"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// CreateSecretOption options when creating secret
|
// CreateOrUpdateSecretOption options when creating or updating secret
|
||||||
// swagger:model
|
// swagger:model
|
||||||
type CreateSecretOption struct {
|
type CreateOrUpdateSecretOption struct {
|
||||||
// Name of the secret to create
|
|
||||||
//
|
|
||||||
// required: true
|
|
||||||
// unique: true
|
|
||||||
Name string `json:"name" binding:"Required;AlphaDashDot;MaxSize(100)"`
|
|
||||||
// Data of the secret to create
|
|
||||||
Data string `json:"data" binding:"Required"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// UpdateSecretOption options when updating secret
|
|
||||||
// swagger:model
|
|
||||||
type UpdateSecretOption struct {
|
|
||||||
// Data of the secret to update
|
// Data of the secret to update
|
||||||
//
|
//
|
||||||
// required: true
|
// required: true
|
||||||
|
|
|
@ -1300,9 +1300,8 @@ func Routes() *web.Route {
|
||||||
})
|
})
|
||||||
m.Group("/actions/secrets", func() {
|
m.Group("/actions/secrets", func() {
|
||||||
m.Get("", reqToken(), reqOrgOwnership(), org.ListActionsSecrets)
|
m.Get("", reqToken(), reqOrgOwnership(), org.ListActionsSecrets)
|
||||||
m.Post("", reqToken(), reqOrgOwnership(), bind(api.CreateSecretOption{}), org.CreateOrgSecret)
|
|
||||||
m.Combo("/{secretname}").
|
m.Combo("/{secretname}").
|
||||||
Put(reqToken(), reqOrgOwnership(), bind(api.UpdateSecretOption{}), org.UpdateOrgSecret).
|
Put(reqToken(), reqOrgOwnership(), bind(api.CreateOrUpdateSecretOption{}), org.CreateOrUpdateOrgSecret).
|
||||||
Delete(reqToken(), reqOrgOwnership(), org.DeleteOrgSecret)
|
Delete(reqToken(), reqOrgOwnership(), org.DeleteOrgSecret)
|
||||||
})
|
})
|
||||||
m.Group("/public_members", func() {
|
m.Group("/public_members", func() {
|
||||||
|
|
|
@ -12,7 +12,6 @@ import (
|
||||||
"code.gitea.io/gitea/modules/web"
|
"code.gitea.io/gitea/modules/web"
|
||||||
"code.gitea.io/gitea/routers/api/v1/utils"
|
"code.gitea.io/gitea/routers/api/v1/utils"
|
||||||
"code.gitea.io/gitea/routers/web/shared/actions"
|
"code.gitea.io/gitea/routers/web/shared/actions"
|
||||||
"code.gitea.io/gitea/services/convert"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// ListActionsSecrets list an organization's actions secrets
|
// ListActionsSecrets list an organization's actions secrets
|
||||||
|
@ -74,55 +73,11 @@ func listActionsSecrets(ctx *context.APIContext) {
|
||||||
ctx.JSON(http.StatusOK, apiSecrets)
|
ctx.JSON(http.StatusOK, apiSecrets)
|
||||||
}
|
}
|
||||||
|
|
||||||
// CreateOrgSecret create one secret of the organization
|
// create or update one secret of the organization
|
||||||
func CreateOrgSecret(ctx *context.APIContext) {
|
func CreateOrUpdateOrgSecret(ctx *context.APIContext) {
|
||||||
// swagger:operation POST /orgs/{org}/actions/secrets organization createOrgSecret
|
|
||||||
// ---
|
|
||||||
// summary: Create a secret in an organization
|
|
||||||
// consumes:
|
|
||||||
// - application/json
|
|
||||||
// produces:
|
|
||||||
// - application/json
|
|
||||||
// parameters:
|
|
||||||
// - name: org
|
|
||||||
// in: path
|
|
||||||
// description: name of organization
|
|
||||||
// type: string
|
|
||||||
// required: true
|
|
||||||
// - name: body
|
|
||||||
// in: body
|
|
||||||
// schema:
|
|
||||||
// "$ref": "#/definitions/CreateSecretOption"
|
|
||||||
// responses:
|
|
||||||
// "201":
|
|
||||||
// "$ref": "#/responses/Secret"
|
|
||||||
// "400":
|
|
||||||
// "$ref": "#/responses/error"
|
|
||||||
// "404":
|
|
||||||
// "$ref": "#/responses/notFound"
|
|
||||||
// "403":
|
|
||||||
// "$ref": "#/responses/forbidden"
|
|
||||||
opt := web.GetForm(ctx).(*api.CreateSecretOption)
|
|
||||||
if err := actions.NameRegexMatch(opt.Name); err != nil {
|
|
||||||
ctx.Error(http.StatusBadRequest, "CreateOrgSecret", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
s, err := secret_model.InsertEncryptedSecret(
|
|
||||||
ctx, ctx.Org.Organization.ID, 0, opt.Name, actions.ReserveLineBreakForTextarea(opt.Data),
|
|
||||||
)
|
|
||||||
if err != nil {
|
|
||||||
ctx.Error(http.StatusInternalServerError, "InsertEncryptedSecret", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.JSON(http.StatusCreated, convert.ToSecret(s))
|
|
||||||
}
|
|
||||||
|
|
||||||
// UpdateOrgSecret update one secret of the organization
|
|
||||||
func UpdateOrgSecret(ctx *context.APIContext) {
|
|
||||||
// swagger:operation PUT /orgs/{org}/actions/secrets/{secretname} organization updateOrgSecret
|
// swagger:operation PUT /orgs/{org}/actions/secrets/{secretname} organization updateOrgSecret
|
||||||
// ---
|
// ---
|
||||||
// summary: Update a secret value in an organization
|
// summary: Create or Update a secret value in an organization
|
||||||
// consumes:
|
// consumes:
|
||||||
// - application/json
|
// - application/json
|
||||||
// produces:
|
// produces:
|
||||||
|
@ -141,19 +96,34 @@ func UpdateOrgSecret(ctx *context.APIContext) {
|
||||||
// - name: body
|
// - name: body
|
||||||
// in: body
|
// in: body
|
||||||
// schema:
|
// schema:
|
||||||
// "$ref": "#/definitions/UpdateSecretOption"
|
// "$ref": "#/definitions/CreateOrUpdateSecretOption"
|
||||||
// responses:
|
// responses:
|
||||||
|
// "201":
|
||||||
|
// description: response when creating a secret
|
||||||
// "204":
|
// "204":
|
||||||
// description: update one secret of the organization
|
// description: response when updating a secret
|
||||||
|
// "400":
|
||||||
|
// "$ref": "#/responses/error"
|
||||||
// "403":
|
// "403":
|
||||||
// "$ref": "#/responses/forbidden"
|
// "$ref": "#/responses/forbidden"
|
||||||
secretName := ctx.Params(":secretname")
|
secretName := ctx.Params(":secretname")
|
||||||
opt := web.GetForm(ctx).(*api.UpdateSecretOption)
|
if err := actions.NameRegexMatch(secretName); err != nil {
|
||||||
|
ctx.Error(http.StatusBadRequest, "CreateOrUpdateOrgSecret", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
opt := web.GetForm(ctx).(*api.CreateOrUpdateSecretOption)
|
||||||
err := secret_model.UpdateSecret(
|
err := secret_model.UpdateSecret(
|
||||||
ctx, ctx.Org.Organization.ID, 0, secretName, opt.Data,
|
ctx, ctx.Org.Organization.ID, 0, secretName, opt.Data,
|
||||||
)
|
)
|
||||||
if secret_model.IsErrSecretNotFound(err) {
|
if secret_model.IsErrSecretNotFound(err) {
|
||||||
ctx.NotFound(err)
|
_, err := secret_model.InsertEncryptedSecret(
|
||||||
|
ctx, ctx.Org.Organization.ID, 0, secretName, actions.ReserveLineBreakForTextarea(opt.Data),
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
ctx.Error(http.StatusInternalServerError, "InsertEncryptedSecret", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Status(http.StatusCreated)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -189,8 +189,5 @@ type swaggerParameterBodies struct {
|
||||||
UpdateRepoAvatarOptions api.UpdateRepoAvatarOption
|
UpdateRepoAvatarOptions api.UpdateRepoAvatarOption
|
||||||
|
|
||||||
// in:body
|
// in:body
|
||||||
CreateSecretOption api.CreateSecretOption
|
CreateOrUpdateSecretOption api.CreateOrUpdateSecretOption
|
||||||
|
|
||||||
// in:body
|
|
||||||
UpdateSecretOption api.UpdateSecretOption
|
|
||||||
}
|
}
|
||||||
|
|
108
templates/swagger/v1_json.tmpl
generated
108
templates/swagger/v1_json.tmpl
generated
|
@ -1586,49 +1586,6 @@
|
||||||
"$ref": "#/responses/SecretList"
|
"$ref": "#/responses/SecretList"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
|
||||||
"post": {
|
|
||||||
"consumes": [
|
|
||||||
"application/json"
|
|
||||||
],
|
|
||||||
"produces": [
|
|
||||||
"application/json"
|
|
||||||
],
|
|
||||||
"tags": [
|
|
||||||
"organization"
|
|
||||||
],
|
|
||||||
"summary": "Create a secret in an organization",
|
|
||||||
"operationId": "createOrgSecret",
|
|
||||||
"parameters": [
|
|
||||||
{
|
|
||||||
"type": "string",
|
|
||||||
"description": "name of organization",
|
|
||||||
"name": "org",
|
|
||||||
"in": "path",
|
|
||||||
"required": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "body",
|
|
||||||
"in": "body",
|
|
||||||
"schema": {
|
|
||||||
"$ref": "#/definitions/CreateSecretOption"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"responses": {
|
|
||||||
"201": {
|
|
||||||
"$ref": "#/responses/Secret"
|
|
||||||
},
|
|
||||||
"400": {
|
|
||||||
"$ref": "#/responses/error"
|
|
||||||
},
|
|
||||||
"403": {
|
|
||||||
"$ref": "#/responses/forbidden"
|
|
||||||
},
|
|
||||||
"404": {
|
|
||||||
"$ref": "#/responses/notFound"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"/orgs/{org}/actions/secrets/{secretname}": {
|
"/orgs/{org}/actions/secrets/{secretname}": {
|
||||||
|
@ -1642,7 +1599,7 @@
|
||||||
"tags": [
|
"tags": [
|
||||||
"organization"
|
"organization"
|
||||||
],
|
],
|
||||||
"summary": "Update a secret value in an organization",
|
"summary": "Create or Update a secret value in an organization",
|
||||||
"operationId": "updateOrgSecret",
|
"operationId": "updateOrgSecret",
|
||||||
"parameters": [
|
"parameters": [
|
||||||
{
|
{
|
||||||
|
@ -1663,13 +1620,19 @@
|
||||||
"name": "body",
|
"name": "body",
|
||||||
"in": "body",
|
"in": "body",
|
||||||
"schema": {
|
"schema": {
|
||||||
"$ref": "#/definitions/UpdateSecretOption"
|
"$ref": "#/definitions/CreateOrUpdateSecretOption"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"responses": {
|
"responses": {
|
||||||
|
"201": {
|
||||||
|
"description": "response when creating a secret"
|
||||||
|
},
|
||||||
"204": {
|
"204": {
|
||||||
"description": "update one secret of the organization"
|
"description": "response when updating a secret"
|
||||||
|
},
|
||||||
|
"400": {
|
||||||
|
"$ref": "#/responses/error"
|
||||||
},
|
},
|
||||||
"403": {
|
"403": {
|
||||||
"$ref": "#/responses/forbidden"
|
"$ref": "#/responses/forbidden"
|
||||||
|
@ -17283,6 +17246,21 @@
|
||||||
},
|
},
|
||||||
"x-go-package": "code.gitea.io/gitea/modules/structs"
|
"x-go-package": "code.gitea.io/gitea/modules/structs"
|
||||||
},
|
},
|
||||||
|
"CreateOrUpdateSecretOption": {
|
||||||
|
"description": "CreateOrUpdateSecretOption options when creating or updating secret",
|
||||||
|
"type": "object",
|
||||||
|
"required": [
|
||||||
|
"data"
|
||||||
|
],
|
||||||
|
"properties": {
|
||||||
|
"data": {
|
||||||
|
"description": "Data of the secret to update",
|
||||||
|
"type": "string",
|
||||||
|
"x-go-name": "Data"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"x-go-package": "code.gitea.io/gitea/modules/structs"
|
||||||
|
},
|
||||||
"CreateOrgOption": {
|
"CreateOrgOption": {
|
||||||
"description": "CreateOrgOption options for creating an organization",
|
"description": "CreateOrgOption options for creating an organization",
|
||||||
"type": "object",
|
"type": "object",
|
||||||
|
@ -17569,27 +17547,6 @@
|
||||||
},
|
},
|
||||||
"x-go-package": "code.gitea.io/gitea/modules/structs"
|
"x-go-package": "code.gitea.io/gitea/modules/structs"
|
||||||
},
|
},
|
||||||
"CreateSecretOption": {
|
|
||||||
"description": "CreateSecretOption options when creating secret",
|
|
||||||
"type": "object",
|
|
||||||
"required": [
|
|
||||||
"name"
|
|
||||||
],
|
|
||||||
"properties": {
|
|
||||||
"data": {
|
|
||||||
"description": "Data of the secret to create",
|
|
||||||
"type": "string",
|
|
||||||
"x-go-name": "Data"
|
|
||||||
},
|
|
||||||
"name": {
|
|
||||||
"description": "Name of the secret to create",
|
|
||||||
"type": "string",
|
|
||||||
"uniqueItems": true,
|
|
||||||
"x-go-name": "Name"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"x-go-package": "code.gitea.io/gitea/modules/structs"
|
|
||||||
},
|
|
||||||
"CreateStatusOption": {
|
"CreateStatusOption": {
|
||||||
"description": "CreateStatusOption holds the information needed to create a new CommitStatus for a Commit",
|
"description": "CreateStatusOption holds the information needed to create a new CommitStatus for a Commit",
|
||||||
"type": "object",
|
"type": "object",
|
||||||
|
@ -21978,21 +21935,6 @@
|
||||||
},
|
},
|
||||||
"x-go-package": "code.gitea.io/gitea/modules/structs"
|
"x-go-package": "code.gitea.io/gitea/modules/structs"
|
||||||
},
|
},
|
||||||
"UpdateSecretOption": {
|
|
||||||
"description": "UpdateSecretOption options when updating secret",
|
|
||||||
"type": "object",
|
|
||||||
"required": [
|
|
||||||
"data"
|
|
||||||
],
|
|
||||||
"properties": {
|
|
||||||
"data": {
|
|
||||||
"description": "Data of the secret to update",
|
|
||||||
"type": "string",
|
|
||||||
"x-go-name": "Data"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"x-go-package": "code.gitea.io/gitea/modules/structs"
|
|
||||||
},
|
|
||||||
"UpdateUserAvatarOption": {
|
"UpdateUserAvatarOption": {
|
||||||
"description": "UpdateUserAvatarUserOption options when updating the user avatar",
|
"description": "UpdateUserAvatarUserOption options when updating the user avatar",
|
||||||
"type": "object",
|
"type": "object",
|
||||||
|
@ -23309,7 +23251,7 @@
|
||||||
"parameterBodies": {
|
"parameterBodies": {
|
||||||
"description": "parameterBodies",
|
"description": "parameterBodies",
|
||||||
"schema": {
|
"schema": {
|
||||||
"$ref": "#/definitions/UpdateSecretOption"
|
"$ref": "#/definitions/CreateOrUpdateSecretOption"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"redirect": {
|
"redirect": {
|
||||||
|
|
Loading…
Reference in a new issue