You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
ngcp-panel/inc/Local/Module/Build.pm

210 lines
6.8 KiB

package Local::Module::Build;
use Moose qw(around extends);
use Child qw(child);
use Capture::Tiny;
use TryCatch;
use MooseX::Method::Signatures;
use LWP::UserAgent;
extends 'Module::Build';
our ($plackup, $webdriver, @cover_opt, $mysqld);
method wait_socket($host, $port, $timeout=90) {
require IO::Socket::IP;
my $timer = 0;
while (1) {
my $sock = IO::Socket::IP->new(
PeerHost => $host,
PeerPort => $port,
Type => IO::Socket::IP::SOCK_STREAM(),
);
last if $sock;
sleep 1;
$timer++;
die sprintf('socket %s:%s is not accessible within %d seconds after start', $host, $port, $timeout)
if $timer > $timeout;
};
}
sub shutdown_servers {
for my $proc ($webdriver, $plackup) {
if ($proc) {
$proc->kill('TERM');
}
}
}
sub _test_preconditions {
my ($self) = @_;
require Getopt::Long;
Getopt::Long::Configure('pass_through');
my %opt = (server => 'http://localhost:5000', webdriver => 'external');
Getopt::Long::GetOptions(\%opt, 'webdriver=s', 'server:s', 'help|?', 'man', 'wd-server=s',
'schema-base-dir=s', 'mysqld-port=s', 'mysql-dump=s@', 'no-junit', 'run-server')
or die 'could not process command-line options';
require Pod::Usage;
Pod::Usage::pod2usage(-exitval => 1, -input => 'Build.PL') if $opt{help};
Pod::Usage::pod2usage(-exitval => 0, -input => 'Build.PL', -verbose => 2) if $opt{man};
if ($opt{'no-junit'}) {
delete $self->tap_harness_args->{formatter_class};
$self->tap_harness_args->{verbosity} = 1;
}
if ($opt{'wd-server'}) {
my ($wd_host, $wd_port) = $opt{'wd-server'} =~ m{([^/:]+):([0-9]+)};
$ENV{TWD_HOST} = $wd_host;
$ENV{TWD_PORT} = $wd_port;
}
if ($opt{'schema-base-dir'}) {
require blib;
blib->import($opt{'schema-base-dir'})
}
$SIG{'INT'} = sub { exit(1) }; # for clean stopping of servers
if ($opt{'mysqld-port'} && $opt{'mysql-dump'}) {
require Test::mysqld;
$mysqld = Test::mysqld->new(
my_cnf => {
'port' => $opt{'mysqld-port'},
},
) or die "couldnt start mysqld";
$ENV{NGCP_PANEL_CUSTOM_DSN} = $mysqld->dsn();
my $dump_files = join(' ', @{ $opt{'mysql-dump'} });
system("cat $dump_files | mysql -uroot --host=127.0.0.1 --port=$opt{'mysqld-port'}");
system(qq/echo "GRANT ALL PRIVILEGES ON *.* TO 'sipwise'\@'localhost' WITH GRANT OPTION;" | mysql -uroot --host=127.0.0.1 --port=$opt{'mysqld-port'}/);
}
unless ($opt{webdriver} eq "external") {
$webdriver = child { exec $opt{webdriver} };
$self->wait_socket(qw(localhost 4444));
}
require URI;
unless ($opt{server} =~ m|^https?://|) {
die "Wrong format of server argument, should start with 'http(s)'.";
}
my $uri = URI->new($opt{server});
if( $opt{'run-server'} ) {
require File::Which;
$ENV{ NGCP_PANEL_CONFIG_LOCAL_SUFFIX } = "testing";
$plackup = child {
my $out_fh = IO::File->new("panel_debug_stdout", "w+");
my $err_fh = IO::File->new("panel_debug_stderr", "w+");
$out_fh->autoflush(1);
$err_fh->autoflush(1);
local $| = 1;
Capture::Tiny::capture {
exec $^X,
'-Ilib',
exists $opt{'schema-base-dir'} ? "-Mblib=$opt{'schema-base-dir'}" : (),
@cover_opt,
scalar File::Which::which('plackup'),
sprintf('--listen=%s:%s', $uri->host, $uri->port),
'ngcp_panel.psgi';
} stdout => $out_fh, stderr => $err_fh;
};
$self->wait_socket($uri->host, $uri->port);
}
$ENV{CATALYST_SERVER} = $opt{server};
if ($self->verbose) {
print("Server is: ".$opt{server}."\n");
}
}
sub _download_certs {
my ($self) = @_;
my $uri = $ENV{CATALYST_SERVER};
use File::Temp qw/tempfile/;
my ($ua, $req, $res);
$ua = LWP::UserAgent->new(cookie_jar => {}, ssl_opts => {verify_hostname => 0});
$res = $ua->post($uri.'/login/admin', {username => 'administrator', password => 'administrator'}, 'Referer' => $uri.'/login/admin');
$res = $ua->get($uri.'/dashboard/');
$res = $ua->get($uri.'/administrator/1/api_key');
if ($res->decoded_content =~ m/gen\.generate/) { # key need to be generated first
$res = $ua->post($uri.'/administrator/1/api_key', {'gen.generate' => 'foo'}, 'Referer' => $uri.'/dashboard');
}
my (undef, $tmp_apiclient_filename) = tempfile;
my (undef, $tmp_apica_filename) = tempfile;
$res = $ua->post($uri.'/administrator/1/api_key', {'pem.download' => 'foo'}, 'Referer' => $uri.'/dashboard', ':content_file' => $tmp_apiclient_filename);
$res = $ua->post($uri.'/administrator/1/api_key', {'ca.download' => 'foo'}, 'Referer' => $uri.'/dashboard', ':content_file' => $tmp_apica_filename);
$ENV{API_SSL_CLIENT_CERT} = $tmp_apiclient_filename;
$ENV{API_SSL_CA_CERT} = $tmp_apica_filename;
print "Client cert: $tmp_apiclient_filename - CA cert: $tmp_apica_filename\n" if $self->verbose;
}
around('ACTION_test', sub {
my $super = shift;
my $self = shift;
$self->_test_preconditions;
try {
$self->$super(@_);
};
});
method ACTION_testcover {
{
my @missing;
for my $module (qw(Devel::Cover sigtrap)) {
push @missing, $module
unless Module::Build::ModuleInfo->find_module_by_name($module);
}
if (@missing) {
warn "modules required for testcover action: @missing\n";
return;
}
}
$self->add_to_cleanup('coverage', 'cover_db');
$self->depends_on('code');
$self->do_system(qw(cover -delete));
@cover_opt = (
'-Msigtrap "handler", sub { exit }, "normal-signals"',
'-MDevel::Cover=+ignore,ngcp_panel.psgi,+ignore,plackup',
);
$self->depends_on('test');
shutdown_servers;
sleep 5;
$self->do_system(qw(cover));
}
method ACTION_test_servers {
$self->depends_on('code');
$self->_test_preconditions;
print "All servers ready for you!\nPress [Enter] to exit.";
<STDIN>;
}
method ACTION_test_selenium {
$self->depends_on('code');
$self->_test_preconditions;
$self->test_files('t/*_selenium.t t/admin-login.t');
$self->generic_test(type => 'default');
}
method ACTION_test_api {
$self->depends_on('code');
$self->_test_preconditions;
$self->_download_certs;
$self->test_files('t/api-*.t');
$self->generic_test(type => 'default');
unlink ($ENV{API_SSL_CLIENT_CERT}, $ENV{API_SSL_CA_CERT}); # created by _download_certs()
}
method ACTION_readme {
require Pod::Readme;
my $parser = Pod::Readme->new();
$parser->parse_from_file('Build.PL', 'README');
}
END { shutdown_servers }
1;