feat: Add backup.sh
This commit is contained in:
parent
06c7c2349a
commit
6c27793897
66
src/baf
66
src/baf
@ -1,66 +0,0 @@
|
|||||||
#! /usr/bin/env bash
|
|
||||||
|
|
||||||
# baf -- backup automation micro-framework.
|
|
||||||
# Copyright (c) 2022 ge <https://nixhacks.net/>
|
|
||||||
#
|
|
||||||
# This program is free software: you can redistribute it and/or modify
|
|
||||||
# it under the terms of the GNU General Public License as published by
|
|
||||||
# the Free Software Foundation, either version 3 of the License, or
|
|
||||||
# (at your option) any later version.
|
|
||||||
#
|
|
||||||
# This program is distributed in the hope that it will be useful,
|
|
||||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
# GNU General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU General Public License
|
|
||||||
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
baf_version='0.0.0'
|
|
||||||
|
|
||||||
BAFLIB="./lib"
|
|
||||||
|
|
||||||
# Source library
|
|
||||||
. ${BAFLIB}/*
|
|
||||||
|
|
||||||
print_help() {
|
|
||||||
cat <<- EOF
|
|
||||||
Backup files and databases.
|
|
||||||
|
|
||||||
Usage: $0 [-hv] <arguments>..
|
|
||||||
|
|
||||||
Options:
|
|
||||||
-h, --help print this help message and exit.
|
|
||||||
-v, --version print version and exit.
|
|
||||||
EOF
|
|
||||||
}
|
|
||||||
|
|
||||||
# Print help if no arguments passed
|
|
||||||
[[ "$@" ]] || { print_help; exit 1; }
|
|
||||||
|
|
||||||
# Transform long options to short ones
|
|
||||||
for arg in "$@"; do
|
|
||||||
shift
|
|
||||||
case "$arg" in
|
|
||||||
--help) set -- "$@" "-h";;
|
|
||||||
--version) set -- "$@" "-v";;
|
|
||||||
*) set -- "$@" "$arg";;
|
|
||||||
esac
|
|
||||||
done
|
|
||||||
|
|
||||||
# Parse short options
|
|
||||||
while getopts ":hvS" opt; do
|
|
||||||
case "$opt" in
|
|
||||||
h) print_help; exit 0;;
|
|
||||||
v) echo "$buf_version"; exit 0;;
|
|
||||||
*) echo "Unknown option $opt" >&2; exit 1;;
|
|
||||||
esac
|
|
||||||
done
|
|
||||||
|
|
||||||
# Parse positional arguments
|
|
||||||
shift $(($OPTIND - 1))
|
|
||||||
while (( "$#" )); do
|
|
||||||
case "$1" in
|
|
||||||
*) posargs+=("$1"); shift;; # Save args
|
|
||||||
esac
|
|
||||||
done
|
|
136
src/bafscript
Executable file
136
src/bafscript
Executable file
@ -0,0 +1,136 @@
|
|||||||
|
#! /usr/bin/env bash
|
||||||
|
|
||||||
|
# bafscript -- backup automation micro-framework.
|
||||||
|
# Copyright (c) 2022 ge <https://nixhacks.net/>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
__version='0.0.0'
|
||||||
|
__config=
|
||||||
|
__log_file='./log.txt'
|
||||||
|
__tar_options='-czf'
|
||||||
|
__name_date_fmt='_%Y%m%d-%H%M'
|
||||||
|
|
||||||
|
if [ -n "$BAFLIB" ]; then
|
||||||
|
__library="$BAFLIB"
|
||||||
|
else
|
||||||
|
__library='./lib'
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Source library
|
||||||
|
for file in "$__library"/*; do
|
||||||
|
. "$file"
|
||||||
|
done
|
||||||
|
|
||||||
|
print_help() {
|
||||||
|
cat <<- EOF
|
||||||
|
Backup files and databases.
|
||||||
|
|
||||||
|
Usage: $0 [-cvlhV] ARGUMENTS..
|
||||||
|
|
||||||
|
Options:
|
||||||
|
-c, --config config file.
|
||||||
|
-v, --verbose verbose output.
|
||||||
|
-l, --log-file log file.
|
||||||
|
-h, --help print this help messagea and exit.
|
||||||
|
-V, --version print version and exit.
|
||||||
|
|
||||||
|
Environment:
|
||||||
|
BAFLIB path to baf library [current: $__library]
|
||||||
|
EOF
|
||||||
|
}
|
||||||
|
|
||||||
|
# ---------------------------------------------------------- #
|
||||||
|
# * CLI Arguments Parser #
|
||||||
|
# ---------------------------------------------------------- #
|
||||||
|
|
||||||
|
optval() {
|
||||||
|
# GNU-style CLI options parser.
|
||||||
|
#
|
||||||
|
# Parse --opt VAL and --opt=VAL options.
|
||||||
|
# Requires 2 arguments: $1, $2.
|
||||||
|
# Return variables:
|
||||||
|
# opt option name.
|
||||||
|
# val option value.
|
||||||
|
# sft value for shift.
|
||||||
|
|
||||||
|
if [[ "$1" =~ .+=.+ ]]; then
|
||||||
|
opt="${1%%=*}"; val="${1#*=}"; sft=1
|
||||||
|
elif [[ ! "$1" =~ .+=$ ]] && [ "$2" ] && [ "${2:0:1}" != "-" ]; then
|
||||||
|
opt="$1"; val="$2"; sft=2
|
||||||
|
else
|
||||||
|
opt="$1"
|
||||||
|
if [[ "$1" =~ .+=$ ]]; then opt="${1:0: -1}"; fi
|
||||||
|
echo "Error: Missing argument for $opt" >&2; exit 1
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# Print help if no arguments passed
|
||||||
|
[[ "$#" == 0 ]] && { print_help; exit 1; }
|
||||||
|
|
||||||
|
# Split combined short options, e.g. '-abc' to '-a' '-b' '-c'
|
||||||
|
for args in "$@"; do
|
||||||
|
shift
|
||||||
|
case "$args" in
|
||||||
|
--*)
|
||||||
|
set -- "$@" "$args";; # save long options
|
||||||
|
-*)
|
||||||
|
args="$(echo "${args:1}" | grep -o . | xargs -I {} echo -n '-{} ')"
|
||||||
|
set -- "$@" $args;;
|
||||||
|
*)
|
||||||
|
set -- "$@" "$args";; # save positional arguments
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
|
# Final arguments parser
|
||||||
|
while (( "$#" )); do
|
||||||
|
case "$1" in
|
||||||
|
-c|--config|--config=*) optval "$1" "$2"; __config="$val"; shift "$sft";;
|
||||||
|
-v|--verbose) __verbose=1; shift;;
|
||||||
|
-l|--log-file|--log-file=*) optval "$1" "$2"; __log_file="$val"; shift "$sft";;
|
||||||
|
-h|--help) print_help; exit 1;;
|
||||||
|
-V|--version) echo "$__version"; exit 0;;
|
||||||
|
-*) echo "Error: Unknown option: $1" >&2; exit 1;;
|
||||||
|
*) __args+=("$1"); shift;; # Save positional args
|
||||||
|
esac
|
||||||
|
done
|
||||||
|
|
||||||
|
# ---------------------------------------------------------- #
|
||||||
|
# * Do backups #
|
||||||
|
# ---------------------------------------------------------- #
|
||||||
|
|
||||||
|
for script in "${__args[@]}"; do
|
||||||
|
source_script "$script"
|
||||||
|
|
||||||
|
# Initialise variables
|
||||||
|
__user_script="$script"
|
||||||
|
backups=() # Array of created backups, contains full pathes
|
||||||
|
|
||||||
|
# Run prepare() before all if set
|
||||||
|
if is_function_set prepare; then
|
||||||
|
prepare
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Run user defined backup() if set or builtin_backup()
|
||||||
|
if is_function_set backup; then
|
||||||
|
backup
|
||||||
|
else
|
||||||
|
builtin_backup
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Run post backup function if set
|
||||||
|
if is_function_set finalise; then
|
||||||
|
finalise
|
||||||
|
fi
|
||||||
|
done
|
251
src/lib/backup.sh
Normal file
251
src/lib/backup.sh
Normal file
@ -0,0 +1,251 @@
|
|||||||
|
#! /usr/bin/env bash
|
||||||
|
|
||||||
|
# backup.sh - functions for backup processing.
|
||||||
|
# Copyright (c) 2022 ge <https://nixhacks.net/>
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
process_source() {
|
||||||
|
# Run handler function for source URI by scheme.
|
||||||
|
#
|
||||||
|
# Usage: process_source URI
|
||||||
|
|
||||||
|
local uri
|
||||||
|
local scheme
|
||||||
|
local handler
|
||||||
|
|
||||||
|
uri="$1"
|
||||||
|
scheme="${uri%%:*}"
|
||||||
|
|
||||||
|
case "$scheme" in
|
||||||
|
file) handler='backup_files';;
|
||||||
|
mysql) handler='backup_mysql';;
|
||||||
|
postgres) handler='backup_postgres';;
|
||||||
|
sqlite) handler='backup_sqlite';;
|
||||||
|
*) echo "Error: Unsupported URI scheme: $scheme" >&2; exit 1;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
# Run handler function
|
||||||
|
"$handler" "$uri"
|
||||||
|
}
|
||||||
|
|
||||||
|
process_target() {
|
||||||
|
# Run handler function for target URI by scheme.
|
||||||
|
#
|
||||||
|
# Usage: process_target URI
|
||||||
|
|
||||||
|
local uri
|
||||||
|
local scheme
|
||||||
|
local handler
|
||||||
|
|
||||||
|
uri="$1"
|
||||||
|
scheme="${uri%%:*}"
|
||||||
|
|
||||||
|
case "$scheme" in
|
||||||
|
file) handler='transfer_files';;
|
||||||
|
ftp) handler='transfer_ftp';;
|
||||||
|
sftp) handler='transfer_sftp';;
|
||||||
|
rsync) handler='transfer_rsync';;
|
||||||
|
s3) handler='transfer_s3';;
|
||||||
|
sj) handler='transfer_sj';;
|
||||||
|
swift) handler='transfer_swift';;
|
||||||
|
dav) handler='transfer_dav';;
|
||||||
|
davs) handler='transfer_davs';;
|
||||||
|
*) echo "Error: Unsupported URI scheme: $scheme" >&2; exit 1;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
# Run handler function
|
||||||
|
"$handler" "$uri"
|
||||||
|
}
|
||||||
|
|
||||||
|
builtin_backup() {
|
||||||
|
# Backup function.
|
||||||
|
#
|
||||||
|
# Usage: builtin_backup
|
||||||
|
|
||||||
|
for source in "${sources[@]}"; do
|
||||||
|
process_source "$source"
|
||||||
|
done
|
||||||
|
|
||||||
|
for target in "${targets[@]}"; do
|
||||||
|
process_target "$target"
|
||||||
|
done
|
||||||
|
}
|
||||||
|
|
||||||
|
# ---------------------------------------------------------- #
|
||||||
|
# * Backup functions #
|
||||||
|
# ---------------------------------------------------------- #
|
||||||
|
|
||||||
|
backup_files() {
|
||||||
|
# Backup local files with tar(1). Handle 'file' URI scheme.
|
||||||
|
#
|
||||||
|
# Usage: backup_files URI
|
||||||
|
|
||||||
|
local uri
|
||||||
|
local src_path
|
||||||
|
local dst_path
|
||||||
|
local archive
|
||||||
|
local file_ext
|
||||||
|
|
||||||
|
uri="$1"
|
||||||
|
dst_path="$__main_target_path"
|
||||||
|
|
||||||
|
parse_uri "$uri"
|
||||||
|
|
||||||
|
if [ -f "$path" ] || [ -d "$path" ]; then
|
||||||
|
src_path="$path"
|
||||||
|
else
|
||||||
|
echo "Error: Path '$path' from URI '$uri' does not exists" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
is_installed tar # Exit if tar is not installed
|
||||||
|
|
||||||
|
# Overwrire __tar_options
|
||||||
|
if [ -n "$tar_options" ]; then
|
||||||
|
__tar_options="$tar_options"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# TODO выбор сжатия, можно в переменной __tar_options заменять букву z на
|
||||||
|
# другую для соответствующего сжатия
|
||||||
|
file_ext=.tar.gz
|
||||||
|
|
||||||
|
archive="${dst_path}/$(gen_backup_name "$file_ext")"
|
||||||
|
|
||||||
|
tar "$__tar_options" "$archive" "$src_path" |& log -p
|
||||||
|
|
||||||
|
# Append path to 'backups' array
|
||||||
|
backups+=("$archive")
|
||||||
|
}
|
||||||
|
|
||||||
|
backup_mysql() {
|
||||||
|
echo Not implemented >&2; exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
backup_postgres() {
|
||||||
|
echo Not implemented >&2; exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
backup_sqlite() {
|
||||||
|
echo Not implemented >&2; exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
# ---------------------------------------------------------- #
|
||||||
|
# * Functions for targets processing #
|
||||||
|
# ---------------------------------------------------------- #
|
||||||
|
|
||||||
|
transfer_files() {
|
||||||
|
# Transfer files to another location from __main_target_path using cp(1)
|
||||||
|
#
|
||||||
|
# Usage: transfer_files URI
|
||||||
|
|
||||||
|
local uri
|
||||||
|
local dst_path
|
||||||
|
|
||||||
|
uri="$1"
|
||||||
|
|
||||||
|
if [[ "$uri" == "$__main_target" ]]; then
|
||||||
|
: # Do nothing. Source and destination is the same
|
||||||
|
else
|
||||||
|
# Copy backups to another destination
|
||||||
|
parse_uri "$uri"
|
||||||
|
|
||||||
|
if [ -f "$path" ] || [ -d "$path" ]; then
|
||||||
|
dst_path="$path"
|
||||||
|
else
|
||||||
|
echo "Error: Path '$path' from URI '$uri' does not exists" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Copy files preserving metadata
|
||||||
|
for backup in "${backups[@]}"; do
|
||||||
|
cp --archive "$backup" "$dst_path"
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
transfer_ftp() {
|
||||||
|
echo Not implemented >&2; exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
transfer_sftp() {
|
||||||
|
echo Not implemented >&2; exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
transfer_rsync() {
|
||||||
|
echo Not implemented >&2; exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
transfer_s3() {
|
||||||
|
echo Not implemented >&2; exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
transfer_sj() {
|
||||||
|
echo Not implemented >&2; exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
transfer_swift() {
|
||||||
|
echo Not implemented >&2; exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
transfer_dav() {
|
||||||
|
echo Not implemented >&2; exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
transfer_davs() {
|
||||||
|
echo Not implemented >&2; exit 1
|
||||||
|
}
|
||||||
|
|
||||||
|
# ---------------------------------------------------------- #
|
||||||
|
# * Helper functions #
|
||||||
|
# ---------------------------------------------------------- #
|
||||||
|
|
||||||
|
is_installed() {
|
||||||
|
# Check if the program is installed.
|
||||||
|
# See good answer: https://stackoverflow.com/a/677212
|
||||||
|
#
|
||||||
|
# Usage: is_installed COMMAND
|
||||||
|
|
||||||
|
local cmd
|
||||||
|
|
||||||
|
cmd="$1"
|
||||||
|
|
||||||
|
if ! command -v "$cmd" >/dev/null 2>&1; then
|
||||||
|
echo "Error: Command $cmd not found." \
|
||||||
|
"Please install $cmd or check your PATH if it's actually installed." >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
gen_backup_name() {
|
||||||
|
# Generate backup file name. Return (echo) string.
|
||||||
|
#
|
||||||
|
# Usage: gen_backup_name NAME_EXT
|
||||||
|
|
||||||
|
local prefix
|
||||||
|
local name
|
||||||
|
local date_fmt
|
||||||
|
local name_ext
|
||||||
|
|
||||||
|
[ -n "$name_prefix" ] || { prefix="${__user_script}_"; }
|
||||||
|
name="$(basename $path)" # 'path' is variable parsed from URI
|
||||||
|
name_ext="$1"
|
||||||
|
|
||||||
|
# Overwrite __name_date_fmt
|
||||||
|
if [ -n "$name_date_fmt" ]; then
|
||||||
|
__name_date_fmt="$name_date_fmt"
|
||||||
|
fi
|
||||||
|
|
||||||
|
date +"${prefix}${name}${__name_date_fmt}${name_ext}"
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user