-
Notifications
You must be signed in to change notification settings - Fork 1.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(boot): add a booter for life cycle scripts
See #2034
- Loading branch information
1 parent
2521c1d
commit 8e4d1ac
Showing
6 changed files
with
147 additions
and
8 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,70 @@ | ||
// Copyright IBM Corp. 2018. All Rights Reserved. | ||
// Node module: @loopback/boot | ||
// This file is licensed under the MIT License. | ||
// License text available at https://opensource.org/licenses/MIT | ||
|
||
import {Constructor, inject} from '@loopback/context'; | ||
import { | ||
Application, | ||
asLifeCycleObserverBinding, | ||
CoreBindings, | ||
isLifeCycleObserverClass, | ||
LifeCycleObserver, | ||
} from '@loopback/core'; | ||
import {ArtifactOptions} from '../interfaces'; | ||
import {BootBindings} from '../keys'; | ||
import {BaseArtifactBooter} from './base-artifact.booter'; | ||
|
||
type LifeCycleObserverClass = Constructor<LifeCycleObserver>; | ||
|
||
/** | ||
* A class that extends BaseArtifactBooter to boot the 'LifeCycleScript' artifact type. | ||
* | ||
* Supported phases: configure, discover, load | ||
* | ||
* @param app Application instance | ||
* @param projectRoot Root of User Project relative to which all paths are resolved | ||
* @param [bootConfig] LifeCycleScript Artifact Options Object | ||
*/ | ||
export class LifeCycleScriptBooter extends BaseArtifactBooter { | ||
observers: LifeCycleObserverClass[]; | ||
|
||
constructor( | ||
@inject(CoreBindings.APPLICATION_INSTANCE) | ||
public app: Application, | ||
@inject(BootBindings.PROJECT_ROOT) projectRoot: string, | ||
@inject(`${BootBindings.BOOT_OPTIONS}#scripts`) | ||
public scriptConfig: ArtifactOptions = {}, | ||
) { | ||
super( | ||
projectRoot, | ||
// Set LifeCycleScript Booter Options if passed in via bootConfig | ||
Object.assign({}, LifeCycleScriptDefaults, scriptConfig), | ||
); | ||
} | ||
|
||
/** | ||
* Uses super method to get a list of Artifact classes. Boot each file by | ||
* creating a DataSourceConstructor and binding it to the application class. | ||
*/ | ||
async load() { | ||
await super.load(); | ||
|
||
this.observers = this.classes.filter(isLifeCycleObserverClass); | ||
for (const observer of this.observers) { | ||
this.app | ||
.bind(`lifeCycleObservers.${observer.name}`) | ||
.toClass(observer) | ||
.apply(asLifeCycleObserverBinding); | ||
} | ||
} | ||
} | ||
|
||
/** | ||
* Default ArtifactOptions for DataSourceBooter. | ||
*/ | ||
export const LifeCycleScriptDefaults: ArtifactOptions = { | ||
dirs: ['scripts'], | ||
extensions: ['.script.js'], | ||
nested: true, | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
// Copyright IBM Corp. 2018. All Rights Reserved. | ||
// Node module: @loopback/boot | ||
// This file is licensed under the MIT License. | ||
// License text available at https://opensource.org/licenses/MIT | ||
|
||
import {LifeCycleObserver} from '@loopback/core'; | ||
|
||
export class MyLifeCycleObserver implements LifeCycleObserver { | ||
status = ''; | ||
|
||
async start() { | ||
this.status = 'started'; | ||
} | ||
|
||
stop() { | ||
this.status = 'stopped'; | ||
} | ||
} |
48 changes: 48 additions & 0 deletions
48
packages/boot/test/integration/lifecycle-script.booter.integration.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,48 @@ | ||
// Copyright IBM Corp. 2018. All Rights Reserved. | ||
// Node module: @loopback/boot | ||
// This file is licensed under the MIT License. | ||
// License text available at https://opensource.org/licenses/MIT | ||
|
||
import {expect, TestSandbox} from '@loopback/testlab'; | ||
import {resolve} from 'path'; | ||
import {BooterApp} from '../fixtures/application'; | ||
import {CoreBindings, BindingScope} from '@loopback/core'; | ||
|
||
describe('lifecycle script booter integration tests', () => { | ||
const SANDBOX_PATH = resolve(__dirname, '../../.sandbox'); | ||
const sandbox = new TestSandbox(SANDBOX_PATH); | ||
|
||
const SCRIPTS_PREFIX = 'lifeCycleObservers'; | ||
const SCRIPTS_TAG = CoreBindings.LIFE_CYCLE_OBSERVER_TAG; | ||
|
||
let app: BooterApp; | ||
|
||
beforeEach('reset sandbox', () => sandbox.reset()); | ||
beforeEach(getApp); | ||
|
||
it('boots scripts when app.boot() is called', async () => { | ||
const expectedBinding = { | ||
key: `${SCRIPTS_PREFIX}.MyLifeCycleObserver`, | ||
tags: [SCRIPTS_TAG], | ||
scope: BindingScope.SINGLETON, | ||
}; | ||
|
||
await app.boot(); | ||
|
||
const bindings = app | ||
.findByTag(SCRIPTS_TAG) | ||
.map(b => ({key: b.key, tags: b.tagNames, scope: b.scope})); | ||
expect(bindings).to.containEql(expectedBinding); | ||
}); | ||
|
||
async function getApp() { | ||
await sandbox.copyFile(resolve(__dirname, '../fixtures/application.js')); | ||
await sandbox.copyFile( | ||
resolve(__dirname, '../fixtures/lifecycle-script.artifact.js'), | ||
'scripts/lifecycle-script.script.js', | ||
); | ||
|
||
const MyApp = require(resolve(SANDBOX_PATH, 'application.js')).BooterApp; | ||
app = new MyApp(); | ||
} | ||
}); |