magic-wormhole/misc/web/timeline.js
2016-06-03 22:32:52 -07:00

1054 lines
37 KiB
JavaScript

var d3; // hush
var container = d3.select("#viz");
var data;
var items;
var globals = {};
var zoom = d3.behavior.zoom().scaleExtent([1, Infinity]);
function zoomin() {
//var w = Number(container.style("width").slice(0,-2));
//console.log("zoomin", w);
//zoom.center([w/2, 20]); // doesn't work yet
zoom.scale(zoom.scale() * 2);
globals.redraw();
}
function zoomout() {
zoom.scale(zoom.scale() * 0.5);
globals.redraw();
}
function is_span(ev, category) {
if (ev.category === category && !!ev.stop)
return true;
return false;
}
function is_event(ev, category) {
if (ev.category === category && !ev.stop)
return true;
return false;
}
const server_message_color = {
"welcome": 0, // receive
"bind": 0, // send
"allocate": 1, // send
"allocated": 1, // receive
"list": 2, // send
"channelids": 2, // receive
"claim": 3, // send
"watch": 4, // send
"deallocate": 5, // send
"deallocated": 5, // receive
"error": 6, // receive
//"add": 8, // send (client message)
//"message": 8, // receive (client message)
"ping": 7, // send
"pong": 7 // receive
};
const proc_map = {
"command dispatch": "dispatch",
"code established": "code-established",
"key established": "key-established",
"transit connected": "transit-connected",
"exit": "exit",
"transit connect": "transit-connect",
"import": "import"
};
const TX_COLUMN = 14;
const RX_COLUMN = 18;
const SERVER_COLUMN0 = 20;
const SERVER_COLUMNS = [20,21,22,23,24,25];
const NUM_SERVER_COLUMNS = 6;
const MAX_COLUMN = 45;
function x_offset(offset, side_name) {
if (side_name === "send")
return offset;
return MAX_COLUMN - offset;
}
function side_text_anchor(side_name) {
if (side_name === "send")
return "end";
return "start";
}
function side_text_dx(side_name) {
if (side_name === "send")
return "-5px";
return "5px";
}
d3.json("data.json", function(d) {
data = d;
// data is {send,receive}{fn,events}
// each event has: {name, start, [stop], [server_rx], [server_tx],
// [id], details={} }
// Display all timestamps relative to the sender's startup event. If all
// clocks are in sync, this will be the same as first_timestamp, but in
// case they aren't, I'd rather use the sender as the reference point.
var first = data.send.events[0].start;
// The X axis is divided up into 50 slots, and then scaled to the screen
// later. The left portion represents the "wormhole send" side, the
// middle is the rendezvous server, and the right portion is the
// "wormhole receive" side.
//
// 0: time axis, tick marks
// 3: sender process events: import, dispatch, exit
// 4: sender major application-level events: code/key establishment,
// transit-connect
// 8: sender stalls: waiting for user, waiting for permission
// 10: sender websocket transmits originate from here
// 15: sender websocket receives terminate here
// 20-25: rendezvous-server message lanes
// 30: receiver websocket receives
// 35: receiver websocket transmits
// 37: receiver stalls
// 41: receiver app-level events
// 42: receiver process events
var first_timestamp = Infinity;
var last_timestamp = 0;
function prepare_data(e, side_name) {
var rel_e = {side_name: side_name, // send or receive
name: e.name,
start: e.start - first,
details: e.details
};
if (e.stop) rel_e.stop = e.stop - first;
// sort events into categories, assign X coordinates to some
if (proc_map[e.name]) {
rel_e.category = "proc";
rel_e.x = x_offset(3, side_name);
rel_e.text = proc_map[e.name];
if (e.name === "import")
rel_e.text += " " + e.details.which;
}
if (e.details.waiting) {
rel_e.category = "wait";
var off = 8;
if (e.details.waiting === "user")
off += 0.5;
rel_e.x = x_offset(off, side_name);
}
// also, calculate the overall time domain while we're at it
[rel_e.start, rel_e.stop].forEach(v => {
if (v) {
if (v > last_timestamp) last_timestamp = v;
if (v < first_timestamp) first_timestamp = v;
}
});
return rel_e;
}
var events = data.send.events.map(e => prepare_data(e, "send"));
events = events.concat(data.receive.events.map(e => prepare_data(e, "receive")));
/* "Client messages" are ones that go all the way from one client to the
other, through the rendezvous channel (and get echoed back to the sender
too). We can correlate three websocket messages for each (the send, the
local receive, and the remote receive) by comparing their "id" strings.
Scan for all client messages, to build a list of central columns. For
each message, we'll have tx/server_rx/server_tx/rx for the sending side,
and server_rx/server_tx/rx for the receiving side. The "add" event
contributes tx, the sender's echo contributes and the "message" event
contributes server_rx, server_tx, and rx.
*/
var side_map = new Map(); // side -> "send"/"receive"
var c2c = new Map(); // msgid => {send,receive}{tx,server_rx,server_tx,rx}
events.forEach(ev => {
var id, phase;
if (ev.name === "ws_send") {
if (ev.details.type !== "add")
return;
id = ev.details.id;
phase = ev.details.phase;
side_map.set(ev.details._side, ev.side_name);
} else if (ev.name === "ws_receive") {
if (ev.details.message.type !== "message")
return;
id = ev.details.message.id;
phase = ev.details.message.phase;
} else
return;
if (!c2c.has(id)) {
c2c.set(id, {phase: phase,
side_id: ev.details._side,
//tx_side_name: assigned when we see 'add'
id: id,
arrivals: []
//col, server_x: assigned later
//server_rx: assigned when we see 'message'
});
}
var cm = c2c.get(id);
if (ev.name === "ws_send") { // add
cm.tx = ev.start;
cm.tx_x = x_offset(TX_COLUMN, ev.side_name);
cm.tx_side_name = ev.side_name;
} else { // message
cm.server_rx = ev.details.message.server_rx - first;
cm.arrivals.push({server_tx: ev.details.message.server_tx - first,
rx: ev.start,
rx_x: x_offset(RX_COLUMN, ev.side_name)});
}
});
// sort c2c messages by initial sending time
var client_messages = Array.from(c2c.values());
client_messages.sort( (a,b) => (a.tx - b.tx) );
// assign columns
// TODO: identify overlaps between the c2c messages, share columns
// between messages which don't overlap
client_messages.forEach((cm,index) => {
cm.col = index % 6;
cm.server_x = 20 + cm.col;
});
console.log("client_messages", client_messages);
console.log(side_map);
console.log(first_timestamp, last_timestamp);
/* "Server messages" are ones that stop or originate at the rendezvous
server. These are of types other than "add" or "message". Although many
of these provoke responses, we do not attempt to correlate these with
any other message. For outbound ws_send messages, we know the send
timestamp, but not the server receipt timestamp. For inbound ws_receive
messages, we know both.
*/
var outbound_sm = new Map();
globals.outbound_sm = outbound_sm;
events
.filter(ev => ev.name === "ws_send")
.forEach(ev => {
// we don't know the server receipt time, so draw a horizontal
// line by setting stop_timestamp=start_timestamp
var sm = {side_name: ev.side_name,
start_timestamp: ev.start,
stop_timestamp: ev.start,
start_x: x_offset(TX_COLUMN, ev.side_name),
end_x: x_offset(20, ev.side_name),
text_x: x_offset(TX_COLUMN, ev.side_name),
text_timestamp: ev.start,
text_dy: "-5px",
type: ev.details.type,
tip: ev.details.type,
ev: ev
};
outbound_sm.set(ev.details.id, sm);
});
events
.filter(ev => ev.name === "ws_receive")
.filter(ev => ev.details.message.type === "ack")
.forEach(ev => {
var id = ev.details.message.id;
var server_tx = ev.details.message.server_tx;
var sm = outbound_sm.get(id);
sm.stop_timestamp = server_tx - first;
});
var server_messages = [];
events
.filter(ev => ev.name === "ws_receive")
.filter(ev => ev.details.message.type !== "message")
.filter(ev => ev.details.message.type !== "ack")
.forEach(ev => {
var sm = {side_name: ev.side_name,
start_timestamp: ev.details.message.server_tx - first,
stop_timestamp: ev.start,
start_x: x_offset(20, ev.side_name),
end_x: x_offset(RX_COLUMN, ev.side_name),
text_x: x_offset(RX_COLUMN, ev.side_name),
text_timestamp: ev.start,
text_dy: "8px",
type: ev.details.message.type,
tip: ev.details.message.type,
ev: ev
};
server_messages.push(sm);
});
server_messages = server_messages.concat(
Array.from(outbound_sm.values())
.filter(sm => sm.type !== "add"));
console.log("server_messages", server_messages);
// TODO: this goes off the edge of the screen, use the viewport instead
var container_width = Number(container.style("width").slice(0,-2));
var container_height = Number(container.style("height").slice(0,-2));
container_height = 700; // no contents, so no height is allocated yet
// scale the X axis to the full width of our container
var x = d3.scale.linear().domain([0, 50]).range([0, container_width]);
// scale the Y axis later
var y = d3.scale.linear().domain([first_timestamp, last_timestamp])
.range([0, container_height]);
zoom.y(y);
zoom.on("zoom", redraw);
var tip = d3.tip()
.attr("class", "d3-tip")
.html(function(d) { return "<span>" + d + "</span>"; })
.direction("s")
;
var chart = container.append("svg:svg")
.attr("id", "outer_chart")
.attr("width", container_width)
.attr("height", container_height)
.attr("pointer-events", "all")
.call(zoom)
.call(tip)
;
var defs = chart.append("svg:defs");
defs.append("svg:marker")
.attr("id", "markerCircle")
.attr("markerWidth", 8)
.attr("markerHeight", 8)
.attr("refX", 5)
.attr("refY", 5)
.append("circle")
.attr("cx", 5)
.attr("cy", 5)
.attr("r", 3)
.attr("style", "stroke: none; fill: #000000;")
;
defs.append("svg:marker")
.attr("id", "markerArrow")
.attr("markerWidth", 26)
.attr("markerHeight", 26)
.attr("refX", 26)
.attr("refY", 12)
.attr("orient", "auto")
.attr("markerUnits", "userSpaceOnUse") // don't scale to stroke-width
.append("path")
.attr("d", "M8,20 L20,12 L8,4")
.attr("style", "stroke: #000000; fill: none")
;
chart.append("svg:line")
.attr("x1", x(0.5)).attr("y1", 0)
.attr("x2", x(0.5)).attr("y2", container_height)
.attr("class", "y_axis")
;
chart.append("svg:g")
.attr("class", "seconds_g")
.attr("transform", "translate("+(x(0.5)+5)+","+(container_height-10)+")")
.append("svg:text")
.text("seconds")
;
chart.append("svg:line")
.attr("x1", x(TX_COLUMN)).attr("y1", y(first_timestamp))
.attr("x2", x(TX_COLUMN)).attr("y2", y(last_timestamp))
.attr("class", "client_tx")
;
chart.append("svg:text")
.attr("x", x(TX_COLUMN)).attr("y", 10)
.attr("text-anchor", "middle")
.text("sender tx");
chart.append("svg:line")
.attr("x1", x(RX_COLUMN)).attr("y1", y(first_timestamp))
.attr("x2", x(RX_COLUMN)).attr("y2", y(last_timestamp))
.attr("class", "client_rx")
;
chart.append("svg:text")
.attr("x", x(RX_COLUMN)).attr("y", 10)
.attr("text-anchor", "middle")
.text("sender rx");
chart.selectAll("line.c2c_column").data(SERVER_COLUMNS)
.enter().append("svg:line")
.attr("class", "c2c_column")
.attr("x1", d => x(d)).attr("y1", y(first_timestamp))
.attr("x2", d => x(d)).attr("y2", y(last_timestamp))
;
chart.append("svg:line")
.attr("x1", x(MAX_COLUMN-RX_COLUMN)).attr("y1", y(first_timestamp))
.attr("x2", x(MAX_COLUMN-RX_COLUMN)).attr("y2", y(last_timestamp))
.attr("class", "client_rx")
;
chart.append("svg:text")
.attr("x", x(MAX_COLUMN-RX_COLUMN)).attr("y", 10)
.attr("text-anchor", "middle")
.text("receiver rx");
chart.append("svg:line")
.attr("x1", x(MAX_COLUMN-TX_COLUMN)).attr("y1", y(first_timestamp))
.attr("x2", x(MAX_COLUMN-TX_COLUMN)).attr("y2", y(last_timestamp))
.attr("class", "client_tx")
;
chart.append("svg:text")
.attr("x", x(MAX_COLUMN-TX_COLUMN)).attr("y", 10)
.attr("text-anchor", "middle")
.text("receiver tx");
// produces list of {p_from, p_to, col, add_arrow, tip}
function cm_line(cm) {
// We draw a bunch of two-point lines
var lines = [];
function push(p_from, p_to, add_arrow) {
lines.push({p_from: p_from, p_to: p_to,
col: cm.col, tip: cm.tip,
add_arrow: add_arrow});
}
// the first goes from the sender to the server_rx, if we know it
// TODO: tolerate not knowing it
var sender_point = [cm.tx_x, cm.tx];
var server_rx_point = [cm.server_x, cm.server_rx];
push(sender_point, server_rx_point, true);
// the second goes from the server_rx to the last server_tx
var last_server_tx = Math.max.apply(null,
cm.arrivals.map(a => a.server_tx));
var last_server_tx_point = [cm.server_x, last_server_tx];
push(server_rx_point, last_server_tx_point, false);
cm.arrivals.forEach(ar => {
var delivery_tx_point = [cm.server_x, ar.server_tx];
var delivery_rx_point = [ar.rx_x, ar.rx];
push(delivery_tx_point, delivery_rx_point, true);
});
return lines;
}
var all_cm_lines = [];
client_messages.forEach(v => {
all_cm_lines = all_cm_lines.concat(cm_line(v));
});
console.log(all_cm_lines);
var cm_colors = d3.scale.category10();
chart.selectAll("line.c2c").data(all_cm_lines)
.enter()
.append("svg:line")
.attr("class", "c2c") // circle-arrow-circle")
.attr("stroke", ls => cm_colors(ls.col))
.attr("style", ls => {
if (ls.add_arrow) return "marker-end: url(#markerArrow);";
return "";
})
.on("mouseover", ls => {
if (ls.tip)
tip.show(ls.tip);
chart.selectAll("circle.c2c").filter(d => d.col == ls.col)
.attr("r", 10);
chart.selectAll("line.c2c")
.classed("active", d => d.col == ls.col);
})
.on("mouseout", ls => {
tip.hide(ls);
chart.selectAll("circle.c2c")
.attr("r", 5);
chart.selectAll("line.c2c")
.classed("active", false);
})
;
chart.selectAll("g.c2c").data(client_messages)
.enter()
.append("svg:g")
.attr("class", "c2c")
.append("svg:text")
.attr("class", "c2c")
.attr("text-anchor", cm => side_text_anchor(cm.tx_side_name))
.attr("dx", cm => side_text_dx(cm.tx_side_name))
.attr("dy", "10px")
.attr("fill", cm => cm_colors(cm.col))
.text(cm => cm.phase);
function cm_dot(cm) {
var dots = [];
var color = cm_colors(cm.col);
var tip = cm.phase;
function push(x,y) {
dots.push({x: x, y: y, col: cm.col, color: color, tip: tip});
}
push(cm.tx_x, cm.tx);
cm.arrivals.forEach(ar => push(ar.rx_x, ar.rx));
return dots;
}
var all_cm_dots = [];
client_messages.forEach(cm => {
all_cm_dots = all_cm_dots.concat(cm_dot(cm));
});
chart.selectAll("circle.c2c").data(all_cm_dots)
.enter()
.append("svg:circle")
.attr("class", "c2c")
.attr("r", 5)
.attr("fill", dot => dot.color)
.on("mouseover", dot => {
if (dot.tip)
tip.show(dot.tip);
chart.selectAll("circle.c2c").filter(d => d.col == dot.col)
.attr("r", 10);
chart.selectAll("line.c2c")
.classed("active", d => d[2] == dot.col);
})
.on("mouseout", dot => {
tip.hide(dot);
chart.selectAll("circle.c2c")
.attr("r", 5);
chart.selectAll("line.c2c")
.classed("active", false);
})
;
// server messages
chart.selectAll("line.server-message").data(server_messages)
.enter()
.append("svg:line")
.attr("class", "server-message")
.attr("stroke", sm => cm_colors(server_message_color[sm.type] || 0))
.attr("style", "marker-end: url(#markerArrow)")
.on("mouseover", sm => {
if (sm.tip)
tip.show(sm.tip);
})
.on("mouseout", sm => {
tip.hide(sm);
})
;
chart.selectAll("g.server-message").data(server_messages)
.enter()
.append("svg:g")
.attr("class", "server-message")
.append("svg:text")
.attr("class", "server-message")
.attr("text-anchor", sm => side_text_anchor(sm.side_name))
.attr("dx", sm => side_text_dx(sm.side_name))
.attr("dy", sm => sm.text_dy)
.attr("fill", sm => cm_colors(server_message_color[sm.type] || 0))
.text(sm => sm.type);
// TODO: add dots on the known send/receive time points
var w = chart.selectAll("g.wait")
.data(events.filter(ev => ev.category === "wait"))
.enter().append("svg:g")
.attr("class", "wait");
w.append("svg:rect")
.attr("class", ev => "wait wait-"+ev.details.waiting)
.attr("width", 10);
var wt = chart.selectAll("g.wait-text")
.data(events.filter(ev => ev.category === "wait"))
.enter().append("svg:g")
.attr("class", "wait-text");
wt.append("svg:text")
.attr("class", ev => "wait-text wait-text-"+ev.details.waiting)
.attr("text-anchor", ev => ev.side_name === "send" ? "end" : "start")
.attr("dx", ev => ev.side_name === "send" ? "-5px" : "15px")
.attr("dy", "5px")
.text(v => v.name+" ("+v.details.waiting+")");
// process-related events
var pe = chart.selectAll("g.proc-event")
.data(events.filter(ev => is_event(ev, "proc")))
.enter().append("svg:g")
.attr("class", "proc-event");
pe.append("svg:circle")
.attr("class", ev => "proc-event proc-event-"+proc_map[ev.name])
.attr("cx", ev => ev.side_name === "send" ? "12px" : "-2px")
.attr("r", 5)
.attr("fill", "red")
.attr("width", 10);
pe.append("svg:text")
.attr("class", ev => "proc-event proc-event-"+proc_map[ev.name])
.attr("text-anchor", ev => ev.side_name === "send" ? "start" : "end")
.attr("dx", ev => ev.side_name === "send" ? "15px" : "-5px")
.attr("dy", "5px")
.attr("transform", "rotate(-30)")
.text(ev => proc_map[ev.name]);
// process-related spans
var ps = chart.selectAll("g.proc-span")
.data(events.filter(ev => is_span(ev, "proc")))
.enter().append("svg:g")
.attr("class", "proc-span");
ps.append("svg:rect")
.attr("class", ev => "proc-span proc-span-"+proc_map[ev.name])
.attr("width", 10);
var pst = chart.selectAll("g.proc-span-text")
.data(events.filter(ev => is_span(ev, "proc")))
.enter().append("svg:g")
.attr("class", "proc-span-text");
pst.append("svg:text")
.attr("class", ev => "proc-span-text proc-span-text-"+proc_map[ev.name])
.attr("text-anchor", ev => ev.side_name === "send" ? "start" : "end")
.attr("dx", ev => ev.side_name === "send" ? "15px" : "-5px")
.attr("dy", "5px")
.text(ev => ev.text);
function ty(d) { return "translate(0,"+y(d)+")"; }
function redraw() {
chart.selectAll("line.c2c")
.attr("x1", ls => x(ls.p_from[0]))
.attr("y1", ls => y(ls.p_from[1]))
.attr("x2", ls => x(ls.p_to[0]))
.attr("y2", ls => y(ls.p_to[1]))
;
chart.selectAll("g.c2c")
.attr("transform", cm =>
"translate("+x(cm.tx_x)+","+y(cm.tx)+")")
;
chart.selectAll("circle.c2c")
.attr("cx", d => x(d.x))
.attr("cy", d => y(d.y))
;
chart.selectAll("line.server-message")
.attr("x1", sm => x(sm.start_x))
.attr("y1", sm => y(sm.start_timestamp))
.attr("x2", sm => x(sm.end_x))
.attr("y2", sm => y(sm.stop_timestamp));
chart.selectAll("g.server-message")
.attr("transform", sm => {
return "translate("+x(sm.text_x)+","+y(sm.text_timestamp)+")";
})
;
chart.selectAll("g.wait")
.attr("transform", ev => {
return "translate("+x(ev.x)+","+y(ev.start)+")";
});
chart.selectAll("rect.wait")
.attr("height", ev => y(ev.stop)-y(ev.start));
chart.selectAll("g.wait-text")
.attr("transform", ev => {
return "translate("+x(ev.x)+","+y((ev.start+ev.stop)/2)+")";
});
chart.selectAll("g.proc-event")
.attr("transform", ev => {
return "translate("+x(ev.x)+","+y(ev.start)+")";
})
;
chart.selectAll("g.proc-span")
.attr("transform", ev => {
return "translate("+x(ev.x)+","+y(ev.start)+")";
})
;
chart.selectAll("rect.proc-span")
.attr("height", ev => y(ev.stop)-y(ev.start));
chart.selectAll("g.proc-span-text")
.attr("transform", ev => {
return "translate("+x(ev.x)+","+y((ev.start+ev.stop)/2)+")";
});
// vertical scale markers: horizontal tick lines at rational
// timestamps
// TODO: clicking on a dot should set the new zero time
var rules = chart.selectAll("g.rule")
.data(y.ticks(10))
.attr("transform", ty);
rules.select("text")
.text(t => y.tickFormat(10, "s")(t)+"s");
var newrules = rules.enter().insert("svg:g")
.attr("class", "rule")
.attr("transform", ty)
;
newrules.append("svg:line")
.attr("class", "rule-tick")
.attr("stroke", "black");
chart.selectAll("line.rule-tick")
.attr("x1", x(0.5)-5)
.attr("x2", x(0.5));
newrules.append("svg:line")
.attr("class", "rule-red")
.attr("stroke", "red")
.attr("stroke-opacity", .3);
chart.selectAll("line.rule-red")
.attr("x1", x(0.5))
.attr("x2", x(MAX_COLUMN));
newrules.append("svg:text")
.attr("class", "rule-text")
.attr("dx", ".1em")
.attr("dy", "-0.2em")
.attr("text-anchor", "start")
.attr("fill", "black")
.text(t => y.tickFormat(10, "s")(t)+"s");
chart.selectAll("text.rule-text")
.attr("x", 6 + 9);
rules.exit().remove();
}
redraw();
return;
function y_off(d) {
return (LANE_HEIGHT * (d.side*(data.lanes.length+1) + d.lane)
+ d.wiggle);
}
var bottom_rule_y = LANE_HEIGHT * data.sides.length * (data.lanes.length+1);
var bottom_y = bottom_rule_y + 45;
//var chart_g = chart.append("svg:g");
// this "backboard" rect lets us catch mouse events anywhere in the
// chart, even between the bars. Without it, we only see events on solid
// objects like bars and text, but not in the gaps between.
chart.append("svg:rect")
.attr("id", "outer_rect")
.attr("width", w).attr("height", bottom_y).attr("fill", "none");
// but the stuff we put inside it should have some room
w = w-50;
chart.selectAll("text.sides-label").data(data.sides).enter()
.append("svg:text")
.attr("class", "sides-label")
.attr("x", "0px")
.attr("y", function(d,idx) {
return y_off({side: idx, lane: data.lanes.length/2,
wiggle: 0}) ;})
.attr("text-anchor", "start") // anchor at top-left
.attr("dy", ".71em")
.attr("fill", "black")
.text(function(d) { return d; })
;
var lanes_by_sides = [];
data.sides.forEach(function(side, side_index) {
data.lanes.forEach(function(lane, lane_index) {
lanes_by_sides.push({side: side, side_index: side_index,
lane: lane, lane_index: lane_index});
});
});
chart.selectAll("text.lanes-label").data(lanes_by_sides).enter()
.append("svg:text")
.attr("class", "lanes-label")
.attr("x", "50px")
.attr("y", function(d) {
return y_off({side: d.side_index, lane: d.lane_index,
wiggle: 0}) ;})
.attr("text-anchor", "start") // anchor at top-left
.attr("dy", ".91em")
.attr("fill", "#f88")
.text(function(d) { return d.lane; })
;
chart.append("svg:text")
.attr("class", "seconds-label")
//.attr("x", w/2).attr("y", y + 35)
.attr("text-anchor", "middle")
.attr("fill", "black")
.text("seconds");
d3.select("#outer_chart").attr("height", bottom_y);
d3.select("#outer_rect").attr("height", bottom_y);
d3.select("#zoom").attr("transform", "translate("+(w-10)+","+10+")");
function reltime(t) {return t-data.bounds.min;}
var last = data.bounds.max - data.bounds.min;
//last = reltime(d3.max(data.dyhb, function(d){return d.finish_time;}));
last = last * 1.05;
// long downloads are likely to have too much info, start small
if (last > 10.0)
last = 10.0;
// d3.time.scale() has no support for ms or us.
var xOFF = d3.time.scale().domain([data.bounds.min, data.bounds.max])
.range([0,w]);
var x = d3.scale.linear().domain([-last*0.05, last])
.range([0,w]);
zoom.x(x);
function tx(d) { return "translate(" +x(d) + ",0)"; }
function left(d) { return x(reltime(d.start_time)); }
function left_server(d) { return x(reltime(d.server_sent)); }
function right(d) {
return d.finish_time ? x(reltime(d.finish_time)) : "1px";
}
function width(d) {
return d.finish_time ? x(reltime(d.finish_time))-x(reltime(d.start_time)) : "1px";
}
function halfwidth(d) {
if (d.finish_time)
return (x(reltime(d.finish_time))-x(reltime(d.start_time)))/2;
return "1px";
}
function middle(d) {
if (d.finish_time)
return (x(reltime(d.start_time))+x(reltime(d.finish_time)))/2;
else
return x(reltime(d.start_time)) + 1;
}
function color(d) { return data.server_info[d.serverid].color; }
function servername(d) { return data.server_info[d.serverid].short; }
function timeformat(duration) {
// TODO: trim to microseconds, maybe humanize
return duration;
}
function oldredraw() {
// at this point zoom/pan must be fixed
var min = data.bounds.min + x.domain()[0];
var max = data.bounds.min + x.domain()[1];
function inside(d) {
var finish_time = d.finish_time || d.start_time;
if (Math.max(d.start_time, min) <= Math.min(finish_time, max))
return true;
return false;
}
// from the data, build a list of bars, dots, and lines
var clipped = {bars: [], dots: [], lines: []};
data.items.filter(inside).forEach(function(d) {
if (!d.finish_time) {
clipped.dots.push(d);
} else {
clipped.bars.push(d);
if (!!d.server_sent) {
clipped.lines.push(d);
}
}
});
globals.clipped = clipped;
//chart.select(".dyhb-label")
// .attr("x", x(0))//"20px")
// .attr("y", y);
// Panning and zooming will re-run this function multiple times, and
// bars will come and go, so we must process all three selections
// (including enter() and exit()).
// TODO: add dots for events that have only start, not finish. Add
// the server-sent bar (a vertical line, half height, centered
// vertically) for events that have server-sent as well as finish.
// This probably requires creating a dot for everything, but making
// it invisible if finished is non-null, likewise for the server-sent
// bar.
// each item gets an SVG group (g.bars), translated left and down
// to match the start time and side/lane of the event
var bars = chart.selectAll("g.bars")
.data(clipped.bars, function(d) { return d.start_time; })
.attr("transform", function(d) {
return "translate("+left(d)+","+y_off(d)+")"; })
;
// update the variable parts of each bar, which depends upon the
// current pan/zoom values
bars.select("rect")
.attr("width", width);
bars.select("text")
.attr("x", halfwidth);
bars.exit().remove();
var new_bars = bars.enter()
.append("svg:g")
.attr("class", "bars")
.attr("transform", function(d) {
return "translate("+left(d)+","+y_off(d)+")"; })
;
// inside the group, we have a rect with a width for the duration of
// the event, and a fixed height. The fill and stroke color depend
// upon the event, and the title has the details. We append the rects
// first, so the text is drawn on top (higher z-order)
//y += 30*(1+d3.max(data.bars, function(d){return d.row;}));
new_bars.append("svg:rect")
.attr("width", width)
.attr("height", RECT_HEIGHT)
.attr("class", function(d) {
var c = ["bar", "lane-" + d.lane];
if (d.details.waiting)
c.push("wait-" + d.details.waiting);
return c.join(" ");
})
.on("mouseover", function(d) {if (d.details_str) tip.show(d);})
.on("mouseout", tip.hide)
//.attr("title", function(d) {return d.details_str;})
;
// each group also has a text, with 'x' set to place it in the middle
// of the rect, and text contents that are drawn in the rect
new_bars.append("svg:text")
.attr("x", halfwidth)
.attr("text-anchor", "middle")
.attr("dy", "0.9em")
//.attr("fill", "black")
.text((d) => d.what)
.on("mouseover", function(d) {if (d.details_str) tip.show(d);})
.on("mouseout", tip.hide)
;
// dots: events that have a single timestamp, rather than a range.
// These get an SVG group, and a circle and some text.
var dots = chart.selectAll("g.dots")
.data(clipped.dots, (d) => d.start_time)
.attr("transform",
(d) => "translate("+left(d)+","+(y_off(d)+LANE_HEIGHT/3)+")")
;
dots.exit().remove();
var new_dots = dots.enter()
.append("svg:g")
.attr("class", "dots")
.attr("transform",
(d) => "translate("+left(d)+","+(y_off(d)+LANE_HEIGHT/3)+")")
;
new_dots.append("svg:circle")
.attr("r", "5")
.attr("class", (d) => "dot lane-"+d.lane)
.attr("fill", "#888")
.attr("stroke", "black")
.on("mouseover", function(d) {if (d.details_str) tip.show(d);})
.on("mouseout", tip.hide)
;
new_dots.append("svg:text")
.attr("x", "5px")
.attr("text-anchor", "start")
.attr("dy", "0.2em")
.text((d) => d.what)
.on("mouseover", function(d) {if (d.details_str) tip.show(d);})
.on("mouseout", tip.hide)
;
// lines: these represent the time at which the server sent a message
// which finished a bar. These get an SVG group, and a line
var linedata = clipped.lines.map(d => [
[d.server_sent, 0],
[d.server_sent, LANE_HEIGHT],
[d.finish_time, 0],
]);
function lineshape(d) {
var l = d3.svg.line()
.x(d => x(d[0]))
.y(d => y_off(d) + 12345);
}
function update_line(sel) {
sel.attr("d", lineshape)
.attr("class", d => "line lane-"+d.lane)
;
}
var lines = chart.selectAll("polyline.lines")
.data(linedata)
.attr("transform",
(d) => "translate("+left(d)+","+y_off(d)+")")
;
lines.exit().remove();
var new_lines = lines.enter()
.append("svg:g")
.attr("class", "lines")
.attr("transform",
(d) => "translate("+left_server(d)+","+(y_off(d))+")")
;
new_lines.append("svg:line")
.attr("x1", 0)
.attr("y1", -5)
.attr("x2", "0")
.attr("y2", LANE_HEIGHT)
.attr("class", (d) => "line lane-"+d.lane)
.attr("stroke", "red")
;
new_lines.append("svg:line")
.attr("x1", 0).attr("y1", -5)
.attr("x2", (d) => x(d.finish_time - d.server_sent))
.attr("y2", 0)
.attr("class", (d) => "line lane-"+d.lane)
.attr("stroke", "red")
;
// horizontal scale markers: vertical lines at rational timestamps
var rules = chart.selectAll("g.rule")
.data(x.ticks(10))
.attr("transform", tx);
rules.select("text").text(x.tickFormat(10));
var newrules = rules.enter().insert("svg:g")
.attr("class", "rule")
.attr("transform", tx)
;
newrules.append("svg:line")
.attr("class", "rule-tick")
.attr("stroke", "black");
chart.selectAll("line.rule-tick")
.attr("y1", bottom_rule_y)
.attr("y2", bottom_rule_y + 6);
newrules.append("svg:line")
.attr("class", "rule-red")
.attr("stroke", "red")
.attr("stroke-opacity", .3);
chart.selectAll("line.rule-red")
.attr("y1", 0)
.attr("y2", bottom_rule_y);
newrules.append("svg:text")
.attr("class", "rule-text")
.attr("dy", ".71em")
.attr("text-anchor", "middle")
.attr("fill", "black")
.text(x.tickFormat(10));
chart.selectAll("text.rule-text")
.attr("y", bottom_rule_y + 9);
rules.exit().remove();
chart.select(".seconds-label")
.attr("x", w/2)
.attr("y", bottom_rule_y + 35);
}
globals.x = x;
globals.redraw = redraw;
zoom.on("zoom", redraw);
d3.select("#zoom_in_button").on("click", zoomin);
d3.select("#zoom_out_button").on("click", zoomout);
d3.select("#reset_button").on("click",
function() {
x.domain([-last*0.05, last]).range([0,w]);
redraw();
});
redraw();
$.get("done", function(_) {});
});
/*
TODO
* identify the largest gaps in the timeline (biggest is probably waiting for
the recipient to start the program, followed by waiting for recipient to
type in code, followed by waiting for recipient to approve transfer, with
the time of actual transfer being anywhere among the others).
* identify groups of events that are separated by those gaps
* put a [1 2 3 4 all] set of buttons at the top of the page
* clicking on each button will zoom the display to 10% beyond the span of
events in the given group, or reset the zoom to include all events
*/