-
Notifications
You must be signed in to change notification settings - Fork 38
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Refactor reading test secrets from env vars (#1028)
* Update loading config fields from env vars Updated loading of config fields from environment variables to enhance reusability for CCIP integration * Update how env vars are read to make it usable in CCIP * Fix * Refactor * Clean up pyroscope config * fix lint * Use info level log
- Loading branch information
Showing
4 changed files
with
244 additions
and
246 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,168 @@ | ||
package config | ||
|
||
import ( | ||
"fmt" | ||
"os" | ||
"regexp" | ||
"strconv" | ||
"strings" | ||
|
||
"github.com/smartcontractkit/chainlink-testing-framework/utils/ptr" | ||
) | ||
|
||
const ( | ||
E2E_TEST_LOKI_TENANT_ID_ENV = "E2E_TEST_LOKI_TENANT_ID" | ||
E2E_TEST_LOKI_ENDPOINT_ENV = "E2E_TEST_LOKI_ENDPOINT" | ||
E2E_TEST_LOKI_BASIC_AUTH_ENV = "E2E_TEST_LOKI_BASIC_AUTH" | ||
E2E_TEST_LOKI_BEARER_TOKEN_ENV = "E2E_TEST_LOKI_BEARER_TOKEN" // #nosec G101 | ||
E2E_TEST_GRAFANA_BASE_URL_ENV = "E2E_TEST_GRAFANA_BASE_URL" | ||
E2E_TEST_GRAFANA_DASHBOARD_URL_ENV = "E2E_TEST_GRAFANA_DASHBOARD_URL" | ||
E2E_TEST_GRAFANA_BEARER_TOKEN_ENV = "E2E_TEST_GRAFANA_BEARER_TOKEN" // #nosec G101 | ||
E2E_TEST_PYROSCOPE_ENABLED_ENV = "E2E_TEST_PYROSCOPE_ENABLED" | ||
E2E_TEST_PYROSCOPE_SERVER_URL_ENV = "E2E_TEST_PYROSCOPE_SERVER_URL" | ||
E2E_TEST_PYROSCOPE_KEY_ENV = "E2E_TEST_PYROSCOPE_KEY" | ||
E2E_TEST_PYROSCOPE_ENVIRONMENT_ENV = "E2E_TEST_PYROSCOPE_ENVIRONMENT" | ||
E2E_TEST_CHAINLINK_IMAGE_ENV = "E2E_TEST_CHAINLINK_IMAGE" | ||
E2E_TEST_CHAINLINK_UPGRADE_IMAGE_ENV = "E2E_TEST_CHAINLINK_UPGRADE_IMAGE" | ||
E2E_TEST_WALLET_KEY_ENV = `E2E_TEST_(.+)_WALLET_KEY$` | ||
E2E_TEST_WALLET_KEYS_ENV = `E2E_TEST_(.+)_WALLET_KEY_(\d+)$` | ||
E2E_TEST_RPC_HTTP_URL_ENV = `E2E_TEST_(.+)_RPC_HTTP_URL$` | ||
E2E_TEST_RPC_HTTP_URLS_ENV = `E2E_TEST_(.+)_RPC_HTTP_URL_(\d+)$` | ||
E2E_TEST_RPC_WS_URL_ENV = `E2E_TEST_(.+)_RPC_WS_URL$` | ||
E2E_TEST_RPC_WS_URLS_ENV = `E2E_TEST_(.+)_RPC_WS_URL_(\d+)$` | ||
) | ||
|
||
func MustReadEnvVar_String(name string) string { | ||
value, err := readEnvVarValue(name, String) | ||
if err != nil { | ||
panic(err) | ||
} | ||
if value == nil { | ||
return "" | ||
} | ||
return value.(string) | ||
} | ||
|
||
func MustReadEnvVar_Boolean(name string) *bool { | ||
value, err := readEnvVarValue(name, Boolean) | ||
if err != nil { | ||
panic(err) | ||
} | ||
if value == nil { | ||
return nil | ||
} | ||
return ptr.Ptr(value.(bool)) | ||
} | ||
|
||
func ReadEnvVarGroupedMap(singleEnvPattern, groupEnvPattern string) map[string][]string { | ||
return mergeMaps(readEnvVarSingleMap(singleEnvPattern), readEnvVarGroupedMap(groupEnvPattern)) | ||
} | ||
|
||
// readEnvVarValue reads an environment variable and returns the value parsed according to the specified type. | ||
func readEnvVarValue(envVarName string, valueType EnvValueType) (interface{}, error) { | ||
// Get the environment variable value | ||
value, isSet := os.LookupEnv(envVarName) | ||
if !isSet { | ||
return nil, nil | ||
} | ||
if isSet && value == "" { | ||
return "", nil // Return "" if the environment variable is not set | ||
} | ||
|
||
// Parse the value according to the specified type | ||
switch valueType { | ||
case Integer: | ||
intVal, err := strconv.Atoi(value) | ||
if err != nil { | ||
return nil, fmt.Errorf("error converting value to integer: %v", err) | ||
} | ||
return intVal, nil | ||
case Boolean: | ||
boolVal, err := strconv.ParseBool(value) | ||
if err != nil { | ||
return nil, fmt.Errorf("error converting value to boolean: %v", err) | ||
} | ||
return boolVal, nil | ||
case Float: | ||
floatVal, err := strconv.ParseFloat(value, 64) | ||
if err != nil { | ||
return nil, fmt.Errorf("error converting value to float: %v", err) | ||
} | ||
return floatVal, nil | ||
default: // String or unrecognized type | ||
return value, nil | ||
} | ||
} | ||
|
||
// readEnvVarGroupedMap scans all environment variables, matches them against | ||
// a specified pattern, and returns a map of grouped values based on the pattern. | ||
// The grouping is defined by the first capture group of the regex. | ||
func readEnvVarGroupedMap(pattern string) map[string][]string { | ||
re := regexp.MustCompile(pattern) | ||
groupedVars := make(map[string][]string) | ||
for _, env := range os.Environ() { | ||
pair := strings.SplitN(env, "=", 2) | ||
if len(pair) != 2 { | ||
continue | ||
} | ||
key, value := pair[0], pair[1] | ||
matches := re.FindStringSubmatch(key) | ||
if len(matches) > 1 && value != "" { | ||
group := matches[1] // Use the first capture group for grouping | ||
groupedVars[group] = append(groupedVars[group], value) | ||
} | ||
} | ||
return groupedVars | ||
} | ||
|
||
func readEnvVarSingleMap(pattern string) map[string]string { | ||
re := regexp.MustCompile(pattern) | ||
singleVars := make(map[string]string) | ||
for _, env := range os.Environ() { | ||
pair := strings.SplitN(env, "=", 2) | ||
if len(pair) != 2 { | ||
continue | ||
} | ||
key, value := pair[0], pair[1] | ||
matches := re.FindStringSubmatch(key) | ||
if len(matches) > 1 && value != "" { | ||
group := matches[1] // Use the first capture group for grouping | ||
singleVars[group] = value | ||
} | ||
} | ||
return singleVars | ||
} | ||
|
||
// Merges a map[string]string with a map[string][]string and returns a new map[string][]string. | ||
// Elements from the single map are inserted at index 0 in the slice of the new map. | ||
func mergeMaps(single map[string]string, multi map[string][]string) map[string][]string { | ||
newMap := make(map[string][]string) | ||
|
||
// First, copy all elements from the multi map to the new map | ||
for key, values := range multi { | ||
newMap[key] = make([]string, len(values)) | ||
copy(newMap[key], values) | ||
} | ||
|
||
// Next, insert or prepend the elements from the single map | ||
for key, value := range single { | ||
if existingValues, exists := newMap[key]; exists { | ||
// Prepend the value from the single map | ||
newMap[key] = append([]string{value}, existingValues...) | ||
} else { | ||
// Initialize a new slice if the key does not exist | ||
newMap[key] = []string{value} | ||
} | ||
} | ||
|
||
return newMap | ||
} | ||
|
||
type EnvValueType int | ||
|
||
const ( | ||
String EnvValueType = iota | ||
Integer | ||
Boolean | ||
Float | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.