2020-07-06 23:18:39 +00:00
package customtemplates
import (
2020-07-08 00:25:37 +00:00
"errors"
2023-06-22 14:08:47 +00:00
"fmt"
2020-07-06 23:18:39 +00:00
"net/http"
"strconv"
portainer "github.com/portainer/portainer/api"
2023-04-04 00:44:42 +00:00
"github.com/portainer/portainer/api/filesystem"
2023-06-22 14:08:47 +00:00
"github.com/portainer/portainer/api/git"
2023-04-04 00:44:42 +00:00
gittypes "github.com/portainer/portainer/api/git/types"
2020-07-08 00:25:37 +00:00
httperrors "github.com/portainer/portainer/api/http/errors"
2020-07-06 23:18:39 +00:00
"github.com/portainer/portainer/api/http/security"
2023-09-01 22:27:02 +00:00
httperror "github.com/portainer/portainer/pkg/libhttp/error"
"github.com/portainer/portainer/pkg/libhttp/request"
"github.com/portainer/portainer/pkg/libhttp/response"
2023-05-26 03:29:09 +00:00
"github.com/asaskevich/govalidator"
2020-07-06 23:18:39 +00:00
)
type customTemplateUpdatePayload struct {
2021-02-23 03:21:39 +00:00
// URL of the template's logo
2023-05-30 03:48:21 +00:00
Logo string ` example:"https://portainer.io/img/logo.svg" `
2021-02-23 03:21:39 +00:00
// Title of the template
Title string ` example:"Nginx" validate:"required" `
// Description of the template
Description string ` example:"High performance web server" validate:"required" `
// A note that will be displayed in the UI. Supports HTML content
Note string ` example:"This is my <b>custom</b> template" `
// Platform associated to the template.
// Valid values are: 1 - 'linux', 2 - 'windows'
2021-09-02 05:28:51 +00:00
// Required for Docker stacks
Platform portainer . CustomTemplatePlatform ` example:"1" enums:"1,2" `
// Type of created stack (1 - swarm, 2 - compose, 3 - kubernetes)
Type portainer . StackType ` example:"1" enums:"1,2,3" validate:"required" `
2023-04-04 00:44:42 +00:00
// URL of a Git repository hosting the Stack file
RepositoryURL string ` example:"https://github.com/openfaas/faas" validate:"required" `
// Reference name of a Git repository hosting the Stack file
RepositoryReferenceName string ` example:"refs/heads/master" `
// Use basic authentication to clone the Git repository
RepositoryAuthentication bool ` example:"true" `
// Username used in basic authentication. Required when RepositoryAuthentication is true
// and RepositoryGitCredentialID is 0
RepositoryUsername string ` example:"myGitUsername" `
// Password used in basic authentication. Required when RepositoryAuthentication is true
// and RepositoryGitCredentialID is 0
RepositoryPassword string ` example:"myGitPassword" `
// GitCredentialID used to identify the bound git credential. Required when RepositoryAuthentication
// is true and RepositoryUsername/RepositoryPassword are not provided
RepositoryGitCredentialID int ` example:"0" `
// Path to the Stack file inside the Git repository
ComposeFilePathInRepository string ` example:"docker-compose.yml" default:"docker-compose.yml" `
2021-02-23 03:21:39 +00:00
// Content of stack file
FileContent string ` validate:"required" `
2022-05-31 10:00:47 +00:00
// Definitions of variables in the stack file
Variables [ ] portainer . CustomTemplateVariableDefinition
2023-05-26 03:29:09 +00:00
// TLSSkipVerify skips SSL verification when cloning the Git repository
TLSSkipVerify bool ` example:"false" `
2023-04-04 00:44:42 +00:00
// IsComposeFormat indicates if the Kubernetes template is created from a Docker Compose file
IsComposeFormat bool ` example:"false" `
2023-11-15 08:45:07 +00:00
// EdgeTemplate indicates if this template purpose for Edge Stack
EdgeTemplate bool ` example:"false" `
2020-07-06 23:18:39 +00:00
}
func ( payload * customTemplateUpdatePayload ) Validate ( r * http . Request ) error {
if govalidator . IsNull ( payload . Title ) {
2020-07-08 00:25:37 +00:00
return errors . New ( "Invalid custom template title" )
2020-07-06 23:18:39 +00:00
}
2023-04-04 00:44:42 +00:00
if govalidator . IsNull ( payload . FileContent ) && govalidator . IsNull ( payload . RepositoryURL ) {
return errors . New ( "Either file content or git repository url need to be provided" )
2020-07-06 23:18:39 +00:00
}
2021-09-02 05:28:51 +00:00
if payload . Type != portainer . KubernetesStack && payload . Platform != portainer . CustomTemplatePlatformLinux && payload . Platform != portainer . CustomTemplatePlatformWindows {
2020-07-08 00:25:37 +00:00
return errors . New ( "Invalid custom template platform" )
2020-07-06 23:18:39 +00:00
}
2021-09-02 05:28:51 +00:00
if payload . Type != portainer . KubernetesStack && payload . Type != portainer . DockerSwarmStack && payload . Type != portainer . DockerComposeStack {
2020-07-08 00:25:37 +00:00
return errors . New ( "Invalid custom template type" )
2020-07-06 23:18:39 +00:00
}
if govalidator . IsNull ( payload . Description ) {
2020-07-08 00:25:37 +00:00
return errors . New ( "Invalid custom template description" )
2020-07-06 23:18:39 +00:00
}
2021-09-29 19:47:39 +00:00
if ! isValidNote ( payload . Note ) {
return errors . New ( "Invalid note. <img> tag is not supported" )
}
2022-05-31 10:00:47 +00:00
2023-04-04 00:44:42 +00:00
if payload . RepositoryAuthentication && ( govalidator . IsNull ( payload . RepositoryUsername ) || govalidator . IsNull ( payload . RepositoryPassword ) ) {
return errors . New ( "Invalid repository credentials. Username and password must be specified when authentication is enabled" )
}
if govalidator . IsNull ( payload . ComposeFilePathInRepository ) {
payload . ComposeFilePathInRepository = filesystem . ComposeFileDefaultName
}
err := validateVariablesDefinitions ( payload . Variables )
if err != nil {
return err
}
return nil
2020-07-06 23:18:39 +00:00
}
2021-02-23 03:21:39 +00:00
// @id CustomTemplateUpdate
// @summary Update a template
// @description Update a template.
// @description **Access policy**: authenticated
// @tags custom_templates
2021-11-30 02:31:16 +00:00
// @security ApiKeyAuth
2021-02-23 03:21:39 +00:00
// @security jwt
// @accept json
// @produce json
// @param id path int true "Template identifier"
// @param body body customTemplateUpdatePayload true "Template details"
// @success 200 {object} portainer.CustomTemplate "Success"
// @failure 400 "Invalid request"
// @failure 403 "Permission denied to access template"
// @failure 404 "Template not found"
// @failure 500 "Server error"
// @router /custom_templates/{id} [put]
2020-07-06 23:18:39 +00:00
func ( handler * Handler ) customTemplateUpdate ( w http . ResponseWriter , r * http . Request ) * httperror . HandlerError {
customTemplateID , err := request . RetrieveNumericRouteVariableValue ( r , "id" )
if err != nil {
2022-09-14 23:42:39 +00:00
return httperror . BadRequest ( "Invalid Custom template identifier route variable" , err )
2020-07-06 23:18:39 +00:00
}
var payload customTemplateUpdatePayload
err = request . DecodeAndValidateJSONPayload ( r , & payload )
if err != nil {
2022-09-14 23:42:39 +00:00
return httperror . BadRequest ( "Invalid request payload" , err )
2020-07-06 23:18:39 +00:00
}
2023-06-22 21:28:07 +00:00
customTemplates , err := handler . DataStore . CustomTemplate ( ) . ReadAll ( )
2020-08-20 22:17:30 +00:00
if err != nil {
2022-09-14 23:42:39 +00:00
return httperror . InternalServerError ( "Unable to retrieve custom templates from the database" , err )
2020-08-20 22:17:30 +00:00
}
for _ , existingTemplate := range customTemplates {
if existingTemplate . ID != portainer . CustomTemplateID ( customTemplateID ) && existingTemplate . Title == payload . Title {
2022-09-14 23:42:39 +00:00
return httperror . InternalServerError ( "Template name must be unique" , errors . New ( "Template name must be unique" ) )
2020-08-20 22:17:30 +00:00
}
}
2023-06-22 21:28:07 +00:00
customTemplate , err := handler . DataStore . CustomTemplate ( ) . Read ( portainer . CustomTemplateID ( customTemplateID ) )
chore(store) EE-1981: Refactor/store/error checking, and other refactoring (#6173)
* use the Store interface IsErrObjectNotFound() to avoid revealing internal errors
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* what happens when you extract the datastore interfaces into their own package
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* Start renaming Storage methods
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* extract the boltdb specific code from the Portainer storage code (example, the others need the same)
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* more extract bolt.Tx from datastore code
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* minimise imports by putting moving the struct definition into the file that needs the Service imports
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* more extraction of boltdb.Tx
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* extract the use of bucket.SetSequence
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* almost done - just endpoint.Synchonise :/
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* so, endpoint.Synchonize looks hard, but i can't find where we use it, so 'delete first refactoring'
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* fix test compile errors
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* test compile fixes after rebase
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* fix a mis-remembering I had wrt deserialisation - last time i used AnyData - jsoniter's bindTo looks interesting for the same reason
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* set us up to make the connection an interface
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* make the db connection a datastore interface, and separate out our datastore services from the bolt ones
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* rename methods to something less oltdb internals specific
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* these errors are not boltdb secific
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* start using the db-backend factory method too
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* export boltdb raw in case we can't export from the service layer
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* add a raw export from boltdb to yaml for broken db's, and an export services to yaml in backup
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* add the version info by hand for now
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* actually, the export from services can be fully typed - its the import that needs to do more work
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* redo raw export, and make import capable of using it
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* add DockerHub
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* migration from anything older than v1.21.0 has been broken for quite a while, deleting the un-tested code
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* fix go test ./... again
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* my goland wasn't setup to gofmt
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* move the two extremely dubious migration tests down into store, so they can use the test store code
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* the migrator is now free of boltdb
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* reverse goland overzealous replcement of internal with boltdb
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* more undo over-zealous goland internal->boltdb
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* yay, now bolt is only mentioned inside the api/database/ dir
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* and this might be the last of the boltdb references?
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* add todo
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* extract the store code into a separate module too
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* don't need the fileService in boltdb anymore
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* use IsErrObjectNotFound()
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* use a string to select what database backend we use
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* make isNew store an ephemeral bool that doesn't stay true after we've initialised it
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* move the import.json wip to a separate file so its more obvious - we'll be using it for testing, emergency fixups, and in the next part of the store work, when we improve migrations and data model lifecycles
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* undo vscode formatting html
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* fix app templates symbol (#6221)
* feat(webhook) EE-2125 send registry auth haeder when update swarms service via webhook (#6220)
* feat(webhook) EE-2125 add some helpers to registry utils
* feat(webhook) EE-2125 persist registryID when creating a webhook
* feat(webhook) EE-2125 send registry auth header when executing a webhook
* feat(webhook) EE-2125 send registryID to backend when creating a service with webhook
* feat(webhook) EE-2125 use the initial registry ID to create webhook on editing service screen
* feat(webhook) EE-2125 update webhook when update registry
* feat(webhook) EE-2125 add endpoint of update webhook
* feat(webhook) EE-2125 code cleanup
* feat(webhook) EE-2125 fix a typo
* feat(webhook) EE-2125 fix circle import issue with unit test
Co-authored-by: Simon Meng <simon.meng@portainer.io>
* fix(kubeconfig): show kubeconfig download button for non admin users [EE-2123] (#6204)
Co-authored-by: Simon Meng <simon.meng@portainer.io>
* fix data-cy for k8s cluster menu (#6226)
LGTM
* feat(stack): make stack created from app template editable EE-1941 (#6104)
feat(stack): make stack from app template editable
* fix(container):disable Duplicate/Edit button when the container is portainer (#6223)
* fix/ee-1909/show-pull-image-error (#6195)
Co-authored-by: sunportainer <ericsun@SG1.local>
* feat(cy): add data-cy to helm install button (#6241)
* feat(cy): add data-cy to add registry button (#6242)
* refactor(app): convert root folder files to es6 (#4159)
* refactor(app): duplicate constants as es6 exports (#4158)
* fix(docker): provide workaround to save network name variable (#6080)
* fix/EE-1862/unable-to-stop-or-remove-stack workaround for var without default value in yaml file
* fix/EE-1862/unable-to-stop-or-remove-stack check yaml file
* fixed func and var names
* wrapper error and used bool for stringset
* UT case for createNetworkEnvFile
* UT case for %s=%s
* powerful StringSet
* wrapper error for extract network name
* wrapper all the return err
* store more env
* put to env file
* make default value None
* feat: gzip static resources (#6258)
* fix(ssl)//handle --sslcert and --sslkey ee-2106 (#6203)
* fix/ee-2106/handle-sslcert-sslkey
Co-authored-by: sunportainer <ericsun@SG1.local>
* fix(server):support disable https only ee-2068 (#6232)
* fix/ee-2068/disable-forcely-https
* feat(store): implement store tests EE-2112 (#6224)
* add store tests
* add some more tests
* Update missing helm user repo methods
* remove redundant comments
* add webhook export
* update webhooks
* use the Store interface IsErrObjectNotFound() to avoid revealing internal errors
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* what happens when you extract the datastore interfaces into their own package
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* Start renaming Storage methods
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* extract the boltdb specific code from the Portainer storage code (example, the others need the same)
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* more extract bolt.Tx from datastore code
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* minimise imports by putting moving the struct definition into the file that needs the Service imports
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* more extraction of boltdb.Tx
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* extract the use of bucket.SetSequence
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* almost done - just endpoint.Synchonise :/
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* so, endpoint.Synchonize looks hard, but i can't find where we use it, so 'delete first refactoring'
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* fix test compile errors
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* test compile fixes after rebase
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* fix a mis-remembering I had wrt deserialisation - last time i used AnyData - jsoniter's bindTo looks interesting for the same reason
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* set us up to make the connection an interface
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* make the db connection a datastore interface, and separate out our datastore services from the bolt ones
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* rename methods to something less oltdb internals specific
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* these errors are not boltdb secific
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* start using the db-backend factory method too
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* export boltdb raw in case we can't export from the service layer
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* add a raw export from boltdb to yaml for broken db's, and an export services to yaml in backup
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* add the version info by hand for now
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* actually, the export from services can be fully typed - its the import that needs to do more work
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* redo raw export, and make import capable of using it
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* add DockerHub
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* migration from anything older than v1.21.0 has been broken for quite a while, deleting the un-tested code
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* fix go test ./... again
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* my goland wasn't setup to gofmt
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* move the two extremely dubious migration tests down into store, so they can use the test store code
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* the migrator is now free of boltdb
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* reverse goland overzealous replcement of internal with boltdb
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* more undo over-zealous goland internal->boltdb
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* yay, now bolt is only mentioned inside the api/database/ dir
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* and this might be the last of the boltdb references?
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* add todo
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* extract the store code into a separate module too
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* don't need the fileService in boltdb anymore
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* use IsErrObjectNotFound()
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* use a string to select what database backend we use
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* make isNew store an ephemeral bool that doesn't stay true after we've initialised it
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* move the import.json wip to a separate file so its more obvious - we'll be using it for testing, emergency fixups, and in the next part of the store work, when we improve migrations and data model lifecycles
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* undo vscode formatting html
Signed-off-by: Sven Dowideit <sven.dowideit@portainer.io>
* Update missing helm user repo methods
* feat(store): implement store tests EE-2112 (#6224)
* add store tests
* add some more tests
* remove redundant comments
* add webhook export
* update webhooks
* fix build issues after rebasing
* move migratorparams
* remove unneeded integer type conversions
* disable the db import/export for now
Co-authored-by: Richard Wei <54336863+WaysonWei@users.noreply.github.com>
Co-authored-by: cong meng <mcpacino@gmail.com>
Co-authored-by: Simon Meng <simon.meng@portainer.io>
Co-authored-by: Marcelo Rydel <marcelorydel26@gmail.com>
Co-authored-by: Hao Zhang <hao.zhang@portainer.io>
Co-authored-by: sunportainer <93502624+sunportainer@users.noreply.github.com>
Co-authored-by: sunportainer <ericsun@SG1.local>
Co-authored-by: wheresolivia <78844659+wheresolivia@users.noreply.github.com>
Co-authored-by: Chaim Lev-Ari <chiptus@users.noreply.github.com>
Co-authored-by: Chao Geng <93526589+chaogeng77977@users.noreply.github.com>
Co-authored-by: Dmitry Salakhov <to@dimasalakhov.com>
Co-authored-by: Matt Hook <hookenz@gmail.com>
2021-12-15 02:26:09 +00:00
if handler . DataStore . IsErrObjectNotFound ( err ) {
2022-09-14 23:42:39 +00:00
return httperror . NotFound ( "Unable to find a custom template with the specified identifier inside the database" , err )
2020-07-06 23:18:39 +00:00
} else if err != nil {
2022-09-14 23:42:39 +00:00
return httperror . InternalServerError ( "Unable to find a custom template with the specified identifier inside the database" , err )
2020-07-06 23:18:39 +00:00
}
securityContext , err := security . RetrieveRestrictedRequestContext ( r )
if err != nil {
2022-09-14 23:42:39 +00:00
return httperror . InternalServerError ( "Unable to retrieve info from request context" , err )
2020-07-06 23:18:39 +00:00
}
access := userCanEditTemplate ( customTemplate , securityContext )
if ! access {
2022-09-14 23:42:39 +00:00
return httperror . Forbidden ( "Access denied to resource" , httperrors . ErrResourceAccessDenied )
2020-07-06 23:18:39 +00:00
}
customTemplate . Title = payload . Title
customTemplate . Logo = payload . Logo
customTemplate . Description = payload . Description
customTemplate . Note = payload . Note
customTemplate . Platform = payload . Platform
customTemplate . Type = payload . Type
2022-05-31 10:00:47 +00:00
customTemplate . Variables = payload . Variables
2023-04-04 00:44:42 +00:00
customTemplate . IsComposeFormat = payload . IsComposeFormat
2023-11-15 08:45:07 +00:00
customTemplate . EdgeTemplate = payload . EdgeTemplate
2023-04-04 00:44:42 +00:00
if payload . RepositoryURL != "" {
if ! govalidator . IsURL ( payload . RepositoryURL ) {
return httperror . BadRequest ( "Invalid repository URL. Must correspond to a valid URL format" , err )
}
gitConfig := & gittypes . RepoConfig {
URL : payload . RepositoryURL ,
ReferenceName : payload . RepositoryReferenceName ,
ConfigFilePath : payload . ComposeFilePathInRepository ,
2023-05-26 03:29:09 +00:00
TLSSkipVerify : payload . TLSSkipVerify ,
2023-04-04 00:44:42 +00:00
}
2023-06-22 14:08:47 +00:00
repositoryUsername := ""
repositoryPassword := ""
2023-04-04 00:44:42 +00:00
if payload . RepositoryAuthentication {
2023-06-22 14:08:47 +00:00
repositoryUsername = payload . RepositoryUsername
repositoryPassword = payload . RepositoryPassword
2023-04-04 00:44:42 +00:00
gitConfig . Authentication = & gittypes . GitAuthentication {
Username : payload . RepositoryUsername ,
Password : payload . RepositoryPassword ,
}
}
2023-06-22 14:08:47 +00:00
cleanBackup , err := git . CloneWithBackup ( handler . GitService , handler . FileService , git . CloneOptions {
ProjectPath : customTemplate . ProjectPath ,
URL : gitConfig . URL ,
ReferenceName : gitConfig . ReferenceName ,
Username : repositoryUsername ,
Password : repositoryPassword ,
TLSSkipVerify : gitConfig . TLSSkipVerify ,
2023-04-04 00:44:42 +00:00
} )
if err != nil {
2023-06-22 14:08:47 +00:00
return httperror . InternalServerError ( "Unable to clone git repository directory" , err )
}
defer cleanBackup ( )
commitHash , err := handler . GitService . LatestCommitID ( gitConfig . URL , gitConfig . ReferenceName , repositoryUsername , repositoryPassword , gitConfig . TLSSkipVerify )
if err != nil {
return httperror . InternalServerError ( "Unable get latest commit id" , fmt . Errorf ( "failed to fetch latest commit id of the template %v: %w" , customTemplate . ID , err ) )
2023-04-04 00:44:42 +00:00
}
gitConfig . ConfigHash = commitHash
customTemplate . GitConfig = gitConfig
} else {
templateFolder := strconv . Itoa ( customTemplateID )
_ , err = handler . FileService . StoreCustomTemplateFileFromBytes ( templateFolder , customTemplate . EntryPoint , [ ] byte ( payload . FileContent ) )
if err != nil {
return httperror . InternalServerError ( "Unable to persist updated custom template file on disk" , err )
}
}
2020-07-06 23:18:39 +00:00
2023-06-22 21:28:07 +00:00
err = handler . DataStore . CustomTemplate ( ) . Update ( customTemplate . ID , customTemplate )
2020-07-06 23:18:39 +00:00
if err != nil {
2022-09-14 23:42:39 +00:00
return httperror . InternalServerError ( "Unable to persist custom template changes inside the database" , err )
2020-07-06 23:18:39 +00:00
}
return response . JSON ( w , customTemplate )
}