From c0ceb621ac5410c9b7407845ce090461d9baacb2 Mon Sep 17 00:00:00 2001 From: nickelc Date: Mon, 6 Jun 2022 16:14:03 +0200 Subject: [PATCH] Introduce a separate data type for autocomplete interactions (#1941) The interaction payload for autocompletes is slightly different. It's missing the `resolved` data map and reports only for the current autocompleted option value the `focused` field and its value as a string. The focused option value is modeled as an enum variant of `CommandDataOptionValue`. ``` enum CommandDataOptionValue { Autocomplete { name: String, kind: CommandOptionType, value: String }, } ``` The focused option value can be retrieved via `AutocompleteData::focused_option()` from the `AutocompleteData::options` vector. --- .../interaction/application_command.rs | 10 - .../application/interaction/autocomplete.rs | 434 +++++++++++++++++- 2 files changed, 428 insertions(+), 16 deletions(-) diff --git a/src/model/application/interaction/application_command.rs b/src/model/application/interaction/application_command.rs index e9691674268..e94e5ecf548 100644 --- a/src/model/application/interaction/application_command.rs +++ b/src/model/application/interaction/application_command.rs @@ -610,10 +610,6 @@ pub struct CommandDataOption { /// The resolved object of the given `value`, if there is one. #[serde(default)] pub resolved: Option, - /// For `Autocomplete` Interactions this will be `true` if - /// this option is currently focused by the user. - #[serde(default)] - pub focused: bool, } impl<'de> Deserialize<'de> for CommandDataOption { @@ -641,18 +637,12 @@ impl<'de> Deserialize<'de> for CommandDataOption { .map_err(DeError::custom)? .unwrap_or_default(); - let focused = match map.get("focused") { - Some(value) => value.as_bool().ok_or_else(|| DeError::custom("expected bool"))?, - None => false, - }; - Ok(Self { name, value, kind, options, resolved: None, - focused, }) } } diff --git a/src/model/application/interaction/autocomplete.rs b/src/model/application/interaction/autocomplete.rs index 88c8c703ff8..9153e8ef2dd 100644 --- a/src/model/application/interaction/autocomplete.rs +++ b/src/model/application/interaction/autocomplete.rs @@ -1,5 +1,8 @@ -use serde::de::Error as DeError; -use serde::{Deserialize, Deserializer}; +use std::fmt; + +use serde::de::{Deserializer, Error as DeError, IgnoredAny, MapAccess}; +use serde::ser::{SerializeStruct, Serializer}; +use serde::{Deserialize, Serialize}; #[cfg(feature = "http")] use crate::builder::CreateAutocompleteResponse; @@ -9,12 +12,22 @@ use crate::internal::prelude::*; #[cfg(feature = "http")] use crate::json; use crate::json::prelude::*; -use crate::model::application::interaction::application_command::CommandData; +use crate::model::application::command::{CommandOptionType, CommandType}; #[cfg(feature = "http")] use crate::model::application::interaction::InteractionResponseType; use crate::model::application::interaction::InteractionType; use crate::model::guild::Member; -use crate::model::id::{ApplicationId, ChannelId, GuildId, InteractionId}; +use crate::model::id::{ + ApplicationId, + AttachmentId, + ChannelId, + CommandId, + GenericId, + GuildId, + InteractionId, + RoleId, + UserId, +}; use crate::model::user::User; use crate::model::Permissions; @@ -32,7 +45,7 @@ pub struct AutocompleteInteraction { #[serde(rename = "type")] pub kind: InteractionType, /// The data of the interaction which was triggered. - pub data: CommandData, + pub data: AutocompleteData, /// The guild Id this interaction was sent from, if there is one. #[serde(skip_serializing_if = "Option::is_none")] pub guild_id: Option, @@ -129,7 +142,7 @@ impl<'de> Deserialize<'de> for AutocompleteInteraction { let data = map .remove("data") .ok_or_else(|| DeError::custom("expected data")) - .and_then(CommandData::deserialize) + .and_then(AutocompleteData::deserialize) .map_err(DeError::custom)?; let guild_id = map @@ -201,3 +214,412 @@ impl<'de> Deserialize<'de> for AutocompleteInteraction { }) } } + +/// The autocomplete data payload. +#[derive(Clone, Debug, Deserialize, Serialize)] +pub struct AutocompleteData { + /// The Id of the invoked command. + pub id: CommandId, + /// The name of the invoked command. + pub name: String, + /// The application command type of the triggered application command. + #[serde(rename = "type")] + pub kind: CommandType, + /// The parameters and the given values. + #[serde(default)] + pub options: Vec, + /// The Id of the guild the command is registered to. + #[serde(skip_serializing_if = "Option::is_none")] + pub guild_id: Option, +} + +impl AutocompleteData { + /// Returns the focused option from [`AutocompleteData::options`]. + #[must_use] + pub fn focused_option(&self) -> Option> { + fn find_option(opts: &[CommandDataOption]) -> Option> { + for opt in opts { + match &opt.value { + CommandDataOptionValue::SubCommand(opts) + | CommandDataOptionValue::SubCommandGroup(opts) => { + let opt = find_option(&*opts); + if opt.is_some() { + return opt; + } + }, + CommandDataOptionValue::Autocomplete { + kind, + value, + } => { + return Some(FocusedOption { + name: &opt.name, + kind: *kind, + value, + }); + }, + _ => {}, + } + } + None + } + find_option(&*self.options) + } +} + +/// The focused option return by [`AutocompleteData::focused_option`]. +#[derive(Clone, Debug)] +pub struct FocusedOption<'a> { + pub name: &'a str, + pub kind: CommandOptionType, + pub value: &'a str, +} + +/// A set of a parameter and a value from the user. +/// +/// All options have names and an option can either be a parameter and input `value` or it can +/// denote a sub-command or group, in which case it will contain a top-level key and another vector +/// of `options`. +#[derive(Clone, Debug)] +#[non_exhaustive] +pub struct CommandDataOption { + /// The name of the parameter. + pub name: String, + /// The given value. + pub value: CommandDataOptionValue, +} + +impl CommandDataOption { + /// Returns the value type. + #[must_use] + pub fn kind(&self) -> CommandOptionType { + self.value.kind() + } +} + +impl<'de> Deserialize<'de> for CommandDataOption { + fn deserialize>(deserializer: D) -> StdResult { + #[derive(Deserialize)] + #[serde(field_identifier, rename_all = "snake_case")] + enum Field { + Name, + Type, + Value, + Options, + Focused, + Unknown(String), + } + + struct Visitor; + + impl<'de> serde::de::Visitor<'de> for Visitor { + type Value = CommandDataOption; + + fn expecting(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result { + formatter.write_str("CommandDataOption") + } + + fn visit_map>(self, mut map: A) -> StdResult { + let mut name = None; + let mut kind = None; + let mut value = None; + let mut options = None; + let mut focused = None; + + while let Some(key) = map.next_key()? { + match key { + Field::Name => { + if name.is_some() { + return Err(DeError::duplicate_field("name")); + } + name = Some(map.next_value()?); + }, + Field::Type => { + if kind.is_some() { + return Err(DeError::duplicate_field("type")); + } + kind = Some(map.next_value()?); + }, + Field::Value => { + if value.is_some() { + return Err(DeError::duplicate_field("value")); + } + value = Some(map.next_value::()?); + }, + Field::Options => { + if options.is_some() { + return Err(DeError::duplicate_field("options")); + } + options = Some(map.next_value()?); + }, + Field::Focused => { + if focused.is_some() { + return Err(DeError::duplicate_field("focused")); + } + focused = Some(map.next_value()?); + }, + Field::Unknown(_) => { + map.next_value::()?; + }, + } + } + + let name = name.ok_or_else(|| DeError::missing_field("name"))?; + let kind = kind.ok_or_else(|| DeError::missing_field("type"))?; + let focused = focused.unwrap_or_default(); + + if focused { + let value = value.ok_or_else(|| DeError::missing_field("value"))?; + let value = String::deserialize(value).map_err(DeError::custom)?; + return Ok(CommandDataOption { + name, + value: CommandDataOptionValue::Autocomplete { + kind, + value, + }, + }); + } + + let value = match kind { + CommandOptionType::Boolean => { + let value = value.ok_or_else(|| DeError::missing_field("value"))?; + let value = bool::deserialize(value).map_err(DeError::custom)?; + CommandDataOptionValue::Boolean(value) + }, + CommandOptionType::Integer => { + let value = value.ok_or_else(|| DeError::missing_field("value"))?; + let value = i64::deserialize(value).map_err(DeError::custom)?; + CommandDataOptionValue::Integer(value) + }, + CommandOptionType::Number => { + let value = value.ok_or_else(|| DeError::missing_field("value"))?; + let value = f64::deserialize(value).map_err(DeError::custom)?; + CommandDataOptionValue::Number(value) + }, + CommandOptionType::String => { + let value = value.ok_or_else(|| DeError::missing_field("value"))?; + let value = String::deserialize(value).map_err(DeError::custom)?; + CommandDataOptionValue::String(value) + }, + CommandOptionType::SubCommand => { + let options = options.ok_or_else(|| DeError::missing_field("options"))?; + CommandDataOptionValue::SubCommand(options) + }, + CommandOptionType::SubCommandGroup => { + let options = options.ok_or_else(|| DeError::missing_field("options"))?; + CommandDataOptionValue::SubCommandGroup(options) + }, + CommandOptionType::Attachment => { + let value = value.ok_or_else(|| DeError::missing_field("value"))?; + let value = AttachmentId::deserialize(value).map_err(DeError::custom)?; + CommandDataOptionValue::Attachment(value) + }, + CommandOptionType::Channel => { + let value = value.ok_or_else(|| DeError::missing_field("value"))?; + let value = ChannelId::deserialize(value).map_err(DeError::custom)?; + CommandDataOptionValue::Channel(value) + }, + CommandOptionType::Mentionable => { + let value = value.ok_or_else(|| DeError::missing_field("value"))?; + let value = GenericId::deserialize(value).map_err(DeError::custom)?; + CommandDataOptionValue::Mentionable(value) + }, + CommandOptionType::Role => { + let value = value.ok_or_else(|| DeError::missing_field("value"))?; + let value = RoleId::deserialize(value).map_err(DeError::custom)?; + CommandDataOptionValue::Role(value) + }, + CommandOptionType::User => { + let value = value.ok_or_else(|| DeError::missing_field("value"))?; + let value = UserId::deserialize(value).map_err(DeError::custom)?; + CommandDataOptionValue::User(value) + }, + CommandOptionType::Unknown => CommandDataOptionValue::Unknown, + }; + + Ok(CommandDataOption { + name, + value, + }) + } + } + + deserializer.deserialize_map(Visitor) + } +} + +impl Serialize for CommandDataOption { + fn serialize(&self, serializer: S) -> StdResult { + let (value_or_options, focused) = match &self.value { + CommandDataOptionValue::Autocomplete { + .. + } => (true, true), + CommandDataOptionValue::SubCommand(o) | CommandDataOptionValue::SubCommandGroup(o) => { + (!o.is_empty(), false) + }, + CommandDataOptionValue::Unknown => (false, false), + _ => (true, false), + }; + let len = 2 + usize::from(value_or_options) + usize::from(focused); + + let mut s = serializer.serialize_struct("CommandDataOption", len)?; + + s.serialize_field("name", &self.name)?; + s.serialize_field("type", &self.value.kind())?; + + match &self.value { + CommandDataOptionValue::Autocomplete { + value, .. + } => { + s.serialize_field("value", value)?; + }, + CommandDataOptionValue::Boolean(v) => s.serialize_field("value", v)?, + CommandDataOptionValue::Integer(v) => s.serialize_field("value", v)?, + CommandDataOptionValue::Number(v) => s.serialize_field("value", v)?, + CommandDataOptionValue::String(v) => s.serialize_field("value", v)?, + CommandDataOptionValue::Attachment(v) => s.serialize_field("value", v)?, + CommandDataOptionValue::Channel(v) => s.serialize_field("value", v)?, + CommandDataOptionValue::Mentionable(v) => s.serialize_field("value", v)?, + CommandDataOptionValue::Role(v) => s.serialize_field("value", v)?, + CommandDataOptionValue::User(v) => s.serialize_field("value", v)?, + CommandDataOptionValue::SubCommand(o) | CommandDataOptionValue::SubCommandGroup(o) => { + if !o.is_empty() { + s.serialize_field("options", o)?; + } + }, + _ => {}, + } + + if focused { + s.serialize_field("focused", &focused)?; + } + + s.end() + } +} + +/// The value of an [`CommandDataOption`]. +#[derive(Clone, Debug)] +#[non_exhaustive] +pub enum CommandDataOptionValue { + Autocomplete { kind: CommandOptionType, value: String }, + Boolean(bool), + Integer(i64), + Number(f64), + String(String), + SubCommand(Vec), + SubCommandGroup(Vec), + Attachment(AttachmentId), + Channel(ChannelId), + Mentionable(GenericId), + Role(RoleId), + User(UserId), + Unknown, +} + +impl CommandDataOptionValue { + /// Returns the value type. + #[must_use] + pub fn kind(&self) -> CommandOptionType { + match self { + Self::Autocomplete { + kind, .. + } => *kind, + Self::Boolean(_) => CommandOptionType::Boolean, + Self::Integer(_) => CommandOptionType::Integer, + Self::Number(_) => CommandOptionType::Number, + Self::String(_) => CommandOptionType::String, + Self::SubCommand(_) => CommandOptionType::SubCommand, + Self::SubCommandGroup(_) => CommandOptionType::SubCommandGroup, + Self::Attachment(_) => CommandOptionType::Attachment, + Self::Channel(_) => CommandOptionType::Channel, + Self::Mentionable(_) => CommandOptionType::Mentionable, + Self::Role(_) => CommandOptionType::Role, + Self::User(_) => CommandOptionType::User, + Self::Unknown => CommandOptionType::Unknown, + } + } + + /// If the value is a boolean, returns the associated f64. Returns None otherwise. + #[must_use] + pub fn as_bool(&self) -> Option { + match *self { + Self::Boolean(b) => Some(b), + _ => None, + } + } + + /// If the value is an integer, returns the associated f64. Returns None otherwise. + #[must_use] + pub fn as_i64(&self) -> Option { + match *self { + Self::Integer(v) => Some(v), + _ => None, + } + } + + /// If the value is a number, returns the associated f64. Returns None otherwise. + #[must_use] + pub fn as_f64(&self) -> Option { + match *self { + Self::Number(v) => Some(v), + _ => None, + } + } + + /// If the value is a string, returns the associated str. Returns None otherwise. + #[must_use] + pub fn as_str(&self) -> Option<&str> { + match self { + Self::String(s) => Some(s), + Self::Autocomplete { + value, .. + } => Some(value), + _ => None, + } + } + + /// If the value is an `AttachmentId`, returns the associated ID. Returns None otherwise. + #[must_use] + pub fn as_attachment_id(&self) -> Option { + match self { + Self::Attachment(id) => Some(*id), + _ => None, + } + } + + /// If the value is an `ChannelId`, returns the associated ID. Returns None otherwise. + #[must_use] + pub fn as_channel_id(&self) -> Option { + match self { + Self::Channel(id) => Some(*id), + _ => None, + } + } + + /// If the value is an `GenericId`, returns the associated ID. Returns None otherwise. + #[must_use] + pub fn as_mentionable(&self) -> Option { + match self { + Self::Mentionable(id) => Some(*id), + _ => None, + } + } + + /// If the value is an `UserId`, returns the associated ID. Returns None otherwise. + #[must_use] + pub fn as_user_id(&self) -> Option { + match self { + Self::User(id) => Some(*id), + _ => None, + } + } + + /// If the value is an `RoleId`, returns the associated ID. Returns None otherwise. + #[must_use] + pub fn as_role_id(&self) -> Option { + match self { + Self::Role(id) => Some(*id), + _ => None, + } + } +}