aboutsummaryrefslogtreecommitdiff
path: root/xmpp-vala/src/module/xep/0313_message_archive_management.vala
diff options
context:
space:
mode:
Diffstat (limited to 'xmpp-vala/src/module/xep/0313_message_archive_management.vala')
-rw-r--r--xmpp-vala/src/module/xep/0313_message_archive_management.vala126
1 files changed, 126 insertions, 0 deletions
diff --git a/xmpp-vala/src/module/xep/0313_message_archive_management.vala b/xmpp-vala/src/module/xep/0313_message_archive_management.vala
new file mode 100644
index 00000000..522f6dca
--- /dev/null
+++ b/xmpp-vala/src/module/xep/0313_message_archive_management.vala
@@ -0,0 +1,126 @@
+using Xmpp.Core;
+
+namespace Xmpp.Xep.MessageArchiveManagement {
+
+public const string NS_URI = "urn:xmpp:mam:2";
+public const string NS_URI_1 = "urn:xmpp:mam:1";
+
+public class Module : XmppStreamModule {
+ public static ModuleIdentity<Module> IDENTITY = new ModuleIdentity<Module>(NS_URI, "0313_message_archive_management");
+
+ public signal void feature_available(XmppStream stream);
+
+ public void query_archive(XmppStream stream, string? jid, DateTime? start, DateTime? end) {
+ if (stream.get_flag(Flag.IDENTITY) == null) return;
+
+ DataForms.DataForm data_form = new DataForms.DataForm();
+ DataForms.DataForm.HiddenField form_type_field = new DataForms.DataForm.HiddenField() { var="FORM_TYPE" };
+ form_type_field.set_value_string(NS_VER(stream));
+ data_form.add_field(form_type_field);
+ if (jid != null) {
+ DataForms.DataForm.Field field = new DataForms.DataForm.Field() { var="with" };
+ field.set_value_string(jid);
+ data_form.add_field(field);
+ }
+ if (start != null) {
+ DataForms.DataForm.Field field = new DataForms.DataForm.Field() { var="start" };
+ field.set_value_string(DateTimeProfiles.to_datetime(start));
+ data_form.add_field(field);
+ }
+ if (end != null) {
+ DataForms.DataForm.Field field = new DataForms.DataForm.Field() { var="end" };
+ field.set_value_string(DateTimeProfiles.to_datetime(end));
+ data_form.add_field(field);
+ }
+ StanzaNode query_node = new StanzaNode.build("query", NS_VER(stream)).add_self_xmlns().put_node(data_form.get_submit_node());
+ Iq.Stanza iq = new Iq.Stanza.set(query_node);
+ stream.get_module(Iq.Module.IDENTITY).send_iq(stream, iq, page_through_results);
+ }
+
+ public override void attach(XmppStream stream) {
+ stream.get_module(Message.Module.IDENTITY).pre_received_message.connect(on_pre_received_message);
+ stream.stream_negotiated.connect(query_availability);
+ }
+
+ public override void detach(XmppStream stream) { }
+
+ public override string get_ns() { return NS_URI; }
+ public override string get_id() { return IDENTITY.id; }
+
+ private void on_pre_received_message(XmppStream stream, Message.Stanza message) {
+// if (message.from != stream.remote_name) return;
+ if (stream.get_flag(Flag.IDENTITY) == null) return;
+
+ StanzaNode? message_node = message.stanza.get_deep_subnode(NS_VER(stream) + ":result", "urn:xmpp:forward:0:forwarded", Message.NS_URI + ":message");
+ if (message_node != null) {
+ StanzaNode? forward_node = message.stanza.get_deep_subnode(NS_VER(stream) + ":result", "urn:xmpp:forward:0:forwarded", DelayedDelivery.NS_URI + ":delay");
+ DateTime? datetime = DelayedDelivery.Module.get_time_for_node(forward_node);
+ message.add_flag(new MessageFlag(datetime));
+
+ message.stanza = message_node;
+ message.rerun_parsing = true;
+ }
+ }
+
+ private static void page_through_results(XmppStream stream, Iq.Stanza iq) {
+ string? last = iq.stanza.get_deep_string_content(NS_VER(stream) + ":fin", "http://jabber.org/protocol/rsm" + ":set", "last");
+ if (last == null) {
+ stream.get_flag(Flag.IDENTITY).cought_up = true;
+ return;
+ }
+
+ Iq.Stanza paging_iq = new Iq.Stanza.set(
+ new StanzaNode.build("query", NS_VER(stream)).add_self_xmlns().put_node(
+ new StanzaNode.build("set", "http://jabber.org/protocol/rsm").add_self_xmlns().put_node(
+ new StanzaNode.build("after", "http://jabber.org/protocol/rsm").put_node(new StanzaNode.text(last))
+ )
+ )
+ );
+ stream.get_module(Iq.Module.IDENTITY).send_iq(stream, paging_iq, page_through_results);
+ }
+
+ private void query_availability(XmppStream stream) {
+ stream.get_module(Xep.ServiceDiscovery.Module.IDENTITY).request_info(stream, get_bare_jid(stream.get_flag(Bind.Flag.IDENTITY).my_jid), (stream, info_result) => {
+ if (info_result.features.contains(NS_URI)) {
+ stream.add_flag(new Flag(NS_URI));
+ } else if (info_result.features.contains(NS_URI_1)) {
+ stream.add_flag(new Flag(NS_URI_1));
+ }
+ if (stream.get_flag(Flag.IDENTITY) != null) feature_available(stream);
+ });
+ }
+
+ private static string NS_VER(XmppStream stream) {
+ return stream.get_flag(Flag.IDENTITY).ns_ver;
+ }
+}
+
+public class Flag : XmppStreamFlag {
+ public static FlagIdentity<Flag> IDENTITY = new FlagIdentity<Flag>(NS_URI, "message_archive_management");
+ public bool cought_up { get; set; default=false; }
+ public string ns_ver;
+
+ public Flag(string ns_ver) {
+ this.ns_ver = ns_ver;
+ }
+
+ public override string get_ns() { return NS_URI; }
+ public override string get_id() { return IDENTITY.id; }
+}
+
+public class MessageFlag : Message.MessageFlag {
+ public const string ID = "message_archive_management";
+
+ public DateTime? server_time { get; private set; }
+
+ public MessageFlag(DateTime? server_time) {
+ this.server_time = server_time;
+ }
+
+ public static MessageFlag? get_flag(Message.Stanza message) { return (MessageFlag) message.get_flag(NS_URI, ID); }
+
+ public override string get_ns() { return NS_URI; }
+ public override string get_id() { return ID; }
+}
+
+}