// Copyright (c) 2024 xmpp-rs contributors. // // This Source Code Form is subject to the terms of the Mozilla Public // License, v. 2.0. If a copy of the MPL was not distributed with this // file, You can obtain one at http://mozilla.org/MPL/2.0/. //! This module implements vCard, for the purpose of vCard-based avatars as defined in //! [XEP-0054](https://xmpp.org/extensions/xep-0054.html). //! //! Only the `` element is supported as a member of this legacy vCard. For more modern and complete //! user profile management, see [XEP-0292](https://xmpp.org/extensions/xep-0292.html): vCard4 Over XMPP. //! //! For vCard updates defined in [XEP-0153](https://xmpp.org/extensions/xep-0153.html), //! see [`vcard_update`][crate::vcard_update] module. use xso::{ error::Error, text::{Base64, StripWhitespace, TextCodec}, AsXml, FromXml, }; use crate::iq::{IqGetPayload, IqResultPayload, IqSetPayload}; use crate::ns; use minidom::Element; /// A photo element. #[derive(FromXml, AsXml, PartialEq, Debug, Clone)] #[xml(namespace = ns::VCARD, name = "PHOTO")] pub struct Photo { /// The type of the photo. #[xml(child)] pub type_: Type, /// The binary data of the photo. #[xml(child)] pub binval: Binval, } /// The type of the photo. #[derive(FromXml, AsXml, PartialEq, Debug, Clone)] #[xml(namespace = ns::VCARD, name = "TYPE")] pub struct Type { /// The type as a plain text string; at least "image/jpeg", "image/gif" and "image/png" SHOULD be supported. #[xml(text)] pub data: String, } /// The binary data of the photo. #[derive(FromXml, AsXml, PartialEq, Debug, Clone)] #[xml(namespace = ns::VCARD, name = "BINVAL")] pub struct Binval { /// The actual data. #[xml(text(codec = Base64.filtered(StripWhitespace)))] pub data: Vec, } /// A `` element; only the `` element is supported for this legacy vCard ; the rest is ignored. pub struct VCard { /// A photo element. pub photo: Option, } impl TryFrom for VCard { type Error = xso::error::Error; fn try_from(value: Element) -> Result { // Check that the root element is if !value.is("vCard", ns::VCARD) { return Err(Error::Other( "Root element is not ", )); } // Parse the element, if any. let photo = value .get_child("PHOTO", ns::VCARD) .map(|photo| Photo::try_from(photo.clone())) .transpose()?; // Return the result. Ok(VCard { photo }) } } impl From for Element { fn from(vcard: VCard) -> Element { Element::builder("vCard", ns::VCARD) .append_all(vcard.photo) .build() } } impl IqGetPayload for VCard {} impl IqSetPayload for VCard {} impl IqResultPayload for VCard {} #[cfg(test)] mod tests { use super::*; use base64::Engine; use std::str::FromStr; #[test] fn test_vcard() { // Create some bytes: let bytes = [0u8, 1, 2, 129]; // Test xml stolen from https://xmpp.org/extensions/xep-0153.html#example-5 let test_vcard = format!( r" 1476-06-09 Italy Verona JulietCapulet jcapulet@shakespeare.lit image/jpeg {} ", base64::prelude::BASE64_STANDARD.encode(&bytes) ); let test_vcard = Element::from_str(&test_vcard).expect("Failed to parse XML"); let test_vcard = VCard::try_from(test_vcard).expect("Failed to parse vCard"); let photo = test_vcard.photo.expect("No photo found"); assert_eq!(photo.type_.data, "image/jpeg".to_string()); assert_eq!(photo.binval.data, bytes); } #[test] fn test_vcard_with_linebreaks() { // Test xml stolen from https://xmpp.org/extensions/xep-0153.html#example-5 // extended to use a multi-line base64 string as is allowed as per RFC 2426 let test_vcard = r" 1476-06-09 Italy Verona JulietCapulet jcapulet@shakespeare.lit image/jpeg Zm9v Cg== "; let test_vcard = Element::from_str(&test_vcard).expect("Failed to parse XML"); let test_vcard = VCard::try_from(test_vcard).expect("Failed to parse vCard"); let photo = test_vcard.photo.expect("No photo found"); assert_eq!(photo.type_.data, "image/jpeg".to_string()); assert_eq!(photo.binval.data, b"foo\n"); } }