# HG changeset patch # User Kim Alvefur # Date 1337628651 -7200 # Node ID 343b115ebbeae9a6de406191d289e9db90dcaa4b # Parent ea994181272115cb9f6dd62fceba9e3ca50bb55e mod_pubsub_feed: Cleanup and update to new APIs in 0.9 diff -r ea9941812721 -r 343b115ebbea mod_pubsub_feed/mod_pubsub_feed.lua --- a/mod_pubsub_feed/mod_pubsub_feed.lua Mon May 21 17:02:15 2012 +0200 +++ b/mod_pubsub_feed/mod_pubsub_feed.lua Mon May 21 21:30:51 2012 +0200 @@ -16,29 +16,34 @@ -- http://pubsubhubbub.googlecode.com/svn/trunk/pubsubhubbub-core-0.3.html local modules = hosts[module.host].modules; -if not modules.pubsub then - --FIXME Should this throw an error() instead? +if not modules.pubsub or module:get_option("component_module") ~= "pubsub" then module:log("warn", "Pubsub needs to be loaded on this host"); + --module:log("debug", "component_module is %s", tostring(module:get_option("component_module"))); + return end - -local t_insert = table.insert; -local add_task = require "util.timer".add_task; local date, time = os.date, os.time; local dt_parse, dt_datetime = require "util.datetime".parse, require "util.datetime".datetime; local uuid = require "util.uuid".generate; local hmac_sha1 = require "util.hmac".sha1; local parse_feed = require "feeds".feed_from_string; local st = require "util.stanza"; +--local dump = require"util.serialization".serialize; + +local xmlns_atom = "http://www.w3.org/2005/Atom"; + +local use_pubsubhubub = module:get_option_boolean("use_pubsubhubub", true); +if use_pubsubhubub then + module:depends"http"; +end local http = require "net.http"; -local httpserver = require "net.httpserver"; local formdecode = http.formdecode; local formencode = http.formencode; local urldecode = http.urldecode; local urlencode = http.urlencode; -local feed_list = {}; +local feed_list = module:shared("feed_list"); local refresh_interval; -- Dynamically reloadable config. @@ -66,64 +71,14 @@ update_config(); module:hook("config-reloaded", update_config); --- Used to kill the timer -local module_unloaded = false; -function module.unload() - module_unloaded = true; -end - --- Config stuff that can't be reloaded, since it would need to re-bind HTTP stuff. - --- If module.host IN A doesn't point to this server, use this to override. -local httphost = module:get_option_string("pubsubhubub_httphost", module.host); --- HTTP by default or not? -local use_pubsubhubub = module:get_option_boolean("use_pubsubhubub", true); - --- Thanks to Maranda for this -local port, base, ssl = 5280, "callback", false; -local ports = module:get_option("feeds_ports") or { port = port, base = base, ssl = ssl }; --- FIXME If ports isn't a table, this will cause an error -local _, first_port = next(ports); -- We base the callback URL on the first port config -if first_port then - if type(first_port) == "number" then - port = first_port; - elseif type(first_port) == "table" then - port, base, ssl = - first_port.port or port, - first_port.path or base, - first_port.ssl or ssl; - elseif type(first_port) == "string" then - base = first_port; - end -end - -local response_codes = { - ["200"] = "OK"; - ["202"] = "Accepted"; - ["400"] = "Bad Request"; - ["403"] = "Forbidden"; - ["404"] = "Not Found"; - ["500"] = "Internal Server Error"; - ["501"] = "Not Implemented"; -}; - -local function http_response(code, headers, body) - return { - status = (type(code) == "number" and code .. " " .. response_codes[tostring(code)]) or code; - headers = headers or {}; - body = body or "

" .. response_codes[tostring(code)] .. "

\n"; - }; -end - local actor = module.host.."/"..module.name; function update_entry(item) local node = item.node; - --module:log("debug", "parsing %d bytes of data in node %s", #item.data or 0, node) + module:log("debug", "parsing %d bytes of data in node %s", #item.data or 0, node) local feed = parse_feed(item.data); - module:log("debug", "updating node %s", node); for _, entry in ipairs(feed) do - entry.attr.xmlns = "http://www.w3.org/2005/Atom"; + entry.attr.xmlns = xmlns_atom; local e_published = entry:get_child_text("published"); e_published = e_published and dt_parse(e_published); @@ -138,11 +93,11 @@ local xitem = st.stanza("item", { id = id }):add_child(entry); -- TODO Put data from /feed into item/source - module:log("debug", "publishing to %s, id %s", node, id); + --module:log("debug", "publishing to %s, id %s", node, id); local ok, err = modules.pubsub.service:publish(node, actor, id, xitem); if not ok then if err == "item-not-found" then -- try again - module:log("debug", "got item-not-found, creating %s and trying again", node); + --module:log("debug", "got item-not-found, creating %s and trying again", node); local ok, err = modules.pubsub.service:create(node, actor); if not ok then module:log("error", "could not create node %s: %s", node, err); @@ -161,7 +116,7 @@ end if use_pubsubhubub and not item.subscription then - module:log("debug", "check if %s has a hub", item.node); + --module:log("debug", "check if %s has a hub", item.node); local hub = feed.links and feed.links.hub; if hub then item.hub = hub; @@ -175,15 +130,13 @@ local headers = { }; if item.data and item.last_update then headers["If-Modified-Since"] = date("!%a, %d %b %Y %H:%M:%S %Z", item.last_update); - --COMPAT We could have saved 6 bytes here, but Microsoft apparently hates %T, so you got this gigantic comment instead. end http.request(item.url, { headers = headers }, function(data, code, req) if code == 200 then item.data = data; if callback then callback(item) end item.last_update = time(); - end - if code == 304 then + elseif code == 304 then item.last_update = time(); end end); @@ -191,28 +144,27 @@ function refresh_feeds() local now = time(); - if module_unloaded then return end --module:log("debug", "Refreshing feeds"); for node, item in pairs(feed_list) do --FIXME Don't fetch feeds which have a subscription -- Otoho, what if the subscription expires or breaks? if item.last_update + refresh_interval < now then - module:log("debug", "checking %s", item.node); + --module:log("debug", "checking %s", item.node); fetch(item, update_entry); end end return refresh_interval; end -local function format_url(secure, host, port, path, node) - return ("%s://%s:%d/%s?node=%s"):format(secure and "https" or "http", host, port, path, urlencode(node)); +local function format_url(node) + return module:http_url(nil, "/callback") .. "?node=" .. urlencode(node); end function subscribe(feed) feed.token = uuid(); feed.secret = uuid(); local body = formencode{ - ["hub.callback"] = format_url(ssl, httphost, port, base, feed.node); + ["hub.callback"] = format_url(feed.node); ["hub.mode"] = "subscribe"; --TODO unsubscribe ["hub.topic"] = feed.url; ["hub.verify"] = "async"; @@ -226,35 +178,37 @@ --FIXME The subscription states and related stuff feed.subscription = "subscribe"; http.request(feed.hub, { body = body }, function(data, code, req) - local code = tostring(code); - module:log("debug", "subscription to %s submitted, status %s", feed.node, code); + module:log("debug", "subscription to %s submitted, status %s", feed.node, tostring(code)); + if code >= 400 then + module:log("error", "There was something wrong with our subscription request, body: %s", tostring(data)); + feed.subscription = "failed"; + end end); end -function handle_http_request(method, body, request) - if module_unloaded then - module:log("warn", "Received a HTTP request after module unload"); - return http_response(500) - -- FIXME if this happens. - end +function handle_http_request(event) + local request = event.request; + local method = request.method; + local body = request.body; + --module:log("debug", "%s request to %s%s with body %s", method, request.url.path, request.url.query and "?" .. request.url.query or "", #body > 0 and body or "empty"); - local query = request.url.query or {}; + local query = request.url.query or {}; --FIXME if query and type(query) == "string" then query = formdecode(query); --module:log("debug", "GET data: %s", dump(query)); end --module:log("debug", "Headers: %s", dump(request.headers)); + local feed = feed_list[query.node]; if method == "GET" then - if query.node and feed_list[query.node] then - local feed = feed_list[query.node]; + if query.node and feed then if query["hub.topic"] ~= feed.url then module:log("debug", "Invalid topic: %s", tostring(query["hub.topic"])) - return http_response(404) + return 404 end if query["hub.mode"] ~= feed.subscription then module:log("debug", "Invalid mode: %s", tostring(query["hub.mode"])) - return http_response(400) + return 400 -- Would this work for unsubscribe? -- Also, if feed.subscription is changed here, -- it would probably invalidate the subscription @@ -262,46 +216,44 @@ end if query["hub.verify_token"] ~= feed.token then module:log("debug", "Invalid verify_token: %s", tostring(query["hub.verify_token"])) - return http_response(403) + return 401 end module:log("debug", "Confirming %s request to %s", feed.subscription, feed.url) - return http_response(200, nil, query["hub.challenge"]) + return query["hub.challenge"]; end - return http_response(400); + return 400; elseif method == "POST" then - if #body > 0 and feed_list[query.node] then + local body = request.body; + if #body > 0 and feed then module:log("debug", "got %d bytes PuSHed for %s", #body, query.node); - local feed = feed_list[query.node]; - local signature = request.headers["x-hub-signature"]; + local signature = request.headers.x_hub_signature; if feed.secret then local localsig = "sha1=" .. hmac_sha1(feed.secret, body, true); if localsig ~= signature then - module:log("debug", "Invalid signature"); - return http_response(403); + module:log("debug", "Invalid signature, got %s but wanted %s", tostring(signature), tostring(localsig)); + return 401; end module:log("debug", "Valid signature"); end feed.data = body; update_entry(feed); feed.last_update = time(); - return http_response(202); + return 202; end - return http_response(400); + return 400; end - return http_response(501); + return 501; end -function init() - module:log("debug", "initiating", module.name); - if use_pubsubhubub then - module:log("debug", "Starting http server on %s", format_url(ssl, httphost, port, base, "NODE")); - httpserver.new_from_config( ports, handle_http_request, { base = "callback" } ); - end - add_task(0, refresh_feeds); +if use_pubsubhubub then + module:provides("http", { + default_path = "/callback"; + route = { + GET = handle_http_request; + POST = handle_http_request; + -- This all? + }; + }); end -if prosody.start_time then -- already started - init(); -else - module:hook_global("server-started", init); -end +module:add_timer(1, refresh_feeds);