123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687 |
- # Copyright © 2008-2011 Raphaël Hertzog <hertzog@debian.org>
- # Copyright © 2008-2015 Guillem Jover <guillem@debian.org>
- #
- # 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, see <https://www.gnu.org/licenses/>.
- package Dpkg::Source::Package;
- =encoding utf8
- =head1 NAME
- Dpkg::Source::Package - manipulate Debian source packages
- =head1 DESCRIPTION
- This module provides an object that can manipulate Debian source
- packages. While it supports both the extraction and the creation
- of source packages, the only API that is officially supported
- is the one that supports the extraction of the source package.
- =cut
- use strict;
- use warnings;
- our $VERSION = '1.02';
- our @EXPORT_OK = qw(
- get_default_diff_ignore_regex
- set_default_diff_ignore_regex
- get_default_tar_ignore_pattern
- );
- use Exporter qw(import);
- use POSIX qw(:errno_h :sys_wait_h);
- use Carp;
- use File::Basename;
- use Dpkg::Gettext;
- use Dpkg::ErrorHandling;
- use Dpkg::Control;
- use Dpkg::Checksums;
- use Dpkg::Version;
- use Dpkg::Compression;
- use Dpkg::Exit qw(run_exit_handlers);
- use Dpkg::Path qw(check_files_are_the_same find_command);
- use Dpkg::IPC;
- use Dpkg::Vendor qw(run_vendor_hook);
- my $diff_ignore_default_regex = '
- # Ignore general backup files
- (?:^|/).*~$|
- # Ignore emacs recovery files
- (?:^|/)\.#.*$|
- # Ignore vi swap files
- (?:^|/)\..*\.sw.$|
- # Ignore baz-style junk files or directories
- (?:^|/),,.*(?:$|/.*$)|
- # File-names that should be ignored (never directories)
- (?:^|/)(?:DEADJOE|\.arch-inventory|\.(?:bzr|cvs|hg|git|mtn-)ignore)$|
- # File or directory names that should be ignored
- (?:^|/)(?:CVS|RCS|\.deps|\{arch\}|\.arch-ids|\.svn|
- \.hg(?:tags|sigs)?|_darcs|\.git(?:attributes|modules|review)?|
- \.mailmap|\.shelf|_MTN|\.be|\.bzr(?:\.backup|tags)?)(?:$|/.*$)
- ';
- # Take out comments and newlines
- $diff_ignore_default_regex =~ s/^#.*$//mg;
- $diff_ignore_default_regex =~ s/\n//sg;
- # Public variables
- # XXX: Backwards compatibility, stop exporting on VERSION 2.00.
- ## no critic (Variables::ProhibitPackageVars)
- our $diff_ignore_default_regexp;
- *diff_ignore_default_regexp = \$diff_ignore_default_regex;
- no warnings 'qw'; ## no critic (TestingAndDebugging::ProhibitNoWarnings)
- our @tar_ignore_default_pattern = qw(
- *.a
- *.la
- *.o
- *.so
- .*.sw?
- */*~
- ,,*
- .[#~]*
- .arch-ids
- .arch-inventory
- .be
- .bzr
- .bzr.backup
- .bzr.tags
- .bzrignore
- .cvsignore
- .deps
- .git
- .gitattributes
- .gitignore
- .gitmodules
- .gitreview
- .hg
- .hgignore
- .hgsigs
- .hgtags
- .mailmap
- .mtn-ignore
- .shelf
- .svn
- CVS
- DEADJOE
- RCS
- _MTN
- _darcs
- {arch}
- );
- ## use critic
- =head1 FUNCTIONS
- =over 4
- =item $string = get_default_diff_ignore_regex()
- Returns the default diff ignore regex.
- =cut
- sub get_default_diff_ignore_regex {
- return $diff_ignore_default_regex;
- }
- =item set_default_diff_ignore_regex($string)
- Set a regex as the new default diff ignore regex.
- =cut
- sub set_default_diff_ignore_regex {
- my $regex = shift;
- $diff_ignore_default_regex = $regex;
- }
- =item @array = get_default_tar_ignore_pattern()
- Returns the default tar ignore pattern, as an array.
- =cut
- sub get_default_tar_ignore_pattern {
- return @tar_ignore_default_pattern;
- }
- =back
- =head1 METHODS
- =over 4
- =item $p = Dpkg::Source::Package->new(filename => $dscfile, options => {})
- Creates a new object corresponding to the source package described
- by the file $dscfile.
- The options hash supports the following options:
- =over 8
- =item skip_debianization
- If set to 1, do not apply Debian changes on the extracted source package.
- =item skip_patches
- If set to 1, do not apply Debian-specific patches. This options is
- specific for source packages using format "2.0" and "3.0 (quilt)".
- =item require_valid_signature
- If set to 1, the check_signature() method will be stricter and will error
- out if the signature can't be verified.
- =item require_strong_checksums
- If set to 1, the check_checksums() method will be stricter and will error
- out if there is no strong checksum.
- =item copy_orig_tarballs
- If set to 1, the extraction will copy the upstream tarballs next the
- target directory. This is useful if you want to be able to rebuild the
- source package after its extraction.
- =back
- =cut
- # Object methods
- sub new {
- my ($this, %args) = @_;
- my $class = ref($this) || $this;
- my $self = {
- fields => Dpkg::Control->new(type => CTRL_PKG_SRC),
- options => {},
- checksums => Dpkg::Checksums->new(),
- };
- bless $self, $class;
- if (exists $args{options}) {
- $self->{options} = $args{options};
- }
- if (exists $args{filename}) {
- $self->initialize($args{filename});
- $self->init_options();
- }
- return $self;
- }
- sub init_options {
- my $self = shift;
- # Use full ignore list by default
- # note: this function is not called by V1 packages
- $self->{options}{diff_ignore_regex} ||= $diff_ignore_default_regex;
- $self->{options}{diff_ignore_regex} .= '|(?:^|/)debian/source/local-.*$';
- if (defined $self->{options}{tar_ignore}) {
- $self->{options}{tar_ignore} = [ @tar_ignore_default_pattern ]
- unless @{$self->{options}{tar_ignore}};
- } else {
- $self->{options}{tar_ignore} = [ @tar_ignore_default_pattern ];
- }
- push @{$self->{options}{tar_ignore}}, 'debian/source/local-options',
- 'debian/source/local-patch-header';
- # Skip debianization while specific to some formats has an impact
- # on code common to all formats
- $self->{options}{skip_debianization} //= 0;
- # Set default compressor for new formats.
- $self->{options}{compression} //= 'xz';
- $self->{options}{comp_level} //= compression_get_property($self->{options}{compression},
- 'default_level');
- $self->{options}{comp_ext} //= compression_get_property($self->{options}{compression},
- 'file_ext');
- }
- sub initialize {
- my ($self, $filename) = @_;
- my ($fn, $dir) = fileparse($filename);
- error(g_('%s is not the name of a file'), $filename) unless $fn;
- $self->{basedir} = $dir || './';
- $self->{filename} = $fn;
- # Read the fields
- my $fields = Dpkg::Control->new(type => CTRL_PKG_SRC);
- $fields->load($filename);
- $self->{fields} = $fields;
- $self->{is_signed} = $fields->get_option('is_pgp_signed');
- foreach my $f (qw(Source Version Files)) {
- unless (defined($fields->{$f})) {
- error(g_('missing critical source control field %s'), $f);
- }
- }
- $self->{checksums}->add_from_control($fields, use_files_for_md5 => 1);
- $self->upgrade_object_type(0);
- }
- sub upgrade_object_type {
- my ($self, $update_format) = @_;
- $update_format //= 1;
- $self->{fields}{'Format'} //= '1.0';
- my $format = $self->{fields}{'Format'};
- if ($format =~ /^([\d\.]+)(?:\s+\((.*)\))?$/) {
- my ($version, $variant) = ($1, $2);
- if (defined $variant and $variant ne lc $variant) {
- error(g_("source package format '%s' is not supported: %s"),
- $format, g_('format variant must be in lowercase'));
- }
- my $major = $version =~ s/\.[\d\.]+$//r;
- my $minor;
- my $module = "Dpkg::Source::Package::V$major";
- $module .= '::' . ucfirst $variant if defined $variant;
- eval qq{
- pop \@INC if \$INC[-1] eq '.';
- require $module;
- \$minor = \$${module}::CURRENT_MINOR_VERSION;
- };
- $minor //= 0;
- if ($update_format) {
- $self->{fields}{'Format'} = "$major.$minor";
- $self->{fields}{'Format'} .= " ($variant)" if defined $variant;
- }
- if ($@) {
- error(g_("source package format '%s' is not supported: %s"),
- $format, $@);
- }
- bless $self, $module;
- } else {
- error(g_("invalid Format field '%s'"), $format);
- }
- }
- =item $p->get_filename()
- Returns the filename of the DSC file.
- =cut
- sub get_filename {
- my $self = shift;
- return $self->{basedir} . $self->{filename};
- }
- =item $p->get_files()
- Returns the list of files referenced by the source package. The filenames
- usually do not have any path information.
- =cut
- sub get_files {
- my $self = shift;
- return $self->{checksums}->get_files();
- }
- =item $p->check_checksums()
- Verify the checksums embedded in the DSC file. It requires the presence of
- the other files constituting the source package. If any inconsistency is
- discovered, it immediately errors out. It will make sure at least one strong
- checksum is present.
- If the object has been created with the "require_strong_checksums" option,
- then any problem will result in a fatal error.
- =cut
- sub check_checksums {
- my $self = shift;
- my $checksums = $self->{checksums};
- my $warn_on_weak = 0;
- # add_from_file verify the checksums if they are already existing
- foreach my $file ($checksums->get_files()) {
- if (not $checksums->has_strong_checksums($file)) {
- if ($self->{options}{require_strong_checksums}) {
- error(g_('source package uses only weak checksums'));
- } else {
- $warn_on_weak = 1;
- }
- }
- $checksums->add_from_file($self->{basedir} . $file, key => $file);
- }
- warning(g_('source package uses only weak checksums')) if $warn_on_weak;
- }
- sub get_basename {
- my ($self, $with_revision) = @_;
- my $f = $self->{fields};
- unless (exists $f->{'Source'} and exists $f->{'Version'}) {
- error(g_('%s and %s fields are required to compute the source basename'),
- 'Source', 'Version');
- }
- my $v = Dpkg::Version->new($f->{'Version'});
- my $vs = $v->as_string(omit_epoch => 1, omit_revision => !$with_revision);
- return $f->{'Source'} . '_' . $vs;
- }
- sub find_original_tarballs {
- my ($self, %opts) = @_;
- $opts{extension} //= compression_get_file_extension_regex();
- $opts{include_main} //= 1;
- $opts{include_supplementary} //= 1;
- my $basename = $self->get_basename();
- my @tar;
- foreach my $dir ('.', $self->{basedir}, $self->{options}{origtardir}) {
- next unless defined($dir) and -d $dir;
- opendir(my $dir_dh, $dir) or syserr(g_('cannot opendir %s'), $dir);
- push @tar, map { "$dir/$_" } grep {
- ($opts{include_main} and
- /^\Q$basename\E\.orig\.tar\.$opts{extension}$/) or
- ($opts{include_supplementary} and
- /^\Q$basename\E\.orig-[[:alnum:]-]+\.tar\.$opts{extension}$/)
- } readdir($dir_dh);
- closedir($dir_dh);
- }
- return @tar;
- }
- =item $bool = $p->is_signed()
- Returns 1 if the DSC files contains an embedded OpenPGP signature.
- Otherwise returns 0.
- =cut
- sub is_signed {
- my $self = shift;
- return $self->{is_signed};
- }
- =item $p->check_signature()
- Implement the same OpenPGP signature check that dpkg-source does.
- In case of problems, it prints a warning or errors out.
- If the object has been created with the "require_valid_signature" option,
- then any problem will result in a fatal error.
- =cut
- sub check_signature {
- my $self = shift;
- my $dsc = $self->get_filename();
- my @exec;
- if (find_command('gpgv2')) {
- push @exec, 'gpgv2';
- } elsif (find_command('gpgv')) {
- push @exec, 'gpgv';
- } elsif (find_command('gpg2')) {
- push @exec, 'gpg2', '--no-default-keyring', '-q', '--verify';
- } elsif (find_command('gpg')) {
- push @exec, 'gpg', '--no-default-keyring', '-q', '--verify';
- }
- if (scalar(@exec)) {
- if (length $ENV{HOME} and -r "$ENV{HOME}/.gnupg/trustedkeys.gpg") {
- push @exec, '--keyring', "$ENV{HOME}/.gnupg/trustedkeys.gpg";
- }
- foreach my $vendor_keyring (run_vendor_hook('package-keyrings')) {
- if (-r $vendor_keyring) {
- push @exec, '--keyring', $vendor_keyring;
- }
- }
- push @exec, $dsc;
- my ($stdout, $stderr);
- spawn(exec => \@exec, wait_child => 1, nocheck => 1,
- to_string => \$stdout, error_to_string => \$stderr,
- timeout => 10);
- if (WIFEXITED($?)) {
- my $gpg_status = WEXITSTATUS($?);
- print { *STDERR } "$stdout$stderr" if $gpg_status;
- if ($gpg_status == 1 or ($gpg_status &&
- $self->{options}{require_valid_signature}))
- {
- error(g_('failed to verify signature on %s'), $dsc);
- } elsif ($gpg_status) {
- warning(g_('failed to verify signature on %s'), $dsc);
- }
- } else {
- subprocerr("@exec");
- }
- } else {
- if ($self->{options}{require_valid_signature}) {
- error(g_('cannot verify signature on %s since GnuPG is not installed'), $dsc);
- } else {
- warning(g_('cannot verify signature on %s since GnuPG is not installed'), $dsc);
- }
- }
- }
- sub describe_cmdline_options {
- return;
- }
- sub parse_cmdline_options {
- my ($self, @opts) = @_;
- foreach my $option (@opts) {
- if (not $self->parse_cmdline_option($option)) {
- warning(g_('%s is not a valid option for %s'), $option, ref $self);
- }
- }
- }
- sub parse_cmdline_option {
- return 0;
- }
- =item $p->extract($targetdir)
- Extracts the source package in the target directory $targetdir. Beware
- that if $targetdir already exists, it will be erased (as long as the
- no_overwrite_dir option is set).
- =cut
- sub extract {
- my ($self, $newdirectory) = @_;
- my ($ok, $error) = version_check($self->{fields}{'Version'});
- if (not $ok) {
- if ($self->{options}{ignore_bad_version}) {
- warning($error);
- } else {
- error($error);
- }
- }
- # Copy orig tarballs
- if ($self->{options}{copy_orig_tarballs}) {
- my $basename = $self->get_basename();
- my ($dirname, $destdir) = fileparse($newdirectory);
- $destdir ||= './';
- my $ext = compression_get_file_extension_regex();
- foreach my $orig (grep { /^\Q$basename\E\.orig(-[[:alnum:]-]+)?\.tar\.$ext$/ }
- $self->get_files())
- {
- my $src = File::Spec->catfile($self->{basedir}, $orig);
- my $dst = File::Spec->catfile($destdir, $orig);
- if (not check_files_are_the_same($src, $dst, 1)) {
- system('cp', '--', $src, $dst);
- subprocerr("cp $src to $dst") if $?;
- }
- }
- }
- # Try extract
- eval { $self->do_extract($newdirectory) };
- if ($@) {
- run_exit_handlers();
- die $@;
- }
- # Store format if non-standard so that next build keeps the same format
- if ($self->{fields}{'Format'} ne '1.0' and
- not $self->{options}{skip_debianization})
- {
- my $srcdir = File::Spec->catdir($newdirectory, 'debian', 'source');
- my $format_file = File::Spec->catfile($srcdir, 'format');
- unless (-e $format_file) {
- mkdir($srcdir) unless -e $srcdir;
- open(my $format_fh, '>', $format_file)
- or syserr(g_('cannot write %s'), $format_file);
- print { $format_fh } $self->{fields}{'Format'} . "\n";
- close($format_fh);
- }
- }
- # Make sure debian/rules is executable
- my $rules = File::Spec->catfile($newdirectory, 'debian', 'rules');
- my @s = lstat($rules);
- if (not scalar(@s)) {
- unless ($! == ENOENT) {
- syserr(g_('cannot stat %s'), $rules);
- }
- warning(g_('%s does not exist'), $rules)
- unless $self->{options}{skip_debianization};
- } elsif (-f _) {
- chmod($s[2] | 0111, $rules)
- or syserr(g_('cannot make %s executable'), $rules);
- } else {
- warning(g_('%s is not a plain file'), $rules);
- }
- }
- sub do_extract {
- croak 'Dpkg::Source::Package does not know how to unpack a ' .
- 'source package; use one of the subclasses';
- }
- # Function used specifically during creation of a source package
- sub before_build {
- my ($self, $dir) = @_;
- }
- sub build {
- my $self = shift;
- eval { $self->do_build(@_) };
- if ($@) {
- run_exit_handlers();
- die $@;
- }
- }
- sub after_build {
- my ($self, $dir) = @_;
- }
- sub do_build {
- croak 'Dpkg::Source::Package does not know how to build a ' .
- 'source package; use one of the subclasses';
- }
- sub can_build {
- my ($self, $dir) = @_;
- return (0, 'can_build() has not been overriden');
- }
- sub add_file {
- my ($self, $filename) = @_;
- my ($fn, $dir) = fileparse($filename);
- if ($self->{checksums}->has_file($fn)) {
- croak "tried to add file '$fn' twice";
- }
- $self->{checksums}->add_from_file($filename, key => $fn);
- $self->{checksums}->export_to_control($self->{fields},
- use_files_for_md5 => 1);
- }
- sub commit {
- my $self = shift;
- eval { $self->do_commit(@_) };
- if ($@) {
- run_exit_handlers();
- die $@;
- }
- }
- sub do_commit {
- my ($self, $dir) = @_;
- info(g_("'%s' is not supported by the source format '%s'"),
- 'dpkg-source --commit', $self->{fields}{'Format'});
- }
- sub write_dsc {
- my ($self, %opts) = @_;
- my $fields = $self->{fields};
- foreach my $f (keys %{$opts{override}}) {
- $fields->{$f} = $opts{override}{$f};
- }
- unless ($opts{nocheck}) {
- foreach my $f (qw(Source Version Architecture)) {
- unless (defined($fields->{$f})) {
- error(g_('missing information for critical output field %s'), $f);
- }
- }
- foreach my $f (qw(Maintainer Standards-Version)) {
- unless (defined($fields->{$f})) {
- warning(g_('missing information for output field %s'), $f);
- }
- }
- }
- foreach my $f (keys %{$opts{remove}}) {
- delete $fields->{$f};
- }
- my $filename = $opts{filename};
- $filename //= $self->get_basename(1) . '.dsc';
- open(my $dsc_fh, '>', $filename)
- or syserr(g_('cannot write %s'), $filename);
- $fields->apply_substvars($opts{substvars});
- $fields->output($dsc_fh);
- close($dsc_fh);
- }
- =back
- =head1 CHANGES
- =head2 Version 1.02 (dpkg 1.18.7)
- New option: require_strong_checksums in check_checksums().
- =head2 Version 1.01 (dpkg 1.17.2)
- New functions: get_default_diff_ignore_regex(), set_default_diff_ignore_regex(),
- get_default_tar_ignore_pattern()
- Deprecated variables: $diff_ignore_default_regexp, @tar_ignore_default_pattern
- =head2 Version 1.00 (dpkg 1.16.1)
- Mark the module as public.
- =cut
- 1;
|