NAME
Resource::Silo - lazy declarative resource container for Perl.
DESCRIPTION
We assume the following setup:
- (i) The application needs to access multiple resources, such as configuration files, databases, queues, service endpoints, credentials, etc.
- (ii) The application has helper scripts that don't need to initialize all the resources at once, as well as a test suite where accessing resources is undesirable unless a fixture or mock is provided.
- (iii) The resource management has to be decoupled from the application logic where possible.
And we propose the following solution:
- (i) All available resources are declared in one place and encapsulated within a single container.
- (ii) Such container is equipped with methods to access resources, as well as an exportable prototyped function for obtaining the one and true instance of it (a.k.a. optional singleton).
- (iii) Every class or script in the project accesses resources through this container and only through it.
SYNOPSIS
The default mode is to create a one-off container for all resources and export if into the calling class via silo
function.
Note that calling use Resource::Silo
from a different module will create a separate container instance. You'll have to re-export (or otherwise provide access to) the silo
function.
This is done on purpose so that multiple projects or modules can coexist within the same interpreter without interference.
package My::App;
use Resource::Silo;
use DBI;
use YAML::LoadFile;
...
resource config => sub { LoadFile( ... ) };
resource dbh => sub {
my $self = shift;
my $conf = $self->config->{dbh};
DBI->connect( $conf->{dsn}, $conf->{user}, $conf->{pass}, { RaiseError => 1 } );
};
resource queue => sub { My::Queue->new( ... ) };
...
my $statement = silo->dbh->prepare( $sql );
my $queue = silo->queue;
For more complicated projects, it may make more sense to create a dedicated class for resource management:
# in the container class
package My::Project::Res;
use Resource::Silo -class; # resource definitions will now create
# eponymous methods in My::Project::Res
resource foo => sub { ... }; # declare resources as in the above example
resource bar => sub { ... };
1;
# in all other modules/packages/scripts:
package My::Project;
use My::Project::Res qw(silo);
silo->foo; # obtain resources
silo->bar;
My::Project::Res->new; # separate empty resource container
EXPORT
The following functions will be exported into the calling module, unconditionally:
silo - a singleton function returning the resource container. Note that this function will be created separately for every calling module, and needs to be re-exported to be shared.
resource - a DSL for defining resources, their initialization and properties. See below.
Additionally, if the -class
argument was added to the use line, the following things happen:
Resource::Silo::Container and Exporter are added to
@ISA
;silo
function is added to@EXPORT
and thus becomes re-exported by default;calling
resource
creates a corresponding method in this package.
RESOURCE DECLARATION
resource
resource 'name' => sub { ... };
resource 'name' => %options;
%options may include:
init => sub { $self, $name, [$argument] }
A coderef to obtain the resource. Required.
If the number of arguments is odd, the last one is popped and considered to be the init function.
argument =>
sub { ... }
||qr( ... )
If specified, assume that the resource in question may have several instances, distinguished by a string argument. Such argument will be passed as the 3rd parameter to the
init
function.This may be useful e.g. for DBIx::Class result sets, or for Redis::Namespace.
A regular expression will always be anchored to match the whole string. A function must return true for the parameter to be valid.
If the argument is omitted, it is assumed to be an empty string.
Example:
use Resource::Silo; use Redis; use Redis::Namespace; resource real_redis => sub { Redis->new }; my %known_namespaces = ( users => 1, sessions => 1, counters => 1, ); resource redis => argument => sub { $known_namespaces{ +shift } }, init => sub { my ($self, $name, $ns) = @_; Redis::Namespace->new( redis => $self->real_redis, namespace => $ns, ); };
derivative => 1 | 0
Assume that resource can be derived from its dependencies, or that it introduces no extra side effects compared to them.
Examples may be Redis::Namespace built on top of a Redis handle or DBIx::Class built on top of DBI connection.
Derivative resources may be instantiated even in locked mode, as they would only initialize if their dependencies have already been initialized or overridden.
ignore_cache => 1 | 0
If set, don't cache resource, always create a fresh one instead. See also "fresh" in Resource::Silo::Container.
preload => 1 | 0
If set, try loading the resource when
silo->ctl->preload
is called. Useful if you want to throw errors when a service is starting, not during request processing.cleanup => sub { $resource_instance }
Undo the init procedure. Usually it is assumed that the resource will do it by itself in the destructor, e.g. that's what a DBI connection would do. However, if it's not the case, or resources refer circularly to one another, a manual "destructor" may be specified.
It only accepts the resource itself as an argument and will be called before erasing the object from the cache.
See also
fork_cleanup
.cleanup_order => $number
The higher the number, the later the resource will get destroyed.
The default is 0, negative numbers are also valid, if that makes sense for you application (e.g. destroy
$my_service_main_object
before the resources it consumes).fork_cleanup => sub { $resource_instance }
Like
cleanup
, but only in case a change in process ID was detected. See "FORKING"This may be useful if cleanup is destructive and shouldn't be performed twice.
TESTING: LOCK AND OVERRIDES
It's usually a bad idea to access real-world resources in one's test suite, especially if it's e.g. a partner's endpoint.
Now the #1 rule when it comes to mocks is to avoid mocks and instead design the modules in such a way that they can be tested in isolation. This however may not always be easily achievable.
Thus, Resource::Silo provides a mechanism to substitute a subset of resources with mocks and forbid the instantiation of the rest, thereby guarding against unwanted side-effects.
The lock
/unlock
methods in Resource::Silo::Container, available via silo->ctl
frontend, temporarily forbid instantiating new resources. The resources already in cache will still be OK though.
The override
method allows to supply substitutes for resources or their initializers.
The derivative
flag in the resource definition may be used to indicate that a resource is safe to instantiate as long as its dependencies are either instantiated or mocked, e.g. a DBIx::Class schema is probably fine as long as the underlying database connection is taken care of.
Here is an example:
use Test::More;
use My::Project qw(silo);
silo->ctl->lock->override(
dbh => DBI->connect( 'dbi:SQLite:database=:memory:', '', '', { RaiseError => 1 ),
);
silo->dbh; # a mocked database
silo->schema; # a DBIx::Class schema reliant on the dbh
silo->endpoint( 'partner' ); # an exception as endpoint wasn't mocked
CAVEATS AND CONSIDERATIONS
See Resource::Silo::Container for resource container implementation. As of current, it is probably a bad idea to use Moose on the same class as Resource::Silo.
CACHING
All resources are cached, the ones with arguments are cached together with the argument.
FORKING
If the process forks, resources such as database handles may become invalid or interfere with other processes' copies. As of current, if a change in the process ID is detected, the resource cache is reset altogether.
This may change in the future as some resources (e.g. configurations or endpoint URLs) are stateless and may be preserved.
CIRCULAR DEPENDENCIES
If a resource depends on other resources, those will be simply created upon request.
It is possible to make several resources depend on each other. Trying to initialize such resource will cause an expection, however.
SEE ALSO
Bread::Board - a more mature IoC / DI framework.
ACKNOWLEDGEMENTS
This module was names after a building in the game Heroes of Might and Magic III.
This module was inspired in part by my work for Cloudbeds. That was a great time and I had great coworkers!
BUGS
Please report any bugs or feature requests to https://github.com/dallaylaen/resource-silo-p5/issues or via RT: https://rt.cpan.org/NoAuth/ReportBug.html?Queue=Resource-Silo.
SUPPORT
You can find documentation for this module with the perldoc command.
perldoc Resource::Silo
You can also look for information at:
RT: CPAN's request tracker (report bugs here)
CPAN Ratings
Search CPAN
COPYRIGHT AND LICENSE
Copyright (c) 2023, Konstantin Uvarin, <khedin@gmail.com>
This program is free software. You can redistribute it and/or modify it under the terms of either: the GNU General Public License as published by the Free Software Foundation, or the Artistic License.
See http://dev.perl.org/licenses/ for more information.