OSSP CVS Repository

ossp - ossp-pkg/shtool/sh.tarball
Not logged in
[Honeypot]  [Browse]  [Directory]  [Home]  [Login
[Reports]  [Search]  [Ticket]  [Timeline
  [Raw

ossp-pkg/shtool/sh.tarball
##
##  tarball -- Roll distribution tarballs
##  Copyright (c) 1999-2008 Ralf S. Engelschall <rse@engelschall.com>
##
##  This file is part of shtool and 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 file 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, write to the Free Software
##  Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
##  USA, or contact Ralf S. Engelschall <rse@engelschall.com>.
##

str_tool="tarball"
str_usage="[-t|--trace] [-v|--verbose] [-o|--output <tarball>] [-c|--compress <prog>] [-d|--directory <dir>] [-u|--user <user>] [-g|--group <group>] [-e|--exclude <pattern>] <path> [<path> ...]"
gen_tmpfile=yes
arg_spec="1+"
opt_spec="t.v.o:c:d:u:g:e:"
opt_alias="t:trace,v:verbose,o:output,c:compress,d:directory,u:user,g:group,e:exclude"
opt_t=no
opt_v=no
opt_o=""
opt_c=""
opt_d=""
opt_u=""
opt_g=""
opt_e="CVS,\\.cvsignore,\\.svn,\\.[oa]\$"

. ./sh.common

srcs="$*"

#   check whether the test command supports the -x option
if [ -x /bin/sh ] 2>/dev/null; then
    minusx="-x"
else
    minusx="-r"
fi

#   find the tools
paths="`echo $PATH |\
        sed -e 's%/*:%:%g' -e 's%/*$%%' \
            -e 's/^:/.:/' -e 's/::/:.:/g' -e 's/:$/:./' \
            -e 's/:/ /g'`"
for spec in find:gfind,find tar:gtar,tar tardy:tardy,tarcust; do
    prg=`echo $spec | sed -e 's/:.*$//'`
    tools=`echo $spec | sed -e 's/^.*://'`
    eval "prg_${prg}=''"
    #   iterate over tools
    for tool in `echo $tools | sed -e 's/,/ /g'`; do
        #   iterate over paths
        for path in $paths; do
            if [ $minusx "$path/$tool" ] && [ ! -d "$path/$tool" ]; then
                eval "prg_${prg}=\"$path/$tool\""
                break
            fi
        done
        eval "val=\$prg_${prg}"
        if [ ".$val" != . ]; then
            break
        fi
    done
done

#   expand source paths
exclude=''
for pat in `echo $opt_e | sed 's/,/ /g'`; do
    exclude="$exclude | grep -v '$pat'"
done
if [ ".$opt_t" = .yes ]; then
    echo "cp /dev/null $tmpfile.lst" 1>&2
fi
cp /dev/null $tmpfile.lst
for src in $srcs; do
    if [ -d $src ]; then
        if [ ".$opt_t" = .yes ]; then
            echo "(cd $src && $prg_find . -type f -depth -print) | sed -e 's:^\\.\$::' -e 's:^\\./::' | cat $exclude >>$tmpfile.lst" 1>&2
        fi
        (cd $src && $prg_find . -type f -depth -print) |\
        sed -e 's:^\.$::' -e 's:^\./::' | eval cat $exclude >>$tmpfile.lst
    else
        if [ ".$opt_t" = .yes ]; then
            echo "echo $src >>$tmpfile.lst" 1>&2
        fi
        echo $src >>$tmpfile.lst
    fi
done
sort <$tmpfile.lst >$tmpfile.lst.n
mv $tmpfile.lst.n $tmpfile.lst
if [ ".$opt_v" = .yes ]; then
    cat $tmpfile.lst | sed -e 's/^/  /' 1>&2
fi

#   determine tarball file and directory name
if [ ".$opt_o" != . ]; then
    tarfile="$opt_o"
    if [ ".$opt_d" != . ]; then
        tarname="$opt_d"
    else
        tarname=`echo $tarfile | sed -e 's/\.tar.*$//' -e 's;.*/\([^/]*\)$;\1;'`
    fi
else
    if [ ".$opt_d" != . ]; then
        tarname="$opt_d"
    elif [ -d "$from" ]; then
        tarname=`echo $from | sed -e 's;.*/\([^/]*\)$;\1;'`
    else
        tarname="out"
    fi
    tarfile="$tarname.tar"
fi

#   roll the tarball
compress=''
if [ ".$opt_c" != . ]; then
    compress="| $opt_c"
fi
if [ ".$prg_tardy" != . ]; then
    #   the elegant hackers way
    tardy_opt="--prefix=$tarname"
    tardy_opt="$tardy_opt --user_number=0 --group_number=0" # security!
    if [ ".$opt_u" != . ]; then
        tardy_opt="$tardy_opt --user_name=$opt_u"
    fi
    if [ ".$opt_g" != . ]; then
        tardy_opt="$tardy_opt --group_name=$opt_g"
    fi
    if [ ".$opt_t" = .yes ]; then
        echo "cat $tmpfile.lst | xargs $prg_tar cf - | $prg_tardy $tardy_opt | cat $compress >$tmpfile.out" 1>&2
    fi
    cat $tmpfile.lst |\
    xargs $prg_tar cf - |\
    $prg_tardy $tardy_opt |\
    eval cat $compress >$tmpfile.out
    if [ ".$opt_t" = .yes ]; then
        echo "cp $tmpfile.out $tarfile" 1>&2
    fi
    cp $tmpfile.out $tarfile
else
    #  the portable standard way
    if [ ".$opt_t" = .yes ]; then
        echo "mkdir $tmpdir/$tarname" 1>&2
    fi
    mkdir $tmpdir/$tarname || shtool_exit 1
    if [ ".$opt_t" = .yes ]; then
        echo "cat $tmpfile.lst | xargs $prg_tar cf - | (cd $tmpdir/$tarname && $prg_tar xf -)" 1>&2
    fi
    cat $tmpfile.lst |\
    xargs $prg_tar cf - |\
    (cd $tmpdir/$tarname && $prg_tar xf -)
    if [ ".$opt_u" != . ]; then
        if [ ".$opt_t" = .yes ]; then
            echo "chown -R $opt_u $tmpdir/$tarname >/dev/null 2>&1" 2>&1
        fi
        chown -R $opt_u $tmpdir/$tarname >/dev/null 2>&1 ||\
        echo "$msgprefix:Warning: cannot set user name \`$opt_u' (would require root privileges)"
    fi
    if [ ".$opt_g" != . ]; then
        if [ ".$opt_t" = .yes ]; then
            echo "chgrp -R $opt_g $tmpdir/$tarname >/dev/null 2>&1" 2>&1
        fi
        chgrp -R $opt_g $tmpdir/$tarname >/dev/null 2>&1 ||\
        echo "$msgprefix:Warning: cannot set group name \`$opt_g' (would require root privileges)"
    fi
    if [ ".$opt_t" = .yes ]; then
        echo "(cd $tmpdir && $prg_find $tarname -type f -depth -print | sort | xargs $prg_tar cf -) | cat $compress >$tmpfile.out" 1>&2
    fi
    (cd $tmpdir && $prg_find $tarname -type f -depth -print | sort | xargs $prg_tar cf -) |\
    eval cat $compress >$tmpfile.out
    if [ ".$opt_t" = .yes ]; then
        echo "cp $tmpfile.out $tarfile" 1>&2
    fi
    cp $tmpfile.out $tarfile
    if [ ".$opt_t" = .yes ]; then
        echo "rm -rf $tmpdir/$tarname" 1>&2
    fi
    rm -rf $tmpdir/$tarname
fi

#   cleanup
if [ ".$opt_t" = .yes ]; then
    echo "rm -f $tmpfile.lst $tmpfile.out" 1>&2
fi
rm -f $tmpfile.lst $tmpfile.out

shtool_exit 0

##
##  manual page
##

=pod

=head1 NAME

B<shtool tarball> - B<GNU shtool> command for rolling standardized tarballs

=head1 SYNOPSIS

B<shtool tarball>
[B<-t>|B<--trace>]
[B<-v>|B<--verbose>]
[B<-o>|B<--output> I<tarball>]
[B<-c>|B<--compress> I<prog>]
[B<-d>|B<--directory> I<directory>]
[B<-u>|B<--user> I<user>]
[B<-g>|B<--group> I<group>]
[B<-e>|B<--exclude> I<pattern>]
I<path> [I<path> ...]

=head1 DESCRIPTION

This command is for rolling input files under I<path> into a
distribution tarballs which can be extracted by tar(1).

The four important aspects of good open source software tarballs are:
(1) unpack into a single top-level directory, (2) top-level directory
corresponds to the tarball filename, (3) tarball files should be sorted
and (4) arbitrary names for file owner and group.

=head1 OPTIONS

The following command line options are available.

=over 4

=item B<-v>, B<--verbose>

Display some processing information.

=item B<-t>, B<--trace>

Enable the output of the essential shell commands which are executed.

=item B<-o>, B<--output> I<tarball>

Output tarball to file I<tarball>.

=item B<-c>, B<--compress> I<prog>

Pipe resulting tarball through compression program I<prog>.

=item B<-d>, B<--directory> I<directory>

Sets the top-level directory into which the tarball unpacks.
By default it is I<tarball> without the trailing C<.tar.*> extension.

=item B<-u>, B<--user> I<user>

The user (owner) of files and directories in the tarball to I<user>.

=item B<-g>, B<--group> I<group>

The group of files and directories in the tarball to I<group>.

=item B<-e>, B<--exclude> I<pattern>

Exclude files and directories matching comma-separated list of
regex I<pattern> from the tarball. Directories are expanded
before the filtering takes place. The default filter pattern is
"C<CVS,\\.cvsignore,\\.svn,\\.[oa]\$>".

=back

=head1 EXAMPLE

 #   Makefile.in
 dist:
     ...
     V=`shtool version -d short ...`; \
     shtool tarball -o foobar-$$V.tar.gz -c 'gzip -9' \
                    -u bar -g gnu -e 'CVS,\.cvsignore' .

=head1 HISTORY

The B<GNU shtool> B<tarball> command was originally written by Ralf S.
Engelschall E<lt>rse@engelschall.comE<gt> in 1999 for B<GNU shtool>.

=head1 SEE ALSO

shtool(1), tar(1), compress(1).

=cut


CVSTrac 2.0.1