blob: 2d0b79d915f0b9c72c3d3de5cdc5b929cfa2adbe [file] [log] [blame]
#!/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