Skip to content

Commit

Permalink
allocator2: preliminary interface definition
Browse files Browse the repository at this point in the history
Informs #103320

Epic: CRDB-25222

Release note: None
  • Loading branch information
sumeerbhola committed Jun 7, 2023
1 parent 05dded4 commit b1b7ea2
Show file tree
Hide file tree
Showing 4 changed files with 179 additions and 1 deletion.
1 change: 1 addition & 0 deletions pkg/kv/kvserver/allocator/allocator2/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ load("@io_bazel_rules_go//go:def.bzl", "go_library", "go_test")
go_library(
name = "allocator2",
srcs = [
"allocator.go",
"allocator_state.go",
"cluster_state.go",
"constraint.go",
Expand Down
172 changes: 172 additions & 0 deletions pkg/kv/kvserver/allocator/allocator2/allocator.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,172 @@
// Copyright 2023 The Cockroach Authors.
//
// Use of this software is governed by the Business Source License
// included in the file licenses/BSL.txt.
//
// As of the Change Date specified in that file, in accordance with
// the Business Source License, use of this software will be governed
// by the Apache License, Version 2.0, included in the file
// licenses/APL.txt.

package allocator2

import "github.com/cockroachdb/cockroach/pkg/roachpb"

// ChangeOptions is passed to ComputeChanges and AdminScatterOne.
type ChangeOptions struct {
// DryRun tells the allocator not to update its internal state with the
// proposed pending changes.
DryRun bool
}

// Allocator is the interface for a distributed allocator. We expect that the
// core of the allocator implementation will not know or care whether the
// allocator is distributed or centralized, but there may be specializations
// for the distributed case at higher layers of the implementation.
// Additionally, there may be:
// - adapter layers that bridge the gap between this interface and the current
// integration with the old allocator.
// - changes to this interface to make the integration for the new allocator
// be less different than integration with the old allocator.
type Allocator interface {
// Methods to update the state of the external world. The allocator starts
// with no knowledge.

// SetStore informs the allocator about a new store, or when something about
// the store descriptor has changed. The allocator's knowledge about the
// nodes in the cluster is a side effect of this method.
SetStore(store roachpb.StoreDescriptor) error

// RemoveNodeAndStores tells the allocator to remove the nodeID and all its
// stores.
//
// TODO(sumeer): do we need a RemoveStore(store roachpb.StoreDescriptor)
// method to handle removing a store from a node?
RemoveNodeAndStores(nodeID roachpb.NodeID) error

// UpdateFailureDetectionSummary tells the allocator about the current
// failure detection state for a node. A node starts in the fdOK state.
UpdateFailureDetectionSummary(nodeID roachpb.NodeID, fd failureDetectionSummary) error

// ProcessNodeLoadResponse provides frequent updates to the state of the
// external world.
//
// Only the node on which this allocator is running will provide a non-empty
// nodeLoadResponse.leaseholderStores slice.
//
// TODO(kvoli,sumeer): The storeLoadMsg may need a combination of gossip changes
// and local synthesis:
//
// - storeLoadMsg.storeRanges is the list of ranges at the node, and is
// needed for accounting for load adjustments, and avoids making the
// assumption that all changes have been applied. However, for expediency
// of integration, we could relax this and synthesize this list as a
// function of (a) timestamp of the gossiped load, (b) timestamp of the
// proposed change, (c) timestamp of the latest leaseholderStores slice
// that shows the change as being enacted.
//
// - storeLoadMsg.storeRanges does not need to be the complete list of
// ranges -- it can be filtered down to only the ranges for which the
// local node is the leaseholder.
//
// - storeLoadMsg.topKRanges could be locally synthesized by estimating the
// load on a follower based on the measured load at the leaseholder. This
// will not give us the follower non-raft CPU, so we will need to assume
// that the CPU for a follower is only the raftCPU, and that it has the
// same value as the raftCPU at the leaseholder.
ProcessNodeLoadResponse(resp *nodeLoadResponse) error

// TODO(sumeer): only a subset of the fields in pendingReplicaChange are
// relevant to the caller. Hide the remaining.

// Methods related to making changes.

// AdjustPendingChangesDisposition is optional feedback to inform the
// allocator of success or failure of proposed changes. For successful
// changes, this is a faster way to know about success than waiting for the
// next ProcessNodeLoadResponse from the local node. For failed changes, in
// the absence of this feedback, proposed changes that have not been enacted
// in N seconds will be garbage collected and assumed to have failed.
//
// Calls to AdjustPendingChangesDisposition must be correctly sequenced with
// full state updates from the local node provided in
// ProcessNodeLoadResponse.
AdjustPendingChangesDisposition(changes []pendingReplicaChange, success bool) error

// ComputeChanges is called periodically and frequently, say every 10s.
//
// Currently, the only proposed changes are rebalancing (include lease
// transfers) of ranges for which this node is a leaseholder, to improve
// load balance across all stores in the cluster. Later we will add support
// for up-replication, constraint satisfaction etc. (the core helper classes
// already support such changes, so this limitation is mainly placed for
// ease of incremental integration).
//
// Unless ChangeOptions.DryRun is true, changes returned are remembered by
// the allocator, to avoid re-proposing the same change and to make
// adjustments to the load.
ComputeChanges(opts ChangeOptions) []*pendingReplicaChange

// AdminRelocateOne is a helper for AdminRelocateRange.
//
// The allocator must know the stores listed in the targets. The
// implementation ignores the allocator's state of the range's replicas and
// leaseholder, and utilizes the information provided in desc, conf, and
// leaseholderStore (leaseholderStore is -1 if there is no leaseholder) --
// for this reason it does not remember the proposed changes. The
// implementation ignores the current load on the targets, though it returns
// an error if any of the targets has a failure detector state != fdOK.
// Additionally, it checks that the voterTargets and nonVoterTargets satisfy
// all the range constraints. Leaseholder preferences are ignored if
// transferLeaseToFirstVoter is true, else the leaseholder is picked
// randomly from the best set of voters (best defined by leaseholder
// preferences. Load is not considered). Diversity scores are ignored since
// the set of voters and non-voters have been explicitly specified.
//
// If the returned slice is empty and error is nil, there are no more
// changes to make.
//
// Note for future integration work: the intention here is to subsume some
// of the code in replica_command.go, and this method will be called from
// Replica.relocateReplicas (RelocateOne code is no longer needed).
//
// NB: leaseholderStore is needed because this function may not be called on
// the leaseholder. Also, even if it is being called at the leaseholder
// node, the node may have multiple stores, and we need to know which store
// is the leaseholder. This also implies that the leaseholder transfer does
// not need to be the last step. For the same reason, the rangeID may not be
// known to the allocator, which is why the range's SpanConfig is provided.
AdminRelocateOne(desc *roachpb.RangeDescriptor, conf *roachpb.SpanConfig,
leaseholderStore roachpb.StoreID,
voterTargets, nonVoterTargets []roachpb.ReplicationTarget,
transferLeaseToFirstVoter bool) ([]pendingReplicaChange, error)

// AdminScatterOne is a helper for AdminScatterRequest. Consider it a more
// aggressive rebalance, with more randomization. The difference wrt
// rebalancing is that (a) the candidate range for rebalancing is being
// specified by the caller, instead of having to be figured out by the
// allocator. Like rebalancing, the candidate store to shed from and the
// store to add to is computed by the callee, and the result is provided in
// []pendingReplicaChange. Exactly one such pair of remove-add will be
// specified in the pending changes. Unless DryRun is true, these pending
// changes are remembered by the callee and the caller should not call again
// for the same range until it has enacted the change and informed the
// callee about the change success or failure via
// AdjustPendingChangesDisposition.
//
// If the caller will call AdminScatterOne in a loop to potentially scatter
// all the replicas, only the last call should specify
// canTransferLease=true, since transferring the lease will make it no
// longer possible to call AdminScatterOne on this node.
//
// TODO(sumeer):
// - understand what is really happening as a consequence of
// ScorerOptionsForScatter.
AdminScatterOne(
rangeID roachpb.RangeID, canTransferLease bool, opts ChangeOptions,
) ([]pendingReplicaChange, error)
}

// Avoid unused lint errors.

var _ = ChangeOptions{}.DryRun
2 changes: 1 addition & 1 deletion pkg/kv/kvserver/allocator/allocator2/cluster_state.go
Original file line number Diff line number Diff line change
Expand Up @@ -226,7 +226,7 @@ type storeIDAndReplicaState struct {
replicaState
}

// rangeState is periodically updated based in reporting by the leaseholder.
// rangeState is periodically updated based on reporting by the leaseholder.
type rangeState struct {
// replicas is the adjusted replicas. It is always consistent with
// the storeState.adjusted.replicas in the corresponding stores.
Expand Down
5 changes: 5 additions & 0 deletions pkg/kv/kvserver/allocator/allocator2/messages.go
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,11 @@ func (rm *rangeMsg) isDeletedRange() bool {
type nodeLoadResponse struct {
// -1 if this nodeLoadMsg is not a diff. Responder can unilaterally send a
// complete load even if the sender asked for a diff.
//
// TODO(sumeer): the diff story is not properly fleshed out for the local
// node in a distributed allocator, since there are also other updates via
// AdjustPendingChangesDisposition. We can start with only full state
// updates, and add diff support later.
lastLoadSeqNum int64
curLoadSeqNum int64

Expand Down

0 comments on commit b1b7ea2

Please sign in to comment.