make room for $VERSION after package declaration (newer [PkgVersion] requires it)
[p5sagit/Devel-REPL.git] / lib / Devel / REPL / Plugin / Turtles.pm
CommitLineData
1716b200 1use strict;
2use warnings;
e958cbc6 3package Devel::REPL::Plugin::Turtles;
e22aa835 4
6f4f9516 5use Devel::REPL::Plugin;
e22aa835 6use Scalar::Util qw(reftype);
aa8b7647 7use namespace::autoclean;
e958cbc6 8
e22aa835 9has default_command_prefix => (
10 isa => "RegexpRef",
11 is => "rw",
12 default => sub { qr/\#/ },
13);
14
15has turtles_matchers => (
78bc5721 16 traits => ['Array'],
e22aa835 17 isa => "ArrayRef[RegexpRef|CodeRef]",
18 is => "rw",
4ea2c254 19 lazy => 1,
e22aa835 20 default => sub { my $prefix = shift->default_command_prefix; [qr/^ $prefix (\w+) \s* (.*) /x] },
8bf72450 21 handles => {
22 add_turtles_matcher => 'unshift',
e22aa835 23 },
24);
25
26around 'formatted_eval' => sub {
0cbfa921 27 my $next = shift;
e22aa835 28 my ($self, $line, @args) = @_;
0cbfa921 29
e22aa835 30 if ( my ( $command, @rest ) = $self->match_turtles($line) ) {
0cbfa921 31 my $method = "command_$command";
e22aa835 32 my $expr_method = "expr_$method";
0cbfa921 33
e22aa835 34 if ( my $expr_code = $self->can($expr_method) ) {
35 if ( my $read_more = $self->can("continue_reading_if_necessary") ) {
36 push @rest, $self->$read_more(pop @rest);
37 }
38 $self->$expr_code($next, @rest);
39 } elsif ( my $cmd_code = $self->can($method) ) {
40 return $self->$cmd_code($next, @rest);
0cbfa921 41 } else {
e22aa835 42 unless ( $line =~ /^\s*#/ ) { # special case for comments
43 return $self->format($self->error_return("REPL Error", "Command '$command' does not exist"));
44 }
e958cbc6 45 }
e22aa835 46 } else {
47 return $self->$next($line, @args);
0cbfa921 48 }
e958cbc6 49};
50
e22aa835 51sub match_turtles {
52 my ( $self, $line ) = @_;
53
54 foreach my $thingy ( @{ $self->turtles_matchers } ) {
55 if ( reftype $thingy eq 'CODE' ) {
56 if ( my @res = $self->$thingy($line) ) {
57 return @res;
58 }
59 } else {
60 if ( my @res = ( $line =~ $thingy ) ) {
61 return @res;
62 }
63 }
64 }
65
66 return;
67}
68
48ddfeae 691;
cfd1094b 70
71__END__
72
73=head1 NAME
74
75Devel::REPL::Plugin::Turtles - Generic command creation using a read hook
76
79e70b9c 77=head1 DESCRIPTION
78
79By default, this plugin allows calling commands using a read hook
80to detect a default_command_prefix followed by the command name,
81say MYCMD as an example. The actual routine to call for the
82command is constructed by looking for subs named 'command_MYCMD'
83or 'expr_MYCMD' and executing them.
84
85=head2 NOTE
86
87The C<default_command_prefix> is C<qr/\#/> so care must be taken
88if other uses for that character are needed (e.g., '#' for the
89shell escape character in the PDL shell.
90
cfd1094b 91=cut
92