package DX::RuleSet;
use Moo;
+use DX::Op::SetupScope;
use DX::Op::CallRule;
use DX::Op::MemberOf;
use DX::Op::ApplyConstraint;
use DX::Op::Observe;
use DX::Op::Not;
use DX::Op::ProposeAction;
+use DX::Op::ModifyAction;
use DX::Op::Materialize;
+use DX::Op::Prop;
+use DX::Op::Predicate;
+use DX::Op::HasAction;
+use DX::Op::FindAll;
+use DX::Op::ForEach;
use List::Util qw(reduce);
has rules => (is => 'ro', default => sub { {} });
+sub add_predicate {
+ my ($self, $name, $vars, @cases) = @_;
+ my $full_name = join('/', $name, scalar @$vars);
+ push @{$self->rules->{$full_name}}, DX::Op::Predicate->new(
+ arg_names => $vars, arg_cases => \@cases
+ );
+}
+
sub add_rule {
my ($self, $name, $vars, @body) = @_;
my $full_name = join('/', $name, scalar @$vars);
sub _make_rule {
my ($self, $vars, @body) = @_;
- my $head = $self->_expand_and_link(DX::Op::Return->new, @body);
- [ $vars, $head ];
+ my $head = $self->expand_and_link(DX::Op::Return->new, @body);
+ DX::Op::SetupScope->new(arg_names => $vars, next => $head);
}
-sub _expand_and_link {
+sub expand_and_link {
my ($self, $last, @body) = @_;
return reduce { $b->but(next => $a) }
$last,
- reverse map $self->expand(@$_), @body;
+ reverse map $self->expand($_), @body;
}
sub expand {
- my ($self, $type, @rest) = @_;
- if ($self->can(my $expand_meth = "_expand_op_${type}")) {
- return $self->$expand_meth(@rest);
+ my ($self, $thing) = @_;
+ if (ref($thing) eq 'ARRAY') {
+ my ($type, @rest) = @$thing;
+ if ($self->can(my $expand_meth = "_expand_op_${type}")) {
+ return $self->$expand_meth(@rest);
+ }
+ return $self->_expand_call(@$thing);
}
- return $self->_expand_call($type, @rest);
+ return $thing;
}
sub _expand_call {
sub _expand_op_fail { return DX::Op::Backtrack->new }
sub _expand_op_not {
- my ($self, @contents) = @_;
+ my ($self, $contents) = @_;
my $cut = DX::Op::Cut->new(next => DX::Op::Backtrack->new);
DX::Op::Not->new(
- body => $self->_expand_and_link($cut, @contents)
+ body => $self->expand_and_link($cut, @$contents)
+ );
+}
+
+sub _expand_op_findall {
+ my ($self, $coll_name, $var_name, $contents) = @_;
+ DX::Op::FindAll->new(
+ coll_name => $coll_name,
+ var_name => $var_name,
+ body => $self->expand_and_link(DX::Op::Return->new, @$contents),
+ );
+}
+
+sub _expand_op_foreach {
+ my ($self, $var_name, $body, $each_body) = @_;
+ DX::Op::ForEach->new(
+ var_name => $var_name,
+ body => $self->expand_and_link(DX::Op::Return->new, @$body),
+ each_body => $self->expand_and_link(DX::Op::Return->new, @$each_body),
);
}
);
}
+sub _expand_op_react {
+ my ($self, $vars, $builder) = @_;
+ DX::Op::ModifyAction->new(
+ vars => $vars,
+ builder => $builder,
+ );
+}
+
sub _expand_op_materialize {
- my ($self, $var_name) = @_;
- DX::Op::Materialize->new(var_name => $var_name);
+ my ($self) = @_;
+ DX::Op::Materialize->new;
+}
+
+sub _expand_op_prop {
+ my ($self, @args) = @_;
+ my %new; @new{qw(of name value)} = @args;
+ DX::Op::Prop->new(%new);
+}
+
+sub _expand_op_has_action {
+ my ($self, @args) = @_;
+ DX::Op::HasAction->new(arg_spec => \@args);
}
1;