using Gee; using Xmpp; using Xmpp.Xep; namespace Xmpp.Xep.InBandBytestreams { private const string NS_URI = "http://jabber.org/protocol/ibb"; private const int SEQ_MODULUS = 65536; public class Module : XmppStreamModule, Iq.Handler { public static Xmpp.ModuleIdentity IDENTITY = new Xmpp.ModuleIdentity(NS_URI, "0047_in_band_bytestreams"); public override void attach(XmppStream stream) { stream.add_flag(new Flag()); stream.get_module(Iq.Module.IDENTITY).register_for_namespace(NS_URI, this); } public override void detach(XmppStream stream) { } public void on_iq_set(XmppStream stream, Iq.Stanza iq) { // the iq module ensures that there's only one child node StanzaNode? node = null; node = (node != null) ? node : iq.stanza.get_subnode("open", NS_URI); node = (node != null) ? node : iq.stanza.get_subnode("data", NS_URI); node = (node != null) ? node : iq.stanza.get_subnode("close", NS_URI); if (node == null) { stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.error(iq, new ErrorStanza.bad_request("unknown IBB action"))); return; } string? sid = node.get_attribute("sid"); if (sid == null) { stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.error(iq, new ErrorStanza.bad_request("missing sid"))); return; } Connection? conn = stream.get_flag(Flag.IDENTITY).get_connection(sid); if (node.name == "open") { if (conn == null) { stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.error(iq, new ErrorStanza.not_acceptable("unexpected IBB connection"))); return; } if (conn.state != WAITING_FOR_CONNECT) { stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.error(iq, new ErrorStanza.bad_request("IBB open for already open connection"))); return; } conn.handle_open(stream, node, iq); } else { if (conn == null || conn.state != Connection.State.CONNECTED) { stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.error(iq, new ErrorStanza.item_not_found())); return; } if (node.name == "close") { conn.handle_close(stream, node, iq); } else { conn.handle_data(stream, node, iq); } } } public override string get_ns() { return NS_URI; } public override string get_id() { return IDENTITY.id; } } public class Connection : IOStream { // TODO(hrxi): Fix reference cycle public class Input : InputStream { private Connection connection; public Input(Connection connection) { this.connection = connection; } public override ssize_t read(uint8[] buffer, Cancellable? cancellable = null) throws IOError { throw new IOError.NOT_SUPPORTED("can't do non-async reads on in-band bytestreams"); } public override async ssize_t read_async(uint8[]? buffer, int io_priority = GLib.Priority.DEFAULT, Cancellable? cancellable = null) throws IOError { return yield connection.read_async(buffer, io_priority, cancellable); } public override bool close(Cancellable? cancellable = null) throws IOError { return connection.close_read(cancellable); } public override async bool close_async(int io_priority = GLib.Priority.DEFAULT, Cancellable? cancellable = null) throws IOError { return yield connection.close_read_async(io_priority, cancellable); } } public class Output : OutputStream { private Connection connection; public Output(Connection connection) { this.connection = connection; } public override ssize_t write(uint8[] buffer, Cancellable? cancellable = null) throws IOError { throw new IOError.NOT_SUPPORTED("can't do non-async writes on in-band bytestreams"); } public override async ssize_t write_async(uint8[]? buffer, int io_priority = GLib.Priority.DEFAULT, Cancellable? cancellable = null) throws IOError { return yield connection.write_async(buffer, io_priority, cancellable); } public override bool close(Cancellable? cancellable = null) throws IOError { return connection.close_write(cancellable); } public override async bool close_async(int io_priority = GLib.Priority.DEFAULT, Cancellable? cancellable = null) throws IOError { return yield connection.close_write_async(io_priority, cancellable); } } private Input input; private Output output; public override InputStream input_stream { get { return input; } } public override OutputStream output_stream { get { return output; } } public enum State { WAITING_FOR_CONNECT, CONNECTING, CONNECTED, DISCONNECTING, DISCONNECTED, ERROR, } public State state { get; private set; } Jid receiver_full_jid; public string sid { get; private set; } int block_size; int local_seq = 0; int remote_ack = 0; internal int remote_seq = 0; bool input_closed = false; bool output_closed = false; // ERROR string? error = null; XmppStream stream; SourceFunc? read_callback = null; SourceFunc? write_callback = null; // Need `Bytes` instead of `uint8[]` because the latter doesn't work in // parameter position of `LinkedList`. LinkedList received = new LinkedList(); private Connection(XmppStream stream, Jid receiver_full_jid, string sid, int block_size, bool initiate) { this.stream = stream; this.receiver_full_jid = receiver_full_jid; this.sid = sid; this.block_size = block_size; this.state = initiate ? State.CONNECTING : State.WAITING_FOR_CONNECT; input = new Input(this); output = new Output(this); } public void set_read_calllback(SourceFunc callback) throws IOError { if (read_callback != null) { throw new IOError.PENDING("only one async read is permitted at a time on an in-band bytestream"); } read_callback = callback; } public void set_write_calllback(SourceFunc callback) throws IOError { if (write_callback != null) { throw new IOError.PENDING("only one async write is permitted at a time on an in-band bytestream"); } write_callback = callback; } public void trigger_read_callback() { if (read_callback != null) { Idle.add((owned) read_callback); read_callback = null; } } public void trigger_write_callback() { if (write_callback != null) { Idle.add((owned) write_callback); write_callback = null; } } public async ssize_t read_async(uint8[]? buffer, int io_priority = GLib.Priority.DEFAULT, Cancellable? cancellable = null) throws IOError { // TODO(hrxi): cancellable? // TODO(hrxi): io_priority? while (true) { if (input_closed) { return 0; } Bytes? chunk = received.poll(); if (chunk != null) { int read = int.min(buffer.length, chunk.length); for (int i = 0; i < read; i++) { buffer[i] = chunk[i]; } if (buffer.length < chunk.length) { received.offer_head(chunk[buffer.length:chunk.length]); } return read; } if (state == DISCONNECTED) { return 0; } set_read_calllback(read_async.callback); yield; } } public async ssize_t write_async(uint8[]? buffer, int io_priority = GLib.Priority.DEFAULT, Cancellable? cancellable = null) throws IOError { while (state == WAITING_FOR_CONNECT || state == CONNECTING) { set_write_calllback(write_async.callback); yield; } throw_if_closed(); assert(state == CONNECTED); // TODO(hrxi): merging? int seq = local_seq; local_seq = (local_seq + 1) % SEQ_MODULUS; if (buffer.length > block_size) { buffer = buffer[0:block_size]; } StanzaNode data = new StanzaNode.build("data", NS_URI) .add_self_xmlns() .put_attribute("sid", sid) .put_attribute("seq", seq.to_string()) .put_node(new StanzaNode.text(Base64.encode(buffer))); Iq.Stanza iq = new Iq.Stanza.set(data) { to=receiver_full_jid }; set_write_calllback(write_async.callback); stream.get_module(Iq.Module.IDENTITY).send_iq(stream, iq, (stream, iq) => { if (iq.is_error()) { set_error("sending failed"); } else if (remote_ack != seq) { set_error("out of order acks"); } else { remote_ack = (remote_ack + 1) % SEQ_MODULUS; if (local_seq == remote_ack) { trigger_write_callback(); } } }); yield; throw_if_error(); return buffer.length; } public bool close_read(Cancellable? cancellable = null) { input_closed = true; if (!output_closed) { return true; } return close_impl(cancellable); } public async bool close_read_async(int io_priority = GLib.Priority.DEFAULT, Cancellable? cancellable = null) throws IOError { input_closed = true; if (!output_closed) { return true; } return yield close_async_impl(io_priority, cancellable); } public bool close_write(Cancellable? cancellable = null) { output_closed = true; if (!input_closed) { return true; } return close_impl(cancellable); } public async bool close_write_async(int io_priority = GLib.Priority.DEFAULT, Cancellable? cancellable = null) throws IOError { output_closed = true; if (!input_closed) { return true; } return yield close_async_impl(io_priority, cancellable); } delegate void OnClose(bool success); private bool close_impl(Cancellable? cancellable = null, OnClose? on_close = null) { if (state == DISCONNECTING || state == DISCONNECTED || state == ERROR) { on_close(true); return true; } if (state == WAITING_FOR_CONNECT) { state = DISCONNECTED; stream.get_flag(Flag.IDENTITY).remove_connection(this); trigger_read_callback(); on_close(true); return true; } state = DISCONNECTING; StanzaNode close = new StanzaNode.build("close", NS_URI) .add_self_xmlns() .put_attribute("sid", sid); Iq.Stanza iq = new Iq.Stanza.set(close) { to=receiver_full_jid }; stream.get_module(Iq.Module.IDENTITY).send_iq(stream, iq, (stream, iq) => { assert(state == DISCONNECTING); if (iq.is_error()) { set_error("disconnecting failed"); } else { state = DISCONNECTED; } stream.get_flag(Flag.IDENTITY).remove_connection(this); trigger_read_callback(); on_close(!iq.is_error()); }); return true; } private async bool close_async_impl(int io_priority = GLib.Priority.DEFAULT, Cancellable? cancellable = null) throws IOError { SourceFunc callback = close_async_impl.callback; close_impl(cancellable, () => { Idle.add((owned) callback); }); yield; throw_if_error(); return true; } public static Connection create(XmppStream stream, Jid receiver_full_jid, string sid, int block_size, bool initiate) { Connection conn = new Connection(stream, receiver_full_jid, sid, block_size, initiate); if (initiate) { StanzaNode open = new StanzaNode.build("open", NS_URI) .add_self_xmlns() .put_attribute("block-size", block_size.to_string()) .put_attribute("sid", sid); Iq.Stanza iq = new Iq.Stanza.set(open) { to=receiver_full_jid }; stream.get_module(Iq.Module.IDENTITY).send_iq(stream, iq, (stream, iq) => { if (conn.state != CONNECTING) { assert(conn.state != CONNECTED); return; } if (!iq.is_error()) { conn.state = CONNECTED; stream.get_flag(Flag.IDENTITY).add_connection(conn); conn.trigger_write_callback(); } else { conn.set_error("connection failed"); } }); } else { stream.get_flag(Flag.IDENTITY).add_connection(conn); } return conn; } void throw_if_error() throws IOError { if (state == ERROR) { throw new IOError.FAILED(error); } } void throw_if_closed() throws IOError { throw_if_error(); if (state == DISCONNECTING || state == DISCONNECTED) { throw new IOError.CLOSED("can't read/write on a closed connection"); } } void set_error(string error) { if (state != WAITING_FOR_CONNECT && state != DISCONNECTING && state != DISCONNECTED && state != ERROR) { close_async.begin(); } state = ERROR; this.error = error; stream.get_flag(Flag.IDENTITY).remove_connection(this); } public void handle_open(XmppStream stream, StanzaNode open, Iq.Stanza iq) { assert(state == WAITING_FOR_CONNECT); int block_size = open.get_attribute_int("block-size"); string? stanza = open.get_attribute("stanza"); if (block_size < 0 || (stanza != null && stanza != "iq" && stanza != "message")) { set_error("invalid open"); stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.error(iq, new ErrorStanza.bad_request("missing block_size or invalid stanza"))); return; } if (stanza != null && stanza != "iq") { set_error("invalid open"); stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.error(iq, new ErrorStanza.feature_not_implemented("cannot use message stanzas for IBB"))); return; } if (block_size > this.block_size) { set_error("invalid open"); stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.error(iq, new ErrorStanza.build(ErrorStanza.TYPE_CANCEL, ErrorStanza.CONDITION_RESOURCE_CONSTRAINT, "opening a connection with a greater than negotiated/acceptable block size", null))); return; } this.block_size = block_size; state = CONNECTED; stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.result(iq)); trigger_write_callback(); } public void handle_data(XmppStream stream, StanzaNode data, Iq.Stanza iq) { assert(state == CONNECTED); if (input_closed) { set_error("unexpected data"); stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.error(iq, new ErrorStanza.not_allowed("unexpected data"))); return; } int seq = data.get_attribute_int("seq"); // TODO(hrxi): return an error on malformed base64 (need to do this // according to the xep) uint8[] content = Base64.decode(data.get_string_content()); if (content.length > block_size) { set_error("data longer than negotiated block size"); stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.error(iq, new ErrorStanza.bad_request("data longer than negotiated block size"))); return; } if (seq < 0 || seq != remote_seq) { set_error("out of order data packets"); stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.error(iq, new ErrorStanza.build(ErrorStanza.TYPE_CANCEL, ErrorStanza.CONDITION_UNEXPECTED_REQUEST, "out of order data packets", null))); return; } remote_seq = (remote_seq + 1) % SEQ_MODULUS; stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.result(iq)); if (content.length != 0) { received.offer(new Bytes.take(content)); trigger_read_callback(); } } public void handle_close(XmppStream stream, StanzaNode close, Iq.Stanza iq) { assert(state == CONNECTED); stream.get_module(Iq.Module.IDENTITY).send_iq(stream, new Iq.Stanza.result(iq)); stream.get_flag(Flag.IDENTITY).remove_connection(this); input_closed = true; output_closed = true; state = DISCONNECTED; trigger_read_callback(); } } public class Flag : XmppStreamFlag { public static FlagIdentity IDENTITY = new FlagIdentity(NS_URI, "in_band_bytestreams"); private HashMap active = new HashMap(); public void add_connection(Connection conn) { active[conn.sid] = conn; } public Connection? get_connection(string sid) { return active.has_key(sid) ? active[sid] : null; } public void remove_connection(Connection conn) { active.unset(conn.sid); } public override string get_ns() { return NS_URI; } public override string get_id() { return IDENTITY.id; } } }