diff --git a/README.md b/README.md
index 18fe734f..9e724dd4 100644
--- a/README.md
+++ b/README.md
@@ -57,9 +57,9 @@ In many cases, the defaults which Ionic provides covers most of the scenarios re
[Default Config Files](https://github.com/driftyco/ionic-app-scripts/tree/master/config)
-### NPM Config
+### package.json Config
-Within the `package.json` file, NPM also provides a handy [config](https://docs.npmjs.com/misc/config#per-package-config-settings) property. Below is an example of setting a custom config file using the `config` property in a project's `package.json`.
+Within the `package.json` file, there's also a handy [config](https://docs.npmjs.com/misc/config#per-package-config-settings) property which can be used. Below is an example of setting a custom config file using the `config` property in a project's `package.json`.
```
"config": {
@@ -88,7 +88,7 @@ npm run build --rollup ./config/rollup.config.js
### Overriding Config Files
-| Config File | NPM Config Property | Cmd-line Flag |
+| Config File | package.json Config | Cmd-line Flag |
|-------------|---------------------|-----------------------|
| CleanCss | `ionic_cleancss` | `--cleancss` or `-e` |
| Copy | `ionic_copy` | `--copy` or `-y` |
@@ -103,7 +103,7 @@ npm run build --rollup ./config/rollup.config.js
### Overriding Config Values
-| Config Values | NPM Config Property | Cmd-line Flag | Defaults |
+| Config Values | package.json Config | Cmd-line Flag | Defaults |
|-----------------|---------------------|---------------|-----------------|
| bundler | `ionic_bundler` | `--bundler` | `webpack` |
| source map type | `ionic_source_map` | `--sourceMap` | `eval` |
@@ -113,9 +113,21 @@ npm run build --rollup ./config/rollup.config.js
| build directory | `ionic_build_dir` | `--buildDir` | `build` |
-### Ionic Environment Variable
+### Ionic Environment Variables
-The `process.env.IONIC_ENV` environment variable can be used to test whether it is a `prod` or `dev` build, which automatically gets set by any command. By default the `build` task is `prod`, and the `watch` task is `dev`. Note that `ionic serve` uses the `watch` task. Additionally, using the `--dev` command line flag will force the build to use `dev`.
+These environment variables are automatically set to [Node's `process.env`](https://nodejs.org/api/process.html#process_process_env) property. These variables can be useful from within custom configuration files, such as custom `webpack.config.js` file.
+
+| Environment Variable | Description |
+|-------------------------|----------------------------------------------------------------------|
+| `IONIC_ENV` | Value can be either `prod` or `dev`. |
+| `IONIC_ROOT_DIR` | The absolute path to the project's root directory. |
+| `IONIC_TMP_DIR` | The absolute path to the project's temporary directory. |
+| `IONIC_SRC_DIR` | The absolute path to the app's source directory. |
+| `IONIC_WWW_DIR` | The absolute path to the app's public distribution directory. |
+| `IONIC_BUILD_DIR` | The absolute path to the app's bundled js and css files. |
+| `IONIC_APP_SCRIPTS_DIR` | The absolute path to the `@ionic/app-scripts` node_module directory. |
+
+The `process.env.IONIC_ENV` environment variable can be used to test whether it is a `prod` or `dev` build, which automatically gets set by any command. By default the `build` task is `prod`, and the `watch` and `serve` tasks are `dev`. Additionally, using the `--dev` command line flag will force the build to use `dev`.
Please take a look at the bottom of the [default Rollup config file](https://github.com/driftyco/ionic-app-scripts/blob/master/config/rollup.config.js) to see how the `IONIC_ENV` environment variable is being used to conditionally change config values for production builds.
diff --git a/bin/ion-dev.css b/bin/ion-dev.css
index a150441e..826e6a6d 100644
--- a/bin/ion-dev.css
+++ b/bin/ion-dev.css
@@ -1,3 +1,272 @@
+#ion-diagnostics * {
+ box-sizing: border-box;
+}
+
+#ion-diagnostics {
+ position: absolute;
+ top: 0;
+ left: 0;
+ width: 100%;
+ height: 100%;
+ z-index: 99999;
+ margin: 0;
+ padding: 15px;
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
+ font-size: 14px;
+ line-height: 1.5;
+ color: #333;
+ background-color: #fff;
+ word-wrap: break-word;
+ box-sizing: border-box;
+ -webkit-font-smoothing: antialiased;
+ font-smoothing: antialiased;
+ text-rendering: optimizeLegibility;
+ text-size-adjust: none;
+ overflow: auto;
+}
+
+#ion-diagnostics .ion-diagnostic {
+ margin-bottom: 40px;
+ border: 1px solid #ddd;
+ border-radius: 3px;
+}
+
+#ion-diagnostics .ion-diagnostic-masthead {
+ padding: 8px 12px 0 12px;
+}
+
+#ion-diagnostics .ion-diagnostic-header {
+ margin: 0;
+ font-size: 18px;
+ color: #222222;
+}
+
+#ion-diagnostics .ion-diagnostic-message {
+ margin-top: 4px;
+ color: #666666;
+}
+
+#ion-diagnostics .ion-diagnostic-file {
+ position: relative;
+ margin-top: 16px;
+ border-top: 1px solid #ddd;
+}
+
+#ion-diagnostics .ion-diagnostic-file-header {
+ padding: 5px 10px;
+ border-bottom: 1px solid #d8d8d8;
+ border-top-left-radius: 2px;
+ border-top-right-radius: 2px;
+ background-color: #f7f7f7;
+}
+
+#ion-diagnostics .ion-diagnostic-blob {
+ overflow-x: auto;
+ overflow-y: hidden;
+ border-bottom-right-radius: 3px;
+ border-bottom-left-radius: 3px;
+}
+
+#ion-diagnostics .ion-diagnostic-table {
+ border-spacing: 0;
+ border-collapse: collapse;
+ -moz-tab-size: 2;
+ -o-tab-size: 2;
+ tab-size: 2;
+}
+
+#ion-diagnostics .ion-diagnostic-table td,
+#ion-diagnostics .ion-diagnostic-table th {
+ padding: 0;
+}
+
+#ion-diagnostics .ion-diagnostic-blob-num {
+ padding-right: 10px;
+ padding-left: 10px;
+ width: 1%;
+ min-width: 50px;
+ font-family: Consolas, "Liberation Mono", Menlo, Courier, monospace;
+ font-size: 12px;
+ line-height: 20px;
+ color: rgba(0,0,0,0.3);
+ text-align: right;
+ white-space: nowrap;
+ vertical-align: top;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+ border: solid #eee;
+ border-width: 0 1px 0 0;
+}
+
+#ion-diagnostics .ion-diagnostic-blob-num::before {
+ content: attr(data-line-number);
+}
+
+#ion-diagnostics .ion-diagnostic-error-line .ion-diagnostic-blob-num {
+ background-color: #ffdddd;
+ border-color: #f1c0c0;
+}
+
+#ion-diagnostics .ion-diagnostic-error-line .ion-diagnostic-blob-code {
+ background: rgba(255, 221, 221, 0.3);
+ z-index: -1;
+}
+
+#ion-diagnostics .ion-diagnostics-error-chr {
+ position: relative;
+}
+
+#ion-diagnostics .ion-diagnostics-error-chr::before {
+ position: absolute;
+ z-index: -1;
+ top: -3px;
+ left: 0px;
+ width: 8px;
+ height: 20px;
+ background-color: #ffdddd;
+ content: "";
+}
+
+#ion-diagnostics .ion-diagnostic-blob-code {
+ position: relative;
+ padding-right: 10px;
+ padding-left: 10px;
+ line-height: 20px;
+ vertical-align: top;
+ overflow: visible;
+ font-family: Consolas, "Liberation Mono", Menlo, Courier, monospace;
+ font-size: 12px;
+ color: #333;
+ word-wrap: normal;
+ white-space: pre;
+}
+
+#ion-diagnostics .ion-diagnostic-blob-code::before {
+ content: "";
+}
+
+#ion-diagnostics .js-keyword,
+#ion-diagnostics .css-prop {
+ color: #183691;
+}
+
+#ion-diagnostics .js-comment,
+#ion-diagnostics .sass-comment {
+ color: #969896;
+}
+
+#ion-diagnostics-system-info {
+ padding-bottom: 20px;
+ font-size: 10px;
+ color: #999;
+}
+
+#ion-diagnostics {
+ -webkit-transition: opacity 150ms ease-out;
+ transition: opacity 150ms ease-out;
+}
+
+#ion-diagnostics.ion-diagnostics-fade-out {
+ opacity: 0;
+}
+
+#ion-diagnostics-toast {
+ position: absolute;
+ top: 10px;
+ right: 10px;
+ left: 10px;
+ z-index: 999999;
+ display: block;
+ margin: auto;
+ max-width: 700px;
+ border-radius: 3px;
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
+ background: rgba(0,0,0,.9);
+ -webkit-transform: translate3d(0px, -60px, 0px);
+ transform: translate3d(0px, -60px, 0px);
+ -webkit-transition: -webkit-transform 75ms ease-out;
+ transition: transform 75ms ease-out;
+ pointer-events: none;
+}
+
+#ion-diagnostics-toast.ion-diagnostics-toast-active {
+ -webkit-transform: translate3d(0px, 0px, 0px);
+ transform: translate3d(0px, 0px, 0px);
+}
+
+#ion-diagnostics-toast .ion-diagnostics-toast-content {
+ display: flex;
+ -webkit-align-items: center;
+ -ms-flex-align: center;
+ align-items: center;
+ pointer-events: auto;
+}
+
+#ion-diagnostics-toast .ion-diagnostics-toast-message {
+ -webkit-flex: 1;
+ -ms-flex: 1;
+ flex: 1;
+ padding: 15px;
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
+ font-size: 14px;
+ color: #fff;
+}
+
+#ion-diagnostics-toast .ion-diagnostics-toast-spinner {
+ position: relative;
+ display: inline-block;
+ width: 56px;
+ height: 28px;
+}
+
+#ion-diagnostics-toast svg:not(:root) {
+ overflow: hidden;
+}
+
+#ion-diagnostics-toast svg {
+ position: absolute;
+ top: 0;
+ left: 0;
+ width: 100%;
+ height: 100%;
+ -webkit-transform: translateZ(0);
+ transform: translateZ(0);
+ -webkit-animation: ion-diagnostics-spinner-rotate 600ms linear infinite;
+ animation: ion-diagnostics-spinner-rotate 600ms linear infinite;
+}
+
+@-webkit-keyframes ion-diagnostics-spinner-rotate {
+ 0% {
+ -webkit-transform: rotate(0deg);
+ transform: rotate(0deg);
+ }
+ 100% {
+ -webkit-transform: rotate(360deg);
+ transform: rotate(360deg);
+ }
+}
+
+@keyframes ion-diagnostics-spinner-rotate {
+ 0% {
+ -webkit-transform: rotate(0deg);
+ transform: rotate(0deg);
+ }
+ 100% {
+ -webkit-transform: rotate(360deg);
+ transform: rotate(360deg);
+ }
+}
+
+#ion-diagnostics-toast circle {
+ fill: transparent;
+ stroke: white;
+ stroke-width: 4px;
+ stroke-dasharray: 128px;
+ stroke-dashoffset: 82px;
+}
+
._ionic-error-view {
position: fixed;
top: 0;
diff --git a/bin/ion-dev.js b/bin/ion-dev.js
index 13603b81..90829eac 100644
--- a/bin/ion-dev.js
+++ b/bin/ion-dev.js
@@ -1,3 +1,4 @@
+window.IonicDevServerConfig = window.IonicDevServerConfig || {};
window.IonicDevServer = {
start: function() {
this.msgQueue = [];
@@ -6,15 +7,28 @@ window.IonicDevServer = {
this.consoleError = console.error;
this.consoleWarn = console.warn;
- if (IonicDevServerConfig && IonicDevServerConfig.sendConsoleLogs) {
+ IonicDevServerConfig.systemInfo.push('Navigator Platform: ' + window.navigator.platform);
+ IonicDevServerConfig.systemInfo.push('User Agent: ' + window.navigator.userAgent);
+
+ if (IonicDevServerConfig.sendConsoleLogs) {
this.patchConsole();
}
- console.log('dev server enabled');
-
this.openConnection();
this.bindKeyboardEvents();
+
+ document.addEventListener("DOMContentLoaded", IonicDevServer.domReady);
+ },
+
+ domReady: function() {
+ document.removeEventListener("DOMContentLoaded", IonicDevServer.domReady);
+ var diagnosticsEle = document.getElementById('ion-diagnostics');
+ if (diagnosticsEle) {
+ IonicDevServer.buildStatus('error');
+ } else {
+ IonicDevServer.buildStatus('success');
+ }
},
handleError: function(err) {
@@ -141,8 +155,8 @@ window.IonicDevServer = {
try {
var msg = JSON.parse(ev.data);
switch (msg.category) {
- case 'taskEvent':
- self.receiveTaskEvent(msg);
+ case 'buildUpdate':
+ self.buildUpdate(msg);
break;
}
} catch (e) {
@@ -208,23 +222,108 @@ window.IonicDevServer = {
}
}
},
- receiveTaskEvent: function(taskEvent) {
- if (taskEvent.data && ['bundle', 'sass', 'transpile', 'template'].indexOf(taskEvent.data.scope) > -1) {
- this.consoleLog(taskEvent.data.msg);
+
+ buildUpdate: function(msg) {
+ var status = 'success';
+
+ if (msg.type === 'started') {
+ status = 'active';
+
+ var toastEle = document.getElementById('ion-diagnostics-toast');
+ if (!toastEle) {
+ toastEle = document.createElement('div');
+ toastEle.id = 'ion-diagnostics-toast';
+ var c = []
+ c.push('
');
+ c.push('
Building...
');
+ c.push('
');
+ c.push('');
+ c.push('
');
+ c.push('
');
+ toastEle.innerHTML = c.join('');
+ document.body.insertBefore(toastEle, document.body.firstChild);
+ }
+ IonicDevServer.toastTimerId = setTimeout(function() {
+ var toastEle = document.getElementById('ion-diagnostics-toast');
+ if (toastEle) {
+ toastEle.classList.add('ion-diagnostics-toast-active');
+ }
+ }, 50);
+
+ } else {
+ status = msg.data.diagnosticsHtml ? 'error' : 'success';
+
+ clearTimeout(IonicDevServer.toastTimerId);
+
+ var toastEle = document.getElementById('ion-diagnostics-toast');
+ if (toastEle) {
+ toastEle.classList.remove('ion-diagnostics-toast-active');
+ }
+
+ var diagnosticsEle = document.getElementById('ion-diagnostics');
+ if (diagnosticsEle && !msg.data.diagnosticsHtml) {
+ diagnosticsEle.classList.add('ion-diagnostics-fade-out');
+ IonicDevServer.diagnosticsTimerId = setTimeout(function() {
+ var diagnosticsEle = document.getElementById('ion-diagnostics');
+ if (diagnosticsEle) {
+ diagnosticsEle.parentElement.removeChild(diagnosticsEle);
+ }
+ }, 100);
+
+ } else if (msg.data.diagnosticsHtml) {
+ clearTimeout(IonicDevServer.diagnosticsTimerId);
+
+ if (!diagnosticsEle) {
+ diagnosticsEle = document.createElement('div');
+ diagnosticsEle.id = 'ion-diagnostics';
+ diagnosticsEle.className = 'ion-diagnostics-fade-out';
+ document.body.insertBefore(diagnosticsEle, document.body.firstChild);
+ IonicDevServer.diagnosticsTimerId = setTimeout(function() {
+ var diagnosticsEle = document.getElementById('ion-diagnostics');
+ if (diagnosticsEle) {
+ diagnosticsEle.classList.remove('ion-diagnostics-fade-out');
+ }
+ }, 24);
+ }
+ diagnosticsEle.innerHTML = msg.data.diagnosticsHtml;
+ }
}
- if (taskEvent.data && taskEvent.data.type === 'failed') {
- Notification.requestPermission().then(function(result) {
- var options = {
- body: taskEvent.data.msg,
- icon: IonicDevServerConfig.notificationIconPath
+
+ IonicDevServer.buildStatus(status);
+ },
+
+ buildStatus: function (status) {
+ var iconLinks = document.querySelectorAll('link[rel="icon"]');
+ for (var i = 0; i < iconLinks.length; i++) {
+ iconLinks[i].parentElement.removeChild(iconLinks[i]);
+ }
+
+ var iconLink = document.createElement('link');
+ iconLink.rel = 'icon';
+ iconLink.type = 'image/png';
+ iconLink.href = IonicDevServer[status + 'Icon'];
+ document.head.appendChild(iconLink);
+
+ if (status === 'error') {
+ var diagnosticsEle = document.getElementById('ion-diagnostics');
+ if (diagnosticsEle) {
+ var systemInfoEle = diagnosticsEle.querySelector('#ion-diagnostics-system-info');
+ if (!systemInfoEle) {
+ systemInfoEle = document.createElement('pre');
+ systemInfoEle.id = 'ion-diagnostics-system-info';
+ systemInfoEle.innerHTML = IonicDevServerConfig.systemInfo.join('\n');
+ diagnosticsEle.appendChild(systemInfoEle);
}
- var notification = new Notification(taskEvent.data.scope, options);
- setTimeout(notification.close.bind(n), 5000);
- });
+ }
}
- }
-};
+ },
+ activeIcon: '',
+ errorIcon: '',
+
+ successIcon: ''
+
+};
IonicDevServer.start();
diff --git a/bin/ionic.png b/bin/ionic.png
deleted file mode 100644
index d82dab01..00000000
Binary files a/bin/ionic.png and /dev/null differ
diff --git a/config/watch.config.js b/config/watch.config.js
index bde65851..8ef6ec8f 100644
--- a/config/watch.config.js
+++ b/config/watch.config.js
@@ -1,9 +1,6 @@
-var fullBuildUpdate = require('../dist/build').fullBuildUpdate;
-var buildUpdate = require('../dist/build').buildUpdate;
-var templateUpdate = require('../dist/template').templateUpdate;
-var copyUpdate = require('../dist/copy').copyUpdate;
-var sassUpdate = require('../dist/sass').sassUpdate;
-var copyConfig = require('./copy.config').include;
+var watch = require('../dist/watch');
+var copy = require('../dist/copy');
+var copyConfig = require('./copy.config');
// https://www.npmjs.com/package/chokidar
@@ -14,44 +11,15 @@ module.exports = {
{
paths: [
- '{{SRC}}/**/*.ts'
+ '{{SRC}}/**/*.(ts|html|scss)'
],
options: { ignored: '{{SRC}}/**/*.spec.ts' },
- eventName: 'all',
- callback: buildUpdate
+ callback: watch.buildUpdate
},
{
- paths: [
- '{{SRC}}/**/*.html'
- ],
- eventName: 'all',
- callback: templateUpdate
- },
-
- {
- paths: [
- '{{SRC}}/**/*.scss'
- ],
- eventName: 'all',
- callback: sassUpdate
- },
-
- {
- paths: copyConfig.map(f => f.src),
- eventName: 'all',
- callback: copyUpdate
- },
-
- {
- paths: [
- '{{SRC}}/**/*',
- ],
- options: {
- ignored: `{{SRC}}/assets/**/*`
- },
- eventName: 'unlinkDir',
- callback: fullBuildUpdate
+ paths: copyConfig.include.map(f => f.src),
+ callback: copy.copyUpdate
}
]
diff --git a/package.json b/package.json
index 2817f4e2..0aaa0bf6 100644
--- a/package.json
+++ b/package.json
@@ -31,7 +31,7 @@
"dependencies": {
"autoprefixer": "6.4.1",
"chalk": "1.1.3",
- "chokidar": "1.6.0",
+ "chokidar": "1.6.1",
"clean-css": "3.4.19",
"conventional-changelog-cli": "^1.2.0",
"cross-spawn": "4.0.0",
@@ -39,6 +39,7 @@
"fs-extra": "0.30.0",
"json-loader": "^0.5.4",
"node-sass": "3.10.1",
+ "os-name": "^2.0.1",
"postcss": "5.2.0",
"proxy-middleware": "^0.15.0",
"rollup": "0.36.3",
diff --git a/src/build.ts b/src/build.ts
index cfcb1c17..caa1ce44 100644
--- a/src/build.ts
+++ b/src/build.ts
@@ -1,15 +1,16 @@
-import { BuildContext } from './util/interfaces';
-import { BuildError, IgnorableError, Logger } from './util/logger';
+import { BuildContext, BuildState } from './util/interfaces';
+import { BuildError, Logger } from './util/logger';
import { bundle, bundleUpdate } from './bundle';
import { clean } from './clean';
import { copy } from './copy';
import { emit, EventType } from './util/events';
import { generateContext } from './util/config';
-import { lint } from './lint';
+import { lint, lintUpdate } from './lint';
import { minifyCss, minifyJs } from './minify';
import { ngc } from './ngc';
import { sass, sassUpdate } from './sass';
-import { transpile, transpileUpdate } from './transpile';
+import { templateUpdate } from './template';
+import { transpile, transpileUpdate, transpileDiagnosticsOnly } from './transpile';
export function build(context: BuildContext) {
@@ -20,8 +21,6 @@ export function build(context: BuildContext) {
return buildWorker(context)
.then(() => {
// congrats, we did it! (•_•) / ( •_•)>⌐■-■ / (⌐■_■)
- context.fullBuildCompleted = true;
- emit(EventType.BuildFinished);
logger.finish();
})
.catch(err => {
@@ -48,7 +47,6 @@ function buildProd(context: BuildContext) {
// async tasks
// these can happen all while other tasks are running
const copyPromise = copy(context);
- const lintPromise = lint(context);
// kick off ngc to run the Ahead of Time compiler
return ngc(context)
@@ -72,10 +70,13 @@ function buildProd(context: BuildContext) {
]);
})
.then(() => {
+ // kick off the tslint after everything else
+ // nothing needs to wait on its completion
+ lint(context);
+
// ensure the async tasks have fully completed before resolving
return Promise.all([
- copyPromise,
- lintPromise
+ copyPromise
]);
})
.catch(err => {
@@ -91,7 +92,6 @@ function buildDev(context: BuildContext) {
// async tasks
// these can happen all while other tasks are running
const copyPromise = copy(context);
- const lintPromise = lint(context);
// just bundle, and if that passes then do the rest at the same time
return transpile(context)
@@ -101,55 +101,175 @@ function buildDev(context: BuildContext) {
.then(() => {
return Promise.all([
sass(context),
- copyPromise,
- lintPromise
+ copyPromise
]);
})
+ .then(() => {
+ // kick off the tslint after everything else
+ // nothing needs to wait on its completion
+ lint(context);
+ return Promise.resolve();
+ })
.catch(err => {
throw new BuildError(err);
});
}
-export function fullBuildUpdate(event: string, filePath: string, context: BuildContext) {
- return buildUpdateWorker(event, filePath, context, true).then(() => {
- context.fullBuildCompleted = true;
- });
-}
export function buildUpdate(event: string, filePath: string, context: BuildContext) {
- return buildUpdateWorker(event, filePath, context, false);
+ return new Promise(resolve => {
+ const logger = new Logger('build');
+
+ buildUpdateId++;
+ emit(EventType.BuildUpdateStarted, buildUpdateId);
+
+ function buildTasksDone(resolveValue: BuildTaskResolveValue) {
+ // all build tasks have been resolved or one of them
+ // bailed early, stopping all others to not run
+
+ parallelTasksPromise.then(() => {
+ // all parallel tasks are also done
+ // so now we're done done
+ emit(EventType.BuildUpdateCompleted, buildUpdateId);
+
+ if (resolveValue.requiresRefresh) {
+ // emit that we need to do a full page refresh
+ emit(EventType.ReloadApp);
+
+ } else {
+ // just emit that only a certain file changed
+ // this one is useful when only a sass changed happened
+ // and the webpack only needs to livereload the css
+ // but does not need to do a full page refresh
+ emit(EventType.FileChange, resolveValue.changedFile);
+ }
+
+ if (filePath.endsWith('.ts')) {
+ // a ts file changed, so let's lint it too, however
+ // this task should run as an after thought
+ lintUpdate(event, filePath, context);
+ }
+
+ logger.finish('green', true);
+ Logger.newLine();
+
+ // we did it!
+ resolve();
+ });
+ }
+
+ // kick off all the build tasks
+ // and the tasks that can run parallel to all the build tasks
+ const buildTasksPromise = buildUpdateTasks(event, filePath, context);
+ const parallelTasksPromise = buildUpdateParallelTasks(event, filePath, context);
+
+ // whether it was resolved or rejected, we need to do the same thing
+ buildTasksPromise
+ .then(buildTasksDone)
+ .catch(() => {
+ buildTasksDone({
+ requiresRefresh: false,
+ changedFile: filePath
+ });
+ });
+ });
}
-function buildUpdateWorker(event: string, filePath: string, context: BuildContext, fullBuild: boolean) {
- const logger = new Logger(`build update`);
+/**
+ * Collection of all the build tasks than need to run
+ * Each task will only run if it's set with eacn BuildState.
+ */
+function buildUpdateTasks(event: string, filePath: string, context: BuildContext) {
+ const resolveValue: BuildTaskResolveValue = {
+ requiresRefresh: false,
+ changedFile: filePath
+ };
- let transpilePromise: Promise = null;
- if (fullBuild) {
- transpilePromise = transpile(context);
- } else {
- transpilePromise = transpileUpdate(event, filePath, context);
- }
- return transpilePromise
+ return Promise.resolve()
.then(() => {
- if (fullBuild) {
- return bundle(context);
+ // TEMPLATE
+ if (context.templateState === BuildState.RequiresUpdate) {
+ resolveValue.requiresRefresh = true;
+ return templateUpdate(event, filePath, context);
}
- return bundleUpdate(event, filePath, context);
- }).then(() => {
- if (fullBuild) {
- return sass(context);
- } else if (event !== 'change' || !context.successfulSass) {
- // if just the TS file changed, then there's no need to do a sass update
- // however, if a new TS file was added or was deleted, then we should do a sass update
- return sassUpdate(event, filePath, context);
+ // no template updates required
+ return Promise.resolve();
+
+ })
+ .then(() => {
+ // TRANSPILE
+ if (context.transpileState === BuildState.RequiresUpdate) {
+ resolveValue.requiresRefresh = true;
+ // we've already had a successful transpile once, only do an update
+ // not that we've also already started a transpile diagnostics only
+ // build that only needs to be completed by the end of buildUpdate
+ return transpileUpdate(event, filePath, context);
+
+ } else if (context.transpileState === BuildState.RequiresBuild) {
+ // run the whole transpile
+ resolveValue.requiresRefresh = true;
+ return transpile(context);
}
- }).then(() => {
- emit(EventType.BuildFinished);
- logger.finish();
- }).catch(err => {
- if (err instanceof IgnorableError) {
- throw err;
+ // no transpiling required
+ return Promise.resolve();
+
+ })
+ .then(() => {
+ // BUNDLE
+ if (context.bundleState === BuildState.RequiresUpdate) {
+ // we need to do a bundle update
+ resolveValue.requiresRefresh = true;
+ return bundleUpdate(event, filePath, context);
+
+ } else if (context.bundleState === BuildState.RequiresBuild) {
+ // we need to do a full bundle build
+ resolveValue.requiresRefresh = true;
+ return bundle(context);
}
- throw logger.fail(err);
+ // no bundling required
+ return Promise.resolve();
+
+ })
+ .then(() => {
+ // SASS
+ if (context.sassState === BuildState.RequiresUpdate) {
+ // we need to do a sass update
+ return sassUpdate(event, filePath, context).then(outputCssFile => {
+ resolveValue.changedFile = outputCssFile;
+ });
+
+ } else if (context.sassState === BuildState.RequiresBuild) {
+ // we need to do a full sass build
+ return sass(context).then(outputCssFile => {
+ resolveValue.changedFile = outputCssFile;
+ });
+ }
+ // no sass build required
+ return Promise.resolve();
+ })
+ .then(() => {
+ return resolveValue;
});
}
+
+interface BuildTaskResolveValue {
+ requiresRefresh: boolean;
+ changedFile: string;
+}
+
+/**
+ * parallelTasks are for any tasks that can run parallel to the entire
+ * build, but we still need to make sure they've completed before we're
+ * all done, it's also possible there are no parallelTasks at all
+ */
+function buildUpdateParallelTasks(event: string, filePath: string, context: BuildContext) {
+ const parallelTasks: Promise[] = [];
+
+ if (context.transpileState === BuildState.RequiresUpdate) {
+ parallelTasks.push(transpileDiagnosticsOnly(context));
+ }
+
+ return Promise.all(parallelTasks);
+}
+
+let buildUpdateId = 0;
diff --git a/src/declarations.d.ts b/src/declarations.d.ts
index 409df78a..b416dd1f 100644
--- a/src/declarations.d.ts
+++ b/src/declarations.d.ts
@@ -1,5 +1,6 @@
declare module 'autoprefixer';
declare module 'mime-types';
+declare module 'os-name';
declare module 'proxy-middleware';
declare module 'rollup-pluginutils';
declare module 'rollup';
diff --git a/src/dev-server/http-server.ts b/src/dev-server/http-server.ts
index 23c741be..2cc80c19 100644
--- a/src/dev-server/http-server.ts
+++ b/src/dev-server/http-server.ts
@@ -6,12 +6,10 @@ import * as fs from 'fs';
import * as url from 'url';
import { ServeConfig, LOGGER_DIR } from './serve-config';
import { Logger } from '../util/logger';
-import { promisify } from '../util/promisify';
import * as proxyMiddleware from 'proxy-middleware';
-import { readDiagnosticsHtmlSync } from '../util/logger-diagnostics';
+import { injectDiagnosticsHtml } from '../util/logger-diagnostics';
import { getProjectJson, IonicProject } from '../util/ionic-project';
-const readFilePromise = promisify(fs.readFile);
/**
* Create HTTP server
@@ -26,7 +24,7 @@ export function createHttpServer(config: ServeConfig): express.Application {
app.get('/', serveIndex);
app.use('/', express.static(config.wwwDir));
- app.use(`/${LOGGER_DIR}`, express.static(path.join(__dirname, '..', '..', 'bin')));
+ app.use(`/${LOGGER_DIR}`, express.static(path.join(__dirname, '..', '..', 'bin'), { maxAge: 31536000 }));
app.get('/cordova.js', serveCordovaJS);
if (config.useProxy) {
@@ -58,33 +56,27 @@ function setupProxies(app: express.Application) {
*/
function serveIndex(req: express.Request, res: express.Response) {
const config: ServeConfig = req.app.get('serveConfig');
- let htmlFile = path.join(config.wwwDir, 'index.html');
- let diagnosticsHtml = readDiagnosticsHtmlSync(config.buildDir);
- function httpResponse(content: any) {
+ // respond with the index.html file
+ const indexFileName = path.join(config.wwwDir, 'index.html');
+ fs.readFile(indexFileName, (err, indexHtml) => {
if (config.useLiveReload) {
- content = injectLiveReloadScript(content, config.host, config.liveReloadPort);
- }
- if (config.useNotifier) {
- content = injectNotificationScript(content, config.notifyOnConsoleLog, config.notificationPort);
+ indexHtml = injectLiveReloadScript(indexHtml, config.host, config.liveReloadPort);
}
- // File found so lets send it back to the response
- res.set('Content-Type', 'text/html');
- res.send(content);
- }
+ indexHtml = injectNotificationScript(config.rootDir, indexHtml, config.notifyOnConsoleLog, config.notificationPort);
- if (diagnosticsHtml) {
- httpResponse(diagnosticsHtml);
- } else {
- readFilePromise(htmlFile).then(httpResponse);
- }
+ indexHtml = injectDiagnosticsHtml(config.buildDir, indexHtml);
+
+ res.set('Content-Type', 'text/html');
+ res.send(indexHtml);
+ });
}
/**
- * http responder for cordova.js fiel
+ * http responder for cordova.js file
*/
function serveCordovaJS(req: express.Request, res: express.Response) {
res.set('Content-Type', 'application/javascript');
res.send('// mock cordova file during development');
-}
\ No newline at end of file
+}
diff --git a/src/dev-server/injector.ts b/src/dev-server/injector.ts
index 423052cf..4438c8d9 100644
--- a/src/dev-server/injector.ts
+++ b/src/dev-server/injector.ts
@@ -1,10 +1,12 @@
+import { getAppScriptsVersion, getSystemInfo } from '../util/helpers';
import { LOGGER_DIR } from './serve-config';
+
const LOGGER_HEADER = '';
-export function injectNotificationScript(content: any, notifyOnConsoleLog: boolean, notificationPort: Number): any {
+export function injectNotificationScript(rootDir: string, content: any, notifyOnConsoleLog: boolean, notificationPort: Number): any {
let contentStr = content.toString();
- const consoleLogScript = getConsoleLoggerScript(notifyOnConsoleLog, notificationPort);
+ const consoleLogScript = getDevLoggerScript(rootDir, notifyOnConsoleLog, notificationPort);
if (contentStr.indexOf(LOGGER_HEADER) > -1) {
// already added script somehow
@@ -24,17 +26,19 @@ export function injectNotificationScript(content: any, notifyOnConsoleLog: boole
return contentStr;
}
-function getConsoleLoggerScript(notifyOnConsoleLog: boolean, notificationPort: Number) {
+function getDevLoggerScript(rootDir: string, notifyOnConsoleLog: boolean, notificationPort: Number) {
+ const appScriptsVersion = getAppScriptsVersion();
const ionDevServer = JSON.stringify({
sendConsoleLogs: notifyOnConsoleLog,
wsPort: notificationPort,
- notificationIconPath: `${LOGGER_DIR}/ionic.png`
+ appScriptsVersion: appScriptsVersion,
+ systemInfo: getSystemInfo(rootDir)
});
return `
${LOGGER_HEADER}
-
-
+
+
`;
}
diff --git a/src/dev-server/live-reload.ts b/src/dev-server/live-reload.ts
index 1dcb21fb..5b8b73e1 100644
--- a/src/dev-server/live-reload.ts
+++ b/src/dev-server/live-reload.ts
@@ -1,3 +1,4 @@
+import { hasDiagnostics } from '../util/logger-diagnostics';
import * as path from 'path';
import * as tinylr from 'tiny-lr';
import { ServeConfig } from './serve-config';
@@ -8,59 +9,27 @@ export function createLiveReloadServer(config: ServeConfig) {
const liveReloadServer = tinylr();
liveReloadServer.listen(config.liveReloadPort, config.host);
- function broadcastChange(filePath: string | string[]) {
- const files = Array.isArray(filePath) ? filePath : [filePath];
- const msg = {
- body: {
- files: files.map(f => '/' + path.relative(config.wwwDir, f))
- }
- };
- liveReloadServer.changed(msg);
- }
-
- let hasFinishedSass = false;
- let hasFinishedBundle = false;
- let hasFinishedBuild = false;
- let hasDoneHardRefresh = false;
-
- events.on(events.EventType.SassFinished, (sassFile: string) => {
- hasFinishedSass = true;
- if (hasFinishedBuild) {
- // only livereload css if a bundle has finished
- // and a build has finished
- // css live reload does not refresh the index page
- broadcastChange(sassFile);
- }
- });
-
- events.on(events.EventType.BundleFinished, (jsFile: string) => {
- hasFinishedBundle = true;
- if (hasFinishedSass && hasFinishedBuild) {
- // only livereload js if sass has finished
- // and a build has finished
- // js live reload refreshes the index page
- hasDoneHardRefresh = true;
- broadcastChange(jsFile);
+ function fileChange(filePath: string | string[]) {
+ // only do a live reload if there are no diagnostics
+ // the notification server takes care of showing diagnostics
+ if (!hasDiagnostics(config.buildDir)) {
+ const files = Array.isArray(filePath) ? filePath : [filePath];
+ liveReloadServer.changed({
+ body: {
+ files: files.map(f => '/' + path.relative(config.wwwDir, f))
+ }
+ });
}
- });
+ }
- events.on(events.EventType.BuildFinished, () => {
- hasFinishedBuild = true;
- if (!hasDoneHardRefresh) {
- hasDoneHardRefresh = true;
- broadcastChange('index.html');
- }
- });
+ events.on(events.EventType.FileChange, fileChange);
- events.on(events.EventType.UpdatedDiagnostics, () => {
- // new diagnostics files have been saved
- // refresh the index file so they render
- broadcastChange('index.html');
+ events.on(events.EventType.ReloadApp, () => {
+ fileChange('index.html');
});
-
- events.on(events.EventType.FileChange, broadcastChange);
}
+
export function injectLiveReloadScript(content: any, host: string, port: Number): any {
let contentStr = content.toString();
const liveReloadScript = getLiveReloadScript(host, port);
diff --git a/src/dev-server/notification-server.ts b/src/dev-server/notification-server.ts
index cdddbf6b..a8a1542f 100644
--- a/src/dev-server/notification-server.ts
+++ b/src/dev-server/notification-server.ts
@@ -1,24 +1,55 @@
// Ionic Dev Server: Server Side Logger
-import { Diagnostic, Logger, TaskEvent } from '../util/logger';
+import { Logger } from '../util/logger';
+import { hasDiagnostics, getDiagnosticsHtmlContent } from '../util/logger-diagnostics';
import { on, EventType } from '../util/events';
import { Server as WebSocketServer } from 'ws';
import { ServeConfig } from './serve-config';
-let wsServer: any;
-const msgToClient: WsMessage[] = [];
-
-export interface WsMessage {
- category: string;
- type: string;
- data: any;
-}
export function createNotificationServer(config: ServeConfig) {
- on(EventType.TaskEvent, (taskEvent: TaskEvent) => {
+ let wsServer: any;
+
+ // queue up all messages to the client
+ function queueMessageSend(msg: WsMessage) {
+ msgToClient.push(msg);
+ drainMessageQueue();
+ }
+
+ // drain the queue messages when the server is ready
+ function drainMessageQueue() {
+ if (wsServer) {
+ let msg: any;
+ while (msg = msgToClient.shift()) {
+ try {
+ wsServer.send(JSON.stringify(msg));
+ } catch (e) {
+ Logger.error(`error sending client ws, ${e}`);
+ }
+ }
+ }
+ }
+
+ // a build update has started, notify the client
+ on(EventType.BuildUpdateStarted, (buildUpdateId) => {
+ const msg: WsMessage = {
+ category: 'buildUpdate',
+ type: 'started',
+ data: {
+ buildUpdateId: buildUpdateId
+ }
+ };
+ queueMessageSend(msg);
+ });
+
+ // a build update has completed, notify the client
+ on(EventType.BuildUpdateCompleted, (buildUpdateId) => {
const msg: WsMessage = {
- category: EventType.TaskEvent,
- type: taskEvent.scope,
- data: taskEvent
+ category: 'buildUpdate',
+ type: 'completed',
+ data: {
+ buildUpdateId: buildUpdateId,
+ diagnosticsHtml: hasDiagnostics(config.buildDir) ? getDiagnosticsHtmlContent(config.buildDir) : null
+ }
};
queueMessageSend(msg);
});
@@ -27,52 +58,36 @@ export function createNotificationServer(config: ServeConfig) {
const wss = new WebSocketServer({ port: config.notificationPort });
wss.on('connection', (ws: any) => {
+ // we've successfully connected
wsServer = ws;
wsServer.on('message', (incomingMessage: string) => {
// incoming message from the client
try {
- printClientMessage(JSON.parse(incomingMessage));
+ printMessageFromClient(JSON.parse(incomingMessage));
} catch (e) {
Logger.error(`error opening ws message: ${incomingMessage}`);
}
});
+ // now that we're connected, send off any messages
+ // we might has already queued up
drainMessageQueue();
});
}
-function queueMessageSend(msg: WsMessage) {
- msgToClient.push(msg);
- drainMessageQueue();
-}
-
-
-function drainMessageQueue() {
- if (wsServer) {
- var msg: any;
- while (msg = msgToClient.shift()) {
- try {
- wsServer.send(JSON.stringify(msg));
- } catch (e) {
- Logger.error(`error sending client ws, ${e}`);
- }
- }
- }
-}
-
-function printClientMessage(msg: WsMessage) {
+function printMessageFromClient(msg: WsMessage) {
if (msg.data) {
switch (msg.category) {
- case 'console':
- printConsole(msg);
- break;
+ case 'console':
+ printConsole(msg);
+ break;
- case 'exception':
- printException(msg);
- break;
+ case 'exception':
+ printException(msg);
+ break;
}
}
}
@@ -82,24 +97,32 @@ function printConsole(msg: WsMessage) {
args[0] = `console.${msg.type}: ${args[0]}`;
switch (msg.type) {
- case 'error':
- Logger.error.apply(this, args);
- break;
+ case 'error':
+ Logger.error.apply(this, args);
+ break;
- case 'warn':
- Logger.warn.apply(this, args);
- break;
+ case 'warn':
+ Logger.warn.apply(this, args);
+ break;
- case 'debug':
- Logger.debug.apply(this, args);
- break;
+ case 'debug':
+ Logger.debug.apply(this, args);
+ break;
- default:
- Logger.info.apply(this, args);
- break;
- }
+ default:
+ Logger.info.apply(this, args);
+ break;
+ }
}
function printException(msg: WsMessage) {
}
+
+const msgToClient: WsMessage[] = [];
+
+export interface WsMessage {
+ category: string;
+ type: string;
+ data: any;
+}
diff --git a/src/dev-server/serve-config.ts b/src/dev-server/serve-config.ts
index 53ebbed6..495572a9 100644
--- a/src/dev-server/serve-config.ts
+++ b/src/dev-server/serve-config.ts
@@ -1,6 +1,7 @@
export interface ServeConfig {
httpPort: number;
host: string;
+ rootDir: string;
wwwDir: string;
buildDir: string;
launchBrowser: boolean;
@@ -9,7 +10,6 @@ export interface ServeConfig {
useLiveReload: boolean;
liveReloadPort: Number;
notificationPort: Number;
- useNotifier: boolean;
useServerLogs: boolean;
notifyOnConsoleLog: boolean;
useProxy: boolean;
diff --git a/src/index.ts b/src/index.ts
index b6502086..394bd8ca 100644
--- a/src/index.ts
+++ b/src/index.ts
@@ -1,4 +1,4 @@
-export { build, buildUpdate, fullBuildUpdate } from './build';
+export { build, buildUpdate } from './build';
export { bundle, bundleUpdate } from './bundle';
export { clean } from './clean';
export { cleancss } from './cleancss';
@@ -8,20 +8,19 @@ export { minify } from './minify';
export { ngc } from './ngc';
export { sass, sassUpdate } from './sass';
export { transpile } from './transpile';
-export { templateUpdate } from './template';
export { uglifyjs } from './uglifyjs';
export { watch } from './watch';
export * from './util/config';
export * from './util/helpers';
export * from './util/interfaces';
-import { Logger, getAppScriptsVersion } from './util/logger';
-import * as chalk from 'chalk';
+import { getAppScriptsVersion } from './util/helpers';
+import { Logger } from './util/logger';
export function run(task: string) {
try {
- Logger.info(chalk.cyan(`ionic-app-scripts ${getAppScriptsVersion()}`));
+ Logger.info(`ionic-app-scripts ${getAppScriptsVersion()}`, 'cyan');
} catch (e) {}
try {
diff --git a/src/lint.ts b/src/lint.ts
index d6e28032..f353e5e5 100644
--- a/src/lint.ts
+++ b/src/lint.ts
@@ -4,7 +4,8 @@ import { BuildError, Logger } from './util/logger';
import { generateContext, getUserConfigFile } from './util/config';
import { join } from 'path';
import { createProgram, findConfiguration, getFileNames } from 'tslint';
-import { runDiagnostics } from './util/logger-tslint';
+import { runTsLintDiagnostics } from './util/logger-tslint';
+import { printDiagnostics, DiagnosticsType } from './util/logger-diagnostics';
import { runWorker } from './worker-client';
import * as Linter from 'tslint';
import * as fs from 'fs';
@@ -22,17 +23,10 @@ export function lint(context?: BuildContext, configFile?: string) {
export function lintWorker(context: BuildContext, configFile: string) {
- const logger = new Logger('lint');
return getLintConfig(context, configFile).then(configFile => {
// there's a valid tslint config, let's continue
return lintApp(context, configFile);
- }).then(() => {
- // always finish and resolve
- logger.finish();
- }).catch(() => {
- // always finish and resolve
- logger.finish();
- });
+ }).catch(() => {});
}
@@ -101,7 +95,8 @@ function lintFile(context: BuildContext, program: ts.Program, filePath: string)
const lintResult = linter.lint();
if (lintResult && lintResult.failures) {
- runDiagnostics(context, lintResult.failures);
+ const diagnostics = runTsLintDiagnostics(context, lintResult.failures);
+ printDiagnostics(context, DiagnosticsType.TsLint, diagnostics, true, false);
}
} catch (e) {
diff --git a/src/ngc.ts b/src/ngc.ts
index 985ccccc..4406b7b1 100644
--- a/src/ngc.ts
+++ b/src/ngc.ts
@@ -1,7 +1,7 @@
import { basename, join } from 'path';
import { BuildContext, TaskInfo } from './util/interfaces';
import { copy as fsCopy, emptyDirSync, outputJsonSync, readFileSync, statSync } from 'fs-extra';
-import { endsWith, objectAssign } from './util/helpers';
+import { objectAssign } from './util/helpers';
import { fillConfigDefaults, generateContext, getUserConfigFile, getNodeBinExecutable } from './util/config';
import { getTsConfigPath } from './transpile';
import { BuildError, Logger } from './util/logger';
@@ -171,7 +171,7 @@ function filterCopyFiles(filePath: any, hoop: any) {
shouldInclude = (EXCLUDE_DIRS.indexOf(basename(filePath)) < 0);
} else {
- shouldInclude = (endsWith(filePath, '.ts') || endsWith(filePath, '.html'));
+ shouldInclude = (filePath.endsWith('.ts') || filePath.endsWith('.html'));
}
} catch (e) {}
diff --git a/src/rollup.ts b/src/rollup.ts
index 7de0305c..fa7d7e49 100644
--- a/src/rollup.ts
+++ b/src/rollup.ts
@@ -1,7 +1,5 @@
-import { BuildContext, TaskInfo } from './util/interfaces';
+import { BuildContext, BuildState, TaskInfo } from './util/interfaces';
import { BuildError, Logger } from './util/logger';
-import { emit, EventType } from './util/events';
-import { endsWith, setModulePathsCache } from './util/helpers';
import { fillConfigDefaults, generateContext, getUserConfigFile, replacePathVars } from './util/config';
import { ionCompiler } from './plugins/ion-compiler';
import { join, isAbsolute, normalize } from 'path';
@@ -16,9 +14,11 @@ export function rollup(context: BuildContext, configFile: string) {
return rollupWorker(context, configFile)
.then(() => {
+ context.bundleState = BuildState.SuccessfulBuild;
logger.finish();
})
.catch(err => {
+ context.bundleState = BuildState.RequiresBuild;
throw logger.fail(err);
});
}
@@ -31,9 +31,11 @@ export function rollupUpdate(event: string, filePath: string, context: BuildCont
return rollupWorker(context, configFile)
.then(() => {
+ context.bundleState = BuildState.SuccessfulBuild;
logger.finish();
})
.catch(err => {
+ context.bundleState = BuildState.RequiresBuild;
throw logger.fail(err);
});
}
@@ -60,10 +62,8 @@ export function rollupWorker(context: BuildContext, configFile: string): Promise
);
}
- if (context.useBundleCache) {
- // tell rollup to use a previous bundle as its starting point
- rollupConfig.cache = cachedBundle;
- }
+ // tell rollup to use a previous bundle as its starting point
+ rollupConfig.cache = cachedBundle;
if (!rollupConfig.onwarn) {
// use our own logger if one wasn't already provided
@@ -84,10 +84,6 @@ export function rollupWorker(context: BuildContext, configFile: string): Promise
// this reference can be used elsewhere in the build (sass)
context.moduleFiles = bundle.modules.map((m) => m.id);
- // async cache all the module paths so we don't need
- // to always bundle to know which modules are used
- setModulePathsCache(context.moduleFiles);
-
// cache our bundle for later use
if (context.isWatch) {
cachedBundle = bundle;
@@ -98,7 +94,6 @@ export function rollupWorker(context: BuildContext, configFile: string): Promise
})
.then(() => {
// clean up any references (overkill yes, but let's play it safe)
- emit(EventType.BundleFinished, rollupConfig.dest);
rollupConfig = rollupConfig.cache = rollupConfig.onwarn = rollupConfig.plugins = null;
resolve();
@@ -130,7 +125,7 @@ export function getOutputDest(context: BuildContext, rollupConfig: RollupConfig)
function checkDeprecations(context: BuildContext, rollupConfig: RollupConfig) {
if (!context.isProd) {
- if (rollupConfig.entry.indexOf('.tmp') > -1 || endsWith(rollupConfig.entry, '.js')) {
+ if (rollupConfig.entry.indexOf('.tmp') > -1 || rollupConfig.entry.endsWith('.js')) {
// warning added 2016-10-05, v0.0.29
throw new BuildError('\nDev builds no longer use the ".tmp" directory. Please update your rollup config\'s\n' +
'entry to use your "src" directory\'s "main.dev.ts" TypeScript file.\n' +
diff --git a/src/sass.ts b/src/sass.ts
index a2ccf5c2..f5574183 100755
--- a/src/sass.ts
+++ b/src/sass.ts
@@ -1,11 +1,11 @@
import { basename, dirname, join, sep } from 'path';
-import { BuildContext, TaskInfo } from './util/interfaces';
+import { BuildContext, BuildState, TaskInfo } from './util/interfaces';
import { BuildError, Logger } from './util/logger';
-import { emit, EventType } from './util/events';
+import { bundle } from './bundle';
import { ensureDirSync, readdirSync, writeFile } from 'fs-extra';
import { fillConfigDefaults, generateContext, getUserConfigFile, replacePathVars } from './util/config';
-import { getModulePathsCache } from './util/helpers';
-import { runDiagnostics, clearSassDiagnostics } from './util/logger-sass';
+import { runSassDiagnostics } from './util/logger-sass';
+import { printDiagnostics, clearDiagnostics, DiagnosticsType } from './util/logger-diagnostics';
import { SassError, render as nodeSassRender, Result } from 'node-sass';
import * as postcss from 'postcss';
import * as autoprefixer from 'autoprefixer';
@@ -17,14 +17,14 @@ export function sass(context?: BuildContext, configFile?: string) {
const logger = new Logger('sass');
- context.successfulSass = false;
-
return sassWorker(context, configFile)
- .then(() => {
- context.successfulSass = true;
+ .then(outFile => {
+ context.sassState = BuildState.SuccessfulBuild;
logger.finish();
+ return outFile;
})
.catch(err => {
+ context.sassState = BuildState.RequiresBuild;
throw logger.fail(err);
});
}
@@ -36,29 +36,28 @@ export function sassUpdate(event: string, filePath: string, context: BuildContex
const logger = new Logger('sass update');
return sassWorker(context, configFile)
- .then(() => {
+ .then(outFile => {
+ context.sassState = BuildState.SuccessfulBuild;
logger.finish();
+ return outFile;
})
.catch(err => {
+ context.sassState = BuildState.RequiresBuild;
throw logger.fail(err);
});
}
export function sassWorker(context: BuildContext, configFile: string) {
- return new Promise((resolve, reject) => {
- if (!context.moduleFiles) {
- // we haven't already gotten the moduleFiles in this process
- // see if we have it cached
- context.moduleFiles = getModulePathsCache();
- if (!context.moduleFiles) {
- reject(new BuildError('Cannot generate Sass files without first bundling JavaScript ' +
- 'files in order to know all used modules. Please build JS files first.'));
- return;
- }
- }
+ const bundlePromise: Promise[] = [];
+ if (!context.moduleFiles) {
+ // sass must always have a list of all the used module files
+ // so ensure we bundle if moduleFiles are currently unknown
+ bundlePromise.push(bundle(context));
+ }
- clearSassDiagnostics(context);
+ return Promise.all(bundlePromise).then(() => {
+ clearDiagnostics(context, DiagnosticsType.Sass);
const sassConfig: SassConfig = fillConfigDefaults(configFile, taskInfo.defaultConfigFile);
@@ -84,15 +83,7 @@ export function sassWorker(context: BuildContext, configFile: string) {
generateSassData(context, sassConfig);
}
- return render(context, sassConfig)
- .then(() => {
- resolve(true);
- }, (reason: any) => {
- reject(reason);
- })
- .catch(err => {
- reject(new BuildError(err));
- });
+ return render(context, sassConfig);
});
}
@@ -219,8 +210,11 @@ function getComponentDirectories(moduleDirectories: string[], sassConfig: SassCo
// filter out module directories we know wouldn't have sibling component sass file
// just a way to reduce the amount of lookups to be done later
return moduleDirectories.filter(moduleDirectory => {
+ // normalize this directory is using / between directories
+ moduleDirectory = moduleDirectory.replace(/\\/g, '/');
+
for (var i = 0; i < sassConfig.excludeModules.length; i++) {
- if (moduleDirectory.indexOf(sassConfig.excludeModules[i]) > -1) {
+ if (moduleDirectory.indexOf('/node_modules/' + sassConfig.excludeModules[i] + '/') > -1) {
return false;
}
}
@@ -229,18 +223,9 @@ function getComponentDirectories(moduleDirectories: string[], sassConfig: SassCo
}
-function render(context: BuildContext, sassConfig: SassConfig) {
+function render(context: BuildContext, sassConfig: SassConfig): Promise {
return new Promise((resolve, reject) => {
- if (context.useSassCache && lastRenderKey !== null) {
- // if the sass data imports are same, don't bother
- const renderKey = getRenderCacheKey(sassConfig);
- if (renderKey === lastRenderKey) {
- resolve();
- return;
- }
- }
-
sassConfig.omitSourceMapUrl = true;
if (sassConfig.sourceMap) {
@@ -249,20 +234,17 @@ function render(context: BuildContext, sassConfig: SassConfig) {
}
nodeSassRender(sassConfig, (sassError: SassError, sassResult: Result) => {
- const diagnostics = runDiagnostics(context, sassError);
+ const diagnostics = runSassDiagnostics(context, sassError);
if (diagnostics.length) {
+ printDiagnostics(context, DiagnosticsType.Sass, diagnostics, true, true);
// sass render error :(
- const buildError = new BuildError();
- buildError.updatedDiagnostics = true;
- emit(EventType.UpdatedDiagnostics);
- reject(buildError);
+ reject(new BuildError());
} else {
// sass render success :)
- renderSassSuccess(context, sassResult, sassConfig).then(() => {
- lastRenderKey = getRenderCacheKey(sassConfig);
- resolve();
+ renderSassSuccess(context, sassResult, sassConfig).then(outFile => {
+ resolve(outFile);
}).catch(err => {
reject(new BuildError(err));
@@ -273,7 +255,7 @@ function render(context: BuildContext, sassConfig: SassConfig) {
}
-function renderSassSuccess(context: BuildContext, sassResult: Result, sassConfig: SassConfig): Promise {
+function renderSassSuccess(context: BuildContext, sassResult: Result, sassConfig: SassConfig): Promise {
if (sassConfig.autoprefixer) {
// with autoprefixer
@@ -361,7 +343,7 @@ function generateSourceMaps(sassResult: Result, sassConfig: SassConfig) {
}
-function writeOutput(context: BuildContext, sassConfig: SassConfig, cssOutput: string, mappingsOutput: string) {
+function writeOutput(context: BuildContext, sassConfig: SassConfig, cssOutput: string, mappingsOutput: string): Promise {
return new Promise((resolve, reject) => {
Logger.debug(`sass start write output: ${sassConfig.outFile}`);
@@ -393,12 +375,9 @@ function writeOutput(context: BuildContext, sassConfig: SassConfig, cssOutput: s
});
}
- // notify a file has changed
- emit(EventType.SassFinished, sassConfig.outFile);
-
// css file all saved
// note that we're not waiting on the css map to finish saving
- resolve();
+ resolve(sassConfig.outFile);
}
});
});
@@ -447,17 +426,6 @@ function defaultSortComponentFilesFn(a: any, b: any): number {
}
-function getRenderCacheKey(sassConfig: SassConfig) {
- return [
- sassConfig.data,
- sassConfig.file,
- ].join('|');
-}
-
-
-let lastRenderKey: string = null;
-
-
const taskInfo: TaskInfo = {
fullArg: '--sass',
shortArg: '-s',
diff --git a/src/serve.ts b/src/serve.ts
index 07ae135c..33ea43ed 100644
--- a/src/serve.ts
+++ b/src/serve.ts
@@ -2,7 +2,6 @@ import { BuildContext } from './util/interfaces';
import { generateContext, getConfigValue, hasConfigValue } from './util/config';
import { Logger } from './util/logger';
import { watch } from './watch';
-import * as chalk from 'chalk';
import open from './util/open';
import { createNotificationServer } from './dev-server/notification-server';
import { createHttpServer } from './dev-server/http-server';
@@ -21,6 +20,7 @@ export function serve(context?: BuildContext) {
const config: ServeConfig = {
httpPort: getHttpServerPort(context),
host: getHttpServerHost(context),
+ rootDir: context.rootDir,
wwwDir: context.wwwDir,
buildDir: context.buildDir,
launchBrowser: launchBrowser(context),
@@ -30,14 +30,13 @@ export function serve(context?: BuildContext) {
liveReloadPort: getLiveReloadServerPort(context),
notificationPort: getNotificationPort(context),
useServerLogs: useServerLogs(context),
- useNotifier: true,
useProxy: useProxy(context),
notifyOnConsoleLog: sendClientConsoleLogs(context)
};
- createHttpServer(config);
- createLiveReloadServer(config);
createNotificationServer(config);
+ createLiveReloadServer(config);
+ createHttpServer(config);
return watch(context)
.then(() => {
@@ -59,7 +58,8 @@ function onReady(config: ServeConfig, context: BuildContext) {
open(openOptions.join(''), browserToLaunch(context));
}
- Logger.info(chalk.green(`dev server running: http://${config.host}:${config.httpPort}/`));
+ Logger.info(`dev server running: http://${config.host}:${config.httpPort}/`, 'green', true);
+ Logger.newLine();
}
function getHttpServerPort(context: BuildContext) {
diff --git a/src/spec/ion-compiler.spec.ts b/src/spec/ion-compiler.spec.ts
index 69bc12ba..2517d7a0 100644
--- a/src/spec/ion-compiler.spec.ts
+++ b/src/spec/ion-compiler.spec.ts
@@ -47,7 +47,7 @@ describe('ion-compiler', () => {
// arrange
let context: BuildContext = {};
context.fileCache = new FileCache();
- context.fileCache.put(importer, {
+ context.fileCache.set(importer, {
path: importer,
content: null
});
@@ -63,7 +63,7 @@ describe('ion-compiler', () => {
// arrange
let context: BuildContext = {};
context.fileCache = new FileCache();
- context.fileCache.put(importer, {
+ context.fileCache.set(importer, {
path: importer,
content: ''
});
@@ -79,7 +79,7 @@ describe('ion-compiler', () => {
// arrange
let context: BuildContext = {};
context.fileCache = new FileCache();
- context.fileCache.put(importer, {
+ context.fileCache.set(importer, {
path: importer,
content: 'fake irrelevant data'
});
@@ -88,7 +88,7 @@ describe('ion-compiler', () => {
const importerBasename = dirname(importer);
const importeeFullPath = resolve(join(importerBasename, importee)) + '.ts';
- context.fileCache.put(importeeFullPath, {
+ context.fileCache.set(importeeFullPath, {
path: importeeFullPath,
content: 'someContent'
});
@@ -105,7 +105,7 @@ describe('ion-compiler', () => {
// arrange
let context: BuildContext = {};
context.fileCache = new FileCache();
- context.fileCache.put(importer, {
+ context.fileCache.set(importer, {
path: importer,
content: 'fake irrelevant data'
});
@@ -114,7 +114,7 @@ describe('ion-compiler', () => {
const importerBasename = dirname(importer);
const importeeFullPath = resolve(join(importerBasename, importee)) + '.ts';
- context.fileCache.put(importeeFullPath, { path: importeeFullPath, content: null});
+ context.fileCache.set(importeeFullPath, { path: importeeFullPath, content: null});
// act
const result = resolveId(importee, importer, context);
@@ -128,7 +128,7 @@ describe('ion-compiler', () => {
// arrange
let context: BuildContext = {};
context.fileCache = new FileCache();
- context.fileCache.put(importer, {
+ context.fileCache.set(importer, {
path: importer,
content: 'fake irrelevant data'
});
@@ -137,7 +137,7 @@ describe('ion-compiler', () => {
const importerBasename = dirname(importer);
const importeeFullPath = join(resolve(join(importerBasename, importee)), 'index.ts');
- context.fileCache.put(importeeFullPath, { path: importeeFullPath, content: null });
+ context.fileCache.set(importeeFullPath, { path: importeeFullPath, content: null });
// act
const result = resolveId(importee, importer, context);
@@ -151,7 +151,7 @@ describe('ion-compiler', () => {
// arrange
let context: BuildContext = {};
context.fileCache = new FileCache();
- context.fileCache.put(importer, {
+ context.fileCache.set(importer, {
path: importer,
content: 'fake irrelevant data'
});
@@ -160,7 +160,7 @@ describe('ion-compiler', () => {
const importerBasename = dirname(importer);
const importeeFullPath = join(resolve(join(importerBasename, importee)), 'index.ts');
- context.fileCache.put(importeeFullPath, { path: importeeFullPath, content: null});
+ context.fileCache.set(importeeFullPath, { path: importeeFullPath, content: null});
// act
const result = resolveId(importee, importer, context);
@@ -173,7 +173,7 @@ describe('ion-compiler', () => {
// arrange
let context: BuildContext = {};
context.fileCache = new FileCache();
- context.fileCache.put(importer, {
+ context.fileCache.set(importer, {
path: importer,
content: 'fake irrelevant data'
});
diff --git a/src/spec/logger.spec.ts b/src/spec/logger.spec.ts
index b4c0c704..e65473c4 100644
--- a/src/spec/logger.spec.ts
+++ b/src/spec/logger.spec.ts
@@ -16,7 +16,6 @@ describe('Logger', () => {
const json = buildErrorCopy.toJson();
expect(json.hasBeenLogged).toEqual(buildError.hasBeenLogged);
- expect(json.updatedDiagnostics).toEqual(buildError.updatedDiagnostics);
expect(json.message).toEqual(buildError.message);
expect(json.name).toEqual(buildError.name);
expect(json.stack).toEqual(buildError.stack);
@@ -30,7 +29,6 @@ describe('Logger', () => {
buildError.stack = 'stack';
const json = buildError.toJson();
expect(json.hasBeenLogged).toEqual(buildError.hasBeenLogged);
- expect(json.updatedDiagnostics).toEqual(buildError.updatedDiagnostics);
expect(json.message).toEqual(buildError.message);
expect(json.name).toEqual(buildError.name);
expect(json.stack).toEqual(buildError.stack);
diff --git a/src/spec/watch.spec.ts b/src/spec/watch.spec.ts
new file mode 100644
index 00000000..f86817e7
--- /dev/null
+++ b/src/spec/watch.spec.ts
@@ -0,0 +1,333 @@
+import { BuildContext, BuildState } from '../util/interfaces';
+import { FileCache } from '../util/file-cache';
+import { runBuildUpdate, ChangedFile } from '../watch';
+import { Watcher, prepareWatcher } from '../watch';
+import * as path from 'path';
+
+
+describe('watch', () => {
+
+ describe('runBuildUpdate', () => {
+
+ it('should get the html file if thats the only one', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.html',
+ ext: '.html'
+ }];
+ const data = runBuildUpdate(context, files);
+ expect(data.filePath).toEqual('file1.html');
+ });
+
+ it('should get the scss file for the filePath over html', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.html',
+ ext: '.html'
+ }, {
+ event: 'change',
+ filePath: 'file1.scss',
+ ext: '.scss'
+ }];
+ const data = runBuildUpdate(context, files);
+ expect(data.filePath).toEqual('file1.scss');
+ });
+
+ it('should get the ts file for the filePath over the others', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.html',
+ ext: '.html'
+ }, {
+ event: 'change',
+ filePath: 'file1.scss',
+ ext: '.scss'
+ }, {
+ event: 'change',
+ filePath: 'file1.ts',
+ ext: '.ts'
+ }];
+ const data = runBuildUpdate(context, files);
+ expect(data.filePath).toEqual('file1.ts');
+ });
+
+ it('should require transpile full build for html file add', () => {
+ const files: ChangedFile[] = [{
+ event: 'add',
+ filePath: 'file1.html',
+ ext: '.html'
+ }];
+ runBuildUpdate(context, files);
+ expect(context.transpileState).toEqual(BuildState.RequiresBuild);
+ });
+
+ it('should require transpile full build for html file change and not already successful bundle', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.html',
+ ext: '.html'
+ }];
+ runBuildUpdate(context, files);
+ expect(context.transpileState).toEqual(BuildState.RequiresBuild);
+ });
+
+ it('should require template update for html file change and already successful bundle', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.html',
+ ext: '.html'
+ }];
+ context.bundleState = BuildState.SuccessfulBuild;
+ runBuildUpdate(context, files);
+ expect(context.templateState).toEqual(BuildState.RequiresUpdate);
+ });
+
+ it('should require sass update for ts file unlink', () => {
+ const files: ChangedFile[] = [{
+ event: 'unlink',
+ filePath: 'file1.ts',
+ ext: '.ts'
+ }];
+ runBuildUpdate(context, files);
+ expect(context.sassState).toEqual(BuildState.RequiresUpdate);
+ });
+
+ it('should require sass update for ts file add', () => {
+ const files: ChangedFile[] = [{
+ event: 'add',
+ filePath: 'file1.ts',
+ ext: '.ts'
+ }];
+ runBuildUpdate(context, files);
+ expect(context.sassState).toEqual(BuildState.RequiresUpdate);
+ });
+
+ it('should require sass update for scss file add', () => {
+ const files: ChangedFile[] = [{
+ event: 'add',
+ filePath: 'file1.scss',
+ ext: '.scss'
+ }];
+ runBuildUpdate(context, files);
+ expect(context.sassState).toEqual(BuildState.RequiresUpdate);
+ });
+
+ it('should require sass update for scss file change', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.scss',
+ ext: '.scss'
+ }];
+ runBuildUpdate(context, files);
+ expect(context.sassState).toEqual(BuildState.RequiresUpdate);
+ });
+
+ it('should require transpile full build for single ts add, but only bundle update when already successful bundle', () => {
+ const files: ChangedFile[] = [{
+ event: 'add',
+ filePath: 'file1.ts',
+ ext: '.ts'
+ }];
+ context.bundleState = BuildState.SuccessfulBuild;
+ runBuildUpdate(context, files);
+ expect(context.transpileState).toEqual(BuildState.RequiresBuild);
+ expect(context.bundleState).toEqual(BuildState.RequiresUpdate);
+ });
+
+ it('should require transpile full build for single ts add', () => {
+ const files: ChangedFile[] = [{
+ event: 'add',
+ filePath: 'file1.ts',
+ ext: '.ts'
+ }];
+ runBuildUpdate(context, files);
+ expect(context.transpileState).toEqual(BuildState.RequiresBuild);
+ expect(context.bundleState).toEqual(BuildState.RequiresBuild);
+ });
+
+ it('should require transpile full build for single ts change and not in file cache', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.ts',
+ ext: '.ts'
+ }];
+ runBuildUpdate(context, files);
+ expect(context.transpileState).toEqual(BuildState.RequiresBuild);
+ expect(context.bundleState).toEqual(BuildState.RequiresBuild);
+ });
+
+ it('should require transpile update only and full bundle build for single ts change and already in file cache and hasnt already had successful bundle', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.ts',
+ ext: '.ts'
+ }];
+ context.bundleState = BuildState.SuccessfulBuild;
+ const resolvedFilePath = path.resolve('file1.ts');
+ context.fileCache.set(resolvedFilePath, { path: 'file1.ts', content: 'content' });
+ runBuildUpdate(context, files);
+ expect(context.transpileState).toEqual(BuildState.RequiresUpdate);
+ expect(context.bundleState).toEqual(BuildState.RequiresUpdate);
+ });
+
+ it('should require transpile update only and bundle update for single ts change and already in file cache and bundle already successful', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.ts',
+ ext: '.ts'
+ }];
+ const resolvedFilePath = path.resolve('file1.ts');
+ context.fileCache.set(resolvedFilePath, { path: 'file1.ts', content: 'content' });
+ runBuildUpdate(context, files);
+ expect(context.transpileState).toEqual(BuildState.RequiresUpdate);
+ expect(context.bundleState).toEqual(BuildState.RequiresBuild);
+ });
+
+ it('should require transpile full build for multiple ts changes', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.ts',
+ ext: '.ts'
+ }, {
+ event: 'change',
+ filePath: 'file2.ts',
+ ext: '.ts'
+ }];
+ runBuildUpdate(context, files);
+ expect(context.transpileState).toEqual(BuildState.RequiresBuild);
+ expect(context.bundleState).toEqual(BuildState.RequiresBuild);
+ });
+
+ it('should not update bundle state if no transpile changes', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.scss',
+ ext: '.scss'
+ }];
+ runBuildUpdate(context, files);
+ expect(context.bundleState).toEqual(undefined);
+ });
+
+ it('should set add event when add and changed files', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file1.ts',
+ ext: '.ts'
+ }, {
+ event: 'add',
+ filePath: 'file2.ts',
+ ext: '.ts'
+ }];
+ const data = runBuildUpdate(context, files);
+ expect(data.event).toEqual('add');
+ });
+
+ it('should set unlink event when only unlinked files', () => {
+ const files: ChangedFile[] = [{
+ event: 'unlink',
+ filePath: 'file.ts',
+ ext: '.ts'
+ }];
+ const data = runBuildUpdate(context, files);
+ expect(data.event).toEqual('unlink');
+ });
+
+ it('should set change event when only changed files', () => {
+ const files: ChangedFile[] = [{
+ event: 'change',
+ filePath: 'file.ts',
+ ext: '.ts'
+ }];
+ const data = runBuildUpdate(context, files);
+ expect(data.event).toEqual('change');
+ });
+
+ it('should do nothing if there are no changed files', () => {
+ expect(runBuildUpdate(context, [])).toEqual(null);
+ expect(runBuildUpdate(context, null)).toEqual(null);
+ });
+
+
+ let context: BuildContext;
+ beforeEach(() => {
+ context = {
+ fileCache: new FileCache()
+ };
+ });
+
+ });
+
+ describe('prepareWatcher', () => {
+
+ it('should do nothing when options.ignored is a function', () => {
+ const ignoreFn = function(){};
+ const watcher: Watcher = { options: { ignored: ignoreFn } };
+ const context: BuildContext = { srcDir: '/some/src/' };
+ prepareWatcher(context, watcher);
+ expect(watcher.options.ignored).toBe(ignoreFn);
+ });
+
+ it('should set replacePathVars when options.ignored is a string', () => {
+ const watcher: Watcher = { options: { ignored: '{{SRC}}/**/*.spec.ts' } };
+ const context: BuildContext = { srcDir: '/some/src/' };
+ prepareWatcher(context, watcher);
+ expect(watcher.options.ignored).toEqual('/some/src/**/*.spec.ts');
+ });
+
+ it('should set replacePathVars when paths is an array', () => {
+ const watcher: Watcher = { paths: [
+ '{{SRC}}/some/path1',
+ '{{SRC}}/some/path2'
+ ] };
+ const context: BuildContext = { srcDir: '/some/src/' };
+ prepareWatcher(context, watcher);
+ expect(watcher.paths.length).toEqual(2);
+ expect(watcher.paths[0]).toEqual('/some/src/some/path1');
+ expect(watcher.paths[1]).toEqual('/some/src/some/path2');
+ });
+
+ it('should set replacePathVars when paths is a string', () => {
+ const watcher: Watcher = { paths: '{{SRC}}/some/path' };
+ const context: BuildContext = { srcDir: '/some/src/' };
+ prepareWatcher(context, watcher);
+ expect(watcher.paths).toEqual('/some/src/some/path');
+ });
+
+ it('should not set options.ignoreInitial if it was provided', () => {
+ const watcher: Watcher = { options: { ignoreInitial: false } };
+ const context: BuildContext = {};
+ prepareWatcher(context, watcher);
+ expect(watcher.options.ignoreInitial).toEqual(false);
+ });
+
+ it('should set options.ignoreInitial to true if it wasnt provided', () => {
+ const watcher: Watcher = { options: {} };
+ const context: BuildContext = {};
+ prepareWatcher(context, watcher);
+ expect(watcher.options.ignoreInitial).toEqual(true);
+ });
+
+ it('should not set options.cwd from context.rootDir if it was provided', () => {
+ const watcher: Watcher = { options: { cwd: '/my/cwd/' } };
+ const context: BuildContext = { rootDir: '/my/root/dir/' };
+ prepareWatcher(context, watcher);
+ expect(watcher.options.cwd).toEqual('/my/cwd/');
+ });
+
+ it('should set options.cwd from context.rootDir if it wasnt provided', () => {
+ const watcher: Watcher = {};
+ const context: BuildContext = { rootDir: '/my/root/dir/' };
+ prepareWatcher(context, watcher);
+ expect(watcher.options.cwd).toEqual(context.rootDir);
+ });
+
+ it('should create watcher options when not provided', () => {
+ const watcher: Watcher = {};
+ const context: BuildContext = {};
+ prepareWatcher(context, watcher);
+ expect(watcher.options).toBeDefined();
+ });
+
+ });
+
+});
diff --git a/src/spec/watcher.spec.ts b/src/spec/watcher.spec.ts
deleted file mode 100644
index e8c30533..00000000
--- a/src/spec/watcher.spec.ts
+++ /dev/null
@@ -1,80 +0,0 @@
-import { BuildContext } from '../util/interfaces';
-import { Watcher, prepareWatcher } from '../watch';
-
-
-describe('watch', () => {
-
- describe('prepareWatcher', () => {
-
- it('should do nothing when options.ignored is a function', () => {
- const ignoreFn = function(){};
- const watcher: Watcher = { options: { ignored: ignoreFn } };
- const context: BuildContext = { srcDir: '/some/src/' };
- prepareWatcher(context, watcher);
- expect(watcher.options.ignored).toBe(ignoreFn);
- });
-
- it('should set replacePathVars when options.ignored is a string', () => {
- const watcher: Watcher = { options: { ignored: '{{SRC}}/**/*.spec.ts' } };
- const context: BuildContext = { srcDir: '/some/src/' };
- prepareWatcher(context, watcher);
- expect(watcher.options.ignored).toEqual('/some/src/**/*.spec.ts');
- });
-
- it('should set replacePathVars when paths is an array', () => {
- const watcher: Watcher = { paths: [
- '{{SRC}}/some/path1',
- '{{SRC}}/some/path2'
- ] };
- const context: BuildContext = { srcDir: '/some/src/' };
- prepareWatcher(context, watcher);
- expect(watcher.paths.length).toEqual(2);
- expect(watcher.paths[0]).toEqual('/some/src/some/path1');
- expect(watcher.paths[1]).toEqual('/some/src/some/path2');
- });
-
- it('should set replacePathVars when paths is a string', () => {
- const watcher: Watcher = { paths: '{{SRC}}/some/path' };
- const context: BuildContext = { srcDir: '/some/src/' };
- prepareWatcher(context, watcher);
- expect(watcher.paths).toEqual('/some/src/some/path');
- });
-
- it('should not set options.ignoreInitial if it was provided', () => {
- const watcher: Watcher = { options: { ignoreInitial: false } };
- const context: BuildContext = {};
- prepareWatcher(context, watcher);
- expect(watcher.options.ignoreInitial).toEqual(false);
- });
-
- it('should set options.ignoreInitial to true if it wasnt provided', () => {
- const watcher: Watcher = { options: {} };
- const context: BuildContext = {};
- prepareWatcher(context, watcher);
- expect(watcher.options.ignoreInitial).toEqual(true);
- });
-
- it('should not set options.cwd from context.rootDir if it was provided', () => {
- const watcher: Watcher = { options: { cwd: '/my/cwd/' } };
- const context: BuildContext = { rootDir: '/my/root/dir/' };
- prepareWatcher(context, watcher);
- expect(watcher.options.cwd).toEqual('/my/cwd/');
- });
-
- it('should set options.cwd from context.rootDir if it wasnt provided', () => {
- const watcher: Watcher = {};
- const context: BuildContext = { rootDir: '/my/root/dir/' };
- prepareWatcher(context, watcher);
- expect(watcher.options.cwd).toEqual(context.rootDir);
- });
-
- it('should create watcher options when not provided', () => {
- const watcher: Watcher = {};
- const context: BuildContext = {};
- prepareWatcher(context, watcher);
- expect(watcher.options).toBeDefined();
- });
-
- });
-
-});
diff --git a/src/template.ts b/src/template.ts
index ef28ed8d..cd462c43 100644
--- a/src/template.ts
+++ b/src/template.ts
@@ -1,62 +1,56 @@
-import { emit, EventType } from './util/events';
-import { BUNDLER_WEBPACK } from './util/config';
-import { BuildContext } from './util/interfaces';
-import { BuildError, IgnorableError, Logger } from './util/logger';
-import { bundleUpdate, getJsOutputDest } from './bundle';
-import { dirname, extname, join, parse, resolve } from 'path';
-import { readdirSync, readFileSync, writeFileSync } from 'fs';
-import { sassUpdate } from './sass';
-import { buildUpdate } from './build';
-
-
-export function templateUpdate(event: string, filePath: string, context: BuildContext) {
- const logger = new Logger('template update');
-
- return templateUpdateWorker(event, filePath, context)
- .then(() => {
- logger.finish();
- })
- .catch(err => {
- if (err instanceof IgnorableError) {
- throw err;
- }
- throw logger.fail(err);
- });
-}
-
-
-function templateUpdateWorker(event: string, filePath: string, context: BuildContext) {
- Logger.debug(`templateUpdate, event: ${event}, path: ${filePath}`);
-
- if (event === 'change') {
- if (context.bundler === BUNDLER_WEBPACK) {
- Logger.debug(`templateUpdate: updating webpack file`);
- const typescriptFile = getTemplatesTypescriptFile(context, filePath);
- if (typescriptFile && typescriptFile.length > 0) {
- return buildUpdate(event, typescriptFile, context);
- }
- } else if (updateBundledJsTemplate(context, filePath)) {
- Logger.debug(`templateUpdate, updated js bundle, path: ${filePath}`);
- // technically, the bundle has changed so emit an event saying so
- emit(EventType.BundleFinished, getJsOutputDest(context));
- return Promise.resolve();
+import { BuildContext, BuildState } from './util/interfaces';
+import { Logger } from './util/logger';
+import { getJsOutputDest } from './bundle';
+import { join, parse, resolve } from 'path';
+import { readFileSync, writeFile } from 'fs';
+
+
+export function templateUpdate(event: string, htmlFilePath: string, context: BuildContext) {
+ return new Promise((resolve) => {
+ const start = Date.now();
+ const bundleOutputDest = getJsOutputDest(context);
+
+ function failed() {
+ context.transpileState = BuildState.RequiresBuild;
+ context.bundleState = BuildState.RequiresUpdate;
+ resolve();
}
- }
- Logger.debug('templateUpdateWorker: Can\'t update template, doing a full rebuild');
- // not sure how it changed, just do a full rebuild without the bundle cache
- context.useBundleCache = false;
- return bundleUpdate(event, filePath, context)
- .then(() => {
- context.useSassCache = true;
- return sassUpdate(event, filePath, context);
- })
- .catch(err => {
- if (err instanceof IgnorableError) {
- throw err;
+ try {
+ let bundleSourceText = readFileSync(bundleOutputDest, 'utf8');
+ let newTemplateContent = readFileSync(htmlFilePath, 'utf8');
+
+ bundleSourceText = replaceBundleJsTemplate(bundleSourceText, newTemplateContent, htmlFilePath);
+
+ if (bundleSourceText) {
+ // awesome, all good and template updated in the bundle file
+ const logger = new Logger(`template update`);
+ logger.setStartTime(start);
+
+ writeFile(bundleOutputDest, bundleSourceText, { encoding: 'utf8'}, (err) => {
+ if (err) {
+ // eww, error saving
+ logger.fail(err);
+ failed();
+
+ } else {
+ // congrats, all gud
+ Logger.debug(`updateBundledJsTemplate, updated: ${htmlFilePath}`);
+ context.templateState = BuildState.SuccessfulBuild;
+ logger.finish();
+ resolve();
+ }
+ });
+
+ } else {
+ failed();
}
- throw new BuildError(err);
- });
+
+ } catch (e) {
+ Logger.debug(`updateBundledJsTemplate error: ${e}`);
+ failed();
+ }
+ });
}
@@ -111,94 +105,38 @@ export function replaceTemplateUrl(match: TemplateUrlMatch, htmlFilePath: string
}
-function updateBundledJsTemplate(context: BuildContext, htmlFilePath: string) {
- Logger.debug(`updateBundledJsTemplate, start: ${htmlFilePath}`);
-
- const outputDest = getJsOutputDest(context);
-
- try {
- let bundleSourceText = readFileSync(outputDest, 'utf8');
- let newTemplateContent = readFileSync(htmlFilePath, 'utf8');
-
- bundleSourceText = replaceBundleJsTemplate(bundleSourceText, newTemplateContent, htmlFilePath);
+export function replaceBundleJsTemplate(bundleSourceText: string, newTemplateContent: string, htmlFilePath: string): string {
+ let prefix = getTemplatePrefix(htmlFilePath);
+ let startIndex = bundleSourceText.indexOf(prefix);
- if (bundleSourceText) {
- writeFileSync(outputDest, bundleSourceText, { encoding: 'utf8'});
- Logger.debug(`updateBundledJsTemplate, updated: ${htmlFilePath}`);
- return true;
- }
+ let isStringified = false;
- } catch (e) {
- Logger.debug(`updateBundledJsTemplate error: ${e}`);
+ if (startIndex === -1) {
+ prefix = stringify(prefix);
+ isStringified = true;
}
- return false;
-}
-
-function getTemplatesTypescriptFile(context: BuildContext, templatePath: string) {
- try {
- const srcDirName = dirname(templatePath);
- const files = readdirSync(srcDirName);
- const typescriptFilesNames = files.filter(file => {
- return extname(file) === '.ts';
- });
- for (const fileName of typescriptFilesNames) {
- const fullPath = join(srcDirName, fileName);
- const fileContent = readFileSync(fullPath).toString();
- const isMatch = tryToMatchTemplateUrl(fileContent, fullPath, srcDirName, templatePath);
- if (isMatch) {
- return fullPath;
- }
- }
- return null;
- } catch (ex) {
- Logger.debug('getTemplatesTypescriptFile: Error occurred - ', ex.message);
+ startIndex = bundleSourceText.indexOf(prefix);
+ if (startIndex === -1) {
return null;
}
-}
-function tryToMatchTemplateUrl(fileContent: string, filePath: string, componentDirPath: string, templateFilePath: string) {
- let lastMatch: string = null;
- let match: TemplateUrlMatch;
-
- while (match = getTemplateMatch(fileContent)) {
- if (match.component === lastMatch) {
- // panic! we don't want to melt any machines if there's a bug
- Logger.debug(`Error matching component: ${match.component}`);
- return false;
- }
- lastMatch = match.component;
-
- if (!match.templateUrl || match.templateUrl === '') {
- Logger.error(`Error @Component templateUrl missing in: "${filePath}"`);
- return false;
- }
-
- const templatUrlPath = resolve(join(componentDirPath, match.templateUrl));
- if (templatUrlPath === templateFilePath) {
- return true;
- }
- }
- return false;
-}
-
-export function replaceBundleJsTemplate(bundleSourceText: string, newTemplateContent: string, htmlFilePath: string): string {
- const prefix = getTemplatePrefix(htmlFilePath);
- const startIndex = bundleSourceText.indexOf(prefix);
-
- if (startIndex === -1) {
- return null;
+ let suffix = getTemplateSuffix(htmlFilePath);
+ if (isStringified) {
+ suffix = stringify(suffix);
}
- const suffix = getTemplateSuffix(htmlFilePath);
const endIndex = bundleSourceText.indexOf(suffix, startIndex + 1);
-
if (endIndex === -1) {
return null;
}
const oldTemplate = bundleSourceText.substring(startIndex, endIndex + suffix.length);
- const newTemplate = getTemplateFormat(htmlFilePath, newTemplateContent);
+ let newTemplate = getTemplateFormat(htmlFilePath, newTemplateContent);
+
+ if (isStringified) {
+ newTemplate = stringify(newTemplate);
+ }
let lastChange: string = null;
while (bundleSourceText.indexOf(oldTemplate) > -1 && bundleSourceText !== lastChange) {
@@ -208,6 +146,11 @@ export function replaceBundleJsTemplate(bundleSourceText: string, newTemplateCon
return bundleSourceText;
}
+function stringify(str: string) {
+ str = JSON.stringify(str);
+ return str.substr(1, str.length - 2);
+}
+
export function getTemplateFormat(htmlFilePath: string, content: string) {
// turn the template into one line and espcape single quotes
diff --git a/src/transpile-worker.ts b/src/transpile-worker.ts
new file mode 100644
index 00000000..1d999bbd
--- /dev/null
+++ b/src/transpile-worker.ts
@@ -0,0 +1,34 @@
+import { BuildContext } from './util/interfaces';
+import { transpileWorker, TranspileWorkerMessage, TranspileWorkerConfig } from './transpile';
+
+
+const context: BuildContext = {};
+
+process.on('message', (incomingMsg: TranspileWorkerMessage) => {
+ context.rootDir = incomingMsg.rootDir;
+ context.buildDir = incomingMsg.buildDir;
+ context.isProd = incomingMsg.isProd;
+
+ const workerConfig: TranspileWorkerConfig = {
+ configFile: incomingMsg.configFile,
+ writeInMemory: false,
+ sourceMaps: false,
+ cache: false,
+ inlineTemplate: false
+ };
+
+ transpileWorker(context, workerConfig)
+ .then(() => {
+ const outgoingMsg: TranspileWorkerMessage = {
+ transpileSuccess: true
+ };
+ process.send(outgoingMsg);
+ })
+ .catch(() => {
+ const outgoingMsg: TranspileWorkerMessage = {
+ transpileSuccess: false
+ };
+ process.send(outgoingMsg);
+ });
+
+});
diff --git a/src/transpile.ts b/src/transpile.ts
index 477a9207..806f2ed2 100644
--- a/src/transpile.ts
+++ b/src/transpile.ts
@@ -1,15 +1,16 @@
import { FileCache } from './util/file-cache';
-import { BuildContext, File } from './util/interfaces';
+import { BuildContext, BuildState } from './util/interfaces';
import { BuildError, Logger } from './util/logger';
import { buildJsSourceMaps } from './bundle';
-import { changeExtension, endsWith } from './util/helpers';
+import { changeExtension } from './util/helpers';
+import { EventEmitter } from 'events';
import { generateContext } from './util/config';
import { inlineTemplate } from './template';
-import { join, normalize, resolve } from 'path';
-import { lintUpdate } from './lint';
import { readFileSync } from 'fs';
-import { runDiagnostics, clearTypeScriptDiagnostics } from './util/logger-typescript';
-import { runWorker } from './worker-client';
+import { runTypeScriptDiagnostics } from './util/logger-typescript';
+import { printDiagnostics, clearDiagnostics, DiagnosticsType } from './util/logger-diagnostics';
+import { fork, ChildProcess } from 'child_process';
+import * as path from 'path';
import * as ts from 'typescript';
@@ -28,20 +29,17 @@ export function transpile(context?: BuildContext) {
return transpileWorker(context, workerConfig)
.then(() => {
+ context.transpileState = BuildState.SuccessfulBuild;
logger.finish();
})
.catch(err => {
+ context.transpileState = BuildState.RequiresBuild;
throw logger.fail(err);
});
}
export function transpileUpdate(event: string, filePath: string, context: BuildContext) {
- if (!filePath.endsWith('.ts') ) {
- // however this ran, the changed file wasn't a .ts file so carry on
- return Promise.resolve();
- }
-
const workerConfig: TranspileWorkerConfig = {
configFile: getTsConfigPath(context),
writeInMemory: true,
@@ -54,9 +52,11 @@ export function transpileUpdate(event: string, filePath: string, context: BuildC
return transpileUpdateWorker(event, filePath, context, workerConfig)
.then(tsFiles => {
+ context.transpileState = BuildState.SuccessfulBuild;
logger.finish();
})
.catch(err => {
+ context.transpileState = BuildState.RequiresBuild;
throw logger.fail(err);
});
}
@@ -70,7 +70,7 @@ export function transpileWorker(context: BuildContext, workerConfig: TranspileWo
// let's do this
return new Promise((resolve, reject) => {
- clearTypeScriptDiagnostics(context);
+ clearDiagnostics(context, DiagnosticsType.TypeScript);
// get the tsconfig data
const tsConfig = getTsConfig(context, workerConfig.configFile);
@@ -101,131 +101,152 @@ export function transpileWorker(context: BuildContext, workerConfig: TranspileWo
}
});
+ // cache the typescript program for later use
+ cachedProgram = program;
+
const tsDiagnostics = program.getSyntacticDiagnostics()
.concat(program.getSemanticDiagnostics())
.concat(program.getOptionsDiagnostics());
- const diagnostics = runDiagnostics(context, tsDiagnostics);
+ const diagnostics = runTypeScriptDiagnostics(context, tsDiagnostics);
if (diagnostics.length) {
- // transpile failed :(
- cachedProgram = null;
+ // darn, we've got some things wrong, transpile failed :(
+ printDiagnostics(context, DiagnosticsType.TypeScript, diagnostics, true, true);
- const buildError = new BuildError();
- buildError.updatedDiagnostics = true;
- reject(buildError);
+ reject(new BuildError());
} else {
// transpile success :)
- // cache the typescript program for later use
- cachedProgram = program;
-
resolve();
}
});
}
+export function canRunTranspileUpdate(event: string, filePath: string, context: BuildContext) {
+ if (event === 'change' && context.fileCache) {
+ return context.fileCache.has(path.resolve(filePath));
+ }
+ return false;
+}
+
+
/**
* Iterative build for one TS file. If it's not an existing file change, or
* something errors out then it falls back to do the full build.
*/
function transpileUpdateWorker(event: string, filePath: string, context: BuildContext, workerConfig: TranspileWorkerConfig) {
- clearTypeScriptDiagnostics(context);
+ return new Promise((resolve, reject) => {
+ clearDiagnostics(context, DiagnosticsType.TypeScript);
- filePath = resolve(filePath);
+ filePath = path.resolve(filePath);
- // let's run tslint on this one file too, but run it in another
- // processor core and don't let it's results hang anything up
- lintUpdate(event, filePath, context);
+ // an existing ts file we already know about has changed
+ // let's "TRY" to do a single module build for this one file
+ const tsConfig = getTsConfig(context, workerConfig.configFile);
- let file: File = null;
- if (context.fileCache) {
- file = context.fileCache.get(filePath);
- }
- if (event === 'change' && file) {
- try {
- // an existing ts file we already know about has changed
- // let's "TRY" to do a single module build for this one file
- const tsConfig = getTsConfig(context, workerConfig.configFile);
+ // build the ts source maps if the bundler is going to use source maps
+ tsConfig.options.sourceMap = buildJsSourceMaps(context);
- // build the ts source maps if the bundler is going to use source maps
- tsConfig.options.sourceMap = buildJsSourceMaps(context);
+ const transpileOptions: ts.TranspileOptions = {
+ compilerOptions: tsConfig.options,
+ fileName: filePath,
+ reportDiagnostics: true
+ };
+
+ // let's manually transpile just this one ts file
+ // load up the source text for this one module
+ const sourceText = readFileSync(filePath, 'utf8');
+
+ // transpile this one module
+ const transpileOutput = ts.transpileModule(sourceText, transpileOptions);
+
+ const diagnostics = runTypeScriptDiagnostics(context, transpileOutput.diagnostics);
+
+ if (diagnostics.length) {
+ printDiagnostics(context, DiagnosticsType.TypeScript, diagnostics, false, true);
- const transpileOptions: ts.TranspileOptions = {
- compilerOptions: tsConfig.options,
- fileName: filePath,
- reportDiagnostics: true
- };
-
- // let's manually transpile just this one ts file
- // load up the source text for this one module
- const sourceText = readFileSync(filePath, 'utf8');
-
- // transpile this one module
- const transpileOutput = ts.transpileModule(sourceText, transpileOptions);
-
- const diagnostics = runDiagnostics(context, transpileOutput.diagnostics);
-
- if (diagnostics.length) {
- // darn, we've got some errors with this transpiling :(
- // but at least we reported the errors like really really fast, so there's that
- Logger.debug(`transpileUpdateWorker: transpileModule, diagnostics: ${diagnostics.length}`);
-
- const buildError = new BuildError();
- buildError.updatedDiagnostics = true;
- return Promise.reject(buildError);
-
- } else if (!transpileOutput.outputText) {
- // derp, not sure how there's no output text, just do a full build
- Logger.debug(`transpileUpdateWorker: transpileModule, missing output text`);
-
- } else {
- // convert the path to have a .js file extension for consistency
- const newPath = changeExtension(filePath, '.js');
-
- const sourceMapFile = { path: newPath + '.map', content: transpileOutput.sourceMapText };
- let jsContent: string = transpileOutput.outputText;
- if (workerConfig.inlineTemplate) {
- // use original path for template inlining
- jsContent = inlineTemplate(transpileOutput.outputText, filePath);
- }
- const jsFile = { path: newPath, content: jsContent };
- const tsFile = { path: filePath, content: sourceText};
-
- context.fileCache.put(sourceMapFile.path, sourceMapFile);
- context.fileCache.put(jsFile.path, jsFile);
- context.fileCache.put(tsFile.path, tsFile);
-
- // cool, the lil transpiling went through, but
- // let's still do the big transpiling (on another processor core)
- // and if there's anything wrong it'll print out messages
- // however, it doesn't hang anything up
- // also make sure it does a little as possible
- const fullBuildWorkerConfig: TranspileWorkerConfig = {
- configFile: workerConfig.configFile,
- writeInMemory: false,
- sourceMaps: false,
- cache: false,
- inlineTemplate: false
- };
- runWorker('transpile', 'transpileWorker', context, fullBuildWorkerConfig);
-
- return Promise.resolve();
+ // darn, we've got some errors with this transpiling :(
+ // but at least we reported the errors like really really fast, so there's that
+ Logger.debug(`transpileUpdateWorker: transpileModule, diagnostics: ${diagnostics.length}`);
+
+ reject(new BuildError());
+
+ } else {
+ // convert the path to have a .js file extension for consistency
+ const newPath = changeExtension(filePath, '.js');
+
+ const sourceMapFile = { path: newPath + '.map', content: transpileOutput.sourceMapText };
+ let jsContent: string = transpileOutput.outputText;
+ if (workerConfig.inlineTemplate) {
+ // use original path for template inlining
+ jsContent = inlineTemplate(transpileOutput.outputText, filePath);
}
+ const jsFile = { path: newPath, content: jsContent };
+ const tsFile = { path: filePath, content: sourceText };
- } catch (e) {
- // umm, oops. Yeah let's just do a full build then
- Logger.debug(`transpileModule error: ${e}`);
- throw new BuildError(e);
+ context.fileCache.set(sourceMapFile.path, sourceMapFile);
+ context.fileCache.set(jsFile.path, jsFile);
+ context.fileCache.set(tsFile.path, tsFile);
+
+ resolve();
}
+ });
+}
+
+
+export function transpileDiagnosticsOnly(context: BuildContext) {
+ return new Promise(resolve => {
+ workerEvent.once('DiagnosticsWorkerDone', () => {
+ resolve();
+ });
+
+ runDiagnosticsWorker(context);
+ });
+}
+
+const workerEvent = new EventEmitter();
+let diagnosticsWorker: ChildProcess = null;
+
+function runDiagnosticsWorker(context: BuildContext) {
+ if (!diagnosticsWorker) {
+ const workerModule = path.join(__dirname, 'transpile-worker.js');
+ diagnosticsWorker = fork(workerModule, [], { env: { FORCE_COLOR: true } });
+
+ Logger.debug(`diagnosticsWorker created, pid: ${diagnosticsWorker.pid}`);
+
+ diagnosticsWorker.on('error', (err: any) => {
+ Logger.error(`diagnosticsWorker error, pid: ${diagnosticsWorker.pid}, error: ${err}`);
+ workerEvent.emit('DiagnosticsWorkerDone');
+ });
+
+ diagnosticsWorker.on('exit', (code: number) => {
+ Logger.debug(`diagnosticsWorker exited, pid: ${diagnosticsWorker.pid}`);
+ diagnosticsWorker = null;
+ });
+
+ diagnosticsWorker.on('message', (msg: TranspileWorkerMessage) => {
+ workerEvent.emit('DiagnosticsWorkerDone');
+ });
}
- // do a full build if it wasn't an existing file that changed
- // or we haven't transpiled the whole thing yet
- // or there were errors trying to transpile just the one module
- Logger.debug(`transpileUpdateWorker: full build, context.tsFiles ${!!context.fileCache}, event: ${event}, file: ${filePath}`);
- return transpileWorker(context, workerConfig);
+ const msg: TranspileWorkerMessage = {
+ rootDir: context.rootDir,
+ buildDir: context.buildDir,
+ isProd: context.isProd,
+ configFile: getTsConfigPath(context)
+ };
+ diagnosticsWorker.send(msg);
+}
+
+
+export interface TranspileWorkerMessage {
+ rootDir?: string;
+ buildDir?: string;
+ isProd?: boolean;
+ configFile?: string;
+ transpileSuccess?: boolean;
}
@@ -237,14 +258,14 @@ function cleanFileNames(context: BuildContext, fileNames: string[]) {
function writeSourceFiles(fileCache: FileCache, sourceFiles: ts.SourceFile[]) {
for (const sourceFile of sourceFiles) {
- fileCache.put(sourceFile.fileName, { path: sourceFile.fileName, content: sourceFile.text });
+ fileCache.set(sourceFile.fileName, { path: sourceFile.fileName, content: sourceFile.text });
}
}
function writeTranspiledFilesCallback(fileCache: FileCache, sourcePath: string, data: string, shouldInlineTemplate: boolean) {
- sourcePath = normalize(sourcePath);
+ sourcePath = path.normalize(sourcePath);
- if (endsWith(sourcePath, '.js')) {
+ if (sourcePath.endsWith('.js')) {
sourcePath = sourcePath.substring(0, sourcePath.length - 3) + '.js';
let file = fileCache.get(sourcePath);
@@ -258,9 +279,9 @@ function writeTranspiledFilesCallback(fileCache: FileCache, sourcePath: string,
file.content = data;
}
- fileCache.put(sourcePath, file);
+ fileCache.set(sourcePath, file);
- } else if (endsWith(sourcePath, '.js.map')) {
+ } else if (sourcePath.endsWith('.js.map')) {
sourcePath = sourcePath.substring(0, sourcePath.length - 7) + '.js.map';
let file = fileCache.get(sourcePath);
@@ -269,7 +290,7 @@ function writeTranspiledFilesCallback(fileCache: FileCache, sourcePath: string,
}
file.content = data;
- fileCache.put(sourcePath, file);
+ fileCache.set(sourcePath, file);
}
}
@@ -295,12 +316,11 @@ export function getTsConfig(context: BuildContext, tsConfigPath?: string): TsCon
ts.sys, context.rootDir,
{}, tsConfigPath);
- const diagnostics = runDiagnostics(context, parsedConfig.errors);
+ const diagnostics = runTypeScriptDiagnostics(context, parsedConfig.errors);
if (diagnostics.length) {
- const buildError = new BuildError();
- buildError.updatedDiagnostics = true;
- throw buildError;
+ printDiagnostics(context, DiagnosticsType.TypeScript, diagnostics, true, true);
+ throw new BuildError();
}
config = {
@@ -318,7 +338,7 @@ export function getTsConfig(context: BuildContext, tsConfigPath?: string): TsCon
let cachedProgram: ts.Program = null;
export function getTsConfigPath(context: BuildContext) {
- return join(context.rootDir, 'tsconfig.json');
+ return path.join(context.rootDir, 'tsconfig.json');
}
export interface TsConfig {
diff --git a/src/util/events.ts b/src/util/events.ts
index 9487de25..b30d59e1 100644
--- a/src/util/events.ts
+++ b/src/util/events.ts
@@ -17,16 +17,13 @@ export function emit(eventType: string, val?: any) {
export const EventType = {
- BuildFinished: 'BuildFinished',
- SassFinished: 'SassFinished',
- BundleFinished: 'BundleFinished',
- FileChange: 'FileChange',
+ BuildUpdateCompleted: 'BuildUpdateCompleted',
+ BuildUpdateStarted: 'BuildUpdateStarted',
FileAdd: 'FileAdd',
+ FileChange: 'FileChange',
FileDelete: 'FileDelete',
DirectoryAdd: 'DirectoryAdd',
DirectoryDelete: 'DirectoryDelete',
- TaskEvent: 'TaskEvent',
- UpdatedDiagnostics: 'UpdatedDiagnostics',
-
+ ReloadApp: 'ReloadApp',
WebpackFilesChanged: 'WebpackFilesChanged'
};
diff --git a/src/util/file-cache.ts b/src/util/file-cache.ts
index 33292d06..cc872048 100644
--- a/src/util/file-cache.ts
+++ b/src/util/file-cache.ts
@@ -1,5 +1,5 @@
import { File } from './interfaces';
-import { emit, EventType } from './events';
+
export class FileCache {
@@ -9,19 +9,21 @@ export class FileCache {
this.map = new Map();
}
- put(key: string, file: File) {
+ set(key: string, file: File) {
file.timestamp = Date.now();
this.map.set(key, file);
- // emit(EventType.DanFileChanged, key);
}
get(key: string): File {
return this.map.get(key);
}
+ has(key: string) {
+ return this.map.has(key);
+ }
+
remove(key: string): Boolean {
const result = this.map.delete(key);
- // emit(EventType.DanFileDeleted, key);
return result;
}
diff --git a/src/util/helpers.ts b/src/util/helpers.ts
index 2e8cee8d..474d3a4b 100644
--- a/src/util/helpers.ts
+++ b/src/util/helpers.ts
@@ -1,11 +1,72 @@
import { BuildContext } from './interfaces';
-import { outputJson, readFile, readJsonSync, writeFile } from 'fs-extra';
-import { BuildError, Logger } from './logger';
+import { readFile, readJsonSync, writeFile } from 'fs-extra';
+import { BuildError } from './logger';
import { basename, dirname, extname, join } from 'path';
-import { tmpdir } from 'os';
+import * as osName from 'os-name';
let _context: BuildContext;
+
+
+let cachedAppScriptsPackageJson: any;
+export function getAppScriptsPackageJson() {
+ if (!cachedAppScriptsPackageJson) {
+ try {
+ cachedAppScriptsPackageJson = readJsonSync(join(__dirname, '..', '..', 'package.json'));
+ } catch (e) {}
+ }
+ return cachedAppScriptsPackageJson;
+}
+
+
+export function getAppScriptsVersion() {
+ const appScriptsPackageJson = getAppScriptsPackageJson();
+ return (appScriptsPackageJson && appScriptsPackageJson.version) ? appScriptsPackageJson.version : '';
+}
+
+function getUserPackageJson(userRootDir: string) {
+ try {
+ return readJsonSync(join(userRootDir, 'package.json'));
+ } catch (e) {}
+ return null;
+}
+
+export function getSystemInfo(userRootDir: string) {
+ const d: string[] = [];
+
+ let ionicAppScripts = getAppScriptsVersion();
+ let ionicFramework: string = null;
+ let ionicNative: string = null;
+ let angularCore: string = null;
+ let angularCompilerCli: string = null;
+
+ try {
+ const userPackageJson = getUserPackageJson(userRootDir);
+ if (userPackageJson) {
+ const userDependencies = userPackageJson.dependencies;
+ if (userDependencies) {
+ ionicFramework = userDependencies['ionic-angular'];
+ ionicNative = userDependencies['ionic-native'];
+ angularCore = userDependencies['@angular/core'];
+ angularCompilerCli = userDependencies['@angular/compiler-cli'];
+ }
+ }
+ } catch (e) {}
+
+ d.push(`Ionic Framework: ${ionicFramework}`);
+ if (ionicNative) {
+ d.push(`Ionic Native: ${ionicNative}`);
+ }
+ d.push(`Ionic App Scripts: ${ionicAppScripts}`);
+ d.push(`Angular Core: ${angularCore}`);
+ d.push(`Angular Compiler CLI: ${angularCompilerCli}`);
+ d.push(`Node: ${process.version.replace('v', '')}`);
+ d.push(`OS Platform: ${osName()}`);
+
+ return d;
+}
+
+
export const objectAssign = (Object.assign) ? Object.assign : function (target: any, source: any) {
const output = Object(target);
@@ -24,14 +85,6 @@ export const objectAssign = (Object.assign) ? Object.assign : function (target:
};
-export function endsWith(str: string, tail: string) {
- if (str && tail) {
- return !tail.length || str.slice(-tail.length).toLowerCase() === tail.toLowerCase();
- }
- return false;
-}
-
-
export function titleCase(str: string) {
return str.charAt(0).toUpperCase() + str.substr(1);
}
@@ -62,42 +115,6 @@ export function readFileAsync(filePath: string): Promise {
});
}
-export function setModulePathsCache(modulePaths: string[]) {
- // async save the module paths for later lookup
- const modulesCachePath = getModulesPathsCachePath();
-
- Logger.debug(`Cached module paths: ${modulePaths && modulePaths.length}, ${modulesCachePath}`);
-
- outputJson(modulesCachePath, modulePaths, (err) => {
- if (err) {
- Logger.error(`Error writing module paths cache: ${err}`);
- }
- });
-}
-
-
-export function getModulesPathsCachePath(): string {
- // make a unique tmp directory for this project's module paths cache file
- let cwd = process.cwd().replace(/-|:|\/|\\|\.|~|;|\s/g, '').toLowerCase();
- if (cwd.length > 40) {
- cwd = cwd.substr(cwd.length - 40);
- }
- return join(tmpdir(), cwd, 'modulepaths.json');
-}
-
-export function getModulePathsCache(): string[] {
- // sync get the cached array of module paths (if they exist)
- let modulePaths: string[] = null;
- const modulesCachePath = getModulesPathsCachePath();
- try {
- modulePaths = readJsonSync(modulesCachePath, { throws: false });
- Logger.debug(`Cached module paths: ${modulePaths && modulePaths.length}, ${modulesCachePath}`);
- } catch (e) {
- Logger.debug(`Cached module paths not found: ${modulesCachePath}`);
- }
- return modulePaths;
-}
-
export function setContext(context: BuildContext) {
_context = context;
}
@@ -120,4 +137,4 @@ export function changeExtension(filePath: string, newExtension: string) {
const extensionlessfileName = basename(filePath, extension);
const newFileName = extensionlessfileName + newExtension;
return join(dir, newFileName);
-}
\ No newline at end of file
+}
diff --git a/src/util/interfaces.ts b/src/util/interfaces.ts
index 60121fe8..e24660e5 100644
--- a/src/util/interfaces.ts
+++ b/src/util/interfaces.ts
@@ -10,16 +10,23 @@ export interface BuildContext {
moduleFiles?: string[];
isProd?: boolean;
isWatch?: boolean;
- isUpdate?: boolean;
- fullBuildCompleted?: boolean;
+
bundler?: string;
- useTranspileCache?: boolean;
- useBundleCache?: boolean;
- useSassCache?: boolean;
fileCache?: FileCache;
- successfulSass?: boolean;
inlineTemplates?: boolean;
webpackWatch?: any;
+
+ sassState?: BuildState;
+ transpileState?: BuildState;
+ templateState?: BuildState;
+ bundleState?: BuildState;
+}
+
+
+export enum BuildState {
+ SuccessfulBuild,
+ RequiresUpdate,
+ RequiresBuild
}
@@ -49,6 +56,7 @@ export interface TaskInfo {
defaultConfigFile: string;
}
+
export interface File {
path: string;
content: string;
diff --git a/src/util/logger-diagnostics.ts b/src/util/logger-diagnostics.ts
index 47d73c4a..f7e0a377 100644
--- a/src/util/logger-diagnostics.ts
+++ b/src/util/logger-diagnostics.ts
@@ -1,21 +1,23 @@
import { BuildContext } from './interfaces';
import { Diagnostic, Logger, PrintLine } from './logger';
-import { join } from 'path';
-import { readFileSync, writeFileSync, unlinkSync } from 'fs';
import { titleCase } from './helpers';
+import { join } from 'path';
+import { readFileSync, unlinkSync, writeFileSync } from 'fs';
import * as chalk from 'chalk';
-export function printDiagnostics(context: BuildContext, type: string, diagnostics: Diagnostic[]) {
- if (diagnostics.length) {
- let content: string[] = [];
- diagnostics.forEach(d => {
- consoleLogDiagnostic(d);
- content.push(generateDiagnosticHtml(d));
- });
+export function printDiagnostics(context: BuildContext, diagnosticsType: string, diagnostics: Diagnostic[], consoleLogDiagnostics: boolean, writeHtmlDiagnostics: boolean) {
+ if (diagnostics && diagnostics.length) {
+
+ if (consoleLogDiagnostics) {
+ diagnostics.forEach(consoleLogDiagnostic);
+ }
- const fileName = getDiagnosticsFileName(context.buildDir, type);
- writeFileSync(fileName, content.join('\n'), { encoding: 'utf8' });
+ if (writeHtmlDiagnostics) {
+ const content = diagnostics.map(generateDiagnosticHtml);
+ const fileName = getDiagnosticsFileName(context.buildDir, diagnosticsType);
+ writeFileSync(fileName, content.join('\n'), { encoding: 'utf8' });
+ }
}
}
@@ -99,87 +101,144 @@ function consoleHighlightError(errorLine: string, errorCharStart: number, errorL
}
-export function clearDiagnosticsHtmlSync(context: BuildContext, type: string) {
+let diagnosticsHtmlCache: {[key: string]: any} = {};
+
+export function clearDiagnosticsCache() {
+ diagnosticsHtmlCache = {};
+}
+
+export function clearDiagnostics(context: BuildContext, type: string) {
try {
+ delete diagnosticsHtmlCache[type];
unlinkSync(getDiagnosticsFileName(context.buildDir, type));
} catch (e) {}
}
-export function readDiagnosticsHtmlSync(buildDir: string) {
- let content = '';
+export function hasDiagnostics(buildDir: string) {
+ loadDiagnosticsHtml(buildDir);
+ const keys = Object.keys(diagnosticsHtmlCache);
+ for (var i = 0; i < keys.length; i++) {
+ if (typeof diagnosticsHtmlCache[keys[i]] === 'string') {
+ return true;
+ }
+ }
+
+ return false;
+}
+
+
+function loadDiagnosticsHtml(buildDir: string) {
try {
- content = readFileSync(getDiagnosticsFileName(buildDir, 'typescript'), 'utf8') + '\n';
- } catch (e) {}
+ if (diagnosticsHtmlCache[DiagnosticsType.TypeScript] === undefined) {
+ diagnosticsHtmlCache[DiagnosticsType.TypeScript] = readFileSync(getDiagnosticsFileName(buildDir, DiagnosticsType.TypeScript), 'utf8');
+ }
+ } catch (e) {
+ diagnosticsHtmlCache[DiagnosticsType.TypeScript] = false;
+ }
try {
- content += readFileSync(getDiagnosticsFileName(buildDir, 'sass'), 'utf8');
- } catch (e) {}
+ if (diagnosticsHtmlCache[DiagnosticsType.Sass] === undefined) {
+ diagnosticsHtmlCache[DiagnosticsType.Sass] = readFileSync(getDiagnosticsFileName(buildDir, DiagnosticsType.Sass), 'utf8');
+ }
+ } catch (e) {
+ diagnosticsHtmlCache[DiagnosticsType.Sass] = false;
+ }
+}
+
+
+export function injectDiagnosticsHtml(buildDir: string, content: any) {
+ if (!hasDiagnostics(buildDir)) {
+ return content;
+ }
+
+ let contentStr = content.toString();
+
+ const diagnosticsHtml: string[] = [];
+ diagnosticsHtml.push(``);
+ diagnosticsHtml.push(getDiagnosticsHtmlContent(buildDir));
+ diagnosticsHtml.push(`
`);
+
+ let match = contentStr.match(/(?![\s\S]*)/i);
+ if (match) {
+ contentStr = contentStr.replace(match[0], match[0] + '\n' + diagnosticsHtml.join('\n'));
+ } else {
+ contentStr = diagnosticsHtml.join('\n') + contentStr;
+ }
+
+ return contentStr;
+}
+
+
+export function getDiagnosticsHtmlContent(buildDir: string) {
+ loadDiagnosticsHtml(buildDir);
- if (content.length) {
- return generateHtml('Build Error', content);
+ const diagnosticsHtml: string[] = [];
+
+ const keys = Object.keys(diagnosticsHtmlCache);
+ for (var i = 0; i < keys.length; i++) {
+ if (typeof diagnosticsHtmlCache[keys[i]] === 'string') {
+ diagnosticsHtml.push(diagnosticsHtmlCache[keys[i]]);
+ }
}
- return null;
+ return diagnosticsHtml.join('\n');
}
function generateDiagnosticHtml(d: Diagnostic) {
const c: string[] = [];
- c.push(``);
+ c.push(`
`);
- c.push(``); // .ion-diagnostic-masthead
- c.push(`
`);
+ c.push(`
`);
- c.push(``);
+ c.push(``);
if (d.lines && d.lines.length) {
- c.push(`
`);
+ c.push(`
`);
- c.push(`
`);
+ c.push(``);
const lines = removeWhitespaceIndent(d.lines);
lines.forEach(l => {
let trCssClass = '';
- let lineNumberCssClass = `blob-num ${d.syntax}-line-number`;
let code = l.text;
if (l.errorCharStart > -1) {
code = htmlHighlightError(code, l.errorCharStart, l.errorLength);
- trCssClass = ' class="error-line"';
+ trCssClass = ' class="ion-diagnostic-error-line"';
}
- code = jsHtmlSyntaxHighlight(code);
-
c.push(``);
- c.push(` | `);
+ c.push(` | `);
- c.push(`${code} | `);
+ c.push(`${code} | `);
c.push(`
`);
});
c.push(`
`);
- c.push(``); // .blob-wrapper
+ c.push(``); // .ion-diagnostic-blob
}
- c.push(``); // .file
+ c.push(``); // .ion-diagnostic-file
- c.push(``); // .diagnostic
+ c.push(``); // .ion-diagnostic
return c.join('\n');
}
@@ -191,7 +250,7 @@ function htmlHighlightError(errorLine: string, errorCharStart: number, errorLeng
for (var i = 0; i < lineLength; i++) {
var chr = errorLine.charAt(i);
if (i >= errorCharStart && i < errorCharStart + errorLength) {
- chr = `${chr === '' ? ' ' : chr}`;
+ chr = `${chr === '' ? ' ' : chr}`;
}
lineChars.push(chr);
}
@@ -200,195 +259,6 @@ function htmlHighlightError(errorLine: string, errorCharStart: number, errorLeng
}
-const DIAGNOSTICS_CSS = `
-
-* {
- box-sizing: border-box;
-}
-
-body {
- font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
- font-size: 14px;
- line-height: 1.5;
- color: #333;
- background-color: #fff;
- word-wrap: break-word;
- margin: 0;
- padding: 0;
-}
-
-main {
- margin: 0;
- padding: 15px;
-}
-
-h2 {
- margin: 0;
- font-size: 18px;
- color: #222222;
-}
-
-p {
- margin-top: 8px;
- color: #666666;
-}
-
-table {
- border-spacing: 0;
- border-collapse: collapse;
-}
-
-td, th {
- padding: 0;
-}
-
-.diagnostic {
- margin-bottom: 40px;
- border: 1px solid #ddd;
- border-radius: 3px;
-}
-
-.diagnostic-header {
- padding: 8px 12px 0 12px;
-}
-
-.file {
- position: relative;
- margin-top: 16px;
- border-top: 1px solid #ddd;
-}
-
-.file-header {
- padding: 5px 10px;
- background-color: #f7f7f7;
- border-bottom: 1px solid #d8d8d8;
- border-top-left-radius: 2px;
- border-top-right-radius: 2px;
-}
-
-.blob-wrapper {
- overflow-x: auto;
- overflow-y: hidden;
- border-bottom-right-radius: 3px;
- border-bottom-left-radius: 3px;
-}
-
-.tab-size {
- -moz-tab-size: 2;
- -o-tab-size: 2;
- tab-size: 2;
-}
-
-.blob-num {
- width: 1%;
- min-width: 50px;
- padding-right: 10px;
- padding-left: 10px;
- font-family: Consolas, "Liberation Mono", Menlo, Courier, monospace;
- font-size: 12px;
- line-height: 20px;
- color: rgba(0,0,0,0.3);
- text-align: right;
- white-space: nowrap;
- vertical-align: top;
- -webkit-user-select: none;
- -moz-user-select: none;
- -ms-user-select: none;
- user-select: none;
- border: solid #eee;
- border-width: 0 1px 0 0;
-}
-
-.blob-num::before {
- content: attr(data-line-number);
-}
-
-.error-line .blob-num {
- background-color: #ffdddd;
- border-color: #f1c0c0;
-}
-
-.error-chr {
- position: relative;
-}
-
-.error-chr:before {
- content: "";
- position: absolute;
- z-index: -1;
- top: -3px;
- left: 0px;
- width: 8px;
- height: 20px;
- background-color: #ffdddd;
-}
-
-.blob-code {
- position: relative;
- padding-right: 10px;
- padding-left: 10px;
- line-height: 20px;
- vertical-align: top;
-}
-
-.blob-code-inner {
- overflow: visible;
- font-family: Consolas, "Liberation Mono", Menlo, Courier, monospace;
- font-size: 12px;
- color: #333;
- word-wrap: normal;
- white-space: pre;
-}
-
-.blob-code-inner::before {
- content: "";
-}
-
-.js-keyword,
-.css-prop {
- color: #183691;
-}
-
-.js-comment,
-.sass-comment {
- color: #969896;
-}
-
-.system-info {
- font-size: 10px;
- color: #999;
-}
-
-`;
-
-
-function generateHtml(title: string, content: string) {
- return `
-
-
-
-
-${escapeHtml(title)}
-
-
-
-
-
-
-
-
-${content}
-
-${getSystemInfo().join('\n')}
-
-
-
-`;
-}
-
-
function jsConsoleSyntaxHighlight(text: string) {
if (text.trim().startsWith('//')) {
return chalk.dim(text);
@@ -431,54 +301,6 @@ function cssConsoleSyntaxHighlight(text: string, errorCharStart: number) {
return chars.join('');
}
-
-function jsHtmlSyntaxHighlight(text: string) {
- if (text.trim().startsWith('//')) {
- return ``;
- }
-
- const words = text.split(' ').map(word => {
- if (JS_KEYWORDS.indexOf(word) > -1) {
- return `${word}`;
- }
- return word;
- });
-
- return words.join(' ');
-}
-
-
-function cssHtmlSyntaxHighlight(text: string, errorCharStart: number) {
- if (text.trim().startsWith('//')) {
- return ``;
- }
-
- let cssProp = true;
- const safeChars = 'abcdefghijklmnopqrstuvwxyz-_';
- const notProp = '.#,:}@$[]/*';
-
- const chars: string[] = [];
-
- for (var i = 0; i < text.length; i++) {
- var c = text.charAt(i);
-
- if (c === ';' || c === '{') {
- cssProp = true;
- } else if (notProp.indexOf(c) > -1) {
- cssProp = false;
- }
- if (cssProp && safeChars.indexOf(c.toLowerCase()) > -1) {
- chars.push(`${c}`);
- continue;
- }
-
- chars.push(c);
- }
-
- return chars.join('');
-}
-
-
function escapeHtml(unsafe: string) {
return unsafe
.replace(/&/g, '&')
@@ -527,7 +349,6 @@ function eachLineHasLeadingWhitespace(lines: PrintLine[]) {
}
-
const JS_KEYWORDS = [
'as',
'break',
@@ -570,18 +391,6 @@ function getDiagnosticsFileName(buildDir: string, type: string) {
}
-export function getSystemInfo() {
- const systemData: string[] = [];
-
- try {
- // const ionicFramework = '2.0.0';
- // systemData.push(`Ionic Framework: ${ionicFramework}`);
- } catch (e) {}
-
- return systemData;
-}
-
-
function isMeaningfulLine(line: string) {
if (line) {
line = line.trim();
@@ -594,4 +403,8 @@ function isMeaningfulLine(line: string) {
const MEH_LINES = [';', ':', '{', '}', '(', ')', '/**', '/*', '*/', '*', '({', '})'];
-const FAVICON = '';
+export const DiagnosticsType = {
+ TypeScript: 'typescript',
+ Sass: 'sass',
+ TsLint: 'tslint'
+};
diff --git a/src/util/logger-sass.ts b/src/util/logger-sass.ts
index 940bb569..d1c068d1 100644
--- a/src/util/logger-sass.ts
+++ b/src/util/logger-sass.ts
@@ -1,25 +1,10 @@
import { BuildContext } from './interfaces';
import { Diagnostic, Logger, PrintLine } from './logger';
-import { printDiagnostics, clearDiagnosticsHtmlSync } from './logger-diagnostics';
import { readFileSync } from 'fs';
import { SassError } from 'node-sass';
-export function runDiagnostics(context: BuildContext, sassError: SassError) {
- const diagnostics = loadDiagnostic(context, sassError);
-
- printDiagnostics(context, 'sass', diagnostics);
-
- return diagnostics;
-}
-
-
-export function clearSassDiagnostics(context: BuildContext) {
- clearDiagnosticsHtmlSync(context, 'sass');
-}
-
-
-function loadDiagnostic(context: BuildContext, sassError: SassError) {
+export function runSassDiagnostics(context: BuildContext, sassError: SassError) {
if (!sassError) {
return [];
}
diff --git a/src/util/logger-tslint.ts b/src/util/logger-tslint.ts
index 5b702e4e..a71e46b9 100644
--- a/src/util/logger-tslint.ts
+++ b/src/util/logger-tslint.ts
@@ -1,16 +1,11 @@
import { BuildContext } from './interfaces';
-import { printDiagnostics } from './logger-diagnostics';
import { Diagnostic, Logger, PrintLine } from './logger';
-export function runDiagnostics(context: BuildContext, failures: RuleFailure[]) {
- const diagnostics = failures.map(failure => {
+export function runTsLintDiagnostics(context: BuildContext, failures: RuleFailure[]) {
+ return failures.map(failure => {
return loadDiagnostic(context, failure);
});
-
- printDiagnostics(context, 'tslint', diagnostics);
-
- return diagnostics;
}
diff --git a/src/util/logger-typescript.ts b/src/util/logger-typescript.ts
index c43f63c8..63f71c46 100644
--- a/src/util/logger-typescript.ts
+++ b/src/util/logger-typescript.ts
@@ -1,5 +1,4 @@
import { BuildContext } from './interfaces';
-import { printDiagnostics, clearDiagnosticsHtmlSync } from './logger-diagnostics';
import { Diagnostic, Logger, PrintLine } from './logger';
import * as ts from 'typescript';
@@ -9,19 +8,10 @@ import * as ts from 'typescript';
* error reporting within a terminal. So, yeah, let's code it up, shall we?
*/
-export function runDiagnostics(context: BuildContext, tsDiagnostics: ts.Diagnostic[]) {
- const diagnostics = tsDiagnostics.map(tsDiagnostic => {
+export function runTypeScriptDiagnostics(context: BuildContext, tsDiagnostics: ts.Diagnostic[]) {
+ return tsDiagnostics.map(tsDiagnostic => {
return loadDiagnostic(context, tsDiagnostic);
});
-
- printDiagnostics(context, 'typescript', diagnostics);
-
- return diagnostics;
-}
-
-
-export function clearTypeScriptDiagnostics(context: BuildContext) {
- clearDiagnosticsHtmlSync(context, 'typescript');
}
diff --git a/src/util/logger.ts b/src/util/logger.ts
index a1ee54b3..c4437c45 100644
--- a/src/util/logger.ts
+++ b/src/util/logger.ts
@@ -1,4 +1,3 @@
-import { emit, EventType } from './events';
import { join } from 'path';
import { isDebugMode } from './config';
import { readJSONSync } from 'fs-extra';
@@ -7,7 +6,6 @@ import * as chalk from 'chalk';
export class BuildError extends Error {
hasBeenLogged = false;
- updatedDiagnostics = false;
constructor(err?: any) {
super();
@@ -26,9 +24,6 @@ export class BuildError extends Error {
if (typeof err.hasBeenLogged === 'boolean') {
this.hasBeenLogged = err.hasBeenLogged;
}
- if (typeof err.updatedDiagnostics === 'boolean') {
- this.updatedDiagnostics = err.updatedDiagnostics;
- }
}
}
@@ -37,8 +32,7 @@ export class BuildError extends Error {
message: this.message,
name: this.name,
stack: this.stack,
- hasBeenLogged: this.hasBeenLogged,
- updatedDiagnostics: this.updatedDiagnostics
+ hasBeenLogged: this.hasBeenLogged
};
}
}
@@ -67,53 +61,41 @@ export class Logger {
msg += memoryUsage();
}
Logger.info(msg);
-
- const taskEvent: TaskEvent = {
- scope: this.scope.split(' ')[0],
- type: 'start',
- msg: `${scope} started ...`
- };
- emit(EventType.TaskEvent, taskEvent);
}
- ready(chalkColor?: Function) {
- this.completed('ready', chalkColor);
+ ready(color?: string, bold?: boolean) {
+ this.completed('ready', color, bold);
}
- finish(chalkColor?: Function) {
- this.completed('finished', chalkColor);
+ finish(color?: string, bold?: boolean) {
+ this.completed('finished', color, bold);
}
- private completed(type: string, chalkColor: Function) {
+ private completed(type: string, color: string, bold: boolean) {
+ const duration = Date.now() - this.start;
+ let time: string;
- const taskEvent: TaskEvent = {
- scope: this.scope.split(' ')[0],
- type: type
- };
-
- taskEvent.duration = Date.now() - this.start;
-
- if (taskEvent.duration > 1000) {
- taskEvent.time = 'in ' + (taskEvent.duration / 1000).toFixed(2) + ' s';
+ if (duration > 1000) {
+ time = 'in ' + (duration / 1000).toFixed(2) + ' s';
} else {
- let ms = parseFloat((taskEvent.duration).toFixed(3));
+ let ms = parseFloat((duration).toFixed(3));
if (ms > 0) {
- taskEvent.time = 'in ' + taskEvent.duration + ' ms';
+ time = 'in ' + duration + ' ms';
} else {
- taskEvent.time = 'in less than 1 ms';
+ time = 'in less than 1 ms';
}
}
- taskEvent.msg = `${this.scope} ${taskEvent.type} ${taskEvent.time}`;
- emit(EventType.TaskEvent, taskEvent);
-
let msg = `${this.scope} ${type}`;
- if (chalkColor) {
- msg = chalkColor(msg);
+ if (color) {
+ msg = (chalk)[color](msg);
+ }
+ if (bold) {
+ msg = chalk.bold(msg);
}
- msg += ' ' + chalk.dim(taskEvent.time);
+ msg += ' ' + chalk.dim(time);
if (isDebugMode()) {
msg += memoryUsage();
@@ -128,14 +110,6 @@ export class Logger {
return;
}
- // only emit the event if it's a valid error
- const taskEvent: TaskEvent = {
- scope: this.scope.split(' ')[0],
- type: 'failed',
- msg: this.scope + ' failed'
- };
- emit(EventType.TaskEvent, taskEvent);
-
if (err instanceof BuildError) {
let failedMsg = `${this.scope} failed`;
if (err.message) {
@@ -161,6 +135,10 @@ export class Logger {
return err;
}
+ setStartTime(startTime: number) {
+ this.start = startTime;
+ }
+
/**
* Does not print out a time prefix or color any text. Only prefix
* with whitespace so the message is lined up with timestamped logs.
@@ -172,15 +150,31 @@ export class Logger {
}
/**
- * Prints out a dim colored timestamp prefix.
+ * Prints out a dim colored timestamp prefix, with optional color
+ * and bold message.
*/
- static info(...msg: any[]) {
- const lines = Logger.wordWrap(msg);
+ static info(msg: string, color?: string, bold?: boolean) {
+ const lines = Logger.wordWrap([msg]);
if (lines.length) {
let prefix = timePrefix();
- lines[0] = chalk.dim(prefix) + lines[0].substr(prefix.length);
+ let lineOneMsg = lines[0].substr(prefix.length);
+ if (color) {
+ lineOneMsg = (chalk)[color](lineOneMsg);
+ }
+ if (bold) {
+ lineOneMsg = chalk.bold(lineOneMsg);
+ }
+ lines[0] = chalk.dim(prefix) + lineOneMsg;
}
- lines.forEach(line => {
+ lines.forEach((line, lineIndex) => {
+ if (lineIndex > 0) {
+ if (color) {
+ line = (chalk)[color](line);
+ }
+ if (bold) {
+ line = chalk.bold(line);
+ }
+ }
console.log(line);
});
}
@@ -355,25 +349,6 @@ function memoryUsage() {
}
-export function getAppScriptsVersion() {
- let rtn = '';
- try {
- const packageJson = readJSONSync(join(__dirname, '..', '..', 'package.json'));
- rtn = packageJson.version || '';
- } catch (e) {}
- return rtn;
-}
-
-
-export interface TaskEvent {
- scope: string;
- type: string;
- duration?: number;
- time?: string;
- msg?: string;
-}
-
-
export interface Diagnostic {
level: string;
syntax: string;
diff --git a/src/watch.ts b/src/watch.ts
index 431c0db2..d1c4ccb2 100644
--- a/src/watch.ts
+++ b/src/watch.ts
@@ -1,9 +1,9 @@
-import { build, fullBuildUpdate } from './build';
-import { BuildContext, TaskInfo } from './util/interfaces';
-import { BuildError, IgnorableError, Logger } from './util/logger';
+import * as buildTask from './build';
+import { BuildContext, BuildState, TaskInfo } from './util/interfaces';
+import { BuildError, Logger } from './util/logger';
import { fillConfigDefaults, generateContext, getUserConfigFile, replacePathVars, setIonicEnvironment } from './util/config';
-import { join, normalize } from 'path';
-import * as chalk from 'chalk';
+import { join, normalize, extname } from 'path';
+import { canRunTranspileUpdate } from './transpile';
import * as chokidar from 'chokidar';
@@ -16,17 +16,20 @@ export function watch(context?: BuildContext, configFile?: string) {
// force watch options
context.isProd = false;
context.isWatch = true;
- context.fullBuildCompleted = false;
+
+ context.sassState = BuildState.RequiresBuild;
+ context.transpileState = BuildState.RequiresBuild;
+ context.bundleState = BuildState.RequiresBuild;
const logger = new Logger('watch');
function buildDone() {
return startWatchers(context, configFile).then(() => {
- logger.ready(chalk.green);
+ logger.ready();
});
}
- return build(context)
+ return buildTask.build(context)
.then(buildDone, buildDone)
.catch(err => {
throw logger.fail(err);
@@ -82,20 +85,9 @@ function startWatcher(index: number, watcher: Watcher, context: BuildContext, wa
filePath = join(context.rootDir, filePath);
- context.isUpdate = true;
-
Logger.debug(`watch callback start, id: ${watchCount}, isProd: ${context.isProd}, event: ${event}, path: ${filePath}`);
- function taskDone() {
- Logger.newLine();
- Logger.info(chalk.green.bold('watch ready'));
- Logger.newLine();
- }
-
const callbackToExecute = function(event: string, filePath: string, context: BuildContext, watcher: Watcher) {
- if (!context.fullBuildCompleted) {
- return fullBuildUpdate(event, filePath, context);
- }
return watcher.callback(event, filePath, context);
};
@@ -103,15 +95,11 @@ function startWatcher(index: number, watcher: Watcher, context: BuildContext, wa
.then(() => {
Logger.debug(`watch callback complete, id: ${watchCount}, isProd: ${context.isProd}, event: ${event}, path: ${filePath}`);
watchCount++;
- taskDone();
})
.catch(err => {
Logger.debug(`watch callback error, id: ${watchCount}, isProd: ${context.isProd}, event: ${event}, path: ${filePath}`);
Logger.debug(`${err}`);
watchCount++;
- if (!(err instanceof IgnorableError)) {
- taskDone();
- }
});
});
@@ -151,6 +139,127 @@ export function prepareWatcher(context: BuildContext, watcher: Watcher) {
}
+let queuedChangedFiles: ChangedFile[] = [];
+let queuedChangeFileTimerId: any;
+export interface ChangedFile {
+ event: string;
+ filePath: string;
+ ext: string;
+}
+
+export function buildUpdate(event: string, filePath: string, context: BuildContext) {
+ const changedFile: ChangedFile = {
+ event: event,
+ filePath: filePath,
+ ext: extname(filePath).toLowerCase()
+ };
+
+ // do not allow duplicates
+ if (!queuedChangedFiles.some(f => f.filePath === filePath)) {
+ queuedChangedFiles.push(changedFile);
+
+ // debounce our build update incase there are multiple files
+ clearTimeout(queuedChangeFileTimerId);
+
+ // run this code in a few milliseconds if another hasn't come in behind it
+ queuedChangeFileTimerId = setTimeout(() => {
+ // figure out what actually needs to be rebuilt
+ const buildData = runBuildUpdate(context, queuedChangedFiles);
+
+ // clear out all the files that are queued up for the build update
+ queuedChangedFiles.length = 0;
+
+ if (buildData) {
+ // cool, we've got some build updating to do ;)
+ buildTask.buildUpdate(buildData.event, buildData.filePath, context);
+ }
+ }, BUILD_UPDATE_DEBOUNCE_MS);
+ }
+
+ return Promise.resolve();
+}
+
+
+export function runBuildUpdate(context: BuildContext, changedFiles: ChangedFile[]) {
+ if (!changedFiles || !changedFiles.length) {
+ return null;
+ }
+
+ // create the data which will be returned
+ const data = {
+ event: changedFiles.map(f => f.event).find(ev => ev !== 'change') || 'change',
+ filePath: changedFiles[0].filePath,
+ changedFiles: changedFiles.map(f => f.filePath)
+ };
+
+ const tsFiles = changedFiles.filter(f => f.ext === '.ts');
+ if (tsFiles.length > 1) {
+ // multiple .ts file changes
+ // if there is more than one ts file changing then
+ // let's just do a full transpile build
+ context.transpileState = BuildState.RequiresBuild;
+
+ } else if (tsFiles.length) {
+ // only one .ts file changed
+ if (canRunTranspileUpdate(tsFiles[0].event, tsFiles[0].filePath, context)) {
+ // .ts file has only changed, it wasn't a file add/delete
+ // we can do the quick typescript update on this changed file
+ context.transpileState = BuildState.RequiresUpdate;
+
+ } else {
+ // .ts file was added or deleted, we need a full rebuild
+ context.transpileState = BuildState.RequiresBuild;
+ }
+ }
+
+ const sassFiles = changedFiles.filter(f => f.ext === '.scss');
+ if (sassFiles.length) {
+ // .scss file was changed/added/deleted, lets do a sass update
+ context.sassState = BuildState.RequiresUpdate;
+ }
+
+ const sassFilesNotChanges = changedFiles.filter(f => f.ext === '.ts' && f.event !== 'change');
+ if (sassFilesNotChanges.length) {
+ // .ts file was either added or deleted, so we'll have to
+ // run sass again to add/remove that .ts file's potential .scss file
+ context.sassState = BuildState.RequiresUpdate;
+ }
+
+ const htmlFiles = changedFiles.filter(f => f.ext === '.html');
+ if (htmlFiles.length) {
+ if (context.bundleState === BuildState.SuccessfulBuild && htmlFiles.every(f => f.event === 'change')) {
+ // .html file was changed
+ // just doing a template update is fine
+ context.templateState = BuildState.RequiresUpdate;
+
+ } else {
+ // .html file was added/deleted
+ // we should do a full transpile build because of this
+ context.transpileState = BuildState.RequiresBuild;
+ }
+ }
+
+ if (context.transpileState === BuildState.RequiresUpdate || context.transpileState === BuildState.RequiresBuild) {
+ if (context.bundleState === BuildState.SuccessfulBuild || context.bundleState === BuildState.RequiresUpdate) {
+ // transpiling needs to happen
+ // and there has already been a successful bundle before
+ // so let's just do a bundle update
+ context.bundleState = BuildState.RequiresUpdate;
+ } else {
+ // transpiling needs to happen
+ // but we've never successfully bundled before
+ // so let's do a full bundle build
+ context.bundleState = BuildState.RequiresBuild;
+ }
+ }
+
+ // guess which file is probably the most important here
+ data.filePath = tsFiles.concat(sassFiles, htmlFiles)[0].filePath;
+
+ return data;
+}
+
+
const taskInfo: TaskInfo = {
fullArg: '--watch',
shortArg: '-w',
@@ -181,3 +290,5 @@ export interface Watcher {
}
let watchCount = 0;
+
+const BUILD_UPDATE_DEBOUNCE_MS = 20;
diff --git a/src/webpack.ts b/src/webpack.ts
index f6a2058f..ae9a3fb2 100644
--- a/src/webpack.ts
+++ b/src/webpack.ts
@@ -1,7 +1,7 @@
import { FileCache } from './util/file-cache';
-import { BuildContext, File, TaskInfo } from './util/interfaces';
+import { BuildContext, BuildState, File, TaskInfo } from './util/interfaces';
import { BuildError, IgnorableError, Logger } from './util/logger';
-import { changeExtension, readFileAsync, setContext, setModulePathsCache } from './util/helpers';
+import { changeExtension, readFileAsync, setContext } from './util/helpers';
import { emit, EventType } from './util/events';
import { fillConfigDefaults, generateContext, getUserConfigFile, replacePathVars } from './util/config';
import { extname, join } from 'path';
@@ -34,9 +34,11 @@ export function webpack(context: BuildContext, configFile: string) {
return webpackWorker(context, configFile)
.then(() => {
+ context.bundleState = BuildState.SuccessfulBuild;
logger.finish();
})
.catch(err => {
+ context.bundleState = BuildState.RequiresBuild;
throw logger.fail(err);
});
}
@@ -69,8 +71,10 @@ export function webpackUpdate(event: string, path: string, context: BuildContext
Logger.debug('webpackUpdate: Incremental Build Done, processing Data');
return webpackBuildComplete(stats, context, webpackConfig);
}).then(() => {
+ context.bundleState = BuildState.SuccessfulBuild;
return logger.finish();
}).catch(err => {
+ context.bundleState = BuildState.RequiresBuild;
if (err instanceof IgnorableError) {
throw err;
}
@@ -112,11 +116,6 @@ function webpackBuildComplete(stats: any, context: BuildContext, webpackConfig:
context.moduleFiles = files;
- // async cache all the module paths so we don't need
- // to always bundle to know which modules are used
- setModulePathsCache(context.moduleFiles);
-
- emit(EventType.BundleFinished, getOutputDest(context, webpackConfig));
return Promise.resolve();
}
diff --git a/src/webpack/ionic-webpack-factory.ts b/src/webpack/ionic-webpack-factory.ts
index ca5400cc..d7d6115f 100644
--- a/src/webpack/ionic-webpack-factory.ts
+++ b/src/webpack/ionic-webpack-factory.ts
@@ -1,7 +1,8 @@
import { IonicEnvironmentPlugin } from './ionic-environment-plugin';
import { getContext } from '../util/helpers';
+
export function getIonicEnvironmentPlugin() {
const context = getContext();
return new IonicEnvironmentPlugin(context.fileCache);
-}
\ No newline at end of file
+}
diff --git a/src/worker-client.ts b/src/worker-client.ts
index 028e44d8..4bb979e4 100644
--- a/src/worker-client.ts
+++ b/src/worker-client.ts
@@ -2,7 +2,6 @@ import { BuildContext, WorkerProcess, WorkerMessage } from './util/interfaces';
import { BuildError, Logger } from './util/logger';
import { fork, ChildProcess } from 'child_process';
import { join } from 'path';
-import { emit, EventType } from './util/events';
export function runWorker(taskModule: string, taskWorker: string, context: BuildContext, workerConfig: any) {
@@ -30,18 +29,10 @@ export function runWorker(taskModule: string, taskWorker: string, context: Build
worker.on('message', (msg: WorkerMessage) => {
if (msg.error) {
- const buildErrorError = new BuildError(msg.error);
- if (buildErrorError.updatedDiagnostics) {
- emit(EventType.UpdatedDiagnostics);
- }
- reject(buildErrorError);
+ reject(new BuildError(msg.error));
} else if (msg.reject) {
- const buildErrorReject = new BuildError(msg.reject);
- if (buildErrorReject.updatedDiagnostics) {
- emit(EventType.UpdatedDiagnostics);
- }
- reject(buildErrorReject);
+ reject(new BuildError(msg.reject));
} else {
resolve(msg.resolve);
diff --git a/src/worker-process.ts b/src/worker-process.ts
index f86c3583..91550905 100644
--- a/src/worker-process.ts
+++ b/src/worker-process.ts
@@ -11,7 +11,7 @@ process.on('message', (msg: WorkerMessage) => {
.then((val: any) => {
taskResolve(msg.taskModule, msg.taskWorker, val);
}, (val: any) => {
- taskReject(msg.taskModule, msg.taskWorker, val)
+ taskReject(msg.taskModule, msg.taskWorker, val);
})
.catch((err: any) => {
taskError(msg.taskModule, msg.taskWorker, err);