2023-03-28 20:12:30 +01:00
|
|
|
// Copyright (c) HashiCorp, Inc.
|
2023-08-11 09:12:13 -04:00
|
|
|
// SPDX-License-Identifier: BUSL-1.1
|
2023-03-28 20:12:30 +01:00
|
|
|
|
2022-07-11 10:13:40 -04:00
|
|
|
package helpers
|
2019-08-07 16:41:33 -05:00
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"unicode"
|
|
|
|
"unicode/utf8"
|
|
|
|
|
|
|
|
"github.com/hashicorp/hcl"
|
|
|
|
)
|
|
|
|
|
|
|
|
// hclDecode is a modified version of hcl.Decode just for the super general
|
|
|
|
// purposes here. There's some strange bug in how hcl.Decode decodes json where
|
|
|
|
//
|
|
|
|
// { "sub" : { "v1" : { "field" : "value1" }, "v2" : { "field" : "value2" } } }
|
|
|
|
//
|
|
|
|
// hcl.Decode-s into:
|
|
|
|
//
|
2022-10-21 15:58:06 -04:00
|
|
|
// map[string]interface {}{
|
|
|
|
// "sub":[]map[string]interface {}{
|
|
|
|
// map[string]interface {}{
|
|
|
|
// "v1":[]map[string]interface {}{
|
|
|
|
// map[string]interface {}{
|
|
|
|
// "field":"value1"
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// },
|
|
|
|
// map[string]interface {}{
|
|
|
|
// "v2":[]map[string]interface {}{
|
|
|
|
// map[string]interface {}{
|
|
|
|
// "field":"value2"
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
2019-08-07 16:41:33 -05:00
|
|
|
//
|
|
|
|
// but json.Unmarshal-s into the more expected:
|
|
|
|
//
|
2022-10-21 15:58:06 -04:00
|
|
|
// map[string]interface {}{
|
|
|
|
// "sub":map[string]interface {}{
|
|
|
|
// "v1":map[string]interface {}{
|
|
|
|
// "field":"value1"
|
|
|
|
// },
|
|
|
|
// "v2":map[string]interface {}{
|
|
|
|
// "field":"value2"
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
2019-08-07 16:41:33 -05:00
|
|
|
//
|
|
|
|
// The strange part is that the following HCL:
|
|
|
|
//
|
|
|
|
// sub { "v1" = { field = "value1" }, "v2" = { field = "value2" } }
|
|
|
|
//
|
|
|
|
// hcl.Decode-s into:
|
|
|
|
//
|
2022-10-21 15:58:06 -04:00
|
|
|
// map[string]interface {}{
|
|
|
|
// "sub":[]map[string]interface {}{
|
|
|
|
// map[string]interface {}{
|
|
|
|
// "v1":[]map[string]interface {}{
|
|
|
|
// map[string]interface {}{
|
|
|
|
// "field":"value1"
|
|
|
|
// }
|
|
|
|
// },
|
|
|
|
// "v2":[]map[string]interface {}{
|
|
|
|
// map[string]interface {}{
|
|
|
|
// "field":"value2"
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
2019-08-07 16:41:33 -05:00
|
|
|
//
|
|
|
|
// Which is the "correct" value assuming you did the patch-slice-of-maps correction.
|
|
|
|
//
|
|
|
|
// Given that HCLv1 is basically frozen and the HCL part of it is fine instead
|
|
|
|
// of trying to track down a weird bug we'll bypass the weird JSON decoder and just use
|
|
|
|
// the stdlib one.
|
|
|
|
func hclDecode(out interface{}, in string) error {
|
|
|
|
data := []byte(in)
|
|
|
|
if isHCL(data) {
|
|
|
|
return hcl.Decode(out, in)
|
|
|
|
}
|
|
|
|
|
|
|
|
return json.Unmarshal(data, out)
|
|
|
|
}
|
|
|
|
|
|
|
|
// this is an inlined variant of hcl.lexMode()
|
|
|
|
func isHCL(v []byte) bool {
|
|
|
|
var (
|
|
|
|
r rune
|
|
|
|
w int
|
|
|
|
offset int
|
|
|
|
)
|
|
|
|
|
|
|
|
for {
|
|
|
|
r, w = utf8.DecodeRune(v[offset:])
|
|
|
|
offset += w
|
|
|
|
if unicode.IsSpace(r) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if r == '{' {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|