feat: instrumented namespace cache

Decorates the NamespaceCache with a set of cache get hit/miss counters,
and put insert/update counters to expose cache behaviour.
pull/24376/head
Dom Dwyer 2022-02-09 16:45:21 +00:00
parent a3e15ade62
commit 92fe507e52
4 changed files with 101 additions and 8 deletions

View File

@ -18,7 +18,7 @@ use data_types::database_rules::{PartitionTemplate, TemplatePart};
use observability_deps::tracing::*; use observability_deps::tracing::*;
use router2::{ use router2::{
dml_handlers::{NamespaceAutocreation, Partitioner, SchemaValidator, ShardedWriteBuffer}, dml_handlers::{NamespaceAutocreation, Partitioner, SchemaValidator, ShardedWriteBuffer},
namespace_cache::{MemoryNamespaceCache, ShardedCache}, namespace_cache::{metrics::InstrumentedCache, MemoryNamespaceCache, ShardedCache},
sequencer::Sequencer, sequencer::Sequencer,
server::{http::HttpDelegate, RouterServer}, server::{http::HttpDelegate, RouterServer},
sharder::JumpHash, sharder::JumpHash,
@ -94,10 +94,14 @@ pub async fn command(config: Config) -> Result<()> {
) )
.await?; .await?;
// Initialise a namespace cache to be shared with the schema validator, and // Initialise an instrumented namespace cache to be shared with the schema
// namespace auto-creator. // validator, and namespace auto-creator that reports cache hit/miss/update
let ns_cache = Arc::new(ShardedCache::new( // metrics.
iter::repeat_with(|| Arc::new(MemoryNamespaceCache::default())).take(10), let ns_cache = Arc::new(InstrumentedCache::new(
Arc::new(ShardedCache::new(
iter::repeat_with(|| Arc::new(MemoryNamespaceCache::default())).take(10),
)),
Arc::clone(&metrics),
)); ));
// Add the schema validator layer. // Add the schema validator layer.
let handler_stack = let handler_stack =

View File

@ -12,7 +12,7 @@ use observability_deps::tracing::*;
use thiserror::Error; use thiserror::Error;
use trace::ctx::SpanContext; use trace::ctx::SpanContext;
use crate::namespace_cache::{MemoryNamespaceCache, NamespaceCache}; use crate::namespace_cache::{metrics::InstrumentedCache, MemoryNamespaceCache, NamespaceCache};
use super::{DmlError, DmlHandler, Partitioned}; use super::{DmlError, DmlHandler, Partitioned};
@ -82,7 +82,7 @@ pub enum SchemaError {
/// ///
/// [#3573]: https://github.com/influxdata/influxdb_iox/issues/3573 /// [#3573]: https://github.com/influxdata/influxdb_iox/issues/3573
#[derive(Debug)] #[derive(Debug)]
pub struct SchemaValidator<D, C = Arc<MemoryNamespaceCache>> { pub struct SchemaValidator<D, C = Arc<InstrumentedCache<MemoryNamespaceCache>>> {
inner: D, inner: D,
catalog: Arc<dyn Catalog>, catalog: Arc<dyn Catalog>,
@ -266,7 +266,10 @@ mod tests {
catalog catalog
} }
fn assert_cache<D>(handler: &SchemaValidator<D>, table: &str, col: &str, want: ColumnType) { fn assert_cache<D, C>(handler: &SchemaValidator<D, C>, table: &str, col: &str, want: ColumnType)
where
C: NamespaceCache,
{
// The cache should be populated. // The cache should be populated.
let ns = handler let ns = handler
.cache .cache

View File

@ -6,6 +6,8 @@ pub use memory::*;
mod sharded_cache; mod sharded_cache;
pub use sharded_cache::*; pub use sharded_cache::*;
pub mod metrics;
use std::{fmt::Debug, sync::Arc}; use std::{fmt::Debug, sync::Arc};
use data_types::DatabaseName; use data_types::DatabaseName;

View File

@ -0,0 +1,84 @@
//! Metric instrumentation for a [`NamespaceCache`] implementation.
use std::sync::Arc;
use data_types::DatabaseName;
use iox_catalog::interface::NamespaceSchema;
use metric::{Metric, U64Counter};
use super::NamespaceCache;
/// An [`InstrumentedCache`] decorates a [`NamespaceCache`] with cache read
/// hit/miss and cache put insert/update metrics.
#[derive(Debug)]
pub struct InstrumentedCache<T> {
inner: T,
/// A cache read hit
get_hit_counter: U64Counter,
/// A cache read miss
get_miss_counter: U64Counter,
/// A cache put for a namespace that did not previously exist.
put_insert_counter: U64Counter,
/// A cache put replacing a namespace that previously had a cache entry.
put_update_counter: U64Counter,
}
impl<T> InstrumentedCache<T> {
/// Instrument `T`, recording cache operations to `registry`.
pub fn new(inner: T, registry: Arc<metric::Registry>) -> Self {
let get_counter: Metric<U64Counter> =
registry.register_metric("namespace_cache_get_count", "cache read requests");
let get_hit_counter = get_counter.recorder(&[("result", "hit")]);
let get_miss_counter = get_counter.recorder(&[("result", "miss")]);
let put_counter: Metric<U64Counter> =
registry.register_metric("namespace_cache_put_count", "cache put requests");
let put_insert_counter = put_counter.recorder(&[("op", "insert")]);
let put_update_counter = put_counter.recorder(&[("op", "update")]);
Self {
inner,
get_hit_counter,
get_miss_counter,
put_insert_counter,
put_update_counter,
}
}
}
impl<T> NamespaceCache for Arc<InstrumentedCache<T>>
where
T: NamespaceCache,
{
fn get_schema(&self, namespace: &DatabaseName<'_>) -> Option<Arc<NamespaceSchema>> {
match self.inner.get_schema(namespace) {
Some(v) => {
self.get_hit_counter.inc(1);
Some(v)
}
None => {
self.get_miss_counter.inc(1);
None
}
}
}
fn put_schema(
&self,
namespace: DatabaseName<'static>,
schema: impl Into<Arc<NamespaceSchema>>,
) -> Option<Arc<NamespaceSchema>> {
match self.inner.put_schema(namespace, schema) {
Some(v) => {
self.put_update_counter.inc(1);
Some(v)
}
None => {
self.put_insert_counter.inc(1);
None
}
}
}
}