2018-07-16 03:49:56 -04:00
/ * *
* @license
* Copyright Google Inc . All Rights Reserved .
*
* Use of this source code is governed by an MIT - style license that can be
* found in the LICENSE file at https : //angular.io/license
* /
2019-08-29 11:47:54 -04:00
/// <reference types="node" />
import * as os from 'os' ;
2019-06-06 15:22:32 -04:00
import { AbsoluteFsPath , FileSystem , absoluteFrom , getFileSystem , join } from '../../../src/ngtsc/file_system' ;
refactor(ngcc): take advantage of early knowledge about format property processability (#32427)
In the past, a task's processability didn't use to be known in advance.
It was possible that a task would be created and added to the queue
during the analysis phase and then later (during the compilation phase)
it would be found out that the task (i.e. the associated format
property) was not processable.
As a result, certain checks had to be delayed, until a task's processing
had started or even until all tasks had been processed. Examples of
checks that had to be delayed are:
- Whether a task can be skipped due to `compileAllFormats: false`.
- Whether there were entry-points for which no format at all was
successfully processed.
It turns out that (as made clear by the refactoring in 9537b2ff8), once
a task starts being processed it is expected to either complete
successfully (with the associated format being processed) or throw an
error (in which case the process will exit). In other words, a task's
processability is known in advance.
This commit takes advantage of this fact by moving certain checks
earlier in the process (e.g. in the analysis phase instead of the
compilation phase), which in turn allows avoiding some unnecessary work.
More specifically:
- When `compileAllFormats` is `false`, tasks are created _only_ for the
first suitable format property for each entry-point, since the rest of
the tasks would have been skipped during the compilation phase anyway.
This has the following advantages:
1. It avoids the slight overhead of generating extraneous tasks and
then starting to process them (before realizing they should be
skipped).
2. In a potential future parallel execution mode, unnecessary tasks
might start being processed at the same time as the first (useful)
task, even if their output would be later discarded, wasting
resources. Alternatively, extra logic would have to be added to
prevent this from happening. The change in this commit avoids these
issues.
- When an entry-point is not processable, an error will be thrown
upfront without having to wait for other tasks to be processed before
failing.
PR Close #32427
2019-08-28 18:33:15 -04:00
import { Folder , MockFileSystem , TestFile , runInEachFileSystem } from '../../../src/ngtsc/file_system/testing' ;
2019-06-06 15:22:32 -04:00
import { loadStandardTestFiles , loadTestFiles } from '../../../test/helpers' ;
2019-04-06 10:35:13 -04:00
import { mainNgcc } from '../../src/main' ;
2019-04-06 10:35:40 -04:00
import { markAsProcessed } from '../../src/packages/build_marker' ;
import { EntryPointJsonProperty , EntryPointPackageJson , SUPPORTED_FORMAT_PROPERTIES } from '../../src/packages/entry_point' ;
refactor(ngcc): add support for asynchronous execution (#32427)
Previously, `ngcc`'s programmatic API would run and complete
synchronously. This was necessary for specific usecases (such as how the
`@angular/cli` invokes `ngcc` as part of the TypeScript module
resolution process), but not for others (e.g. running `ivy-ngcc` as a
`postinstall` script).
This commit adds a new option (`async`) that enables turning on
asynchronous execution. I.e. it signals that the caller is OK with the
function call to complete asynchronously, which allows `ngcc` to
potentially run in a more efficient mode.
Currently, there is no difference in the way tasks are executed in sync
vs async mode, but this change sets the ground for adding new execution
options (that require asynchronous operation), such as processing tasks
in parallel on multiple processes.
NOTE:
When using the programmatic API, the default value for `async` is
`false`, thus retaining backwards compatibility.
When running `ngcc` from the command line (i.e. via the `ivy-ngcc`
script), it runs in async mode (to be able to take advantage of future
optimizations), but that is transparent to the caller.
PR Close #32427
2019-08-19 15:58:22 -04:00
import { Transformer } from '../../src/packages/transformer' ;
2019-08-12 11:15:24 -04:00
import { DirectPackageJsonUpdater , PackageJsonUpdater } from '../../src/writing/package_json_updater' ;
2019-04-06 10:35:13 -04:00
import { MockLogger } from '../helpers/mock_logger' ;
feat(ngcc): add a migration for undecorated child classes (#33362)
In Angular View Engine, there are two kinds of decorator inheritance:
1) both the parent and child classes have decorators
This case is supported by InheritDefinitionFeature, which merges some fields
of the definitions (such as the inputs or queries).
2) only the parent class has a decorator
If the child class is missing a decorator, the compiler effectively behaves
as if the parent class' decorator is applied to the child class as well.
This is the "undecorated child" scenario, and this commit adds a migration
to ngcc to support this pattern in Ivy.
This migration has 2 phases. First, the NgModules of the application are
scanned for classes in 'declarations' which are missing decorators, but
whose base classes do have decorators. These classes are the undecorated
children. This scan is performed recursively, so even if a declared class
has a base class that itself inherits a decorator, this case is handled.
Next, a synthetic decorator (either @Component or @Directive) is created
on the child class. This decorator copies some critical information such
as 'selector' and 'exportAs', as well as supports any decorated fields
(@Input, etc). A flag is passed to the decorator compiler which causes a
special feature `CopyDefinitionFeature` to be included on the compiled
definition. This feature copies at runtime the remaining aspects of the
parent definition which `InheritDefinitionFeature` does not handle,
completing the "full" inheritance of the child class' decorator from its
parent class.
PR Close #33362
2019-10-23 15:00:49 -04:00
import { genNodeModules } from './util' ;
2018-07-16 03:49:56 -04:00
2019-08-29 11:47:54 -04:00
2019-06-06 15:22:32 -04:00
const testFiles = loadStandardTestFiles ( { fakeCore : false , rxjs : true } ) ;
2018-07-25 06:06:32 -04:00
2019-06-06 15:22:32 -04:00
runInEachFileSystem ( ( ) = > {
describe ( 'ngcc main()' , ( ) = > {
let _ : typeof absoluteFrom ;
let fs : FileSystem ;
2019-08-12 11:15:24 -04:00
let pkgJsonUpdater : PackageJsonUpdater ;
2019-03-20 09:47:58 -04:00
2019-06-06 15:22:32 -04:00
beforeEach ( ( ) = > {
_ = absoluteFrom ;
fs = getFileSystem ( ) ;
2019-08-12 11:15:24 -04:00
pkgJsonUpdater = new DirectPackageJsonUpdater ( fs ) ;
2019-06-06 15:22:32 -04:00
initMockFileSystem ( fs , testFiles ) ;
2019-08-29 11:47:54 -04:00
// Force single-process execution in unit tests by mocking available CPUs to 1.
spyOn ( os , 'cpus' ) . and . returnValue ( [ { model : 'Mock CPU' } ] ) ;
2019-06-10 08:52:11 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
it ( 'should run ngcc without errors for esm2015' , ( ) = > {
expect ( ( ) = > mainNgcc ( { basePath : '/node_modules' , propertiesToConsider : [ 'esm2015' ] } ) )
. not . toThrow ( ) ;
2019-06-10 08:52:11 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
it ( 'should run ngcc without errors for esm5' , ( ) = > {
expect ( ( ) = > mainNgcc ( {
basePath : '/node_modules' ,
propertiesToConsider : [ 'esm5' ] ,
logger : new MockLogger ( ) ,
} ) )
. not . toThrow ( ) ;
2019-03-20 09:47:59 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
it ( 'should run ngcc without errors when "main" property is not present' , ( ) = > {
mainNgcc ( {
basePath : '/dist' ,
propertiesToConsider : [ 'main' , 'es2015' ] ,
logger : new MockLogger ( ) ,
} ) ;
2019-03-20 09:47:59 -04:00
2019-06-06 15:22:32 -04:00
expect ( loadPackage ( 'local-package' , _ ( '/dist' ) ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
2019-03-20 09:47:59 -04:00
es2015 : '0.0.0-PLACEHOLDER' ,
2019-06-06 15:22:32 -04:00
typings : '0.0.0-PLACEHOLDER' ,
2019-03-20 09:47:59 -04:00
} ) ;
2019-03-20 09:47:58 -04:00
} ) ;
2019-03-20 09:47:58 -04:00
refactor(ngcc): take advantage of early knowledge about format property processability (#32427)
In the past, a task's processability didn't use to be known in advance.
It was possible that a task would be created and added to the queue
during the analysis phase and then later (during the compilation phase)
it would be found out that the task (i.e. the associated format
property) was not processable.
As a result, certain checks had to be delayed, until a task's processing
had started or even until all tasks had been processed. Examples of
checks that had to be delayed are:
- Whether a task can be skipped due to `compileAllFormats: false`.
- Whether there were entry-points for which no format at all was
successfully processed.
It turns out that (as made clear by the refactoring in 9537b2ff8), once
a task starts being processed it is expected to either complete
successfully (with the associated format being processed) or throw an
error (in which case the process will exit). In other words, a task's
processability is known in advance.
This commit takes advantage of this fact by moving certain checks
earlier in the process (e.g. in the analysis phase instead of the
compilation phase), which in turn allows avoiding some unnecessary work.
More specifically:
- When `compileAllFormats` is `false`, tasks are created _only_ for the
first suitable format property for each entry-point, since the rest of
the tasks would have been skipped during the compilation phase anyway.
This has the following advantages:
1. It avoids the slight overhead of generating extraneous tasks and
then starting to process them (before realizing they should be
skipped).
2. In a potential future parallel execution mode, unnecessary tasks
might start being processed at the same time as the first (useful)
task, even if their output would be later discarded, wasting
resources. Alternatively, extra logic would have to be added to
prevent this from happening. The change in this commit avoids these
issues.
- When an entry-point is not processable, an error will be thrown
upfront without having to wait for other tasks to be processed before
failing.
PR Close #32427
2019-08-28 18:33:15 -04:00
it ( 'should throw, if some of the entry-points are unprocessable' , ( ) = > {
const createEntryPoint = ( name : string , prop : EntryPointJsonProperty ) : TestFile [ ] = > {
return [
{
name : _ ( ` /dist/ ${ name } /package.json ` ) ,
contents : ` {"name": " ${ name } ", "typings": "./index.d.ts", " ${ prop } ": "./index.js"} ` ,
} ,
{ name : _ ( ` /dist/ ${ name } /index.js ` ) , contents : 'var DUMMY_DATA = true;' } ,
{ name : _ ( ` /dist/ ${ name } /index.d.ts ` ) , contents : 'export type DummyData = boolean;' } ,
{ name : _ ( ` /dist/ ${ name } /index.metadata.json ` ) , contents : 'DUMMY DATA' } ,
] ;
} ;
loadTestFiles ( [
. . . createEntryPoint ( 'processable-1' , 'es2015' ) ,
. . . createEntryPoint ( 'unprocessable-2' , 'main' ) ,
. . . createEntryPoint ( 'unprocessable-3' , 'main' ) ,
] ) ;
expect ( ( ) = > mainNgcc ( {
basePath : '/dist' ,
propertiesToConsider : [ 'es2015' , 'fesm5' , 'module' ] ,
logger : new MockLogger ( ) ,
} ) )
. toThrowError (
'Unable to process any formats for the following entry-points (tried es2015, fesm5, module): \n' +
` - ${ _ ( '/dist/unprocessable-2' ) } \ n ` +
` - ${ _ ( '/dist/unprocessable-3' ) } ` ) ;
} ) ;
refactor(ngcc): add support for asynchronous execution (#32427)
Previously, `ngcc`'s programmatic API would run and complete
synchronously. This was necessary for specific usecases (such as how the
`@angular/cli` invokes `ngcc` as part of the TypeScript module
resolution process), but not for others (e.g. running `ivy-ngcc` as a
`postinstall` script).
This commit adds a new option (`async`) that enables turning on
asynchronous execution. I.e. it signals that the caller is OK with the
function call to complete asynchronously, which allows `ngcc` to
potentially run in a more efficient mode.
Currently, there is no difference in the way tasks are executed in sync
vs async mode, but this change sets the ground for adding new execution
options (that require asynchronous operation), such as processing tasks
in parallel on multiple processes.
NOTE:
When using the programmatic API, the default value for `async` is
`false`, thus retaining backwards compatibility.
When running `ngcc` from the command line (i.e. via the `ivy-ngcc`
script), it runs in async mode (to be able to take advantage of future
optimizations), but that is transparent to the caller.
PR Close #32427
2019-08-19 15:58:22 -04:00
it ( 'should throw, if an error happens during processing' , ( ) = > {
spyOn ( Transformer . prototype , 'transform' ) . and . throwError ( 'Test error.' ) ;
expect ( ( ) = > mainNgcc ( {
basePath : '/dist' ,
targetEntryPointPath : 'local-package' ,
propertiesToConsider : [ 'main' , 'es2015' ] ,
logger : new MockLogger ( ) ,
} ) )
. toThrowError ( ` Test error. ` ) ;
expect ( loadPackage ( '@angular/core' ) . __processed_by_ivy_ngcc__ ) . toBeUndefined ( ) ;
expect ( loadPackage ( 'local-package' , _ ( '/dist' ) ) . __processed_by_ivy_ngcc__ ) . toBeUndefined ( ) ;
} ) ;
fix(ngcc): generate correct metadata for classes with getter/setter properties (#33514)
While processing class metadata, ngtsc generates a `setClassMetadata()`
call which (among other things) contains info about property decorators.
Previously, processing getter/setter pairs with some of ngcc's
`ReflectionHost`s resulted in multiple metadata entries for the same
property, which resulted in duplicate object keys, which in turn causes
an error in ES5 strict mode.
This commit fixes it by ensuring that there are no duplicate property
names in the `setClassMetadata()` calls.
In addition, `generateSetClassMetadataCall()` is updated to treat
`ClassMember#decorators: []` the same as `ClassMember.decorators: null`
(i.e. omitting the `ClassMember` from the generated `setClassMetadata()`
call). Alternatively, ngcc's `ReflectionHost`s could be updated to do
this transformation (`decorators: []` --> `decorators: null`) when
reflecting on class members, but this would require changes in many
places and be less future-proof.
For example, given a class such as:
```ts
class Foo {
@Input() get bar() { return 'bar'; }
set bar(value: any) {}
}
```
...previously the generated `setClassMetadata()` call would look like:
```ts
ɵsetClassMetadata(..., {
bar: [{type: Input}],
bar: [],
});
```
The same class will now result in a call like:
```ts
ɵsetClassMetadata(..., {
bar: [{type: Input}],
});
```
Fixes #30569
PR Close #33514
2019-10-30 09:50:19 -04:00
it ( 'should generate correct metadata for decorated getter/setter properties' , ( ) = > {
genNodeModules ( {
'test-package' : {
'/index.ts' : `
import { Directive , Input , NgModule } from '@angular/core' ;
@Directive ( { selector : '[foo]' } )
export class FooDirective {
@Input ( ) get bar() { return 'bar' ; }
set bar ( value : string ) { }
}
@NgModule ( {
declarations : [ FooDirective ] ,
} )
export class FooModule { }
` ,
} ,
} ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : 'test-package' ,
propertiesToConsider : [ 'main' ] ,
} ) ;
const jsContents = fs . readFile ( _ ( ` /node_modules/test-package/index.js ` ) ) . replace ( /\s+/g , ' ' ) ;
expect ( jsContents )
. toContain (
'/*@__PURE__*/ ɵngcc0.ɵsetClassMetadata(FooDirective, ' +
'[{ type: Directive, args: [{ selector: \'[foo]\' }] }], ' +
'function () { return []; }, ' +
'{ bar: [{ type: Input }] });' ) ;
} ) ;
2019-11-04 12:29:01 -05:00
it ( 'should not add `const` in ES5 generated code' , ( ) = > {
genNodeModules ( {
'test-package' : {
'/index.ts' : `
import { Directive , Input , NgModule } from '@angular/core' ;
@Directive ( {
selector : '[foo]' ,
host : { bar : '' } ,
} )
export class FooDirective {
}
@NgModule ( {
declarations : [ FooDirective ] ,
} )
export class FooModule { }
` ,
} ,
} ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : 'test-package' ,
propertiesToConsider : [ 'main' ] ,
} ) ;
const jsContents = fs . readFile ( _ ( ` /node_modules/test-package/index.js ` ) ) ;
expect ( jsContents ) . not . toMatch ( /\bconst \w+\s*=/ ) ;
expect ( jsContents ) . toMatch ( /\bvar _c0 =/ ) ;
} ) ;
refactor(ngcc): add support for asynchronous execution (#32427)
Previously, `ngcc`'s programmatic API would run and complete
synchronously. This was necessary for specific usecases (such as how the
`@angular/cli` invokes `ngcc` as part of the TypeScript module
resolution process), but not for others (e.g. running `ivy-ngcc` as a
`postinstall` script).
This commit adds a new option (`async`) that enables turning on
asynchronous execution. I.e. it signals that the caller is OK with the
function call to complete asynchronously, which allows `ngcc` to
potentially run in a more efficient mode.
Currently, there is no difference in the way tasks are executed in sync
vs async mode, but this change sets the ground for adding new execution
options (that require asynchronous operation), such as processing tasks
in parallel on multiple processes.
NOTE:
When using the programmatic API, the default value for `async` is
`false`, thus retaining backwards compatibility.
When running `ngcc` from the command line (i.e. via the `ivy-ngcc`
script), it runs in async mode (to be able to take advantage of future
optimizations), but that is transparent to the caller.
PR Close #32427
2019-08-19 15:58:22 -04:00
describe ( 'in async mode' , ( ) = > {
it ( 'should run ngcc without errors for fesm2015' , async ( ) = > {
const promise = mainNgcc ( {
basePath : '/node_modules' ,
propertiesToConsider : [ 'fesm2015' ] ,
async : true ,
} ) ;
expect ( promise ) . toEqual ( jasmine . any ( Promise ) ) ;
await promise ;
} ) ;
refactor(ngcc): take advantage of early knowledge about format property processability (#32427)
In the past, a task's processability didn't use to be known in advance.
It was possible that a task would be created and added to the queue
during the analysis phase and then later (during the compilation phase)
it would be found out that the task (i.e. the associated format
property) was not processable.
As a result, certain checks had to be delayed, until a task's processing
had started or even until all tasks had been processed. Examples of
checks that had to be delayed are:
- Whether a task can be skipped due to `compileAllFormats: false`.
- Whether there were entry-points for which no format at all was
successfully processed.
It turns out that (as made clear by the refactoring in 9537b2ff8), once
a task starts being processed it is expected to either complete
successfully (with the associated format being processed) or throw an
error (in which case the process will exit). In other words, a task's
processability is known in advance.
This commit takes advantage of this fact by moving certain checks
earlier in the process (e.g. in the analysis phase instead of the
compilation phase), which in turn allows avoiding some unnecessary work.
More specifically:
- When `compileAllFormats` is `false`, tasks are created _only_ for the
first suitable format property for each entry-point, since the rest of
the tasks would have been skipped during the compilation phase anyway.
This has the following advantages:
1. It avoids the slight overhead of generating extraneous tasks and
then starting to process them (before realizing they should be
skipped).
2. In a potential future parallel execution mode, unnecessary tasks
might start being processed at the same time as the first (useful)
task, even if their output would be later discarded, wasting
resources. Alternatively, extra logic would have to be added to
prevent this from happening. The change in this commit avoids these
issues.
- When an entry-point is not processable, an error will be thrown
upfront without having to wait for other tasks to be processed before
failing.
PR Close #32427
2019-08-28 18:33:15 -04:00
it ( 'should reject, if some of the entry-points are unprocessable' , async ( ) = > {
const createEntryPoint = ( name : string , prop : EntryPointJsonProperty ) : TestFile [ ] = > {
return [
{
name : _ ( ` /dist/ ${ name } /package.json ` ) ,
contents : ` {"name": " ${ name } ", "typings": "./index.d.ts", " ${ prop } ": "./index.js"} ` ,
} ,
{ name : _ ( ` /dist/ ${ name } /index.js ` ) , contents : 'var DUMMY_DATA = true;' } ,
{ name : _ ( ` /dist/ ${ name } /index.d.ts ` ) , contents : 'export type DummyData = boolean;' } ,
{ name : _ ( ` /dist/ ${ name } /index.metadata.json ` ) , contents : 'DUMMY DATA' } ,
] ;
} ;
loadTestFiles ( [
. . . createEntryPoint ( 'processable-1' , 'es2015' ) ,
. . . createEntryPoint ( 'unprocessable-2' , 'main' ) ,
. . . createEntryPoint ( 'unprocessable-3' , 'main' ) ,
] ) ;
const promise = mainNgcc ( {
basePath : '/dist' ,
propertiesToConsider : [ 'es2015' , 'fesm5' , 'module' ] ,
logger : new MockLogger ( ) ,
async : true ,
} ) ;
await promise . then (
( ) = > Promise . reject ( 'Expected promise to be rejected.' ) ,
err = > expect ( err ) . toEqual ( new Error (
'Unable to process any formats for the following entry-points (tried es2015, fesm5, module): \n' +
` - ${ _ ( '/dist/unprocessable-2' ) } \ n ` +
` - ${ _ ( '/dist/unprocessable-3' ) } ` ) ) ) ;
} ) ;
refactor(ngcc): add support for asynchronous execution (#32427)
Previously, `ngcc`'s programmatic API would run and complete
synchronously. This was necessary for specific usecases (such as how the
`@angular/cli` invokes `ngcc` as part of the TypeScript module
resolution process), but not for others (e.g. running `ivy-ngcc` as a
`postinstall` script).
This commit adds a new option (`async`) that enables turning on
asynchronous execution. I.e. it signals that the caller is OK with the
function call to complete asynchronously, which allows `ngcc` to
potentially run in a more efficient mode.
Currently, there is no difference in the way tasks are executed in sync
vs async mode, but this change sets the ground for adding new execution
options (that require asynchronous operation), such as processing tasks
in parallel on multiple processes.
NOTE:
When using the programmatic API, the default value for `async` is
`false`, thus retaining backwards compatibility.
When running `ngcc` from the command line (i.e. via the `ivy-ngcc`
script), it runs in async mode (to be able to take advantage of future
optimizations), but that is transparent to the caller.
PR Close #32427
2019-08-19 15:58:22 -04:00
it ( 'should reject, if an error happens during processing' , async ( ) = > {
spyOn ( Transformer . prototype , 'transform' ) . and . throwError ( 'Test error.' ) ;
const promise = mainNgcc ( {
basePath : '/dist' ,
targetEntryPointPath : 'local-package' ,
propertiesToConsider : [ 'main' , 'es2015' ] ,
logger : new MockLogger ( ) ,
async : true ,
} ) ;
await promise . then (
( ) = > Promise . reject ( 'Expected promise to be rejected.' ) ,
err = > expect ( err ) . toEqual ( new Error ( 'Test error.' ) ) ) ;
expect ( loadPackage ( '@angular/core' ) . __processed_by_ivy_ngcc__ ) . toBeUndefined ( ) ;
expect ( loadPackage ( 'local-package' , _ ( '/dist' ) ) . __processed_by_ivy_ngcc__ ) . toBeUndefined ( ) ;
} ) ;
} ) ;
2019-06-06 15:22:32 -04:00
describe ( 'with targetEntryPointPath' , ( ) = > {
it ( 'should only compile the given package entry-point (and its dependencies).' , ( ) = > {
const STANDARD_MARKERS = {
main : '0.0.0-PLACEHOLDER' ,
module : '0.0.0-PLACEHOLDER' ,
es2015 : '0.0.0-PLACEHOLDER' ,
esm5 : '0.0.0-PLACEHOLDER' ,
esm2015 : '0.0.0-PLACEHOLDER' ,
fesm5 : '0.0.0-PLACEHOLDER' ,
fesm2015 : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ;
mainNgcc ( { basePath : '/node_modules' , targetEntryPointPath : '@angular/common/http/testing' } ) ;
expect ( loadPackage ( '@angular/common/http/testing' ) . __processed_by_ivy_ngcc__ )
. toEqual ( STANDARD_MARKERS ) ;
// * `common/http` is a dependency of `common/http/testing`, so is compiled.
expect ( loadPackage ( '@angular/common/http' ) . __processed_by_ivy_ngcc__ )
. toEqual ( STANDARD_MARKERS ) ;
// * `core` is a dependency of `common/http`, so is compiled.
expect ( loadPackage ( '@angular/core' ) . __processed_by_ivy_ngcc__ ) . toEqual ( STANDARD_MARKERS ) ;
// * `common` is a private (only in .js not .d.ts) dependency so is compiled.
expect ( loadPackage ( '@angular/common' ) . __processed_by_ivy_ngcc__ ) . toEqual ( STANDARD_MARKERS ) ;
// * `common/testing` is not a dependency so is not compiled.
expect ( loadPackage ( '@angular/common/testing' ) . __processed_by_ivy_ngcc__ ) . toBeUndefined ( ) ;
2019-04-06 10:35:40 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
it ( 'should mark a non-Angular package target as processed' , ( ) = > {
mainNgcc ( { basePath : '/node_modules' , targetEntryPointPath : 'test-package' } ) ;
// `test-package` has no Angular but is marked as processed.
expect ( loadPackage ( 'test-package' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
es2015 : '0.0.0-PLACEHOLDER' ,
2019-08-04 10:51:37 -04:00
esm2015 : '0.0.0-PLACEHOLDER' ,
esm5 : '0.0.0-PLACEHOLDER' ,
fesm2015 : '0.0.0-PLACEHOLDER' ,
fesm5 : '0.0.0-PLACEHOLDER' ,
main : '0.0.0-PLACEHOLDER' ,
module : '0.0.0-PLACEHOLDER' ,
2019-04-06 10:35:40 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
// * `core` is a dependency of `test-package`, but it is not processed, since test-package
// was not processed.
expect ( loadPackage ( '@angular/core' ) . __processed_by_ivy_ngcc__ ) . toBeUndefined ( ) ;
2019-04-06 10:35:40 -04:00
} ) ;
2019-07-31 07:54:12 -04:00
it ( 'should report an error if a dependency of the target does not exist' , ( ) = > {
expect ( ( ) = > {
mainNgcc ( { basePath : '/node_modules' , targetEntryPointPath : 'invalid-package' } ) ;
} )
. toThrowError (
'The target entry-point "invalid-package" has missing dependencies:\n - @angular/missing\n' ) ;
} ) ;
2019-04-06 10:35:40 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
describe ( 'early skipping of target entry-point' , ( ) = > {
describe ( '[compileAllFormats === true]' , ( ) = > {
it ( 'should skip all processing if all the properties are marked as processed' , ( ) = > {
const logger = new MockLogger ( ) ;
markPropertiesAsProcessed ( '@angular/common/http/testing' , SUPPORTED_FORMAT_PROPERTIES ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : '@angular/common/http/testing' , logger ,
} ) ;
expect ( logger . logs . debug ) . toContain ( [
'The target entry-point has already been processed'
] ) ;
} ) ;
2019-04-06 10:35:40 -04:00
2019-06-06 15:22:32 -04:00
it ( 'should process the target if any `propertyToConsider` is not marked as processed' ,
( ) = > {
const logger = new MockLogger ( ) ;
markPropertiesAsProcessed ( '@angular/common/http/testing' , [ 'esm2015' , 'fesm2015' ] ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : '@angular/common/http/testing' ,
propertiesToConsider : [ 'fesm2015' , 'esm5' , 'esm2015' ] , logger ,
} ) ;
expect ( logger . logs . debug ) . not . toContain ( [
'The target entry-point has already been processed'
] ) ;
} ) ;
} ) ;
2019-04-06 10:35:40 -04:00
2019-06-06 15:22:32 -04:00
describe ( '[compileAllFormats === false]' , ( ) = > {
it ( 'should process the target if the first matching `propertyToConsider` is not marked as processed' ,
( ) = > {
const logger = new MockLogger ( ) ;
markPropertiesAsProcessed ( '@angular/common/http/testing' , [ 'esm2015' ] ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : '@angular/common/http/testing' ,
propertiesToConsider : [ 'esm5' , 'esm2015' ] ,
compileAllFormats : false , logger ,
} ) ;
expect ( logger . logs . debug ) . not . toContain ( [
'The target entry-point has already been processed'
] ) ;
2019-04-06 10:35:40 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
it ( 'should skip all processing if the first matching `propertyToConsider` is marked as processed' ,
( ) = > {
const logger = new MockLogger ( ) ;
markPropertiesAsProcessed ( '@angular/common/http/testing' , [ 'esm2015' ] ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : '@angular/common/http/testing' ,
// Simulate a property that does not exist on the package.json and will be ignored.
propertiesToConsider : [ 'missing' , 'esm2015' , 'esm5' ] ,
compileAllFormats : false , logger ,
} ) ;
expect ( logger . logs . debug ) . toContain ( [
'The target entry-point has already been processed'
] ) ;
} ) ;
} ) ;
2019-08-05 06:36:51 -04:00
it ( 'should skip all processing if the first matching `propertyToConsider` is marked as processed' ,
( ) = > {
const logger = new MockLogger ( ) ;
markPropertiesAsProcessed ( '@angular/common/http/testing' , [ 'esm2015' ] ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : '@angular/common/http/testing' ,
// Simulate a property that does not exist on the package.json and will be ignored.
propertiesToConsider : [ 'missing' , 'esm2015' , 'esm5' ] ,
compileAllFormats : false , logger ,
} ) ;
expect ( logger . logs . debug ) . toContain ( [
'The target entry-point has already been processed'
] ) ;
} ) ;
2019-04-06 10:35:40 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
function markPropertiesAsProcessed ( packagePath : string , properties : EntryPointJsonProperty [ ] ) {
const basePath = _ ( '/node_modules' ) ;
const targetPackageJsonPath = join ( basePath , packagePath , 'package.json' ) ;
const targetPackage = loadPackage ( packagePath ) ;
2019-08-12 11:15:24 -04:00
markAsProcessed (
pkgJsonUpdater , targetPackage , targetPackageJsonPath , [ 'typings' , . . . properties ] ) ;
2019-06-06 15:22:32 -04:00
}
2019-04-06 10:35:40 -04:00
2019-06-06 15:22:32 -04:00
describe ( 'with propertiesToConsider' , ( ) = > {
2019-08-05 17:53:38 -04:00
it ( 'should complain if none of the properties in the `propertiesToConsider` list is supported' ,
( ) = > {
const propertiesToConsider = [ 'es1337' , 'fesm42' ] ;
const errorMessage =
'No supported format property to consider among [es1337, fesm42]. Supported ' +
'properties: fesm2015, fesm5, es2015, esm2015, esm5, main, module' ;
expect ( ( ) = > mainNgcc ( { basePath : '/node_modules' , propertiesToConsider } ) )
. toThrowError ( errorMessage ) ;
} ) ;
2019-06-06 15:22:32 -04:00
it ( 'should only compile the entry-point formats given in the `propertiesToConsider` list' ,
( ) = > {
mainNgcc ( {
basePath : '/node_modules' ,
propertiesToConsider : [ 'main' , 'esm5' , 'module' , 'fesm5' ] ,
logger : new MockLogger ( ) ,
2019-04-06 10:35:13 -04:00
2019-06-06 15:22:32 -04:00
} ) ;
2019-03-20 09:47:58 -04:00
2019-06-06 15:22:32 -04:00
// The ES2015 formats are not compiled as they are not in `propertiesToConsider`.
expect ( loadPackage ( '@angular/core' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
esm5 : '0.0.0-PLACEHOLDER' ,
main : '0.0.0-PLACEHOLDER' ,
module : '0.0.0-PLACEHOLDER' ,
fesm5 : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
expect ( loadPackage ( '@angular/common' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
esm5 : '0.0.0-PLACEHOLDER' ,
main : '0.0.0-PLACEHOLDER' ,
module : '0.0.0-PLACEHOLDER' ,
fesm5 : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
expect ( loadPackage ( '@angular/common/testing' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
esm5 : '0.0.0-PLACEHOLDER' ,
main : '0.0.0-PLACEHOLDER' ,
module : '0.0.0-PLACEHOLDER' ,
fesm5 : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
expect ( loadPackage ( '@angular/common/http' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
esm5 : '0.0.0-PLACEHOLDER' ,
main : '0.0.0-PLACEHOLDER' ,
module : '0.0.0-PLACEHOLDER' ,
fesm5 : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
2019-03-20 09:47:58 -04:00
} ) ;
2019-08-05 06:36:51 -04:00
it ( 'should mark all matching properties as processed in order not to compile them on a subsequent run' ,
( ) = > {
const logger = new MockLogger ( ) ;
const logs = logger . logs . debug ;
// `fesm2015` and `es2015` map to the same file: `./fesm2015/common.js`
mainNgcc ( {
basePath : '/node_modules/@angular/common' ,
propertiesToConsider : [ 'fesm2015' ] , logger ,
} ) ;
expect ( logs ) . not . toContain ( [ 'Skipping @angular/common : es2015 (already compiled).' ] ) ;
expect ( loadPackage ( '@angular/common' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
es2015 : '0.0.0-PLACEHOLDER' ,
fesm2015 : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
// Now, compiling `es2015` should be a no-op.
mainNgcc ( {
basePath : '/node_modules/@angular/common' ,
propertiesToConsider : [ 'es2015' ] , logger ,
} ) ;
expect ( logs ) . toContain ( [ 'Skipping @angular/common : es2015 (already compiled).' ] ) ;
} ) ;
2019-06-06 15:22:32 -04:00
} ) ;
2019-03-20 09:47:58 -04:00
2019-06-06 15:22:32 -04:00
describe ( 'with compileAllFormats set to false' , ( ) = > {
it ( 'should only compile the first matching format' , ( ) = > {
mainNgcc ( {
basePath : '/node_modules' ,
propertiesToConsider : [ 'module' , 'fesm5' , 'esm5' ] ,
compileAllFormats : false ,
logger : new MockLogger ( ) ,
} ) ;
// * In the Angular packages fesm5 and module have the same underlying format,
// so both are marked as compiled.
// * The `esm5` is not compiled because we stopped after the `fesm5` format.
expect ( loadPackage ( '@angular/core' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
fesm5 : '0.0.0-PLACEHOLDER' ,
module : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
expect ( loadPackage ( '@angular/common' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
fesm5 : '0.0.0-PLACEHOLDER' ,
module : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
expect ( loadPackage ( '@angular/common/testing' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
fesm5 : '0.0.0-PLACEHOLDER' ,
module : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
expect ( loadPackage ( '@angular/common/http' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
fesm5 : '0.0.0-PLACEHOLDER' ,
module : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
2019-03-20 09:47:58 -04:00
} ) ;
2019-04-02 06:51:39 -04:00
2019-06-06 15:22:32 -04:00
it ( 'should cope with compiling the same entry-point multiple times with different formats' ,
( ) = > {
mainNgcc ( {
basePath : '/node_modules' ,
propertiesToConsider : [ 'module' ] ,
compileAllFormats : false ,
logger : new MockLogger ( ) ,
2019-04-06 10:35:13 -04:00
2019-06-06 15:22:32 -04:00
} ) ;
expect ( loadPackage ( '@angular/core' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
2019-08-05 06:36:51 -04:00
fesm5 : '0.0.0-PLACEHOLDER' ,
2019-06-06 15:22:32 -04:00
module : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
// If ngcc tries to write out the typings files again, this will throw an exception.
mainNgcc ( {
basePath : '/node_modules' ,
propertiesToConsider : [ 'esm5' ] ,
compileAllFormats : false ,
logger : new MockLogger ( ) ,
} ) ;
expect ( loadPackage ( '@angular/core' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
esm5 : '0.0.0-PLACEHOLDER' ,
2019-08-05 06:36:51 -04:00
fesm5 : '0.0.0-PLACEHOLDER' ,
2019-06-06 15:22:32 -04:00
module : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
2019-04-02 06:51:39 -04:00
} ) ;
2019-06-06 15:22:32 -04:00
} ) ;
2019-03-20 09:47:59 -04:00
2019-06-06 15:22:32 -04:00
describe ( 'with createNewEntryPointFormats' , ( ) = > {
it ( 'should create new files rather than overwriting the originals' , ( ) = > {
const ANGULAR_CORE_IMPORT_REGEX = /import \* as ɵngcc\d+ from '@angular\/core';/ ;
mainNgcc ( {
basePath : '/node_modules' ,
createNewEntryPointFormats : true ,
propertiesToConsider : [ 'esm5' ] ,
logger : new MockLogger ( ) ,
2019-04-06 10:35:13 -04:00
2019-06-06 15:22:32 -04:00
} ) ;
2019-03-20 09:47:59 -04:00
2019-06-06 15:22:32 -04:00
// Updates the package.json
expect ( loadPackage ( '@angular/common' ) . esm5 ) . toEqual ( './esm5/common.js' ) ;
expect ( ( loadPackage ( '@angular/common' ) as any ) . esm5_ivy_ngcc )
. toEqual ( '__ivy_ngcc__/esm5/common.js' ) ;
// Doesn't touch original files
expect ( fs . readFile ( _ ( ` /node_modules/@angular/common/esm5/src/common_module.js ` ) ) )
. not . toMatch ( ANGULAR_CORE_IMPORT_REGEX ) ;
// Or create a backup of the original
expect (
fs . exists ( _ ( ` /node_modules/@angular/common/esm5/src/common_module.js.__ivy_ngcc_bak ` ) ) )
. toBe ( false ) ;
// Creates new files
expect (
fs . readFile ( _ ( ` /node_modules/@angular/common/__ivy_ngcc__/esm5/src/common_module.js ` ) ) )
. toMatch ( ANGULAR_CORE_IMPORT_REGEX ) ;
// Copies over files (unchanged) that did not need compiling
expect ( fs . exists ( _ ( ` /node_modules/@angular/common/__ivy_ngcc__/esm5/src/version.js ` ) ) ) ;
expect ( fs . readFile ( _ ( ` /node_modules/@angular/common/__ivy_ngcc__/esm5/src/version.js ` ) ) )
. toEqual ( fs . readFile ( _ ( ` /node_modules/@angular/common/esm5/src/version.js ` ) ) ) ;
// Overwrites .d.ts files (as usual)
expect ( fs . readFile ( _ ( ` /node_modules/@angular/common/common.d.ts ` ) ) )
. toMatch ( ANGULAR_CORE_IMPORT_REGEX ) ;
expect ( fs . exists ( _ ( ` /node_modules/@angular/common/common.d.ts.__ivy_ngcc_bak ` ) ) ) . toBe ( true ) ;
} ) ;
2019-08-07 20:23:46 -04:00
it ( 'should update `package.json` for all matching format properties' , ( ) = > {
mainNgcc ( {
basePath : '/node_modules/@angular/core' ,
createNewEntryPointFormats : true ,
propertiesToConsider : [ 'fesm2015' , 'fesm5' ] ,
} ) ;
const pkg : any = loadPackage ( '@angular/core' ) ;
// `es2015` is an alias of `fesm2015`.
expect ( pkg . fesm2015 ) . toEqual ( './fesm2015/core.js' ) ;
expect ( pkg . es2015 ) . toEqual ( './fesm2015/core.js' ) ;
expect ( pkg . fesm2015_ivy_ngcc ) . toEqual ( '__ivy_ngcc__/fesm2015/core.js' ) ;
expect ( pkg . es2015_ivy_ngcc ) . toEqual ( '__ivy_ngcc__/fesm2015/core.js' ) ;
// `module` is an alias of `fesm5`.
expect ( pkg . fesm5 ) . toEqual ( './fesm5/core.js' ) ;
expect ( pkg . module ) . toEqual ( './fesm5/core.js' ) ;
expect ( pkg . fesm5_ivy_ngcc ) . toEqual ( '__ivy_ngcc__/fesm5/core.js' ) ;
expect ( pkg . module _ivy_ngcc ) . toEqual ( '__ivy_ngcc__/fesm5/core.js' ) ;
} ) ;
2019-03-20 09:47:59 -04:00
} ) ;
2019-03-29 06:13:14 -04:00
2019-08-04 13:20:38 -04:00
describe ( 'diagnostics' , ( ) = > {
it ( 'should fail with formatted diagnostics when an error diagnostic is produced' , ( ) = > {
loadTestFiles ( [
{
name : _ ( '/node_modules/fatal-error/package.json' ) ,
contents : '{"name": "fatal-error", "es2015": "./index.js", "typings": "./index.d.ts"}' ,
} ,
{ name : _ ( '/node_modules/fatal-error/index.metadata.json' ) , contents : 'DUMMY DATA' } ,
{
name : _ ( '/node_modules/fatal-error/index.js' ) ,
contents : `
import { Component } from '@angular/core' ;
export class FatalError { }
FatalError . decorators = [
{ type : Component , args : [ { selector : 'fatal-error' } ] }
] ;
` ,
} ,
{
name : _ ( '/node_modules/fatal-error/index.d.ts' ) ,
contents : `
export declare class FatalError { }
` ,
} ,
] ) ;
expect ( ( ) = > mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : 'fatal-error' ,
propertiesToConsider : [ 'es2015' ]
} ) )
. toThrowError (
/^Failed to compile entry-point fatal-error due to compilation errors:\nnode_modules\/fatal-error\/index\.js\(5,17\): error TS-992001: component is missing a template\r?\n$/ ) ;
} ) ;
} ) ;
2019-06-06 15:22:32 -04:00
describe ( 'logger' , ( ) = > {
it ( 'should log info message to the console by default' , ( ) = > {
const consoleInfoSpy = spyOn ( console , 'info' ) ;
mainNgcc ( { basePath : '/node_modules' , propertiesToConsider : [ 'esm2015' ] } ) ;
expect ( consoleInfoSpy )
. toHaveBeenCalledWith ( 'Compiling @angular/common/http : esm2015 as esm2015' ) ;
} ) ;
2019-03-29 06:13:14 -04:00
2019-06-06 15:22:32 -04:00
it ( 'should use a custom logger if provided' , ( ) = > {
const logger = new MockLogger ( ) ;
mainNgcc ( {
basePath : '/node_modules' ,
propertiesToConsider : [ 'esm2015' ] , logger ,
} ) ;
expect ( logger . logs . info ) . toContain ( [ 'Compiling @angular/common/http : esm2015 as esm2015' ] ) ;
2019-04-06 10:35:13 -04:00
} ) ;
2019-03-29 06:13:14 -04:00
} ) ;
2019-04-28 15:47:57 -04:00
2019-06-06 15:22:32 -04:00
describe ( 'with pathMappings' , ( ) = > {
it ( 'should find and compile packages accessible via the pathMappings' , ( ) = > {
mainNgcc ( {
basePath : '/node_modules' ,
propertiesToConsider : [ 'es2015' ] ,
pathMappings : { paths : { '*' : [ 'dist/*' ] } , baseUrl : '/' } ,
} ) ;
expect ( loadPackage ( '@angular/core' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
es2015 : '0.0.0-PLACEHOLDER' ,
2019-08-05 06:36:51 -04:00
fesm2015 : '0.0.0-PLACEHOLDER' ,
2019-06-06 15:22:32 -04:00
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
expect ( loadPackage ( 'local-package' , _ ( '/dist' ) ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
2019-05-21 10:23:24 -04:00
es2015 : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
} ) ;
} ) ;
describe ( 'with configuration files' , ( ) = > {
it ( 'should process a configured deep-import as an entry-point' , ( ) = > {
loadTestFiles ( [
{
name : _ ( '/ngcc.config.js' ) ,
contents : ` module.exports = { packages: {
'deep_import' : {
entryPoints : {
'./entry_point' : { override : { typings : '../entry_point.d.ts' , es2015 : '../entry_point.js' } }
}
}
} } ; ` ,
} ,
{
name : _ ( '/node_modules/deep_import/package.json' ) ,
contents : '{"name": "deep-import", "es2015": "./index.js", "typings": "./index.d.ts"}' ,
} ,
{
name : _ ( '/node_modules/deep_import/entry_point.js' ) ,
contents : `
import { Component } from '@angular/core' ;
@Component ( { selector : 'entry-point' } )
export class EntryPoint { }
` ,
} ,
{
name : _ ( '/node_modules/deep_import/entry_point.d.ts' ) ,
contents : `
import { Component } from '@angular/core' ;
@Component ( { selector : 'entry-point' } )
export class EntryPoint { }
` ,
} ,
] ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : 'deep_import/entry_point' ,
propertiesToConsider : [ 'es2015' ]
} ) ;
// The containing package is not processed
expect ( loadPackage ( 'deep_import' ) . __processed_by_ivy_ngcc__ ) . toBeUndefined ( ) ;
// But the configured entry-point and its dependency (@angular/core) are processed.
expect ( loadPackage ( 'deep_import/entry_point' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
es2015 : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
expect ( loadPackage ( '@angular/core' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
es2015 : '0.0.0-PLACEHOLDER' ,
2019-08-05 06:36:51 -04:00
fesm2015 : '0.0.0-PLACEHOLDER' ,
2019-05-21 10:23:24 -04:00
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
} ) ;
it ( 'should not process ignored entry-points' , ( ) = > {
loadTestFiles ( [
{
name : _ ( '/ngcc.config.js' ) ,
contents : ` module.exports = { packages: {
'@angular/core' : {
entryPoints : {
'./testing' : { ignore : true }
} ,
} ,
'@angular/common' : {
entryPoints : {
'.' : { ignore : true }
} ,
}
} } ; ` ,
} ,
] ) ;
mainNgcc ( { basePath : '/node_modules' , propertiesToConsider : [ 'es2015' ] } ) ;
// We process core but not core/testing.
expect ( loadPackage ( '@angular/core' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
es2015 : '0.0.0-PLACEHOLDER' ,
2019-08-05 06:36:51 -04:00
fesm2015 : '0.0.0-PLACEHOLDER' ,
2019-05-21 10:23:24 -04:00
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
expect ( loadPackage ( '@angular/core/testing' ) . __processed_by_ivy_ngcc__ ) . toBeUndefined ( ) ;
// We do not compile common but we do compile its sub-entry-points.
expect ( loadPackage ( '@angular/common' ) . __processed_by_ivy_ngcc__ ) . toBeUndefined ( ) ;
expect ( loadPackage ( '@angular/common/http' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
2019-06-06 15:22:32 -04:00
es2015 : '0.0.0-PLACEHOLDER' ,
2019-08-05 06:36:51 -04:00
fesm2015 : '0.0.0-PLACEHOLDER' ,
2019-06-06 15:22:32 -04:00
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
2019-04-28 15:47:57 -04:00
} ) ;
fix(ngcc): ignore format properties that exist but are undefined (#32205)
Previously, `ngcc` assumed that if a format property was defined in
`package.json` it would point to a valid format-path (i.e. a file that
is an entry-point for a specific format). This is generally the case,
except if a format property is set to a non-string value (such as
`package.json`) - either directly in the `package.json` (which is unusual)
or in ngcc.config.js (which is a valid usecase, when one wants a
format property to be ignored by `ngcc`).
For example, the following config file would cause `ngcc` to throw:
```
module.exports = {
packages: {
'test-package': {
entryPoints: {
'.': {
override: {
fesm2015: undefined,
},
},
},
},
},
};
```
This commit fixes it by ensuring that only format properties whose value
is a string are considered by `ngcc`.
For reference, this regression was introduced in #32052.
Fixes #32188
PR Close #32205
2019-08-20 03:43:08 -04:00
it ( 'should support removing a format property by setting it to `undefined`' , ( ) = > {
loadTestFiles ( [
{
name : _ ( '/ngcc.config.js' ) ,
contents : `
module .exports = {
packages : {
'test-package' : {
entryPoints : {
'.' : {
override : {
fesm2015 : undefined ,
} ,
} ,
} ,
} ,
} ,
} ;
` ,
} ,
{
name : _ ( '/node_modules/test-package/package.json' ) ,
contents : `
{
"name" : "test-package" ,
"fesm2015" : "./index.es2015.js" ,
"fesm5" : "./index.es5.js" ,
"typings" : "./index.d.ts"
}
` ,
} ,
{
name : _ ( '/node_modules/test-package/index.es5.js' ) ,
contents : `
var TestService = ( function ( ) {
function TestService() {
}
return TestService ;
} ( ) ) ;
` ,
} ,
{
name : _ ( '/node_modules/test-package/index.d.js' ) ,
contents : `
export declare class TestService { }
` ,
} ,
] ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : 'test-package' ,
propertiesToConsider : [ 'fesm2015' , 'fesm5' ] ,
} ) ;
expect ( loadPackage ( 'test-package' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
fesm5 : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
} ) ;
2019-04-28 15:47:57 -04:00
} ) ;
2019-01-25 13:48:27 -05:00
feat(ngcc): add a migration for undecorated child classes (#33362)
In Angular View Engine, there are two kinds of decorator inheritance:
1) both the parent and child classes have decorators
This case is supported by InheritDefinitionFeature, which merges some fields
of the definitions (such as the inputs or queries).
2) only the parent class has a decorator
If the child class is missing a decorator, the compiler effectively behaves
as if the parent class' decorator is applied to the child class as well.
This is the "undecorated child" scenario, and this commit adds a migration
to ngcc to support this pattern in Ivy.
This migration has 2 phases. First, the NgModules of the application are
scanned for classes in 'declarations' which are missing decorators, but
whose base classes do have decorators. These classes are the undecorated
children. This scan is performed recursively, so even if a declared class
has a base class that itself inherits a decorator, this case is handled.
Next, a synthetic decorator (either @Component or @Directive) is created
on the child class. This decorator copies some critical information such
as 'selector' and 'exportAs', as well as supports any decorated fields
(@Input, etc). A flag is passed to the decorator compiler which causes a
special feature `CopyDefinitionFeature` to be included on the compiled
definition. This feature copies at runtime the remaining aspects of the
parent definition which `InheritDefinitionFeature` does not handle,
completing the "full" inheritance of the child class' decorator from its
parent class.
PR Close #33362
2019-10-23 15:00:49 -04:00
describe ( 'undecorated child class migration' , ( ) = > {
it ( 'should generate a directive definition with CopyDefinitionFeature for an undecorated child directive' ,
( ) = > {
genNodeModules ( {
'test-package' : {
'/index.ts' : `
import { Directive , NgModule } from '@angular/core' ;
@Directive ( {
selector : '[base]' ,
} )
export class BaseDir { }
export class DerivedDir extends BaseDir { }
@NgModule ( {
declarations : [ DerivedDir ] ,
} )
export class Module { }
` ,
} ,
} ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : 'test-package' ,
propertiesToConsider : [ 'main' ] ,
} ) ;
const jsContents = fs . readFile ( _ ( ` /node_modules/test-package/index.js ` ) ) ;
expect ( jsContents )
. toContain (
'DerivedDir.ɵdir = ɵngcc0.ɵɵdefineDirective({ type: DerivedDir, selectors: [["", "base", ""]], ' +
'features: [ɵngcc0.ɵɵInheritDefinitionFeature, ɵngcc0.ɵɵCopyDefinitionFeature] });' ) ;
const dtsContents = fs . readFile ( _ ( ` /node_modules/test-package/index.d.ts ` ) ) ;
expect ( dtsContents )
. toContain (
'static ɵdir: ɵngcc0.ɵɵDirectiveDefWithMeta<DerivedDir, "[base]", never, {}, {}, never>;' ) ;
} ) ;
it ( 'should generate a component definition with CopyDefinitionFeature for an undecorated child component' ,
( ) = > {
genNodeModules ( {
'test-package' : {
'/index.ts' : `
import { Component , NgModule } from '@angular/core' ;
@Component ( {
selector : '[base]' ,
template : '<span>This is the base template</span>' ,
} )
export class BaseCmp { }
export class DerivedCmp extends BaseCmp { }
@NgModule ( {
declarations : [ DerivedCmp ] ,
} )
export class Module { }
` ,
} ,
} ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : 'test-package' ,
propertiesToConsider : [ 'main' ] ,
} ) ;
const jsContents = fs . readFile ( _ ( ` /node_modules/test-package/index.js ` ) ) ;
expect ( jsContents ) . toContain ( 'DerivedCmp.ɵcmp = ɵngcc0.ɵɵdefineComponent' ) ;
expect ( jsContents )
. toContain (
'features: [ɵngcc0.ɵɵInheritDefinitionFeature, ɵngcc0.ɵɵCopyDefinitionFeature]' ) ;
const dtsContents = fs . readFile ( _ ( ` /node_modules/test-package/index.d.ts ` ) ) ;
expect ( dtsContents )
. toContain (
'static ɵcmp: ɵngcc0.ɵɵComponentDefWithMeta<DerivedCmp, "[base]", never, {}, {}, never>;' ) ;
} ) ;
it ( 'should generate directive definitions with CopyDefinitionFeature for undecorated child directives in a long inheritance chain' ,
( ) = > {
genNodeModules ( {
'test-package' : {
'/index.ts' : `
import { Directive , NgModule } from '@angular/core' ;
@Directive ( {
selector : '[base]' ,
} )
export class BaseDir { }
export class DerivedDir1 extends BaseDir { }
export class DerivedDir2 extends DerivedDir1 { }
export class DerivedDir3 extends DerivedDir2 { }
@NgModule ( {
declarations : [ DerivedDir3 ] ,
} )
export class Module { }
` ,
} ,
} ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : 'test-package' ,
propertiesToConsider : [ 'main' ] ,
} ) ;
const dtsContents = fs . readFile ( _ ( ` /node_modules/test-package/index.d.ts ` ) ) ;
expect ( dtsContents )
. toContain (
'static ɵdir: ɵngcc0.ɵɵDirectiveDefWithMeta<DerivedDir1, "[base]", never, {}, {}, never>;' ) ;
expect ( dtsContents )
. toContain (
'static ɵdir: ɵngcc0.ɵɵDirectiveDefWithMeta<DerivedDir2, "[base]", never, {}, {}, never>;' ) ;
expect ( dtsContents )
. toContain (
'static ɵdir: ɵngcc0.ɵɵDirectiveDefWithMeta<DerivedDir3, "[base]", never, {}, {}, never>;' ) ;
} ) ;
} ) ;
2019-10-14 16:04:42 -04:00
describe ( 'aliasing re-exports in commonjs' , ( ) = > {
it ( 'should add re-exports to commonjs files' , ( ) = > {
loadTestFiles ( [
{
name : _ ( '/node_modules/test-package/package.json' ) ,
contents : `
{
"name" : "test-package" ,
"main" : "./index.js" ,
"typings" : "./index.d.ts"
}
` ,
} ,
{
name : _ ( '/node_modules/test-package/index.js' ) ,
contents : `
var __export = null ;
__export ( require ( "./module" ) ) ;
` ,
} ,
{
name : _ ( '/node_modules/test-package/index.d.ts' ) ,
contents : `
export * from "./module" ;
` ,
} ,
{
name : _ ( '/node_modules/test-package/index.metadata.json' ) ,
contents : '{}' ,
} ,
{
name : _ ( '/node_modules/test-package/module.js' ) ,
contents : `
var __decorate = null ;
var core_1 = require ( "@angular/core" ) ;
var directive_1 = require ( "./directive" ) ;
2019-11-07 13:34:40 -05:00
var LocalDir = /** @class */ ( function ( ) {
function LocalDir() {
}
LocalDir = __decorate ( [
core_1 . Directive ( {
selector : '[local]' ,
} )
] , LocalDir ) ;
return LocalDir ;
} ( ) ) ;
2019-10-14 16:04:42 -04:00
var FooModule = /** @class */ ( function ( ) {
function FooModule() {
}
FooModule = __decorate ( [
core_1 . NgModule ( {
2019-11-07 13:34:40 -05:00
declarations : [ directive_1 . Foo , LocalDir ] ,
exports : [ directive_1 . Foo , LocalDir ] ,
2019-10-14 16:04:42 -04:00
} )
] , FooModule ) ;
return FooModule ;
} ( ) ) ;
2019-11-07 13:34:40 -05:00
exports . LocalDir = LocalDir ;
2019-10-14 16:04:42 -04:00
exports . FooModule = FooModule ;
` ,
} ,
{
name : _ ( '/node_modules/test-package/module.d.ts' ) ,
contents : `
2019-11-07 13:34:40 -05:00
export declare class LocalDir { }
2019-10-14 16:04:42 -04:00
export declare class FooModule { }
` ,
} ,
{
name : _ ( '/node_modules/test-package/module.metadata.json' ) ,
contents : '{}' ,
} ,
{
name : _ ( '/node_modules/test-package/directive.js' ) ,
contents : `
var __decorate = null ;
var core_1 = require ( "@angular/core" ) ;
var Foo = /** @class */ ( function ( ) {
function Foo() {
}
Foo = __decorate ( [
core_1 . Directive ( {
selector : '[foo]' ,
} )
] , Foo ) ;
return Foo ;
} ( ) ) ;
exports . Foo = Foo ;
` ,
} ,
{
name : _ ( '/node_modules/test-package/directive.d.ts' ) ,
contents : `
export declare class Foo { }
` ,
} ,
{
name : _ ( '/node_modules/test-package/directive.metadata.json' ) ,
contents : '{}' ,
} ,
{
name : _ ( '/ngcc.config.js' ) ,
contents : `
module .exports = {
packages : {
'test-package' : {
entryPoints : {
'.' : {
generateDeepReexports : true
} ,
} ,
} ,
} ,
} ;
` ,
}
] ) ;
mainNgcc ( {
basePath : '/node_modules' ,
targetEntryPointPath : 'test-package' ,
propertiesToConsider : [ 'main' ] ,
} ) ;
expect ( loadPackage ( 'test-package' ) . __processed_by_ivy_ngcc__ ) . toEqual ( {
main : '0.0.0-PLACEHOLDER' ,
typings : '0.0.0-PLACEHOLDER' ,
} ) ;
const jsContents = fs . readFile ( _ ( ` /node_modules/test-package/module.js ` ) ) ;
const dtsContents = fs . readFile ( _ ( ` /node_modules/test-package/module.d.ts ` ) ) ;
expect ( jsContents ) . toContain ( ` var ɵngcc1 = require('./directive'); ` ) ;
expect ( jsContents ) . toContain ( 'exports.ɵngExportɵFooModuleɵFoo = ɵngcc1.Foo;' ) ;
expect ( dtsContents )
. toContain ( ` export {Foo as ɵngExportɵFooModuleɵFoo} from './directive'; ` ) ;
2019-11-07 13:34:40 -05:00
expect ( dtsContents . match ( /ɵngExportɵFooModuleɵFoo/g ) ! . length ) . toBe ( 1 ) ;
expect ( dtsContents ) . not . toContain ( ` ɵngExportɵFooModuleɵLocalDir ` ) ;
2019-10-14 16:04:42 -04:00
} ) ;
} ) ;
2019-06-06 15:22:32 -04:00
function loadPackage (
packageName : string , basePath : AbsoluteFsPath = _ ( '/node_modules' ) ) : EntryPointPackageJson {
return JSON . parse ( fs . readFile ( fs . resolve ( basePath , packageName , 'package.json' ) ) ) ;
}
2018-08-28 06:32:01 -04:00
2019-06-06 15:22:32 -04:00
function initMockFileSystem ( fs : FileSystem , testFiles : Folder ) {
if ( fs instanceof MockFileSystem ) {
fs . init ( testFiles ) ;
}
// a random test package that no metadata.json file so not compiled by Angular.
loadTestFiles ( [
{
name : _ ( '/node_modules/test-package/package.json' ) ,
contents : '{"name": "test-package", "es2015": "./index.js", "typings": "./index.d.ts"}'
} ,
{
name : _ ( '/node_modules/test-package/index.js' ) ,
contents :
'import {AppModule} from "@angular/common"; export class MyApp extends AppModule {};'
} ,
{
name : _ ( '/node_modules/test-package/index.d.ts' ) ,
contents :
'import {AppModule} from "@angular/common"; export declare class MyApp extends AppModule;'
} ,
] ) ;
// An Angular package that has been built locally and stored in the `dist` directory.
loadTestFiles ( [
{
name : _ ( '/dist/local-package/package.json' ) ,
contents : '{"name": "local-package", "es2015": "./index.js", "typings": "./index.d.ts"}'
} ,
{ name : _ ( '/dist/local-package/index.metadata.json' ) , contents : 'DUMMY DATA' } ,
{
name : _ ( '/dist/local-package/index.js' ) ,
contents :
` import {Component} from '@angular/core'; \ nexport class AppComponent {}; \ nAppComponent.decorators = [ \ n{ type: Component, args: [{selector: 'app', template: '<h2>Hello</h2>'}] } \ n]; `
} ,
{
name : _ ( '/dist/local-package/index.d.ts' ) ,
contents : ` export declare class AppComponent {}; `
} ,
] ) ;
2019-07-31 07:54:12 -04:00
// An Angular package that has a missing dependency
loadTestFiles ( [
{
name : _ ( '/node_modules/invalid-package/package.json' ) ,
contents : '{"name": "invalid-package", "es2015": "./index.js", "typings": "./index.d.ts"}'
} ,
{
name : _ ( '/node_modules/invalid-package/index.js' ) ,
contents : `
import { AppModule } from "@angular/missing" ;
import { Component } from '@angular/core' ;
export class AppComponent { } ;
AppComponent . decorators = [
{ type : Component , args : [ { selector : 'app' , template : '<h2>Hello</h2>' } ] }
] ;
`
} ,
{
name : _ ( '/node_modules/invalid-package/index.d.ts' ) ,
contents : ` export declare class AppComponent {} `
} ,
{ name : _ ( '/node_modules/invalid-package/index.metadata.json' ) , contents : 'DUMMY DATA' } ,
] ) ;
2018-08-28 06:32:01 -04:00
}
} ) ;
2019-06-06 15:22:32 -04:00
} ) ;