Mercurial > prosody-modules
comparison mod_http_upload/mod_http_upload.lua @ 1772:45f7e3c2557f
mod_http_upload: Implementation of Conversations HTTP upload file transfer mode
author | Kim Alvefur <zash@zash.se> |
---|---|
date | Sat, 18 Jul 2015 23:05:09 +0200 |
parents | |
children | 25c28644fae8 |
comparison
equal
deleted
inserted
replaced
1771:398a007da84e | 1772:45f7e3c2557f |
---|---|
1 -- mod_http_upload | |
2 -- | |
3 -- Copyright (C) 2015 Kim Alvefur | |
4 -- | |
5 -- This file is MIT/X11 licensed. | |
6 -- | |
7 -- Implementation of HTTP Upload file transfer mechanism used by Conversations | |
8 -- | |
9 | |
10 -- imports | |
11 local st = require"util.stanza"; | |
12 local lfs = require"lfs"; | |
13 local join_path = require"util.paths".join; | |
14 local uuid = require"util.uuid".generate; | |
15 | |
16 -- depends | |
17 module:depends("http"); | |
18 | |
19 -- namespace | |
20 local xmlns_http_upload = "eu:siacs:conversations:http:upload"; | |
21 | |
22 module:add_feature(xmlns_http_upload); | |
23 | |
24 -- state | |
25 local pending_slots = module:shared("upload_slots"); | |
26 | |
27 local storage_path = join_path(prosody.paths.data, module.name); | |
28 lfs.mkdir(storage_path); | |
29 | |
30 -- hooks | |
31 module:hook("iq/host/"..xmlns_http_upload..":request", function (event) | |
32 local stanza, origin = event.stanza, event.origin; | |
33 -- local clients only | |
34 if origin.type ~= "c2s" then | |
35 origin.send(st.error_reply(stanza, "cancel", "not-authorized")); | |
36 return true; | |
37 end | |
38 -- validate | |
39 local filename = stanza.tags[1]:get_child_text("filename"); | |
40 if not filename or filename:find("/") then | |
41 origin.send(st.error_reply(stanza, "modify", "bad-request")); | |
42 return true; | |
43 end | |
44 local reply = st.reply(stanza); | |
45 reply:tag("slot", { xmlns = xmlns_http_upload }); | |
46 local random = uuid(); | |
47 pending_slots[random.."/"..filename] = origin.full_jid; | |
48 local url = module:http_url() .. "/" .. random .. "/" .. filename; | |
49 reply:tag("get"):text(url):up(); | |
50 reply:tag("put"):text(url):up(); | |
51 origin.send(reply); | |
52 return true; | |
53 end); | |
54 | |
55 -- http service | |
56 local function upload_data(event, path) | |
57 if not pending_slots[path] then | |
58 return 401; | |
59 end | |
60 local random, filename = path:match("^([^/]+)/([^/]+)$"); | |
61 if not random then | |
62 return 400; | |
63 end | |
64 local dirname = join_path(storage_path, random); | |
65 if not lfs.mkdir(dirname) then | |
66 module:log("error", "Could not create directory %s for upload", dirname); | |
67 return 500; | |
68 end | |
69 local full_filename = join_path(dirname, filename); | |
70 local fh, ferr = io.open(full_filename, "w"); | |
71 if not fh then | |
72 module:log("error", "Could not open file %s for upload: %s", full_filename, ferr); | |
73 return 500; | |
74 end | |
75 local ok, err = fh:write(event.request.body); | |
76 if not ok then | |
77 module:log("error", "Could not write to file %s for upload: %s", full_filename, err); | |
78 os.remove(full_filename); | |
79 return 500; | |
80 end | |
81 ok, err = fh:close(); | |
82 if not ok then | |
83 module:log("error", "Could not write to file %s for upload: %s", full_filename, err); | |
84 os.remove(full_filename); | |
85 return 500; | |
86 end | |
87 module:log("info", "File uploaded by %s to slot %s", pending_slots[path], random); | |
88 pending_slots[path] = nil; | |
89 return 200; | |
90 end | |
91 | |
92 local serve_uploaded_files = module:depends("http_files").serve(storage_path); | |
93 | |
94 local function size_only(request, data) | |
95 request.headers.content_size = #data; | |
96 return 200; | |
97 end | |
98 | |
99 local function serve_head(event, path) | |
100 event.send = size_only; | |
101 return serve_uploaded_files(event, path); | |
102 end | |
103 | |
104 module:provides("http", { | |
105 route = { | |
106 ["GET /*"] = serve_uploaded_files; | |
107 ["HEAD /*"] = serve_head; | |
108 ["PUT /*"] = upload_data; | |
109 }; | |
110 }); |