summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--lib/Bread/Board.pm94
-rw-r--r--t/020_sugar.t48
2 files changed, 142 insertions, 0 deletions
diff --git a/lib/Bread/Board.pm b/lib/Bread/Board.pm
index 7a56200..39b5a4b 100644
--- a/lib/Bread/Board.pm
+++ b/lib/Bread/Board.pm
@@ -5,6 +5,8 @@ module Bread::Board;
class Container {...}
class Dependency {...}
+role Lifecycle { }
+
role Traversable {
has Traversable $.parent is rw;
@@ -51,6 +53,7 @@ role Traversable {
role Service does Traversable {
has Str $.name;
+ has $.lifecycle;
# PERL6: doing anything at all with the type object for a role with
# required methods is broken
@@ -175,8 +178,13 @@ class ConstructorInjection does Service does HasParameters does HasDependencies
%params.<dependencies> = $deps;
}
my $self = callwith(|%params);
+
# XXX see above
$self._set_dependency_parents;
+
+ $self does $self.lifecycle
+ if $self.lifecycle ~~ Lifecycle;
+
return $self;
}
@@ -229,8 +237,13 @@ class BlockInjection does Service does HasParameters does HasDependencies {
%params.<dependencies> = $deps;
}
my $self = callwith(|%params);
+
# XXX see above
$self._set_dependency_parents;
+
+ $self does $self.lifecycle
+ if $self.lifecycle ~~ Lifecycle;
+
return $self;
}
@@ -307,6 +320,11 @@ class Container does Traversable {
return $.sub_containers.{$name};
}
+ method add_service (Service $s) {
+ $.services.{$s.name} = $s;
+ $s.parent = self;
+ }
+
method has_services {
return $.services > 0;
}
@@ -322,6 +340,82 @@ class Container does Traversable {
// self.get_service($name)
// die "Couldn't find service or container for $name in $.name";
}
+
+ method resolve (Str :$service) {
+ return self.fetch($service).get;
+ }
+}
+
+role Singleton does Lifecycle is export {
+ has $!instance;
+ has Bool $!has_instance;
+
+ method get {
+ if !$!has_instance {
+ $!instance = callsame;
+ $!has_instance = True;
+ }
+ return $!instance;
+ }
+}
+
+our $CC;
+
+proto container is export {*}
+multi container (Container $c, Callable $body) {
+ $CC.add_sub_container($c)
+ if $CC;
+ temp $CC = $c;
+ $body.();
+ $c;
+}
+multi container (Str $name, Callable $body) {
+ container(Container.new(name => $name), $body);
+}
+multi container (Callable $body) {
+ container(Container.new, $body);
+}
+
+sub depends_on (Str $path) is export {
+ Dependency.new(service_path => $path);
+}
+
+proto service is export {*}
+multi service (*%params) {
+ my $service;
+
+ if (%params.<value>:exists) {
+ $service = Literal.new(|%params);
+ }
+ elsif (%params.<block>:exists) {
+ $service = BlockInjection.new(|%params);
+ }
+ elsif (%params.<class>:exists) {
+ $service = ConstructorInjection.new(|%params);
+ }
+ else {
+ die "Couldn't create a service from {%params}";
+ }
+
+ $CC.add_service($service)
+ if $CC;
+
+ return $service;
+}
+multi service (Str $name, *%params) {
+ service(name => $name, |%params);
+}
+multi service (Any $value) {
+ service(value => $value);
+}
+multi service (Str $name, Any $value) {
+ service(name => $name, value => $value);
+}
+multi service (Str $name, Parcel $params) {
+ service(name => $name, |$params.hash);
+}
+multi service (Parcel $params) {
+ service(|$params.hash);
}
# vim:ft=perl6:foldmethod=manual
diff --git a/t/020_sugar.t b/t/020_sugar.t
new file mode 100644
index 0000000..04074fd
--- /dev/null
+++ b/t/020_sugar.t
@@ -0,0 +1,48 @@
+use v6;
+use Test;
+
+use Bread::Board;
+
+class FileLogger {
+ has $.log_file;
+}
+
+class MyApplication {
+ has FileLogger $.logger;
+}
+
+my $c = container 'MyApp', {
+ service 'log_file', 'logfile.log';
+
+ service 'logger', (
+ class => FileLogger,
+ lifecycle => Singleton,
+ dependencies => {
+ log_file => depends_on('log_file'),
+ },
+ );
+
+ service 'application', (
+ class => MyApplication,
+ dependencies => {
+ logger => depends_on('logger'),
+ },
+ );
+};
+
+#my $logger = $c.resolve(service => 'logger');
+my $logger = $c.fetch('logger').get;
+isa_ok($logger, FileLogger);
+is($logger.log_file, 'logfile.log');
+
+is($c.fetch('logger/log_file').service, $c.fetch('log_file'));
+is($c.fetch('logger/log_file').get, 'logfile.log');
+
+my $app = $c.resolve(service => 'application');
+isa_ok($app, MyApplication);
+isa_ok($app.logger, FileLogger);
+is($app.logger, $logger);
+
+done;
+
+# vim:ft=perl6:foldmethod=manual