mirror of
https://github.com/portainer/portainer.git
synced 2025-07-19 13:29:41 +02:00
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
This commit is contained in:
parent
19a09b4730
commit
76916b0ad6
3 changed files with 242 additions and 1 deletions
|
@ -2,4 +2,49 @@ package exec
|
||||||
|
|
||||||
import "regexp"
|
import "regexp"
|
||||||
|
|
||||||
var stackNameNormalizeRegex = regexp.MustCompile("[^-_a-z0-9]+")
|
var stackNameNormalizeRegex = regexp.MustCompile("[^-_a-z0-9]+")
|
||||||
|
|
||||||
|
type StringSet map[string]bool
|
||||||
|
|
||||||
|
func NewStringSet() StringSet {
|
||||||
|
return make(StringSet)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s StringSet) Add(x string) {
|
||||||
|
s[x] = true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s StringSet) Remove(x string) {
|
||||||
|
if s.Contains(x) {
|
||||||
|
delete(s, x)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s StringSet) Contains(x string) bool {
|
||||||
|
_, ok := s[x]
|
||||||
|
return ok
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s StringSet) Len() int {
|
||||||
|
return len(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s StringSet) List() []string {
|
||||||
|
list := make([]string, s.Len())
|
||||||
|
|
||||||
|
i := 0
|
||||||
|
for k, _ := range s {
|
||||||
|
list[i] = k
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
|
||||||
|
return list
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s StringSet) Union(x StringSet) {
|
||||||
|
if x.Len() != 0 {
|
||||||
|
for k := range x {
|
||||||
|
s.Add(k)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"path"
|
"path"
|
||||||
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
|
@ -12,6 +13,7 @@ import (
|
||||||
libstack "github.com/portainer/docker-compose-wrapper"
|
libstack "github.com/portainer/docker-compose-wrapper"
|
||||||
"github.com/portainer/docker-compose-wrapper/compose"
|
"github.com/portainer/docker-compose-wrapper/compose"
|
||||||
|
|
||||||
|
"github.com/docker/cli/cli/compose/loader"
|
||||||
portainer "github.com/portainer/portainer/api"
|
portainer "github.com/portainer/portainer/api"
|
||||||
"github.com/portainer/portainer/api/http/proxy"
|
"github.com/portainer/portainer/api/http/proxy"
|
||||||
"github.com/portainer/portainer/api/http/proxy/factory"
|
"github.com/portainer/portainer/api/http/proxy/factory"
|
||||||
|
@ -97,6 +99,12 @@ func (manager *ComposeStackManager) fetchEndpointProxy(endpoint *portainer.Endpo
|
||||||
}
|
}
|
||||||
|
|
||||||
func createEnvFile(stack *portainer.Stack) (string, error) {
|
func createEnvFile(stack *portainer.Stack) (string, error) {
|
||||||
|
// workaround for EE-1862. It will have to be removed when
|
||||||
|
// docker/compose upgraded to v2.x.
|
||||||
|
if err := createNetworkEnvFile(stack); err != nil {
|
||||||
|
return "", errors.Wrap(err, "failed to create network env file")
|
||||||
|
}
|
||||||
|
|
||||||
if stack.Env == nil || len(stack.Env) == 0 {
|
if stack.Env == nil || len(stack.Env) == 0 {
|
||||||
return "", nil
|
return "", nil
|
||||||
}
|
}
|
||||||
|
@ -115,3 +123,137 @@ func createEnvFile(stack *portainer.Stack) (string, error) {
|
||||||
|
|
||||||
return "stack.env", nil
|
return "stack.env", nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func createNetworkEnvFile(stack *portainer.Stack) error {
|
||||||
|
networkNameSet := NewStringSet()
|
||||||
|
|
||||||
|
for _, filePath := range stackutils.GetStackFilePaths(stack) {
|
||||||
|
networkNames, err := extractNetworkNames(filePath)
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrap(err, "failed to extract network name")
|
||||||
|
}
|
||||||
|
|
||||||
|
if networkNames == nil || networkNames.Len() == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
networkNameSet.Union(networkNames)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, s := range networkNameSet.List() {
|
||||||
|
if _, ok := os.LookupEnv(s); ok {
|
||||||
|
networkNameSet.Remove(s)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if networkNameSet.Len() == 0 && stack.Env == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
envfile, err := os.OpenFile(path.Join(stack.ProjectPath, ".env"),
|
||||||
|
os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0600)
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrap(err, "failed to open env file")
|
||||||
|
}
|
||||||
|
|
||||||
|
defer envfile.Close()
|
||||||
|
|
||||||
|
var scanEnvSettingFunc = func(name string) (string, bool) {
|
||||||
|
if stack.Env != nil {
|
||||||
|
for _, v := range stack.Env {
|
||||||
|
if name == v.Name {
|
||||||
|
return v.Value, true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return "", false
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, s := range networkNameSet.List() {
|
||||||
|
if _, ok := scanEnvSettingFunc(s); !ok {
|
||||||
|
stack.Env = append(stack.Env, portainer.Pair{
|
||||||
|
Name: s,
|
||||||
|
Value: "None",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if stack.Env != nil {
|
||||||
|
for _, v := range stack.Env {
|
||||||
|
envfile.WriteString(
|
||||||
|
fmt.Sprintf("%s=%s\n", v.Name, v.Value))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func extractNetworkNames(filePath string) (StringSet, error) {
|
||||||
|
if info, err := os.Stat(filePath); errors.Is(err,
|
||||||
|
os.ErrNotExist) || info.IsDir() {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
stackFileContent, err := os.ReadFile(filePath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "failed to open yaml file")
|
||||||
|
}
|
||||||
|
|
||||||
|
config, err := loader.ParseYAML(stackFileContent)
|
||||||
|
if err != nil {
|
||||||
|
// invalid stack file
|
||||||
|
return nil, errors.Wrap(err, "invalid stack file")
|
||||||
|
}
|
||||||
|
|
||||||
|
var version string
|
||||||
|
if _, ok := config["version"]; ok {
|
||||||
|
version, _ = config["version"].(string)
|
||||||
|
}
|
||||||
|
|
||||||
|
var networks map[string]interface{}
|
||||||
|
if value, ok := config["networks"]; ok {
|
||||||
|
if value == nil {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if networks, ok = value.(map[string]interface{}); !ok {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
networkContent, err := loader.LoadNetworks(networks, version)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil // skip the error
|
||||||
|
}
|
||||||
|
|
||||||
|
re := regexp.MustCompile(`^\$\{?([^\}]+)\}?$`)
|
||||||
|
networkNames := NewStringSet()
|
||||||
|
|
||||||
|
for _, v := range networkContent {
|
||||||
|
matched := re.FindAllStringSubmatch(v.Name, -1)
|
||||||
|
if matched != nil && matched[0] != nil {
|
||||||
|
if strings.Contains(matched[0][1], ":-") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.Contains(matched[0][1], "?") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if strings.Contains(matched[0][1], "-") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
networkNames.Add(matched[0][1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if networkNames.Len() == 0 {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return networkNames, nil
|
||||||
|
}
|
||||||
|
|
|
@ -64,3 +64,57 @@ func Test_createEnvFile(t *testing.T) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func Test_createNetworkEnvFile(t *testing.T) {
|
||||||
|
dir := t.TempDir()
|
||||||
|
buf := []byte(`
|
||||||
|
version: '3.6'
|
||||||
|
services:
|
||||||
|
nginx-example:
|
||||||
|
image: nginx:latest
|
||||||
|
networks:
|
||||||
|
default:
|
||||||
|
name: ${test}
|
||||||
|
driver: bridge
|
||||||
|
`)
|
||||||
|
if err := ioutil.WriteFile(path.Join(dir,
|
||||||
|
"docker-compose.yml"), buf, 0644); err != nil {
|
||||||
|
t.Fatalf("Failed to create yaml file: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
stackWithoutEnv := &portainer.Stack{
|
||||||
|
ProjectPath: dir,
|
||||||
|
EntryPoint: "docker-compose.yml",
|
||||||
|
Env: []portainer.Pair{},
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := createNetworkEnvFile(stackWithoutEnv); err != nil {
|
||||||
|
t.Fatalf("Failed to create network env file: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
content, err := ioutil.ReadFile(path.Join(dir, ".env"))
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to read network env file: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
assert.Equal(t, "test=None\n", string(content))
|
||||||
|
|
||||||
|
stackWithEnv := &portainer.Stack{
|
||||||
|
ProjectPath: dir,
|
||||||
|
EntryPoint: "docker-compose.yml",
|
||||||
|
Env: []portainer.Pair{
|
||||||
|
{Name: "test", Value: "test-value"},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := createNetworkEnvFile(stackWithEnv); err != nil {
|
||||||
|
t.Fatalf("Failed to create network env file: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
content, err = ioutil.ReadFile(path.Join(dir, ".env"))
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to read network env file: %s", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
assert.Equal(t, "test=test-value\n", string(content))
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue