Start working on API refactor

This commit is contained in:
Calvin Montgomery 2013-08-07 23:44:41 -04:00
parent 9c83a4dd3e
commit d266175d5b
3 changed files with 126 additions and 180 deletions

293
api.js
View File

@ -28,183 +28,144 @@ module.exports = function (Server) {
return raw; return raw;
} }
var API = function () { function getChannelData(channel) {
var data = {
name: channel.name,
loaded: true
};
data.pagetitle = channel.opts.pagetitle;
data.media = channel.playlist.current ?
channel.playlist.current.media.pack() :
{};
data.usercount = channel.users.length;
data.afkcount = channel.afkers.length;
data.users = [];
for(var i in channel.users)
if(channel.users[i].name !== "")
data.users.push(channel.users[i].name);
data.chat = [];
for(var i in channel.chatbuffer)
data.chat.push(channel.chatbuffer[i]);
return data;
} }
API.prototype = {
handle: function (path, req, res) {
var parts = path.split("/");
var last = parts[parts.length - 1];
var params = {};
if(last.indexOf("?") != -1) {
parts[parts.length - 1] = last.substring(0, last.indexOf("?"));
var plist = last.substring(last.indexOf("?") + 1).split("&");
for(var i = 0; i < plist.length; i++) {
var kv = plist[i].split("=");
if(kv.length != 2) {
res.send(400);
return;
}
params[unescape(kv[0])] = unescape(kv[1]);
}
}
for(var i = 0; i < parts.length; i++) {
parts[i] = unescape(parts[i]);
}
if(parts.length != 2) { var app = Server.app;
res.send(400);
return;
}
if(parts[0] == "json") { /* REGION channels */
res.callback = params.callback || false;
if(!(parts[1] in this.jsonHandlers)) {
res.end(JSON.stringify({
error: "Unknown endpoint: " + parts[1]
}, null, 4));
return;
}
this.jsonHandlers[parts[1]](params, req, res);
}
else if(parts[0] == "plain") {
if(!(parts[1] in this.plainHandlers)) {
res.send(404);
return;
}
this.plainHandlers[parts[1]](params, req, res);
}
else {
res.send(400);
}
},
sendJSON: function (res, obj) { /* data about a specific channel */
var response = JSON.stringify(obj, null, 4); app.get("/api/channels/:channel", function (req, res) {
if(res.callback) { var name = req.params.channel;
response = res.callback + "(" + response + ")"; if(!name.match(/^[\w-_]+$/)) {
} res.send(404);
var len = unescape(encodeURIComponent(response)).length; return;
}
res.setHeader("Content-Type", "application/json"); var data = {
res.setHeader("Content-Length", len); name: name,
res.end(response); loaded: false
}, };
sendPlain: function (res, str) { if(Server.channelLoaded(name))
if(res.callback) { data = getChannelData(name);
str = res.callback + "('" + str + "')";
}
var len = unescape(encodeURIComponent(str)).length;
res.setHeader("Content-Type", "text/plain"); res.type("application/json");
res.setHeader("Content-Length", len); res.jsonp(data);
res.end(response); });
},
handleChannelData: function (params, req, res) { /* data about all channels (filter= public or all) */
var clist = params.channel || ""; app.get("/api/allchannels/:filter", function (req, res) {
clist = clist.split(","); var filter = req.params.filter;
var data = []; if(filter !== "public" && filter !== "all") {
for(var j = 0; j < clist.length; j++) { res.send(400);
var cname = clist[j]; return;
if(!cname.match(/^[a-zA-Z0-9-_]+$/)) { }
continue;
}
var d = { var query = req.query;
name: cname,
loaded: Server.channelLoaded(cname)
};
if(d.loaded) { // Listing non-public channels requires authenticating as an admin
var chan = Server.getChannel(cname); if(filter !== "public") {
d.pagetitle = chan.opts.pagetitle; var name = query.name || "";
d.media = chan.playlist.current ? chan.playlist.current.media.pack() : {}; var session = query.session || "";
d.usercount = chan.users.length; var row = Auth.login(name, "", session);
d.users = [];
for(var i = 0; i < chan.users.length; i++) {
if(chan.users[i].name) {
d.users.push(chan.users[i].name);
}
}
d.chat = [];
for(var i = 0; i < chan.chatbuffer.length; i++) {
d.chat.push(chan.chatbuffer[i]);
}
}
data.push(d);
}
this.sendJSON(res, data);
},
handleChannelList: function (params, req, res) {
if(params.filter == "public") {
var all = Server.channels;
var clist = [];
for(var key in all) {
if(all[key].opts.show_public) {
clist.push(all[key].name);
}
}
this.handleChannelData({channel: clist.join(",")}, req, res);
}
var session = params.session || "";
var name = params.name || "";
var pw = params.pw || "";
var row = Auth.login(name, pw, session);
if(!row || row.global_rank < 255) { if(!row || row.global_rank < 255) {
res.send(403); res.send(403);
return; return;
} }
var clist = []; }
for(var key in Server.channels) {
clist.push(Server.channels[key].name); var channels = [];
for(var key in Server.channels) {
var channel = Server.channels[key];
if(channel.opts.show_public) {
channels.push(getChannelData(channel));
} else if(filter !== "public") {
channels.push(getChannelData(channel));
} }
this.handleChannelData({channel: clist.join(",")}, req, res); }
},
handleLogin: function (params, req, res) { res.type("application/jsonp");
var session = params.session || ""; res.jsonp(channels);
var name = params.name || ""; });
var pw = params.pw || "";
if(pw == "" && session == "") { /* ENDREGION channels */
if(!Auth.isRegistered(name)) {
this.sendJSON(res, {
success: true,
session: ""
});
return;
}
else {
this.sendJSON(res, {
success: false,
error: "That username is already taken"
});
return;
}
}
var row = Auth.login(name, pw, session); /* REGION authentication */
if(row) {
if(row.global_rank >= 255) /* login */
ActionLog.record(getIP(req), name, "login-success"); app.post("/api/login", function (req, res) {
this.sendJSON(res, { res.type("application/jsonp");
success: true, var name = req.body.name;
session: row.session_hash var pw = req.body.pw;
var session = req.body.session;
// for some reason CyTube previously allowed guest logins
// over the API...wat
if(!pw && !session) {
res.jsonp({
success: false,
error_code: "need_pw_or_session",
error: "You must provide a password"
});
return;
}
var row = Auth.login(name, pw, session);
if(!row) {
if(session && !pw) {
res.jsonp({
success: false,
error_code: "invalid_session",
error: "Session expired"
}); });
} return;
else { } else {
ActionLog.record(getIP(req), name, "login-failure"); ActionLog.record(getIP(req), name, "login-failure",
this.sendJSON(res, { "invalid_password");
error: "Invalid username/password", res.jsonp({
success: false success: false,
error_code: "invalid_password",
error: "Provided username/password pair is invalid"
}); });
return;
} }
}, }
// record the login if the user is an administrator
if(row.global_rank >= 255)
ActionLog.record(getIP(req), name, "login-success");
res.jsonp({
success: true,
name: name,
session: row.session_hash
});
});
var x = {
handlePasswordChange: function (params, req, res) { handlePasswordChange: function (params, req, res) {
var name = params.name || ""; var name = params.name || "";
var oldpw = params.oldpw || ""; var oldpw = params.oldpw || "";
@ -593,27 +554,5 @@ module.exports = function (Server) {
} }
}; };
var api = new API(); return null;
api.plainHandlers = {
"readlog" : api.handleReadLog.bind(api)
};
api.jsonHandlers = {
"channeldata" : api.handleChannelData.bind(api),
"listloaded" : api.handleChannelList.bind(api),
"login" : api.handleLogin.bind(api),
"register" : api.handleRegister.bind(api),
"changepass" : api.handlePasswordChange.bind(api),
"resetpass" : api.handlePasswordReset.bind(api),
"recoverpw" : api.handlePasswordRecover.bind(api),
"setprofile" : api.handleProfileChange.bind(api),
"getprofile" : api.handleProfileGet.bind(api),
"listuserchannels": api.handleListUserChannels.bind(api),
"setemail" : api.handleEmailChange.bind(api),
"admreports" : api.handleAdmReports.bind(api),
"readactionlog" : api.handleReadActionLog.bind(api)
};
return api;
} }

View File

@ -42,9 +42,11 @@ var defaults = {
} }
function save(cfg, file) { function save(cfg, file) {
if(!cfg.loaded)
return;
var x = {}; var x = {};
for(var k in cfg) { for(var k in cfg) {
if(k !== "nodemailer") if(k !== "nodemailer" && k !== "loaded")
x[k] = cfg[k]; x[k] = cfg[k];
} }
fs.writeFile(file, JSON.stringify(x, null, 4), function (err) { fs.writeFile(file, JSON.stringify(x, null, 4), function (err) {
@ -92,6 +94,8 @@ exports.load = function (Server, file, callback) {
); );
} }
cfg["loaded"] = true;
save(cfg, file); save(cfg, file);
Server.cfg = cfg; Server.cfg = cfg;
callback(); callback();

View File

@ -90,6 +90,7 @@ var Server = {
init: function () { init: function () {
this.httpaccess = new Logger.Logger("httpaccess.log"); this.httpaccess = new Logger.Logger("httpaccess.log");
this.app = express(); this.app = express();
this.app.use(express.bodyParser());
// channel path // channel path
this.app.get("/r/:channel(*)", function (req, res, next) { this.app.get("/r/:channel(*)", function (req, res, next) {
var c = req.params.channel; var c = req.params.channel;
@ -104,10 +105,12 @@ var Server = {
// api path // api path
this.api = require("./api")(this); this.api = require("./api")(this);
/*
this.app.get("/api/:apireq(*)", function (req, res, next) { this.app.get("/api/:apireq(*)", function (req, res, next) {
this.logHTTP(req); this.logHTTP(req);
this.api.handle(req.url.substring(5), req, res); this.api.handle(req.url.substring(5), req, res);
}.bind(this)); }.bind(this));
*/
this.app.get("/", function (req, res, next) { this.app.get("/", function (req, res, next) {
this.logHTTP(req); this.logHTTP(req);