mod dump; pub mod error; mod store; use std::collections::{HashMap, HashSet}; use std::path::Path; use std::sync::Arc; use error::{AuthControllerError, Result}; use maplit::hashset; use meilisearch_types::index_uid_pattern::IndexUidPattern; use meilisearch_types::keys::{Action, CreateApiKey, Key, PatchApiKey}; use meilisearch_types::milli::update::Setting; use serde::{Deserialize, Serialize}; pub use store::open_auth_store_env; use store::{generate_key_as_hexa, HeedAuthStore}; use time::OffsetDateTime; use uuid::Uuid; #[derive(Clone)] pub struct AuthController { store: Arc, master_key: Option, } impl AuthController { pub fn new(db_path: impl AsRef, master_key: &Option) -> Result { let store = HeedAuthStore::new(db_path)?; if store.is_empty()? { generate_default_keys(&store)?; } Ok(Self { store: Arc::new(store), master_key: master_key.clone() }) } /// Return the size of the `AuthController` database in bytes. pub fn size(&self) -> Result { self.store.size() } pub fn create_key(&self, create_key: CreateApiKey) -> Result { match self.store.get_api_key(create_key.uid)? { Some(_) => Err(AuthControllerError::ApiKeyAlreadyExists(create_key.uid.to_string())), None => self.store.put_api_key(create_key.to_key()), } } pub fn update_key(&self, uid: Uuid, patch: PatchApiKey) -> Result { let mut key = self.get_key(uid)?; match patch.description { Setting::NotSet => (), description => key.description = description.set(), }; match patch.name { Setting::NotSet => (), name => key.name = name.set(), }; key.updated_at = OffsetDateTime::now_utc(); self.store.put_api_key(key) } pub fn get_key(&self, uid: Uuid) -> Result { self.store .get_api_key(uid)? .ok_or_else(|| AuthControllerError::ApiKeyNotFound(uid.to_string())) } pub fn get_optional_uid_from_encoded_key(&self, encoded_key: &[u8]) -> Result> { match &self.master_key { Some(master_key) => { self.store.get_uid_from_encoded_key(encoded_key, master_key.as_bytes()) } None => Ok(None), } } pub fn get_uid_from_encoded_key(&self, encoded_key: &str) -> Result { self.get_optional_uid_from_encoded_key(encoded_key.as_bytes())? .ok_or_else(|| AuthControllerError::ApiKeyNotFound(encoded_key.to_string())) } pub fn get_key_filters( &self, uid: Uuid, search_rules: Option, ) -> Result { let mut filters = AuthFilter::default(); let key = self.get_key(uid)?; filters.key_authorized_indexes = SearchRules::Set(key.indexes.into_iter().collect()); filters.search_rules = match search_rules { Some(search_rules) => search_rules, None => filters.key_authorized_indexes.clone(), }; filters.allow_index_creation = self.is_key_authorized(uid, Action::IndexesAdd, None)?; Ok(filters) } pub fn list_keys(&self) -> Result> { self.store.list_api_keys() } pub fn delete_key(&self, uid: Uuid) -> Result<()> { if self.store.delete_api_key(uid)? { Ok(()) } else { Err(AuthControllerError::ApiKeyNotFound(uid.to_string())) } } pub fn get_master_key(&self) -> Option<&String> { self.master_key.as_ref() } /// Generate a valid key from a key id using the current master key. /// Returns None if no master key has been set. pub fn generate_key(&self, uid: Uuid) -> Option { self.master_key.as_ref().map(|master_key| generate_key_as_hexa(uid, master_key.as_bytes())) } /// Check if the provided key is authorized to make a specific action /// without checking if the key is valid. pub fn is_key_authorized( &self, uid: Uuid, action: Action, index: Option<&str>, ) -> Result { match self .store // check if the key has access to all indexes. .get_expiration_date(uid, action, None)? .or(match index { // else check if the key has access to the requested index. Some(index) => self.store.get_expiration_date(uid, action, Some(index))?, // or to any index if no index has been requested. None => self.store.prefix_first_expiration_date(uid, action)?, }) { // check expiration date. Some(Some(exp)) => Ok(OffsetDateTime::now_utc() < exp), // no expiration date. Some(None) => Ok(true), // action or index forbidden. None => Ok(false), } } /// Delete all the keys in the DB. pub fn raw_delete_all_keys(&mut self) -> Result<()> { self.store.delete_all_keys() } /// Delete all the keys in the DB. pub fn raw_insert_key(&mut self, key: Key) -> Result<()> { self.store.put_api_key(key)?; Ok(()) } } pub struct AuthFilter { search_rules: SearchRules, key_authorized_indexes: SearchRules, pub allow_index_creation: bool, } impl Default for AuthFilter { fn default() -> Self { Self { search_rules: SearchRules::default(), key_authorized_indexes: SearchRules::default(), allow_index_creation: true, } } } impl AuthFilter { pub fn is_index_authorized(&self, index: &str) -> bool { self.key_authorized_indexes.is_index_authorized(index) && self.search_rules.is_index_authorized(index) } pub fn get_index_search_rules(&self, index: &str) -> Option { if !self.is_index_authorized(index) { return None; } self.search_rules.get_index_search_rules(index) } /// Return the list of indexes such that `self.is_index_authorized(index) == true`, /// or `None` if all indexes satisfy this condition. /// /// FIXME: this works only when there are no tenant tokens, otherwise it ignores the rules of the API key. /// /// It is better to use `is_index_authorized` when possible. pub fn authorized_indexes(&self) -> Option> { self.search_rules.authorized_indexes() } } /// Transparent wrapper around a list of allowed indexes with the search rules to apply for each. #[derive(Debug, Serialize, Deserialize, Clone)] #[serde(untagged)] pub enum SearchRules { Set(HashSet), Map(HashMap>), } impl Default for SearchRules { fn default() -> Self { Self::Set(hashset! { IndexUidPattern::all() }) } } impl SearchRules { fn is_index_authorized(&self, index: &str) -> bool { match self { Self::Set(set) => { set.contains("*") || set.contains(index) || set.iter().any(|pattern| pattern.matches_str(index)) } Self::Map(map) => { map.contains_key("*") || map.contains_key(index) || map.keys().any(|pattern| pattern.matches_str(index)) } } } fn get_index_search_rules(&self, index: &str) -> Option { match self { Self::Set(_) => { if self.is_index_authorized(index) { Some(IndexSearchRules::default()) } else { None } } Self::Map(map) => { // We must take the most retrictive rule of this index uid patterns set of rules. map.iter() .filter(|(pattern, _)| pattern.matches_str(index)) .max_by_key(|(pattern, _)| (pattern.is_exact(), pattern.len())) .and_then(|(_, rule)| rule.clone()) } } } /// Return the list of indexes such that `self.is_index_authorized(index) == true`, /// or `None` if all indexes satisfy this condition. fn authorized_indexes(&self) -> Option> { match self { SearchRules::Set(set) => { if set.contains("*") { None } else { Some(set.iter().cloned().collect()) } } SearchRules::Map(map) => { if map.contains_key("*") { None } else { Some(map.keys().cloned().collect()) } } } } } impl IntoIterator for SearchRules { type Item = (IndexUidPattern, IndexSearchRules); type IntoIter = Box>; fn into_iter(self) -> Self::IntoIter { match self { Self::Set(array) => { Box::new(array.into_iter().map(|i| (i, IndexSearchRules::default()))) } Self::Map(map) => { Box::new(map.into_iter().map(|(i, isr)| (i, isr.unwrap_or_default()))) } } } } /// Contains the rules to apply on the top of the search query for a specific index. /// /// filter: search filter to apply in addition to query filters. #[derive(Debug, Serialize, Deserialize, Default, Clone)] pub struct IndexSearchRules { pub filter: Option, } fn generate_default_keys(store: &HeedAuthStore) -> Result<()> { store.put_api_key(Key::default_admin())?; store.put_api_key(Key::default_search())?; Ok(()) } pub const MASTER_KEY_MIN_SIZE: usize = 16; const MASTER_KEY_GEN_SIZE: usize = 32; pub fn generate_master_key() -> String { use rand::rngs::OsRng; use rand::RngCore; // We need to use a cryptographically-secure source of randomness. That's why we're using the OsRng; https://crates.io/crates/getrandom let mut csprng = OsRng; let mut buf = vec![0; MASTER_KEY_GEN_SIZE]; csprng.fill_bytes(&mut buf); // let's encode the random bytes to base64 to make them human-readable and not too long. // We're using the URL_SAFE alphabet that will produce keys without =, / or other unusual characters. base64::encode_config(buf, base64::URL_SAFE_NO_PAD) }