tgbotrs 0.2.0

A fully-featured, auto-generated Telegram Bot API library for Rust. All 285 types and 165 methods - strongly typed, fully async.
Documentation
use bytes::Bytes;
use serde::{Deserialize, Serialize, Serializer};

/// A file to be sent via the Telegram Bot API.
///
/// - `FileId` - reference a file already on Telegram's servers
/// - `Url` - let Telegram download the file from a URL
/// - `Memory` - upload raw bytes directly via multipart
#[derive(Debug, Clone)]
pub enum InputFile {
    FileId(String),
    Url(String),
    Memory { filename: String, data: Bytes },
}

impl InputFile {
    pub fn file_id(id: impl Into<String>) -> Self {
        InputFile::FileId(id.into())
    }

    pub fn url(url: impl Into<String>) -> Self {
        InputFile::Url(url.into())
    }

    pub fn memory(filename: impl Into<String>, data: impl Into<Bytes>) -> Self {
        InputFile::Memory {
            filename: filename.into(),
            data: data.into(),
        }
    }
}

// Memory files are handled separately when building multipart requests.
// FileId and Url serialize to their string value.
impl Serialize for InputFile {
    fn serialize<S: Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
        match self {
            InputFile::FileId(id) => serializer.serialize_str(id),
            InputFile::Url(url) => serializer.serialize_str(url),
            InputFile::Memory { filename, .. } => {
                serializer.serialize_str(&format!("attach://{}", filename))
            }
        }
    }
}

impl<'de> Deserialize<'de> for InputFile {
    fn deserialize<D: serde::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
        let s = String::deserialize(deserializer)?;
        if s.starts_with("http://") || s.starts_with("https://") {
            Ok(InputFile::Url(s))
        } else {
            Ok(InputFile::FileId(s))
        }
    }
}

impl From<String> for InputFile {
    fn from(s: String) -> Self {
        InputFile::FileId(s)
    }
}
impl From<&str> for InputFile {
    fn from(s: &str) -> Self {
        InputFile::FileId(s.to_string())
    }
}

/// A field that accepts either an [`InputFile`] or a plain string (file_id / URL).
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(untagged)]
pub enum InputFileOrString {
    File(InputFile),
    String(String),
}

impl From<InputFile> for InputFileOrString {
    fn from(f: InputFile) -> Self {
        InputFileOrString::File(f)
    }
}
impl From<String> for InputFileOrString {
    fn from(s: String) -> Self {
        InputFileOrString::String(s)
    }
}
impl From<&str> for InputFileOrString {
    fn from(s: &str) -> Self {
        InputFileOrString::String(s.to_string())
    }
}