mirror of
https://github.com/status-im/consul.git
synced 2025-01-24 20:51:10 +00:00
5fb9df1640
* Adding explicit MPL license for sub-package This directory and its subdirectories (packages) contain files licensed with the MPLv2 `LICENSE` file in this directory and are intentionally licensed separately from the BSL `LICENSE` file at the root of this repository. * Adding explicit MPL license for sub-package This directory and its subdirectories (packages) contain files licensed with the MPLv2 `LICENSE` file in this directory and are intentionally licensed separately from the BSL `LICENSE` file at the root of this repository. * Updating the license from MPL to Business Source License Going forward, this project will be licensed under the Business Source License v1.1. Please see our blog post for more details at <Blog URL>, FAQ at www.hashicorp.com/licensing-faq, and details of the license at www.hashicorp.com/bsl. * add missing license headers * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 * Update copyright file headers to BUSL-1.1 --------- Co-authored-by: hashicorp-copywrite[bot] <110428419+hashicorp-copywrite[bot]@users.noreply.github.com>
164 lines
5.5 KiB
Go
164 lines
5.5 KiB
Go
// Copyright (c) HashiCorp, Inc.
|
|
// SPDX-License-Identifier: BUSL-1.1
|
|
|
|
package resource
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/oklog/ulid/v2"
|
|
"google.golang.org/grpc/codes"
|
|
"google.golang.org/grpc/status"
|
|
"google.golang.org/protobuf/types/known/anypb"
|
|
|
|
"github.com/hashicorp/consul/acl"
|
|
"github.com/hashicorp/consul/internal/resource"
|
|
"github.com/hashicorp/consul/internal/storage"
|
|
"github.com/hashicorp/consul/proto-public/pbresource"
|
|
)
|
|
|
|
// Deletes a resource.
|
|
// - To delete a resource regardless of the stored version, set Version = ""
|
|
// - Supports deleting a resource by name, hence Id.Uid may be empty.
|
|
// - Delete of a previously deleted or non-existent resource is a no-op to support idempotency.
|
|
// - Errors with Aborted if the requested Version does not match the stored Version.
|
|
// - Errors with PermissionDenied if ACL check fails
|
|
func (s *Server) Delete(ctx context.Context, req *pbresource.DeleteRequest) (*pbresource.DeleteResponse, error) {
|
|
if err := validateDeleteRequest(req); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
reg, err := s.resolveType(req.Id.Type)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// TODO(spatel): Refactor _ and entMeta in NET-4919
|
|
authz, authzContext, err := s.getAuthorizer(tokenFromContext(ctx), acl.DefaultEnterpriseMeta())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Retrieve resource since ACL hook requires it. Furthermore, we'll need the
|
|
// read to be strongly consistent if the passed in Version or Uid are empty.
|
|
consistency := storage.EventualConsistency
|
|
if req.Version == "" || req.Id.Uid == "" {
|
|
consistency = storage.StrongConsistency
|
|
}
|
|
existing, err := s.Backend.Read(ctx, consistency, req.Id)
|
|
switch {
|
|
case errors.Is(err, storage.ErrNotFound):
|
|
// Deletes are idempotent so no-op when not found
|
|
return &pbresource.DeleteResponse{}, nil
|
|
case err != nil:
|
|
return nil, status.Errorf(codes.Internal, "failed read: %v", err)
|
|
}
|
|
|
|
// Check ACLs
|
|
err = reg.ACLs.Write(authz, authzContext, existing)
|
|
switch {
|
|
case acl.IsErrPermissionDenied(err):
|
|
return nil, status.Error(codes.PermissionDenied, err.Error())
|
|
case err != nil:
|
|
return nil, status.Errorf(codes.Internal, "failed write acl: %v", err)
|
|
}
|
|
|
|
deleteVersion := req.Version
|
|
deleteId := req.Id
|
|
if deleteVersion == "" || deleteId.Uid == "" {
|
|
deleteVersion = existing.Version
|
|
deleteId = existing.Id
|
|
}
|
|
|
|
if err := s.maybeCreateTombstone(ctx, deleteId); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = s.Backend.DeleteCAS(ctx, deleteId, deleteVersion)
|
|
switch {
|
|
case err == nil:
|
|
return &pbresource.DeleteResponse{}, nil
|
|
case errors.Is(err, storage.ErrCASFailure):
|
|
return nil, status.Error(codes.Aborted, err.Error())
|
|
default:
|
|
return nil, status.Errorf(codes.Internal, "failed delete: %v", err)
|
|
}
|
|
}
|
|
|
|
// Create a tombstone to capture the intent to delete child resources.
|
|
// Tombstones are created preemptively to prevent partial failures even though
|
|
// we are currently unaware of the success/failure/no-op of DeleteCAS. In
|
|
// the failure and no-op cases the tombstone is effectively a no-op and will
|
|
// still be deleted from the system by the reaper controller.
|
|
func (s *Server) maybeCreateTombstone(ctx context.Context, deleteId *pbresource.ID) error {
|
|
// Don't create a tombstone when the resource being deleted is itself a tombstone.
|
|
if resource.EqualType(resource.TypeV1Tombstone, deleteId.Type) {
|
|
return nil
|
|
}
|
|
|
|
data, err := anypb.New(&pbresource.Tombstone{Owner: deleteId})
|
|
if err != nil {
|
|
return status.Errorf(codes.Internal, "failed creating tombstone: %v", err)
|
|
}
|
|
|
|
// Since a tombstone is an internal resource type that should not be visible
|
|
// or accessible by users, we're writing to the backend directly instead of
|
|
// using the resource service's Write endpoint. This bypasses resource level
|
|
// concerns that are either not relevant (valiation and mutation hooks) or
|
|
// futher complicate the implementation (user provided tokens having
|
|
// awareness of tombstone ACLs).
|
|
//
|
|
// ErrCASFailure should never happen since an empty Version is always passed.
|
|
//
|
|
// TODO(spatel): Probably a good idea to block writes of TypeV1Tombstone
|
|
// on the ResourceService.Write() endpoint to lock things down?
|
|
_, err = s.Backend.WriteCAS(ctx, &pbresource.Resource{
|
|
Id: &pbresource.ID{
|
|
Type: resource.TypeV1Tombstone,
|
|
Tenancy: deleteId.Tenancy,
|
|
Name: tombstoneName(deleteId),
|
|
Uid: ulid.Make().String(),
|
|
},
|
|
Generation: ulid.Make().String(),
|
|
Data: data,
|
|
Metadata: map[string]string{
|
|
"generated_at": time.Now().Format(time.RFC3339),
|
|
},
|
|
})
|
|
|
|
switch {
|
|
case err == nil:
|
|
// Success!
|
|
return nil
|
|
case errors.Is(err, storage.ErrWrongUid):
|
|
// Backend has detected that we're trying to change the Uid for an
|
|
// existing tombstone (probably created from a previously failed Delete
|
|
// where the tombstone WriteCAS succeeded but the resource DeleteCAS
|
|
// failed). The fact that the tombstone already exists means we're good.
|
|
return nil
|
|
default:
|
|
return status.Errorf(codes.Internal, "failed writing tombstone: %v", err)
|
|
}
|
|
}
|
|
|
|
func validateDeleteRequest(req *pbresource.DeleteRequest) error {
|
|
if req.Id == nil {
|
|
return status.Errorf(codes.InvalidArgument, "id is required")
|
|
}
|
|
|
|
if err := validateId(req.Id, "id"); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Maintains a deterministic mapping between a resource and it's tombstone's
|
|
// name by embedding the resources's Uid in the name.
|
|
func tombstoneName(deleteId *pbresource.ID) string {
|
|
// deleteId.Name is just included for easier identification
|
|
return fmt.Sprintf("tombstone-%v-%v", deleteId.Name, deleteId.Uid)
|
|
}
|