2014-08-14 16:35:37 +00:00
|
|
|
var _ = require("lodash");
|
2016-04-26 10:51:11 +00:00
|
|
|
var package = require("../package.json");
|
2014-10-03 09:57:35 +00:00
|
|
|
var bcrypt = require("bcrypt-nodejs");
|
2014-08-14 16:35:37 +00:00
|
|
|
var Client = require("./client");
|
|
|
|
var ClientManager = require("./clientManager");
|
2014-09-26 22:12:53 +00:00
|
|
|
var express = require("express");
|
2014-08-14 16:35:37 +00:00
|
|
|
var fs = require("fs");
|
|
|
|
var io = require("socket.io");
|
2016-04-03 05:12:49 +00:00
|
|
|
var dns = require("dns");
|
2014-09-13 12:23:17 +00:00
|
|
|
var Helper = require("./helper");
|
2014-10-03 23:33:44 +00:00
|
|
|
var config = {};
|
2014-08-14 16:35:37 +00:00
|
|
|
|
|
|
|
var manager = new ClientManager();
|
|
|
|
|
2014-10-11 12:35:28 +00:00
|
|
|
module.exports = function(options) {
|
2014-10-03 23:33:44 +00:00
|
|
|
config = Helper.getConfig();
|
2014-10-11 12:35:28 +00:00
|
|
|
config = _.extend(config, options);
|
2014-08-14 16:35:37 +00:00
|
|
|
|
2014-09-26 22:12:53 +00:00
|
|
|
var app = express()
|
2016-05-01 17:27:10 +00:00
|
|
|
.use(allRequests)
|
2014-08-14 16:35:37 +00:00
|
|
|
.use(index)
|
2014-10-03 23:33:44 +00:00
|
|
|
.use(express.static("client"));
|
2014-11-01 20:06:01 +00:00
|
|
|
|
2014-09-26 23:26:21 +00:00
|
|
|
var server = null;
|
|
|
|
var https = config.https || {};
|
|
|
|
var protocol = https.enable ? "https" : "http";
|
2014-10-11 12:35:28 +00:00
|
|
|
var port = config.port;
|
|
|
|
var host = config.host;
|
2016-03-19 16:48:36 +00:00
|
|
|
var transports = config.transports || ["polling", "websocket"];
|
2014-09-26 23:26:21 +00:00
|
|
|
|
2016-03-09 08:50:20 +00:00
|
|
|
if (!https.enable) {
|
2014-09-26 23:26:21 +00:00
|
|
|
server = require("http");
|
|
|
|
server = server.createServer(app).listen(port, host);
|
|
|
|
} else {
|
2016-03-09 12:04:05 +00:00
|
|
|
server = require("spdy");
|
2014-09-26 23:26:21 +00:00
|
|
|
server = server.createServer({
|
2016-04-27 08:13:25 +00:00
|
|
|
key: fs.readFileSync(Helper.expandHome(https.key)),
|
|
|
|
cert: fs.readFileSync(Helper.expandHome(https.certificate))
|
2015-09-30 22:39:57 +00:00
|
|
|
}, app).listen(port, host);
|
2014-09-26 23:26:21 +00:00
|
|
|
}
|
|
|
|
|
2014-10-11 17:33:28 +00:00
|
|
|
if ((config.identd || {}).enable) {
|
|
|
|
require("./identd").start(config.identd.port);
|
2014-10-11 10:09:27 +00:00
|
|
|
}
|
|
|
|
|
2016-02-29 01:19:11 +00:00
|
|
|
var sockets = io(server, {
|
2014-11-01 20:06:01 +00:00
|
|
|
transports: transports
|
|
|
|
});
|
|
|
|
|
2014-08-14 16:35:37 +00:00
|
|
|
sockets.on("connect", function(socket) {
|
|
|
|
if (config.public) {
|
|
|
|
auth.call(socket);
|
|
|
|
} else {
|
|
|
|
init(socket);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2014-09-24 22:23:54 +00:00
|
|
|
manager.sockets = sockets;
|
|
|
|
|
2016-04-26 10:51:11 +00: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");
|
|
|
|
}
|
2014-08-14 16:35:37 +00:00
|
|
|
|
|
|
|
if (!config.public) {
|
2014-09-24 22:23:54 +00:00
|
|
|
manager.loadUsers();
|
|
|
|
if (config.autoload) {
|
|
|
|
manager.autoload();
|
|
|
|
}
|
2014-08-14 16:35:37 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-04-03 05:12:49 +00:00
|
|
|
function getClientIp(req) {
|
|
|
|
if (!config.reverseProxy) {
|
|
|
|
return req.connection.remoteAddress;
|
|
|
|
} else {
|
|
|
|
return req.headers["x-forwarded-for"] || req.connection.remoteAddress;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-01 17:27:10 +00:00
|
|
|
function allRequests(req, res, next) {
|
|
|
|
res.setHeader("X-Content-Type-Options", "nosniff");
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
2014-08-14 16:35:37 +00:00
|
|
|
function index(req, res, next) {
|
2016-05-01 09:41:17 +00:00
|
|
|
if (req.url.split("?")[0] !== "/") {
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
2014-08-14 16:35:37 +00:00
|
|
|
return fs.readFile("client/index.html", "utf-8", function(err, file) {
|
|
|
|
var data = _.merge(
|
2016-04-26 10:51:11 +00:00
|
|
|
package,
|
2014-08-14 16:35:37 +00:00
|
|
|
config
|
|
|
|
);
|
2016-02-14 17:09:51 +00:00
|
|
|
var template = _.template(file);
|
2016-05-01 17:27:10 +00:00
|
|
|
res.setHeader("Content-Security-Policy", "default-src *; style-src * 'unsafe-inline'; script-src 'self'; child-src 'none'; object-src 'none'; form-action 'none'; referrer no-referrer;");
|
2014-09-13 04:54:17 +00:00
|
|
|
res.setHeader("Content-Type", "text/html");
|
|
|
|
res.writeHead(200);
|
2016-02-14 17:09:51 +00:00
|
|
|
res.end(template(data));
|
2014-08-14 16:35:37 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2014-09-15 21:13:03 +00:00
|
|
|
function init(socket, client, token) {
|
2014-08-14 16:35:37 +00:00
|
|
|
if (!client) {
|
|
|
|
socket.emit("auth");
|
|
|
|
socket.on("auth", auth);
|
|
|
|
} else {
|
|
|
|
socket.on(
|
|
|
|
"input",
|
|
|
|
function(data) {
|
2014-09-09 19:31:23 +00:00
|
|
|
client.input(data);
|
2014-08-14 16:35:37 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
socket.on(
|
2014-09-10 19:23:56 +00:00
|
|
|
"more",
|
2014-08-14 16:35:37 +00:00
|
|
|
function(data) {
|
2014-09-10 19:23:56 +00:00
|
|
|
client.more(data);
|
2014-08-14 16:35:37 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
socket.on(
|
|
|
|
"conn",
|
|
|
|
function(data) {
|
2016-04-03 05:12:49 +00:00
|
|
|
// prevent people from overriding webirc settings
|
|
|
|
data.ip = null;
|
|
|
|
data.hostname = null;
|
2014-08-14 16:35:37 +00:00
|
|
|
client.connect(data);
|
|
|
|
}
|
|
|
|
);
|
2016-02-17 00:14:43 +00:00
|
|
|
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"
|
|
|
|
});
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2014-09-21 16:46:43 +00:00
|
|
|
socket.on(
|
|
|
|
"open",
|
|
|
|
function(data) {
|
|
|
|
client.open(data);
|
|
|
|
}
|
2014-09-24 19:42:36 +00:00
|
|
|
);
|
|
|
|
socket.on(
|
|
|
|
"sort",
|
|
|
|
function(data) {
|
|
|
|
client.sort(data);
|
|
|
|
}
|
|
|
|
);
|
2016-02-17 02:29:44 +00:00
|
|
|
socket.on(
|
|
|
|
"names",
|
|
|
|
function(data) {
|
|
|
|
client.names(data);
|
|
|
|
}
|
|
|
|
);
|
2014-08-14 16:35:37 +00:00
|
|
|
socket.join(client.id);
|
|
|
|
socket.emit("init", {
|
2014-09-21 16:46:43 +00:00
|
|
|
active: client.activeChannel,
|
2014-09-15 21:13:03 +00:00
|
|
|
networks: client.networks,
|
|
|
|
token: token || ""
|
2014-08-14 16:35:37 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-03 05:12:49 +00:00
|
|
|
function reverseDnsLookup(socket, client, token) {
|
|
|
|
client.ip = getClientIp(socket.request);
|
|
|
|
|
|
|
|
dns.reverse(client.ip, function(err, host) {
|
|
|
|
if (!err && host.length) {
|
|
|
|
client.hostname = host[0];
|
|
|
|
} else {
|
|
|
|
client.hostname = client.ip;
|
|
|
|
}
|
|
|
|
|
|
|
|
init(socket, client, token);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2014-08-14 16:35:37 +00:00
|
|
|
function auth(data) {
|
|
|
|
var socket = this;
|
|
|
|
if (config.public) {
|
2016-02-29 01:19:11 +00:00
|
|
|
var client = new Client(manager);
|
2014-08-14 16:35:37 +00:00
|
|
|
manager.clients.push(client);
|
|
|
|
socket.on("disconnect", function() {
|
|
|
|
manager.clients = _.without(manager.clients, client);
|
|
|
|
client.quit();
|
|
|
|
});
|
2016-04-03 05:12:49 +00:00
|
|
|
if (config.webirc) {
|
|
|
|
reverseDnsLookup(socket, client);
|
|
|
|
} else {
|
|
|
|
init(socket, client);
|
|
|
|
}
|
2014-08-14 16:35:37 +00:00
|
|
|
} else {
|
2014-09-11 20:37:16 +00:00
|
|
|
var success = false;
|
2014-08-14 16:35:37 +00:00
|
|
|
_.each(manager.clients, function(client) {
|
2014-09-15 21:13:03 +00:00
|
|
|
if (data.token) {
|
2015-09-30 22:39:57 +00:00
|
|
|
if (data.token === client.token) {
|
2014-09-15 21:13:03 +00:00
|
|
|
success = true;
|
|
|
|
}
|
2015-09-30 22:39:57 +00:00
|
|
|
} else if (client.config.user === data.user) {
|
2014-09-14 19:13:34 +00:00
|
|
|
if (bcrypt.compareSync(data.password || "", client.config.password)) {
|
2014-09-11 20:37:16 +00:00
|
|
|
success = true;
|
|
|
|
}
|
2014-08-14 16:35:37 +00:00
|
|
|
}
|
2014-09-15 21:13:03 +00:00
|
|
|
if (success) {
|
|
|
|
var token;
|
|
|
|
if (data.remember || data.token) {
|
|
|
|
token = client.token;
|
|
|
|
}
|
2016-04-03 05:12:49 +00:00
|
|
|
if (config.webirc !== null && !client.config["ip"]) {
|
|
|
|
reverseDnsLookup(socket, client, token);
|
|
|
|
} else {
|
|
|
|
init(socket, client, token);
|
|
|
|
}
|
2014-09-16 17:42:49 +00:00
|
|
|
return false;
|
2014-09-15 21:13:03 +00:00
|
|
|
}
|
2014-08-14 16:35:37 +00:00
|
|
|
});
|
|
|
|
if (!success) {
|
2014-10-08 20:16:10 +00:00
|
|
|
socket.emit("auth");
|
2014-08-14 16:35:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|