thelounge/src/server.js

221 lines
4.9 KiB
JavaScript
Raw Normal View History

var _ = require("lodash");
2016-04-26 12:51:11 +02:00
var package = require("../package.json");
2014-10-03 11:57:35 +02:00
var bcrypt = require("bcrypt-nodejs");
var Client = require("./client");
var ClientManager = require("./clientManager");
2014-09-27 00:12:53 +02:00
var express = require("express");
var fs = require("fs");
var io = require("socket.io");
var Helper = require("./helper");
2014-10-04 01:33:44 +02:00
var config = {};
var manager = new ClientManager();
2014-10-11 14:35:28 +02:00
module.exports = function(options) {
2014-10-04 01:33:44 +02:00
config = Helper.getConfig();
2014-10-11 14:35:28 +02:00
config = _.extend(config, options);
2014-09-27 00:12:53 +02:00
var app = express()
.use(index)
2014-10-04 01:33:44 +02:00
.use(express.static("client"));
2014-10-14 22:05:16 +02:00
app.enable("trust proxy");
2014-09-27 01:26:21 +02:00
var server = null;
var https = config.https || {};
var protocol = https.enable ? "https" : "http";
2014-10-11 14:35:28 +02:00
var port = config.port;
var host = config.host;
2016-03-19 17:48:36 +01:00
var transports = config.transports || ["polling", "websocket"];
2014-09-27 01:26:21 +02:00
if (!https.enable) {
2014-09-27 01:26:21 +02:00
server = require("http");
server = server.createServer(app).listen(port, host);
} else {
2016-03-09 13:04:05 +01:00
server = require("spdy");
2014-09-27 01:26:21 +02:00
server = server.createServer({
key: fs.readFileSync(https.key),
cert: fs.readFileSync(https.certificate)
2015-10-01 00:39:57 +02:00
}, app).listen(port, host);
2014-09-27 01:26:21 +02:00
}
2014-10-11 19:33:28 +02:00
if ((config.identd || {}).enable) {
require("./identd").start(config.identd.port);
}
var sockets = io(server, {
transports: transports
});
sockets.on("connect", function(socket) {
if (config.public) {
auth.call(socket);
} else {
init(socket);
}
});
2014-09-25 00:23:54 +02:00
manager.sockets = sockets;
2016-04-26 12:51:11 +02:00
log.info("The Lounge v" + package.version + " is now running on", protocol + "://" + config.host + ":" + config.port + "/");
log.info("Press ctrl-c to stop\n");
if (!require("semver").satisfies(process.version, package.engines.node)) {
log.warn("The oldest supported Node.js version is ", package.engines.node);
log.warn("We strongly encourage you to upgrade, see https://nodejs.org/en/download/package-manager/ for more details\n");
}
if (!config.public) {
2014-09-25 00:23:54 +02:00
manager.loadUsers();
if (config.autoload) {
manager.autoload();
}
}
};
function index(req, res, next) {
2015-10-01 00:39:57 +02:00
if (req.url.split("?")[0] !== "/") return next();
return fs.readFile("client/index.html", "utf-8", function(err, file) {
var data = _.merge(
2016-04-26 12:51:11 +02:00
package,
config
);
2016-02-14 18:09:51 +01:00
var template = _.template(file);
res.setHeader("Content-Type", "text/html");
res.writeHead(200);
2016-02-14 18:09:51 +01:00
res.end(template(data));
});
}
2014-09-15 23:13:03 +02:00
function init(socket, client, token) {
if (!client) {
socket.emit("auth");
socket.on("auth", auth);
} else {
socket.on(
"input",
function(data) {
2014-09-09 21:31:23 +02:00
client.input(data);
}
);
socket.on(
2014-09-10 21:23:56 +02:00
"more",
function(data) {
2014-09-10 21:23:56 +02:00
client.more(data);
}
);
socket.on(
"conn",
function(data) {
client.connect(data);
}
);
if (!config.public) {
socket.on(
"change-password",
function(data) {
var old = data.old_password;
var p1 = data.new_password;
var p2 = data.verify_password;
if (typeof old === "undefined" || old === "") {
socket.emit("change-password", {
error: "Please enter your current password"
});
return;
}
if (typeof p1 === "undefined" || p1 === "") {
socket.emit("change-password", {
error: "Please enter a new password"
});
return;
}
if (p1 !== p2) {
socket.emit("change-password", {
error: "Both new password fields must match"
});
return;
}
if (!bcrypt.compareSync(old || "", client.config.password)) {
socket.emit("change-password", {
error: "The current password field does not match your account password"
});
return;
}
var salt = bcrypt.genSaltSync(8);
var hash = bcrypt.hashSync(p1, salt);
if (client.setPassword(hash)) {
socket.emit("change-password", {
success: "Successfully updated your password"
});
return;
}
socket.emit("change-password", {
error: "Failed to update your password"
});
}
);
}
socket.on(
"open",
function(data) {
client.open(data);
}
2014-09-24 21:42:36 +02:00
);
socket.on(
"sort",
function(data) {
client.sort(data);
}
);
socket.on(
"names",
function(data) {
client.names(data);
}
);
socket.join(client.id);
socket.emit("init", {
active: client.activeChannel,
2014-09-15 23:13:03 +02:00
networks: client.networks,
token: token || ""
});
}
}
function auth(data) {
var socket = this;
if (config.public) {
var client = new Client(manager);
manager.clients.push(client);
socket.on("disconnect", function() {
manager.clients = _.without(manager.clients, client);
client.quit();
});
init(socket, client);
} else {
2014-09-11 22:37:16 +02:00
var success = false;
_.each(manager.clients, function(client) {
2014-09-15 23:13:03 +02:00
if (data.token) {
2015-10-01 00:39:57 +02:00
if (data.token === client.token) {
2014-09-15 23:13:03 +02:00
success = true;
}
2015-10-01 00:39:57 +02:00
} else if (client.config.user === data.user) {
2014-09-14 21:13:34 +02:00
if (bcrypt.compareSync(data.password || "", client.config.password)) {
2014-09-11 22:37:16 +02:00
success = true;
}
}
2014-09-15 23:13:03 +02:00
if (success) {
var token;
if (data.remember || data.token) {
token = client.token;
}
init(socket, client, token);
2014-09-16 19:42:49 +02:00
return false;
2014-09-15 23:13:03 +02:00
}
});
if (!success) {
2014-10-08 22:16:10 +02:00
socket.emit("auth");
}
}
}