mirror of
https://github.com/status-im/consul.git
synced 2025-01-09 21:35:52 +00:00
07f54fe3b8
When one resource contains an inner field that is of type *pbresource.Reference we want the Tenancy to be reasonably defaulted by the following rules: 1. The final values will be limited by the scope of the referenced type. 2. Values will be inferred from the parent's tenancy, and if that is insufficient then using the default tenancy for the type's scope. 3. Namespace will only be used from a parent if the reference and the parent share a partition, otherwise the default namespace will be used. Until we tackle peering, this hard codes an assumption of peer name being local. The logic for defaulting may need adjustment when that is addressed.
176 lines
4.6 KiB
Go
176 lines
4.6 KiB
Go
// Copyright (c) HashiCorp, Inc.
|
|
// SPDX-License-Identifier: BUSL-1.1
|
|
|
|
package resource
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
"google.golang.org/protobuf/proto"
|
|
|
|
"github.com/hashicorp/consul/proto-public/pbresource"
|
|
)
|
|
|
|
const (
|
|
DefaultPartitionName = "default"
|
|
DefaultNamespaceName = "default"
|
|
)
|
|
|
|
// Scope describes the tenancy scope of a resource.
|
|
type Scope int
|
|
|
|
const (
|
|
// There is no default scope, it must be set explicitly.
|
|
ScopeUndefined Scope = iota
|
|
// ScopeCluster describes a resource that is scoped to a cluster.
|
|
ScopeCluster
|
|
// ScopePartition describes a resource that is scoped to a partition.
|
|
ScopePartition
|
|
// ScopeNamespace applies to a resource that is scoped to a partition and namespace.
|
|
ScopeNamespace
|
|
)
|
|
|
|
func (s Scope) String() string {
|
|
switch s {
|
|
case ScopeUndefined:
|
|
return "undefined"
|
|
case ScopeCluster:
|
|
return "cluster"
|
|
case ScopePartition:
|
|
return "partition"
|
|
case ScopeNamespace:
|
|
return "namespace"
|
|
}
|
|
panic(fmt.Sprintf("string mapping missing for scope %v", int(s)))
|
|
}
|
|
|
|
// Normalize lowercases the partition and namespace.
|
|
func Normalize(tenancy *pbresource.Tenancy) {
|
|
if tenancy == nil {
|
|
return
|
|
}
|
|
tenancy.Partition = strings.ToLower(tenancy.Partition)
|
|
tenancy.Namespace = strings.ToLower(tenancy.Namespace)
|
|
}
|
|
|
|
// DefaultClusteredTenancy returns the default tenancy for a cluster scoped resource.
|
|
func DefaultClusteredTenancy() *pbresource.Tenancy {
|
|
return &pbresource.Tenancy{
|
|
// TODO(spatel): Remove as part of "peer is not part of tenancy" ADR
|
|
PeerName: "local",
|
|
}
|
|
}
|
|
|
|
// DefaultPartitionedTenancy returns the default tenancy for a partition scoped resource.
|
|
func DefaultPartitionedTenancy() *pbresource.Tenancy {
|
|
return &pbresource.Tenancy{
|
|
Partition: DefaultPartitionName,
|
|
// TODO(spatel): Remove as part of "peer is not part of tenancy" ADR
|
|
PeerName: "local",
|
|
}
|
|
}
|
|
|
|
// DefaultNamespedTenancy returns the default tenancy for a namespace scoped resource.
|
|
func DefaultNamespacedTenancy() *pbresource.Tenancy {
|
|
return &pbresource.Tenancy{
|
|
Partition: DefaultPartitionName,
|
|
Namespace: DefaultNamespaceName,
|
|
// TODO(spatel): Remove as part of "peer is not part of tenancy" ADR
|
|
PeerName: "local",
|
|
}
|
|
}
|
|
|
|
// DefaultReferenceTenancy will default/normalize the Tenancy of the provided
|
|
// Reference in the context of some parent resource containing that Reference.
|
|
// The default tenancy for the Reference's type is also provided in cases where
|
|
// "default" is needed selectively or the parent is more precise than the
|
|
// child.
|
|
func DefaultReferenceTenancy(ref *pbresource.Reference, parentTenancy, scopeTenancy *pbresource.Tenancy) {
|
|
if ref == nil {
|
|
return
|
|
}
|
|
if ref.Tenancy == nil {
|
|
ref.Tenancy = &pbresource.Tenancy{}
|
|
}
|
|
|
|
if parentTenancy != nil {
|
|
dup := proto.Clone(parentTenancy).(*pbresource.Tenancy)
|
|
parentTenancy = dup
|
|
}
|
|
|
|
defaultTenancy(ref.Tenancy, parentTenancy, scopeTenancy)
|
|
}
|
|
|
|
func defaultTenancy(itemTenancy, parentTenancy, scopeTenancy *pbresource.Tenancy) {
|
|
if itemTenancy == nil {
|
|
panic("item tenancy is required")
|
|
}
|
|
if scopeTenancy == nil {
|
|
panic("scope tenancy is required")
|
|
}
|
|
|
|
if itemTenancy.PeerName == "" {
|
|
itemTenancy.PeerName = "local"
|
|
}
|
|
Normalize(itemTenancy)
|
|
|
|
if parentTenancy != nil {
|
|
// Recursively normalize this tenancy as well.
|
|
defaultTenancy(parentTenancy, nil, scopeTenancy)
|
|
}
|
|
|
|
// use scope defaults for parent
|
|
if parentTenancy == nil {
|
|
parentTenancy = scopeTenancy
|
|
}
|
|
Normalize(parentTenancy)
|
|
|
|
if !equalOrEmpty(itemTenancy.PeerName, "local") {
|
|
panic("peering is not supported yet for resource tenancies")
|
|
}
|
|
if !equalOrEmpty(parentTenancy.PeerName, "local") {
|
|
panic("peering is not supported yet for parent tenancies")
|
|
}
|
|
if !equalOrEmpty(scopeTenancy.PeerName, "local") {
|
|
panic("peering is not supported yet for scopes")
|
|
}
|
|
|
|
// Only retain the parts of the parent that apply to this resource.
|
|
if scopeTenancy.Partition == "" {
|
|
parentTenancy.Partition = ""
|
|
itemTenancy.Partition = ""
|
|
}
|
|
if scopeTenancy.Namespace == "" {
|
|
parentTenancy.Namespace = ""
|
|
itemTenancy.Namespace = ""
|
|
}
|
|
|
|
if parentTenancy.Partition == "" {
|
|
// (cluster scoped)
|
|
} else {
|
|
if itemTenancy.Partition == "" {
|
|
itemTenancy.Partition = parentTenancy.Partition
|
|
}
|
|
if parentTenancy.Namespace == "" {
|
|
// (partition scoped)
|
|
} else {
|
|
// (namespace scoped)
|
|
|
|
if itemTenancy.Namespace == "" {
|
|
if itemTenancy.Partition == parentTenancy.Partition {
|
|
// safe to copy the namespace
|
|
itemTenancy.Namespace = parentTenancy.Namespace
|
|
} else {
|
|
// cross-peer, the namespace must come from the scope default
|
|
itemTenancy.Namespace = scopeTenancy.Namespace
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func equalOrEmpty(a, b string) bool {
|
|
return (a == b) || (a == "") || (b == "")
|
|
}
|