2009-01-26 21:12:53 +01:00
|
|
|
// $Id: server.c,v 1.64 2008/05/13 09:51:07 lynx Exp $ // vim:syntax=lpc
|
2009-01-26 20:21:29 +01:00
|
|
|
//
|
|
|
|
// yes, psyced is also a web server, like every decent piece of code. ;)
|
|
|
|
//
|
2009-01-26 21:12:53 +01:00
|
|
|
#include <ht/http.h>
|
2009-01-26 20:21:29 +01:00
|
|
|
#include <net.h>
|
|
|
|
#include <server.h>
|
|
|
|
#include <text.h>
|
|
|
|
|
|
|
|
#include "header.i"
|
|
|
|
|
2010-02-28 23:38:35 +01:00
|
|
|
volatile string url, file, qs, version, method, body = "";
|
2009-01-26 20:21:29 +01:00
|
|
|
volatile mapping headers;
|
2010-02-28 23:38:35 +01:00
|
|
|
volatile int length;
|
2009-01-26 20:21:29 +01:00
|
|
|
|
|
|
|
// we're using #'closures to point to the functions we're giving the
|
|
|
|
// next_input_to(). as i don't want to restructure the whole file, i need
|
|
|
|
// to predefine some functions.
|
|
|
|
//
|
|
|
|
// quite stupid indeed, as they don't got any modifiers or whatever :)
|
|
|
|
parse_url(input);
|
|
|
|
parse_header(input);
|
2010-02-28 23:38:35 +01:00
|
|
|
parse_body(input);
|
2009-01-26 20:21:29 +01:00
|
|
|
devNull();
|
|
|
|
|
2009-01-26 21:12:53 +01:00
|
|
|
qScheme() { return "html"; }
|
2009-01-26 20:21:29 +01:00
|
|
|
|
|
|
|
logon() {
|
|
|
|
D2(D("»»» New SmallHTTP user\n");)
|
|
|
|
|
|
|
|
// bigger buffer (for psyc logo)
|
|
|
|
set_buffer_size(32768);
|
|
|
|
// unfortunately limited to a compilation limit
|
|
|
|
// so we would have to push large files in chunks
|
|
|
|
// using heart_beat() or something like that TODO
|
|
|
|
|
|
|
|
next_input_to(#'parse_url);
|
|
|
|
call_out(#'quit, TIME_LOGIN_IDLE);
|
|
|
|
}
|
|
|
|
|
|
|
|
disconnected(remainder) {
|
|
|
|
// TODO: shouldn't ignore remainder
|
|
|
|
D2(D("««« SmallHTTP got disconnected.\n");)
|
|
|
|
destruct(ME);
|
|
|
|
return 1; // expected death of socket
|
|
|
|
}
|
|
|
|
|
|
|
|
// gets called from async apps
|
|
|
|
done() { quit(); }
|
|
|
|
|
|
|
|
#if DEBUG > 1
|
|
|
|
quit() {
|
|
|
|
D2(D("««« SmallHTTP user done.\n");)
|
|
|
|
::quit();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void create() {
|
|
|
|
if (clonep(ME)) headers = ([ ]);
|
|
|
|
}
|
|
|
|
|
|
|
|
parse_wait(null) { // waiting to send my error message here
|
|
|
|
if (null == "") {
|
|
|
|
http_error("HTTP/1.0", 405, "Invalid Request (Welcome Proxyscanner)");
|
|
|
|
quit();
|
|
|
|
}
|
|
|
|
next_input_to(#'parse_wait);
|
|
|
|
}
|
|
|
|
|
|
|
|
parse_url(input) {
|
|
|
|
P3(("=== SmallHTTP got: %O\n", input))
|
2010-02-28 23:38:35 +01:00
|
|
|
unless (sscanf(input, "%s%t%s%tHTTP/%s", method, url, version)) quit();
|
|
|
|
switch (method) {
|
|
|
|
case "CONNECT":
|
2009-01-26 20:21:29 +01:00
|
|
|
next_input_to(#'parse_wait);
|
|
|
|
return;
|
2010-02-28 23:38:35 +01:00
|
|
|
case "GET":
|
|
|
|
case "POST":
|
|
|
|
break;
|
|
|
|
default:
|
2009-01-26 20:21:29 +01:00
|
|
|
quit();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
version = "HTTP/" + version;
|
|
|
|
|
|
|
|
P2(("=== SmallHTTP user requested url: %O\n", url))
|
|
|
|
next_input_to(#'parse_header);
|
|
|
|
}
|
|
|
|
|
|
|
|
parse_header(input) {
|
|
|
|
string key, val;
|
|
|
|
|
2010-02-22 08:33:32 +01:00
|
|
|
P4(("parse_header(%O)\n", input))
|
2009-01-26 20:21:29 +01:00
|
|
|
|
|
|
|
unless (input == "") {
|
|
|
|
if (sscanf(input, "%s:%1.0t%s", key, val)) {
|
|
|
|
headers[lower_case(key)] = val;
|
|
|
|
}
|
|
|
|
|
|
|
|
next_input_to(#'parse_header);
|
|
|
|
} else {
|
2010-02-28 23:38:35 +01:00
|
|
|
if (method == "POST" && (length = to_int(headers["content-length"])) &&
|
|
|
|
headers["content-type"] == "application/x-www-form-urlencoded") {
|
|
|
|
input_to(#'parse_body, INPUT_IGNORE_BANG | INPUT_CHARMODE | INPUT_NO_TELNET);
|
|
|
|
} else {
|
|
|
|
process();
|
|
|
|
next_input_to(#'devNull);
|
|
|
|
}
|
2009-01-26 20:21:29 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-02-28 23:38:35 +01:00
|
|
|
parse_body(input) {
|
|
|
|
//P4(("parse_body(%O)\n", input))
|
|
|
|
body += input;
|
|
|
|
if (strlen(body) == length)
|
|
|
|
process();
|
|
|
|
else
|
|
|
|
input_to(#'parse_body, INPUT_IGNORE_BANG | INPUT_CHARMODE | INPUT_NO_TELNET);
|
|
|
|
}
|
|
|
|
|
2009-01-26 20:21:29 +01:00
|
|
|
process() {
|
|
|
|
string t, ext;
|
|
|
|
mapping query = ([]);
|
|
|
|
object o;
|
|
|
|
int done = 1;
|
|
|
|
|
|
|
|
// take defaults from cookie, then override by query string
|
2009-01-26 21:12:53 +01:00
|
|
|
// lynXism cookie behaviour, normal one is below
|
2009-01-26 20:21:29 +01:00
|
|
|
t = headers["cookie"];
|
|
|
|
P4(("found cookie: %O\n", t))
|
|
|
|
if (t && sscanf(t, "psyced=\"%s\"", t)) {
|
|
|
|
P3(("got cookie: %O\n", t))
|
2010-02-23 16:15:44 +01:00
|
|
|
query = url_parse_query(query, t);
|
2009-01-26 20:21:29 +01:00
|
|
|
P4(("parsed cookie: %O\n", query))
|
|
|
|
}
|
2009-01-26 21:12:53 +01:00
|
|
|
#ifdef GENERIC_COOKIES // we might need them someday..?
|
|
|
|
// if within the same domain other cookies are being used, like
|
|
|
|
// by including google-analytics, then we might be receiving them
|
|
|
|
// here and have no friggin' idea what they are good for.
|
|
|
|
// thus: we *need* a way to ensure a cookie is our own.
|
|
|
|
// FIXME: this is not really compliant
|
|
|
|
else if (t) {
|
|
|
|
mapping cook = ([ ]);
|
|
|
|
string k, v;
|
|
|
|
while(t && sscanf(t, "%s=%s;%t%s", k, v, t) >= 2) {
|
|
|
|
cook[k] = v;
|
|
|
|
}
|
|
|
|
if (sscanf(t, "%s=%s", k, v))
|
|
|
|
cook[lower_case(k)] = v; // case insensitive
|
|
|
|
cook[0] = headers["cookie"]; // save cookie-string
|
|
|
|
headers["cookie"] = cook;
|
|
|
|
}
|
|
|
|
#endif
|
2009-01-26 20:21:29 +01:00
|
|
|
if (sscanf(url, "%s?%s", file, qs)) {
|
|
|
|
P3(("got query: %O\n", qs))
|
2010-02-23 16:15:44 +01:00
|
|
|
query = url_parse_query(query, qs);
|
2009-01-26 20:21:29 +01:00
|
|
|
} else {
|
|
|
|
file = url;
|
|
|
|
}
|
2010-02-28 23:38:35 +01:00
|
|
|
if (method == "POST" && headers["content-type"] == "application/x-www-form-urlencoded") {
|
|
|
|
query = url_parse_query(query, body);
|
|
|
|
}
|
2009-01-26 20:21:29 +01:00
|
|
|
P4(("parsed query: %O\n", query))
|
|
|
|
switch (file) {
|
|
|
|
case "/favicon.ico":
|
|
|
|
#if 0
|
|
|
|
htredirect(version, "http://www.psyced.org/favicon.ico",
|
|
|
|
"This one looks neat", 1);
|
|
|
|
quit();
|
|
|
|
return 1;
|
|
|
|
#else
|
|
|
|
file = "/static/favicon.ico";
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case "/":
|
|
|
|
case "":
|
|
|
|
// should we look for text/wml in the accept: and go directly
|
|
|
|
// to /net/wap/index ?
|
|
|
|
//
|
|
|
|
http_ok(version);
|
|
|
|
sTextPath(0, query["lang"], "html");
|
|
|
|
write( //T("_HTML_head", "<title>" CHATNAME "</title><body>\n"
|
|
|
|
// "<center><table width=404><tr><td>") +
|
|
|
|
T("_PAGES_index", "<i><a href=\"http://www.psyc.eu\"><img src="
|
|
|
|
"\"static/psyc.gif\" width=464 height=93 border=0></a><p>"
|
|
|
|
"<a href=\"http://www.psyced.org\">psyced</a></i> -"
|
|
|
|
" your multicast capable web application server.") );
|
|
|
|
// T("_HTML_tail", "</td></tr></table></center></body>"));
|
|
|
|
quit();
|
|
|
|
return 1;
|
|
|
|
case "/static": // really don't like to do this, but the IE stores directories
|
|
|
|
// (history) without trailing slash, even if the url originaly
|
|
|
|
// has one, at least IIRC.
|
|
|
|
htredirect(version, "/static/", "use the trailing slash", 1);
|
|
|
|
quit();
|
|
|
|
return 1;
|
|
|
|
case "/static/":
|
|
|
|
file = "/static/index.html";
|
|
|
|
break;
|
2010-02-23 13:10:06 +01:00
|
|
|
case "/oauth":
|
|
|
|
object oauth;
|
2010-02-23 13:32:23 +01:00
|
|
|
http_ok(version);
|
2010-02-23 23:42:54 +01:00
|
|
|
//PT((">>> looking up token %O in shm: %O\n", query["oauth_token"], shared_memory("oauth_request_tokens")))
|
|
|
|
if (oauth = shared_memory("oauth_request_tokens")[query["oauth_token"]]) {
|
2010-02-23 13:10:06 +01:00
|
|
|
//PT((">>> oauth: %O\n", oauth))
|
|
|
|
oauth->verified(query["oauth_verifier"]);
|
|
|
|
m_delete(shared_memory("oauth_request_tokens"), query["oauth_token"]);
|
2010-02-23 18:46:50 +01:00
|
|
|
write("OAuth succeeded, you can now return to your client.");
|
2010-02-23 13:32:23 +01:00
|
|
|
} else {
|
|
|
|
write("OAuth failed: token not found");
|
2010-02-23 13:10:06 +01:00
|
|
|
}
|
2010-02-23 13:32:23 +01:00
|
|
|
quit();
|
2010-02-23 13:10:06 +01:00
|
|
|
return 1;
|
2009-01-26 20:21:29 +01:00
|
|
|
}
|
2010-02-24 05:55:08 +01:00
|
|
|
string name;
|
2009-01-26 20:21:29 +01:00
|
|
|
switch (file[1]) {
|
|
|
|
case '~':
|
2010-02-24 05:55:08 +01:00
|
|
|
string channel, nick = file[2..];
|
|
|
|
if (sscanf(file, "/~%s/%s", nick, channel)) {
|
|
|
|
name = "~" + nick + "#" + channel;
|
|
|
|
} else if (o = summon_person(nick, NET_PATH "user")) {
|
|
|
|
o->htinfo(version, query, headers, qs, channel);
|
|
|
|
quit();
|
|
|
|
return 1;
|
2009-01-26 20:21:29 +01:00
|
|
|
}
|
2010-02-24 05:55:08 +01:00
|
|
|
//fall thru
|
2009-01-26 20:21:29 +01:00
|
|
|
case '@':
|
2010-02-24 05:55:08 +01:00
|
|
|
unless(name) name = file[2..];
|
|
|
|
o = find_place(name);
|
2009-01-26 20:21:29 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (abbrev("/static/", file)) {
|
|
|
|
if (file_size(file) > 0) {
|
|
|
|
if (sscanf(file, "%!s.%s", ext)) {
|
|
|
|
while (sscanf(ext, "%!s.%s", ext)) ;
|
|
|
|
}
|
|
|
|
http_ok(version, content_type(ext), 0);
|
|
|
|
binary_message(read_file(file));
|
|
|
|
quit();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
} else if (sscanf(file, "/%s/%s.page", ext, t) == 2) {
|
|
|
|
http_ok(version);
|
|
|
|
sTextPath(0, query["lang"] || ext, "html");
|
|
|
|
t = replace(t, "/", "_");
|
|
|
|
write(T("_HTML_head", "<title>" CHATNAME "</title><body>\n"
|
|
|
|
"<center><table width=404><tr><td>") +
|
|
|
|
T("_PAGES_"+t, "[no such page]\n") +
|
|
|
|
T("_HTML_tail", "</td></tr></table></center></body>"));
|
|
|
|
quit();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (index(file, ':') != -1) {
|
|
|
|
http_error(version, 501, "Not Implemented. Whatever you are trying "
|
|
|
|
"there, this server won't help you.");
|
|
|
|
quit();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-02-24 05:55:08 +01:00
|
|
|
unless (o) o = file -> load();
|
2009-01-26 20:21:29 +01:00
|
|
|
if (objectp(o) || o = find_object(file))
|
|
|
|
done = o->htget(version, query, headers, qs) != HTMORE;
|
|
|
|
|
|
|
|
if (done)
|
|
|
|
quit();
|
|
|
|
else
|
|
|
|
remove_call_out(#'quit);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// wozu binary_message nochmal durch eine funktion jagen? lieber so nennen:
|
|
|
|
emit(a) { return binary_message(a); }
|
|
|
|
|
|
|
|
devNull() {
|
|
|
|
next_input_to(#'devNull);
|
|
|
|
|
|
|
|
D2(D("=== SmallHTTP just ignored some input\n");)
|
|
|
|
}
|