Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added config file for the Ruff Python linter and fixed additional errors #2473

Open
wants to merge 54 commits into
base: main
Choose a base branch
from
Open
Changes from all commits
Commits
Show all changes
54 commits
Select commit Hold shift + click to select a range
c659fe1
Fixed syntax in readable_bash.py.
tdulcet Jan 27, 2024
2dd083f
Added Ruff config for Python code.
tdulcet Jan 8, 2025
de9781a
Fixed UP031 (printf-string-formatting): Use format specifiers instead…
tdulcet Jan 8, 2025
70aff05
Fixed FURB118 (reimplemented-operator)
tdulcet Jan 8, 2025
fda9b09
Fixed PLW0120 (useless-else-on-loop): `else` clause on loop without a…
tdulcet Jan 8, 2025
18f4baf
Fixed RET505 (superfluous-else-return)
tdulcet Jan 8, 2025
9cd5f4a
Fixed PLR6104 (non-augmented-assignment): Use `+=` to perform an augm…
tdulcet Jan 8, 2025
1d79595
Fixed FURB188 (slice-to-remove-prefix-or-suffix): Prefer `removeprefi…
tdulcet Jan 8, 2025
d9d6e96
Fixed RUF051 (if-key-in-dict-del): Use `pop` instead of `key in dict`…
tdulcet Jan 8, 2025
0c44319
Fixed SIM103 (needless-bool): Return the condition `not "admin" not i…
tdulcet Jan 8, 2025
3f7e733
Fixed F401 (unused-import): `contextlib` imported but unused
tdulcet Jan 8, 2025
d31716a
Fixed RET507 (superfluous-else-continue): Unnecessary `elif` after `c…
tdulcet Jan 8, 2025
4e3e85c
Fixed RUF039 (unraw-re-pattern)
tdulcet Jan 8, 2025
80406de
Fixed RUF031 (incorrectly-parenthesized-tuple-in-subscript): Avoid pa…
tdulcet Jan 8, 2025
32189af
Fixed PGH004 (blanket-noqa): Use a colon when specifying `noqa` rule …
tdulcet Jan 8, 2025
2ddbe12
Fixed SIM101 (duplicate-isinstance-call): Multiple `isinstance` calls…
tdulcet Jan 8, 2025
4a7ea32
Fixed FURB142 (for-loop-set-mutations): Use of `set.add()` in a for loop
tdulcet Jan 8, 2025
386055b
Fixed UP032 (f-string): Use f-string instead of `format` call
tdulcet Jan 8, 2025
de5fc42
Fixed FURB110 (if-exp-instead-of-or-operator): Replace ternary `if` e…
tdulcet Jan 8, 2025
fd2e4fd
Fixed UP015 (redundant-open-modes): Unnecessary open mode parameters
tdulcet Jan 8, 2025
ca452e9
Fixed PLW1514 (unspecified-encoding): `open` in text mode without exp…
tdulcet Jan 8, 2025
430ee05
Fixed PLW1514 (unspecified-encoding): `open` in text mode without exp…
tdulcet Mar 21, 2025
a7f265c
Fixed PLR6201 (literal-membership): Use a set literal when testing fo…
tdulcet Jan 8, 2025
eaf9668
Fixed W605 (invalid-escape-sequence)
tdulcet Jan 8, 2025
85deb42
Fixed Q003 (avoidable-escaped-quote): Change outer quotes to avoid es…
tdulcet Jan 8, 2025
6b5638f
Fixed RUF055 (unnecessary-regular-expression): Plain string pattern p…
tdulcet Jan 8, 2025
73b748f
Fixed RET504 (unnecessary-assign): Unnecessary assignment to `v` befo…
tdulcet Jan 8, 2025
205f84b
Fixed RET506 (superfluous-else-raise): Unnecessary `elif` after `rais…
tdulcet Jan 8, 2025
c3343ce
Fixed EM102 (f-string-in-exception): Exception must not use an f-stri…
tdulcet Jan 8, 2025
9d81ff5
Fixed RUF010 (explicit-f-string-type-conversion): Use explicit conver…
tdulcet Jan 8, 2025
cd764e5
Fixed PLC1901 (compare-to-empty-string)
tdulcet Jan 12, 2025
1d6560a
Fixed UP031 (printf-string-formatting): Use format specifiers instead…
tdulcet Jan 12, 2025
c82b84c
Fixed TRY003 (raise-vanilla-args): Avoid specifying long messages out…
tdulcet Jan 12, 2025
4e1dd3f
Fixed RET505 (superfluous-else-return): Unnecessary `elif` after `ret…
tdulcet Jan 12, 2025
1c98189
Fixed G004 (logging-f-string): Logging statement uses f-string
tdulcet Jan 12, 2025
854e8f4
Fixed TRY300 (try-consider-else): Consider moving this statement to a…
tdulcet Jan 12, 2025
7ff9d08
Fixed B007 (unused-loop-control-variable)
tdulcet Jan 12, 2025
386e8a4
Fixed RUF005 (collection-literal-concatenation): Consider iterable un…
tdulcet Jan 12, 2025
9fb9af5
Fixed SIM115 (open-file-with-context-handler): Use a context manager …
tdulcet Jan 12, 2025
b305871
Fixed ARG005 (unused-lambda-argument): Unused lambda argument: `alias`
tdulcet Jan 12, 2025
6883f1d
Fixed RUF039 (unraw-re-pattern)
tdulcet Jan 12, 2025
ed2ad44
Fixed F841 (unused-variable): Local variable `conffile` is assigned t…
tdulcet Jan 12, 2025
0564d1a
Fixed UP032 (f-string): Use f-string instead of `format` call
tdulcet Jan 12, 2025
3d3311a
Fixed SIM117 (multiple-with-statements): Use a single `with` statemen…
tdulcet Jan 12, 2025
4e9c108
Fixed PERF102 (incorrect-dict-iterator): When using only the keys of …
tdulcet Jan 12, 2025
7e0015e
Explicitly removed temporary file and deleted outdated comment.
tdulcet Feb 15, 2025
1438be2
Fixed RET505 (superfluous-else-return): Unnecessary `else` after `ret…
tdulcet Mar 21, 2025
6fc0493
Fixed UP015 (redundant-open-modes): Unnecessary mode argument
tdulcet Mar 21, 2025
0402b23
Fixed FURB129 (readlines-in-for): Instead of calling `readlines()`, i…
tdulcet Mar 21, 2025
a502f3e
Fixed RUF059 (unused-unpacked-variable)
tdulcet Mar 21, 2025
44ba1e9
Fixed EM101 (raw-string-in-exception): Exception must not use a strin…
tdulcet Mar 21, 2025
6e23d9d
Fixed SIM118 (in-dict-keys): Use `key in dict` instead of `key in dic…
tdulcet Mar 21, 2025
5428356
Fixed FURB122 (for-loop-writes): Use of `f.write` in a for loop
tdulcet Mar 21, 2025
5afb9ab
Fixed EM102 (f-string-in-exception): Exception must not use an f-stri…
tdulcet Mar 21, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions management/auth.py
Original file line number Diff line number Diff line change
@@ -52,7 +52,7 @@ def decode(s):
msg = "Authorization header invalid."
raise ValueError(msg)

if username.strip() == "" and password.strip() == "":
if not username.strip() and not password.strip():
msg = "No email address, password, session key, or API key provided."
raise ValueError(msg)

@@ -73,7 +73,7 @@ def decode(s):
self.sessions[sessionid] = session

# If no password was given, but a username was given, we're missing some information.
elif password.strip() == "":
elif not password.strip():
msg = "Enter a password."
raise ValueError(msg)

81 changes: 39 additions & 42 deletions management/backup.py
Original file line number Diff line number Diff line change
@@ -14,6 +14,7 @@
from exclusiveprocess import Lock

from utils import load_environment, shell, wait_for_service
import operator

def backup_status(env):
# If backups are disabled, return no status.
@@ -32,14 +33,14 @@ def backup_status(env):
def reldate(date, ref, clip):
if ref < date: return clip
rd = dateutil.relativedelta.relativedelta(ref, date)
if rd.years > 1: return "%d years, %d months" % (rd.years, rd.months)
if rd.years == 1: return "%d year, %d months" % (rd.years, rd.months)
if rd.months > 1: return "%d months, %d days" % (rd.months, rd.days)
if rd.months == 1: return "%d month, %d days" % (rd.months, rd.days)
if rd.days >= 7: return "%d days" % rd.days
if rd.days > 1: return "%d days, %d hours" % (rd.days, rd.hours)
if rd.days == 1: return "%d day, %d hours" % (rd.days, rd.hours)
return "%d hours, %d minutes" % (rd.hours, rd.minutes)
if rd.years > 1: return f"{rd.years:d} years, {rd.months:d} months"
if rd.years == 1: return f"{rd.years:d} year, {rd.months:d} months"
if rd.months > 1: return f"{rd.months:d} months, {rd.days:d} days"
if rd.months == 1: return f"{rd.months:d} month, {rd.days:d} days"
if rd.days >= 7: return f"{rd.days:d} days"
if rd.days > 1: return f"{rd.days:d} days, {rd.hours:d} hours"
if rd.days == 1: return f"{rd.days:d} day, {rd.hours:d} hours"
return f"{rd.hours:d} hours, {rd.minutes:d} minutes"

# Get duplicity collection status and parse for a list of backups.
def parse_line(line):
@@ -91,7 +92,7 @@ def parse_line(line):

# Ensure the rows are sorted reverse chronologically.
# This is relied on by should_force_full() and the next step.
backups = sorted(backups.values(), key = lambda b : b["date"], reverse=True)
backups = sorted(backups.values(), key = operator.itemgetter("date"), reverse=True)

# Get the average size of incremental backups, the size of the
# most recent full backup, and the date of the most recent
@@ -129,7 +130,7 @@ def parse_line(line):
# It still can't be deleted until it's old enough.
est_deleted_on = max(est_time_of_next_full, first_date + datetime.timedelta(days=config["min_age_in_days"]))

deleted_in = "approx. %d days" % round((est_deleted_on-now).total_seconds()/60/60/24 + .5)
deleted_in = f"approx. {round((est_deleted_on-now).total_seconds()/60/60/24 + .5):d} days"

# When will a backup be deleted? Set the deleted_in field of each backup.
saw_full = False
@@ -177,10 +178,8 @@ def should_force_full(config, env):
if dateutil.parser.parse(bak["date"]) + datetime.timedelta(days=config["min_age_in_days"]*10+1) < datetime.datetime.now(dateutil.tz.tzlocal()):
return True
return False
else:
# If we got here there are no (full) backups, so make one.
# (I love for/else blocks. Here it's just to show off.)
return True
# If we got here there are no (full) backups, so make one.
return True

def get_passphrase(env):
# Get the encryption passphrase. secret_key.txt is 2048 random
@@ -192,7 +191,9 @@ def get_passphrase(env):
backup_root = os.path.join(env["STORAGE_ROOT"], 'backup')
with open(os.path.join(backup_root, 'secret_key.txt'), encoding="utf-8") as f:
passphrase = f.readline().strip()
if len(passphrase) < 43: raise Exception("secret_key.txt's first line is too short!")
if len(passphrase) < 43:
msg = "secret_key.txt's first line is too short!"
raise Exception(msg)

return passphrase

@@ -236,7 +237,7 @@ def get_duplicity_additional_args(env):
f"--ssh-options='-i /root/.ssh/id_rsa_miab -p {port}'",
f"--rsync-options='-e \"/usr/bin/ssh -oStrictHostKeyChecking=no -oBatchMode=yes -p {port} -i /root/.ssh/id_rsa_miab\"'",
]
elif get_target_type(config) == 's3':
if get_target_type(config) == 's3':
# See note about hostname in get_duplicity_target_url.
# The region name, which is required by some non-AWS endpoints,
# is saved inside the username portion of the URL.
@@ -346,7 +347,7 @@ def service_command(service, command, quit=None):
shell('check_call', [
"/usr/bin/duplicity",
"remove-older-than",
"%dD" % config["min_age_in_days"],
"{:d}D".format(config["min_age_in_days"]),
"--verbosity", "error",
"--archive-dir", backup_cache_dir,
"--force",
@@ -447,7 +448,7 @@ def list_target_files(config):
if target.scheme == "file":
return [(fn, os.path.getsize(os.path.join(target.path, fn))) for fn in os.listdir(target.path)]

elif target.scheme == "rsync":
if target.scheme == "rsync":
rsync_fn_size_re = re.compile(r'.* ([^ ]*) [^ ]* [^ ]* (.*)')
rsync_target = '{host}:{path}'

@@ -463,9 +464,8 @@ def list_target_files(config):

target_path = target.path
if not target_path.endswith('/'):
target_path = target_path + '/'
if target_path.startswith('/'):
target_path = target_path[1:]
target_path += "/"
target_path = target_path.removeprefix('/')

rsync_command = [ 'rsync',
'-e',
@@ -485,23 +485,22 @@ def list_target_files(config):
if match:
ret.append( (match.groups()[1], int(match.groups()[0].replace(',',''))) )
return ret
if 'Permission denied (publickey).' in listing:
reason = "Invalid user or check you correctly copied the SSH key."
elif 'No such file or directory' in listing:
reason = f"Provided path {target_path} is invalid."
elif 'Network is unreachable' in listing:
reason = f"The IP address {target.hostname} is unreachable."
elif 'Could not resolve hostname' in listing:
reason = f"The hostname {target.hostname} cannot be resolved."
else:
if 'Permission denied (publickey).' in listing:
reason = "Invalid user or check you correctly copied the SSH key."
elif 'No such file or directory' in listing:
reason = f"Provided path {target_path} is invalid."
elif 'Network is unreachable' in listing:
reason = f"The IP address {target.hostname} is unreachable."
elif 'Could not resolve hostname' in listing:
reason = f"The hostname {target.hostname} cannot be resolved."
else:
reason = ("Unknown error."
"Please check running 'management/backup.py --verify'"
"from mailinabox sources to debug the issue.")
msg = f"Connection to rsync host failed: {reason}"
raise ValueError(msg)
reason = ("Unknown error."
"Please check running 'management/backup.py --verify'"
"from mailinabox sources to debug the issue.")
msg = f"Connection to rsync host failed: {reason}"
raise ValueError(msg)

elif target.scheme == "s3":
if target.scheme == "s3":
import boto3.s3
from botocore.exceptions import ClientError

@@ -513,7 +512,7 @@ def list_target_files(config):
if path == '/':
path = ''

if bucket == "":
if not bucket:
msg = "Enter an S3 bucket name."
raise ValueError(msg)

@@ -531,7 +530,7 @@ def list_target_files(config):
except ClientError as e:
raise ValueError(e)
return backup_list
elif target.scheme == 'b2':
if target.scheme == 'b2':
from b2sdk.v1 import InMemoryAccountInfo, B2Api
from b2sdk.v1.exception import NonExistentBucket
info = InMemoryAccountInfo()
@@ -550,8 +549,7 @@ def list_target_files(config):
raise ValueError(msg)
return [(key.file_name, key.size) for key, _ in bucket.ls()]

else:
raise ValueError(config["target"])
raise ValueError(config["target"])


def backup_set_custom(env, target, target_user, target_pass, min_age):
@@ -605,8 +603,7 @@ def get_backup_config(env, for_save=False, for_ui=False):
# authentication details. The user will have to re-enter it.
if for_ui:
for field in ("target_user", "target_pass"):
if field in config:
del config[field]
config.pop(field, None)

# helper fields for the admin
config["file_target_directory"] = os.path.join(backup_root, 'encrypted')
2 changes: 1 addition & 1 deletion management/cli.py
Original file line number Diff line number Diff line change
@@ -120,7 +120,7 @@ def setup_key_auth(mgmt_uri):

elif sys.argv[1] == "user" and sys.argv[2] == "quota" and len(sys.argv) == 4:
# Get a user's quota
print(mgmt("/mail/users/quota?text=1&email=%s" % sys.argv[3]))
print(mgmt(f"/mail/users/quota?text=1&email={sys.argv[3]}"))

elif sys.argv[1] == "user" and sys.argv[2] == "quota" and len(sys.argv) == 5:
# Set a user's quota
Loading