better debugging output
[scpubgit/DKit.git] / bin / dx-shell
CommitLineData
eed368c9 1#!/usr/bin/env perl
2
3use strictures 1;
4use curry;
5use DX::Solver;
6use DX::Lib::FS;
7use Term::ReadLine;
a26f6378 8use Devel::Dwarn;
ffacb8aa 9use Sub::Quote;
db732a14 10use YAML ();
11use Safe::Isa;
eed368c9 12
13my $solver = DX::Solver->new(observation_policy => sub { 1 });
14
15DX::Lib::FS->new->load_into($solver);
16
17#::Dwarn(
18# [ $solver->query([ 'D' ], [ directory_at => 'D' => \'t' ])->results ]
19# ->[0]->value_for('D')
20#);
21
22use Tcl;
23
24my $tcl = Tcl->new;
25
26my $split = $tcl->curry::weak::SplitList;
27
7ca660cb 28my ($r, $res, @last_q);
eed368c9 29
7ca660cb 30my $last_mode;
31
32sub show {
33 $r = ($res->isa('DX::Result') ? $res : $res->next);
34 unless ($r) { warn "false\n"; return; }
db732a14 35 if (my @act = $r->actions) {
36 warn YAML::Dump([ map $_->as_structure, @act ]);
37 }
38 my $values = $r->all_values;
39 unless (keys %$values) {
40 warn "---\ntrue\n"; return;
41 }
42 warn YAML::Dump({
43 map +($_ => ($values->{$_}->$_does('DX::Role::Set')
44 ? [ $values->{$_}->all ]
45 : $values->{$_} )),
46 keys %$values
47 });
7ca660cb 48}
49
50sub do_query {
51 $res = $solver->$last_mode(@last_q);
52 show();
eed368c9 53}
54
a26f6378 55sub expand_def {
56 my ($vars, $body) = @_;
eed368c9 57 my @varnames = $split->($vars);
97c0c46e 58 return (\@varnames, expand_body($body));
59}
60
61sub expand_body {
7ca660cb 62 my (@body) = @_;
eed368c9 63 local our @Body_Parts;
7ca660cb 64 $tcl->Eval(@body);
97c0c46e 65 return @Body_Parts;
a26f6378 66}
67
7ca660cb 68sub q_command {
69 my ($this_mode, undef, undef, undef, $body) = @_;
70 $last_mode = $this_mode;
71 if ($body) {
72 @last_q = expand_body($body);
73 }
eed368c9 74 do_query();
75 return;
7ca660cb 76}
77
78$tcl->CreateCommand('?' => sub { q_command(query => @_) });
79$tcl->CreateCommand('?!' => sub { q_command(solve => @_) });
80$tcl->CreateCommand('!?' => sub { q_command(solve => @_) });
81$tcl->CreateCommand('!' => sub { q_command(ensure => @_) });
eed368c9 82
83$tcl->CreateCommand(run => sub {
7ca660cb 84 foreach my $ind ($r->independent_actions) {
eed368c9 85 $solver->run_action($ind);
86 }
87 do_query();
88 return;
89});
90
ffacb8aa 91$tcl->CreateCommand(dump => sub {
92 my (undef, undef, undef, $to_dump) = @_;
93 my $filter = quote_sub($to_dump);
7ca660cb 94 Dwarn($filter->($r));
ffacb8aa 95});
96
eed368c9 97sub mangle_args {
98 my @args = @_;
99 map { $_ =~ /^'(.*)'$/ ? \do { my $x = $1 } : $_ } @args;
100}
101
102my $rule_sub = sub {
103 my (undef, undef, $name, @args) = @_;
104 push our @Body_Parts, [ $name => mangle_args(@args) ];
105 return;
106};
107
a26f6378 108$tcl->CreateCommand(rule => sub {
109 my (undef, undef, undef, $rule, $vars, $body) = @_;
110 $solver->add_rule($rule => expand_def($vars, $body));
111 $tcl->CreateCommand($rule => $rule_sub);
112 return;
113});
114
eed368c9 115foreach my $rule (keys %{$solver->rule_set->rules}) {
116 $rule =~ s/\/\d+$//;
117 $tcl->CreateCommand($rule => $rule_sub);
118}
a26f6378 119
97c0c46e 120$tcl->CreateCommand(foreach => sub {
121 my (undef, undef, undef, $var, $body, $each_body) = @_;
122 push our @Body_Parts, [
123 foreach => $var => map [ expand_body($_) ], $body, $each_body
124 ];
125 return;
126});
7ca660cb 127
128$tcl->CreateCommand(findall => sub {
129 my (undef, undef, undef, $coll_var, $name_var, $body) = @_;
130 push our @Body_Parts, [
131 findall => $coll_var => $name_var => expand_body($body)
132 ];
133 return;
134});
db732a14 135
136$tcl->CreateCommand(n => \&show);
eed368c9 137
138#$tcl->Eval(q{query D {directory_at D 't'; mode D '0755'; }});
139
a26f6378 140if ($ARGV[0]) {
141 $tcl->EvalFile($ARGV[0]);
142}
143
eed368c9 144my $rl = Term::ReadLine->new;
145
146my $cmd;
147
2e60bb64 148while (defined(my $line = $rl->readline(length($cmd)?'> ':'$ '))) {
149 $cmd .= "${line}\n";
eed368c9 150 if ($tcl->call(info => complete => $cmd)) {
a57a26e3 151 unless (eval { $tcl->Eval($cmd); 1 }) {
152 warn $@;
153 }
eed368c9 154 $cmd = '';
155 }
156}