<!doctype html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content="width=device-width, initial-scale=1"> <title>HTTPing | Access servers based on network response time!</title> <script type="text/javascript"> /* Configuration start */ var test_url_list = [ { name : 'Site 1', testurl: "http://site1.app.example.com/MYTESTFILE.txt", redirect: "http://site1.app.example.com"}, { name : 'Site 2', testurl: "http://site2.app.example.com/MYTESTFILE.txt", redirect: "http://site2.app.example.com"} ]; /* Configuration finishes */ /* DO NOT EDIT BELOW */ (function(globals) { var define, requireModule; (function() { var registry = {}, seen = {}; define = function(name, deps, callback) { registry[name] = { deps: deps, callback: callback }; }; requireModule = function(name) { if (seen[name]) { return seen[name]; } seen[name] = {}; var mod = registry[name]; if (!mod) { throw new Error("Module '" + name + "' not found."); } var deps = mod.deps, callback = mod.callback, reified = [], exports; for (var i=0, l=deps.length; i<l; i++) { if (deps[i] === 'exports') { reified.push(exports = {}); } else { reified.push(requireModule(deps[i])); } } var value = callback.apply(this, reified); return seen[name] = exports || value; }; })(); define("rsvp/all", ["rsvp/promise","exports"], function(__dependency1__, __exports__) { "use strict"; var Promise = __dependency1__.Promise; /* global toString */ function promiseAtLeast(expected_count, promises) { if (Object.prototype.toString.call(promises) !== "[object Array]") { throw new TypeError('You must pass an array to all.'); } function canceller() { var promise; for (var i = 0; i < promises.length; i++) { promise = promises[i]; if (promise && typeof promise.then === 'function' && typeof promise.cancel === 'function') { promise.cancel(); } } } return new Promise(function(resolve, reject) { var results = [], remaining = promises.length, promise, remaining_count = promises.length - expected_count; if (remaining === 0) { if (expected_count === 1) { resolve(); } else { resolve([]); } } function resolver(index) { return function(value) { resolveAll(index, value); }; } function resolveAll(index, value) { results[index] = value; if (--remaining === remaining_count) { if (remaining_count === 0) { resolve(results); } else { resolve(value); canceller(); } } } function cancelAll(rejectionValue) { reject(rejectionValue); canceller(); } for (var i = 0; i < promises.length; i++) { promise = promises[i]; if (promise && typeof promise.then === 'function') { promise.then(resolver(i), cancelAll); } else { resolveAll(i, promise); } } }, canceller ); } function all(promises) { return promiseAtLeast(promises.length, promises); } function any(promises) { return promiseAtLeast(1, promises); } __exports__.all = all; __exports__.any = any; }); define("rsvp/async", ["exports"], function(__exports__) { "use strict"; var browserGlobal = (typeof window !== 'undefined') ? window : {}; var BrowserMutationObserver = browserGlobal.MutationObserver || browserGlobal.WebKitMutationObserver; var async; var local = (typeof global !== 'undefined') ? global : this; function checkNativePromise() { if (typeof Promise === "function" && typeof Promise.resolve === "function") { try { /* global Promise */ var promise = new Promise(function(){}); if ({}.toString.call(promise) === "[object Promise]") { return true; } } catch (e) {} } return false; } function useNativePromise() { var nativePromise = Promise.resolve(); return function(callback, arg) { nativePromise.then(function () { callback(arg); }); }; } // old node function useNextTick() { return function(callback, arg) { process.nextTick(function() { callback(arg); }); }; } // node >= 0.10.x function useSetImmediate() { return function(callback, arg) { /* global setImmediate */ setImmediate(function(){ callback(arg); }); }; } function useMutationObserver() { var queue = []; var observer = new BrowserMutationObserver(function() { var toProcess = queue.slice(); queue = []; toProcess.forEach(function(tuple) { var callback = tuple[0], arg= tuple[1]; callback(arg); }); }); var element = document.createElement('div'); observer.observe(element, { attributes: true }); // Chrome Memory Leak: https://bugs.webkit.org/show_bug.cgi?id=93661 window.addEventListener('unload', function(){ observer.disconnect(); observer = null; }, false); return function(callback, arg) { queue.push([callback, arg]); element.setAttribute('drainQueue', 'drainQueue'); }; } function useSetTimeout() { return function(callback, arg) { local.setTimeout(function() { callback(arg); }, 1); }; } if (checkNativePromise()) { async = useNativePromise(); } else if (typeof setImmediate === 'function') { async = useSetImmediate(); } else if (typeof process !== 'undefined' && {}.toString.call(process) === '[object process]') { async = useNextTick(); } else if (BrowserMutationObserver) { async = useMutationObserver(); } else { async = useSetTimeout(); } __exports__.async = async; }); define("rsvp/cancellation_error", ["exports"], function(__exports__) { "use strict"; // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Error function CancellationError(message) { this.name = "cancel"; if ((message !== undefined) && (typeof message !== "string")) { throw new TypeError('You must pass a string.'); } this.message = message || "Default Message"; } CancellationError.prototype = new Error(); CancellationError.prototype.constructor = CancellationError; __exports__.CancellationError = CancellationError; }); define("rsvp/config", ["rsvp/async","exports"], function(__dependency1__, __exports__) { "use strict"; var async = __dependency1__.async; var config = {}; config.async = async; __exports__.config = config; }); define("rsvp/defer", ["rsvp/promise","exports"], function(__dependency1__, __exports__) { "use strict"; var Promise = __dependency1__.Promise; function defer() { var deferred = { // pre-allocate shape resolve: undefined, reject: undefined, promise: undefined }; deferred.promise = new Promise(function(resolve, reject) { deferred.resolve = resolve; deferred.reject = reject; }); return deferred; } __exports__.defer = defer; }); define("rsvp/events", ["exports"], function(__exports__) { "use strict"; var Event = function(type, options) { this.type = type; for (var option in options) { if (!options.hasOwnProperty(option)) { continue; } this[option] = options[option]; } }; var indexOf = function(callbacks, callback) { for (var i=0, l=callbacks.length; i<l; i++) { if (callbacks[i][0] === callback) { return i; } } return -1; }; var callbacksFor = function(object) { var callbacks = object._promiseCallbacks; if (!callbacks) { callbacks = object._promiseCallbacks = {}; } return callbacks; }; var EventTarget = { mixin: function(object) { object.on = this.on; object.off = this.off; object.trigger = this.trigger; return object; }, on: function(eventNames, callback, binding) { var allCallbacks = callbacksFor(this), callbacks, eventName; eventNames = eventNames.split(/\s+/); binding = binding || this; while (eventName = eventNames.shift()) { callbacks = allCallbacks[eventName]; if (!callbacks) { callbacks = allCallbacks[eventName] = []; } if (indexOf(callbacks, callback) === -1) { callbacks.push([callback, binding]); } } }, off: function(eventNames, callback) { var allCallbacks = callbacksFor(this), callbacks, eventName, index; eventNames = eventNames.split(/\s+/); while (eventName = eventNames.shift()) { if (!callback) { allCallbacks[eventName] = []; continue; } callbacks = allCallbacks[eventName]; index = indexOf(callbacks, callback); if (index !== -1) { callbacks.splice(index, 1); } } }, trigger: function(eventName, options) { var allCallbacks = callbacksFor(this), callbacks, callbackTuple, callback, binding, event; if (callbacks = allCallbacks[eventName]) { // Don't cache the callbacks.length since it may grow for (var i=0; i<callbacks.length; i++) { callbackTuple = callbacks[i]; callback = callbackTuple[0]; binding = callbackTuple[1]; if (typeof options !== 'object') { options = { detail: options }; } event = new Event(eventName, options); callback.call(binding, event); } } } }; __exports__.EventTarget = EventTarget; }); define("rsvp/hash", ["rsvp/defer","exports"], function(__dependency1__, __exports__) { "use strict"; var defer = __dependency1__.defer; function size(object) { var s = 0; for (var prop in object) { s++; } return s; } function hash(promises) { var results = {}, deferred = defer(), remaining = size(promises); if (remaining === 0) { deferred.resolve({}); } var resolver = function(prop) { return function(value) { resolveAll(prop, value); }; }; var resolveAll = function(prop, value) { results[prop] = value; if (--remaining === 0) { deferred.resolve(results); } }; var rejectAll = function(error) { deferred.reject(error); }; for (var prop in promises) { if (promises[prop] && typeof promises[prop].then === 'function') { promises[prop].then(resolver(prop), rejectAll); } else { resolveAll(prop, promises[prop]); } } return deferred.promise; } __exports__.hash = hash; }); define("rsvp/node", ["rsvp/promise","rsvp/all","exports"], function(__dependency1__, __dependency2__, __exports__) { "use strict"; var Promise = __dependency1__.Promise; var all = __dependency2__.all; function makeNodeCallbackFor(resolve, reject) { return function (error, value) { if (error) { reject(error); } else if (arguments.length > 2) { resolve(Array.prototype.slice.call(arguments, 1)); } else { resolve(value); } }; } function denodeify(nodeFunc) { return function() { var nodeArgs = Array.prototype.slice.call(arguments), resolve, reject; var thisArg = this; var promise = new Promise(function(nodeResolve, nodeReject) { resolve = nodeResolve; reject = nodeReject; }); all(nodeArgs).then(function(nodeArgs) { nodeArgs.push(makeNodeCallbackFor(resolve, reject)); try { nodeFunc.apply(thisArg, nodeArgs); } catch(e) { reject(e); } }); return promise; }; } __exports__.denodeify = denodeify; }); define("rsvp/promise", ["rsvp/config","rsvp/events","rsvp/cancellation_error","exports"], function(__dependency1__, __dependency2__, __dependency3__, __exports__) { "use strict"; var config = __dependency1__.config; var EventTarget = __dependency2__.EventTarget; var CancellationError = __dependency3__.CancellationError; function objectOrFunction(x) { return isFunction(x) || (typeof x === "object" && x !== null); } function isFunction(x){ return typeof x === "function"; } var Promise = function(resolver, canceller) { var promise = this, resolved = false; if (typeof resolver !== 'function') { throw new TypeError('You must pass a resolver function as the sole argument to the promise constructor'); } if ((canceller !== undefined) && (typeof canceller !== 'function')) { throw new TypeError('You can only pass a canceller function' + ' as the second argument to the promise constructor'); } if (!(promise instanceof Promise)) { return new Promise(resolver, canceller); } var resolvePromise = function(value) { if (resolved) { return; } resolved = true; resolve(promise, value); }; var rejectPromise = function(value) { if (resolved) { return; } resolved = true; reject(promise, value); }; this.on('promise:failed', function(event) { this.trigger('error', { detail: event.detail }); }, this); this.on('error', onerror); this.cancel = function () { // For now, simply reject the promise and does not propagate the cancel // to parent or children if (resolved) { return; } if (canceller !== undefined) { try { canceller(); } catch (e) { rejectPromise(e); return; } } // Trigger cancel? rejectPromise(new CancellationError()); }; try { resolver(resolvePromise, rejectPromise); } catch(e) { rejectPromise(e); } }; function onerror(event) { if (config.onerror) { config.onerror(event.detail); } } var invokeCallback = function(type, promise, callback, event) { var hasCallback = isFunction(callback), value, error, succeeded, failed; if (promise.isFulfilled) { return; } if (promise.isRejected) { return; } if (hasCallback) { try { value = callback(event.detail); succeeded = true; } catch(e) { failed = true; error = e; } } else { value = event.detail; succeeded = true; } if (handleThenable(promise, value)) { return; } else if (hasCallback && succeeded) { resolve(promise, value); } else if (failed) { reject(promise, error); } else if (type === 'resolve') { resolve(promise, value); } else if (type === 'reject') { reject(promise, value); } }; Promise.prototype = { constructor: Promise, isRejected: undefined, isFulfilled: undefined, rejectedReason: undefined, fulfillmentValue: undefined, then: function(done, fail) { this.off('error', onerror); var thenPromise = new this.constructor(function() {}, function () { thenPromise.trigger('promise:cancelled', {}); }); if (this.isFulfilled) { config.async(function(promise) { invokeCallback('resolve', thenPromise, done, { detail: promise.fulfillmentValue }); }, this); } if (this.isRejected) { config.async(function(promise) { invokeCallback('reject', thenPromise, fail, { detail: promise.rejectedReason }); }, this); } this.on('promise:resolved', function(event) { invokeCallback('resolve', thenPromise, done, event); }); this.on('promise:failed', function(event) { invokeCallback('reject', thenPromise, fail, event); }); return thenPromise; }, fail: function(fail) { return this.then(null, fail); }, always: function(fail) { return this.then(fail, fail); } }; EventTarget.mixin(Promise.prototype); function resolve(promise, value) { if (promise === value) { fulfill(promise, value); } else if (!handleThenable(promise, value)) { fulfill(promise, value); } } function handleThenable(promise, value) { var then = null, resolved; try { if (promise === value) { throw new TypeError("A promises callback cannot return that same promise."); } if (objectOrFunction(value)) { then = value.then; if (isFunction(then)) { promise.on('promise:cancelled', function(event) { if (isFunction(value.cancel)) { value.cancel(); } }); then.call(value, function(val) { if (resolved) { return true; } resolved = true; if (value !== val) { resolve(promise, val); } else { fulfill(promise, val); } }, function(val) { if (resolved) { return true; } resolved = true; reject(promise, val); }); return true; } } } catch (error) { reject(promise, error); return true; } return false; } function fulfill(promise, value) { config.async(function() { if (promise.isFulfilled) { return; } if (promise.isRejected) { return; } promise.trigger('promise:resolved', { detail: value }); promise.isFulfilled = true; promise.fulfillmentValue = value; }); } function reject(promise, value) { config.async(function() { if (promise.isFulfilled) { return; } if (promise.isRejected) { return; } promise.trigger('promise:failed', { detail: value }); promise.isRejected = true; promise.rejectedReason = value; }); } __exports__.Promise = Promise; }); define("rsvp/queue", ["rsvp/promise","rsvp/resolve","exports"], function(__dependency1__, __dependency2__, __exports__) { "use strict"; var Promise = __dependency1__.Promise; var resolve = __dependency2__.resolve; // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Error function ResolvedQueueError(message) { this.name = "resolved"; if ((message !== undefined) && (typeof message !== "string")) { throw new TypeError('You must pass a string.'); } this.message = message || "Default Message"; } ResolvedQueueError.prototype = new Error(); ResolvedQueueError.prototype.constructor = ResolvedQueueError; var Queue = function() { var queue = this, promise_list = [], promise, fulfill, reject, resolved; if (!(this instanceof Queue)) { return new Queue(); } function canceller() { for (var i = 0; i < 2; i++) { promise_list[i].cancel(); } } promise = new Promise(function(done, fail) { fulfill = function (fulfillmentValue) { if (resolved) {return;} queue.isFulfilled = true; queue.fulfillmentValue = fulfillmentValue; resolved = true; return done(fulfillmentValue); }; reject = function (rejectedReason) { if (resolved) {return;} queue.isRejected = true; queue.rejectedReason = rejectedReason ; resolved = true; return fail(rejectedReason); }; }, canceller); promise_list.push(resolve()); promise_list.push(promise_list[0].then(function () { promise_list.splice(0, 2); if (promise_list.length === 0) { fulfill(); } })); queue.cancel = function () { if (resolved) {return;} resolved = true; promise.cancel(); promise.fail(function (rejectedReason) { queue.isRejected = true; queue.rejectedReason = rejectedReason; }); }; queue.then = function () { return promise.then.apply(promise, arguments); }; queue.push = function(done, fail) { var last_promise = promise_list[promise_list.length - 1], next_promise; if (resolved) { throw new ResolvedQueueError(); } next_promise = last_promise.then(done, fail); promise_list.push(next_promise); // Handle pop promise_list.push(next_promise.then(function (fulfillmentValue) { promise_list.splice(0, 2); if (promise_list.length === 0) { fulfill(fulfillmentValue); } else { return fulfillmentValue; } }, function (rejectedReason) { promise_list.splice(0, 2); if (promise_list.length === 0) { reject(rejectedReason); } else { throw rejectedReason; } })); return this; }; }; Queue.prototype = Object.create(Promise.prototype); Queue.prototype.constructor = Queue; __exports__.Queue = Queue; __exports__.ResolvedQueueError = ResolvedQueueError; }); define("rsvp/reject", ["rsvp/promise","exports"], function(__dependency1__, __exports__) { "use strict"; var Promise = __dependency1__.Promise; function reject(reason) { return new Promise(function (resolve, reject) { reject(reason); }); } __exports__.reject = reject; }); define("rsvp/resolve", ["rsvp/promise","exports"], function(__dependency1__, __exports__) { "use strict"; var Promise = __dependency1__.Promise; function resolve(thenable) { return new Promise(function(resolve, reject) { if (typeof thenable === "object" && thenable !== null) { var then = thenable.then; if ((then !== undefined) && (typeof then === "function")) { return then.apply(thenable, [resolve, reject]); } } return resolve(thenable); }, function () { if ((thenable !== undefined) && (thenable.cancel !== undefined)) { thenable.cancel(); } }); } __exports__.resolve = resolve; }); define("rsvp/rethrow", ["exports"], function(__exports__) { "use strict"; var local = (typeof global === "undefined") ? this : global; function rethrow(reason) { local.setTimeout(function() { throw reason; }); throw reason; } __exports__.rethrow = rethrow; }); define("rsvp/timeout", ["rsvp/promise","exports"], function(__dependency1__, __exports__) { "use strict"; var Promise = __dependency1__.Promise; function promiseSetTimeout(millisecond, should_reject, message) { var timeout_id; function resolver(resolve, reject) { timeout_id = setTimeout(function () { if (should_reject) { reject(message); } else { resolve(message); } }, millisecond); } function canceller() { clearTimeout(timeout_id); } return new Promise(resolver, canceller); } function delay(millisecond, message) { return promiseSetTimeout(millisecond, false, message); } function timeout(millisecond) { return promiseSetTimeout(millisecond, true, "Timed out after " + millisecond + " ms"); } Promise.prototype.delay = function(millisecond) { return this.then(function (fulfillmentValue) { return delay(millisecond, fulfillmentValue); }); }; __exports__.delay = delay; __exports__.timeout = timeout; }); define("rsvp", ["rsvp/events","rsvp/cancellation_error","rsvp/promise","rsvp/node","rsvp/all","rsvp/queue","rsvp/timeout","rsvp/hash","rsvp/rethrow","rsvp/defer","rsvp/config","rsvp/resolve","rsvp/reject","exports"], function(__dependency1__, __dependency2__, __dependency3__, __dependency4__, __dependency5__, __dependency6__, __dependency7__, __dependency8__, __dependency9__, __dependency10__, __dependency11__, __dependency12__, __dependency13__, __exports__) { "use strict"; var EventTarget = __dependency1__.EventTarget; var CancellationError = __dependency2__.CancellationError; var Promise = __dependency3__.Promise; var denodeify = __dependency4__.denodeify; var all = __dependency5__.all; var any = __dependency5__.any; var Queue = __dependency6__.Queue; var ResolvedQueueError = __dependency6__.ResolvedQueueError; var delay = __dependency7__.delay; var timeout = __dependency7__.timeout; var hash = __dependency8__.hash; var rethrow = __dependency9__.rethrow; var defer = __dependency10__.defer; var config = __dependency11__.config; var resolve = __dependency12__.resolve; var reject = __dependency13__.reject; function configure(name, value) { config[name] = value; } __exports__.CancellationError = CancellationError; __exports__.Promise = Promise; __exports__.EventTarget = EventTarget; __exports__.all = all; __exports__.any = any; __exports__.Queue = Queue; __exports__.ResolvedQueueError = ResolvedQueueError; __exports__.delay = delay; __exports__.timeout = timeout; __exports__.hash = hash; __exports__.rethrow = rethrow; __exports__.defer = defer; __exports__.denodeify = denodeify; __exports__.configure = configure; __exports__.resolve = resolve; __exports__.reject = reject; }); window.RSVP = requireModule("rsvp"); })(window); </script> <script type="text/javascript"> /*jslint indent: 2, maxlen: 80, nomen: true */ /*global console, window, document, RSVP, XMLHttpRequest */ (function (document, RSVP, XMLHttpRequest) { "use strict"; var promise_list = []; function ajax(param) { var xhr = new XMLHttpRequest(); return new RSVP.Promise(function (resolve, reject, notify) { var k; xhr.open(param.type || "GET", param.url, true); xhr.responseType = param.dataType || ""; if (typeof param.headers === 'object' && param.headers !== null) { for (k in param.headers) { if (param.headers.hasOwnProperty(k)) { xhr.setRequestHeader(k, param.headers[k]); } } } xhr.addEventListener("load", function (e) { if (e.target.status >= 400) { return reject(e); } return resolve(e); }); xhr.addEventListener("error", reject); xhr.addEventListener("progress", notify); if (typeof param.xhrFields === 'object' && param.xhrFields !== null) { for (k in param.xhrFields) { if (param.xhrFields.hasOwnProperty(k)) { xhr[k] = param.xhrFields[k]; } } } if (typeof param.beforeSend === 'function') { param.beforeSend(xhr); } xhr.send(param.data); }, function () { xhr.abort(); }); } function testURL(url) { return RSVP.Queue() .push(function () { return ajax({ url: url }); }) .push(function (evt) { return evt.target.responseText; }, function () { return "FAIL"; }); } function launchTest(test_case) { var start = new Date().getTime(), url = test_case.testurl; return new RSVP.Queue() .push(function () { return RSVP.any([RSVP.delay(4000), testURL(url + "?nocache=" + start)]); }) .push(function (result) { var elapsed = new Date().getTime() - start; if (result === undefined) { return {url: test_case.redirect, time: 1000000001, name: test_case.name}; } if (result === "FAIL") { return {url: test_case.redirect, time: 1000000000, name: test_case.name}; } return {url: test_case.redirect, name: test_case.name, time: elapsed}; }); } function runOnce(test_queue) { return new RSVP.Queue() .push(function () { var u; for (u in test_url_list) { if (test_url_list.hasOwnProperty(u)) { test_queue.push(launchTest(test_url_list[u])); } } return RSVP.all(test_queue); }) .push(undefined, function (reason) { console.log(reason); return "OK"; }); } function renderPartialResult(result, main_div, winner_div) { var u, y, interaction, winner = 0, t_i, table_dict = {}, interaction_dict = {}, msg = ""; msg += "<table width=100%>"; for (u in result) { if (result.hasOwnProperty(u)) { interaction = parseInt(u / test_url_list.length, 10); if (!interaction_dict.hasOwnProperty(interaction)) { interaction_dict[interaction] = 1; } if (!table_dict.hasOwnProperty(result[u].name)) { table_dict[result[u].name] = { total: 0, name: result[u].name, url: result[u].url }; } if (result[u].time === 1000000000) { table_dict[result[u].name][interaction] = "FAIL"; table_dict[result[u].name].total = "REJECT"; } else if (result[u].time === 1000000001) { table_dict[result[u].name][interaction] = "TIMEOUT"; table_dict[result[u].name].total = "REJECT"; } else { table_dict[result[u].name][interaction] = result[u].time; if (table_dict[result[u].name].total !== "REJECT") { table_dict[result[u].name].total += result[u].time; } } } } msg += "</tr> <th> </th>"; t_i = 0; for (u in interaction_dict) { if (interaction_dict.hasOwnProperty(u)) { msg += '<th> Test ' + u + "</th>"; t_i += 1; } } msg += "<th class='avg'> AVERAGE </th></tr>"; for (u in table_dict) { if (table_dict.hasOwnProperty(u)) { if (winner === 0) { winner = table_dict[u]; } if (table_dict[u].total !== "REJECT") { table_dict[u].average = table_dict[u].total / t_i; if (winner.average === "REJECT") { winner = table_dict[u]; } if (winner.average > table_dict[u].average) { winner = table_dict[u]; } } else { table_dict[u].average = "REJECT"; } } } for (u in table_dict) { if (table_dict.hasOwnProperty(u)) { if (u === winner.name) { msg += "<tr class='winner'>"; } else { msg += "<tr>"; } msg += "<td>"; if (table_dict[u].url) { msg += '<a href="' + table_dict[u].url + '">' + u + '</a>'; } else { msg += u; } msg += "</td>"; for (y in interaction_dict) { if (interaction_dict.hasOwnProperty(y)) { msg += "<td>" + table_dict[u][y] + "</td>"; } } msg += "<td>" + table_dict[u].average + "</td>"; msg += "</tr>"; } } msg += "</table>"; msg = "<h2> The best frontend for you is ... <strong>" + winner.name + " </strong> </h2>" + msg; main_div.innerHTML = msg; msg = ""; msg += "<p><a class=btn href=" + winner.url + "> Click to Connect via "; msg += winner.name + "</a> </p>"; winner_div.innerHTML = msg; return winner; } function runAll() { return runOnce(promise_list) .push(function () { return runOnce(promise_list) .push(function () { return runOnce(promise_list) .push(function () { return runOnce(promise_list) .push(function () { return runOnce(promise_list) .push(function (result) { var winner, div = document.getElementsByClassName("table"), winner_div = document.getElementsByClassName("link"); winner = renderPartialResult(result, div[0], winner_div[0]); return winner; }); }); }); }); }); } runAll(); }(document, RSVP, XMLHttpRequest)); </script> <style type="text/css"> /* === animation === */ @-ms-keyframes spin { from { -ms-transform: rotate(0deg); } to { -ms-transform: rotate(360deg); } } @-moz-keyframes spin { from { -moz-transform: rotate(0deg); } to { -moz-transform: rotate(360deg); } } @-webkit-keyframes spin { from { -webkit-transform: rotate(0deg); } to { -webkit-transform: rotate(360deg); } } @keyframes spin { from { transform:rotate(0deg); } to { transform:rotate(360deg); } } /* === font (embedded icons (hehe) and text === */ @charset "UTF-8"; body { font-family: Arial; } @font-face { font-family: "untitled-font-1"; src: url('untitled-font-1.eot') format('embedded-opentype'); /* src:url("fonts/untitled-font-1.eot"); src:url("fonts/untitled-font-1.eot?#iefix") format("embedded-opentype"), url("fonts/untitled-font-1.woff") format("woff"), url("fonts/untitled-font-1.ttf") format("truetype"), url("fonts/untitled-font-1.svg#untitled-font-1") format("svg"); */ font-weight: normal; font-style: normal; } @font-face { font-family: 'untitled-font-1'; src: url(data:application/x-font-woff;charset=utf-8;base64,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) format('woff'), url('untitled-font-1.ttf') format('truetype'), url('untitled-font-1.svg#untitled-font-1') format('svg'); font-weight: normal; font-style: normal; } /* === set font on classes === */ [class^="custom-icon-"]:before, [class*=" custom-icon-"]:before { font-family: "untitled-font-1" !important; font-style: normal !important; font-weight: normal !important; font-variant: normal !important; text-transform: none !important; speak: none; line-height: 1; -webkit-font-smoothing: antialiased; -moz-osx-font-smoothing: grayscale; } .custom-icon-spinner:before { content: "a"; } .custom-icon-user:before { content: "b"; } .custom-icon-forward:before { content: "c"; } .custom-icon-harddrive:before { content: "e"; } /* === logo === */ .custom-grandenet-logo { background: url() no-repeat center center; display: block; } .custom-header-logo { width: 200px; height: 120px; margin: 0 auto; } .custom-icon-logo { width: 100px; height: 60px; } /* === corners === */ .custom-corner-all { border: 1px solid #9FD9F2; -webkit-border-radius: .375em; border-radius: .375em; } /* === wrapper === */ .custom-wrapper { max-width: 40em; padding: 0 1em; -moz-box-sizing: border-box; -webkit-box-sizing: border-box; box-sizing: border-box; display: block; margin: 0 auto; } @media (max-width: 40em) { .custom-wrapper { max-width: none; width: 100%; } } /* === description === */ .custom-description { padding: .5em 0; } .custom-description > span { display: inline-block; vertical-align: middle; text-align: center; width: 15%; line-height: 90px; height: 90px; } .custom-description .custom-icon { color: #24A9E1; font-size: 2em; } .custom-description > .custom-icon-logo { width: 35%; } /* === ipv4/6 info === */ .custom-description .custom-icon:nth-child(2):after { content: "IPv4"; } .custom-description .custom-icon:nth-child(4):after { content: "IPv6"; } .custom-description .custom-icon:nth-child(2):after, .custom-description .custom-icon:nth-child(4):after { display: block; line-height: 1em; font-family: Arial; font-size: .6em; margin-top: -32px; } @media (max-width: 40em) { .custom-description > .custom-grandenet-logo { background-position: 13px 0px; background-size: 9em; } .custom-description > .custom-grandenet-logo { width: 30%; } } /* === button === */ .btn { background-color:#FFC131; background-image:-webkit-linear-gradient(left top, #FFC131 42%, #FFB100 70%); background-image:linear-gradient(left top, #FFC131 42%, #FFB100 70%); display: block; text-align: center; padding: .5em 0; -webkit-border-radius: .375em; border-radius: .375em; color: black; text-decoration: none; font-weight: bold; } .btn:hover { background: #FFB100; text-decoration: none; font-weight: bold; } .btn:hover { background: #FFC131; text-decoration: none; } /* === table === */ table { background: rgba(159,217,242,.15); border: 1px solid #24A9E1; border-radius: .375em; padding: .5em; margin-bottom: 1em; } table th, table td { text-align: right; } table tr td:first-child { text-align: left; } table tr { border: 1px solid #24A9E1; } /* result */ .table h2 { font-weight: normal; text-align: center; } .table p { display: inline-block; vertical-align: middle; max-width: 80%; } .winner th, .winner td { background: #24A9E1; } @media (max-width: 26em) { table { padding: 0; } table tr td, table tr th { font-size: .6em; } } /* spinner */ .custom-icon-spinner { margin-right: .25em; padding-top: 10px; -webkit-animation-name: spin; -webkit-animation-duration: 600ms; -webkit-animation-iteration-count: infinite; -webkit-animation-timing-function: linear; animation-name: spin; animation-duration: 600ms; animation-iteration-count: infinite; animation-timing-function: linear; font-size: 2em; display: inline-block; width: auto; height: auto; vertical-align: middle; } </style> </head> <body> <div class="custom-wrapper custom-corner-all"> <header> <span class="custom-grandenet-logo custom-header-logo"></span> </header> <div class="table"> <span class="custom-icon custom-icon-spinner"></span> <p>Testing all possible frontends, please wait...</p> </div> <div class="custom-description"> <span class="custom-corner-all custom-icon custom-icon-user"></span> <span class="custom-icon custom-icon-forward"></span> <span class="custom-corner-all custom-grandenet-logo custom-icon-logo"></span> <span class="custom-icon custom-icon-forward"></span> <span class="custom-corner-all custom-icon custom-icon-harddrive"></span> </div> <div class=link></div> </div> </body> </html>