mirror of https://github.com/calzoneman/sync.git
Start working on API refactor
This commit is contained in:
parent
9c83a4dd3e
commit
d266175d5b
293
api.js
293
api.js
|
@ -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;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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);
|
||||||
|
|
Loading…
Reference in New Issue