271c31c9d4
This allows stateful or configurable codecs without having to express all configuration in the type name itself. For example, we could have a Base64 type with configurable Base64 engines without having to duplicate the Base64 type itself. (Note that the different engines in the Base64 crate are values, not types.)
166 lines
4.9 KiB
Rust
166 lines
4.9 KiB
Rust
// 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 `<PHOTO>` 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<u8>,
|
|
}
|
|
|
|
/// A `<vCard>` element; only the `<PHOTO>` element is supported for this legacy vCard ; the rest is ignored.
|
|
pub struct VCard {
|
|
/// A photo element.
|
|
pub photo: Option<Photo>,
|
|
}
|
|
|
|
impl TryFrom<Element> for VCard {
|
|
type Error = xso::error::Error;
|
|
|
|
fn try_from(value: Element) -> Result<Self, Self::Error> {
|
|
// Check that the root element is <vCard>
|
|
if !value.is("vCard", ns::VCARD) {
|
|
return Err(Error::Other(
|
|
"Root element is not <vCard xmlns='vcard-temp'>",
|
|
));
|
|
}
|
|
|
|
// Parse the <PHOTO> 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<VCard> 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"<vCard xmlns='vcard-temp'>
|
|
<BDAY>1476-06-09</BDAY>
|
|
<ADR>
|
|
<CTRY>Italy</CTRY>
|
|
<LOCALITY>Verona</LOCALITY>
|
|
<HOME/>
|
|
</ADR>
|
|
<NICKNAME/>
|
|
<N><GIVEN>Juliet</GIVEN><FAMILY>Capulet</FAMILY></N>
|
|
<EMAIL>jcapulet@shakespeare.lit</EMAIL>
|
|
<PHOTO>
|
|
<TYPE>image/jpeg</TYPE>
|
|
<BINVAL>{}</BINVAL>
|
|
</PHOTO>
|
|
</vCard>",
|
|
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"<vCard xmlns='vcard-temp'>
|
|
<BDAY>1476-06-09</BDAY>
|
|
<ADR>
|
|
<CTRY>Italy</CTRY>
|
|
<LOCALITY>Verona</LOCALITY>
|
|
<HOME/>
|
|
</ADR>
|
|
<NICKNAME/>
|
|
<N><GIVEN>Juliet</GIVEN><FAMILY>Capulet</FAMILY></N>
|
|
<EMAIL>jcapulet@shakespeare.lit</EMAIL>
|
|
<PHOTO>
|
|
<TYPE>image/jpeg</TYPE>
|
|
<BINVAL>Zm9v
|
|
Cg==</BINVAL>
|
|
</PHOTO>
|
|
</vCard>";
|
|
|
|
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");
|
|
}
|
|
}
|