- Compatibility with dry-struct 1.0 and dry-types 1.0 (flash-gordon)
- [BREAKING]
:decorate
plugin was moved from dry-system to dry-container (available in 0.7.0+). To upgrade removeuse :decorate
and changedecorate
calls fromdecorate(key, decorator: something)
todecorate(key, with: something)
(flash-gordon) - [internal] Compatibility with dry-struct 0.7.0 and dry-types 0.15.0
- Support for stopping bootable components with
Container.stop(component_name)
(GustavoCaso)
- When using a non-finalized container, you can now resolve multiple different container objects registered using the same root key as a bootable component (timriley)
-
You can now set a custom inflector on the container level. As a result, the
Loader
's constructor accepts two arguments:path
andinflector
, update your custom loaders accordingly (flash-gordon)class MyContainer < Dry::System::Container configure do |config| config.inflector = Dry::Inflector.new do |inflections| inflections.acronym('API') end end end
- A helpful error will be raised if an invalid setting value is provided (GustavoCaso)
- When using setting plugin, will use default values from types (GustavoCaso)
- Minimal supported ruby version was bumped to
2.3
(flash-gordon) dry-struct
was updated to~> 0.5
(flash-gordon)
- Default namespace no longer breaks resolving dependencies with identifier that includes part of the namespace (ie
mail.mailer
) (GustavoCaso)
- Plugin dependencies are now auto-required and a meaningful error is raised when a dep failed to load (solnic)
- Plugin API (solnic)
:env
plugin which adds support for settingenv
config value (solnic):logging
plugin which adds a default logger (solnic):decorate
plugin for decorating registered objects (solnic):notifications
plugin adding pub/sub bus to containers (solnic):monitoring
plugin which addsmonitor
method for monitoring object method calls (solnic):bootsnap
plugin which adds support for bootsnap (solnic)
- [BREAKING] renamed
Container.{require=>require_from_root}
(GustavoCaso)
#bootable?
and#boot_file
methods were moved fromComponent
toBooter
(GustavoCaso)
- Aliasing an external component works correctly (solnic)
- Manually calling
:init
will also finalize a component (solnic)
- Support for external bootable components (solnic)
- Built-in
:system
components including:settings
component (solnic)
- Lazy-loading components work when a container has
default_namespace
configured (GustavoCaso)
- [BREAKING] Improved boot DSL with support for namespacing and lifecycle before/after callbacks (solnic)
Container.enable_stubs!
calls super too, which actually addsstub
API (solnic)- Issues with lazy-loading and import in stub mode are gone (solnic)
Container.enable_stubs!
for test environments which enables stubbing components (GustavoCaso)
- Component identifiers can now include same name more than once ie
foo.stuff.foo
(GustavoCaso) Container#boot!
was renamed toContainer#start
(davydovanton)Container#boot
was renamed toContainer#init
(davydovanton)
- Accept string values for Container's
root
config (timriley)
-
Added
manual_registrar
container setting (along with defaultManualRegistrar
implementation), andregistrations_dir
setting. These provide support for a well-established place for keeping files with manual container registrations (timriley) -
AutoRegistrar parses initial lines of Ruby source files for "magic comments" when auto-registering components. An
# auto_register: false
magic comment will prevent a Ruby file from being auto-registered (timriley) -
Container.auto_register!
, when called with a block, yields a configuration object to control the auto-registration behavior for that path, with support for configuring 2 different aspects of auto-registration behavior (both optional):class MyContainer < Dry::System::Container auto_register!('lib') do |config| config.instance do |component| # custom logic for initializing a component end config.exclude do |component| # return true to skip auto-registration of the component, e.g. # component.path =~ /entities/ end end end
-
A helpful error will be raised if a bootable component's finalize block name doesn't match its boot file name (GustavoCaso)
- The
default_namespace
container setting now supports multi-level namespaces (GustavoCaso) Container.auto_register!
yields a configuration block instead of a block for returning a custom instance (see above) (GustavoCaso)Container.import
now requires an explicit local name for the imported container (e.g.import(local_name: AnotherContainer)
) (timriley)
- Lazy load components as they are resolved, rather than on injection (timriley)
- Perform registration even though component already required (blelump)
- Undefined locals or method calls will raise proper exceptions in Lifecycle DSL (aradunovic)
This is a major refactoring with better internal APIs and improved support
for multi-container setups. As part of this release dry-system
has been renamed to dry-system
.
- Boot DSL with:
- Lifecycle triggers:
init
,start
andstop
(solnic) use
method which auto-boots a dependency and makes it available in the booting context (solnic)
- Lifecycle triggers:
- When a component relies on a bootable component, and is being loaded in isolation, the component will be booted automatically (solnic)
- [BREAKING]
Dry::Component::Container
is nowDry::System::Container
(solnic) - [BREAKING] Configurable
loader
is now a class that accepts container's config and responds to#constant
and#instance
(solnic) - [BREAKING]
core_dir
renameda tosystem_dir
and defaults tosystem
(solnic) - [BREAKING]
auto_register!
yieldsComponent
objects (solnic)
- Use new
Dry::Container#merge
which accepts:namespace
option (AMHOL) - Auto-registration is handled by a
System::AutoRegistrar
object configured in a container (solnic) - Booting is now handled by
System::Booter
object configured in a container (solnic) - Importing containers is now handled by
System::Importer
object configured in a container (solnic)
- Return immediately from
Container.load_component
if the requested component key already exists in the container. This fixes a crash when requesting to load a manually registered component with a name that doesn't map to a filename (timriley in #24)
- Ensure file components can be loaded when they're requested for the first time using their shorthand container identifier (i.e. with the container's default namespace removed) (timriley)
- Require the 0.4.0 release of dry-auto_inject for the features below (in 0.4.0) to work properly (timriley)
-
Support for supplying a default namespace to a container, which is passed to the container's injector to allow for convenient shorthand access to registered objects in the same namespace (timriley in #20)
# Set up container with default namespace module Admin class Container < Dry::Component::Container configure do |config| config.root = Pathname.new(__dir__).join("../..") config.default_namespace = "admin" end end Import = Container.injector end module Admin class CreateUser # "users.repository" will resolve an Admin::Users::Repository instance, # where previously you had to identify it as "admin.users.repository" include Admin::Import["users.repository"] end end
-
Support for supplying to options directly to dry-auto_inject's
Builder
viaDry::Component::Container#injector(options)
. This allows you to provide dry-auto_inject customizations like your own container of injection strategies (timriley in #20) -
Support for accessing all available injector strategies, not just the defaults (e.g.
MyContainer.injector.some_custom_strategy
) (timriley in #19)
- Subclasses of
Dry::Component::Container
no longer have anInjector
constant automatically defined within them. The recommended approach is to save your own injector object to a constant, which allows you to pass options to it at the same time, e.g.MyApp::Import = MyApp::Container.injector(my_options)
(timriley in #19)
Removed two pieces that are moving to dry-web:
- Removed
env
setting fromContainer
(timriley) - Removed
Dry::Component::Config
andoptions
setting fromContainer
(timriley) - Changed
Component#configure
behavior so it can be run multiple times for configuration to be applied in multiple passes (timriley)
- Component core directory is now
component/
by default (timriley) - Injector default stragegy is now whatever dry-auto_inject's default is (rather than hard-coding a particular default strategy for dry-system) (timriley)
- Fixed bug where specified auto-inject strategies were not respected (timriley)
-
Provide a dependency injector as an
Inject
constant inside any subclass ofDry::Component::Container
. This injector supports all ofdry-auto_inject
's default injection strategies, and will lazily load any dependencies as they are injected. It also supports arbitrarily switching strategies, so they can be used in different classes as required (e.g.include MyComponent::Inject.args["dep"]
) (timriley) -
Support aliased dependency names when calling the injector object (e.g.
MyComponent::Inject[foo: "my_app.foo", bar: "another.thing"]
) (timriley) -
Allow a custom dependency loader to be set on a container via its config (AMHOL)
class MyContainer < Dry::Component::Container configure do |config| # other config config.loader = MyLoader end end
Container.boot
now only makes a simplerequire
for the boot file (solnic)- Container object is passed to
Container.finalize
blocks (solnic) - Allow
Pathname
objects passed toContainer.require
(solnic) - Support lazily loading missing dependencies from imported containers (solnic)
Container.import_module
renamed to.injector
(timriley)- Default injection strategy is now
kwargs
, courtesy of the new dry-auto_inject default (timriley)
- Containers have a
name
setting (solnic) - Containers can be imported into one another (solnic)
- Container name is used to determine the name of its config file (solnic)
First public release, extracted from rodakase project