diff options
Diffstat (limited to 'scripts/dpkg-buildpackage.pl')
-rwxr-xr-x | scripts/dpkg-buildpackage.pl | 1095 |
1 files changed, 1095 insertions, 0 deletions
diff --git a/scripts/dpkg-buildpackage.pl b/scripts/dpkg-buildpackage.pl new file mode 100755 index 0000000..7b8181b --- /dev/null +++ b/scripts/dpkg-buildpackage.pl @@ -0,0 +1,1095 @@ +#!/usr/bin/perl +# +# dpkg-buildpackage +# +# Copyright © 1996 Ian Jackson +# Copyright © 2000 Wichert Akkerman +# Copyright © 2006-2010, 2012-2015 Guillem Jover <guillem@debian.org> +# Copyright © 2007 Frank Lichtenheld +# +# 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/>. + +use strict; +use warnings; + +use File::Temp qw(tempdir); +use File::Basename; +use File::Copy; +use File::Glob qw(bsd_glob GLOB_TILDE GLOB_NOCHECK); +use POSIX qw(:sys_wait_h); + +use Dpkg (); +use Dpkg::Gettext; +use Dpkg::ErrorHandling; +use Dpkg::BuildTypes; +use Dpkg::BuildAPI qw(get_build_api); +use Dpkg::BuildOptions; +use Dpkg::BuildProfiles qw(set_build_profiles); +use Dpkg::Conf; +use Dpkg::Compression; +use Dpkg::Checksums; +use Dpkg::Package; +use Dpkg::Version; +use Dpkg::Control; +use Dpkg::Control::Info; +use Dpkg::Changelog::Parse; +use Dpkg::OpenPGP; +use Dpkg::OpenPGP::ErrorCodes; +use Dpkg::OpenPGP::KeyHandle; +use Dpkg::Path qw(find_command); +use Dpkg::IPC; +use Dpkg::Vendor qw(run_vendor_hook); + +textdomain('dpkg-dev'); + +sub showversion { + printf g_("Debian %s version %s.\n"), $Dpkg::PROGNAME, $Dpkg::PROGVERSION; + + print 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>...]') + . "\n\n" . g_( +'Options: + --build=<type>[,...] specify the build <type>: full, source, binary, + any, all (default is \'full\'). + -F, --build=full normal full build (source and binary; default). + -g, --build=source,all source and arch-indep build. + -G, --build=source,any source and arch-specific build. + -b, --build=binary binary-only, no source files. + -B, --build=any binary-only, only arch-specific files. + -A, --build=all binary-only, only arch-indep files. + -S, --build=source source-only, no binary files. + -nc, --no-pre-clean do not pre clean source tree (implies -b). + --pre-clean pre clean source tree (default). + --no-post-clean do not post clean source tree (default). + -tc, --post-clean post clean source tree. + --sanitize-env sanitize the build environment. + -D, --check-builddeps check build dependencies and conflicts (default). + -d, --no-check-builddeps do not check build dependencies and conflicts. + --ignore-builtin-builddeps + do not check builtin build dependencies. + -P, --build-profiles=<profiles> + assume comma-separated build <profiles> as active. + --rules-requires-root assume legacy Rules-Requires-Root field value. + -R, --rules-file=<rules> rules file to execute (default is debian/rules). + -T, --rules-target=<target> call debian/rules <target>. + --as-root ensure -T calls the target with root rights. + -j, --jobs[=<jobs>|auto] jobs to run simultaneously (passed to <rules>), + (default; default is auto, opt-in mode). + -J, --jobs-try[=<jobs>|auto] + alias for -j, --jobs. + --jobs-force[=<jobs>|auto] + jobs to run simultaneously (passed to <rules>), + (default is auto, forced mode). + -r, --root-command=<command> + command to gain root rights (default is fakeroot). + --check-command=<command> + command to check the .changes file (no default). + --check-option=<opt> pass <opt> to check <command>. + --hook-<name>=<command> set <command> as the hook <name>, known hooks: + preinit init preclean source build binary + buildinfo changes postclean check sign done + --buildinfo-file=<file> set the .buildinfo filename to generate. + --buildinfo-option=<opt> + pass option <opt> to dpkg-genbuildinfo. + --changes-file=<file> set the .changes filename to generate. + --sign-backend=<backend> + OpenPGP backend to use to sign + (default is auto). + -p, --sign-command=<command> + command to sign .dsc and/or .changes files + (default is gpg). + --sign-keyfile=<file> the key file to use for signing. + -k, --sign-keyid=<keyid> the key id to use for signing. + --sign-key=<keyid> alias for -k, --sign-keyid. + -ap, --sign-pause add pause before starting signature process. + -us, --unsigned-source unsigned source package. + -ui, --unsigned-buildinfo unsigned .buildinfo file. + -uc, --unsigned-changes unsigned .buildinfo and .changes file. + --no-sign do not sign any file. + --force-sign force signing the resulting files. + --admindir=<directory> change the administrative directory. + -?, --help show this help message. + --version show the version.') + . "\n\n" . g_( +'Options passed to dpkg-architecture: + -a, --host-arch <arch> set the host Debian architecture. + -t, --host-type <type> set the host GNU system type. + --target-arch <arch> set the target Debian architecture. + --target-type <type> set the target GNU system type.') + . "\n\n" . g_( +'Options passed to dpkg-genchanges: + -si source includes orig, if new upstream (default). + -sa source includes orig, always. + -sd source is diff and .dsc only. + -v<version> changes since version <version>. + -m, --source-by=<maint> maintainer for this source or build is <maint>. + --build-by=<maint> ditto. + -e, --release-by=<maint> maintainer for this change or release is <maint>. + --changed-by=<maint> ditto. + -C<descfile> changes are described in <descfile>. + --changes-option=<opt> pass option <opt> to dpkg-genchanges.') + . "\n\n" . g_( +'Options passed to dpkg-source: + -sn force Debian native source format. + -s[sAkurKUR] see dpkg-source for explanation. + -z, --compression-level=<level> + compression level to use for source. + -Z, --compression=<compressor> + compression to use for source (gz|xz|bzip2|lzma). + -i, --diff-ignore[=<regex>] ignore diffs of files matching <regex>. + -I, --tar-ignore[=<pattern>] + filter out files when building tarballs. + --source-option=<opt> pass option <opt> to dpkg-source. +'), $Dpkg::PROGNAME; +} + +my $admindir; +my @debian_rules = ('debian/rules'); +my @rootcommand = (); +my $signbackend; +my $signcommand; +my $preclean = 1; +my $postclean = 0; +my $sanitize_env = 0; +my $parallel; +my $parallel_force = 0; +my $checkbuilddep = 1; +my $check_builtin_builddep = 1; +my @source_opts; +my $check_command = $ENV{DEB_CHECK_COMMAND}; +my @check_opts; +my $signpause; +my $signkeyfile = $ENV{DEB_SIGN_KEYFILE}; +my $signkeyid = $ENV{DEB_SIGN_KEYID}; +my $signforce = 0; +my $signreleased = 1; +my $signsource = 1; +my $signbuildinfo = 1; +my $signchanges = 1; +my $buildtarget = 'build'; +my $binarytarget = 'binary'; +my $host_arch = ''; +my $host_type = ''; +my $target_arch = ''; +my $target_type = ''; +my @build_profiles = (); +my $rrr_override; +my @call_target = (); +my $call_target_as_root = 0; +my $since; +my $maint; +my $changedby; +my $desc; +my $buildinfo_file; +my @buildinfo_opts; +my $changes_file; +my @changes_opts; +my %target_legacy_root = map { $_ => 1 } qw( + clean + binary + binary-arch + binary-indep +); +my %target_official = map { $_ => 1 } qw( + clean + build + build-arch + build-indep + binary + binary-arch + binary-indep +); +my @hook_names = qw( + preinit + init + preclean + source + build + binary + buildinfo + changes + postclean + check + sign + done +); +my %hook; +$hook{$_} = undef foreach @hook_names; + + +my $conf = Dpkg::Conf->new(); +$conf->load_config('buildpackage.conf'); + +# Inject config options for command-line parser. +unshift @ARGV, @{$conf}; + +my $build_opts = Dpkg::BuildOptions->new(); + +if ($build_opts->has('nocheck')) { + $check_command = undef; +} elsif (not find_command($check_command)) { + $check_command = undef; +} + +while (@ARGV) { + $_ = shift @ARGV; + + if (/^(?:--help|-\?)$/) { + usage; + exit 0; + } elsif (/^--version$/) { + showversion; + exit 0; + } elsif (/^--admindir$/) { + $admindir = shift @ARGV; + } elsif (/^--admindir=(.*)$/) { + $admindir = $1; + } elsif (/^--source-option=(.*)$/) { + push @source_opts, $1; + } elsif (/^--buildinfo-file=(.*)$/) { + $buildinfo_file = $1; + usageerr(g_('missing .buildinfo filename')) if not length $buildinfo_file; + } elsif (/^--buildinfo-option=(.*)$/) { + my $buildinfo_opt = $1; + if ($buildinfo_opt =~ m/^-O(.*)/) { + warning(g_('passing %s via %s is not supported; please use %s instead'), + '-O', '--buildinfo-option', '--buildinfo-file'); + $buildinfo_file = $1; + } else { + push @buildinfo_opts, $buildinfo_opt; + } + } elsif (/^--changes-file=(.*)$/) { + $changes_file = $1; + usageerr(g_('missing .changes filename')) if not length $changes_file; + } elsif (/^--changes-option=(.*)$/) { + my $changes_opt = $1; + if ($changes_opt =~ m/^-O(.*)/) { + warning(g_('passing %s via %s is not supported; please use %s instead'), + '-O', '--changes-option', '--changes-file'); + $changes_file = $1; + } else { + push @changes_opts, $changes_opt; + } + } elsif (/^--jobs(?:-try)?$/) { + $parallel = ''; + $parallel_force = 0; + } elsif (/^(?:-[jJ]|--jobs(?:-try)?=)(\d*|auto)$/) { + $parallel = $1 || ''; + $parallel_force = 0; + } elsif (/^--jobs-force(?:=(\d*|auto))?$/) { + $parallel = $1 || ''; + $parallel_force = 1; + } elsif (/^(?:-r|--root-command=)(.*)$/) { + my $arg = $1; + @rootcommand = split ' ', $arg; + } elsif (/^--check-command=(.*)$/) { + $check_command = $1; + } elsif (/^--check-option=(.*)$/) { + push @check_opts, $1; + } elsif (/^--hook-([^=]+)=(.*)$/) { + my ($hook_name, $hook_cmd) = ($1, $2); + usageerr(g_('unknown hook name %s'), $hook_name) + if not exists $hook{$hook_name}; + usageerr(g_('missing hook %s command'), $hook_name) + if not defined $hook_cmd; + $hook{$hook_name} = $hook_cmd; + } elsif (/^(--buildinfo-id)=.*$/) { + # Deprecated option + warning(g_('%s is deprecated; it is without effect'), $1); + } elsif (/^--sign-backend=(.*)$/) { + $signbackend = $1; + } elsif (/^(?:-p|--sign-command=)(.*)$/) { + $signcommand = $1; + } elsif (/^--sign-keyfile=(.*)$/) { + $signkeyfile = $1; + } elsif (/^(?:-k|--sign-keyid=|--sign-key=)(.*)$/) { + $signkeyid = $1; + } elsif (/^--(no-)?check-builddeps$/) { + $checkbuilddep = !(defined $1 and $1 eq 'no-'); + } elsif (/^-([dD])$/) { + $checkbuilddep = ($1 eq 'D'); + } elsif (/^--ignore-builtin-builddeps$/) { + $check_builtin_builddep = 0; + } elsif (/^-s(gpg|pgp)$/) { + # Deprecated option + warning(g_('-s%s is deprecated; always using gpg style interface'), $1); + } elsif (/^--force-sign$/) { + $signforce = 1; + } elsif (/^--no-sign$/) { + $signforce = 0; + $signsource = 0; + $signbuildinfo = 0; + $signchanges = 0; + } elsif (/^-us$/ or /^--unsigned-source$/) { + $signsource = 0; + } elsif (/^-ui$/ or /^--unsigned-buildinfo$/) { + $signbuildinfo = 0; + } elsif (/^-uc$/ or /^--unsigned-changes$/) { + $signbuildinfo = 0; + $signchanges = 0; + } elsif (/^-ap$/ or /^--sign-pausa$/) { + $signpause = 1; + } elsif (/^-a$/ or /^--host-arch$/) { + $host_arch = shift; + } elsif (/^-a(.*)$/ or /^--host-arch=(.*)$/) { + $host_arch = $1; + } elsif (/^-P(.*)$/ or /^--build-profiles=(.*)$/) { + my $arg = $1; + @build_profiles = split /,/, $arg; + } elsif (/^-s[iad]$/) { + push @changes_opts, $_; + } elsif (/^--(?:compression-level|compression)=.+$/) { + push @source_opts, $_; + } elsif (/^--(?:diff-ignore|tar-ignore)(?:=.+)?$/) { + push @source_opts, $_; + } elsif (/^-(?:s[nsAkurKUR]|[zZ].*|i.*|I.*)$/) { + push @source_opts, $_; # passed to dpkg-source + } elsif (/^-tc$/ or /^--post-clean$/) { + $postclean = 1; + } elsif (/^--no-post-clean$/) { + $postclean = 0; + } elsif (/^--sanitize-env$/) { + $sanitize_env = 1; + } elsif (/^-t$/ or /^--host-type$/) { + $host_type = shift; # Order DOES matter! + } elsif (/^-t(.*)$/ or /^--host-type=(.*)$/) { + $host_type = $1; # Order DOES matter! + } elsif (/^--target-arch$/) { + $target_arch = shift; + } elsif (/^--target-arch=(.*)$/) { + $target_arch = $1; + } elsif (/^--target-type$/) { + $target_type = shift; + } elsif (/^--target-type=(.*)$/) { + $target_type = $1; + } elsif (/^(?:--target|--rules-target|-T)$/) { + push @call_target, split /,/, shift @ARGV; + } elsif (/^(?:--target=|--rules-target=|-T)(.+)$/) { + my $arg = $1; + push @call_target, split /,/, $arg; + } elsif (/^--rules-requires-root$/) { + $rrr_override = 'binary-targets'; + } elsif (/^--as-root$/) { + $call_target_as_root = 1; + } elsif (/^--pre-clean$/) { + $preclean = 1; + } elsif (/^-nc$/ or /^--no-pre-clean$/) { + $preclean = 0; + } elsif (/^--build=(.*)$/) { + set_build_type_from_options($1, $_); + } elsif (/^-b$/) { + set_build_type(BUILD_BINARY, $_); + } elsif (/^-B$/) { + set_build_type(BUILD_ARCH_DEP, $_); + } elsif (/^-A$/) { + set_build_type(BUILD_ARCH_INDEP, $_); + } elsif (/^-S$/) { + set_build_type(BUILD_SOURCE, $_); + } elsif (/^-G$/) { + set_build_type(BUILD_SOURCE | BUILD_ARCH_DEP, $_); + } elsif (/^-g$/) { + set_build_type(BUILD_SOURCE | BUILD_ARCH_INDEP, $_); + } elsif (/^-F$/) { + set_build_type(BUILD_FULL, $_); + } elsif (/^-v(.*)$/) { + $since = $1; + } elsif (/^-m(.*)$/ or /^--(?:source|build)-by=(.*)$/) { + $maint = $1; + } elsif (/^-e(.*)$/ or /^--(?:changed|release)-by=(.*)$/) { + $changedby = $1; + } elsif (/^-C(.*)$/) { + $desc = $1; + } elsif (m/^-[EW]$/) { + # Deprecated option + warning(g_('%s is deprecated; it is without effect'), $_); + } elsif (/^-R(.*)$/ or /^--rules-file=(.*)$/) { + my $arg = $1; + @debian_rules = split ' ', $arg; + } else { + usageerr(g_('unknown option or argument %s'), $_); + } +} + +if (@call_target) { + my $targets = join ',', @call_target; + set_build_type_from_targets($targets, '--rules-target', nocheck => 1); +} + +if (build_has_all(BUILD_BINARY)) { + $buildtarget = 'build'; + $binarytarget = 'binary'; +} elsif (build_has_any(BUILD_ARCH_DEP)) { + $buildtarget = 'build-arch'; + $binarytarget = 'binary-arch'; +} elsif (build_has_any(BUILD_ARCH_INDEP)) { + $buildtarget = 'build-indep'; + $binarytarget = 'binary-indep'; +} + +if (not $preclean) { + # -nc without -b/-B/-A/-S/-F implies -b + set_build_type(BUILD_BINARY) if build_has_any(BUILD_DEFAULT); + # -nc with -S implies no dependency checks + $checkbuilddep = 0 if build_is(BUILD_SOURCE); +} + +if ($call_target_as_root and @call_target == 0) { + error(g_('option %s is only meaningful with option %s'), + '--as-root', '--rules-target'); +} + +if ($check_command and not find_command($check_command)) { + error(g_("check-command '%s' not found"), $check_command); +} + +if ($signcommand and not find_command($signcommand)) { + error(g_("sign-command '%s' not found"), $signcommand); +} + +# Default to auto if none of parallel=N, -J or -j have been specified. +if (not defined $parallel and not $build_opts->has('parallel')) { + $parallel = 'auto'; +} + +# +# Prepare the environment. +# + +run_hook('preinit'); + +if (defined $parallel) { + if ($parallel eq 'auto') { + # Most Unices. + $parallel = qx(getconf _NPROCESSORS_ONLN 2>/dev/null); + # Fallback for at least Irix. + $parallel = qx(getconf _NPROC_ONLN 2>/dev/null) if $?; + # Fallback to serial execution if cannot infer the number of online + # processors. + $parallel = '1' if $?; + chomp $parallel; + } + if ($parallel_force) { + $ENV{MAKEFLAGS} //= ''; + $ENV{MAKEFLAGS} .= " -j$parallel"; + } + $build_opts->set('parallel', $parallel); + $build_opts->export(); +} + +if ($build_opts->has('terse')) { + $ENV{MAKEFLAGS} //= ''; + $ENV{MAKEFLAGS} .= ' --no-print-directory'; +} + +set_build_profiles(@build_profiles) if @build_profiles; + +my $changelog = changelog_parse(); +my $ctrl = Dpkg::Control::Info->new(); + +# Check whether we are doing some kind of rootless build, and sanity check +# the fields values. +my %rules_requires_root = parse_rules_requires_root($ctrl); + +my $pkg = mustsetvar($changelog->{source}, g_('source package')); +my $version = mustsetvar($changelog->{version}, g_('source version')); +my $v = Dpkg::Version->new($version); +my ($ok, $error) = version_check($v); +error($error) unless $ok; + +my $sversion = $v->as_string(omit_epoch => 1); +my $uversion = $v->version(); + +my $distribution = mustsetvar($changelog->{distribution}, g_('source distribution')); + +my $maintainer; +if ($changedby) { + $maintainer = $changedby; +} elsif ($maint) { + $maintainer = $maint; +} else { + $maintainer = mustsetvar($changelog->{maintainer}, g_('source changed by')); +} + +# <https://reproducible-builds.org/specs/source-date-epoch/> +$ENV{SOURCE_DATE_EPOCH} ||= $changelog->{timestamp} || time; + +my @arch_opts; +push @arch_opts, ('--host-arch', $host_arch) if $host_arch; +push @arch_opts, ('--host-type', $host_type) if $host_type; +push @arch_opts, ('--target-arch', $target_arch) if $target_arch; +push @arch_opts, ('--target-type', $target_type) if $target_type; + +open my $arch_env, '-|', 'dpkg-architecture', '-f', @arch_opts + or subprocerr('dpkg-architecture'); +while (<$arch_env>) { + chomp; + my ($key, $value) = split /=/, $_, 2; + $ENV{$key} = $value; +} +close $arch_env or subprocerr('dpkg-architecture'); + +my $arch; +if (build_has_any(BUILD_ARCH_DEP)) { + $arch = mustsetvar($ENV{DEB_HOST_ARCH}, g_('host architecture')); +} elsif (build_has_any(BUILD_ARCH_INDEP)) { + $arch = 'all'; +} elsif (build_has_any(BUILD_SOURCE)) { + $arch = 'source'; +} + +my $pv = "${pkg}_$sversion"; +my $pva = "${pkg}_${sversion}_$arch"; + +my $signkeytype; +my $signkeyhandle; +if (defined $signkeyfile) { + $signkeytype = 'keyfile'; + $signkeyhandle = bsd_glob($signkeyfile, GLOB_TILDE | GLOB_NOCHECK); +} elsif (defined $signkeyid) { + $signkeytype = 'autoid'; + $signkeyhandle = $signkeyid; +} else { + $signkeytype = 'userid'; + $signkeyhandle = $maintainer; +} +my $signkey = Dpkg::OpenPGP::KeyHandle->new( + type => $signkeytype, + handle => $signkeyhandle, +); +signkey_validate(); + +my $openpgp = Dpkg::OpenPGP->new( + backend => $signbackend // 'auto', + cmd => $signcommand // 'auto', + needs => { + keystore => $signkey->needs_keystore(), + }, +); + +if (not $openpgp->can_use_secrets($signkey)) { + $signsource = 0; + $signbuildinfo = 0; + $signchanges = 0; +} elsif ($signforce) { + $signsource = 1; + $signbuildinfo = 1; + $signchanges = 1; +} elsif (($signsource or $signbuildinfo or $signchanges) and + $distribution eq 'UNRELEASED') { + $signreleased = 0; + $signsource = 0; + $signbuildinfo = 0; + $signchanges = 0; +} + +if ($signsource && build_has_none(BUILD_SOURCE)) { + $signsource = 0; +} + +# Sanitize build environment. +if ($sanitize_env) { + run_vendor_hook('sanitize-environment'); +} + +# +# Preparation of environment stops here +# + +run_hook('init'); + +if (not -x 'debian/rules') { + warning(g_('debian/rules is not executable; fixing that')); + chmod(0755, 'debian/rules'); # No checks of failures, non fatal +} + +if (scalar @call_target == 0) { + run_cmd('dpkg-source', @source_opts, '--before-build', '.'); +} + +if ($checkbuilddep) { + my @checkbuilddep_opts; + + push @checkbuilddep_opts, '-A' if build_has_none(BUILD_ARCH_DEP); + push @checkbuilddep_opts, '-B' if build_has_none(BUILD_ARCH_INDEP); + push @checkbuilddep_opts, '-I' if not $check_builtin_builddep; + push @checkbuilddep_opts, "--admindir=$admindir" if $admindir; + + system('dpkg-checkbuilddeps', @checkbuilddep_opts); + if (not WIFEXITED($?)) { + subprocerr('dpkg-checkbuilddeps'); + } elsif (WEXITSTATUS($?)) { + warning(g_('build dependencies/conflicts unsatisfied; aborting')); + warning(g_('(Use -d flag to override.)')); + exit 3; + } +} + +foreach my $call_target (@call_target) { + run_rules_cond_root($call_target); +} +exit 0 if scalar @call_target; + +run_hook('preclean', { + enabled => $preclean, +}); + +if ($preclean) { + run_rules_cond_root('clean'); +} + +run_hook('source', { + enabled => build_has_any(BUILD_SOURCE), + env => { + DPKG_BUILDPACKAGE_HOOK_SOURCE_OPTIONS => join(' ', @source_opts), + }, +}); + +if (build_has_any(BUILD_SOURCE)) { + warning(g_('building a source package without cleaning up as you asked; ' . + 'it might contain undesired files')) if not $preclean; + run_cmd('dpkg-source', @source_opts, '-b', '.'); +} + +my $build_types = get_build_options_from_type(); + +if (build_has_any(BUILD_BINARY)) { + # XXX Use some heuristics to decide whether to use build-{arch,indep} + # targets. This is a temporary measure to not break too many packages + # on a flag day. + build_target_fallback($ctrl); +} + +# If we are building rootless, there is no need to call the build target +# independently as non-root. +if (build_has_any(BUILD_BINARY) && rules_requires_root($binarytarget)) { + run_hook('build', { + env => { + DPKG_BUILDPACKAGE_HOOK_BUILD_TARGET => $buildtarget, + }, + }); + run_cmd(@debian_rules, $buildtarget); +} else { + run_hook('build', { + enabled => 0, + }); +} + +if (build_has_any(BUILD_BINARY)) { + run_hook('binary', { + env => { + DPKG_BUILDPACKAGE_HOOK_BINARY_TARGET => $binarytarget, + }, + }); + run_rules_cond_root($binarytarget); +} + +$buildinfo_file //= "../$pva.buildinfo"; + +push @buildinfo_opts, "--build=$build_types" if build_has_none(BUILD_DEFAULT); +push @buildinfo_opts, "--admindir=$admindir" if $admindir; +push @buildinfo_opts, "-O$buildinfo_file" if $buildinfo_file; + +run_hook('buildinfo', { + env => { + DPKG_BUILDPACKAGE_HOOK_BUILDINFO_OPTIONS => join(' ', @buildinfo_opts), + }, +}); +run_cmd('dpkg-genbuildinfo', @buildinfo_opts); + +$changes_file //= "../$pva.changes"; + +push @changes_opts, "--build=$build_types" if build_has_none(BUILD_DEFAULT); +push @changes_opts, "-m$maint" if defined $maint; +push @changes_opts, "-e$changedby" if defined $changedby; +push @changes_opts, "-v$since" if defined $since; +push @changes_opts, "-C$desc" if defined $desc; +push @changes_opts, "-O$changes_file"; + +my $changes = Dpkg::Control->new(type => CTRL_FILE_CHANGES); + +run_hook('changes', { + env => { + DPKG_BUILDPACKAGE_HOOK_CHANGES_OPTIONS => join(' ', @changes_opts), + }, +}); +run_cmd('dpkg-genchanges', @changes_opts); +$changes->load($changes_file); + +run_hook('postclean', { + enabled => $postclean, +}); + +if ($postclean) { + run_rules_cond_root('clean'); +} + +run_cmd('dpkg-source', @source_opts, '--after-build', '.'); + +info(describe_build($changes->{'Files'})); + +run_hook('check', { + enabled => $check_command, + env => { + DPKG_BUILDPACKAGE_HOOK_CHECK_OPTIONS => join(' ', @check_opts), + }, +}); + +if ($check_command) { + run_cmd($check_command, @check_opts, $changes_file); +} + +if ($signpause && ($signsource || $signbuildinfo || $signchanges)) { + print g_("Press <enter> to start the signing process.\n"); + getc(); +} + +run_hook('sign', { + enabled => $signsource || $signbuildinfo || $signchanges, +}); + +if ($signsource) { + signfile("$pv.dsc"); + + # Recompute the checksums as the .dsc has changed now. + my $buildinfo = Dpkg::Control->new(type => CTRL_FILE_BUILDINFO); + $buildinfo->load($buildinfo_file); + my $checksums = Dpkg::Checksums->new(); + $checksums->add_from_control($buildinfo); + $checksums->add_from_file("../$pv.dsc", update => 1, key => "$pv.dsc"); + $checksums->export_to_control($buildinfo); + $buildinfo->save($buildinfo_file); +} +if ($signbuildinfo) { + signfile("$pva.buildinfo"); +} +if ($signsource or $signbuildinfo) { + # Recompute the checksums as the .dsc and/or .buildinfo have changed. + my $checksums = Dpkg::Checksums->new(); + $checksums->add_from_control($changes); + $checksums->add_from_file("../$pv.dsc", update => 1, key => "$pv.dsc") + if $signsource; + $checksums->add_from_file($buildinfo_file, update => 1, key => "$pva.buildinfo"); + $checksums->export_to_control($changes); + delete $changes->{'Checksums-Md5'}; + update_files_field($changes, $checksums, "$pv.dsc") + if $signsource; + update_files_field($changes, $checksums, "$pva.buildinfo"); + $changes->save($changes_file); +} +if ($signchanges) { + signfile("$pva.changes"); +} + +if (not $signreleased) { + warning(g_('not signing UNRELEASED build; use --force-sign to override')); +} + +run_hook('done'); + +sub mustsetvar { + my ($var, $text) = @_; + + error(g_('unable to determine %s'), $text) + unless defined($var); + + info("$text $var"); + return $var; +} + +sub setup_rootcommand { + if ($< == 0) { + warning(g_('using a gain-root-command while being root')) if @rootcommand; + } else { + push @rootcommand, 'fakeroot' unless @rootcommand; + } + + if (@rootcommand and not find_command($rootcommand[0])) { + if ($rootcommand[0] eq 'fakeroot' and $< != 0) { + error(g_("fakeroot not found, either install the fakeroot\n" . + 'package, specify a command with the -r option, ' . + 'or run this as root')); + } else { + error(g_("gain-root-command '%s' not found"), $rootcommand[0]); + } + } +} + +sub parse_rules_requires_root { + my $ctrl = shift; + + my %rrr; + my $rrr; + my $rrr_default; + my $keywords_base; + my $keywords_impl; + + if (get_build_api($ctrl) >= 1) { + $rrr_default = 'no'; + } else { + $rrr_default = 'binary-targets'; + } + + my $ctrl_src = $ctrl->get_source(); + $rrr = $rrr_override // $ctrl_src->{'Rules-Requires-Root'} // $rrr_default; + + foreach my $keyword (split ' ', $rrr) { + if ($keyword =~ m{/}) { + if ($keyword =~ m{^dpkg/target/(.*)$}p and $target_official{$1}) { + error(g_('disallowed target in %s field keyword "%s"'), + 'Rules-Requires-Root', $keyword); + } elsif ($keyword =~ m{^dpkg/(.*)$} and $1 ne 'target-subcommand') { + error(g_('%s field keyword "%s" is unknown in dpkg namespace'), + 'Rules-Requires-Root', $keyword); + } + $keywords_impl++; + } else { + if ($keyword ne lc $keyword and + (lc $keyword eq 'no' or lc $keyword eq 'binary-targets')) { + error(g_('%s field keyword "%s" is uppercase; use "%s" instead'), + 'Rules-Requires-Root', $keyword, lc $keyword); + } elsif (lc $keyword eq 'yes') { + error(g_('%s field keyword "%s" is invalid; use "%s" instead'), + 'Rules-Requires-Root', $keyword, 'binary-targets'); + } elsif ($keyword ne 'no' and $keyword ne 'binary-targets') { + warning(g_('%s field keyword "%s" is unknown'), + 'Rules-Requires-Root', $keyword); + } + $keywords_base++; + } + + if ($rrr{$keyword}++) { + error(g_('field %s contains duplicate keyword "%s"'), + 'Rules-Requires-Root', $keyword); + } + } + + if ($call_target_as_root or not exists $rrr{no}) { + setup_rootcommand(); + } + + # Notify the children we do support R³. + $ENV{DEB_RULES_REQUIRES_ROOT} = join ' ', sort keys %rrr; + + if ($keywords_base > 1 or $keywords_base and $keywords_impl) { + error(g_('%s field contains both global and implementation specific keywords'), + 'Rules-Requires-Root'); + } elsif ($keywords_impl) { + # Set only on <implementations-keywords>. + $ENV{DEB_GAIN_ROOT_CMD} = join ' ', @rootcommand; + } else { + # We should not provide the variable otherwise. + delete $ENV{DEB_GAIN_ROOT_CMD}; + } + + return %rrr; +} + +sub run_cmd { + my @cmd = @_; + + printcmd(@cmd); + system @cmd and subprocerr("@cmd"); +} + +sub rules_requires_root { + my $target = shift; + + return 1 if $call_target_as_root; + return 1 if $rules_requires_root{"dpkg/target/$target"}; + return 1 if $rules_requires_root{'binary-targets'} and $target_legacy_root{$target}; + return 0; +} + +sub run_rules_cond_root { + my $target = shift; + + my @cmd; + push @cmd, @rootcommand if rules_requires_root($target); + push @cmd, @debian_rules, $target; + + run_cmd(@cmd); +} + +sub run_hook { + my ($name, $opts) = @_; + my $cmd = $hook{$name}; + $opts->{enabled} //= 1; + + return if not $cmd; + + info("running hook $name"); + + my %hook_vars = ( + '%' => '%', + 'a' => $opts->{enabled} ? 1 : 0, + 'p' => $pkg // q{}, + 'v' => $version // q{}, + 's' => $sversion // q{}, + 'u' => $uversion // q{}, + ); + + my $subst_hook_var = sub { + my $var = shift; + + if (exists $hook_vars{$var}) { + return $hook_vars{$var}; + } else { + warning(g_('unknown %% substitution in hook: %%%s'), $var); + return "\%$var"; + } + }; + + $cmd =~ s/\%(.)/$subst_hook_var->($1)/eg; + + $opts->{env}{DPKG_BUILDPACKAGE_HOOK_NAME} = $name; + + # Set any environment variables for this hook invocation. + local @ENV{keys %{$opts->{env}}} = values %{$opts->{env}}; + + run_cmd($cmd); +} + +sub update_files_field { + my ($ctrl, $checksums, $filename) = @_; + + my $md5sum_regex = checksums_get_property('md5', 'regex'); + my $md5sum = $checksums->get_checksum($filename, 'md5'); + my $size = $checksums->get_size($filename); + my $file_regex = qr/$md5sum_regex\s+\d+\s+(\S+\s+\S+\s+\Q$filename\E)/; + + $ctrl->{'Files'} =~ s/^$file_regex$/$md5sum $size $1/m; +} + +sub signkey_validate { + return unless $signkey->type eq 'keyid'; + + if (length $signkey->handle <= 8) { + error(g_('short OpenPGP key IDs are broken; ' . + 'please use key fingerprints in %s or %s instead'), + '-k', 'DEB_SIGN_KEYID'); + } elsif (length $signkey->handle <= 16) { + warning(g_('long OpenPGP key IDs are strongly discouraged; ' . + 'please use key fingerprints in %s or %s instead'), + '-k', 'DEB_SIGN_KEYID'); + } +} + +sub signfile { + my $file = shift; + + printcmd("signfile $file"); + + my $signdir = tempdir('dpkg-sign.XXXXXXXX', CLEANUP => 1); + my $signfile = "$signdir/$file"; + + # Make sure the file to sign ends with a newline. + copy("../$file", $signfile); + open my $signfh, '>>', $signfile or syserr(g_('cannot open %s'), $signfile); + print { $signfh } "\n"; + close $signfh or syserr(g_('cannot close %s'), $signfile); + + my $status = $openpgp->inline_sign($signfile, "$signfile.asc", $signkey); + if ($status == OPENPGP_OK) { + move("$signfile.asc", "../$file") + or syserror(g_('cannot move %s to %s'), "$signfile.asc", "../$file"); + } else { + error(g_('failed to sign %s file: %s'), $file, + openpgp_errorcode_to_string($status)); + } + + return $status +} + +sub fileomitted { + my ($files, $regex) = @_; + + return $files !~ m/$regex$/m +} + +sub describe_build { + my $files = shift; + my $ext = compression_get_file_extension_regex(); + + if (fileomitted($files, qr/\.deb/)) { + # source-only upload + if (fileomitted($files, qr/\.diff\.$ext/) and + fileomitted($files, qr/\.debian\.tar\.$ext/)) { + return g_('source-only upload: Debian-native package'); + } elsif (fileomitted($files, qr/\.orig\.tar\.$ext/)) { + return g_('source-only, diff-only upload (original source NOT included)'); + } else { + return g_('source-only upload (original source is included)'); + } + } elsif (fileomitted($files, qr/\.dsc/)) { + return g_('binary-only upload (no source included)'); + } elsif (fileomitted($files, qr/\.diff\.$ext/) and + fileomitted($files, qr/\.debian\.tar\.$ext/)) { + return g_('full upload; Debian-native package (full source is included)'); + } elsif (fileomitted($files, qr/\.orig\.tar\.$ext/)) { + return g_('binary and diff upload (original source NOT included)'); + } else { + return g_('full upload (original source is included)'); + } +} + +sub build_target_fallback { + my $ctrl = shift; + + # If we are building rootless, there is no need to call the build target + # independently as non-root. + return if not rules_requires_root($binarytarget); + + return if $buildtarget eq 'build'; + return if scalar @debian_rules != 1; + + # Avoid further heuristics in newer dpkg-build-api levels. + return if get_build_api($ctrl) >= 1; + + # Check if we are building both arch:all and arch:any packages, in which + # case we now require working build-indep and build-arch targets. + my $pkg_arch = 0; + + foreach my $bin ($ctrl->get_packages()) { + if ($bin->{Architecture} eq 'all') { + $pkg_arch |= BUILD_ARCH_INDEP; + } else { + $pkg_arch |= BUILD_ARCH_DEP; + } + } + + return if $pkg_arch == BUILD_BINARY; + + # Check if the build-{arch,indep} targets are supported. If not, fallback + # to build. + my $pid = spawn(exec => [ $Dpkg::PROGMAKE, '-f', @debian_rules, '-qn', $buildtarget ], + from_file => '/dev/null', to_file => '/dev/null', + error_to_file => '/dev/null'); + my $cmdline = "make -f @debian_rules -qn $buildtarget"; + wait_child($pid, nocheck => 1, cmdline => $cmdline); + my $exitcode = WEXITSTATUS($?); + subprocerr($cmdline) unless WIFEXITED($?); + if ($exitcode == 2) { + warning(g_("%s must be updated to support the 'build-arch' and " . + "'build-indep' targets (at least '%s' seems to be " . + 'missing)'), "@debian_rules", $buildtarget); + $buildtarget = 'build'; + } +} |