|
| 1 | +// Package formatting provides internal utilities for formatting Terraform/Terragrunt CLI arguments. |
| 2 | +package formatting |
| 3 | + |
| 4 | +import ( |
| 5 | + "fmt" |
| 6 | + "reflect" |
| 7 | + "strconv" |
| 8 | + "strings" |
| 9 | +) |
| 10 | + |
| 11 | +// FormatBackendConfigAsArgs formats backend configuration as Terraform CLI args. |
| 12 | +// Example: {"bucket": "my-bucket"} -> ["-backend-config=bucket=my-bucket"] |
| 13 | +func FormatBackendConfigAsArgs(vars map[string]interface{}) []string { |
| 14 | + return formatTerraformArgs(vars, "-backend-config", false, true) |
| 15 | +} |
| 16 | + |
| 17 | +// FormatPluginDirAsArgs formats plugin directory as a Terraform CLI arg. |
| 18 | +// Returns nil if pluginDir is empty. |
| 19 | +func FormatPluginDirAsArgs(pluginDir string) []string { |
| 20 | + if pluginDir == "" { |
| 21 | + return nil |
| 22 | + } |
| 23 | + return []string{fmt.Sprintf("-plugin-dir=%v", pluginDir)} |
| 24 | +} |
| 25 | + |
| 26 | +// formatTerraformArgs formats vars as CLI args with the given prefix. |
| 27 | +func formatTerraformArgs(vars map[string]interface{}, prefix string, useSpaceAsSeparator bool, omitNil bool) []string { |
| 28 | + var args []string |
| 29 | + |
| 30 | + for key, value := range vars { |
| 31 | + var argValue string |
| 32 | + if omitNil && value == nil { |
| 33 | + argValue = key |
| 34 | + } else { |
| 35 | + hclString := toHclString(value, false) |
| 36 | + argValue = fmt.Sprintf("%s=%s", key, hclString) |
| 37 | + } |
| 38 | + if useSpaceAsSeparator { |
| 39 | + args = append(args, prefix, argValue) |
| 40 | + } else { |
| 41 | + args = append(args, fmt.Sprintf("%s=%s", prefix, argValue)) |
| 42 | + } |
| 43 | + } |
| 44 | + |
| 45 | + return args |
| 46 | +} |
| 47 | + |
| 48 | +// toHclString converts Go values to HCL-formatted strings for Terraform CLI arguments. |
| 49 | +// Handles primitives, slices, and maps. Example: []int{1,2,3} -> "[1, 2, 3]" |
| 50 | +func toHclString(value interface{}, isNested bool) string { |
| 51 | + if slice, isSlice := tryToConvertToGenericSlice(value); isSlice { |
| 52 | + return sliceToHclString(slice) |
| 53 | + } else if m, isMap := tryToConvertToGenericMap(value); isMap { |
| 54 | + return mapToHclString(m) |
| 55 | + } else { |
| 56 | + return primitiveToHclString(value, isNested) |
| 57 | + } |
| 58 | +} |
| 59 | + |
| 60 | +// tryToConvertToGenericSlice converts any slice type to []interface{} using reflection. |
| 61 | +func tryToConvertToGenericSlice(value interface{}) ([]interface{}, bool) { |
| 62 | + reflectValue := reflect.ValueOf(value) |
| 63 | + if reflectValue.Kind() != reflect.Slice { |
| 64 | + return []interface{}{}, false |
| 65 | + } |
| 66 | + |
| 67 | + genericSlice := make([]interface{}, reflectValue.Len()) |
| 68 | + |
| 69 | + for i := 0; i < reflectValue.Len(); i++ { |
| 70 | + genericSlice[i] = reflectValue.Index(i).Interface() |
| 71 | + } |
| 72 | + |
| 73 | + return genericSlice, true |
| 74 | +} |
| 75 | + |
| 76 | +// tryToConvertToGenericMap converts any map[string]T to map[string]interface{} using reflection. |
| 77 | +func tryToConvertToGenericMap(value interface{}) (map[string]interface{}, bool) { |
| 78 | + reflectValue := reflect.ValueOf(value) |
| 79 | + if reflectValue.Kind() != reflect.Map { |
| 80 | + return map[string]interface{}{}, false |
| 81 | + } |
| 82 | + |
| 83 | + reflectType := reflect.TypeOf(value) |
| 84 | + if reflectType.Key().Kind() != reflect.String { |
| 85 | + return map[string]interface{}{}, false |
| 86 | + } |
| 87 | + |
| 88 | + genericMap := make(map[string]interface{}, reflectValue.Len()) |
| 89 | + |
| 90 | + mapKeys := reflectValue.MapKeys() |
| 91 | + for _, key := range mapKeys { |
| 92 | + genericMap[key.String()] = reflectValue.MapIndex(key).Interface() |
| 93 | + } |
| 94 | + |
| 95 | + return genericMap, true |
| 96 | +} |
| 97 | + |
| 98 | +func sliceToHclString(slice []interface{}) string { |
| 99 | + hclValues := []string{} |
| 100 | + |
| 101 | + for _, value := range slice { |
| 102 | + hclValue := toHclString(value, true) |
| 103 | + hclValues = append(hclValues, hclValue) |
| 104 | + } |
| 105 | + |
| 106 | + return fmt.Sprintf("[%s]", strings.Join(hclValues, ", ")) |
| 107 | +} |
| 108 | + |
| 109 | +func mapToHclString(m map[string]interface{}) string { |
| 110 | + keyValuePairs := []string{} |
| 111 | + |
| 112 | + for key, value := range m { |
| 113 | + keyValuePair := fmt.Sprintf(`"%s" = %s`, key, toHclString(value, true)) |
| 114 | + keyValuePairs = append(keyValuePairs, keyValuePair) |
| 115 | + } |
| 116 | + |
| 117 | + return fmt.Sprintf("{%s}", strings.Join(keyValuePairs, ", ")) |
| 118 | +} |
| 119 | + |
| 120 | +func primitiveToHclString(value interface{}, isNested bool) string { |
| 121 | + if value == nil { |
| 122 | + return "null" |
| 123 | + } |
| 124 | + |
| 125 | + switch v := value.(type) { |
| 126 | + |
| 127 | + case bool: |
| 128 | + return strconv.FormatBool(v) |
| 129 | + |
| 130 | + case string: |
| 131 | + // If string is nested in a larger data structure (e.g. list of string, map of string), ensure value is quoted |
| 132 | + if isNested { |
| 133 | + return fmt.Sprintf("\"%v\"", v) |
| 134 | + } |
| 135 | + |
| 136 | + return fmt.Sprintf("%v", v) |
| 137 | + |
| 138 | + default: |
| 139 | + return fmt.Sprintf("%v", v) |
| 140 | + } |
| 141 | +} |
0 commit comments