Skip to content

Commit

Permalink
Improvments and fixes
Browse files Browse the repository at this point in the history
- Improved validation during encode and decode "purl" string
- Fixed CPAN repository URL
- FIX Qualifiers are case sensitive (#4)
- FIX PURLs containing multiple namespaces segments parse incorrectly (#5)
- FIX Incorrect parsing of PURLs that begin with "pkg:/" (#6)
- Improved "t/99-official-purl-test-suite.t" test
  • Loading branch information
giterlizzi committed Nov 9, 2023
1 parent 99ef062 commit b7116b0
Show file tree
Hide file tree
Showing 7 changed files with 180 additions and 42 deletions.
11 changes: 11 additions & 0 deletions Changes
Original file line number Diff line number Diff line change
@@ -1,5 +1,16 @@
Revision history for Perl extension URI::PackageURL.

2.03 2023-11-09
- Improved validation during encode and decode "purl" string
- Fixed CPAN repository URL
- FIX Qualifiers are case sensitive
(giterlizzi/perl-URI-PackageURL#4)
- FIX PURLs containing multiple namespaces segments parse incorrectly
(giterlizzi/perl-URI-PackageURL#5)
- FIX Incorrect parsing of PURLs that begin with "pkg:/"
(giterlizzi/perl-URI-PackageURL#6)
- Improved "t/99-official-purl-test-suite.t" test

2.02 2023-09-22
- Added core "JSON" module prerequisite in Makefile.PL (#4)

Expand Down
58 changes: 53 additions & 5 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -10,19 +10,66 @@ use URI::PackageURL;
# OO-interface

# Encode components in PackageURL string
$purl = URI::PackageURL->new(type => cpan, namespace => 'GDT', name => 'URI-PackageURL', version => '2.02');
$purl = URI::PackageURL->new(type => cpan, namespace => 'GDT', name => 'URI-PackageURL', version => '2.03');

say $purl; # pkg:cpan/GDT/URI-PackageURL@2.02
say $purl; # pkg:cpan/GDT/URI-PackageURL@2.03

# Parse PackageURL string
$purl = URI::PackageURL->from_string('pkg:cpan/GDT/URI-PackageURL@2.02');
$purl = URI::PackageURL->from_string('pkg:cpan/GDT/URI-PackageURL@2.03');

# exported funtions

$purl = decode_purl('pkg:cpan/GDT/URI-PackageURL@2.02');
$purl = decode_purl('pkg:cpan/GDT/URI-PackageURL@2.03');
say $purl->type; # cpan

$purl_string = encode_purl(type => cpan, namespace => 'GDT', name => 'URI::PackageURL', version => '2.02');
$purl_string = encode_purl(type => cpan, namespace => 'GDT', name => 'URI::PackageURL', version => '2.03');
```


## purl-tool a CLI for URI::PackageURL module

Inspect and export "purl" string in various formats (JSON, YAML, Data::Dumper, ENV):

```console
$ purl-tool pkg:cpan/GDT/URI-PackageURL@2.03 --json | jq
{
"name": "URI-PackageURL",
"namespace": "GDT",
"qualifiers": {},
"subpath": null,
"type": "cpan",
"version": "2.03"
}
```


Download package using "purl" string:

```console
$ wget $(purl-tool pkg:cpan/GDT/URI-PackageURL@2.03 --download-url)
```


Use "purl" string in your shell-scripts:

```.bash
#!bash

set -e

PURL="pkg:cpan/GDT/URI-PackageURL@2.03"

eval $(purl-tool "$PURL" --env)

echo "Download $PURL_NAME $PURL_VERSION"
wget $PURL_DOWNLOAD_URL

echo "Build and install module $PURL_NAME $PURL_VERSION"
tar xvf $PURL_NAME-$PURL_VERSION.tar.gz

cd $PURL_NAME-$PURL_VERSION
perl Makefile.PL
make && make install
```

## Install
Expand All @@ -45,6 +92,7 @@ Using App::cpanminus:

- `perldoc URI::PackageURL`
- https://metacpan.org/release/URI-PackageURL
- /~https://github.com/package-url/purl-spec


## Copyright
Expand Down
6 changes: 3 additions & 3 deletions bin/purl-tool
Original file line number Diff line number Diff line change
Expand Up @@ -40,9 +40,9 @@ purl-tool - PackageURL tool
Examples:
purl-tool pkg:cpan/GDT/URI-PackageURL@2.02 --json | jq
purl-tool pkg:cpan/GDT/URI-PackageURL@2.03 --json | jq
wget $(purl-tool pkg:cpan/GDT/URI-PackageURL@2.02 --download-url)
wget $(purl-tool pkg:cpan/GDT/URI-PackageURL@2.03 --download-url)
=head1 DESCRIPTION
Expand All @@ -53,7 +53,7 @@ C<purl-tool> PackageURL tool
Parse the given PackageURL string and return JSON and send the STDOUT to L<jq>:
B<purl-tool pkg:cpan/GDT/URI-PackageURL@2.02 --json | jq>
B<purl-tool pkg:cpan/GDT/URI-PackageURL@2.03 --json | jq>
Download the package from the repository using PackageURL string:
Expand Down
71 changes: 56 additions & 15 deletions lib/URI/PackageURL.pm
Original file line number Diff line number Diff line change
Expand Up @@ -9,9 +9,11 @@ use Carp;
use Exporter qw(import);
use URI::PackageURL::Util qw(purl_to_urls);

use constant PURL_DEBUG => $ENV{PURL_DEBUG};

use overload '""' => 'to_string', fallback => 1;

our $VERSION = '2.02';
our $VERSION = '2.03';
our @EXPORT = qw(encode_purl decode_purl);

my $PURL_REGEXP = qr{^pkg:[A-Za-z\\.\\-\\+][A-Za-z0-9\\.\\-\\+]*/.+};
Expand All @@ -28,6 +30,8 @@ sub new {
my $qualifiers = delete $params{qualifiers} // {};
my $subpath = delete $params{subpath};

Carp::croak "Invalid PackageURL: '$scheme' is not a valid scheme" if (!$scheme eq 'pkg');

$type = lc $type;

if (grep { $_ eq $type } qw(alpm apk bitbucket composer deb github gitlab hex npm oci pypi)) {
Expand All @@ -42,6 +46,7 @@ sub new {

foreach my $qualifier (keys %{$qualifiers}) {
Carp::croak "Invalid PackageURL: '$qualifier' is not a valid qualifier" if ($qualifier =~ /\s/);
Carp::croak "Invalid PackageURL: '$qualifier' is not a valid qualifier" if ($qualifier =~ /\%/);
}

$name =~ s/_/-/g if $type eq 'pypi';
Expand Down Expand Up @@ -71,6 +76,25 @@ sub new {

}

if ($type eq 'mlflow') {

# The "name" case sensitivity depends on the server implementation:
# - Azure ML: it is case sensitive and must be kept as-is in the package URL.
# - Databricks: it is case insensitive and must be lowercased in the package URL.

if (defined $qualifiers->{repository_url} && $qualifiers->{repository_url} =~ /azuredatabricks/) {
$name = lc $name;
}

}

if ($type eq 'huggingface') {

# The version is the model revision Git commit hash. It is case insensitive and must be lowercased in the package URL.
$version = lc $version;

}

my $self = {
scheme => $scheme,
type => $type,
Expand Down Expand Up @@ -105,6 +129,11 @@ sub from_string {

my ($class, $string) = @_;

# Strip slash / after scheme
while ($string =~ m|^pkg:/|) {
$string =~ s|^pkg:/|pkg:|;
}

if ($string !~ /$PURL_REGEXP/) {
Carp::croak 'Malformed PackageURL string';
}
Expand Down Expand Up @@ -158,7 +187,7 @@ sub from_string {
$value = [split(',', $value)];
}

$components{qualifiers}->{$key} = $value;
$components{qualifiers}->{lc $key} = $value;

}

Expand Down Expand Up @@ -200,8 +229,8 @@ sub from_string {
# Apply type-specific normalization to the name if needed
# This is the name

my @s6 = split('/', $s5[0], 2);
$components{name} = (scalar @s6 > 1) ? _url_decode($s6[1]) : _url_decode($s6[0]);
my @s6 = split('/', $s5[0], -1);
$components{name} = _url_decode(pop @s6);


# Split the remainder on '/'
Expand All @@ -212,9 +241,17 @@ sub from_string {
# Join segments back with a '/'
# This is the namespace

if (scalar @s6 > 1) {
my @s7 = split('/', $s6[0]);
$components{namespace} = join '/', map { _url_decode($_) } @s7;
if (@s6) {
$components{namespace} = join '/', map { _url_decode($_) } @s6;
}

if (PURL_DEBUG) {
say STDERR "-- S1: @s1";
say STDERR "-- S2: @s2";
say STDERR "-- S3: @s3";
say STDERR "-- S4: @s4";
say STDERR "-- S5: @s5";
say STDERR "-- S6: @s6";
}

return $class->new(%components);
Expand Down Expand Up @@ -244,7 +281,7 @@ sub to_string {
# Qualifiers
if (my $qualifiers = $self->qualifiers) {

my @qualifiers = map { sprintf('%s=%s', $_, _url_encode($qualifiers->{$_})) } sort keys %{$qualifiers};
my @qualifiers = map { sprintf('%s=%s', lc $_, _url_encode($qualifiers->{$_})) } sort keys %{$qualifiers};
push @purl, ('?', join('&', @qualifiers)) if (@qualifiers);

}
Expand Down Expand Up @@ -286,9 +323,13 @@ sub _url_encode {
}

sub _url_decode {

my $string = shift;
return unless $string;

$string =~ s/%([0-9a-fA-F]{2})/chr hex $1/ge;
return $string;

}

1;
Expand All @@ -309,21 +350,21 @@ URI::PackageURL - Perl extension for Package URL (aka "purl")
type => cpan,
namespace => 'GDT',
name => 'URI-PackageURL',
version => '2.02'
version => '2.03'
);
say $purl; # pkg:cpan/GDT/URI-PackageURL@2.02
say $purl; # pkg:cpan/GDT/URI-PackageURL@2.03
# Parse PackageURL string
$purl = URI::PackageURL->from_string('pkg:cpan/URI-PackageURL@2.02');
$purl = URI::PackageURL->from_string('pkg:cpan/URI-PackageURL@2.03');
# exported funtions
$purl = decode_purl('pkg:cpan/GDT/URI-PackageURL@2.02');
$purl = decode_purl('pkg:cpan/GDT/URI-PackageURL@2.03');
say $purl->type; # cpan
$purl_string = encode_purl(type => cpan, name => 'URI-PackageURL', version => '2.02');
say $purl_string; # pkg:cpan/URI-PackageURL@2.02
$purl_string = encode_purl(type => cpan, name => 'URI-PackageURL', version => '2.03');
say $purl_string; # pkg:cpan/URI-PackageURL@2.03
=head1 DESCRIPTION
Expand Down Expand Up @@ -439,7 +480,7 @@ Helper method for JSON modules (L<JSON>, L<JSON::PP>, L<JSON::XS>, L<Mojo::JSON>
use Mojo::JSON qw(encode_json);
say encode_json($purl); # {"name":"URI-PackageURL","namespace":"GDT","qualifiers":null,"subpath":null,"type":"cpan","version":"2.02"}
say encode_json($purl); # {"name":"URI-PackageURL","namespace":"GDT","qualifiers":null,"subpath":null,"type":"cpan","version":"2.03"}
=back
Expand Down
2 changes: 1 addition & 1 deletion lib/URI/PackageURL/CLI.pm
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ use Data::Dumper;

use URI::PackageURL;

our $VERSION = '2.02';
our $VERSION = '2.03';

sub cli_error {
my ($error) = @_;
Expand Down
4 changes: 2 additions & 2 deletions lib/URI/PackageURL/Util.pm
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ use warnings;
use Carp;
use Exporter qw(import);

our $VERSION = '2.02';
our $VERSION = '2.03';
our @EXPORT = qw(purl_to_urls);

sub purl_to_urls {
Expand Down Expand Up @@ -206,7 +206,7 @@ sub _cpan_urls {

$name =~ s/\:\:/-/g; # TODO

my $urls = {repository => "https://metacpan.org/pod/$name"};
my $urls = {repository => "https://metacpan.org/dist/$name"};

if ($name && $version && $author) {

Expand Down
Loading

0 comments on commit b7116b0

Please sign in to comment.