// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
/**
* @fileoverview
* @suppress {checkTypes} By default, JSCompile is not run on test files.
* However, you can modify |remoting_webapp_files.gypi| locally to include
* the test in the package to expedite local development. This suppress
* is here so that JSCompile won't complain.
*
* Provides basic functionality for JavaScript based browser test.
*
* To define a browser test, create a class under the browserTest namespace.
* You can pass arbitrary object literals to the browser test from the C++ test
* harness as the test data. Each browser test class should implement the run
* method.
* For example:
*
* browserTest.My_Test = function() {};
* browserTest.My_Test.prototype.run(myObjectLiteral) = function() { ... };
*
* The browser test is async in nature. It will keep running until
* browserTest.fail("My error message.") or browserTest.pass() is called.
*
* For example:
*
* browserTest.My_Test.prototype.run(myObjectLiteral) = function() {
* window.setTimeout(function() {
* if (doSomething(myObjectLiteral)) {
* browserTest.pass();
* } else {
* browserTest.fail('My error message.');
* }
* }, 1000);
* };
*
* You will then invoke the test in C++ by calling:
*
* RunJavaScriptTest(web_content, "My_Test", "{"
* "pin: '123123'"
* "}");
*/
'use strict';
var browserTest = {};
browserTest.init = function() {
// The domAutomationController is used to communicate progress back to the
// C++ calling code. It will only exist if chrome is run with the flag
// --dom-automation. It is stubbed out here so that browser test can be run
// under the regular app.
browserTest.automationController_ = window.domAutomationController || {
send: function(json) {
var result = JSON.parse(json);
if (result.succeeded) {
console.log('Test Passed.');
} else {
console.error('Test Failed.\n' +
result.error_message + '\n' + result.stack_trace);
}
}
};
};
browserTest.expect = function(expr, message) {
if (!expr) {
message = (message) ? '<' + message + '>' : '';
browserTest.fail('Expectation failed.' + message);
}
};
browserTest.fail = function(error) {
var error_message = error;
var stack_trace = base.debug.callstack();
if (error instanceof Error) {
error_message = error.toString();
stack_trace = error.stack;
}
// To run browserTest locally:
// 1. Go to |remoting_webapp_files| and look for
// |remoting_webapp_js_browser_test_files| and uncomment it
// 2. gclient runhooks
// 3. rebuild the webapp
// 4. Run it in the console browserTest.runTest(browserTest.MyTest, {});
// 5. The line below will trap the test in the debugger in case of
// failure.
debugger;
browserTest.automationController_.send(JSON.stringify({
succeeded: false,
error_message: error_message,
stack_trace: stack_trace
}));
};
browserTest.pass = function() {
browserTest.automationController_.send(JSON.stringify({
succeeded: true,
error_message: '',
stack_trace: ''
}));
};
browserTest.clickOnControl = function(id) {
var element = document.getElementById(id);
browserTest.expect(element);
element.click();
};
/** @enum {number} */
browserTest.Timeout = {
NONE: -1,
DEFAULT: 5000
};
browserTest.onUIMode = function(expectedMode, opt_timeout) {
if (expectedMode == remoting.currentMode) {
// If the current mode is the same as the expected mode, return a fulfilled
// promise. For some reason, if we fulfill the promise in the same
// callstack, V8 will assert at V8RecursionScope.h(66) with
// ASSERT(!ScriptForbiddenScope::isScriptForbidden()).
// To avoid the assert, execute the callback in a different callstack.
return base.Promise.sleep(0);
}
return new Promise (function(fulfill, reject) {
var uiModeChanged = remoting.testEvents.Names.uiModeChanged;
var timerId = null;
if (opt_timeout === undefined) {
opt_timeout = browserTest.Timeout.DEFAULT;
}
function onTimeout() {
remoting.testEvents.removeEventListener(uiModeChanged, onUIModeChanged);
reject('Timeout waiting for ' + expectedMode);
}
function onUIModeChanged(mode) {
if (mode == expectedMode) {
remoting.testEvents.removeEventListener(uiModeChanged, onUIModeChanged);
window.clearTimeout(timerId);
timerId = null;
fulfill();
}
}
if (opt_timeout != browserTest.Timeout.NONE) {
timerId = window.setTimeout(onTimeout, opt_timeout);
}
remoting.testEvents.addEventListener(uiModeChanged, onUIModeChanged);
});
};
browserTest.expectMe2MeError = function(errorTag) {
var AppMode = remoting.AppMode;
var Timeout = browserTest.Timeout;
var onConnected = browserTest.onUIMode(AppMode.IN_SESSION, Timeout.None);
var onFailure = browserTest.onUIMode(AppMode.CLIENT_CONNECT_FAILED_ME2ME);
onConnected = onConnected.then(function() {
return Promise.reject(
'Expected the Me2Me connection to fail.');
});
onFailure = onFailure.then(function() {
var errorDiv = document.getElementById('connect-error-message');
var actual = errorDiv.innerText;
var expected = l10n.getTranslationOrError(errorTag);
browserTest.clickOnControl('client-finished-me2me-button');
if (actual != expected) {
return Promise.reject('Unexpected failure. actual:' + actual +
' expected:' + expected);
}
});
return Promise.race([onConnected, onFailure]);
};
browserTest.expectMe2MeConnected = function() {
var AppMode = remoting.AppMode;
// Timeout if the session is not connected within 30 seconds.
var SESSION_CONNECTION_TIMEOUT = 30000;
var onConnected = browserTest.onUIMode(AppMode.IN_SESSION,
SESSION_CONNECTION_TIMEOUT);
var onFailure = browserTest.onUIMode(AppMode.CLIENT_CONNECT_FAILED_ME2ME,
browserTest.Timeout.NONE);
onFailure = onFailure.then(function() {
var errorDiv = document.getElementById('connect-error-message');
var errorMsg = errorDiv.innerText;
return Promise.reject('Unexpected error - ' + errorMsg);
});
return Promise.race([onConnected, onFailure]);
};
browserTest.runTest = function(testClass, data) {
try {
var test = new testClass();
browserTest.expect(typeof test.run == 'function');
test.run(data);
} catch (e) {
browserTest.fail(e);
}
};
browserTest.init();