forked from ungleich-public/ccollect
add updated patches from john
Signed-off-by: Nico Schottelius <nico@ikn.schottelius.org>
This commit is contained in:
parent
02264020f5
commit
6595fe7b97
15 changed files with 2718 additions and 0 deletions
15
contrib/jlawless-2009-06-03/README_g-i.txt
Normal file
15
contrib/jlawless-2009-06-03/README_g-i.txt
Normal file
|
@ -0,0 +1,15 @@
|
|||
Hello Nico,
|
||||
|
||||
I have attached three more patches for ccollect. Each patch
|
||||
has comments explaining its motivation.
|
||||
|
||||
All of these patches work-for-me (but I continue to test
|
||||
them). I would be interested in your opinion on, for example, the
|
||||
general approach used in i.patch which changes the way options are
|
||||
handled. I think it is a big improvement. If, however, you wanted
|
||||
the code to go in a different direction, let me know before we
|
||||
diverge too far.
|
||||
|
||||
Regards,
|
||||
|
||||
John
|
663
contrib/jlawless-2009-06-03/ccollect-i.sh
Executable file
663
contrib/jlawless-2009-06-03/ccollect-i.sh
Executable file
|
@ -0,0 +1,663 @@
|
|||
#!/bin/sh
|
||||
#
|
||||
# 2005-2009 Nico Schottelius (nico-ccollect at schottelius.org)
|
||||
#
|
||||
# This file is part of ccollect.
|
||||
#
|
||||
# ccollect 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 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# ccollect 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.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with ccollect. If not, see <http://www.gnu.org/licenses/>.
|
||||
#
|
||||
# Initially written for SyGroup (www.sygroup.ch)
|
||||
# Date: Mon Nov 14 11:45:11 CET 2005
|
||||
|
||||
#
|
||||
# Standard variables (stolen from cconf)
|
||||
#
|
||||
__pwd="$(pwd -P)"
|
||||
__mydir="${0%/*}"; __abs_mydir="$(cd "$__mydir" && pwd -P)"
|
||||
__myname=${0##*/}; __abs_myname="$__abs_mydir/$__myname"
|
||||
|
||||
#
|
||||
# where to find our configuration and temporary file
|
||||
#
|
||||
CCOLLECT_CONF=${CCOLLECT_CONF:-/etc/ccollect}
|
||||
CSOURCES=${CCOLLECT_CONF}/sources
|
||||
CDEFAULTS=${CCOLLECT_CONF}/defaults
|
||||
CPREEXEC="${CDEFAULTS}/pre_exec"
|
||||
CPOSTEXEC="${CDEFAULTS}/post_exec"
|
||||
|
||||
TMP=$(mktemp "/tmp/${__myname}.XXXXXX")
|
||||
VERSION=0.7.1
|
||||
RELEASE="2009-02-02"
|
||||
HALF_VERSION="ccollect ${VERSION}"
|
||||
FULL_VERSION="ccollect ${VERSION} (${RELEASE})"
|
||||
|
||||
#TSORT="tc" ; NEWER="cnewer"
|
||||
TSORT="t" ; NEWER="newer"
|
||||
|
||||
#
|
||||
# CDATE: how we use it for naming of the archives
|
||||
# DDATE: how the user should see it in our output (DISPLAY)
|
||||
#
|
||||
CDATE="date +%Y%m%d-%H%M"
|
||||
DDATE="date +%Y-%m-%d-%H:%M:%S"
|
||||
|
||||
#
|
||||
# unset parallel execution
|
||||
#
|
||||
PARALLEL=""
|
||||
|
||||
#
|
||||
# catch signals
|
||||
#
|
||||
trap "rm -f \"${TMP}\"" 1 2 15
|
||||
|
||||
#
|
||||
# Functions
|
||||
#
|
||||
|
||||
# time displaying echo
|
||||
_techo()
|
||||
{
|
||||
echo "$(${DDATE}): $@"
|
||||
}
|
||||
|
||||
# exit on error
|
||||
_exit_err()
|
||||
{
|
||||
_techo "$@"
|
||||
rm -f "${TMP}"
|
||||
exit 1
|
||||
}
|
||||
|
||||
add_name()
|
||||
{
|
||||
awk "{ print \"[${name}] \" \$0 }"
|
||||
}
|
||||
|
||||
pcmd()
|
||||
{
|
||||
if [ "$remote_host" ]; then
|
||||
ssh "$remote_host" "$@"
|
||||
else
|
||||
"$@"
|
||||
fi
|
||||
}
|
||||
|
||||
#
|
||||
# Version
|
||||
#
|
||||
display_version()
|
||||
{
|
||||
echo "${FULL_VERSION}"
|
||||
exit 0
|
||||
}
|
||||
|
||||
#
|
||||
# Tell how to use us
|
||||
#
|
||||
usage()
|
||||
{
|
||||
echo "${__myname}: <interval name> [args] <sources to backup>"
|
||||
echo ""
|
||||
echo " ccollect creates (pseudo) incremental backups"
|
||||
echo ""
|
||||
echo " -h, --help: Show this help screen"
|
||||
echo " -p, --parallel: Parallelise backup processes"
|
||||
echo " -a, --all: Backup all sources specified in ${CSOURCES}"
|
||||
echo " -v, --verbose: Be very verbose (uses set -x)"
|
||||
echo " -V, --version: Print version information"
|
||||
echo ""
|
||||
echo " This is version ${VERSION}, released on ${RELEASE}"
|
||||
echo " (the first version was written on 2005-12-05 by Nico Schottelius)."
|
||||
echo ""
|
||||
echo " Retrieve latest ccollect at http://unix.schottelius.org/ccollect/"
|
||||
exit 0
|
||||
}
|
||||
|
||||
#
|
||||
# Select interval if AUTO
|
||||
#
|
||||
# For this to work nicely, you have to choose interval names that sort nicely
|
||||
# such as int1, int2, int3 or a_daily, b_weekly, c_monthly, etc.
|
||||
#
|
||||
auto_interval()
|
||||
{
|
||||
if [ -d "${backup}/intervals" -a -n "$(ls "${backup}/intervals" 2>/dev/null)" ] ; then
|
||||
intervals_dir="${backup}/intervals"
|
||||
elif [ -d "${CDEFAULTS}/intervals" -a -n "$(ls "${CDEFAULTS}/intervals" 2>/dev/null)" ] ; then
|
||||
intervals_dir="${CDEFAULTS}/intervals"
|
||||
else
|
||||
_exit_err "No intervals are defined. Skipping."
|
||||
fi
|
||||
echo intervals_dir=${intervals_dir}
|
||||
|
||||
trial_interval="$(ls -1r "${intervals_dir}/" | head -n 1)" || \
|
||||
_exit_err "Failed to list contents of ${intervals_dir}/."
|
||||
_techo "Considering interval ${trial_interval}"
|
||||
most_recent="$(pcmd ls -${TSORT}p1 "${ddir}" | grep "^${trial_interval}.*/$" | head -n 1)" || \
|
||||
_exit_err "Failed to list contents of ${ddir}/."
|
||||
_techo " Most recent ${trial_interval}: '${most_recent}'"
|
||||
if [ -n "${most_recent}" ]; then
|
||||
no_intervals="$(ls -1 "${intervals_dir}/" | wc -l)"
|
||||
n=1
|
||||
while [ "${n}" -le "${no_intervals}" ]; do
|
||||
trial_interval="$(ls -p1 "${intervals_dir}/" | tail -n+${n} | head -n 1)"
|
||||
_techo "Considering interval '${trial_interval}'"
|
||||
c_interval="$(cat "${intervals_dir}/${trial_interval}" 2>/dev/null)"
|
||||
m=$((${n}+1))
|
||||
set -- "${ddir}" -maxdepth 1
|
||||
while [ "${m}" -le "${no_intervals}" ]; do
|
||||
interval_m="$(ls -1 "${intervals_dir}/" | tail -n+${m} | head -n 1)"
|
||||
most_recent="$(pcmd ls -${TSORT}p1 "${ddir}" | grep "^${interval_m}\..*/$" | head -n 1)"
|
||||
_techo " Most recent ${interval_m}: '${most_recent}'"
|
||||
if [ -n "${most_recent}" ] ; then
|
||||
set -- "$@" -$NEWER "${ddir}/${most_recent}"
|
||||
fi
|
||||
m=$((${m}+1))
|
||||
done
|
||||
count=$(pcmd find "$@" -iname "${trial_interval}*" | wc -l)
|
||||
_techo " Found $count more recent backups of ${trial_interval} (limit: ${c_interval})"
|
||||
if [ "$count" -lt "${c_interval}" ] ; then
|
||||
break
|
||||
fi
|
||||
n=$((${n}+1))
|
||||
done
|
||||
fi
|
||||
export INTERVAL="${trial_interval}"
|
||||
D_FILE_INTERVAL="${intervals_dir}/${INTERVAL}"
|
||||
D_INTERVAL=$(cat "${D_FILE_INTERVAL}" 2>/dev/null)
|
||||
}
|
||||
|
||||
#
|
||||
# need at least interval and one source or --all
|
||||
#
|
||||
if [ $# -lt 2 ]; then
|
||||
if [ "$1" = "-V" -o "$1" = "--version" ]; then
|
||||
display_version
|
||||
else
|
||||
usage
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# check for configuraton directory
|
||||
#
|
||||
[ -d "${CCOLLECT_CONF}" ] || _exit_err "No configuration found in " \
|
||||
"\"${CCOLLECT_CONF}\" (is \$CCOLLECT_CONF properly set?)"
|
||||
|
||||
#
|
||||
# Filter arguments
|
||||
#
|
||||
export INTERVAL="$1"; shift
|
||||
i=1
|
||||
no_sources=0
|
||||
|
||||
#
|
||||
# Create source "array"
|
||||
#
|
||||
while [ "$#" -ge 1 ]; do
|
||||
eval arg=\"\$1\"; shift
|
||||
|
||||
if [ "${NO_MORE_ARGS}" = 1 ]; then
|
||||
eval source_${no_sources}=\"${arg}\"
|
||||
no_sources=$((${no_sources}+1))
|
||||
|
||||
# make variable available for subscripts
|
||||
eval export source_${no_sources}
|
||||
else
|
||||
case "${arg}" in
|
||||
-a|--all)
|
||||
ALL=1
|
||||
;;
|
||||
-v|--verbose)
|
||||
VERBOSE=1
|
||||
;;
|
||||
-p|--parallel)
|
||||
PARALLEL=1
|
||||
;;
|
||||
-h|--help)
|
||||
usage
|
||||
;;
|
||||
--)
|
||||
NO_MORE_ARGS=1
|
||||
;;
|
||||
*)
|
||||
eval source_${no_sources}=\"$arg\"
|
||||
no_sources=$(($no_sources+1))
|
||||
;;
|
||||
esac
|
||||
fi
|
||||
|
||||
i=$(($i+1))
|
||||
done
|
||||
|
||||
# also export number of sources
|
||||
export no_sources
|
||||
|
||||
#
|
||||
# be really, really, really verbose
|
||||
#
|
||||
if [ "${VERBOSE}" = 1 ]; then
|
||||
set -x
|
||||
fi
|
||||
|
||||
#
|
||||
# Look, if we should take ALL sources
|
||||
#
|
||||
if [ "${ALL}" = 1 ]; then
|
||||
# reset everything specified before
|
||||
no_sources=0
|
||||
|
||||
#
|
||||
# get entries from sources
|
||||
#
|
||||
cwd=$(pwd -P)
|
||||
( cd "${CSOURCES}" && ls > "${TMP}" ); ret=$?
|
||||
|
||||
[ "${ret}" -eq 0 ] || _exit_err "Listing of sources failed. Aborting."
|
||||
|
||||
while read tmp; do
|
||||
eval source_${no_sources}=\"${tmp}\"
|
||||
no_sources=$((${no_sources}+1))
|
||||
done < "${TMP}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Need at least ONE source to backup
|
||||
#
|
||||
if [ "${no_sources}" -lt 1 ]; then
|
||||
usage
|
||||
else
|
||||
_techo "${HALF_VERSION}: Beginning backup using interval ${INTERVAL}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Look for pre-exec command (general)
|
||||
#
|
||||
if [ -x "${CPREEXEC}" ]; then
|
||||
_techo "Executing ${CPREEXEC} ..."
|
||||
"${CPREEXEC}"; ret=$?
|
||||
_techo "Finished ${CPREEXEC} (return code: ${ret})."
|
||||
|
||||
[ "${ret}" -eq 0 ] || _exit_err "${CPREEXEC} failed. Aborting"
|
||||
fi
|
||||
|
||||
#
|
||||
# check default configuration
|
||||
#
|
||||
|
||||
D_FILE_INTERVAL="${CDEFAULTS}/intervals/${INTERVAL}"
|
||||
D_INTERVAL=$(cat "${D_FILE_INTERVAL}" 2>/dev/null)
|
||||
|
||||
|
||||
#
|
||||
# Let's do the backup
|
||||
#
|
||||
i=0
|
||||
while [ "${i}" -lt "${no_sources}" ]; do
|
||||
|
||||
#
|
||||
# Get current source
|
||||
#
|
||||
eval name=\"\$source_${i}\"
|
||||
i=$((${i}+1))
|
||||
|
||||
export name
|
||||
|
||||
#
|
||||
# start ourself, if we want parallel execution
|
||||
#
|
||||
if [ "${PARALLEL}" ]; then
|
||||
"$0" "${INTERVAL}" "${name}" &
|
||||
continue
|
||||
fi
|
||||
|
||||
#
|
||||
# Start subshell for easy log editing
|
||||
#
|
||||
(
|
||||
#
|
||||
# Stderr to stdout, so we can produce nice logs
|
||||
#
|
||||
exec 2>&1
|
||||
|
||||
#
|
||||
# Configuration
|
||||
#
|
||||
backup="${CSOURCES}/${name}"
|
||||
c_source="${backup}/source"
|
||||
c_dest="${backup}/destination"
|
||||
c_pre_exec="${backup}/pre_exec"
|
||||
c_post_exec="${backup}/post_exec"
|
||||
for opt in exclude verbose very_verbose rsync_options summary delete_incomplete remote_host ; do
|
||||
if [ -f "${backup}/$opt" -o -f "${backup}/no_$opt" ]; then
|
||||
eval c_$opt=\"${backup}/$opt\"
|
||||
else
|
||||
eval c_$opt=\"${CDEFAULTS}/$opt\"
|
||||
fi
|
||||
done
|
||||
|
||||
#
|
||||
# Marking backups: If we abort it's not removed => Backup is broken
|
||||
#
|
||||
c_marker=".ccollect-marker"
|
||||
|
||||
#
|
||||
# Times
|
||||
#
|
||||
begin_s=$(date +%s)
|
||||
|
||||
#
|
||||
# unset possible options
|
||||
#
|
||||
VERBOSE=""
|
||||
VVERBOSE=""
|
||||
|
||||
_techo "Beginning to backup"
|
||||
|
||||
#
|
||||
# Standard configuration checks
|
||||
#
|
||||
if [ ! -e "${backup}" ]; then
|
||||
_exit_err "Source does not exist."
|
||||
fi
|
||||
|
||||
#
|
||||
# configuration _must_ be a directory
|
||||
#
|
||||
if [ ! -d "${backup}" ]; then
|
||||
_exit_err "\"${name}\" is not a cconfig-directory. Skipping."
|
||||
fi
|
||||
|
||||
#
|
||||
# first execute pre_exec, which may generate destination or other
|
||||
# parameters
|
||||
#
|
||||
if [ -x "${c_pre_exec}" ]; then
|
||||
_techo "Executing ${c_pre_exec} ..."
|
||||
"${c_pre_exec}"; ret="$?"
|
||||
_techo "Finished ${c_pre_exec} (return code ${ret})."
|
||||
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "${c_pre_exec} failed. Skipping."
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# Destination is a path
|
||||
#
|
||||
if [ ! -f "${c_dest}" ]; then
|
||||
_exit_err "Destination ${c_dest} is not a file. Skipping."
|
||||
else
|
||||
ddir=$(cat "${c_dest}"); ret="$?"
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "Destination ${c_dest} is not readable. Skipping."
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# interval definition: First try source specific, fallback to default
|
||||
#
|
||||
if [ "${INTERVAL}" = "AUTO" ] ; then
|
||||
auto_interval
|
||||
_techo "Selected interval: '$INTERVAL'"
|
||||
fi
|
||||
c_interval="$(cat "${backup}/intervals/${INTERVAL}" 2>/dev/null)"
|
||||
|
||||
if [ -z "${c_interval}" ]; then
|
||||
c_interval="${D_INTERVAL}"
|
||||
|
||||
if [ -z "${c_interval}" ]; then
|
||||
_exit_err "No definition for interval \"${INTERVAL}\" found. Skipping."
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# Source checks
|
||||
#
|
||||
if [ ! -f "${c_source}" ]; then
|
||||
_exit_err "Source description \"${c_source}\" is not a file. Skipping."
|
||||
else
|
||||
source=$(cat "${c_source}"); ret="$?"
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "Source ${c_source} is not readable. Skipping."
|
||||
fi
|
||||
fi
|
||||
# Verify source is up and accepting connections before deleting any old backups
|
||||
rsync "$source" >/dev/null || _exit_err "Source ${source} is not readable. Skipping."
|
||||
|
||||
#
|
||||
# do we backup to a remote host? then set pre-cmd
|
||||
#
|
||||
if [ -f "${c_remote_host}" ]; then
|
||||
# adjust ls and co
|
||||
remote_host=$(cat "${c_remote_host}"); ret="$?"
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "Remote host file ${c_remote_host} exists, but is not readable. Skipping."
|
||||
fi
|
||||
destination="${remote_host}:${ddir}"
|
||||
else
|
||||
remote_host=""
|
||||
destination="${ddir}"
|
||||
fi
|
||||
export remote_host
|
||||
|
||||
#
|
||||
# check for existence / use real name
|
||||
#
|
||||
( pcmd cd "$ddir" ) || _exit_err "Cannot change to ${ddir}. Skipping."
|
||||
|
||||
|
||||
# NEW method as of 0.6:
|
||||
# - insert ccollect default parameters
|
||||
# - insert options
|
||||
# - insert user options
|
||||
|
||||
#
|
||||
# rsync standard options
|
||||
#
|
||||
|
||||
set -- "$@" "--archive" "--delete" "--numeric-ids" "--relative" \
|
||||
"--delete-excluded" "--sparse"
|
||||
|
||||
#
|
||||
# exclude list
|
||||
#
|
||||
if [ -f "${c_exclude}" ]; then
|
||||
set -- "$@" "--exclude-from=${c_exclude}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Output a summary
|
||||
#
|
||||
if [ -f "${c_summary}" ]; then
|
||||
set -- "$@" "--stats"
|
||||
fi
|
||||
|
||||
#
|
||||
# Verbosity for rsync
|
||||
#
|
||||
if [ -f "${c_very_verbose}" ]; then
|
||||
set -- "$@" "-vv"
|
||||
elif [ -f "${c_verbose}" ]; then
|
||||
set -- "$@" "-v"
|
||||
fi
|
||||
|
||||
#
|
||||
# extra options for rsync provided by the user
|
||||
#
|
||||
if [ -f "${c_rsync_options}" ]; then
|
||||
while read line; do
|
||||
set -- "$@" "$line"
|
||||
done < "${c_rsync_options}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Check for incomplete backups
|
||||
#
|
||||
pcmd ls -1 "$ddir/${INTERVAL}"*".${c_marker}" 2>/dev/null | while read marker; do
|
||||
incomplete="$(echo ${marker} | sed "s/\\.${c_marker}\$//")"
|
||||
_techo "Incomplete backup: ${incomplete}"
|
||||
if [ -f "${c_delete_incomplete}" ]; then
|
||||
_techo "Deleting ${incomplete} ..."
|
||||
pcmd rm $VVERBOSE -rf "${incomplete}" || \
|
||||
_exit_err "Removing ${incomplete} failed."
|
||||
pcmd rm $VVERBOSE -f "${marker}" || \
|
||||
_exit_err "Removing ${marker} failed."
|
||||
fi
|
||||
done
|
||||
|
||||
#
|
||||
# check if maximum number of backups is reached, if so remove
|
||||
# use grep and ls -p so we only look at directories
|
||||
#
|
||||
count="$(pcmd ls -p1 "${ddir}" | grep "^${INTERVAL}\..*/\$" | wc -l \
|
||||
| sed 's/^ *//g')" || _exit_err "Counting backups failed"
|
||||
|
||||
_techo "Existing backups: ${count} Total keeping backups: ${c_interval}"
|
||||
|
||||
if [ "${count}" -ge "${c_interval}" ]; then
|
||||
substract=$((${c_interval} - 1))
|
||||
remove=$((${count} - ${substract}))
|
||||
_techo "Removing ${remove} backup(s)..."
|
||||
|
||||
pcmd ls -${TSORT}p1r "$ddir" | grep "^${INTERVAL}\..*/\$" | \
|
||||
head -n "${remove}" > "${TMP}" || \
|
||||
_exit_err "Listing old backups failed"
|
||||
|
||||
i=0
|
||||
while read to_remove; do
|
||||
eval remove_$i=\"${to_remove}\"
|
||||
i=$(($i+1))
|
||||
done < "${TMP}"
|
||||
|
||||
j=0
|
||||
while [ "$j" -lt "$i" ]; do
|
||||
eval to_remove=\"\$remove_$j\"
|
||||
_techo "Removing ${to_remove} ..."
|
||||
pcmd rm ${VVERBOSE} -rf "${ddir}/${to_remove}" || \
|
||||
_exit_err "Removing ${to_remove} failed."
|
||||
j=$(($j+1))
|
||||
done
|
||||
fi
|
||||
|
||||
|
||||
#
|
||||
# Check for backup directory to clone from: Always clone from the latest one!
|
||||
#
|
||||
# Depending on your file system, you may want to sort on:
|
||||
# 1. mtime (modification time) with TSORT=t, or
|
||||
# 2. ctime (last change time, usually) with TSORT=tc
|
||||
last_dir="$(pcmd ls -${TSORT}p1 "${ddir}" | grep '/$' | head -n 1)" || \
|
||||
_exit_err "Failed to list contents of ${ddir}."
|
||||
|
||||
#
|
||||
# clone from old backup, if existing
|
||||
#
|
||||
if [ "${last_dir}" ]; then
|
||||
set -- "$@" "--link-dest=${ddir}/${last_dir}"
|
||||
_techo "Hard linking from ${last_dir}"
|
||||
fi
|
||||
|
||||
|
||||
# set time when we really begin to backup, not when we began to remove above
|
||||
destination_date=$(${CDATE})
|
||||
destination_dir="${ddir}/${INTERVAL}.${destination_date}.$$"
|
||||
destination_full="${destination}/${INTERVAL}.${destination_date}.$$"
|
||||
|
||||
# give some info
|
||||
_techo "Beginning to backup, this may take some time..."
|
||||
|
||||
_techo "Creating ${destination_dir} ..."
|
||||
pcmd mkdir ${VVERBOSE} "${destination_dir}" || \
|
||||
_exit_err "Creating ${destination_dir} failed. Skipping."
|
||||
|
||||
#
|
||||
# added marking in 0.6 (and remove it, if successful later)
|
||||
#
|
||||
pcmd touch "${destination_dir}.${c_marker}"
|
||||
|
||||
#
|
||||
# the rsync part
|
||||
#
|
||||
_techo "Transferring files..."
|
||||
rsync "$@" "${source}" "${destination_full}"; ret=$?
|
||||
# Correct the modification time:
|
||||
pcmd touch "${destination_dir}"
|
||||
|
||||
#
|
||||
# remove marking here
|
||||
#
|
||||
if [ "$ret" -ne 12 ] ; then
|
||||
pcmd rm "${destination_dir}.${c_marker}" || \
|
||||
_exit_err "Removing ${destination_dir}/${c_marker} failed."
|
||||
fi
|
||||
|
||||
_techo "Finished backup (rsync return code: $ret)."
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_techo "Warning: rsync exited non-zero, the backup may be broken (see rsync errors)."
|
||||
fi
|
||||
|
||||
#
|
||||
# post_exec
|
||||
#
|
||||
if [ -x "${c_post_exec}" ]; then
|
||||
_techo "Executing ${c_post_exec} ..."
|
||||
"${c_post_exec}"; ret=$?
|
||||
_techo "Finished ${c_post_exec}."
|
||||
|
||||
if [ ${ret} -ne 0 ]; then
|
||||
_exit_err "${c_post_exec} failed."
|
||||
fi
|
||||
fi
|
||||
|
||||
# Calculation
|
||||
end_s=$(date +%s)
|
||||
|
||||
full_seconds=$((${end_s} - ${begin_s}))
|
||||
hours=$((${full_seconds} / 3600))
|
||||
seconds=$((${full_seconds} - (${hours} * 3600)))
|
||||
minutes=$((${seconds} / 60))
|
||||
seconds=$((${seconds} - (${minutes} * 60)))
|
||||
|
||||
_techo "Backup lasted: ${hours}:${minutes}:${seconds} (h:m:s)"
|
||||
|
||||
) | add_name
|
||||
done
|
||||
|
||||
#
|
||||
# Be a good parent and wait for our children, if they are running wild parallel
|
||||
#
|
||||
if [ "${PARALLEL}" ]; then
|
||||
_techo "Waiting for children to complete..."
|
||||
wait
|
||||
fi
|
||||
|
||||
#
|
||||
# Look for post-exec command (general)
|
||||
#
|
||||
if [ -x "${CPOSTEXEC}" ]; then
|
||||
_techo "Executing ${CPOSTEXEC} ..."
|
||||
"${CPOSTEXEC}"; ret=$?
|
||||
_techo "Finished ${CPOSTEXEC} (return code: ${ret})."
|
||||
|
||||
if [ ${ret} -ne 0 ]; then
|
||||
_techo "${CPOSTEXEC} failed."
|
||||
fi
|
||||
fi
|
||||
|
||||
rm -f "${TMP}"
|
||||
_techo "Finished ${WE}"
|
||||
|
||||
# vim: set shiftwidth=3 tabstop=3 expandtab :
|
74
contrib/jlawless-2009-06-03/g.patch
Normal file
74
contrib/jlawless-2009-06-03/g.patch
Normal file
|
@ -0,0 +1,74 @@
|
|||
# I found that ccollect was not deleting incomplete backups despite the
|
||||
# delete_incomplete option being specified. I traced the problem to:
|
||||
#
|
||||
# < pcmd rm $VVERBOSE -rf "${ddir}/${realincomplete}" || \
|
||||
#
|
||||
# which, at least on all the systems I tested, should read:
|
||||
#
|
||||
# > pcmd rm $VVERBOSE -rf "${realincomplete}" || \
|
||||
#
|
||||
# Also, the marker file is not deleted. I didn't see any reason to keep
|
||||
# those files around (what do you think?), so I deleted them also:
|
||||
#
|
||||
# > pcmd rm $VVERBOSE -rf "${ddir}/${realincomplete}" || \
|
||||
# > _exit_err "Removing ${realincomplete} failed."
|
||||
#
|
||||
# As long as I was messing with the delete incomplete code and therefore need
|
||||
# to test it, I took the liberty of simplifying it. The v0.7.1 code uses
|
||||
# multiple loops with multiple loop counters and creates many variables. I
|
||||
# simplified that to a single loop:
|
||||
#
|
||||
# > pcmd ls -1 "$ddir/${INTERVAL}"*".${c_marker}" 2>/dev/null | while read marker; do
|
||||
# > incomplete="$(echo ${marker} | sed "s/\\.${c_marker}\$//")"
|
||||
# > _techo "Incomplete backup: ${incomplete}"
|
||||
# > if [ "${DELETE_INCOMPLETE}" = "yes" ]; then
|
||||
# > _techo "Deleting ${incomplete} ..."
|
||||
# > pcmd rm $VVERBOSE -rf "${incomplete}" || \
|
||||
# > _exit_err "Removing ${incomplete} failed."
|
||||
# > pcmd rm $VVERBOSE -f "${marker}" || \
|
||||
# > _exit_err "Removing ${marker} failed."
|
||||
# > fi
|
||||
# > done
|
||||
#
|
||||
# The final code (a) fixes the delete bug, (b) also deletes the marker, and
|
||||
# (c) is eight lines shorter than the original.
|
||||
#
|
||||
--- ccollect-f.sh 2009-05-12 12:49:28.000000000 -0700
|
||||
+++ ccollect-g.sh 2009-06-03 14:32:03.000000000 -0700
|
||||
@@ -516,28 +516,20 @@
|
||||
fi
|
||||
|
||||
#
|
||||
# Check for incomplete backups
|
||||
#
|
||||
- pcmd ls -1 "$ddir/${INTERVAL}"*".${c_marker}" > "${TMP}" 2>/dev/null
|
||||
-
|
||||
- i=0
|
||||
- while read incomplete; do
|
||||
- eval incomplete_$i=\"$(echo ${incomplete} | sed "s/\\.${c_marker}\$//")\"
|
||||
- i=$(($i+1))
|
||||
- done < "${TMP}"
|
||||
-
|
||||
- j=0
|
||||
- while [ "$j" -lt "$i" ]; do
|
||||
- eval realincomplete=\"\$incomplete_$j\"
|
||||
- _techo "Incomplete backup: ${realincomplete}"
|
||||
+ pcmd ls -1 "$ddir/${INTERVAL}"*".${c_marker}" 2>/dev/null | while read marker; do
|
||||
+ incomplete="$(echo ${marker} | sed "s/\\.${c_marker}\$//")"
|
||||
+ _techo "Incomplete backup: ${incomplete}"
|
||||
if [ "${DELETE_INCOMPLETE}" = "yes" ]; then
|
||||
- _techo "Deleting ${realincomplete} ..."
|
||||
- pcmd rm $VVERBOSE -rf "${ddir}/${realincomplete}" || \
|
||||
- _exit_err "Removing ${realincomplete} failed."
|
||||
+ _techo "Deleting ${incomplete} ..."
|
||||
+ pcmd rm $VVERBOSE -rf "${incomplete}" || \
|
||||
+ _exit_err "Removing ${incomplete} failed."
|
||||
+ pcmd rm $VVERBOSE -f "${marker}" || \
|
||||
+ _exit_err "Removing ${marker} failed."
|
||||
fi
|
||||
- j=$(($j+1))
|
||||
done
|
||||
|
||||
#
|
||||
# check if maximum number of backups is reached, if so remove
|
||||
# use grep and ls -p so we only look at directories
|
18
contrib/jlawless-2009-06-03/h.patch
Normal file
18
contrib/jlawless-2009-06-03/h.patch
Normal file
|
@ -0,0 +1,18 @@
|
|||
# A line in my f.patch was missing needed quotation marks.
|
||||
# This fixes that.
|
||||
#
|
||||
--- ccollect-g.sh 2009-06-03 14:32:03.000000000 -0700
|
||||
+++ ccollect-h.sh 2009-06-03 14:32:19.000000000 -0700
|
||||
@@ -412,11 +412,11 @@
|
||||
fi
|
||||
|
||||
#
|
||||
# interval definition: First try source specific, fallback to default
|
||||
#
|
||||
- if [ ${INTERVAL} = "AUTO" ] ; then
|
||||
+ if [ "${INTERVAL}" = "AUTO" ] ; then
|
||||
auto_interval
|
||||
_techo "Selected interval: '$INTERVAL'"
|
||||
fi
|
||||
c_interval="$(cat "${backup}/intervals/${INTERVAL}" 2>/dev/null)"
|
||||
|
134
contrib/jlawless-2009-06-03/i.patch
Normal file
134
contrib/jlawless-2009-06-03/i.patch
Normal file
|
@ -0,0 +1,134 @@
|
|||
# I have many sources that use the same options so I put those
|
||||
# options in the defaults directory. I found that ccollect was
|
||||
# ignoring most of them. I thought that this was a bug so I wrote
|
||||
# some code to correct this:
|
||||
#
|
||||
# > for opt in exclude verbose very_verbose rsync_options summary delete_incomplete remote_host ; do
|
||||
# > if [ -f "${backup}/$opt" -o -f "${backup}/no_$opt" ]; then
|
||||
# > eval c_$opt=\"${backup}/$opt\"
|
||||
# > else
|
||||
# > eval c_$opt=\"${CDEFAULTS}/$opt\"
|
||||
# > fi
|
||||
# > done
|
||||
#
|
||||
# This also adds a new feature: if some option, say verbose, is
|
||||
# specified in the defaults directory, it can be turned off for
|
||||
# particular sources by specifying no_verbose as a source option.
|
||||
#
|
||||
# A side effect of this approach is that it forces script variable
|
||||
# names to be consistent with option file names. Thus, there are
|
||||
# several changes such as:
|
||||
#
|
||||
# < if [ -f "${c_rsync_extra}" ]; then
|
||||
# > if [ -f "${c_rsync_options}" ]; then
|
||||
#
|
||||
# and
|
||||
#
|
||||
# < if [ -f "${c_vverbose}" ]; then
|
||||
# > if [ -f "${c_very_verbose}" ]; then
|
||||
#
|
||||
# After correcting the bug and adding the "no_" feature, the code is
|
||||
# 12 lines shorter.
|
||||
#
|
||||
--- ccollect-h.sh 2009-06-01 15:59:11.000000000 -0700
|
||||
+++ ccollect-i.sh 2009-06-03 14:27:58.000000000 -0700
|
||||
@@ -336,20 +336,19 @@
|
||||
# Configuration
|
||||
#
|
||||
backup="${CSOURCES}/${name}"
|
||||
c_source="${backup}/source"
|
||||
c_dest="${backup}/destination"
|
||||
- c_exclude="${backup}/exclude"
|
||||
- c_verbose="${backup}/verbose"
|
||||
- c_vverbose="${backup}/very_verbose"
|
||||
- c_rsync_extra="${backup}/rsync_options"
|
||||
- c_summary="${backup}/summary"
|
||||
c_pre_exec="${backup}/pre_exec"
|
||||
c_post_exec="${backup}/post_exec"
|
||||
- f_incomplete="delete_incomplete"
|
||||
- c_incomplete="${backup}/${f_incomplete}"
|
||||
- c_remote_host="${backup}/remote_host"
|
||||
+ for opt in exclude verbose very_verbose rsync_options summary delete_incomplete remote_host ; do
|
||||
+ if [ -f "${backup}/$opt" -o -f "${backup}/no_$opt" ]; then
|
||||
+ eval c_$opt=\"${backup}/$opt\"
|
||||
+ else
|
||||
+ eval c_$opt=\"${CDEFAULTS}/$opt\"
|
||||
+ fi
|
||||
+ done
|
||||
|
||||
#
|
||||
# Marking backups: If we abort it's not removed => Backup is broken
|
||||
#
|
||||
c_marker=".ccollect-marker"
|
||||
@@ -360,16 +359,12 @@
|
||||
begin_s=$(date +%s)
|
||||
|
||||
#
|
||||
# unset possible options
|
||||
#
|
||||
- EXCLUDE=""
|
||||
- RSYNC_EXTRA=""
|
||||
- SUMMARY=""
|
||||
VERBOSE=""
|
||||
VVERBOSE=""
|
||||
- DELETE_INCOMPLETE=""
|
||||
|
||||
_techo "Beginning to backup"
|
||||
|
||||
#
|
||||
# Standard configuration checks
|
||||
@@ -462,17 +457,10 @@
|
||||
# check for existence / use real name
|
||||
#
|
||||
( pcmd cd "$ddir" ) || _exit_err "Cannot change to ${ddir}. Skipping."
|
||||
|
||||
|
||||
- #
|
||||
- # Check whether to delete incomplete backups
|
||||
- #
|
||||
- if [ -f "${c_incomplete}" -o -f "${CDEFAULTS}/${f_incomplete}" ]; then
|
||||
- DELETE_INCOMPLETE="yes"
|
||||
- fi
|
||||
-
|
||||
# NEW method as of 0.6:
|
||||
# - insert ccollect default parameters
|
||||
# - insert options
|
||||
# - insert user options
|
||||
|
||||
@@ -498,32 +486,32 @@
|
||||
fi
|
||||
|
||||
#
|
||||
# Verbosity for rsync
|
||||
#
|
||||
- if [ -f "${c_vverbose}" ]; then
|
||||
+ if [ -f "${c_very_verbose}" ]; then
|
||||
set -- "$@" "-vv"
|
||||
elif [ -f "${c_verbose}" ]; then
|
||||
set -- "$@" "-v"
|
||||
fi
|
||||
|
||||
#
|
||||
# extra options for rsync provided by the user
|
||||
#
|
||||
- if [ -f "${c_rsync_extra}" ]; then
|
||||
+ if [ -f "${c_rsync_options}" ]; then
|
||||
while read line; do
|
||||
set -- "$@" "$line"
|
||||
- done < "${c_rsync_extra}"
|
||||
+ done < "${c_rsync_options}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Check for incomplete backups
|
||||
#
|
||||
pcmd ls -1 "$ddir/${INTERVAL}"*".${c_marker}" 2>/dev/null | while read marker; do
|
||||
incomplete="$(echo ${marker} | sed "s/\\.${c_marker}\$//")"
|
||||
_techo "Incomplete backup: ${incomplete}"
|
||||
- if [ "${DELETE_INCOMPLETE}" = "yes" ]; then
|
||||
+ if [ -f "${c_delete_incomplete}" ]; then
|
||||
_techo "Deleting ${incomplete} ..."
|
||||
pcmd rm $VVERBOSE -rf "${incomplete}" || \
|
||||
_exit_err "Removing ${incomplete} failed."
|
||||
pcmd rm $VVERBOSE -f "${marker}" || \
|
||||
_exit_err "Removing ${marker} failed."
|
296
contrib/jlawless-2009-06-03/old/README_a-f.txt
Normal file
296
contrib/jlawless-2009-06-03/old/README_a-f.txt
Normal file
|
@ -0,0 +1,296 @@
|
|||
Dear Nico Schottelius,
|
||||
|
||||
I have started using ccollect and I very much like its design:
|
||||
it is elegant and effective.
|
||||
|
||||
In the process of getting ccollect setup and running, I made
|
||||
five changes, including one major new feature, that I hope you will
|
||||
find useful.
|
||||
|
||||
First, I added the following before any old backup gets deleted:
|
||||
|
||||
> # Verify source is up and accepting connections before deleting any old backups
|
||||
> rsync "$source" >/dev/null || _exit_err "Source ${source} is not readable. Skipping."
|
||||
|
||||
I think that this quick test is a much better than, say, pinging
|
||||
the source in a pre-exec script: this tests not only that the
|
||||
source is up and connected to the net, it also verifies (1) that
|
||||
ssh is up and accepting our key (if we are using ssh), and (2) that
|
||||
the source directory is mounted (if it needs to be mounted) and
|
||||
readable.
|
||||
|
||||
Second, I found ccollect's use of ctime problematic. After
|
||||
copying an old backup over to my ccollect destination, I adjusted
|
||||
mtime and atime where needed using touch, e.g.:
|
||||
|
||||
touch -d"28 Apr 2009 3:00" destination/daily.01
|
||||
|
||||
However, as far as I know, there is no way to correct a bad ctime.
|
||||
I ran into this issue repeatedly while adjusting my backup
|
||||
configuration. (For example, "cp -a" preserves mtime but not
|
||||
ctime. Even worse, "cp -al old new" also changes ctime on old.)
|
||||
|
||||
Another potential problem with ctime is that it is file-system
|
||||
dependent: I have read that Windows sets ctime to create-time not
|
||||
last change-time.
|
||||
|
||||
However, It is simple to give a new backup the correct mtime.
|
||||
After the rsync step, I added the command:
|
||||
|
||||
553a616,617
|
||||
> # Correct the modification time:
|
||||
> pcmd touch "${destination_dir}"
|
||||
|
||||
Even if ccollect continues to use ctime for sorting, I see no
|
||||
reason not to have the backup directory have the correct mtime.
|
||||
|
||||
To allow the rest of the code to use either ctime or mtime, I
|
||||
added definitions:
|
||||
|
||||
44a45,47
|
||||
> #TSORT="tc" ; NEWER="cnewer"
|
||||
> TSORT="t" ; NEWER="newer"
|
||||
|
||||
(It would be better if this choice was user-configurable because
|
||||
those with existing backup directories should continue to use ctime
|
||||
until the mtimes of their directories are correct. The correction
|
||||
would happen passively over time as new backups created using the
|
||||
above touch command and the old ones are deleted.)
|
||||
|
||||
With these definitions, the proper link-dest directory can then be
|
||||
found using this minor change (and comment update):
|
||||
|
||||
516,519c579,582
|
||||
< # Use ls -1c instead of -1t, because last modification maybe the same on all
|
||||
< # and metadate update (-c) is updated by rsync locally.
|
||||
< #
|
||||
< last_dir="$(pcmd ls -tcp1 "${ddir}" | grep '/$' | head -n 1)" || \
|
||||
---
|
||||
> # Depending on your file system, you may want to sort on:
|
||||
> # 1. mtime (modification time) with TSORT=t, or
|
||||
> # 2. ctime (last change time, usually) with TSORT=tc
|
||||
> last_dir="$(pcmd ls -${TSORT}p1 "${ddir}" | grep '/$' | head -n 1)" || \
|
||||
|
||||
Thirdly, after I copied my old backups over to my ccollect
|
||||
destination directory, I found that ccollect would delete a
|
||||
recent backup not an old backup! My problem was that, unknown to
|
||||
me, the algorithm to find the oldest backup (for deletion) was
|
||||
inconsistent with that used to find the newest (for link-dest). I
|
||||
suggest that these two should be consistent. Because time-sorting
|
||||
seemed more consistent with the ccollect documentation, I suggest:
|
||||
|
||||
492,493c555,556
|
||||
< pcmd ls -p1 "$ddir" | grep "^${INTERVAL}\..*/\$" | \
|
||||
< sort -n | head -n "${remove}" > "${TMP}" || \
|
||||
---
|
||||
> pcmd ls -${TSORT}p1r "$ddir" | grep "^${INTERVAL}\..*/\$" | \
|
||||
> head -n "${remove}" > "${TMP}" || \
|
||||
|
||||
Fourthly, in my experience, rsync error code 12 means complete
|
||||
failure, usually because the source refuses the ssh connection.
|
||||
So, I left the marker in that case:
|
||||
|
||||
558,559c622,625
|
||||
< pcmd rm "${destination_dir}.${c_marker}" || \
|
||||
< _exit_err "Removing ${destination_dir}/${c_marker} failed."
|
||||
---
|
||||
> if [ "$ret" -ne 12 ] ; then
|
||||
> pcmd rm "${destination_dir}.${c_marker}" || \
|
||||
> _exit_err "Removing ${destination_dir}/${c_marker} failed."
|
||||
> fi
|
||||
|
||||
(A better solution might allow a user-configurable list of error
|
||||
codes that are treated the same as a fail.)
|
||||
|
||||
Fifth, because I was frustrated by the problems of having a
|
||||
cron-job decide which interval to backup, I added a major new
|
||||
feature: the modified ccollect can now automatically select an
|
||||
interval to use for backup.
|
||||
|
||||
Cron-job controlled backup works well if all machines are up and
|
||||
running all the time and nothing ever goes wrong. I have, however,
|
||||
some machines that are occasionally turned off, or that are mobile
|
||||
and only sometimes connected to local net. For these machines, the
|
||||
use of cron-jobs to select intervals can be a disaster.
|
||||
|
||||
There are several ways one could automatically choose an
|
||||
appropriate interval. The method I show below has the advantage
|
||||
that it works with existing ccollect configuration files. The only
|
||||
requirement is that interval names be chosen to sort nicely (under
|
||||
ls). For example, I currently use:
|
||||
|
||||
$ ls -1 intervals
|
||||
a_daily
|
||||
b_weekly
|
||||
c_monthly
|
||||
d_quarterly
|
||||
e_yearly
|
||||
$ cat intervals/*
|
||||
6
|
||||
3
|
||||
2
|
||||
3
|
||||
30
|
||||
|
||||
A simpler example would be:
|
||||
|
||||
$ ls -1 intervals
|
||||
int1
|
||||
int2
|
||||
int3
|
||||
$ cat intervals/*
|
||||
2
|
||||
3
|
||||
4
|
||||
|
||||
The algorithm works as follows:
|
||||
|
||||
If no backup exists for the least frequent interval (int3 in the
|
||||
simpler example), then use that interval. Otherwise, use the
|
||||
most frequent interval (int1) unless there are "$(cat
|
||||
intervals/int1)" int1 backups more recent than any int2 or int3
|
||||
backup, in which case select int2 unless there are "$(cat
|
||||
intervals/int2)" int2 backups more recent than any int3 backups
|
||||
in which case choose int3.
|
||||
|
||||
This algorithm works well cycling through all the backups for my
|
||||
always connected machines as well as for my usually connected
|
||||
machines, and rarely connected machines. (For a rarely connected
|
||||
machine, interval names like "b_weekly" lose their English meaning
|
||||
but it still does a reasonable job of rotating through the
|
||||
intervals.)
|
||||
|
||||
In addition to being more robust, the automatic interval
|
||||
selection means that crontab is greatly simplified: only one line
|
||||
is needed. I use:
|
||||
|
||||
30 3 * * * ccollect.sh AUTO host1 host2 host3 | tee -a /var/log/ccollect-full.log | ccollect_analyse_logs.sh iwe
|
||||
|
||||
Some users might prefer a calendar-driven algorithm such as: do
|
||||
a yearly backup the first time a machine is connected during a new
|
||||
year; do a monthly backup the first that a machine is connected
|
||||
during a month; etc. This, however, would require a change to the
|
||||
ccollect configuration files. So, I didn't pursue the idea any
|
||||
further.
|
||||
|
||||
The code checks to see if the user specified the interval as
|
||||
AUTO. If so, the auto_interval function is called to select the
|
||||
interval:
|
||||
|
||||
347a417,420
|
||||
> if [ ${INTERVAL} = "AUTO" ] ; then
|
||||
> auto_interval
|
||||
> _techo "Selected interval: '$INTERVAL'"
|
||||
> fi
|
||||
|
||||
The code for auto_interval is as follows (note that it allows 'more
|
||||
recent' to be defined by either ctime or mtime as per the TSORT
|
||||
variable):
|
||||
|
||||
125a129,182
|
||||
> # Select interval if AUTO
|
||||
> #
|
||||
> # For this to work nicely, you have to choose interval names that sort nicely
|
||||
> # such as int1, int2, int3 or a_daily, b_weekly, c_monthly, etc.
|
||||
> #
|
||||
> auto_interval()
|
||||
> {
|
||||
> if [ -d "${backup}/intervals" -a -n "$(ls "${backup}/intervals" 2>/dev/null)" ] ; then
|
||||
> intervals_dir="${backup}/intervals"
|
||||
> elif [ -d "${CDEFAULTS}/intervals" -a -n "$(ls "${CDEFAULTS}/intervals" 2>/dev/null)" ] ; then
|
||||
> intervals_dir="${CDEFAULTS}/intervals"
|
||||
> else
|
||||
> _exit_err "No intervals are defined. Skipping."
|
||||
> fi
|
||||
> echo intervals_dir=${intervals_dir}
|
||||
>
|
||||
> trial_interval="$(ls -1r "${intervals_dir}/" | head -n 1)" || \
|
||||
> _exit_err "Failed to list contents of ${intervals_dir}/."
|
||||
> _techo "Considering interval ${trial_interval}"
|
||||
> most_recent="$(pcmd ls -${TSORT}p1 "${ddir}" | grep "^${trial_interval}.*/$" | head -n 1)" || \
|
||||
> _exit_err "Failed to list contents of ${ddir}/."
|
||||
> _techo " Most recent ${trial_interval}: '${most_recent}'"
|
||||
> if [ -n "${most_recent}" ]; then
|
||||
> no_intervals="$(ls -1 "${intervals_dir}/" | wc -l)"
|
||||
> n=1
|
||||
> while [ "${n}" -le "${no_intervals}" ]; do
|
||||
> trial_interval="$(ls -p1 "${intervals_dir}/" | tail -n+${n} | head -n 1)"
|
||||
> _techo "Considering interval '${trial_interval}'"
|
||||
> c_interval="$(cat "${intervals_dir}/${trial_interval}" 2>/dev/null)"
|
||||
> m=$((${n}+1))
|
||||
> set -- "${ddir}" -maxdepth 1
|
||||
> while [ "${m}" -le "${no_intervals}" ]; do
|
||||
> interval_m="$(ls -1 "${intervals_dir}/" | tail -n+${m} | head -n 1)"
|
||||
> most_recent="$(pcmd ls -${TSORT}p1 "${ddir}" | grep "^${interval_m}\..*/$" | head -n 1)"
|
||||
> _techo " Most recent ${interval_m}: '${most_recent}'"
|
||||
> if [ -n "${most_recent}" ] ; then
|
||||
> set -- "$@" -$NEWER "${ddir}/${most_recent}"
|
||||
> fi
|
||||
> m=$((${m}+1))
|
||||
> done
|
||||
> count=$(pcmd find "$@" -iname "${trial_interval}*" | wc -l)
|
||||
> _techo " Found $count more recent backups of ${trial_interval} (limit: ${c_interval})"
|
||||
> if [ "$count" -lt "${c_interval}" ] ; then
|
||||
> break
|
||||
> fi
|
||||
> n=$((${n}+1))
|
||||
> done
|
||||
> fi
|
||||
> export INTERVAL="${trial_interval}"
|
||||
> D_FILE_INTERVAL="${intervals_dir}/${INTERVAL}"
|
||||
> D_INTERVAL=$(cat "${D_FILE_INTERVAL}" 2>/dev/null)
|
||||
> }
|
||||
>
|
||||
> #
|
||||
|
||||
While I consider the auto_interval code to be developmental, I have
|
||||
been using it for my nightly backups and it works for me.
|
||||
|
||||
One last change: For auto_interval to work, it needs "ddir" to
|
||||
be defined first. Consequently, I had to move the following code
|
||||
so it gets run before auto_interval is called:
|
||||
|
||||
369,380c442,443
|
||||
<
|
||||
< #
|
||||
< # Destination is a path
|
||||
< #
|
||||
< if [ ! -f "${c_dest}" ]; then
|
||||
< _exit_err "Destination ${c_dest} is not a file. Skipping."
|
||||
< else
|
||||
< ddir=$(cat "${c_dest}"); ret="$?"
|
||||
< if [ "${ret}" -ne 0 ]; then
|
||||
< _exit_err "Destination ${c_dest} is not readable. Skipping."
|
||||
< fi
|
||||
< fi
|
||||
345a403,414
|
||||
> # Destination is a path
|
||||
> #
|
||||
> if [ ! -f "${c_dest}" ]; then
|
||||
> _exit_err "Destination ${c_dest} is not a file. Skipping."
|
||||
> else
|
||||
> ddir=$(cat "${c_dest}"); ret="$?"
|
||||
> if [ "${ret}" -ne 0 ]; then
|
||||
> _exit_err "Destination ${c_dest} is not readable. Skipping."
|
||||
> fi
|
||||
> fi
|
||||
>
|
||||
> #
|
||||
|
||||
I have some other ideas but this is all I have implemented at
|
||||
the moment. Files are attached.
|
||||
|
||||
Thanks again for developing ccollect and let me know what you
|
||||
think.
|
||||
|
||||
Regards,
|
||||
|
||||
John
|
||||
|
||||
--
|
||||
John L. Lawless, Ph.D.
|
||||
Redwood Scientific, Inc.
|
||||
1005 Terra Nova Blvd
|
||||
Pacifica, CA 94044-4300
|
||||
1-650-738-8083
|
||||
|
15
contrib/jlawless-2009-06-03/old/a.patch
Normal file
15
contrib/jlawless-2009-06-03/old/a.patch
Normal file
|
@ -0,0 +1,15 @@
|
|||
--- ccollect-0.7.1.sh 2009-02-02 03:39:42.000000000 -0800
|
||||
+++ ccollect-0.7.1-a.sh 2009-05-24 21:30:38.000000000 -0700
|
||||
@@ -364,10 +364,12 @@
|
||||
source=$(cat "${c_source}"); ret="$?"
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "Source ${c_source} is not readable. Skipping."
|
||||
fi
|
||||
fi
|
||||
+ # Verify source is up and accepting connections before deleting any old backups
|
||||
+ rsync "$source" >/dev/null || _exit_err "Source ${source} is not readable. Skipping."
|
||||
|
||||
#
|
||||
# Destination is a path
|
||||
#
|
||||
if [ ! -f "${c_dest}" ]; then
|
15
contrib/jlawless-2009-06-03/old/b.patch
Normal file
15
contrib/jlawless-2009-06-03/old/b.patch
Normal file
|
@ -0,0 +1,15 @@
|
|||
--- ccollect-0.7.1-a.sh 2009-05-24 21:30:38.000000000 -0700
|
||||
+++ ccollect-0.7.1-b.sh 2009-05-24 21:32:00.000000000 -0700
|
||||
@@ -551,10 +551,12 @@
|
||||
# the rsync part
|
||||
#
|
||||
|
||||
_techo "Transferring files..."
|
||||
rsync "$@" "${source}" "${destination_full}"; ret=$?
|
||||
+ # Correct the modification time:
|
||||
+ pcmd touch "${destination_dir}"
|
||||
|
||||
#
|
||||
# remove marking here
|
||||
#
|
||||
pcmd rm "${destination_dir}.${c_marker}" || \
|
35
contrib/jlawless-2009-06-03/old/c.patch
Normal file
35
contrib/jlawless-2009-06-03/old/c.patch
Normal file
|
@ -0,0 +1,35 @@
|
|||
--- ccollect-0.7.1-b.sh 2009-05-24 21:32:00.000000000 -0700
|
||||
+++ ccollect-0.7.1-c.sh 2009-05-24 21:39:43.000000000 -0700
|
||||
@@ -40,10 +40,13 @@
|
||||
VERSION=0.7.1
|
||||
RELEASE="2009-02-02"
|
||||
HALF_VERSION="ccollect ${VERSION}"
|
||||
FULL_VERSION="ccollect ${VERSION} (${RELEASE})"
|
||||
|
||||
+#TSORT="tc" ; NEWER="cnewer"
|
||||
+TSORT="t" ; NEWER="newer"
|
||||
+
|
||||
#
|
||||
# CDATE: how we use it for naming of the archives
|
||||
# DDATE: how the user should see it in our output (DISPLAY)
|
||||
#
|
||||
CDATE="date +%Y%m%d-%H%M"
|
||||
@@ -513,14 +516,14 @@
|
||||
|
||||
|
||||
#
|
||||
# Check for backup directory to clone from: Always clone from the latest one!
|
||||
#
|
||||
- # Use ls -1c instead of -1t, because last modification maybe the same on all
|
||||
- # and metadate update (-c) is updated by rsync locally.
|
||||
- #
|
||||
- last_dir="$(pcmd ls -tcp1 "${ddir}" | grep '/$' | head -n 1)" || \
|
||||
+ # Depending on your file system, you may want to sort on:
|
||||
+ # 1. mtime (modification time) with TSORT=t, or
|
||||
+ # 2. ctime (last change time, usually) with TSORT=tc
|
||||
+ last_dir="$(pcmd ls -${TSORT}p1 "${ddir}" | grep '/$' | head -n 1)" || \
|
||||
_exit_err "Failed to list contents of ${ddir}."
|
||||
|
||||
#
|
||||
# clone from old backup, if existing
|
||||
#
|
615
contrib/jlawless-2009-06-03/old/ccollect-0.7.1.sh
Executable file
615
contrib/jlawless-2009-06-03/old/ccollect-0.7.1.sh
Executable file
|
@ -0,0 +1,615 @@
|
|||
#!/bin/sh
|
||||
#
|
||||
# 2005-2009 Nico Schottelius (nico-ccollect at schottelius.org)
|
||||
#
|
||||
# This file is part of ccollect.
|
||||
#
|
||||
# ccollect 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 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# ccollect 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.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with ccollect. If not, see <http://www.gnu.org/licenses/>.
|
||||
#
|
||||
# Initially written for SyGroup (www.sygroup.ch)
|
||||
# Date: Mon Nov 14 11:45:11 CET 2005
|
||||
|
||||
#
|
||||
# Standard variables (stolen from cconf)
|
||||
#
|
||||
__pwd="$(pwd -P)"
|
||||
__mydir="${0%/*}"; __abs_mydir="$(cd "$__mydir" && pwd -P)"
|
||||
__myname=${0##*/}; __abs_myname="$__abs_mydir/$__myname"
|
||||
|
||||
#
|
||||
# where to find our configuration and temporary file
|
||||
#
|
||||
CCOLLECT_CONF=${CCOLLECT_CONF:-/etc/ccollect}
|
||||
CSOURCES=${CCOLLECT_CONF}/sources
|
||||
CDEFAULTS=${CCOLLECT_CONF}/defaults
|
||||
CPREEXEC="${CDEFAULTS}/pre_exec"
|
||||
CPOSTEXEC="${CDEFAULTS}/post_exec"
|
||||
|
||||
TMP=$(mktemp "/tmp/${__myname}.XXXXXX")
|
||||
VERSION=0.7.1
|
||||
RELEASE="2009-02-02"
|
||||
HALF_VERSION="ccollect ${VERSION}"
|
||||
FULL_VERSION="ccollect ${VERSION} (${RELEASE})"
|
||||
|
||||
#
|
||||
# CDATE: how we use it for naming of the archives
|
||||
# DDATE: how the user should see it in our output (DISPLAY)
|
||||
#
|
||||
CDATE="date +%Y%m%d-%H%M"
|
||||
DDATE="date +%Y-%m-%d-%H:%M:%S"
|
||||
|
||||
#
|
||||
# unset parallel execution
|
||||
#
|
||||
PARALLEL=""
|
||||
|
||||
#
|
||||
# catch signals
|
||||
#
|
||||
trap "rm -f \"${TMP}\"" 1 2 15
|
||||
|
||||
#
|
||||
# Functions
|
||||
#
|
||||
|
||||
# time displaying echo
|
||||
_techo()
|
||||
{
|
||||
echo "$(${DDATE}): $@"
|
||||
}
|
||||
|
||||
# exit on error
|
||||
_exit_err()
|
||||
{
|
||||
_techo "$@"
|
||||
rm -f "${TMP}"
|
||||
exit 1
|
||||
}
|
||||
|
||||
add_name()
|
||||
{
|
||||
awk "{ print \"[${name}] \" \$0 }"
|
||||
}
|
||||
|
||||
pcmd()
|
||||
{
|
||||
if [ "$remote_host" ]; then
|
||||
ssh "$remote_host" "$@"
|
||||
else
|
||||
"$@"
|
||||
fi
|
||||
}
|
||||
|
||||
#
|
||||
# Version
|
||||
#
|
||||
display_version()
|
||||
{
|
||||
echo "${FULL_VERSION}"
|
||||
exit 0
|
||||
}
|
||||
|
||||
#
|
||||
# Tell how to use us
|
||||
#
|
||||
usage()
|
||||
{
|
||||
echo "${__myname}: <interval name> [args] <sources to backup>"
|
||||
echo ""
|
||||
echo " ccollect creates (pseudo) incremental backups"
|
||||
echo ""
|
||||
echo " -h, --help: Show this help screen"
|
||||
echo " -p, --parallel: Parallelise backup processes"
|
||||
echo " -a, --all: Backup all sources specified in ${CSOURCES}"
|
||||
echo " -v, --verbose: Be very verbose (uses set -x)"
|
||||
echo " -V, --version: Print version information"
|
||||
echo ""
|
||||
echo " This is version ${VERSION}, released on ${RELEASE}"
|
||||
echo " (the first version was written on 2005-12-05 by Nico Schottelius)."
|
||||
echo ""
|
||||
echo " Retrieve latest ccollect at http://unix.schottelius.org/ccollect/"
|
||||
exit 0
|
||||
}
|
||||
|
||||
#
|
||||
# need at least interval and one source or --all
|
||||
#
|
||||
if [ $# -lt 2 ]; then
|
||||
if [ "$1" = "-V" -o "$1" = "--version" ]; then
|
||||
display_version
|
||||
else
|
||||
usage
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# check for configuraton directory
|
||||
#
|
||||
[ -d "${CCOLLECT_CONF}" ] || _exit_err "No configuration found in " \
|
||||
"\"${CCOLLECT_CONF}\" (is \$CCOLLECT_CONF properly set?)"
|
||||
|
||||
#
|
||||
# Filter arguments
|
||||
#
|
||||
export INTERVAL="$1"; shift
|
||||
i=1
|
||||
no_sources=0
|
||||
|
||||
#
|
||||
# Create source "array"
|
||||
#
|
||||
while [ "$#" -ge 1 ]; do
|
||||
eval arg=\"\$1\"; shift
|
||||
|
||||
if [ "${NO_MORE_ARGS}" = 1 ]; then
|
||||
eval source_${no_sources}=\"${arg}\"
|
||||
no_sources=$((${no_sources}+1))
|
||||
|
||||
# make variable available for subscripts
|
||||
eval export source_${no_sources}
|
||||
else
|
||||
case "${arg}" in
|
||||
-a|--all)
|
||||
ALL=1
|
||||
;;
|
||||
-v|--verbose)
|
||||
VERBOSE=1
|
||||
;;
|
||||
-p|--parallel)
|
||||
PARALLEL=1
|
||||
;;
|
||||
-h|--help)
|
||||
usage
|
||||
;;
|
||||
--)
|
||||
NO_MORE_ARGS=1
|
||||
;;
|
||||
*)
|
||||
eval source_${no_sources}=\"$arg\"
|
||||
no_sources=$(($no_sources+1))
|
||||
;;
|
||||
esac
|
||||
fi
|
||||
|
||||
i=$(($i+1))
|
||||
done
|
||||
|
||||
# also export number of sources
|
||||
export no_sources
|
||||
|
||||
#
|
||||
# be really, really, really verbose
|
||||
#
|
||||
if [ "${VERBOSE}" = 1 ]; then
|
||||
set -x
|
||||
fi
|
||||
|
||||
#
|
||||
# Look, if we should take ALL sources
|
||||
#
|
||||
if [ "${ALL}" = 1 ]; then
|
||||
# reset everything specified before
|
||||
no_sources=0
|
||||
|
||||
#
|
||||
# get entries from sources
|
||||
#
|
||||
cwd=$(pwd -P)
|
||||
( cd "${CSOURCES}" && ls > "${TMP}" ); ret=$?
|
||||
|
||||
[ "${ret}" -eq 0 ] || _exit_err "Listing of sources failed. Aborting."
|
||||
|
||||
while read tmp; do
|
||||
eval source_${no_sources}=\"${tmp}\"
|
||||
no_sources=$((${no_sources}+1))
|
||||
done < "${TMP}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Need at least ONE source to backup
|
||||
#
|
||||
if [ "${no_sources}" -lt 1 ]; then
|
||||
usage
|
||||
else
|
||||
_techo "${HALF_VERSION}: Beginning backup using interval ${INTERVAL}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Look for pre-exec command (general)
|
||||
#
|
||||
if [ -x "${CPREEXEC}" ]; then
|
||||
_techo "Executing ${CPREEXEC} ..."
|
||||
"${CPREEXEC}"; ret=$?
|
||||
_techo "Finished ${CPREEXEC} (return code: ${ret})."
|
||||
|
||||
[ "${ret}" -eq 0 ] || _exit_err "${CPREEXEC} failed. Aborting"
|
||||
fi
|
||||
|
||||
#
|
||||
# check default configuration
|
||||
#
|
||||
|
||||
D_FILE_INTERVAL="${CDEFAULTS}/intervals/${INTERVAL}"
|
||||
D_INTERVAL=$(cat "${D_FILE_INTERVAL}" 2>/dev/null)
|
||||
|
||||
|
||||
#
|
||||
# Let's do the backup
|
||||
#
|
||||
i=0
|
||||
while [ "${i}" -lt "${no_sources}" ]; do
|
||||
|
||||
#
|
||||
# Get current source
|
||||
#
|
||||
eval name=\"\$source_${i}\"
|
||||
i=$((${i}+1))
|
||||
|
||||
export name
|
||||
|
||||
#
|
||||
# start ourself, if we want parallel execution
|
||||
#
|
||||
if [ "${PARALLEL}" ]; then
|
||||
"$0" "${INTERVAL}" "${name}" &
|
||||
continue
|
||||
fi
|
||||
|
||||
#
|
||||
# Start subshell for easy log editing
|
||||
#
|
||||
(
|
||||
#
|
||||
# Stderr to stdout, so we can produce nice logs
|
||||
#
|
||||
exec 2>&1
|
||||
|
||||
#
|
||||
# Configuration
|
||||
#
|
||||
backup="${CSOURCES}/${name}"
|
||||
c_source="${backup}/source"
|
||||
c_dest="${backup}/destination"
|
||||
c_exclude="${backup}/exclude"
|
||||
c_verbose="${backup}/verbose"
|
||||
c_vverbose="${backup}/very_verbose"
|
||||
c_rsync_extra="${backup}/rsync_options"
|
||||
c_summary="${backup}/summary"
|
||||
c_pre_exec="${backup}/pre_exec"
|
||||
c_post_exec="${backup}/post_exec"
|
||||
f_incomplete="delete_incomplete"
|
||||
c_incomplete="${backup}/${f_incomplete}"
|
||||
c_remote_host="${backup}/remote_host"
|
||||
|
||||
#
|
||||
# Marking backups: If we abort it's not removed => Backup is broken
|
||||
#
|
||||
c_marker=".ccollect-marker"
|
||||
|
||||
#
|
||||
# Times
|
||||
#
|
||||
begin_s=$(date +%s)
|
||||
|
||||
#
|
||||
# unset possible options
|
||||
#
|
||||
EXCLUDE=""
|
||||
RSYNC_EXTRA=""
|
||||
SUMMARY=""
|
||||
VERBOSE=""
|
||||
VVERBOSE=""
|
||||
DELETE_INCOMPLETE=""
|
||||
|
||||
_techo "Beginning to backup"
|
||||
|
||||
#
|
||||
# Standard configuration checks
|
||||
#
|
||||
if [ ! -e "${backup}" ]; then
|
||||
_exit_err "Source does not exist."
|
||||
fi
|
||||
|
||||
#
|
||||
# configuration _must_ be a directory
|
||||
#
|
||||
if [ ! -d "${backup}" ]; then
|
||||
_exit_err "\"${name}\" is not a cconfig-directory. Skipping."
|
||||
fi
|
||||
|
||||
#
|
||||
# first execute pre_exec, which may generate destination or other
|
||||
# parameters
|
||||
#
|
||||
if [ -x "${c_pre_exec}" ]; then
|
||||
_techo "Executing ${c_pre_exec} ..."
|
||||
"${c_pre_exec}"; ret="$?"
|
||||
_techo "Finished ${c_pre_exec} (return code ${ret})."
|
||||
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "${c_pre_exec} failed. Skipping."
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# interval definition: First try source specific, fallback to default
|
||||
#
|
||||
c_interval="$(cat "${backup}/intervals/${INTERVAL}" 2>/dev/null)"
|
||||
|
||||
if [ -z "${c_interval}" ]; then
|
||||
c_interval="${D_INTERVAL}"
|
||||
|
||||
if [ -z "${c_interval}" ]; then
|
||||
_exit_err "No definition for interval \"${INTERVAL}\" found. Skipping."
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# Source checks
|
||||
#
|
||||
if [ ! -f "${c_source}" ]; then
|
||||
_exit_err "Source description \"${c_source}\" is not a file. Skipping."
|
||||
else
|
||||
source=$(cat "${c_source}"); ret="$?"
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "Source ${c_source} is not readable. Skipping."
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# Destination is a path
|
||||
#
|
||||
if [ ! -f "${c_dest}" ]; then
|
||||
_exit_err "Destination ${c_dest} is not a file. Skipping."
|
||||
else
|
||||
ddir=$(cat "${c_dest}"); ret="$?"
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "Destination ${c_dest} is not readable. Skipping."
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# do we backup to a remote host? then set pre-cmd
|
||||
#
|
||||
if [ -f "${c_remote_host}" ]; then
|
||||
# adjust ls and co
|
||||
remote_host=$(cat "${c_remote_host}"); ret="$?"
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "Remote host file ${c_remote_host} exists, but is not readable. Skipping."
|
||||
fi
|
||||
destination="${remote_host}:${ddir}"
|
||||
else
|
||||
remote_host=""
|
||||
destination="${ddir}"
|
||||
fi
|
||||
export remote_host
|
||||
|
||||
#
|
||||
# check for existence / use real name
|
||||
#
|
||||
( pcmd cd "$ddir" ) || _exit_err "Cannot change to ${ddir}. Skipping."
|
||||
|
||||
|
||||
#
|
||||
# Check whether to delete incomplete backups
|
||||
#
|
||||
if [ -f "${c_incomplete}" -o -f "${CDEFAULTS}/${f_incomplete}" ]; then
|
||||
DELETE_INCOMPLETE="yes"
|
||||
fi
|
||||
|
||||
# NEW method as of 0.6:
|
||||
# - insert ccollect default parameters
|
||||
# - insert options
|
||||
# - insert user options
|
||||
|
||||
#
|
||||
# rsync standard options
|
||||
#
|
||||
|
||||
set -- "$@" "--archive" "--delete" "--numeric-ids" "--relative" \
|
||||
"--delete-excluded" "--sparse"
|
||||
|
||||
#
|
||||
# exclude list
|
||||
#
|
||||
if [ -f "${c_exclude}" ]; then
|
||||
set -- "$@" "--exclude-from=${c_exclude}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Output a summary
|
||||
#
|
||||
if [ -f "${c_summary}" ]; then
|
||||
set -- "$@" "--stats"
|
||||
fi
|
||||
|
||||
#
|
||||
# Verbosity for rsync
|
||||
#
|
||||
if [ -f "${c_vverbose}" ]; then
|
||||
set -- "$@" "-vv"
|
||||
elif [ -f "${c_verbose}" ]; then
|
||||
set -- "$@" "-v"
|
||||
fi
|
||||
|
||||
#
|
||||
# extra options for rsync provided by the user
|
||||
#
|
||||
if [ -f "${c_rsync_extra}" ]; then
|
||||
while read line; do
|
||||
set -- "$@" "$line"
|
||||
done < "${c_rsync_extra}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Check for incomplete backups
|
||||
#
|
||||
pcmd ls -1 "$ddir/${INTERVAL}"*".${c_marker}" > "${TMP}" 2>/dev/null
|
||||
|
||||
i=0
|
||||
while read incomplete; do
|
||||
eval incomplete_$i=\"$(echo ${incomplete} | sed "s/\\.${c_marker}\$//")\"
|
||||
i=$(($i+1))
|
||||
done < "${TMP}"
|
||||
|
||||
j=0
|
||||
while [ "$j" -lt "$i" ]; do
|
||||
eval realincomplete=\"\$incomplete_$j\"
|
||||
_techo "Incomplete backup: ${realincomplete}"
|
||||
if [ "${DELETE_INCOMPLETE}" = "yes" ]; then
|
||||
_techo "Deleting ${realincomplete} ..."
|
||||
pcmd rm $VVERBOSE -rf "${ddir}/${realincomplete}" || \
|
||||
_exit_err "Removing ${realincomplete} failed."
|
||||
fi
|
||||
j=$(($j+1))
|
||||
done
|
||||
|
||||
#
|
||||
# check if maximum number of backups is reached, if so remove
|
||||
# use grep and ls -p so we only look at directories
|
||||
#
|
||||
count="$(pcmd ls -p1 "${ddir}" | grep "^${INTERVAL}\..*/\$" | wc -l \
|
||||
| sed 's/^ *//g')" || _exit_err "Counting backups failed"
|
||||
|
||||
_techo "Existing backups: ${count} Total keeping backups: ${c_interval}"
|
||||
|
||||
if [ "${count}" -ge "${c_interval}" ]; then
|
||||
substract=$((${c_interval} - 1))
|
||||
remove=$((${count} - ${substract}))
|
||||
_techo "Removing ${remove} backup(s)..."
|
||||
|
||||
pcmd ls -p1 "$ddir" | grep "^${INTERVAL}\..*/\$" | \
|
||||
sort -n | head -n "${remove}" > "${TMP}" || \
|
||||
_exit_err "Listing old backups failed"
|
||||
|
||||
i=0
|
||||
while read to_remove; do
|
||||
eval remove_$i=\"${to_remove}\"
|
||||
i=$(($i+1))
|
||||
done < "${TMP}"
|
||||
|
||||
j=0
|
||||
while [ "$j" -lt "$i" ]; do
|
||||
eval to_remove=\"\$remove_$j\"
|
||||
_techo "Removing ${to_remove} ..."
|
||||
pcmd rm ${VVERBOSE} -rf "${ddir}/${to_remove}" || \
|
||||
_exit_err "Removing ${to_remove} failed."
|
||||
j=$(($j+1))
|
||||
done
|
||||
fi
|
||||
|
||||
|
||||
#
|
||||
# Check for backup directory to clone from: Always clone from the latest one!
|
||||
#
|
||||
# Use ls -1c instead of -1t, because last modification maybe the same on all
|
||||
# and metadate update (-c) is updated by rsync locally.
|
||||
#
|
||||
last_dir="$(pcmd ls -tcp1 "${ddir}" | grep '/$' | head -n 1)" || \
|
||||
_exit_err "Failed to list contents of ${ddir}."
|
||||
|
||||
#
|
||||
# clone from old backup, if existing
|
||||
#
|
||||
if [ "${last_dir}" ]; then
|
||||
set -- "$@" "--link-dest=${ddir}/${last_dir}"
|
||||
_techo "Hard linking from ${last_dir}"
|
||||
fi
|
||||
|
||||
|
||||
# set time when we really begin to backup, not when we began to remove above
|
||||
destination_date=$(${CDATE})
|
||||
destination_dir="${ddir}/${INTERVAL}.${destination_date}.$$"
|
||||
destination_full="${destination}/${INTERVAL}.${destination_date}.$$"
|
||||
|
||||
# give some info
|
||||
_techo "Beginning to backup, this may take some time..."
|
||||
|
||||
_techo "Creating ${destination_dir} ..."
|
||||
pcmd mkdir ${VVERBOSE} "${destination_dir}" || \
|
||||
_exit_err "Creating ${destination_dir} failed. Skipping."
|
||||
|
||||
#
|
||||
# added marking in 0.6 (and remove it, if successful later)
|
||||
#
|
||||
pcmd touch "${destination_dir}.${c_marker}"
|
||||
|
||||
#
|
||||
# the rsync part
|
||||
#
|
||||
|
||||
_techo "Transferring files..."
|
||||
rsync "$@" "${source}" "${destination_full}"; ret=$?
|
||||
|
||||
#
|
||||
# remove marking here
|
||||
#
|
||||
pcmd rm "${destination_dir}.${c_marker}" || \
|
||||
_exit_err "Removing ${destination_dir}/${c_marker} failed."
|
||||
|
||||
_techo "Finished backup (rsync return code: $ret)."
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_techo "Warning: rsync exited non-zero, the backup may be broken (see rsync errors)."
|
||||
fi
|
||||
|
||||
#
|
||||
# post_exec
|
||||
#
|
||||
if [ -x "${c_post_exec}" ]; then
|
||||
_techo "Executing ${c_post_exec} ..."
|
||||
"${c_post_exec}"; ret=$?
|
||||
_techo "Finished ${c_post_exec}."
|
||||
|
||||
if [ ${ret} -ne 0 ]; then
|
||||
_exit_err "${c_post_exec} failed."
|
||||
fi
|
||||
fi
|
||||
|
||||
# Calculation
|
||||
end_s=$(date +%s)
|
||||
|
||||
full_seconds=$((${end_s} - ${begin_s}))
|
||||
hours=$((${full_seconds} / 3600))
|
||||
seconds=$((${full_seconds} - (${hours} * 3600)))
|
||||
minutes=$((${seconds} / 60))
|
||||
seconds=$((${seconds} - (${minutes} * 60)))
|
||||
|
||||
_techo "Backup lasted: ${hours}:${minutes}:${seconds} (h:m:s)"
|
||||
|
||||
) | add_name
|
||||
done
|
||||
|
||||
#
|
||||
# Be a good parent and wait for our children, if they are running wild parallel
|
||||
#
|
||||
if [ "${PARALLEL}" ]; then
|
||||
_techo "Waiting for children to complete..."
|
||||
wait
|
||||
fi
|
||||
|
||||
#
|
||||
# Look for post-exec command (general)
|
||||
#
|
||||
if [ -x "${CPOSTEXEC}" ]; then
|
||||
_techo "Executing ${CPOSTEXEC} ..."
|
||||
"${CPOSTEXEC}"; ret=$?
|
||||
_techo "Finished ${CPOSTEXEC} (return code: ${ret})."
|
||||
|
||||
if [ ${ret} -ne 0 ]; then
|
||||
_techo "${CPOSTEXEC} failed."
|
||||
fi
|
||||
fi
|
||||
|
||||
rm -f "${TMP}"
|
||||
_techo "Finished ${WE}"
|
683
contrib/jlawless-2009-06-03/old/ccollect-f.sh
Executable file
683
contrib/jlawless-2009-06-03/old/ccollect-f.sh
Executable file
|
@ -0,0 +1,683 @@
|
|||
#!/bin/sh
|
||||
#
|
||||
# 2005-2009 Nico Schottelius (nico-ccollect at schottelius.org)
|
||||
#
|
||||
# This file is part of ccollect.
|
||||
#
|
||||
# ccollect 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 3 of the License, or
|
||||
# (at your option) any later version.
|
||||
#
|
||||
# ccollect 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.
|
||||
#
|
||||
# You should have received a copy of the GNU General Public License
|
||||
# along with ccollect. If not, see <http://www.gnu.org/licenses/>.
|
||||
#
|
||||
# Initially written for SyGroup (www.sygroup.ch)
|
||||
# Date: Mon Nov 14 11:45:11 CET 2005
|
||||
|
||||
#
|
||||
# Standard variables (stolen from cconf)
|
||||
#
|
||||
__pwd="$(pwd -P)"
|
||||
__mydir="${0%/*}"; __abs_mydir="$(cd "$__mydir" && pwd -P)"
|
||||
__myname=${0##*/}; __abs_myname="$__abs_mydir/$__myname"
|
||||
|
||||
#
|
||||
# where to find our configuration and temporary file
|
||||
#
|
||||
CCOLLECT_CONF=${CCOLLECT_CONF:-/etc/ccollect}
|
||||
CSOURCES=${CCOLLECT_CONF}/sources
|
||||
CDEFAULTS=${CCOLLECT_CONF}/defaults
|
||||
CPREEXEC="${CDEFAULTS}/pre_exec"
|
||||
CPOSTEXEC="${CDEFAULTS}/post_exec"
|
||||
|
||||
TMP=$(mktemp "/tmp/${__myname}.XXXXXX")
|
||||
VERSION=0.7.1
|
||||
RELEASE="2009-02-02"
|
||||
HALF_VERSION="ccollect ${VERSION}"
|
||||
FULL_VERSION="ccollect ${VERSION} (${RELEASE})"
|
||||
|
||||
#TSORT="tc" ; NEWER="cnewer"
|
||||
TSORT="t" ; NEWER="newer"
|
||||
|
||||
#
|
||||
# CDATE: how we use it for naming of the archives
|
||||
# DDATE: how the user should see it in our output (DISPLAY)
|
||||
#
|
||||
CDATE="date +%Y%m%d-%H%M"
|
||||
DDATE="date +%Y-%m-%d-%H:%M:%S"
|
||||
|
||||
#
|
||||
# unset parallel execution
|
||||
#
|
||||
PARALLEL=""
|
||||
|
||||
#
|
||||
# catch signals
|
||||
#
|
||||
trap "rm -f \"${TMP}\"" 1 2 15
|
||||
|
||||
#
|
||||
# Functions
|
||||
#
|
||||
|
||||
# time displaying echo
|
||||
_techo()
|
||||
{
|
||||
echo "$(${DDATE}): $@"
|
||||
}
|
||||
|
||||
# exit on error
|
||||
_exit_err()
|
||||
{
|
||||
_techo "$@"
|
||||
rm -f "${TMP}"
|
||||
exit 1
|
||||
}
|
||||
|
||||
add_name()
|
||||
{
|
||||
awk "{ print \"[${name}] \" \$0 }"
|
||||
}
|
||||
|
||||
pcmd()
|
||||
{
|
||||
if [ "$remote_host" ]; then
|
||||
ssh "$remote_host" "$@"
|
||||
else
|
||||
"$@"
|
||||
fi
|
||||
}
|
||||
|
||||
#
|
||||
# Version
|
||||
#
|
||||
display_version()
|
||||
{
|
||||
echo "${FULL_VERSION}"
|
||||
exit 0
|
||||
}
|
||||
|
||||
#
|
||||
# Tell how to use us
|
||||
#
|
||||
usage()
|
||||
{
|
||||
echo "${__myname}: <interval name> [args] <sources to backup>"
|
||||
echo ""
|
||||
echo " ccollect creates (pseudo) incremental backups"
|
||||
echo ""
|
||||
echo " -h, --help: Show this help screen"
|
||||
echo " -p, --parallel: Parallelise backup processes"
|
||||
echo " -a, --all: Backup all sources specified in ${CSOURCES}"
|
||||
echo " -v, --verbose: Be very verbose (uses set -x)"
|
||||
echo " -V, --version: Print version information"
|
||||
echo ""
|
||||
echo " This is version ${VERSION}, released on ${RELEASE}"
|
||||
echo " (the first version was written on 2005-12-05 by Nico Schottelius)."
|
||||
echo ""
|
||||
echo " Retrieve latest ccollect at http://unix.schottelius.org/ccollect/"
|
||||
exit 0
|
||||
}
|
||||
|
||||
#
|
||||
# Select interval if AUTO
|
||||
#
|
||||
# For this to work nicely, you have to choose interval names that sort nicely
|
||||
# such as int1, int2, int3 or a_daily, b_weekly, c_monthly, etc.
|
||||
#
|
||||
auto_interval()
|
||||
{
|
||||
if [ -d "${backup}/intervals" -a -n "$(ls "${backup}/intervals" 2>/dev/null)" ] ; then
|
||||
intervals_dir="${backup}/intervals"
|
||||
elif [ -d "${CDEFAULTS}/intervals" -a -n "$(ls "${CDEFAULTS}/intervals" 2>/dev/null)" ] ; then
|
||||
intervals_dir="${CDEFAULTS}/intervals"
|
||||
else
|
||||
_exit_err "No intervals are defined. Skipping."
|
||||
fi
|
||||
echo intervals_dir=${intervals_dir}
|
||||
|
||||
trial_interval="$(ls -1r "${intervals_dir}/" | head -n 1)" || \
|
||||
_exit_err "Failed to list contents of ${intervals_dir}/."
|
||||
_techo "Considering interval ${trial_interval}"
|
||||
most_recent="$(pcmd ls -${TSORT}p1 "${ddir}" | grep "^${trial_interval}.*/$" | head -n 1)" || \
|
||||
_exit_err "Failed to list contents of ${ddir}/."
|
||||
_techo " Most recent ${trial_interval}: '${most_recent}'"
|
||||
if [ -n "${most_recent}" ]; then
|
||||
no_intervals="$(ls -1 "${intervals_dir}/" | wc -l)"
|
||||
n=1
|
||||
while [ "${n}" -le "${no_intervals}" ]; do
|
||||
trial_interval="$(ls -p1 "${intervals_dir}/" | tail -n+${n} | head -n 1)"
|
||||
_techo "Considering interval '${trial_interval}'"
|
||||
c_interval="$(cat "${intervals_dir}/${trial_interval}" 2>/dev/null)"
|
||||
m=$((${n}+1))
|
||||
set -- "${ddir}" -maxdepth 1
|
||||
while [ "${m}" -le "${no_intervals}" ]; do
|
||||
interval_m="$(ls -1 "${intervals_dir}/" | tail -n+${m} | head -n 1)"
|
||||
most_recent="$(pcmd ls -${TSORT}p1 "${ddir}" | grep "^${interval_m}\..*/$" | head -n 1)"
|
||||
_techo " Most recent ${interval_m}: '${most_recent}'"
|
||||
if [ -n "${most_recent}" ] ; then
|
||||
set -- "$@" -$NEWER "${ddir}/${most_recent}"
|
||||
fi
|
||||
m=$((${m}+1))
|
||||
done
|
||||
count=$(pcmd find "$@" -iname "${trial_interval}*" | wc -l)
|
||||
_techo " Found $count more recent backups of ${trial_interval} (limit: ${c_interval})"
|
||||
if [ "$count" -lt "${c_interval}" ] ; then
|
||||
break
|
||||
fi
|
||||
n=$((${n}+1))
|
||||
done
|
||||
fi
|
||||
export INTERVAL="${trial_interval}"
|
||||
D_FILE_INTERVAL="${intervals_dir}/${INTERVAL}"
|
||||
D_INTERVAL=$(cat "${D_FILE_INTERVAL}" 2>/dev/null)
|
||||
}
|
||||
|
||||
#
|
||||
# need at least interval and one source or --all
|
||||
#
|
||||
if [ $# -lt 2 ]; then
|
||||
if [ "$1" = "-V" -o "$1" = "--version" ]; then
|
||||
display_version
|
||||
else
|
||||
usage
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# check for configuraton directory
|
||||
#
|
||||
[ -d "${CCOLLECT_CONF}" ] || _exit_err "No configuration found in " \
|
||||
"\"${CCOLLECT_CONF}\" (is \$CCOLLECT_CONF properly set?)"
|
||||
|
||||
#
|
||||
# Filter arguments
|
||||
#
|
||||
export INTERVAL="$1"; shift
|
||||
i=1
|
||||
no_sources=0
|
||||
|
||||
#
|
||||
# Create source "array"
|
||||
#
|
||||
while [ "$#" -ge 1 ]; do
|
||||
eval arg=\"\$1\"; shift
|
||||
|
||||
if [ "${NO_MORE_ARGS}" = 1 ]; then
|
||||
eval source_${no_sources}=\"${arg}\"
|
||||
no_sources=$((${no_sources}+1))
|
||||
|
||||
# make variable available for subscripts
|
||||
eval export source_${no_sources}
|
||||
else
|
||||
case "${arg}" in
|
||||
-a|--all)
|
||||
ALL=1
|
||||
;;
|
||||
-v|--verbose)
|
||||
VERBOSE=1
|
||||
;;
|
||||
-p|--parallel)
|
||||
PARALLEL=1
|
||||
;;
|
||||
-h|--help)
|
||||
usage
|
||||
;;
|
||||
--)
|
||||
NO_MORE_ARGS=1
|
||||
;;
|
||||
*)
|
||||
eval source_${no_sources}=\"$arg\"
|
||||
no_sources=$(($no_sources+1))
|
||||
;;
|
||||
esac
|
||||
fi
|
||||
|
||||
i=$(($i+1))
|
||||
done
|
||||
|
||||
# also export number of sources
|
||||
export no_sources
|
||||
|
||||
#
|
||||
# be really, really, really verbose
|
||||
#
|
||||
if [ "${VERBOSE}" = 1 ]; then
|
||||
set -x
|
||||
fi
|
||||
|
||||
#
|
||||
# Look, if we should take ALL sources
|
||||
#
|
||||
if [ "${ALL}" = 1 ]; then
|
||||
# reset everything specified before
|
||||
no_sources=0
|
||||
|
||||
#
|
||||
# get entries from sources
|
||||
#
|
||||
cwd=$(pwd -P)
|
||||
( cd "${CSOURCES}" && ls > "${TMP}" ); ret=$?
|
||||
|
||||
[ "${ret}" -eq 0 ] || _exit_err "Listing of sources failed. Aborting."
|
||||
|
||||
while read tmp; do
|
||||
eval source_${no_sources}=\"${tmp}\"
|
||||
no_sources=$((${no_sources}+1))
|
||||
done < "${TMP}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Need at least ONE source to backup
|
||||
#
|
||||
if [ "${no_sources}" -lt 1 ]; then
|
||||
usage
|
||||
else
|
||||
_techo "${HALF_VERSION}: Beginning backup using interval ${INTERVAL}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Look for pre-exec command (general)
|
||||
#
|
||||
if [ -x "${CPREEXEC}" ]; then
|
||||
_techo "Executing ${CPREEXEC} ..."
|
||||
"${CPREEXEC}"; ret=$?
|
||||
_techo "Finished ${CPREEXEC} (return code: ${ret})."
|
||||
|
||||
[ "${ret}" -eq 0 ] || _exit_err "${CPREEXEC} failed. Aborting"
|
||||
fi
|
||||
|
||||
#
|
||||
# check default configuration
|
||||
#
|
||||
|
||||
D_FILE_INTERVAL="${CDEFAULTS}/intervals/${INTERVAL}"
|
||||
D_INTERVAL=$(cat "${D_FILE_INTERVAL}" 2>/dev/null)
|
||||
|
||||
|
||||
#
|
||||
# Let's do the backup
|
||||
#
|
||||
i=0
|
||||
while [ "${i}" -lt "${no_sources}" ]; do
|
||||
|
||||
#
|
||||
# Get current source
|
||||
#
|
||||
eval name=\"\$source_${i}\"
|
||||
i=$((${i}+1))
|
||||
|
||||
export name
|
||||
|
||||
#
|
||||
# start ourself, if we want parallel execution
|
||||
#
|
||||
if [ "${PARALLEL}" ]; then
|
||||
"$0" "${INTERVAL}" "${name}" &
|
||||
continue
|
||||
fi
|
||||
|
||||
#
|
||||
# Start subshell for easy log editing
|
||||
#
|
||||
(
|
||||
#
|
||||
# Stderr to stdout, so we can produce nice logs
|
||||
#
|
||||
exec 2>&1
|
||||
|
||||
#
|
||||
# Configuration
|
||||
#
|
||||
backup="${CSOURCES}/${name}"
|
||||
c_source="${backup}/source"
|
||||
c_dest="${backup}/destination"
|
||||
c_exclude="${backup}/exclude"
|
||||
c_verbose="${backup}/verbose"
|
||||
c_vverbose="${backup}/very_verbose"
|
||||
c_rsync_extra="${backup}/rsync_options"
|
||||
c_summary="${backup}/summary"
|
||||
c_pre_exec="${backup}/pre_exec"
|
||||
c_post_exec="${backup}/post_exec"
|
||||
f_incomplete="delete_incomplete"
|
||||
c_incomplete="${backup}/${f_incomplete}"
|
||||
c_remote_host="${backup}/remote_host"
|
||||
|
||||
#
|
||||
# Marking backups: If we abort it's not removed => Backup is broken
|
||||
#
|
||||
c_marker=".ccollect-marker"
|
||||
|
||||
#
|
||||
# Times
|
||||
#
|
||||
begin_s=$(date +%s)
|
||||
|
||||
#
|
||||
# unset possible options
|
||||
#
|
||||
EXCLUDE=""
|
||||
RSYNC_EXTRA=""
|
||||
SUMMARY=""
|
||||
VERBOSE=""
|
||||
VVERBOSE=""
|
||||
DELETE_INCOMPLETE=""
|
||||
|
||||
_techo "Beginning to backup"
|
||||
|
||||
#
|
||||
# Standard configuration checks
|
||||
#
|
||||
if [ ! -e "${backup}" ]; then
|
||||
_exit_err "Source does not exist."
|
||||
fi
|
||||
|
||||
#
|
||||
# configuration _must_ be a directory
|
||||
#
|
||||
if [ ! -d "${backup}" ]; then
|
||||
_exit_err "\"${name}\" is not a cconfig-directory. Skipping."
|
||||
fi
|
||||
|
||||
#
|
||||
# first execute pre_exec, which may generate destination or other
|
||||
# parameters
|
||||
#
|
||||
if [ -x "${c_pre_exec}" ]; then
|
||||
_techo "Executing ${c_pre_exec} ..."
|
||||
"${c_pre_exec}"; ret="$?"
|
||||
_techo "Finished ${c_pre_exec} (return code ${ret})."
|
||||
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "${c_pre_exec} failed. Skipping."
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# Destination is a path
|
||||
#
|
||||
if [ ! -f "${c_dest}" ]; then
|
||||
_exit_err "Destination ${c_dest} is not a file. Skipping."
|
||||
else
|
||||
ddir=$(cat "${c_dest}"); ret="$?"
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "Destination ${c_dest} is not readable. Skipping."
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# interval definition: First try source specific, fallback to default
|
||||
#
|
||||
if [ ${INTERVAL} = "AUTO" ] ; then
|
||||
auto_interval
|
||||
_techo "Selected interval: '$INTERVAL'"
|
||||
fi
|
||||
c_interval="$(cat "${backup}/intervals/${INTERVAL}" 2>/dev/null)"
|
||||
|
||||
if [ -z "${c_interval}" ]; then
|
||||
c_interval="${D_INTERVAL}"
|
||||
|
||||
if [ -z "${c_interval}" ]; then
|
||||
_exit_err "No definition for interval \"${INTERVAL}\" found. Skipping."
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
# Source checks
|
||||
#
|
||||
if [ ! -f "${c_source}" ]; then
|
||||
_exit_err "Source description \"${c_source}\" is not a file. Skipping."
|
||||
else
|
||||
source=$(cat "${c_source}"); ret="$?"
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "Source ${c_source} is not readable. Skipping."
|
||||
fi
|
||||
fi
|
||||
# Verify source is up and accepting connections before deleting any old backups
|
||||
rsync "$source" >/dev/null || _exit_err "Source ${source} is not readable. Skipping."
|
||||
|
||||
#
|
||||
# do we backup to a remote host? then set pre-cmd
|
||||
#
|
||||
if [ -f "${c_remote_host}" ]; then
|
||||
# adjust ls and co
|
||||
remote_host=$(cat "${c_remote_host}"); ret="$?"
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_exit_err "Remote host file ${c_remote_host} exists, but is not readable. Skipping."
|
||||
fi
|
||||
destination="${remote_host}:${ddir}"
|
||||
else
|
||||
remote_host=""
|
||||
destination="${ddir}"
|
||||
fi
|
||||
export remote_host
|
||||
|
||||
#
|
||||
# check for existence / use real name
|
||||
#
|
||||
( pcmd cd "$ddir" ) || _exit_err "Cannot change to ${ddir}. Skipping."
|
||||
|
||||
|
||||
#
|
||||
# Check whether to delete incomplete backups
|
||||
#
|
||||
if [ -f "${c_incomplete}" -o -f "${CDEFAULTS}/${f_incomplete}" ]; then
|
||||
DELETE_INCOMPLETE="yes"
|
||||
fi
|
||||
|
||||
# NEW method as of 0.6:
|
||||
# - insert ccollect default parameters
|
||||
# - insert options
|
||||
# - insert user options
|
||||
|
||||
#
|
||||
# rsync standard options
|
||||
#
|
||||
|
||||
set -- "$@" "--archive" "--delete" "--numeric-ids" "--relative" \
|
||||
"--delete-excluded" "--sparse"
|
||||
|
||||
#
|
||||
# exclude list
|
||||
#
|
||||
if [ -f "${c_exclude}" ]; then
|
||||
set -- "$@" "--exclude-from=${c_exclude}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Output a summary
|
||||
#
|
||||
if [ -f "${c_summary}" ]; then
|
||||
set -- "$@" "--stats"
|
||||
fi
|
||||
|
||||
#
|
||||
# Verbosity for rsync
|
||||
#
|
||||
if [ -f "${c_vverbose}" ]; then
|
||||
set -- "$@" "-vv"
|
||||
elif [ -f "${c_verbose}" ]; then
|
||||
set -- "$@" "-v"
|
||||
fi
|
||||
|
||||
#
|
||||
# extra options for rsync provided by the user
|
||||
#
|
||||
if [ -f "${c_rsync_extra}" ]; then
|
||||
while read line; do
|
||||
set -- "$@" "$line"
|
||||
done < "${c_rsync_extra}"
|
||||
fi
|
||||
|
||||
#
|
||||
# Check for incomplete backups
|
||||
#
|
||||
pcmd ls -1 "$ddir/${INTERVAL}"*".${c_marker}" > "${TMP}" 2>/dev/null
|
||||
|
||||
i=0
|
||||
while read incomplete; do
|
||||
eval incomplete_$i=\"$(echo ${incomplete} | sed "s/\\.${c_marker}\$//")\"
|
||||
i=$(($i+1))
|
||||
done < "${TMP}"
|
||||
|
||||
j=0
|
||||
while [ "$j" -lt "$i" ]; do
|
||||
eval realincomplete=\"\$incomplete_$j\"
|
||||
_techo "Incomplete backup: ${realincomplete}"
|
||||
if [ "${DELETE_INCOMPLETE}" = "yes" ]; then
|
||||
_techo "Deleting ${realincomplete} ..."
|
||||
pcmd rm $VVERBOSE -rf "${ddir}/${realincomplete}" || \
|
||||
_exit_err "Removing ${realincomplete} failed."
|
||||
fi
|
||||
j=$(($j+1))
|
||||
done
|
||||
|
||||
#
|
||||
# check if maximum number of backups is reached, if so remove
|
||||
# use grep and ls -p so we only look at directories
|
||||
#
|
||||
count="$(pcmd ls -p1 "${ddir}" | grep "^${INTERVAL}\..*/\$" | wc -l \
|
||||
| sed 's/^ *//g')" || _exit_err "Counting backups failed"
|
||||
|
||||
_techo "Existing backups: ${count} Total keeping backups: ${c_interval}"
|
||||
|
||||
if [ "${count}" -ge "${c_interval}" ]; then
|
||||
substract=$((${c_interval} - 1))
|
||||
remove=$((${count} - ${substract}))
|
||||
_techo "Removing ${remove} backup(s)..."
|
||||
|
||||
pcmd ls -${TSORT}p1r "$ddir" | grep "^${INTERVAL}\..*/\$" | \
|
||||
head -n "${remove}" > "${TMP}" || \
|
||||
_exit_err "Listing old backups failed"
|
||||
|
||||
i=0
|
||||
while read to_remove; do
|
||||
eval remove_$i=\"${to_remove}\"
|
||||
i=$(($i+1))
|
||||
done < "${TMP}"
|
||||
|
||||
j=0
|
||||
while [ "$j" -lt "$i" ]; do
|
||||
eval to_remove=\"\$remove_$j\"
|
||||
_techo "Removing ${to_remove} ..."
|
||||
pcmd rm ${VVERBOSE} -rf "${ddir}/${to_remove}" || \
|
||||
_exit_err "Removing ${to_remove} failed."
|
||||
j=$(($j+1))
|
||||
done
|
||||
fi
|
||||
|
||||
|
||||
#
|
||||
# Check for backup directory to clone from: Always clone from the latest one!
|
||||
#
|
||||
# Depending on your file system, you may want to sort on:
|
||||
# 1. mtime (modification time) with TSORT=t, or
|
||||
# 2. ctime (last change time, usually) with TSORT=tc
|
||||
last_dir="$(pcmd ls -${TSORT}p1 "${ddir}" | grep '/$' | head -n 1)" || \
|
||||
_exit_err "Failed to list contents of ${ddir}."
|
||||
|
||||
#
|
||||
# clone from old backup, if existing
|
||||
#
|
||||
if [ "${last_dir}" ]; then
|
||||
set -- "$@" "--link-dest=${ddir}/${last_dir}"
|
||||
_techo "Hard linking from ${last_dir}"
|
||||
fi
|
||||
|
||||
|
||||
# set time when we really begin to backup, not when we began to remove above
|
||||
destination_date=$(${CDATE})
|
||||
destination_dir="${ddir}/${INTERVAL}.${destination_date}.$$"
|
||||
destination_full="${destination}/${INTERVAL}.${destination_date}.$$"
|
||||
|
||||
# give some info
|
||||
_techo "Beginning to backup, this may take some time..."
|
||||
|
||||
_techo "Creating ${destination_dir} ..."
|
||||
pcmd mkdir ${VVERBOSE} "${destination_dir}" || \
|
||||
_exit_err "Creating ${destination_dir} failed. Skipping."
|
||||
|
||||
#
|
||||
# added marking in 0.6 (and remove it, if successful later)
|
||||
#
|
||||
pcmd touch "${destination_dir}.${c_marker}"
|
||||
|
||||
#
|
||||
# the rsync part
|
||||
#
|
||||
_techo "Transferring files..."
|
||||
rsync "$@" "${source}" "${destination_full}"; ret=$?
|
||||
# Correct the modification time:
|
||||
pcmd touch "${destination_dir}"
|
||||
|
||||
#
|
||||
# remove marking here
|
||||
#
|
||||
if [ "$ret" -ne 12 ] ; then
|
||||
pcmd rm "${destination_dir}.${c_marker}" || \
|
||||
_exit_err "Removing ${destination_dir}/${c_marker} failed."
|
||||
fi
|
||||
|
||||
_techo "Finished backup (rsync return code: $ret)."
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_techo "Warning: rsync exited non-zero, the backup may be broken (see rsync errors)."
|
||||
fi
|
||||
|
||||
#
|
||||
# post_exec
|
||||
#
|
||||
if [ -x "${c_post_exec}" ]; then
|
||||
_techo "Executing ${c_post_exec} ..."
|
||||
"${c_post_exec}"; ret=$?
|
||||
_techo "Finished ${c_post_exec}."
|
||||
|
||||
if [ ${ret} -ne 0 ]; then
|
||||
_exit_err "${c_post_exec} failed."
|
||||
fi
|
||||
fi
|
||||
|
||||
# Calculation
|
||||
end_s=$(date +%s)
|
||||
|
||||
full_seconds=$((${end_s} - ${begin_s}))
|
||||
hours=$((${full_seconds} / 3600))
|
||||
seconds=$((${full_seconds} - (${hours} * 3600)))
|
||||
minutes=$((${seconds} / 60))
|
||||
seconds=$((${seconds} - (${minutes} * 60)))
|
||||
|
||||
_techo "Backup lasted: ${hours}:${minutes}:${seconds} (h:m:s)"
|
||||
|
||||
) | add_name
|
||||
done
|
||||
|
||||
#
|
||||
# Be a good parent and wait for our children, if they are running wild parallel
|
||||
#
|
||||
if [ "${PARALLEL}" ]; then
|
||||
_techo "Waiting for children to complete..."
|
||||
wait
|
||||
fi
|
||||
|
||||
#
|
||||
# Look for post-exec command (general)
|
||||
#
|
||||
if [ -x "${CPOSTEXEC}" ]; then
|
||||
_techo "Executing ${CPOSTEXEC} ..."
|
||||
"${CPOSTEXEC}"; ret=$?
|
||||
_techo "Finished ${CPOSTEXEC} (return code: ${ret})."
|
||||
|
||||
if [ ${ret} -ne 0 ]; then
|
||||
_techo "${CPOSTEXEC} failed."
|
||||
fi
|
||||
fi
|
||||
|
||||
rm -f "${TMP}"
|
||||
_techo "Finished ${WE}"
|
||||
|
||||
# vim: set shiftwidth=3 tabstop=3 expandtab :
|
17
contrib/jlawless-2009-06-03/old/d.patch
Normal file
17
contrib/jlawless-2009-06-03/old/d.patch
Normal file
|
@ -0,0 +1,17 @@
|
|||
--- ccollect-0.7.1-c.sh 2009-05-24 21:39:43.000000000 -0700
|
||||
+++ ccollect-0.7.1-d.sh 2009-05-24 21:47:09.000000000 -0700
|
||||
@@ -492,12 +492,12 @@
|
||||
if [ "${count}" -ge "${c_interval}" ]; then
|
||||
substract=$((${c_interval} - 1))
|
||||
remove=$((${count} - ${substract}))
|
||||
_techo "Removing ${remove} backup(s)..."
|
||||
|
||||
- pcmd ls -p1 "$ddir" | grep "^${INTERVAL}\..*/\$" | \
|
||||
- sort -n | head -n "${remove}" > "${TMP}" || \
|
||||
+ pcmd ls -${TSORT}p1r "$ddir" | grep "^${INTERVAL}\..*/\$" | \
|
||||
+ head -n "${remove}" > "${TMP}" || \
|
||||
_exit_err "Listing old backups failed"
|
||||
|
||||
i=0
|
||||
while read to_remove; do
|
||||
eval remove_$i=\"${to_remove}\"
|
19
contrib/jlawless-2009-06-03/old/e.patch
Normal file
19
contrib/jlawless-2009-06-03/old/e.patch
Normal file
|
@ -0,0 +1,19 @@
|
|||
--- ccollect-0.7.1-d.sh 2009-05-24 21:47:09.000000000 -0700
|
||||
+++ ccollect-0.7.1-e.sh 2009-05-24 22:18:16.000000000 -0700
|
||||
@@ -560,12 +560,14 @@
|
||||
pcmd touch "${destination_dir}"
|
||||
|
||||
#
|
||||
# remove marking here
|
||||
#
|
||||
- pcmd rm "${destination_dir}.${c_marker}" || \
|
||||
- _exit_err "Removing ${destination_dir}/${c_marker} failed."
|
||||
+ if [ "$ret" -ne 12 ] ; then
|
||||
+ pcmd rm "${destination_dir}.${c_marker}" || \
|
||||
+ _exit_err "Removing ${destination_dir}/${c_marker} failed."
|
||||
+ fi
|
||||
|
||||
_techo "Finished backup (rsync return code: $ret)."
|
||||
if [ "${ret}" -ne 0 ]; then
|
||||
_techo "Warning: rsync exited non-zero, the backup may be broken (see rsync errors)."
|
||||
fi
|
119
contrib/jlawless-2009-06-03/old/f.patch
Normal file
119
contrib/jlawless-2009-06-03/old/f.patch
Normal file
|
@ -0,0 +1,119 @@
|
|||
--- ccollect-0.7.1-e.sh 2009-05-24 22:18:16.000000000 -0700
|
||||
+++ ccollect-0.7.1-f.sh 2009-05-24 22:19:50.000000000 -0700
|
||||
@@ -124,10 +124,64 @@
|
||||
echo " Retrieve latest ccollect at http://unix.schottelius.org/ccollect/"
|
||||
exit 0
|
||||
}
|
||||
|
||||
#
|
||||
+# Select interval if AUTO
|
||||
+#
|
||||
+# For this to work nicely, you have to choose interval names that sort nicely
|
||||
+# such as int1, int2, int3 or a_daily, b_weekly, c_monthly, etc.
|
||||
+#
|
||||
+auto_interval()
|
||||
+{
|
||||
+ if [ -d "${backup}/intervals" -a -n "$(ls "${backup}/intervals" 2>/dev/null)" ] ; then
|
||||
+ intervals_dir="${backup}/intervals"
|
||||
+ elif [ -d "${CDEFAULTS}/intervals" -a -n "$(ls "${CDEFAULTS}/intervals" 2>/dev/null)" ] ; then
|
||||
+ intervals_dir="${CDEFAULTS}/intervals"
|
||||
+ else
|
||||
+ _exit_err "No intervals are defined. Skipping."
|
||||
+ fi
|
||||
+ echo intervals_dir=${intervals_dir}
|
||||
+
|
||||
+ trial_interval="$(ls -1r "${intervals_dir}/" | head -n 1)" || \
|
||||
+ _exit_err "Failed to list contents of ${intervals_dir}/."
|
||||
+ _techo "Considering interval ${trial_interval}"
|
||||
+ most_recent="$(pcmd ls -${TSORT}p1 "${ddir}" | grep "^${trial_interval}.*/$" | head -n 1)" || \
|
||||
+ _exit_err "Failed to list contents of ${ddir}/."
|
||||
+ _techo " Most recent ${trial_interval}: '${most_recent}'"
|
||||
+ if [ -n "${most_recent}" ]; then
|
||||
+ no_intervals="$(ls -1 "${intervals_dir}/" | wc -l)"
|
||||
+ n=1
|
||||
+ while [ "${n}" -le "${no_intervals}" ]; do
|
||||
+ trial_interval="$(ls -p1 "${intervals_dir}/" | tail -n+${n} | head -n 1)"
|
||||
+ _techo "Considering interval '${trial_interval}'"
|
||||
+ c_interval="$(cat "${intervals_dir}/${trial_interval}" 2>/dev/null)"
|
||||
+ m=$((${n}+1))
|
||||
+ set -- "${ddir}" -maxdepth 1
|
||||
+ while [ "${m}" -le "${no_intervals}" ]; do
|
||||
+ interval_m="$(ls -1 "${intervals_dir}/" | tail -n+${m} | head -n 1)"
|
||||
+ most_recent="$(pcmd ls -${TSORT}p1 "${ddir}" | grep "^${interval_m}\..*/$" | head -n 1)"
|
||||
+ _techo " Most recent ${interval_m}: '${most_recent}'"
|
||||
+ if [ -n "${most_recent}" ] ; then
|
||||
+ set -- "$@" -$NEWER "${ddir}/${most_recent}"
|
||||
+ fi
|
||||
+ m=$((${m}+1))
|
||||
+ done
|
||||
+ count=$(pcmd find "$@" -iname "${trial_interval}*" | wc -l)
|
||||
+ _techo " Found $count more recent backups of ${trial_interval} (limit: ${c_interval})"
|
||||
+ if [ "$count" -lt "${c_interval}" ] ; then
|
||||
+ break
|
||||
+ fi
|
||||
+ n=$((${n}+1))
|
||||
+ done
|
||||
+ fi
|
||||
+ export INTERVAL="${trial_interval}"
|
||||
+ D_FILE_INTERVAL="${intervals_dir}/${INTERVAL}"
|
||||
+ D_INTERVAL=$(cat "${D_FILE_INTERVAL}" 2>/dev/null)
|
||||
+}
|
||||
+
|
||||
+#
|
||||
# need at least interval and one source or --all
|
||||
#
|
||||
if [ $# -lt 2 ]; then
|
||||
if [ "$1" = "-V" -o "$1" = "--version" ]; then
|
||||
display_version
|
||||
@@ -344,12 +398,28 @@
|
||||
_exit_err "${c_pre_exec} failed. Skipping."
|
||||
fi
|
||||
fi
|
||||
|
||||
#
|
||||
+ # Destination is a path
|
||||
+ #
|
||||
+ if [ ! -f "${c_dest}" ]; then
|
||||
+ _exit_err "Destination ${c_dest} is not a file. Skipping."
|
||||
+ else
|
||||
+ ddir=$(cat "${c_dest}"); ret="$?"
|
||||
+ if [ "${ret}" -ne 0 ]; then
|
||||
+ _exit_err "Destination ${c_dest} is not readable. Skipping."
|
||||
+ fi
|
||||
+ fi
|
||||
+
|
||||
+ #
|
||||
# interval definition: First try source specific, fallback to default
|
||||
#
|
||||
+ if [ ${INTERVAL} = "AUTO" ] ; then
|
||||
+ auto_interval
|
||||
+ _techo "Selected interval: '$INTERVAL'"
|
||||
+ fi
|
||||
c_interval="$(cat "${backup}/intervals/${INTERVAL}" 2>/dev/null)"
|
||||
|
||||
if [ -z "${c_interval}" ]; then
|
||||
c_interval="${D_INTERVAL}"
|
||||
|
||||
@@ -371,22 +441,10 @@
|
||||
fi
|
||||
# Verify source is up and accepting connections before deleting any old backups
|
||||
rsync "$source" >/dev/null || _exit_err "Source ${source} is not readable. Skipping."
|
||||
|
||||
#
|
||||
- # Destination is a path
|
||||
- #
|
||||
- if [ ! -f "${c_dest}" ]; then
|
||||
- _exit_err "Destination ${c_dest} is not a file. Skipping."
|
||||
- else
|
||||
- ddir=$(cat "${c_dest}"); ret="$?"
|
||||
- if [ "${ret}" -ne 0 ]; then
|
||||
- _exit_err "Destination ${c_dest} is not readable. Skipping."
|
||||
- fi
|
||||
- fi
|
||||
-
|
||||
- #
|
||||
# do we backup to a remote host? then set pre-cmd
|
||||
#
|
||||
if [ -f "${c_remote_host}" ]; then
|
||||
# adjust ls and co
|
||||
remote_host=$(cat "${c_remote_host}"); ret="$?"
|
Loading…
Reference in a new issue