wheeeeeeeeee
[scpubgit/DKit.git] / lib / DX / RuleSet.pm
CommitLineData
b40e416a 1package DX::RuleSet;
2
3use Moo;
4use DX::Op::CallRule;
5use DX::Op::MemberOf;
6use DX::Op::MemberLookup;
7use DX::Op::ApplyConstraint;
8use DX::Op::Return;
9use List::Util qw(reduce);
10
11has rules => (is => 'ro', default => sub { {} });
12
13sub add_rule {
14 my ($self, $name, $vars, @body) = @_;
15 my $full_name = join('/', $name, scalar @$vars);
16 push @{$self->rules->{$full_name}}, $self->_make_rule($vars, @body);
17 return $self;
18}
19
20sub _make_rule {
21 my ($self, $vars, @body) = @_;
22 my $head = reduce { $b->but(next => $a) }
23 DX::Op::Return->new,
24 reverse map $self->expand(@$_), @body;
25 [ $vars, $head ];
26}
27
28sub expand {
29 my ($self, $type, @rest) = @_;
30 $self->${\"_expand_${type}"}(@rest);
31}
32
33sub _expand_call {
34 my ($self, $name, @args) = @_;
35 DX::Op::CallRule->new(rule_name => $name, rule_args => \@args);
36}
37
38sub _expand_member_of {
39 my ($self, $member_var, $coll_name) = @_;
40 DX::Op::MemberOf->new(
41 member_var => $member_var,
42 coll_name => $coll_name,
43 );
44}
45
46sub _expand_member_lookup {
47 my ($self, $member_var, $coll_name, $key_name, $key_var) = @_;
48 DX::Op::MemberLookup->new(
49 member_var => $member_var,
50 coll_name => $coll_name,
51 key_name => $key_name,
52 key_var => $key_var
53 );
54}
55
56sub _expand_constrain {
57 my ($self, $vars, $constraint) = @_;
58 DX::Op::ApplyConstraint->new(
59 vars => $vars,
60 constraint => $constraint
61 );
62}
63
641;