Magellan Linux

Diff of /branches/mage-next/src/mage4.functions.sh

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 1209 by niro, Fri Jan 28 20:37:27 2011 UTC revision 1648 by niro, Fri Jan 13 20:51:18 2012 UTC
# Line 2  Line 2 
2  # Magellan Linux Installer Functions (mage.functions.sh)  # Magellan Linux Installer Functions (mage.functions.sh)
3  # $Header: /home/cvsd/magellan-cvs/magellan-src/mage/usr/lib/mage/mage4.functions.sh,v 1.38 2008-10-05 10:32:24 niro Exp $  # $Header: /home/cvsd/magellan-cvs/magellan-src/mage/usr/lib/mage/mage4.functions.sh,v 1.38 2008-10-05 10:32:24 niro Exp $
4    
5    COLRED="\033[1;6m\033[31m"
6    COLGREEN="\033[1;6m\033[32m"
7    COLYELLOW="\033[1;6m\033[33m"
8    COLBLUE="\033[1;6m\033[34m"
9    COLMAGENTA="\033[1;6m\033[35m"
10    COLWHITE="\033[1;6m\033[37m"
11    COLGRAY="\033[0;6m\033[37m"
12    COLBOLD="\033[1m"
13    COLDEFAULT="\033[0m"
14    
15    if [[ ${NOCOLORS} = true ]]
16    then
17     COLRED=""
18     COLGREEN=""
19     COLYELLOW=""
20     COLBLUE=""
21     COLMAGENTA=""
22     COLWHITE=""
23     COLGRAY=""
24     COLBOLD=""
25     COLDEFAULT=""
26    fi
27    
28  mage_setup()  mage_setup()
29  {  {
30   [ ! -d ${MROOT}${INSTALLDB} ] && \   [ ! -d ${MROOT}${INSTALLDB} ] && \
# Line 15  mage_setup() Line 38  mage_setup()
38   return 0   return 0
39  }  }
40    
41    mchecksum()
42    {
43     local i
44     local rundir
45     local file
46     local method
47     local cmd
48     local retval
49    
50     # very basic getops
51     for i in $*
52     do
53     case $1 in
54     --rundir|-r) shift; rundir="$1" ;;
55     --file|-f) shift; file="$1" ;;
56     --method|-m) shift; method="$1" ;;
57     esac
58     shift
59     done
60    
61     # sanity checks
62     [[ -z ${rundir} ]] && die "mchecksum(): rundir missing"
63     [[ -z ${file} ]] && die "mchecksum(): file missing"
64     [[ -z ${method} ]] && die "mchecksum(): method missing"
65    
66     case ${method} in
67     md5) cmd="md5sum" ;;
68     sha256) cmd="sha256sum" ;;
69     *) die "mchecksum(): unknown method '${method}'" ;;
70     esac
71    
72     if [[ -d ${rundir} ]]
73     then
74     pushd ${rundir} &> /dev/null
75     ${cmd} -c ${file} &> /dev/null
76     retval="$?"
77     popd &> /dev/null
78     else
79     retval=1
80     fi
81    
82     return "${retval}"
83    }
84    
85  unpack_packages()  unpack_packages()
86  {  {
87   local list="$@"   local list="$@"
# Line 23  unpack_packages() Line 90  unpack_packages()
90   local pkgtype   local pkgtype
91   local count_current   local count_current
92   local count_total   local count_total
93     local tar_opts
94    
95   # get count of total packages   # get count of total packages
96   declare -i count_current=0   declare -i count_current=0
# Line 54  unpack_packages() Line 122  unpack_packages()
122   continue   continue
123   fi   fi
124    
125     # busybox?
126     if need_busybox_support tar
127     then
128     tar_opts="xjf"
129     else
130     tar_opts="xjmf"
131     fi
132    
133   echo -e " ${COLBLUE}***${COLDEFAULT} unpacking (${count_current}/${count_total}): ${pkg} ... "   echo -e " ${COLBLUE}***${COLDEFAULT} unpacking (${count_current}/${count_total}): ${pkg} ... "
134   tar xjmf ${PKGDIR}/${pkg} -C ${BUILDDIR} || die "Unpacking package ${pkg}"   tar ${tar_opts} ${PKGDIR}/${pkg} -C ${BUILDDIR} || die "Unpacking package ${pkg}"
135   done   done
136    
137   # add a crlf for a better view   # add a crlf for a better view
# Line 130  install_directories() Line 206  install_directories()
206   while read pathto posix user group   while read pathto posix user group
207   do   do
208   [ -z "${pathto}" ] && continue   [ -z "${pathto}" ] && continue
209   [[ ${VERBOSE} = on ]] && echo -e "\t>>> DIR:  ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t>>> DIR:  ${MROOT}${pathto}"
   
210    
211   # monitors /etc/env.d -> env-rebuild   # monitors /etc/env.d -> env-rebuild
212   [[ ${pathto} = /etc/env.d ]] && export MAGE_ENV_REBUILD=true   [[ ${pathto} = /etc/env.d ]] && export MAGE_ENV_REBUILD=true
# Line 207  install_files() Line 282  install_files()
282   case ${retval} in   case ${retval} in
283   # file is not protected - (over)write it   # file is not protected - (over)write it
284   0|3)   0|3)
285   [[ ${VERBOSE} = on ]] && echo -e "\t>>> FILE: ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t>>> FILE: ${MROOT}${pathto}"
286   install -m "${posix}" -o "${user}" -g "${group}" \   install -m "${posix}" -o "${user}" -g "${group}" \
287   ${BUILDDIR}/${pkgname}/binfiles/"${pathto}" \   ${BUILDDIR}/${pkgname}/binfiles/"${pathto}" \
288   "${MROOT}${pathto}"   "${MROOT}${pathto}"
# Line 219  install_files() Line 294  install_files()
294   "${user}" \   "${user}" \
295   "${group}" \   "${group}" \
296   "$(fix_mtime "${BUILDDIR}/${pkgname}"/.mtime \   "$(fix_mtime "${BUILDDIR}/${pkgname}"/.mtime \
297   "${MROOT}${pathto}")" \   "${MROOT}${pathto}")" \
298   "${md5sum}"   "${md5sum}"
299   ;;   ;;
300    
301   # file is protected, write backup file   # file is protected, write backup file
302   2)   2)
303   if [[ ${VERBOSE} = on ]]   if mqueryfeature "verbose"
304   then   then
305   echo -en "${COLRED}"   echo -en "${COLRED}"
306   echo -n "! prot "   echo -n "! prot "
# Line 246  install_files() Line 321  install_files()
321   "${user}" \   "${user}" \
322   "${group}" \   "${group}" \
323   "$(fix_mtime "${BUILDDIR}/${pkgname}"/.mtime \   "$(fix_mtime "${BUILDDIR}/${pkgname}"/.mtime \
324   "${dest_protected}")" \   "${dest_protected}")" \
325   "${md5sum}"   "${md5sum}"
326    
327   # update global MAGE_PROTECT_COUNTER   # update global MAGE_PROTECT_COUNTER
# Line 256  install_files() Line 331  install_files()
331    
332   # file is protected but ignored, delete the update/do nothing   # file is protected but ignored, delete the update/do nothing
333   4)   4)
334   if [[ ${VERBOSE} = on ]]   if mqueryfeature "verbose"
335   then   then
336   echo -en "${COLRED}"   echo -en "${COLRED}"
337   echo -n "! ignr "   echo -n "! ignr "
338   echo -en "${COLDEFAULT}"   echo -en "${COLDEFAULT}"
339   echo " === FILE: ${MROOT}${pathto}"   echo " === FILE: ${MROOT}${pathto}"
340   fi   fi
341   # simply do nothing here   # simply do nothing here - only fix mtime
342     fix_descriptor ${pkgname}/.files \
343     "${pathto}" \
344     "${posix}" \
345     "${user}" \
346     "${group}" \
347     "$(fix_mtime "${BUILDDIR}/${pkgname}"/.mtime \
348     "${MROOT}${pathto}")" \
349     "${md5sum}"
350   ;;   ;;
351   esac   esac
352   done < ${BUILDDIR}/${pkgname}/.files   done < ${BUILDDIR}/${pkgname}/.files
# Line 307  install_symlinks() Line 390  install_symlinks()
390   while read pathto posix link mtime   while read pathto posix link mtime
391   do   do
392   [ -z "${pathto}" ] && continue   [ -z "${pathto}" ] && continue
393   [[ ${VERBOSE} = on ]] && echo -e "\t>>> LINK: ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t>>> LINK: ${MROOT}${pathto}"
394    
395   ln -snf "${link}" "${MROOT}${pathto}"   ln -snf "${link}" "${MROOT}${pathto}"
396    
# Line 354  install_blockdevices() Line 437  install_blockdevices()
437   # sets fieldseperator to "§" instead of " "   # sets fieldseperator to "§" instead of " "
438   IFS=§   IFS=§
439    
440   while read pathto posix user group   while read pathto posix major minor user group
441   do   do
442   [ -z "${pathto}" ] && continue   [ -z "${pathto}" ] && continue
443   [[ ${VERBOSE} = on ]] && echo -e "\t>>> PIPE: ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t>>> PIPE: ${MROOT}${pathto}"
444    
445   mkfifo -m "${posix}" "${MROOT}${pathto}"   mknod -m "${posix}" "${MROOT}${pathto}"
446   chown "${user}:${group}" "${MROOT}${pathto}"   # make it optional atm !!
447     if [[ ! -z ${user} ]] && [[ ! -z ${group} ]]
448     then
449     chown "${user}:${group}" "${MROOT}${pathto}" b "${major}" "${minor}"
450     fi
451   done < ${BUILDDIR}/${pkgname}/.pipes   done < ${BUILDDIR}/${pkgname}/.pipes
452    
453   # very important: unsetting the '§' fieldseperator   # very important: unsetting the '§' fieldseperator
# Line 397  install_characterdevices() Line 484  install_characterdevices()
484   while read pathto posix major minor user group   while read pathto posix major minor user group
485   do   do
486   [ -z "${pathto}" ] && continue   [ -z "${pathto}" ] && continue
487   [[ ${VERBOSE} = on ]] && echo -e "\t>>> CHAR: ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t>>> CHAR: ${MROOT}${pathto}"
488    
489   mknod -m ${posix} "${MROOT}${pathto}" c ${major} ${minor}   mknod -m ${posix} "${MROOT}${pathto}" b "${major}" "${minor}"
490   chown "${user}:${group}" "${MROOT}${pathto}"  
491     # make it optional atm !!
492     if [[ ! -z ${user} ]] && [[ ! -z ${group} ]]
493     then
494     chown "${user}:${group}" "${MROOT}${pathto}"
495     fi
496   done < ${BUILDDIR}/${pkgname}/.char   done < ${BUILDDIR}/${pkgname}/.char
497    
498   # very important: unsetting the '§' fieldseperator   # very important: unsetting the '§' fieldseperator
# Line 426  install_fifos() Line 518  install_fifos()
518   # check needed global vars   # check needed global vars
519   [ -z "${BUILDDIR}" ] && die "install_fifos() \$BUILDDIR not set."   [ -z "${BUILDDIR}" ] && die "install_fifos() \$BUILDDIR not set."
520    
521   [ ! -f ${BUILDDIR}/${pkgname}/.fifo ] && die "install_fifos() .fifo not found"   # make it optional atm !!
522     #[ ! -f ${BUILDDIR}/${pkgname}/.fifo ] && die "install_fifos() .fifo not found"
523     [ ! -f ${BUILDDIR}/${pkgname}/.fifo ] && return
524    
525   # sets fieldseperator to "§" instead of " "   # sets fieldseperator to "§" instead of " "
526   IFS=§   IFS=§
# Line 434  install_fifos() Line 528  install_fifos()
528   while read pathto posix user group   while read pathto posix user group
529   do   do
530   [ -z "${pathto}" ] && continue   [ -z "${pathto}" ] && continue
531   [[ ${VERBOSE} = on ]] && echo -e "\t>>> FIFO: ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t>>> FIFO: ${MROOT}${pathto}"
532    
533   mkfifo -m "${posix}" "${MROOT}${pathto}"   mkfifo -m "${posix}" "${MROOT}${pathto}"
534   chown "${user}:${group}" "${MROOT}${pathto}"   chown "${user}:${group}" "${MROOT}${pathto}"
# Line 456  build_doinstall() Line 550  build_doinstall()
550    
551   # sanity checks; abort if not given   # sanity checks; abort if not given
552   [ -z "${pkgname}" ] && die "build_doinstall() \$pkgname not given."   [ -z "${pkgname}" ] && die "build_doinstall() \$pkgname not given."
553    
554   # this is only a wrapper   # this is only a wrapper
555    
556   # NOTE:   # NOTE:
# Line 553  install_database_entry() Line 647  install_database_entry()
647   local i   local i
648   for i in .char .dirs .files .pipes .symlinks .fifo   for i in .char .dirs .files .pipes .symlinks .fifo
649   do   do
650   install -m 0644 ${BUILDDIR}/${pkgname}/${i} \   # make .fifo optional atm
651   ${dbrecorddir}/${i}   if [[ -f ${BUILDDIR}/${pkgname}/${i} ]]
652     then
653     install -m 0644 ${BUILDDIR}/${pkgname}/${i} ${dbrecorddir}/${i}
654     fi
655   done   done
656   ;;   ;;
657   esac   esac
# Line 765  remove_symlinks() Line 862  remove_symlinks()
862   [ -z "${pathto}" ] && continue   [ -z "${pathto}" ] && continue
863   if [ ! -L "${MROOT}${pathto}" ]   if [ ! -L "${MROOT}${pathto}" ]
864   then   then
865   [[ ${VERBOSE} = on ]] && \   mqueryfeature "verbose" && \
866   echo -e "${COLRED}! exist${COLDEFAULT} === LINK: ${MROOT}${pathto}"   echo -e "${COLRED}! exist${COLDEFAULT} === LINK: ${MROOT}${pathto}"
867   continue   continue
868   fi   fi
# Line 777  remove_symlinks() Line 874  remove_symlinks()
874   # 1=keep me   #   # 1=keep me   #
875   case ${retval} in   case ${retval} in
876   0)   0)
877   [[ ${VERBOSE} = on ]] && echo -e "\t<<< LINK: ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t<<< LINK: ${MROOT}${pathto}"
878   rm "${MROOT}${pathto}"   rm "${MROOT}${pathto}"
879   ;;   ;;
880    
881   1)   1)
882   [[ ${VERBOSE} = on ]] && \   mqueryfeature "verbose" && \
883   echo -e "${COLRED}! mtime${COLDEFAULT} === LINK: ${MROOT}${pathto}"   echo -e "${COLRED}! mtime${COLDEFAULT} === LINK: ${MROOT}${pathto}"
884   ;;   ;;
885   esac   esac
# Line 849  remove_files() Line 946  remove_files()
946    
947   if [ ! -e "${MROOT}${pathto}" ]   if [ ! -e "${MROOT}${pathto}" ]
948   then   then
949   [[ ${VERBOSE} = on ]] && \   mqueryfeature "verbose" && \
950   echo -e "${COLRED}! exist${COLDEFAULT} === FILE: ${MROOT}${pathto}"   echo -e "${COLRED}! exist${COLDEFAULT} === FILE: ${MROOT}${pathto}"
951   continue   continue
952   fi   fi
# Line 875  remove_files() Line 972  remove_files()
972   case ${retval} in   case ${retval} in
973   # file is not protected - delete it   # file is not protected - delete it
974   0|3)   0|3)
975   [[ ${VERBOSE} = on ]] && echo -e "\t<<< FILE: ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t<<< FILE: ${MROOT}${pathto}"
976   rm "${MROOT}${pathto}"   rm "${MROOT}${pathto}"
977   ;;   ;;
978    
979   # file is protected, do not delete   # file is protected, do not delete
980   2)   2)
981   if [[ ${VERBOSE} = on ]]   if mqueryfeature "verbose"
982   then   then
983   echo -en "${COLRED}"   echo -en "${COLRED}"
984   echo -n "! prot "   echo -n "! prot "
# Line 892  remove_files() Line 989  remove_files()
989    
990   # file is protected but ignored, delete the update/do nothing   # file is protected but ignored, delete the update/do nothing
991   4)   4)
992   if [[ ${VERBOSE} = on ]]   if mqueryfeature "verbose"
993   then   then
994   echo -en "${COLRED}"   echo -en "${COLRED}"
995   echo -n "! ignr "   echo -n "! ignr "
# Line 904  remove_files() Line 1001  remove_files()
1001   esac   esac
1002   ;;   ;;
1003   1)   1)
1004   [[ ${VERBOSE} = on ]] && \   mqueryfeature "verbose" && \
1005   echo -e "${COLRED}! mtime${COLDEFAULT} === FILE: ${MROOT}${pathto}"   echo -e "${COLRED}! mtime${COLDEFAULT} === FILE: ${MROOT}${pathto}"
1006   ;;   ;;
1007   esac   esac
# Line 966  remove_blockdevices() Line 1063  remove_blockdevices()
1063   do   do
1064   [ -z "${pathto}" ] && continue   [ -z "${pathto}" ] && continue
1065    
1066   [[ ${VERBOSE} = on ]] && echo -e "\t<<< PIPE: ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t<<< PIPE: ${MROOT}${pathto}"
1067   rm "${MROOT}${pathto}"   rm "${MROOT}${pathto}"
1068   done < ${MROOT}${INSTALLDB}/${pfull}/.pipes   done < ${MROOT}${INSTALLDB}/${pfull}/.pipes
1069    
# Line 1026  remove_characterdevices() Line 1123  remove_characterdevices()
1123   do   do
1124   [ -z "${pathto}" ] && continue   [ -z "${pathto}" ] && continue
1125    
1126   [[ ${VERBOSE} = on ]] && echo -e "\t<<< CHAR: ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t<<< CHAR: ${MROOT}${pathto}"
1127   rm "${MROOT}${pathto}"   rm "${MROOT}${pathto}"
1128   done < ${MROOT}${INSTALLDB}/${pfull}/.char   done < ${MROOT}${INSTALLDB}/${pfull}/.char
1129    
# Line 1077  remove_fifos() Line 1174  remove_fifos()
1174   # check needed global vars   # check needed global vars
1175   [ -z "${BUILDDIR}" ] && die "remove_fifos() \$BUILDDIR not set."   [ -z "${BUILDDIR}" ] && die "remove_fifos() \$BUILDDIR not set."
1176    
1177   [ ! -f ${MROOT}${INSTALLDB}/${pfull}/.fifo ] && die "remove_fifos() .fifo not found"   # make it optional atm !!
1178     #[ ! -f ${MROOT}${INSTALLDB}/${pfull}/.fifo ] && die "remove_fifos() .fifo not found"
1179     [ ! -f ${MROOT}${INSTALLDB}/${pfull}/.fifo ] && return
1180    
1181   # sets fieldseperator to "§" instead of " "   # sets fieldseperator to "§" instead of " "
1182   IFS=§   IFS=§
# Line 1086  remove_fifos() Line 1185  remove_fifos()
1185   do   do
1186   [ -z "${pathto}" ] && continue   [ -z "${pathto}" ] && continue
1187    
1188   [[ ${VERBOSE} = on ]] && echo -e "\t<<< FIFO: ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t<<< FIFO: ${MROOT}${pathto}"
1189   rm "${MROOT}${pathto}"   rm "${MROOT}${pathto}"
1190   done < ${MROOT}${INSTALLDB}/${pfull}/.fifo   done < ${MROOT}${INSTALLDB}/${pfull}/.fifo
1191    
# Line 1147  remove_directories() Line 1246  remove_directories()
1246    
1247   if [ ! -d "${MROOT}${pathto}" ]   if [ ! -d "${MROOT}${pathto}" ]
1248   then   then
1249   [[ ${VERBOSE} = on ]] && \   mqueryfeature "verbose" && \
1250   echo -e "${COLRED}! exist${COLDEFAULT} === DIR:  ${MROOT}${pathto}"   echo -e "${COLRED}! exist${COLDEFAULT} === DIR:  ${MROOT}${pathto}"
1251   continue   continue
1252   fi   fi
# Line 1155  remove_directories() Line 1254  remove_directories()
1254   # exclude .keep directories   # exclude .keep directories
1255   if [ -f "${MROOT}${pathto}/.keep" ]   if [ -f "${MROOT}${pathto}/.keep" ]
1256   then   then
1257   [[ ${VERBOSE} = on ]] && \   mqueryfeature "verbose" && \
1258   echo -e "${COLRED}! .keep${COLDEFAULT} === DIR:  ${MROOT}${pathto}"   echo -e "${COLRED}! .keep${COLDEFAULT} === DIR:  ${MROOT}${pathto}"
1259   continue   continue
1260   fi   fi
# Line 1168  remove_directories() Line 1267  remove_directories()
1267    
1268   if rmdir "${MROOT}${pathto}" &> /dev/null   if rmdir "${MROOT}${pathto}" &> /dev/null
1269   then   then
1270   [[ ${VERBOSE} = on ]] && echo -e "\t<<< DIR:  ${MROOT}${pathto}"   mqueryfeature "verbose" && echo -e "\t<<< DIR:  ${MROOT}${pathto}"
1271   else   else
1272   [[ ${VERBOSE} = on ]] && \   mqueryfeature "verbose" && \
1273   echo -e "${COLRED}! empty${COLDEFAULT} === DIR:  ${MROOT}${pathto}"   echo -e "${COLRED}! empty${COLDEFAULT} === DIR:  ${MROOT}${pathto}"
1274   fi   fi
1275   done   done
# Line 1228  build_douninstall() Line 1327  build_douninstall()
1327   done   done
1328  }  }
1329    
1330    # convertmirrors [uri]
1331    convertmirrors()
1332    {
1333     local uri="$1"
1334     local scheme
1335     local mirror
1336     local mirrors
1337     local addon
1338     local real_uri
1339     local output
1340    
1341     # needs
1342     [[ -z ${MIRRORS} ]] && die "convertmirrors(): no mirrors defined!"
1343     [[ -z ${SOURCEFORGE_MIRRORS} ]] && die "convertmirrors(): no sourceforge mirrors defined!"
1344     [[ -z ${GNU_MIRRORS} ]] && die "convertmirrors(): no gnu mirrors defined!"
1345     [[ -z ${GNOME_MIRRORS} ]] && die "convertmirrors(): no gnome mirrors defined!"
1346     [[ -z ${KDE_MIRRORS} ]] && die "convertmirrors(): no kde mirrors defined!"
1347    
1348     # check known uri schemes
1349     case ${uri} in
1350     http://*|https://*|ftp://*|ftps://*) mirrors="" ;;
1351     mirror://*) mirrors="${MIRRORS}"; scheme="mirror://"; addon="/sources" ;;
1352     package://*) mirrors="${MIRRORS}"; scheme="package://"; addon="/${PACKAGES_SERVER_PATH}" ;;
1353     gnu://*) mirrors="${GNU_MIRRORS}"; scheme="gnu://" ;;
1354     sourceforge://*) mirrors="${SOURCEFORGE_MIRRORS}"; scheme="sourceforge://" ;;
1355     gnome://*) mirrors="${GNOME_MIRRORS}"; scheme="gnome://" ;;
1356     kde://*) mirrors="${KDE_MIRRORS}"; scheme="kde://" ;;
1357     *) die "convertmirror(): unsupported uri scheme in '${uri}'!" ;;
1358     esac
1359    
1360     if [[ ! -z ${mirrors} ]]
1361     then
1362     for mirror in ${mirrors}
1363     do
1364     # add a whitespace to the output
1365     [[ -z ${output} ]] || output+=" "
1366     output+="${mirror}${addon}/${uri/${scheme}/}"
1367     done
1368     else
1369     output="${uri}"
1370     fi
1371    
1372     echo "${output}"
1373    }
1374    
1375    mdownload()
1376    {
1377     local i
1378     local uri
1379     local real_uris
1380     local mirror
1381     local outputfile
1382     local outputdir
1383     local retval
1384     local wget_opts
1385    
1386     # very basic getops
1387     for i in $*
1388     do
1389     case $1 in
1390     --uri|-u) shift; uri="$1" ;;
1391     --dir|-d) shift; outputdir="$1" ;;
1392     esac
1393     shift
1394     done
1395    
1396     # sanity checks; abort if not given
1397     [[ -z ${uri} ]] && die "mdownload(): no uri given!"
1398     [[ -z ${outputdir} ]] && die "mdownload(): no dir given!"
1399    
1400     # convert mirrored uris to the real ones
1401     real_uris="$(convertmirrors ${uri})"
1402    
1403     # verbose or not
1404     mqueryfeature "!verbose" && wget_opts+=" --quiet"
1405    
1406     # filter wget options if busybox was found
1407     wget_opts+=" $(busybox_filter_wget_options ${WGET_FETCH_OPTIONS})"
1408    
1409     # create outputdir
1410     [[ ! -d ${outputdir} ]] && install -d "${outputdir}"
1411    
1412     for mirror in ${real_uris}
1413     do
1414     # get the name of the output file
1415     outputfile="${mirror##*/}"
1416    
1417     wget ${wget_opts} --output-document="${outputdir}/${outputfile}" "${mirror}"
1418     retval="$?"
1419     if [[ ${retval} = 0 ]]
1420     then
1421     break
1422     else
1423     continue
1424     fi
1425     done
1426    
1427     # return wget retval
1428     return "${retval}"
1429    }
1430    
1431  # fetch_packages /path/to/mage/file1 /path/to/mage/file2  # fetch_packages /path/to/mage/file1 /path/to/mage/file2
1432  fetch_packages()  fetch_packages()
1433  {  {
1434     local i
1435   local list="$@"   local list="$@"
1436   local pkg   local pkg
1437   local mirr   local mirr
# Line 1239  fetch_packages() Line 1440  fetch_packages()
1440   local opt   local opt
1441   local count_current   local count_current
1442   local count_total   local count_total
1443     local wget_opts
1444    
1445   [ -z "${MIRRORS}" ] && die "You have no mirrors defined. Please edit your ${MAGERC}."   [ -z "${MIRRORS}" ] && die "You have no mirrors defined. Please edit your ${MAGERC}."
1446    
1447     # filter wget command if busybox was found
1448     wget_opts="$(busybox_filter_wget_options ${WGET_FETCH_OPTIONS})"
1449    
1450   # get count of total packages   # get count of total packages
1451   declare -i count_current=0   declare -i count_current=0
1452   declare -i count_total=0   declare -i count_total=0
# Line 1280  fetch_packages() Line 1485  fetch_packages()
1485   continue   continue
1486   fi   fi
1487    
1488   for mirr in ${MIRRORS}   echo -ne " ${COLBLUE}***${COLDEFAULT}"
1489   do   echo -e " fetching (${count_current}/${count_total}): ${pkg} ... "
1490   echo -ne " ${COLBLUE}***${COLDEFAULT}"   mdownload --uri "package://${pkg}" --dir "${PKGDIR}" || die "Could not download ${pkg}"
  #echo -e " fetching (${count_current}/${count_total}): ${mirr}/${pkg} ... "  
  echo -e " fetching (${count_current}/${count_total}): ${pkg} ... "  
  [[ ${VERBOSE} = off ]] && opt="--quiet"  
  wget \  
  ${WGET_FETCH_OPTIONS} \  
  --directory-prefix=${PKGDIR} \  
  ${opt} ${mirr}/${PACKAGES_SERVER_PATH}/${pkg}  
  if [[ $? = 0 ]]  
  then  
  break  
  else  
  continue  
  fi  
  done  
   
1491   if [ ! -f ${PKGDIR}/${pkg} ]   if [ ! -f ${PKGDIR}/${pkg} ]
1492   then   then
1493   die "Could not download ${pkg}"   die "Package '${pkg}' after download not found in '${PKGDIR}'"
1494   fi   fi
1495   done   done
1496    
# Line 1328  syncmage() Line 1518  syncmage()
1518   done   done
1519    
1520   # clean up backup files (foo~)   # clean up backup files (foo~)
1521   find ${MAGEDIR} -name *~ -exec rm '{}' ';'   find ${MAGEDIR} -name \*~ -exec rm '{}' ';'
1522    
1523   # check if a newer mage version is available   # check if a newer mage version is available
1524   is_newer_mage_version_available   is_newer_mage_version_available
# Line 1341  syncmage_tarball() Line 1531  syncmage_tarball()
1531   local temp="$(mktemp -d)"   local temp="$(mktemp -d)"
1532   local mirr mymirr   local mirr mymirr
1533   local opt   local opt
1534     local tar_opts
1535     local wget_opts
1536    
1537   # try to get the md5 marked as latest on the server   # try to get the md5 marked as latest on the server
1538   latest_md5="mage-latest.md5"   latest_md5="mage-latest.md5"
# Line 1348  syncmage_tarball() Line 1540  syncmage_tarball()
1540   # try to get the tarball marked as latest on the server   # try to get the tarball marked as latest on the server
1541   latest_tarball="mage-latest.tar.bz2"   latest_tarball="mage-latest.tar.bz2"
1542    
1543     # filter wget command if busybox was found
1544     wget_opts="$(busybox_filter_wget_options ${WGET_FETCH_OPTIONS})"
1545    
1546   for mirr in ${MIRRORS}   for mirr in ${MIRRORS}
1547   do   do
1548   # path without distribution   # path without distribution
# Line 1355  syncmage_tarball() Line 1550  syncmage_tarball()
1550    
1551   echo -ne "${COLBLUE} --- ${COLDEFAULT}"   echo -ne "${COLBLUE} --- ${COLDEFAULT}"
1552   echo "fetching latest md5 from ${mymirr} ..."   echo "fetching latest md5 from ${mymirr} ..."
1553   [[ ${VERBOSE} = off ]] && opt="--quiet"   mqueryfeature "!verbose" && opt="--quiet"
1554   wget \   wget \
1555   ${WGET_FETCH_OPTIONS} \   ${wget_opts} \
1556   --directory-prefix=${temp} \   --directory-prefix=${temp} \
1557   ${opt} ${mymirr}/rsync/tarballs/${latest_md5}   ${opt} ${mymirr}/rsync/tarballs/${latest_md5}
1558    
1559   echo -ne "${COLBLUE} --- ${COLDEFAULT}"   echo -ne "${COLBLUE} --- ${COLDEFAULT}"
1560   echo "fetching latest tarball from ${mymirr} ..."   echo "fetching latest tarball from ${mymirr} ..."
1561   wget \   wget \
1562   ${WGET_FETCH_OPTIONS} \   ${wget_opts} \
1563   --directory-prefix=${temp} \   --directory-prefix=${temp} \
1564   ${opt} ${mymirr}/rsync/tarballs/${latest_tarball}   ${opt} ${mymirr}/rsync/tarballs/${latest_tarball}
1565   if [[ $? = 0 ]]   if [[ $? = 0 ]]
# Line 1384  syncmage_tarball() Line 1579  syncmage_tarball()
1579   else   else
1580   echo -ne "${COLBLUE} --- ${COLDEFAULT}"   echo -ne "${COLBLUE} --- ${COLDEFAULT}"
1581   echo -n "checking md5sum... "   echo -n "checking md5sum... "
1582   ( cd ${temp}; md5sum --check ${latest_md5} ) || die "md5 for ${latest_tarball} failed"   ( cd ${temp}; md5sum -c ${latest_md5} ) || die "md5 for ${latest_tarball} failed"
1583   fi   fi
1584    
1585   if [[ -d ${MAGEDIR} ]]   if [[ -d ${MAGEDIR} ]]
# Line 1394  syncmage_tarball() Line 1589  syncmage_tarball()
1589   rm -rf ${MAGEDIR}   rm -rf ${MAGEDIR}
1590   fi   fi
1591    
1592     if need_busybox_support tar
1593     then
1594     tar_opts="xjf"
1595     else
1596     tar_opts="xjmf"
1597     fi
1598    
1599   echo -ne "${COLBLUE} --- ${COLDEFAULT}"   echo -ne "${COLBLUE} --- ${COLDEFAULT}"
1600   echo "updating mage-tree from tarball ..."   echo "updating mage-tree from tarball ..."
1601   # unpack in dirname of MAGEDIR, as the tarball has already the mage   # unpack in dirname of MAGEDIR, as the tarball has already the mage
1602   tar xjmf ${temp}/${latest_tarball} -C ${MAGEDIR%/*} || die "Unpacking tarball"   tar ${tar_opts} ${temp}/${latest_tarball} -C ${MAGEDIR%/*} || die "Unpacking tarball"
1603    
1604   if [[ -d ${temp} ]]   if [[ -d ${temp} ]]
1605   then   then
# Line 1443  xtitleclean() Line 1645  xtitleclean()
1645  }  }
1646    
1647    
1648  # cuts full pathnames or versioniezed names down to basename  # cuts full pathnames or versionized names down to basename
1649  choppkgname()  choppkgname()
1650  {  {
1651   #we want this only if full name was used   #we want this only if full name was used
# Line 1526  get_highest_magefile() Line 1728  get_highest_magefile()
1728   local magefile   local magefile
1729    
1730   # do not list the content of a directory, only the name (-d)   # do not list the content of a directory, only the name (-d)
1731   for magefile in $(ls --format=single-column -v -d ${MAGEDIR}/${PCAT}/${PNAME}/*)   for magefile in $(ls --format=single-column -v -d ${MAGEDIR}/${PCAT}/${PNAME}/* 2> /dev/null)
1732   do   do
1733   [[ -z ${magefile} ]] && continue   [[ -z ${magefile} ]] && continue
1734   # we exclude subdirs (for stuff like a md5sum dir)   # we exclude subdirs (for stuff like a md5sum dir)
# Line 1535  get_highest_magefile() Line 1737  get_highest_magefile()
1737   then   then
1738   HIGHEST_MAGEFILE=${magefile}   HIGHEST_MAGEFILE=${magefile}
1739   #for debug only   #for debug only
1740   [[ ${MAGEDEBUG} = on ]] && echo "HIGHEST_MAGEFILE=${HIGHEST_MAGEFILE}"   mqueryfeature "debug" && echo "HIGHEST_MAGEFILE=${HIGHEST_MAGEFILE}"
1741   fi   fi
1742   done   done
1743    
 # do not so anything  
 # # stop here if HIGHEST_MAGEFILE is zero  
 # # this package must be unstable or old  
 # if [ -z "${HIGHEST_MAGEFILE}" ]  
 # then  
 # echo  
 # echo -n "All packages named "  
 # echo -en ${COLRED}\""${PKGNAME%-*-*-*}\""${COLDEFAULT}  
 # echo -n " are marked "  
 # echo -en ${COLRED}"*UNSTABLE*"${COLDEFAULT}  
 # echo "."  
 # echo "You need to declare USE_UNSTABLE=true to install this."  
 # echo  
 # echo "Example:"  
 # echo "         USE_UNSTABLE=true mage install ${PKGNAME%-*-*-*}"  
 # echo  
 # echo "Be warned that these packages are not stable and may cause serious problems."  
 # echo "You should know what you are doing, so don't complain about any damage."  
 # echo  
 # return 1  
 # fi  
   
1744   echo "${HIGHEST_MAGEFILE}"   echo "${HIGHEST_MAGEFILE}"
1745   return 0   return 0
1746  }  }
# Line 1965  minclude() Line 2145  minclude()
2145   then   then
2146   for i in $*   for i in $*
2147   do   do
2148   [[ ${MAGEDEBUG} = on ]] && \   mqueryfeature "debug" && \
2149   echo "--- Including ${MAGEDIR}/include/${i}.minc"   echo "--- Including ${MAGEDIR}/include/${i}.minc"
2150   source ${MAGEDIR}/include/${i}.minc   source ${MAGEDIR}/include/${i}.minc
2151   done   done
2152   [[ ${MAGEDEBUG} = on ]] && echo   mqueryfeature "debug" && echo
2153   fi   fi
2154  }  }
2155    
# Line 2413  mage_install() Line 2593  mage_install()
2593   if [[ -n ${MAGE_TARGETS} ]]   if [[ -n ${MAGE_TARGETS} ]]
2594   then   then
2595   # basic svn compat   # basic svn compat
2596   if [[ -d ${SMAGESCRIPTSDIR}/trunk ]]   if [[ -d ${SMAGESCRIPTSDIR}/.svn ]]
2597   then   then
2598   for i in ${SMAGESCRIPTSDIR}/trunk/*/${pname/${MAGE_TARGETS}/}/${pname/${MAGE_TARGETS}/}-${pver}-${pbuild}.smage2   for i in ${SMAGESCRIPTSDIR}/*/${pname/${MAGE_TARGETS}/}/${pname/${MAGE_TARGETS}/}-${pver}-${pbuild}.smage2
2599   do   do
2600   smage2file="${i}"   smage2file="${i}"
2601   done   done
# Line 2426  mage_install() Line 2606  mage_install()
2606   elif [[ -n ${SPLIT_PACKAGE_BASE} ]]   elif [[ -n ${SPLIT_PACKAGE_BASE} ]]
2607   then   then
2608   # basic svn compat   # basic svn compat
2609   if [[ -d ${SMAGESCRIPTSDIR}/trunk ]]   if [[ -d ${SMAGESCRIPTSDIR}/.svn ]]
2610   then   then
2611   for i in ${SMAGESCRIPTSDIR}/trunk/*/${SPLIT_PACKAGE_BASE}/${SPLIT_PACKAGE_BASE}-${pver}-${pbuild}.smage2   for i in ${SMAGESCRIPTSDIR}/*/${SPLIT_PACKAGE_BASE}/${SPLIT_PACKAGE_BASE}-${pver}-${pbuild}.smage2
2612   do   do
2613   smage2file="${i}"   smage2file="${i}"
2614   done   done
# Line 2438  mage_install() Line 2618  mage_install()
2618    
2619   else   else
2620   # basic svn compat   # basic svn compat
2621   if [[ -d ${SMAGESCRIPTSDIR}/trunk ]]   if [[ -d ${SMAGESCRIPTSDIR}/.svn ]]
2622   then   then
2623   for i in ${SMAGESCRIPTSDIR}/trunk/*/${pname}/${pname}-${pver}-${pbuild}.smage2   for i in ${SMAGESCRIPTSDIR}/*/${pname}/${pname}-${pver}-${pbuild}.smage2
2624   do   do
2625   smage2file="${i}"   smage2file="${i}"
2626   done   done
# Line 2579  md5sum_packages() Line 2759  md5sum_packages()
2759   then   then
2760   echo -ne "${COLBLUE} *** ${COLDEFAULT}"   echo -ne "${COLBLUE} *** ${COLDEFAULT}"
2761   echo -ne "checking md5sum (${count_current}/${count_total}): "   echo -ne "checking md5sum (${count_current}/${count_total}): "
2762   ( cd ${PKGDIR}; md5sum --check ${md5file}) || die "md5 for ${pkgfile} failed"   ( cd ${PKGDIR}; md5sum -c ${md5file}) || die "md5 for ${pkgfile} failed"
2763   else   else
2764   echo -ne "${COLBLUE} --- ${COLDEFAULT}"   echo -ne "${COLBLUE} --- ${COLDEFAULT}"
2765   echo -e "!! no md5sum file found for ${pkgfile} :("   echo -e "!! no md5sum file found for ${pkgfile} :("
# Line 2819  pkgsearch() Line 2999  pkgsearch()
2999   local state   local state
3000   local descriptiom   local descriptiom
3001   local homepage   local homepage
3002     local license
3003   local i   local i
3004   local all_installed   local all_installed
3005   local ipver   local ipver
# Line 2855  pkgsearch() Line 3036  pkgsearch()
3036   state="$(get_value_from_magefile STATE ${magefile})"   state="$(get_value_from_magefile STATE ${magefile})"
3037   description="$(get_value_from_magefile DESCRIPTION ${magefile})"   description="$(get_value_from_magefile DESCRIPTION ${magefile})"
3038   homepage="$(get_value_from_magefile HOMEPAGE ${magefile})"   homepage="$(get_value_from_magefile HOMEPAGE ${magefile})"
3039     license="$(get_value_from_magefile LICENSE ${magefile})"
3040    
3041   # all installed   # all installed
3042   for i in $(get_uninstall_candidates --pname ${pname} --pcat ${pcat})   for i in $(get_uninstall_candidates --pname ${pname} --pcat ${pcat})
3043   do   do
3044   ipver="$(magename2pver ${i})"   ipver="$(magename2pver ${i})"
3045   ipbuild="$(magename2pbuild ${i})"   ipbuild="$(magename2pbuild ${i})"
3046    
3047   if [[ -z ${all_installed} ]]   if [[ -z ${all_installed} ]]
3048   then   then
3049   all_installed="${ipver}-${ipbuild}"   all_installed="${ipver}-${ipbuild}"
# Line 2870  pkgsearch() Line 3052  pkgsearch()
3052   fi   fi
3053   done   done
3054   [[ -z ${all_installed} ]] && all_installed="none"   [[ -z ${all_installed} ]] && all_installed="none"
3055    
3056   case ${state} in   case ${state} in
3057   stable) state=${COLGREEN}"[s] ";;   stable) state=${COLGREEN}"[s] ";;
3058   testing) state=${COLYELLOW}"[t] ";;   testing) state=${COLYELLOW}"[t] ";;
# Line 2915  EOF Line 3097  EOF
3097   echo "      Installed versions: ${all_installed}"   echo "      Installed versions: ${all_installed}"
3098   echo "      Description: ${description}"   echo "      Description: ${description}"
3099   echo "      Homepage: ${homepage}"   echo "      Homepage: ${homepage}"
3100     if [[ ! -z ${license} ]]
3101     then
3102     echo "      License:  ${license}"
3103     fi
3104   echo "      Depends: ${deps}"   echo "      Depends: ${deps}"
3105   echo "      SDepends: ${sdeps}"   echo "      SDepends: ${sdeps}"
3106   echo   echo
# Line 2955  export_inherits() Line 3141  export_inherits()
3141   eval "${functions}() { ${include}_${functions} ; }"   eval "${functions}() { ${include}_${functions} ; }"
3142    
3143   # debug   # debug
3144   [[ ${MAGEDEBUG} = on ]] && typeset -f "${functions}"   mqueryfeature "debug" && typeset -f "${functions}"
3145    
3146   shift   shift
3147   done   done
# Line 3027  EOF Line 3213  EOF
3213   return 0   return 0
3214  }  }
3215    
3216    # need_busybox_support ${cmd}
3217    # return 0 (no error = needs busybox support) or return 1 (error = no busybox support required)
3218    need_busybox_support()
3219    {
3220     local cmd
3221     cmd="$1"
3222    
3223     if [[ -x /bin/busybox ]]
3224     then
3225     if [[ $(readlink $(which ${cmd})) = /bin/busybox ]]
3226     then
3227     # needs busybox support
3228     return 0
3229     fi
3230     fi
3231    
3232     # no busybox
3233     return 1
3234    }
3235    
3236    # busybox_filter_wget_options ${wget_opts}
3237    busybox_filter_wget_options()
3238    {
3239     local opts="$@"
3240     local i
3241     local fixed_opts
3242    
3243     if need_busybox_support wget
3244     then
3245     for i in ${opts}
3246     do
3247     # show only the allowed ones
3248     case ${i} in
3249     -c|--continue) fixed_opts+=" -c" ;;
3250     -s|--spider) fixed_opts+=" -s" ;;
3251     -q|--quiet) fixed_opts+=" -q" ;;
3252     -O|--output-document) shift; fixed_opts+=" -O $1" ;;
3253     --header) shift; fixed_opts+=" --header $1" ;;
3254     -Y|--proxy) shift; fixed_opts+=" -Y $1" ;;
3255     -P) shift; fixed_opts+=" -P $1" ;;
3256     --no-check-certificate) fixed_opts+=" --no-check-certificate ${i}" ;;
3257     -U|--user-agent) shift; fixed_opts+=" -U ${i}" ;;
3258     # simply drop all other opts
3259     *) continue ;;
3260     esac
3261     done
3262    
3263     echo "${fixed_opts}"
3264     else
3265     echo "${opts}"
3266     fi
3267    }
3268    
3269    have_root_privileges()
3270    {
3271     local retval
3272    
3273     if [[ $(id -u) = 0 ]]
3274     then
3275     retval=0
3276     else
3277     retval=1
3278     fi
3279    
3280     return ${retval}
3281    }
3282    
3283    known_mage_feature()
3284    {
3285     local feature="$1"
3286     local retval
3287    
3288     case "${feature}" in
3289     autosvc|!autosvc) retval=0 ;;
3290     buildlog|!buildlog) retval=0 ;;
3291     ccache|!ccache) retval=0 ;;
3292     check|!check) retval=0 ;;
3293     compressdoc|!compressdoc) retval=0 ;;
3294     debug|!debug) retval=0 ;;
3295     distcc|!distcc) retval=0 ;;
3296     kernelsrcunpack|!kernelsrcunpack) retval=0 ;;
3297     libtool|!libtool) retval=0 ;;
3298     linuxsymlink|!linuxsymlink) retval=0 ;;
3299     pkgbuild|!pkgbuild) retval=0 ;;
3300     purge|!purge) retval=0 ;;
3301     qalint|!qalint) retval=0 ;;
3302     regentree|!regentree) retval=0 ;;
3303     resume|!resume) retval=0 ;;
3304     srcpkgbuild|!srcpkgbuild) retval=0 ;;
3305     srcpkgtarball|!srcpkgtarball) retval=0 ;;
3306     static|!static) retval=0 ;;
3307     stepbystep|!stepbystep) retval=0 ;;
3308     strip|!strip) retval=0 ;;
3309     verbose|!verbose) retval=0 ;;
3310     *) retval=1 ;;
3311     esac
3312    
3313     return "${retval}"
3314    }
3315    
3316    load_mage_features()
3317    {
3318     for i in ${MAGE_FEATURES_GLOBAL[*]} ${MAGE_FEATURES[*]}
3319     do
3320     FVERBOSE=off msetfeature ${i}
3321     done
3322    }
3323    
3324    msetfeature()
3325    {
3326     local feature
3327     local count
3328     local i
3329     local found
3330    
3331     for feature in $@
3332     do
3333     found=0
3334     count="${#MAGE_FEATURES_CURRENT[*]}"
3335    
3336     if ! known_mage_feature "${feature}"
3337     then
3338     [[ ${FVERBOSE} = off ]] || echo -e "${COLRED}Unknown feature '${feature}', ignoring it${COLDEFAULT}"
3339     return 3
3340     fi
3341    
3342     for ((i=0; i<count; i++))
3343     do
3344     if [[ ${MAGE_FEATURES_CURRENT[${i}]} = ${feature} ]]
3345     then
3346     [[ ${FVERBOSE} = off ]] || echo -e "${COLBLUE}---${COLGREEN} Feature '${feature}' already enabled${COLDEFAULT}"
3347     MAGE_FEATURES_CURRENT[${i}]="${feature}"
3348     found=1
3349     elif [[ ${MAGE_FEATURES_CURRENT[${i}]} = !${feature} ]]
3350     then
3351     [[ ${FVERBOSE} = off ]] || echo -e "${COLBLUE}---${COLGREEN} Feature '${feature}' currently disabled, enabling it!${COLDEFAULT}"
3352     MAGE_FEATURES_CURRENT[${i}]="${feature}"
3353     found=1
3354     elif [[ ${MAGE_FEATURES_CURRENT[${i}]} = ${feature//!} ]]
3355     then
3356     [[ ${FVERBOSE} = off ]] || echo -e "${COLBLUE}---${COLGREEN} Feature '${feature//!}' currently enabled, disabling it!${COLDEFAULT}"
3357     MAGE_FEATURES_CURRENT[${i}]="${feature}"
3358     found=1
3359     fi
3360     done
3361    
3362     # if the feature was not found after proccessing the whole array
3363     # it was not declared. in this case enable it
3364     if [[ ${found} = 0 ]]
3365     then
3366     [[ ${FVERBOSE} = off ]] || echo -e "${COLBLUE}---${COLGREEN} Feature '${feature}' was not declared, enabling it!${COLDEFAULT}"
3367     MAGE_FEATURES_CURRENT=( ${MAGE_FEATURES_CURRENT[*]} "${feature}" )
3368     fi
3369    
3370     export MAGE_FEATURE_CURRENT
3371     done
3372    }
3373    
3374    mqueryfeature()
3375    {
3376     local feature="$1"
3377     local retval=1
3378     local i
3379    
3380     if known_mage_feature "${feature}"
3381     then
3382     for i in ${MAGE_FEATURES_CURRENT[*]}
3383     do
3384     if [[ ${i} = ${feature} ]]
3385     then
3386     retval=0
3387     break # found break here
3388     fi
3389     done
3390     else
3391     [[ ${FVERBOSE} = off ]] || echo -e "${COLRED}Unknown feature '${feature}', ignoring it${COLDEFAULT}"
3392     retval=3
3393     fi
3394    
3395     return ${retval}
3396    }
3397    
3398    mprintfeatures()
3399    {
3400     echo "Global features:  ${MAGE_FEATURES_GLOBAL[*]}"
3401     echo "Local features:   ${MAGE_FEATURES[*]}"
3402     echo "Current features: ${MAGE_FEATURES_CURRENT[*]}"
3403    }

Legend:
Removed from v.1209  
changed lines
  Added in v.1648