From 2253c9168b78b7e24fa57e737794c58e70b596e8 Mon Sep 17 00:00:00 2001 From: Jordan Date: Tue, 13 Aug 2019 11:45:28 -0700 Subject: [PATCH] feat: default service identifier target to class when omitted --- src/decorator/fluent_provide.ts | 5 +- src/decorator/provide.ts | 6 +- test/decorator/fluent_provide.test.ts | 25 +++----- test/index.test.ts | 90 +++++++++++++++++++++++++++ 4 files changed, 103 insertions(+), 23 deletions(-) diff --git a/src/decorator/fluent_provide.ts b/src/decorator/fluent_provide.ts index c764879..012e8b4 100644 --- a/src/decorator/fluent_provide.ts +++ b/src/decorator/fluent_provide.ts @@ -6,9 +6,8 @@ import ProvideDoneSyntax from "../syntax/provide_done_syntax"; import interfaces from "../interfaces/interfaces"; import { interfaces as inversifyInterfaces } from "inversify"; -function fluentProvide(serviceIdentifier: inversifyInterfaces.ServiceIdentifier) { - - let bindingWhenOnSyntax = (bind: inversifyInterfaces.Bind, target: any) => bind(serviceIdentifier).to(target); +function fluentProvide(serviceIdentifier?: inversifyInterfaces.ServiceIdentifier) { + let bindingWhenOnSyntax = (bind: inversifyInterfaces.Bind, target: any) => bind(serviceIdentifier || target).to(target); let bindingConstraintFunction = (bind: inversifyInterfaces.Bind, target: any) => (bindingWhenOnSyntax(bind, target))._binding; let provideDoneSyntax = new ProvideDoneSyntax(bindingConstraintFunction); diff --git a/src/decorator/provide.ts b/src/decorator/provide.ts index 3e98974..7350278 100644 --- a/src/decorator/provide.ts +++ b/src/decorator/provide.ts @@ -4,12 +4,11 @@ import interfaces from "../interfaces/interfaces"; import { METADATA_KEY } from "../constants"; function provide( - serviceIdentifier: inversifyInterfaces.ServiceIdentifier, + serviceIdentifier?: inversifyInterfaces.ServiceIdentifier, force?: boolean ) { return function (target: any) { - const isAlreadyDecorated = Reflect.hasOwnMetadata(inversify_METADATA_KEY.PARAM_TYPES, target); const redecorateWithInject = force === true; @@ -30,7 +29,8 @@ function provide( } const currentMetadata: interfaces.ProvideSyntax = { - constraint: (bind: inversifyInterfaces.Bind, bindTarget: any) => bind(serviceIdentifier).to(bindTarget), + constraint: (bind: inversifyInterfaces.Bind, bindTarget: any) => + bind(serviceIdentifier || bindTarget).to(bindTarget), implementationType: target }; diff --git a/test/decorator/fluent_provide.test.ts b/test/decorator/fluent_provide.test.ts index b758ea2..2e488a7 100644 --- a/test/decorator/fluent_provide.test.ts +++ b/test/decorator/fluent_provide.test.ts @@ -44,23 +44,14 @@ describe("fluentProvide", () => { }); - it("Should work if @provide is applied more than once with force flag", () => { - - const provideSingleton = (identifier: any) => { - return fluentProvide(identifier) - .inSingletonScope() - .done(true); // IMPORTANT! - }; - - function shouldThrow() { - @provideSingleton("Ninja") - @provideSingleton("SilentNinja") - class Ninja { } - return Ninja; - } - - expect(shouldThrow).not.to.throw(); - + it("Should work if @fluentProvide is applied with no arguments", () => { + function shouldThrow() { + @fluentProvide().inSingletonScope().done() + class Ninja {} + return Ninja; + } + + expect(shouldThrow).not.to.throw(); }); }); diff --git a/test/index.test.ts b/test/index.test.ts index 97e7711..709b978 100644 --- a/test/index.test.ts +++ b/test/index.test.ts @@ -131,6 +131,51 @@ describe("inversify-binding-decorators", () => { }); + it("Should be able to declare bindings using classes as identifiers on empty provide arguments", () => { + let container = new Container(); + + @provide() + class Katana { + public hit() { + return "cut!"; + } + } + + @provide() + class Shuriken { + public throw() { + return "hit!"; + } + } + + @provide() + class Ninja { + public katana: Katana; + public shuriken: Shuriken; + + public constructor(katana: Katana, shuriken: Shuriken) { + this.katana = katana; + this.shuriken = shuriken; + } + + public fight() { + return this.katana.hit(); + } + public sneak() { + return this.shuriken.throw(); + } + } + + container.load(buildProviderModule()); + let ninja = container.get(Ninja); + + expect(ninja instanceof Ninja).eql(true); + expect(ninja.katana instanceof Katana).eql(true); + expect(ninja.shuriken instanceof Shuriken).eql(true); + expect(ninja.fight()).eql("cut!"); + expect(ninja.sneak()).eql("hit!"); + }); + it("Should be able to declare bindings using symbols as identifiers", () => { let container = new Container(); @@ -482,4 +527,49 @@ describe("inversify-binding-decorators", () => { }); + it("Should be able to declare bindings using classes as identifiers on empty provide arguments", () => { + let container = new Container(); + + @fluentProvide().inSingletonScope().done() + class Katana { + public hit() { + return "cut!"; + } + } + + @fluentProvide().inSingletonScope().done() + class Shuriken { + public throw() { + return "hit!"; + } + } + + @provide() + class Ninja { + public katana: Katana; + public shuriken: Shuriken; + + public constructor(katana: Katana, shuriken: Shuriken) { + this.katana = katana; + this.shuriken = shuriken; + } + + public fight() { + return this.katana.hit(); + } + public sneak() { + return this.shuriken.throw(); + } + } + + container.load(buildProviderModule()); + let ninja = container.get(Ninja); + + expect(ninja instanceof Ninja).eql(true); + expect(ninja.katana instanceof Katana).eql(true); + expect(ninja.shuriken instanceof Shuriken).eql(true); + expect(ninja.fight()).eql("cut!"); + expect(ninja.sneak()).eql("hit!"); + }); + });