2017-02-07 22:33:23 +01:00
|
|
|
package project
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"path/filepath"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/docker/libcompose/config"
|
|
|
|
"github.com/docker/libcompose/logger"
|
2018-01-22 12:16:03 +01:00
|
|
|
"github.com/sirupsen/logrus"
|
2017-02-07 22:33:23 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var projectRegexp = regexp.MustCompile("[^a-zA-Z0-9_.-]")
|
|
|
|
|
|
|
|
// Context holds context meta information about a libcompose project, like
|
|
|
|
// the project name, the compose file, etc.
|
|
|
|
type Context struct {
|
|
|
|
ComposeFiles []string
|
|
|
|
ComposeBytes [][]byte
|
|
|
|
ProjectName string
|
|
|
|
isOpen bool
|
|
|
|
ServiceFactory ServiceFactory
|
|
|
|
NetworksFactory NetworksFactory
|
2017-07-06 16:28:13 +02:00
|
|
|
VolumesFactory VolumesFactory
|
2017-02-07 22:33:23 +01:00
|
|
|
EnvironmentLookup config.EnvironmentLookup
|
|
|
|
ResourceLookup config.ResourceLookup
|
|
|
|
LoggerFactory logger.Factory
|
|
|
|
IgnoreMissingConfig bool
|
|
|
|
Project *Project
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) readComposeFiles() error {
|
|
|
|
if c.ComposeBytes != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.Debugf("Opening compose files: %s", strings.Join(c.ComposeFiles, ","))
|
|
|
|
|
|
|
|
// Handle STDIN (`-f -`)
|
|
|
|
if len(c.ComposeFiles) == 1 && c.ComposeFiles[0] == "-" {
|
|
|
|
composeBytes, err := ioutil.ReadAll(os.Stdin)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Errorf("Failed to read compose file from stdin: %v", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.ComposeBytes = [][]byte{composeBytes}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, composeFile := range c.ComposeFiles {
|
|
|
|
composeBytes, err := ioutil.ReadFile(composeFile)
|
|
|
|
if err != nil && !os.IsNotExist(err) {
|
|
|
|
logrus.Errorf("Failed to open the compose file: %s", composeFile)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err != nil && !c.IgnoreMissingConfig {
|
|
|
|
logrus.Errorf("Failed to find the compose file: %s", composeFile)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
c.ComposeBytes = append(c.ComposeBytes, composeBytes)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) determineProject() error {
|
|
|
|
name, err := c.lookupProjectName()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
c.ProjectName = normalizeName(name)
|
|
|
|
|
|
|
|
if c.ProjectName == "" {
|
|
|
|
return fmt.Errorf("Falied to determine project name")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) lookupProjectName() (string, error) {
|
|
|
|
if c.ProjectName != "" {
|
|
|
|
return c.ProjectName, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if envProject := os.Getenv("COMPOSE_PROJECT_NAME"); envProject != "" {
|
|
|
|
return envProject, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
file := "."
|
|
|
|
if len(c.ComposeFiles) > 0 {
|
|
|
|
file = c.ComposeFiles[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
f, err := filepath.Abs(file)
|
|
|
|
if err != nil {
|
|
|
|
logrus.Errorf("Failed to get absolute directory for: %s", file)
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
f = toUnixPath(f)
|
|
|
|
|
|
|
|
parent := path.Base(path.Dir(f))
|
|
|
|
if parent != "" && parent != "." {
|
|
|
|
return parent, nil
|
|
|
|
} else if wd, err := os.Getwd(); err != nil {
|
|
|
|
return "", err
|
|
|
|
} else {
|
|
|
|
return path.Base(toUnixPath(wd)), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func normalizeName(name string) string {
|
|
|
|
r := regexp.MustCompile("[^a-z0-9]+")
|
|
|
|
return r.ReplaceAllString(strings.ToLower(name), "")
|
|
|
|
}
|
|
|
|
|
|
|
|
func toUnixPath(p string) string {
|
|
|
|
return strings.Replace(p, "\\", "/", -1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Context) open() error {
|
|
|
|
if c.isOpen {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := c.readComposeFiles(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := c.determineProject(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
c.isOpen = true
|
|
|
|
return nil
|
|
|
|
}
|