test(platform-server): add initial e2e tests for platform-server (#15061)
This commit is contained in:
parent
13686bb518
commit
ff60c041f6
|
@ -40,6 +40,7 @@ env:
|
|||
matrix:
|
||||
# Order: a slower build first, so that we don't occupy an idle travis worker waiting for others to complete.
|
||||
- CI_MODE=e2e
|
||||
- CI_MODE=e2e_2
|
||||
- CI_MODE=js
|
||||
- CI_MODE=saucelabs_required
|
||||
- CI_MODE=browserstack_required
|
||||
|
|
|
@ -0,0 +1,6 @@
|
|||
built/
|
||||
*/src/*.d.ts
|
||||
*/src/*.js
|
||||
**/*.ngfactory.ts
|
||||
**/*.ngsummary.json
|
||||
npm-debug.log
|
|
@ -0,0 +1,7 @@
|
|||
To add a new server side rendering E2E test
|
||||
|
||||
- Add a new server side rendered application to src/
|
||||
- Edit webpack.client.config.js to add new entry point for the new client bundle
|
||||
- The index.html can access the client bundle from /built/<bundle-name>.js
|
||||
- Edit src/server.ts to add the server side application to a new URL
|
||||
- Add a protractor test in e2e/ to test with the new URL
|
|
@ -0,0 +1,16 @@
|
|||
#!/usr/bin/env bash
|
||||
|
||||
set -eu -o pipefail
|
||||
|
||||
rm -rf built
|
||||
|
||||
ngc
|
||||
|
||||
# This is to mainlt copy the index.html to be packaged into the server.
|
||||
cp -r src/* built/src
|
||||
|
||||
# Bundle the server which hosts all the server side apps.
|
||||
webpack --config webpack.server.config.js
|
||||
|
||||
# Bundle the clients into individual bundles.
|
||||
webpack --config webpack.client.config.js
|
|
@ -0,0 +1,38 @@
|
|||
/**
|
||||
* @license
|
||||
* Copyright Google Inc. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by an MIT-style license that can be
|
||||
* found in the LICENSE file at https://angular.io/license
|
||||
*/
|
||||
|
||||
import {browser, by, element} from 'protractor';
|
||||
|
||||
import {verifyNoBrowserErrors} from './util';
|
||||
|
||||
describe('Hello world E2E Tests', function() {
|
||||
it('should display: Hello world!', function() {
|
||||
// Load the page without waiting for Angular since it is not boostrapped automatically.
|
||||
browser.driver.get(browser.baseUrl + 'helloworld');
|
||||
|
||||
const style = browser.driver.findElement(by.css('style[ng-transition="hlw"]'));
|
||||
expect(style.getText()).not.toBeNull();
|
||||
|
||||
// Test the contents from the server.
|
||||
const serverDiv = browser.driver.findElement(by.css('div'));
|
||||
expect(serverDiv.getText()).toEqual('Hello world!');
|
||||
|
||||
// Bootstrap the client side app.
|
||||
browser.executeScript('doBootstrap()');
|
||||
|
||||
// Retest the contents after the client bootstraps.
|
||||
expect(element(by.css('div')).getText()).toEqual('Hello world!');
|
||||
|
||||
// Make sure the server styles got replaced by client side ones.
|
||||
expect(element(by.css('style[ng-transition="hlw"]')).isPresent()).toBe(false);
|
||||
expect(element(by.css('style')).getText()).toBe('');
|
||||
|
||||
// Make sure there were no client side errors.
|
||||
verifyNoBrowserErrors();
|
||||
});
|
||||
});
|
|
@ -0,0 +1,22 @@
|
|||
/**
|
||||
* @license
|
||||
* Copyright Google Inc. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by an MIT-style license that can be
|
||||
* found in the LICENSE file at https://angular.io/license
|
||||
*/
|
||||
|
||||
exports.config = {
|
||||
specs: ['../built/e2e/*-spec.js'],
|
||||
capabilities: {
|
||||
browserName: 'chrome',
|
||||
chromeOptions: {
|
||||
args: ['--no-sandbox'],
|
||||
binary: process.env.CHROME_BIN,
|
||||
}
|
||||
},
|
||||
directConnect: true,
|
||||
baseUrl: 'http://localhost:9876/',
|
||||
framework: 'jasmine',
|
||||
useAllAngular2AppRoots: true
|
||||
};
|
|
@ -0,0 +1,8 @@
|
|||
{
|
||||
"compilerOptions": {
|
||||
"outDir": "../built/e2e",
|
||||
"types": ["jasmine"],
|
||||
// TODO(alexeagle): was required for Protractor 4.0.11
|
||||
"skipLibCheck": true
|
||||
}
|
||||
}
|
|
@ -0,0 +1,25 @@
|
|||
/**
|
||||
* @license
|
||||
* Copyright Google Inc. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by an MIT-style license that can be
|
||||
* found in the LICENSE file at https://angular.io/license
|
||||
*/
|
||||
/* tslint:disable:no-console */
|
||||
import * as webdriver from 'selenium-webdriver';
|
||||
declare var browser: any;
|
||||
declare var expect: any;
|
||||
|
||||
export function verifyNoBrowserErrors() {
|
||||
browser.manage().logs().get('browser').then(function(browserLog: any[]) {
|
||||
const errors: any[] = [];
|
||||
browserLog.filter(logEntry => {
|
||||
const msg = logEntry.message;
|
||||
console.log('>> ' + msg);
|
||||
if (logEntry.level.value >= webdriver.logging.Level.INFO.value) {
|
||||
errors.push(msg);
|
||||
}
|
||||
});
|
||||
expect(errors).toEqual([]);
|
||||
});
|
||||
}
|
|
@ -0,0 +1,41 @@
|
|||
{
|
||||
"name": "platform-server-integration",
|
||||
"version": "0.0.0",
|
||||
"license": "MIT",
|
||||
"description": "Integration tests for @angular/platform-server",
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "https://github.com/angular/angular.git"
|
||||
},
|
||||
"dependencies": {
|
||||
"@angular/common": "file:../../../dist/packages-dist/common",
|
||||
"@angular/compiler": "file:../../../dist/packages-dist/compiler",
|
||||
"@angular/compiler-cli": "file:../../../dist/packages-dist/compiler-cli",
|
||||
"@angular/core": "file:../../../dist/packages-dist/core",
|
||||
"@angular/http": "file:../../../dist/packages-dist/http",
|
||||
"@angular/platform-browser": "file:../../../dist/packages-dist/platform-browser",
|
||||
"@angular/platform-server": "file:../../../dist/packages-dist/platform-server",
|
||||
"express": "^4.14.1",
|
||||
"rxjs": "file:../../../node_modules/rxjs",
|
||||
"typescript": "2.1.6",
|
||||
"zone.js": "0.7.6"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/jasmine": "2.5.41",
|
||||
"babel-core": "^6.23.1",
|
||||
"babel-loader": "^6.4.0",
|
||||
"babel-preset-es2015": "^6.22.0",
|
||||
"concurrently": "3.1.0",
|
||||
"protractor": "file:../../../node_modules/protractor",
|
||||
"raw-loader": "^0.5.1",
|
||||
"webpack": "^2.2.1"
|
||||
},
|
||||
"scripts": {
|
||||
"postinstall": "webdriver-manager update",
|
||||
"build": "./build.sh",
|
||||
"test": "npm run build && concurrently \"npm run serve\" \"npm run protractor\" --kill-others --success first",
|
||||
"serve": "node built/server-bundle.js",
|
||||
"preprotractor": "tsc -p e2e",
|
||||
"protractor": "protractor e2e/protractor.config.js"
|
||||
}
|
||||
}
|
|
@ -0,0 +1,13 @@
|
|||
#!/usr/bin/env bash
|
||||
|
||||
set -eu -o pipefail
|
||||
|
||||
cd `dirname $0`
|
||||
|
||||
echo "#################################"
|
||||
echo "Running platform-server end to end tests"
|
||||
echo "#################################"
|
||||
|
||||
npm install
|
||||
|
||||
npm run test
|
|
@ -0,0 +1,20 @@
|
|||
/**
|
||||
* @license
|
||||
* Copyright Google Inc. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by an MIT-style license that can be
|
||||
* found in the LICENSE file at https://angular.io/license
|
||||
*/
|
||||
|
||||
import {NgModule} from '@angular/core';
|
||||
import {ServerModule} from '@angular/platform-server';
|
||||
|
||||
import {HelloWorldModule} from './app';
|
||||
import {HelloWorldComponent} from './hello-world.component';
|
||||
|
||||
@NgModule({
|
||||
bootstrap: [HelloWorldComponent],
|
||||
imports: [HelloWorldModule, ServerModule],
|
||||
})
|
||||
export class HelloWorldServerModule {
|
||||
}
|
|
@ -0,0 +1,20 @@
|
|||
/**
|
||||
* @license
|
||||
* Copyright Google Inc. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by an MIT-style license that can be
|
||||
* found in the LICENSE file at https://angular.io/license
|
||||
*/
|
||||
|
||||
import {NgModule} from '@angular/core';
|
||||
import {BrowserModule} from '@angular/platform-browser';
|
||||
|
||||
import {HelloWorldComponent} from './hello-world.component';
|
||||
|
||||
@NgModule({
|
||||
declarations: [HelloWorldComponent],
|
||||
bootstrap: [HelloWorldComponent],
|
||||
imports: [BrowserModule.withServerTransition({appId: 'hlw'})],
|
||||
})
|
||||
export class HelloWorldModule {
|
||||
}
|
|
@ -0,0 +1,17 @@
|
|||
/**
|
||||
* @license
|
||||
* Copyright Google Inc. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by an MIT-style license that can be
|
||||
* found in the LICENSE file at https://angular.io/license
|
||||
*/
|
||||
|
||||
import 'zone.js/dist/zone.js';
|
||||
|
||||
import {enableProdMode} from '@angular/core';
|
||||
import {platformBrowser} from '@angular/platform-browser';
|
||||
import {HelloWorldModuleNgFactory} from './app.ngfactory';
|
||||
|
||||
window['doBootstrap'] = function() {
|
||||
platformBrowser().bootstrapModuleFactory(HelloWorldModuleNgFactory);
|
||||
};
|
|
@ -0,0 +1,24 @@
|
|||
/**
|
||||
* @license
|
||||
* Copyright Google Inc. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by an MIT-style license that can be
|
||||
* found in the LICENSE file at https://angular.io/license
|
||||
*/
|
||||
|
||||
import {Component} from '@angular/core';
|
||||
|
||||
@Component({
|
||||
selector: 'hello-world-app',
|
||||
template: `
|
||||
<div>Hello {{ name }}!</div>
|
||||
`,
|
||||
styles: [`
|
||||
div {
|
||||
font-weight: bold;
|
||||
}
|
||||
`]
|
||||
})
|
||||
export class HelloWorldComponent {
|
||||
name: string = 'world';
|
||||
}
|
|
@ -0,0 +1,10 @@
|
|||
<html>
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>Hello World</title>
|
||||
<script src="built/helloworld-bundle.js"></script>
|
||||
</head>
|
||||
<body>
|
||||
<hello-world-app></hello-world-app>
|
||||
</body>
|
||||
</html>
|
|
@ -0,0 +1,40 @@
|
|||
/**
|
||||
* @license
|
||||
* Copyright Google Inc. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by an MIT-style license that can be
|
||||
* found in the LICENSE file at https://angular.io/license
|
||||
*/
|
||||
/* tslint:disable:no-console */
|
||||
require('zone.js/dist/zone-node.js');
|
||||
|
||||
import {enableProdMode, NgModuleFactory} from '@angular/core';
|
||||
import {renderModuleFactory} from '@angular/platform-server';
|
||||
import * as express from 'express';
|
||||
|
||||
import {HelloWorldServerModuleNgFactory} from './helloworld/app.server.ngfactory';
|
||||
const helloworld = require('raw-loader!./helloworld/index.html');
|
||||
|
||||
const app = express();
|
||||
|
||||
function render<T>(moduleFactory: NgModuleFactory<T>, html: string) {
|
||||
return (req, res) => {
|
||||
renderModuleFactory(moduleFactory, {
|
||||
document: html,
|
||||
url: req.url,
|
||||
}).then((response) => { res.send(response); });
|
||||
};
|
||||
}
|
||||
|
||||
enableProdMode();
|
||||
|
||||
// Client bundles will be statically served from the built/ directory.
|
||||
app.use('/built', express.static('built'));
|
||||
|
||||
// Keep the browser logs free of errors.
|
||||
app.get('/favicon.ico', (req, res) => { res.send(''); });
|
||||
|
||||
//-----------ADD YOUR SERVER SIDE RENDERED APP HERE ----------------------
|
||||
app.get('/helloworld', render(HelloWorldServerModuleNgFactory, helloworld));
|
||||
|
||||
app.listen(9876, function() { console.log('Server listening on port 9876!'); });
|
|
@ -0,0 +1,27 @@
|
|||
{
|
||||
"angularCompilerOptions": {
|
||||
"annotationsAs": "static fields",
|
||||
"annotateForClosureCompiler": true
|
||||
},
|
||||
|
||||
"compilerOptions": {
|
||||
"module": "es2015",
|
||||
"moduleResolution": "node",
|
||||
"strictNullChecks": true,
|
||||
"target": "es6",
|
||||
"noImplicitAny": false,
|
||||
"sourceMap": false,
|
||||
"experimentalDecorators": true,
|
||||
"outDir": "built/src",
|
||||
"declaration": true,
|
||||
"typeRoots": ["node_modules/@types"]
|
||||
},
|
||||
|
||||
"exclude": [
|
||||
"vendor",
|
||||
"node_modules",
|
||||
"built",
|
||||
"dist",
|
||||
"e2e"
|
||||
]
|
||||
}
|
|
@ -0,0 +1,18 @@
|
|||
/**
|
||||
* @license
|
||||
* Copyright Google Inc. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by an MIT-style license that can be
|
||||
* found in the LICENSE file at https://angular.io/license
|
||||
*/
|
||||
|
||||
const path = require('path');
|
||||
|
||||
module.exports = {
|
||||
entry: {
|
||||
helloworld: './built/src/helloworld/client.js',
|
||||
},
|
||||
output: {path: path.join(__dirname, 'built'), filename: '[name]-bundle.js'},
|
||||
module: {loaders: [{test: /\.js$/, loader: 'babel-loader?presets[]=es2015'}]},
|
||||
resolve: {extensions: ['.js']}
|
||||
};
|
|
@ -0,0 +1,14 @@
|
|||
/**
|
||||
* @license
|
||||
* Copyright Google Inc. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by an MIT-style license that can be
|
||||
* found in the LICENSE file at https://angular.io/license
|
||||
*/
|
||||
|
||||
module.exports = {
|
||||
target: 'node',
|
||||
entry: './built/src/server.js',
|
||||
output: {filename: './built/server-bundle.js'},
|
||||
resolve: {extensions: ['.js']},
|
||||
};
|
|
@ -26,6 +26,7 @@
|
|||
"types": ["angularjs"]
|
||||
},
|
||||
"exclude": [
|
||||
"compiler-cli/integrationtest"
|
||||
"compiler-cli/integrationtest",
|
||||
"platform-server/integrationtest"
|
||||
]
|
||||
}
|
||||
|
|
|
@ -38,7 +38,7 @@ travisFoldEnd "tsc a bunch of useless stuff"
|
|||
|
||||
|
||||
# Build integration tests
|
||||
if [[ ${CI_MODE:-} == "e2e" ]]; then
|
||||
if [[ ${CI_MODE:-} == "e2e_2" ]]; then
|
||||
travisFoldStart "build.integration"
|
||||
cd "`dirname $0`/../../integration"
|
||||
./build_rxjs_es6.sh
|
||||
|
|
|
@ -41,7 +41,7 @@ travisFoldStart "npm-install"
|
|||
travisFoldEnd "npm-install"
|
||||
|
||||
|
||||
if [[ ${TRAVIS} && (${CI_MODE} == "e2e" || ${CI_MODE} == "aio" || ${CI_MODE} == "docs_test") ]]; then
|
||||
if [[ ${TRAVIS} && (${CI_MODE} == "e2e" || ${CI_MODE} == "e2e_2" || ${CI_MODE} == "aio" || ${CI_MODE} == "docs_test") ]]; then
|
||||
# Install version of yarn that we are locked against
|
||||
travisFoldStart "install-yarn"
|
||||
curl -o- -L https://yarnpkg.com/install.sh | bash -s -- --version "${YARN_VERSION}"
|
||||
|
@ -61,7 +61,7 @@ fi
|
|||
|
||||
|
||||
# Install Chromium
|
||||
if [[ ${CI_MODE} == "js" || ${CI_MODE} == "e2e" || ${CI_MODE} == "aio" ]]; then
|
||||
if [[ ${CI_MODE} == "js" || ${CI_MODE} == "e2e" || ${CI_MODE} == "e2e_2" || ${CI_MODE} == "aio" ]]; then
|
||||
travisFoldStart "install-chromium"
|
||||
(
|
||||
${thisDir}/install-chromium.sh
|
||||
|
|
|
@ -0,0 +1,36 @@
|
|||
#!/usr/bin/env bash
|
||||
|
||||
# Second shard for the e2e tests. Balance it with runtime of test-e2e.sh
|
||||
|
||||
set -u -e -o pipefail
|
||||
|
||||
# Setup environment
|
||||
readonly thisDir=$(cd $(dirname $0); pwd)
|
||||
source ${thisDir}/_travis-fold.sh
|
||||
|
||||
|
||||
travisFoldStart "test.e2e.buildPackages"
|
||||
./build.sh
|
||||
travisFoldEnd "test.e2e.buildPackages"
|
||||
|
||||
|
||||
if [[ ${TRAVIS:-} ]]; then
|
||||
travisFoldStart "test.e2e.xvfb-start"
|
||||
sh -e /etc/init.d/xvfb start
|
||||
travisFoldEnd "test.e2e.xvfb-start"
|
||||
fi
|
||||
|
||||
|
||||
travisFoldStart "test.e2e.integration"
|
||||
./integration/run_tests.sh
|
||||
travisFoldEnd "test.e2e.integration"
|
||||
|
||||
|
||||
travisFoldStart "test.e2e.offlineCompiler"
|
||||
#TODO(alexeagle): move offline_compiler_test to integration/
|
||||
${thisDir}/offline_compiler_test.sh
|
||||
travisFoldEnd "test.e2e.offlineCompiler"
|
||||
|
||||
travisFoldStart "test.e2e.platform-server"
|
||||
./packages/platform-server/integrationtest/run_tests.sh
|
||||
travisFoldEnd "test.e2e.platform-server"
|
|
@ -1,5 +1,7 @@
|
|||
#!/usr/bin/env bash
|
||||
|
||||
# First shard for the e2e tests. Balance it with runtime of test-e2e-2.sh
|
||||
|
||||
set -u -e -o pipefail
|
||||
|
||||
# Setup environment
|
||||
|
@ -18,18 +20,6 @@ if [[ ${TRAVIS:-} ]]; then
|
|||
travisFoldEnd "test.e2e.xvfb-start"
|
||||
fi
|
||||
|
||||
|
||||
travisFoldStart "test.e2e.integration"
|
||||
./integration/run_tests.sh
|
||||
travisFoldEnd "test.e2e.integration"
|
||||
|
||||
|
||||
travisFoldStart "test.e2e.offlineCompiler"
|
||||
#TODO(alexeagle): move offline_compiler_test to integration/
|
||||
${thisDir}/offline_compiler_test.sh
|
||||
travisFoldEnd "test.e2e.offlineCompiler"
|
||||
|
||||
|
||||
travisFoldStart "test.e2e.publicApi"
|
||||
$(npm bin)/gulp public-api:enforce
|
||||
travisFoldEnd "test.e2e.publicApi"
|
||||
|
|
|
@ -22,6 +22,9 @@ case ${CI_MODE} in
|
|||
e2e)
|
||||
${thisDir}/test-e2e.sh
|
||||
;;
|
||||
e2e_2)
|
||||
${thisDir}/test-e2e-2.sh
|
||||
;;
|
||||
saucelabs_required)
|
||||
${thisDir}/test-saucelabs.sh
|
||||
;;
|
||||
|
|
Loading…
Reference in New Issue