Skip to content

Commit

Permalink
Rework manual deserialization implementations (serenity-rs#1938, sere…
Browse files Browse the repository at this point in the history
  • Loading branch information
GnomedDev authored and arqunis committed Oct 24, 2023
1 parent a4181d0 commit 464ed63
Show file tree
Hide file tree
Showing 13 changed files with 290 additions and 1,226 deletions.
46 changes: 18 additions & 28 deletions src/model/application/component.rs
Original file line number Diff line number Diff line change
@@ -1,8 +1,10 @@
use serde::de::{Deserialize, Deserializer, Error as DeError};
use serde::ser::{Serialize, Serializer};

use crate::json::{from_value, JsonMap, Value};
use crate::internal::prelude::*;
use crate::json::from_value;
use crate::model::channel::ReactionType;
use crate::model::utils::deserialize_val;

/// The type of a component
#[derive(Copy, Clone, Debug, Hash, Eq, PartialEq, PartialOrd, Ord)]
Expand Down Expand Up @@ -44,39 +46,27 @@ pub enum ActionRowComponent {
}

impl<'de> Deserialize<'de> for ActionRowComponent {
fn deserialize<D: Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
let map = JsonMap::deserialize(deserializer)?;

let kind = map
.get("type")
.ok_or_else(|| DeError::custom("expected type"))
.and_then(ComponentType::deserialize)
.map_err(DeError::custom)?;

match kind {
ComponentType::Button => from_value::<Button>(Value::from(map))
.map(ActionRowComponent::Button)
.map_err(DeError::custom),
ComponentType::SelectMenu => from_value::<SelectMenu>(Value::from(map))
.map(ActionRowComponent::SelectMenu)
.map_err(DeError::custom),
ComponentType::InputText => from_value::<InputText>(Value::from(map))
.map(ActionRowComponent::InputText)
.map_err(DeError::custom),
_ => Err(DeError::custom("Unknown component type")),
fn deserialize<D: Deserializer<'de>>(deserializer: D) -> std::result::Result<Self, D::Error> {
let value = Value::deserialize(deserializer)?;
let map = value.as_object().ok_or_else(|| DeError::custom("expected JsonMap"))?;

let raw_kind = map.get("type").ok_or_else(|| DeError::missing_field("type"))?;
match deserialize_val(raw_kind.clone())? {
ComponentType::Button => from_value(value).map(ActionRowComponent::Button),
ComponentType::InputText => from_value(value).map(ActionRowComponent::InputText),
ComponentType::SelectMenu => from_value(value).map(ActionRowComponent::SelectMenu),
_ => return Err(DeError::custom("Unknown component type")),
}
.map_err(DeError::custom)
}
}

impl Serialize for ActionRowComponent {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
fn serialize<S: Serializer>(&self, serializer: S) -> std::result::Result<S::Ok, S::Error> {
match self {
Self::Button(c) => Button::serialize(c, serializer),
Self::SelectMenu(c) => SelectMenu::serialize(c, serializer),
Self::InputText(c) => InputText::serialize(c, serializer),
Self::Button(c) => c.serialize(serializer),
Self::InputText(c) => c.serialize(serializer),
Self::SelectMenu(c) => c.serialize(serializer),
}
}
}
Expand Down
213 changes: 30 additions & 183 deletions src/model/application/interaction/application_command.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,11 +14,10 @@ use crate::http::Http;
use crate::internal::prelude::*;
#[cfg(feature = "http")]
use crate::json;
use crate::json::prelude::*;
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::application::interaction::{add_guild_id_to_resolved, InteractionType};
use crate::model::channel::{Attachment, Message, PartialChannel};
use crate::model::guild::{Member, PartialMember, Role};
use crate::model::id::{
Expand All @@ -34,7 +33,11 @@ use crate::model::id::{
UserId,
};
use crate::model::user::User;
use crate::model::utils::deserialize_options_with_resolved;
use crate::model::utils::{
deserialize_options_with_resolved,
remove_from_map,
remove_from_map_opt,
};
use crate::model::Permissions;

/// An interaction when a user invokes a slash command.
Expand Down Expand Up @@ -330,119 +333,31 @@ impl<'de> Deserialize<'de> for ApplicationCommandInteraction {
fn deserialize<D: Deserializer<'de>>(deserializer: D) -> StdResult<Self, D::Error> {
let mut map = JsonMap::deserialize(deserializer)?;

let id = map.get("guild_id").and_then(Value::as_str).and_then(|x| x.parse::<u64>().ok());

if let Some(guild_id) = id {
if let Some(member) = map.get_mut("member").and_then(Value::as_object_mut) {
member.insert("guild_id".to_string(), from_number(guild_id));
}

if let Some(data) = map.get_mut("data") {
if let Some(resolved) = data.get_mut("resolved") {
if let Some(roles) = resolved.get_mut("roles") {
if let Some(values) = roles.as_object_mut() {
for value in values.values_mut() {
value.as_object_mut().expect("couldn't deserialize").insert(
"guild_id".to_string(),
Value::from(guild_id.to_string()),
);
}
}
}
}
}
}

let id = map
.remove("id")
.ok_or_else(|| DeError::custom("expected id"))
.and_then(InteractionId::deserialize)
.map_err(DeError::custom)?;

let application_id = map
.remove("application_id")
.ok_or_else(|| DeError::custom("expected application id"))
.and_then(ApplicationId::deserialize)
.map_err(DeError::custom)?;

let kind = map
.remove("type")
.ok_or_else(|| DeError::custom("expected type"))
.and_then(InteractionType::deserialize)
.map_err(DeError::custom)?;

let data = map
.remove("data")
.ok_or_else(|| DeError::custom("expected data"))
.and_then(CommandData::deserialize)
.map_err(DeError::custom)?;

let guild_id = map
.remove("guild_id")
.map(GuildId::deserialize)
.transpose()
.map_err(DeError::custom)?;

let channel_id = map
.remove("channel_id")
.ok_or_else(|| DeError::custom("expected channel_id"))
.and_then(ChannelId::deserialize)
.map_err(DeError::custom)?;
let guild_id = remove_from_map_opt::<GuildId, _>(&mut map, "guild_id")?;

let member =
map.remove("member").map(Member::deserialize).transpose().map_err(DeError::custom)?;

let user =
map.remove("user").map(User::deserialize).transpose().map_err(DeError::custom)?;
if let Some(guild_id) = guild_id {
add_guild_id_to_resolved(&mut map, guild_id);
}

let user = user
let member = remove_from_map_opt::<Member, _>(&mut map, "member")?;
let user = remove_from_map_opt(&mut map, "user")?
.or_else(|| member.as_ref().map(|m| m.user.clone()))
.ok_or_else(|| DeError::custom("expected user or member"))?;

let token = map
.remove("token")
.ok_or_else(|| DeError::custom("expected token"))
.and_then(String::deserialize)
.map_err(DeError::custom)?;

let version = map
.remove("version")
.ok_or_else(|| DeError::custom("expected version"))
.and_then(u8::deserialize)
.map_err(DeError::custom)?;

let app_permissions = map
.remove("app_permissions")
.map(Permissions::deserialize)
.transpose()
.map_err(DeError::custom)?;

let guild_locale = map
.remove("guild_locale")
.map(String::deserialize)
.transpose()
.map_err(DeError::custom)?;

let locale = map
.remove("locale")
.ok_or_else(|| DeError::custom("expected locale"))
.and_then(String::deserialize)
.map_err(DeError::custom)?;

Ok(Self {
id,
application_id,
kind,
data,
guild_id,
channel_id,
member,
user,
token,
version,
app_permissions,
locale,
guild_locale,
id: remove_from_map(&mut map, "id")?,
application_id: remove_from_map(&mut map, "application_id")?,
kind: remove_from_map(&mut map, "type")?,
data: remove_from_map(&mut map, "data")?,
channel_id: remove_from_map(&mut map, "channel_id")?,
token: remove_from_map(&mut map, "token")?,
version: remove_from_map(&mut map, "version")?,
app_permissions: remove_from_map_opt(&mut map, "app_permissions")?,
locale: remove_from_map(&mut map, "locale")?,
guild_locale: remove_from_map_opt(&mut map, "guild_locale")?,
})
}
}
Expand Down Expand Up @@ -510,24 +425,7 @@ impl<'de> Deserialize<'de> for CommandData {
fn deserialize<D: Deserializer<'de>>(deserializer: D) -> StdResult<Self, D::Error> {
let mut map = JsonMap::deserialize(deserializer)?;

let name = map
.remove("name")
.ok_or_else(|| DeError::custom("expected value"))
.and_then(String::deserialize)
.map_err(DeError::custom)?;

let id = map
.remove("id")
.ok_or_else(|| DeError::custom("expected value"))
.and_then(CommandId::deserialize)
.map_err(DeError::custom)?;

let resolved = map
.remove("resolved")
.map(CommandDataResolved::deserialize)
.transpose()
.map_err(DeError::custom)?
.unwrap_or_default();
let resolved = remove_from_map_opt(&mut map, "resolved")?.unwrap_or_default();

let options = map
.remove("options")
Expand All @@ -536,30 +434,14 @@ impl<'de> Deserialize<'de> for CommandData {
.map_err(DeError::custom)?
.unwrap_or_default();

let kind = map
.remove("type")
.ok_or_else(|| DeError::custom("expected type"))
.and_then(CommandType::deserialize)
.map_err(DeError::custom)?;

let guild_id = match map.remove("guild_id") {
Some(id) => Option::<GuildId>::deserialize(id).map_err(DeError::custom)?,
None => None,
};

let target_id = match map.remove("target_id") {
Some(id) => Option::<TargetId>::deserialize(id).map_err(DeError::custom)?,
None => None,
};

Ok(Self {
id,
name,
kind,
resolved,
options,
guild_id,
target_id,
resolved,
id: remove_from_map(&mut map, "id")?,
name: remove_from_map(&mut map, "name")?,
kind: remove_from_map(&mut map, "type")?,
guild_id: remove_from_map_opt(&mut map, "guild_id")?.flatten(),
target_id: remove_from_map_opt(&mut map, "target_id")?.flatten(),
})
}
}
Expand Down Expand Up @@ -608,7 +490,7 @@ pub struct CommandDataResolved {
/// Their resolved objects can be found on [`CommandData::resolved`].
///
/// [Discord docs](https://discord.com/developers/docs/interactions/application-commands#application-command-object-application-command-interaction-data-option-structure).
#[derive(Clone, Debug, Serialize)]
#[derive(Clone, Debug, Deserialize, Serialize)]
#[non_exhaustive]
pub struct CommandDataOption {
/// The name of the parameter.
Expand All @@ -629,41 +511,6 @@ pub struct CommandDataOption {
pub resolved: Option<CommandDataOptionValue>,
}

impl<'de> Deserialize<'de> for CommandDataOption {
fn deserialize<D: Deserializer<'de>>(deserializer: D) -> StdResult<Self, D::Error> {
let mut map = JsonMap::deserialize(deserializer)?;

let name = map
.remove("name")
.ok_or_else(|| DeError::custom("expected value"))
.and_then(String::deserialize)
.map_err(DeError::custom)?;

let value = map.remove("value");

let kind = map
.remove("type")
.ok_or_else(|| DeError::custom("expected type"))
.and_then(CommandOptionType::deserialize)
.map_err(DeError::custom)?;

let options = map
.remove("options")
.map(Vec::deserialize)
.transpose()
.map_err(DeError::custom)?
.unwrap_or_default();

Ok(Self {
name,
value,
kind,
options,
resolved: None,
})
}
}

/// The resolved value of an [`CommandDataOption`].
#[derive(Clone, Debug, Serialize, Deserialize)]
#[non_exhaustive]
Expand Down
Loading

0 comments on commit 464ed63

Please sign in to comment.