dr-kd / PPIx-Refactor

Perl refactoring tools based on PPI
2 stars 0 forks source link

package PPIx::Refactor;

use Moo; use Path::Tiny; BEGIN { use File::Path; use constant { CACHE => '/tmp/ppix-refactor_cache', }; our $cache = CACHE; File::Path::mkpath($cache) unless -e CACHE; }

use PPI; use PPI::Cache path => CACHE; use PPI::Find;

=head1 NAME

PPIx::Refactor - Hooks for refactoring perl via L

=head1 SYNOPSIS

use PPIx::Refactor;
my $p = PPIx::Refactor->new(file => '/path/to/perl/code/file.pl',
                            ppi_find => sub {
                                my ($elem, $doc) = @_;
                                return 1 if $elem->class eq 'PPI::Statement::Sub',
                                return 0;
                            }
                            [ writer => \&found ]);
my $finds = $p->finds; # for examining them interactively
$p->rewrite; # rewrites the file in place.  You are using version control yes?

=head1 SUMMARY

This is a really simple module to make rewriting perl code via L debugger friendly and easy. See the test in L<t/refactor.t|https://github.com/singingfish/PPIx-Refactor/blob/master/t/refactor.t> of this distribution for a working example. Pretty much all the real work happens in the coderef you set up in C<< $p->ppi_find >> and C<< $p->writer >>.

For an example of a simple script for checking statements in code for being syntactically identical (i.e. a crude copypasta detector) see C< similar_statements.pl > in the examples directory of the distribution.

NOTE L is used to store a cached representation of the source parse in C</tmp/pppix-refactor_cache>

=head2 RATIONALE

Rewriting code via ppi is a fiddly pain. L provides a minimal interface so you can concentrate on the fiddlyness and minimise the pain.

=head2 TODO

Would be nice to specify a rewriter via roles, and it would be nice to have $self in C<< $p->ppi_find >>. On the other hand rewrite/refactoring code like this can either be simple throwaways, or really really complicated. This code is so far optimised for the throwaway case.

=cut

=head2 ATTRIBUTES

=head3 file

required string that coerces into a Path::Tiny

=cut

has file => ( is => 'ro', coerce => sub { path($_[0]); } );

=head3 doc

lazily built PPI::Document

=cut

has doc => ( is => 'ro', lazy => 1, default => sub { my ($self) = @_; return PPI::Document::File->new($self->file->stringify); }, );

=head3 element

If you're using prior finds (e.g. subroutines you're trying to analyse) you'll want to pass an element into new rather than a doc. Element defaults to the document you passed in.

=cut

has element => ( is => 'ro', lazy => 1, builder => sub { $_[0]->doc; }, );

=head3 ppi_find

required coderef with which to find the elements of interest

=cut

has ppi_find => ( is => 'ro',

isa CodeRef

required => 1,

);

=head3 writer

optional coderef with which to rewrite the code.

=cut

has writer => ( is => 'ro', default => sub {}, );

=head3 finds

lazy built arrayref of all the elements of interest found

=cut

has finds => ( is => 'ro', lazy => 1, default => sub { my ($self) = @_; my $find = PPI::Find->new($self->ppi_find); my @results = $find->in($self->element); return \@results; } );

=head1 METHODS

=head2 $self->rewrite

Worker sub that rewrites the code. Operates on what it finds in C<<$self->finds>>

=cut

sub rewrite { my ($self) = @_; $self->writer->($self->finds); $self->element->save($self->file); }

=head2 $self->dump($elem, $whitespace);

For debugging. Prints a dump of the passed in element. If C<$whitespace> is true it will include whitespace in the dump. Defaults to false.

=cut

sub dump { my ($self, $doc, $whitespace) = @_; $whitespace ||=0; my $dump = PPI::Dumper->new($doc, whitespace => $whitespace); $dump->print; }

=head1 AUTHOR

Kieren Diment, C<< >>

=head1 BUGS

Please report any bugs or feature requests via github: Lhttps://github.com/singingfish/PPIx-Refactor/issues.

=head1 SUPPORT

Jump on to #web-simple on irc.perl.org

=head1 ACKNOWLEDGEMENTS

=head1 LICENSE AND COPYRIGHT

Copyright 2015 Kieren Diment.

This program is free software; you can redistribute it and/or modify it under the same terms as perl itself. =cut

1;