Skip to content

Commit

Permalink
Tablets: handle schema / topology changes
Browse files Browse the repository at this point in the history
There are some situations where tablet information can become stale.
This can have various consequences:
- it would prevent old `Node` objects from being destroyed and make
  them continue to being used in LBP. 
- some of the replicas may be ignored indefinitely by the driver,
  increasing load on the rest.


To prevent this we need to update tablet info during topology change
with newly fetched data about cluster.
  • Loading branch information
Lorak-mmk committed Mar 28, 2024
1 parent 4cf6236 commit b7892ee
Show file tree
Hide file tree
Showing 2 changed files with 254 additions and 19 deletions.
47 changes: 44 additions & 3 deletions scylla/src/transport/cluster.rs
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ use itertools::Itertools;
use scylla_cql::errors::{BadQuery, NewSessionError};
use scylla_cql::frame::response::result::TableSpec;
use scylla_cql::types::serialize::row::SerializedValues;
use std::collections::HashMap;
use std::collections::{HashMap, HashSet};
use std::net::SocketAddr;
use std::sync::Arc;
use std::time::Duration;
Expand Down Expand Up @@ -291,7 +291,7 @@ impl ClusterData {
known_peers: &HashMap<Uuid, Arc<Node>>,
used_keyspace: &Option<VerifiedKeyspaceName>,
host_filter: Option<&dyn HostFilter>,
#[cfg(feature = "unstable-tablets")] tablets: TabletsInfo,
#[cfg(feature = "unstable-tablets")] mut tablets: TabletsInfo,
) -> Self {
// Create new updated known_peers and ring
let mut new_known_peers: HashMap<Uuid, Arc<Node>> =
Expand Down Expand Up @@ -351,6 +351,40 @@ impl ClusterData {
}
}

#[cfg(feature = "unstable-tablets")]
{
let mut removed_nodes = HashSet::new();
for old_peer in known_peers {
if !new_known_peers.contains_key(old_peer.0) {
removed_nodes.insert(*old_peer.0);
}
}

let table_predicate = |spec: &TableSpec| {
if let Some(ks) = metadata.keyspaces.get(spec.ks_name.as_str()) {
ks.tables.contains_key(spec.table_name.as_str())
} else {
false
}
};

let mut recreated_nodes = HashMap::new();
for (old_peer_id, old_peer_node) in known_peers {
if let Some(new_peer_node) = new_known_peers.get(old_peer_id) {
if !Arc::ptr_eq(old_peer_node, new_peer_node) {
recreated_nodes.insert(*old_peer_id, Arc::clone(new_peer_node));
}
}
}

tablets.perform_maintanance(
&table_predicate,
&removed_nodes,
&new_known_peers,
&recreated_nodes,
)
}

Self::update_rack_count(&mut datacenters);

let keyspaces = metadata.keyspaces;
Expand Down Expand Up @@ -508,7 +542,14 @@ impl ClusterData {
let replica_translator = |uuid: Uuid| self.known_peers.get(&uuid).map(Arc::clone);

for (table, raw_tablet) in raw_tablets.into_iter() {
let tablet = Tablet::from_raw_tablet(&raw_tablet, replica_translator);
// Should we skip tablets that belong to keyspace not present in
// self.keyspaces? The keyspace could have been, without drivers knowledge:
// 1. Dropped - in which case we'll remove it's info soon (when refreshing
// topology) anyway.
// 2. Created - no harm in storing the info now.
//
// So I think we can safely skip checking keyspace presence.
let tablet = Tablet::from_raw_tablet(raw_tablet, replica_translator);
self.locator.tablets.add_tablet(table, tablet);
}
}
Expand Down
Loading

0 comments on commit b7892ee

Please sign in to comment.