spin_factor_key_value/runtime_config/
spin.rsuse crate::{RuntimeConfig, StoreManager};
use anyhow::Context as _;
use serde::de::DeserializeOwned;
use serde::{Deserialize, Serialize};
use spin_factors::runtime_config::toml::GetTomlValue;
use std::{collections::HashMap, sync::Arc};
pub trait MakeKeyValueStore: 'static + Send + Sync {
const RUNTIME_CONFIG_TYPE: &'static str;
type RuntimeConfig: DeserializeOwned;
type StoreManager: StoreManager;
fn make_store(&self, runtime_config: Self::RuntimeConfig)
-> anyhow::Result<Self::StoreManager>;
}
type StoreFromToml =
Arc<dyn Fn(toml::Table) -> anyhow::Result<Arc<dyn StoreManager>> + Send + Sync>;
fn store_from_toml_fn<T: MakeKeyValueStore>(provider_type: T) -> StoreFromToml {
Arc::new(move |table| {
let runtime_config: T::RuntimeConfig = table
.try_into()
.context("could not parse key-value runtime config")?;
let provider = provider_type
.make_store(runtime_config)
.context("could not make key-value store from runtime config")?;
Ok(Arc::new(provider))
})
}
#[derive(Default, Clone)]
pub struct RuntimeConfigResolver {
store_types: HashMap<&'static str, StoreFromToml>,
defaults: HashMap<&'static str, StoreConfig>,
}
impl RuntimeConfigResolver {
pub fn new() -> Self {
<Self as Default>::default()
}
pub fn add_default_store<T>(
&mut self,
label: &'static str,
config: T::RuntimeConfig,
) -> anyhow::Result<()>
where
T: MakeKeyValueStore,
T::RuntimeConfig: Serialize,
{
self.defaults.insert(
label,
StoreConfig::new(T::RUNTIME_CONFIG_TYPE.to_owned(), config)?,
);
Ok(())
}
pub fn register_store_type<T: MakeKeyValueStore>(
&mut self,
store_type: T,
) -> anyhow::Result<()> {
if self
.store_types
.insert(T::RUNTIME_CONFIG_TYPE, store_from_toml_fn(store_type))
.is_some()
{
anyhow::bail!(
"duplicate key value store type {:?}",
T::RUNTIME_CONFIG_TYPE
);
}
Ok(())
}
pub fn resolve(&self, table: Option<&impl GetTomlValue>) -> anyhow::Result<RuntimeConfig> {
let mut runtime_config = self.resolve_from_toml(table)?.unwrap_or_default();
for (&label, config) in &self.defaults {
if !runtime_config.store_managers.contains_key(label) {
let store_manager = self
.store_manager_from_config(config.clone())
.with_context(|| {
format!("could not configure key-value store with label '{label}'")
})?;
runtime_config.add_store_manager(label.to_owned(), store_manager);
}
}
Ok(runtime_config)
}
fn resolve_from_toml(
&self,
table: Option<&impl GetTomlValue>,
) -> anyhow::Result<Option<RuntimeConfig>> {
let Some(table) = table.and_then(|t| t.get("key_value_store")) else {
return Ok(None);
};
let table: HashMap<String, StoreConfig> = table.clone().try_into()?;
let mut runtime_config = RuntimeConfig::default();
for (label, config) in table {
let store_manager = self.store_manager_from_config(config).with_context(|| {
format!("could not configure key-value store with label '{label}'")
})?;
runtime_config.add_store_manager(label.clone(), store_manager);
}
Ok(Some(runtime_config))
}
fn store_manager_from_config(
&self,
config: StoreConfig,
) -> anyhow::Result<Arc<dyn StoreManager>> {
let config_type = config.type_.as_str();
let maker = self.store_types.get(config_type).with_context(|| {
format!("the store type '{config_type}' was not registered with the config resolver")
})?;
maker(config.config)
}
}
#[derive(Deserialize, Clone)]
pub struct StoreConfig {
#[serde(rename = "type")]
pub type_: String,
#[serde(flatten)]
pub config: toml::Table,
}
impl StoreConfig {
pub fn new<T>(type_: String, config: T) -> anyhow::Result<Self>
where
T: Serialize,
{
Ok(Self {
type_,
config: toml::value::Table::try_from(config)?,
})
}
}