1Panel/agent/utils/docker/compose.go

151 lines
3.6 KiB
Go

package docker
import (
"bufio"
"bytes"
"context"
"fmt"
"github.com/compose-spec/compose-go/v2/loader"
"github.com/compose-spec/compose-go/v2/types"
"github.com/docker/compose/v2/pkg/api"
"github.com/joho/godotenv"
"gopkg.in/yaml.v3"
"path"
"regexp"
"strings"
)
type ComposeService struct {
api.Service
}
func GetComposeProject(projectName, workDir string, yml []byte, env []byte, skipNormalization bool) (*types.Project, error) {
var configFiles []types.ConfigFile
configFiles = append(configFiles, types.ConfigFile{
Filename: "docker-compose.yml",
Content: yml},
)
envMap, err := godotenv.UnmarshalBytes(env)
if err != nil {
return nil, err
}
details := types.ConfigDetails{
WorkingDir: workDir,
ConfigFiles: configFiles,
Environment: envMap,
}
projectName = strings.ToLower(projectName)
reg, _ := regexp.Compile(`[^a-z0-9_-]+`)
projectName = reg.ReplaceAllString(projectName, "")
project, err := loader.LoadWithContext(context.Background(), details, func(options *loader.Options) {
options.SetProjectName(projectName, true)
options.ResolvePaths = true
options.SkipNormalization = skipNormalization
})
if err != nil {
return nil, err
}
project.ComposeFiles = []string{path.Join(workDir, "docker-compose.yml")}
return project, nil
}
type ComposeProject struct {
Version string
Services map[string]Service `yaml:"services"`
}
type Service struct {
Image string `yaml:"image"`
Environment Environment `yaml:"environment"`
Volumes []string `json:"volumes"`
Restart string `json:"restart"`
}
type Environment struct {
Variables map[string]string
}
func (e *Environment) UnmarshalYAML(value *yaml.Node) error {
e.Variables = make(map[string]string)
switch value.Kind {
case yaml.MappingNode:
for i := 0; i < len(value.Content); i += 2 {
key := value.Content[i].Value
val := value.Content[i+1].Value
e.Variables[key] = val
}
case yaml.SequenceNode:
for _, item := range value.Content {
var kv string
if err := item.Decode(&kv); err != nil {
return err
}
parts := strings.SplitN(kv, "=", 2)
if len(parts) == 2 {
e.Variables[parts[0]] = parts[1]
} else {
e.Variables[parts[0]] = ""
}
}
default:
return fmt.Errorf("unsupported environment format")
}
return nil
}
func GetImagesFromDockerCompose(env, yml []byte) ([]string, error) {
envVars, err := loadEnvFile(env)
if err != nil {
return nil, fmt.Errorf("load env failed: %v", err)
}
var compose ComposeProject
if err := yaml.Unmarshal(yml, &compose); err != nil {
return nil, fmt.Errorf("parse docker-compose file failed: %v", err)
}
var images []string
for _, service := range compose.Services {
if service.Image != "" {
resolvedImage := replaceEnvVars(service.Image, envVars)
images = append(images, resolvedImage)
}
}
return images, nil
}
func loadEnvFile(env []byte) (map[string]string, error) {
envVars := make(map[string]string)
scanner := bufio.NewScanner(bytes.NewReader(env))
for scanner.Scan() {
line := strings.TrimSpace(scanner.Text())
if line == "" || strings.HasPrefix(line, "#") {
continue
}
parts := strings.SplitN(line, "=", 2)
if len(parts) == 2 {
key := strings.TrimSpace(parts[0])
value := strings.TrimSpace(parts[1])
value = strings.Trim(value, `"'`)
envVars[key] = value
}
}
return envVars, scanner.Err()
}
func replaceEnvVars(input string, envVars map[string]string) string {
re := regexp.MustCompile(`\$\{([^}]+)\}`)
return re.ReplaceAllStringFunc(input, func(match string) string {
varName := match[2 : len(match)-1]
if value, exists := envVars[varName]; exists {
return value
}
return match
})
}