-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathenv_config.go
93 lines (83 loc) · 2.65 KB
/
env_config.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
package aceptadora
import (
"bufio"
"fmt"
"os"
"strings"
"testing"
"github.com/stretchr/testify/require"
)
// SetEnv loads the configuration from a file, choosing the proper one depending on the provided matchers.
// Every matcher returning true as second value will be executed.
// Matchers are executed in their provided order
func SetEnv(t *testing.T, matchers ...ConfigPathMatcher) {
for _, f := range matchers {
if path, shouldBeUsed := f(); shouldBeUsed {
env, err := loadConfigFromFile(path)
require.NoError(t, err, "Can't load config from %s: %s", path, err)
for k, v := range env {
os.Setenv(k, v)
}
t.Logf("Loaded env from %q", path)
}
}
}
type ConfigPathMatcher func() (path string, use bool)
// OneOfEnvConfigs provides a matcher that returns as the result the result of the first matching matcher.
// Useful when several matchers would match, like for instance a Github Actions environment is being a Linux at the same time.
func OneOfEnvConfigs(matchers ...ConfigPathMatcher) ConfigPathMatcher {
return func() (string, bool) {
for _, match := range matchers {
if path, matches := match(); matches {
return path, true
}
}
return "", false
}
}
// EnvConfigWhenEnvVarPresent is a matcher for SetEnv that matches when the provided env var name is present
func EnvConfigWhenEnvVarPresent(path, envVarName string) ConfigPathMatcher {
return func() (string, bool) {
_, isGitlab := os.LookupEnv(envVarName)
return path, isGitlab
}
}
// EnvConfigAlways is a matcher for SetEnv that is always true, useful to load the common acceptance.env config once
// the env-specifics are loaded.
func EnvConfigAlways(path string) ConfigPathMatcher {
return func() (string, bool) {
return path, true
}
}
// loadConfigFromFile parses the configuration file into a map, expanding the env var references to their values.
func loadConfigFromFile(filepath string) (map[string]string, error) {
file, err := os.Open(filepath)
if err != nil {
return nil, err
}
defer file.Close()
config := make(map[string]string)
scanner := bufio.NewScanner(file)
for scanner.Scan() {
line := strings.TrimSpace(scanner.Text())
line = os.ExpandEnv(line)
if len(line) > 0 && line[0] != '#' {
s := strings.SplitN(line, "=", 2)
config[s[0]] = s[1]
}
}
if err := scanner.Err(); err != nil {
return nil, fmt.Errorf("error parsing config file: %v", err)
}
return config, nil
}
// mergeConfigs merges, overwriting values, the provided configs into a single one.
func mergeConfigs(cfgs ...map[string]string) map[string]string {
cfg := map[string]string{}
for _, another := range cfgs {
for k, v := range another {
cfg[k] = v
}
}
return cfg
}