add plugin infrastructure
This commit is contained in:
parent
8883f343b7
commit
83fdd9866c
9 changed files with 401 additions and 2 deletions
|
@ -6,6 +6,7 @@ authors = ["lumi <lumi@pew.im>"]
|
|||
[dependencies]
|
||||
xml-rs = "*"
|
||||
openssl = "*"
|
||||
base64 = "*"
|
||||
|
||||
[dependencies.minidom]
|
||||
git = "https://gitlab.com/lumi/minidom-rs.git"
|
||||
|
|
|
@ -2,10 +2,22 @@ extern crate xmpp;
|
|||
|
||||
use xmpp::jid::Jid;
|
||||
use xmpp::client::ClientBuilder;
|
||||
use xmpp::plugins::messaging::{MessagingPlugin, MessageEvent};
|
||||
use xmpp::plugins::presence::{PresencePlugin, Show};
|
||||
|
||||
use std::env;
|
||||
|
||||
fn main() {
|
||||
let jid: Jid = env::var("JID").unwrap().parse().unwrap();
|
||||
let client = ClientBuilder::new(jid).connect().unwrap();
|
||||
let mut client = ClientBuilder::new(jid).connect().unwrap();
|
||||
client.register_plugin(MessagingPlugin::new());
|
||||
client.register_plugin(PresencePlugin::new());
|
||||
client.connect_plain(&env::var("PASS").unwrap()).unwrap();
|
||||
client.plugin::<PresencePlugin>().set_presence(Show::Available, None).unwrap();
|
||||
loop {
|
||||
let event = client.next_event().unwrap();
|
||||
if let Some(evt) = event.downcast::<MessageEvent>() {
|
||||
println!("{:?}", evt);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
122
src/client.rs
122
src/client.rs
|
@ -2,10 +2,18 @@ use jid::Jid;
|
|||
use transport::{Transport, SslTransport};
|
||||
use error::Error;
|
||||
use ns;
|
||||
use plugin::{Plugin, PluginProxyBinding};
|
||||
use event::AbstractEvent;
|
||||
|
||||
use base64;
|
||||
|
||||
use minidom::Element;
|
||||
|
||||
use xml::writer::XmlEvent as WriterEvent;
|
||||
use xml::reader::XmlEvent as ReaderEvent;
|
||||
|
||||
use std::sync::mpsc::{Receiver, channel};
|
||||
|
||||
pub struct ClientBuilder {
|
||||
jid: Jid,
|
||||
host: Option<String>,
|
||||
|
@ -38,9 +46,15 @@ impl ClientBuilder {
|
|||
.attr("to", &self.jid.domain)
|
||||
.default_ns(ns::CLIENT)
|
||||
.ns("stream", ns::STREAM))?;
|
||||
let (sender_out, sender_in) = channel();
|
||||
let (dispatcher_out, dispatcher_in) = channel();
|
||||
Ok(Client {
|
||||
jid: self.jid,
|
||||
transport: transport
|
||||
transport: transport,
|
||||
plugins: Vec::new(),
|
||||
binding: PluginProxyBinding::new(sender_out, dispatcher_out),
|
||||
sender_in: sender_in,
|
||||
dispatcher_in: dispatcher_in,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
@ -48,10 +62,116 @@ impl ClientBuilder {
|
|||
pub struct Client {
|
||||
jid: Jid,
|
||||
transport: SslTransport,
|
||||
plugins: Vec<Box<Plugin>>,
|
||||
binding: PluginProxyBinding,
|
||||
sender_in: Receiver<Element>,
|
||||
dispatcher_in: Receiver<AbstractEvent>,
|
||||
}
|
||||
|
||||
impl Client {
|
||||
pub fn jid(&self) -> &Jid {
|
||||
&self.jid
|
||||
}
|
||||
|
||||
pub fn register_plugin<P: Plugin + 'static>(&mut self, mut plugin: P) {
|
||||
plugin.bind(self.binding.clone());
|
||||
self.plugins.push(Box::new(plugin));
|
||||
}
|
||||
|
||||
pub fn plugin<P: Plugin>(&self) -> &P {
|
||||
for plugin in &self.plugins {
|
||||
let any = plugin.as_any();
|
||||
if let Some(ret) = any.downcast_ref::<P>() {
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
panic!("plugin does not exist!");
|
||||
}
|
||||
|
||||
pub fn next_event(&mut self) -> Result<AbstractEvent, Error> {
|
||||
self.flush_send_queue()?;
|
||||
loop {
|
||||
if let Ok(evt) = self.dispatcher_in.try_recv() {
|
||||
return Ok(evt);
|
||||
}
|
||||
let elem = self.transport.read_element()?;
|
||||
for plugin in self.plugins.iter_mut() {
|
||||
plugin.handle(&elem);
|
||||
// TODO: handle plugin return
|
||||
}
|
||||
self.flush_send_queue()?;
|
||||
}
|
||||
}
|
||||
|
||||
pub fn flush_send_queue(&mut self) -> Result<(), Error> { // TODO: not sure how great of an
|
||||
// idea it is to flush in this
|
||||
// manner…
|
||||
while let Ok(elem) = self.sender_in.try_recv() {
|
||||
self.transport.write_element(&elem)?;
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
|
||||
pub fn connect_plain(&mut self, password: &str) -> Result<(), Error> {
|
||||
// TODO: this is very ugly
|
||||
loop {
|
||||
let e = self.transport.read_event().unwrap();
|
||||
match e {
|
||||
ReaderEvent::StartElement { .. } => {
|
||||
break;
|
||||
},
|
||||
_ => (),
|
||||
}
|
||||
}
|
||||
let mut did_sasl = false;
|
||||
loop {
|
||||
let n = self.transport.read_element().unwrap();
|
||||
if n.is("features", ns::STREAM) {
|
||||
if did_sasl {
|
||||
let mut elem = Element::builder("iq")
|
||||
.attr("id", "bind")
|
||||
.attr("type", "set")
|
||||
.build();
|
||||
let bind = Element::builder("bind")
|
||||
.ns(ns::BIND)
|
||||
.build();
|
||||
elem.append_child(bind);
|
||||
self.transport.write_element(&elem)?;
|
||||
}
|
||||
else {
|
||||
let mut auth = Vec::new();
|
||||
auth.push(0);
|
||||
auth.extend(self.jid.node.as_ref().expect("JID has no node").bytes());
|
||||
auth.push(0);
|
||||
auth.extend(password.bytes());
|
||||
let mut elem = Element::builder("auth")
|
||||
.ns(ns::SASL)
|
||||
.attr("mechanism", "PLAIN")
|
||||
.build();
|
||||
elem.append_text_node(base64::encode(&auth));
|
||||
self.transport.write_element(&elem)?;
|
||||
did_sasl = true;
|
||||
}
|
||||
}
|
||||
else if n.is("success", ns::SASL) {
|
||||
self.transport.reset_stream();
|
||||
self.transport.write_event(WriterEvent::start_element("stream:stream")
|
||||
.attr("to", &self.jid.domain)
|
||||
.default_ns(ns::CLIENT)
|
||||
.ns("stream", ns::STREAM))?;
|
||||
loop {
|
||||
let e = self.transport.read_event()?;
|
||||
match e {
|
||||
ReaderEvent::StartElement { .. } => {
|
||||
break;
|
||||
},
|
||||
_ => (),
|
||||
}
|
||||
}
|
||||
}
|
||||
else if n.is("iq", ns::CLIENT) && n.has_child("bind", ns::BIND) {
|
||||
return Ok(());
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
25
src/event.rs
Normal file
25
src/event.rs
Normal file
|
@ -0,0 +1,25 @@
|
|||
use std::fmt::Debug;
|
||||
|
||||
use std::any::Any;
|
||||
|
||||
pub struct AbstractEvent {
|
||||
inner: Box<Any>,
|
||||
}
|
||||
|
||||
impl AbstractEvent {
|
||||
pub fn new<E: Event>(event: E) -> AbstractEvent {
|
||||
AbstractEvent {
|
||||
inner: Box::new(event),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn downcast<E: Event + 'static>(&self) -> Option<&E> {
|
||||
self.inner.downcast_ref::<E>()
|
||||
}
|
||||
|
||||
pub fn is<E: Event + 'static>(&self) -> bool {
|
||||
self.inner.is::<E>()
|
||||
}
|
||||
}
|
||||
|
||||
pub trait Event: Any + Debug {}
|
|
@ -1,11 +1,15 @@
|
|||
extern crate xml;
|
||||
extern crate openssl;
|
||||
extern crate minidom;
|
||||
extern crate base64;
|
||||
|
||||
pub mod ns;
|
||||
pub mod transport;
|
||||
pub mod error;
|
||||
pub mod jid;
|
||||
pub mod client;
|
||||
pub mod plugin;
|
||||
pub mod event;
|
||||
pub mod plugins;
|
||||
|
||||
mod locked_io;
|
||||
|
|
89
src/plugin.rs
Normal file
89
src/plugin.rs
Normal file
|
@ -0,0 +1,89 @@
|
|||
use event::{Event, AbstractEvent};
|
||||
|
||||
use std::any::Any;
|
||||
|
||||
use std::sync::mpsc::Sender;
|
||||
|
||||
use std::mem;
|
||||
|
||||
use minidom::Element;
|
||||
|
||||
#[derive(Clone)]
|
||||
pub struct PluginProxyBinding {
|
||||
sender: Sender<Element>,
|
||||
dispatcher: Sender<AbstractEvent>,
|
||||
}
|
||||
|
||||
impl PluginProxyBinding {
|
||||
pub fn new(sender: Sender<Element>, dispatcher: Sender<AbstractEvent>) -> PluginProxyBinding {
|
||||
PluginProxyBinding {
|
||||
sender: sender,
|
||||
dispatcher: dispatcher,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub enum PluginProxy {
|
||||
Unbound,
|
||||
BoundTo(PluginProxyBinding),
|
||||
}
|
||||
|
||||
impl PluginProxy {
|
||||
pub fn new() -> PluginProxy {
|
||||
PluginProxy::Unbound
|
||||
}
|
||||
|
||||
pub fn bind(&mut self, inner: PluginProxyBinding) {
|
||||
if let PluginProxy::BoundTo(_) = *self {
|
||||
panic!("trying to bind an already bound plugin proxy!");
|
||||
}
|
||||
mem::replace(self, PluginProxy::BoundTo(inner));
|
||||
}
|
||||
|
||||
fn with_binding<R, F: FnOnce(&PluginProxyBinding) -> R>(&self, f: F) -> R {
|
||||
match *self {
|
||||
PluginProxy::Unbound => {
|
||||
panic!("trying to use an unbound plugin proxy!");
|
||||
},
|
||||
PluginProxy::BoundTo(ref binding) => {
|
||||
f(binding)
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
pub fn dispatch<E: Event>(&self, event: E) {
|
||||
self.with_binding(move |binding| {
|
||||
binding.dispatcher.send(AbstractEvent::new(event))
|
||||
.unwrap(); // TODO: may want to return the error
|
||||
});
|
||||
}
|
||||
|
||||
pub fn send(&self, elem: Element) {
|
||||
self.with_binding(move |binding| {
|
||||
binding.sender.send(elem).unwrap(); // TODO: as above, may want to return the error
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
|
||||
pub enum PluginReturn {
|
||||
Continue,
|
||||
Unload,
|
||||
}
|
||||
|
||||
pub trait Plugin: Any + PluginAny {
|
||||
fn get_proxy(&mut self) -> &mut PluginProxy;
|
||||
fn handle(&mut self, _elem: &Element) -> PluginReturn { PluginReturn::Continue }
|
||||
|
||||
fn bind(&mut self, inner: PluginProxyBinding) {
|
||||
self.get_proxy().bind(inner);
|
||||
}
|
||||
}
|
||||
|
||||
pub trait PluginAny {
|
||||
fn as_any(&self) -> &Any;
|
||||
}
|
||||
|
||||
impl<T: Any + Sized> PluginAny for T {
|
||||
fn as_any(&self) -> &Any { self }
|
||||
}
|
45
src/plugins/messaging.rs
Normal file
45
src/plugins/messaging.rs
Normal file
|
@ -0,0 +1,45 @@
|
|||
use plugin::{Plugin, PluginReturn, PluginProxy};
|
||||
use event::Event;
|
||||
use minidom::Element;
|
||||
use jid::Jid;
|
||||
use ns;
|
||||
|
||||
#[derive(Debug)]
|
||||
pub struct MessageEvent {
|
||||
from: Jid,
|
||||
to: Jid,
|
||||
body: String,
|
||||
}
|
||||
|
||||
impl Event for MessageEvent {}
|
||||
|
||||
pub struct MessagingPlugin {
|
||||
proxy: PluginProxy,
|
||||
}
|
||||
|
||||
impl MessagingPlugin {
|
||||
pub fn new() -> MessagingPlugin {
|
||||
MessagingPlugin {
|
||||
proxy: PluginProxy::new(),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Plugin for MessagingPlugin {
|
||||
fn get_proxy(&mut self) -> &mut PluginProxy {
|
||||
&mut self.proxy
|
||||
}
|
||||
|
||||
fn handle(&mut self, elem: &Element) -> PluginReturn {
|
||||
if elem.is("message", ns::CLIENT) && elem.attr("type") == Some("chat") {
|
||||
if let Some(body) = elem.get_child("body", ns::CLIENT) {
|
||||
self.proxy.dispatch(MessageEvent { // TODO: safety!!!
|
||||
from: elem.attr("from").unwrap().parse().unwrap(),
|
||||
to: elem.attr("to").unwrap().parse().unwrap(),
|
||||
body: body.text(),
|
||||
});
|
||||
}
|
||||
}
|
||||
PluginReturn::Continue
|
||||
}
|
||||
}
|
2
src/plugins/mod.rs
Normal file
2
src/plugins/mod.rs
Normal file
|
@ -0,0 +1,2 @@
|
|||
pub mod messaging;
|
||||
pub mod presence;
|
101
src/plugins/presence.rs
Normal file
101
src/plugins/presence.rs
Normal file
|
@ -0,0 +1,101 @@
|
|||
use error::Error;
|
||||
use plugin::{Plugin, PluginProxy};
|
||||
|
||||
use minidom::Element;
|
||||
|
||||
use ns;
|
||||
|
||||
use std::fmt;
|
||||
|
||||
use std::str::FromStr;
|
||||
|
||||
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
|
||||
pub enum Show {
|
||||
Available,
|
||||
Away,
|
||||
ExtendedAway,
|
||||
DoNotDisturb,
|
||||
Chat,
|
||||
Unavailable,
|
||||
}
|
||||
|
||||
impl fmt::Display for Show {
|
||||
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
||||
match *self {
|
||||
Show::Away => write!(fmt, "away"),
|
||||
Show::ExtendedAway => write!(fmt, "xa"),
|
||||
Show::DoNotDisturb => write!(fmt, "dnd"),
|
||||
Show::Chat => write!(fmt, "chat"),
|
||||
|
||||
// will never be seen inside a <show>, maybe should crash?
|
||||
Show::Available => write!(fmt, "available"),
|
||||
Show::Unavailable => write!(fmt, "unavailable"),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(Debug, Copy, Clone, PartialEq, Eq)]
|
||||
pub struct InvalidShow;
|
||||
|
||||
impl FromStr for Show {
|
||||
type Err = InvalidShow;
|
||||
|
||||
fn from_str(s: &str) -> Result<Show, InvalidShow> {
|
||||
Ok(match s {
|
||||
"away" => Show::Away,
|
||||
"xa" => Show::ExtendedAway,
|
||||
"dnd" => Show::DoNotDisturb,
|
||||
"chat" => Show::Chat,
|
||||
|
||||
_ => { return Err(InvalidShow); }
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
pub struct PresencePlugin {
|
||||
proxy: PluginProxy,
|
||||
}
|
||||
|
||||
impl PresencePlugin {
|
||||
pub fn new() -> PresencePlugin {
|
||||
PresencePlugin {
|
||||
proxy: PluginProxy::new(),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn set_presence(&self, show: Show, status: Option<String>) -> Result<(), Error> {
|
||||
if show == Show::Unavailable {
|
||||
self.proxy.send(Element::builder("presence")
|
||||
.ns(ns::CLIENT)
|
||||
.attr("type", "unavailable")
|
||||
.build());
|
||||
}
|
||||
else {
|
||||
let mut stanza = Element::builder("presence")
|
||||
.ns(ns::CLIENT)
|
||||
.build();
|
||||
if let Some(stat) = status {
|
||||
let mut elem = Element::builder("status")
|
||||
.ns(ns::CLIENT)
|
||||
.build();
|
||||
elem.append_text_node(stat);
|
||||
stanza.append_child(elem);
|
||||
}
|
||||
let mut elem = Element::builder("show")
|
||||
.ns(ns::CLIENT)
|
||||
.build();
|
||||
if show != Show::Available {
|
||||
elem.append_text_node(show.to_string());
|
||||
}
|
||||
stanza.append_child(elem);
|
||||
self.proxy.send(stanza);
|
||||
}
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
impl Plugin for PresencePlugin {
|
||||
fn get_proxy(&mut self) -> &mut PluginProxy {
|
||||
&mut self.proxy
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue