A C++98 dependency injection framework.
If you do not need strict C++98 compatibility, I suggest you look at Google's fruit instead.
The design and name are inspired by Google's excellent Guice framework, but neither Google nor Guice is otherwise affiliated in any way.
I refer the reader to Guice's documentation for an introduction to dependency injection as a concept, and why they might be interested in using it.
In Sauce, one defines bindings that map interface types to implementation types. Each
binding is declared in the context of a module which is used to organize and refer to
collections of bindings at a time. Modules may be function pointers, or classes
providing a certain
operator(). Groups of modules may be used together, to avoid
At runtime, one collects desired modules into a modules object, which produces
injectors. One can then ask an injector to provide a value (instance) of a desired
type (again supplied as a template parameter.) When providing a value, implicit
transitive dependencies are provided as well. All values are exchanged with shared
(std|std::tr1|boost)::shared_ptrs are supported) and the injector takes care
of disposing the value when the smart pointer deletes itself.
Requesting the injector for an unbound type results in a runtime exception. No RTTI is used (but we use a portable, homebrew version of same.)
Sauce is available with a liberal, BSD-ish license.
$ vagrant up $ vagrant ssh vagrant@sauce:~$ cd sauce vagrant@sauce:~/sauce$ ./configure # create make files vagrant@sauce:~/sauce$ make check # compile and run unit tests
A side-effect of using an injector to hide implementation choices from dependents is the
discouraged use of stack allocation and the
new operator for dependencies. The
operator (et. al.) has an additional guarantee past the ensuring the instance's concrete
type: the instance received is unique, and not shared. This raises the question: will
the injector always provide new, successive instances, or will it ever reuse some?
It turns out the most useful answer is "both". Depending on the context, it may be appropriate to always create new instances upon request, to always share a solitary instance with everyone (such as in the Singleton pattern), or to do something in between.
Sauce supports scopes to answer this need. While within a scope, a participating
binding will only ever create a single instance of its bound type: if the dependency is
provided more than once, the same instance is reused. One enters a scope with the
enter method on the injector, to receive a scoped injector. Requests made to this
injector will benefit from the open scope. To leave the scope, stop using the scoped
injector (either by simply dropping it on the floor, or by calling
exit to recover the
original.) The injectors returned from
Modules instances are created implicitly in
SingletonScope, which can not be exited (at pain of runtime exception.)
Entered scopes form a stack: entering the
RequestScope from a
will result in an injector that is within both scopes. Put differently, such an injector
will try to provide both
SessionScope dependencies from cache. They
are a stack in the sense that injectors beneath the top are guaranteed to survive those
above them. It is illegal to reenter a scope already on the stack; a runtime exception
It is possible (and encouraged!) to reenter a scope many times from a single injector in
parallel. For example, one may enter
RequestScope from the same
injector many times concurrently, to create many contemporary
These will all cache
RequestScope dependencies separately, but share the same
SessionScope cache. Such shared scopes are not thread-safe by default, but may be made
so by supplying a locker type and a lockable instance when creating the initial, root
mutex are suitable for this purpose.
Sometimes it is convenient to force the creation of all dependencies up front in a given scope (such as singleton scope.) This can help programs fail fast, by exposing environmental issues or other problems at start up. Sauce supports this by optionally eagerly injecting arbitrary scopes (with an injector method.) One may only eagerly inject dependencies in open scopes.
Unlike Guice, Sauce expects the developer to enter and eagerly inject scopes explicitly, at their convenience. No entrance or eager injection occurs implicitly.
Circular dependency detection Setter injection Named, overloaded bindings Eager-loaded singletons Injectable Providers for lazy resolution
- Implicit bindings implied by integration within interfaces or implementations
On-demand injection for provided instances
- Member field injection (meh)
- Static injection (meh)
These peeps are amaze for helping make Sauce better! Buy them all the drink of their choice!
- Casey B.
- Jakub S.
- Markus E.
(If you're up here and want me to tweak/link, make a ticket or otherwise prod me.)
Copyright (c) 2011-2017 Phil Smith. See LICENSE for details.