#!/usr/bin/env zsh ## pkg - /home package manager and zsh playground ## Copyright (C) 2008 Daniel Friesel ## ## Permission to use, copy, modify, and/or distribute this software for any ## purpose with or without fee is hereby granted, provided that the above ## copyright notice and this permission notice appear in all copies. ## ## THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES ## WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF ## MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ## ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES ## WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ## ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF ## OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. typeset -a -U triggers ## ## Internal functions for displaying stuff ## info=$'\e[0;36m' error=$'\e[0;31m' reset=$'\e[0m' function info { (( SILENT )) || echo -ne "${info}$*${reset}" } function warn { echo -ne "${error}$*${reset}" > /dev/stderr } function die { echo -ne "${error}$*${reset}" > /dev/stderr exit 100 } function say { (( SILENT )) || echo $* } function check_installed { [[ -n $1 && -d $PDIR/$1 ]] || die "Not installed: '$1'\n" } function check_valid { [[ -d $PDIR/$1/.git ]] || die "Not a valid package name: '$1'\n" } function clear_line { (( SILENT )) || echo -ne "\r\e[2K" } # Read local configuration if [[ -f $HOME/.pkg.conf ]] { . $HOME/.pkg.conf } # Default values... or not [[ -n $PDIR ]] || PDIR="$HOME/packages" [[ -n $PKG_ROOT ]] || die "No PKG_ROOT specified. Please edit ~/.pkg.conf\n" [[ -n $CL_OPTIONS ]] || CL_OPTIONS=(-q) [[ -n $SILENT ]] || SILENT=0 [[ -n $AUTOUPDATE ]] || AUTOUPDATE=1 export PDIR export PKG_ROOT ## ## Make sure everything's sane ## Warn otherwise ## if [[ ! -d $PDIR ]] { die "$PDIR not found!!\n" } ## ## Some additional variables related to PKG_ROOT ## # Protocol if [[ ${PKG_ROOT#ssh} != $PKG_ROOT ]] { PKG_PROTO='ssh' } elif [[ ${PKG_ROOT#'/'} != $PKG_ROOT ]] { PKG_PROTO='file' } else { false } # Host if [[ $PKG_PROTO = 'ssh' ]] { PKG_HOST=${${PKG_ROOT#'ssh://'}%%/*} } # Remote path if [[ $PKG_PROTO = 'ssh' ]] { PKG_PATH=${PKG_ROOT#"ssh://$PKG_HOST"} } elif [[ $PKG_PROTO = 'file' ]] { PKG_PATH=$PKG_ROOT } ## ## Ask the user for confirmation ## # Default reply: Yes function confirm_yes { echo -n "$* [Y/n] " read -k 1 [[ $REPLY != $'\n' ]] && echo if [[ $REPLY = 'y' || $REPLY = 'Y' || $REPLY = $'\n' ]] { true } else { false } } # Default reply: No function confirm_no { echo -n "$* [y/N] " read -q } ## ## Major internal functions ## function progress { (( SILENT )) && return current=$1 max=$2 desc=$3 desc2=$4 output='' currentper=$(( (current*100)/max )) item () { for j in {0..$1}; { (( j > 0 )) && output+=$2 } } clear_line c=$(( currentper/5 )) a=$(( 20-c )) output+="${info}$desc${reset} [" item $c '=' item $a ' ' output+="] $currentper% $desc2" echo -ne $output } ## VCS Wrappers function vcs_to_list { if [[ -d $1/.git ]] { echo -n "$1 git " echo ${$(git --git-dir=$1/.git log -n 1)[2]} } else { warn "Unsupported or no repository: $1\n" } } function vcs_add { cd $PDIR case $(list_type $1) in git) git clone "$PKG_ROOT/$1" ;; *) die "$1: Cannot handle repository format '$(list_type $1)'\n" ;; esac } function vcs_log { git log } function vcs_upgrade { git pull $PKG_ROOT/${PWD:t} master } function vcs_push { git push $PKG_ROOT/${PWD:t} master } function vcs_status { git status } ## List stuff function list_is_installed { grep ^"$1 " $PDIR/.list } function list_exists { grep ^"$1 " $PDIR/.list-remote } function list_incoming { [[ $(list_local_version $1) != $(list_remote_version $1) ]] } function list_type { echo ${$(grep ^"$1 " $PDIR/.list-remote)[2]} } function list_type_local { echo ${$(grep ^"$1 " $PDIR/.list)[2]} } function list_update_remote { export PDIR if [[ $PKG_PROTO = 'ssh' ]] { ssh $PKG_HOST "PDIR='$PDIR' $PKG_PATH/core/include/pkglist $PKG_PATH" > .list-remote if [[ ${#$(cat .list-remote)} == 0 ]] { scp -q $PKG_HOST:$PKG_PATH/.list .list-remote warn "The pkglist version on your packages_root is outdated, please update\n" } } elif [[ $PKG_PROTO = 'file' ]] { $PKG_PATH/core/include/pkglist $PKG_PATH > .list-remote if [[ ${#$(cat .list-remote)} == 0 ]] { cp $PKG_ROOT/.list .list-remote warn "The pkglist version on your packages_root is outdated, please update\n" } } } function list_update_local { cd $PDIR rm -f .list all=${#$(echo $PDIR/*(/))} current=0 for i in *(/); { (( current++ )) progress $current $all 'Updating package list' $i vcs_to_list $i >> .list } } function list_update_package { cd $PDIR LIST=$(grep -v "^$1 " .list) echo $LIST > .list vcs_to_list $1 >> .list } function list_remove_package { cd $PDIR LIST=$(grep -v "^$1 " .list) echo $LIST > .list } function list_local_version { echo ${$(grep "^$1 " $PDIR/.list)[3]} } function list_remote_version { echo ${$(grep "^$1 " $PDIR/.list-remote)[3]} } # Return an understandable priority from the numeric one function real_priority { case $1 in 6) echo "essential" ;; 5) echo "important" ;; 4) echo "required" ;; 3) echo "standard" ;; 2) echo "optional" ;; 1) echo "extra" ;; *) echo ;; esac } # Execute a hook function exec_hook { package=$1 hook=$2 if [[ -r $PDIR/$package/hooks/$hook ]] { info "Executing $package $hook hook\n" cd $PDIR/$package (source hooks/$hook) } } # Check dependencies, conflicts etc. function check_prereqs { package=$1 [[ -r $PDIR/$package/prereqs ]] || return 0 cd $PDIR/$package info "checking prerequisites\n" typeset -a install is_installed () { [[ -d $PDIR/$1 ]] } perlmodule () { perl -M$1 < /dev/null 2> /dev/null } file_in_path () { which $1 > /dev/null } offer_install () { install+=$1 } source prereqs if [[ -n $install ]] { info "$1 requires the following packages: $install\n" if (confirm_yes "Install them?") { for i in $install; { pkg_add $i } } } } # Write a packages' files to .collected # Currently, this is only documentation function populate_collected { cd $PDIR/$1 || return info "Enabling documentation " if [[ -d man ]] { for i in man/*/*; { section=${i:h:t} manpage=${i:t} if (podchecker man/$section/$manpage &> /dev/null) { pod2man -s $section -c "$1 package" -r $HOME man/$section/$manpage > $PDIR/.collected/man/man$section/$manpage.$section say -n "+" } else { say -n "." } } } if [[ -d bin ]] { for i in bin/*; { if (podchecker $i &> /dev/null) { pod2man $i > $PDIR/.collected/man/man1/${i:t}.1 say -n "+" } else { say -n "." } } } clear_line if [[ -d bin ]] { for i in bin/*(*); { if [[ $(readlink $HOME/$i) != "../$1/$i" ]] { rm -f $HOME/$i ln -s ../${PDIR//$HOME\/}/$1/$i $HOME/$i } } } } # Remove a packages' files from .collected # Assuming there are no packages with colliding files # TODO: Make sure there are none function genocide_collected { cd $PDIR/$1 || return info "Removing documentation" if [[ -d man ]] { for i in man/*/*; { section=${i:h:t} manual=${i:t} if [[ -e $PDIR/.collected/man/man$section/$manual ]] { rm $PDIR/.collected/man/man$section/$manual } } } if [[ -d bin ]] { for i in bin/*; { rm -f $PDIR/.collected/man/man1/${i:t}.1 } } clear_line if [[ -d bin ]] { for i in bin/*(*); { if [[ $(readlink $HOME/$i) = "../${PDIR//$HOME\/}/$1/$i" ]] { rm -f $HOME/$i } } } } function update_provides { [[ -d $PDIR/$1/provides ]] || return cd $PDIR/$1/provides for package in *; { if [[ -d $PDIR/$package ]] { [[ $2 = 'remove' ]] && rm -r $package triggers+=$package } } } function apply_triggers { for package in $triggers; { exec_hook $package 'post-update' } } function wrap { function=$1 arg=$2 progress=$3 if [[ -n $2 ]] { $function $2 } else { cd $PDIR [[ -n $progress ]] && all=$(wc -l < $PDIR/.list) [[ -n $progress ]] && current=0 for i in *(/); { (( current++ )) [[ -n $progress ]] && progress $current $all $progress $i $function $i } [[ -n $progress ]] && clear_line } } ## ## Finally - the functions actually doing something ## function pkg_add { if [[ -d $PDIR/$1 ]] { info "Package '$1' is already installed!\n" return 100 } info "Retrieving package $1...\n" cd $PDIR || return 255 vcs_add $1 || return 255 cd $1 || return 255 check_prereqs $1 if [[ -r $PDIR/$1/Makefile ]] { info "Building binaries\n" make } exec_hook $1 'post-add' exec_hook $1 'post-update' checklinks $CL_OPTIONS populate_collected $1 update_provides $1 list_update_package $1 } function pkg_push { check_installed $1 check_valid $1 cd $PDIR/$1 list_incoming $1 if [[ $? = 0 ]] { clear_line info "Pushing $1\n" vcs_push check_prereqs $1 if [[ -r Makefile ]] { info "Building binaries\n" make } triggers+=$1 checklinks $CL_OPTIONS populate_collected $1 update_provides $1 } } function pkg_remove { check_installed $1 check_valid $1 cd $PDIR/$1 if [[ -r priority ]] { if (( $(cat priority) > 3 )) { confirm_no "Package '$1' is $(real_priority $(cat priority)). Really remove?" || return } } exec_hook $1 "pre-remove" genocide_collected $1 checklinks -r list_remove_package $1 update_provides $1 'remove' rm -rf $PDIR/$1 info "Package removed.\n" } function pkg_upgrade { check_installed $1 check_valid $1 cd $PDIR/$1 if [[ $(list_type $1) != $(list_type_local $1) ]] { clear_line warn "Incompatible systems. Please reinstall: $1\n" return 9 } if {list_incoming $1} { clear_line info "Updating $1 to $(list_remote_version $1)\n" vcs_upgrade check_prereqs $1 if [[ -r Makefile ]] { info "Building binaries\n" make } triggers+=$1 checklinks $CL_OPTIONS populate_collected $1 update_provides $1 list_update_package $1 } } function pkg_list_installed { cut -d ' ' -f 1 $PDIR/.list } function pkg_list_available { cut -d ' ' -f 1 $PDIR/.list-remote } function pkg_status { check_installed $1 cd $PDIR/$1 vcs_status=$(PAGER='' vcs_status $1) if [[ -n $vcs_status && $(echo $vcs_status | tail -n 1) != 'nothing to commit (working directory clean)' ]] { clear_line info "$1:\n" echo $vcs_status if [[ -r .autocommit ]] { git add . git commit -m 'autocommit by pkg' } } } function pkg_refresh { check_installed $1 cd $PDIR/$1 populate_collected $1 checklinks $CL_OPTIONS triggers+=$1 } function pkg_update { pkg_update_remote pkg_update_local } function pkg_update_remote { cd $PDIR info "Updating remote package list..." list_update_remote clear_line } function pkg_update_local { info "Updating local package list..." list_update_local clear_line } # Various information related to a package function pkg_info { cd $PDIR # Fetch the infos NAME=$1 LOCAL_VERSION=$(list_local_version $1) REMOTE_VERSION=$(list_remote_version $1) REPO_TYPE=$(list_type $1) if [[ -d $1 ]] { cd $1 if [[ -r priority ]] { PRIORITY=$(cat priority) PRIORITY_NAME=$(real_priority "$PRIORITY") } if [[ -r tags ]] { TAGS=$(cat tags | tr "\n" " " | sed 's/ /, /g' | sed 's/, $//') } if [[ -d hooks ]] { HOOKS=$(ls hooks) } if [[ -r Makefile ]] { MAKEFILE=1 } SIZE=$(du -sh .$(list_type $1) | grep -o '.*[KMG]') if [[ -r description ]] { DESCRIPTION=$(cat description) } state='installed' if list_incoming $1; then state+=', needs update' else state+=', up-to-date' fi } else { state='not installed' } function show_info { [[ -z $2 ]] && return info "$1: " echo $2 } show_info "Package" $NAME show_info 'State' $state [[ -n $PRIORITY ]] && show_info "Priority" "$PRIORITY ($PRIORITY_NAME)" show_info "Local Version" $LOCAL_VERSION show_info "Remote Version" $REMOTE_VERSION show_info "Repository Type" $REPO_TYPE show_info "Repository Size" $SIZE show_info "Tags" $TAGS show_info "Hooks" $HOOKS show_info "Description" $DESCRIPTION } function pkg_log { check_installed $1 cd $PDIR/$1 vcs_log } ## ## Now what shall we do... ## case $1 in add) pkg_add $2 ;; delete) pkg_remove $2 ;; info) pkg_info $2 ;; install) pkg_add $2 ;; list) pkg_list_installed ;; list-all) pkg_list_available ;; local-update) pkg_update_local ;; log) pkg_log $2 ;; push) (( AUTOUPDATE )) && pkg_update wrap pkg_push "$2" "Pushing" ;; refresh) wrap pkg_refresh "$2" "Refreshing" ;; remote-update) pkg_update_remote ;; remove) pkg_remove $2 ;; status) wrap pkg_status "$2" "Checking package status" ;; update) pkg_update ;; upgrade) (( AUTOUPDATE )) && pkg_update_remote wrap pkg_upgrade "$2" "Looking for updates" ;; eval) shift; eval $* ;; *) die "wait, what?\npkg: unknown action: '$1'\n" ;; esac apply_triggers