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
|
|
|
package datastore
|
2021-09-27 00:52:50 +00:00
|
|
|
|
|
|
|
import (
|
2022-04-14 04:25:13 +00:00
|
|
|
"bytes"
|
2023-12-03 20:12:41 +00:00
|
|
|
"encoding/json"
|
2021-09-27 00:52:50 +00:00
|
|
|
"fmt"
|
2022-04-14 04:25:13 +00:00
|
|
|
"io"
|
|
|
|
"os"
|
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
|
|
|
"path/filepath"
|
2021-09-27 00:52:50 +00:00
|
|
|
"testing"
|
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
"github.com/Masterminds/semver"
|
2023-11-27 22:12:17 +00:00
|
|
|
portainer "github.com/portainer/portainer/api"
|
2022-04-14 04:25:13 +00:00
|
|
|
"github.com/portainer/portainer/api/database/boltdb"
|
2023-10-24 16:55:11 +00:00
|
|
|
"github.com/portainer/portainer/api/database/models"
|
2023-12-03 20:12:41 +00:00
|
|
|
"github.com/portainer/portainer/api/datastore/migrator"
|
2022-09-16 16:18:44 +00:00
|
|
|
|
|
|
|
"github.com/google/go-cmp/cmp"
|
|
|
|
"github.com/rs/zerolog/log"
|
2021-09-27 00:52:50 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestMigrateData(t *testing.T) {
|
2023-12-03 20:12:41 +00:00
|
|
|
tests := []struct {
|
2022-11-18 00:18:09 +00:00
|
|
|
testName string
|
|
|
|
srcPath string
|
|
|
|
wantPath string
|
|
|
|
overrideInstanceId bool
|
2022-04-14 04:25:13 +00:00
|
|
|
}{
|
|
|
|
{
|
2022-11-18 00:18:09 +00:00
|
|
|
testName: "migrate version 24 to latest",
|
|
|
|
srcPath: "test_data/input_24.json",
|
|
|
|
wantPath: "test_data/output_24_to_latest.json",
|
|
|
|
overrideInstanceId: true,
|
2022-04-14 04:25:13 +00:00
|
|
|
},
|
|
|
|
}
|
2023-12-03 20:12:41 +00:00
|
|
|
for _, test := range tests {
|
2022-04-14 04:25:13 +00:00
|
|
|
t.Run(test.testName, func(t *testing.T) {
|
2022-11-18 00:18:09 +00:00
|
|
|
err := migrateDBTestHelper(t, test.srcPath, test.wantPath, test.overrideInstanceId)
|
2022-04-14 04:25:13 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf(
|
|
|
|
"Failed migrating mock database %v: %v",
|
|
|
|
test.srcPath,
|
|
|
|
err,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-11-27 22:12:17 +00:00
|
|
|
t.Run("MigrateData for New Store & Re-Open Check", func(t *testing.T) {
|
|
|
|
newStore, store := MustNewTestStore(t, true, false)
|
|
|
|
if !newStore {
|
|
|
|
t.Error("Expect a new DB")
|
|
|
|
}
|
2021-10-01 07:35:43 +00:00
|
|
|
|
2023-11-27 22:12:17 +00:00
|
|
|
testVersion(store, portainer.APIVersion, t)
|
|
|
|
store.Close()
|
2021-10-01 07:35:43 +00:00
|
|
|
|
2023-11-27 22:12:17 +00:00
|
|
|
newStore, _ = store.Open()
|
|
|
|
if newStore {
|
|
|
|
t.Error("Expect store to NOT be new DB")
|
|
|
|
}
|
|
|
|
})
|
2021-10-01 07:35:43 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
t.Run("MigrateData should create backup file upon update", func(t *testing.T) {
|
|
|
|
_, store := MustNewTestStore(t, true, false)
|
|
|
|
store.VersionService.UpdateVersion(&models.Version{SchemaVersion: "1.0", Edition: int(portainer.PortainerCE)})
|
2023-11-27 22:12:17 +00:00
|
|
|
store.MigrateData()
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
backupfilename := store.backupFilename()
|
|
|
|
if exists, _ := store.fileService.FileExists(backupfilename); !exists {
|
|
|
|
t.Errorf("Expect backup file to be created %s", backupfilename)
|
|
|
|
}
|
2023-11-27 22:12:17 +00:00
|
|
|
})
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
t.Run("MigrateData should recover and restore backup during migration critical failure", func(t *testing.T) {
|
|
|
|
os.Setenv("PORTAINER_TEST_MIGRATE_FAIL", "FAIL")
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
version := "2.15"
|
|
|
|
_, store := MustNewTestStore(t, true, false)
|
|
|
|
store.VersionService.UpdateVersion(&models.Version{SchemaVersion: version, Edition: int(portainer.PortainerCE)})
|
2023-11-27 22:12:17 +00:00
|
|
|
store.MigrateData()
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
store.Open()
|
|
|
|
testVersion(store, version, t)
|
2023-11-27 22:12:17 +00:00
|
|
|
})
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-11-27 22:12:17 +00:00
|
|
|
t.Run("MigrateData should fail to create backup if database file is set to updating", func(t *testing.T) {
|
2023-12-03 20:12:41 +00:00
|
|
|
_, store := MustNewTestStore(t, true, false)
|
2023-11-27 22:12:17 +00:00
|
|
|
store.VersionService.StoreIsUpdating(true)
|
|
|
|
store.MigrateData()
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
// If you get an error, it usually means that the backup folder doesn't exist (no backups). Expected!
|
|
|
|
// If the backup file is not blank, then it means a backup was created. We don't want that because we
|
|
|
|
// only create a backup when the version changes.
|
|
|
|
backupfilename := store.backupFilename()
|
|
|
|
if exists, _ := store.fileService.FileExists(backupfilename); exists {
|
|
|
|
t.Errorf("Backup file should not exist for dirty database")
|
2023-11-27 22:12:17 +00:00
|
|
|
}
|
|
|
|
})
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
t.Run("MigrateData should not create backup on startup if portainer version matches db", func(t *testing.T) {
|
2023-11-27 22:12:17 +00:00
|
|
|
_, store := MustNewTestStore(t, true, false)
|
2023-12-03 20:12:41 +00:00
|
|
|
|
|
|
|
// Set migrator the count to match our migrations array (simulate no changes).
|
|
|
|
// Should not create a backup
|
|
|
|
v, err := store.VersionService.Version()
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unable to read version from db: %s", err)
|
|
|
|
t.FailNow()
|
2023-11-27 22:12:17 +00:00
|
|
|
}
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
migratorParams := store.newMigratorParameters(v)
|
|
|
|
m := migrator.NewMigrator(migratorParams)
|
|
|
|
latestMigrations := m.LatestMigrations()
|
|
|
|
|
|
|
|
if latestMigrations.Version.Equal(semver.MustParse(portainer.APIVersion)) {
|
|
|
|
v.MigratorCount = len(latestMigrations.MigrationFuncs)
|
|
|
|
store.VersionService.UpdateVersion(v)
|
|
|
|
}
|
|
|
|
|
|
|
|
store.MigrateData()
|
|
|
|
|
|
|
|
// If you get an error, it usually means that the backup folder doesn't exist (no backups). Expected!
|
|
|
|
// If the backup file is not blank, then it means a backup was created. We don't want that because we
|
|
|
|
// only create a backup when the version changes.
|
|
|
|
backupfilename := store.backupFilename()
|
|
|
|
if exists, _ := store.fileService.FileExists(backupfilename); exists {
|
|
|
|
t.Errorf("Backup file should not exist for dirty database")
|
|
|
|
}
|
2023-11-27 22:12:17 +00:00
|
|
|
})
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
t.Run("MigrateData should create backup on startup if portainer version matches db and migrationFuncs counts differ", func(t *testing.T) {
|
|
|
|
_, store := MustNewTestStore(t, true, false)
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
// Set migrator count very large to simulate changes
|
|
|
|
// Should not create a backup
|
|
|
|
v, err := store.VersionService.Version()
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unable to read version from db: %s", err)
|
|
|
|
t.FailNow()
|
|
|
|
}
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
v.MigratorCount = 1000
|
|
|
|
store.VersionService.UpdateVersion(v)
|
|
|
|
store.MigrateData()
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
// If you get an error, it usually means that the backup folder doesn't exist (no backups). Expected!
|
|
|
|
// If the backup file is not blank, then it means a backup was created. We don't want that because we
|
|
|
|
// only create a backup when the version changes.
|
|
|
|
backupfilename := store.backupFilename()
|
|
|
|
if exists, _ := store.fileService.FileExists(backupfilename); !exists {
|
|
|
|
t.Errorf("DB backup should exist and there should be no error")
|
|
|
|
}
|
|
|
|
})
|
2021-09-27 00:52:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestRollback(t *testing.T) {
|
|
|
|
t.Run("Rollback should restore upgrade after backup", func(t *testing.T) {
|
2023-12-03 20:12:41 +00:00
|
|
|
version := "2.11"
|
2021-09-27 00:52:50 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
v := models.Version{
|
|
|
|
SchemaVersion: version,
|
2022-11-18 00:18:09 +00:00
|
|
|
}
|
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
_, store := MustNewTestStore(t, false, false)
|
|
|
|
store.VersionService.UpdateVersion(&v)
|
|
|
|
|
|
|
|
_, err := store.Backup()
|
2021-09-27 00:52:50 +00:00
|
|
|
if err != nil {
|
2022-09-16 16:18:44 +00:00
|
|
|
log.Fatal().Err(err).Msg("")
|
2021-09-27 00:52:50 +00:00
|
|
|
}
|
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
v.SchemaVersion = "2.14"
|
|
|
|
// Change the current edition
|
|
|
|
err = store.VersionService.UpdateVersion(&v)
|
2021-09-27 00:52:50 +00:00
|
|
|
if err != nil {
|
2022-09-16 16:18:44 +00:00
|
|
|
log.Fatal().Err(err).Msg("")
|
2021-09-27 00:52:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
err = store.Rollback(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("Rollback failed: %s", err)
|
|
|
|
t.Fail()
|
2022-11-18 00:18:09 +00:00
|
|
|
return
|
|
|
|
}
|
2022-09-16 16:18:44 +00:00
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
store.Open()
|
|
|
|
testVersion(store, version, t)
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Run("Rollback should restore upgrade after backup", func(t *testing.T) {
|
|
|
|
version := "2.15"
|
|
|
|
|
|
|
|
v := models.Version{
|
|
|
|
SchemaVersion: version,
|
|
|
|
Edition: int(portainer.PortainerCE),
|
|
|
|
}
|
|
|
|
|
|
|
|
_, store := MustNewTestStore(t, true, false)
|
|
|
|
store.VersionService.UpdateVersion(&v)
|
|
|
|
|
|
|
|
_, err := store.Backup()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal().Err(err).Msg("")
|
|
|
|
}
|
|
|
|
|
|
|
|
v.SchemaVersion = "2.14"
|
|
|
|
// Change the current edition
|
|
|
|
err = store.VersionService.UpdateVersion(&v)
|
2022-11-18 00:18:09 +00:00
|
|
|
if err != nil {
|
2023-12-03 20:12:41 +00:00
|
|
|
log.Fatal().Err(err).Msg("")
|
|
|
|
}
|
|
|
|
|
|
|
|
err = store.Rollback(true)
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("Rollback failed: %s", err)
|
2022-11-18 00:18:09 +00:00
|
|
|
t.Fail()
|
2021-09-27 00:52:50 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-12-03 20:12:41 +00:00
|
|
|
store.Open()
|
|
|
|
testVersion(store, version, t)
|
2021-09-27 00:52:50 +00:00
|
|
|
})
|
|
|
|
}
|
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
|
|
|
|
2022-04-14 04:25:13 +00:00
|
|
|
// migrateDBTestHelper loads a json representation of a bolt database from srcPath,
|
|
|
|
// parses it into a database, runs a migration on that database, and then
|
|
|
|
// compares it with an expected output database.
|
2022-11-18 00:18:09 +00:00
|
|
|
func migrateDBTestHelper(t *testing.T, srcPath, wantPath string, overrideInstanceId bool) error {
|
2022-04-14 04:25:13 +00:00
|
|
|
srcJSON, err := os.ReadFile(srcPath)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed loading source JSON file %v: %v", srcPath, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse source json to db.
|
2022-11-18 00:18:09 +00:00
|
|
|
// When we create a new test store, it sets its version field automatically to latest.
|
2023-05-30 14:02:22 +00:00
|
|
|
_, store := MustNewTestStore(t, true, false)
|
2022-11-18 00:18:09 +00:00
|
|
|
|
|
|
|
fmt.Println("store.path=", store.GetConnection().GetDatabaseFilePath())
|
|
|
|
store.connection.DeleteObject("version", []byte("VERSION"))
|
|
|
|
|
|
|
|
// defer teardown()
|
2022-04-14 04:25:13 +00:00
|
|
|
err = importJSON(t, bytes.NewReader(srcJSON), store)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Run the actual migrations on our input database.
|
|
|
|
err = store.MigrateData()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-11-18 00:18:09 +00:00
|
|
|
if overrideInstanceId {
|
|
|
|
// old versions of portainer did not have instance-id. Because this gets generated
|
|
|
|
// we need to override the expected output to match the expected value to pass the test
|
|
|
|
v, err := store.VersionService.Version()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
v.InstanceID = "463d5c47-0ea5-4aca-85b1-405ceefee254"
|
|
|
|
err = store.VersionService.UpdateVersion(v)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-14 04:25:13 +00:00
|
|
|
// Assert that our database connection is using bolt so we can call
|
|
|
|
// exportJson rather than ExportRaw. The exportJson function allows us to
|
|
|
|
// strip out the metadata which we don't want for our tests.
|
|
|
|
// TODO: update connection interface in CE to allow us to use ExportRaw and pass meta false
|
|
|
|
err = store.connection.Close()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("err closing bolt connection: %v", err)
|
|
|
|
}
|
|
|
|
con, ok := store.connection.(*boltdb.DbConnection)
|
|
|
|
if !ok {
|
|
|
|
t.Fatalf("backing database is not using boltdb, but the migrations test requires it")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert database back to json.
|
|
|
|
databasePath := con.GetDatabaseFilePath()
|
|
|
|
if _, err := os.Stat(databasePath); err != nil {
|
2023-05-05 00:19:47 +00:00
|
|
|
return fmt.Errorf("stat on %s failed: %w", databasePath, err)
|
2022-04-14 04:25:13 +00:00
|
|
|
}
|
|
|
|
|
2022-09-28 17:56:32 +00:00
|
|
|
gotJSON, err := con.ExportJSON(databasePath, false)
|
2022-04-14 04:25:13 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Logf(
|
|
|
|
"failed re-exporting database %s to JSON: %v",
|
|
|
|
databasePath,
|
|
|
|
err,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
wantJSON, err := os.ReadFile(wantPath)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed loading want JSON file %v: %v", wantPath, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Compare the result we got with the one we wanted.
|
|
|
|
if diff := cmp.Diff(wantJSON, gotJSON); diff != "" {
|
2022-10-24 20:41:30 +00:00
|
|
|
gotPath := filepath.Join(os.TempDir(), "portainer-migrator-test-fail.json")
|
2022-04-14 04:25:13 +00:00
|
|
|
os.WriteFile(
|
|
|
|
gotPath,
|
|
|
|
gotJSON,
|
|
|
|
0600,
|
|
|
|
)
|
|
|
|
t.Errorf(
|
|
|
|
"migrate data from %s to %s failed\nwrote migrated input to %s\nmismatch (-want +got):\n%s",
|
|
|
|
srcPath,
|
|
|
|
wantPath,
|
|
|
|
gotPath,
|
|
|
|
diff,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// importJSON reads input JSON and commits it to a portainer datastore.Store.
|
|
|
|
// Errors are logged with the testing package.
|
|
|
|
func importJSON(t *testing.T, r io.Reader, store *Store) error {
|
|
|
|
objects := make(map[string]interface{})
|
|
|
|
|
|
|
|
// Parse json into map of objects.
|
|
|
|
d := json.NewDecoder(r)
|
|
|
|
d.UseNumber()
|
|
|
|
err := d.Decode(&objects)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get database connection from store.
|
|
|
|
con := store.connection
|
|
|
|
|
|
|
|
for k, v := range objects {
|
|
|
|
switch k {
|
|
|
|
case "version":
|
|
|
|
versions, ok := v.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
t.Logf("failed casting %s to map[string]interface{}", k)
|
|
|
|
}
|
|
|
|
|
2022-11-18 00:18:09 +00:00
|
|
|
// New format db
|
|
|
|
version, ok := versions["VERSION"]
|
|
|
|
if ok {
|
|
|
|
err := con.CreateObjectWithStringId(
|
|
|
|
k,
|
|
|
|
[]byte("VERSION"),
|
|
|
|
version,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("failed writing VERSION in %s: %v", k, err)
|
|
|
|
}
|
2022-04-14 04:25:13 +00:00
|
|
|
}
|
|
|
|
|
2022-11-18 00:18:09 +00:00
|
|
|
// old format db
|
2022-04-14 04:25:13 +00:00
|
|
|
|
2022-11-18 00:18:09 +00:00
|
|
|
dbVersion, ok := versions["DB_VERSION"]
|
|
|
|
if ok {
|
|
|
|
numDBVersion, ok := dbVersion.(json.Number)
|
|
|
|
if !ok {
|
|
|
|
t.Logf("failed parsing DB_VERSION as json number from %s", k)
|
|
|
|
}
|
|
|
|
|
|
|
|
intDBVersion, err := numDBVersion.Int64()
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("failed casting %v to int: %v", numDBVersion, intDBVersion)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = con.CreateObjectWithStringId(
|
|
|
|
k,
|
|
|
|
[]byte("DB_VERSION"),
|
|
|
|
int(intDBVersion),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("failed writing DB_VERSION in %s: %v", k, err)
|
|
|
|
}
|
2022-04-14 04:25:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
instanceID, ok := versions["INSTANCE_ID"]
|
2022-11-18 00:18:09 +00:00
|
|
|
if ok {
|
|
|
|
err = con.CreateObjectWithStringId(
|
|
|
|
k,
|
|
|
|
[]byte("INSTANCE_ID"),
|
|
|
|
instanceID,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("failed writing INSTANCE_ID in %s: %v", k, err)
|
|
|
|
}
|
2022-04-14 04:25:13 +00:00
|
|
|
}
|
|
|
|
|
2022-11-18 00:18:09 +00:00
|
|
|
edition, ok := versions["EDITION"]
|
|
|
|
if ok {
|
|
|
|
err = con.CreateObjectWithStringId(
|
|
|
|
k,
|
|
|
|
[]byte("EDITION"),
|
|
|
|
edition,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("failed writing EDITION in %s: %v", k, err)
|
|
|
|
}
|
2022-04-14 04:25:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
case "dockerhub":
|
|
|
|
obj, ok := v.([]interface{})
|
|
|
|
if !ok {
|
|
|
|
t.Logf("failed to cast %s to []interface{}", k)
|
|
|
|
}
|
|
|
|
err := con.CreateObjectWithStringId(
|
|
|
|
k,
|
|
|
|
[]byte("DOCKERHUB"),
|
|
|
|
obj[0],
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("failed writing DOCKERHUB in %s: %v", k, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
case "ssl":
|
|
|
|
obj, ok := v.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
t.Logf("failed to case %s to map[string]interface{}", k)
|
|
|
|
}
|
|
|
|
err := con.CreateObjectWithStringId(
|
|
|
|
k,
|
|
|
|
[]byte("SSL"),
|
|
|
|
obj,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("failed writing SSL in %s: %v", k, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
case "settings":
|
|
|
|
obj, ok := v.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
t.Logf("failed to case %s to map[string]interface{}", k)
|
|
|
|
}
|
|
|
|
err := con.CreateObjectWithStringId(
|
|
|
|
k,
|
|
|
|
[]byte("SETTINGS"),
|
|
|
|
obj,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("failed writing SETTINGS in %s: %v", k, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
case "tunnel_server":
|
|
|
|
obj, ok := v.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
t.Logf("failed to case %s to map[string]interface{}", k)
|
|
|
|
}
|
|
|
|
err := con.CreateObjectWithStringId(
|
|
|
|
k,
|
|
|
|
[]byte("INFO"),
|
|
|
|
obj,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("failed writing INFO in %s: %v", k, err)
|
|
|
|
}
|
|
|
|
case "templates":
|
|
|
|
continue
|
|
|
|
|
|
|
|
default:
|
|
|
|
objlist, ok := v.([]interface{})
|
|
|
|
if !ok {
|
|
|
|
t.Logf("failed to cast %s to []interface{}", k)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, obj := range objlist {
|
|
|
|
value, ok := obj.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
t.Logf("failed to cast %v to map[string]interface{}", obj)
|
|
|
|
} else {
|
|
|
|
var ok bool
|
|
|
|
var id interface{}
|
|
|
|
switch k {
|
|
|
|
case "endpoint_relations":
|
|
|
|
// TODO: need to make into an int, then do that weird
|
|
|
|
// stringification
|
|
|
|
id, ok = value["EndpointID"]
|
|
|
|
default:
|
|
|
|
id, ok = value["Id"]
|
|
|
|
}
|
|
|
|
if !ok {
|
|
|
|
// endpoint_relations: EndpointID
|
|
|
|
t.Logf("missing Id field: %s", k)
|
|
|
|
id = "error"
|
|
|
|
}
|
|
|
|
n, ok := id.(json.Number)
|
|
|
|
if !ok {
|
|
|
|
t.Logf("failed to cast %v to json.Number in %s", id, k)
|
|
|
|
} else {
|
|
|
|
key, err := n.Int64()
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("failed to cast %v to int in %s", n, k)
|
|
|
|
} else {
|
|
|
|
err := con.CreateObjectWithId(
|
|
|
|
k,
|
|
|
|
int(key),
|
|
|
|
value,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
t.Logf("failed writing %v in %s: %v", key, k, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|