diff options
Diffstat (limited to '')
-rw-r--r-- | update-smart-drivedb.in | 931 |
1 files changed, 931 insertions, 0 deletions
diff --git a/update-smart-drivedb.in b/update-smart-drivedb.in new file mode 100644 index 0000000..c4a8da0 --- /dev/null +++ b/update-smart-drivedb.in @@ -0,0 +1,931 @@ +#! /bin/sh +# +# smartmontools drive database update script +# +# Home page of code is: https://www.smartmontools.org +# +# Copyright (C) 2010-22 Christian Franke +# +# SPDX-License-Identifier: GPL-2.0-or-later +# +# $Id: update-smart-drivedb.in 5359 2022-04-24 15:11:30Z chrfranke $ +# + +set -e + +# Set by config.status +@ENABLE_SCRIPTPATH_TRUE@export PATH="@scriptpath@" +PACKAGE="@PACKAGE@" +VERSION="@VERSION@" +prefix="@prefix@" +exec_prefix="@exec_prefix@" +sbindir="@sbindir@" +datarootdir="@datarootdir@" +datadir="@datadir@" +localstatedir="@localstatedir@" +drivedbinstdir="@drivedbinstdir@" +drivedbdir="@drivedbdir@" + +# Download tools +os_dltools="@os_dltools@" + +# Default drivedb.h update branch +default_branch="@DRIVEDB_BRANCH@" + +# Default drivedb location +default_drivedb="$drivedbdir/drivedb.h" + +# GnuPG used to verify signature (disabled if empty) +gpg="@gnupg@" + +# Default command used for syntax check +default_smartctl="$sbindir/smartctl" + +# PATH information for help and error messages +@ENABLE_SCRIPTPATH_FALSE@pathinfo='$PATH' +@ENABLE_SCRIPTPATH_TRUE@pathinfo="'$PATH'" + +myname=$0 + +print_help() +{ +@ENABLE_SCRIPTPATH_TRUE@ pathinfo=" +@ENABLE_SCRIPTPATH_TRUE@ $pathinfo" + cat <<EOF +smartmontools $VERSION drive database update script + +Usage: $myname [OPTIONS] [DESTFILE] + + -s, --smartctl SMARTCTL + Use SMARTCTL for syntax check ('-s -' to disable) + [default: $default_smartctl] + -t, --tool [DIR/]TOOL + Use TOOL for download: $os_dltools + [default: first one found in $pathinfo] + -u, --url-of LOCATION + Use URL of LOCATION for download: + github (GitHub mirror of SVN repository) + sf (Sourceforge code browser) + svn (SVN repository) [default] + svni (SVN repository via HTTP instead of HTTPS) + trac (Trac code browser) + --url URL Download from URL + --file FILE Copy from local FILE +EOF + test "$drivedbinstdir" = "$drivedbdir" || cat <<EOF + --install Copy from originally installed drive database file + This is the same as: + '--no-verify --file $drivedbinstdir/drivedb.h' +EOF + cat <<EOF + --trunk Download from SVN trunk (requires '--no-verify') + --branch X.Y Download from branches/RELEASE_X_Y_DRIVEDB + --cacert FILE Use CA certificates from FILE to verify the peer + --capath DIR Use CA certificate files from DIR to verify the peer + --insecure Don't abort download if certificate verification fails + --no-verify Don't verify signature + --force Allow downgrades + --export-key Print the OpenPGP/GPG public key block + --dryrun Print download commands only + -q, --quiet Suppress info messages + -v, --verbose Verbose output + -h, --help Print this help text + +Updates $default_drivedb +or DESTFILE from branches/$default_branch of smartmontools +SVN repository. +EOF +} + +error() +{ + echo "$myname: $*" >&2 + test -z "$usageerr" || echo "Try '$myname -h' for help" >&2 + exit 1 +} + +err_notfound() +{ + case $1 in + */*) error "$1: not found $2" ;; + *) error "$1: not found in $pathinfo $2" ;; + esac +} + +# check_optarg "$@" +check_optarg() +{ + test $# -gt 1 || error "option '$1' requires an argument" +} + +warning() +{ + echo "$myname: (Warning) $*" >&2 +} + +# selecturl URL_OF +selecturl() +{ + case $1 in + github) # https://github.com/smartmontools/smartmontools/raw/origin/$branch/smartmontools/drivedb.h + # https://github.com/smartmontools/smartmontools/raw/master/smartmontools/drivedb.h + # redirected to: + url='https://raw.githubusercontent.com/smartmontools/smartmontools/master/smartmontools/drivedb.h' ;; + sf) url='https://sourceforge.net/p/smartmontools/code/HEAD/tree/trunk/smartmontools/drivedb.h?format=raw' ;; + svn) url='https://svn.code.sf.net/p/smartmontools/code/trunk/smartmontools/drivedb.h' ;; + svni) url='http://svn.code.sf.net/p/smartmontools/code/trunk/smartmontools/drivedb.h' ;; + trac) url='https://www.smartmontools.org/export/HEAD/trunk/smartmontools/drivedb.h' ;; + *) error "$1: is none of 'github sf svn svni trac'" + esac +} + +inpath() +{ + local d rc save + rc=1 + save=$IFS + IFS=':' + for d in $PATH; do + test -f "$d/$1" || continue + test -x "$d/$1" || continue + rc=0 + break + done + IFS=$save + return $rc +} + +iecho() +{ + test -n "$quiet" || echo "$*" +} + +vecho() +{ + test -n "$q" || echo "$*" +} + +# vrun COMMAND ARGS... +vrun() +{ + if [ -n "$dryrun" ]; then + echo "$*" + elif [ -n "$q" ]; then + "$@" 2>/dev/null + else + echo "$*" + "$@" + fi +} + +# vrun2 OUTFILE COMMAND ARGS... +vrun2() +{ + local f err rc + f=$1; shift + rc=0 + if [ -n "$dryrun" ]; then + echo "$* > $f" + else + vecho "$* > $f" + err=`"$@" 2>&1 > $f` || rc=$? + if [ -n "$err" ]; then + vecho "$err" >&2 + test $rc != 0 || rc=42 + fi + fi + return $rc +} + +# download URL FILE +download() +{ + local f u rc + u=$1; f=$2 + rc=0 + + case ${tool##*/} in + curl*) + # "Accept-Encoding" header avoids caching problems with svn URL + vrun "$tool" ${q:+-s} -f --max-redirs 0 \ + -H "Accept-Encoding: identity" \ + ${cacert:+--cacert "$cacert"} \ + ${capath:+--capath "$capath"} \ + ${insecure:+--insecure} \ + -o "$f" "$u" || rc=$? + ;; + + wget*) + # wget >= 1.16.1 sets "Accept-Encoding: identity" header by default + vrun "$tool" $q --max-redirect=0 \ + ${cacert:+--ca-certificate="$cacert"} \ + ${capath:+--ca-directory="$capath"} \ + ${insecure:+--no-check-certificate} \ + -O "$f" "$u" || rc=$? + ;; + + lynx*) + test -z "$cacert" || vrun export SSL_CERT_FILE="$cacert" + test -z "$capath" || vrun export SSL_CERT_DIR="$capath" + # Check also stderr as lynx does not return != 0 on HTTP error + vrun2 "$f" "$tool" -stderr -noredir -source "$u" || rc=$? + ;; + + svn*) + vrun "$tool" $q export \ + --non-interactive --no-auth-cache \ + ${cacert:+--config-option "servers:global:ssl-trust-default-ca=no"} \ + ${cacert:+--config-option "servers:global:ssl-authority-files=$cacert"} \ + ${insecure:+--trust-server-cert} \ + "$u" "$f" || rc=$? + ;; + + fetch*) # FreeBSD + vrun "$tool" $q --no-redirect \ + ${cacert:+--ca-cert "$cacert"} \ + ${capath:+--ca-path "$capath"} \ + ${insecure:+--no-verify-hostname} \ + -o "$f" "$u" || rc=$? + ;; + + ftp*) # OpenBSD + vrun "$tool" \ + ${cacert:+-S cafile="$cacert"} \ + ${capath:+-S capath="$capath"} \ + ${insecure:+-S dont} \ + -o "$f" "$u" || rc=$? + ;; + + *) error "$tool: unknown (internal error)" ;; + esac + return $rc +} + +# check_file FILE FIRST_CHAR MIN_SIZE MAX_SIZE +check_file() +{ + local firstchar f maxsize minsize size + f=$1; firstchar=$2; minsize=$3; maxsize=$4 + + # Check first chars + case `dd if="$f" bs=1 count=1 2>/dev/null` in + $firstchar) ;; + \<) echo "HTML error message"; return 1 ;; + *) echo "unknown file contents"; return 1 ;; + esac + + # Check file size + size=`wc -c < "$f"` + if test "$size" -lt $minsize; then + echo "too small file size $size bytes" + return 1 + fi + if test "$size" -gt $maxsize; then + echo "too large file size $size bytes" + return 1 + fi + return 0 +} + +# unexpand_svn_id < INFILE > OUTFILE +unexpand_svn_id() +{ + sed 's,\$''Id'': drivedb\.h [0-9][0-9]* 2[-0-9]* [012][:0-9]*Z [a-z][a-z0-9]* \$,$''Id''$,' +} + +# selectkey BRANCH +selectkey() +{ + case $1 in + RELEASE_5_4[0-3]_DRIVEDB|RELEASE_6_[0-6]_DRIVEDB) +# Smartmontools Signing Key (ext. to 2024) <smartmontools-database@listi.jpberlin.de> +# Smartmontools Signing Key (through 2018) <smartmontools-database@listi.jpberlin.de> +# Smartmontools Signing Key (through 2018) <smartmontools-database@lists.sourceforge.net> +# Key ID DFD22559 +public_key="\ +-----BEGIN PGP PUBLIC KEY BLOCK----- + +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 +1TC5jSVOIEqOE4TwzRz1Y/dfqSU= +=3Lcg +-----END PGP PUBLIC KEY BLOCK----- +" + ;; + + RELEASE_7_[023]_DRIVEDB) +# Smartmontools Signing Key (through 2025) <smartmontools-database@listi.jpberlin.de> +# Smartmontools Signing Key (through 2020) <smartmontools-database@listi.jpberlin.de> +# Key ID 721042C5 +public_key="\ +-----BEGIN PGP PUBLIC KEY BLOCK----- + +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 +90iSgZdt5cZgoH04weXCMwDugn/+Q3rzKvRUTrEfSOivJYg65D/mhbz6HoUTs4JD +SstTYa9qNCwKQGRSeis4PAgu0hCpnDAhZuN3Ja5AFC2Wi2szQ7R+Zx/JucIBm5S4 +U30W66MtsyUHeulSJ3AV3HrbFfnqu6zfQM4XLw7MpAtQUNJceS/lWfGIquAp3tY/ +IjZIHwgZqKB3czWDhM83wBzCWgAmxyzIrpb4MBYJ5PGuCyC7R/YTdtPJXxsPQl2l +znsX/9ssa4kBHAQTAQIABgUCX+CfSAAKCRDzh2PO39IlWVcuB/9UkLaPtGY4sDDV +/A7qjSvSy93mv8gkaIj9dhqoZw+r7cLiEtX04Cz9PqocOFgCYJXKrufHNNkHke2A +jE9EJfRKiPU/bkeWmrACvtrOd/DZbdmXfxTOekOr516D2ip/U8GBPw6zxfCQVot6 +htpBpB6zzMDtzMOeLnkOxoxR4EMu5K6eJ48bHvG/lbGBByyfRzhtqPh6AAA9G1CC +IdhNkaA5W1qums3N1mCXrTBnWyjaFhdnttGQfrMdHvTQ77HeL0c2axT2y5PYfrXY +2ZfZowYLEtFXRSTpDaJfgG+qem3N+pMv6SMOG/4CvlH4/3Hq0aCNvKcY5KUXfIgT +xmc3/n/wtFNTbWFydG1vbnRvb2xzIFNpZ25pbmcgS2V5ICh0aHJvdWdoIDIwMjAp +IDxzbWFydG1vbnRvb2xzLWRhdGFiYXNlQGxpc3RpLmpwYmVybGluLmRlPokCPgQT +AQIAKAIbAwYLCQgHAwIGFQgCCQoLBBYCAwECHgECF4AFAl/gnzAFCQ0umpsACgkQ +6nSrJXIQQsVK7RAAqbZfT3wZEfJkw8MK2JlvgGWH76fHKn5ZoH5i0mA4AvN4QLbU +5Q20HmqHnO9mfAZQ6u4Tn/aFcYT7nlSsEsEmFX+s5QU2y6m2Tx9ThDbZ03ezREOS +0wNf0FOQunV9ZVPT/7cKIgWJa5mZy+LClor9OHllyGUfs9tKNzwxaHh1zBrCNJow +Fi/1bkWy3iMc7vZhWHASwPSp64KHjB4UdMz2hV4pROiUhWi7BY0exIHyZrkcANMP +Hhl9lP32ZvNbOy8osBdPUgXyK3HePD+ftcwJMkoc4mFQXYi9UY7NQpk7STRO10cx +Kq/CgDDvYxbnViRjQoJ0sfwKCaOfsnY/gea7I0aCx8uNISYpHO9iMidd/tJ7+lgx +NiKZTI0EppHYvkyMY15/NGb0gTJbYjuVYdbqDS9mnLuLQAjAX43+n9ND2NjX1o0q +Z9bBqV2VFioNmnxKqGphhRFX9jEzTklieOjhpRrd8v9ljprT6vLFNpYpeLkel8om +VFXrHxrfzKtVFcto5wqHVOcyZyE2zm1QmsS8qvWOTrNfY6p2q9MA2rysqdfgfvN7 +pNDaXutK6ooQi6YlyyTA2ANnHFKa0ncRH+dg+5OF9rhNvM7RyaBXgxF7+5gnU5Gb +VQRKbJ+LOtSKkj0pApR5AKSwyGslZ2bNVlKsADWhk5xj8QlHVlNWiht+i/6JARwE +EwECAAYFAlwmhpwACgkQL83sC9OvGqsVOggAqLB5eQrUv8E9ikD6kJCito827bzD +WF29yD7PvfhjXaz5in54jOVpwg3o9CsqIjjRW0/1bBVswC8ZL0sAdZ+GDSDMw5F2 +IpkD77gjnFY79M/e6C9xYyxYzHC7emDPSz9IroOvdkkEgrB+OABKkaOCcS18P4Lk +3WNHaPw5c7aI0z1iJP52EmSfvB8r86mtUFJB+f15eD/4vaRfkZLFjF9FQ3kgEK1U ++rV4s1O2bCFfP3WPDcc83NgwRUvtXmcSOSOIoXnemJzyJr+JnqCWVET4XWF6i20m +RFXVEpWtf5AkJYgR3z/jW0djELbBWA/35bAnpXy5pDHv9NbZsTkBZxK/kokBHAQT +AQIABgUCXCaGnQAKCRAY7NpGy/a6xn4lB/90tXTnZsgmoftol9uivfQrPdR88WmO +ZLYmUeQAd1rqSFMxe+KzO/qLuU8s6OF4nznwL2cPfbGZxezM4PiYmAmbbEU/3gTO +NwjVBBA0Gfimy/fITEezFtCigo1thkaJ195g/dqY+zE3Vt4rzC03j1vx8mUHRPU6 +kkvKj8cP0j+XHX2xQDsTXTstfnom29wBmGnvSZ9HgcdL71e1VXJXwikmnO3P4J/1 +C2LeCOlWrGqWZ2c0WBLKdJnsYUx7Dm/OvkkB4lF+zWp98zS8jS/5h+1apVgEzrdT +MvT8ydTkUr7ObKGkIhK+L+Xo5BD+V9Qf6xKGYPwhhdj/E5/kyjULrm10iQEcBBMB +AgAGBQJcJoadAAoJEPOHY87f0iVZfiUH/3yKS5wGvTeRInse8+W1WzKuto3XzqXL +ngb9QXWw7nCwqmNS7PbzDnufQi2ThKrMfcK14WgNYABNZPU75I+6bcb0oCB5tloo +IUEV/2Ut/5Hl/83zFFoNA/kQKVz8kIDqgRcxC+zY2VJ4eTKHyQDvXygVk8wnKTBa +e3gX+CIZqJHPXiiygHlbl31Mi3G1Iaxu57dP6ocV0vX1dytKSwd4Rbviwwb4L76o +/tVT9t3GwFM15uK1SqtnAaiaktEdMi3XI4d01H3VUVz/iR0XQbf13RZoEM6CJWms +Q/qvYlwkbKOdlahjoHrFlkhADSBaO9N1OZp3OYDjziIujMdt2IPKnmM= +=7MQk +-----END PGP PUBLIC KEY BLOCK----- +" + ;; + + *) error "No known public key for branches/$1" ;; + esac +} + +# gpg_verify FILE.asc FILE +gpg_verify() +{ + local gnupgtmp i out rc + + # Create temp home dir + gnupgtmp="$tmpdir/.gnupg.$$.tmp" + rm -f -r "$gnupgtmp" + mkdir "$gnupgtmp" || exit 1 + chmod 0700 "$gnupgtmp" + + # Import public key + if ! out=`echo "$public_key" | "$gpg" --batch --no-tty --homedir="$gnupgtmp" --import 2>&1`; then + echo "$out" >&2 + exit 1 + fi + vecho "$out" + + # Verify + rc=0 + out=`"$gpg" --batch --no-tty --homedir="$gnupgtmp" --verify "$1" "$2" </dev/null 2>&1` || rc=1 + if [ $rc = 0 ]; then + vecho "$out" + else + # print gpg error always + echo "$out" >&2 + fi + + # Stop the gpg-agent possibly started by gpg + if [ -n "$gpgconf" ]; then + out=`"$gpgconf" --homedir="$gnupgtmp" --kill gpg-agent </dev/null 2>&1` || echo "$out" >&2 + fi + + # Remove temp home dir, retry on failure + i=0 + while ! out=`rm -f -r "$gnupgtmp" 2>&1`; do + i=$((i+1)) + if [ $i -ge 10 ]; then + echo "$out" >&2; break + fi + vecho "$out" + sleep 1 + done + + return $rc +} + +# get_db_version DRIVEDB > VERSION +get_db_version() +{ + local r v x + x=`sed -n '/^[ {]*"VERSION: *[^"]*"/{ + s,^[ {]*"VERSION: \([1-9][./0-9]* [^"]*\)".*$,\1,p + q + }' "$1"` || return 1 + v=${x%% *} + test -n "$v" || return 0 + if [ "${v%/*}" = "$v" ]; then # trunk: get rev from expanded SVN-Id + r=`echo "$x" | sed -n 's,^[^$]*\$''Id: drivedb\.h \([1-9][0-9]*\) .*$,\1,p'` + test -n "$r" || r="?" + v="$v/$r" + fi + echo "$v" +} + +# mv_all PREFIX OLD NEW +mv_all() +{ + mv -f "${1}${2}" "${1}${3}" && \ + mv -f "${1}${2}.raw" "${1}${3}.raw" && \ + if [ -f "${1}${2}.raw.asc" ]; then + mv -f "${1}${2}.raw.asc" "${1}${3}.raw.asc" + else + rm -f "${1}${3}.raw.asc" + fi +} + +# Parse options +smartctl=$default_smartctl +tool= +url_of= +url= +file= +quiet= +q="-q" +dryrun= +trunk= +branch=$default_branch +cacert= +capath= +insecure= +no_verify= +force= +expkey= +usageerr=t + +while true; do case $1 in + -s|--smartctl) + check_optarg "$@"; shift + smartctl=$1 ;; + + -t|--tool) + check_optarg "$@"; shift + tool=$1 ;; + + -u|--url-of) + check_optarg "$@"; shift + url_of=$1 ;; + + -q|--quiet) + quiet=t ;; + + -v|--verbose) + q= ;; + + --url) + check_optarg "$@"; shift + url=$1 ;; + + --file) + check_optarg "$@"; shift + file=$1 ;; + + --install) + test "$drivedbinstdir" != "$drivedbdir" \ + || error "'$1' is not supported in this configuration" + file="$drivedbinstdir/drivedb.h" + no_verify=t ;; + + --dryrun) + dryrun=t ;; + + --trunk) + trunk=t ;; + + --branch) + check_optarg "$@"; shift + branch=`echo "$1" | sed -n 's,^\([567]\)\.\([0-9][0-9]*\)$,RELEASE_\1_\2_DRIVEDB,p'` + test -n "$branch" || error "invalid branch version '$1'" ;; + + --cacert) + check_optarg "$@"; shift + cacert=$1 ;; + + --capath) + check_optarg "$@"; shift + capath=$1 ;; + + --insecure) + insecure=t ;; + + --no-verify) + no_verify=t ;; + + --force) + force=t ;; + + --export-key) + expkey=t ;; + + -h|--help) + print_help + exit 0 ;; + + -*) + error "unknown option '$1'" ;; + + *) + break ;; +esac; shift; done + +if [ -n "$expkey" ]; then + selectkey "$branch" + cat <<EOF +$public_key +EOF + exit 0 +fi + +case $# in + 0) drivedb=$default_drivedb ;; + 1) drivedb=$1 ;; + *) error "only one DESTFILE argument is allowed" ;; +esac + +# Check selected source +case "${url_of:+url_of}${url:+url}${file:+file}" in + ''|url_of) + test -n "$url_of" || url_of=svn + selecturl "$url_of" + if [ -z "$trunk" ]; then # select branch + url=`echo "$url" | sed -e "s,/trunk/,/branches/$branch/," \ + -e "s,/master/,/origin/$branch/,"` + elif [ -z "$no_verify" ]; then + error "'--trunk' requires '--no-verify'" + fi ;; + url) + test -z "`echo "$url" | sed -e 's,^[a-z][a-z0-9]*:[^ ][^ ]*$,,'`" \ + || error "$url: Invalid URL" ;; + file) ;; + *) error "only one of '-u', '--url', '--file' is allowed" ;; +esac + +# Determine path of signature file +file_asc= +url_asc= +if [ -z "$no_verify" ]; then case $url in + '') file_asc="$file.raw.asc" ;; + *\?*) url_asc=`echo "$url" | sed 's,?,.raw.asc?,'` ;; + *) url_asc="$url.raw.asc" ;; +esac; fi + +if [ -z "$file" ]; then + if [ -z "$tool" ]; then + # Find download tool in PATH + for t in $os_dltools; do + if inpath "$t"; then + tool=$t + break + fi + done + test -n "$tool" || error "found none of '$os_dltools' in $pathinfo" + else + # Check tool name + found= + for t in $os_dltools; do + case ${tool##*/} in + $t*) found=t; break ;; + esac + done + test -n "$found" || error "$tool: is none of '$os_dltools'" + fi +fi + +# Check option compatibility +case "$tool:$url_of" in + svn:svn*) ;; + svn:*) error "'-t svn' requires '-u svn' or '-u svni'" ;; +esac +case "$tool:${capath:+set}" in + svn:set) warning "'--capath' is ignored if '-t svn' is used" ;; +esac +case "$url_of:$insecure" in + svni:t) insecure= ;; + svni:*) error "'-u svni' requires '--insecure'" ;; +esac +case "$tool:$insecure" in + lynx:t) warning "'--insecure' is ignored if '-t lynx' is used" ;; +esac + +# Check for smartctl +if [ "$smartctl" != "-" ]; then + "$smartctl" -V >/dev/null 2>&1 \ + || err_notfound "$smartctl" "('-s -' to ignore)" +fi + +# Check for GnuPG +gpgconf= +if [ -z "$no_verify" ]; then + test -n "$gpg" \ + || error "GnuPG is not available ('--no-verify' to ignore)" + "$gpg" --version >/dev/null 2>&1 \ + || err_notfound "$gpg" "('--no-verify' to ignore)" + selectkey "$branch" + case $gpg in + */*) gpgconf="${gpg%/*}/gpgconf" ;; + *) gpgconf="gpgconf" ;; + esac + "$gpgconf" --version >/dev/null 2>&1 || gpgconf= +fi +usageerr= + +# Use destination directory as temp directory for gpg +tmpdir=`dirname "$drivedb"` + +# Remove possible garbage from last download +test -n "$dryrun" || rm -f "$drivedb.new" "$drivedb.new.raw" "$drivedb.new.raw.asc" || exit 1 + +if [ -n "$url" ]; then + # Download + vecho "Download drivedb.h with $tool" + rc=0 + download "$url" "$drivedb.new" || rc=$? + if [ $rc != 0 ]; then + rm -f "$drivedb.new" + error "drivedb.h: download failed ($tool: exit $rc)" + fi + + if [ -n "$url_asc" ]; then + vecho "Download drivedb.h.raw.asc with $tool" + rc=0 + download "$url_asc" "$drivedb.new.raw.asc" || rc=$? + if [ $rc != 0 ]; then + rm -f "$drivedb.new" "$drivedb.new.raw.asc" + error "drivedb.h.raw.asc: download failed ($tool: exit $rc) ('--no-verify' to ignore)" + fi + fi +else + # Copy from local file + if [ ! -f "$file" ]; then + error "$file: file not found" + fi + if [ -n "$file_asc" ] && [ ! -f "$file_asc" ]; then + error "$file_asc: file not found ('--no-verify' to ignore)" + fi + + if ! vrun cp "$file" "$drivedb.new"; then + error "$file: copy failed" + fi + if [ -n "$file_asc" ]; then + if ! vrun cp "$file_asc" "$drivedb.new.raw.asc"; then + rm -f "$drivedb.new" + error "$file_asc: copy failed" + fi + fi +fi + +test -z "$dryrun" || exit 0 + +# Check files, adjust timestamps and permissions +if ! errmsg=`check_file "$drivedb.new" '/' 10000 1000000`; then + rm -f "$drivedb.new.raw.asc" + mv -f "$drivedb.new" "$drivedb.error" + error "$drivedb.error: $errmsg" +fi +touch "$drivedb.new" +chmod 0644 "$drivedb.new" + +if [ -f "$drivedb.new.raw.asc" ]; then + if ! errmsg=`check_file "$drivedb.new.raw.asc" '-' 200 2000`; then + rm -f "$drivedb.new" + mv -f "$drivedb.new.raw.asc" "$drivedb.error.raw.asc" + error "$drivedb.error.raw.asc: $errmsg" + fi + touch "$drivedb.new.raw.asc" + chmod 0644 "$drivedb.new.raw.asc" +fi + +# Create raw file with unexpanded SVN Id +# (This assumes newlines are LF and not CR/LF) +unexpand_svn_id < "$drivedb.new" > "$drivedb.new.raw" +chmod 0644 "$drivedb.new.raw" + +# Check whether installed file is identical +equal= +if [ -f "$drivedb" ]; then + if [ ! -f "$drivedb.raw" ]; then + # Create missing raw file + unexpand_svn_id < "$drivedb" > "$drivedb.raw" + chmod 0644 "$drivedb.raw" + fi + # Ignore missing Id keyword expansion in new file + if cmp "$drivedb.raw" "$drivedb.new.raw" >/dev/null 2>&1 \ + && { cmp "$drivedb" "$drivedb.new" >/dev/null 2>&1 \ + || cmp "$drivedb.raw" "$drivedb.new" >/dev/null 2>&1; } + then + equal=t + fi +fi + +if [ -z "$no_verify" ]; then + # Verify raw file + if ! gpg_verify "$drivedb.new.raw.asc" "$drivedb.new.raw"; then + mv_all "$drivedb" ".new" ".error" + test -z "$equal" \ + || warning "$drivedb: *** installed file is identical to broken new file ***" + error "$drivedb.error.raw: *** BAD signature or outdated key ***" + fi +fi + +# Get version +newver=`get_db_version "$drivedb.new"` +if [ -z "$newver" ]; then + if [ -z "$force" ]; then + mv_all "$drivedb" ".new" ".error" + error "$drivedb.error: no VERSION information found ('--force' to ignore)" + fi + newver="?/?" +elif [ "${newver##*/}" = "?" ]; then + if [ -z "$trunk" ]; then + mv_all "$drivedb" ".new" ".error" + error "$drivedb.error: VERSION information is incomplete ('--trunk' to ignore)" + fi +fi + +if [ "$smartctl" != "-" ]; then + # Check syntax + if ! "$smartctl" -B "$drivedb.new" -P showall >/dev/null; then + mv_all "$drivedb" ".new" ".error" + error "$drivedb.error: rejected by $smartctl, probably no longer compatible" + fi + vecho "$smartctl: syntax OK" +fi + +# Always install if missing +rm -f "$drivedb.lastcheck" +if [ ! -f "$drivedb" ]; then + mv_all "$drivedb" ".new" "" + iecho "$drivedb $newver newly installed${no_verify:+ (NOT VERIFIED)}" + exit 0 +fi + +# Keep old file if identical +if [ -n "$equal" ]; then + if [ -f "$drivedb.new.raw.asc" ] \ + && ! cmp "$drivedb.new.raw.asc" "$drivedb.raw.asc" >/dev/null 2>&1; then + mv -f "$drivedb.new.raw.asc" "$drivedb.raw.asc" + iecho "$drivedb.raw.asc $newver updated" + fi + rm -f "$drivedb.new" "$drivedb.new.raw" "$drivedb.new.raw.asc" + touch "$drivedb.lastcheck" + iecho "$drivedb $newver is already up to date${no_verify:+ (NOT VERIFIED)}" + exit 0 +fi + +# Check branch and file version +oldver=`get_db_version "$drivedb"` +test -n "$oldver" || oldver="?/?" +if [ "${newver##*/}" = "?" ] \ + || [ "${oldver##*/}" = "?" ] \ + || [ "${newver%/*}" != "${oldver%/*}" ]; then + # Always install from trunk or other branch + updmsg="replaced with" +elif [ "${newver##*/}" -lt "${oldver##*/}" ]; then + # Install older file only if '--force' is used + if [ -z "$force" ]; then + rm -f "$drivedb.new" "$drivedb.new.raw" "$drivedb.new.raw.asc" + iecho "$drivedb $oldver not downgraded to $newver ('--force' to override)" + exit 0 + fi + updmsg="downgraded to" +else + updmsg="updated to" +fi + +mv_all "$drivedb" "" ".old" +mv_all "$drivedb" ".new" "" +iecho "$drivedb $oldver $updmsg $newver${no_verify:+ (NOT VERIFIED)}" |