You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
boundary/internal/servers/controller/handlers/mask_manager.go

93 lines
2.6 KiB

package handlers
import (
"fmt"
"strings"
"google.golang.org/protobuf/proto"
"google.golang.org/protobuf/reflect/protoreflect"
"google.golang.org/protobuf/types/descriptorpb"
pb "github.com/hashicorp/boundary/internal/gen/controller/protooptions"
)
type MaskManager map[string]string
// NewMaskManager returns a mask manager that can translate field masks into the first proto from all subsequent
// protos assuming they are both using the mask_mapping custom option. Error is returned if no mappings are
// found or if one of the passed protos has a mapping that doesn't reciprocate.
func NewMaskManager(dest protoreflect.ProtoMessage, src ...protoreflect.ProtoMessage) (MaskManager, error) {
srcToDest, err := mapFromProto(src...)
if err != nil {
return nil, err
}
destToSrc, err := mapFromProto(dest)
if err != nil {
return nil, err
}
result := make(map[string]string)
for k, v := range srcToDest {
ov, ok := destToSrc[v]
if !ok || ov != k {
return nil, fmt.Errorf("mapping src field %q maps to %q, dest %q maps to %q", k, v, v, ov)
}
result[k] = v
}
// Now check to make sure there aren't any dangling dest mappings.
for k, v := range destToSrc {
if ov, ok := srcToDest[v]; !ok || ov != k {
return nil, fmt.Errorf("mapping dest field %q maps to %q, src %q maps to %q", k, v, v, ov)
}
}
if len(result) == 0 {
return nil, fmt.Errorf("size of mask mapping is 0")
}
return result, nil
}
func mapFromProto(ps ...protoreflect.ProtoMessage) (map[string]string, error) {
mapping := make(map[string]string)
for _, p := range ps {
m := p.ProtoReflect()
fields := m.Descriptor().Fields()
for i := 0; i < fields.Len(); i++ {
f := fields.Get(i)
opts := f.Options().(*descriptorpb.FieldOptions)
if nameMap := proto.GetExtension(opts, pb.E_MaskMapping).(*pb.MaskMapping); !proto.Equal(nameMap, &pb.MaskMapping{}) && nameMap != nil {
if _, ok := mapping[nameMap.GetThis()]; ok {
return nil, fmt.Errorf("duplicate mapping from field %q with the mapping key %q", f.Name(), nameMap.GetThis())
}
mapping[nameMap.GetThis()] = nameMap.GetThat()
}
}
}
return mapping, nil
}
// Translate takes a field mask's paths and returns paths translated for the destination's protobuf.
func (m MaskManager) Translate(paths []string) []string {
var result []string
for _, v := range paths {
vSplit := strings.Split(v, ",")
for _, v := range vSplit {
if ov, ok := m[strings.TrimSpace(v)]; ok {
result = append(result, ov)
}
}
}
return result
}
func MaskContains(paths []string, s string) bool {
for _, p := range paths {
if p == s {
return true
}
}
return false
}