diff --git a/CHANGELOG.md b/CHANGELOG.md
new file mode 100644
index 0000000..5b4213d
--- /dev/null
+++ b/CHANGELOG.md
@@ -0,0 +1,5 @@
+**2.0.0**
+- Servers are now referenced by their name on the graph controls instead of their IP.
+- Servers now display their name on hover instead of their IP.
+- Graph controls are now saved and loaded automatically.
+- Moved server configuration into servers.json from config.json.
diff --git a/README.md b/README.md
index 08d09b7..a2757be 100644
--- a/README.md
+++ b/README.md
@@ -16,6 +16,9 @@ You can see an up-to-date copy of the master branch running on http://minetrack.
Database logging is disabled by default. You can enable it in ```config.json``` by setting ```logToDatabase``` to true.
This requires sqlite3 drivers to be installed.
+#### What's being changed?
+For the changelog, check out the [CHANGELOG](CHANGELOG.md) file.
+
#### How do I get added?
At the moment, the main "minetrack.me" site is only accepting networks that average roughly 1,000 players at peak for **PC** servers, and 100 players at peak for **PE** servers. This is due to limited capacity and it's functionality. If you fit this description, you're welcome to open a Pull Request adding the network to ```config.json```.
diff --git a/app.js b/app.js
index 6501a98..7b235c1 100644
--- a/app.js
+++ b/app.js
@@ -104,15 +104,16 @@ function handlePing(network, res, err) {
// Don't have too much data!
util.trimOldPings(graphData);
- if (!graphData[network.ip]) {
- graphData[network.ip] = [];
+ if (!graphData[network.name]) {
+ graphData[network.name] = [];
}
- graphData[network.ip].push([timeMs, res ? res.players.online : 0]);
+ graphData[network.name].push([timeMs, res ? res.players.online : 0]);
// Send the update.
server.io.sockets.emit('updateHistoryGraph', {
ip: network.ip,
+ name: network.name,
players: (res ? res.players.online : 0),
timestamp: timeMs
});
diff --git a/assets/js/site.js b/assets/js/site.js
index 11d9e8f..0175218 100644
--- a/assets/js/site.js
+++ b/assets/js/site.js
@@ -214,6 +214,13 @@ function setAllGraphVisibility(visible) {
historyPlot.setupGrid();
historyPlot.draw();
+
+ // Update our localStorage
+ if (visible) {
+ resetGraphControls();
+ } else {
+ saveGraphControls(Object.keys(displayedGraphData));
+ }
}
function toggleControlsDrawer() {
@@ -270,11 +277,30 @@ $(document).ready(function() {
});
socket.on('historyGraph', function(rawData) {
- displayedGraphData = rawData;
+ var shownServers = loadGraphControls();
+
+ if (shownServers) {
+ var keys = Object.keys(rawData);
+
+ hiddenGraphData = [];
+ displayedGraphData = [];
+
+ for (var i = 0; i < keys.length; i++) {
+ var name = keys[i];
+
+ if (shownServers.indexOf(name) !== -1) {
+ displayedGraphData[name] = rawData[name];
+ } else {
+ hiddenGraphData[name] = rawData[name];
+ }
+ }
+ } else {
+ displayedGraphData = rawData;
+ }
$('#big-graph').css('height', '400px');
- historyPlot = $.plot('#big-graph', convertGraphData(rawData), bigChartOptions);
+ historyPlot = $.plot('#big-graph', convertGraphData(displayedGraphData), bigChartOptions);
$('#big-graph').bind('plothover', handlePlotHover);
@@ -286,7 +312,13 @@ $(document).ready(function() {
keys.sort();
for (var i = 0; i < keys.length; i++) {
- html += '
' + keys[i] + '
';
+ var checkedString = '';
+
+ if (displayedGraphData[keys[i]]) {
+ checkedString = 'checked=checked';
+ }
+
+ html += '
' + keys[i] + '
';
if (sinceBreak >= 7) {
sinceBreak = 0;
@@ -303,19 +335,19 @@ $(document).ready(function() {
socket.on('updateHistoryGraph', function(rawData) {
// Prevent race conditions.
- if (!graphDuration) {
+ if (!graphDuration || !displayedGraphData || !hiddenGraphData) {
return;
}
// If it's not in our display group, use the hidden group instead.
- var targetGraphData = displayedGraphData[rawData.ip] ? displayedGraphData : hiddenGraphData;
+ var targetGraphData = displayedGraphData[rawData.name] ? displayedGraphData : hiddenGraphData;
trimOldPings(targetGraphData, graphDuration);
- targetGraphData[rawData.ip].push([rawData.timestamp, rawData.players]);
+ targetGraphData[rawData.name].push([rawData.timestamp, rawData.players]);
// Redraw if we need to.
- if (displayedGraphData[rawData.ip]) {
+ if (displayedGraphData[rawData.name]) {
historyPlot.setData(convertGraphData(displayedGraphData));
historyPlot.setupGrid();
@@ -460,9 +492,17 @@ $(document).ready(function() {
delete hiddenGraphData[serverIp];
}
+ // Redraw the graph
historyPlot.setData(convertGraphData(displayedGraphData));
historyPlot.setupGrid();
historyPlot.draw();
+
+ // Update our localStorage
+ if (Object.keys(hiddenGraphData).length === 0) {
+ resetGraphControls();
+ } else {
+ saveGraphControls(Object.keys(displayedGraphData));
+ }
});
});
diff --git a/assets/js/util.js b/assets/js/util.js
index b128990..449effc 100644
--- a/assets/js/util.js
+++ b/assets/js/util.js
@@ -2,6 +2,30 @@ var MISSING_FAVICON_BASE64 = "
var tooltip = $('#tooltip');
+function saveGraphControls(displayedServers) {
+ if (typeof(localStorage) !== undefined) {
+ var json = JSON.stringify(displayedServers);
+
+ localStorage.setItem('displayedServers', json);
+ }
+}
+
+function loadGraphControls() {
+ if (typeof(localStorage) !== undefined) {
+ var item = localStorage.getItem('displayedServers');
+
+ if (item) {
+ return JSON.parse(item);
+ }
+ }
+}
+
+function resetGraphControls() {
+ if (typeof(localStorage) !== undefined) {
+ localStorage.removeItem('displayedServers');
+ }
+}
+
function getTimestamp(ms, timeOnly) {
var date = new Date(0);
@@ -96,7 +120,7 @@ function convertGraphData(rawData) {
function stringToColor(base) {
var hash;
- for (var i = 0, hash = 0; i < base.length; i++) {
+ for (var i = base.length - 1, hash = 0; i >= 0; i--) {
hash = base.charCodeAt(i) + ((hash << 5) - hash);
}
diff --git a/lib/util.js b/lib/util.js
index 0e9914b..de946a1 100644
--- a/lib/util.js
+++ b/lib/util.js
@@ -1,17 +1,39 @@
+var logger = require('./logger');
+
var config = require('../config.json');
var servers = require('../servers.json');
-// Checks if we have a server in config.json with the IP.
-function serverWithIpExists(ip) {
+var serverNameLookup = {};
+
+// Finds a server in servers.json with a matching IP.
+// If it finds one, it caches the result for faster future lookups.
+function getServerNameByIp(ip) {
+ var lookupName = serverNameLookup[ip];
+
+ if (lookupName) {
+ return lookupName;
+ }
+
for (var i = 0; i < servers.length; i++) {
var entry = servers[i];
if (entry.ip === ip) {
- return true;
+ serverNameLookup[entry.ip] = entry.name;
+
+ return entry.name;
}
}
+}
- return false;
+// Returns a list of configured server IPs from servers.json
+function getServerIps() {
+ var ips = [];
+
+ for (var i = 0; i < servers.length; i++) {
+ ips.push(servers[i].ip);
+ }
+
+ return ips;
}
// This method is a monstrosity.
@@ -20,16 +42,7 @@ function serverWithIpExists(ip) {
function trimUselessPings(data) {
var keys = Object.keys(data);
- var keysToRemove = [];
-
for (var i = 0; i < keys.length; i++) {
- // Don't bother we servers we deleted from config.json
- if (!serverWithIpExists(keys[i])) {
- keysToRemove.push(keys[i]);
-
- continue;
- }
-
var listing = data[keys[i]];
var lastTimestamp = 0;
@@ -55,11 +68,6 @@ function trimUselessPings(data) {
data[keys[i]] = filteredListing;
}
-
- // Delete data for any networks we don't care about anymore.
- for (var i = 0; i < keysToRemove.length; i++) {
- delete data[keysToRemove[i]];
- }
}
exports.trimOldPings = function(data) {
@@ -98,16 +106,25 @@ exports.setIntervalNoDelay = function(func, delay) {
};
exports.convertPingsToGraph = function(sqlData) {
+ var serverIps = getServerIps();
var graphData = {};
+ var startTime = exports.getCurrentTimeMs();
+
for (var i = 0; i < sqlData.length; i++) {
var entry = sqlData[i];
- if (!graphData[entry.ip]) {
- graphData[entry.ip] = [];
+ if (serverIps.indexOf(entry.ip) === -1) {
+ continue;
}
- graphData[entry.ip].push([entry.timestamp, entry.playerCount]);
+ var name = getServerNameByIp(entry.ip);
+
+ if (!graphData[name]) {
+ graphData[name] = [];
+ }
+
+ graphData[name].push([entry.timestamp, entry.playerCount]);
}
// Break it into minutes.
@@ -116,5 +133,7 @@ exports.convertPingsToGraph = function(sqlData) {
// Drop old data.
exports.trimOldPings(graphData);
+ logger.info('Converted data structure in ' + (exports.getCurrentTimeMs() - startTime) + 'ms');
+
return graphData;
};
\ No newline at end of file
diff --git a/package.json b/package.json
index b277459..f33df8b 100644
--- a/package.json
+++ b/package.json
@@ -1,6 +1,6 @@
{
"name": "minetrack",
- "version": "1.0.0",
+ "version": "2.0.0",
"description": "A Minecraft server tracker that lets you focus on the basics.",
"main": "app.js",
"dependencies": {