mirror of
https://github.com/meilisearch/MeiliSearch
synced 2024-12-29 16:01:40 +01:00
366 lines
13 KiB
Rust
366 lines
13 KiB
Rust
use std::convert::Infallible;
|
|
use std::hash::Hash;
|
|
use std::str::FromStr;
|
|
|
|
use deserr::{DeserializeError, Deserr, MergeWithError, ValuePointerRef};
|
|
use enum_iterator::Sequence;
|
|
use milli::update::Setting;
|
|
use serde::{Deserialize, Serialize};
|
|
use time::format_description::well_known::Rfc3339;
|
|
use time::macros::{format_description, time};
|
|
use time::{Date, OffsetDateTime, PrimitiveDateTime};
|
|
use uuid::Uuid;
|
|
|
|
use crate::deserr::{immutable_field_error, DeserrError, DeserrJsonError};
|
|
use crate::error::deserr_codes::*;
|
|
use crate::error::{Code, ErrorCode, ParseOffsetDateTimeError};
|
|
use crate::index_uid_pattern::{IndexUidPattern, IndexUidPatternFormatError};
|
|
|
|
pub type KeyId = Uuid;
|
|
|
|
impl<C: Default + ErrorCode> MergeWithError<IndexUidPatternFormatError> for DeserrJsonError<C> {
|
|
fn merge(
|
|
_self_: Option<Self>,
|
|
other: IndexUidPatternFormatError,
|
|
merge_location: deserr::ValuePointerRef,
|
|
) -> std::ops::ControlFlow<Self, Self> {
|
|
DeserrError::error::<Infallible>(
|
|
None,
|
|
deserr::ErrorKind::Unexpected { msg: other.to_string() },
|
|
merge_location,
|
|
)
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Deserr)]
|
|
#[deserr(error = DeserrJsonError, rename_all = camelCase, deny_unknown_fields)]
|
|
pub struct CreateApiKey {
|
|
#[deserr(default, error = DeserrJsonError<InvalidApiKeyDescription>)]
|
|
pub description: Option<String>,
|
|
#[deserr(default, error = DeserrJsonError<InvalidApiKeyName>)]
|
|
pub name: Option<String>,
|
|
#[deserr(default = Uuid::new_v4(), error = DeserrJsonError<InvalidApiKeyUid>, try_from(&String) = Uuid::from_str -> uuid::Error)]
|
|
pub uid: KeyId,
|
|
#[deserr(error = DeserrJsonError<InvalidApiKeyActions>, missing_field_error = DeserrJsonError::missing_api_key_actions)]
|
|
pub actions: Vec<Action>,
|
|
#[deserr(error = DeserrJsonError<InvalidApiKeyIndexes>, missing_field_error = DeserrJsonError::missing_api_key_indexes)]
|
|
pub indexes: Vec<IndexUidPattern>,
|
|
#[deserr(error = DeserrJsonError<InvalidApiKeyExpiresAt>, try_from(Option<String>) = parse_expiration_date -> ParseOffsetDateTimeError, missing_field_error = DeserrJsonError::missing_api_key_expires_at)]
|
|
pub expires_at: Option<OffsetDateTime>,
|
|
}
|
|
|
|
impl CreateApiKey {
|
|
pub fn to_key(self) -> Key {
|
|
let CreateApiKey { description, name, uid, actions, indexes, expires_at } = self;
|
|
let now = OffsetDateTime::now_utc();
|
|
Key {
|
|
description,
|
|
name,
|
|
uid,
|
|
actions,
|
|
indexes,
|
|
expires_at,
|
|
created_at: now,
|
|
updated_at: now,
|
|
}
|
|
}
|
|
}
|
|
|
|
fn deny_immutable_fields_api_key(
|
|
field: &str,
|
|
accepted: &[&str],
|
|
location: ValuePointerRef,
|
|
) -> DeserrJsonError {
|
|
match field {
|
|
"uid" => immutable_field_error(field, accepted, Code::ImmutableApiKeyUid),
|
|
"actions" => immutable_field_error(field, accepted, Code::ImmutableApiKeyActions),
|
|
"indexes" => immutable_field_error(field, accepted, Code::ImmutableApiKeyIndexes),
|
|
"expiresAt" => immutable_field_error(field, accepted, Code::ImmutableApiKeyExpiresAt),
|
|
"createdAt" => immutable_field_error(field, accepted, Code::ImmutableApiKeyCreatedAt),
|
|
"updatedAt" => immutable_field_error(field, accepted, Code::ImmutableApiKeyUpdatedAt),
|
|
_ => deserr::take_cf_content(DeserrJsonError::<BadRequest>::error::<Infallible>(
|
|
None,
|
|
deserr::ErrorKind::UnknownKey { key: field, accepted },
|
|
location,
|
|
)),
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Deserr)]
|
|
#[deserr(error = DeserrJsonError, rename_all = camelCase, deny_unknown_fields = deny_immutable_fields_api_key)]
|
|
pub struct PatchApiKey {
|
|
#[deserr(default, error = DeserrJsonError<InvalidApiKeyDescription>)]
|
|
pub description: Setting<String>,
|
|
#[deserr(default, error = DeserrJsonError<InvalidApiKeyName>)]
|
|
pub name: Setting<String>,
|
|
}
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Deserialize, Serialize)]
|
|
pub struct Key {
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
pub description: Option<String>,
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
pub name: Option<String>,
|
|
pub uid: KeyId,
|
|
pub actions: Vec<Action>,
|
|
pub indexes: Vec<IndexUidPattern>,
|
|
#[serde(with = "time::serde::rfc3339::option")]
|
|
pub expires_at: Option<OffsetDateTime>,
|
|
#[serde(with = "time::serde::rfc3339")]
|
|
pub created_at: OffsetDateTime,
|
|
#[serde(with = "time::serde::rfc3339")]
|
|
pub updated_at: OffsetDateTime,
|
|
}
|
|
|
|
impl Key {
|
|
pub fn default_admin() -> Self {
|
|
let now = OffsetDateTime::now_utc();
|
|
let uid = Uuid::new_v4();
|
|
Self {
|
|
name: Some("Default Admin API Key".to_string()),
|
|
description: Some("Use it for anything that is not a search operation. Caution! Do not expose it on a public frontend".to_string()),
|
|
uid,
|
|
actions: vec![Action::All],
|
|
indexes: vec![IndexUidPattern::all()],
|
|
expires_at: None,
|
|
created_at: now,
|
|
updated_at: now,
|
|
}
|
|
}
|
|
|
|
pub fn default_search() -> Self {
|
|
let now = OffsetDateTime::now_utc();
|
|
let uid = Uuid::new_v4();
|
|
Self {
|
|
name: Some("Default Search API Key".to_string()),
|
|
description: Some("Use it to search from the frontend".to_string()),
|
|
uid,
|
|
actions: vec![Action::Search],
|
|
indexes: vec![IndexUidPattern::all()],
|
|
expires_at: None,
|
|
created_at: now,
|
|
updated_at: now,
|
|
}
|
|
}
|
|
}
|
|
|
|
fn parse_expiration_date(
|
|
string: Option<String>,
|
|
) -> std::result::Result<Option<OffsetDateTime>, ParseOffsetDateTimeError> {
|
|
let Some(string) = string else {
|
|
return Ok(None)
|
|
};
|
|
let datetime = if let Ok(datetime) = OffsetDateTime::parse(&string, &Rfc3339) {
|
|
datetime
|
|
} else if let Ok(primitive_datetime) = PrimitiveDateTime::parse(
|
|
&string,
|
|
format_description!(
|
|
"[year repr:full base:calendar]-[month repr:numerical]-[day]T[hour]:[minute]:[second]"
|
|
),
|
|
) {
|
|
primitive_datetime.assume_utc()
|
|
} else if let Ok(primitive_datetime) = PrimitiveDateTime::parse(
|
|
&string,
|
|
format_description!(
|
|
"[year repr:full base:calendar]-[month repr:numerical]-[day] [hour]:[minute]:[second]"
|
|
),
|
|
) {
|
|
primitive_datetime.assume_utc()
|
|
} else if let Ok(date) = Date::parse(
|
|
&string,
|
|
format_description!("[year repr:full base:calendar]-[month repr:numerical]-[day]"),
|
|
) {
|
|
PrimitiveDateTime::new(date, time!(00:00)).assume_utc()
|
|
} else {
|
|
return Err(ParseOffsetDateTimeError(string));
|
|
};
|
|
if datetime > OffsetDateTime::now_utc() {
|
|
Ok(Some(datetime))
|
|
} else {
|
|
Err(ParseOffsetDateTimeError(string))
|
|
}
|
|
}
|
|
|
|
#[derive(Copy, Clone, Serialize, Deserialize, Debug, Eq, PartialEq, Hash, Sequence, Deserr)]
|
|
#[repr(u8)]
|
|
pub enum Action {
|
|
#[serde(rename = "*")]
|
|
#[deserr(rename = "*")]
|
|
All = 0,
|
|
#[serde(rename = "search")]
|
|
#[deserr(rename = "search")]
|
|
Search,
|
|
#[serde(rename = "documents.*")]
|
|
#[deserr(rename = "documents.*")]
|
|
DocumentsAll,
|
|
#[serde(rename = "documents.add")]
|
|
#[deserr(rename = "documents.add")]
|
|
DocumentsAdd,
|
|
#[serde(rename = "documents.get")]
|
|
#[deserr(rename = "documents.get")]
|
|
DocumentsGet,
|
|
#[serde(rename = "documents.delete")]
|
|
#[deserr(rename = "documents.delete")]
|
|
DocumentsDelete,
|
|
#[serde(rename = "indexes.*")]
|
|
#[deserr(rename = "indexes.*")]
|
|
IndexesAll,
|
|
#[serde(rename = "indexes.create")]
|
|
#[deserr(rename = "indexes.create")]
|
|
IndexesAdd,
|
|
#[serde(rename = "indexes.get")]
|
|
#[deserr(rename = "indexes.get")]
|
|
IndexesGet,
|
|
#[serde(rename = "indexes.update")]
|
|
#[deserr(rename = "indexes.update")]
|
|
IndexesUpdate,
|
|
#[serde(rename = "indexes.delete")]
|
|
#[deserr(rename = "indexes.delete")]
|
|
IndexesDelete,
|
|
#[serde(rename = "indexes.swap")]
|
|
#[deserr(rename = "indexes.swap")]
|
|
IndexesSwap,
|
|
#[serde(rename = "tasks.*")]
|
|
#[deserr(rename = "tasks.*")]
|
|
TasksAll,
|
|
#[serde(rename = "tasks.cancel")]
|
|
#[deserr(rename = "tasks.cancel")]
|
|
TasksCancel,
|
|
#[serde(rename = "tasks.delete")]
|
|
#[deserr(rename = "tasks.delete")]
|
|
TasksDelete,
|
|
#[serde(rename = "tasks.get")]
|
|
#[deserr(rename = "tasks.get")]
|
|
TasksGet,
|
|
#[serde(rename = "settings.*")]
|
|
#[deserr(rename = "settings.*")]
|
|
SettingsAll,
|
|
#[serde(rename = "settings.get")]
|
|
#[deserr(rename = "settings.get")]
|
|
SettingsGet,
|
|
#[serde(rename = "settings.update")]
|
|
#[deserr(rename = "settings.update")]
|
|
SettingsUpdate,
|
|
#[serde(rename = "stats.*")]
|
|
#[deserr(rename = "stats.*")]
|
|
StatsAll,
|
|
#[serde(rename = "stats.get")]
|
|
#[deserr(rename = "stats.get")]
|
|
StatsGet,
|
|
#[serde(rename = "metrics.*")]
|
|
#[deserr(rename = "metrics.*")]
|
|
MetricsAll,
|
|
#[serde(rename = "metrics.get")]
|
|
#[deserr(rename = "metrics.get")]
|
|
MetricsGet,
|
|
#[serde(rename = "dumps.*")]
|
|
#[deserr(rename = "dumps.*")]
|
|
DumpsAll,
|
|
#[serde(rename = "dumps.create")]
|
|
#[deserr(rename = "dumps.create")]
|
|
DumpsCreate,
|
|
#[serde(rename = "version")]
|
|
#[deserr(rename = "version")]
|
|
Version,
|
|
#[serde(rename = "keys.create")]
|
|
#[deserr(rename = "keys.create")]
|
|
KeysAdd,
|
|
#[serde(rename = "keys.get")]
|
|
#[deserr(rename = "keys.get")]
|
|
KeysGet,
|
|
#[serde(rename = "keys.update")]
|
|
#[deserr(rename = "keys.update")]
|
|
KeysUpdate,
|
|
#[serde(rename = "keys.delete")]
|
|
#[deserr(rename = "keys.delete")]
|
|
KeysDelete,
|
|
#[serde(rename = "experimental.get")]
|
|
#[deserr(rename = "experimental.get")]
|
|
ExperimentalFeaturesGet,
|
|
#[serde(rename = "experimental.update")]
|
|
#[deserr(rename = "experimental.update")]
|
|
ExperimentalFeaturesUpdate,
|
|
}
|
|
|
|
impl Action {
|
|
pub const fn from_repr(repr: u8) -> Option<Self> {
|
|
use actions::*;
|
|
match repr {
|
|
ALL => Some(Self::All),
|
|
SEARCH => Some(Self::Search),
|
|
DOCUMENTS_ALL => Some(Self::DocumentsAll),
|
|
DOCUMENTS_ADD => Some(Self::DocumentsAdd),
|
|
DOCUMENTS_GET => Some(Self::DocumentsGet),
|
|
DOCUMENTS_DELETE => Some(Self::DocumentsDelete),
|
|
INDEXES_ALL => Some(Self::IndexesAll),
|
|
INDEXES_CREATE => Some(Self::IndexesAdd),
|
|
INDEXES_GET => Some(Self::IndexesGet),
|
|
INDEXES_UPDATE => Some(Self::IndexesUpdate),
|
|
INDEXES_DELETE => Some(Self::IndexesDelete),
|
|
INDEXES_SWAP => Some(Self::IndexesSwap),
|
|
TASKS_ALL => Some(Self::TasksAll),
|
|
TASKS_CANCEL => Some(Self::TasksCancel),
|
|
TASKS_DELETE => Some(Self::TasksDelete),
|
|
TASKS_GET => Some(Self::TasksGet),
|
|
SETTINGS_ALL => Some(Self::SettingsAll),
|
|
SETTINGS_GET => Some(Self::SettingsGet),
|
|
SETTINGS_UPDATE => Some(Self::SettingsUpdate),
|
|
STATS_ALL => Some(Self::StatsAll),
|
|
STATS_GET => Some(Self::StatsGet),
|
|
METRICS_ALL => Some(Self::MetricsAll),
|
|
METRICS_GET => Some(Self::MetricsGet),
|
|
DUMPS_ALL => Some(Self::DumpsAll),
|
|
DUMPS_CREATE => Some(Self::DumpsCreate),
|
|
VERSION => Some(Self::Version),
|
|
KEYS_CREATE => Some(Self::KeysAdd),
|
|
KEYS_GET => Some(Self::KeysGet),
|
|
KEYS_UPDATE => Some(Self::KeysUpdate),
|
|
KEYS_DELETE => Some(Self::KeysDelete),
|
|
EXPERIMENTAL_FEATURES_GET => Some(Self::ExperimentalFeaturesGet),
|
|
EXPERIMENTAL_FEATURES_UPDATE => Some(Self::ExperimentalFeaturesUpdate),
|
|
_otherwise => None,
|
|
}
|
|
}
|
|
|
|
pub const fn repr(&self) -> u8 {
|
|
*self as u8
|
|
}
|
|
}
|
|
|
|
pub mod actions {
|
|
use super::Action::*;
|
|
|
|
pub(crate) const ALL: u8 = All.repr();
|
|
pub const SEARCH: u8 = Search.repr();
|
|
pub const DOCUMENTS_ALL: u8 = DocumentsAll.repr();
|
|
pub const DOCUMENTS_ADD: u8 = DocumentsAdd.repr();
|
|
pub const DOCUMENTS_GET: u8 = DocumentsGet.repr();
|
|
pub const DOCUMENTS_DELETE: u8 = DocumentsDelete.repr();
|
|
pub const INDEXES_ALL: u8 = IndexesAll.repr();
|
|
pub const INDEXES_CREATE: u8 = IndexesAdd.repr();
|
|
pub const INDEXES_GET: u8 = IndexesGet.repr();
|
|
pub const INDEXES_UPDATE: u8 = IndexesUpdate.repr();
|
|
pub const INDEXES_DELETE: u8 = IndexesDelete.repr();
|
|
pub const INDEXES_SWAP: u8 = IndexesSwap.repr();
|
|
pub const TASKS_ALL: u8 = TasksAll.repr();
|
|
pub const TASKS_CANCEL: u8 = TasksCancel.repr();
|
|
pub const TASKS_DELETE: u8 = TasksDelete.repr();
|
|
pub const TASKS_GET: u8 = TasksGet.repr();
|
|
pub const SETTINGS_ALL: u8 = SettingsAll.repr();
|
|
pub const SETTINGS_GET: u8 = SettingsGet.repr();
|
|
pub const SETTINGS_UPDATE: u8 = SettingsUpdate.repr();
|
|
pub const STATS_ALL: u8 = StatsAll.repr();
|
|
pub const STATS_GET: u8 = StatsGet.repr();
|
|
pub const METRICS_ALL: u8 = MetricsAll.repr();
|
|
pub const METRICS_GET: u8 = MetricsGet.repr();
|
|
pub const DUMPS_ALL: u8 = DumpsAll.repr();
|
|
pub const DUMPS_CREATE: u8 = DumpsCreate.repr();
|
|
pub const VERSION: u8 = Version.repr();
|
|
pub const KEYS_CREATE: u8 = KeysAdd.repr();
|
|
pub const KEYS_GET: u8 = KeysGet.repr();
|
|
pub const KEYS_UPDATE: u8 = KeysUpdate.repr();
|
|
pub const KEYS_DELETE: u8 = KeysDelete.repr();
|
|
pub const EXPERIMENTAL_FEATURES_GET: u8 = ExperimentalFeaturesGet.repr();
|
|
pub const EXPERIMENTAL_FEATURES_UPDATE: u8 = ExperimentalFeaturesUpdate.repr();
|
|
}
|