acion infrastructure
[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;
71217e42 9use DX::Op::Backtrack;
5ef4d923 10use DX::Op::Observe;
71217e42 11use DX::Op::Not;
12use DX::Op::ProposeAction;
13use DX::Op::Materialize;
896fd92e 14use DX::Op::Prop;
859049a3 15use DX::Op::Exists;
b40e416a 16use List::Util qw(reduce);
17
18has rules => (is => 'ro', default => sub { {} });
19
20sub add_rule {
21 my ($self, $name, $vars, @body) = @_;
22 my $full_name = join('/', $name, scalar @$vars);
23 push @{$self->rules->{$full_name}}, $self->_make_rule($vars, @body);
24 return $self;
25}
26
27sub _make_rule {
28 my ($self, $vars, @body) = @_;
deec7cc4 29 my $head = $self->expand_and_link(DX::Op::Return->new, @body);
b40e416a 30 [ $vars, $head ];
31}
32
deec7cc4 33sub expand_and_link {
71217e42 34 my ($self, $last, @body) = @_;
35 return reduce { $b->but(next => $a) }
36 $last,
deec7cc4 37 reverse map $self->expand($_), @body;
71217e42 38}
39
b40e416a 40sub expand {
deec7cc4 41 my ($self, $thing) = @_;
42 if (ref($thing) eq 'ARRAY') {
43 my ($type, @rest) = @$thing;
44 if ($self->can(my $expand_meth = "_expand_op_${type}")) {
45 return $self->$expand_meth(@rest);
46 }
47 return $self->_expand_call(@$thing);
6d533c9d 48 }
deec7cc4 49 return $thing;
b40e416a 50}
51
52sub _expand_call {
53 my ($self, $name, @args) = @_;
54 DX::Op::CallRule->new(rule_name => $name, rule_args => \@args);
55}
56
71217e42 57sub _expand_op_cut { return DX::Op::Cut->new }
734376d9 58
71217e42 59sub _expand_op_fail { return DX::Op::Backtrack->new }
60
61sub _expand_op_not {
62 my ($self, @contents) = @_;
63 my $cut = DX::Op::Cut->new(next => DX::Op::Backtrack->new);
64 DX::Op::Not->new(
deec7cc4 65 body => $self->expand_and_link($cut, @contents)
71217e42 66 );
67}
68
69sub _expand_op_member_of {
385fa954 70 my ($self, $member_var, $coll_var) = @_;
b40e416a 71 DX::Op::MemberOf->new(
72 member_var => $member_var,
385fa954 73 coll_var => $coll_var,
b40e416a 74 );
75}
76
71217e42 77sub _expand_op_constrain {
b40e416a 78 my ($self, $vars, $constraint) = @_;
79 DX::Op::ApplyConstraint->new(
80 vars => $vars,
81 constraint => $constraint
82 );
83}
84
71217e42 85sub _expand_op_observe {
5ef4d923 86 my ($self, $vars, $builder) = @_;
87 DX::Op::Observe->new(
88 vars => $vars,
89 builder => $builder,
90 );
91}
92
71217e42 93sub _expand_op_act {
94 my ($self, $vars, $builder) = @_;
95 DX::Op::ProposeAction->new(
96 vars => $vars,
97 builder => $builder,
98 );
99}
100
101sub _expand_op_materialize {
102 my ($self, $var_name) = @_;
103 DX::Op::Materialize->new(var_name => $var_name);
104}
105
896fd92e 106sub _expand_op_prop {
107 my ($self, @args) = @_;
108 my %new; @new{qw(of name value)} = @args;
109 DX::Op::Prop->new(%new);
110}
111
859049a3 112sub _expand_op_exists {
113 my ($self, $vars, @body) = @_;
114 DX::Op::Exists->new(
115 vars => $vars,
deec7cc4 116 body => $self->expand_and_link(DX::Op::Return->new, @body)
859049a3 117 );
118}
119
b40e416a 1201;