norstalgia/ipviking/scripts/ipviking.js

2095 lines
62 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters!

This file contains ambiguous Unicode characters that may be confused with others in your current locale. If your use case is intentional and legitimate, you can safely ignore this warning. Use the Escape button to highlight these characters.

"use strict";
// 限制可被引用的域名
var approvedDomains = ['norse-corp.com'];
if (top.location != self.location && approvedDomains.indexOf(top.location.hostname) === -1) {
top.location = self.location.href
}
// 每6个小时刷新页面
var refreshSeconds = 60 * 60 * 6; // 6 hours
setTimeout("location.reload()", refreshSeconds * 1000);
// 呈现消息
function showMessage(message) {
document.getElementById('message-text').innerHTML = message;
document.getElementById('message-panel').style.display = "block";
}
// 隐藏消息
function hideMessage() {
document.getElementById('message-panel').style.display = 'none';
}
(function(window) {
var VSN = "1.1";
// 限制使用Chrome浏览器
/*
var isChrome = navigator.userAgent.toLowerCase().indexOf('chrome') > -1;
if(!isChrome) {
showMessage("The IPViking digital attack map only supports the Google Chrome browser.");
return;
}
*/
var settings = {
// 解析成数字的数据属性
numberProps: ["dport", "latitude", "longitude", "latitude2", "longitude2"],
// 作为统计信息更新的闪光的颜色
triggerColor: "red",
// 范围文本最低透明度
minTextOpacity: 0.35,
// 这个数字开始后丢弃攻击
maxAttacks: 100,
radius: 5,
countryColor: d3.scale.log()
.domain([1, 1200])
.range([d3.rgb(30, 30, 30), d3.rgb(30, 65, 140)]),
tableBarWidth: d3.scale.log()
.domain([1, 500])
.range([1, 130]),
// 布局设置
linkAnchor: false,
linkSiblings: false,
// 表格行数
topTableRows: 10,
portTableRows: 8,
consoleTableRows: 8,
pruneInterval: 3600,
dataPruneInterval: 60,
// Websocket 设置
//wsHost: "ws://64.19.78.244:443/",
wsHost: "ws://127.0.0.1:9999",
psk: "18c989796c61724d4661b019f2779848dd69ae62",
wsTimeout: 30000
};
/*
* HTML 接口
*/
d3.selectAll(".vsn").text(VSN);
var timestampedData = [];
function prune() {
// 使用 lodash 库, 用 _.select 实现折半搜索来找到时间对数范围的起始值
var now = new Date().getTime() / 1000;
for (var i in timestampedData) {
if (timestampedData[i].pruneTS > now) break;
}
var expiredData = [];
if (i > 0) {
expiredData = timestampedData.splice(0, i);
}
for (var n = 0; n < expiredData.length; n++) {
// todo:
// statsManager.remove(timestampData[n]);
linkModels.forEach(function(model) {
model.remove(expiredData[n]);
});
}
}
var displayLabel = {
// 显示常规信息的标签
elt: d3.select("#display-label"),
set: function(text) {
this.elt.text(text);
},
clear: function() {
this.elt.text("");
}
};
if (!window.chrome) {
displayLabel.set("Too slow? Try Chrome.");
}
d3.selectAll(".info-btn").on("click", function() {
d3.event.preventDefault();
var drawerContent = d3.select("#drawer");
if (drawerContent.style("display") === "none") {
drawerContent
.transition()
.style("display", "block");
d3.selectAll(".info-btn").classed("blue-bg", true);
d3.selectAll(".info-btn").classed("gray-bg", false);
d3.selectAll(".info-text").classed('icon-info', false);
d3.selectAll(".info-text").classed('icon-close', true);
} else {
drawerContent
.transition()
.style("display", "none");
d3.selectAll(".info-btn").classed("gray-bg", true);
d3.selectAll(".info-btn").classed("blue-bg", false);
d3.selectAll(".info-text").classed('icon-info', true);
d3.selectAll(".info-text").classed('icon-close', false);
}
});
var loadingToggle = (function() {
// 切换为加载HTML状态
var loading = true;
return function() {
if (loading) {
d3.select("#content")
.transition()
.duration(1000)
.style("opacity", 1);
d3.select("#loading")
.transition()
.duration(1000)
.style("opacity", 0);
} else {
d3.select("#content").style("opacity", 0);
d3.select("#loading")
.transition()
.duration(1000)
.style("opacity", 1);
}
}
})();
loadingToggle();
/*
* 定义变量
*/
// 监听 .toggles
(function() {
var toggles = d3.selectAll(".toggle");
var data = toggles[0]
.map(function(elt) {
return d3.select(elt.getAttribute("data-target"));
});
toggles
.data(data)
.on("click", function(d) {
d3.event.preventDefault();
if (d.style("display") === "none") {
d.style("display", "block");
} else {
d.style("display", "none");
}
});
})();
// 设置d3的地图
var width = window.innerWidth,
height = window.innerHeight;
// 从经纬转换为像素坐标
var projection = d3.geo.mercator()
.scale(width / 8.5)
.translate([width / 2, height / 1.7]);
// 使用projection将geojson绘制为svg的path
var path = d3.geo.path().projection(projection);
// SVG -- 我们的空白画布
var svg = d3.select("#content").append("svg")
.attr("class", "overlay")
.attr("width", width)
.attr("height", height);
svg.append("defs")
.append("filter")
.attr("id", "blur")
.append("feGaussianBlur")
.attr("stdDeviation", 2);
// 添加Attacks到 .attacks svg group
var node = svg.selectAll(".node"),
link = svg.selectAll(".link");
// 目标城市集群
var _clusters = d3.map();
var colorizer = d3.scale.category20();
// 端口到属性的映射表
var ports;
/**************
* 内部API
*/
function spanWrap(content, classes) {
// Returns the content wrapped in the span
return '<span class="' + (classes ? classes.join(" ") : "") + '">' +
content + '</span>'
}
function dist(x1, y1, x2, y2) {
// Returns the distance between two points
return Math.sqrt(Math.pow(x2 - x1, 2) + Math.pow(y2 - y1, 2));
}
function rgbaString(c, a) {
// Helper to get the color as an rgba string
return "rgba(" + c.r + "," + c.g + "," + c.b + "," + a + ")";
}
function parsePorts(rawPorts) {
// Given the csv list of ports, process it
var ports = [];
for (var i = 0; i < rawPorts.length; ++i) {
var port = parseInt(rawPorts[i].port);
if (port in ports) {
ports[port] = ports[port] + ", " + rawPorts[i].service;
} else {
ports[port] = rawPorts[i].service;
}
}
// Fix certain port strings
ports[80] = "http";
return ports;
}
var getID = (function() {
// Generate unique enough IDs
var i = 0;
return function() {
return i++;
}
})();
function flagPath(iso) {
// Return the path to the flag for the given countrycode
if (iso === "O1") {
return "images/militarywhite.svg";
} else {
return "images/flags/" + iso + ".png";
}
}
function flagTag(iso) {
return '<span class="flag f16 ' + iso.toLowerCase() + '"></span>';
}
function isNumber(n) {
// Returns true if n is a number
return !isNaN(parseFloat(n)) && isFinite(n);
}
var particler = function() {
// 返回 生成着色粒子的函数
var particle = new Image(),
tempFileCanvas = d3.select("#content")
.append("canvas")
.attr("class", "buffer")
.node();
particle.src = "";
tempFileCanvas.width = 64;
tempFileCanvas.height = 64;
return function(r, g, b, a) {
var imgCtx = tempFileCanvas.getContext("2d"),
imgData, i;
imgCtx.drawImage(particle, 0, 0);
//if(particle.width > self.innerWidth){particle.width=self.innerWidth;} if(particle.width < 1){particle.width=1;}
//if(particle.height > self.innerHeight){particle.height=self.innerHeight;} if(particle.height < 1){particle.height=1;}
imgData = imgCtx.getImageData(2, 2, 64, 64);
//imgData = imgCtx.getImageData(0, 0, particle.width, particle.height);
i = imgData.data.length;
while ((i -= 4) > -1) {
imgData.data[i + 3] = imgData.data[i] * a;
if (imgData.data[i + 3]) {
imgData.data[i] = r;
imgData.data[i + 1] = g;
imgData.data[i + 2] = b;
}
}
imgCtx.putImageData(imgData, 0, 0);
return tempFileCanvas;
}
}();
var nodeModel = {
/*
* 提供了一个API用于管理各种力量的布局节点
*
* Node Types:
* - Attack Nodes {type: attack}
* - Target Nodes {type: target}
* - Anchor Nodes, ie City {type: anchor}
*/
// Configuration
linkSiblings: true,
linkAnchor: false,
target: true,
interval: 50,
pathLength: 15,
targetMaxAge: 200,
scaleTargetVel: d3.scale.log().domain([1, 40]).range([40, 100]),
// Constants
ATTACKS: "attacks",
TARGETS: "targets",
ANCHORS: "anchors",
lastPrune: new Date().getTime() / 1000,
// The list of nodes, shared with the force layout
nodes: undefined,
// The list of links, shared with the force layout
links: undefined,
// Force layout to make the elements move correctly
force: d3.layout.force()
.size([width, height])
.friction(0.25)
.gravity(0)
.charge(-10)
.chargeDistance(50)
.linkDistance(15)
.linkStrength(function(d) {
return d.linkStrength || 0.5;
}),
prune: function() {
var now = new Date().getTime() / 1000;
if (now - this.lastPrune > 10) {
this.lastPrune = now;
for (var i in this.nodes) {
if (this.nodes[i].pruneTS > now) {
if (i > 0) {
this.nodes.splice(0, i);
}
break;
}
}
for (var i in this.links) {
if (this.links[i].pruneTS > now) {
if (i > 0) {
this.links.splice(0, i);
}
break;
}
}
}
},
get: function(type) {
return this.nodes.filter(function(n) {
return n.type === type;
});
},
_mapKey: function(d) {
return d.city + d.latitude + d.longitude;
},
_remove: function(n, j) {
for (var i = 0; i < this.links.length; i++) {
if (this.links[i].source.id === n.id ||
this.links[i].target.id === n.id) {
this.links.splice(i--, 1);
}
}
if (typeof j !== 'undefined') this.nodes.splice(j, 1);
// Remove a node, and its associated links
for (var i = 0; i < this.nodes.length; i++) {
if (n.id === this.nodes[i].id) {
this.nodes.splice(i--, 1);
}
}
},
_shift: function(type) {
for (var i = 0; i < this.nodes.length; i++) {
if (this.nodes[i].type === type) {
this._remove(this.nodes[i], i);
break;
}
}
},
_getsertAnchorFor: function(attack) {
// Get the anchor for the given node, inserting it if its not present
var key = this._mapKey(attack),
anchor = this.nodes.filter(function(n) {
return n.key === key;
})[0];
if (anchor) {
return anchor;
} else {
var newAnchor = {
id: getID(),
key: key,
type: this.ANCHORS,
x: attack.cx,
y: attack.cy,
cx: attack.cx,
cy: attack.cy,
country: attack.country,
city: attack.city,
fixed: true,
pruneTS: (new Date()).getTime() / 1000 + settings.dataPruneInterval
}
this.nodes.push(newAnchor);
return newAnchor;
}
},
pushAttack: function(attack) {
/*
* Push a new Attack
* Nodes are inserted at the source location and linked to an anchor
* centered at this source, and all of the adjacent nodes. Links
* are stronger to nodes of the same type.
*/
// Clear out old nodes
while (this.get(this.ATTACKS).length > 50) {
this._shift(this.ATTACKS);
}
if (this.linkSiblings) {
var key = this._mapKey(attack),
that = this;
this.nodes.forEach(function(n) {
if (that._mapKey(n) === key) { //&& n.dport === d.dport ) {
that.links.push({
source: n,
target: attack,
pruneTS: (new Date()).getTime() / 1000 + settings.dataPruneInterval,
linkStrength: n.dport === attack.dport ? 0.5 : 0.25
});
}
});
}
// Anchor
var anchor = this._getsertAnchorFor(attack);
if (this.linkAnchor) {
this.links.push({
source: anchor,
target: attack,
linkStrength: 1.0
});
}
// Target
if (this.target) {
var initialVelocity = -0.0001;
var target = {
type: this.TARGETS,
age: 0,
path: [],
h: dist(attack.x, attack.y, attack.targetX, attack.targetY),
id: getID(),
x: attack.x,
y: attack.y,
cx: attack.targetX,
cy: attack.targetY,
startX: attack.x,
startY: attack.y,
city: attack.city2,
country: attack.country2,
theta: Math.atan((attack.targetY - attack.y) /
(attack.targetX - attack.x)),
dport: attack.dport,
pruneTS: (new Date()).getTime() / 1000 + settings.dataPruneInterval
};
// this.links.push({source: this._getsertAnchorFor(target),
// target: target, linkStrength: 1.0});
this.nodes.push(target);
}
// Decorate and add the attack node
attack.type = this.ATTACKS;
attack.age = 0;
this.nodes.push(attack);
// TODO - is this necessary?!
this.force.start();
},
step: function() {
// Step the simulation
this.nodes.forEach(function(n) {
n.age++;
});
this.get(this.TARGETS)
.filter(function(t) {
return t.age > this.targetMaxAge ||
"arrivalAge" in t && t["arrivalAge"] + 40 < t.age;
}, this)
.forEach(function(t) {
this._remove(t);
}, this);
},
start: function() {
// Start the layout
var that = this;
// Initialize the array references
this.nodes = [];
this.links = [];
this.force
.nodes(this.nodes)
.links(this.links)
.on("tick", (function() {
//var targetTrack;
return function(e) {
// Tick the force layout
that.step();
that.get(that.ATTACKS).forEach(function(d) {
var scale = 0.1;
d.x += scale * (d.cx - d.x) * e.alpha;
d.y += scale * (d.cy - d.y) * e.alpha;
});
that.get(that.TARGETS).forEach(function(d) {
//DEBUGGING
// if (!targetTrack) targetTrack = d.id;
// Update the target's path
d.path.unshift({
x: d.x,
y: d.y
});
if (d.path.length > that.pathLength)
d.path.pop();
if (d.arrivalAge) {
d.fixed = true;
} else {
var travelled = dist(d.x, d.y, d.startX, d.startY),
//v = (Math.sqrt(travelled * 50) + 180) * e.alpha,
v = that.scaleTargetVel(d.age) * e.alpha,
toTarget = dist(d.cx, d.cy, d.x, d.y);
if (v <= toTarget) {
var theta = Math.atan2(d.cy - d.y, d.cx - d.x);
//r = v / d.h;
d.x += v * Math.cos(theta);
d.y += v * Math.sin(theta);
} else {
//debugger;
// Arrived at target
d.x = d.cx;
d.y = d.cy;
d.arrivalAge = d.age;
}
}
});
that.force.resume();
}
})())
.start();
// Prevent the alpha from 'cooling' to 0
d3.timer(this.force.resume);
}
};
// 缓存和查询城市<=>城市链接的抽象模型
var LinksModel = {
// {ORIGINCOUNTRY: {ORIGINCITY: {TRGTCOUNTRY: {TRGTCOUNTRY: {DPORT: COUNT}}}}}
// Created in via .extend: _links: {},
// {COUNTRY: {CITY: {latitude: LAT, longitude: LON}}}
_locs: {},
insertLink: function(origin, target, port) {
if (!(origin.country in this._links)) {
this._links[origin.country] = {};
}
if (!(origin.city in this._links[origin.country])) {
this._links[origin.country][origin.city] = {};
}
var originLinks = this._links[origin.country][origin.city];
if (!(target.country in originLinks)) {
originLinks[target.country] = {};
}
if (!(target.city in originLinks[target.country])) {
originLinks[target.country][target.city] = {};
}
var targetLinks = originLinks[target.country][target.city]
if (!(port in targetLinks)) {
targetLinks[port] = 1;
} else {
targetLinks[port] = targetLinks[port] + 1;
}
},
removeLink: function(origin, target, port) {
var link1 = this._links[origin.country];
if (!link1) return;
var link2 = this._links[origin.country][origin.city];
if (!link2) return;
var target1 = link2[target.country];
if (!target1) return;
var target2 = link2[target.country][target.city];
if (!target2) return;
if (target2[port] > 1) {
target2[port]--;
} else if (target2[port] !== undefined) {
delete target2[port];
}
},
insertLoc: function(loc) {
if (!(loc.country in this._locs)) {
this._locs[loc.country] = {}
}
if (!(loc.city in this._locs)) {
this._locs[loc.country][loc.city] = {
latitude: loc.latitude,
longitude: loc.longitude
};
}
},
removeLoc: function(loc, origin, target, port) {
var link1 = this._links[origin.country];
if (!link1) return;
var link2 = this._links[origin.country][origin.city];
if (!link2) return;
var target1 = link2[target.country];
if (!target1) return;
var target2 = link2[target.country][target.city]
if (!target2) return;
if (!target2[port]) {
delete this._links[origin.country][origin.city];
delete this._locs[loc.country][loc.city]
}
},
_distanceBetween: function(pt1, pt2) {
return Math.sqrt(Math.pow(pt1[0] - pt2[0], 2) +
Math.pow(pt1[1] - pt2[1], 2));
},
getCity: function(country, city) {
if (country in this._locs && city in this._locs[country]) {
var loc = this._locs[country][city],
pt = projection([loc.longitude, loc.latitude]),
info = {
country: country,
city: city,
latitude: loc.latitude,
longitude: loc.longitude,
pt: pt
};
if (country in this._links && city in this._links[country]) {
info.counts = this._links[country][city];
}
return info;
}
},
getCities: function() {
// Returns list of all cities
var cities = [];
for (var country in this._locs) {
for (var city in this._locs[country]) {
cities.push(this.getCity(country, city));
}
}
return cities;
},
getCountry: function(country) {
var cities = [];
for (var city in this._links[country] || {}) {
cities.push(this.getCity(country, city));
}
return cities;
},
pixelsFromNearest: function(pt) {
// Returns the pixels from the nearest source
var closest;
for (var country in this._links) {
for (var city in this._links[country]) {
var info = this.getCity(country, city);
if (info) {
var distance = this._distanceBetween(pt, info.pt);
if (!closest || distance < closest.distance) {
info.distance = distance;
closest = info;
}
}
}
}
return closest
},
total: function(counts) {
// Total up a counts array, recursive
if (isNumber(counts)) {
return counts;
} else {
var sum = 0;
for (var key in counts) {
sum += this.total(counts[key]);
}
return sum;
}
},
cityToLinks: function(city, strokeOrigin, strokeTarget) {
if (!city) return [];
var links = [];
for (var country in city.counts) {
if (!city.counts[country]) continue;
for (var cityKey in city.counts[country]) {
var info = this.getCity(country, cityKey),
counts = city.counts[country][cityKey];
if (!info || !counts || !Object.keys(counts).length) continue;
for (var dport in counts) {
var r = circleScale(counts[dport]),
color = colorizer(dport),
source = {
x: city.pt[0],
y: city.pt[1],
r: r
},
target = {
x: info.pt[0],
y: info.pt[1],
r: r
};
if (strokeTarget) {
source.strokeStyle = color;
} else {
source.fillStyle = color;
}
if (strokeOrigin) {
target.strokeStyle = color;
} else {
target.fillStyle = color;
}
links.push({
count: counts[dport],
source: source,
target: target,
width: lineScale(counts[dport]),
color: color
});
}
}
}
return links.sort(function(l1, l2) {
return l2.count - l1.count;
});
},
dPortLinks: function(dport) {
var links = []
for (var sourceCountry in this._links) {
for (var sourceCity in this._links[sourceCountry]) {
var s = this.getCity(sourceCountry, sourceCity);
for (var targetCountry in s.counts) {
for (var targetCity in s.counts[targetCountry]) {
var t = this.getCity(targetCountry, targetCity);
for (var targetdport in s.counts[targetCountry][targetCity]) {
var c = s.counts[targetCountry][targetCity][targetdport];
if (dport && dport === targetdport) {
links.push({
count: c,
source: s,
target: t,
dport: dport
});
}
}
}
}
}
}
return links;
},
extend: function(o) {
o.__proto__ = this;
o._links = {};
return o;
}
};
// Set up the link models, one for origins, and one for targets
var linkModels = {
origins: LinksModel.extend({
insert: function(d) {
this.insertLink({
country: d.country,
city: d.city
}, {
country: d.country2,
city: d.city2
},
d.dport);
this.insertLoc({
country: d.country,
city: d.city,
latitude: d.latitude,
longitude: d.longitude
});
},
remove: function(d) {
this.removeLink({
country: d.country,
city: d.city
}, {
country: d.country2,
city: d.city2
},
d.dport);
this.removeLoc({
country: d.country,
city: d.city,
latitude: d.latitude,
longitude: d.longitude
}, {
country: d.country,
city: d.city
}, {
country: d.country2,
city: d.city2
},
d.dport);
}
}),
targets: LinksModel.extend({
insert: function(d) {
this.insertLink({
country: d.country2,
city: d.city2
}, {
country: d.country,
city: d.city
},
d.dport);
this.insertLoc({
country: d.country2,
city: d.city2,
latitude: d.latitude2,
longitude: d.longitude2
});
},
remove: function(d) {
this.removeLink({
country: d.country2,
city: d.city2
}, {
country: d.country,
city: d.city
},
d.dport);
this.removeLoc({
country: d.country2,
city: d.city2,
latitude: d.latitude2,
longitude: d.longitude2
}, {
country: d.country2,
city: d.city2
}, {
country: d.country,
city: d.city
},
d.dport);
}
})
};
var countryModel = {
// The raw list of data
_raw: undefined,
_iso2: undefined,
_iso3: undefined,
_countries: undefined,
push: function(country) {
// Push a new country
this._raw.push(country);
if (country.iso2) this._iso2[country.iso2] = country;
if (country.iso3) this._iso3[country.iso3] = country;
if (country.country) this._countries[country.country] = country;
},
set: function(raw) {
this._raw = [];
this._iso2 = {};
this._iso3 = {};
this._countries = {};
for (var i = 0; i < raw.length; i++) {
this.push(raw[i]);
}
},
getByIso2: function(iso2) {
return this._iso2[iso2.toUpperCase()];
},
getByIso3: function(iso3) {
return this._iso3[iso3.toUpperCase()];
},
getByCountry: function(country) {
return this._countries[country];
}
};
/*
* painter handles rendering to the canvas
*/
// Prepare canvas and buffer
var canvas = d3.select("#content").append("canvas")
.text("This browser doesn't support Canvas elements")
.attr("id", "visible-canvas")
.attr("class", "overlay")
.attr("width", width)
.attr("height", height)
var bufCanvas = d3.select("#content").append("canvas")
.attr("class", "buffer overlay")
.attr("width", width)
.attr("height", height);
// 使得绘制的元素比例一致
var logScale = d3.scale.log()
.domain([1, 200])
.range([1, 10]);
var lineScale = function(x) {
return logScale(x)
};
var circleScale = function(x) {
return Math.ceil(1.4 * logScale(x))
};
var colorScale = (function() {
var log = d3.scale.log()
.domain([1, 600])
.range([1, 100]);
return function(v) {
return log(v);
}
})();
var painter = {
// The various drawings
drawings: {
// Draw the attack nodes
nodes: {
// The visible nodes
active: true,
nodeModel: nodeModel,
// Canvas composition: "lighter", "darker", ...
compositeOperation: undefined,
getRadius: function(d) {
// Given a node, return the radius
var growthEnd = 60,
growthMax = 80,
growthStep = growthMax / growthEnd,
shrinkEnd = 120,
shrinkMin = 20,
shrinkStep = (growthMax - shrinkMin) / (shrinkEnd - growthEnd);
if (d.age >= 0 && d.age < growthEnd) {
return growthStep * d.age;
} else if (d.age < shrinkEnd) {
return growthMax - shrinkStep * (d.age - growthEnd);
} else {
return shrinkMin;
}
},
draw: function(context) {
nodeModel.prune();
if (this.compositeOperation) {
context.globalCompositeOperation = this.compositeOperation;
}
var that = this;
nodeModel.get(nodeModel.ATTACKS)
.forEach(function(n) {
var c = d3.rgb(colorizer(n.dport)),
r = that.getRadius(n);
context.drawImage(particler(c.r, c.g, c.b, 1),
n.x - r / 2, n.y - r / 2, r, r);
});
}
},
// Draw new node pings
pings: {
active: true,
order: -1,
duration: 80,
scaleRadius: d3.scale.linear().domain([1, 80]).range([1, 48]),
scaleOpacity: d3.scale.linear().domain([1, 80]).range([1, 0]),
draw: function(context) {
var pi = Math.PI;
for (var i = 0; i < nodeModel.nodes.length; i++) {
var n = nodeModel.nodes[i];
if (n.type === nodeModel.ATTACKS && n.age < this.duration) {
context.globalAlpha = this.scaleOpacity(n.age);
context.strokeStyle = colorizer(n.dport);
context.lineWidth = 3;
context.beginPath();
context.arc(n.x, n.y, this.scaleRadius(n.age),
0, 2 * pi);
context.stroke();
}
}
}
},
// Draw the origin nodes
origins: {
active: true,
order: -2,
draw: function(context) {
context.globalAlpha = 0.65;
context.fillStyle = "#fff";
var cities = linkModels.origins.getCities();
var ceil = Math.ceil;
var pi = Math.PI;
for (var i = 0; i < cities.length; i++) {
var total = linkModels.origins.total(cities[i].counts),
latlng = projection([cities[i].longitude,
cities[i].latitude
]),
r = ceil(circleScale(total));
context.beginPath();
context.arc(latlng[0], latlng[1], r, 0, pi * 2);
context.fill();
}
}
},
// Draw the city nodes
targets: {
active: true,
order: -2,
draw: function(context) {
context.globalAlpha = 0.65;
context.strokeStyle = "#fff";
context.fillStyle = "#fff";
var cities = linkModels.targets.getCities();
var ceil = Math.ceil;
var pi = Math.PI;
for (var i = 0; i < cities.length; i++) {
var total = linkModels.targets.total(cities[i].counts),
latlng = projection([cities[i].longitude,
cities[i].latitude
]),
r = ceil(circleScale(total));
context.beginPath();
context.arc(latlng[0], latlng[1], r, 0, pi * 2);
context.stroke();
context.fillRect(latlng[0] - 1, latlng[1] - 1, 2, 2);
}
}
},
// Draw the target impact
targetImpact: {
active: true,
impactRadiusScale: d3.scale.linear().domain([1, 40]).range([1, 30]),
impactOpacityScale: d3.scale.linear().domain([1, 40]).range([1, 0]),
impactWidth: 1,
draw: function(ctx) {
var pi = Math.PI;
ctx.fillStyle = "#f00";
nodeModel.get(nodeModel.TARGETS)
.forEach(function(n) {
var c = d3.rgb(colorizer(n.dport)),
afterArrival = n.age - n["arrivalAge"];
r = 10;
if ("arrivalAge" in n) {
var r = this.impactRadiusScale(afterArrival);
ctx.save();
ctx.globalAlpha = this.impactOpacityScale(afterArrival);
ctx.strokeStyle = c.toString();
ctx.lineWidth = this.impactWidth;
ctx.beginPath();
ctx.arc(n.x, n.y, r, 0, 2 * pi);
ctx.closePath();
ctx.stroke();
ctx.restore();
}
if (n.path.length > 0) {
var point = n.path[n.path.length - 1];
if (n.x != point.x && n.y != point.y) {
ctx.save();
var grd = ctx.createLinearGradient(
n.x, n.y, point.x, point.y);
grd.addColorStop(0, rgbaString(c, 1));
grd.addColorStop(1, rgbaString(c, 0));
ctx.lineCap = 'round';
ctx.lineWidth = 2;
ctx.beginPath();
ctx.moveTo(n.x, n.y);
ctx.strokeStyle = grd;
ctx.lineTo(point.x, point.y);
ctx.closePath();
ctx.stroke();
ctx.restore();
}
}
ctx.drawImage(particler(c.r, c.g, c.b, 1),
n.x - r / 2, n.y - r / 2, r, r);
}, this);
}
},
// Draw the provided links
// A link: {source: _, target: _, color: _, width: _}
// source/target: {x: _, y: _, strokeStyle: _, fillStyle: _}
links: {
active: true,
order: -1,
data: [],
draw: function(context) {
var pi = Math.PI;
for (var i = 0; i < this.data.length; i++) {
context.beginPath();
context.strokeStyle = this.data[i].color;
context.moveTo(this.data[i].source.x, this.data[i].source.y);
context.lineTo(this.data[i].target.x, this.data[i].target.y);
context.lineWidth = this.data[i].width;
context.lineCap = "round";
context.stroke();
context.lineWidth = 2;
context.beginPath();
context.arc(this.data[i].source.x, this.data[i].source.y,
this.data[i].source.r || 5, 0, pi * 2);
if (this.data[i].source.fillStyle) {
context.fillStyle = this.data[i].source.fillStyle;
context.fill();
}
if (this.data[i].source.strokeStyle) {
context.strokeStyle = this.data[i].source.strokeStyle;
context.stroke();
}
context.beginPath();
context.fillStyle = this.data[i].target.fillStyle || "#fff";
context.arc(this.data[i].target.x, this.data[i].target.y,
this.data[i].target.r || 5, 0, pi * 2);
if (this.data[i].target.fillStyle) {
context.fillStyle = this.data[i].target.fillStyle;
context.fill();
}
if (this.data[i].target.strokeStyle) {
context.strokeStyle = this.data[i].target.strokeStyle;
context.stroke();
}
}
}
}
},
// State
_activeCanvas: {
canvas: canvas,
context: canvas.node().getContext("2d")
},
_clearContext: function(context) {
context.save();
context.clearRect(0, 0, width, height);
context.restore();
},
_drawSort: function(d1, d2) {
return d1.order || 0 - d2.order || 0;
},
redraw: function() {
this.drawStart = new Date().getTime();
// Draw the active drawings
this._clearContext(this._activeCanvas.context);
for (var drawing in this.drawings) {
if (this.drawings[drawing].active) {
this.drawings[drawing].draw(this._activeCanvas.context);
}
}
var that = this;
var nextFrame = 1000 / 30 - (new Date().getTime() - this.drawStart);
if (nextFrame < 0) nextFrame = 0;
this._timeout = setTimeout(function() {
that.redraw()
}, nextFrame);
},
start: function(interval) {
this.redraw();
},
stop: function() {
clearTimeout(this._timeout);
}
};
// Event handling for setting the painter's links
var _ = (function() {
var previous;
function cityKey(n) {
return [n.country, n.city];
}
canvas.on("mousemove", function() {
// TODO -- this is a hacky mess
var nearestOrigin = linkModels.origins.pixelsFromNearest(d3.mouse(this)),
nearestTarget = linkModels.targets.pixelsFromNearest(d3.mouse(this)),
nearest, msg, model, mapfn, strokeOrigin, strokeTarget;
if (nearestOrigin || nearestTarget) {
if (!nearestTarget || nearestOrigin.distance <= nearestTarget.distance) {
nearest = nearestOrigin;
msg = "Attacks originating from: ";
model = linkModels.origins;
strokeOrigin = true;
}
if (!nearestOrigin || nearestTarget.distance < nearestOrigin.distance) {
nearest = nearestTarget;
msg = "Attacks targeting: ";
model = linkModels.targets;
strokeTarget = true;
}
}
if (nearest && nearest.distance < 20) {
var key = cityKey(nearest);
if (key !== previous) {
painter.drawings.links.data =
model.cityToLinks(nearest, strokeOrigin, strokeTarget);
displayLabel.set(msg +
(nearest.city === "" ? "<unknown>" : nearest.city) + ", " + countryModel.getByIso2(nearest.country).country);
}
} else {
previous = key;
painter.drawings.links.data.length = 0;
displayLabel.clear();
}
});
})();
/*
* The legend
*/
(function() {
var legend = d3.select("#legend-container")
.append("div")
.attr("id", "legend");
var attack = legend.append("div"),
width = 20,
height = 20;
attack.append("h4").text("每个粒子代表一个攻击");
attack.append("canvas")
.attr("width", width * 2)
.attr("height", height * 2)
.node().getContext("2d")
.drawImage(particler(255, 255, 255, 1),
width / 2, height / 2, width, height);
var clusters = legend.append("div").attr("class", "clusters");
clusters.append("h4").text("这是攻击来源");
var height = 30;
var clusterList = clusters.append("ul").selectAll("li")
.data([1, 10, 200])
.enter().append("li");
clusterList.append("svg")
.style("width", function(d) {
return circleScale(d) * 2;
})
.style("height", height)
.append("circle")
.attr("fill", "white")
.attr("cy", function(d) {
return height - circleScale(d);
})
.attr("cx", function(d) {
return circleScale(d);
})
.attr("r", function(d) {
return circleScale(d);
});
clusterList.append("p")
.text(function(d) {
return d;
});
var countryColors = legend.append("div").attr("class", "country-colors");
countryColors.append("h4").text("这是被攻击国家");
var r = 4;
var countryColorList = countryColors.append("ul").selectAll("li")
.data([1, 5, 25, 100, 500])
.enter().append("li");
countryColorList.append("svg")
.style("width", r * 2)
.style("height", r * 2)
.append("circle")
.attr("fill", function(d) {
return settings.countryColor(d);
})
.attr("cy", r)
.attr("cx", r)
.attr("r", r);
countryColorList.append("p")
.text(function(d) {
return d;
});
})(window);
/*
* Stats the state machine
*/
function Stats(params) {
this.state = params.state || d3.map();
this.elt = params.elt || d3.select("body");
this.tag = params.tag || "div";
this.insert = function(incoming) {
// Insert a new item, updating the state. params.insert should mutate
params.insert(incoming, this.state);
};
this.data = function() {
// Get the data as a list
if (params.data) return params.data(this.state);
else this.state;
}
this.redraw = function() {
if (params.redraw) {
params.redraw()
} else {
this.elt.selectAll(this.tag)
.data(this.data())
.enter().append(this.tag)
.text(function(d) {
return d
});
}
}
}
var statsManager = {
insert: function(incoming) {
for (var i = 0; i < this.stats.length; ++i) {
this.stats[i].insert(incoming);
}
},
redraw: function() {
for (var i = 0; i < this.stats.length; i++) {
this.stats[i].redraw();
}
},
stats: [
new Stats({
// Color countries
state: {
sources: {},
targets: {}
},
insert: function(d, state) {
function pushState(m, key) {
if (key in m) {
m[key] = m[key] + 1;
} else {
m[key] = 1;
}
}
//pushState(state.sources, d.country);
pushState(state.targets, d.country2);
//d3.select("#" + d.country)
// .attr("fill", d3.rgb(colorScale(state.sources[d.country]), 0, 0).toString());
d3.select("#" + d.country2)
.attr("fill", settings.countryColor(state.targets[d.country2]));
},
redraw: function() {}
}),
// Origin data table
new Stats({
state: d3.map(),
insert: function(incoming, state) {
this.updated = incoming.country;
if (state.has(incoming.country)) {
state.set(incoming.country,
state.get(incoming.country) + 1);
} else {
state.set(incoming.country, 1);
}
},
redraw: function() {
var data = this.state.entries()
.sort(function(d1, d2) {
return d2.value - d1.value;
})
.slice(0, settings.topTableRows),
updated = this.updated;
var rows = d3.select("#left-data").selectAll("tr.row")
.data(data, function(d) {
return d.key;
});
rows.enter()
.append("tr")
.attr("class", "row")
.on("mouseenter", function(d) {
var country = countryModel.getByIso2(d.key);
displayLabel.set("Attacks originating from: " +
(country ? country.country : d.key));
// On mouseenter, use all country data to create links
painter.drawings.links.data =
linkModels.origins.getCountry(d.key)
.reduce(function(acc, city) {
return acc.concat(
linkModels.origins.cityToLinks(
city, true, false));
}, []);
})
.on("mouseleave", function() {
displayLabel.clear();
});
rows.sort(function(d1, d2) {
return d2.value - d1.value;
});
rows.exit().remove();
rows.filter(function(d) {
return d.key == updated
})
.style("color", settings.triggerColor)
.transition()
.duration(1000)
.style("color", "white");
var cols = rows.selectAll("td")
.data(function(d) {
var country = countryModel.getByIso2(d.key);
return [
'<div class="bar" style="width: ' +
settings.tableBarWidth(d.value + 1) + '"></div>',
spanWrap(d.value, ["numeric"]),
flagTag(d.key),
(country ? country.country : d.key)
];
});
cols.enter().append("td");
cols.html(function(d) {
return d;
});
cols.exit().remove();
}
}),
// Targeted data table
new Stats({
state: d3.map(),
insert: function(incoming, state) {
this.updated = incoming.country2;
if (state.has(incoming.country2)) {
state.set(incoming.country2,
state.get(incoming.country2) + 1);
} else {
state.set(incoming.country2, 1);
}
},
redraw: function() {
var data = this.state.entries()
.sort(function(d1, d2) {
return d2.value - d1.value;
})
.slice(0, settings.topTableRows),
updated = this.updated;
var rows = d3.select("#right-data").selectAll("tr.row")
.data(data, function(d) {
return d.key;
});
rows.enter()
.append("tr")
.attr("class", "row")
.on("mouseenter", function(d) {
var country = countryModel.getByIso2(d.key);
displayLabel.set("Attacks targeting: " +
(country ? country.country : d.key));
// On mouseenter, use all country data to create links
painter.drawings.links.data =
linkModels.targets.getCountry(d.key)
.reduce(function(acc, city) {
return acc.concat(
linkModels.targets.cityToLinks(
city, false, true));
}, []);
})
.on("mouseleave", function() {
displayLabel.clear();
});
rows.sort(function(d1, d2) {
return d2.value - d1.value;
});
rows.exit().remove();
rows.filter(function(d) {
return d.key == updated
})
.style("color", "blue")
.transition()
.duration(1000)
.style("color", "white");
var cols = rows.selectAll("td")
.data(function(d) {
var country = countryModel.getByIso2(d.key);
return [
'<div class="bar" style="width: ' +
settings.tableBarWidth(d.value + 1) + '"></div>',
spanWrap(d.value, ["numeric"]),
flagTag(d.key),
(country ? country.country : d.key)
];
});
cols.enter().append("td");
cols.html(function(d) {
return d;
});
cols.exit().remove();
}
}),
new Stats({
state: d3.map(),
insert: function(incoming) {
this.updated = incoming.dport;
if (this.state.has(incoming.dport)) {
this.state.set(incoming.dport,
this.state.get(incoming.dport) + 1);
} else {
this.state.set(incoming.dport, 1);
}
},
redraw: function() {
var data = this.state.entries()
.sort(function(d1, d2) {
return d2.value - d1.value;
})
.slice(0, settings.portTableRows),
updated = this.updated;
var rows = d3.select("#bottom-right-data").selectAll("tr.row")
// XXX - I'm not sure why d is undef the first time through.
.data(data, function(d, i) {
return d ? d.key : i;
});
rows.enter()
.append("tr")
.attr("class", "row")
.on("mouseenter", function(d) {
var port = ports[d.key];
displayLabel.set("Attacks made over: " +
(port ? port : "unknown") +
" [" + d.key + "]");
painter.drawings.links.data =
linkModels.origins.dPortLinks(d.key).map(function(c) {
var r = circleScale(c.count),
color = colorizer(c.dport);
return {
source: {
x: c.source.pt[0],
y: c.source.pt[1],
fillStyle: color,
r: r
},
target: {
x: c.target.pt[0],
y: c.target.pt[1],
strokeStyle: color,
r: r
},
width: lineScale(c.count),
color: colorizer(c.dport)
}
});
})
.on("mouseleave", function(d) {
displayLabel.clear();
})
rows.sort(function(d1, d2) {
return d2.value - d1.value;
});
rows.exit().remove();
rows.filter(function(d) {
return d.key == updated
})
.style("color", function(d) {
return colorizer(d.key);
})
.transition()
.duration(1000)
.style("color", "white");
var cols = rows.selectAll("td")
.data(function(d) {
return [
'<div class="bar" style="width: ' +
settings.tableBarWidth(d.value + 1) + '"></div>',
spanWrap(d.value, ["numeric"]),
'<span class="port-circle" style="color:' + colorizer(d.key) +
'">●</span>',
ports[d.key] ? ports[d.key] : "unknown",
spanWrap(d.key, ["numeric"])
];
});
//function(d) { return d; });
cols.enter().append("td");
cols.html(function(d) {
return d;
});
cols.exit().remove();
}
}),
new Stats({
// #console stats manager
state: [],
insert: function(incoming, state) {
state.push(incoming);
while (state.length > settings.consoleTableRows) {
state.shift();
}
return state;
},
redraw: function() {
var that = this;
var rows = d3.select("#events-data").selectAll("tr.row")
.data(this.state, function(d) {
return d.id;
});
rows.enter().append("tr")
.style("color", function(d) {
return colorizer(d.dport);
})
.attr("class", "row");
rows.exit().remove();
var cols = rows.selectAll("td")
.data(function(d) {
return [
d.datetime,
spanWrap(d.org, ["org", "overflow"]),
spanWrap(
(d.city === "" ? "unknown" : d.city) + ", " +
countryModel.getByIso2(d.country).country, ["location", "overflow"]),
d.md5,
spanWrap(
(d.city2 === "" ? "unknown" : d.city2) + ", " +
countryModel.getByIso2(d.country2).country, ["location", "overflow", "numeric"]),
spanWrap(d.service || "unknown", ["service", "overflow"]),
spanWrap(d.dport, ["numeric"])
];
});
cols.enter().append("td")
.html(function(d) {
return d;
});
cols.exit().remove();
}
})
]
};
// Keep track of the rate
var rateTicker = {
data: [],
interval: 90000,
graph: d3.select("#content")
.append("svg")
.attr("id", "rate-graph"),
x: d3.time.scale().range([800, 0]),
opacity: d3.time.scale().range([0, 1]),
push: function(d) {
this.data.push({
date: Date.now(),
key: d.dport
});
},
clean: function(toDate) {
// Clean out old data
while (this.data.length > 0 && this.data[0].date < toDate) {
this.data.shift();
}
},
step: function() {
var that = this,
now = Date.now(),
range = [now - this.interval, now];
this.clean(range[0]);
this.x.domain(range);
this.opacity.domain(range);
var simpleLine = function(h) {
return function(d) {
var cx = that.x(d.date);
return d3.svg.line()([
[cx, 0],
[cx, h || 20]
])
}
};
var ticks = this.graph.selectAll("path.tick")
.data(this.data, function(d) {
return d.date;
});
ticks.enter().append("path")
.attr("class", "tick")
.attr("d", simpleLine())
.style("stroke", function(d) {
return colorizer(d.key);
});
ticks.exit().remove();
ticks
//.transition().duration(30)
.attr("d", simpleLine())
.style("opacity", function(d) {
return that.opacity(d.date);
});
},
start: function() {
var that = this;
setInterval(function() {
that.step();
}, 50);
}
};
var wsDiscTime = 0;
function start(loc, psk) {
var webSocket = new WebSocket(loc || settings.wsHost);
var pauser = {
elt: d3.selectAll(".controls"),
_buffer: [],
unbuffer: function(d) {
while (this._buffer.length > 0) {
this.insert(this._buffer.shift());
}
statsManager.redraw();
},
insert: function(d) {
nodeModel.pushAttack(d);
statsManager.insert(d);
for (var model in linkModels) {
linkModels[model].insert(d);
}
rateTicker.push(d);
d.pruneTS = new Date().getTime() / 1000 + settings.pruneInterval;
timestampedData.push(d);
},
push: function(d) {
if (this.paused()) {
this._buffer.push(d);
} else {
this.insert(d);
statsManager.redraw();
}
},
paused: function() {
return this.elt.node().dataset.paused === "true";
},
toggle: function() {
var dataset = this.elt.node().dataset;
var button = d3.selectAll(".play-pause");
if (this.paused()) {
button.classed("icon-play", false);
button.classed("icon-pause", true);
this.elt.node().dataset.paused = "false";
this.unbuffer();
} else {
button.classed("icon-pause", false);
button.classed("icon-play", true);
this.elt.node().dataset.paused = "true";
}
}
};
pauser.elt.on("click", function() {
pauser.toggle();
});
webSocket.onopen = function() {
wsDiscTime = 0;
d3.select("#events-data").selectAll("tr.row").remove();
setInterval(function () {
webSocket.send(psk || settings.psk);
}, 1000);
};
webSocket.onmessage = function(evt) {
if (!evt) {
return;
}
// Parse the json to a js obj and clean the data
var datum = eval("(" + evt.data + ")");
if (datum.longitude == 0 && datum.latitude == 0) {
datum.longitude = -5;
datum.latitude = -50;
}
var startLoc = projection([datum.longitude, datum.latitude]);
var endLoc = projection([datum.longitude2, datum.latitude2]);
if (datum.error) {
showMessage("ERROR: " + datum.error.msg);
}
for (var prop in datum) {
if (settings.numberProps.indexOf(prop) !== -1) {
datum[prop] = Number(datum[prop]);
}
}
function cleanCountry(country) {
// Clean incoming country code
if (country === "USA") {
return "US";
}
return country
}
datum.country = cleanCountry(datum.country);
datum.country2 = cleanCountry(datum.country2);
datum.service = datum.dport in ports ? ports[datum.dport] : undefined;
datum.cx = startLoc[0];
datum.cy = startLoc[1];
datum.x = startLoc[0];
datum.y = startLoc[1];
datum.targetX = endLoc[0];
datum.targetY = endLoc[1];
datum.id = getID();
datum.datetime = (new Date()).toISOString()
.replace("T", "&ensp;")
.slice(0, -2);
pauser.push(datum);
};
webSocket.onclose = function() {
//try to reconnect in 5 seconds
var interval = 1000;
wsDiscTime += 1000;
if (wsDiscTime > settings.wsTimeout) {
showMessage("We are having difficulties in the WebSocket connectivity. We will continue trying...");
wsDiscTime = 0;
}
setTimeout(function() {
console.log("websocket closed, reconnecting in " + interval + "ms");
start(loc, psk);
}, interval);
};
return webSocket;
}
/*
* Load external data, and manage loading state
*/
queue()
.defer(d3.json, "data/readme-world.json")
.defer(d3.tsv, "data/port-names.tsv")
.defer(d3.csv, "data/country-codes.csv")
.await(function(error, world, rawPorts, countryCodes) {
// Update the countryModel
countryModel.set(countryCodes);
countryModel.push({
iso2: "O1",
country: "Mil/Gov"
});
// Temporary mapping to key the map
var mapCodes = {};
countryCodes.forEach(function(d) {
mapCodes[Number(d.isonum)] = d.iso2;
});
// Enter the countries
svg.append("g")
.attr("class", "world")
.selectAll("path")
.data(topojson.feature(world, world.objects.countries).features)
.enter().insert("path")
.attr("class", "country")
.attr("id", function(d) {
return mapCodes[d.id];
})
.attr("fill", settings.countryColor(0))
.attr("d", path);
ports = parsePorts(rawPorts);
loadingToggle();
var webSocket = start();
nodeModel.start();
painter.start();
rateTicker.start();
});
// Export the external API
window.IPViking = {
settings: settings
};
var unknownLoc = projection([-5, -50]);
d3.selectAll("#unknown-icon").style({
'left': unknownLoc[0] - 18,
'top': unknownLoc[1] - 18
});
setInterval(function() {
prune();
}, 30000);
})(window);