common/variables.go (136 lines of code) (raw):
package common
import (
"errors"
"fmt"
"os"
"path"
"strconv"
"strings"
)
type JobVariable struct {
Key string `json:"key"`
Value string `json:"value"`
Public bool `json:"public"`
Internal bool `json:"-"`
File bool `json:"file"`
Masked bool `json:"masked"`
Raw bool `json:"raw"`
}
type JobVariables []JobVariable
func (b JobVariable) String() string {
return fmt.Sprintf("%s=%s", b.Key, b.Value)
}
const tempProjectDirVariableKey = "RUNNER_TEMP_PROJECT_DIR"
// tmpFile will return a canonical temp file path by prepending the job
// variables Key with the value of `RUNNER_TEMP_PROJECT_DIR` (typically the
// build's temporary directory). The returned path must be further expanded
// by/for each shell that uses it.
func (b JobVariables) tmpFile(s string) string {
return path.Join(b.Value(tempProjectDirVariableKey), s)
}
func (b JobVariables) PublicOrInternal() (variables JobVariables) {
for _, variable := range b {
if variable.Public || variable.Internal {
variables = append(variables, variable)
}
}
return variables
}
func (b JobVariables) StringList() (variables []string) {
for _, variable := range b {
// For file-type secrets, substitute the path to the secret for the secret
// value.
if variable.File {
v := variable
v.Value = b.value(v.Key, true)
variables = append(variables, v.String())
} else {
variables = append(variables, variable.String())
}
}
return variables
}
// Get returns the value of a variable, or if a file type variable, the
// pathname to the saved file containing the value,
func (b JobVariables) Get(key string) string {
return b.value(key, true)
}
// Set sets newJobVars on the JobVariables, replacing an original variable if one exists with the same key.
func (b *JobVariables) Set(newJobVars ...JobVariable) {
for _, newJobVar := range newJobVars {
b.set(newJobVar)
}
}
func (b *JobVariables) set(newJobVar JobVariable) {
for i, v := range *b {
if v.Key == newJobVar.Key {
(*b)[i] = newJobVar
return
}
}
*b = append(*b, newJobVar)
}
// Value is similar to Get(), but always returns the key value, regardless
// of the variable type. File variables therefore return the file contents
// and not the path name of the file.
func (b JobVariables) Value(key string) string {
return b.value(key, false)
}
// value returns the contents of the variable by key.
//
// If the variable type is 'file' and the 'pathnames' parameter is true, then
// the pathname of the file containing the contents is returned instead.
func (b JobVariables) value(key string, pathnames bool) string {
switch key {
case "$":
return key
case "*", "#", "@", "!", "?", "0", "1", "2", "3", "4", "5", "6", "7", "8", "9":
return ""
}
for i := len(b) - 1; i >= 0; i-- {
if b[i].Key == key {
if b[i].File && pathnames {
return b.tmpFile(b[i].Key)
}
return b[i].Value
}
}
return ""
}
// Bool tries to get the boolean value of a variable
// "true" and "false" strings are parsed as well as numeric values
// where only the value of "1" is considered to be true
func (b JobVariables) Bool(key string) bool {
value := b.Get(key)
parsedBool, err := strconv.ParseBool(strings.ToLower(value))
if err == nil {
return parsedBool
}
parsedInt, err := strconv.ParseInt(value, 10, 32)
if err == nil {
return parsedInt == 1
}
return false
}
// OverwriteKey overwrites an existing key with a new variable.
func (b JobVariables) OverwriteKey(key string, variable JobVariable) {
for i, v := range b {
if v.Key == key {
b[i] = variable
return
}
}
}
func (b JobVariables) ExpandValue(value string) string {
return os.Expand(value, b.Get)
}
func (b JobVariables) Expand() JobVariables {
var variables JobVariables
for _, variable := range b {
if !variable.Raw {
variable.Value = b.ExpandValue(variable.Value)
}
variables = append(variables, variable)
}
return variables
}
func (b JobVariables) Masked() (masked []string) {
for _, variable := range b {
if variable.Masked {
masked = append(masked, variable.Value)
}
}
return
}
func ParseVariable(text string) (variable JobVariable, err error) {
keyValue := strings.SplitN(text, "=", 2)
if len(keyValue) != 2 {
err = errors.New("missing =")
return
}
variable = JobVariable{
Key: keyValue[0],
Value: keyValue[1],
}
return
}