cut support
[scpubgit/DKit.git] / lib / DX / RuleSet.pm
CommitLineData
b40e416a 1package DX::RuleSet;
2
3use Moo;
4use DX::Op::CallRule;
5use DX::Op::MemberOf;
b40e416a 6use DX::Op::ApplyConstraint;
7use DX::Op::Return;
734376d9 8use DX::Op::Cut;
b40e416a 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) = @_;
6d533c9d 30 if ($self->can(my $expand_meth = "_expand_${type}")) {
31 return $self->$expand_meth(@rest);
32 }
33 return $self->_expand_call($type, @rest);
b40e416a 34}
35
36sub _expand_call {
37 my ($self, $name, @args) = @_;
38 DX::Op::CallRule->new(rule_name => $name, rule_args => \@args);
39}
40
734376d9 41sub _expand_cut { return DX::Op::Cut->new }
42
b40e416a 43sub _expand_member_of {
385fa954 44 my ($self, $member_var, $coll_var) = @_;
b40e416a 45 DX::Op::MemberOf->new(
46 member_var => $member_var,
385fa954 47 coll_var => $coll_var,
b40e416a 48 );
49}
50
b40e416a 51sub _expand_constrain {
52 my ($self, $vars, $constraint) = @_;
53 DX::Op::ApplyConstraint->new(
54 vars => $vars,
55 constraint => $constraint
56 );
57}
58
591;