KeyMangler example app
[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;
2c467394 14use DX::Op::ModifyAction;
71217e42 15use DX::Op::Materialize;
896fd92e 16use DX::Op::Prop;
37e9670d 17use DX::Op::Predicate;
577a2146 18use DX::Op::HasAction;
138613a8 19use DX::Op::FindAll;
f5a02c91 20use DX::Op::ForEach;
b40e416a 21use List::Util qw(reduce);
22
23has rules => (is => 'ro', default => sub { {} });
24
37e9670d 25sub add_predicate {
26 my ($self, $name, $vars, @cases) = @_;
27 my $full_name = join('/', $name, scalar @$vars);
28 push @{$self->rules->{$full_name}}, DX::Op::Predicate->new(
29 arg_names => $vars, arg_cases => \@cases
30 );
31}
32
b40e416a 33sub add_rule {
34 my ($self, $name, $vars, @body) = @_;
35 my $full_name = join('/', $name, scalar @$vars);
36 push @{$self->rules->{$full_name}}, $self->_make_rule($vars, @body);
37 return $self;
38}
39
40sub _make_rule {
41 my ($self, $vars, @body) = @_;
deec7cc4 42 my $head = $self->expand_and_link(DX::Op::Return->new, @body);
02ae4168 43 DX::Op::SetupScope->new(arg_names => $vars, next => $head);
b40e416a 44}
45
deec7cc4 46sub expand_and_link {
71217e42 47 my ($self, $last, @body) = @_;
48 return reduce { $b->but(next => $a) }
49 $last,
deec7cc4 50 reverse map $self->expand($_), @body;
71217e42 51}
52
b40e416a 53sub expand {
deec7cc4 54 my ($self, $thing) = @_;
55 if (ref($thing) eq 'ARRAY') {
56 my ($type, @rest) = @$thing;
57 if ($self->can(my $expand_meth = "_expand_op_${type}")) {
58 return $self->$expand_meth(@rest);
59 }
60 return $self->_expand_call(@$thing);
6d533c9d 61 }
deec7cc4 62 return $thing;
b40e416a 63}
64
65sub _expand_call {
66 my ($self, $name, @args) = @_;
67 DX::Op::CallRule->new(rule_name => $name, rule_args => \@args);
68}
69
71217e42 70sub _expand_op_cut { return DX::Op::Cut->new }
734376d9 71
71217e42 72sub _expand_op_fail { return DX::Op::Backtrack->new }
73
74sub _expand_op_not {
91d1f239 75 my ($self, $contents) = @_;
71217e42 76 my $cut = DX::Op::Cut->new(next => DX::Op::Backtrack->new);
77 DX::Op::Not->new(
91d1f239 78 body => $self->expand_and_link($cut, @$contents)
71217e42 79 );
80}
81
138613a8 82sub _expand_op_findall {
91d1f239 83 my ($self, $coll_name, $var_name, $contents) = @_;
f5a02c91 84 DX::Op::FindAll->new(
138613a8 85 coll_name => $coll_name,
86 var_name => $var_name,
91d1f239 87 body => $self->expand_and_link(DX::Op::Return->new, @$contents),
138613a8 88 );
89}
90
f5a02c91 91sub _expand_op_foreach {
92 my ($self, $var_name, $body, $each_body) = @_;
93 DX::Op::ForEach->new(
94 var_name => $var_name,
95 body => $self->expand_and_link(DX::Op::Return->new, @$body),
96 each_body => $self->expand_and_link(DX::Op::Return->new, @$each_body),
97 );
98}
99
71217e42 100sub _expand_op_member_of {
385fa954 101 my ($self, $member_var, $coll_var) = @_;
b40e416a 102 DX::Op::MemberOf->new(
103 member_var => $member_var,
385fa954 104 coll_var => $coll_var,
b40e416a 105 );
106}
107
71217e42 108sub _expand_op_constrain {
b40e416a 109 my ($self, $vars, $constraint) = @_;
110 DX::Op::ApplyConstraint->new(
111 vars => $vars,
112 constraint => $constraint
113 );
114}
115
71217e42 116sub _expand_op_observe {
5ef4d923 117 my ($self, $vars, $builder) = @_;
118 DX::Op::Observe->new(
119 vars => $vars,
120 builder => $builder,
121 );
122}
123
71217e42 124sub _expand_op_act {
125 my ($self, $vars, $builder) = @_;
126 DX::Op::ProposeAction->new(
127 vars => $vars,
128 builder => $builder,
129 );
130}
131
2c467394 132sub _expand_op_react {
133 my ($self, $vars, $builder) = @_;
134 DX::Op::ModifyAction->new(
135 vars => $vars,
136 builder => $builder,
137 );
138}
139
71217e42 140sub _expand_op_materialize {
7d384eca 141 my ($self) = @_;
142 DX::Op::Materialize->new;
71217e42 143}
144
896fd92e 145sub _expand_op_prop {
146 my ($self, @args) = @_;
147 my %new; @new{qw(of name value)} = @args;
148 DX::Op::Prop->new(%new);
149}
150
577a2146 151sub _expand_op_has_action {
152 my ($self, @args) = @_;
153 DX::Op::HasAction->new(arg_spec => \@args);
154}
155
b40e416a 1561;