thelounge/src/server.js

406 lines
9.9 KiB
JavaScript
Raw Normal View History

"use strict";
var _ = require("lodash");
var pkg = require("../package.json");
var Client = require("./client");
var ClientManager = require("./clientManager");
2014-09-26 22:12:53 +00:00
var express = require("express");
var expressHandlebars = require("express-handlebars");
var fs = require("fs");
var path = require("path");
var io = require("socket.io");
2016-04-03 05:12:49 +00:00
var dns = require("dns");
var Helper = require("./helper");
2016-07-30 01:20:38 +00:00
var ldap = require("ldapjs");
2016-11-18 17:25:23 +00:00
var colors = require("colors/safe");
2017-06-11 19:33:04 +00:00
const net = require("net");
const Identification = require("./identification");
2016-04-26 20:41:08 +00:00
var manager = null;
2016-07-30 01:20:38 +00:00
var authFunction = localAuth;
module.exports = function() {
log.info(`The Lounge ${colors.green(Helper.getVersion())} \
(node ${colors.green(process.versions.node)} on ${colors.green(process.platform)} ${process.arch})`);
log.info(`Configuration file: ${colors.green(Helper.CONFIG_PATH)}`);
if (!fs.existsSync("client/js/bundle.js")) {
log.error(`The client application was not built. Run ${colors.bold("NODE_ENV=production npm run build")} to resolve this.`);
process.exit();
}
2014-09-26 22:12:53 +00:00
var app = express()
.use(allRequests)
.use(index)
.use(express.static("client"))
.use("/storage/", express.static(Helper.getStoragePath(), {
redirect: false,
maxAge: 86400 * 1000,
}))
.engine("html", expressHandlebars({
extname: ".html",
helpers: {
tojson: (c) => JSON.stringify(c)
}
}))
.set("view engine", "html")
.set("views", path.join(__dirname, "..", "client"));
var config = Helper.config;
2014-09-26 23:26:21 +00:00
var server = null;
2016-07-30 01:20:38 +00:00
if (config.public && (config.ldap || {}).enable) {
log.warn("Server is public and set to use LDAP. Set to private mode if trying to use LDAP authentication.");
2016-07-30 01:20:38 +00:00
}
if (!config.https.enable) {
2014-09-26 23:26:21 +00:00
server = require("http");
server = server.createServer(app);
2014-09-26 23:26:21 +00:00
} else {
const keyPath = Helper.expandHome(config.https.key);
const certPath = Helper.expandHome(config.https.certificate);
2017-04-10 18:49:58 +00:00
const caPath = Helper.expandHome(config.https.ca);
2017-04-13 21:05:28 +00:00
if (!keyPath.length || !fs.existsSync(keyPath)) {
log.error("Path to SSL key is invalid. Stopping server...");
process.exit();
}
2017-04-13 21:05:28 +00:00
if (!certPath.length || !fs.existsSync(certPath)) {
log.error("Path to SSL certificate is invalid. Stopping server...");
process.exit();
}
2017-04-13 21:05:28 +00:00
2017-04-10 18:49:58 +00:00
if (caPath.length && !fs.existsSync(caPath)) {
log.error("Path to SSL ca bundle is invalid. Stopping server...");
process.exit();
}
2017-04-13 21:05:28 +00:00
server = require("spdy");
2014-09-26 23:26:21 +00:00
server = server.createServer({
key: fs.readFileSync(keyPath),
2017-04-10 18:49:58 +00:00
cert: fs.readFileSync(certPath),
ca: caPath ? fs.readFileSync(caPath) : undefined
}, app);
2014-09-26 23:26:21 +00:00
}
server.listen({
port: config.port,
host: config.host,
}, () => {
const protocol = config.https.enable ? "https" : "http";
var address = server.address();
log.info(`Available on ${colors.green(protocol + "://" + address.address + ":" + address.port + "/")} \
in ${config.public ? "public" : "private"} mode`);
});
if (!config.public && (config.ldap || {}).enable) {
2016-07-30 01:20:38 +00:00
authFunction = ldapAuth;
}
var sockets = io(server, {
2016-12-18 15:53:28 +00:00
serveClient: false,
transports: config.transports
});
sockets.on("connect", function(socket) {
if (config.public) {
auth.call(socket);
} else {
init(socket);
}
});
manager = new ClientManager();
new Identification((identHandler) => {
manager.init(identHandler, sockets);
});
};
2017-06-11 19:33:04 +00:00
function getClientIp(request) {
let ip = request.connection.remoteAddress;
2016-07-31 00:54:09 +00:00
2017-06-11 19:33:04 +00:00
if (Helper.config.reverseProxy) {
const forwarded = (request.headers["x-forwarded-for"] || "").split(/\s*,\s*/).filter(Boolean);
if (forwarded.length && net.isIP(forwarded[0])) {
ip = forwarded[0];
}
2016-04-03 05:12:49 +00:00
}
2016-07-31 00:54:09 +00:00
return ip.replace(/^::ffff:/, "");
2016-04-03 05:12:49 +00:00
}
function allRequests(req, res, next) {
res.setHeader("X-Content-Type-Options", "nosniff");
return next();
}
function index(req, res, next) {
if (req.url.split("?")[0] !== "/") {
return next();
}
var data = _.merge(
pkg,
Helper.config
);
data.gitCommit = Helper.getGitCommit();
data.themes = fs.readdirSync("client/themes/").filter(function(themeFile) {
return themeFile.endsWith(".css");
}).map(function(css) {
const filename = css.slice(0, -4);
return {
name: filename.charAt(0).toUpperCase() + filename.slice(1),
filename: filename
};
});
const policies = [
"default-src *",
"connect-src 'self' ws: wss:",
"style-src * 'unsafe-inline'",
"script-src 'self'",
"child-src 'self'",
"object-src 'none'",
"form-action 'none'",
];
// If prefetch is enabled, but storage is not, we have to allow mixed content
if (Helper.config.prefetchStorage || !Helper.config.prefetch) {
policies.push("img-src 'self'");
policies.unshift("block-all-mixed-content");
}
res.setHeader("Content-Security-Policy", policies.join("; "));
res.setHeader("Referrer-Policy", "no-referrer");
res.render("index", data);
}
2016-05-31 21:28:31 +00:00
function init(socket, client) {
if (!client) {
socket.emit("auth", {success: true});
socket.on("auth", auth);
} else {
2016-09-25 06:52:16 +00:00
socket.emit("authorized");
client.ip = getClientIp(socket.request);
socket.on("disconnect", function() {
client.clientDetach(socket.id);
});
client.clientAttach(socket.id);
socket.on(
"input",
function(data) {
2014-09-09 19:31:23 +00:00
client.input(data);
}
);
socket.on(
2014-09-10 19:23:56 +00:00
"more",
function(data) {
2014-09-10 19:23:56 +00:00
client.more(data);
}
);
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;
client.connect(data);
}
);
2016-07-30 01:20:38 +00:00
if (!Helper.config.public && !Helper.config.ldap.enable) {
socket.on(
"change-password",
function(data) {
var old = data.old_password;
var p1 = data.new_password;
var p2 = data.verify_password;
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;
}
2016-05-31 21:28:31 +00:00
Helper.password
.compare(old || "", client.config.password)
.then((matching) => {
if (!matching) {
socket.emit("change-password", {
error: "The current password field does not match your account password"
});
return;
}
const hash = Helper.password.hash(p1);
client.setPassword(hash, (success) => {
const obj = {};
if (success) {
obj.success = "Successfully updated your password, all your other sessions were logged out";
obj.token = client.config.token;
} else {
obj.error = "Failed to update your password";
}
socket.emit("change-password", obj);
});
}).catch((error) => {
log.error(`Error while checking users password. Error: ${error}`);
});
}
);
}
socket.on(
"open",
function(data) {
client.open(socket.id, data);
}
2014-09-24 19:42:36 +00: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.lastActiveChannel,
2014-09-15 21:13:03 +00:00
networks: client.networks,
token: client.config.token || null
});
}
}
2016-05-31 21:28:31 +00:00
function reverseDnsLookup(socket, client) {
2016-04-03 05:12:49 +00:00
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;
}
2016-05-31 21:28:31 +00:00
init(socket, client);
2016-04-03 05:12:49 +00:00
});
}
2016-07-30 01:20:38 +00:00
function localAuth(client, user, password, callback) {
2016-10-21 19:00:43 +00:00
if (!client || !password) {
return callback(false);
}
if (!client.config.password) {
log.error(`User ${colors.bold(user)} with no local password set tried to sign in. (Probably a LDAP user)`);
2016-10-21 19:00:43 +00:00
return callback(false);
}
Helper.password
.compare(password, client.config.password)
.then((matching) => {
if (matching && Helper.password.requiresUpdate(client.config.password)) {
const hash = Helper.password.hash(password);
2016-10-21 19:00:43 +00:00
client.setPassword(hash, (success) => {
if (success) {
log.info(`User ${colors.bold(client.name)} logged in and their hashed password has been updated to match new security requirements`);
}
});
2016-10-21 19:00:43 +00:00
}
callback(matching);
}).catch((error) => {
log.error(`Error while checking users password. Error: ${error}`);
2016-10-21 19:00:43 +00:00
});
2016-07-30 01:20:38 +00:00
}
function ldapAuth(client, user, password, callback) {
var userDN = user.replace(/([,\\/#+<>;"= ])/g, "\\$1");
2016-07-30 01:20:38 +00:00
var bindDN = Helper.config.ldap.primaryKey + "=" + userDN + "," + Helper.config.ldap.baseDN;
var ldapclient = ldap.createClient({
url: Helper.config.ldap.url
});
ldapclient.on("error", function(err) {
log.error("Unable to connect to LDAP server", err);
callback(!err);
});
2016-07-30 01:20:38 +00:00
ldapclient.bind(bindDN, password, function(err) {
if (!err && !client) {
if (!manager.addUser(user, null)) {
log.error("Unable to create new user", user);
2016-07-30 01:20:38 +00:00
}
}
ldapclient.unbind();
2016-07-30 01:20:38 +00:00
callback(!err);
});
}
function auth(data) {
var socket = this;
2016-07-30 01:20:38 +00:00
var client;
if (Helper.config.public) {
2016-07-30 01:20:38 +00:00
client = new Client(manager);
manager.clients.push(client);
socket.on("disconnect", function() {
manager.clients = _.without(manager.clients, client);
client.quit();
});
if (Helper.config.webirc) {
2016-04-03 05:12:49 +00:00
reverseDnsLookup(socket, client);
} else {
init(socket, client);
}
} else {
2016-07-30 01:20:38 +00:00
client = manager.findClient(data.user, data.token);
var signedIn = data.token && client && data.token === client.config.token;
2016-07-30 01:20:38 +00:00
var token;
if (client && (data.remember || data.token)) {
token = client.config.token;
2016-07-30 01:20:38 +00:00
}
var authCallback = function(success) {
2014-09-15 21:13:03 +00:00
if (success) {
2016-07-30 01:20:38 +00:00
if (!client) {
// LDAP just created a user
manager.loadUser(data.user);
client = manager.findClient(data.user);
}
2016-10-09 08:54:44 +00:00
if (Helper.config.webirc !== null && !client.config.ip) {
2016-05-31 21:28:31 +00:00
reverseDnsLookup(socket, client);
2016-04-03 05:12:49 +00:00
} else {
2016-07-30 01:20:38 +00:00
init(socket, client, token);
2016-04-03 05:12:49 +00:00
}
2016-07-30 01:20:38 +00:00
} else {
socket.emit("auth", {success: success});
2014-09-15 21:13:03 +00:00
}
2016-07-30 01:20:38 +00:00
};
if (signedIn) {
authCallback(true);
} else {
authFunction(client, data.user, data.password, authCallback);
}
}
}