|
@@ -0,0 +1,250 @@
|
|
|
+#!/bin/bash
|
|
|
+
|
|
|
+OFFSET=30
|
|
|
+image=/mnt/backup.img
|
|
|
+type=83
|
|
|
+next=$OFFSET
|
|
|
+rootpart=1
|
|
|
+ROOTFS_TYPE=ext4
|
|
|
+bootfs=
|
|
|
+logfile="/var/log/backup.log"
|
|
|
+# script configuration
|
|
|
+CWD="/usr/lib/nand-sata-install"
|
|
|
+EX_LIST="${CWD}/exclude.txt"
|
|
|
+[[ -f /usr/lib/u-boot/platform_install.sh ]] && source /usr/lib/u-boot/platform_install.sh
|
|
|
+[[ -f /etc/orangepi-release ]] && source /etc/orangepi-release
|
|
|
+
|
|
|
+if [[ $EUID -ne 0 ]]; then
|
|
|
+ echo 'This tool must run as root. Exiting ...' >&2
|
|
|
+ exit 14
|
|
|
+fi
|
|
|
+
|
|
|
+[[ -f "$image" ]] && rm "$image"
|
|
|
+
|
|
|
+# define makefs and mount options
|
|
|
+declare -A mkopts mountopts
|
|
|
+# for ARMv7 remove 64bit feature from default mke2fs format features
|
|
|
+if [[ $LINUXFAMILY =~ sun50iw6|sun50iw2|sun50iw1 && $BRANCH == legacy ]]; then
|
|
|
+ mkopts[ext2]='-O ^64bit -qF'
|
|
|
+ mkopts[ext3]='-O ^64bit -qF'
|
|
|
+ mkopts[ext4]='-O ^64bit -qF'
|
|
|
+else
|
|
|
+ mkopts[ext2]='-qF'
|
|
|
+ mkopts[ext3]='-qF'
|
|
|
+ mkopts[ext4]='-qF'
|
|
|
+fi
|
|
|
+mkopts[btrfs]='-f'
|
|
|
+mkopts[f2fs]='-f'
|
|
|
+
|
|
|
+mountopts[ext2]='defaults,noatime,commit=600,errors=remount-ro,x-gvfs-hide 0 1'
|
|
|
+mountopts[ext3]='defaults,noatime,commit=600,errors=remount-ro,x-gvfs-hide 0 1'
|
|
|
+mountopts[ext4]='defaults,noatime,commit=600,errors=remount-ro,x-gvfs-hide 0 1'
|
|
|
+mountopts[btrfs]='defaults,noatime,commit=600,compress=lzo,x-gvfs-hide 0 2'
|
|
|
+mountopts[f2fs]='defaults,noatime,x-gvfs-hide 0 2'
|
|
|
+
|
|
|
+
|
|
|
+function display_alert()
|
|
|
+{
|
|
|
+
|
|
|
+ local tmp=""
|
|
|
+ [[ -n $2 ]] && tmp="[\e[0;33m $2 \x1B[0m]"
|
|
|
+
|
|
|
+ case $3 in
|
|
|
+ err)
|
|
|
+ echo -e "[\e[0;31m error \x1B[0m] $1 $tmp"
|
|
|
+ ;;
|
|
|
+
|
|
|
+ wrn)
|
|
|
+ echo -e "[\e[0;35m warn \x1B[0m] $1 $tmp"
|
|
|
+ ;;
|
|
|
+
|
|
|
+ ext)
|
|
|
+ echo -e "[\e[0;32m o.k. \x1B[0m] \e[1;32m$1\x1B[0m $tmp"
|
|
|
+ ;;
|
|
|
+
|
|
|
+ info)
|
|
|
+ echo -e "[\e[0;32m o.k. \x1B[0m] $1 $tmp"
|
|
|
+ ;;
|
|
|
+
|
|
|
+ *)
|
|
|
+ echo -e "[\e[0;32m .... \x1B[0m] $1 $tmp"
|
|
|
+ ;;
|
|
|
+ esac
|
|
|
+}
|
|
|
+
|
|
|
+stop_running_services()
|
|
|
+{
|
|
|
+ systemctl --state=running | awk -F" " '/.service/ {print $1}' | sort -r | \
|
|
|
+ grep -E -e "$1" | while read ; do
|
|
|
+ echo -e "\nStopping ${REPLY} \c"
|
|
|
+ systemctl stop ${REPLY} 2>&1
|
|
|
+ done
|
|
|
+}
|
|
|
+
|
|
|
+display_alert "Starting backup to image" "$image" "info"
|
|
|
+
|
|
|
+rootfs_size=$(df -BM | grep ^/dev | head -1 | awk '{print $3}' | tr -cd '[0-9]. \n')
|
|
|
+display_alert "Current rootfs size" "$rootfs_size MiB" "info"
|
|
|
+
|
|
|
+imagesize=$(($rootfs_size + $OFFSET))
|
|
|
+sdsize=$(bc -l <<< "scale=0; ((($imagesize * 1.35) / 1 + 0) / 4 + 1) * 4")
|
|
|
+
|
|
|
+mnt_free=$(df -BM /mnt | grep ^/dev | head -1 | awk '{print $4}' | tr -cd '[0-9]. \n')
|
|
|
+
|
|
|
+if [ "$mnt_free" -lt "$sdsize" ]; then
|
|
|
+ display_alert "Not enough space in /mnt" "Required: ${sdsize}MiB, Available: ${mnt_free}MiB" "err"
|
|
|
+ exit 1
|
|
|
+fi
|
|
|
+
|
|
|
+cleanup() {
|
|
|
+ if [[ -d "${TempDir}" ]] && findmnt -r "${TempDir}/rootfs" > /dev/null; then
|
|
|
+ display_alert "Unmounting" "${TempDir}/rootfs" "info"
|
|
|
+ umount "${TempDir}"/bootfs
|
|
|
+ umount "${TempDir}"/rootfs
|
|
|
+ rm -r ${TempDir}
|
|
|
+ losetup -d $LOOP
|
|
|
+ fi
|
|
|
+
|
|
|
+ [[ -d "${TempDir}" ]] && rm "${TempDir}"
|
|
|
+
|
|
|
+ exit
|
|
|
+}
|
|
|
+
|
|
|
+trap cleanup EXIT INT
|
|
|
+
|
|
|
+display_alert "Creating blank image for rootfs" "$sdsize MiB" "info"
|
|
|
+dd if=/dev/zero bs=1M status=none count=$sdsize | pv -p -b -r -s $(($sdsize * 1024 * 1024)) -N "[ .... ] dd" | dd status=none of=$image
|
|
|
+
|
|
|
+display_alert "Creating partitions" "${bootfs:+/boot: $bootfs }root: $ROOTFS_TYPE" "info"
|
|
|
+{
|
|
|
+ echo "$rootpart : name=\"rootfs\", start=${next}MiB, type=${type}"
|
|
|
+} | sfdisk $image >> "$logfile" 2>&1
|
|
|
+
|
|
|
+LOOP=$(losetup -f)
|
|
|
+[[ -z $LOOP ]] && echo "error" && exit
|
|
|
+
|
|
|
+losetup $LOOP $image
|
|
|
+
|
|
|
+partprobe $LOOP
|
|
|
+
|
|
|
+rootdevice="${LOOP}p${rootpart}"
|
|
|
+
|
|
|
+display_alert "Creating rootfs" "$ROOTFS_TYPE on $rootdevice"
|
|
|
+
|
|
|
+mkfs.ext4 -q -m 2 -O '^64bit,^metadata_csum' -L opi_root $rootdevice
|
|
|
+
|
|
|
+[[ $ROOTFS_TYPE == ext4 ]] && tune2fs -o journal_data_writeback $rootdevice > /dev/null
|
|
|
+
|
|
|
+TempDir=$(mktemp -d /mnt/${0##*/}.XXXXXX || exit 2)
|
|
|
+
|
|
|
+sync && mkdir -p "${TempDir}"/bootfs "${TempDir}"/rootfs
|
|
|
+
|
|
|
+( mount -o compress-force=zlib "${rootdevice}" "${TempDir}"/rootfs 2> /dev/null || mount "${rootdevice}" "${TempDir}"/rootfs )
|
|
|
+mount $rootdevice "${TempDir}"/bootfs
|
|
|
+
|
|
|
+# stop running services
|
|
|
+echo -e "\nFiles currently open for writing:" >> $logfile
|
|
|
+lsof / | awk 'NR==1 || $4~/[0-9][uw]/' | grep -v "^COMMAND" >> $logfile
|
|
|
+echo -e "\nTrying to stop running services to minimize open files:\c" >> $logfile
|
|
|
+stop_running_services "nfs-|smbd|nmbd|winbind|ftpd|netatalk|monit|cron|webmin|rrdcached" >> $logfile
|
|
|
+stop_running_services "fail2ban|ramlog|folder2ram|postgres|mariadb|mysql|postfix|mail|nginx|apache|snmpd" >> $logfile
|
|
|
+pkill dhclient 2>/dev/null
|
|
|
+LANG=C echo -e "\n\nChecking again for open files:" >> $logfile
|
|
|
+lsof / | awk 'NR==1 || $4~/[0-9][uw]/' | grep -v "^COMMAND" >> $logfile
|
|
|
+
|
|
|
+TODO=$(rsync -ahvrltDn --delete --stats --exclude-from=$EX_LIST / "${TempDir}"/rootfs | grep "Number of files:"|awk '{print $4}' | tr -d '.,')
|
|
|
+echo -e "\nCopying ${TODO} files to $rootdevice. \c" >> $logfile
|
|
|
+#display_alert "Copying ${TODO} files to" "/"
|
|
|
+
|
|
|
+# creating rootfs
|
|
|
+# Speed copy increased x10
|
|
|
+ # Variables for interfacing with rsync progress
|
|
|
+nsi_conn_path="${TempDir}/nand-sata-install"
|
|
|
+nsi_conn_done="${nsi_conn_path}/done"
|
|
|
+nsi_conn_progress="${nsi_conn_path}/progress"
|
|
|
+mkdir -p "${nsi_conn_path}"
|
|
|
+echo 0 >"${nsi_conn_progress}"
|
|
|
+echo no >"${nsi_conn_done}"
|
|
|
+
|
|
|
+ # Launch rsync in background
|
|
|
+{ \
|
|
|
+rsync -avrltD --delete --exclude-from=$EX_LIST / "${TempDir}"/rootfs | \
|
|
|
+nl | awk '{ printf "%.0f\n", 100*$1/"'"$TODO"'" }' \
|
|
|
+> "${nsi_conn_progress}" ;
|
|
|
+ # save exit code from rsync
|
|
|
+echo ${PIPESTATUS[0]} >"${nsi_conn_done}"
|
|
|
+} &
|
|
|
+
|
|
|
+ # while variables
|
|
|
+rsync_copy_finish=0
|
|
|
+rsync_progress=0
|
|
|
+prev_progress=0
|
|
|
+rsync_done=""
|
|
|
+while [ "${rsync_copy_finish}" -eq 0 ]; do
|
|
|
+ # Sometimes reads the progress file while writing and only partial numbers (like 1 when is 15)
|
|
|
+ prev_progress=${rsync_progress}
|
|
|
+ rsync_progress=$(tail -n1 "${nsi_conn_progress}")
|
|
|
+ if [[ -z ${rsync_progress} ]]; then
|
|
|
+ rsync_progress=${prev_progress}
|
|
|
+ fi
|
|
|
+ if [ ${prev_progress} -gt ${rsync_progress} ]; then
|
|
|
+ rsync_progress=${prev_progress}
|
|
|
+ fi
|
|
|
+
|
|
|
+ echo "${rsync_progress}"
|
|
|
+ # finish the while if the rsync is finished
|
|
|
+ rsync_done=$(cat ${nsi_conn_done})
|
|
|
+ if [[ "${rsync_done}" != "no" ]]; then
|
|
|
+ if [[ ${rsync_done} -eq 0 ]]; then
|
|
|
+ rm -rf "${nsi_conn_path}"
|
|
|
+ rsync_copy_finish=1
|
|
|
+ else
|
|
|
+ # if rsync return error
|
|
|
+ echo "Error: could not copy rootfs files, exiting"
|
|
|
+ exit 4
|
|
|
+ fi
|
|
|
+ else
|
|
|
+ sleep 0.5
|
|
|
+ fi
|
|
|
+
|
|
|
+done | \
|
|
|
+while read i; do
|
|
|
+ width=30
|
|
|
+ num_hash=$((i * width / 100))
|
|
|
+ bar=$(printf "%-${width}s" "#" | tr ' ' '#')
|
|
|
+ printf "\r[\e[0;32m .... \x1B[0m] Copying "$TODO" file to [\e[0;33m / \x1B[0m] [%-${width}s] %3d%%" "${bar:0:num_hash}" "$i"
|
|
|
+done
|
|
|
+echo ""
|
|
|
+
|
|
|
+#pv -l -s 100 -p -e -N "[ .... ] Copying 48304 files to [ / ]" > /dev/null
|
|
|
+
|
|
|
+#rsync --info=progress2 -arltD --delete --exclude-from=$EX_LIST / "${TempDir}"/rootfs
|
|
|
+
|
|
|
+rsync -avrltD --delete --exclude-from=$EX_LIST / "${TempDir}"/rootfs >/dev/null 2>&1
|
|
|
+
|
|
|
+display_alert "Re-enabling" "orangepi-resize-filesystem"
|
|
|
+chroot ${TempDir}/rootfs/ /bin/bash -c "systemctl --quiet enable orangepi-resize-filesystem" 2>&1 > /dev/null
|
|
|
+
|
|
|
+rm -f "${TempDir}"/rootfs/etc/fstab
|
|
|
+# Restore TMP and swap
|
|
|
+echo "# <file system> <mount point> <type> <options> <dump> <pass>" > "${TempDir}"/rootfs/etc/fstab
|
|
|
+echo "tmpfs /tmp tmpfs defaults,nosuid 0 0" >> "${TempDir}"/rootfs/etc/fstab
|
|
|
+grep swap /etc/fstab >> "${TempDir}"/rootfs/etc/fstab
|
|
|
+
|
|
|
+cp -R /boot "${TempDir}"/bootfs
|
|
|
+targetuuid=$(blkid -o export $rootdevice | grep -w UUID)
|
|
|
+
|
|
|
+sed -e 's,rootdev=.*,rootdev='"$targetuuid"',g' -i "${TempDir}"/bootfs//boot/orangepiEnv.txt
|
|
|
+grep -q '^rootdev' "${TempDir}"/bootfs/boot/orangepiEnv.txt || echo "rootdev=$targetuuid" >> "${TempDir}"/bootfs/boot/orangepiEnv.txt
|
|
|
+echo "$targetuuid / $ROOTFS_TYPE ${mountopts[$ROOTFS_TYPE]}" >> "${TempDir}"/rootfs/etc/fstab
|
|
|
+
|
|
|
+
|
|
|
+if [[ $(type -t write_uboot_platform) != function ]]; then
|
|
|
+ display_alert "no u-boot package found" "write bootloader"
|
|
|
+ exit
|
|
|
+fi
|
|
|
+
|
|
|
+display_alert "Writing U-boot bootloader" "$LOOP" "info"
|
|
|
+write_uboot_platform "$DIR" $LOOP
|
|
|
+
|
|
|
+display_alert "Backup completed" "${image}" "info"
|