package Mojolicious::Command; use Mojo::Base -base; use Carp 'croak'; use Mojo::File 'path'; use Mojo::Loader 'data_section'; use Mojo::Server; use Mojo::Template; has app => sub { Mojo::Server->new->build_app('Mojo::HelloWorld') }; has description => 'No description'; has 'quiet'; has usage => "Usage: APPLICATION\n"; sub chmod_file { my ($self, $path, $mod) = @_; chmod $mod, $path or croak qq{Can't chmod file "$path": $!}; return $self->_loud(" [chmod] $path " . sprintf('%lo', $mod)); } sub chmod_rel_file { $_[0]->chmod_file($_[0]->rel_file($_[1]), $_[2]) } sub create_dir { my ($self, $path) = @_; return $self->_loud(" [exist] $path") if -d $path; path($path)->make_path; return $self->_loud(" [mkdir] $path"); } sub create_rel_dir { $_[0]->create_dir($_[0]->rel_file($_[1])) } sub extract_usage { Mojo::Util::extract_usage((caller)[1]) } sub help { print shift->usage } sub rel_file { path->child(split('/', pop)) } sub render_data { my ($self, $name) = (shift, shift); Mojo::Template->new->name("template $name from DATA section") ->render(data_section(ref $self, $name), @_); } sub render_to_file { my ($self, $data, $path) = (shift, shift, shift); return $self->write_file($path, $self->render_data($data, @_)); } sub render_to_rel_file { my $self = shift; $self->render_to_file(shift, $self->rel_file(shift), @_); } sub run { croak 'Method "run" not implemented by subclass' } sub write_file { my ($self, $path, $data) = @_; return $self->_loud(" [exist] $path") if -f $path; $self->create_dir(path($path)->dirname); path($path)->spurt($data); return $self->_loud(" [write] $path"); } sub write_rel_file { $_[0]->write_file($_[0]->rel_file($_[1]), $_[2]) } sub _loud { my ($self, $msg) = @_; say $msg unless $self->quiet; return $self; } 1; =encoding utf8 =head1 NAME Mojolicious::Command - Command base class =head1 SYNOPSIS # Lowercase command name package Mojolicious::Command::mycommand; use Mojo::Base 'Mojolicious::Command'; # Short description has description => 'My first Mojo command'; # Usage message from SYNOPSIS has usage => sub { shift->extract_usage }; sub run { my ($self, @args) = @_; # Magic here! :) } 1; =head1 SYNOPSIS Usage: APPLICATION mycommand [OPTIONS] Options: -s, --something Does something =cut =head1 DESCRIPTION L is an abstract base class for L commands. See L for a list of commands that are available by default. =head1 ATTRIBUTES L implements the following attributes. =head2 app my $app = $command->app; $command = $command->app(Mojolicious->new); Application for command, defaults to a L object. # Introspect say "Template path: $_" for @{$command->app->renderer->paths}; =head2 description my $description = $command->description; $command = $command->description('Foo'); Short description of command, used for the command list. =head2 quiet my $bool = $command->quiet; $command = $command->quiet($bool); Limited command output. =head2 usage my $usage = $command->usage; $command = $command->usage('Foo'); Usage information for command, used for the help screen. =head1 METHODS L inherits all methods from L and implements the following new ones. =head2 chmod_file $command = $command->chmod_file('/home/sri/foo.txt', 0644); Change mode of a file. =head2 chmod_rel_file $command = $command->chmod_rel_file('foo/foo.txt', 0644); Portably change mode of a file relative to the current working directory. =head2 create_dir $command = $command->create_dir('/home/sri/foo/bar'); Create a directory. =head2 create_rel_dir $command = $command->create_rel_dir('foo/bar/baz'); Portably create a directory relative to the current working directory. =head2 extract_usage my $usage = $command->extract_usage; Extract usage message from the SYNOPSIS section of the file this method was called from with L. =head2 help $command->help; Print usage information for command. =head2 rel_file my $path = $command->rel_file('foo/bar.txt'); Return a L object relative to the current working directory. =head2 render_data my $data = $command->render_data('foo_bar'); my $data = $command->render_data('foo_bar', @args); Render a template from the C section of the command class with L and L. =head2 render_to_file $command = $command->render_to_file('foo_bar', '/home/sri/foo.txt'); $command = $command->render_to_file('foo_bar', '/home/sri/foo.txt', @args); Render a template from the C section of the command class with L to a file and create directory if necessary. =head2 render_to_rel_file $command = $command->render_to_rel_file('foo_bar', 'foo/bar.txt'); $command = $command->render_to_rel_file('foo_bar', 'foo/bar.txt', @args); Portably render a template from the C section of the command class with L to a file relative to the current working directory and create directory if necessary. =head2 run $command->run; $command->run(@ARGV); Run command. Meant to be overloaded in a subclass. =head2 write_file $command = $command->write_file('/home/sri/foo.txt', 'Hello World!'); Write text to a file and create directory if necessary. =head2 write_rel_file $command = $command->write_rel_file('foo/bar.txt', 'Hello World!'); Portably write text to a file relative to the current working directory and create directory if necessary. =head1 SEE ALSO L, L, L. =cut