2023-03-28 19:39:22 +01:00
|
|
|
// Copyright (c) HashiCorp, Inc.
|
|
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
|
2023-03-09 13:40:23 -06:00
|
|
|
package resource
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
2023-04-06 10:40:04 +01:00
|
|
|
"github.com/hashicorp/go-hclog"
|
2023-03-09 13:40:23 -06:00
|
|
|
"google.golang.org/grpc"
|
2023-03-27 14:37:54 -05:00
|
|
|
"google.golang.org/grpc/codes"
|
2023-03-27 16:25:27 -05:00
|
|
|
"google.golang.org/grpc/metadata"
|
2023-03-27 14:37:54 -05:00
|
|
|
"google.golang.org/grpc/status"
|
2023-04-06 10:40:04 +01:00
|
|
|
"google.golang.org/protobuf/proto"
|
2023-03-09 13:40:23 -06:00
|
|
|
|
2023-04-11 06:10:14 -05:00
|
|
|
"github.com/hashicorp/consul/acl"
|
|
|
|
"github.com/hashicorp/consul/acl/resolver"
|
2023-03-27 10:35:39 -05:00
|
|
|
"github.com/hashicorp/consul/internal/resource"
|
|
|
|
"github.com/hashicorp/consul/internal/storage"
|
2023-03-09 13:40:23 -06:00
|
|
|
"github.com/hashicorp/consul/proto-public/pbresource"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Server struct {
|
|
|
|
Config
|
|
|
|
}
|
|
|
|
|
|
|
|
type Config struct {
|
2023-04-06 10:40:04 +01:00
|
|
|
Logger hclog.Logger
|
|
|
|
Registry Registry
|
2023-04-04 17:30:06 +01:00
|
|
|
|
|
|
|
// Backend is the storage backend that will be used for resource persistence.
|
2023-04-11 06:10:14 -05:00
|
|
|
Backend Backend
|
|
|
|
ACLResolver ACLResolver
|
2023-03-27 14:37:54 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
//go:generate mockery --name Registry --inpackage
|
|
|
|
type Registry interface {
|
|
|
|
resource.Registry
|
2023-03-27 10:35:39 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
//go:generate mockery --name Backend --inpackage
|
|
|
|
type Backend interface {
|
|
|
|
storage.Backend
|
2023-03-09 13:40:23 -06:00
|
|
|
}
|
|
|
|
|
2023-04-11 06:10:14 -05:00
|
|
|
//go:generate mockery --name ACLResolver --inpackage
|
|
|
|
type ACLResolver interface {
|
|
|
|
ResolveTokenAndDefaultMeta(string, *acl.EnterpriseMeta, *acl.AuthorizerContext) (resolver.Result, error)
|
|
|
|
}
|
|
|
|
|
2023-03-09 13:40:23 -06:00
|
|
|
func NewServer(cfg Config) *Server {
|
|
|
|
return &Server{cfg}
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ pbresource.ResourceServiceServer = (*Server)(nil)
|
|
|
|
|
|
|
|
func (s *Server) Register(grpcServer *grpc.Server) {
|
|
|
|
pbresource.RegisterResourceServiceServer(grpcServer, s)
|
|
|
|
}
|
|
|
|
|
2023-04-11 06:10:14 -05:00
|
|
|
// Get token from grpc metadata or AnonymounsTokenId if not found
|
|
|
|
func tokenFromContext(ctx context.Context) string {
|
|
|
|
md, ok := metadata.FromIncomingContext(ctx)
|
|
|
|
if !ok {
|
|
|
|
return acl.AnonymousTokenID
|
|
|
|
}
|
|
|
|
|
|
|
|
vals := md.Get("x-consul-token")
|
|
|
|
if len(vals) == 0 {
|
|
|
|
return acl.AnonymousTokenID
|
|
|
|
}
|
|
|
|
return vals[0]
|
|
|
|
}
|
|
|
|
|
2023-03-27 14:37:54 -05:00
|
|
|
func (s *Server) resolveType(typ *pbresource.Type) (*resource.Registration, error) {
|
2023-04-06 10:40:04 +01:00
|
|
|
v, ok := s.Registry.Resolve(typ)
|
2023-03-27 14:37:54 -05:00
|
|
|
if ok {
|
|
|
|
return &v, nil
|
|
|
|
}
|
|
|
|
return nil, status.Errorf(
|
|
|
|
codes.InvalidArgument,
|
|
|
|
"resource type %s not registered", resource.ToGVK(typ),
|
|
|
|
)
|
2023-03-09 13:40:23 -06:00
|
|
|
}
|
2023-03-27 16:25:27 -05:00
|
|
|
|
|
|
|
func readConsistencyFrom(ctx context.Context) storage.ReadConsistency {
|
|
|
|
md, ok := metadata.FromIncomingContext(ctx)
|
|
|
|
if !ok {
|
|
|
|
return storage.EventualConsistency
|
|
|
|
}
|
|
|
|
|
|
|
|
vals := md.Get("x-consul-consistency-mode")
|
|
|
|
if len(vals) == 0 {
|
|
|
|
return storage.EventualConsistency
|
|
|
|
}
|
|
|
|
|
|
|
|
if vals[0] == "consistent" {
|
|
|
|
return storage.StrongConsistency
|
|
|
|
}
|
|
|
|
return storage.EventualConsistency
|
|
|
|
}
|
2023-04-06 10:40:04 +01:00
|
|
|
|
2023-04-11 06:10:14 -05:00
|
|
|
func (s *Server) getAuthorizer(token string) (acl.Authorizer, error) {
|
|
|
|
authz, err := s.ACLResolver.ResolveTokenAndDefaultMeta(token, nil, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, status.Errorf(codes.Internal, "failed getting authorizer: %v", err)
|
|
|
|
}
|
|
|
|
return authz, nil
|
|
|
|
}
|
|
|
|
|
2023-04-14 08:19:46 -05:00
|
|
|
func isGRPCStatusError(err error) bool {
|
|
|
|
if err == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
_, ok := status.FromError(err)
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
2023-04-17 16:33:20 -05:00
|
|
|
func validateId(id *pbresource.ID, errorPrefix string) error {
|
|
|
|
var field string
|
|
|
|
switch {
|
|
|
|
case id.Type == nil:
|
|
|
|
field = "type"
|
|
|
|
case id.Tenancy == nil:
|
|
|
|
field = "tenancy"
|
|
|
|
case id.Name == "":
|
|
|
|
field = "name"
|
|
|
|
}
|
|
|
|
|
|
|
|
if field != "" {
|
|
|
|
return status.Errorf(codes.InvalidArgument, "%s.%s is required", errorPrefix, field)
|
|
|
|
}
|
|
|
|
|
2023-04-24 08:14:51 -05:00
|
|
|
// Revisit defaulting and non-namespaced resources post-1.16
|
|
|
|
var expected string
|
2023-04-17 16:33:20 -05:00
|
|
|
switch {
|
2023-04-24 08:14:51 -05:00
|
|
|
case id.Tenancy.Partition != "default":
|
|
|
|
field, expected = "partition", "default"
|
|
|
|
case id.Tenancy.Namespace != "default":
|
|
|
|
field, expected = "namespace", "default"
|
|
|
|
case id.Tenancy.PeerName != "local":
|
|
|
|
field, expected = "peername", "local"
|
2023-04-17 16:33:20 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if field != "" {
|
2023-04-24 08:14:51 -05:00
|
|
|
return status.Errorf(codes.InvalidArgument, "%s.tenancy.%s must be %s", errorPrefix, field, expected)
|
2023-04-17 16:33:20 -05:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-04-06 10:40:04 +01:00
|
|
|
func clone[T proto.Message](v T) T { return proto.Clone(v).(T) }
|