-
Notifications
You must be signed in to change notification settings - Fork 18
/
option.go
129 lines (110 loc) · 3.78 KB
/
option.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
122
123
124
125
126
127
128
129
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
// SPDX-License-Identifier: Apache-2.0
// Package option customizes how tests are run.
package option
import (
"errors"
"fmt"
"os"
"os/exec"
"strings"
)
type feature int
const (
environmentVariablePassthrough feature = iota
nerdctlVersion feature = iota
)
// Option customizes how tests are run.
//
// If a testing function needs special customizations other than the ones specified in Option,
// we can use composition to extend it.
// For example, to test login functionality,
// we may create a struct named LoginOption that embeds Option and contains additional fields like Username and Password.
type Option struct {
subject []string
env []string
features map[feature]any
}
// New does some sanity checks on the arguments before initializing an Option.
//
// subject specifies the subject to be tested.
// It is intentionally not designed as an (optional) Modifier because it must contain at least one element.
// Essentially it is used as a prefix when invoking all the binaries during testing.
//
// For example, if subject is ["foo", "bar"], then to test pulling a image, the command name would be "foo",
// and the command args would be something like ["bar", "pull", "alpine"].
func New(subject []string, modifiers ...Modifier) (*Option, error) {
if len(subject) == 0 {
return nil, errors.New("missing subject")
}
o := &Option{
subject: subject,
features: map[feature]any{
environmentVariablePassthrough: true,
nerdctlVersion: nerdctl2xx,
},
}
for _, modifier := range modifiers {
modifier.modify(o)
}
return o, nil
}
// NewCmd creates a command using the stored option and the provided args.
func (o *Option) NewCmd(args ...string) *exec.Cmd {
cmdName := o.subject[0]
cmdArgs := append(o.subject[1:], args...) //nolint:gocritic // appendAssign does not apply to our case.
cmd := exec.Command(cmdName, cmdArgs...) //nolint:gosec // G204 is not an issue because cmdName is fully controlled by the user.
cmd.Env = append(os.Environ(), o.env...)
return cmd
}
// UpdateEnv updates the environment variable for the key name of the input.
func (o *Option) UpdateEnv(envKey, envValue string) {
env := fmt.Sprintf("%s=%s", envKey, envValue)
if i, exists := containsEnv(o.env, envKey); exists {
o.env[i] = env
} else {
o.env = append(o.env, env)
}
}
// DeleteEnv deletes the environment variable for the key name of the input.
func (o *Option) DeleteEnv(envKey string) {
if i, exists := containsEnv(o.env, envKey); exists {
o.env = append(o.env[:i], o.env[i+1:]...)
}
}
// containsEnv determines whether an environment variable exists.
func containsEnv(envs []string, targetEnvKey string) (int, bool) {
for i, env := range envs {
if strings.Split(env, "=")[0] == targetEnvKey {
return i, true
}
}
return -1, false
}
// SupportsEnvVarPassthrough is used by tests to check if the option
// supports [feature.environmentVariablePassthrough].
func (o *Option) SupportsEnvVarPassthrough() bool {
if value, exists := o.features[environmentVariablePassthrough]; exists {
if boolValue, ok := value.(bool); ok {
return boolValue
}
}
return false
}
// IsNerdctlV1 is used by tests to check if the option supports [feature.nerdctlVersion] == nerdctl1xx.
func (o *Option) IsNerdctlV1() bool {
return o.isNerdctlVersion(isNerdctl1xx)
}
// IsNerdctlV2 is used by tests to check if the option supports [feature.nerdctlVersion] == nerdctl2xx.
func (o *Option) IsNerdctlV2() bool {
return o.isNerdctlVersion(isNerdctl2xx)
}
func (o *Option) isNerdctlVersion(cmp func(string) bool) bool {
var version string
if value, exists := o.features[nerdctlVersion]; !exists {
version = defaultNerdctlVersion
} else if value, ok := value.(string); ok {
version = value
}
return cmp(version)
}