summaryrefslogtreecommitdiff
path: root/scripts
diff options
context:
space:
mode:
Diffstat (limited to 'scripts')
-rw-r--r--scripts/Makefile19
-rwxr-xr-xscripts/apk_create195
-rwxr-xr-xscripts/libutil.sh275
3 files changed, 489 insertions, 0 deletions
diff --git a/scripts/Makefile b/scripts/Makefile
new file mode 100644
index 0000000..817205b
--- /dev/null
+++ b/scripts/Makefile
@@ -0,0 +1,19 @@
+# Makefile - one file to rule them all, one file to bind them
+#
+# Copyright (C) 2007 Timo Teräs <timo.teras@iki.fi>
+# All rights reserved.
+#
+# This program is free software; you can redistribute it and/or modify it
+# under the terms of the GNU General Public License version 3 as published
+# by the Free Software Foundation. See http://www.gnu.org/ for details.
+
+all:
+
+clean:
+
+install::
+ $(INSTALLDIR) $(DESTDIR)$(SBINDIR)
+ $(INSTALL) apk_create $(DESTDIR)$(SBINDIR)
+ $(INSTALLDIR) $(DESTDIR)/lib/apk
+ $(INSTALL) libutil.sh $(DESTDIR)/lib/apk
+
diff --git a/scripts/apk_create b/scripts/apk_create
new file mode 100755
index 0000000..6bbb9ee
--- /dev/null
+++ b/scripts/apk_create
@@ -0,0 +1,195 @@
+#!/bin/sh
+#
+# apk_create - a utility for creating software packages.
+#
+# Copyright (c) 2005 Natanael Copa
+#
+# Distributed under GPL-2
+#
+
+PROGRAM=apk_create
+
+#load the libraries
+. "${APK_LIBS:=/lib/apk}/libutil.sh"
+#. "$APK_LIBS/libfetch.sh"
+#. "$APK_LIBS/libindex.sh"
+#. "$APK_LIBS/libdb.sh"
+#. "$APK_LIBS/libpkgf.sh"
+
+# print usage and die
+usage() {
+ echo "$PROGRAM $VERSION"
+ echo "
+Usage: apk_create [hvS] [-a [+]arch] [-c [+]desc] [-i iscript] [-I piscript]
+ [-k dscript] [-K pdscript] [-l [+]license] [-m metadir] [-o owners]
+ [-p package] [-P [+]pkgs] [-s srcdir] [-u uscript] [-U puscript]
+ [-x excludepattern] [-X excludefile ] [-w [+]uri] pkg-filename
+
+Options:
+ -a Read package architecture from arch or use the argument itself, if
+ preceded with an '+'.
+ -c Read package description from desc or use the argument itself, if
+ preceded with an '+'.
+ -f Use fakeroot when creating the package.
+ -h Show help and exit.
+ -i Use iscript as pre-install script.
+ -I Use piscript as post-install script.
+ -k Use dscript as deinstall script.
+ -K Use pdscript as post-deinstall script.
+ -l Read package license from license or use the argument itself, if
+ preceded with an '+'.
+ -m Use all metafiles found in metadir.
+ -o Use file as OWNERS file.
+ -p Use package as source in addition to or instead of source dir.
+ -P Read space separated dependencies from pkgs or use the argument itself,
+ if preceded with an '+'.
+ -s Set source directory to srcdir.
+ -S Strip ELF files with sstrip.
+ -T Include files listed in includefile.
+ -u Use uscript as pre-update script.
+ -U Use puscript as post-upgrade script.
+ -v Turn on verbose output.
+ -w Read webpage URI from uri or use the argument itself. if preceded
+ with an '+'.
+ -x Exclude files matching excludepattern.
+ -X Exclude files listed in excludefile.
+ "
+ exit 1
+}
+
+# if $1 start with a '+', strip the '+' and echo the reset
+# otherwise read contents from file $1
+get_arg_or_file() {
+ local tmp
+ # we can do this with awk
+ # but we want it more readable...
+ # echo "$1" | awk '
+ # /^\+/ { print substr($0, 2)}
+ # /^[^\+]/ { readline < $0 ; print}'
+ if echo "$1" | grep '^+' > /dev/null; then
+ echo "$1" | sed 's/^\+//'
+ return 0
+ fi
+ cat "$1" || die "Could not read file '$1'"
+}
+
+#parse args
+unset vflag
+
+
+while getopts "a:c:fhi:I:k:K:l:m:o:p:P:Ss:T:u:U:vX:x:w:" opt ; do
+ case "$opt" in
+ a) ARCH=`get_arg_or_file "$OPTARG"`;;
+ h) usage;;
+ c) DESC=`get_arg_or_file "$OPTARG"`;;
+ i) PRE_INSTALL="$OPTARG";;
+ I) POST_INSTALL="$OPTARG";;
+ k) DSCRIPT="$OPTARG";;
+ K) POST_DEINSTALL="$OPTARG";;
+ l) LICENSE=`get_arg_or_file "$OPTARG"`;;
+ m) METADIR="$OPTARG";;
+ o) OWNERS="$OPTARG";;
+ p) case "$OPTARG" in
+ *.tar.bz) UNTAR="tar -zxf \"$OPTARG\"";;
+ *.tar.bz2) UNTAR="tar -jxf \"$OPTARG\"";;
+ *) die "Only .tar.gz and .tar.bz2 are supported";;
+ esac;;
+ P) DEPEND=`get_arg_or_file "$OPTARG"`;;
+ s) SRCDIR="$OPTARG";;
+ S) SSTRIP="-S";;
+ T) INCLUDE_FROM="$INCLUDE_FROM --include-from=$OPTARG";;
+ u) PRE_UPDATE="$OPTARG";;
+ U) POST_UPDATE="$OPTARG";;
+ v) VERBOSE="-v" ;;
+ x) EXCLUDE="$EXCLUDE --exclude=$OPTARG";;
+ X) EXCLUDE_FROM="$EXCLUDE_FROM --exclude-from=$OPTARG";;
+ w) WWW=`get_arg_or_file "$OPTARG"`;;
+ \?) usage;;
+ esac
+done
+shift `expr $OPTIND - 1`
+
+# if -s and -p is not specified, use current dir as default.
+[ -z "$SRCDIR$UNTAR" ] && SRCDIR="."
+
+[ $# -ne 1 ] && usage
+
+which rsync > /dev/null || die "This program depends on rsync."
+
+if [ "`whoami`" != "root" ] ; then
+ SUDO=sudo
+fi
+
+# find absolute path to package
+case "$1" in
+ /*) PKGVF="$1";;
+ *) PKGVF="$PWD/$1";;
+esac
+
+PV="`basename \"$PKGVF\" .apk`"
+echo "$PV" | grep -- '-[0-9].*' > /dev/null || die "Package name $PKGVF does not have any version."
+P="`echo $PV | sed 's/-[0-9].*//'`"
+V="`echo $PV | sed 's/^'$P'-//'`"
+
+# clean temp dir
+tmp="$APK_TMPDIR/$PV"
+rm -rf "$tmp"
+
+# copy files
+if [ "$UNTAR" ] ; then
+ $UNTAR $INCLUDE_FROM $EXCLUDE $EXCLUDE_FROM -C "$tmp" || \
+ die "Failed to unpack"
+fi
+
+if [ "$SRCDIR" ] ; then
+ rsync -ra $INCLUDE_FROM $EXCLUDE $EXCLUDE_FROM "$SRCDIR/." "$tmp" ||\
+ die "Failed to copy files from '$SRCDIR/.' to '$tmp'".
+fi
+
+# run sstrip
+[ "$SSTRIP" ] && find "$tmp" -type f \
+ | xargs -n1 file \
+ | grep ELF \
+ | grep -v relocatable \
+ | cut -d : -f 1 \
+ | while read f ; do
+ $SUDO sstrip "$f"
+ done
+
+# create meta data
+dbdir=`beautify_path "$tmp/$APK_DBDIR"`
+db="$dbdir/$PV"
+mkdir -p "$db"
+
+if [ "$METADIR" ] ; then
+ cp $METADIR/* "$db/" || die "Failed to copy files from '$METADIR'"
+fi
+[ "$ARCH" ] && echo "$ARCH" > "$db/ARCH"
+[ "$DEPEND" ] && echo "$DEPEND" > "$db/DEPEND"
+[ "$DESC" ] && echo "$DESC" > "$db/DESC"
+[ "$LICENSE" ] && echo "$LICENSE" > "$db/LICENSE"
+[ "$WWW" ] && echo "$WWW" > "$db/WWW"
+
+[ "$PRE_INSTALL" ] && cp -a "$PRE_INSTALL" "$db/pre-install"
+[ "$POST_INSTALL" ] && cp -a "$POST_INSTALL" "$db/post-install"
+[ "$DSCRIPT" ] && cp -a "$DSCRIPT" "$db/pre-deinstall"
+[ "$POST_DEINSTALL" ] && cp -a "$POST_DEINSTALL" "$db/post-deinstall"
+[ "$PRE_UPDATE" ] && cp -a "$PRE_UPDATE" "$db/pre-update"
+[ "$POST_UPDATE" ] && cp -a "$POST_UPDATE" "$db/post-update"
+[ "$OWNERS" ] && cp -a "$OWNERS" "$db/OWNERS"
+
+# check if var/db/apk contains more than it should
+[ `ls "$dbdir" | wc -l` -gt 1 ] && die "$APK_DBDIR should only contain the directory $PV."
+
+# check if arch, desc, license and www info is there
+[ -f "$db/ARCH" ] || die "Architecture not set. Use -a."
+[ -f "$db/DESC" ] || die "Description not set. Use -c."
+[ -f "$db/LICENSE" ] || die "License not set. Use -l."
+[ -f "$db/WWW" ] || eecho "Warning: Homepage not set (use -w)."
+
+# now, lets create the package
+cd "$tmp"
+tar --owner=root --group=root $VERBOSE -zcf "$PKGVF" *
+
+# cleanup
+rm -rf "$tmp"
diff --git a/scripts/libutil.sh b/scripts/libutil.sh
new file mode 100755
index 0000000..229b275
--- /dev/null
+++ b/scripts/libutil.sh
@@ -0,0 +1,275 @@
+#!/bin/sh
+
+# libutil.sh - Utility functions
+#
+# Copyright(c) 2005 Natanael Copa
+#
+# Distributed under GPL-2
+#
+
+VERSION=0.13.1
+
+# echo to stderr
+eecho() {
+ echo $* >&2
+}
+
+# echo to stderr and die
+die() {
+ echo -n "$PROGRAM: " >&2
+ eecho $*
+ exit 1
+}
+
+die_unless_force() {
+ echo "$PROGRAM: $*" >&2
+ [ -z "$FORCE" ] && exit 1
+}
+
+# remove double / and ./ in pathnames
+beautify_path() {
+ echo "$1" | sed 's:/^[^\.]\./::g; s:/\{2,\}:/:g; s:/\./:/:g'
+}
+
+# check if parameter is an uri or not
+is_uri() {
+ echo "$1" | grep "^[a-z][a-z0-9+]*:/" >/dev/null
+}
+
+# check if parameter is an apk package (contains a / or .apk at the end)
+is_apk() {
+ #echo "$1" | grep '/' >/dev/null && return 0
+ [ -z "${1%%*/*}" ] && return 0
+
+ #echo "$1" | grep ".apk$" >/dev/null
+ [ -z "${1%%*.apk}" ]
+}
+
+# check if path start with a '/'
+is_absolute_path() {
+ test -z "${1##/*}"
+}
+
+# if path dont start with '/' then append $PWD
+get_absolute_path() {
+ if is_absolute_path "$1" ; then
+ echo "$1"
+ else
+ beautify_path "$PWD/$1"
+ fi
+}
+
+# check if parameter has version number (i.e. if it is an pkgv or pkg)
+has_version() {
+ echo "$1" | grep -- '-[0-9].*' >/dev/null
+}
+
+# check if parameter has some kind of wildcard
+has_wildcard() {
+ echo "$1" | grep "[\*\?\[]" >/dev/null
+}
+
+# get the scheme for an uri (echo everything before the first ':')
+get_uri_scheme() {
+ echo "$1" | cut -d : -f 1
+}
+
+# remove version number from package name
+rm_ver() {
+ echo "$1" | sed 's/\(.*\)-[0-9].*/\1/'
+}
+
+# get version number from package name or file
+get_ver() {
+ basename "$1" .apk | sed 's/.*-\([0-9].*\)/\1/'
+}
+
+# initialize a temp directory
+# $1 contains the variable name for the directory
+# the directory will automatically be deleted upon exit
+init_tmpdir() {
+ local omask=`umask`
+ local __tmpd="$APK_TMPDIR/$PROGRAM-${$}-`date +%s`"
+ umask 077 || die "umask"
+ mkdir "$__tmpd" || exit 1
+ trap "rm -fr \"$__tmpd\"; exit" 0
+ umask $omask
+ eval "$1=\"$__tmpd\""
+}
+
+# remove files and empty dirs in specified list.
+# also remove APK_LBUFILES from default.tdb
+# environment:
+# ROOT: all files are relative this path
+# VERBOSE: echo filenames to stdout
+# DRYRUN: don't delete anything, just simulate
+my_rm() {
+ rm "$1" 2>/dev/null || busybox rm "$1"
+}
+
+list_uninstall() {
+ local f p
+ local root=${ROOT:-"/"}
+ sort -r "$1" | while read f ; do
+ p="`beautify_path \"$root/$f\"`"
+ if [ "$DRYRUN" ] ; then
+ [ "$VERBOSE" ] && echo "$p"
+ else
+ if [ -d "$p" ] ; then
+ # try to remove dir, but ignore errors. It might
+ if rmdir "$p" 2>/dev/null ; then
+ [ "$VERBOSE" ] && echo "$p"
+ [ "$2" ] && echo "$f" >> "$2"
+ fi
+ else
+ my_rm "$p" && [ "$VERBOSE" ] && echo "$p"
+ [ "$2" ] && echo "$f" >> "$2"
+ fi
+ fi
+ done
+ return 0
+}
+
+# list all lines that occur in first list but not second
+# the files cannot contain duplicate lines.
+list_subtract() {
+ (
+ # first we find all uniq lines
+ cat "$1" "$2" | sort | uniq -u
+
+ # then we combine uniq lines with first file ...
+ cat "$1"
+
+ # ...and find all duplicates. Those only exist in first file
+ ) | sort | uniq -d
+}
+
+# insert an element first in APK_PATH if its not already there
+insert_apk_path() {
+ if [ "$1" != "`echo "$APK_PATH" | cut -d\; -f1`" ] ; then
+ [ "$APK_PATH" ] && APK_PATH=";$APK_PATH"
+ APK_PATH="$1$APK_PATH"
+ fi
+}
+
+lbu_filter() {
+ # Ok... I give up. shell is too slow. lets do it in awk.
+ awk 'BEGIN {
+ APK_LBUDIRS="'"$APK_LBUDIRS"'";
+ numdirs = split(APK_LBUDIRS, lbudir, ":");
+ #precalc lengths to save a few cpu cycles in loop
+ for (i = 1; i <= numdirs; i++)
+ len[i] = length(lbudir[i]);
+ }
+
+ # main loop
+ {
+ for (i = 1; i <= numdirs; i++) {
+ if (index($0, lbudir[i]) == 1 && (len[i] == length() || substr($0, len[i] + 1, 1) == "/")) {
+ print $0;
+ }
+ }
+ }'
+}
+
+is_lbu_file() {
+ # just run test
+ [ "$(echo "$1" | lbu_filter)" ]
+}
+
+# assign a value to a global var, either from environment or
+# from configuraion file
+# usage: get_var VARIBALE_NAME DEFAULT_VALUE
+get_var() {
+ local var
+ # first we check if the envvar is set
+ eval "var=\$$1"
+ if [ "$var" ] ; then
+ echo "$var"
+ elif [ -f ${APKTOOLS_CONF:="$ROOT/etc/apk.conf"} ] ; then
+ # then we check the conf file
+ var=`awk -F = '/^'$1'=/ { print $2 }' "$APKTOOLS_CONF"`
+ if [ "$var" ] ; then
+ echo "$var"
+ else
+ # else we use the default
+ echo "$2"
+ fi
+ else
+ # no conf file found use default
+ echo "$2"
+ fi
+}
+
+##########################################################
+# last_pkgf
+# find the latest package in a list, return 1 if not found
+last_pkgf() {
+ local pkgf last status
+ while read pkgf ; do
+ apk_version -q -t "$pkgf" "$last"
+ [ $? -eq 2 ] && last="$pkgf"
+ done
+ [ -z "$last" ] && return 1
+ echo "$last"
+}
+
+###########################################################
+# dump global variables
+dump_env() {
+ echo "ROOT=$ROOT"
+ echo "APKTOOLS_CONF=$APKTOOLS_CONF"
+ echo "APK_PATH=$APK_PATH"
+ echo "APK_DBDIR=$APK_DBDIR"
+ echo "APK_TMPDIR=$APK_TMPDIR"
+ echo "APK_FETCH=$APK_FETCH"
+ echo "APK_DATA=$APK_DATA"
+ echo "APK_DATALEVEL=$APK_DATALEVEL"
+ echo "APK_LIBS=$APK_LIBS"
+ echo "PACKAGES=$PACKAGES"
+
+ echo "APKDB=$APKDB"
+ echo "APK_NOCOMPRESS=$APK_NOCOMPRESS"
+ echo "REP_DIR=$REP_DIR"
+ echo "REP_SCHEME=$REP_SCHEME"
+ echo "CACHED_INDEX=$CACHED_INDEX"
+}
+
+#############################################################################
+# init_globals sets up the global variables
+
+APK_PREFIX_IN_PKG="`get_var APK_PREFIX_IN_PKG ''`"
+
+ROOT="`get_var ROOT /`"
+echo "$ROOT" | grep -v "^/" > /dev/null && ROOT="$PWD/$ROOT"
+
+APKTOOLS_CONF="`get_var APKTOOLS_CONF \"$(beautify_path /etc/apk/apk.conf)\"`"
+APK_PATH=`get_var APK_PATH ""`
+APK_DBDIR="`get_var APK_DBDIR \"$(beautify_path \"$ROOT/var/db/apk\")\"`"
+APK_DBDIR_IN_PKG="`get_var APK_DBDIR_IN_PKG ${APK_PREFIX_IN_PKG}var/db/apk`"
+APK_TMPDIR="`get_var \"APK_TMPDIR\" /tmp`"
+APK_ADD_TMP="`get_var \"APK_ADD_TMP\" \"$ROOT/usr/tmp\"`"
+APK_DATA="`get_var APK_DATA \"$(beautify_path \"$ROOT/var/lib/apk\")\"`"
+APK_KEEPCACHE="`get_var APK_KEEPCACHE no`"
+APK_LIBS="`get_var APK_LIBS /lib/apk`"
+PACKAGES="`get_var PACKAGES \"$(beautify_path \"$ROOT/var/cache/packages\")\"`"
+
+APKDB="`beautify_path \"$APK_DBDIR\"`"
+APK_NOCOMPRESS=`get_var APK_NOCOMPRESS ""`
+
+INDEX="INDEX.md5.gz"
+CACHED_INDEX="$APK_DATA/$INDEX"
+
+APK_SUM=`get_var APK_SUM md5`
+APK_MKSUM=`get_var APK_MKSUM "${APK_SUM}sum"`
+APK_CHKSUM=`get_var APK_CHKSUM "${APK_SUM}sum -c"`
+
+APK_DEFAULT_TDB=`get_var APK_DEFAULT_TDB "$APK_DATA/default.tdb"`
+SFIC=`which sfic 2>/dev/null`
+APK_GZSIGN_CERT=`get_var APK_GZSIGN_KEY /etc/apk/apk.crt`
+
+# confdirs are a : spearate list of dirs relative $ROOT that are to be
+# considered for local backups.
+# for example: APK_LBUDIRS="etc:usr/local/etc"
+APK_LBUDIRS=`get_var APK_LBUDIRS 'etc'`
+