Skip to content

wartman/capsule

Repository files navigation

Capsule

Capsule is a minimal, easy to use dependency injection library.

Note: The previous version can be found here.

Features

  • Simple, opinionated API.
  • All the complicated stuff is handled by macros -- at runtime Capsule is just a few simple classes.
  • Using capsule.Modules and capsule.Container.build will check your dependencies at compile time -- no more runtime exceptions if you forget to add something, and you'll be warned if any changes to your code requires a new dependency.

Getting Started

Install using Lix:

lix install gh:wartman/capsule

Install using haxelib:

Not available yet

Add -lib capsule to your hxml file and you're ready to go!

Guide

The examples folder is a good place to see how Capsule works too!

Here's a quick look at the API in action:

import capsule.Container;
import capsule.Module;

interface Foo {
  public function getFoo():String;
}

interface Bar {
  public function getBar():String;
}

interface FooBar {
  public function getFooBar():String;
}

class DefaultFoo implements Foo {
  public function new() {}

  public function getFoo() {
    return 'foo';
  }
}

class DefaultBar implements Bar {
  public function new() {}

  public function getBar() {
    return 'bar';
  }
}

class DefaultFooBar implements FooBar {
  final foo:Foo;
  final bar:Bar;

  public function new(foo, bar) {
    this.foo = foo;
    this.bar = bar;
  }

  public function getFooBar() {
    return '${foo.geFoo()}${bar.getBar()}';
  }
}

class FooAndBarModule implements Module {
  public function new() {}

  public function provide(container:Container) {
    container.map(Foo).to(DefaultFoo);
    container.map(Bar).to(DefaultBar);
  }
}

class FooBarModule implements Module {
  public function new() {}

  public function provide(container:Container) {
    container.map(FooBar).to(DefaultFooBar);
  }
}

function main() {
  var container = Container.build(
    new FooAndBarModule(),
    new FooBarModule()
  );
  trace(container.get(FooBar).getFooBar()); // => "foobar"
}

This should all be pretty straightforward, but there are some important things to call out.

The first is that Container.build is a macro that ensures the dependencies of all capsule.Modules passed to it are satisfied. If, for example, we omitted the FooAndBarModule from the above example:

function main() {
  var container = Container.build(
    // new FooAndBarModule(),
    new FooBarModule()
  );
  trace(container.get(FooBar).getFooBar());
}

...our code wouldn't compile. Instead, we'd get an error telling us that the Foo and Bar dependencies were not satisfied. You don't need to use Capsule with Container.build and Modules, but it's probably a good idea.

Something that the example doesn't cover is how to handle generic types. Haxe only lets us use the angle bracket syntax (e.g. Map<String, String>) in a few places, so Capsule hacks the function-call syntax to get around this:

capsule.map(Map(String, String)).to([ 'foo' => 'bar', 'bin' => 'bax' ]);

If you're new to Haxe, please note that this is NOT standard syntax. It'll only work in capsule.map(...), capsule.get(...) and capsule.map(...).to(...).

Another thing not covered in the example are the different kinds of values you can map to. The simplest is mapping to a Class, which automatically injects its constructor:

container.map(FooBar).to(FooBar);

However, say we wanted to provide a different implementation of Foo only for FooBar. We could map to a function instead:

container.map(FooBar).to(function (bar:Bar) {
  return new FooBar(new SomeOtherFoo(), bar);
});

Function parameters will all be injected by the container and tracked by Modules, just like mapping to a class. Note that any function will work here, so something like this is fine:

container.map(FooBar).to(FooBar.createWithCustomFoo);

Internally Capsule is actually mapping everything to functions -- container.map(FooBar).to(FooBar) is the same as container.map(FooBar).to(FooBar.new), and if you poke around in the source code you'll see that's exactly what's happening.

You can also just map a type to a value, like we did with Map<String, String>.

// This will work:
container.map(String).to('foo');

Unlike the other mappings, value mappings will always return the same value. Function and Class mappings will be called every time, returning a new instance/value. This isn't always what we want, so you can mark a mapping as shared with the share method:

container.map(FooBar).to(DefaultFooBar).share();

Because this is such a common pattern, you can also use the toShared shortcut to do the same thing:

container.map(FooBar).toShared(DefaultFooBar);

This will ensure that an instance is only created once, and is returned whenever it's requested thereafter.

If you need to extend a mapping -- say you need to register a route to a router in some notional web app -- you can call getMapping from your Container and extend it:

container.getMapping(Router).extend(router -> {
  router.add(new Route('/foo/bar'));
  // You MUST return a Router from this function. Note that this means
  // you're also able to change the value of a mapping using `extend`.
  return router;
});

Importantly, you can extend a mapping that does not exist yet. The following code will work just fine:

container.getMapping(Router).extend(router -> {
  router.add(new Route('/foo/bar'));
  // You MUST return a Router from this function. Note that this means
  // you're also able to change the value of a mapping using `extend`.
  return router;
});
container.map(Router).toShared(Router);

This is done to ensure that you don't need to worry about the order you map things in -- everything should just work.

Changelog

0.4.0

  • Removed the confusing getChild stuff. It wasn't very useful and it mostly added lots of strange complexity and, worse, led to using Containers in very weird ways. Instead, you can now clone a Container if you really need to.

0.3.0

  • Breaks anything that used the old version of Capsule. Is that a feature?
    • I promise it's for the best.
  • Removed all @:inject.* meta. Instead, dependencies are only injected into constructors (or derived from any function's arguments). This is to simplify the API and ensure that code doesn't require Capsule to work.
  • All functions passed to the Mapping.to(...) macro are injectable now, not just lambdas.
  • capsule.Module replaces capsule.ServiceProvider and tracks dependencies at compile time.