205 lines
37 KiB
JavaScript
205 lines
37 KiB
JavaScript
"use strict";
|
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
};
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
const load_assets_1 = __importDefault(require("../../load-assets"));
|
|
const http_1 = require("../../utils/http");
|
|
const remotes_queue_1 = __importDefault(require("./remotes-queue"));
|
|
const testcafe_hammerhead_1 = require("testcafe-hammerhead");
|
|
const service_routes_1 = __importDefault(require("./service-routes"));
|
|
const empty_page_markup_1 = __importDefault(require("../../proxyless/empty-page-markup"));
|
|
const error_route_1 = __importDefault(require("../../proxyless/error-route"));
|
|
class BrowserConnectionGateway {
|
|
constructor(proxy, options) {
|
|
this._connections = {};
|
|
this._remotesQueue = new remotes_queue_1.default();
|
|
this.connectUrl = proxy.resolveRelativeServiceUrl(service_routes_1.default.connect);
|
|
this.retryTestPages = options.retryTestPages;
|
|
this.proxyless = options.proxyless;
|
|
this.proxy = proxy;
|
|
this._registerRoutes(proxy);
|
|
}
|
|
_dispatch(url, proxy, handler, method = 'GET', shouldAcceptCrossOrigin) {
|
|
// @ts-ignore Need to improve typings of the 'testcafe-hammerhead' module
|
|
proxy[method](url, (req, res, serverInfo, params) => {
|
|
const connection = this._connections[params.id];
|
|
(0, http_1.preventCaching)(res);
|
|
if (shouldAcceptCrossOrigin)
|
|
(0, testcafe_hammerhead_1.acceptCrossOrigin)(res);
|
|
if (connection)
|
|
handler(req, res, connection);
|
|
else
|
|
(0, http_1.respond404)(res);
|
|
});
|
|
}
|
|
_registerRoutes(proxy) {
|
|
const { idlePageScript, idlePageStyle, idlePageLogo, serviceWorkerScript, } = (0, load_assets_1.default)();
|
|
this._dispatch(`${service_routes_1.default.connect}/{id}`, proxy, BrowserConnectionGateway._onConnection);
|
|
this._dispatch(`${service_routes_1.default.heartbeat}/{id}`, proxy, BrowserConnectionGateway._onHeartbeat, 'GET', this.proxyless);
|
|
this._dispatch(`${service_routes_1.default.idle}/{id}`, proxy, BrowserConnectionGateway._onIdle);
|
|
this._dispatch(`${service_routes_1.default.idleForced}/{id}`, proxy, BrowserConnectionGateway._onIdleForced);
|
|
this._dispatch(`${service_routes_1.default.status}/{id}`, proxy, BrowserConnectionGateway._onStatusRequest);
|
|
this._dispatch(`${service_routes_1.default.statusDone}/{id}`, proxy, BrowserConnectionGateway._onStatusRequestOnTestDone, 'GET', this.proxyless);
|
|
this._dispatch(`${service_routes_1.default.initScript}/{id}`, proxy, BrowserConnectionGateway._onInitScriptRequest);
|
|
this._dispatch(`${service_routes_1.default.initScript}/{id}`, proxy, BrowserConnectionGateway._onInitScriptResponse, 'POST');
|
|
this._dispatch(`${service_routes_1.default.activeWindowId}/{id}`, proxy, BrowserConnectionGateway._onGetActiveWindowIdRequest, 'GET', this.proxyless);
|
|
this._dispatch(`${service_routes_1.default.activeWindowId}/{id}`, proxy, BrowserConnectionGateway._onSetActiveWindowIdRequest, 'POST', this.proxyless);
|
|
this._dispatch(`${service_routes_1.default.closeWindow}/{id}`, proxy, BrowserConnectionGateway._onCloseWindowRequest, 'POST');
|
|
this._dispatch(`${service_routes_1.default.openFileProtocol}/{id}`, proxy, BrowserConnectionGateway._onOpenFileProtocolRequest, 'POST');
|
|
this._dispatch(`${service_routes_1.default.dispatchProxylessEvent}/{id}`, proxy, BrowserConnectionGateway._onDispatchProxylessEvent, 'POST', this.proxyless);
|
|
proxy.GET(service_routes_1.default.connect, (req, res) => this._connectNextRemoteBrowser(req, res));
|
|
proxy.GET(service_routes_1.default.connectWithTrailingSlash, (req, res) => this._connectNextRemoteBrowser(req, res));
|
|
proxy.GET(service_routes_1.default.serviceWorker, { content: serviceWorkerScript, contentType: 'application/x-javascript' });
|
|
proxy.GET(service_routes_1.default.assets.index, { content: idlePageScript, contentType: 'application/x-javascript' });
|
|
proxy.GET(service_routes_1.default.assets.styles, { content: idlePageStyle, contentType: 'text/css' });
|
|
proxy.GET(service_routes_1.default.assets.logo, { content: idlePageLogo, contentType: 'image/svg+xml' });
|
|
if (this.proxyless) {
|
|
proxy.GET(error_route_1.default, (req, res) => {
|
|
res.writeHead(200, { 'Content-Type': 'text/html' });
|
|
res.end(empty_page_markup_1.default);
|
|
});
|
|
}
|
|
}
|
|
// Helpers
|
|
static _ensureConnectionReady(res, connection) {
|
|
if (!connection.isReady()) {
|
|
(0, http_1.respond500)(res, 'The connection is not ready yet.');
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
static _fetchRequestData(req, callback) {
|
|
let data = '';
|
|
req.on('data', chunk => {
|
|
data += chunk;
|
|
});
|
|
req.on('end', () => {
|
|
callback(data.toString());
|
|
});
|
|
}
|
|
// Route handlers
|
|
static async _onConnection(req, res, connection) {
|
|
if (connection.isReady())
|
|
(0, http_1.respond500)(res, 'The connection is already established.');
|
|
else {
|
|
const userAgent = req.headers['user-agent'];
|
|
await connection.establish(userAgent);
|
|
(0, http_1.redirect)(res, connection.idleUrl);
|
|
}
|
|
}
|
|
static _onHeartbeat(req, res, connection) {
|
|
if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
|
|
const status = connection.heartbeat();
|
|
(0, http_1.respondWithJSON)(res, status);
|
|
}
|
|
}
|
|
static _onIdle(req, res, connection) {
|
|
if (BrowserConnectionGateway._ensureConnectionReady(res, connection))
|
|
res.end(connection.renderIdlePage());
|
|
}
|
|
static async _onIdleForced(req, res, connection) {
|
|
if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
|
|
const status = await connection.getStatus(true);
|
|
(0, http_1.redirect)(res, status.url);
|
|
}
|
|
}
|
|
static async _onStatusRequest(req, res, connection) {
|
|
return BrowserConnectionGateway._onStatusRequestCore(req, res, connection, false);
|
|
}
|
|
static async _onStatusRequestOnTestDone(req, res, connection) {
|
|
return BrowserConnectionGateway._onStatusRequestCore(req, res, connection, true);
|
|
}
|
|
static async _onStatusRequestCore(req, res, connection, isTestDone) {
|
|
if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
|
|
const status = await connection.getStatus(isTestDone);
|
|
(0, http_1.respondWithJSON)(res, status);
|
|
}
|
|
}
|
|
static _onInitScriptRequest(req, res, connection) {
|
|
if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
|
|
const script = connection.getInitScript();
|
|
(0, http_1.respondWithJSON)(res, script);
|
|
}
|
|
}
|
|
static _onInitScriptResponse(req, res, connection) {
|
|
if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
|
|
BrowserConnectionGateway._fetchRequestData(req, data => {
|
|
connection.handleInitScriptResult(data);
|
|
res.end();
|
|
});
|
|
}
|
|
}
|
|
static _onGetActiveWindowIdRequest(req, res, connection) {
|
|
if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
|
|
(0, http_1.respondWithJSON)(res, {
|
|
activeWindowId: connection.activeWindowId,
|
|
});
|
|
}
|
|
}
|
|
static _onSetActiveWindowIdRequest(req, res, connection) {
|
|
if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
|
|
BrowserConnectionGateway._fetchRequestData(req, data => {
|
|
const parsedData = JSON.parse(data);
|
|
connection.activeWindowId = parsedData.windowId;
|
|
(0, http_1.respondWithJSON)(res);
|
|
});
|
|
}
|
|
}
|
|
static _onCloseWindowRequest(req, res, connection) {
|
|
if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
|
|
connection.provider.closeBrowserChildWindow(connection.id)
|
|
.then(() => {
|
|
(0, http_1.respondWithJSON)(res);
|
|
});
|
|
}
|
|
}
|
|
static _onOpenFileProtocolRequest(req, res, connection) {
|
|
if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
|
|
BrowserConnectionGateway._fetchRequestData(req, data => {
|
|
const parsedData = JSON.parse(data);
|
|
connection.openFileProtocol(parsedData.url)
|
|
.then(() => {
|
|
(0, http_1.respondWithJSON)(res);
|
|
});
|
|
});
|
|
}
|
|
}
|
|
static _onDispatchProxylessEvent(req, res, connection) {
|
|
if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {
|
|
BrowserConnectionGateway._fetchRequestData(req, data => {
|
|
const { type, options } = JSON.parse(data);
|
|
connection.dispatchProxylessEvent(type, options)
|
|
.then(() => {
|
|
(0, http_1.respondWithJSON)(res);
|
|
});
|
|
});
|
|
}
|
|
}
|
|
async _connectNextRemoteBrowser(req, res) {
|
|
(0, http_1.preventCaching)(res);
|
|
const remoteConnection = await this._remotesQueue.shift();
|
|
if (remoteConnection)
|
|
(0, http_1.redirect)(res, remoteConnection.url);
|
|
else
|
|
(0, http_1.respond500)(res, 'There are no available _connections to establish.');
|
|
}
|
|
// API
|
|
startServingConnection(connection) {
|
|
this._connections[connection.id] = connection;
|
|
if (connection.browserInfo.providerName === 'remote')
|
|
this._remotesQueue.add(connection);
|
|
}
|
|
stopServingConnection(connection) {
|
|
delete this._connections[connection.id];
|
|
if (connection.browserInfo.providerName === 'remote')
|
|
this._remotesQueue.remove(connection);
|
|
}
|
|
async close() {
|
|
for (const id in this._connections)
|
|
await this._connections[id].close();
|
|
}
|
|
}
|
|
exports.default = BrowserConnectionGateway;
|
|
module.exports = exports.default;
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"gateway.js","sourceRoot":"","sources":["../../../src/browser/connection/gateway.ts"],"names":[],"mappings":"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","sourcesContent":["import loadAssets from '../../load-assets';\nimport {\n    respond404,\n    respond500,\n    respondWithJSON,\n    redirect,\n    preventCaching,\n} from '../../utils/http';\n\nimport RemotesQueue from './remotes-queue';\nimport { Proxy, acceptCrossOrigin } from 'testcafe-hammerhead';\nimport { Dictionary } from '../../configuration/interfaces';\nimport BrowserConnection from './index';\nimport { IncomingMessage, ServerResponse } from 'http';\nimport SERVICE_ROUTES from './service-routes';\nimport EMPTY_PAGE_MARKUP from '../../proxyless/empty-page-markup';\nimport PROXYLESS_ERROR_ROUTE from '../../proxyless/error-route';\n\nexport default class BrowserConnectionGateway {\n    private _connections: Dictionary<BrowserConnection> = {};\n    private _remotesQueue: RemotesQueue;\n    public readonly connectUrl: string;\n    public retryTestPages: boolean;\n    private readonly proxyless: boolean;\n    public readonly proxy: Proxy;\n\n    public constructor (proxy: Proxy, options: { retryTestPages: boolean; proxyless: boolean }) {\n        this._remotesQueue   = new RemotesQueue();\n        this.connectUrl      = proxy.resolveRelativeServiceUrl(SERVICE_ROUTES.connect);\n        this.retryTestPages  = options.retryTestPages;\n        this.proxyless       = options.proxyless;\n        this.proxy           = proxy;\n\n        this._registerRoutes(proxy);\n    }\n\n    private _dispatch (url: string, proxy: Proxy, handler: Function, method = 'GET', shouldAcceptCrossOrigin?: boolean): void {\n        // @ts-ignore Need to improve typings of the 'testcafe-hammerhead' module\n        proxy[method](url, (req: IncomingMessage, res: ServerResponse, serverInfo, params: Dictionary<string>) => {\n            const connection = this._connections[params.id];\n\n            preventCaching(res);\n\n            if (shouldAcceptCrossOrigin)\n                acceptCrossOrigin(res);\n\n            if (connection)\n                handler(req, res, connection);\n            else\n                respond404(res);\n        });\n    }\n\n    private _registerRoutes (proxy: Proxy): void {\n        const {\n            idlePageScript,\n            idlePageStyle,\n            idlePageLogo,\n            serviceWorkerScript,\n        } = loadAssets();\n\n        this._dispatch(`${SERVICE_ROUTES.connect}/{id}`, proxy, BrowserConnectionGateway._onConnection);\n        this._dispatch(`${SERVICE_ROUTES.heartbeat}/{id}`, proxy, BrowserConnectionGateway._onHeartbeat, 'GET', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.idle}/{id}`, proxy, BrowserConnectionGateway._onIdle);\n        this._dispatch(`${SERVICE_ROUTES.idleForced}/{id}`, proxy, BrowserConnectionGateway._onIdleForced);\n        this._dispatch(`${SERVICE_ROUTES.status}/{id}`, proxy, BrowserConnectionGateway._onStatusRequest);\n        this._dispatch(`${SERVICE_ROUTES.statusDone}/{id}`, proxy, BrowserConnectionGateway._onStatusRequestOnTestDone, 'GET', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.initScript}/{id}`, proxy, BrowserConnectionGateway._onInitScriptRequest);\n        this._dispatch(`${SERVICE_ROUTES.initScript}/{id}`, proxy, BrowserConnectionGateway._onInitScriptResponse, 'POST');\n        this._dispatch(`${SERVICE_ROUTES.activeWindowId}/{id}`, proxy, BrowserConnectionGateway._onGetActiveWindowIdRequest, 'GET', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.activeWindowId}/{id}`, proxy, BrowserConnectionGateway._onSetActiveWindowIdRequest, 'POST', this.proxyless);\n        this._dispatch(`${SERVICE_ROUTES.closeWindow}/{id}`, proxy, BrowserConnectionGateway._onCloseWindowRequest, 'POST');\n        this._dispatch(`${SERVICE_ROUTES.openFileProtocol}/{id}`, proxy, BrowserConnectionGateway._onOpenFileProtocolRequest, 'POST');\n        this._dispatch(`${SERVICE_ROUTES.dispatchProxylessEvent}/{id}`, proxy, BrowserConnectionGateway._onDispatchProxylessEvent, 'POST', this.proxyless);\n\n        proxy.GET(SERVICE_ROUTES.connect, (req: IncomingMessage, res: ServerResponse) => this._connectNextRemoteBrowser(req, res));\n        proxy.GET(SERVICE_ROUTES.connectWithTrailingSlash, (req: IncomingMessage, res: ServerResponse) => this._connectNextRemoteBrowser(req, res));\n\n        proxy.GET(SERVICE_ROUTES.serviceWorker, { content: serviceWorkerScript, contentType: 'application/x-javascript' });\n        proxy.GET(SERVICE_ROUTES.assets.index, { content: idlePageScript, contentType: 'application/x-javascript' });\n        proxy.GET(SERVICE_ROUTES.assets.styles, { content: idlePageStyle, contentType: 'text/css' });\n        proxy.GET(SERVICE_ROUTES.assets.logo, { content: idlePageLogo, contentType: 'image/svg+xml' });\n\n        if (this.proxyless) {\n            proxy.GET(PROXYLESS_ERROR_ROUTE, (req: IncomingMessage, res: ServerResponse) => {\n                res.writeHead(200, { 'Content-Type': 'text/html' });\n                res.end(EMPTY_PAGE_MARKUP);\n            });\n        }\n    }\n\n    // Helpers\n    private static _ensureConnectionReady (res: ServerResponse, connection: BrowserConnection): boolean {\n        if (!connection.isReady()) {\n            respond500(res, 'The connection is not ready yet.');\n            return false;\n        }\n\n        return true;\n    }\n\n    private static _fetchRequestData (req: IncomingMessage, callback: (data: string) => void): void {\n        let data = '';\n\n        req.on('data', chunk => {\n            data += chunk;\n        });\n\n        req.on('end', () => {\n            callback(data.toString());\n        });\n    }\n\n    // Route handlers\n    private static async _onConnection (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        if (connection.isReady())\n            respond500(res, 'The connection is already established.');\n\n        else {\n            const userAgent = req.headers['user-agent'] as string;\n\n            await connection.establish(userAgent);\n            redirect(res, connection.idleUrl);\n        }\n    }\n\n    private static _onHeartbeat (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const status = connection.heartbeat();\n\n            respondWithJSON(res, status);\n        }\n    }\n\n    private static _onIdle (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection))\n            res.end(connection.renderIdlePage());\n    }\n\n    private static async _onIdleForced (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const status = await connection.getStatus(true);\n\n            redirect(res, status.url);\n        }\n    }\n\n    private static async _onStatusRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        return BrowserConnectionGateway._onStatusRequestCore(req, res, connection, false);\n    }\n\n    private static async _onStatusRequestOnTestDone (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): Promise<void> {\n        return BrowserConnectionGateway._onStatusRequestCore(req, res, connection, true);\n    }\n\n    private static async _onStatusRequestCore (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection, isTestDone: boolean): Promise<void> {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const status = await connection.getStatus(isTestDone);\n\n            respondWithJSON(res, status);\n        }\n    }\n\n    private static _onInitScriptRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            const script = connection.getInitScript();\n\n            respondWithJSON(res, script);\n        }\n    }\n\n    private static _onInitScriptResponse (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                connection.handleInitScriptResult(data);\n\n                res.end();\n            });\n        }\n    }\n\n    private static _onGetActiveWindowIdRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            respondWithJSON(res, {\n                activeWindowId: connection.activeWindowId,\n            });\n        }\n    }\n\n    private static _onSetActiveWindowIdRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                const parsedData = JSON.parse(data);\n\n                connection.activeWindowId = parsedData.windowId;\n\n                respondWithJSON(res);\n            });\n        }\n    }\n\n    private static _onCloseWindowRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            connection.provider.closeBrowserChildWindow(connection.id)\n                .then(() => {\n                    respondWithJSON(res);\n                });\n        }\n    }\n\n    private static _onOpenFileProtocolRequest (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                const parsedData = JSON.parse(data);\n\n                connection.openFileProtocol(parsedData.url)\n                    .then(() => {\n                        respondWithJSON(res);\n                    });\n            });\n        }\n    }\n\n    private static _onDispatchProxylessEvent (req: IncomingMessage, res: ServerResponse, connection: BrowserConnection): void {\n        if (BrowserConnectionGateway._ensureConnectionReady(res, connection)) {\n            BrowserConnectionGateway._fetchRequestData(req, data => {\n                const { type, options } = JSON.parse(data);\n\n                connection.dispatchProxylessEvent(type, options)\n                    .then(() => {\n                        respondWithJSON(res);\n                    });\n            });\n        }\n    }\n\n    private async _connectNextRemoteBrowser (req: IncomingMessage, res: ServerResponse): Promise<void> {\n        preventCaching(res);\n\n        const remoteConnection = await this._remotesQueue.shift();\n\n        if (remoteConnection)\n            redirect(res, remoteConnection.url);\n        else\n            respond500(res, 'There are no available _connections to establish.');\n    }\n\n    // API\n    public startServingConnection (connection: BrowserConnection): void {\n        this._connections[connection.id] = connection;\n\n        if (connection.browserInfo.providerName === 'remote')\n            this._remotesQueue.add(connection);\n    }\n\n    public stopServingConnection (connection: BrowserConnection): void {\n        delete this._connections[connection.id];\n\n        if (connection.browserInfo.providerName === 'remote')\n            this._remotesQueue.remove(connection);\n    }\n\n    public async close (): Promise<void> {\n        for (const id in this._connections)\n            await this._connections[id].close();\n    }\n}\n\n"]}
|