| File: | Dpkg/Source/Functions.pm |
| Coverage: | 65.8% |
| line | stmt | bran | cond | sub | pod | time | code |
|---|---|---|---|---|---|---|---|
| 1 | # Copyright © 2008-2010, 2012-2015 Guillem Jover <guillem@debian.org> | ||||||
| 2 | # | ||||||
| 3 | # This program is free software; you can redistribute it and/or modify | ||||||
| 4 | # it under the terms of the GNU General Public License as published by | ||||||
| 5 | # the Free Software Foundation; either version 2 of the License, or | ||||||
| 6 | # (at your option) any later version. | ||||||
| 7 | # | ||||||
| 8 | # This program is distributed in the hope that it will be useful, | ||||||
| 9 | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
| 10 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||||||
| 11 | # GNU General Public License for more details. | ||||||
| 12 | # | ||||||
| 13 | # You should have received a copy of the GNU General Public License | ||||||
| 14 | # along with this program. If not, see <https://www.gnu.org/licenses/>. | ||||||
| 15 | |||||||
| 16 | =encoding utf8 | ||||||
| 17 | |||||||
| 18 - 29 | =head1 NAME Dpkg::Source::Functions - miscellaneous source package handling functions =head1 DESCRIPTION This module provides a set of miscellaneous helper functions to handle source packages. B<Note>: This is a private module, its API can change at any time. =cut | ||||||
| 30 | |||||||
| 31 | package Dpkg::Source::Functions 0.01; | ||||||
| 32 | |||||||
| 33 | 24 24 24 | 57 14 323 | use strict; | ||||
| 34 | 24 24 24 | 86 20 725 | use warnings; | ||||
| 35 | |||||||
| 36 | our @EXPORT_OK = qw( | ||||||
| 37 | erasedir | ||||||
| 38 | fixperms | ||||||
| 39 | chmod_if_needed | ||||||
| 40 | fs_time | ||||||
| 41 | is_binary | ||||||
| 42 | ); | ||||||
| 43 | |||||||
| 44 | 24 24 24 | 81 14 322 | use Exporter qw(import); | ||||
| 45 | 24 24 24 | 1695 8040 1061 | use Errno qw(ENOENT); | ||||
| 46 | |||||||
| 47 | 24 24 24 | 378 26 859 | use Dpkg::ErrorHandling; | ||||
| 48 | 24 24 24 | 44 22 596 | use Dpkg::Gettext; | ||||
| 49 | 24 24 24 | 2122 18 560 | use Dpkg::File; | ||||
| 50 | 24 24 24 | 1006 20 6600 | use Dpkg::IPC; | ||||
| 51 | |||||||
| 52 | sub erasedir { | ||||||
| 53 | 6 | 0 | 9 | my $dir = shift; | |||
| 54 | 6 | 18 | if (not lstat($dir)) { | ||||
| 55 | 0 | 0 | return if $! == ENOENT; | ||||
| 56 | 0 | 0 | syserr(g_('cannot stat directory %s (before removal)'), $dir); | ||||
| 57 | } | ||||||
| 58 | 6 | 13496 | system 'rm', '-rf', '--', $dir; | ||||
| 59 | 6 | 109 | subprocerr("rm -rf $dir") if $?; | ||||
| 60 | 6 | 75 | if (not stat($dir)) { | ||||
| 61 | 6 | 196 | return if $! == ENOENT; | ||||
| 62 | 0 | 0 | syserr(g_("unable to check for removal of directory '%s'"), $dir); | ||||
| 63 | } | ||||||
| 64 | 0 | 0 | error(g_("rm -rf failed to remove '%s'"), $dir); | ||||
| 65 | } | ||||||
| 66 | |||||||
| 67 | sub fixperms { | ||||||
| 68 | 3 | 0 | 3 | my $dir = shift; | |||
| 69 | 3 | 3 | my ($mode, $modes_set); | ||||
| 70 | # Unfortunately tar insists on applying our umask _to the original | ||||||
| 71 | # permissions_ rather than mostly-ignoring the original | ||||||
| 72 | # permissions. We fix it up with chmod -R (which saves us some | ||||||
| 73 | # work) but we have to construct a u+/- string which is a bit | ||||||
| 74 | # of a palaver. (Numeric doesn't work because we need [ugo]+X | ||||||
| 75 | # and [ugo]=<stuff> doesn't work because that unsets sgid on dirs.) | ||||||
| 76 | 3 | 17 | $mode = 0777 & ~umask; | ||||
| 77 | 3 | 8 | for my $i (0 .. 2) { | ||||
| 78 | 9 | 10 | $modes_set .= ',' if $i; | ||||
| 79 | 9 | 9 | $modes_set .= qw(u g o)[$i]; | ||||
| 80 | 9 | 11 | for my $j (0 .. 2) { | ||||
| 81 | 27 | 23 | $modes_set .= $mode & (0400 >> ($i * 3 + $j)) ? '+' : '-'; | ||||
| 82 | 27 | 21 | $modes_set .= qw(r w X)[$j]; | ||||
| 83 | } | ||||||
| 84 | } | ||||||
| 85 | 3 | 4983 | system('chmod', '-R', '--', $modes_set, $dir); | ||||
| 86 | 3 | 63 | subprocerr("chmod -R -- $modes_set $dir") if $?; | ||||
| 87 | } | ||||||
| 88 | |||||||
| 89 | # Only change the pathname permissions if they differ from the desired. | ||||||
| 90 | # | ||||||
| 91 | # To be able to build a source tree, a user needs write permissions on it, | ||||||
| 92 | # but not necessarily ownership of those files. | ||||||
| 93 | sub chmod_if_needed { | ||||||
| 94 | 0 | 0 | 0 | my ($newperms, $pathname) = @_; | |||
| 95 | 0 | 0 | my $oldperms = (stat $pathname)[2] & 07777; | ||||
| 96 | |||||||
| 97 | 0 | 0 | return 1 if $oldperms == $newperms; | ||||
| 98 | 0 | 0 | return chmod $newperms, $pathname; | ||||
| 99 | } | ||||||
| 100 | |||||||
| 101 | # Touch the file and read the resulting mtime. | ||||||
| 102 | # | ||||||
| 103 | # If the file doesn't exist, create it, read the mtime and unlink it. | ||||||
| 104 | # | ||||||
| 105 | # Use this instead of time() when the timestamp is going to be | ||||||
| 106 | # used to set file timestamps. This avoids confusion when an | ||||||
| 107 | # NFS server and NFS client disagree about what time it is. | ||||||
| 108 | sub fs_time($) { | ||||||
| 109 | 6 | 0 | 6 | my $file = shift; | |||
| 110 | 6 | 6 | my $is_temp = 0; | ||||
| 111 | 6 | 38 | if (not -e $file) { | ||||
| 112 | 0 | 0 | file_touch($file); | ||||
| 113 | 0 | 0 | $is_temp = 1; | ||||
| 114 | } else { | ||||||
| 115 | 6 | 44 | utime(undef, undef, $file) or | ||||
| 116 | syserr(g_('cannot change timestamp for %s'), $file); | ||||||
| 117 | } | ||||||
| 118 | 6 | 20 | stat($file) or syserr(g_('cannot read timestamp from %s'), $file); | ||||
| 119 | 6 | 10 | my $mtime = (stat(_))[9]; | ||||
| 120 | 6 | 6 | unlink($file) if $is_temp; | ||||
| 121 | 6 | 12 | return $mtime; | ||||
| 122 | } | ||||||
| 123 | |||||||
| 124 | sub is_binary($) { | ||||||
| 125 | 0 | 0 | my $file = shift; | ||||
| 126 | |||||||
| 127 | # Perform the same check as diff(1), look for a NUL character in the first | ||||||
| 128 | # 4 KiB of the file. | ||||||
| 129 | 0 | open my $fh, '<', $file | |||||
| 130 | or syserr(g_('cannot open file %s for binary detection'), $file); | ||||||
| 131 | 0 | read $fh, my $buf, 4096, 0; | |||||
| 132 | 0 | my $res = index $buf, "\0"; | |||||
| 133 | 0 | close $fh; | |||||
| 134 | |||||||
| 135 | 0 | return $res >= 0; | |||||
| 136 | } | ||||||
| 137 | |||||||
| 138 - 144 | =head1 CHANGES =head2 Version 0.xx This is a private module. =cut | ||||||
| 145 | |||||||
| 146 | 1; | ||||||