spin_factor_key_value/runtime_config/
spin.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
//! Runtime configuration implementation used by Spin CLI.

use 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};

/// Defines the construction of a key value store from a serialized runtime config.
pub trait MakeKeyValueStore: 'static + Send + Sync {
    /// Unique type identifier for the store.
    const RUNTIME_CONFIG_TYPE: &'static str;
    /// Runtime configuration for the store.
    type RuntimeConfig: DeserializeOwned;
    /// The store manager for the store.
    type StoreManager: StoreManager;

    /// Creates a new store manager from the runtime configuration.
    fn make_store(&self, runtime_config: Self::RuntimeConfig)
        -> anyhow::Result<Self::StoreManager>;
}

/// A function that creates a store manager from a TOML table.
type StoreFromToml =
    Arc<dyn Fn(toml::Table) -> anyhow::Result<Arc<dyn StoreManager>> + Send + Sync>;

/// Creates a `StoreFromToml` function from a `MakeKeyValueStore` implementation.
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))
    })
}

/// Converts from toml based runtime configuration into a [`RuntimeConfig`].
///
/// The various store types (i.e., the "type" field in the toml field) are
/// registered with the resolver using `add_store_type`. The default store for a
/// label is registered using `add_default_store`.
#[derive(Default, Clone)]
pub struct RuntimeConfigResolver {
    /// A map of store types to a function that returns the appropriate store
    /// manager from runtime config TOML.
    store_types: HashMap<&'static str, StoreFromToml>,
    /// A map of default store configurations for a label.
    defaults: HashMap<&'static str, StoreConfig>,
}

impl RuntimeConfigResolver {
    /// Create a new RuntimeConfigResolver.
    pub fn new() -> Self {
        <Self as Default>::default()
    }

    /// Adds a default store configuration for a label.
    ///
    /// Users must ensure that the store type for `config` has been registered with
    /// the resolver using [`Self::register_store_type`].
    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(())
    }

    /// Registers a store type to the resolver.
    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(())
    }

    /// Resolves a toml table into a runtime config.
    ///
    /// The default stores are also added to the runtime config.
    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))
    }

    /// Given a [`StoreConfig`], returns a store manager.
    ///
    /// Errors if there is no [`MakeKeyValueStore`] registered for the store config's type
    /// or if the store manager cannot be created from the 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)?,
        })
    }
}