blob: 0bb81ae43b16287acb03c71233651bdd8e7493cb [file] [log] [blame]
// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1
package marks
import (
"sort"
"strings"
"github.com/hashicorp/terraform/internal/lang/format"
"github.com/zclconf/go-cty/cty"
)
// PathsWithMark produces a list of paths identified as having a specified
// mark in a given set of [cty.PathValueMarks] that presumably resulted from
// deeply-unmarking a [cty.Value].
//
// This is for situations where a subsystem needs to give special treatment
// to one specific mark value, as opposed to just handling all marks
// generically as cty operations would. The second return value is a
// subset of the given [cty.PathValueMarks] values which contained marks
// other than the one requested, so that a caller that can't preserve other
// marks at all can more easily return an error explaining that.
func PathsWithMark(pvms []cty.PathValueMarks, wantMark any) (withWanted []cty.Path, withOthers []cty.PathValueMarks) {
if len(pvms) == 0 {
// No-allocations path for the common case where there are no marks at all.
return nil, nil
}
for _, pvm := range pvms {
if _, ok := pvm.Marks[wantMark]; ok {
withWanted = append(withWanted, pvm.Path)
}
for mark := range pvm.Marks {
if mark != wantMark {
withOthers = append(withOthers, pvm)
// only add a path with unwanted marks a single time
break
}
}
}
return withWanted, withOthers
}
// RemoveAll take a series of PathValueMarks and removes the unwanted mark from
// all paths. Paths with no remaining marks will be removed entirely. The
// PathValuesMarks passed in are not cloned, and RemoveAll will modify the
// original values, so the prior set of marks should not be retained for use.
func RemoveAll(pvms []cty.PathValueMarks, remove any) []cty.PathValueMarks {
if len(pvms) == 0 {
// No-allocations path for the common case where there are no marks at all.
return nil
}
var res []cty.PathValueMarks
for _, pvm := range pvms {
delete(pvm.Marks, remove)
if len(pvm.Marks) > 0 {
res = append(res, pvm)
}
}
return res
}
// MarkPaths transforms the given value by marking each of the given paths
// with the given mark value.
func MarkPaths(val cty.Value, mark any, paths []cty.Path) cty.Value {
if len(paths) == 0 {
// No-allocations path for the common case where there are no marked paths at all.
return val
}
// For now we'll use cty's slightly lower-level function to achieve this
// result. This is a little inefficient due to an additional dynamic
// allocation for the intermediate data structure, so if that becomes
// a problem in practice then we may wish to write a more direct
// implementation here.
markses := make([]cty.PathValueMarks, len(paths))
marks := cty.NewValueMarks(mark)
for i, path := range paths {
markses[i] = cty.PathValueMarks{
Path: path,
Marks: marks,
}
}
return val.MarkWithPaths(markses)
}
// MarksEqual compares 2 unordered sets of PathValue marks for equality, with
// the comparison using the cty.PathValueMarks.Equal method.
func MarksEqual(a, b []cty.PathValueMarks) bool {
if len(a) == 0 && len(b) == 0 {
return true
}
if len(a) != len(b) {
return false
}
less := func(s []cty.PathValueMarks) func(i, j int) bool {
return func(i, j int) bool {
cmp := strings.Compare(format.CtyPath(s[i].Path), format.CtyPath(s[j].Path))
switch {
case cmp < 0:
return true
case cmp > 0:
return false
}
// the sort only needs to be consistent, so use the GoString format
// to get a comparable value
return s[i].Marks.GoString() < s[j].Marks.GoString()
}
}
sort.Slice(a, less(a))
sort.Slice(b, less(b))
for i := 0; i < len(a); i++ {
if !a[i].Equal(b[i]) {
return false
}
}
return true
}