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