Security News
Fluent Assertions Faces Backlash After Abandoning Open Source Licensing
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
Encase is a library for using dependency injection within ruby applications. It provides a lightweight IOC Container that manages dependencies between your classes. It was written to assist with wiring of domain objects outside Rails applications to enable faster test suites.
Consider a Worker
class that has a dependency on a Logger
. We would
like this dependency to be available to the worker when it is created.
First we create a container object and declare the dependencies.
require 'encase/container'
container = Container.new
container.configure do
object :logger, Logger.new
factory :worker, Worker
end
Then we declare the logger
dependency in the Worker class by including
the Encase
module into it and using the needs
declaration.
require 'encase'
class Worker
include Encase
needs :logger
end
That's it! Now we can create a new Worker
by looking up the :worker
key on the Container.
my_worker = container.lookup('worker')
my_worker.logger.is_a?(Logger) # true
Containers are configured using a mini DSL in a configure
method.
Container items are stored with ruby symbols
and are used to later lookup the object.
container.configure do
# declarations go here
end
The declarations supported are listed below.
Objects are pre existing entities that have already been created in your
system. They are stored and returned as is without any modification. To
store objects use the object
declaration.
container.configure do
object :key, value
end
A Factory can be stored with the container to create instances of a class with it's dependencies auto-injected. On every lookup a new instance of that class will be returned.
container.configure do
factory :key, TheClass
end
A Singleton is similar to a Factory. However it caches the instance created on the first lookup and returns that instance on every subsequent lookups.
container.configure do
singleton :key, TheSingletonClass
end
To specify dependencies of a class, you use the needs
declaration. It
takes an array of symbols corresponding to the keys of the dependencies
stored in the container. Multiple needs
declarations are also supported.
require 'encase'
class Worker
include Encase
needs :a, :b, :c
needs :one
needs :two
needs :three
end
Encase allows storage of dependencies lazily. This can be useful if the dependencies aren't ready at the time of container configuration. But will ready before lookup.
Lazy initialization is done by passing a block
to the DSL declaration
instead of a value. Here :key
's block will be evaluated before the
first lookup.
container.configure do
object :key do
value
end
end
The block can optionally take an argument equal to the container object itself. You can use this to conditionally resolve the value based on other objects in the container or elsewhere.
container.configure do
object :key do |c|
value
end
end
Containers can also be nested within other containers. This allows grouping dependencies within different contexts of your application. When looking up keys, parent containers are queried when a key is not found in a child container.
parent_container = Container.new
parent_container.configure do
object :logger, Logger.new
factory :worker, Worker
end
child_container = parent_container.child
child_container.configure do
factory :worker, CustomWorker
end
child_container.lookup(:logger) # from parent_container
child_container.lookup(:worker) # CustomWorker
Here the child_container
will use CustomWorker
for resolving
:worker
. While the :logger
will be looked up from the
parent_container
Encase creates accessor
methods corresponding to each declared need
in the class. A container
accessor is also injected into the class for
looking up other dependencies at runtime.
class Worker
include Encase
needs :one, :two, :three
end
worker = container.lookup('worker')
worker.one
worker.two
worker.three
worker.container
An on_inject
event hook is provided to container items after their
dependencies are injected. Any post injection initialization can be
carried out here.
class Worker
include Encase
needs :logger
def on_inject
logger.log('Worker is ready')
end
end
Encase significantly simplifies testability of objects. In the earlier
example in order to test that the logger is indeed called by the worker
we can register the worker as a mock
. Then verify that this mock was called.
describe Worker do
let(:container) {
container = Container.new
container.configure do
factory :worker, Worker
end
container
}
it 'logs message to the logger' do
logger = double()
logger.should_receive(:log).with(anything())
container.object logger, double(:log => true)
worker = container.lookup(:worker)
worker.start()
end
end
Using the Encase created accessor
methods it is also possible to
manually assign the dependencies. Below logger
is directly assigned to
the worker
without requiring a container.
it 'logs message to the logger' do
logger = double()
logger.should_receive(:log).with(anything())
worker = Worker.new
worker.logger = logger
worker.start()
end
Add this line to your application's Gemfile:
gem 'encase'
And then execute:
$ bundle install
Encase has been tested to work on these platforms.
See contributing guidelines for Portkey.
FAQs
Unknown package
We found that encase demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
Research
Security News
Socket researchers uncover the risks of a malicious Python package targeting Discord developers.
Security News
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.