| #!/usr/bin/perl |
| # |
| # dpkg-genchanges |
| # |
| # 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 <http://www.gnu.org/licenses/>. |
| |
| use strict; |
| use warnings; |
| |
| use Encode; |
| use POSIX; |
| use POSIX qw(:errno_h :signal_h); |
| use Dpkg; |
| use Dpkg::Gettext; |
| use Dpkg::Checksums; |
| use Dpkg::ErrorHandling; |
| use Dpkg::Arch qw(get_host_arch debarch_eq debarch_is); |
| use Dpkg::Compression; |
| use Dpkg::Control::Info; |
| use Dpkg::Control::Fields; |
| use Dpkg::Control; |
| use Dpkg::Substvars; |
| use Dpkg::Vars; |
| use Dpkg::Changelog::Parse; |
| use Dpkg::Version; |
| |
| textdomain("dpkg-dev"); |
| |
| my $controlfile = 'debian/control'; |
| my $changelogfile = 'debian/changelog'; |
| my $changelogformat; |
| my $fileslistfile = 'debian/files'; |
| my $uploadfilesdir = '..'; |
| my $sourcestyle = 'i'; |
| my $quiet = 0; |
| my $host_arch = get_host_arch(); |
| my $changes_format = "1.8"; |
| |
| my %f2p; # - file to package map |
| my %p2f; # - package to file map, has entries for "packagename" |
| my %pa2f; # - likewise, has entries for "packagename architecture" |
| my %p2ver; # - package to version map |
| my %p2arch; # - package to arch map |
| my %f2sec; # - file to section map |
| my %f2seccf; # - likewise, from control file |
| my %f2pri; # - file to priority map |
| my %f2pricf; # - likewise, from control file |
| my %sourcedefault; # - default values as taken from source (used for Section, |
| # Priority and Maintainer) |
| |
| my @descriptions; |
| my @fileslistfiles; |
| |
| my $checksums = Dpkg::Checksums->new(); |
| my %remove; # - fields to remove |
| my %override; |
| my %archadded; |
| my @archvalues; |
| my $dsc; |
| my $changesdescription; |
| my $forcemaint; |
| my $forcechangedby; |
| my $since; |
| |
| my $substvars_loaded = 0; |
| my $substvars = Dpkg::Substvars->new(); |
| $substvars->set("Format", $changes_format); |
| |
| use constant SOURCE => 1; |
| use constant ARCH_DEP => 2; |
| use constant ARCH_INDEP => 4; |
| use constant BIN => ARCH_DEP | ARCH_INDEP; |
| use constant ALL => BIN | SOURCE; |
| my $include = ALL; |
| |
| sub is_sourceonly() { return $include == SOURCE; } |
| sub is_binaryonly() { return !($include & SOURCE); } |
| sub binary_opt() { return (($include == BIN) ? '-b' : |
| (($include == ARCH_DEP) ? '-B' : |
| (($include == ARCH_INDEP) ? '-A' : |
| internerr("binary_opt called with include=$include")))); |
| } |
| |
| sub version { |
| printf _g("Debian %s version %s.\n"), $progname, $version; |
| |
| printf _g(" |
| Copyright (C) 1996 Ian Jackson. |
| Copyright (C) 2000,2001 Wichert Akkerman."); |
| |
| printf _g(" |
| This is free software; see the GNU General Public License version 2 or |
| later for copying conditions. There is NO warranty. |
| "); |
| } |
| |
| sub usage { |
| printf _g( |
| "Usage: %s [<option> ...] |
| |
| Options: |
| -b binary-only build - no source files. |
| -B arch-specific - no source or arch-indep files. |
| -A only arch-indep - no source or arch-specific files. |
| -S source-only upload. |
| -c<controlfile> get control info from this file. |
| -l<changelogfile> get per-version info from this file. |
| -f<fileslistfile> get .deb files list from this file. |
| -v<sinceversion> include all changes later than version. |
| -C<changesdescription> use change description from this file. |
| -m<maintainer> override control's maintainer value. |
| -e<maintainer> override changelog's maintainer value. |
| -u<uploadfilesdir> directory with files (default is \`..'). |
| -si (default) src includes orig if new upstream. |
| -sa source includes orig src. |
| -sd source is diff and .dsc only. |
| -q quiet - no informational messages on stderr. |
| -F<changelogformat> force change log format. |
| -V<name>=<value> set a substitution variable. |
| -T<varlistfile> read variables here, not debian/substvars. |
| -D<field>=<value> override or add a field and value. |
| -U<field> remove a field. |
| -h, --help show this help message. |
| --version show the version. |
| "), $progname; |
| } |
| |
| |
| while (@ARGV) { |
| $_=shift(@ARGV); |
| if (m/^-b$/) { |
| is_sourceonly && usageerr(_g("cannot combine %s and %s"), $_, "-S"); |
| $include = BIN; |
| } elsif (m/^-B$/) { |
| is_sourceonly && usageerr(_g("cannot combine %s and %s"), $_, "-S"); |
| $include = ARCH_DEP; |
| printf STDERR _g("%s: arch-specific upload - not including arch-independent packages")."\n", $progname; |
| } elsif (m/^-A$/) { |
| is_sourceonly && usageerr(_g("cannot combine %s and %s"), $_, "-S"); |
| $include = ARCH_INDEP; |
| printf STDERR _g("%s: arch-indep upload - not including arch-specific packages")."\n", $progname; |
| } elsif (m/^-S$/) { |
| is_binaryonly && usageerr(_g("cannot combine %s and %s"), binary_opt, "-S"); |
| $include = SOURCE; |
| } elsif (m/^-s([iad])$/) { |
| $sourcestyle= $1; |
| } elsif (m/^-q$/) { |
| $quiet= 1; |
| } elsif (m/^-c(.*)$/) { |
| $controlfile = $1; |
| } elsif (m/^-l(.*)$/) { |
| $changelogfile = $1; |
| } elsif (m/^-C(.*)$/) { |
| $changesdescription = $1; |
| } elsif (m/^-f(.*)$/) { |
| $fileslistfile = $1; |
| } elsif (m/^-v(.*)$/) { |
| $since = $1; |
| } elsif (m/^-T(.*)$/) { |
| $substvars->load($1) if -e $1; |
| $substvars_loaded = 1; |
| } elsif (m/^-m(.*)$/s) { |
| $forcemaint = $1; |
| } elsif (m/^-e(.*)$/s) { |
| $forcechangedby = $1; |
| } elsif (m/^-F([0-9a-z]+)$/) { |
| $changelogformat = $1; |
| } elsif (m/^-D([^\=:]+)[=:](.*)$/s) { |
| $override{$1} = $2; |
| } elsif (m/^-u(.*)$/) { |
| $uploadfilesdir = $1; |
| } elsif (m/^-U([^\=:]+)$/) { |
| $remove{$1} = 1; |
| } elsif (m/^-V(\w[-:0-9A-Za-z]*)[=:](.*)$/s) { |
| $substvars->set($1, $2); |
| } elsif (m/^-(h|-help)$/) { |
| usage(); |
| exit(0); |
| } elsif (m/^--version$/) { |
| version(); |
| exit(0); |
| } else { |
| usageerr(_g("unknown option \`%s'"), $_); |
| } |
| } |
| |
| # Retrieve info from the current changelog entry |
| my %options = (file => $changelogfile); |
| $options{"changelogformat"} = $changelogformat if $changelogformat; |
| $options{"since"} = $since if defined($since); |
| my $changelog = changelog_parse(%options); |
| # Change options to retrieve info of the former changelog entry |
| delete $options{"since"}; |
| $options{"count"} = 1; |
| $options{"offset"} = 1; |
| my $prev_changelog = changelog_parse(%options); |
| # Other initializations |
| my $control = Dpkg::Control::Info->new($controlfile); |
| my $fields = Dpkg::Control->new(type => CTRL_FILE_CHANGES); |
| $substvars->set_version_substvars($changelog->{"Version"}); |
| $substvars->set_arch_substvars(); |
| $substvars->load("debian/substvars") if -e "debian/substvars" and not $substvars_loaded; |
| |
| if (defined($prev_changelog) and |
| version_compare_relation($changelog->{"Version"}, REL_LT, |
| $prev_changelog->{"Version"})) |
| { |
| warning(_g("the current version (%s) is earlier than the previous one (%s)"), |
| $changelog->{"Version"}, $prev_changelog->{"Version"}) |
| # ~bpo and ~vola are backports and have lower version number by definition |
| unless $changelog->{"Version"} =~ /~(?:bpo|vola)/; |
| } |
| |
| if (not is_sourceonly) { |
| open(FL, "<", $fileslistfile) || syserr(_g("cannot read files list file")); |
| while(<FL>) { |
| if (m/^(([-+.0-9a-z]+)_([^_]+)_([-\w]+)\.u?deb) (\S+) (\S+)$/) { |
| defined($p2f{"$2 $4"}) && |
| warning(_g("duplicate files list entry for package %s (line %d)"), |
| $2, $.); |
| $f2p{$1}= $2; |
| $pa2f{"$2 $4"}= $1; |
| $p2f{$2} ||= []; |
| push @{$p2f{$2}}, $1; |
| $p2ver{$2}= $3; |
| defined($f2sec{$1}) && |
| warning(_g("duplicate files list entry for file %s (line %d)"), |
| $1, $.); |
| $f2sec{$1}= $5; |
| $f2pri{$1}= $6; |
| push(@archvalues,$4) unless !$4 || $archadded{$4}++; |
| push(@fileslistfiles,$1); |
| } elsif (m/^([-+.0-9a-z]+_[^_]+_([-\w]+)\.[a-z0-9.]+) (\S+) (\S+)$/) { |
| # A non-deb package |
| $f2sec{$1}= $3; |
| $f2pri{$1}= $4; |
| push(@archvalues,$2) unless !$2 || $archadded{$2}++; |
| push(@fileslistfiles,$1); |
| } elsif (m/^([-+.,_0-9a-zA-Z]+) (\S+) (\S+)$/) { |
| defined($f2sec{$1}) && |
| warning(_g("duplicate files list entry for file %s (line %d)"), |
| $1, $.); |
| $f2sec{$1}= $2; |
| $f2pri{$1}= $3; |
| push(@fileslistfiles,$1); |
| } else { |
| error(_g("badly formed line in files list file, line %d"), $.); |
| } |
| } |
| close(FL); |
| } |
| |
| # Scan control info of source package |
| my $src_fields = $control->get_source(); |
| foreach $_ (keys %{$src_fields}) { |
| my $v = $src_fields->{$_}; |
| if (m/^Source$/) { |
| set_source_package($v); |
| } elsif (m/^Section$|^Priority$/i) { |
| $sourcedefault{$_} = $v; |
| } else { |
| field_transfer_single($src_fields, $fields); |
| } |
| } |
| |
| # Scan control info of all binary packages |
| foreach my $pkg ($control->get_packages()) { |
| my $p = $pkg->{"Package"}; |
| my $a = $pkg->{"Architecture"} || ""; |
| my $d = $pkg->{"Description"} || "no description available"; |
| $d = $1 if $d =~ /^(.*)\n/; |
| my $pkg_type = $pkg->{"Package-Type"} || |
| $pkg->get_custom_field("Package-Type") || "deb"; |
| |
| my @f; # List of files for this binary package |
| push @f, @{$p2f{$p}} if defined $p2f{$p}; |
| |
| # Add description of all binary packages |
| my $desc = encode_utf8(sprintf("%-10s - %-.65s", $p, decode_utf8($d))); |
| $desc .= " (udeb)" if $pkg_type eq "udeb"; |
| push @descriptions, $desc; |
| |
| if (not defined($p2f{$p})) { |
| # No files for this package... warn if it's unexpected |
| if ((debarch_eq('all', $a) and ($include & ARCH_INDEP)) || |
| (grep(debarch_is($host_arch, $_), split(/\s+/, $a)) |
| and ($include & ARCH_DEP))) { |
| warning(_g("package %s in control file but not in files list"), |
| $p); |
| } |
| next; # and skip it |
| } |
| |
| $p2arch{$p} = $a; |
| |
| foreach $_ (keys %{$pkg}) { |
| my $v = $pkg->{$_}; |
| |
| if (m/^Section$/) { |
| $f2seccf{$_} = $v foreach (@f); |
| } elsif (m/^Priority$/) { |
| $f2pricf{$_} = $v foreach (@f); |
| } elsif (m/^Architecture$/) { |
| if (grep(debarch_is($host_arch, $_), split(/\s+/, $v)) |
| and ($include & ARCH_DEP)) { |
| $v = $host_arch; |
| } elsif (!debarch_eq('all', $v)) { |
| $v = ''; |
| } |
| push(@archvalues,$v) unless !$v || $archadded{$v}++; |
| } elsif (m/^Description$/) { |
| # Description in changes is computed, do not copy this field |
| } else { |
| field_transfer_single($pkg, $fields); |
| } |
| } |
| } |
| |
| # Scan fields of dpkg-parsechangelog |
| foreach $_ (keys %{$changelog}) { |
| my $v = $changelog->{$_}; |
| if (m/^Source$/i) { |
| set_source_package($v); |
| } elsif (m/^Maintainer$/i) { |
| $fields->{"Changed-By"} = $v; |
| } else { |
| field_transfer_single($changelog, $fields); |
| } |
| } |
| |
| if ($changesdescription) { |
| open(X, "<", $changesdescription) || syserr(_g("read changesdescription")); |
| $fields->{'Changes'} = "\n" . join("", <X>); |
| close(X); |
| } |
| |
| for my $pa (keys %pa2f) { |
| my ($pp, $aa) = (split / /, $pa); |
| defined($control->get_pkg_by_name($pp)) || |
| warning(_g("package %s listed in files list but not in control info"), |
| $pp); |
| } |
| |
| for my $p (keys %p2f) { |
| my @f = @{$p2f{$p}}; |
| |
| foreach my $f (@f) { |
| my $sec = $f2seccf{$f}; |
| $sec ||= $sourcedefault{'Section'}; |
| if (!defined($sec)) { |
| $sec = '-'; |
| warning(_g("missing Section for binary package %s; using '-'"), $p); |
| } |
| $sec eq $f2sec{$f} || error(_g("package %s has section %s in " . |
| "control file but %s in files list"), |
| $p, $sec, $f2sec{$f}); |
| my $pri = $f2pricf{$f}; |
| $pri ||= $sourcedefault{'Priority'}; |
| if (!defined($pri)) { |
| $pri = '-'; |
| warning(_g("missing Priority for binary package %s; using '-'"), $p); |
| } |
| $pri eq $f2pri{$f} || error(_g("package %s has priority %s in " . |
| "control file but %s in files list"), |
| $p, $pri, $f2pri{$f}); |
| } |
| } |
| |
| my $origsrcmsg; |
| |
| if (!is_binaryonly) { |
| my $sec = $sourcedefault{'Section'}; |
| if (!defined($sec)) { |
| $sec = '-'; |
| warning(_g("missing Section for source files")); |
| } |
| my $pri = $sourcedefault{'Priority'}; |
| if (!defined($pri)) { |
| $pri = '-'; |
| warning(_g("missing Priority for source files")); |
| } |
| |
| (my $sversion = $substvars->get('source:Version')) =~ s/^\d+://; |
| $dsc= "$uploadfilesdir/${sourcepackage}_${sversion}.dsc"; |
| |
| my $dsc_fields = Dpkg::Control->new(type => CTRL_PKG_SRC); |
| $dsc_fields->load($dsc) || error(_g("%s is empty", $dsc)); |
| $checksums->add_from_file($dsc, key => "$sourcepackage\_$sversion.dsc"); |
| $checksums->add_from_control($dsc_fields, use_files_for_md5 => 1); |
| |
| for my $f ($checksums->get_files()) { |
| $f2sec{$f} = $sec; |
| $f2pri{$f} = $pri; |
| } |
| |
| # Compare upstream version to previous upstream version to decide if |
| # the .orig tarballs must be included |
| my $include_tarball; |
| if (defined($prev_changelog)) { |
| my $cur = Dpkg::Version->new($changelog->{"Version"}); |
| my $prev = Dpkg::Version->new($prev_changelog->{"Version"}); |
| $include_tarball = ($cur->version() ne $prev->version()) ? 1 : 0; |
| } else { |
| # No previous entry means first upload, tarball required |
| $include_tarball = 1; |
| } |
| |
| my $ext = $compression_re_file_ext; |
| if ((($sourcestyle =~ m/i/ && not($include_tarball)) || |
| $sourcestyle =~ m/d/) && |
| grep(m/\.(debian\.tar|diff)\.$ext$/, $checksums->get_files())) |
| { |
| $origsrcmsg= _g("not including original source code in upload"); |
| foreach my $f (grep m/\.orig(-.+)?\.tar\.$ext$/, $checksums->get_files()) { |
| $checksums->remove_file($f); |
| } |
| } else { |
| if ($sourcestyle =~ m/d/ && |
| !grep(m/\.(debian\.tar|diff)\.$ext$/, $checksums->get_files())) { |
| warning(_g("ignoring -sd option for native Debian package")); |
| } |
| $origsrcmsg= _g("including full source code in upload"); |
| } |
| } else { |
| $origsrcmsg= _g("binary-only upload - not including any source code"); |
| } |
| |
| print(STDERR "$progname: $origsrcmsg\n") || |
| syserr(_g("write original source message")) unless $quiet; |
| |
| $fields->{'Format'} = $substvars->get("Format"); |
| |
| if (!defined($fields->{'Date'})) { |
| chomp(my $date822 = `date -R`); |
| $? && subprocerr("date -R"); |
| $fields->{'Date'}= $date822; |
| } |
| |
| $fields->{'Binary'} = join(' ', map { $_->{'Package'} } $control->get_packages()); |
| # Avoid overly long line by splitting over multiple lines |
| if (length($fields->{'Binary'}) > 980) { |
| $fields->{'Binary'} =~ s/(.{0,980}) /$1\n/g; |
| } |
| |
| unshift(@archvalues,'source') unless is_binaryonly; |
| @archvalues = ('all') if $include == ARCH_INDEP; |
| @archvalues = grep {!debarch_eq('all',$_)} @archvalues |
| unless $include & ARCH_INDEP; |
| $fields->{'Architecture'} = join(' ',@archvalues); |
| |
| $fields->{'Description'} = "\n" . join("\n", sort @descriptions); |
| |
| $fields->{'Files'} = ''; |
| |
| my %filedone; |
| |
| for my $f ($checksums->get_files(), @fileslistfiles) { |
| my $arch_all = debarch_eq('all', $p2arch{$f2p{$f}}) if defined($f2p{$f}); |
| |
| next if (defined($arch_all) && ($include == ARCH_DEP and $arch_all)); |
| next if (defined($arch_all) && ($include == ARCH_INDEP and not $arch_all)); |
| next if $filedone{$f}++; |
| my $uf = "$uploadfilesdir/$f"; |
| $checksums->add_from_file($uf, key => $f); |
| $fields->{'Files'} .= "\n" . $checksums->get_checksum($f, "md5") . |
| " " . $checksums->get_size($f) . |
| " $f2sec{$f} $f2pri{$f} $f"; |
| } |
| $checksums->export_to_control($fields); |
| # redundant with the Files field |
| delete $fields->{"Checksums-Md5"}; |
| |
| $fields->{'Source'}= $sourcepackage; |
| if ($fields->{'Version'} ne $substvars->get('source:Version')) { |
| $fields->{'Source'} .= " (" . $substvars->get('source:Version') . ")"; |
| } |
| |
| $fields->{'Maintainer'} = $forcemaint if defined($forcemaint); |
| $fields->{'Changed-By'} = $forcechangedby if defined($forcechangedby); |
| |
| for my $f (qw(Version Distribution Maintainer Changes)) { |
| defined($fields->{$f}) || |
| error(_g("missing information for critical output field %s"), $f); |
| } |
| |
| for my $f (qw(Urgency)) { |
| defined($fields->{$f}) || |
| warning(_g("missing information for output field %s"), $f); |
| } |
| |
| for my $f (keys %override) { |
| $fields->{$f} = $override{$f}; |
| } |
| for my $f (keys %remove) { |
| delete $fields->{$f}; |
| } |
| |
| $fields->output(\*STDOUT); # Note: no substitution of variables |