Changeset

288:9233d7ee3c09

mod_admin_web: Initial PoC commit
author Florian Zeitz <florob@babelmonkeys.de>
date Fri, 17 Dec 2010 03:34:53 +0100 (2010-12-17)
parents 287:6144fe6161f1
children 289:415034fd38c2
files mod_admin_web/admin_web/get_deps.lua mod_admin_web/admin_web/mod_admin_web.lua mod_admin_web/admin_web/www_files/index.html mod_admin_web/admin_web/www_files/js/main.js mod_admin_web/admin_web/www_files/style.css
diffstat 5 files changed, 405 insertions(+), 0 deletions(-) [+]
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/mod_admin_web/admin_web/get_deps.lua	Fri Dec 17 03:34:53 2010 +0100
@@ -0,0 +1,30 @@
+#!/usr/bin/env lua
+
+files = {
+	["www_files/strophejs.tar.gz"] = "http://download.github.com/metajack-strophejs-release-1.0-0-g1581b37.tar.gz";
+	["www_files/js/jquery-1.4.4.min.js"] = "http://code.jquery.com/jquery-1.4.4.min.js";
+}
+
+function fetch(url)
+	local http = require "socket.http";
+	local body, status = http.request(url);
+	if status == 200 then
+		return body;
+	end
+	return false, "HTTP status code: "..tostring(status);
+end
+
+for filename, url in pairs(files) do
+	file = io.open(filename, "w+");
+	data, error = fetch(url);
+	if data then
+		file:write(data);
+	else
+		print("Error: " .. error);
+	end
+	file:close();
+end
+
+os.execute("cd www_files && tar xzf strophejs.tar.gz");
+os.execute("cd www_files/metajack-strophejs-3ada7f5 && make strophe.js && cp strophe.js ../js/strophe.js");
+os.execute("rm -r www_files/strophejs.tar.gz www_files/metajack-strophejs-3ada7f5");
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/mod_admin_web/admin_web/mod_admin_web.lua	Fri Dec 17 03:34:53 2010 +0100
@@ -0,0 +1,150 @@
+-- Copyright (C) 2010 Florian Zeitz
+--
+-- This file is MIT/X11 licensed. Please see the
+-- COPYING file in the source package for more information.
+--
+
+-- <session xmlns="http://prosody.im/streams/s2s" jid="example.com">
+--   <encrypted/>
+--   <compressed/>
+--   <in/> / <out/>
+-- </session>
+
+local stanza = require "util.stanza";
+local uuid_generate = require "util.uuid".generate;
+local httpserver = require "net.httpserver";
+local lfs = require "lfs";
+local open = io.open;
+local stat = lfs.attributes;
+
+local host = module:get_host();
+local service = config.get("*", "core", "webadmin_pubsub_host") or ("pubsub." .. host);
+
+local http_base = (prosody.paths.plugins or "./plugins/") .. "admin_web/www_files";
+
+local xmlns_sessions = "http://prosody.im/streams/s2s";
+
+local response_400 = { status = "400 Bad Request", body = "<h1>Bad Request</h1>Sorry, we didn't understand your request :(" };
+local response_403 = { status = "403 Forbidden", body = "<h1>Forbidden</h1>You don't have permission to view the contents of this directory :(" };
+local response_404 = { status = "404 Not Found", body = "<h1>Page Not Found</h1>Sorry, we couldn't find what you were looking for :(" };
+
+local mime_map = {
+	html = "text/html";
+	xml = "text/xml";
+	js = "text/javascript";
+	css = "text/css";
+};
+
+local idmap = {};
+
+function add_host(session, type)
+	local name = (type == "out" and session.to_host) or (type == "in" and session.from_host);
+	local id = idmap[name.."_"..type];
+	if not id then
+		id = uuid_generate();
+		idmap[name.."_"..type] = id;
+	end
+	local item = stanza.stanza("item", { id = id }):tag("session", {xmlns = xmlns_sessions, jid = name})
+		:tag(type):up();
+	if session.secure then
+		item:tag("encrypted"):up();
+	end
+	if session.compressed then
+		item:tag("compressed"):up();
+	end
+	hosts[service].modules.pubsub.service:publish(xmlns_sessions, host, id, item);
+	module:log("debug", "Added host " .. name .. " s2s" .. type);
+end
+
+function del_host(session, type)
+	local name = (type == "out" and session.to_host) or (type == "in" and session.from_host);
+	local id = idmap[name.."_"..type];
+	if id then
+		local notifier = stanza.stanza("retract", { id = id });
+		hosts[service].modules.pubsub.service:retract(xmlns_sessions, host, id, notifier);
+	end
+end
+
+local function preprocess_path(path)
+	if path:sub(1,1) ~= "/" then
+		path = "/"..path;
+	end
+	local level = 0;
+	for component in path:gmatch("([^/]+)/") do
+		if component == ".." then
+			level = level - 1;
+		elseif component ~= "." then
+			level = level + 1;
+		end
+		if level < 0 then
+			return nil;
+		end
+	end
+	return path;
+end
+
+function serve_file(path)
+	local full_path = http_base..path;
+	if stat(full_path, "mode") == "directory" then
+		if stat(full_path.."/index.html", "mode") == "file" then
+			return serve_file(path.."/index.html");
+		end
+		return response_403;
+	end
+	local f, err = open(full_path, "rb");
+	if not f then return response_404; end
+	local data = f:read("*a");
+	f:close();
+	if not data then
+		return response_403;
+	end
+	local ext = path:match("%.([^.]*)$");
+	local mime = mime_map[ext]; -- Content-Type should be nil when not known
+	return {
+		headers = { ["Content-Type"] = mime; };
+		body = data;
+	};
+end
+
+local function handle_file_request(method, body, request)
+	local path = preprocess_path(request.url.path);
+	if not path then return response_400; end
+	path = path:gsub("^/[^/]+", ""); -- Strip /admin/
+	return serve_file(path);
+end
+
+function module.load()
+	local host_session = prosody.hosts[host];
+	local http_conf = config.get("*", "core", "webadmin_http_ports");
+
+	hosts[service].modules.pubsub.service:create(xmlns_sessions, host);
+
+	for remotehost, session in pairs(host_session.s2sout) do
+		if session.type ~= "s2sout_unauthed" then
+			add_host(session, "out");
+		end
+	end
+	for session in pairs(incoming_s2s) do
+		if session.to_host == host then
+			add_host(session, "in");
+		end
+	end
+
+	httpserver.new_from_config(http_conf, handle_file_request, { base = "admin" });
+end
+
+module:hook("s2sout-established", function(event)
+	add_host(event.session, "out");
+end);
+
+module:hook("s2sin-established", function(event)
+	add_host(event.session, "in");
+end);
+
+module:hook("s2sout-destroyed", function(event)
+	del_host(event.session, "out");
+end);
+
+module:hook("s2sin-destroyed", function(event)
+	del_host(event.session, "in");
+end);
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/mod_admin_web/admin_web/www_files/index.html	Fri Dec 17 03:34:53 2010 +0100
@@ -0,0 +1,42 @@
+<?xml version="1.0" encoding="utf-8" ?>
+<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN"
+	"http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">
+<html xmlns="http://www.w3.org/1999/xhtml">
+<head>
+<title>S2S</title>
+<link rel="stylesheet" type="text/css" href="style.css" />
+<meta http-equiv="Content-Type" content="application/xhtml+xml; charset=utf-8" />
+<script type="text/javascript" src="js/jquery-1.4.4.min.js"></script>
+<script type="text/javascript" src="js/strophe.js"></script>
+<script type="text/javascript" src="js/main.js"></script>
+</head>
+<body>
+<div id='login'>
+  <form id='cred' name='cred'>
+    <label for='jid'>JID:</label><br/>
+    <input type='text' id='jid' />
+    <br />
+    <label for='pass'>Password:</label><br/>
+    <input type='password' id='pass' />
+    <br />
+    <input type='submit' id='connect' value='connect' />
+  </form>
+</div>
+
+<div id='main'>
+  <div class="container">
+    Incomming S2S connections:
+    <ul id="s2sin"></ul>
+  </div>
+  <div class="container">
+    Outgoing S2S connections:
+    <ul id="s2sout"></ul>
+  </div>
+</div>
+
+<div id='log_container'>
+  <a id='log_toggle' href='#'>Status Log :</a>
+  <div id='log'></div>
+</div>
+</body>
+</html>
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/mod_admin_web/admin_web/www_files/js/main.js	Fri Dec 17 03:34:53 2010 +0100
@@ -0,0 +1,136 @@
+var BOSH_SERVICE = 'http://localhost:5280/http-bind/';
+var show_log = false;
+
+Strophe.addNamespace('S2SPUBSUB', 'http://prosody.im/streams/s2s');
+Strophe.addNamespace('PUBSUB', 'http://jabber.org/protocol/pubsub');
+Strophe.addNamespace('CAPS', 'http://jabber.org/protocol/caps');
+
+var localJID = null;
+var connection   = null;
+
+var pubsubHost = null; /* TODO: Replace this inside Lua */
+
+function log(msg) {
+    var entry = $('<div></div>').append(document.createTextNode(msg));
+    $('#log').append(entry);
+}
+
+function rawInput(data) {
+    log('RECV: ' + data);
+}
+
+function rawOutput(data) {
+    log('SENT: ' + data);
+}
+
+function _cbNewS2S(e) {
+    var items, retract, id, jid;
+    items = e.getElementsByTagName('item');
+    for (i = 0; i < items.length; i++) {
+        id = items[i].attributes['id'].value;
+        jid = items[i].getElementsByTagName('session')[0].attributes['jid'].value;
+        if (items[i].getElementsByTagName('out')[0]) {
+            $('#s2sout').append('<li id="' + id + '">' + jid + '</li>');
+        } else {
+            $('#s2sin').append('<li id="' + id + '">' + jid + '</li>');
+        }
+    }
+    retract = e.getElementsByTagName('retract')[0];
+    if (retract) {
+        id = retract.attributes['id'].value;
+        $('#' + id).remove();
+    }
+    return true;
+}
+
+function onConnect(status) {
+    if (status == Strophe.Status.CONNECTING) {
+        log('Strophe is connecting.');
+    } else if (status == Strophe.Status.CONNFAIL) {
+        log('Strophe failed to connect.');
+        showConnect();
+    } else if (status == Strophe.Status.DISCONNECTING) {
+        log('Strophe is disconnecting.');
+    } else if (status == Strophe.Status.DISCONNECTED) {
+        log('Strophe is disconnected.');
+        showConnect();
+    } else if (status == Strophe.Status.AUTHFAIL) {
+        log('Authentication failed');
+        if (connection) {
+            connection.disconnect();
+        }
+    } else if (status == Strophe.Status.CONNECTED) {
+        log('Strophe is connected.');
+        showDisconnect();
+	pubsubHost = 'pubsub.' + connection.domain;
+        connection.send($iq({to: pubsubHost, type: 'set', id: connection.getUniqueId()}).c('pubsub', {xmlns: Strophe.NS.PUBSUB})
+                .c('subscribe', {node: Strophe.NS.S2SPUBSUB, jid: connection.jid}));
+        connection.addHandler(_cbNewS2S, Strophe.NS.PUBSUB + '#event', 'message');
+        connection.sendIQ($iq({to: pubsubHost, type: 'get', id: connection.getUniqueId()}).c('pubsub', {xmlns: Strophe.NS.PUBSUB})
+                .c('items', {node: Strophe.NS.S2SPUBSUB}), _cbNewS2S);
+    }
+}
+
+function showConnect() {
+    var jid = $('#jid');
+    var pass = $('#pass');
+    var button = $('#connect').get(0);
+
+    button.value = 'connect';
+    pass.show();
+    jid.show();
+    $('.container').hide();
+    $('#cred label').show();
+    $('#cred br').show();
+    $('ul').empty();
+}
+
+function showDisconnect() {
+    var jid = $('#jid');
+    var pass = $('#pass');
+    var button = $('#connect').get(0);
+
+    button.value = 'disconnect';
+    pass.hide();
+    jid.hide();
+    $('.container').show();
+    $('#cred label').hide();
+    $('#cred br').hide();
+}
+
+$(document).ready(function () {
+    connection = new Strophe.Connection(BOSH_SERVICE);
+    if (show_log) {
+        $('#log_container').show();
+        connection.rawInput = rawInput;
+        connection.rawOutput = rawOutput;
+    }
+
+    $("#log_toggle").click(function () {
+        $("#log").toggle();
+      });
+
+    $('#cred').bind('submit', function (event) {
+        var button = $('#connect').get(0);
+        var jid = $('#jid');
+        var pass = $('#pass');
+        localJID = jid.get(0).value;
+
+        if (button.value == 'connect') {
+            $('#log').empty();
+            connection.connect(localJID,
+               pass.get(0).value,
+               onConnect);
+        } else {
+            connection.disconnect();
+        }
+        event.preventDefault();
+    });
+
+});
+
+window.onunload = window.onbeforeunload = function() {
+    if (connection) {
+        connection.disconnect();
+    }
+}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/mod_admin_web/admin_web/www_files/style.css	Fri Dec 17 03:34:53 2010 +0100
@@ -0,0 +1,47 @@
+label {
+  margin-right: 0.5em
+}
+
+input, textarea, select {
+  margin: 0.25em;
+  margin-left: 1em
+}
+
+body {
+  padding: 1em;
+  background: #f29b00;
+  color: #000000
+}
+
+a {
+  color: #0000FF
+}
+
+#log_container {
+  clear: both;
+  display: none
+}
+
+.container {
+  margin: 2em;
+  padding: 0.5em;
+  border: solid;
+  float: left;
+  display: none
+}
+
+#login{
+  float: left;
+  margin-right: 2em;
+  padding: 1em;
+  background: #6197DF;
+  color: #000000
+}
+
+#cred input[type="submit"] {
+  margin-left: 0em
+}
+
+#main {
+  float: left
+}