2020-04-15 13:12:43 +02:00
|
|
|
package versioning
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Docker defines an artifact based on a Dockerfile
|
|
|
|
type Docker struct {
|
2020-05-06 22:07:27 +02:00
|
|
|
artifact Artifact
|
|
|
|
content []byte
|
2020-11-10 17:14:55 +01:00
|
|
|
utils Utils
|
2020-05-06 22:07:27 +02:00
|
|
|
options *Options
|
|
|
|
path string
|
|
|
|
versionSource string
|
|
|
|
versioningScheme string
|
|
|
|
readFile func(string) ([]byte, error)
|
|
|
|
writeFile func(string, []byte, os.FileMode) error
|
2020-04-15 13:12:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Docker) init() {
|
|
|
|
if d.readFile == nil {
|
2023-08-16 12:57:04 +02:00
|
|
|
d.readFile = os.ReadFile
|
2020-04-15 13:12:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if d.writeFile == nil {
|
2023-08-16 12:57:04 +02:00
|
|
|
d.writeFile = os.WriteFile
|
2020-04-15 13:12:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Docker) initDockerfile() {
|
|
|
|
if len(d.path) == 0 {
|
|
|
|
d.path = "Dockerfile"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// VersioningScheme returns the relevant versioning scheme
|
|
|
|
func (d *Docker) VersioningScheme() string {
|
2020-05-06 22:07:27 +02:00
|
|
|
if len(d.versioningScheme) == 0 {
|
|
|
|
return "docker"
|
|
|
|
}
|
|
|
|
return d.versioningScheme
|
2020-04-15 13:12:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetVersion returns the current version of the artifact
|
|
|
|
func (d *Docker) GetVersion() (string, error) {
|
|
|
|
d.init()
|
|
|
|
var err error
|
|
|
|
|
|
|
|
switch d.versionSource {
|
|
|
|
case "FROM":
|
|
|
|
var err error
|
|
|
|
d.initDockerfile()
|
|
|
|
d.content, err = d.readFile(d.path)
|
|
|
|
if err != nil {
|
|
|
|
return "", errors.Wrapf(err, "failed to read file '%v'", d.path)
|
|
|
|
}
|
|
|
|
version := d.versionFromBaseImageTag()
|
|
|
|
if len(version) == 0 {
|
|
|
|
return "", fmt.Errorf("no version information available in FROM statement")
|
|
|
|
}
|
|
|
|
return version, nil
|
2020-04-29 13:42:14 +02:00
|
|
|
case "":
|
|
|
|
if len(d.path) == 0 {
|
|
|
|
d.path = "VERSION"
|
|
|
|
}
|
|
|
|
d.versionSource = "custom"
|
|
|
|
fallthrough
|
2020-04-15 13:12:43 +02:00
|
|
|
case "custom", "dub", "golang", "maven", "mta", "npm", "pip", "sbt":
|
2020-04-29 13:42:14 +02:00
|
|
|
if d.options == nil {
|
|
|
|
d.options = &Options{}
|
|
|
|
}
|
2020-11-10 17:14:55 +01:00
|
|
|
d.artifact, err = GetArtifact(d.versionSource, d.path, d.options, d.utils)
|
2020-04-15 13:12:43 +02:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return d.artifact.GetVersion()
|
|
|
|
default:
|
|
|
|
d.initDockerfile()
|
|
|
|
d.content, err = d.readFile(d.path)
|
|
|
|
if err != nil {
|
|
|
|
return "", errors.Wrapf(err, "failed to read file '%v'", d.path)
|
|
|
|
}
|
|
|
|
version := d.versionFromEnv(d.versionSource)
|
|
|
|
if len(version) == 0 {
|
|
|
|
return "", fmt.Errorf("no version information available in ENV '%v'", d.versionSource)
|
|
|
|
}
|
|
|
|
return version, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetVersion updates the version of the artifact
|
|
|
|
func (d *Docker) SetVersion(version string) error {
|
|
|
|
d.init()
|
|
|
|
|
|
|
|
dir := ""
|
|
|
|
|
|
|
|
if d.artifact != nil {
|
|
|
|
err := d.artifact.SetVersion(version)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
dir = filepath.Dir(d.path)
|
|
|
|
}
|
|
|
|
|
|
|
|
err := d.writeFile(filepath.Join(dir, "VERSION"), []byte(version), 0700)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "failed to write file 'VERSION'")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Docker) versionFromEnv(env string) string {
|
|
|
|
lines := strings.Split(string(d.content), "\n")
|
|
|
|
for _, line := range lines {
|
|
|
|
if strings.HasPrefix(line, "ENV") && strings.Fields(line)[1] == env {
|
|
|
|
return strings.Fields(line)[2]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *Docker) versionFromBaseImageTag() string {
|
|
|
|
lines := strings.Split(string(d.content), "\n")
|
|
|
|
for _, line := range lines {
|
|
|
|
if strings.HasPrefix(line, "FROM") {
|
|
|
|
imageParts := strings.Split(line, ":")
|
|
|
|
partsCount := len(imageParts)
|
|
|
|
if partsCount == 1 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
version := imageParts[partsCount-1]
|
|
|
|
return strings.TrimSpace(version)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
2020-05-25 19:48:59 +02:00
|
|
|
|
|
|
|
// GetCoordinates returns the coordinates
|
|
|
|
func (d *Docker) GetCoordinates() (Coordinates, error) {
|
2021-02-10 16:18:00 +01:00
|
|
|
result := Coordinates{}
|
2021-02-03 14:52:48 +01:00
|
|
|
|
|
|
|
result.GroupID = ""
|
|
|
|
result.ArtifactID, _ = d.GetArtifactID()
|
|
|
|
|
|
|
|
result.Version = ""
|
|
|
|
// cannot properly resolve version unless all options are provided. Can we ensure proper parameterization?
|
|
|
|
// result.Version, err = d.GetVersion()
|
|
|
|
// if err != nil {
|
|
|
|
// return nil, err
|
|
|
|
// }
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetArtifactID returns the current ID of the artifact
|
|
|
|
func (d *Docker) GetArtifactID() (string, error) {
|
|
|
|
d.init()
|
|
|
|
|
|
|
|
artifactID := strings.ReplaceAll(strings.ReplaceAll(strings.ReplaceAll(d.options.DockerImage, "/", "_"), ":", "_"), ".", "_")
|
|
|
|
return artifactID, nil
|
2020-05-25 19:48:59 +02:00
|
|
|
}
|