-
Notifications
You must be signed in to change notification settings - Fork 3.8k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
allocator2: preliminary interface definition
Informs #103320 Epic: CRDB-25222 Release note: None
- Loading branch information
1 parent
05dded4
commit b1b7ea2
Showing
4 changed files
with
179 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters