Qual é o melhor método para agrupar Angular (versão 2, 4, 6, ...) para produção em um servidor web ao vivo.
Por favor, inclua a versão Angular nas respostas para que possamos acompanhar melhor quando ela se mover para lançamentos posteriores.
2.x, 4.x, 5.x, 6.x, 7.x, 8.x
(TypeScript) com CLI Angularnpm install -g @angular/cli
ng build --prod
(executar em linha de comando quando o diretório é projectFolder
)
flag prod
bundle for production (ver a Documentação angular para a lista de opções incluídas com a bandeira de produção).
Comprima usando compressão Brotli os recursos usando o seguinte comando
"for i in dist/*; do brotli $i; done
bundles are generated by default to projectFolder/dist(/$projectFolder for 6)
tamanhos com Angular 8.2.11
com CLI 8.3.13
e opção CSS sem roteamento Angular.
dist/index.html
da sua aplicação.dist/runtime-[es-version].[hash].bundle.js
webpack loaderdist/style.[hash].bundle.css
as definições de estiloVocê pode obter uma prévia da sua aplicação utilizando o comando `ng serve --prod' que inicia um servidor HTTP local de forma que a aplicação com arquivos de produção seja acessível utilizando http://localhost:4200.
Para uma utilização de produção, você tem que distribuir todos os arquivos da pasta dist
no servidor HTTP de sua escolha.
2.0.1 Final
utilizando Gulp (TypeScript - Target: ES5)npm install
(rodar em cmd quando direcory é projectFolder)npm run bundle
(correr em cmd quando direcory é projectFolder)*bundles are generated to projectFolder / bundles /**
var gulp = require('gulp'),
tsc = require('gulp-typescript'),
Builder = require('systemjs-builder'),
inlineNg2Template = require('gulp-inline-ng2-template');
gulp.task('bundle', ['bundle-app', 'bundle-dependencies'], function(){});
gulp.task('inline-templates', function () {
return gulp.src('app/**/*.ts')
.pipe(inlineNg2Template({ useRelativePaths: true, indent: 0, removeLineBreaks: true}))
.pipe(tsc({
"target": "ES5",
"module": "system",
"moduleResolution": "node",
"sourceMap": true,
"emitDecoratorMetadata": true,
"experimentalDecorators": true,
"removeComments": true,
"noImplicitAny": false
}))
.pipe(gulp.dest('dist/app'));
});
gulp.task('bundle-app', ['inline-templates'], function() {
// optional constructor options
// sets the baseURL and loads the configuration file
var builder = new Builder('', 'dist-systemjs.config.js');
return builder
.bundle('dist/app/**/* - [@angular/**/*.js] - [rxjs/**/*.js]', 'bundles/app.bundle.js', { minify: true})
.then(function() {
console.log('Build complete');
})
.catch(function(err) {
console.log('Build error');
console.log(err);
});
});
gulp.task('bundle-dependencies', ['inline-templates'], function() {
// optional constructor options
// sets the baseURL and loads the configuration file
var builder = new Builder('', 'dist-systemjs.config.js');
return builder
.bundle('dist/app/**/*.js - [dist/app/**/*.js]', 'bundles/dependencies.bundle.js', { minify: true})
.then(function() {
console.log('Build complete');
})
.catch(function(err) {
console.log('Build error');
console.log(err);
});
});
{
"name": "angular2-quickstart",
"version": "1.0.0",
"scripts": {
***
"gulp": "gulp",
"rimraf": "rimraf",
"bundle": "gulp bundle",
"postbundle": "rimraf dist"
},
"license": "ISC",
"dependencies": {
***
},
"devDependencies": {
"rimraf": "^2.5.2",
"gulp": "^3.9.1",
"gulp-typescript": "2.13.6",
"gulp-inline-ng2-template": "2.0.1",
"systemjs-builder": "^0.15.16"
}
}
(function(global) {
// map tells the System loader where to look for things
var map = {
'app': 'app',
'rxjs': 'node_modules/rxjs',
'angular2-in-memory-web-api': 'node_modules/angular2-in-memory-web-api',
'@angular': 'node_modules/@angular'
};
// packages tells the System loader how to load when no filename and/or no extension
var packages = {
'app': { main: 'app/boot.js', defaultExtension: 'js' },
'rxjs': { defaultExtension: 'js' },
'angular2-in-memory-web-api': { defaultExtension: 'js' }
};
var packageNames = [
'@angular/common',
'@angular/compiler',
'@angular/core',
'@angular/forms',
'@angular/http',
'@angular/platform-browser',
'@angular/platform-browser-dynamic',
'@angular/router',
'@angular/router-deprecated',
'@angular/testing',
'@angular/upgrade',
];
// add package entries for angular packages in the form '@angular/common': { main: 'index.js', defaultExtension: 'js' }
packageNames.forEach(function(pkgName) {
packages[pkgName] = { main: 'index.js', defaultExtension: 'js' };
});
var config = {
map: map,
packages: packages
};
// filterSystemConfig - index.asp's chance to modify config before we register it.
if (global.filterSystemConfig) { global.filterSystemConfig(config); }
System.config(config);
})(this);
var map = {
'app': 'dist/app',
};
dist-systemjs.config.js' após as tags do pacote ainda permitiria que o programa fosse executado, mas o pacote de dependências seria ignorado e as dependências seriam carregadas da pasta
node_modules'.<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8"/>
<meta name="viewport" content="width=device-width, initial-scale=1"/>
<base href="/"/>
<title>Angular</title>
<link rel="stylesheet" type="text/css" href="style.css"/>
</head>
<body>
<my-app>
loading...
</my-app>
<!-- Polyfill(s) for older browsers -->
<script src="node_modules/core-js/client/shim.min.js"></script>
<script src="node_modules/zone.js/dist/zone.min.js"></script>
<script src="node_modules/reflect-metadata/Reflect.js"></script>
<script src="node_modules/systemjs/dist/system.js"></script>
<script src="dist-systemjs.config.js"></script>
<!-- Project Bundles. Note that these have to be loaded AFTER the systemjs.config script -->
<script src="bundles/dependencies.bundle.js"></script>
<script src="bundles/app.bundle.js"></script>
<script>
System.import('app/boot').catch(function (err) {
console.error(err);
});
</script>
</body>
</html>
O melhor que pude fazer até agora :)
Angular.io tem tutorial de início rápido. Eu copiei este tutorial e ampliei com algumas tarefas simples de gulp para empacotar tudo para a pasta dist, que pode ser copiada para o servidor e funcionar assim mesmo. Eu tentei otimizar tudo para funcionar bem no Jenkis CI, assim os node_modules podem ser armazenados em cache e don't precisam ser copiados.
Código fonte com aplicativo de amostra no Github: https://github.com/Anjmao/angular2-production-workflow
Nó: Embora possa sempre criar o seu próprio processo de construção, mas recomendo vivamente o uso do angular-cli, porque tem todos os fluxos de trabalho necessários e funciona perfeitamente agora. Nós já estamos usando-o em produção e don'não temos nenhum problema com o angular-cli.