mirror of
git://projects.qi-hardware.com/openwrt-xburst.git
synced 2025-01-07 18:00:15 +02:00
e3f2df6671
strip directory entries from package file list when offline installing .ipk archives, this prevents opkg from removing directories like /usr later on when trying to uninstall preinstalled packages (partial fix for #6772) git-svn-id: svn://svn.openwrt.org/openwrt/trunk@19936 3c298f89-4303-0410-b956-a3cf2f4a3e73
1184 lines
28 KiB
Bash
Executable File
1184 lines
28 KiB
Bash
Executable File
#!/bin/sh
|
|
# ipkg - the itsy package management system
|
|
#
|
|
# Copyright (C) 2001 Carl D. Worth
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; either version 2, or (at your option)
|
|
# any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
|
|
set -e
|
|
|
|
# By default do not do globbing. Any command wanting globbing should
|
|
# explicitly enable it first and disable it afterwards.
|
|
set -o noglob
|
|
|
|
ipkg_is_upgrade () {
|
|
local A B a b
|
|
A=$(echo $1 | sed -r "s/([0-9]+)[^[:alnum:]]*/ \1 /g").
|
|
B=$(echo $2 | sed -r "s/([0-9]+)[^[:alnum:]]*/ \1 /g").
|
|
while [ \! -z "$A" ] && [ \! -z "$B" ]; do {
|
|
set $A; a=$1; shift; A=$*
|
|
set $B; b=$1; shift; B=$*
|
|
[ "$a" -lt "$b" ] 2>&- && return 1
|
|
{ [ "$a" -gt "$b" ] 2>&- || [ "$a" ">" "$b" ]; } && return
|
|
}; done
|
|
return 1
|
|
}
|
|
|
|
which md5sum 2>&1 >/dev/null || alias md5sum=md5
|
|
|
|
ipkg_srcs() {
|
|
local srcre="$1"
|
|
sed -ne "s/^src[[:space:]]\+$srcre[[:space:]]\+//p" < $IPKG_CONF
|
|
}
|
|
|
|
ipkg_src_names() {
|
|
sed -ne "s/^src[[:space:]]\+\([^[:space:]]\+\).*/\1/p" < $IPKG_CONF
|
|
}
|
|
|
|
ipkg_src_byname() {
|
|
local src="$1"
|
|
ipkg_srcs $src | head -n 1
|
|
}
|
|
|
|
ipkg_dests() {
|
|
local destre="`echo $1 | ipkg_protect_slashes`"
|
|
sed -ne "/^dest[[:space:]]\+$destre/{
|
|
s/^dest[[:space:]]\+[^[:space:]]\+[[:space:]]\+//
|
|
s/^/`echo $IPKG_OFFLINE_ROOT | ipkg_protect_slashes`/
|
|
p
|
|
}" < $IPKG_CONF
|
|
}
|
|
|
|
ipkg_dest_names() {
|
|
sed -ne "s/^dest[[:space:]]\+\([^[:space:]]\+\).*/\1/p" < $IPKG_CONF
|
|
}
|
|
|
|
ipkg_dests_all() {
|
|
ipkg_dests '.*'
|
|
}
|
|
|
|
ipkg_state_dirs() {
|
|
ipkg_dests_all | sed "s|\$|/$IPKG_DIR_PREFIX|"
|
|
}
|
|
|
|
ipkg_dest_default() {
|
|
ipkg_dests_all | head -n 1
|
|
}
|
|
|
|
ipkg_dest_default_name() {
|
|
ipkg_dest_names | head -n 1
|
|
}
|
|
|
|
ipkg_dest_byname() {
|
|
local dest="$1"
|
|
ipkg_dests $dest | head -n 1
|
|
}
|
|
|
|
ipkg_option() {
|
|
local option="$1"
|
|
sed -ne "s/^option[[:space:]]\+$option[[:space:]]\+//p" < $IPKG_CONF
|
|
}
|
|
|
|
ipkg_load_configuration() {
|
|
if [ -z "$IPKG_CONF_DIR" ]; then
|
|
IPKG_CONF_DIR=/etc
|
|
fi
|
|
|
|
IPKG_CONF="$IPKG_CONF_DIR/ipkg.conf"
|
|
|
|
if [ -z "$IPKG_OFFLINE_ROOT" ]; then
|
|
IPKG_OFFLINE_ROOT="`ipkg_option offline_root`"
|
|
fi
|
|
# Export IPKG_OFFLINE_ROOT for use by update-alternatives
|
|
export IPKG_OFFLINE_ROOT
|
|
if [ -n "$DEST_NAME" ]; then
|
|
IPKG_ROOT="`ipkg_dest_byname $DEST_NAME`"
|
|
if [ -z "$IPKG_ROOT" ]; then
|
|
if [ -d "$IPKG_OFFLINE_ROOT$DEST_NAME" ]; then
|
|
IPKG_ROOT="$IPKG_OFFLINE_ROOT$DEST_NAME";
|
|
else
|
|
echo "ipkg: invalid destination specification: $DEST_NAME
|
|
Valid destinations are directories or one of the dest names from $IPKG_CONF:" >&2
|
|
ipkg_dest_names >&2
|
|
return 1
|
|
fi
|
|
fi
|
|
else
|
|
IPKG_ROOT="`ipkg_dest_default`"
|
|
fi
|
|
|
|
# Global ipkg state directories
|
|
IPKG_DIR_PREFIX=usr/lib/opkg
|
|
IPKG_LISTS_DIR=$IPKG_OFFLINE_ROOT/$IPKG_DIR_PREFIX/lists
|
|
IPKG_PENDING_DIR=$IPKG_OFFLINE_ROOT/$IPKG_DIR_PREFIX/pending
|
|
if [ -z "$IPKG_TMP" ]; then
|
|
IPKG_TMP=$IPKG_ROOT/tmp/ipkg
|
|
fi
|
|
|
|
[ -e "$IPKG_TMP" ] || mkdir -p $IPKG_TMP
|
|
|
|
# Destination specific ipkg meta-data directory
|
|
IPKG_STATE_DIR=$IPKG_ROOT/$IPKG_DIR_PREFIX
|
|
|
|
# Proxy Support
|
|
IPKG_PROXY_USERNAME="`ipkg_option proxy_username`"
|
|
IPKG_PROXY_PASSWORD="`ipkg_option proxy_password`"
|
|
IPKG_HTTP_PROXY="`ipkg_option http_proxy`"
|
|
IPKG_FTP_PROXY="`ipkg_option ftp_proxy`"
|
|
IPKG_NO_PROXY="`ipkg_option no_proxy`"
|
|
if [ -n "$IPKG_HTTP_PROXY" ]; then
|
|
export http_proxy="$IPKG_HTTP_PROXY"
|
|
fi
|
|
|
|
if [ -n "$IPKG_FTP_PROXY" ]; then
|
|
export ftp_proxy="$IPKG_FTP_PROXY"
|
|
fi
|
|
|
|
if [ -n "$IPKG_NO_PROXY" ]; then
|
|
export no_proxy="$IPKG_NO_PROXY"
|
|
fi
|
|
|
|
IPKG_STATUS_FIELDS='\(Package\|Status\|Essential\|Version\|Conffiles\|Root\|Architecture\)'
|
|
}
|
|
|
|
ipkg_usage() {
|
|
[ $# -gt 0 ] && echo "ipkg: $*"
|
|
echo "
|
|
usage: ipkg [options...] sub-command [arguments...]
|
|
where sub-command is one of:
|
|
|
|
Package Manipulation:
|
|
update Update list of available packages
|
|
upgrade Upgrade all installed packages to latest version
|
|
install <pkg> Download and install <pkg> (and dependencies)
|
|
install <file.ipk> Install package <file.ipk>
|
|
install <file.deb> Install package <file.deb>
|
|
remove <pkg> Remove package <pkg>
|
|
|
|
Informational Commands:
|
|
list List available packages and descriptions
|
|
files <pkg> List all files belonging to <pkg>
|
|
search <file> Search for a packaging providing <file>
|
|
info [pkg [<field>]] Display all/some info fields for <pkg> or all
|
|
status [pkg [<field>]] Display all/some status fields for <pkg> or all
|
|
depends <pkg> Print uninstalled package dependencies for <pkg>
|
|
|
|
Options:
|
|
-d <dest_name> Use <dest_name> as the the root directory for
|
|
-dest <dest_name> package installation, removal, upgrading.
|
|
<dest_name> should be a defined dest name from the
|
|
configuration file, (but can also be a directory
|
|
name in a pinch).
|
|
-o <offline_root> Use <offline_root> as the root for offline installation.
|
|
-offline <offline_root>
|
|
|
|
Force Options (use when ipkg is too smart for its own good):
|
|
-force-depends Make dependency checks warnings instead of errors
|
|
-force-defaults Use default options for questions asked by ipkg.
|
|
(no prompts). Note that this will not prevent
|
|
package installation scripts from prompting.
|
|
" >&2
|
|
exit 1
|
|
}
|
|
|
|
ipkg_dir_part() {
|
|
local dir="`echo $1 | sed -ne 's/\(.*\/\).*/\1/p'`"
|
|
if [ -z "$dir" ]; then
|
|
dir="./"
|
|
fi
|
|
echo $dir
|
|
}
|
|
|
|
ipkg_file_part() {
|
|
echo $1 | sed 's/.*\///'
|
|
}
|
|
|
|
ipkg_protect_slashes() {
|
|
sed -e 's/\//\\\//g'
|
|
}
|
|
|
|
ipkg_download() {
|
|
local src="$1"
|
|
local dest="$2"
|
|
|
|
local src_file="`ipkg_file_part $src`"
|
|
local dest_dir="`ipkg_dir_part $dest`"
|
|
if [ -z "$dest_dir" ]; then
|
|
dest_dir="$IPKG_TMP"
|
|
fi
|
|
|
|
local dest_file="`ipkg_file_part $dest`"
|
|
if [ -z "$dest_file" ]; then
|
|
dest_file="$src_file"
|
|
fi
|
|
|
|
# Proxy support
|
|
local proxyuser=""
|
|
local proxypassword=""
|
|
local proxyoption=""
|
|
|
|
if [ -n "$IPKG_PROXY_USERNAME" ]; then
|
|
proxyuser="--proxy-user=\"$IPKG_PROXY_USERNAME\""
|
|
proxypassword="--proxy-passwd=\"$IPKG_PROXY_PASSWORD\""
|
|
fi
|
|
|
|
if [ -n "$IPKG_PROXY_HTTP" -o -n "$IPKG_PROXY_FTP" ]; then
|
|
proxyoption="--proxy=on"
|
|
fi
|
|
|
|
rm -f $IPKG_TMP/$src_file
|
|
case "$src" in
|
|
http://* | ftp://*)
|
|
if ! wget --passive-ftp $proxyoption $proxyuser $proxypassword -P $IPKG_TMP $src; then
|
|
echo "ipkg_download: ERROR: Failed to retrieve $src, returning $err"
|
|
return 1
|
|
fi
|
|
mv $IPKG_TMP/$src_file $dest_dir/$dest_file 2>/dev/null
|
|
;;
|
|
file:/* )
|
|
ln -s `echo $src | sed 's/^file://'` $dest_dir/$dest_file 2>/dev/null
|
|
;;
|
|
*)
|
|
echo "DEBUG: $src"
|
|
;;
|
|
esac
|
|
|
|
return 0
|
|
}
|
|
|
|
ipkg_update() {
|
|
if [ ! -e "$IPKG_LISTS_DIR" ]; then
|
|
mkdir -p $IPKG_LISTS_DIR
|
|
fi
|
|
|
|
local err=
|
|
for src_name in `ipkg_src_names`; do
|
|
local src="`ipkg_src_byname $src_name`"
|
|
if ! ipkg_download $src/Packages $IPKG_LISTS_DIR/$src_name; then
|
|
echo "ipkg_update: Error downloading $src/Packages to $IPKG_LISTS_DIR/$src_name" >&2
|
|
err=t
|
|
else
|
|
echo "Updated list of available packages in $IPKG_LISTS_DIR/$src_name"
|
|
fi
|
|
done
|
|
|
|
[ -n "$err" ] && return 1
|
|
|
|
return 0
|
|
}
|
|
|
|
ipkg_list() {
|
|
for src in `ipkg_src_names`; do
|
|
if ipkg_require_list $src; then
|
|
# black magic...
|
|
sed -ne "
|
|
/^Package:/{
|
|
s/^Package:[[:space:]]*\<\([a-z0-9.+-]*$1[a-z0-9.+-]*\).*/\1/
|
|
h
|
|
}
|
|
/^Description:/{
|
|
s/^Description:[[:space:]]*\(.*\)/\1/
|
|
H
|
|
g
|
|
s/\\
|
|
/ - /
|
|
p
|
|
}
|
|
" $IPKG_LISTS_DIR/$src
|
|
fi
|
|
done
|
|
}
|
|
|
|
ipkg_extract_paragraph() {
|
|
local pkg="$1"
|
|
sed -ne "/Package:[[:space:]]*$pkg[[:space:]]*\$/,/^\$/p"
|
|
}
|
|
|
|
ipkg_extract_field() {
|
|
local field="$1"
|
|
# blacker magic...
|
|
sed -ne "
|
|
: TOP
|
|
/^$field:/{
|
|
p
|
|
n
|
|
b FIELD
|
|
}
|
|
d
|
|
: FIELD
|
|
/^$/b TOP
|
|
/^[^[:space:]]/b TOP
|
|
p
|
|
n
|
|
b FIELD
|
|
"
|
|
}
|
|
|
|
ipkg_extract_value() {
|
|
sed -e "s/^[^:]*:[[:space:]]*//"
|
|
}
|
|
|
|
ipkg_require_list() {
|
|
[ $# -lt 1 ] && return 1
|
|
local src="$1"
|
|
if [ ! -f "$IPKG_LISTS_DIR/$src" ]; then
|
|
echo "ERROR: File not found: $IPKG_LISTS_DIR/$src" >&2
|
|
echo " You probably want to run \`ipkg update'" >&2
|
|
return 1
|
|
fi
|
|
return 0
|
|
}
|
|
|
|
ipkg_info() {
|
|
for src in `ipkg_src_names`; do
|
|
if ipkg_require_list $src; then
|
|
case $# in
|
|
0)
|
|
cat $IPKG_LISTS_DIR/$src
|
|
;;
|
|
1)
|
|
ipkg_extract_paragraph $1 < $IPKG_LISTS_DIR/$src
|
|
;;
|
|
*)
|
|
ipkg_extract_paragraph $1 < $IPKG_LISTS_DIR/$src | ipkg_extract_field $2
|
|
;;
|
|
esac
|
|
fi
|
|
done
|
|
}
|
|
|
|
ipkg_status_sd() {
|
|
[ $# -lt 1 ] && return 0
|
|
sd="$1"
|
|
shift
|
|
if [ -f $sd/status ]; then
|
|
case $# in
|
|
0)
|
|
cat $sd/status
|
|
;;
|
|
1)
|
|
ipkg_extract_paragraph $1 < $sd/status
|
|
;;
|
|
*)
|
|
ipkg_extract_paragraph $1 < $sd/status | ipkg_extract_field $2
|
|
;;
|
|
esac
|
|
fi
|
|
return 0
|
|
}
|
|
|
|
ipkg_status_all() {
|
|
for sd in `ipkg_state_dirs`; do
|
|
ipkg_status_sd $sd $*
|
|
done
|
|
}
|
|
|
|
ipkg_status() {
|
|
if [ -n "$DEST_NAME" ]; then
|
|
ipkg_status_sd $IPKG_STATE_DIR $*
|
|
else
|
|
ipkg_status_all $*
|
|
fi
|
|
}
|
|
|
|
ipkg_status_matching_sd() {
|
|
local sd="$1"
|
|
local re="$2"
|
|
if [ -f $sd/status ]; then
|
|
sed -ne "
|
|
: TOP
|
|
/^Package:/{
|
|
s/^Package:[[:space:]]*//
|
|
s/[[:space:]]*$//
|
|
h
|
|
}
|
|
/$re/{
|
|
g
|
|
p
|
|
b NEXT
|
|
}
|
|
d
|
|
: NEXT
|
|
/^$/b TOP
|
|
n
|
|
b NEXT
|
|
" < $sd/status
|
|
fi
|
|
return 0
|
|
}
|
|
|
|
ipkg_status_matching_all() {
|
|
for sd in `ipkg_state_dirs`; do
|
|
ipkg_status_matching_sd $sd $*
|
|
done
|
|
}
|
|
|
|
ipkg_status_matching() {
|
|
if [ -n "$DEST_NAME" ]; then
|
|
ipkg_status_matching_sd $IPKG_STATE_DIR $*
|
|
else
|
|
ipkg_status_matching_all $*
|
|
fi
|
|
}
|
|
|
|
ipkg_status_installed_sd() {
|
|
local sd="$1"
|
|
local pkg="$2"
|
|
ipkg_status_sd $sd $pkg Status | grep -q "Status: install ok installed"
|
|
}
|
|
|
|
ipkg_status_installed_all() {
|
|
local ret=1
|
|
for sd in `ipkg_state_dirs`; do
|
|
if `ipkg_status_installed_sd $sd $*`; then
|
|
ret=0
|
|
fi
|
|
done
|
|
return $ret
|
|
}
|
|
|
|
ipkg_status_mentioned_sd() {
|
|
local sd="$1"
|
|
local pkg="$2"
|
|
[ -n "`ipkg_status_sd $sd $pkg Status`" ]
|
|
}
|
|
|
|
ipkg_files() {
|
|
local pkg="$1"
|
|
if [ -n "$DEST_NAME" ]; then
|
|
dests=$IPKG_ROOT
|
|
else
|
|
dests="`ipkg_dests_all`"
|
|
fi
|
|
for dest in $dests; do
|
|
if [ -f $dest/$IPKG_DIR_PREFIX/info/$pkg.list ]; then
|
|
dest_sed="`echo $dest | ipkg_protect_slashes`"
|
|
sed -e "s/^/$dest_sed/" < $dest/$IPKG_DIR_PREFIX/info/$pkg.list
|
|
fi
|
|
done
|
|
}
|
|
|
|
ipkg_search() {
|
|
local pattern="$1"
|
|
|
|
for dest_name in `ipkg_dest_names`; do
|
|
dest="`ipkg_dest_byname $dest_name`"
|
|
dest_sed="`echo $dest | ipkg_protect_slashes`"
|
|
|
|
set +o noglob
|
|
local list_files="`ls -1 $dest/$IPKG_DIR_PREFIX/info/*.list 2>/dev/null`"
|
|
set -o noglob
|
|
for file in $list_files; do
|
|
if sed "s/^/$dest_sed/" $file | grep -q $pattern; then
|
|
local pkg="`echo $file | sed "s/^.*\/\(.*\)\.list/\1/"`"
|
|
[ "$dest_name" != `ipkg_dest_default_name` ] && pkg="$pkg ($dest_name)"
|
|
sed "s/^/$dest_sed/" $file | grep $pattern | sed "s/^/$pkg: /"
|
|
fi
|
|
done
|
|
done
|
|
}
|
|
|
|
ipkg_status_remove_sd() {
|
|
local sd="$1"
|
|
local pkg="$2"
|
|
|
|
if [ ! -f $sd/status ]; then
|
|
mkdir -p $sd
|
|
touch $sd/status
|
|
fi
|
|
sed -ne "/Package:[[:space:]]*$pkg[[:space:]]*\$/,/^\$/!p" < $sd/status > $sd/status.new
|
|
mv $sd/status.new $sd/status
|
|
}
|
|
|
|
ipkg_status_remove_all() {
|
|
for sd in `ipkg_state_dirs`; do
|
|
ipkg_status_remove_sd $sd $*
|
|
done
|
|
}
|
|
|
|
ipkg_status_remove() {
|
|
if [ -n "$DEST_NAME" ]; then
|
|
ipkg_status_remove_sd $IPKG_STATE_DIR $*
|
|
else
|
|
ipkg_status_remove_all $*
|
|
fi
|
|
}
|
|
|
|
ipkg_status_update_sd() {
|
|
local sd="$1"
|
|
local pkg="$2"
|
|
|
|
ipkg_status_remove_sd $sd $pkg
|
|
ipkg_extract_field "$IPKG_STATUS_FIELDS" >> $sd/status
|
|
echo "" >> $sd/status
|
|
}
|
|
|
|
ipkg_status_update() {
|
|
ipkg_status_update_sd $IPKG_STATE_DIR $*
|
|
}
|
|
|
|
ipkg_unsatisfied_dependences() {
|
|
local pkg=$1
|
|
local deps="`ipkg_get_depends $pkg`"
|
|
local remaining_deps=
|
|
for dep in $deps; do
|
|
local installed="`ipkg_get_installed $dep`"
|
|
if [ "$installed" != "installed" ] ; then
|
|
remaining_deps="$remaining_deps $dep"
|
|
fi
|
|
done
|
|
## echo "ipkg_unsatisfied_dependences pkg=$pkg $remaining_deps" > /dev/console
|
|
echo $remaining_deps
|
|
}
|
|
|
|
ipkg_safe_pkg_name() {
|
|
local pkg=$1
|
|
local spkg="`echo pkg_$pkg | sed -e y/-+./___/`"
|
|
echo $spkg
|
|
}
|
|
|
|
ipkg_set_depends() {
|
|
local pkg=$1; shift
|
|
local new_deps="$*"
|
|
pkg="`ipkg_safe_pkg_name $pkg`"
|
|
## setvar ${pkg}_depends "$new_deps"
|
|
echo $new_deps > $IPKG_TMP/${pkg}.depends
|
|
}
|
|
|
|
ipkg_get_depends() {
|
|
local pkg=$1
|
|
pkg="`ipkg_safe_pkg_name $pkg`"
|
|
cat $IPKG_TMP/${pkg}.depends
|
|
## eval "echo \$${pkg}_depends"
|
|
}
|
|
|
|
ipkg_set_installed() {
|
|
local pkg=$1
|
|
pkg="`ipkg_safe_pkg_name $pkg`"
|
|
echo installed > $IPKG_TMP/${pkg}.installed
|
|
## setvar ${pkg}_installed "installed"
|
|
}
|
|
|
|
ipkg_set_uninstalled() {
|
|
local pkg=$1
|
|
pkg="`ipkg_safe_pkg_name $pkg`"
|
|
### echo ipkg_set_uninstalled $pkg > /dev/console
|
|
echo uninstalled > $IPKG_TMP/${pkg}.installed
|
|
## setvar ${pkg}_installed "uninstalled"
|
|
}
|
|
|
|
ipkg_get_installed() {
|
|
local pkg=$1
|
|
pkg="`ipkg_safe_pkg_name $pkg`"
|
|
if [ -f $IPKG_TMP/${pkg}.installed ]; then
|
|
cat $IPKG_TMP/${pkg}.installed
|
|
fi
|
|
## eval "echo \$${pkg}_installed"
|
|
}
|
|
|
|
ipkg_depends() {
|
|
local new_pkgs="$*"
|
|
local all_deps=
|
|
local installed_pkgs="`ipkg_status_matching_all 'Status:.*[[:space:]]installed'`"
|
|
for pkg in $installed_pkgs; do
|
|
ipkg_set_installed $pkg
|
|
done
|
|
while [ -n "$new_pkgs" ]; do
|
|
all_deps="$all_deps $new_pkgs"
|
|
local new_deps=
|
|
for pkg in $new_pkgs; do
|
|
if echo $pkg | grep -q '[^A-Za-z0-9.+-]'; then
|
|
echo "ipkg_depends: ERROR: Package name $pkg contains illegal characters (should be [A-Za-z0-9.+-])" >&2
|
|
return 1
|
|
fi
|
|
# TODO: Fix this. For now I am ignoring versions and alternations in dependencies.
|
|
new_deps="$new_deps "`ipkg_info $pkg '\(Pre-\)\?Depends' | ipkg_extract_value | sed -e 's/([^)]*)//g
|
|
s/\(|[[:space:]]*[a-z0-9.+-]\+[[:space:]]*\)\+//g
|
|
s/,/ /g
|
|
s/ \+/ /g'`
|
|
ipkg_set_depends $pkg $new_deps
|
|
done
|
|
|
|
new_deps=`echo $new_deps | sed -e 's/[[:space:]]\+/\n/g' | sort | uniq`
|
|
|
|
local maybe_new_pkgs=
|
|
for pkg in $new_deps; do
|
|
if ! echo $installed_pkgs | grep -q "\<$pkg\>"; then
|
|
maybe_new_pkgs="$maybe_new_pkgs $pkg"
|
|
fi
|
|
done
|
|
|
|
new_pkgs=
|
|
for pkg in $maybe_new_pkgs; do
|
|
if ! echo $all_deps | grep -q "\<$pkg\>"; then
|
|
if [ -z "`ipkg_info $pkg`" ]; then
|
|
echo "ipkg_depends: Warning: $pkg mentioned in dependency but no package found in $IPKG_LISTS_DIR" >&2
|
|
ipkg_set_installed $pkg
|
|
else
|
|
new_pkgs="$new_pkgs $pkg"
|
|
ipkg_set_uninstalled $pkg
|
|
fi
|
|
else
|
|
ipkg_set_uninstalled $pkg
|
|
fi
|
|
done
|
|
done
|
|
|
|
echo $all_deps
|
|
}
|
|
|
|
ipkg_get_install_dest() {
|
|
local dest="$1"
|
|
shift
|
|
local sd=$dest/$IPKG_DIR_PREFIX
|
|
local info_dir=$sd/info
|
|
|
|
local requested_pkgs="$*"
|
|
local pkgs="`ipkg_depends $*`"
|
|
|
|
mkdir -p $info_dir
|
|
for pkg in $pkgs; do
|
|
if ! ipkg_status_mentioned_sd $sd $pkg; then
|
|
echo "Package: $pkg
|
|
Status: install ok not-installed" | ipkg_status_update_sd $sd $pkg
|
|
fi
|
|
done
|
|
## mark the packages that we were directly requested to install as uninstalled
|
|
for pkg in $requested_pkgs; do ipkg_set_uninstalled $pkg; done
|
|
|
|
local new_pkgs=
|
|
local pkgs_installed=0
|
|
while [ -n "pkgs" ]; do
|
|
curcheck=0
|
|
## echo "pkgs to install: {$pkgs}" > /dev/console
|
|
for pkg in $pkgs; do
|
|
curcheck="`expr $curcheck + 1`"
|
|
local is_installed="`ipkg_get_installed $pkg`"
|
|
if [ "$is_installed" = "installed" ]; then
|
|
echo "$pkg is installed"
|
|
continue
|
|
fi
|
|
|
|
local remaining_deps="`ipkg_unsatisfied_dependences $pkg`"
|
|
if [ -n "$remaining_deps" ]; then
|
|
new_pkgs="$new_pkgs $pkg"
|
|
### echo "Dependences not satisfied for $pkg: $remaining_deps"
|
|
if [ $curcheck -ne `echo $pkgs|wc -w` ]; then
|
|
continue
|
|
fi
|
|
fi
|
|
|
|
local filename=
|
|
for src in `ipkg_src_names`; do
|
|
if ipkg_require_list $src; then
|
|
filename="`ipkg_extract_paragraph $pkg < $IPKG_LISTS_DIR/$src | ipkg_extract_field Filename | ipkg_extract_value`"
|
|
[ -n "$filename" ] && break
|
|
fi
|
|
done
|
|
|
|
if [ -z "$filename" ]; then
|
|
echo "ipkg_get_install: ERROR: Cannot find package $pkg in $IPKG_LISTS_DIR"
|
|
echo "ipkg_get_install: Check the spelling and maybe run \`ipkg update'."
|
|
ipkg_status_remove_sd $sd $pkg
|
|
return 1;
|
|
fi
|
|
|
|
local tmp_pkg_file="$IPKG_TMP/"`ipkg_file_part $filename`
|
|
if ! ipkg_download `ipkg_src_byname $src`/$filename $tmp_pkg_file; then
|
|
echo "ipkg_get_install: Perhaps you need to run \`ipkg update'?"
|
|
return 1
|
|
fi
|
|
|
|
if ! ipkg_install_file_dest $dest $tmp_pkg_file; then
|
|
echo "ipkg_get_install: ERROR: Failed to install $tmp_pkg_file"
|
|
echo "ipkg_get_install: I'll leave it there for you to try a manual installation"
|
|
return 1
|
|
fi
|
|
|
|
ipkg_set_installed $pkg
|
|
pkgs_installed="`expr $pkgs_installed + 1`"
|
|
rm $tmp_pkg_file
|
|
done
|
|
### echo "Installed $pkgs_installed package(s) this round"
|
|
if [ $pkgs_installed -eq 0 ]; then
|
|
if [ -z "$new_pkgs" ]; then
|
|
break
|
|
fi
|
|
fi
|
|
pkgs_installed=0
|
|
pkgs="$new_pkgs"
|
|
new_pkgs=
|
|
curcheck=0
|
|
done
|
|
}
|
|
|
|
ipkg_get_install() {
|
|
ipkg_get_install_dest $IPKG_ROOT $*
|
|
}
|
|
|
|
ipkg_install_file_dest() {
|
|
local dest="$1"
|
|
local filename="$2"
|
|
local sd=$dest/$IPKG_DIR_PREFIX
|
|
local info_dir=$sd/info
|
|
|
|
if [ ! -f "$filename" ]; then
|
|
echo "ipkg_install_file: ERROR: File $filename not found"
|
|
return 1
|
|
fi
|
|
|
|
local pkg="`ipkg_file_part $filename | sed 's/\([a-z0-9.+-]\+\)_.*/\1/'`"
|
|
local ext="`echo $filename | sed 's/.*\.//'`"
|
|
local pkg_extract_stdout
|
|
if [ "$ext" = "ipk" ]; then
|
|
pkg_extract_stdout="tar -xzOf"
|
|
elif [ "$ext" = "deb" ]; then
|
|
pkg_extract_stdout="ar p"
|
|
else
|
|
echo "ipkg_install_file: ERROR: File $filename has unknown extension $ext (not .ipk or .deb)"
|
|
return 1
|
|
fi
|
|
|
|
# Check dependencies
|
|
local depends="`ipkg_depends $pkg | sed -e "s/\<$pkg\>//"`"
|
|
|
|
# Don't worry about deps that are scheduled for installation
|
|
local missing_deps=
|
|
for dep in $depends; do
|
|
if ! ipkg_status_all $dep | grep -q 'Status:[[:space:]]install'; then
|
|
missing_deps="$missing_deps $dep"
|
|
fi
|
|
done
|
|
|
|
if [ ! -z "$missing_deps" ]; then
|
|
if [ -n "$FORCE_DEPENDS" ]; then
|
|
echo "ipkg_install_file: Warning: $pkg depends on the following uninstalled programs: $missing_deps"
|
|
else
|
|
echo "ipkg_install_file: ERROR: $pkg depends on the following uninstalled programs:
|
|
$missing_deps"
|
|
echo "ipkg_install_file: You may want to use \`ipkg install' to install these."
|
|
return 1
|
|
fi
|
|
fi
|
|
|
|
mkdir -p $IPKG_TMP/$pkg/control
|
|
mkdir -p $IPKG_TMP/$pkg/data
|
|
mkdir -p $info_dir
|
|
|
|
if ! $pkg_extract_stdout $filename ./control.tar.gz | (cd $IPKG_TMP/$pkg/control; zcat | tar -xf - ) ; then
|
|
echo "ipkg_install_file: ERROR unpacking control.tar.gz from $filename"
|
|
return 1
|
|
fi
|
|
|
|
if [ -n "$IPKG_OFFLINE_ROOT" ]; then
|
|
if grep -q '^InstallsOffline:[[:space:]]*no' $IPKG_TMP/$pkg/control/control; then
|
|
echo "*** Warning: Package $pkg may not be installed in offline mode"
|
|
echo "*** Warning: Scheduling $filename for pending installation (installing into $IPKG_PENDING_DIR)"
|
|
echo "Package: $pkg
|
|
Status: install ok pending" | ipkg_status_update_sd $sd $pkg
|
|
mkdir -p $IPKG_PENDING_DIR
|
|
cp -f $filename $IPKG_PENDING_DIR
|
|
rm -r $IPKG_TMP/$pkg/control
|
|
rm -r $IPKG_TMP/$pkg/data
|
|
rmdir $IPKG_TMP/$pkg
|
|
return 0
|
|
fi
|
|
fi
|
|
|
|
|
|
printf "Unpacking $pkg..."
|
|
set +o noglob
|
|
for file in $IPKG_TMP/$pkg/control/*; do
|
|
local base_file="`ipkg_file_part $file`"
|
|
mv $file $info_dir/$pkg.$base_file
|
|
done
|
|
set -o noglob
|
|
rm -r $IPKG_TMP/$pkg/control
|
|
|
|
if ! $pkg_extract_stdout $filename ./data.tar.gz | (cd $IPKG_TMP/$pkg/data; zcat | tar -xf - ) ; then
|
|
echo "ipkg_install_file: ERROR unpacking data.tar.gz from $filename"
|
|
return 1
|
|
fi
|
|
echo "Done."
|
|
|
|
printf "Configuring $pkg..."
|
|
export PKG_ROOT=$dest
|
|
if [ -x "$info_dir/$pkg.preinst" ]; then
|
|
if ! $info_dir/$pkg.preinst install; then
|
|
echo "$info_dir/$pkg.preinst failed. Aborting installation of $pkg"
|
|
rm -rf $IPKG_TMP/$pkg/data
|
|
rmdir $IPKG_TMP/$pkg
|
|
return 1
|
|
fi
|
|
fi
|
|
|
|
local old_conffiles="`ipkg_status_sd $sd $pkg Conffiles | ipkg_extract_value`"
|
|
local new_conffiles=
|
|
if [ -f "$info_dir/$pkg.conffiles" ]; then
|
|
for conffile in `cat $info_dir/$pkg.conffiles`; do
|
|
if [ -f "$dest/$conffile" ] && ! echo " $old_conffiles " | grep -q " $conffile "`md5sum $dest/$conffile | sed 's/ .*//'`; then
|
|
local use_maintainers_conffile=
|
|
if [ -z "$FORCE_DEFAULTS" ]; then
|
|
while true; do
|
|
printf "Configuration file \`$conffile'
|
|
==> File on system created by you or by a script.
|
|
==> File also in package provided by package maintainer.
|
|
What would you like to do about it ? Your options are:
|
|
Y or I : install the package maintainer's version
|
|
N or O : keep your currently-installed version
|
|
D : show the differences between the versions (if diff is installed)
|
|
The default action is to keep your current version.
|
|
*** `ipkg_file_part $conffile` (Y/I/N/O/D) [default=N] ? "
|
|
read response
|
|
case "$response" in
|
|
[YyIi] | [Yy][Ee][Ss])
|
|
use_maintainers_conffile=t
|
|
break
|
|
;;
|
|
[Dd])
|
|
echo "
|
|
diff -u $dest/$conffile $IPKG_TMP/$pkg/data/$conffile"
|
|
diff -u $dest/$conffile $IPKG_TMP/$pkg/data/$conffile || true
|
|
echo "[Press ENTER to continue]"
|
|
read junk
|
|
;;
|
|
*)
|
|
break
|
|
;;
|
|
esac
|
|
done
|
|
fi
|
|
if [ -n "$use_maintainers_conffile" ]; then
|
|
local md5sum="`md5sum $IPKG_TMP/$pkg/data/$conffile | sed 's/ .*//'`"
|
|
new_conffiles="$new_conffiles $conffile $md5sum"
|
|
else
|
|
new_conffiles="$new_conffiles $conffile <custom>"
|
|
rm $IPKG_TMP/$pkg/data/$conffile
|
|
fi
|
|
else
|
|
md5sum="`md5sum $IPKG_TMP/$pkg/data/$conffile | sed 's/ .*//'`"
|
|
new_conffiles="$new_conffiles $conffile $md5sum"
|
|
fi
|
|
done
|
|
fi
|
|
|
|
local owd="`pwd`"
|
|
(cd $IPKG_TMP/$pkg/data/; tar cf - . | (cd $owd; cd $dest; tar xf -))
|
|
rm -rf $IPKG_TMP/$pkg/data
|
|
rmdir $IPKG_TMP/$pkg
|
|
rm -f $info_dir/$pkg.list
|
|
$pkg_extract_stdout $filename ./data.tar.gz | zcat | tar tf - | sed -e '/\/$/d; s/^\.//' > $info_dir/$pkg.list
|
|
|
|
if [ -x "$info_dir/$pkg.postinst" ]; then
|
|
$info_dir/$pkg.postinst configure
|
|
fi
|
|
|
|
if [ -n "$new_conffiles" ]; then
|
|
new_conffiles='Conffiles: '`echo $new_conffiles | ipkg_protect_slashes`
|
|
fi
|
|
local sed_safe_offline_root="`echo ${IPKG_OFFLINE_ROOT} | ipkg_protect_slashes`"
|
|
local sed_safe_root="`echo $dest | sed -e "s/^${sed_safe_offline_root}//" | ipkg_protect_slashes`"
|
|
sed -e "s/\(Package:.*\)/\1\\
|
|
Status: install ok installed\\
|
|
Root: ${sed_safe_root}\\
|
|
${new_conffiles}/" $info_dir/$pkg.control | ipkg_status_update_sd $sd $pkg
|
|
|
|
rm -f $info_dir/$pkg.control
|
|
rm -f $info_dir/$pkg.conffiles
|
|
rm -f $info_dir/$pkg.preinst
|
|
rm -f $info_dir/$pkg.postinst
|
|
|
|
echo "Done."
|
|
}
|
|
|
|
ipkg_install_file() {
|
|
ipkg_install_file_dest $IPKG_ROOT $*
|
|
}
|
|
|
|
ipkg_install() {
|
|
while [ $# -gt 0 ]; do
|
|
local pkg="$1"
|
|
shift
|
|
|
|
case "$pkg" in
|
|
http://* | ftp://*)
|
|
local tmp_pkg_file="$IPKG_TMP/"`ipkg_file_part $pkg`
|
|
if ipkg_download $pkg $tmp_pkg_file; then
|
|
ipkg_install_file $tmp_pkg_file
|
|
rm $tmp_pkg_file
|
|
fi
|
|
;;
|
|
file:/*.ipk | file://*.deb)
|
|
local ipkg_filename="`echo $pkg|sed 's/^file://'`"
|
|
ipkg_install_file $ipkg_filename
|
|
;;
|
|
*.ipk | *.deb)
|
|
ipkg_install_file $pkg
|
|
;;
|
|
*)
|
|
ipkg_get_install $pkg || true
|
|
;;
|
|
esac
|
|
done
|
|
}
|
|
|
|
ipkg_install_pending() {
|
|
[ -n "$IPKG_OFFLINE_ROOT" ] && return 0
|
|
|
|
if [ -d "$IPKG_PENDING_DIR" ]; then
|
|
set +o noglob
|
|
local pending="`ls -1d $IPKG_PENDING_DIR/*.ipk 2> /dev/null`" || true
|
|
set -o noglob
|
|
if [ -n "$pending" ]; then
|
|
echo "The following packages in $IPKG_PENDING_DIR will now be installed:"
|
|
echo $pending
|
|
for filename in $pending; do
|
|
if ipkg_install_file $filename; then
|
|
rm $filename
|
|
fi
|
|
done
|
|
fi
|
|
fi
|
|
return 0
|
|
}
|
|
|
|
ipkg_install_wanted() {
|
|
local wanted="`ipkg_status_matching 'Status:[[:space:]]*install.*not-installed'`"
|
|
|
|
if [ -n "$wanted" ]; then
|
|
echo "The following package were previously requested but have not been installed:"
|
|
echo $wanted
|
|
|
|
if [ -n "$FORCE_DEFAULTS" ]; then
|
|
echo "Installing them now."
|
|
else
|
|
printf "Install them now [Y/n] ? "
|
|
read response
|
|
case "$response" in
|
|
[Nn] | [Nn][Oo])
|
|
return 0
|
|
;;
|
|
esac
|
|
fi
|
|
|
|
ipkg_install $wanted
|
|
fi
|
|
|
|
return 0
|
|
}
|
|
|
|
ipkg_upgrade_pkg() {
|
|
local pkg="$1"
|
|
local avail_ver="`ipkg_info $pkg Version | ipkg_extract_value | head -n 1`"
|
|
|
|
is_installed=
|
|
for dest_name in `ipkg_dest_names`; do
|
|
local dest="`ipkg_dest_byname $dest_name`"
|
|
local sd=$dest/$IPKG_DIR_PREFIX
|
|
local inst_ver="`ipkg_status_sd $sd $pkg Version | ipkg_extract_value`"
|
|
if [ -n "$inst_ver" ]; then
|
|
is_installed=t
|
|
|
|
if [ -z "$avail_ver" ]; then
|
|
echo "Assuming locally installed package $pkg ($inst_ver) is up to date"
|
|
return 0
|
|
fi
|
|
|
|
if [ "$avail_ver" = "$inst_ver" ]; then
|
|
echo "Package $pkg ($inst_ver) installed in $dest_name is up to date"
|
|
elif ipkg_is_upgrade "$avail_ver" "$inst_ver"; then
|
|
echo "Upgrading $pkg ($dest_name) from $inst_ver to $avail_ver"
|
|
ipkg_get_install_dest $dest $pkg
|
|
else
|
|
echo "Not downgrading package $pkg from $inst_ver to $avail_ver"
|
|
fi
|
|
fi
|
|
done
|
|
|
|
if [ -z "$is_installed" ]; then
|
|
echo "Package $pkg does not appear to be installed"
|
|
return 0
|
|
fi
|
|
|
|
}
|
|
|
|
ipkg_upgrade() {
|
|
if [ $# -lt 1 ]; then
|
|
local pkgs="`ipkg_status_matching 'Status:.*[[:space:]]installed'`"
|
|
else
|
|
pkgs="$*"
|
|
fi
|
|
|
|
for pkg in $pkgs; do
|
|
ipkg_upgrade_pkg $pkg
|
|
done
|
|
}
|
|
|
|
ipkg_remove_pkg_dest() {
|
|
local dest="$1"
|
|
local pkg="$2"
|
|
local sd=$dest/$IPKG_DIR_PREFIX
|
|
local info_dir=$sd/info
|
|
|
|
if ! ipkg_status_installed_sd $sd $pkg; then
|
|
echo "ipkg_remove: Package $pkg does not appear to be installed in $dest"
|
|
if ipkg_status_mentioned_sd $sd $pkg; then
|
|
echo "Purging mention of $pkg from the ipkg database"
|
|
ipkg_status_remove_sd $sd $pkg
|
|
fi
|
|
return 1
|
|
fi
|
|
|
|
echo "ipkg_remove: Removing $pkg... "
|
|
|
|
local files="`cat $info_dir/$pkg.list`"
|
|
|
|
export PKG_ROOT=$dest
|
|
if [ -x "$info_dir/$pkg.prerm" ]; then
|
|
$info_dir/$pkg.prerm remove
|
|
fi
|
|
|
|
local conffiles="`ipkg_status_sd $sd $pkg Conffiles | ipkg_extract_value`"
|
|
|
|
local dirs_to_remove=
|
|
for file in $files; do
|
|
if [ -d "$dest/$file" ]; then
|
|
dirs_to_remove="$dirs_to_remove $dest/$file"
|
|
else
|
|
if echo " $conffiles " | grep -q " $file "; then
|
|
if echo " $conffiles " | grep -q " $file "`md5sum $dest/$file | sed 's/ .*//'`; then
|
|
rm -f $dest/$file
|
|
fi
|
|
else
|
|
rm -f $dest/$file
|
|
fi
|
|
fi
|
|
done
|
|
|
|
local removed_a_dir=t
|
|
while [ -n "$removed_a_dir" ]; do
|
|
removed_a_dir=
|
|
local new_dirs_to_remove=
|
|
for dir in $dirs_to_remove; do
|
|
if rmdir $dir >/dev/null 2>&1; then
|
|
removed_a_dir=t
|
|
else
|
|
new_dirs_to_remove="$new_dirs_to_remove $dir"
|
|
fi
|
|
done
|
|
dirs_to_remove="$new_dirs_to_remove"
|
|
done
|
|
|
|
if [ -n "$dirs_to_remove" ]; then
|
|
echo "ipkg_remove: Warning: Not removing the following directories since they are not empty:" >&2
|
|
echo "$dirs_to_remove" | sed -e 's/\/[/]\+/\//g' >&2
|
|
fi
|
|
|
|
if [ -x "$info_dir/$pkg.postrm" ]; then
|
|
$info_dir/$pkg.postrm remove
|
|
fi
|
|
|
|
ipkg_status_remove_sd $sd $pkg
|
|
set +o noglob
|
|
rm -f $info_dir/$pkg.*
|
|
set -o noglob
|
|
|
|
echo "Done."
|
|
}
|
|
|
|
ipkg_remove_pkg() {
|
|
local pkg="$1"
|
|
for dest in `ipkg_dests_all`; do
|
|
local sd=$dest/$IPKG_DIR_PREFIX
|
|
if ipkg_status_mentioned_sd $sd $pkg; then
|
|
ipkg_remove_pkg_dest $dest $pkg
|
|
fi
|
|
done
|
|
}
|
|
|
|
ipkg_remove() {
|
|
while [ $# -gt 0 ]; do
|
|
local pkg="$1"
|
|
shift
|
|
if [ -n "$DEST_NAME" ]; then
|
|
ipkg_remove_pkg_dest $IPKG_ROOT $pkg
|
|
else
|
|
ipkg_remove_pkg $pkg
|
|
fi
|
|
done
|
|
}
|
|
|
|
###########
|
|
# ipkg main
|
|
###########
|
|
|
|
# Parse options
|
|
while [ $# -gt 0 ]; do
|
|
arg="$1"
|
|
case $arg in
|
|
-d | -dest)
|
|
[ $# -gt 1 ] || ipkg_usage "option $arg requires an argument"
|
|
DEST_NAME="$2"
|
|
shift
|
|
;;
|
|
-o | -offline)
|
|
[ $# -gt 1 ] || ipkg_usage "option $arg requires an argument"
|
|
IPKG_OFFLINE_ROOT="$2"
|
|
shift
|
|
;;
|
|
-force-depends)
|
|
FORCE_DEPENDS=t
|
|
;;
|
|
-force-defaults)
|
|
FORCE_DEFAULTS=t
|
|
;;
|
|
-*)
|
|
ipkg_usage "unknown option $arg"
|
|
;;
|
|
*)
|
|
break
|
|
;;
|
|
esac
|
|
shift
|
|
done
|
|
|
|
[ $# -lt 1 ] && ipkg_usage "ipkg must have one sub-command argument"
|
|
cmd="$1"
|
|
shift
|
|
|
|
ipkg_load_configuration
|
|
mkdir -p /tmp/ipkg
|
|
|
|
case "$cmd" in
|
|
update|upgrade|list|info|status|install_pending)
|
|
;;
|
|
install|depends|remove|files|search)
|
|
[ $# -lt 1 ] && ipkg_usage "ERROR: the \`\`$cmd'' command requires an argument"
|
|
;;
|
|
*)
|
|
echo "ERROR: unknown sub-command \`$cmd'"
|
|
ipkg_usage
|
|
;;
|
|
esac
|
|
|
|
# Only install pending if we have an interactive sub-command
|
|
case "$cmd" in
|
|
upgrade|install)
|
|
ipkg_install_pending
|
|
ipkg_install_wanted
|
|
;;
|
|
esac
|
|
|
|
ipkg_$cmd $*
|
|
for a in `ls $IPKG_TMP`; do
|
|
rm -rf $IPKG_TMP/$a
|
|
done
|