2013-09-23 16:13:38 -05:00
<!doctype html>
< html lang = "en" >
< head >
< meta charset = "UTF-8" >
2015-07-29 05:32:28 +00:00
< meta name = "referrer" content = "no-referrer" >
2013-09-23 16:13:38 -05:00
< meta http-equiv = "X-UA-Compatible" content = "IE=edge" >
< base target = "_blank" >
2015-07-17 23:42:51 +00:00
< title > grab-site dashboard< / title >
2013-09-23 16:13:38 -05:00
< / head >
< body >
< style >
2014-10-09 07:49:16 +00:00
html {
/* Always show scrollbar to prevent jumpiness when filtering */
overflow-y: scroll;
}
2013-09-23 16:13:38 -05:00
html, body {
background-color: #D0C0AE;
2015-11-30 05:16:29 +00:00
font-family: Roboto, Tahoma, Arial, sans-serif;
2013-09-23 16:13:38 -05:00
font-size: 13px;
}
2014-10-09 05:09:42 +00:00
#filter-box {
2014-10-09 08:14:13 +00:00
background-color: #eee;
2014-10-09 05:09:42 +00:00
border: 1px solid #999;
2014-10-09 08:14:13 +00:00
padding: 1px 3px 1px 3px;
2014-10-09 05:09:42 +00:00
font-size: 18px;
border-radius: 3px;
}
2014-10-09 05:20:11 +00:00
.button {
font-size: 18px;
}
2013-09-23 16:13:38 -05:00
.padded-page {
padding: 20px 27px 20px 27px;
}
2015-01-03 07:25:07 +00:00
@media all and (min-width: 1440px) {
.padded-page {
padding: 20px 54px 47px 54px;
}
}
2013-09-23 16:13:38 -05:00
.header {
2015-11-30 05:16:29 +00:00
font-family: Roboto, Arial, sans-serif;
2013-09-23 16:13:38 -05:00
font-weight: bold;
font-size: 18px;
margin: 0 0 20px 0;
display: flex;
align-items: flex-end;
justify-content: space-between;
flex-flow: row nowrap;
}
.job-header {
display: flex;
align-items: flex-end;
justify-content: space-between;
flex-flow: row nowrap;
2015-04-20 18:57:03 +00:00
cursor: default;
}
.stats-elements:hover {
background-color: #D8CBBC;
2013-09-23 16:13:38 -05:00
}
.job-info {
white-space: nowrap;
overflow: hidden;
}
2015-03-27 07:56:04 +00:00
.job-info-done {
color: #767676;
}
.job-info-aborted {
color: #9B00D7 !important;
}
.job-info-fatal {
color: #DD0000 !important;
}
2014-10-09 07:04:46 +00:00
.inline-stat {
2014-10-09 05:32:47 +00:00
/* Needed for 'Align!' feature */
display: inline-block;
2014-10-09 05:56:58 +00:00
/* Needed to avoid extra vertical padding */
vertical-align: bottom;
2014-10-09 07:04:46 +00:00
/* Needed to avoid collapsing of leading space */
white-space: pre;
}
.job-url {
2015-11-30 05:16:29 +00:00
font-family: Roboto, Arial, sans-serif;
2013-09-23 16:13:38 -05:00
font-size: 14px;
font-weight: bold;
text-decoration: none;
2015-03-27 07:56:04 +00:00
color: inherit;
2013-09-23 16:13:38 -05:00
}
2014-10-09 05:48:07 +00:00
.job-url-aligned {
2014-10-09 07:04:46 +00:00
width: 260px;
2014-10-09 05:48:07 +00:00
overflow: hidden;
text-overflow: ellipsis;
}
.job-note-aligned {
display: none;
}
2014-10-09 05:56:58 +00:00
.job-nick-aligned {
width: 60px;
overflow: hidden;
text-overflow: hidden;
}
2014-10-09 07:04:46 +00:00
.job-mb-aligned {
2016-05-22 10:22:11 +00:00
width: 78px;
2014-10-09 07:04:46 +00:00
overflow: hidden;
text-overflow: hidden;
text-align: right;
}
.job-responses-aligned {
width: 92px;
overflow: hidden;
text-overflow: hidden;
text-align: right;
}
2016-05-22 10:18:20 +00:00
.job-responses-per-second-aligned {
width: 27px;
overflow: hidden;
text-overflow: hidden;
text-align: right;
}
2014-10-09 07:04:46 +00:00
.job-in-queue-aligned {
width: 92px;
overflow: hidden;
text-overflow: hidden;
text-align: right;
}
.job-delay-aligned {
width: 130px;
overflow: hidden;
text-overflow: hidden;
text-align: right;
}
2014-10-09 07:34:37 +00:00
.job-ignores {
font-weight: bold;
}
.job-igoff {
font-weight: normal !important;
}
2013-09-23 16:13:38 -05:00
.job-ident {
2015-11-30 05:16:29 +00:00
font-family: Roboto, Tahoma, Arial, sans-serif;
2013-09-23 16:13:38 -05:00
margin: 0 1px 0 0;
2014-10-09 07:14:33 +00:00
padding: 0;
2013-09-23 16:13:38 -05:00
border: 0;
background-color: #D0C0AE;
2014-10-09 07:14:33 +00:00
color: #645444;
2013-09-23 16:13:38 -05:00
text-align: right;
}
.log-window {
2015-04-22 17:53:06 +00:00
transition: height 0.18s ease;
2013-09-23 16:13:38 -05:00
background-color: #FFF7E1;
overflow-y: scroll;
height: 192px;
border: 1px solid #999;
margin: 0 0 1em 0;
border-radius: 3px;
}
2015-01-03 10:21:16 +00:00
.log-window-hidden {
height: 0;
2015-01-03 10:50:42 +00:00
border-width: 0 1px 0 1px;
2015-01-03 10:21:16 +00:00
margin: 0;
}
2013-09-23 16:13:38 -05:00
.log-window-stopped {
border: 1px solid #222;
box-shadow: 2px 2px 4px #888;
}
2015-01-03 09:29:10 +00:00
.log-window-expanded {
height: 384px;
}
2013-09-23 16:13:38 -05:00
.line-normal {
2015-04-20 19:24:51 +00:00
display: block;
2013-09-23 16:13:38 -05:00
white-space: pre;
width: 100%;
padding: 0 0 0 5px;
box-sizing: border-box;
}
.line-error {
2015-04-20 19:24:51 +00:00
display: block;
2013-09-23 16:13:38 -05:00
white-space: pre;
width: 100%;
background-color: #FFB9B9;
padding: 0 0 0 5px;
box-sizing: border-box;
}
.line-warning {
2015-04-20 19:24:51 +00:00
display: block;
2013-09-23 16:13:38 -05:00
white-space: pre;
width: 100%;
background-color: #F7DB7D;
padding: 0 0 0 5px;
box-sizing: border-box;
}
.line-redirect {
2015-04-20 19:24:51 +00:00
display: block;
2013-09-23 16:13:38 -05:00
white-space: pre;
width: 100%;
background-color: #E7CEEA;
padding: 0 0 0 5px;
box-sizing: border-box;
}
.line-ignore {
white-space: pre;
width: 100%;
color: #999;
padding: 0 0 0 5px;
box-sizing: border-box;
}
.line-stdout {
white-space: pre;
width: 100%;
background-color: #DCD8CB;
padding: 0 0 0 5px;
box-sizing: border-box;
}
a {
color: #000;
text-decoration: none;
}
a.ignore {
color: #999 !important;
}
.underlined-a {
text-decoration: underline;
}
.bold {
font-weight: bold;
}
#help {
background-color: #FFF7E1;
2015-11-30 05:16:29 +00:00
font-family: Roboto, Arial, sans-serif;
2013-09-23 16:13:38 -05:00
font-size: 14px;
border-radius: 5px;
padding: 0.01em 1em 0.01em 1em;
margin-bottom: 1em;
}
#help p {
padding: 0.20em 0 0.20em 0;
}
#help p a {
text-decoration: underline;
}
.undisplayed {
display: none;
}
2015-04-20 15:01:45 +00:00
#context-menu {
2015-04-20 17:15:19 +00:00
padding: 2px 0 2px 0;
2015-04-20 15:01:45 +00:00
background-color: white;
2015-04-20 17:15:19 +00:00
border: 1px solid #BABABA;
box-shadow: 2px 2px 3px #8E8E8E;
2015-04-20 15:01:45 +00:00
position: fixed;
left: 0;
top: 0;
display: none;
2015-04-25 11:42:21 +00:00
cursor: default;
2015-04-20 15:01:45 +00:00
}
2015-04-20 16:32:43 +00:00
.context-menu-entry {
2015-04-20 17:46:10 +00:00
display: block;
2015-04-25 12:13:47 +00:00
white-space: nowrap;
overflow: hidden;
max-width: 960px;
2015-04-20 16:32:43 +00:00
height: 26px;
line-height: 26px;
padding-left: 26px;
2015-04-20 17:15:19 +00:00
padding-right: 26px;
2015-11-30 05:16:29 +00:00
font-family: Roboto, 'Segoe UI', 'Helvetica Neue', sans-serif;
2015-04-20 16:32:43 +00:00
font-size: 12px;
2015-04-20 17:49:48 +00:00
cursor: default;
2015-04-25 17:45:10 +00:00
-webkit-touch-callout: none;
-webkit-user-select: none;
-khtml-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
2015-04-20 16:32:43 +00:00
}
.context-menu-entry:hover {
background-color: #4281F4;
color: #fff;
}
2015-04-20 17:15:19 +00:00
#clipboard-scratchpad {
height: 1px;
width: 1px;
padding: 0;
border: 0;
position: absolute;
top: 0;
}
2013-09-23 16:13:38 -05:00
< / style >
2015-04-20 15:01:45 +00:00
< div id = "context-menu" > < / div >
2013-09-23 16:13:38 -05:00
< div class = "padded-page" >
< div class = "header" >
< div >
2015-07-18 04:17:08 +00:00
grab-site server tracking < span id = "num-crawls" > 0 crawls< / span > .
2015-04-07 02:33:33 +00:00
Show: < input id = "filter-box" type = "text" size = "21" >
2014-10-09 09:01:13 +00:00
< input onclick = "ds.setFilter('');" type = "button" value = " All " class = "button" >
2015-01-01 06:33:09 +00:00
< input onclick = "ds.setFilter('^$');" type = "button" value = "None" class = "button" >
2013-09-23 16:13:38 -05:00
< / div >
< div >
2014-10-09 05:32:47 +00:00
< input type = "button" onclick = "ds.toggleAlign()" class = "button" value = "Align!" >
2014-10-09 05:20:11 +00:00
< input type = "button" onclick = "ds.toggleHelp()" class = "button" value = "Help!" >
2013-09-23 16:13:38 -05:00
< / div >
< / div >
< div id = "critical-info" >
< noscript >
2015-07-17 23:42:51 +00:00
Need JavaScript (ES5+) and WebSocket
2013-09-23 16:13:38 -05:00
< / noscript >
< div id = "help" class = "undisplayed" >
< p >
2015-07-17 23:42:51 +00:00
This page shows all of the crawls that are being reported to this server by your grab-site processes.
2013-09-23 16:13:38 -05:00
< / p >
< p >
To pause scrolling, move your mouse inside a log window.
< / p >
2014-10-09 09:01:13 +00:00
< p >
2015-03-29 02:56:30 +00:00
To show just one job, click anywhere on its stats line.
2014-10-09 09:01:13 +00:00
< / p >
2015-05-09 01:12:55 +00:00
< p >
To clear all finished jobs, reload the page.
< / p >
2015-03-28 02:28:00 +00:00
< p >
Keyboard shortcuts:
< / p >
< ul >
< li > < kbd > j< / kbd > - show next job window
< li > < kbd > k< / kbd > - show previous job window
< li > < kbd > a< / kbd > - show all job windows
< li > < kbd > n< / kbd > - hide all job windows
< li > < kbd > f< / kbd > - move focus to filter box
2015-04-20 19:14:50 +00:00
< li > < kbd > v< / kbd > - open the job URL of the first-shown job window
2015-04-09 20:56:36 +00:00
< li > < kbd > ?< / kbd > - show/hide help text
2015-03-28 02:28:00 +00:00
< / ul >
2015-03-27 07:56:04 +00:00
< p >
2015-05-09 01:12:55 +00:00
The color coding for the job stats line is:
2015-03-27 07:56:04 +00:00
in progress,
< span class = "job-info-done" > finished normally< / span > ,
< span class = "job-info-aborted" > finished with abort< / span > ,
< span class = "job-info-fatal" > finished with fatal exception< / span > .
< / p >
2013-09-23 16:13:38 -05:00
< p >
Mouse over the job start date or the response count for additional information.
< / p >
< p >
If your adblocker is enabled for this domain, you will see slower performance, and some URLs will not be displayed.
< / p >
2015-04-20 21:03:01 +00:00
< p >
2015-07-06 05:22:34 +00:00
In Chrome 42+, Firefox 41+, and IE10+, a custom context menu is enabled when right-clicking URLs in the log windows below. It can be disabled by adding < kbd > < span class = "url-q-or-amp" > ?< / span > contextMenu=0< / kbd > to the dashboard URL. Firefox users: if you see both the normal and custom context menu, make sure < code > dom.event.contextmenu.enabled< / code > is set to < code > true< / code > in < code > about:config< / code > .
2015-04-20 21:03:01 +00:00
< / p >
2015-05-09 01:00:32 +00:00
< p >
2015-05-09 01:12:55 +00:00
For easier text selection in the log windows, add < kbd > < span class = "url-q-or-amp" > ?< / span > moreDom=1< / kbd > to the dashboard URL. (This uses ~25% more memory.)
2015-05-09 01:00:32 +00:00
< / p >
2013-09-23 16:13:38 -05:00
< p >
2016-05-27 13:53:35 +00:00
< a href = "https://ludios.org/grab-site/" > grab-site source code< / a > .
2015-07-06 05:28:55 +00:00
< / p >
2013-09-23 16:13:38 -05:00
< / div >
< / div >
< div id = "traffic" > < / div >
< div id = "logs" > < / div >
< / div >
< script >
"use strict";
var assert = function(condition, message) {
if(!condition) {
throw message || "Assertion failed";
}
};
var byId = function(id) {
return document.getElementById(id);
};
var text = function(s) {
return document.createTextNode(s);
};
2014-09-10 03:20:34 -05:00
/**
* Adaptation of ActiveSupport's #blank?.
*
* Returns true if the object is undefined, null, or is a string whose
* post-trim length is zero. Otherwise, returns false.
*/
var isBlank = function(o) {
2014-09-10 11:14:10 -05:00
return !o || o.trim().length === 0;
2014-09-10 03:20:34 -05:00
}
2013-09-23 16:13:38 -05:00
/**
* appendChild but accepts strings and arrays of children|strings
*/
var appendAny = function(e, thing) {
if(Array.isArray(thing)) {
for(var i=0; i < thing.length ; i + + ) {
appendAny(e, thing[i]);
}
} else if(typeof thing == "string") {
e.appendChild(text(thing));
} else {
2015-03-27 07:56:04 +00:00
if(thing == null) {
throw Error("thing is " + JSON.stringify(thing));
}
2013-09-23 16:13:38 -05:00
e.appendChild(thing);
}
};
/**
* Create DOM element with attributes and children from Array< node | string > |node|string
*/
var h = function(elem, attrs, thing) {
var e = document.createElement(elem);
if(attrs != null) {
for(var attr in attrs) {
if(attr == "spellcheck" || attr == "readonly") {
e.setAttribute(attr, attrs[attr]);
2015-04-20 16:32:43 +00:00
} else if(attr == "class") {
throw new Error("Did you mean className?");
2013-09-23 16:13:38 -05:00
} else {
e[attr] = attrs[attr];
}
}
}
if(thing != null) {
appendAny(e, thing);
}
return e;
};
var href = function(href, text) {
var a = h("a");
a.href = href;
a.textContent = text;
return a;
};
2015-04-20 16:32:43 +00:00
var removeChildren = function(elem) {
while(elem.firstChild) {
elem.removeChild(elem.firstChild);
}
};
2013-09-23 16:13:38 -05:00
var prettyJson = function(obj) {
return JSON.stringify(obj, undefined, 2);
};
// Copied from Coreweb/js_coreweb/cw/string.js
/**
* Like Python's s.split(delim, num) and s.split(delim)
* This does *NOT* implement Python's no-argument s.split()
*
* @param {string} s The string to split.
* @param {string} sep The separator to split by.
* @param {number} maxsplit Maximum number of times to split.
*
* @return {!Array.< string > } The splitted string, as an array.
*/
var split = function(s, sep, maxsplit) {
assert(typeof sep == "string",
"arguments[1] of split must be a separator string");
if(maxsplit === undefined || maxsplit < 0 ) {
return s.split(sep);
}
var pieces = s.split(sep);
var head = pieces.splice(0, maxsplit);
// after the splice, pieces is shorter and no longer has the `head` elements.
if(pieces.length > 0) {
var tail = pieces.join(sep);
head.push(tail); // no longer just the head.
}
return head;
};
2015-01-01 06:33:09 +00:00
// Copied from closure-library's goog.string.startsWith
var startsWith = function(str, prefix) {
return str.lastIndexOf(prefix, 0) == 0;
}
// Copied from closure-library's goog.string.endsWith
var endsWith = function(str, suffix) {
var l = str.length - suffix.length;
return l >= 0 & & str.indexOf(suffix, l) == l;
};
2015-04-25 11:54:02 +00:00
// Based on closure-library's goog.string.regExpEscape
2015-01-01 06:33:09 +00:00
var regExpEscape = function(s) {
2015-04-25 11:54:02 +00:00
var escaped = String(s).replace(/([-()\[\]{}+?*.$\^|,:#< !\\])/g, '\\$1').
2015-01-01 06:33:09 +00:00
replace(/\x08/g, '\\x08');
2015-04-25 11:54:02 +00:00
if(s.indexOf('[') == -1 & & s.indexOf(']') == -1) {
// If there were no character classes, there can't have been any need
// to escape -, to unescape them.
escaped = escaped.replace(/\\-/g, "-");
}
return escaped;
2015-01-01 06:33:09 +00:00
};
2013-09-23 16:13:38 -05:00
/**
* [[1, 2], [3, 4]] -> {1: 2, 3: 4}
*/
var intoObject = function(arr) {
var obj = {};
arr.forEach(function(e) {
obj[e[0]] = e[1];
});
return obj;
};
var getQueryArgs = function() {
var pairs = location.search.replace("?", "").split("&");
if(pairs == "") {
return {};
}
return intoObject(pairs.map(function(e) { return split(e, "=", 1); }));
};
2015-04-20 15:41:03 +00:00
var getChromeMajorVersion = function() {
return Number(navigator.userAgent.match(/Chrome\/(\d+)/)[1]);
};
2015-07-06 05:22:34 +00:00
var getFirefoxMajorVersion = function() {
return Number(navigator.userAgent.match(/Firefox\/(\d+)/)[1]);
};
2015-05-20 00:23:55 +00:00
var getTridentMajorVersion = function() {
return Number(navigator.userAgent.match(/Trident\/(\d+)/)[1]);
};
2015-04-20 15:28:38 +00:00
var isChrome = navigator.userAgent.indexOf("Chrome/") != -1;
var isSafari = !isChrome & & navigator.userAgent.indexOf("Safari") != -1;
2015-03-27 13:45:03 +00:00
var isFirefox = navigator.userAgent.indexOf("Firefox") != -1;
2015-05-20 00:23:55 +00:00
var isTrident = navigator.userAgent.indexOf("Trident/") != -1;
2013-09-23 16:13:38 -05:00
2014-10-09 05:09:42 +00:00
var addAnyChangeListener = function(elem, func) {
2014-10-09 07:49:16 +00:00
// DOM0 handler for convenient use by Clear button
elem.onchange = func;
2014-10-09 05:09:42 +00:00
elem.addEventListener('keydown', func, false);
elem.addEventListener('paste', func, false);
elem.addEventListener('input', func, false);
};
2014-10-09 05:48:07 +00:00
var arrayFrom = function(arrayLike) {
return Array.prototype.slice.call(arrayLike);
};
2014-10-09 05:09:42 +00:00
/**
* Returns a function that gets the given property on any object passed in
*/
var prop = function(name) {
return function(obj) {
return obj[name];
};
};
2014-10-09 05:48:07 +00:00
/**
* Returns a function that adds the given class to any element passed in
*/
var classAdder = function(name) {
return function(elem) {
elem.classList.add(name);
};
};
/**
* Returns a function that removes the given class to any element passed in
*/
var classRemover = function(name) {
return function(elem) {
elem.classList.remove(name);
};
};
2015-03-27 11:44:33 +00:00
var removeFromArray = function(arr, item) {
var idx = arr.indexOf(item);
if(idx != -1) {
arr.splice(idx, 1);
}
};
2015-03-27 12:11:25 +00:00
// Based on http://stackoverflow.com/a/18520276
var findInArray = function(arr, test, ctx) {
var result = null;
arr.some(function(el, i) {
return test.call(ctx, el, i, arr) ? ((result = i), true) : false;
});
return result;
};
2013-09-23 16:13:38 -05:00
/*** End of utility code ***/
var JobsTracker = function() {
this.known = {};
this.sorted = [];
this.finishedArray = [];
this.finishedSet = {};
2015-03-27 12:48:42 +00:00
this.fatalExceptionSet = {};
2013-09-23 16:13:38 -05:00
};
JobsTracker.prototype.countActive = function() {
return this.sorted.length - this.finishedArray.length;
};
JobsTracker.prototype.resort = function() {
this.sorted.sort(function(a, b) { return a["started_at"] > b["started_at"] ? -1 : 1 });
};
/**
* Returns true if a new job was added
*/
JobsTracker.prototype.handleJobData = function(jobData) {
var ident = jobData["ident"];
var alreadyKnown = ident in this.known;
if(!alreadyKnown) {
this.known[ident] = true;
this.sorted.push(jobData);
this.resort();
}
2015-03-27 11:44:33 +00:00
return !alreadyKnown;
};
2015-03-27 12:48:42 +00:00
JobsTracker.prototype.markFinished = function(ident) {
2015-03-27 11:44:33 +00:00
if(!(ident in this.finishedSet)) {
2013-09-23 16:13:38 -05:00
this.finishedSet[ident] = true;
this.finishedArray.push(ident);
}
2015-03-27 11:44:33 +00:00
};
2015-03-27 12:48:42 +00:00
JobsTracker.prototype.markUnfinished = function(ident) {
2015-03-27 11:44:33 +00:00
if(ident in this.finishedSet) {
delete this.finishedSet[ident];
removeFromArray(this.finishedArray, ident);
}
2015-03-27 12:48:42 +00:00
// Job was restarted, so unmark fatal exception
if(ident in this.fatalExceptionSet) {
delete this.fatalExceptionSet[ident];
}
};
JobsTracker.prototype.markFatalException = function(ident) {
this.fatalExceptionSet[ident] = true;
};
JobsTracker.prototype.hasFatalException = function(ident) {
return ident in this.fatalExceptionSet;
2013-09-23 16:13:38 -05:00
};
2014-09-10 02:26:28 -05:00
var JobRenderInfo = function(logWindow, logSegment, statsElements, jobNote, lineCountWindow, lineCountSegments) {
2013-09-23 16:13:38 -05:00
this.logWindow = logWindow;
this.logSegment = logSegment;
this.statsElements = statsElements;
2014-09-10 02:26:28 -05:00
this.jobNote = jobNote;
2013-09-23 16:13:38 -05:00
this.lineCountWindow = lineCountWindow;
this.lineCountSegments = lineCountSegments;
};
var Reusable = {
2015-04-25 22:04:03 +00:00
obj_className_line_normal: {"className": "line-normal"},
obj_className_line_error: {"className": "line-error"},
obj_className_line_warning: {"className": "line-warning"},
obj_className_line_redirect: {"className": "line-redirect"},
//
2013-09-23 16:13:38 -05:00
obj_className_line_ignore: {"className": "line-ignore"},
obj_className_line_stdout: {"className": "line-stdout"},
obj_className_bold: {"className": "bold"}
};
// http://stackoverflow.com/questions/2901102/how-to-print-a-number-with-commas-as-thousands-separators-in-javascript
2014-09-10 23:27:27 +00:00
var numberWithCommas = function(s_or_n) {
return ("" + s_or_n).replace(/\B(?=(\d{3})+(?!\d))/g, ",");
2013-09-23 16:13:38 -05:00
};
var toStringTenths = function(n) {
var s = "" + (Math.round(10 * n) / 10);
if(s.indexOf(".") == -1) {
s += ".0";
}
return s;
};
var getTotalResponses = function(jobData) {
return (
parseInt(jobData["r1xx"]) +
parseInt(jobData["r2xx"]) +
parseInt(jobData["r3xx"]) +
parseInt(jobData["r4xx"]) +
parseInt(jobData["r5xx"]) +
parseInt(jobData["runk"]));
};
var getSummaryResponses = function(jobData) {
return (
2014-09-10 23:27:27 +00:00
"1xx: " + numberWithCommas(jobData["r1xx"]) + "\n" +
"2xx: " + numberWithCommas(jobData["r2xx"]) + "\n" +
"3xx: " + numberWithCommas(jobData["r3xx"]) + "\n" +
"4xx: " + numberWithCommas(jobData["r4xx"]) + "\n" +
"5xx: " + numberWithCommas(jobData["r5xx"]) + "\n" +
"Unknown: " + numberWithCommas(jobData["runk"]));
2013-09-23 16:13:38 -05:00
};
2015-04-20 20:30:39 +00:00
var JobsRenderer = function(container, filterBox, historyLines, showNicks, contextMenuRenderer) {
2013-09-23 16:13:38 -05:00
this.container = container;
2014-10-09 05:09:42 +00:00
this.filterBox = filterBox;
2014-10-09 08:23:30 +00:00
addAnyChangeListener(this.filterBox, this.applyFilter.bind(this));
2015-03-27 11:28:33 +00:00
this.filterBox.onkeypress = function(ev) {
// So that j or k in input box does not result in job window switching
ev.stopPropagation();
}
2013-09-23 16:13:38 -05:00
this.historyLines = historyLines;
this.showNicks = showNicks;
2015-04-20 20:30:39 +00:00
this.contextMenuRenderer = contextMenuRenderer;
2013-09-23 16:13:38 -05:00
this.linesPerSegment = Math.max(1, Math.round(this.historyLines / 10));
this.jobs = new JobsTracker();
// ident -> JobRenderInfo
this.renderInfo = {};
this.mouseInside = null;
this.numCrawls = byId('num-crawls');
2014-10-09 06:05:49 +00:00
this.aligned = false;
2013-09-23 16:13:38 -05:00
};
JobsRenderer.prototype._getNextJobInSorted = function(ident) {
for(var i=0; i < this.jobs.sorted.length ; i + + ) {
var e = this.jobs.sorted[i];
if(e["ident"] == ident) {
return this.jobs.sorted[i+1];
}
}
return null;
};
JobsRenderer.prototype._createLogSegment = function() {
return h('div');
};
JobsRenderer.prototype._createLogContainer = function(jobData) {
var ident = jobData["ident"];
var beforeJob = this._getNextJobInSorted(ident);
var beforeElement = beforeJob == null ? null : byId("log-container-" + beforeJob["ident"]);
var logSegment = this._createLogSegment();
var logWindowAttrs = {
"className": "log-window",
"id": "log-window-" + ident,
"onmouseenter": function(ev) {
this.mouseInside = ident;
ev.target.classList.add('log-window-stopped');
}.bind(this),
"onmouseleave": function(ev) {
2015-04-20 20:30:39 +00:00
var leave = function() {
this.mouseInside = null;
ev.target.classList.remove('log-window-stopped');
}.bind(this);
// When our custom context menu pops up, it causes onmouseleave on the
// log window, so make our leave callback fire only after the context
// menu is closed.
if(this.contextMenuRenderer.visible) {
this.contextMenuRenderer.callAfterBlur(leave);
} else {
leave();
}
2013-09-23 16:13:38 -05:00
}.bind(this)
}
// If you reach the end of a log window, the browser annoyingly
// starts to scroll the page instead. We prevent this behavior here.
// If the user wants to scroll the page, they need to move their
// mouse outside a log window first.
if(!isSafari) {
logWindowAttrs["onwheel"] = function(ev) {
// Note: offsetHeight is "wrong" by 2px but it doesn't matter
//console.log(ev, logWindow.scrollTop, (logWindow.scrollHeight - logWindow.offsetHeight));
if(ev.deltaY < 0 & & logWindow . scrollTop = = 0 ) {
ev.preventDefault();
} else if(ev.deltaY > 0 & & logWindow.scrollTop >= (logWindow.scrollHeight - logWindow.offsetHeight)) {
ev.preventDefault();
}
}
} else {
// Safari 7.0.5 can't preventDefault or stopPropagation an onwheel event,
// so use onmousewheel instead.
logWindowAttrs["onmousewheel"] = function(ev) {
//console.log(ev, logWindow.scrollTop, (logWindow.scrollHeight - logWindow.offsetHeight));
if(ev.wheelDeltaY > 0 & & logWindow.scrollTop == 0) {
ev.preventDefault();
} else if(ev.wheelDeltaY < 0 & & logWindow . scrollTop > = (logWindow.scrollHeight - logWindow.offsetHeight)) {
ev.preventDefault();
}
}
}
var statsElements = {
2014-10-09 08:19:09 +00:00
mb: h("span", {"className": "inline-stat job-mb"}, "?"),
2014-10-09 07:04:46 +00:00
responses: h("span", {"className": "inline-stat job-responses"}, "?"),
2016-05-22 10:18:20 +00:00
responsesPerSecond: h("span", {"className": "inline-stat job-responses-per-second"}, "?"),
2014-10-09 07:04:46 +00:00
queueLength: h("span", {"className": "inline-stat job-in-queue"}, "? in q."),
2013-09-23 16:13:38 -05:00
connections: h("span", null, "?"),
2014-10-09 07:04:46 +00:00
delay: h("span", {"className": "inline-stat job-delay"}, "? ms delay"),
2015-03-27 07:56:04 +00:00
ignores: h("span", {"className": "job-ignores"}, "?"),
jobInfo: null /* set later */
2013-09-23 16:13:38 -05:00
};
2015-03-27 07:56:04 +00:00
var startedISOString = new Date(parseFloat(jobData["started_at"]) * 1000).toISOString();
2014-10-09 05:48:07 +00:00
var jobNote = h("span", {"className": "job-note"}, null);
2014-09-10 02:26:28 -05:00
2015-03-27 07:56:04 +00:00
statsElements.jobInfo = h(
"span", {"className": "job-info"}, [
h("a", {"className": "inline-stat job-url", "href": jobData["url"]}, jobData["url"]),
2015-03-27 08:16:21 +00:00
// Clicking anywhere in this area will set the filter to a regexp that
// matches only this job URL, thus hiding everything but this job.
2015-03-27 07:56:04 +00:00
h("span", {
2015-04-20 18:57:03 +00:00
"className": "stats-elements",
2015-03-27 07:56:04 +00:00
"onclick": function() {
var filter = ds.getFilter();
if(RegExp(filter).test(jobData["url"]) & & startsWith(filter, "^") & & endsWith(filter, "$")) {
// If we're already showing just this log window, go back
// to showing nothing.
ds.setFilter("^$");
} else {
ds.setFilter("^" + regExpEscape(jobData["url"]) + "$");
}
}
}, [
" on ",
h("span", {"title": startedISOString}, startedISOString.split("T")[0].substr(5)),
h("span", {"className": "inline-stat job-nick"}, (this.showNicks ? " by " + jobData["started_by"] : "")),
jobNote,
"; ",
statsElements.mb,
" MB in ",
statsElements.responses,
" at ",
statsElements.responsesPerSecond,
"/s, ",
statsElements.queueLength,
"; ",
statsElements.connections,
" con. w/ ",
statsElements.delay,
"; ",
statsElements.ignores
])
]
);
2013-09-23 16:13:38 -05:00
var logWindow = h('div', logWindowAttrs, logSegment);
var div = h(
'div',
{"id": "log-container-" + ident}, [
h("div", {"className": "job-header"}, [
2015-03-27 07:56:04 +00:00
statsElements.jobInfo,
2013-09-23 16:13:38 -05:00
h("input", {
"className": "job-ident",
"type": "text",
"value": ident,
"size": "28",
"spellcheck": "false",
"readonly": "",
"onclick": function() { this.select(); }
})
]),
logWindow
]
);
2014-09-10 02:26:28 -05:00
this.renderInfo[ident] = new JobRenderInfo(logWindow, logSegment, statsElements, jobNote, 0, [0]);
2013-09-23 16:13:38 -05:00
this.container.insertBefore(div, beforeElement);
2014-10-09 07:56:52 +00:00
// Set appropriate CSS classes - we might be in aligned mode already
this.updateAlign();
2015-01-03 09:29:10 +00:00
// Filter hasn't changed, but we might need to filter out the new job, or
// add/remove log-window-expanded class
2014-10-09 08:23:30 +00:00
this.applyFilter();
2013-09-23 16:13:38 -05:00
}
JobsRenderer.prototype._renderDownloadLine = function(data, logSegment) {
2015-07-18 09:49:03 +00:00
var code = data["response_code"];
if(code >= 400 & & code < 500 ) {
2015-04-20 19:24:51 +00:00
var attrs = {"className": "line-warning", "href": data["url"]};
2015-07-18 09:49:03 +00:00
} else if(code === 0 || code >= 500) {
2015-04-20 19:24:51 +00:00
var attrs = {"className": "line-error", "href": data["url"]};
2015-07-18 09:49:03 +00:00
} else if(code & & code >= 300 & & code < 400 ) {
2015-04-20 19:24:51 +00:00
var attrs = {"className": "line-redirect", "href": data["url"]};
2013-09-23 16:13:38 -05:00
} else {
2015-04-20 19:24:51 +00:00
var attrs = {"className": "line-normal", "href": data["url"]};
2013-09-23 16:13:38 -05:00
}
2015-04-20 19:24:51 +00:00
logSegment.appendChild(
2015-07-18 09:49:03 +00:00
h("a", attrs, code + " " + data["wget_code"] + " " + data["url"])
2015-04-20 19:24:51 +00:00
);
2013-09-23 16:13:38 -05:00
return 1;
};
2015-04-25 22:04:03 +00:00
/**
* Like _renderDownloadLine, but makes it easier to start a text selection from the
* left or right of the URL.
*/
JobsRenderer.prototype._moreDomRenderDownloadLine = function(data, logSegment) {
2015-07-18 09:49:03 +00:00
var code = data["response_code"];
if(code >= 400 & & code < 500 ) {
2015-04-25 22:04:03 +00:00
var attrs = Reusable.obj_className_line_warning;
2015-07-18 09:49:03 +00:00
} else if(code === 0 || code >= 500) {
2015-04-25 22:04:03 +00:00
var attrs = Reusable.obj_className_line_error;
2015-07-18 09:49:03 +00:00
} else if(code & & code >= 300 & & code < 400 ) {
2015-04-25 22:04:03 +00:00
var attrs = Reusable.obj_className_line_redirect;
} else {
var attrs = Reusable.obj_className_line_normal;
}
logSegment.appendChild(h("div", attrs, [
2015-07-18 09:49:03 +00:00
code + " " + data["wget_code"] + " ",
2015-04-25 22:11:37 +00:00
h("a", {"href": data["url"], "className": "log-url"}, data["url"])
2015-04-25 22:04:03 +00:00
]));
return 1;
};
2013-09-23 16:13:38 -05:00
JobsRenderer.prototype._renderIgnoreLine = function(data, logSegment) {
var attrs = Reusable.obj_className_line_ignore;
logSegment.appendChild(h("div", attrs, [
h('span', null, " IGNOR "),
h('a', {"href": data["url"], "className": "ignore"}, data["url"]),
h('span', Reusable.obj_className_bold, " by "),
data["pattern"]
]));
return 1;
};
2015-03-27 11:44:33 +00:00
JobsRenderer.prototype._renderStdoutLine = function(data, logSegment, info, ident) {
2013-09-23 16:13:38 -05:00
var cleanedMessage = data["message"].replace(/[\r\n]+$/, "");
2015-07-28 12:33:59 +00:00
// Format DUPE/OF messages a little more nicely
cleanedMessage = cleanedMessage.replace(/^DUPE /, " DUPE ").replace(/\n OF /, "\n OF ");
2013-09-23 16:13:38 -05:00
var renderedLines = 0;
if(!cleanedMessage) {
return renderedLines;
}
var lines = cleanedMessage.split("\n");
for(var i=0; i < lines.length ; i + + ) {
var line = lines[i];
if(!line) {
continue;
}
logSegment.appendChild(h("div", Reusable.obj_className_line_stdout, line));
renderedLines += 1;
2015-03-27 07:56:04 +00:00
2015-07-19 22:11:01 +00:00
if(/^Finished grab \S+ \S+ with exit code ([0-13-8])$/.test(line)) {
2015-03-27 07:56:04 +00:00
info.statsElements.jobInfo.classList.add('job-info-done');
2015-03-27 11:44:33 +00:00
this.jobs.markFinished(ident);
2015-07-19 20:42:29 +00:00
} else if(/^Finished grab \S+ \S+ with exit code |^CRITICAL (Sorry|Please report)|^ERROR Fatal exception|No space left on device|^Fatal Python error:|^(Thread|Current thread) 0x/.test(line)) {
2015-03-27 07:56:04 +00:00
info.statsElements.jobInfo.classList.add('job-info-fatal');
2015-03-27 12:48:42 +00:00
this.jobs.markFatalException(ident);
2015-07-19 20:42:29 +00:00
} else if(/Script requested immediate stop/.test(line)) {
2015-03-27 08:04:24 +00:00
// Note: above message can be in:
// ERROR Script requested immediate stop
// or after an ERROR Fatal exception:
// wpull.hook.HookStop: Script requested immediate stop.
info.statsElements.jobInfo.classList.remove('job-info-fatal');
2015-03-27 07:56:04 +00:00
info.statsElements.jobInfo.classList.add('job-info-aborted');
}
2013-09-23 16:13:38 -05:00
}
return renderedLines;
};
JobsRenderer.prototype.handleData = function(data) {
var jobData = data["job_data"];
var added = this.jobs.handleJobData(jobData);
2015-07-18 04:17:08 +00:00
var jobsActive = this.jobs.countActive();
this.numCrawls.textContent =
jobsActive === 1 ?
"1 crawl" :
jobsActive + " crawls";
2013-09-23 16:13:38 -05:00
if(added) {
this._createLogContainer(jobData);
}
var type = data["type"];
var ident = jobData["ident"];
var info = this.renderInfo[ident];
if(!info) {
console.warn("No render info for " + ident);
return;
}
var totalResponses = parseInt(getTotalResponses(jobData));
if(type == "download") {
var linesRendered = this._renderDownloadLine(data, info.logSegment);
2015-07-18 05:24:54 +00:00
} else if(type == "stdout" || type == "stderr") {
2015-03-27 11:44:33 +00:00
var linesRendered = this._renderStdoutLine(data, info.logSegment, info, ident);
2013-09-23 16:13:38 -05:00
} else if(type == "ignore") {
var linesRendered = this._renderIgnoreLine(data, info.logSegment);
} else {
assert(false, "Unexpected message type " + type);
}
// Update stats
info.statsElements.mb.textContent =
numberWithCommas(
toStringTenths(
(parseInt(jobData["bytes_downloaded"]) / (1024 * 1024)).toString()));
info.statsElements.responses.textContent =
2014-09-10 23:27:27 +00:00
numberWithCommas(totalResponses) + " resp.";
2013-09-23 16:13:38 -05:00
info.statsElements.responses.title = getSummaryResponses(jobData);
var duration = Date.now()/1000 - parseFloat(jobData["started_at"]);
info.statsElements.responsesPerSecond.textContent =
toStringTenths(totalResponses/duration);
2014-09-10 00:06:17 -05:00
if (jobData["items_queued"] & & jobData["items_downloaded"]) {
var totalQueued = parseInt(jobData["items_queued"], 10);
var totalDownloaded = parseInt(jobData["items_downloaded"], 10);
info.statsElements.queueLength.textContent =
2014-09-10 23:27:27 +00:00
numberWithCommas((totalQueued - totalDownloaded) + " in q.");
2014-09-10 00:06:17 -05:00
info.statsElements.queueLength.title =
2014-09-10 23:27:27 +00:00
numberWithCommas(totalQueued) + " queued\n" +
numberWithCommas(totalDownloaded) + " downloaded";
2014-09-10 00:06:17 -05:00
}
2013-09-23 16:13:38 -05:00
info.statsElements.connections.textContent = jobData["concurrency"];
2014-10-09 08:21:33 +00:00
2013-09-23 16:13:38 -05:00
var delayMin = parseInt(jobData["delay_min"]);
var delayMax = parseInt(jobData["delay_max"]);
info.statsElements.delay.textContent =
2014-10-09 07:04:46 +00:00
(delayMin == delayMax ?
2013-09-23 16:13:38 -05:00
delayMin :
2014-10-09 07:04:46 +00:00
delayMin + "-" + delayMax) + " ms delay";
2014-10-09 08:21:33 +00:00
2014-10-09 07:34:37 +00:00
if(jobData["suppress_ignore_reports"]) {
2014-10-09 08:21:33 +00:00
info.statsElements.ignores.textContent = 'igoff';
2014-10-09 07:34:37 +00:00
if(!info.statsElements.ignores.classList.contains('job-igoff')) {
info.statsElements.ignores.classList.add('job-igoff');
}
} else {
2014-10-09 08:21:33 +00:00
info.statsElements.ignores.textContent = 'igon';
2014-10-09 07:34:37 +00:00
if(info.statsElements.ignores.classList.contains('job-igoff')) {
info.statsElements.ignores.classList.remove('job-igoff');
}
}
2013-09-23 16:13:38 -05:00
2014-09-10 02:26:28 -05:00
// Update note
2014-09-10 23:29:23 +00:00
info.jobNote.textContent =
isBlank(jobData["note"]) ?
2014-09-10 03:20:34 -05:00
"" :
" (" + jobData["note"] + ")";
2014-09-10 02:26:28 -05:00
2013-09-23 16:13:38 -05:00
info.lineCountWindow += linesRendered;
info.lineCountSegments[info.lineCountSegments.length - 1] += linesRendered;
if(info.lineCountSegments[info.lineCountSegments.length - 1] >= this.linesPerSegment) {
//console.log("Created new segment", info);
var newSegment = this._createLogSegment();
info.logWindow.appendChild(newSegment);
info.logSegment = newSegment;
info.lineCountSegments.push(0);
}
if(this.mouseInside != ident) {
2015-03-27 12:48:42 +00:00
// Don't remove any scrollback information when the job has a fatal exception,
// so that the user can find the traceback and report a bug.
if(!this.jobs.hasFatalException(ident)) {
// We may have to remove more than one segment, if the user
// has paused the log window for a while.
while(info.lineCountWindow >= this.historyLines + this.linesPerSegment) {
var firstLogSegment = info.logWindow.firstChild;
assert(firstLogSegment != null, "info.logWindow.firstChild is null; " +
JSON.stringify({
"lineCountWindow": info.lineCountWindow,
"lineCountSegments": info.lineCountSegments}));
info.logWindow.removeChild(firstLogSegment);
info.lineCountWindow -= info.lineCountSegments[0];
info.lineCountSegments.shift();
}
2013-09-23 16:13:38 -05:00
}
// Scroll to the bottom
2015-03-27 08:50:24 +00:00
// To avoid serious performance problems in Firefox, we use a big number
// instead of info.logWindow.scrollHeight.
info.logWindow.scrollTop = 999999;
2013-09-23 16:13:38 -05:00
}
};
2014-10-09 08:23:30 +00:00
JobsRenderer.prototype.applyFilter = function() {
2014-10-09 05:09:42 +00:00
var query = this.filterBox.value;
2014-10-09 06:05:49 +00:00
var matches = 0;
2015-01-03 09:29:10 +00:00
var matchedWindows = [];
var unmatchedWindows = [];
2015-03-27 12:11:25 +00:00
this.firstFilterMatch = null;
2014-10-09 05:09:42 +00:00
for(var i=0; i < this.jobs.sorted.length ; i + + ) {
var job = this.jobs.sorted[i];
2015-01-03 09:29:10 +00:00
var w = this.renderInfo[job["ident"]].logWindow;
2015-01-01 06:33:09 +00:00
if(!RegExp(query).test(job["url"])) {
2015-01-03 10:21:16 +00:00
w.classList.add("log-window-hidden");
2015-03-27 13:45:03 +00:00
// Firefox exhibits serious performance problems when adding
// lines to our 0px-high log windows, so add display: none
// (effectively killing the animation)
if(isFirefox) {
w.style.display = "none";
}
// Remove this class, else an ugly border may be visible
2015-01-03 11:20:37 +00:00
w.classList.remove('log-window-stopped');
2015-01-03 09:29:10 +00:00
unmatchedWindows.push(w);
2014-10-09 05:09:42 +00:00
} else {
2015-01-03 10:21:16 +00:00
w.classList.remove("log-window-hidden");
2015-03-27 13:45:03 +00:00
if(isFirefox) {
w.style.display = "block";
}
2014-10-09 06:05:49 +00:00
matches += 1;
2015-01-03 09:29:10 +00:00
matchedWindows.push(w);
2015-03-27 13:45:03 +00:00
if(this.firstFilterMatch == null) {
2015-04-20 19:14:50 +00:00
this.firstFilterMatch = job;
2015-03-27 13:45:03 +00:00
}
2014-10-09 05:09:42 +00:00
}
}
2014-10-09 06:07:36 +00:00
2015-01-03 09:29:10 +00:00
// If there's only one visible log window, expand it so that more lines are visible.
unmatchedWindows.map(classRemover('log-window-expanded'));
matchedWindows.map(classRemover('log-window-expanded'));
if(matches == 1) {
matchedWindows.map(classAdder('log-window-expanded'));
}
2014-10-09 06:05:49 +00:00
if(matches < this.jobs.sorted.length ) {
2014-10-09 06:07:36 +00:00
// If you're not seeing all of the log windows, you're probably seeing very
// few of them, so you probably want alignment enabled.
2014-10-09 06:05:49 +00:00
this.aligned = true;
this.updateAlign();
} else {
2014-10-09 06:07:36 +00:00
// You're seeing all of the log windows, so alignment doesn't help as much
// as seeing the full info.
2014-10-09 06:05:49 +00:00
this.aligned = false;
this.updateAlign();
}
};
2015-03-27 12:11:25 +00:00
JobsRenderer.prototype.showNextPrev = function(offset) {
2015-04-20 19:14:50 +00:00
var idx;
if(this.firstFilterMatch == null) {
idx = null;
} else {
idx = findInArray(this.jobs.sorted, function(el, i) {
return el["ident"] == this.firstFilterMatch["ident"];
}.bind(this));
}
2015-03-27 12:11:25 +00:00
if(idx == null) {
2015-03-27 12:28:16 +00:00
// If no job windows are shown, set up index to make j show the first job window,
// k the last job window.
2015-03-27 14:52:55 +00:00
idx = this.jobs.sorted.length;
2015-03-27 12:11:25 +00:00
}
idx = idx + offset;
2015-03-27 14:52:55 +00:00
// When reaching either end, hide all job windows. When going past
// the end, wrap around.
2015-03-27 12:11:25 +00:00
if(idx == -1) {
2015-03-27 14:52:55 +00:00
idx = this.jobs.sorted.length;
} else if(idx == this.jobs.sorted.length + 1) {
2015-03-27 12:11:25 +00:00
idx = 0;
}
2015-03-27 14:52:55 +00:00
if(idx == this.jobs.sorted.length) {
ds.setFilter("^$");
} else {
var newShownJob = this.jobs.sorted[idx];
ds.setFilter("^" + regExpEscape(newShownJob["url"]) + "$");
}
2015-03-27 12:11:25 +00:00
};
2014-10-09 06:05:49 +00:00
JobsRenderer.prototype.updateAlign = function() {
var adderOrRemover = this.aligned ? classAdder : classRemover;
arrayFrom(document.querySelectorAll('.job-url')).map(adderOrRemover('job-url-aligned'));
arrayFrom(document.querySelectorAll('.job-note')).map(adderOrRemover('job-note-aligned'));
arrayFrom(document.querySelectorAll('.job-nick')).map(adderOrRemover('job-nick-aligned'));
2014-10-09 07:04:46 +00:00
arrayFrom(document.querySelectorAll('.job-mb')).map(adderOrRemover('job-mb-aligned'));
arrayFrom(document.querySelectorAll('.job-responses')).map(adderOrRemover('job-responses-aligned'));
2016-05-22 10:18:20 +00:00
arrayFrom(document.querySelectorAll('.job-responses-per-second')).map(adderOrRemover('job-responses-per-second-aligned'));
2014-10-09 07:04:46 +00:00
arrayFrom(document.querySelectorAll('.job-in-queue')).map(adderOrRemover('job-in-queue-aligned'));
arrayFrom(document.querySelectorAll('.job-delay')).map(adderOrRemover('job-delay-aligned'));
2014-10-09 06:05:49 +00:00
};
JobsRenderer.prototype.toggleAlign = function() {
this.aligned = !this.aligned;
this.updateAlign();
2014-10-09 05:09:42 +00:00
};
2013-09-23 16:13:38 -05:00
2015-04-20 15:28:55 +00:00
/**
* This context menu pops up when you right-click on a URL in
2015-07-28 12:01:20 +00:00
* a log window, helping you copy a regexp based on the URL
2015-04-20 15:28:55 +00:00
* you right-clicked.
*/
2015-04-20 15:01:45 +00:00
var ContextMenuRenderer = function() {
2015-04-20 20:30:39 +00:00
this.visible = false;
this.callAfterBlurFns = [];
2015-04-20 15:01:45 +00:00
this.element = byId('context-menu');
};
2015-04-20 15:28:55 +00:00
/**
* Returns true if the event target is a URL in a log window
*/
2015-04-20 17:47:08 +00:00
ContextMenuRenderer.prototype.clickedOnLogWindowURL = function(ev) {
2015-04-20 19:24:51 +00:00
var cn = ev.target.className;
2015-04-25 22:11:37 +00:00
return cn == "line-normal" || cn == "line-error" || cn == "line-warning" || cn == "line-redirect" || cn == "log-url";
2015-04-20 15:01:45 +00:00
};
2015-04-20 17:15:19 +00:00
ContextMenuRenderer.prototype.makeCopyTextFn = function(text) {
return function() {
var clipboardScratchpad = byId('clipboard-scratchpad');
clipboardScratchpad.value = text;
clipboardScratchpad.focus();
clipboardScratchpad.select();
document.execCommand('copy');
}.bind(this);
};
2015-04-25 11:25:02 +00:00
ContextMenuRenderer.prototype.getPathVariants = function(path) {
var paths = [path];
2015-04-25 11:28:02 +00:00
// Avoid generating a duplicate suggestion
path = path.replace(/\/$/, "");
2015-04-25 12:00:06 +00:00
while(path & & path.lastIndexOf('/') != -1) {
2015-04-25 11:25:02 +00:00
path = path.replace(/\/[^\/]*$/, "");
paths.push(path + '/');
}
2015-04-25 12:00:06 +00:00
return paths;
2015-04-25 11:25:02 +00:00
};
2015-04-20 19:51:39 +00:00
ContextMenuRenderer.prototype.getSuggestedCommands = function(ident, url) {
2015-04-25 11:07:08 +00:00
var schema = url.split(':')[0];
2015-04-20 19:51:39 +00:00
var domain = url.split('/')[2];
2015-04-25 11:25:02 +00:00
var withoutQuery = url.split('?')[0];
var path = '/' + split(withoutQuery, '/', 3)[3];
2015-04-25 11:07:08 +00:00
var reSchema = startsWith(schema, 'http') ? 'https?' : 'ftp';
2015-04-25 11:25:02 +00:00
return this.getPathVariants(path).map(function(p) {
2015-07-28 12:01:20 +00:00
return "^" + reSchema + "://" + regExpEscape(domain + p);
});
2015-04-20 20:30:39 +00:00
};
2015-04-20 19:51:39 +00:00
ContextMenuRenderer.prototype.makeEntries = function(ident, url) {
var commands = this.getSuggestedCommands(ident, url).map(function(c) {
2015-04-20 20:30:39 +00:00
return h(
'span',
{'onclick': this.makeCopyTextFn(c)},
2015-07-28 12:01:20 +00:00
"Copy " + c
2015-04-20 20:30:39 +00:00
);
2015-04-20 19:51:39 +00:00
}.bind(this));
2015-04-20 16:32:43 +00:00
return [
2015-04-20 18:57:12 +00:00
// Unfortunately, this does not open it in a background tab
// like the real context menu does.
2015-04-20 17:15:19 +00:00
h('a', {'href': url}, "Open link in new tab")
,h('span', {'onclick': this.makeCopyTextFn(url)}, "Copy link address")
2015-04-20 19:51:39 +00:00
].concat(commands);
2015-04-20 17:15:19 +00:00
};
2015-04-20 16:32:43 +00:00
2015-04-20 15:28:55 +00:00
ContextMenuRenderer.prototype.onContextMenu = function(ev) {
//console.log(ev);
2015-04-20 17:47:08 +00:00
if(!this.clickedOnLogWindowURL(ev)) {
2015-04-20 15:01:45 +00:00
this.blur();
return;
}
2015-04-20 16:32:43 +00:00
ev.preventDefault();
2015-04-20 20:30:39 +00:00
this.visible = true;
2015-04-20 15:01:45 +00:00
this.element.style.display = "block";
this.element.style.left = ev.clientX + "px";
this.element.style.top = ev.clientY + "px";
2015-04-20 16:32:43 +00:00
removeChildren(this.element);
2015-04-20 17:15:19 +00:00
// We put the clipboard-scratchpad in the fixed-positioned
// context menu instead of elsewhere on the page, because
// we must focus the input box to automatically copy its text,
// and the focus operation scrolls to the element on the page,
// and we want to avoid such scrolling.
appendAny(this.element, h('input', {'type': 'text', 'id': 'clipboard-scratchpad'}));
2015-04-20 16:32:43 +00:00
var url = ev.target.href;
2015-04-25 22:11:37 +00:00
try {
var ident = ev.target.parentNode.parentNode.id.match(/^log-window-(.*)/)[1];
} catch(e) {
// moreDom=1
var ident = ev.target.parentNode.parentNode.parentNode.id.match(/^log-window-(.*)/)[1];
}
2015-04-20 19:51:39 +00:00
var entries = this.makeEntries(ident, url);
2015-04-20 16:32:43 +00:00
for(var i=0; i < entries.length ; i + + ) {
2015-04-20 17:46:10 +00:00
var entry = entries[i];
entry.classList.add('context-menu-entry');
appendAny(this.element, entry);
2015-04-20 16:32:43 +00:00
}
2015-04-25 11:42:21 +00:00
// If the bottom of the context menu is outside the viewport, move the context
// menu up, so that it appears to have opened from its bottom-left corner.
// + 1 pixel so that the pointer lands inside the element and turns on cursor: default
if(ev.clientY + this.element.offsetHeight > document.documentElement.clientHeight) {
this.element.style.top = (ev.clientY - this.element.offsetHeight + 1) + "px";
}
2015-04-20 15:01:45 +00:00
};
ContextMenuRenderer.prototype.blur = function() {
2015-04-20 20:30:39 +00:00
this.visible = false;
2015-04-20 15:01:45 +00:00
this.element.style.display = "none";
2015-04-20 20:30:39 +00:00
this.callAfterBlurFns.map(function(fn) { fn(); });
this.callAfterBlurFns = [];
};
// TODO: decouple - fire an onblur event instead
ContextMenuRenderer.prototype.callAfterBlur = function(fn) {
this.callAfterBlurFns.push(fn);
2015-04-20 15:01:45 +00:00
};
2013-09-23 16:13:38 -05:00
var BatchingQueue = function(callable, minInterval) {
this.callable = callable;
this._minInterval = minInterval;
this.queue = [];
this._timeout = null;
this._boundRunCallable = this._runCallable.bind(this);
};
BatchingQueue.prototype.setMinInterval = function(minInterval) {
this._minInterval = minInterval;
};
BatchingQueue.prototype._runCallable = function() {
this._timeout = null;
var queue = this.queue;
this.queue = [];
this.callable(queue);
};
BatchingQueue.prototype.callNow = function() {
if(this._timeout !== null) {
clearTimeout(this._timeout);
this._timeout = null;
}
this._runCallable();
};
BatchingQueue.prototype.push = function(v) {
this.queue.push(v);
if(this._timeout === null) {
this._timeout = setTimeout(this._boundRunCallable, this._minInterval);
}
};
var Decayer = function(initial, multiplier, max) {
this.initial = initial;
this.multiplier = multiplier;
this.max = max;
this.reset();
};
Decayer.prototype.reset = function() {
// First call to .decay() will multiply, but we want to get the `intitial`
// value on the first call to .decay(), so divide.
this.current = this.initial / this.multiplier;
return this.current;
};
Decayer.prototype.decay = function() {
this.current = Math.min(this.current * this.multiplier, this.max);
return this.current;
};
var Dashboard = function() {
this.messageCount = 0;
var args = getQueryArgs();
var historyLines =
args["historyLines"] ?
Number(args["historyLines"]) :
navigator.userAgent.match(/Mobi/) ? 250 : 1000;
var batchTimeWhenVisible =
args["batchTimeWhenVisible"] ?
Number(args["batchTimeWhenVisible"]) :
125;
var showNicks =
args["showNicks"] ?
2015-03-28 02:34:20 +00:00
Boolean(Number(args["showNicks"])) :
2013-09-23 16:13:38 -05:00
false;
2015-04-20 15:41:03 +00:00
var contextMenu =
args["contextMenu"] ?
Boolean(Number(args["contextMenu"])) :
/**
* The context menu is useless without document.execCommand('copy'),
2015-07-13 21:35:22 +00:00
* which is available in IE10+, Chrome 42+, and Firefox 41+
2015-04-20 15:41:03 +00:00
*/
2015-05-20 00:23:55 +00:00
((isTrident & & getTridentMajorVersion() >= 6) ||
2015-07-06 05:22:34 +00:00
(isChrome & & getChromeMajorVersion() >= 42) ||
(isFirefox & & getFirefoxMajorVersion() >= 41));
2015-04-25 22:04:03 +00:00
var moreDom =
args["moreDom"] ?
Boolean(Number(args["moreDom"])) :
false;
2015-04-25 21:52:46 +00:00
// Append to page title to make it possible to identify the tab in Chrome's task manager
if(args["title"]) {
document.title += " - " + args["title"];
}
2015-04-25 22:04:03 +00:00
if(moreDom) {
JobsRenderer.prototype._renderDownloadLine = JobsRenderer.prototype._moreDomRenderDownloadLine;
}
2015-07-18 04:10:15 +00:00
if(args["host"]) {
this.host = args["host"];
} else {
2016-02-24 23:46:21 +00:00
// If no ?host=, connect to this grab-site server instead of some other server.
2016-02-24 15:22:31 -06:00
this.host = location.host;
2015-07-18 04:10:15 +00:00
}
2013-09-23 16:13:38 -05:00
this.dumpTraffic = args["dumpMax"] & & Number(args["dumpMax"]) > 0;
if(this.dumpTraffic) {
this.dumpMax = Number(args["dumpMax"]);
}
2015-04-20 20:30:39 +00:00
this.contextMenuRenderer = new ContextMenuRenderer(document);
if(contextMenu) {
document.oncontextmenu = this.contextMenuRenderer.onContextMenu.bind(this.contextMenuRenderer);
document.onclick = this.contextMenuRenderer.blur.bind(this.contextMenuRenderer);
// onkeydown picks up ESC, onkeypress doesn't (tested Chrome 44)
document.onkeydown = function(ev) {
if(ev.keyCode == 27) { // ESC
this.contextMenuRenderer.blur();
}
}.bind(this);
2015-04-25 12:20:46 +00:00
// In Chrome, the native context menu disappears when you wheel around, so
// match that behavior for our own context menu.
if(isChrome) {
document.onwheel = function(ev) {
this.contextMenuRenderer.blur();
}.bind(this);
}
2015-04-20 20:30:39 +00:00
}
2014-10-09 07:49:16 +00:00
this.jobsRenderer = new JobsRenderer(
2015-04-20 20:30:39 +00:00
byId('logs'), byId('filter-box'), historyLines, showNicks, this.contextMenuRenderer);
2013-09-23 16:13:38 -05:00
var batchTimeWhenHidden = 5000;
2015-03-27 11:28:33 +00:00
document.onkeypress = this.keyPress.bind(this);
2015-04-20 15:41:03 +00:00
2015-04-20 21:03:01 +00:00
// Adjust help text based on URL
Array.prototype.slice.call(document.querySelectorAll('.url-q-or-amp')).map(function(elem) {
if(window.location.search.indexOf("?") != -1) {
elem.textContent = "&";
}
});
2015-06-08 06:54:49 +00:00
if(!showNicks) {
document.write('< style > . job-nick-aligned { width : 0 ; } < / style > ');
}
2015-07-18 02:11:18 +00:00
this.queue = new BatchingQueue(function(queue) {
//console.log("Queue has ", queue.length, "items");
for(var i=0; i < queue.length ; i + + ) {
this.handleData(JSON.parse(queue[i]));
}
}.bind(this), batchTimeWhenVisible);
this.decayer = new Decayer(1000, 1.5, 60000);
this.connectWebSocket();
document.addEventListener("visibilitychange", function() {
if(document.hidden) {
//console.log("Page has become hidden");
this.queue.setMinInterval(batchTimeWhenHidden);
} else {
//console.log("Page has become visible");
this.queue.setMinInterval(batchTimeWhenVisible);
this.queue.callNow();
}
}.bind(this), false);
2015-03-27 11:28:33 +00:00
};
Dashboard.prototype.keyPress = function(ev) {
2015-03-27 12:11:25 +00:00
//console.log(ev);
2015-07-29 23:04:20 +00:00
// If you press ctrl-f or alt-f in Firefox (tested: 41), it dispatches
// the keypress event for 'f'. We want only the modifier-free
// keypresses.
if(ev.ctrlKey || ev.altKey || ev.metaKey || ev.shiftKey) {
return;
}
2015-03-27 12:11:25 +00:00
if(ev.which == 106) { // j
this.jobsRenderer.showNextPrev(1);
} else if(ev.which == 107) { // k
this.jobsRenderer.showNextPrev(-1);
2015-03-28 02:28:00 +00:00
} else if(ev.which == 97) { // a
ds.setFilter('');
} else if(ev.which == 110) { // n
ds.setFilter('^$');
} else if(ev.which == 102) { // f
ev.preventDefault();
byId('filter-box').focus();
byId('filter-box').select();
2015-04-20 19:14:50 +00:00
} else if(ev.which == 118) { // v
window.open(this.jobsRenderer.firstFilterMatch["url"]);
2015-04-09 20:56:36 +00:00
} else if(ev.which == 63) { // ?
ds.toggleHelp();
2015-03-27 11:28:33 +00:00
}
2013-09-23 16:13:38 -05:00
};
Dashboard.prototype.handleData = function(data) {
this.messageCount += 1;
if(this.dumpTraffic & & this.messageCount < = this.dumpMax) {
byId('traffic').appendChild(h("pre", null, prettyJson(data)));
}
this.jobsRenderer.handleData(data);
};
Dashboard.prototype.connectWebSocket = function() {
2014-09-10 06:28:30 +00:00
this.ws = new WebSocket("ws://" + this.host + "/stream");
2013-09-23 16:13:38 -05:00
this.ws.onmessage = function(ev) {
this.queue.push(ev["data"]);
}.bind(this);
this.ws.onopen = function(ev) {
console.log("WebSocket opened:", ev);
2015-07-18 13:29:19 +00:00
this.ws.send(JSON.stringify({
"type": "hello",
"mode": "dashboard",
"user_agent": navigator.userAgent
}));
2013-09-23 16:13:38 -05:00
this.decayer.reset();
}.bind(this);
this.ws.onclose = function(ev) {
console.log("WebSocket closed:", ev);
var delay = this.decayer.decay();
console.log("Reconnecting in", delay, "ms");
setTimeout(this.connectWebSocket.bind(this), delay);
}.bind(this);
};
2014-10-09 05:32:47 +00:00
Dashboard.prototype.toggleAlign = function() {
2014-10-09 06:05:49 +00:00
this.jobsRenderer.toggleAlign();
2014-10-09 05:32:47 +00:00
};
2013-09-23 16:13:38 -05:00
Dashboard.prototype.toggleHelp = function() {
var help = byId('help');
if(help.classList.contains('undisplayed')) {
help.classList.remove('undisplayed');
} else {
help.classList.add('undisplayed');
}
};
2014-10-09 09:01:13 +00:00
Dashboard.prototype.getFilter = function(value) {
return byId('filter-box').value;
};
Dashboard.prototype.setFilter = function(value) {
byId('filter-box').value = value;
byId('filter-box').onchange();
};
2013-09-23 16:13:38 -05:00
var ds = new Dashboard();
< / script >
< / body >
< / html >