From 234ec10ef25e99e7eb58be1dece54aed212a11d7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Miroslav=20Bajto=C5=A1?= Date: Mon, 8 Apr 2019 13:37:13 +0200 Subject: [PATCH] feat(build): add more TypeScript "strict" checks Enable all "strict" checks with the exception of `strictPropertyInitialization` which would require significant changes. The following additional checks are added: - noImplicitThis - alwaysStrict - strictFunctionTypes In the future, any checks added to TypeScript "strict" mode will be automatically enabled for LoopBack projects too. --- benchmark/src/benchmark.ts | 9 +++++++-- packages/build/config/tsconfig.common.json | 7 ++++--- .../class-level-bindings.acceptance.ts | 4 ++-- .../src/__tests__/unit/resolver.unit.ts | 12 ++++++++--- packages/context/src/context-view.ts | 20 ++++++++++++------- .../src/__tests__/unit/type-resolver.unit.ts | 10 ++++++++-- 6 files changed, 43 insertions(+), 19 deletions(-) diff --git a/benchmark/src/benchmark.ts b/benchmark/src/benchmark.ts index efebf07fe0ed..cd1ea48df420 100644 --- a/benchmark/src/benchmark.ts +++ b/benchmark/src/benchmark.ts @@ -5,7 +5,7 @@ import * as byline from 'byline'; import {ChildProcess, spawn} from 'child_process'; -import pEvent from 'p-event'; +import pEvent, {Emitter} from 'p-event'; import {Autocannon, EndpointStats} from './autocannon'; import {Client} from './client'; import {scenarios} from './scenarios'; @@ -111,5 +111,10 @@ function startWorker() { async function closeWorker(worker: ChildProcess) { worker.kill(); - await pEvent(worker, 'close'); + await pEvent( + // workaround for a bug in pEvent types which makes them + // incompatible with "strictFunctionTypes" + worker as Emitter<[unknown]>, + 'close', + ); } diff --git a/packages/build/config/tsconfig.common.json b/packages/build/config/tsconfig.common.json index eb58cee2e096..5aa85c2b0b3b 100644 --- a/packages/build/config/tsconfig.common.json +++ b/packages/build/config/tsconfig.common.json @@ -3,11 +3,12 @@ "compilerOptions": { "emitDecoratorMetadata": true, "experimentalDecorators": true, - "noImplicitAny": true, - "strictNullChecks": true, "resolveJsonModule": true, - "strictBindCallApply": true, "skipLibCheck": true, + "strict": true, + + // FIXME(bajtos) LB4 is not compatible with this setting yet + "strictPropertyInitialization": false, "lib": ["es2018", "esnext.asynciterable"], "module": "commonjs", diff --git a/packages/context/src/__tests__/acceptance/class-level-bindings.acceptance.ts b/packages/context/src/__tests__/acceptance/class-level-bindings.acceptance.ts index 046c83cd69dd..925731bb3595 100644 --- a/packages/context/src/__tests__/acceptance/class-level-bindings.acceptance.ts +++ b/packages/context/src/__tests__/acceptance/class-level-bindings.acceptance.ts @@ -458,8 +458,8 @@ describe('Context bindings - Injecting dependencies of classes', () => { 'location', {}, // Set up a custom resolve() to access information from the session - (c: Context, injection: Injection, session: ResolutionSession) => { - resolutionPath = session.getResolutionPath(); + (c: Context, injection: Injection, session?: ResolutionSession) => { + resolutionPath = session ? session.getResolutionPath() : undefined; return 'San Jose'; }, ) diff --git a/packages/context/src/__tests__/unit/resolver.unit.ts b/packages/context/src/__tests__/unit/resolver.unit.ts index bd4603030f80..5bf4e29b1c09 100644 --- a/packages/context/src/__tests__/unit/resolver.unit.ts +++ b/packages/context/src/__tests__/unit/resolver.unit.ts @@ -213,7 +213,9 @@ describe('constructor injection', () => { 'p', {}, // Set up a custom resolve() to access information from the session - (c: Context, injection: Injection, session: ResolutionSession) => { + (c: Context, injection: Injection, session?: ResolutionSession) => { + if (!session) + throw new Error('Resolver was called with no ResolutionSession'); bindingPath = session.getBindingPath(); resolutionPath = session.getResolutionPath(); }, @@ -251,7 +253,9 @@ describe('constructor injection', () => { 'p', {}, // Set up a custom resolve() to access information from the session - (c: Context, injection: Injection, session: ResolutionSession) => { + (c: Context, injection: Injection, session?: ResolutionSession) => { + if (!session) + throw new Error('Resolver was called with no ResolutionSession'); bindingPath = session.getBindingPath(); resolutionPath = session.getResolutionPath(); decorators = session.injectionStack.map(i => i.metadata!.decorator); @@ -290,7 +294,9 @@ describe('constructor injection', () => { 'p', {}, // Set up a custom resolve() to access information from the session - (c: Context, injection: Injection, session: ResolutionSession) => { + (c: Context, injection: Injection, session?: ResolutionSession) => { + if (!session) + throw new Error('Resolver was called with no ResolutionSession'); injectionPath = session.getInjectionPath(); }, ) diff --git a/packages/context/src/context-view.ts b/packages/context/src/context-view.ts index ce5b9cf48f0f..2cdb8ec5f605 100644 --- a/packages/context/src/context-view.ts +++ b/packages/context/src/context-view.ts @@ -40,11 +40,16 @@ export class ContextView extends EventEmitter protected _cachedValues: T[] | undefined; private _subscription: Subscription | undefined; - constructor( - protected readonly context: Context, - public readonly filter: BindingFilter, - ) { + // Workaround to pass TypeScript's "strictFunctionTypes" check: + // Type 'BindingFilter' is not assignable to type 'BindingFilter'. + // Type 'unknown' is not assignable to type 'T'. + readonly filter: BindingFilter; + + constructor(protected readonly context: Context, filter: BindingFilter) { super(); + + // Workaround to pass TypeScript's "strictFunctionTypes" check + this.filter = filter as BindingFilter; } /** @@ -85,10 +90,11 @@ export class ContextView extends EventEmitter /** * Find matching bindings and refresh the cache */ - protected findBindings() { + protected findBindings(): Readonly>[] { debug('Finding matching bindings'); - this._cachedBindings = this.context.find(this.filter); - return this._cachedBindings; + const found = this.context.find(this.filter); + this._cachedBindings = found; + return found; } /** diff --git a/packages/repository/src/__tests__/unit/type-resolver.unit.ts b/packages/repository/src/__tests__/unit/type-resolver.unit.ts index ceb469741bd1..a3079a4c7842 100644 --- a/packages/repository/src/__tests__/unit/type-resolver.unit.ts +++ b/packages/repository/src/__tests__/unit/type-resolver.unit.ts @@ -4,7 +4,12 @@ // License text available at https://opensource.org/licenses/MIT import {expect} from '@loopback/testlab'; -import {isBuiltinType, isTypeResolver, resolveType} from '../../type-resolver'; +import { + isBuiltinType, + isTypeResolver, + resolveType, + TypeResolver, +} from '../../type-resolver'; class SomeModel { constructor(public name: string) {} @@ -60,7 +65,8 @@ describe('isTypeResolver', () => { describe('resolveType', () => { it('resolves the arg when the value is a resolver', () => { - const ctor = resolveType(() => SomeModel); + const resolver: TypeResolver = () => SomeModel; + const ctor = resolveType(resolver); expect(ctor).to.eql(SomeModel); const inst = new ctor('a-name');