]> pd.if.org Git - zpackage/blobdiff - bin/zpm-merge
move programs to bin for build
[zpackage] / bin / zpm-merge
diff --git a/bin/zpm-merge b/bin/zpm-merge
new file mode 100755 (executable)
index 0000000..48d826f
--- /dev/null
@@ -0,0 +1,253 @@
+#!/bin/sh
+
+# merge
+# collects info from package file and copies it into another package file
+
+# merge [ -d ZPMDB ] [-f $pkgfile] pkg
+# -F include file content
+# -S don't include script content
+
+warn() {
+       printf 'zpm-merge:' 1>&2
+       printf ' %s' "$@" 1>&2
+       printf '\n' 1>&2
+       exit 1
+}
+
+die() {
+       warn "$@"
+       exit 1
+}
+
+verbose=0
+mergefiles=0
+mergescripts=1
+mergeall=1
+update=0
+remove_older=0
+only_if_newer=0
+one_at_a_time=0
+
+target=${ZPMDB:=/var/lib/zpm/local.db}
+
+# TODO option to only update if new
+# new is hash is different or build time is later
+# TODO option to merge all packages found in pkgfile
+while getopts :f:vd:FSs:auonO opt; do
+       case $opt in
+               f) pkgfile="$OPTARG" ;;
+               v) verbose=$(( verbose + 1 )) ;;
+               d) target="$OPTARG" ;;
+               F) mergefiles=1 ;;
+               S) mergescripts=0 ;;
+               s) newstatus="$OPTARG" ;;
+               a) mergeall=1 ;;
+               u) update=1 ;;
+               o) remove_older=1 ;;
+               n) only_if_newer=1 ;;
+               O) one_at_a_time=1 ;;
+               *) echo 'zpm-merge unknown option' $OPTARG; exit 1 ;;
+       esac
+done
+shift $((OPTIND - 1))
+
+# TODO test target
+if [ ! -e "$target" ]; then
+       die "target file $target does not exist"
+fi
+
+zpm test -v "$target" || exit 1
+
+if [ $# -eq 0 ]; then
+       mergeall=1
+else
+       mergeall=0
+fi
+
+# TODO if pkgfile is specified, allow multiple packages as args
+if [ $mergeall -eq 0 ]; then
+       pkgid=$1
+       shift
+
+       if [ -z "$pkgid" ]; then
+               die "must specify pkgid"
+       fi
+
+       eval "$(zpm parse -E $pkgid)"
+
+       if [ -z "$pkgfile" ]; then
+               pkgfile=$ZPM_PACKAGE_FILE
+       fi
+
+       # calculate package id, pkgfile, etc
+       # cases R = full package id, F = specified package file
+
+       # immediate error
+       # --- 000 error, must specify something
+       if [ -z "$release" ] && [ -z "$pkgfile" ]; then
+               die must specify package file or complete package id
+       fi
+
+       # try to get from package file
+       if [ -z "$release" ]; then
+               pkgid=$(zpm findpkg -f $pkgfile $pkgid)
+               if [ -z "$pkgid" ]; then
+                       die cannot find package id
+               fi
+               eval "$(zpm parse -E $pkgid)"
+       fi
+
+       # --F 001 error, wouldn't know which pkgid to create, could derive from file?
+       if [ -z "$release" ]; then
+               die must specify complete package id
+       fi
+
+       # set file from pkgid
+       # -R- 010 set file from pkgid, create in file, error if no file
+       if [ -z "$pkgfile" ]; then
+               pkgfile="$pkgid.zpm"
+       fi
+
+       if [ $verbose -gt 1 ]; then
+               echo merging $pkgfile $pkgid into $target
+       fi
+       pkglist=$pkgid
+else
+       pkglist=$(zpm list -f "$pkgfile")
+fi
+
+zpm test -v "$pkgfile" || exit 1
+
+if [ -n "$newstatus" ]; then
+       newstatus=$(zpm quote "$newstatus")
+fi
+
+mergeone() {
+       pkgid=$1
+       package=$(zpm parse -n "$pkgid")
+       where="where printf('%s-%s-%s', P.package, P.version, P.release) = '$pkgid'"
+       if [ $update -eq 1 ]; then
+               printf "delete from packages_pkgid where pkgid = '%s';\n" "$pkgid"
+       fi
+       if [ $remove_older -eq 1 ]; then
+               printf "delete from packages_pkgid where package = '$package' and pkgid < '%s' collate vercmp;\n" "$pkgid"
+       fi
+       printf "insert or rollback into packages select * from remote.packages P %s;\n" "$where"
+       printf "insert or rollback into packagefiles select * from remote.packagefiles P %s;\n" "$where"
+       printf "insert or rollback into scripts select * from remote.scripts P %s;\n" "$where"
+       if [ $mergefiles -eq 1 ]; then
+               printf "insert into files select F.* from remote.files F\n"
+               printf "inner join remote.packagefiles_pkgid P on P.hash = F.hash %s\n" "$where"
+               printf "on conflict (hash) do nothing;\n";
+       fi
+       # scripts
+       if [ $mergescripts -eq 1 ]; then
+               printf "insert into files select F.* from remote.files F\n"
+               printf "inner join remote.scripts_pkgid P on P.hash = F.hash where P.pkgid = '%s'\n" "$pkgid"
+               printf "on conflict (hash) do nothing;\n";
+       fi
+       if [ -n "$newstatus" ]; then
+               newstatus=$(zpm quote "$newstatus")
+               printf "update packages as P set status = '$newstatus' %s;\n" "$where"
+       fi
+}
+
+check_have() {
+       merged=$(zpm shell "$target" "select 1 from packages_pkgid where pkgid = '$1'")
+       test -n "$merged"
+}
+
+filter_out_have() {
+       mlist=
+       for pkgid in "$@"; do
+               if ! check_have "$pkgid" ; then
+                       mlist="$mlist $pkgid"
+               fi
+       done
+       mlist=${mlist#' '}
+       printf '%s' "$mlist"
+}
+
+filter_have() {
+       mlist=
+       for pkgid in "$@"; do
+               if check_have "$pkgid" ; then
+                       mlist="$mlist $pkgid"
+               fi
+       done
+       mlist=${mlist#' '}
+       printf '%s' "$mlist"
+}
+
+check_newer() {
+       pkg=$1
+       package=$(zpm parse -n "$pkg")
+       newer=$(zpm shell "$target" "select '$pkg' > (select max(pkgid collate vercmp) from packages_pkgid where package = '$package') collate vercmp")
+       if [ "$newer" = '1' ] || [ -z "$newer" ]; then
+               return 0
+       fi
+       return 1
+}
+
+filter_newer() {
+       mlist=
+       for pkgid in "$@"; do
+               if check_newer "$pkgid" ; then
+                       mlist="$mlist $pkgid"
+               fi
+       done
+       mlist=${mlist#' '}
+       printf '%s' "$mlist"
+}
+
+merge_list() {
+# TODO file tags and package tags
+(
+       printf ".bail on\n"
+       printf "attach '%s' as remote;\n" "$pkgfile"
+       printf "begin;\n"
+
+       for pkgid in "$@"; do
+               mergeone "$pkgid"
+       done
+
+       printf "insert or ignore into elfneeded select * from remote.elfneeded;\n"
+       printf "insert or ignore into elflibraries select * from remote.elflibraries;\n"
+       printf "insert or ignore into package_signatures select * from remote.package_signatures PS where PS.packagehash in (select hash from remote.packages);\n"
+
+       printf "commit;\n"
+) | zpm shell $target
+}
+
+# check for already merged packages
+if [ $update -eq 0 ] && [ $only_if_newer -eq 0 ]; then
+       have=$(filter_have $pkglist)
+       if [ -n "$have" ]; then
+               die "already merged: $have"
+       fi
+fi
+
+if [ $update -eq 0 ]; then
+       pkglist=$(filter_out_have $pkglist)
+fi
+
+if [ $only_if_newer -eq 1 ]; then
+       pkglist=$(filter_newer $pkglist)
+       if [ -z "$pkglist" ]; then
+               exit 0
+       fi
+fi
+
+if [ $one_at_a_time -eq 1 ]; then
+       for pkgid in $pkglist; do
+               if [ $verbose -ne 0 ]; then
+                       printf '%s\n' $pkgid
+               fi
+               merge_list $pkgid
+       done
+else
+       if [ $verbose -ne 0 ]; then
+               printf "%s\n" $pkglist
+       fi
+       merge_list $pkglist
+fi