2014-02-23 15:17:20 +01:00
|
|
|
#!/usr/bin/env perl
|
|
|
|
|
|
|
|
# This chunk of stuff was generated by App::FatPacker. To find the original
|
|
|
|
# file's code, look for the end of this BEGIN block or the string 'FATPACK'
|
|
|
|
BEGIN {
|
|
|
|
my %fatpacked;
|
|
|
|
|
|
|
|
$fatpacked{"MetaCPAN/API/Tiny.pm"} = <<'METACPAN_API_TINY';
|
|
|
|
package MetaCPAN::API::Tiny;
|
|
|
|
{
|
|
|
|
$MetaCPAN::API::Tiny::VERSION = '1.131730';
|
|
|
|
}
|
|
|
|
use strict;
|
|
|
|
use warnings;
|
|
|
|
# ABSTRACT: A Tiny API client for MetaCPAN
|
|
|
|
|
|
|
|
use Carp;
|
|
|
|
use JSON::PP 'encode_json', 'decode_json';
|
|
|
|
use HTTP::Tiny;
|
|
|
|
|
|
|
|
|
|
|
|
sub new {
|
|
|
|
my ($class, @args) = @_;
|
|
|
|
|
|
|
|
$#_ % 2 == 0
|
|
|
|
or croak 'Arguments must be provided as name/value pairs';
|
|
|
|
|
|
|
|
my %params = @args;
|
|
|
|
|
|
|
|
die 'ua_args must be an array reference'
|
|
|
|
if $params{ua_args} && ref($params{ua_args}) ne 'ARRAY';
|
|
|
|
|
|
|
|
my $self = +{
|
|
|
|
base_url => $params{base_url} || 'http://api.metacpan.org/v0',
|
|
|
|
ua => $params{ua} || HTTP::Tiny->new(
|
|
|
|
$params{ua_args}
|
|
|
|
? @{$params{ua_args}}
|
|
|
|
: (agent => 'MetaCPAN::API::Tiny/'
|
|
|
|
. ($MetaCPAN::API::VERSION || 'xx'))),
|
|
|
|
};
|
|
|
|
|
|
|
|
return bless($self, $class);
|
|
|
|
}
|
|
|
|
|
|
|
|
sub _build_extra_params {
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
@_ % 2 == 0
|
|
|
|
or croak 'Incorrect number of params, must be key/value';
|
|
|
|
|
|
|
|
my %extra = @_;
|
|
|
|
my $ua = $self->{ua};
|
|
|
|
|
|
|
|
foreach my $key (keys %extra)
|
|
|
|
{
|
|
|
|
# The implementation in HTTP::Tiny uses + instead of %20, fix that
|
|
|
|
$extra{$key} = $ua->_uri_escape($extra{$key});
|
|
|
|
$extra{$key} =~ s/\+/%20/g;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $params = join '&', map { "$_=" . $extra{$_} } sort keys %extra;
|
|
|
|
|
|
|
|
return $params;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# /source/{author}/{release}/{path}
|
|
|
|
sub source {
|
|
|
|
my $self = shift;
|
|
|
|
my %opts = @_ ? @_ : ();
|
|
|
|
my $url = '';
|
|
|
|
my $error = "Provide 'author' and 'release' and 'path'";
|
|
|
|
|
|
|
|
%opts or croak $error;
|
|
|
|
|
|
|
|
if (
|
|
|
|
defined ( my $author = $opts{'author'} ) &&
|
|
|
|
defined ( my $release = $opts{'release'} ) &&
|
|
|
|
defined ( my $path = $opts{'path'} )
|
|
|
|
) {
|
|
|
|
$url = "source/$author/$release/$path";
|
|
|
|
} else {
|
|
|
|
croak $error;
|
|
|
|
}
|
|
|
|
|
|
|
|
$url = $self->{base_url} . "/$url";
|
|
|
|
|
|
|
|
my $result = $self->{ua}->get($url);
|
|
|
|
$result->{'success'}
|
|
|
|
or croak "Failed to fetch '$url': " . $result->{'reason'};
|
|
|
|
|
|
|
|
return $result->{'content'};
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# /release/{distribution}
|
|
|
|
# /release/{author}/{release}
|
|
|
|
sub release {
|
|
|
|
my $self = shift;
|
|
|
|
my %opts = @_ ? @_ : ();
|
|
|
|
my $url = '';
|
|
|
|
my $error = "Either provide 'distribution', or 'author' and 'release', " .
|
|
|
|
"or 'search'";
|
|
|
|
|
|
|
|
%opts or croak $error;
|
|
|
|
|
|
|
|
my %extra_opts = ();
|
|
|
|
|
|
|
|
if ( defined ( my $dist = $opts{'distribution'} ) ) {
|
|
|
|
$url = "release/$dist";
|
|
|
|
} elsif (
|
|
|
|
defined ( my $author = $opts{'author'} ) &&
|
|
|
|
defined ( my $release = $opts{'release'} )
|
|
|
|
) {
|
|
|
|
$url = "release/$author/$release";
|
|
|
|
} elsif ( defined ( my $search_opts = $opts{'search'} ) ) {
|
|
|
|
ref $search_opts && ref $search_opts eq 'HASH'
|
|
|
|
or croak $error;
|
|
|
|
|
|
|
|
%extra_opts = %{$search_opts};
|
|
|
|
$url = 'release/_search';
|
|
|
|
} else {
|
|
|
|
croak $error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $self->fetch( $url, %extra_opts );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# /pod/{module}
|
|
|
|
# /pod/{author}/{release}/{path}
|
|
|
|
sub pod {
|
|
|
|
my $self = shift;
|
|
|
|
my %opts = @_ ? @_ : ();
|
|
|
|
my $url = '';
|
|
|
|
my $error = "Either provide 'module' or 'author and 'release' and 'path'";
|
|
|
|
|
|
|
|
%opts or croak $error;
|
|
|
|
|
|
|
|
if ( defined ( my $module = $opts{'module'} ) ) {
|
|
|
|
$url = "pod/$module";
|
|
|
|
} elsif (
|
|
|
|
defined ( my $author = $opts{'author'} ) &&
|
|
|
|
defined ( my $release = $opts{'release'} ) &&
|
|
|
|
defined ( my $path = $opts{'path'} )
|
|
|
|
) {
|
|
|
|
$url = "pod/$author/$release/$path";
|
|
|
|
} else {
|
|
|
|
croak $error;
|
|
|
|
}
|
|
|
|
|
|
|
|
# check content-type
|
|
|
|
my %extra = ();
|
|
|
|
if ( defined ( my $type = $opts{'content-type'} ) ) {
|
|
|
|
$type =~ m{^ text/ (?: html|plain|x-pod|x-markdown ) $}x
|
|
|
|
or croak 'Incorrect content-type provided';
|
|
|
|
|
|
|
|
$extra{headers}{'content-type'} = $type;
|
|
|
|
}
|
|
|
|
|
|
|
|
$url = $self->{base_url}. "/$url";
|
|
|
|
|
|
|
|
my $result = $self->{ua}->get( $url, \%extra );
|
|
|
|
$result->{'success'}
|
|
|
|
or croak "Failed to fetch '$url': " . $result->{'reason'};
|
|
|
|
|
|
|
|
return $result->{'content'};
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# /module/{module}
|
|
|
|
sub module {
|
|
|
|
my $self = shift;
|
|
|
|
my $name = shift;
|
|
|
|
|
|
|
|
$name or croak 'Please provide a module name';
|
|
|
|
|
|
|
|
return $self->fetch("module/$name");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# file() is a synonym of module
|
|
|
|
sub file { goto &module }
|
|
|
|
|
|
|
|
|
|
|
|
# /author/{author}
|
|
|
|
sub author {
|
|
|
|
my $self = shift;
|
|
|
|
my ( $pause_id, $url, %extra_opts );
|
|
|
|
|
|
|
|
if ( @_ == 1 ) {
|
|
|
|
$url = 'author/' . shift;
|
|
|
|
} elsif ( @_ == 2 ) {
|
|
|
|
my %opts = @_;
|
|
|
|
|
|
|
|
if ( defined $opts{'pauseid'} ) {
|
|
|
|
$url = "author/" . $opts{'pauseid'};
|
|
|
|
} elsif ( defined $opts{'search'} ) {
|
|
|
|
my $search_opts = $opts{'search'};
|
|
|
|
|
|
|
|
ref $search_opts && ref $search_opts eq 'HASH'
|
|
|
|
or croak "'search' key must be hashref";
|
|
|
|
|
|
|
|
%extra_opts = %{$search_opts};
|
|
|
|
$url = 'author/_search';
|
|
|
|
} else {
|
|
|
|
croak 'Unknown option given';
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
croak 'Please provide an author PAUSEID or a "search"';
|
|
|
|
}
|
|
|
|
|
|
|
|
return $self->fetch( $url, %extra_opts );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sub fetch {
|
|
|
|
my $self = shift;
|
|
|
|
my $url = shift;
|
|
|
|
my $extra = $self->_build_extra_params(@_);
|
|
|
|
my $base = $self->{base_url};
|
|
|
|
my $req_url = $extra ? "$base/$url?$extra" : "$base/$url";
|
|
|
|
|
|
|
|
my $result = $self->{ua}->get($req_url);
|
|
|
|
return $self->_decode_result( $result, $req_url );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
sub post {
|
|
|
|
my $self = shift;
|
|
|
|
my $url = shift;
|
|
|
|
my $query = shift;
|
|
|
|
my $base = $self->{base_url};
|
|
|
|
|
|
|
|
defined $url
|
|
|
|
or croak 'First argument of URL must be provided';
|
|
|
|
|
|
|
|
ref $query and ref $query eq 'HASH'
|
|
|
|
or croak 'Second argument of query hashref must be provided';
|
|
|
|
|
|
|
|
my $query_json = encode_json( $query );
|
|
|
|
my $result = $self->{ua}->request(
|
|
|
|
'POST',
|
|
|
|
"$base/$url",
|
|
|
|
{
|
|
|
|
headers => { 'Content-Type' => 'application/json' },
|
|
|
|
content => $query_json,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
return $self->_decode_result( $result, $url, $query_json );
|
|
|
|
}
|
|
|
|
|
|
|
|
sub _decode_result {
|
|
|
|
my $self = shift;
|
|
|
|
my ( $result, $url, $original ) = @_;
|
|
|
|
my $decoded_result;
|
|
|
|
|
|
|
|
ref $result and ref $result eq 'HASH'
|
|
|
|
or croak 'First argument must be hashref';
|
|
|
|
|
|
|
|
defined $url
|
|
|
|
or croak 'Second argument of a URL must be provided';
|
|
|
|
|
|
|
|
if ( defined ( my $success = $result->{'success'} ) ) {
|
|
|
|
my $reason = $result->{'reason'} || '';
|
|
|
|
$reason .= ( defined $original ? " (request: $original)" : '' );
|
|
|
|
|
|
|
|
$success or croak "Failed to fetch '$url': $reason";
|
|
|
|
} else {
|
|
|
|
croak 'Missing success in return value';
|
|
|
|
}
|
|
|
|
|
|
|
|
defined ( my $content = $result->{'content'} )
|
|
|
|
or croak 'Missing content in return value';
|
|
|
|
|
|
|
|
eval { $decoded_result = decode_json $content; 1 }
|
|
|
|
or do { croak "Couldn't decode '$content': $@" };
|
|
|
|
|
|
|
|
return $decoded_result;
|
|
|
|
}
|
|
|
|
|
|
|
|
1;
|
|
|
|
|
|
|
|
__END__
|
|
|
|
|
|
|
|
=pod
|
|
|
|
|
|
|
|
=head1 NAME
|
|
|
|
|
|
|
|
MetaCPAN::API::Tiny - A Tiny API client for MetaCPAN
|
|
|
|
|
|
|
|
=head1 VERSION
|
|
|
|
|
|
|
|
version 1.131730
|
|
|
|
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
|
|
|
|
This is the Tiny version of L<MetaCPAN::API>. It implements a compatible API
|
|
|
|
with a few notable exceptions:
|
|
|
|
|
|
|
|
=over 4
|
|
|
|
|
|
|
|
=item Attributes are direct hash access
|
|
|
|
|
|
|
|
The attributes defined using Mo(o|u)se are now accessed via the blessed hash
|
|
|
|
directly. There are no accessors defined to access this elements.
|
|
|
|
|
|
|
|
=item Exception handling
|
|
|
|
|
|
|
|
Instead of using Try::Tiny, raw evals are used. This could potentially cause
|
|
|
|
issues, so just be aware.
|
|
|
|
|
|
|
|
=item Testing
|
|
|
|
|
|
|
|
Test::Fatal was replaced with an eval implementation of exception().
|
|
|
|
Test::TinyMocker usage is retained, but may be absorbed since it is pure perl
|
|
|
|
|
|
|
|
=back
|
|
|
|
|
|
|
|
=head1 CLASS_METHODS
|
|
|
|
|
|
|
|
=head2 new
|
|
|
|
|
|
|
|
new is the constructor for MetaCPAN::API::Tiny. In the non-tiny version of this
|
|
|
|
module, this is provided via Any::Moose built from the attributes defined. In
|
|
|
|
the tiny version, we define our own constructor. It takes the same arguments
|
|
|
|
and provides similar checks to MetaCPAN::API with regards to arguments passed.
|
|
|
|
|
|
|
|
=head1 PUBLIC_METHODS
|
|
|
|
|
|
|
|
=head2 source
|
|
|
|
|
|
|
|
my $source = $mcpan->source(
|
|
|
|
author => 'DOY',
|
|
|
|
release => 'Moose-2.0201',
|
|
|
|
path => 'lib/Moose.pm',
|
|
|
|
);
|
|
|
|
|
|
|
|
Searches MetaCPAN for a module or a specific release and returns the plain source.
|
|
|
|
|
|
|
|
=head2 release
|
|
|
|
|
|
|
|
my $result = $mcpan->release( distribution => 'Moose' );
|
|
|
|
|
|
|
|
# or
|
|
|
|
my $result = $mcpan->release( author => 'DOY', release => 'Moose-2.0001' );
|
|
|
|
|
|
|
|
Searches MetaCPAN for a dist.
|
|
|
|
|
|
|
|
You can do complex searches using 'search' parameter:
|
|
|
|
|
|
|
|
# example lifted from MetaCPAN docs
|
|
|
|
my $result = $mcpan->release(
|
|
|
|
search => {
|
|
|
|
author => "OALDERS AND ",
|
|
|
|
filter => "status:latest",
|
|
|
|
fields => "name",
|
|
|
|
size => 1,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
=head2 pod
|
|
|
|
|
|
|
|
my $result = $mcpan->pod( module => 'Moose' );
|
|
|
|
|
|
|
|
# or
|
|
|
|
my $result = $mcpan->pod(
|
|
|
|
author => 'DOY',
|
|
|
|
release => 'Moose-2.0201',
|
|
|
|
path => 'lib/Moose.pm',
|
|
|
|
);
|
|
|
|
|
|
|
|
Searches MetaCPAN for a module or a specific release and returns the POD.
|
|
|
|
|
|
|
|
=head2 module
|
|
|
|
|
|
|
|
my $result = $mcpan->module('MetaCPAN::API');
|
|
|
|
|
|
|
|
Searches MetaCPAN and returns a module's ".pm" file.
|
|
|
|
|
|
|
|
=head2 file
|
|
|
|
|
|
|
|
A synonym of L</module>
|
|
|
|
|
|
|
|
=head2 author
|
|
|
|
|
|
|
|
my $result1 = $mcpan->author('XSAWYERX');
|
|
|
|
my $result2 = $mcpan->author( pauseid => 'XSAWYERX' );
|
|
|
|
|
|
|
|
Searches MetaCPAN for a specific author.
|
|
|
|
|
|
|
|
You can do complex searches using 'search' parameter:
|
|
|
|
|
|
|
|
# example lifted from MetaCPAN docs
|
|
|
|
my $result = $mcpan->author(
|
|
|
|
search => {
|
|
|
|
q => 'profile.name:twitter',
|
|
|
|
size => 1,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
=head2 fetch
|
|
|
|
|
|
|
|
my $result = $mcpan->fetch('/release/distribution/Moose');
|
|
|
|
|
|
|
|
# with parameters
|
|
|
|
my $more = $mcpan->fetch(
|
|
|
|
'/release/distribution/Moose',
|
|
|
|
param => 'value',
|
|
|
|
);
|
|
|
|
|
|
|
|
This is a helper method for API implementations. It fetches a path from MetaCPAN, decodes the JSON from the content variable and returns it.
|
|
|
|
|
|
|
|
You don't really need to use it, but you can in case you want to write your own extension implementation to MetaCPAN::API.
|
|
|
|
|
|
|
|
It accepts an additional hash as "GET" parameters.
|
|
|
|
|
|
|
|
=head2 post
|
|
|
|
|
|
|
|
# /release&content={"query":{"match_all":{}},"filter":{"prefix":{"archive":"Cache-Cache-1.06"}}}
|
|
|
|
my $result = $mcpan->post(
|
|
|
|
'release',
|
|
|
|
{
|
|
|
|
query => { match_all => {} },
|
|
|
|
filter => { prefix => { archive => 'Cache-Cache-1.06' } },
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
The POST equivalent of the "fetch()" method. It gets the path and JSON request.
|
|
|
|
|
|
|
|
=head1 THANKS
|
|
|
|
|
|
|
|
Overall the tests and code were ripped directly from MetaCPAN::API and
|
|
|
|
tiny-fied. A big thanks to Sawyer X for writing the original module.
|
|
|
|
|
|
|
|
=head1 AUTHOR
|
|
|
|
|
|
|
|
Nicholas R. Perez <nperez@cpan.org>
|
|
|
|
|
|
|
|
=head1 COPYRIGHT AND LICENSE
|
|
|
|
|
|
|
|
This software is copyright (c) 2013 by Nicholas R. Perez <nperez@cpan.org>.
|
|
|
|
|
|
|
|
This is free software; you can redistribute it and/or modify it under
|
|
|
|
the same terms as the Perl 5 programming language system itself.
|
|
|
|
|
|
|
|
=cut
|
|
|
|
METACPAN_API_TINY
|
|
|
|
|
|
|
|
s/^ //mg for values %fatpacked;
|
|
|
|
|
|
|
|
unshift @INC, sub {
|
|
|
|
if (my $fat = $fatpacked{$_[1]}) {
|
|
|
|
if ($] < 5.008) {
|
|
|
|
return sub {
|
|
|
|
return 0 unless length $fat;
|
|
|
|
$fat =~ s/^([^\n]*\n?)//;
|
|
|
|
$_ = $1;
|
|
|
|
return 1;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
open my $fh, '<', \$fat
|
|
|
|
or die "FatPacker error loading $_[1] (could be a perl installation issue?)";
|
|
|
|
return $fh;
|
|
|
|
}
|
|
|
|
return
|
|
|
|
};
|
|
|
|
|
|
|
|
} # END OF FATPACK CODE
|
|
|
|
|
|
|
|
|
2016-06-16 18:26:24 +02:00
|
|
|
use 5.010;
|
2014-02-23 15:17:20 +01:00
|
|
|
use strict;
|
|
|
|
use warnings;
|
|
|
|
use Fatal qw(open close);
|
|
|
|
|
|
|
|
use Getopt::Long;
|
|
|
|
use Pod::Usage;
|
|
|
|
use File::Basename;
|
|
|
|
use Module::CoreList;
|
2014-07-17 18:32:58 +02:00
|
|
|
use HTTP::Tiny;
|
2014-11-20 20:35:27 +01:00
|
|
|
use Safe;
|
2014-02-23 15:17:20 +01:00
|
|
|
use MetaCPAN::API::Tiny;
|
2018-05-12 09:03:00 +02:00
|
|
|
use Digest::SHA qw(sha256_hex);
|
2018-10-10 08:46:09 +02:00
|
|
|
use Text::Wrap;
|
|
|
|
$Text::Wrap::columns = 62;
|
2014-02-23 15:17:20 +01:00
|
|
|
|
2017-09-27 10:32:50 +02:00
|
|
|
# Below, 5.026 should be aligned with the version of perl actually
|
2016-06-16 18:26:24 +02:00
|
|
|
# bundled in Buildroot:
|
2017-09-27 10:32:50 +02:00
|
|
|
die <<"MSG" if $] < 5.026;
|
2016-06-16 18:26:24 +02:00
|
|
|
This script needs a host perl with the same major version as Buildroot target perl.
|
|
|
|
|
|
|
|
Your current host perl is:
|
|
|
|
$^X
|
|
|
|
version $]
|
|
|
|
|
|
|
|
You may install a local one by running:
|
2017-09-27 10:32:50 +02:00
|
|
|
perlbrew install perl-5.26.0
|
2016-06-16 18:26:24 +02:00
|
|
|
MSG
|
|
|
|
|
2014-10-01 20:36:59 +02:00
|
|
|
my ($help, $man, $quiet, $force, $recommend, $test, $host);
|
2014-04-04 19:10:33 +02:00
|
|
|
my $target = 1;
|
2014-02-23 15:17:20 +01:00
|
|
|
GetOptions( 'help|?' => \$help,
|
|
|
|
'man' => \$man,
|
|
|
|
'quiet|q' => \$quiet,
|
|
|
|
'force|f' => \$force,
|
2014-04-04 19:10:33 +02:00
|
|
|
'host!' => \$host,
|
|
|
|
'target!' => \$target,
|
2014-10-01 20:36:59 +02:00
|
|
|
'recommend' => \$recommend,
|
|
|
|
'test' => \$test
|
2014-02-23 15:17:20 +01:00
|
|
|
) or pod2usage(-exitval => 1);
|
|
|
|
pod2usage(-exitval => 0) if $help;
|
|
|
|
pod2usage(-exitval => 0, -verbose => 2) if $man;
|
|
|
|
pod2usage(-exitval => 1) if scalar @ARGV == 0;
|
|
|
|
|
|
|
|
my %dist; # name -> metacpan data
|
|
|
|
my %need_target; # name -> 1 if target package is needed
|
|
|
|
my %need_host; # name -> 1 if host package is needed
|
2014-07-17 18:32:58 +02:00
|
|
|
my %need_dlopen; # name -> 1 if requires dynamic library
|
2014-02-23 15:17:20 +01:00
|
|
|
my %deps_build; # name -> list of host dependencies
|
|
|
|
my %deps_runtime; # name -> list of target dependencies
|
2016-03-13 18:37:28 +01:00
|
|
|
my %deps_optional; # name -> list of optional target dependencies
|
2018-05-12 09:03:00 +02:00
|
|
|
my %license_files; # name -> hash of license files
|
2014-11-20 20:35:27 +01:00
|
|
|
my %checksum; # author -> list of checksum
|
2017-06-14 16:15:31 +02:00
|
|
|
my $mirror = 'http://cpan.metacpan.org'; # a CPAN mirror
|
2017-06-14 16:15:29 +02:00
|
|
|
my $mcpan = MetaCPAN::API::Tiny->new(base_url => 'http://fastapi.metacpan.org/v1');
|
2014-07-17 18:32:58 +02:00
|
|
|
my $ua = HTTP::Tiny->new();
|
|
|
|
|
2018-08-13 18:15:52 +02:00
|
|
|
my %white_list = (
|
2018-10-06 18:55:14 +02:00
|
|
|
'ExtUtils-Config' => 1,
|
2018-10-06 18:55:16 +02:00
|
|
|
'ExtUtils-InstallPaths' => 1,
|
2018-10-06 18:55:15 +02:00
|
|
|
'ExtUtils-Helpers' => 1,
|
2018-10-06 18:55:22 +02:00
|
|
|
'File-ShareDir-Install' => 1,
|
2018-08-13 18:15:52 +02:00
|
|
|
'Module-Build' => 1,
|
|
|
|
'Module-Build-Tiny' => 1,
|
|
|
|
);
|
|
|
|
my @info = ();
|
|
|
|
|
2014-11-20 20:35:27 +01:00
|
|
|
sub get_checksum {
|
|
|
|
my ($url) = @_;
|
2017-09-27 10:32:50 +02:00
|
|
|
my ($path) = $url =~ m|^[^:/?#]+://[^/?#]*([^?#]*)|;
|
|
|
|
my ($basename, $dirname) = fileparse( $path );
|
2014-11-20 20:35:27 +01:00
|
|
|
unless ($checksum{$dirname}) {
|
2017-06-14 16:15:31 +02:00
|
|
|
my $url = $mirror . $dirname . q{CHECKSUMS};
|
|
|
|
my $response = $ua->get($url);
|
2014-11-20 20:35:27 +01:00
|
|
|
$checksum{$dirname} = $response->{content};
|
|
|
|
}
|
|
|
|
my $chksum = Safe->new->reval($checksum{$dirname});
|
|
|
|
return $chksum->{$basename}, $basename;
|
|
|
|
}
|
|
|
|
|
2014-07-18 15:43:36 +02:00
|
|
|
sub is_xs {
|
|
|
|
my ($manifest) = @_;
|
2014-07-17 18:32:58 +02:00
|
|
|
# This heuristic determines if a module is a native extension, by searching
|
|
|
|
# some file extension types in the MANIFEST of the distribution.
|
|
|
|
# It was inspired by http://deps.cpantesters.org/static/purity.html
|
2015-09-06 10:32:50 +02:00
|
|
|
return $manifest =~ m/\.(swg|xs|c|h|i)[\n\s]/;
|
2014-07-18 15:43:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
sub find_license_files {
|
|
|
|
my ($manifest) = @_;
|
|
|
|
my @license_files;
|
|
|
|
foreach (split /\n/, $manifest) {
|
|
|
|
next if m|/|;
|
|
|
|
push @license_files, $_ if m/(ARTISTIC|COPYING|COPYRIGHT|LICENSE)/i;
|
|
|
|
}
|
2015-10-26 18:45:01 +01:00
|
|
|
if (scalar @license_files == 0 && $manifest =~ m/(README)[\n\s]/i) {
|
|
|
|
@license_files = ($1);
|
|
|
|
}
|
2018-05-12 09:03:00 +02:00
|
|
|
return @license_files;
|
2014-07-17 18:32:58 +02:00
|
|
|
}
|
2014-02-23 15:17:20 +01:00
|
|
|
|
|
|
|
sub fetch {
|
2016-03-13 18:37:29 +01:00
|
|
|
my ($name, $need_target, $need_host, $top) = @_;
|
2014-02-23 15:17:20 +01:00
|
|
|
$need_target{$name} = $need_target if $need_target;
|
|
|
|
$need_host{$name} = $need_host if $need_host;
|
2016-03-13 18:37:29 +01:00
|
|
|
unless ($dist{$name} && !$top) {
|
2014-02-23 15:17:20 +01:00
|
|
|
say qq{fetch ${name}} unless $quiet;
|
|
|
|
my $result = $mcpan->release( distribution => $name );
|
|
|
|
$dist{$name} = $result;
|
2018-05-12 09:03:00 +02:00
|
|
|
$license_files{$name} = {};
|
2017-08-20 10:03:31 +02:00
|
|
|
eval {
|
2018-05-12 09:03:00 +02:00
|
|
|
my $author = $result->{author};
|
|
|
|
my $release = $name . q{-} . $result->{version};
|
|
|
|
my $manifest = $mcpan->source( author => $author, release => $release, path => 'MANIFEST' );
|
2017-08-20 10:03:31 +02:00
|
|
|
$need_dlopen{$name} = is_xs( $manifest );
|
2018-05-12 09:03:00 +02:00
|
|
|
foreach my $fname (find_license_files( $manifest )) {
|
|
|
|
my $license = $mcpan->source( author => $author, release => $release, path => $fname );
|
|
|
|
$license_files{$name}->{$fname} = sha256_hex( $license );
|
|
|
|
}
|
2017-08-20 10:03:31 +02:00
|
|
|
};
|
|
|
|
if ($@) {
|
|
|
|
warn $@;
|
|
|
|
}
|
2014-12-09 20:32:03 +01:00
|
|
|
my %build = ();
|
|
|
|
my %runtime = ();
|
2016-03-13 18:37:28 +01:00
|
|
|
my %optional = ();
|
2014-02-23 15:17:20 +01:00
|
|
|
foreach my $dep (@{$result->{dependency}}) {
|
|
|
|
my $modname = ${$dep}{module};
|
|
|
|
next if $modname eq q{perl};
|
|
|
|
next if $modname =~ m|^Alien|;
|
|
|
|
next if $modname =~ m|^Win32|;
|
2016-03-13 18:37:29 +01:00
|
|
|
next if !($test && $top) && $modname =~ m|^Test|;
|
2014-12-03 19:05:37 +01:00
|
|
|
next if Module::CoreList::is_core( $modname, undef, $] );
|
2014-02-23 15:17:20 +01:00
|
|
|
# we could use the host Module::CoreList data, because host perl and
|
|
|
|
# target perl have the same major version
|
|
|
|
next if ${$dep}{phase} eq q{develop};
|
2018-08-20 13:16:22 +02:00
|
|
|
next if ${$dep}{phase} eq q{x_Dist_Zilla};
|
2016-03-13 18:37:29 +01:00
|
|
|
next if !($test && $top) && ${$dep}{phase} eq q{test};
|
2014-02-23 15:17:20 +01:00
|
|
|
my $distname = $mcpan->module( $modname )->{distribution};
|
|
|
|
if (${$dep}{phase} eq q{runtime}) {
|
2016-03-13 18:37:28 +01:00
|
|
|
if (${$dep}{relationship} eq q{requires}) {
|
|
|
|
$runtime{$distname} = 1;
|
|
|
|
}
|
|
|
|
else {
|
2016-03-13 18:37:29 +01:00
|
|
|
$optional{$distname} = 1 if $recommend && $top;
|
2016-03-13 18:37:28 +01:00
|
|
|
}
|
2014-02-23 15:17:20 +01:00
|
|
|
}
|
|
|
|
else { # configure, build
|
2014-12-09 20:32:03 +01:00
|
|
|
$build{$distname} = 1;
|
2018-08-13 18:15:52 +02:00
|
|
|
push @info, qq{[$name] suspicious dependency on $distname}
|
|
|
|
unless exists $white_list{$distname};
|
2014-02-23 15:17:20 +01:00
|
|
|
}
|
|
|
|
}
|
2014-12-09 20:32:03 +01:00
|
|
|
$deps_build{$name} = [keys %build];
|
|
|
|
$deps_runtime{$name} = [keys %runtime];
|
2016-03-13 18:37:28 +01:00
|
|
|
$deps_optional{$name} = [keys %optional];
|
2014-12-09 20:32:03 +01:00
|
|
|
foreach my $distname (@{$deps_build{$name}}) {
|
2014-07-17 18:32:58 +02:00
|
|
|
fetch( $distname, 0, 1 );
|
|
|
|
}
|
2014-12-09 20:32:03 +01:00
|
|
|
foreach my $distname (@{$deps_runtime{$name}}) {
|
2014-07-17 18:32:58 +02:00
|
|
|
fetch( $distname, $need_target, $need_host );
|
|
|
|
$need_dlopen{$name} ||= $need_dlopen{$distname};
|
|
|
|
}
|
2016-03-13 18:37:28 +01:00
|
|
|
foreach my $distname (@{$deps_optional{$name}}) {
|
|
|
|
fetch( $distname, $need_target, $need_host );
|
|
|
|
}
|
2014-02-23 15:17:20 +01:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach my $distname (@ARGV) {
|
2014-04-04 19:10:33 +02:00
|
|
|
# Command-line's distributions
|
2016-03-13 18:37:29 +01:00
|
|
|
fetch( $distname, !!$target, !!$host, 1 );
|
2014-02-23 15:17:20 +01:00
|
|
|
}
|
|
|
|
say scalar keys %dist, q{ packages fetched.} unless $quiet;
|
|
|
|
|
|
|
|
# Buildroot package name: lowercase
|
|
|
|
sub fsname {
|
|
|
|
my $name = shift;
|
2015-07-23 18:47:16 +02:00
|
|
|
$name =~ s|_|-|g;
|
2014-02-23 15:17:20 +01:00
|
|
|
return q{perl-} . lc $name;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Buildroot variable name: uppercase
|
|
|
|
sub brname {
|
|
|
|
my $name = shift;
|
|
|
|
$name =~ s|-|_|g;
|
|
|
|
return uc $name;
|
|
|
|
}
|
|
|
|
|
2018-09-22 08:53:23 +02:00
|
|
|
# Buildroot requires license name as in http://spdx.org/licenses/
|
|
|
|
sub brlicense {
|
|
|
|
my $license = shift;
|
|
|
|
$license =~ s|apache_1_1|Apache-1.1|;
|
|
|
|
$license =~ s|apache_2_0|Apache-2.0|;
|
|
|
|
$license =~ s|artistic_2|Artistic-2.0|;
|
|
|
|
$license =~ s|artistic|Artistic-1.0|;
|
|
|
|
$license =~ s|lgpl_2_1|LGPL-2.1|;
|
|
|
|
$license =~ s|lgpl_3_0|LGPL-3.0|;
|
|
|
|
$license =~ s|gpl_2|GPL-2.0|;
|
|
|
|
$license =~ s|gpl_3|GPL-3.0|;
|
|
|
|
$license =~ s|mit|MIT|;
|
|
|
|
$license =~ s|mozilla_1_1|Mozilla-1.1|;
|
|
|
|
$license =~ s|openssl|OpenSSL|;
|
|
|
|
$license =~ s|perl_5|Artistic or GPL-1.0+|;
|
|
|
|
return $license;
|
|
|
|
}
|
|
|
|
|
2014-02-23 15:17:20 +01:00
|
|
|
while (my ($distname, $dist) = each %dist) {
|
|
|
|
my $fsname = fsname( $distname );
|
|
|
|
my $dirname = q{package/} . $fsname;
|
|
|
|
my $cfgname = $dirname . q{/Config.in};
|
|
|
|
my $mkname = $dirname . q{/} . $fsname . q{.mk};
|
2014-11-20 20:35:27 +01:00
|
|
|
my $hashname = $dirname . q{/} . $fsname . q{.hash};
|
2014-02-23 15:17:20 +01:00
|
|
|
my $brname = brname( $fsname );
|
|
|
|
mkdir $dirname unless -d $dirname;
|
|
|
|
if ($need_target{$distname} && ($force || !-f $cfgname)) {
|
2018-10-10 08:46:09 +02:00
|
|
|
my $abstract = wrap( q{}, qq{\t }, $dist->{abstract} );
|
2014-06-16 21:35:56 +02:00
|
|
|
my $homepage = $dist->{resources}->{homepage} || qq{https://metacpan.org/release/${distname}};
|
2014-02-23 15:17:20 +01:00
|
|
|
say qq{write ${cfgname}} unless $quiet;
|
|
|
|
open my $fh, q{>}, $cfgname;
|
|
|
|
say {$fh} qq{config BR2_PACKAGE_${brname}};
|
|
|
|
say {$fh} qq{\tbool "${fsname}"};
|
2014-12-03 22:41:29 +01:00
|
|
|
say {$fh} qq{\tdepends on !BR2_STATIC_LIBS} if $need_dlopen{$distname};
|
2014-05-18 16:09:35 +02:00
|
|
|
foreach my $dep (sort @{$deps_runtime{$distname}}) {
|
2014-02-23 15:17:20 +01:00
|
|
|
my $brdep = brname( fsname( $dep ) );
|
|
|
|
say {$fh} qq{\tselect BR2_PACKAGE_${brdep}};
|
|
|
|
}
|
2014-06-16 21:35:56 +02:00
|
|
|
say {$fh} qq{\thelp};
|
|
|
|
say {$fh} qq{\t ${abstract}\n} if $abstract;
|
|
|
|
say {$fh} qq{\t ${homepage}};
|
2014-07-17 18:32:58 +02:00
|
|
|
if ($need_dlopen{$distname}) {
|
|
|
|
say {$fh} qq{\ncomment "${fsname} needs a toolchain w/ dynamic library"};
|
2014-12-03 22:41:29 +01:00
|
|
|
say {$fh} qq{\tdepends on BR2_STATIC_LIBS};
|
2014-07-17 18:32:58 +02:00
|
|
|
}
|
2014-02-23 15:17:20 +01:00
|
|
|
close $fh;
|
|
|
|
}
|
|
|
|
if ($force || !-f $mkname) {
|
|
|
|
my $version = $dist->{version};
|
2017-09-27 10:32:50 +02:00
|
|
|
my ($path) = $dist->{download_url} =~ m|^[^:/?#]+://[^/?#]*([^?#]*)|;
|
2014-02-23 15:17:20 +01:00
|
|
|
# this URL contains only the scheme, auth and path parts (but no query and fragment parts)
|
|
|
|
# the scheme is not used, because the job is done by the BR download infrastructure
|
|
|
|
# the auth part is not used, because we use $(BR2_CPAN_MIRROR)
|
2017-09-27 10:32:50 +02:00
|
|
|
my ($filename, $directories, $suffix) = fileparse( $path, q{tar.gz}, q{tgz} );
|
2014-08-29 10:51:53 +02:00
|
|
|
$directories =~ s|/$||;
|
2015-09-04 10:31:13 +02:00
|
|
|
my $dependencies = join q{ }, map( { q{host-} . fsname( $_ ); } sort @{$deps_build{$distname}} ),
|
2014-05-18 16:09:35 +02:00
|
|
|
map( { fsname( $_ ); } sort @{$deps_runtime{$distname}} );
|
2015-09-04 10:31:13 +02:00
|
|
|
my $host_dependencies = join q{ }, map { q{host-} . fsname( $_ ); } sort( @{$deps_build{$distname}},
|
2014-05-18 16:09:35 +02:00
|
|
|
@{$deps_runtime{$distname}} );
|
2018-09-22 08:53:23 +02:00
|
|
|
my $license = brlicense( ref $dist->{license} eq 'ARRAY'
|
|
|
|
? join q{ or }, @{$dist->{license}}
|
|
|
|
: $dist->{license} );
|
2018-05-12 09:03:00 +02:00
|
|
|
my $license_files = join q{ }, keys %{$license_files{$distname}};
|
2018-08-13 18:15:53 +02:00
|
|
|
if ($license_files && (!$license || $license eq q{unknown})) {
|
|
|
|
push @info, qq{[$distname] undefined LICENSE, see $license_files};
|
|
|
|
$license = q{???};
|
|
|
|
}
|
2014-02-23 15:17:20 +01:00
|
|
|
say qq{write ${mkname}} unless $quiet;
|
|
|
|
open my $fh, q{>}, $mkname;
|
|
|
|
say {$fh} qq{################################################################################};
|
|
|
|
say {$fh} qq{#};
|
|
|
|
say {$fh} qq{# ${fsname}};
|
|
|
|
say {$fh} qq{#};
|
|
|
|
say {$fh} qq{################################################################################};
|
|
|
|
say {$fh} qq{};
|
|
|
|
say {$fh} qq{${brname}_VERSION = ${version}};
|
|
|
|
say {$fh} qq{${brname}_SOURCE = ${distname}-\$(${brname}_VERSION).${suffix}};
|
|
|
|
say {$fh} qq{${brname}_SITE = \$(BR2_CPAN_MIRROR)${directories}};
|
2015-09-04 10:31:13 +02:00
|
|
|
say {$fh} qq{${brname}_DEPENDENCIES = ${dependencies}} if $need_target{$distname} && $dependencies;
|
|
|
|
say {$fh} qq{HOST_${brname}_DEPENDENCIES = ${host_dependencies}} if $need_host{$distname} && $host_dependencies;
|
2018-08-13 18:15:53 +02:00
|
|
|
say {$fh} qq{${brname}_LICENSE = ${license}} if $license;
|
2014-07-18 15:43:36 +02:00
|
|
|
say {$fh} qq{${brname}_LICENSE_FILES = ${license_files}} if $license_files;
|
2014-02-23 15:17:20 +01:00
|
|
|
say {$fh} qq{};
|
2016-03-13 18:37:28 +01:00
|
|
|
foreach (sort @{$deps_optional{$distname}}) {
|
|
|
|
next if grep { $_ eq $distname; } @{$deps_runtime{$_}}; # avoid cyclic dependencies
|
|
|
|
my $opt_brname = brname( $_ );
|
|
|
|
my $opt_fsname = fsname( $_ );
|
|
|
|
say {$fh} qq{ifeq (\$(BR2_PACKAGE_PERL_${opt_brname}),y)};
|
|
|
|
say {$fh} qq{${brname}_DEPENDENCIES += ${opt_fsname}};
|
|
|
|
say {$fh} qq{endif};
|
|
|
|
say {$fh} qq{};
|
|
|
|
}
|
2014-02-23 15:17:20 +01:00
|
|
|
say {$fh} qq{\$(eval \$(perl-package))} if $need_target{$distname};
|
|
|
|
say {$fh} qq{\$(eval \$(host-perl-package))} if $need_host{$distname};
|
|
|
|
close $fh;
|
|
|
|
}
|
2014-11-20 20:35:27 +01:00
|
|
|
if ($force || !-f $hashname) {
|
2017-09-27 10:32:50 +02:00
|
|
|
my ($checksum, $filename) = get_checksum($dist->{download_url});
|
2014-11-20 20:35:27 +01:00
|
|
|
my $md5 = $checksum->{md5};
|
|
|
|
my $sha256 = $checksum->{sha256};
|
|
|
|
say qq{write ${hashname}} unless $quiet;
|
|
|
|
open my $fh, q{>}, $hashname;
|
2017-06-14 16:15:31 +02:00
|
|
|
say {$fh} qq{# retrieved by scancpan from ${mirror}/};
|
2014-11-20 20:35:27 +01:00
|
|
|
say {$fh} qq{md5 ${md5} ${filename}};
|
|
|
|
say {$fh} qq{sha256 ${sha256} ${filename}};
|
2018-05-12 09:03:00 +02:00
|
|
|
if (scalar keys %{$license_files{$distname}}) {
|
|
|
|
say {$fh} q{};
|
|
|
|
say {$fh} qq{# computed by scancpan};
|
|
|
|
while (my ($license, $digest) = each %{$license_files{$distname}}) {
|
|
|
|
say {$fh} qq{sha256 ${digest} ${license}};
|
|
|
|
}
|
|
|
|
}
|
2014-11-20 20:35:27 +01:00
|
|
|
close $fh;
|
|
|
|
}
|
2014-02-23 15:17:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
my %pkg;
|
|
|
|
my $cfgname = q{package/Config.in};
|
|
|
|
if (-f $cfgname) {
|
|
|
|
open my $fh, q{<}, $cfgname;
|
|
|
|
while (<$fh>) {
|
|
|
|
chomp;
|
|
|
|
$pkg{$_} = 1 if m|package/perl-|;
|
|
|
|
}
|
|
|
|
close $fh;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach my $distname (keys %need_target) {
|
|
|
|
my $fsname = fsname( $distname );
|
2014-06-23 20:14:44 +02:00
|
|
|
$pkg{qq{\tsource "package/${fsname}/Config.in"}} = 1;
|
2014-02-23 15:17:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
say qq{${cfgname} must contain the following lines:};
|
|
|
|
say join qq{\n}, sort keys %pkg;
|
|
|
|
|
2018-08-13 18:15:52 +02:00
|
|
|
say join qq{\n}, @info;
|
|
|
|
|
2014-02-23 15:17:20 +01:00
|
|
|
__END__
|
|
|
|
|
|
|
|
=head1 NAME
|
|
|
|
|
2017-07-01 18:40:41 +02:00
|
|
|
utils/scancpan Try-Tiny Moo
|
2014-02-23 15:17:20 +01:00
|
|
|
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
|
|
|
|
supports/scripts/scancpan [options] [distname ...]
|
|
|
|
|
|
|
|
Options:
|
|
|
|
-help
|
|
|
|
-man
|
|
|
|
-quiet
|
|
|
|
-force
|
2014-04-04 19:10:33 +02:00
|
|
|
-target/-notarget
|
|
|
|
-host/-nohost
|
2014-02-23 15:17:20 +01:00
|
|
|
-recommend
|
2014-10-01 20:36:59 +02:00
|
|
|
-test
|
2014-02-23 15:17:20 +01:00
|
|
|
|
|
|
|
=head1 OPTIONS
|
|
|
|
|
|
|
|
=over 8
|
|
|
|
|
|
|
|
=item B<-help>
|
|
|
|
|
|
|
|
Prints a brief help message and exits.
|
|
|
|
|
|
|
|
=item B<-man>
|
|
|
|
|
|
|
|
Prints the manual page and exits.
|
|
|
|
|
|
|
|
=item B<-quiet>
|
|
|
|
|
|
|
|
Executes without output
|
|
|
|
|
|
|
|
=item B<-force>
|
|
|
|
|
|
|
|
Forces the overwriting of existing files.
|
|
|
|
|
2014-04-04 19:10:33 +02:00
|
|
|
=item B<-target/-notarget>
|
|
|
|
|
|
|
|
Switches package generation for the target variant (the default is C<-target>).
|
|
|
|
|
|
|
|
=item B<-host/-nohost>
|
|
|
|
|
|
|
|
Switches package generation for the host variant (the default is C<-nohost>).
|
|
|
|
|
2014-02-23 15:17:20 +01:00
|
|
|
=item B<-recommend>
|
|
|
|
|
|
|
|
Adds I<recommended> dependencies.
|
|
|
|
|
2014-10-01 20:36:59 +02:00
|
|
|
=item B<-test>
|
|
|
|
|
|
|
|
Adds dependencies for test.
|
|
|
|
|
2014-02-23 15:17:20 +01:00
|
|
|
=back
|
|
|
|
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
|
|
|
|
This script creates templates of the Buildroot package files for all the
|
|
|
|
Perl/CPAN distributions required by the specified distnames. The
|
|
|
|
dependencies and metadata are fetched from https://metacpan.org/.
|
|
|
|
|
|
|
|
After running this script, it is necessary to check the generated files.
|
2014-04-04 19:10:33 +02:00
|
|
|
For distributions that link against a target library, you have to add the
|
2014-02-23 15:17:20 +01:00
|
|
|
buildroot package name for that library to the DEPENDENCIES variable.
|
|
|
|
|
|
|
|
See the Buildroot documentation for details on the usage of the Perl
|
|
|
|
infrastructure.
|
|
|
|
|
|
|
|
The major version of the host perl must be aligned on the target one,
|
|
|
|
in order to work with the right CoreList data.
|
|
|
|
|
|
|
|
=head1 LICENSE
|
|
|
|
|
2018-05-12 09:03:00 +02:00
|
|
|
Copyright (C) 2013-2018 by Francois Perrad <francois.perrad@gadz.org>
|
2014-02-23 15:17:20 +01:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
|
|
|
|
This script is a part of Buildroot.
|
|
|
|
|
|
|
|
This script requires the module C<MetaCPAN::API::Tiny> (version 1.131730)
|
|
|
|
which was included at the beginning of this file by the tool C<fatpack>.
|
|
|
|
|
|
|
|
See L<http://search.cpan.org/~nperez/MetaCPAN-API-Tiny-1.131730/>.
|
|
|
|
|
|
|
|
See L<http://search.cpan.org/search?query=App-FatPacker&mode=dist>.
|
|
|
|
|
|
|
|
These both libraries are free software and may be distributed under the same
|
|
|
|
terms as perl itself.
|
|
|
|
|
|
|
|
And perl may be distributed under the terms of Artistic v1 or GPL v1 license.
|
|
|
|
|
|
|
|
=cut
|