App-lcpan
view release on metacpan or search on metacpan
lib/App/lcpan/Cmd/changes.pm view on Meta::CPAN
# schema => 'bool',
#}.
},
examples => [
{
summary => 'Use module name',
argv => ['Data::CSel::Parser'],
test => 0,
'x.doc.show_result' => 0,
},
#{
# summary => 'Use dist name, parse',
# argv => ['--parse', 'App-PMUtils'],
# test => 0,
# 'x.doc.show_result' => 0,
#},
],
};
sub handle_cmd {
my %args = @_;
my $state = App::lcpan::_init(\%args, 'ro');
my $dbh = $state->{dbh};
my $mod_or_dist_or_script = $args{module_or_dist_or_script};
$mod_or_dist_or_script =~ s!/!::!g; # XXX this should be done by coercer
my @join;
my @where;
my @bind;
my @file_ids;
{
# search in module first
unless ($mod_or_dist_or_script =~ /-/) {
my $sth = $dbh->prepare("SELECT file_id FROM module WHERE name=?");
$sth->execute($mod_or_dist_or_script);
while (my ($e) = $sth->fetchrow_array) {
push @file_ids, $e;
}
}
# search in dist or script
unless ($mod_or_dist_or_script =~ /::/) {
my $dist_found;
my $sth = $dbh->prepare("SELECT id FROM file WHERE dist_name=? ORDER BY dist_version_numified DESC LIMIT 1");
$sth->execute($mod_or_dist_or_script);
while (my ($e) = $sth->fetchrow_array) {
$dist_found++;
push @file_ids, $e;
}
unless ($dist_found) {
my $sth = $dbh->prepare("SELECT file_id FROM script WHERE name=?");
$sth->execute($mod_or_dist_or_script);
while (my ($e) = $sth->fetchrow_array) {
push @file_ids, $e;
}
}
}
return [404, "No such module/dist/script"] unless @file_ids;
push @where, "file.id IN (".join(",", @file_ids).")";
}
my $sql = "SELECT
content.path content_path,
file.cpanid author,
file.name release
FROM content
LEFT JOIN file ON content.file_id=file.id
".(@join ? join(" ", @join) : "")."
".(@where ? " WHERE ".join(" AND ", @where) : "")."
ORDER BY content.path";
my $sth = $dbh->prepare($sql);
$sth->execute(@bind);
my $first_row;
while (my $row = $sth->fetchrow_hashref) {
$first_row //= $row;
next unless $row->{content_path} =~ m!\A
(?:[^/]+/)?
(changes|changelog)
(?:\.(\w+))?\z!ix;
# XXX handle YAML file
my $path = App::lcpan::_fullpath(
$row->{release}, $state->{cpan}, $row->{author});
# XXX needs to be refactored into common code (see also doc subcommand)
my $content;
if ($path =~ /\.zip$/i) {
require Archive::Zip;
my $zip = Archive::Zip->new;
$zip->read($path) == Archive::Zip::AZ_OK()
or return [500, "Can't read zip file '$path'"];
$content = $zip->contents($row->{content_path});
} else {
require Archive::Tar;
my $tar;
eval {
$tar = Archive::Tar->new;
$content = $tar->read($path); # can still die untrapped when out of mem
};
return [500, "Can't read tar file '$path': $@"] if $@;
my ($obj) = $tar->get_files($row->{content_path});
$content = $obj->get_content;
}
return [200, "OK", $content, {
'func.content_path' => $row->{content_path},
'cmdline.skip_format'=>1,
"cmdline.page_result"=>1,
}];
}
if ($first_row) {
return [404, "No Changes file found in $first_row->{release}"];
} else {
return [404, "No such module or dist"];
}
}
1;
# ABSTRACT: Show Changes of distribution/module
__END__
=pod
=encoding UTF-8
=head1 NAME
App::lcpan::Cmd::changes - Show Changes of distribution/module
=head1 VERSION
This document describes version 1.074 of App::lcpan::Cmd::changes (from Perl distribution App-lcpan), released on 2023-09-26.
=head1 FUNCTIONS
=head2 handle_cmd
Usage:
handle_cmd(%args) -> [$status_code, $reason, $payload, \%result_meta]
Show Changes of distributionE<sol>module.
Examples:
=over
=item * Use module name:
handle_cmd(module_or_dist_or_script => "Data::CSel::Parser");
=back
This command will find a file named Changes/CHANGES/ChangeLog or other similar
name in the top-level directory inside the release tarballs and show it.
This function is not exported.
Arguments ('*' denotes required arguments):
=over 4
=item * B<cpan> => I<dirname>
Location of your local CPAN mirror, e.g. E<sol>pathE<sol>toE<sol>cpan.
Defaults to C<~/cpan>.
=item * B<index_name> => I<filename> (default: "index.db")
Filename of index.
( run in 1.888 second using v1.01-cache-2.11-cpan-39bf76dae61 )