mirror of
https://github.com/teloxide/teloxide.git
synced 2025-01-10 20:12:25 +01:00
Merge branch 'dev' into remove_macro_use
This commit is contained in:
commit
2953310f13
1 changed files with 132 additions and 12 deletions
|
@ -1,23 +1,143 @@
|
|||
use serde::{Deserialize, Serialize};
|
||||
use serde::{de::Error, Deserialize, Deserializer, Serialize, Serializer};
|
||||
|
||||
use crate::types::True;
|
||||
|
||||
/// This object represents one button of the reply keyboard. For filter text
|
||||
/// buttons String can be used instead of this object to specify text of the
|
||||
/// button. Optional fields are mutually exclusive.
|
||||
/// button.
|
||||
///
|
||||
/// [The official docs](https://core.telegram.org/bots/api#keyboardbutton).
|
||||
#[derive(Clone, Debug, Deserialize, Eq, Hash, PartialEq, Serialize)]
|
||||
#[serde_with_macros::skip_serializing_none]
|
||||
#[derive(Debug, Serialize, Deserialize, Hash, PartialEq, Eq, Clone)]
|
||||
pub struct KeyboardButton {
|
||||
/// Text of the button. If none of the optional fields are used, it will
|
||||
/// be sent as a message when the button is pressed.
|
||||
pub text: String,
|
||||
|
||||
/// If `true`, the user's phone number will be sent as a contact
|
||||
/// when the button is pressed. Available in private chats only.
|
||||
#[serde(skip_serializing_if = "Option::is_none")]
|
||||
pub request_contact: Option<bool>,
|
||||
|
||||
/// If `true`, the user's current location will be sent when the
|
||||
/// button is pressed. Available in private chats only.
|
||||
#[serde(skip_serializing_if = "Option::is_none")]
|
||||
pub request_location: Option<bool>,
|
||||
/// Request something from user.
|
||||
/// - If `Some(Contact)`, the user's phone number will be sent as a contact
|
||||
/// when the button is pressed. Available in private chats only
|
||||
/// - If `Some(Location)`, the user's current location will be sent when
|
||||
/// the button is pressed. Available in private chats only
|
||||
#[serde(flatten)]
|
||||
pub request: Option<ButtonRequest>,
|
||||
}
|
||||
|
||||
// Serialize + Deserialize are implemented by hand
|
||||
#[derive(Debug, Hash, PartialEq, Eq, Clone)]
|
||||
pub enum ButtonRequest {
|
||||
Location,
|
||||
Contact,
|
||||
}
|
||||
|
||||
/// Helper struct for (de)serializing [`ButtonRequest`](ButtonRequest)
|
||||
#[serde_with_macros::skip_serializing_none]
|
||||
#[derive(Serialize, Deserialize)]
|
||||
struct RawRequest {
|
||||
/// Optional. If True, the user's phone number will be sent as a contact
|
||||
/// when the button is pressed. Available in private chats only
|
||||
#[serde(rename = "request_contact")]
|
||||
contact: Option<True>,
|
||||
|
||||
/// Optional. If True, the user's current location will be sent when the
|
||||
/// button is pressed. Available in private chats only
|
||||
#[serde(rename = "request_location")]
|
||||
location: Option<True>,
|
||||
}
|
||||
|
||||
impl<'de> Deserialize<'de> for ButtonRequest {
|
||||
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
||||
where
|
||||
D: Deserializer<'de>,
|
||||
{
|
||||
let raw = RawRequest::deserialize(deserializer)?;
|
||||
match raw {
|
||||
RawRequest {
|
||||
contact: Some(_),
|
||||
location: Some(_),
|
||||
} => Err(D::Error::custom(
|
||||
"`request_contact` and `request_location` fields are mutually \
|
||||
exclusive, but both were provided",
|
||||
)),
|
||||
RawRequest {
|
||||
contact: Some(_), ..
|
||||
} => Ok(Self::Contact),
|
||||
RawRequest {
|
||||
location: Some(_), ..
|
||||
} => Ok(Self::Location),
|
||||
_ => Err(D::Error::custom(
|
||||
"Either one of `request_contact` and `request_location` \
|
||||
fields is required",
|
||||
)),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Serialize for ButtonRequest {
|
||||
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
||||
where
|
||||
S: Serializer,
|
||||
{
|
||||
match self {
|
||||
Self::Contact => RawRequest {
|
||||
contact: Some(True),
|
||||
location: None,
|
||||
}
|
||||
.serialize(serializer),
|
||||
Self::Location => RawRequest {
|
||||
contact: None,
|
||||
location: Some(True),
|
||||
}
|
||||
.serialize(serializer),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
|
||||
#[test]
|
||||
fn serialize_no_request() {
|
||||
let button = KeyboardButton {
|
||||
text: String::from(""),
|
||||
request: None,
|
||||
};
|
||||
let expected = r#"{"text":""}"#;
|
||||
let actual = serde_json::to_string(&button).unwrap();
|
||||
assert_eq!(expected, actual);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn serialize_request_contact() {
|
||||
let button = KeyboardButton {
|
||||
text: String::from(""),
|
||||
request: Some(ButtonRequest::Contact),
|
||||
};
|
||||
let expected = r#"{"text":"","request_contact":true}"#;
|
||||
let actual = serde_json::to_string(&button).unwrap();
|
||||
assert_eq!(expected, actual);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn deserialize_no_request() {
|
||||
let json = r#"{"text":""}"#;
|
||||
let expected = KeyboardButton {
|
||||
text: String::from(""),
|
||||
request: None,
|
||||
};
|
||||
let actual = serde_json::from_str(json).unwrap();
|
||||
assert_eq!(expected, actual);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn deserialize_request_contact() {
|
||||
let json = r#"{"text":"","request_contact":true}"#;
|
||||
let expected = KeyboardButton {
|
||||
text: String::from(""),
|
||||
request: Some(ButtonRequest::Contact),
|
||||
};
|
||||
let actual = serde_json::from_str(json).unwrap();
|
||||
assert_eq!(expected, actual);
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue