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