init III
This commit is contained in:
173
Perl OTRS/Kernel/cpan-lib/Module/CPANfile/Environment.pm
Normal file
173
Perl OTRS/Kernel/cpan-lib/Module/CPANfile/Environment.pm
Normal file
@@ -0,0 +1,173 @@
|
||||
package Module::CPANfile::Environment;
|
||||
use strict;
|
||||
use warnings;
|
||||
use Module::CPANfile::Prereqs;
|
||||
use Carp ();
|
||||
|
||||
my @bindings = qw(
|
||||
on requires recommends suggests conflicts
|
||||
feature
|
||||
osname
|
||||
mirror
|
||||
configure_requires build_requires test_requires author_requires
|
||||
);
|
||||
|
||||
my $file_id = 1;
|
||||
|
||||
sub new {
|
||||
my($class, $file) = @_;
|
||||
bless {
|
||||
file => $file,
|
||||
phase => 'runtime', # default phase
|
||||
feature => undef,
|
||||
features => {},
|
||||
prereqs => Module::CPANfile::Prereqs->new,
|
||||
mirrors => [],
|
||||
}, $class;
|
||||
}
|
||||
|
||||
sub bind {
|
||||
my $self = shift;
|
||||
my $pkg = caller;
|
||||
|
||||
for my $binding (@bindings) {
|
||||
no strict 'refs';
|
||||
*{"$pkg\::$binding"} = sub { $self->$binding(@_) };
|
||||
}
|
||||
}
|
||||
|
||||
sub parse {
|
||||
my($self, $code) = @_;
|
||||
|
||||
my $err;
|
||||
{
|
||||
local $@;
|
||||
$file_id++;
|
||||
$self->_evaluate(<<EVAL);
|
||||
package Module::CPANfile::Sandbox$file_id;
|
||||
no warnings;
|
||||
BEGIN { \$_environment->bind }
|
||||
|
||||
# line 1 "$self->{file}"
|
||||
$code;
|
||||
EVAL
|
||||
$err = $@;
|
||||
}
|
||||
|
||||
if ($err) { die "Parsing $self->{file} failed: $err" };
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
sub _evaluate {
|
||||
my $_environment = $_[0];
|
||||
eval $_[1];
|
||||
}
|
||||
|
||||
sub prereqs { $_[0]->{prereqs} }
|
||||
|
||||
sub mirrors { $_[0]->{mirrors} }
|
||||
|
||||
# DSL goes from here
|
||||
|
||||
sub on {
|
||||
my($self, $phase, $code) = @_;
|
||||
local $self->{phase} = $phase;
|
||||
$code->();
|
||||
}
|
||||
|
||||
sub feature {
|
||||
my($self, $identifier, $description, $code) = @_;
|
||||
|
||||
# shortcut: feature identifier => sub { ... }
|
||||
if (@_ == 3 && ref($description) eq 'CODE') {
|
||||
$code = $description;
|
||||
$description = $identifier;
|
||||
}
|
||||
|
||||
unless (ref $description eq '' && ref $code eq 'CODE') {
|
||||
Carp::croak("Usage: feature 'identifier', 'Description' => sub { ... }");
|
||||
}
|
||||
|
||||
local $self->{feature} = $identifier;
|
||||
$self->prereqs->add_feature($identifier, $description);
|
||||
|
||||
$code->();
|
||||
}
|
||||
|
||||
sub osname { die "TODO" }
|
||||
|
||||
sub mirror {
|
||||
my($self, $url) = @_;
|
||||
push @{$self->{mirrors}}, $url;
|
||||
}
|
||||
|
||||
sub requirement_for {
|
||||
my($self, $module, @args) = @_;
|
||||
|
||||
my $requirement = 0;
|
||||
$requirement = shift @args if @args % 2;
|
||||
|
||||
return Module::CPANfile::Requirement->new(
|
||||
name => $module,
|
||||
version => $requirement,
|
||||
@args,
|
||||
);
|
||||
}
|
||||
|
||||
sub requires {
|
||||
my $self = shift;
|
||||
$self->add_prereq(requires => @_);
|
||||
}
|
||||
|
||||
sub recommends {
|
||||
my $self = shift;
|
||||
$self->add_prereq(recommends => @_);
|
||||
}
|
||||
|
||||
sub suggests {
|
||||
my $self = shift;
|
||||
$self->add_prereq(suggests => @_);
|
||||
}
|
||||
|
||||
sub conflicts {
|
||||
my $self = shift;
|
||||
$self->add_prereq(conflicts => @_);
|
||||
}
|
||||
|
||||
sub add_prereq {
|
||||
my($self, $type, $module, @args) = @_;
|
||||
|
||||
$self->prereqs->add(
|
||||
feature => $self->{feature},
|
||||
phase => $self->{phase},
|
||||
type => $type,
|
||||
module => $module,
|
||||
requirement => $self->requirement_for($module, @args),
|
||||
);
|
||||
}
|
||||
|
||||
# Module::Install compatible shortcuts
|
||||
|
||||
sub configure_requires {
|
||||
my($self, @args) = @_;
|
||||
$self->on(configure => sub { $self->requires(@args) });
|
||||
}
|
||||
|
||||
sub build_requires {
|
||||
my($self, @args) = @_;
|
||||
$self->on(build => sub { $self->requires(@args) });
|
||||
}
|
||||
|
||||
sub test_requires {
|
||||
my($self, @args) = @_;
|
||||
$self->on(test => sub { $self->requires(@args) });
|
||||
}
|
||||
|
||||
sub author_requires {
|
||||
my($self, @args) = @_;
|
||||
$self->on(develop => sub { $self->requires(@args) });
|
||||
}
|
||||
|
||||
1;
|
||||
|
||||
15
Perl OTRS/Kernel/cpan-lib/Module/CPANfile/Prereq.pm
Normal file
15
Perl OTRS/Kernel/cpan-lib/Module/CPANfile/Prereq.pm
Normal file
@@ -0,0 +1,15 @@
|
||||
package Module::CPANfile::Prereq;
|
||||
use strict;
|
||||
|
||||
sub new {
|
||||
my($class, %options) = @_;
|
||||
bless \%options, $class;
|
||||
}
|
||||
|
||||
sub feature { $_[0]->{feature} }
|
||||
sub phase { $_[0]->{phase} }
|
||||
sub type { $_[0]->{type} }
|
||||
sub module { $_[0]->{module} }
|
||||
sub requirement { $_[0]->{requirement} }
|
||||
|
||||
1;
|
||||
118
Perl OTRS/Kernel/cpan-lib/Module/CPANfile/Prereqs.pm
Normal file
118
Perl OTRS/Kernel/cpan-lib/Module/CPANfile/Prereqs.pm
Normal file
@@ -0,0 +1,118 @@
|
||||
package Module::CPANfile::Prereqs;
|
||||
use strict;
|
||||
use Carp ();
|
||||
use CPAN::Meta::Feature;
|
||||
use Module::CPANfile::Prereq;
|
||||
|
||||
sub from_cpan_meta {
|
||||
my($class, $prereqs) = @_;
|
||||
|
||||
my $self = $class->new;
|
||||
|
||||
for my $phase (keys %$prereqs) {
|
||||
for my $type (keys %{ $prereqs->{$phase} }) {
|
||||
while (my($module, $requirement) = each %{ $prereqs->{$phase}{$type} }) {
|
||||
$self->add(
|
||||
phase => $phase,
|
||||
type => $type,
|
||||
module => $module,
|
||||
requirement => Module::CPANfile::Requirement->new(name => $module, version => $requirement),
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$self;
|
||||
}
|
||||
|
||||
sub new {
|
||||
my $class = shift;
|
||||
bless {
|
||||
prereqs => {},
|
||||
features => {},
|
||||
}, $class;
|
||||
}
|
||||
|
||||
sub add_feature {
|
||||
my($self, $identifier, $description) = @_;
|
||||
$self->{features}{$identifier} = { description => $description };
|
||||
}
|
||||
|
||||
sub add {
|
||||
my($self, %args) = @_;
|
||||
|
||||
my $feature = $args{feature} || '';
|
||||
push @{$self->{prereqs}{$feature}},
|
||||
Module::CPANfile::Prereq->new(%args);
|
||||
}
|
||||
|
||||
sub as_cpan_meta {
|
||||
my $self = shift;
|
||||
$self->{cpanmeta} ||= $self->build_cpan_meta;
|
||||
}
|
||||
|
||||
sub build_cpan_meta {
|
||||
my($self, $feature) = @_;
|
||||
CPAN::Meta::Prereqs->new($self->specs($feature));
|
||||
}
|
||||
|
||||
sub specs {
|
||||
my($self, $feature) = @_;
|
||||
|
||||
$feature = ''
|
||||
unless defined $feature;
|
||||
|
||||
my $prereqs = $self->{prereqs}{$feature} || [];
|
||||
my $specs = {};
|
||||
|
||||
for my $prereq (@$prereqs) {
|
||||
$specs->{$prereq->phase}{$prereq->type}{$prereq->module} =
|
||||
$prereq->requirement->version;
|
||||
}
|
||||
|
||||
return $specs;
|
||||
}
|
||||
|
||||
sub merged_requirements {
|
||||
my $self = shift;
|
||||
|
||||
my $reqs = CPAN::Meta::Requirements->new;
|
||||
for my $prereq (@{$self->{prereqs}}) {
|
||||
$reqs->add_string_requirement($prereq->module, $prereq->requirement->version);
|
||||
}
|
||||
|
||||
$reqs;
|
||||
}
|
||||
|
||||
sub find {
|
||||
my($self, $module) = @_;
|
||||
|
||||
for my $feature ('', keys %{$self->{features}}) {
|
||||
for my $prereq (@{$self->{prereqs}{$feature}}) {
|
||||
return $prereq if $prereq->module eq $module;
|
||||
}
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
sub identifiers {
|
||||
my $self = shift;
|
||||
keys %{$self->{features}};
|
||||
}
|
||||
|
||||
sub feature {
|
||||
my($self, $identifier) = @_;
|
||||
|
||||
my $data = $self->{features}{$identifier}
|
||||
or Carp::croak("Unknown feature '$identifier'");
|
||||
|
||||
my $prereqs = $self->build_cpan_meta($identifier);
|
||||
|
||||
CPAN::Meta::Feature->new($identifier, {
|
||||
description => $data->{description},
|
||||
prereqs => $prereqs->as_string_hash,
|
||||
});
|
||||
}
|
||||
|
||||
1;
|
||||
25
Perl OTRS/Kernel/cpan-lib/Module/CPANfile/Requirement.pm
Normal file
25
Perl OTRS/Kernel/cpan-lib/Module/CPANfile/Requirement.pm
Normal file
@@ -0,0 +1,25 @@
|
||||
package Module::CPANfile::Requirement;
|
||||
use strict;
|
||||
|
||||
sub new {
|
||||
my ($class, %args) = @_;
|
||||
|
||||
$args{version} ||= 0;
|
||||
|
||||
bless +{
|
||||
name => delete $args{name},
|
||||
version => delete $args{version},
|
||||
options => \%args,
|
||||
}, $class;
|
||||
}
|
||||
|
||||
sub name { $_[0]->{name} }
|
||||
sub version { $_[0]->{version} }
|
||||
|
||||
sub options { $_[0]->{options} }
|
||||
|
||||
sub has_options {
|
||||
keys %{$_[0]->{options}} > 0;
|
||||
}
|
||||
|
||||
1;
|
||||
Reference in New Issue
Block a user