2020-03-05 00:25:24 +00:00
|
|
|
|
use futures::stream::StreamExt;
|
2019-02-26 18:28:41 +00:00
|
|
|
|
use std::env::args;
|
|
|
|
|
use std::fs::{create_dir_all, File};
|
|
|
|
|
use std::io::{self, Write};
|
|
|
|
|
use std::process::exit;
|
|
|
|
|
use std::str::FromStr;
|
2024-08-10 13:05:42 +00:00
|
|
|
|
use tokio_xmpp::{Client, Stanza};
|
2019-02-26 18:28:41 +00:00
|
|
|
|
use xmpp_parsers::{
|
|
|
|
|
avatar::{Data as AvatarData, Metadata as AvatarMetadata},
|
|
|
|
|
caps::{compute_disco, hash_caps, Caps},
|
|
|
|
|
disco::{DiscoInfoQuery, DiscoInfoResult, Feature, Identity},
|
|
|
|
|
hashes::Algo,
|
|
|
|
|
iq::{Iq, IqType},
|
2024-07-24 18:39:27 +00:00
|
|
|
|
jid::{BareJid, Jid},
|
2019-02-26 18:28:41 +00:00
|
|
|
|
ns,
|
|
|
|
|
presence::{Presence, Type as PresenceType},
|
|
|
|
|
pubsub::{
|
|
|
|
|
event::PubSubEvent,
|
|
|
|
|
pubsub::{Items, PubSub},
|
|
|
|
|
NodeName,
|
|
|
|
|
},
|
2019-10-22 23:32:41 +00:00
|
|
|
|
stanza_error::{DefinedCondition, ErrorType, StanzaError},
|
2019-02-26 18:28:41 +00:00
|
|
|
|
};
|
|
|
|
|
|
2020-03-05 00:25:24 +00:00
|
|
|
|
#[tokio::main]
|
|
|
|
|
async fn main() {
|
2023-06-05 11:06:52 +00:00
|
|
|
|
env_logger::init();
|
|
|
|
|
|
2019-02-26 18:28:41 +00:00
|
|
|
|
let args: Vec<String> = args().collect();
|
|
|
|
|
if args.len() != 3 {
|
|
|
|
|
println!("Usage: {} <jid> <password>", args[0]);
|
|
|
|
|
exit(1);
|
|
|
|
|
}
|
2023-06-01 09:58:04 +00:00
|
|
|
|
let jid = BareJid::from_str(&args[1]).expect(&format!("Invalid JID: {}", &args[1]));
|
2020-03-05 00:25:24 +00:00
|
|
|
|
let password = args[2].clone();
|
2019-02-26 18:28:41 +00:00
|
|
|
|
|
|
|
|
|
// Client instance
|
2023-06-01 09:58:04 +00:00
|
|
|
|
let mut client = Client::new(jid.clone(), password);
|
2019-02-26 18:28:41 +00:00
|
|
|
|
|
|
|
|
|
let disco_info = make_disco();
|
|
|
|
|
|
|
|
|
|
// Main loop, processes events
|
2024-08-20 14:54:48 +00:00
|
|
|
|
while let Some(event) = client.next().await {
|
|
|
|
|
if event.is_online() {
|
|
|
|
|
println!("Online!");
|
2020-03-05 00:25:24 +00:00
|
|
|
|
|
2024-08-20 14:54:48 +00:00
|
|
|
|
let caps = get_disco_caps(&disco_info, "https://gitlab.com/xmpp-rs/tokio-xmpp");
|
|
|
|
|
let presence = make_presence(caps);
|
|
|
|
|
client.send_stanza(presence.into()).await.unwrap();
|
|
|
|
|
} else if let Some(stanza) = event.into_stanza() {
|
|
|
|
|
match stanza {
|
|
|
|
|
Stanza::Iq(iq) => {
|
|
|
|
|
if let IqType::Get(payload) = iq.payload {
|
|
|
|
|
if payload.is("query", ns::DISCO_INFO) {
|
|
|
|
|
let query = DiscoInfoQuery::try_from(payload);
|
|
|
|
|
match query {
|
|
|
|
|
Ok(query) => {
|
|
|
|
|
let mut disco = disco_info.clone();
|
|
|
|
|
disco.node = query.node;
|
|
|
|
|
let iq = Iq::from_result(iq.id, Some(disco))
|
|
|
|
|
.with_to(iq.from.unwrap());
|
|
|
|
|
client.send_stanza(iq.into()).await.unwrap();
|
|
|
|
|
}
|
|
|
|
|
Err(err) => {
|
|
|
|
|
client
|
2024-08-10 13:05:42 +00:00
|
|
|
|
.send_stanza(
|
|
|
|
|
make_error(
|
|
|
|
|
iq.from.unwrap(),
|
|
|
|
|
iq.id,
|
|
|
|
|
ErrorType::Modify,
|
|
|
|
|
DefinedCondition::BadRequest,
|
|
|
|
|
&format!("{}", err),
|
|
|
|
|
)
|
|
|
|
|
.into(),
|
|
|
|
|
)
|
|
|
|
|
.await
|
2024-08-20 14:54:48 +00:00
|
|
|
|
.unwrap();
|
2020-03-05 00:25:24 +00:00
|
|
|
|
}
|
2024-08-10 13:05:42 +00:00
|
|
|
|
}
|
2024-08-20 14:54:48 +00:00
|
|
|
|
} else {
|
|
|
|
|
// We MUST answer unhandled get iqs with a service-unavailable error.
|
2020-03-05 00:25:24 +00:00
|
|
|
|
client
|
2024-08-10 13:05:42 +00:00
|
|
|
|
.send_stanza(
|
|
|
|
|
make_error(
|
|
|
|
|
iq.from.unwrap(),
|
|
|
|
|
iq.id,
|
|
|
|
|
ErrorType::Cancel,
|
|
|
|
|
DefinedCondition::ServiceUnavailable,
|
|
|
|
|
"No handler defined for this kind of iq.",
|
|
|
|
|
)
|
|
|
|
|
.into(),
|
|
|
|
|
)
|
2020-03-05 00:25:24 +00:00
|
|
|
|
.await
|
|
|
|
|
.unwrap();
|
2019-02-26 18:28:41 +00:00
|
|
|
|
}
|
2024-08-20 14:54:48 +00:00
|
|
|
|
} else if let IqType::Result(Some(payload)) = iq.payload {
|
|
|
|
|
if payload.is("pubsub", ns::PUBSUB) {
|
|
|
|
|
let pubsub = PubSub::try_from(payload).unwrap();
|
|
|
|
|
let from = iq.from.clone().unwrap_or(jid.clone().into());
|
|
|
|
|
handle_iq_result(pubsub, &from);
|
|
|
|
|
}
|
|
|
|
|
} else if let IqType::Set(_) = iq.payload {
|
|
|
|
|
// We MUST answer unhandled set iqs with a service-unavailable error.
|
|
|
|
|
client
|
|
|
|
|
.send_stanza(
|
|
|
|
|
make_error(
|
|
|
|
|
iq.from.unwrap(),
|
|
|
|
|
iq.id,
|
|
|
|
|
ErrorType::Cancel,
|
|
|
|
|
DefinedCondition::ServiceUnavailable,
|
|
|
|
|
"No handler defined for this kind of iq.",
|
|
|
|
|
)
|
|
|
|
|
.into(),
|
|
|
|
|
)
|
|
|
|
|
.await
|
|
|
|
|
.unwrap();
|
2019-02-26 18:28:41 +00:00
|
|
|
|
}
|
2024-08-20 14:54:48 +00:00
|
|
|
|
}
|
|
|
|
|
Stanza::Message(message) => {
|
|
|
|
|
let from = message.from.clone().unwrap();
|
|
|
|
|
if let Some(body) = message.get_best_body(vec!["en"]) {
|
|
|
|
|
if body.0 == "die" {
|
|
|
|
|
println!("Secret die command triggered by {}", from);
|
|
|
|
|
break;
|
2020-03-05 00:25:24 +00:00
|
|
|
|
}
|
2024-08-20 14:54:48 +00:00
|
|
|
|
}
|
|
|
|
|
for child in message.payloads {
|
|
|
|
|
if child.is("event", ns::PUBSUB_EVENT) {
|
|
|
|
|
let event = PubSubEvent::try_from(child).unwrap();
|
|
|
|
|
if let PubSubEvent::PublishedItems { node, items } = event {
|
|
|
|
|
if node.0 == ns::AVATAR_METADATA {
|
|
|
|
|
for item in items.into_iter() {
|
|
|
|
|
let payload = item.payload.clone().unwrap();
|
|
|
|
|
if payload.is("metadata", ns::AVATAR_METADATA) {
|
|
|
|
|
// TODO: do something with these metadata.
|
|
|
|
|
let _metadata =
|
|
|
|
|
AvatarMetadata::try_from(payload).unwrap();
|
|
|
|
|
println!(
|
|
|
|
|
"[1m{}[0m has published an avatar, downloading...",
|
|
|
|
|
from.clone()
|
|
|
|
|
);
|
|
|
|
|
let iq = download_avatar(from.clone());
|
|
|
|
|
client.send_stanza(iq.into()).await.unwrap();
|
2020-03-05 00:25:24 +00:00
|
|
|
|
}
|
2019-02-26 18:28:41 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-08-20 14:54:48 +00:00
|
|
|
|
// Nothing to do here.
|
|
|
|
|
Stanza::Presence(_) => (),
|
2019-02-26 18:28:41 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-05 00:25:24 +00:00
|
|
|
|
fn make_error(
|
|
|
|
|
to: Jid,
|
|
|
|
|
id: String,
|
|
|
|
|
type_: ErrorType,
|
|
|
|
|
condition: DefinedCondition,
|
|
|
|
|
text: &str,
|
2024-08-10 13:05:42 +00:00
|
|
|
|
) -> Iq {
|
2020-03-05 00:25:24 +00:00
|
|
|
|
let error = StanzaError::new(type_, condition, "en", text);
|
2024-08-10 13:05:42 +00:00
|
|
|
|
Iq::from_error(id, error).with_to(to)
|
2020-03-05 00:25:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-26 18:28:41 +00:00
|
|
|
|
fn make_disco() -> DiscoInfoResult {
|
|
|
|
|
let identities = vec![Identity::new("client", "bot", "en", "tokio-xmpp")];
|
|
|
|
|
let features = vec![
|
|
|
|
|
Feature::new(ns::DISCO_INFO),
|
|
|
|
|
Feature::new(format!("{}+notify", ns::AVATAR_METADATA)),
|
|
|
|
|
];
|
|
|
|
|
DiscoInfoResult {
|
|
|
|
|
node: None,
|
|
|
|
|
identities,
|
|
|
|
|
features,
|
|
|
|
|
extensions: vec![],
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn get_disco_caps(disco: &DiscoInfoResult, node: &str) -> Caps {
|
|
|
|
|
let caps_data = compute_disco(disco);
|
|
|
|
|
let hash = hash_caps(&caps_data, Algo::Sha_1).unwrap();
|
|
|
|
|
Caps::new(node, hash)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Construct a <presence/>
|
|
|
|
|
fn make_presence(caps: Caps) -> Presence {
|
2019-10-22 23:32:41 +00:00
|
|
|
|
let mut presence = Presence::new(PresenceType::None).with_priority(-1);
|
2019-02-26 18:28:41 +00:00
|
|
|
|
presence.set_status("en", "Downloading avatars.");
|
|
|
|
|
presence.add_payload(caps);
|
|
|
|
|
presence
|
|
|
|
|
}
|
|
|
|
|
|
2019-09-09 12:57:08 +00:00
|
|
|
|
fn download_avatar(from: Jid) -> Iq {
|
2019-10-22 23:32:41 +00:00
|
|
|
|
Iq::from_get(
|
|
|
|
|
"coucou",
|
|
|
|
|
PubSub::Items(Items {
|
|
|
|
|
max_items: None,
|
|
|
|
|
node: NodeName(String::from(ns::AVATAR_DATA)),
|
|
|
|
|
subid: None,
|
|
|
|
|
items: Vec::new(),
|
|
|
|
|
}),
|
|
|
|
|
)
|
2019-09-09 12:57:08 +00:00
|
|
|
|
.with_to(from)
|
2019-02-26 18:28:41 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn handle_iq_result(pubsub: PubSub, from: &Jid) {
|
|
|
|
|
if let PubSub::Items(items) = pubsub {
|
|
|
|
|
if items.node.0 == ns::AVATAR_DATA {
|
|
|
|
|
for item in items.items {
|
2019-09-09 12:57:08 +00:00
|
|
|
|
match (item.id.clone(), item.payload.clone()) {
|
|
|
|
|
(Some(id), Some(payload)) => {
|
|
|
|
|
let data = AvatarData::try_from(payload).unwrap();
|
2019-02-26 18:28:41 +00:00
|
|
|
|
save_avatar(from, id.0, &data.data).unwrap();
|
|
|
|
|
}
|
2019-09-09 12:57:08 +00:00
|
|
|
|
_ => {}
|
2019-02-26 18:28:41 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-05 00:25:24 +00:00
|
|
|
|
// TODO: may use tokio?
|
2019-02-26 18:28:41 +00:00
|
|
|
|
fn save_avatar(from: &Jid, id: String, data: &[u8]) -> io::Result<()> {
|
2019-09-09 12:57:08 +00:00
|
|
|
|
let directory = format!("data/{}", from);
|
|
|
|
|
let filename = format!("data/{}/{}", from, id);
|
2024-09-16 21:32:49 +00:00
|
|
|
|
println!(
|
|
|
|
|
"Saving avatar from [1m{}[0m to [4m{}[0m.",
|
|
|
|
|
from, filename
|
|
|
|
|
);
|
2019-02-26 18:28:41 +00:00
|
|
|
|
create_dir_all(directory)?;
|
|
|
|
|
let mut file = File::create(filename)?;
|
|
|
|
|
file.write_all(data)
|
|
|
|
|
}
|