-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathbuild_configuration_parser.go
121 lines (102 loc) · 3.23 KB
/
build_configuration_parser.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
package gobuild
import (
"fmt"
"os"
"path/filepath"
"strings"
"github.com/mattn/go-shellwords"
"github.com/paketo-buildpacks/packit/v2/fs"
)
//go:generate faux --interface TargetManager --output fakes/target_manager.go
type TargetManager interface {
CleanAndValidate(targets []string, workingDir string) ([]string, error)
GenerateDefaults(workingDir string) ([]string, error)
}
type BuildConfiguration struct {
Targets []string
Flags []string
ImportPath string
WorkspaceUseModules []string
}
type BuildConfigurationParser struct {
targetManager TargetManager
}
func NewBuildConfigurationParser(targetManager TargetManager) BuildConfigurationParser {
return BuildConfigurationParser{
targetManager: targetManager,
}
}
func (p BuildConfigurationParser) Parse(buildpackVersion, workingDir string) (BuildConfiguration, error) {
bpYML, err := fs.Exists(filepath.Join(workingDir, "buildpack.yml"))
if err != nil {
return BuildConfiguration{}, fmt.Errorf("failed to check for buildpack.yml: %w", err)
}
if bpYML {
return BuildConfiguration{}, fmt.Errorf("working directory contains deprecated 'buildpack.yml'; use environment variables for configuration")
}
var buildConfiguration BuildConfiguration
if val, ok := os.LookupEnv("BP_GO_TARGETS"); ok {
buildConfiguration.Targets = filepath.SplitList(val)
}
if len(buildConfiguration.Targets) > 0 {
buildConfiguration.Targets, err = p.targetManager.CleanAndValidate(buildConfiguration.Targets, workingDir)
if err != nil {
return BuildConfiguration{}, err
}
} else {
buildConfiguration.Targets, err = p.targetManager.GenerateDefaults(workingDir)
if err != nil {
return BuildConfiguration{}, err
}
}
buildConfiguration.Flags, err = parseFlagsFromEnvVars(buildConfiguration.Flags)
if err != nil {
return BuildConfiguration{}, err
}
if val, ok := os.LookupEnv("BP_GO_BUILD_IMPORT_PATH"); ok {
buildConfiguration.ImportPath = val
}
if val, ok := os.LookupEnv("BP_GO_WORK_USE"); ok {
buildConfiguration.WorkspaceUseModules = filepath.SplitList(val)
}
return buildConfiguration, nil
}
func containsFlag(flags []string, match string) bool {
for _, flag := range flags {
if strings.HasPrefix(flag, match) {
return true
}
}
return false
}
func parseFlagsFromEnvVars(flags []string) ([]string, error) {
shellwordsParser := shellwords.NewParser()
shellwordsParser.ParseEnv = true
if buildFlags, ok := os.LookupEnv("BP_GO_BUILD_FLAGS"); ok {
var err error
flags, err = shellwordsParser.Parse(buildFlags)
if err != nil {
return nil, err
}
}
if ldFlags, ok := os.LookupEnv("BP_GO_BUILD_LDFLAGS"); ok {
parsed, err := shellwordsParser.Parse(fmt.Sprintf(`-ldflags="%s"`, ldFlags))
if err != nil {
return nil, err
}
if len(parsed) != 1 {
return nil, fmt.Errorf("BP_GO_BUILD_LDFLAGS value (%s) could not be parsed: value contains multiple words", ldFlags)
}
for i, flag := range flags {
if strings.HasPrefix(flag, "-ldflags") {
// Replace value from BP_GO_BUILD_FLAGS or buildpack.yml with value from
// BP_GO_BUILD_LDFLAGS because BP_GO_BUILD_LDFLAGS takes precedence
flags[i] = parsed[0]
}
}
if !containsFlag(flags, "-ldflags") {
flags = append(flags, parsed[0])
}
}
return flags, nil
}