#!/bin/sh # # pkgfs - Builds source distribution files and stows/unstows them into # a master directory, thanks to Xstow. # #- # Copyright (c) 2008 Juan Romero Pardines. # All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions # are met: # 1. Redistributions of source code must retain the above copyright # notice, this list of conditions and the following disclaimer. # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR # IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES # OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. # IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, # INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT # NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY # THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF # THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. #- # # TODO # - milosn says: "pkgfs.sh -c pkgfs.conf build glib" work # - Multiple distfiles in a package. # - Multiple URLs to download source distribution files. # - Support GNU/BSD-makefile style source distribution files. # - Fix PKGFS_{C,CXX}FLAGS, aren't passed to the environment yet. # - Support adding filters to templates to avoid creating useless links. # # Default path to configuration file, can be overriden # via the environment or command line. # : ${PKGFS_CONFIG_FILE:=/usr/local/etc/pkgfs.conf} : ${PKGFS_REGISTERED_PKG_DB:=.pkgfs-registered-pkgfs.db} : ${progname:=$(basename $0)} : ${topdir:=$(/bin/pwd -P 2>/dev/null)} : ${fetch_cmd:=/usr/bin/ftp -a} : ${cksum_cmd:=/usr/bin/cksum -a rmd160} : ${awk_cmd:=/usr/bin/awk} : ${mkdir_cmd:=/bin/mkdir -p} : ${tar_cmd:=/usr/bin/tar} : ${unzip_cmd:=/usr/pkg/bin/unzip} : ${rm_cmd:=/bin/rm} : ${mv_cmd:=/bin/mv} : ${cp_cmd:=/bin/cp} : ${sed_cmd=/usr/bin/sed} : ${grep_cmd=/usr/bin/grep} : ${gunzip_cmd:=/usr/bin/gunzip} : ${bunzip2_cmd:=/usr/bin/bunzip2} : ${patch_cmd:=/usr/bin/patch} : ${find_cmd:=/usr/bin/find} : ${file_cmd:=/usr/bin/file} : ${ln_cmd:=/bin/ln} : ${chmod_cmd:=/bin/chmod} : ${db_cmd:=/usr/bin/db -q} : ${xstow_version:=xstow-0.6.1-unstable} : ${xstow_args:=-ap} usage() { cat << _EOF $progname: [-bCef] [-c ] Targets info Show information about . install Build and install package from . remove Remove package completely (unstow and remove from destdir) stow Create symlinks from in master directory. unstow Remove symlinks from in master directory. Options -b Only build the source distribution file(s). -C Do not remove build directory after successful build. -c Path to global configuration file. If not specified /usr/local/etc/pkgfs.conf is used. -e Only extract the source distribution file(s). -f Only fetch the source distribution file(s). _EOF exit 1 } check_path() { eval local orig="$1" case "$orig" in /) ;; /*) orig="${orig%/}" ;; *) orig="$topdir/${orig%/}" ;; esac path_fixed="$orig" } run_file() { local file="$1" check_path "$file" . $path_fixed } info_tmpl() { local tmpl="$1" if [ -z "$tmpl" -o ! -f "$tmpl" ]; then echo -n "*** ERROR: invalid template file '$tmpl'," echo ", aborting ***" exit 1 fi run_file ${tmpl} echo " pkgfs template definitions:" echo echo " pkgname: $pkgname" for i in "${distfiles}"; do [ -n "$i" ] && echo " distfile: $i" done echo " URL: $url" echo " maintainer: $maintainer" [ -n $checksum ] && echo " checksum: $checksum" echo " build_style: $build_style" echo " short_desc: $short_desc" echo "$long_desc" } apply_tmpl_patches() { if [ -z "$PKGFS_TEMPLATESDIR" ]; then echo -n "*** WARNING: PKGFS_TEMPLATESDIR is not set, " echo "won't apply patches ***" return 1 fi # # If package needs some patches applied before building, # apply them now. # if [ -n "$patch_files" ]; then for i in ${patch_files}; do patch="$PKGFS_TEMPLATESDIR/$i" if [ ! -f "$patch" ]; then echo "*** WARNING: unexistent patch '$i' ***" continue fi # Try to guess if its a compressed patch. if $(echo $patch|$grep_cmd -q .gz); then $gunzip_cmd $patch patch=${patch%%.gz} elif $(echo $patch|$grep_cmd -q .bz2); then $bunzip2_cmd $patch patch=${patch%%.bz2} elif $(echo $patch|$grep_cmd -q .diff); then # nada else echo "*** WARNING: unknown patch type '$i' ***" continue fi cd $pkg_builddir && $patch_cmd < $patch 2>/dev/null if [ "$?" -eq 0 ]; then echo ">> Patch applied: \`$i'." else echo -n "*** ERROR: couldn't apply patch '$i'" echo ", aborting ***" exit 1 fi done fi } check_config_vars() { run_file ${PKGFS_CONFIG_FILE} PKGFS_CONFIG_FILE=$path_fixed if [ ! -f "$PKGFS_CONFIG_FILE" ]; then echo -n "*** ERROR: cannot find configuration file: " echo "'$PKGFS_CONFIG_FILE' ***" exit 1 fi if [ -z "$PKGFS_DEPSDIR" ]; then echo -n "**** ERROR: PKGFS_DEPSDIR not set in configuration " echo "file, aborting ***" exit 1 fi local PKGFS_VARS="PKGFS_MASTERDIR PKGFS_DESTDIR PKGFS_BUILDDIR \ PKGFS_SRC_DISTDIR" for f in ${PKGFS_VARS}; do eval val="\$$f" if [ -z "$val" ]; then echo "**** ERROR: '$f' not set in configuration " echo "file, aborting ***" exit 1 fi if [ ! -d "$f" ]; then $mkdir_cmd "$val" if [ "$?" -ne 0 ]; then echo -n "*** ERROR: couldn't create '$f'" echo "directory, aborting ***" exit 1 fi fi done } reset_tmpl_vars() { local TMPL_VARS="pkgname extract_sufx distfiles url configure_args \ make_build_args make_install_args build_style \ short_desc maintainer long_desc checksum wrksrc \ patch_files configure_env make_cmd" for i in ${TMPL_VARS}; do eval unset "$i" done } check_tmpl_vars() { local pkg="$1" local dfile="" [ -z "$pkg" ] && return 1 REQ_VARS="pkgname extract_sufx url build_style" # Check if required vars weren't set. for i in ${REQ_VARS}; do eval val="\$$i" if [ -z "$val" -o -z "$i" ]; then echo -n "*** ERROR: $i not set (incomplete template" echo " build file), aborting ***" exit 1 fi done if [ -z "$distfiles" ]; then dfile="$pkgname$extract_sufx" elif [ -n "${distfiles}" ]; then dfile="$distfiles$extract_sufx" else echo "*** ERROR unsupported fetch state ***" exit 1 fi dfile="$PKGFS_SRC_DISTDIR/$dfile" case "$extract_sufx" in .tar.bz2|.tar.gz|.tgz|.tbz) extract_cmd="$tar_cmd xfz $dfile -C $PKGFS_BUILDDIR" ;; .tar) extract_cmd="$tar_cmd xf $dfile -C $PKGFS_BUILDDIR" ;; .zip) extract_cmd="$unzip_cmd -x $dfile -C $PKGFS_BUILDDIR" ;; *) echo -n "*** ERROR: unknown 'extract_sufx' argument in build " echo "file ***" exit 1 ;; esac } check_rmd160_cksum() { local file="$1" [ -z "$file" ] && return 1 if [ -z "${distfiles}" ]; then dfile="$pkgname$extract_sufx" elif [ -n "${distfiles}" ]; then dfile="$distfiles$extract_sufx" else dfile="$file$extract_sufx" fi origsum="$checksum" dfile="$PKGFS_SRC_DISTDIR/$dfile" filesum="$($cksum_cmd $dfile | $awk_cmd '{print $4}')" if [ "$origsum" != "$filesum" ]; then echo "*** WARNING: checksum doesn't match (rmd160) ***" return 1 fi } fetch_tmpl_sources() { local file="" local file2="" [ -z "$pkgname" ] && return 1 if [ -z "$distfiles" ]; then file="$pkgname" else file="$distfiles" fi for f in "$file"; do file2="$f$extract_sufx" if [ -f "$PKGFS_SRC_DISTDIR/$file2" ]; then check_rmd160_cksum $f if [ "$?" -eq 0 ]; then if [ -n "$only_fetch" ]; then echo "=> checksum ok" exit 0 fi return 0 fi fi echo ">>> Fetching \'$file2' source tarball" cd $PKGFS_SRC_DISTDIR && $fetch_cmd $url/$file2 if [ "$?" -ne 0 ]; then if [ ! -f $PKGFS_SRC_DISTDIR/$file2 ]; then echo -n "*** ERROR: couldn't fetch '$file2', " echo "aborting ***" else echo -n "*** ERROR: there was an error " echo "fetching '$file2', aborting ***" fi exit 1 else if [ -n "$only_fetch" ]; then echo "=> checksum ok" exit 0 fi fi done } extract_tmpl_sources() { [ -z "$pkgname" ] && return 1 echo ">>> Extracting \`$pkgname' into $PKGFS_BUILDDIR." $extract_cmd if [ "$?" -ne 0 ]; then echo -n "*** ERROR: there was an error extracting the " echo "distfile, aborting *** " exit 1 fi unset extract_cmd [ -n "$only_extract" ] && exit 0 } fixup_tmpl_libtool() { local lt_file="$pkg_builddir/libtool" # # If package has a libtool file replace it with ours, so that # we use the master directory while relinking, all will be fine # once the package is stowned. # if [ -f "$lt_file" -a -f "$PKGFS_MASTERDIR/bin/libtool" ]; then $rm_cmd -f $pkg_builddir/libtool $rm_cmd -f $pkg_builddir/ltmain.sh $ln_cmd -s $PKGFS_MASTERDIR/bin/libtool $lt_file $ln_cmd -s $PKGFS_MASTERDIR/share/libtool/config/ltmain.sh \ $pkg_builddir/ltmain.sh fi } build_tmpl_sources() { local pkg_builddir="" [ -z "$pkgname" ] && return 1 export PATH="/bin:/sbin:/usr/bin:/usr/sbin:$PKGFS_MASTERDIR/bin:$PKGFS_MASTERDIR/sbin" if [ -z "$wrksrc" ]; then if [ -z "$distfiles" ]; then pkg_builddir=$PKGFS_BUILDDIR/$pkgname else pkg_builddir=$PKGFS_BUILDDIR/$distfiles fi else pkg_builddir=$PKGFS_BUILDDIR/$wrksrc fi if [ ! -d "$pkg_builddir" ]; then echo "*** ERROR: unexistent build directory, aborting ***" exit 1 fi # Apply patches if requested by template file apply_tmpl_patches echo ">>> Building \`$pkgname' (be patient, may take a while)" # # For now, just set LDFLAGS. # export LDFLAGS="-L$PKGFS_MASTERDIR/lib -Wl,-R$PKGFS_MASTERDIR/lib" # # Packages using GNU autoconf # if [ "$build_style" = "gnu_configure" ]; then for i in ${configure_env}; do [ -n "$i" ] && export $i done cd $pkg_builddir ./configure --prefix="$PKGFS_MASTERDIR" ${configure_args} \ --mandir="$PKGFS_DESTDIR/$pkgname/man" # # Packages using propietary configure scripts. # elif [ "$build_style" = "configure" ]; then cd $pkg_builddir if [ -n "$configure_script" ]; then ./$configure_script ${configure_args} else ./configure ${configure_args} fi fi if [ "$?" -ne 0 ]; then echo "*** ERROR building (configure state) $pkgname ***" exit 1 fi if [ -z "$make_cmd" ]; then MAKE_CMD="/usr/bin/make" else MAKE_CMD="$make_cmd" fi # Fixup libtool script if necessary fixup_tmpl_libtool ${MAKE_CMD} ${make_build_args} if [ "$?" -ne 0 ]; then echo "*** ERROR building (make stage) $pkgname ***" exit 1 fi ${MAKE_CMD} ${make_install_args} \ install prefix="$PKGFS_DESTDIR/$pkgname" if [ "$?" -ne 0 ]; then echo "*** ERROR instaling $pkgname ***" exit 1 fi unset LDFLAGS echo ">>> Installed \`$pkgname' into the master directory." if [ -d "$pkg_builddir" -a -z "$dontrm_builddir" ]; then $rm_cmd -rf $pkg_builddir [ "$?" -eq 0 ] && \ echo ">> Removed \`$pkgname' build directory." fi cd $PKGFS_BUILDDIR } stow_tmpl() { local pkg="$1" [ -z "$pkg" ] && return 2 $PKGFS_XSTOW_CMD -dir $PKGFS_DESTDIR -target $PKGFS_MASTERDIR \ ${xstow_args} $PKGFS_DESTDIR/$pkg if [ "$?" -ne 0 ]; then echo "*** ERROR: couldn't create symlinks for \`$pkg' ***" exit 1 else echo ">>> Created \`$pkg' symlinks into master directory." fi installed_tmpl_handler register $pkg } unstow_tmpl() { local pkg="$1" if [ -z "$pkg" ]; then echo "*** ERROR: template wasn't specified? ***" exit 1 fi local tmppkg="${pkg%-[0-9]*}" if [ "$tmppkg" = "xstow" ]; then echo "*** INFO: You aren't allowed to unstow \`$xstow_version'." exit 1 fi $PKGFS_XSTOW_CMD -dir $PKGFS_DESTDIR -target $PKGFS_MASTERDIR \ -D $PKGFS_DESTDIR/$pkg if [ "$?" -ne 0 ]; then exit 1 else echo ">>> Removed \`$pkg' symlinks from master directory." fi installed_tmpl_handler unregister $pkg } add_dependency_tolist() { local pkgdepf="$1" local reg_pkgdb="${PKGFS_DESTDIR}/${PKGFS_REGISTERED_PKG_DB}" [ -z "$pkgdepf" ] && return 1 [ -n "$prev_depf" ] && pkgdepf=${prev_depf} for i in $($db_cmd btree $pkgdepf deps); do # # Skip key # [ "$i" = "deps" ] && continue echo -n " $i: " # # Check if dep already installed. # if [ -r "$reg_pkgdb" ]; then check_installed_tmpl $i if [ "$?" -eq 0 ]; then echo "already installed." continue fi echo "not installed" # Added dep into list deps_list="$i $deps_list" [ -n "$prev_depf" ] && unset prev_depf # # Check if dependency needs more deps. # depdbf="$PKGFS_DEPSDIR/$i-deps.db" if [ -r "$PKGFS_DEPSDIR/$i-deps.db" ]; then add_dependency_tolist ${depdbf} prev_depf="$depdbf" fi fi done } install_dependency_tmpl() { local pkgdepf="$1" local tmpdepf="$pkgdepf" local tmppkgname= deps_list= [ -z "$pkgdepf" ] && return 1 doing_deps=true tmp_pkgn=${pkgdepf%%-deps.db} echo ">>> Required dependencies for $(basename $tmp_pkgn):" add_dependency_tolist $pkgdepf for i in ${deps_list}; do echo ">>> Installing dependency: $i" install_tmpl "${PKGFS_TEMPLATESDIR}/${i%%-deps.db}.tmpl" done } install_xstow_tmpl() { [ -x "$PKGFS_XSTOW_CMD" ] && return 0 reset_tmpl_vars run_file "$PKGFS_TEMPLATESDIR/$xstow_version.tmpl" cur_tmpl=$path_fixed check_tmpl_vars ${xstow_version} fetch_tmpl_sources extract_tmpl_sources build_tmpl_sources PKGFS_XSTOW_CMD="$PKGFS_DESTDIR/$xstow_version/bin/xstow" stow_tmpl $xstow_version $sed_cmd -e "s|PKGFS_XSTOW_.*|PKGFS_XSTOW_CMD=$PKGFS_MASTERDIR/bin/xstow|" \ $path_fixed > $path_fixed.in && \ $mv_cmd $path_fixed.in $path_fixed # # Continue with origin package that called us. # run_file ${origin_tmpl} cur_tmpl=${path_fixed} } installed_tmpl_handler() { local action="$1" local pkg="$2" [ -z "$action" -o -z "$pkg" ] && return 1 # # This function is called every time a package has been stowned # or unstowned. # There's a db(3) btree database file in # PKGFS_DESTDIR/PKGFS_REGISTER_PKG_DB that stores which package # is installed and stowned. # if [ "$action" = "register" ]; then $db_cmd -w btree ${PKGFS_DESTDIR}/${PKGFS_REGISTERED_PKG_DB} \ $pkg stowned if [ "$?" -ne 0 ]; then echo -n "*** ERROR: couldn't register stowned \`$pkg'" echo " in db file ***" exit 1 fi elif [ "$action" = "unregister" ]; then $db_cmd -d btree ${PKGFS_DESTDIR}/${PKGFS_REGISTERED_PKG_DB} $pkg if [ "$?" -ne 0 ]; then echo -n "*** ERROR: \`$pkg' stowned not registered " echo "in db file? ***" exit 1 fi else return 1 fi } check_installed_tmpl() { local pkg="$1" local db_file="${PKGFS_DESTDIR}/${PKGFS_REGISTERED_PKG_DB}" [ -z "$pkg" ] && return 1 $db_cmd btree $db_file $pkg 2>&1 >/dev/null return $? } install_tmpl() { cur_tmpl="$1" if [ -z "$cur_tmpl" -o ! -f "$cur_tmpl" ]; then echo -n "*** ERROR: invalid template file '$cur_tmpl'," echo " aborting ***" exit 1 fi reset_tmpl_vars run_file ${cur_tmpl} cur_tmpl=${path_fixed} # # If we are the originator package save the path this template in # other var for future use. # if [ -z "$origin_tmpl" ]; then origin_tmpl=${path_fixed} fi # # Install xstow if it's not there. # install_xstow_tmpl # # Check vars for current template. # check_tmpl_vars ${pkgname} # # Handle required dependency for this template iff it # is not installed and if we are being invoked by a dependency. # local pkgdepf="$PKGFS_DEPSDIR/$pkgname-deps.db" if [ -r "$pkgdepf" -a -z "$doing_deps" ]; then install_dependency_tmpl ${pkgdepf} # # At this point all required deps are installed, and # only remaining is the origin template; install it. # doing_deps= reset_tmpl_vars run_file ${origin_tmpl} check_tmpl_vars ${pkgname} fi if [ -n "$only_build" ]; then build_tmpl_sources exit 0 fi # # Fetch, extract, stow and reset vars for a template. # fetch_tmpl_sources extract_tmpl_sources build_tmpl_sources stow_tmpl ${pkgname} } remove_tmpl() { local pkg="$1" if [ -z "$pkg" ]; then echo "*** ERROR: unexistent package, aborting ***" exit 1 fi if [ ! -d "$PKGFS_DESTDIR/$pkg" ]; then echo "*** ERROR: cannot find package on $PKGFS_DESTDIR ***" exit 1 fi unstow_tmpl ${pkg} $rm_cmd -rf $PKGFS_DESTDIR/$pkg return "$?" } # # main() # args=$(getopt bCc:ef $*) [ "$?" -ne 0 ] && usage set -- $args while [ "$#" -gt 0 ]; do case "$1" in -b) only_build=yes ;; -C) dontrm_builddir=yes ;; -c) PKGFS_CONFIG_FILE="$2" shift ;; -e) only_extract=yes ;; -f) only_fetch=yes ;; --) shift break ;; esac shift done [ "$#" -gt 2 ] && usage target="$1" if [ -z "$target" ]; then echo "*** ERROR missing target ***" usage fi # Check configuration vars before anyting else. check_config_vars # Main switch case "$target" in info) info_tmpl "$2" ;; install) install_tmpl "$2" ;; remove) remove_tmpl "$2" ;; stow) stow_tmpl "$2" ;; unstow) unstow_tmpl "$2" ;; *) echo "*** ERROR invalid target '$target' ***" usage esac # Agur exit 0