Mercurial > prosody-modules
comparison mod_debug_omemo/mod_debug_omemo.lua @ 4682:e4e5474420e6
mod_debug_omemo: OMEMO debugging tool
author | Matthew Wild <mwild1@gmail.com> |
---|---|
date | Mon, 13 Sep 2021 19:24:13 +0100 |
parents | |
children | 07b6f444bafb |
comparison
equal
deleted
inserted
replaced
4681:edbd84bbd8fb | 4682:e4e5474420e6 |
---|---|
1 local array = require "util.array"; | |
2 local jid = require "util.jid"; | |
3 local set = require "util.set"; | |
4 local st = require "util.stanza"; | |
5 local url_escape = require "util.http".urlencode; | |
6 | |
7 local base_url = "https://"..module.host.."/"; | |
8 | |
9 local render_html_template = require"util.interpolation".new("%b{}", st.xml_escape, { | |
10 urlescape = url_escape; | |
11 lower = string.lower; | |
12 classname = function (s) return (s:gsub("%W+", "-")); end; | |
13 relurl = function (s) | |
14 if s:match("^%w+://") then | |
15 return s; | |
16 end | |
17 return base_url.."/"..s; | |
18 end; | |
19 }); | |
20 local render_url = require "util.interpolation".new("%b{}", url_escape, { | |
21 urlescape = url_escape; | |
22 noscheme = function (url) | |
23 return (url:gsub("^[^:]+:", "")); | |
24 end; | |
25 }); | |
26 | |
27 local mod_pep = module:depends("pep"); | |
28 | |
29 local mam = module:open_store("archive", "archive"); | |
30 | |
31 local function get_user_omemo_info(username) | |
32 local everything_valid = true; | |
33 local any_device = false; | |
34 local omemo_status = {}; | |
35 local omemo_devices; | |
36 local pep_service = mod_pep.get_pep_service(username); | |
37 if pep_service and pep_service.nodes then | |
38 local ok, _, device_list = pep_service:get_last_item("eu.siacs.conversations.axolotl.devicelist", true); | |
39 if ok and device_list then | |
40 device_list = device_list:get_child("list", "eu.siacs.conversations.axolotl"); | |
41 end | |
42 if device_list then | |
43 omemo_devices = {}; | |
44 for device_entry in device_list:childtags("device") do | |
45 any_device = true; | |
46 local device_info = {}; | |
47 local device_id = tonumber(device_entry.attr.id or ""); | |
48 if device_id then | |
49 device_info.id = device_id; | |
50 local bundle_id = ("eu.siacs.conversations.axolotl.bundles:%d"):format(device_id); | |
51 local have_bundle, _, bundle = pep_service:get_last_item(bundle_id, true); | |
52 if have_bundle and bundle and bundle:get_child("bundle", "eu.siacs.conversations.axolotl") then | |
53 device_info.have_bundle = true; | |
54 local config_ok, bundle_config = pep_service:get_node_config(bundle_id, true); | |
55 if config_ok and bundle_config then | |
56 device_info.bundle_config = bundle_config; | |
57 if bundle_config.max_items == 1 | |
58 and bundle_config.access_model == "open" | |
59 and bundle_config.persist_items == true | |
60 and bundle_config.publish_model == "publishers" then | |
61 device_info.valid = true; | |
62 end | |
63 end | |
64 end | |
65 end | |
66 if device_info.valid == nil then | |
67 device_info.valid = false; | |
68 everything_valid = false; | |
69 end | |
70 table.insert(omemo_devices, device_info); | |
71 end | |
72 | |
73 local config_ok, list_config = pep_service:get_node_config("eu.siacs.conversations.axolotl.devicelist", true); | |
74 if config_ok and list_config then | |
75 omemo_status.config = list_config; | |
76 if list_config.max_items == 1 | |
77 and list_config.access_model == "open" | |
78 and list_config.persist_items == true | |
79 and list_config.publish_model == "publishers" then | |
80 omemo_status.config_valid = true; | |
81 end | |
82 end | |
83 if omemo_status.config_valid == nil then | |
84 omemo_status.config_valid = false; | |
85 everything_valid = false; | |
86 end | |
87 end | |
88 end | |
89 omemo_status.valid = everything_valid and any_device; | |
90 return { | |
91 status = omemo_status; | |
92 devices = omemo_devices; | |
93 }; | |
94 end | |
95 | |
96 local access_model_text = { | |
97 open = "Public"; | |
98 whitelist = "Private"; | |
99 roster = "Contacts only"; | |
100 presence = "Contacts only"; | |
101 }; | |
102 | |
103 local function render_message(event, path) | |
104 local username, message_id = path:match("^([^/]+)/(.+)$"); | |
105 if not username then | |
106 return 400; | |
107 end | |
108 local message; | |
109 for _, result in mam:find(username, { key = message_id }) do | |
110 message = result; | |
111 end | |
112 if not message then | |
113 return 404; | |
114 end | |
115 | |
116 local user_omemo_status = get_user_omemo_info(username); | |
117 | |
118 local user_rids = set.new(array.pluck(user_omemo_status.devices or {}, "id")) / tostring; | |
119 | |
120 local message_omemo_header = message:find("{eu.siacs.conversations.axolotl}encrypted/header"); | |
121 local message_rids = set.new(); | |
122 local rid_info = {}; | |
123 if message_omemo_header then | |
124 for key_el in message_omemo_header:childtags("key") do | |
125 local rid = key_el.attr.rid; | |
126 if rid then | |
127 message_rids:add(rid); | |
128 local prekey = key_el.attr.prekey; | |
129 rid_info = { | |
130 prekey = prekey and (prekey == "1" or prekey:lower() == "true"); | |
131 }; | |
132 end | |
133 end | |
134 end | |
135 | |
136 local rids = user_rids + message_rids; | |
137 | |
138 local direction = jid.bare(message.attr.to) == (username.."@"..module.host) and "incoming" or "outgoing"; | |
139 | |
140 local is_encrypted = not not message_omemo_header; | |
141 | |
142 local sender_id = message_omemo_header and message_omemo_header.attr.sid or nil; | |
143 | |
144 local f = module:load_resource("view.tpl.html"); | |
145 if not f then | |
146 return 500; | |
147 end | |
148 local tpl = f:read("*a"); | |
149 | |
150 local data = { user = username, rids = {} }; | |
151 for rid in rids do | |
152 data.rids[rid] = { | |
153 status = message_rids:contains(rid) and "Encrypted" or user_rids:contains(rid) and "Missing" or nil; | |
154 prekey = rid_info.prekey; | |
155 }; | |
156 end | |
157 | |
158 data.message = { | |
159 type = message.attr.type or "normal"; | |
160 direction = direction; | |
161 encryption = is_encrypted and "encrypted" or "unencrypted"; | |
162 }; | |
163 | |
164 data.omemo = { | |
165 sender_id = sender_id; | |
166 status = user_omemo_status.status.valid and "no known issues" or "problems"; | |
167 }; | |
168 | |
169 data.omemo.devices = {}; | |
170 for _, device_info in ipairs(user_omemo_status.devices) do | |
171 data.omemo.devices[("%d"):format(device_info.id)] = { | |
172 status = device_info.valid and "OK" or "Problem"; | |
173 bundle = device_info.have_bundle and "Published" or "Missing"; | |
174 access_model = access_model_text[device_info.bundle_config and device_info.bundle_config.access_model or nil]; | |
175 }; | |
176 end | |
177 | |
178 event.response.headers.content_type = "text/html; charset=utf-8"; | |
179 return render_html_template(tpl, data); | |
180 end | |
181 | |
182 local function check_omemo_fallback(event) | |
183 local message = event.stanza; | |
184 | |
185 local message_omemo_header = message:find("{eu.siacs.conversations.axolotl}encrypted/header"); | |
186 if not message_omemo_header then return; end | |
187 | |
188 local to_bare = jid.bare(message.attr.to); | |
189 | |
190 local archive_stanza_id; | |
191 for stanza_id_tag in message:childtags("stanza-id", "urn:xmpp:sid:0") do | |
192 if stanza_id_tag.attr.by == to_bare then | |
193 archive_stanza_id = stanza_id_tag.attr.id; | |
194 end | |
195 end | |
196 if not archive_stanza_id then | |
197 return; | |
198 end | |
199 | |
200 local debug_url = render_url(module:http_url().."/view/{username}/{message_id}", { | |
201 username = jid.node(to_bare); | |
202 message_id = archive_stanza_id; | |
203 }); | |
204 | |
205 local body = message:get_child("body"); | |
206 if not body then | |
207 body = st.stanza("body") | |
208 :text("This message is encrypted using OMEMO, but could not be decrypted by your device.\nFor more information see: "..debug_url); | |
209 message:reset():add_child(body); | |
210 else | |
211 body:text("\n\nOMEMO debug information: "..debug_url); | |
212 end | |
213 end | |
214 | |
215 module:hook("message/bare", check_omemo_fallback, 1); | |
216 module:hook("message/full", check_omemo_fallback, 1); | |
217 | |
218 module:depends("http") | |
219 module:provides("http", { | |
220 route = { | |
221 ["GET /view/*"] = render_message; | |
222 }; | |
223 }); |