diff options
author | Paul Eggert <eggert@cs.ucla.edu> | 2006-12-01 20:52:52 +0000 |
---|---|---|
committer | Paul Eggert <eggert@cs.ucla.edu> | 2006-12-01 20:52:52 +0000 |
commit | 873e56c3ba906ae217d25e85d7515e8f4c775e32 (patch) | |
tree | 297a0996d95686a314392427d442d35c5bea73ff /gzexe.in | |
parent | d28d9e4506cd5f353766535f9d9460cb28eb9b72 (diff) | |
download | gzip-873e56c3ba906ae217d25e85d7515e8f4c775e32.tar.gz |
* NEWS: Describe gzexe changes noted below.
* Makefile.am (.in): Don't bother with SEDCMD. This stuff isn't needed
any more (the hosts it caters to are long extinct), and was questionable
anyway since the code assumes the skip= line was line 2.
* configure.ac (AC_SYS_INTERPRETER, SEDCMD): Remove; no longer needed.
* gzexe.in: Sweep the code and fix some bugs.
My, what sharp teeth you have, gzexe!
The straw that broke this camel's back was Matthew Burgess's bug report
<http://lists.gnu.org/archive/html/bug-gzip/2006-11/msg00012.html>.
(IFS): Set it to the standard value, both here and in the
script gzexe generates.
Check for missing operand after parsing options, not before.
This fixes the case for "cpexe --".
Check for tail -n problem separately in gzexe and in the executable
it generates, in case it's a different 'tail'.
(trap): Remove $tmp only if $tmp is not the empty string.
Preserve exit status.
Don't use the nonstandard (and rarely available) "cpmod" utility.
Don't use "set -C"; it's no longer useful and it breaks things
in some cases.
(main loop): Handle file names beginning with "-".
Exit with status of failing program, not with 1.
Fix some bugs in printing diagnostics, and in quoting.
Require the skip= line to have at least one digit.
Use test -u and test -g rather than using the (less-reliable) ls.
Refuse to compress more programs, e.g., sh.
Use mktemp rather than tempfile. Build a copy of
the compressed or uncompressed executable in the same
directory as the executable, as that's less likely to go wrong
if disk space is low. Have the executable exit with status 127,
not 1, if the decompression process fails; this is more compatible
with meta-programs like nohup. Have the executable
uncompress to a temp file with the same basename as the executable;
this is more likely to go right. Fix a race condition where the
executable temporarily did not exist (in either old or new forms).
Check for race conditions better when mv fails. Do not attempt
to use cp to do the real work, only to copy permissions to a temp
file, since we don't want to trash running executables.
Diffstat (limited to 'gzexe.in')
-rw-r--r-- | gzexe.in | 224 |
1 files changed, 122 insertions, 102 deletions
@@ -1,4 +1,3 @@ -: #!/bin/sh # gzexe: compressor for Unix executables. # Use this only for binaries that you do not use frequently. @@ -14,7 +13,7 @@ # On Ultrix, /bin/sh is too buggy, change the first line to: #!/bin/sh5 -# Copyright (C) 1998, 2002, 2004 Free Software Foundation +# Copyright (C) 1998, 2002, 2004, 2006 Free Software Foundation # Copyright (C) 1993 Jean-loup Gailly # This program is free software; you can redistribute it and/or modify @@ -31,6 +30,11 @@ # with this program; if not, write to the Free Software Foundation, Inc., # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. +tab=' ' +nl=' +' +IFS=" $tab$nl" + version='gzexe (gzip) @VERSION@ Copyright (C) 2006 Free Software Foundation, Inc. This is free software. You may redistribute copies of it under the terms of @@ -50,10 +54,6 @@ Report bugs to <bug-gzip@gnu.org>.' PATH="BINDIR:$PATH" -if test $# = 0; then - echo "$usage" - exit 1 -fi decomp=0 res=0 @@ -67,134 +67,154 @@ while :; do esac done -tmp=gz$$ -trap "rm -f $tmp; exit 1" 1 2 3 5 10 13 15 - -set -C -echo hi > $tmp || exit -if test -z "`(${CPMOD-cpmod} $tmp $tmp) 2>&1`"; then - cpmod=${CPMOD-cpmod} -fi - -tail="" -IFS="${IFS= }"; saveifs="$IFS"; IFS="${IFS}:" -for dir in $PATH; do - test -z "$dir" && dir=. - if test -f $dir/tail; then - tail="$dir/tail" - break - fi -done -IFS="$saveifs" -if test -z "$tail"; then - echo cannot find tail +if test $# -eq 0; then + echo >&2 "$0: missing operand +Try \`$0 --help' for more information." exit 1 fi -case `echo foo | $tail -n +1 2>/dev/null` in -foo) tail="$tail -n";; -esac + +tmp= +trap ' + res=$? + test -n "$tmp" && rm -f "$tmp" + (exit $res); exit $res +' 0 1 2 3 5 10 13 15 for i do - if test ! -f "$i" ; then - echo ${x}: $i not a file - res=1 + case $i in + -*) file=./$i;; + *) file=$i;; + esac + if test ! -f "$file" || test ! -r "$file"; then + res=$? + echo >&2 "$0: $i is not a readable regular file" continue fi if test $decomp -eq 0; then - if sed -e 1d -e 2q "$i" | grep "^skip=[0-9]*$" >/dev/null; then - echo "${x}: $i is already gzexe'd" + if sed -e 1d -e 2q "$file" | grep "^skip=[0-9][0-9]*$" >/dev/null; then + echo >&2 "$0: $i is already gzexe'd" continue fi fi - if ls -l "$i" | grep '^...[sS]' > /dev/null; then - echo "${x}: $i has setuid permission, unchanged" + if test -u "$file"; then + echo >&2 "$0: $i has setuid permission, unchanged" continue fi - if ls -l "$i" | grep '^......[sS]' > /dev/null; then - echo "${x}: $i has setgid permission, unchanged" + if test -g "$file"; then + echo >&2 "$0: $i has setgid permission, unchanged" continue fi - case "$i" in - */gzip | */tail | */sed | */chmod | */ln | */sleep | */rm) - echo "${x}: $i would depend on itself"; continue ;; + case /$file in + */basename | */cat | */chmod | */cp | \ + */dirname | */echo | */expr | */gzip | \ + */ln | */mkdir | */mktemp | */mv | */rm | */rmdir | \ + */sed | */sh | */sleep | */test | */tail) + echo >&2 "$0: $i would depend on itself"; continue;; esac - if test -z "$cpmod"; then - cp -p "$i" $tmp 2>/dev/null || cp "$i" $tmp - if test -w $tmp 2>/dev/null; then - writable=1 - else - writable=0 - chmod u+w $tmp 2>/dev/null - fi - : >| $tmp + + dir=`dirname "$file"` || dir=$TMPDIR + test -d "$dir" && test -w "$dir" && test -x "$dir" || dir=/tmp + test -n "$tmp" && rm -f "$tmp" + tmp=`TMPDIR=$dir mktemp -t gzexeXXXXXX` || tmp=$dir/gzexe$$ + cp -p "$file" "$tmp" 2>/dev/null || cp "$file" "$tmp" || { + res=$? + echo >&2 "$0: cannot copy $file" + continue + } + if test -w "$tmp"; then + writable=1 + else + writable=0 + chmod u+w "$tmp" || { + res=$? + echo >&2 "$0: cannot chmod $tmp" + continue + } fi if test $decomp -eq 0; then - (sed 1q $0 && - sed "s|^if tail|if $tail|" <<'EOF' && -skip=26 -set -C + (cat <<'EOF' && +#!/bin/sh +skip=43 + +tab=' ' +nl=' +' +IFS=" $tab$nl" + umask=`umask` umask 77 -if (tempfile --version) >/dev/null 2>&1 -then gztmp=`tempfile -p gztmp` || exit -else gztmp=/tmp/gztmp$$ -fi -if tail +$skip "$0" | "BINDIR"/gzip -cd > "$gztmp"; then + +gztmpdir= +trap ' + res=$? + test -n "$gztmpdir" && rm -fr "$gztmpdir" + (exit $res); exit $res +' 0 1 2 3 5 10 13 15 + +gztmpdir=`(mktemp -dt) 2>/dev/null` || + { gztmpdir=/tmp/gztmp$$; mkdir $gztmpdir; } || + { (exit 127); exit 127; } + +gztmp=$gztmpdir/$0 +case $0 in +-* | */*' +') mkdir -p "$gztmp" && rmdir "$gztmp";; +*/*) gztmp=$gztmpdir/`basename "$0"`;; +esac || { (exit 127); exit 127; } + +case `echo X | tail -n +1 2>/dev/null` in +X) tail_n=-n;; +*) tail_n=;; +esac +if tail $tail_n +$skip <"$0" | gzip -cd > "$gztmp"; then umask $umask - /bin/chmod 700 "$gztmp" - prog=`echo "$gztmp" | /bin/sed 's|[^/]*$||'; echo $0 | /bin/sed 's|.*/||'` - if /bin/ln "$gztmp" "$prog" 2>/dev/null; then - trap '/bin/rm -f "$gztmp" "$prog"; exit $res' 0 - (/bin/sleep 5; /bin/rm -f "$gztmp" "$prog") 2>/dev/null & - "$prog" ${1+"$@"}; res=$? - else - trap '/bin/rm -f "$gztmp"; exit $res' 0 - (/bin/sleep 5; /bin/rm -f "$gztmp") 2>/dev/null & - "$gztmp" ${1+"$@"}; res=$? - fi + chmod 700 "$gztmp" + (sleep 5; rm -fr "$gztmpdir") 2>/dev/null & + "$gztmp" ${1+"$@"}; res=$? else - echo Cannot decompress $0; exit 1 + echo >&2 "Cannot decompress $0" + (exit 127); res=127 fi; exit $res EOF - gzip -cv9 "$i") > $tmp || { - /bin/rm -f $tmp - echo ${x}: compression not possible for $i, file unchanged. - res=1 + gzip -cv9 "$file") > "$tmp" || { + res=$? + echo >&2 "$0: compression not possible for $i, file unchanged." continue } else # decompression - skip=26 - if sed -e 1d -e 2q "$i" | grep "^skip=[0-9][0-9]*$" >/dev/null; then - eval `sed -e 1d -e 2q "$i"` - fi - if $tail +$skip "$i" | gzip -cd > $tmp; then - : - else - echo ${x}: $i probably not in gzexe format, file unchanged. - res=1 + skip=43 + skip_line=`sed -e 1d -e 2q "$file"` + case $skip_line in + skip=[0-9] | skip=[0-9][0-9] | skip=[0-9][0-9][0-9]) + eval "$skip_line";; + esac + case `echo X | tail -n +1 2>/dev/null` in + X) tail_n=-n;; + *) tail_n=;; + esac + tail $tail_n +$skip "$file" | gzip -cd > "$tmp" || { + res=$? + echo >&2 "$0: $i probably not in gzexe format, file unchanged." continue - fi + } fi - rm -f "$i~" - mv "$i" "$i~" || { - echo ${x}: cannot backup $i as $i~ - rm -f $tmp - res=1 + test $writable -eq 1 || chmod u-w "$tmp" || { + res=$? + echo >&2 "$0: $tmp: cannot chmod" continue } - mv $tmp "$i" || cp -p $tmp "$i" 2>/dev/null || cp $tmp "$i" || { - echo ${x}: cannot create $i - rm -f $tmp - res=1 + ln -f "$file" "$file~" || { + res=$? + echo >&2 "$0: cannot backup $i as $i~" continue } - rm -f $tmp - if test -n "$cpmod"; then - $cpmod "$i~" "$i" 2>/dev/null - elif test $writable -eq 0; then - chmod u-w $i 2>/dev/null - fi + mv -f "$tmp" "$file" || { + res=$? + echo >&2 "$0: cannot rename $tmp to $i" + continue + } + tmp= done -exit $res +(exit $res); exit $res |