Group
Extension

Mojolicious/t/mojolicious/commands.t

use Mojo::Base -strict;

BEGIN {
  $ENV{PLACK_ENV}    = undef;
  $ENV{MOJO_REACTOR} = 'Mojo::Reactor::Poll';
}

use Test::More;

use Mojo::File qw(curfile);
use lib curfile->sibling('lib')->to_string;

use Mojo::File qw(path tempdir);

package Mojolicious::Command::my_fake_test_command;

package Mojolicious::Command::my_test_command;
use Mojo::Base 'Mojolicious::Command';

has description => 'See, it works';

package main;

# Make sure @ARGV is not changed
{
  local $ENV{MOJO_MODE};
  local @ARGV = qw(-m production -x whatever);
  require Mojolicious::Commands;
  is $ENV{MOJO_MODE}, 'production', 'right mode';
  is_deeply \@ARGV, [qw(-m production -x whatever)], 'unchanged';
}

# Environment detection
my $commands = Mojolicious::Commands->new;
{
  local $ENV{PLACK_ENV} = 'production';
  is $commands->detect, 'psgi', 'right environment';
}
{
  local $ENV{PATH_INFO} = '/test';
  is $commands->detect, 'cgi', 'right environment';
}
{
  local $ENV{GATEWAY_INTERFACE} = 'CGI/1.1';
  is $commands->detect, 'cgi', 'right environment';
}
{
  local @ENV{qw(PLACK_ENV PATH_INFO GATEWAY_INTERFACE)};
  is $commands->detect, undef, 'no environment';
}
{
  local $ENV{PLACK_ENV} = 'production';
  is ref Mojolicious::Commands->new->run, 'CODE', 'right reference';
  local $ENV{MOJO_NO_DETECT} = 1;
  isnt ref Mojolicious::Commands->new->run, 'CODE', 'not a CODE reference';
}

# Run command
is ref Mojolicious::Commands->new->run('psgi'), 'CODE', 'right reference';

# Start application
{
  local $ENV{MOJO_APP_LOADER} = 1;
  is ref Mojolicious::Commands->start_app('MojoliciousTest'), 'MojoliciousTest', 'right class';
}

# Start application with command
{
  is ref Mojolicious::Commands->start_app(MojoliciousTest => 'psgi'), 'CODE', 'right reference';
}

# Start application with application specific commands
my $app;
{
  local $ENV{MOJO_APP_LOADER} = 1;
  $app = Mojolicious::Commands->start_app('MojoliciousTest');
}
is $app->start('test_command'),   'works!',   'right result';
is $app->start('test-command'),   'works!',   'right result';
is $app->start('_test2-command'), 'works 2!', 'right result';
{
  is(Mojolicious::Commands->start_app(MojoliciousTest => 'test_command'),   'works!',   'right result');
  is(Mojolicious::Commands->start_app(MojoliciousTest => 'test-command'),   'works!',   'right result');
  is(Mojolicious::Commands->start_app(MojoliciousTest => '_test2-command'), 'works 2!', 'right result');
}

# Application specific help
my $buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  local $ENV{HARNESS_ACTIVE} = 0;
  $app->start;
}
like $buffer, qr/Usage: APPLICATION COMMAND \[OPTIONS\].*_test2-command.*cgi.*test-comm/s, 'right output';

subtest 'Commands starting with a dash are not allowed' => sub {
  local $ENV{HARNESS_ACTIVE} = 0;
  eval { $app->start('-test2-command') };
  like $@, qr/Invalid command "-test2-command"\./, 'not allowed';
};

# Do not pick up options for detected environments
{
  local $ENV{MOJO_MODE};
  local $ENV{PLACK_ENV} = 'testing';
  local @ARGV = qw(psgi -m production);
  is ref Mojolicious::Commands->start_app('MojoliciousTest'), 'CODE', 'right reference';
  is $ENV{MOJO_MODE},                                         undef,  'no mode';
}

# mojo
is_deeply $commands->namespaces, ['Mojolicious::Command::Author', 'Mojolicious::Command'], 'right namespaces';
ok $commands->description, 'has a description';
like $commands->message, qr/COMMAND/, 'has a message';
like $commands->hint,    qr/help/,    'has a hint';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  local $ENV{HARNESS_ACTIVE} = 0;
  $commands->run;
}
like $buffer,   qr/Usage: APPLICATION COMMAND \[OPTIONS\].*daemon.*my-test-command.*version/s, 'right output';
like $buffer,   qr/See, it works/,        'description has been picked up';
unlike $buffer, qr/my-fake-test-command/, 'fake command has been ignored';

# help
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $commands->run('help', 'generate', 'lite-app');
}
like $buffer, qr/Usage: APPLICATION generate lite-app \[OPTIONS\] \[NAME\]/, 'right output';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $commands->run('generate', 'help', 'lite-app');
}
like $buffer, qr/Usage: APPLICATION generate lite-app \[OPTIONS\] \[NAME\]/, 'right output';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $commands->run('generate', 'app', '-h');
}
like $buffer, qr/Usage: APPLICATION generate app \[OPTIONS\] \[NAME\]/, 'right output';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $commands->run('generate', 'lite-app', '--help');
}
like $buffer, qr/Usage: APPLICATION generate lite-app \[OPTIONS\] \[NAME\]/, 'right output';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  local $ENV{HARNESS_ACTIVE} = 0;
  $commands->run('help');
}
like $buffer, qr/Usage: APPLICATION COMMAND \[OPTIONS\]/, 'right output';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  local $ENV{HARNESS_ACTIVE} = 0;
  $commands->run('--help');
}
like $buffer, qr/Usage: APPLICATION COMMAND \[OPTIONS\]/, 'right output';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  local $ENV{HARNESS_ACTIVE} = 0;
  $commands->run('-h');
}
like $buffer, qr/Usage: APPLICATION COMMAND \[OPTIONS\]/, 'right output';

# get
require Mojolicious::Command::get;
my $get = Mojolicious::Command::get->new;
ok $get->description, 'has a description';
like $get->usage, qr/get/, 'has usage information';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $get->run('/');
}
like $buffer, qr/Your Mojo is working!/, 'right output';
my $template = '<p></p><p><%= param "just" %> <%= $c->req->headers->header("X-Test") %></p>';
$get->app->plugins->once(before_dispatch => sub { shift->render(inline => $template) });
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $get->run('-f', 'just=works', '-H', 'X-Test: fine', '/html', 'p', 1, 'text');
}
like $buffer, qr/works fine/, 'right output';
$get->app->plugins->once(before_dispatch => sub { shift->render(json => {works => 'too'}) });
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $get->run('/json', '/works');
}
like $buffer, qr/too/, 'right output';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDERR = $handle;
  eval { $get->run('--unknown') };
  like $@, qr/Usage: APPLICATION get/, 'unknown option';
}
like $buffer, qr/Unknown option: unknown/, 'right output';

# cgi
require Mojolicious::Command::cgi;
my $cgi = Mojolicious::Command::cgi->new;
ok $cgi->description, 'has a description';
like $cgi->usage, qr/cgi/, 'has usage information';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDERR = $handle;
  eval { $cgi->run('--unknown') };
  like $@, qr/Usage: APPLICATION cgi/, 'unknown option';
}
like $buffer, qr/Unknown option: unknown/, 'right output';

# cpanify
require Mojolicious::Command::Author::cpanify;
my $cpanify = Mojolicious::Command::Author::cpanify->new;
ok $cpanify->description, 'has a description';
like $cpanify->usage, qr/cpanify/, 'has usage information';
$cpanify->app->ua->server->app($cpanify->app);
$cpanify->app->ua->unsubscribe('start')->once(
  start => sub {
    my ($ua, $tx) = @_;
    $tx->req->via_proxy(0)->url($ua->server->url->path('/'));
  }
);
$cpanify->app->plugins->once(before_dispatch => sub { shift->render(data => '', status => 200) });
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $cpanify->run('-u', 'sri', '-p', 's3cret', __FILE__);
}
like $buffer, qr/Upload successful!/, 'right output';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDERR = $handle;
  eval { $cpanify->run('--unknown') };
  like $@, qr/Usage: APPLICATION cpanify/, 'unknown option';
}
like $buffer, qr/Unknown option: unknown/, 'right output';

subtest 'daemon' => sub {
  require Mojolicious::Command::daemon;

  subtest 'Description' => sub {
    my $command = Mojolicious::Command::daemon->new;
    ok $command->description, 'has a description';
    like $command->usage, qr/daemon/, 'has usage information';
  };

  subtest 'Unknown option' => sub {
    my $command = Mojolicious::Command::daemon->new;
    $buffer = '';
    {
      open my $handle, '>', \$buffer;
      local *STDERR = $handle;
      eval { $command->run('--unknown') };
      like $@, qr/Usage: APPLICATION daemon/, 'unknown option';
    }
    like $buffer, qr/Unknown option: unknown/, 'right output';
  };

  subtest 'Proxy boolean' => sub {
    my $command = Mojolicious::Command::daemon->new;
    my $daemon  = $command->build_server('-p');
    ok $daemon->reverse_proxy, 'right value';
    is_deeply $daemon->trusted_proxies, [], 'right value';
  };

  subtest 'Trusted proxies' => sub {
    my $command = Mojolicious::Command::daemon->new;
    my $daemon  = $command->build_server('-p', '127.0.0.0/8', '-p', '10.0.0.0/8');
    ok $daemon->reverse_proxy, 'right value';
    is_deeply $daemon->trusted_proxies, ['127.0.0.0/8', '10.0.0.0/8'], 'right value';
  };

  subtest 'Trusted proxies from environment' => sub {
    local $ENV{MOJO_TRUSTED_PROXIES} = '127.0.0.0/8,10.0.0.0/8';
    my $command = Mojolicious::Command::daemon->new;
    my $daemon  = $command->build_server;
    ok $daemon->reverse_proxy, 'right value';
    is_deeply $daemon->trusted_proxies, ['127.0.0.0/8', '10.0.0.0/8'], 'right value';
  };

  subtest 'Proxy boolean and trusted' => sub {
    my $command = Mojolicious::Command::daemon->new;
    my $daemon  = $command->build_server('-p', '-p', '127.0.0.0/8', '-p', '10.0.0.0/8');
    ok $daemon->reverse_proxy, 'right value';
    is_deeply $daemon->trusted_proxies, ['127.0.0.0/8', '10.0.0.0/8'], 'right value';
  };
};

# eval
require Mojolicious::Command::eval;
my $eval = Mojolicious::Command::eval->new;
ok $eval->description, 'has a description';
like $eval->usage, qr/eval/, 'has usage information';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $eval->run('-v', 'app->controller_class');
}
like $buffer, qr/Mojolicious::Controller/, 'right output';
eval { $eval->run('-v', 'die "TEST"') };
like $@, qr/TEST/, 'right output';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $eval->run('-v', 'Mojo::Promise->new->resolve("Zoidberg")');
}
like $buffer, qr/Zoidberg/, 'right output';
eval { $eval->run('-v', 'Mojo::Promise->new->reject("DOOM")') };
like $@, qr/DOOM/, 'right output';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDERR = $handle;
  eval { $eval->run('--unknown') };
  like $@, qr/Usage: APPLICATION eval/, 'unknown option';
}
like $buffer, qr/Unknown option: unknown/, 'right output';

# generate
require Mojolicious::Command::Author::generate;
my $generator = Mojolicious::Command::Author::generate->new;
is_deeply $generator->namespaces, ['Mojolicious::Command::Author::generate'], 'right namespaces';
ok $generator->description, 'has a description';
like $generator->message, qr/generate/, 'has a message';
like $generator->hint,    qr/help/,     'has a hint';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  local $ENV{HARNESS_ACTIVE} = 0;
  $generator->run;
}
like $buffer, qr/Usage: APPLICATION generate GENERATOR \[OPTIONS\].*lite-app.*plugin/s, 'right output';

# generate app
require Mojolicious::Command::Author::generate::app;
$app = Mojolicious::Command::Author::generate::app->new;
ok $app->description, 'has a description';
like $app->usage, qr/app/, 'has usage information';
my $cwd = path;
my $dir = tempdir;
chdir $dir;
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $app->run;
}
like $buffer, qr/my_app/, 'right output';
ok -e $app->rel_file('my_app/script/my_app'),                     'script exists';
ok -e $app->rel_file('my_app/lib/MyApp.pm'),                      'application class exists';
ok -e $app->rel_file('my_app/lib/MyApp/Controller/Example.pm'),   'controller exists';
ok -e $app->rel_file('my_app/my_app.yml'),                        'config file exists';
ok -e $app->rel_file('my_app/t/basic.t'),                         'test exists';
ok -d $app->rel_file('my_app/public/assets'),                     'assets directory exists';
ok -e $app->rel_file('my_app/public/index.html'),                 'static file exists';
ok -e $app->rel_file('my_app/templates/layouts/default.html.ep'), 'layout exists';
ok -e $app->rel_file('my_app/templates/example/welcome.html.ep'), 'template exists';
chdir $cwd;

subtest 'generate dockerfile' => sub {
  require Mojolicious::Command::Author::generate::dockerfile;
  my $dockerfile = Mojolicious::Command::Author::generate::dockerfile->new;
  ok $dockerfile->description, 'has a description';
  like $dockerfile->usage, qr/dockerfile/, 'has usage information';
  my $dir = tempdir CLEANUP => 1;
  chdir $dir;
  $buffer = '';
  {
    open my $handle, '>', \$buffer;
    local *STDOUT = $handle;
    $dockerfile->run;
  }
  like $buffer, qr/Dockerfile/, 'right output';
  ok -e $app->rel_file('Dockerfile'), 'Dockerfile exists';
  chdir $cwd;
};

# generate lite_app
require Mojolicious::Command::Author::generate::lite_app;
$app = Mojolicious::Command::Author::generate::lite_app->new;
ok $app->description, 'has a description';
like $app->usage, qr/lite-app/, 'has usage information';
$dir = tempdir CLEANUP => 1;
chdir $dir;
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $app->run;
}
like $buffer, qr/myapp\.pl/, 'right output';
ok -e $app->rel_file('myapp.pl'), 'app exists';
chdir $cwd;

# generate makefile
require Mojolicious::Command::Author::generate::makefile;
my $makefile = Mojolicious::Command::Author::generate::makefile->new;
ok $makefile->description, 'has a description';
like $makefile->usage, qr/makefile/, 'has usage information';
$dir = tempdir CLEANUP => 1;
chdir $dir;
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $makefile->run;
}
like $buffer, qr/Makefile\.PL/, 'right output';
ok -e $app->rel_file('Makefile.PL'), 'Makefile.PL exists';
chdir $cwd;

# generate plugin
require Mojolicious::Command::Author::generate::plugin;
my $plugin = Mojolicious::Command::Author::generate::plugin->new;
ok $plugin->description, 'has a description';
like $plugin->usage, qr/plugin/, 'has usage information';
$dir = tempdir CLEANUP => 1;
chdir $dir;
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $plugin->run;
}
like $buffer, qr/MyPlugin\.pm/, 'right output';
ok -e $app->rel_file('Mojolicious-Plugin-MyPlugin/lib/Mojolicious/Plugin/MyPlugin.pm'), 'class exists';
ok -e $app->rel_file('Mojolicious-Plugin-MyPlugin/t/basic.t'),                          'test exists';
ok -e $app->rel_file('Mojolicious-Plugin-MyPlugin/Makefile.PL'),                        'Makefile.PL exists';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $plugin->run('-f', 'MyApp::Ext::Test');
}
like $buffer, qr/Test\.pm/, 'right output';
ok -e $app->rel_file('MyApp-Ext-Test/lib/MyApp/Ext/Test.pm'), 'class exists';
ok -e $app->rel_file('MyApp-Ext-Test/t/basic.t'),             'test exists';
ok -e $app->rel_file('MyApp-Ext-Test/Makefile.PL'),           'Makefile.PL exists';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDERR = $handle;
  eval { $plugin->run('--unknown') };
  like $@, qr/Usage: APPLICATION generate plugin/, 'unknown option';
}
like $buffer, qr/Unknown option: unknown/, 'right output';
chdir $cwd;

# inflate
require Mojolicious::Command::Author::inflate;
my $inflate = Mojolicious::Command::Author::inflate->new;
ok $inflate->description, 'has a description';
like $inflate->usage, qr/inflate/, 'has usage information';

subtest 'prefork' => sub {
  require Mojolicious::Command::prefork;

  subtest 'Description' => sub {
    my $command = Mojolicious::Command::prefork->new;
    ok $command->description, 'has a description';
    like $command->usage, qr/prefork/, 'has usage information';
  };

  subtest 'Unknown option' => sub {
    my $command = Mojolicious::Command::prefork->new;
    $buffer = '';
    {
      open my $handle, '>', \$buffer;
      local *STDERR = $handle;
      eval { $command->run('--unknown') };
      like $@, qr/Usage: APPLICATION prefork/, 'unknown option';
    }
    like $buffer, qr/Unknown option: unknown/, 'right output';
  };

  subtest 'Proxy boolean' => sub {
    my $command = Mojolicious::Command::prefork->new;
    my $prefork = $command->build_server('-p');
    ok $prefork->reverse_proxy, 'right value';
    is_deeply $prefork->trusted_proxies, [], 'right value';
  };

  subtest 'Trusted proxies' => sub {
    my $command = Mojolicious::Command::prefork->new;
    my $prefork = $command->build_server('-p', '127.0.0.0/8', '-p', '10.0.0.0/8');
    ok $prefork->reverse_proxy, 'right value';
    is_deeply $prefork->trusted_proxies, ['127.0.0.0/8', '10.0.0.0/8'], 'right value';
  };

  subtest 'Trusted proxies from environment' => sub {
    local $ENV{MOJO_TRUSTED_PROXIES} = '127.0.0.0/8,10.0.0.0/8';
    my $command = Mojolicious::Command::prefork->new;
    my $prefork = $command->build_server;
    ok $prefork->reverse_proxy, 'right value';
    is_deeply $prefork->trusted_proxies, ['127.0.0.0/8', '10.0.0.0/8'], 'right value';
  };

  subtest 'Proxy boolean and trusted' => sub {
    my $command = Mojolicious::Command::prefork->new;
    my $prefork = $command->build_server('-p', '-p', '127.0.0.0/8', '-p', '10.0.0.0/8');
    ok $prefork->reverse_proxy, 'right value';
    is_deeply $prefork->trusted_proxies, ['127.0.0.0/8', '10.0.0.0/8'], 'right value';
  };
};

# psgi
require Mojolicious::Command::psgi;
my $psgi = Mojolicious::Command::psgi->new;
ok $psgi->description, 'has a description';
like $psgi->usage, qr/psgi/, 'has usage information';

# routes
require Mojolicious::Command::routes;
my $routes = Mojolicious::Command::routes->new;
ok $routes->description, 'has a description';
like $routes->usage, qr/routes/, 'has usage information';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $routes->run;
}
like $buffer,   qr!/\*whatever!, 'right output';
unlike $buffer, qr!/\(\.\+\)\?!, 'not verbose';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $routes->run('-v');
}
like $buffer, qr!/\*whatever!, 'right output';
like $buffer, qr!/\(\.\+\)\?!, 'verbose';
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDERR = $handle;
  eval { $routes->run('--unknown') };
  like $@, qr/Usage: APPLICATION routes/, 'unknown option';
}
like $buffer, qr/Unknown option: unknown/, 'right output';

# version
require Mojolicious::Command::version;
my $version = Mojolicious::Command::version->new;
ok $version->description, 'has a description';
like $version->usage, qr/version/, 'has usage information';
$version->app->ua->server->app($version->app);
$version->app->ua->once(
  start => sub {
    my ($ua, $tx) = @_;
    $tx->req->via_proxy(0)->url($ua->server->url->path('/'));
  }
);
$version->app->plugins->once(before_dispatch => sub { shift->render(json => {version => 1000}) });
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $version->run;
}
like $buffer, qr/Perl/,                                               'right output';
like $buffer, qr/You might want to update your Mojolicious to 1000!/, 'right output';

# Hooks
$app = Mojolicious->new;
$app->hook(
  before_command => sub {
    my ($command, $args) = @_;
    return unless $command->isa('Mojolicious::Command::eval');
    $command->app->config->{test} = 'works!';
    unshift @$args, '-v';
  }
);
$buffer = '';
{
  open my $handle, '>', \$buffer;
  local *STDOUT = $handle;
  $app->start('eval', 'app->config->{test}');
}
like $buffer, qr/works!/, 'right output';

done_testing();


Powered by Groonga
Maintained by Kenichi Ishigaki <ishigaki@cpan.org>. If you find anything, submit it on GitHub.