2021-01-30 19:25:00 +01:00
|
|
|
# ContentDB
|
|
|
|
# Copyright (C) 2018-21 rubenwardy
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU Affero 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 Affero General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU Affero General Public License
|
|
|
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2021-05-06 15:45:59 +02:00
|
|
|
|
2022-01-25 21:48:37 +01:00
|
|
|
import re
|
2021-05-06 15:45:59 +02:00
|
|
|
import secrets
|
2023-05-12 01:17:15 +02:00
|
|
|
from typing import Dict
|
|
|
|
|
2023-05-12 01:45:27 +02:00
|
|
|
import deep_compare
|
2023-06-19 20:32:36 +02:00
|
|
|
from flask import current_app
|
2023-05-12 01:45:27 +02:00
|
|
|
|
2021-01-30 19:25:00 +01:00
|
|
|
from .flask import *
|
|
|
|
from .models import *
|
|
|
|
from .user import *
|
|
|
|
|
|
|
|
YESES = ["yes", "true", "1", "on"]
|
|
|
|
|
2021-02-02 01:07:41 +01:00
|
|
|
|
2024-06-22 12:11:57 +02:00
|
|
|
def is_username_valid(username: str) -> bool:
|
2022-10-13 20:23:18 +02:00
|
|
|
return username is not None and len(username) >= 2 and \
|
2022-10-14 13:52:29 +02:00
|
|
|
re.match(r"^[A-Za-z0-9._-]*$", username) and not re.match(r"^\.*$", username)
|
2022-01-21 22:20:04 +01:00
|
|
|
|
|
|
|
|
2024-06-22 12:11:57 +02:00
|
|
|
def make_valid_username(username: str) -> str:
|
|
|
|
return re.sub(r"[^A-Za-z0-9._-]+", "_", username)
|
|
|
|
|
|
|
|
|
2023-06-19 22:27:49 +02:00
|
|
|
def is_yes(val):
|
2021-01-30 19:25:00 +01:00
|
|
|
return val and val.lower() in YESES
|
|
|
|
|
2021-02-02 01:07:41 +01:00
|
|
|
|
2023-06-19 22:27:49 +02:00
|
|
|
def is_no(val):
|
|
|
|
return val and not is_yes(val)
|
2021-01-30 19:25:00 +01:00
|
|
|
|
2021-02-02 01:07:41 +01:00
|
|
|
|
2023-06-19 22:27:49 +02:00
|
|
|
def nonempty_or_none(str):
|
2021-01-30 19:25:00 +01:00
|
|
|
if str is None or str == "":
|
|
|
|
return None
|
|
|
|
|
|
|
|
return str
|
2021-02-02 01:07:41 +01:00
|
|
|
|
|
|
|
|
2024-06-22 14:22:37 +02:00
|
|
|
def normalize_line_endings(value: Optional[str]) -> Optional[str]:
|
|
|
|
if value is None:
|
|
|
|
return None
|
|
|
|
|
2024-08-26 12:56:30 +02:00
|
|
|
return value.replace("\r\n", "\n")
|
2024-06-22 14:22:37 +02:00
|
|
|
|
|
|
|
|
2023-06-19 22:27:49 +02:00
|
|
|
def should_return_json():
|
2021-02-02 01:07:41 +01:00
|
|
|
return "application/json" in request.accept_mimetypes and \
|
|
|
|
not "text/html" in request.accept_mimetypes
|
|
|
|
|
|
|
|
|
2023-06-19 22:27:49 +02:00
|
|
|
def random_string(n):
|
2021-05-06 15:45:59 +02:00
|
|
|
return secrets.token_hex(int(n / 2))
|
2023-01-03 13:17:01 +01:00
|
|
|
|
|
|
|
|
|
|
|
def has_blocked_domains(text: str, username: str, location: str) -> bool:
|
|
|
|
if text is None:
|
|
|
|
return False
|
|
|
|
|
|
|
|
blocked_domains = current_app.config["BLOCKED_DOMAINS"]
|
|
|
|
for domain in blocked_domains:
|
|
|
|
if domain in text:
|
|
|
|
from app.tasks.webhooktasks import post_discord_webhook
|
|
|
|
post_discord_webhook.delay(username,
|
2023-01-12 17:43:08 +01:00
|
|
|
f"Attempted to post blocked domain {domain} in {location}",
|
2023-01-03 13:17:01 +01:00
|
|
|
True)
|
|
|
|
return True
|
|
|
|
|
|
|
|
return False
|
2023-05-12 01:17:15 +02:00
|
|
|
|
|
|
|
|
2023-05-12 01:45:27 +02:00
|
|
|
def diff_dictionaries(one: Dict, two: Dict) -> List:
|
2023-05-12 01:17:15 +02:00
|
|
|
if len(set(one.keys()).difference(set(two.keys()))) != 0:
|
|
|
|
raise "Mismatching keys"
|
|
|
|
|
|
|
|
retval = []
|
|
|
|
|
|
|
|
for key, before in one.items():
|
|
|
|
after = two[key]
|
|
|
|
|
2023-05-12 01:45:27 +02:00
|
|
|
if isinstance(before, dict):
|
2023-05-12 01:17:15 +02:00
|
|
|
diff = diff_dictionaries(before, after)
|
|
|
|
if len(diff) != 0:
|
|
|
|
retval.append({
|
|
|
|
"key": key,
|
|
|
|
"changes": diff,
|
|
|
|
})
|
2023-05-12 01:45:27 +02:00
|
|
|
elif not deep_compare.CompareVariables.compare(before, after):
|
2023-05-12 01:17:15 +02:00
|
|
|
retval.append({
|
|
|
|
"key": key,
|
|
|
|
"before": before,
|
|
|
|
"after": after,
|
|
|
|
})
|
|
|
|
|
|
|
|
return retval
|
|
|
|
|
|
|
|
|
|
|
|
def describe_difference(diff: List, available_space: int) -> typing.Optional[str]:
|
|
|
|
if len(diff) == 0 or available_space <= 0:
|
|
|
|
return None
|
|
|
|
|
|
|
|
if len(diff) == 1 and "before" in diff[0] and "after" in diff[0]:
|
|
|
|
key = diff[0]["key"]
|
|
|
|
before = diff[0]["before"]
|
|
|
|
after = diff[0]["after"]
|
|
|
|
|
|
|
|
if isinstance(before, str) and isinstance(after, str):
|
2023-05-12 01:20:38 +02:00
|
|
|
if len(before) + len(after) <= available_space + 30:
|
|
|
|
return f"{key}: {before} -> {after}"
|
2023-05-12 01:17:15 +02:00
|
|
|
|
2023-05-12 01:20:38 +02:00
|
|
|
elif isinstance(before, list) and isinstance(after, list):
|
2023-05-12 01:17:15 +02:00
|
|
|
removed = []
|
|
|
|
added = []
|
|
|
|
for x in before:
|
|
|
|
if x not in after:
|
|
|
|
removed.append(x)
|
|
|
|
for x in after:
|
|
|
|
if x not in before:
|
|
|
|
added.append(x)
|
|
|
|
|
|
|
|
parts = ["-" + str(x) for x in removed] + ["+" + str(x) for x in added]
|
|
|
|
return f"{key}: {', '.join(parts)}"
|
|
|
|
|
|
|
|
return ", ".join([x["key"] for x in diff])
|