X-Git-Url: https://pd.if.org/git/?a=blobdiff_plain;f=zpm-install;h=64b064bf7170d90e4b85f9e12b53374f41a82946;hb=62e2d81f8e5c1a2284660752524753638ae59766;hp=1886cf1760076da90b54002989cacb78432ba2b2;hpb=4d5cd305f3b6768810445c05de4cade0d4a7154e;p=zpackage diff --git a/zpm-install b/zpm-install index 1886cf1..64b064b 100755 --- a/zpm-install +++ b/zpm-install @@ -1,288 +1,240 @@ #!/bin/sh -# do one thing and do it well -# this program installs packages from a package file -# by default, it will install the latest version of each package -# found in the file. - -# todo - -# if package specifiers are given as arguments after the package file -# it will only install those packages. - -# package specifiers are of the form pkg[-ver[-rel]], i.e the -# release can be omitted, and the version-release can be omitted. -# examples: -# zpm install -f foo-0.1-1.zpm -# zpm install -f foo.zpm foo-0.1 - -# todo: file tag handling - -# what we need to install a package: - -# recording repo: where we record what we've done -# source: where we get the package and files from -# install root: really just a prefix, but thought about differently, -# and we might need to do a chroot -# package name to install, could be "all" or "all most recent" -# but for now, assume only one package in a file? - -# zpm install [options] - -pkgver=${ZPMPACKAGEVER:-1.0} -pkgrel=${ZPMPACKAGEREL:-1} - -pkgroot=/ - -# allocate an install id, perhaps hash package and timestamp -# installid=$(echo $(date) $pkglist | zpm hash) - -# extract pre-scripts and run them -# get list of paths to install -# for each path, if directory, create and set mode -# for each path, create leading directories -# extract to something like /var/tmp/.hash, with low perms -# only atomic if same filesystem, so use .installid.hash in the -# install directory -# set perms and such -# move into place -# after all the files, extract post scripts and run them - -# also need to mark package as installing so if it fails part way -# through, it can be finished later -# probably also want an option to "backup" any packages being upgraded -# so it's trivial to downgrade or to revert if the install fails - -# option for "multipackage" just to let the system know that's what you meant -# option to take filenames from stdin -# parse package, version, release from file if not given -# TODO what's the difference between prefix and pkgroot -# need an option to not chown the files -# option to install but not merge/track - -# options -# -R install root, if installing as root, will chroot? -# -C no chroot, even if root -# -N no pre-scripts -# -X no post-scripts -# -f source repository file -# -d local (recording) repository database -# -D don't locally record -# -t only files matching tags -# -T exclude files matching tags - -chroot=1 - -while getopts :R:CNXf:d:Dt:T:u:g: opt; do +warn() { + echo $* 1>&2 +} + +die() { + echo $* 1>&2 + zpm log -i -a 'aborting install' -t "$pkgid" "$*" + exit 1 +} + +dryrun=0 +verbose=0 +runscripts=1 +runconfigure=1 +absorb=0 +overwrite=0 +syncopts='' + +# zpm-install [-SCn] [ -d localdb ] [ -f pkgfile ] [ -R installroot ] pkgstr ... +while getopts f:d:R:nSCvAO opt; do case $opt in - R) pkgroot="$OPTARG" ;; - C) chroot=0 ;; f) pkgfile="$OPTARG" ;; - u) user="$OPTARG" ;; - g) group="$OPTARG" ;; - d) ZPMDB="$OPTARG" ;; + d) localdb="$OPTARG" ;; + R) rootdir="$OPTARG" ;; + S) runscripts=0 ;; + C) runconfigure=0 ;; + A) absorb=1; syncopts="${syncopts} -A" ;; + O) overwrite=1; syncopts="${syncopts} -O" ;; + n) dryrun=1 ;; + v) verbose=1 ;; + *) die "usage ..." ;; esac done +shift $(( OPTIND - 1)) -shift $((OPTIND - 1)) +pkgid=$1 -for cf in /etc/zpmrc ~/.zpmrc ./.zpmrc; do - test -r $cf && . $cf -done +if [ -z "$pkgid" ]; then + die "must specify pkgid" +fi -: ${ZPMDB:=/var/lib/zpm/db.zpm} +eval "$(zpm parse -E $pkgid)" -export ZPMDB +if [ -z "$pkgfile" ]; then + pkgfile=$ZPM_PACKAGE_FILE +fi -die() { - printf 'zpm-install:' 1>&2 - printf ' %s' $* 1>&2 - printf '\n' 1>&2 - exit 1 -} +# cases R = full package id, F = specified package file -if [ -z "$pkgfile" ]; then - # actually, if no pkgfile, get pkgfile from repo - # but need to loop over finding package files then - # so this program probably needs to be "install from pkgfile" - # and a separate one that will loop over a package - # spec list and find from repos - die "must specify package file" - pkgfile="$package-$pkgver-$pkgrel.zpm" +# immediate error +# -- 00 error, must specify something +if [ -z "$release" ] && [ -z "$pkgfile" ]; then + die must specify package file or complete package id fi -set -e -zpm test -v $pkgfile +# TODO look in package file +# -F 01 error, wouldn't know which pkgid to create, could derive from file? +if [ -z "$release" ]; then + # must have package file, or would have died above + pkgid=$(zpm findpkg -f $pkgfile $pkgid) + if [ -n "$pkgid" ]; then + eval "$(zpm parse -E $pkgid)" + fi +fi -if [ $# -gt 0 ]; then - pkglist="$@" -else - pkglist=$(zpm findpkg $pkgfile) +if [ -z "$pkgid" ]; then + die "$0 can't figure out a package id" fi -pathlist() { - zpm shell $pkgfile<' $pkg ;; - n) printf 'ok to install %s\n' $pkg ;; - esac + # remove the package if it exists. can't be in an installed + # state, would have bailed above. So, if it exists, can only + # be 'upgraded'. This should be fast, because we won't run + # a garbage collect, so any possible duplicate files between + # an upgraded or removed files, if they're there, won't need + # to be merged. + + # only merge if localdb and pkgfile are different + if [ "$pkgfile" != "$ZPMDB" ]; then + zpm rmpackage $pkgid + zpm merge -f $pkgfile -s installing $pkgid + if [ $? -ne 0 ]; then + die "merging $pkgid failed" + fi + else + zpm pkg $pkgid status=installing + fi - zpm merge $pkgfile $pkg - # TODO find scripts marked as pre-install - #run_preinstall + #zpm shell $ZPMDB 'select * from install_status' 1>&2 + if [ $dryrun -gt 0 ]; then + #zpm list -v + #zpm shell $ZPMDB 'select * from install_status' + zpm syncfs $syncopts -nv -f $pkgfile + zpm pkg $pkgid status=dryrun + continue + fi - # TODO if this is an upgrade, run pre-upgrade - #if [ $upgrade != 'n' ]; then - # run_preupgrade - #fi + if [ $verbose -gt 0 ]; then + syncopts="${syncopts} -v" + fi - # TODO get total size of install so we can do a progress bar - #tsize=totalsize + zpm syncfs $syncopts -f $pkgfile - # add package info to local package db - #zpm merge $pkgfile $pkg + if [ $? -ne 0 ]; then + zpm pkg $pkgid status=failed + die 'zpm-syncfs failed'; + fi - # check for conflicts + if [ $runscripts -gt 0 ]; then + zpm script -f $pkgfile -p post-install $pkgid $current + fi - # mark package in localdb as installing - # zpm setmark installing $pkg - # install all the files for a package - # TODO install directories first in order of path length - echo installing $pkglist - filecount=$(pathlist | wc -l) - echo $filecount files - #die 'aborting for test' - #pathlist - pathlist | xargs zpm ipkgfile ${user+-u $user} ${group+-g $group} -R "$pkgroot" -f $pkgfile -- $pkg + if [ -n "$current" ]; then + zpm pkg $pkgid status=installed :$current status=updated + else + zpm pkg $pkgid status=installed + fi - # TODO find scripts marked as post-install - # TODO if this is an upgrade, run post-upgrade + if [ $(id -u) -eq 0 ]; then + if [ ! -d $rootdir/etc ]; then + warn "no etc directory in $rootdir, skipping ldconfig" + elif [ -f $rootdir/sbin/ldconfig ]; then + $rootdir/sbin/ldconfig -r ${rootdir:-/} + elif [ -f /sbin/ldconfig ]; then + /sbin/ldconfig -r ${rootdir:-/} + else + true + fi + fi - # mark as ready for install - #zpm setmark ready $pkg + # TODO skip configure if not on a terminal, regardless of settings + # TODO will need force option + if [ $runconfigure -gt 0 ]; then + zpm script -f $pkgfile -p configure $pkgid $current + else + true + fi done - -#zpm setmark installed $pkglist