2024-08-16 16:40:24 +01:00
|
|
|
import ../model/config_type
|
2024-08-17 18:00:30 +01:00
|
|
|
import ../model/state_type
|
2024-08-18 18:45:37 +01:00
|
|
|
import ../model/borg_type
|
2024-08-17 18:00:30 +01:00
|
|
|
import ../notifier/notifier
|
2024-08-18 18:45:37 +01:00
|
|
|
import ../utils/actions
|
|
|
|
import strutils
|
|
|
|
import strformat
|
|
|
|
import sequtils
|
2024-08-16 16:40:24 +01:00
|
|
|
import osproc
|
|
|
|
import times
|
|
|
|
import os
|
|
|
|
import nativesockets
|
|
|
|
|
|
|
|
|
|
|
|
proc genArchiveName(): string =
|
|
|
|
let hostname = getHostname()
|
2024-08-18 16:19:16 +01:00
|
|
|
let ts = getTime().format("yyyy-MM-dd'T'HH:mm:ss'.'ffffff")
|
2024-08-18 18:45:37 +01:00
|
|
|
return fmt"{hostname}-{ts}"
|
2024-08-16 16:40:24 +01:00
|
|
|
|
|
|
|
proc genCommand(cmd: string, repo: string, others: seq[string]): string =
|
2024-08-18 18:45:37 +01:00
|
|
|
let args = others.join(" ")
|
|
|
|
let cmd = fmt"{BORG_BIN} {cmd} {repo} {args}"
|
|
|
|
return cmd
|
2024-08-16 16:40:24 +01:00
|
|
|
|
|
|
|
proc run(cmd: string): int =
|
2024-08-18 18:45:37 +01:00
|
|
|
echo fmt"Trying to run : {cmd}"
|
2024-08-16 16:40:24 +01:00
|
|
|
try:
|
|
|
|
let res = execProcess(cmd)
|
|
|
|
echo res
|
|
|
|
except:
|
|
|
|
echo getCurrentExceptionMsg()
|
|
|
|
|
|
|
|
proc runDiscard(cmd: string): int =
|
2024-08-18 18:45:37 +01:00
|
|
|
echo fmt"Trying to run : {cmd}"
|
2024-08-16 16:40:24 +01:00
|
|
|
try:
|
|
|
|
let res = execCmd(cmd)
|
|
|
|
return res
|
|
|
|
except:
|
|
|
|
echo getCurrentExceptionMsg()
|
|
|
|
return 1
|
|
|
|
|
|
|
|
proc initRepo(nc: NorgConfig, repo: Repository): int =
|
|
|
|
return runDiscard genCommand(cmd = "init", repo = repo.path, others = nc.args.others)
|
|
|
|
|
2024-08-18 18:45:37 +01:00
|
|
|
proc createArchive(nc: NorgConfig, repo: Repository, archivename: string, retry: int = 0): int =
|
|
|
|
let others = concat(nc.source_directories, nc.args.others)
|
|
|
|
let res = run genCommand(cmd = "create", repo = archivename, others = others)
|
|
|
|
if res == 1:
|
|
|
|
sleep 15 * 1000 # 15 seconds
|
|
|
|
if retry == nc.retries:
|
|
|
|
return 1
|
|
|
|
else:
|
|
|
|
return createArchive(nc, repo, archivename, retry + 1)
|
|
|
|
return res
|
|
|
|
|
2024-08-16 16:40:24 +01:00
|
|
|
proc backupSources(nc: NorgConfig, repo: Repository): int =
|
2024-08-18 15:12:44 +01:00
|
|
|
let start_time = now()
|
|
|
|
notify(nc.notifiers, state=Running)
|
|
|
|
let end_time = now()
|
2024-08-18 18:45:37 +01:00
|
|
|
let archivename = repo.path & "::" & genArchiveName()
|
|
|
|
let res = createArchive(nc, repo, archivename)
|
2024-08-18 15:43:11 +01:00
|
|
|
let total = (end_time - start_time).inMilliSeconds()
|
|
|
|
case res
|
2024-08-18 15:12:44 +01:00
|
|
|
of 0:
|
|
|
|
notify(nc.notifiers, state=Success, runtime=total)
|
|
|
|
of 1:
|
|
|
|
notify(nc.notifiers, state=Failure, runtime=total)
|
|
|
|
else:
|
|
|
|
notify(nc.notifiers, state=Failure, runtime=total)
|
2024-08-18 18:45:37 +01:00
|
|
|
return res
|
2024-08-17 18:00:30 +01:00
|
|
|
|
2024-08-16 16:40:24 +01:00
|
|
|
|
|
|
|
proc listArchives(nc: NorgConfig, repo: Repository): int =
|
|
|
|
return run genCommand(cmd = "list", repo = repo.path, others = nc.args.others)
|
|
|
|
|
|
|
|
proc mountArchive(nc: NorgConfig, repo: Repository): int =
|
|
|
|
let archive = repo.path & "::" & nc.args.others[0]
|
|
|
|
let others = nc.args.others[1..^1]
|
|
|
|
let ok = runDiscard genCommand(cmd = "mount", repo = archive, others = others)
|
|
|
|
if ok == 0:
|
2024-08-18 18:45:37 +01:00
|
|
|
echo fmt"Mounted {archive} at {others[0]}"
|
2024-08-16 16:40:24 +01:00
|
|
|
else:
|
|
|
|
echo "Failed to mount ", archive
|
|
|
|
|
|
|
|
proc unmountArchive(nc: NorgConfig): int =
|
|
|
|
let ok = runDiscard genCommand(cmd = "umount", repo = "", others = nc.args.others)
|
|
|
|
if ok == 0:
|
|
|
|
echo "Unmounted ", nc.args.others[0]
|
|
|
|
else:
|
|
|
|
echo "Failed to unmount ", nc.args.others[0]
|
|
|
|
|
2024-08-18 15:43:11 +01:00
|
|
|
proc isEmpty(dir: string): bool =
|
|
|
|
var count = 0
|
|
|
|
for idx, f in walkDir(dir): return false
|
|
|
|
#count += 1
|
|
|
|
return count == 0
|
|
|
|
|
|
|
|
proc extractArchive(nc: NorgConfig, repo: Repository): int =
|
2024-08-18 18:45:37 +01:00
|
|
|
let archive = fmt"{repo.path}::{nc.args.others[0]}"
|
2024-08-18 15:43:11 +01:00
|
|
|
var others = nc.args.others[1..^1]
|
|
|
|
if nc.args.extract_destination != "":
|
|
|
|
discard existsOrCreateDir(nc.args.extract_destination)
|
|
|
|
setCurrentDir(nc.args.extract_destination)
|
|
|
|
let dir = getCurrentDir()
|
|
|
|
if dir.isEmpty():
|
|
|
|
echo "Restoring..."
|
|
|
|
let ok = run(genCommand(cmd = "extract", repo = archive, others = others))
|
|
|
|
return ok
|
|
|
|
else:
|
|
|
|
echo "Not restoring to non-empty destination\r\nPlease use the --destination flag"
|
|
|
|
|
2024-08-18 18:45:37 +01:00
|
|
|
proc pruneRepo(nc: NorgConfig, repo: Repository): int =
|
|
|
|
echo "Not Yet Implemented."
|
|
|
|
discard
|
|
|
|
|
|
|
|
proc compactRepo(nc: NorgConfig, repo: Repository): int =
|
|
|
|
echo "Not Yet Implemented."
|
|
|
|
discard
|
|
|
|
|
|
|
|
proc checkRepo(nc: NorgConfig, repo: Repository): int =
|
|
|
|
echo "Not Yet Implemented."
|
|
|
|
discard
|
|
|
|
|
2024-08-16 16:40:24 +01:00
|
|
|
proc execute*(nc: NorgConfig) =
|
2024-08-18 18:45:37 +01:00
|
|
|
run_actions(norg_config.actions.before_everything)
|
2024-08-16 16:40:24 +01:00
|
|
|
for repo in nc.repositories:
|
2024-08-18 18:45:37 +01:00
|
|
|
run_actions(norg_config.actions.before_actions)
|
2024-08-16 16:40:24 +01:00
|
|
|
case nc.args.borg_cmd
|
2024-08-18 18:45:37 +01:00
|
|
|
of INIT:
|
2024-08-16 16:40:24 +01:00
|
|
|
discard initRepo(nc, repo)
|
2024-08-18 18:45:37 +01:00
|
|
|
of CREATE:
|
|
|
|
run_actions(norg_config.actions.before_backup)
|
2024-08-16 16:40:24 +01:00
|
|
|
discard backupSources(nc, repo)
|
2024-08-18 18:45:37 +01:00
|
|
|
run_actions(norg_config.actions.after_backup)
|
|
|
|
of LIST:
|
2024-08-16 16:40:24 +01:00
|
|
|
discard listArchives(nc, repo)
|
2024-08-18 18:45:37 +01:00
|
|
|
of MOUNT:
|
2024-08-16 16:40:24 +01:00
|
|
|
discard mountArchive(nc, repo)
|
2024-08-18 18:45:37 +01:00
|
|
|
of UMOUNT:
|
2024-08-16 16:40:24 +01:00
|
|
|
discard unmountArchive(nc)
|
2024-08-18 18:45:37 +01:00
|
|
|
of EXTRACT:
|
|
|
|
run_actions(norg_config.actions.before_extract)
|
2024-08-18 15:43:11 +01:00
|
|
|
discard extractArchive(nc, repo)
|
2024-08-18 18:45:37 +01:00
|
|
|
run_actions(norg_config.actions.after_extract)
|
|
|
|
of PRUNE:
|
|
|
|
run_actions(norg_config.actions.before_prune)
|
|
|
|
discard pruneRepo(nc, repo)
|
|
|
|
run_actions(norg_config.actions.after_prune)
|
|
|
|
of COMPACT:
|
|
|
|
run_actions(norg_config.actions.before_compact)
|
|
|
|
discard compactRepo(nc, repo)
|
|
|
|
run_actions(norg_config.actions.after_compact)
|
|
|
|
of CHECK:
|
|
|
|
run_actions(norg_config.actions.before_check)
|
|
|
|
discard checkRepo(nc, repo)
|
|
|
|
run_actions(norg_config.actions.after_check)
|
|
|
|
run_actions(norg_config.actions.after_actions)
|
|
|
|
delEncryptionPassphraseInfo()
|
|
|
|
run_actions(norg_config.actions.after_everything)
|