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