Skip to content

Commit

Permalink
add support for multi-instances
Browse files Browse the repository at this point in the history
  • Loading branch information
mrjk committed Dec 3, 2024
1 parent a58d743 commit 87c111b
Show file tree
Hide file tree
Showing 4 changed files with 63 additions and 30 deletions.
65 changes: 35 additions & 30 deletions app/backup.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ def __init__(self, docker_client: docker.DockerClient):

self.containers_completed = set()
self.containers_skipped = set()
self.prefix = self.env.LABEL_PREFIX

# Grab the backup starting time
self.start_time = datetime.now()
Expand All @@ -57,6 +58,9 @@ def log_this(self, log_message, log_priority="INFO", log_type=LogType.DEFAULT) -
"""Wrapper for log this"""
return self.logger.log_this(log_message, log_priority, log_type)

def get_label(self, container, target, default=None):
return container.labels.get(f"{self.prefix}.{target}", default)

def verify_nautcical_mounted_source_location(self, src_dir: str):
self.log_this(f"Verifying source directory '{src_dir}'...", "DEBUG", LogType.INIT)
if not os.path.isdir(src_dir):
Expand Down Expand Up @@ -141,7 +145,7 @@ def _should_skip_container(self, c: Container) -> bool:
# Convert the strings into lists
skip_containers_set = set(SKIP_CONTAINERS.split(","))

nautical_backup_enable_str = str(c.labels.get("nautical-backup.enable", ""))
nautical_backup_enable_str = str(self.get_label(c, "enable", ""))
if nautical_backup_enable_str.lower() == "false":
nautical_backup_enable = False
elif nautical_backup_enable_str.lower() == "true":
Expand All @@ -155,7 +159,7 @@ def _should_skip_container(self, c: Container) -> bool:

if self.env.REQUIRE_LABEL == True and nautical_backup_enable is not True:
self.log_this(
f"Skipping {c.name} as 'nautical-backup.enable=true' was not found and REQUIRE_LABEL is true.", "DEBUG"
f"Skipping {c.name} as '{self.prefix}.enable=true' was not found and REQUIRE_LABEL is true.", "DEBUG"
)
return True

Expand All @@ -170,6 +174,7 @@ def _should_skip_container(self, c: Container) -> bool:
# No reason to skip
return False


def group_containers(self) -> Dict[str, List[Container]]:
containers: List[Container] = self.docker.containers.list() # type: ignore
starting_container_amt = len(containers)
Expand All @@ -192,15 +197,15 @@ def group_containers(self) -> Dict[str, List[Container]]:

# Create a default group, so ungrouped items are not grouped together
default_group = f"{self.default_group_pfx_sfx}{str(c.id)[0:12]}{self.default_group_pfx_sfx}"
group = str(c.labels.get("nautical-backup.group", default_group))
group = str(self.get_label(c, "group", default_group))
if not group or group == "":
group = default_group

# Split the group string into a list of groups by comma
groups = group.split(",")
for g in groups:
# Get priority. Default=100
priority = int(c.labels.get(f"nautical-backup.group.{g}.priority", 100))
priority = int(self.get_label(c, f"group.{g}.priority", 100))

if g not in containers_by_group_in_process:
containers_by_group_in_process[g] = [(priority, c)]
Expand Down Expand Up @@ -240,33 +245,33 @@ def _run_exec(

if attached_to_container == True and c:
if when == BeforeAfterorDuring.BEFORE:
curl_command = str(c.labels.get("nautical-backup.curl.before", ""))
command = str(c.labels.get("nautical-backup.exec.before", curl_command))
curl_command = str(self.get_label(c, "curl.before", ""))
command = str(self.get_label(c, "exec.before", curl_command))

if curl_command and curl_command != "":
self.log_this(
"Deprecated: 'nautical-backup.curl.before' has been moved to 'nautical-backup.exec.before'",
f"Deprecated: '{self.prefix}.curl.before' has been moved to '{self.prefix}.exec.before'",
"WARN",
)
if command and command != "":
self.log_this("Running PRE-backup exec command for $name", "DEBUG")
elif when == BeforeAfterorDuring.DURING:
curl_command = str(c.labels.get("nautical-backup.curl.during", ""))
command = str(c.labels.get("nautical-backup.exec.during", curl_command))
curl_command = str(self.get_label(c, "curl.during", ""))
command = str(self.get_label(c, "exec.during", curl_command))

if curl_command and curl_command != "":
self.log_this(
"Deprecated: 'nautical-backup.curl.during' has been moved to 'nautical-backup.exec.during'",
f"Deprecated: '{self.prefix}.curl.during' has been moved to '{self.prefix}.exec.during'",
"WARN",
)
if command and command != "":
self.log_this("Running DURING-backup exec command for $name", "DEBUG")
elif when == BeforeAfterorDuring.AFTER:
curl_command = str(c.labels.get("nautical-backup.curl.after", ""))
command = str(c.labels.get("nautical-backup.exec.after", curl_command))
curl_command = str(self.get_label(c, "curl.after", ""))
command = str(self.get_label(c, "exec.after", curl_command))
if curl_command and curl_command != "":
self.log_this(
"Deprecated: 'nautical-backup.curl.after' has been moved to 'nautical-backup.exec.after'",
f"Deprecated: '{self.prefix}.curl.after' has been moved to '{self.prefix}.exec.after'",
"WARN",
)
if command and command != "":
Expand Down Expand Up @@ -322,13 +327,13 @@ def _run_exec(
def _run_lifecyle_hook(self, c: Container, when: BeforeOrAfter):
"""Runs a commend inside the child container"""
if when == BeforeOrAfter.BEFORE:
command = str(c.labels.get("nautical-backup.lifecycle.before", ""))
timeout = str(c.labels.get("nautical-backup.lifecycle.before.timeout", "60"))
command = str(self.get_label(c, "lifecycle.before", ""))
timeout = str(self.get_label(c, "lifecycle.before.timeout", "60"))
if command and command != "":
self.log_this("Running DURING-backup lifecycle hook for $name", "DEBUG")
elif when == BeforeOrAfter.AFTER:
timeout = str(c.labels.get("nautical-backup.lifecycle.after.timeout", "60"))
command = str(c.labels.get("nautical-backup.lifecycle.after", ""))
timeout = str(self.get_label(c, "lifecycle.after.timeout", "60"))
command = str(self.get_label(c, "lifecycle.after", ""))
if command and command != "":
self.log_this("Running AFTER-backup lifecycle hook for $name", "DEBUG")
else:
Expand All @@ -350,7 +355,7 @@ def _stop_container(self, c: Container, attempt=1) -> bool:
self.log_this(f"Container {c.name} is in SKIP_STOPPING list. Will not stop container.", "DEBUG")
return True

stop_before_backup = str(c.labels.get("nautical-backup.stop-before-backup", "true"))
stop_before_backup = str(self.get_label(c, "stop-before-backup", "true"))
if stop_before_backup.lower() == "false":
self.log_this(f"Skipping stopping of {c.name} because of label" "DEBUG")
return True
Expand Down Expand Up @@ -419,7 +424,7 @@ def _get_src_dir(self, c: Container, log=False) -> Tuple[Path, str]:
if log == True:
self.log_this(f"Overriding source directory for {c.id} to '{new_src_dir}'")

label_src = str(c.labels.get("nautical-backup.override-source-dir", ""))
label_src = str(self.get_label(c, "override-source-dir", ""))
if label_src and label_src != "":
if log == True:
self.log_this(f"Overriding source directory for {c.name} to '{label_src}' from label")
Expand All @@ -433,7 +438,7 @@ def _get_dest_dir(self, c: Container, src_dir_name: str) -> Tuple[Path, str]:
dest_dir_full: Path = base_dest_dir / str(c.name)
dest_dir_name = str(c.name)

keep_src_dir_name_label = str(c.labels.get("nautical-backup.keep_src_dir_name", "")).lower()
keep_src_dir_name_label = str(self.get_label(c, "keep_src_dir_name", "")).lower()

# This allows the user to set the KEEP_SRC_DIR_NAME environment variable to override the label's default if set
# But the label will still override the environment variable if set to false/true
Expand All @@ -460,7 +465,7 @@ def _get_dest_dir(self, c: Container, src_dir_name: str) -> Tuple[Path, str]:
dest_dir_name = new_dest_dir
self.log_this(f"Overriding destination directory for {c.id} to '{new_dest_dir}'")

label_dest = str(c.labels.get("nautical-backup.override-destination-dir", ""))
label_dest = str(self.get_label(c, "override-destination-dir", ""))
if label_dest and label_dest != "":
self.log_this(f"Overriding destination directory for {c.name} to '{label_dest}' from label")
dest_dir_full = base_dest_dir / label_dest
Expand Down Expand Up @@ -545,7 +550,7 @@ def _backup_additional_folders_standalone(self, when: BeforeOrAfter, base_dest_d
self._run_rsync(None, rsync_args, src_dir, dest_dir)

def _backup_additional_folders(self, c: Container, base_dest_dir: Path):
additional_folders = str(c.labels.get("nautical-backup.additional-folders", ""))
additional_folders = str(self.get_label(c, "additional-folders", ""))
base_src_dir = Path(self.env.SOURCE_LOCATION)

rsync_args = self._get_rsync_args(c, log=False)
Expand Down Expand Up @@ -577,7 +582,7 @@ def _backup_container_folders(self, c: Container, dest_path: Optional[Path] = No
else: # Only container given (no secondary dest)
dest_path = Path(self.env.DEST_LOCATION)

src_dir_required = str(c.labels.get("nautical-backup.source-dir-required", "true")).lower()
src_dir_required = str(self.get_label(c, "source-dir-required", "true")).lower()
if src_dir_required == "true":
self.verify_destination_location(dest_path)
if not dest_dir.exists():
Expand All @@ -595,7 +600,7 @@ def _backup_container_folders(self, c: Container, dest_path: Optional[Path] = No
else:
self.log_this(f"Source directory {src_dir} does not exist. Skipping", "DEBUG")

additional_folders_when = str(c.labels.get("nautical-backup.additional-folders.when", "during")).lower()
additional_folders_when = str(self.get_label(c, "additional-folders.when", "during")).lower()
if not additional_folders_when or additional_folders_when == "during":
self._backup_additional_folders(c, dest_path)

Expand All @@ -622,7 +627,7 @@ def _get_rsync_args(self, c: Optional[Container], log=False) -> str:
default_rsync_args = ""

if c:
use_default_args = str(c.labels.get("nautical-backup.use-default-rsync-args", "")).lower()
use_default_args = str(self.get_label(c, "use-default-rsync-args", "")).lower()
if use_default_args == "false":
if log == True:
self.log_this(f"Disabling default rsync arguments ({self.env.DEFAULT_RNC_ARGS})", "DEBUG")
Expand All @@ -635,7 +640,7 @@ def _get_rsync_args(self, c: Optional[Container], log=False) -> str:
used_default_args = False

if c:
custom_rsync_args_label = str(c.labels.get("nautical-backup.rsync-custom-args", ""))
custom_rsync_args_label = str(self.get_label(c, "rsync-custom-args", ""))
if custom_rsync_args_label != "":
if log == True:
self.log_this(f"Setting custom rsync args from label ({custom_rsync_args_label})", "DEBUG")
Expand Down Expand Up @@ -694,14 +699,14 @@ def backup(self):
self._run_exec(c, BeforeAfterorDuring.BEFORE, attached_to_container=True)
self._run_lifecyle_hook(c, BeforeOrAfter.BEFORE)

additional_folders_when = str(c.labels.get("nautical-backup.additional-folders.when", "during")).lower()
additional_folders_when = str(self.get_label(c, "additional-folders.when", "during")).lower()
if additional_folders_when == "before":
for dir in dest_dirs:
self._backup_additional_folders(c, dir)

src_dir, src_dir_no_path = self._get_src_dir(c)
if not src_dir.exists():
src_dir_required = str(c.labels.get("nautical-backup.source-dir-required", "true")).lower()
src_dir_required = str(self.get_label(c, "source-dir-required", "true")).lower()
if src_dir_required == "false":
self.log_this(
f"{c.name} - Source directory '{src_dir}' does not exist, but that's okay", "DEBUG"
Expand All @@ -718,7 +723,7 @@ def backup(self):
# Backup containers
c.reload() # Refresh the status for this container
if c.status != "exited":
stop_before_backup = str(c.labels.get("nautical-backup.stop-before-backup", "true"))
stop_before_backup = str(self.get_label(c, "stop-before-backup", "true"))

# Allow the user to skip stopping the container before backup
# Here we allow the Enviorment variable to supercede the EMPTY label
Expand Down Expand Up @@ -750,7 +755,7 @@ def backup(self):
self._run_lifecyle_hook(c, BeforeOrAfter.AFTER)
self._run_exec(c, BeforeAfterorDuring.AFTER, attached_to_container=True)

additional_folders_when = str(c.labels.get("nautical-backup.additional-folders.when", "during")).lower()
additional_folders_when = str(self.get_label(c, "additional-folders.when", "during")).lower()
if additional_folders_when == "after":
for dir in dest_dirs:
self._backup_additional_folders(c, dir)
Expand Down
3 changes: 3 additions & 0 deletions app/defaults.env
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,9 @@ USE_DEFAULT_RSYNC_ARGS=true
# Require the Docker Label `nautical-backup.enable=true` to be present on each container or it will be skipped
REQUIRE_LABEL=false

# Label prefix
LABEL_PREFIX=nautical-backup

# Set the default log level to INFO
LOG_LEVEL=INFO

Expand Down
1 change: 1 addition & 0 deletions app/nautical_env.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ def __init__(self) -> None:
self.REQUIRE_LABEL = False
if os.environ.get("REQUIRE_LABEL", "False").lower() == "true":
self.REQUIRE_LABEL = True
self.LABEL_PREFIX = os.environ.get("LABEL_PREFIX", "nautical-backup")

self.NAUTICAL_DB_PATH = os.environ.get("NAUTICAL_DB_PATH", "")

Expand Down
24 changes: 24 additions & 0 deletions docs/arguments.md
Original file line number Diff line number Diff line change
Expand Up @@ -252,6 +252,30 @@ REQUIRE_LABEL=true

See the [Enable or Disable Nautical](./labels.md#enable-or-disable-nautical) Label Section for more details.

## Label Prefix
By default, nautical-backup will only scan containers having labels starting with `nautical-backup.*`. To be able to run multiple instances of nautical-backup, you may want to customize the label prefix to avoid colision among instances.

> **Default**: nautical-backup
```properties
services:
# Multiple instance of nautical
nautical-inst1:
environment:
- LABEL_PREFIX=nautical-backup.inst1
nautical-inst2:
environment:
- LABEL_PREFIX=nautical-backup.inst2

# Multi targets
backuped-inst1:
labels:
- nautical-backup.inst1.enable=true
backuped-inst2:
labels:
- nautical-backup.inst2.enable=true
```

## Override Source Directory
Allows a source directory and container-name that do not match.

Expand Down

0 comments on commit 87c111b

Please sign in to comment.