2024-07-02 19:35:28 +00:00
|
|
|
from abc import ABC, abstractmethod
|
|
|
|
import re
|
|
|
|
from typing import Optional
|
|
|
|
|
2024-07-23 14:05:29 +00:00
|
|
|
from selfprivacy_api.utils import check_if_subdomain_is_taken
|
|
|
|
|
2024-07-02 19:35:28 +00:00
|
|
|
|
2024-07-03 13:35:40 +00:00
|
|
|
class ServiceConfigItem(ABC):
|
2024-07-02 19:35:28 +00:00
|
|
|
id: str
|
|
|
|
description: str
|
|
|
|
widget: str
|
|
|
|
type: str
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def get_value(self, service_options):
|
|
|
|
pass
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def set_value(self, value, service_options):
|
|
|
|
pass
|
|
|
|
|
2024-07-23 14:05:29 +00:00
|
|
|
@abstractmethod
|
|
|
|
def validate_value(self, value):
|
|
|
|
return True
|
|
|
|
|
2024-07-02 19:35:28 +00:00
|
|
|
def as_dict(self, service_options):
|
|
|
|
return {
|
|
|
|
"id": self.id,
|
|
|
|
"type": self.type,
|
|
|
|
"description": self.description,
|
|
|
|
"widget": self.widget,
|
|
|
|
"value": self.get_value(service_options),
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-07-03 13:35:40 +00:00
|
|
|
class StringServiceConfigItem(ServiceConfigItem):
|
2024-07-02 19:35:28 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
id: str,
|
|
|
|
default_value: str,
|
|
|
|
description: str,
|
|
|
|
regex: Optional[str] = None,
|
|
|
|
widget: Optional[str] = None,
|
2024-07-23 14:05:29 +00:00
|
|
|
allow_empty: bool = False,
|
2024-07-02 19:35:28 +00:00
|
|
|
):
|
2024-07-23 14:05:29 +00:00
|
|
|
if widget == "subdomain" and not regex:
|
|
|
|
raise ValueError("Subdomain widget requires regex")
|
2024-07-02 19:35:28 +00:00
|
|
|
self.id = id
|
|
|
|
self.type = "string"
|
|
|
|
self.default_value = default_value
|
|
|
|
self.description = description
|
|
|
|
self.regex = re.compile(regex) if regex else None
|
|
|
|
self.widget = widget if widget else "text"
|
2024-07-23 14:05:29 +00:00
|
|
|
self.allow_empty = allow_empty
|
2024-07-02 19:35:28 +00:00
|
|
|
|
|
|
|
def get_value(self, service_options):
|
|
|
|
return service_options.get(self.id, self.default_value)
|
|
|
|
|
|
|
|
def set_value(self, value, service_options):
|
2024-07-23 14:05:29 +00:00
|
|
|
if not self.validate_value(value):
|
|
|
|
raise ValueError(f"Value {value} is not valid")
|
2024-07-02 19:35:28 +00:00
|
|
|
if self.regex and not self.regex.match(value):
|
|
|
|
raise ValueError(f"Value {value} does not match regex {self.regex}")
|
|
|
|
service_options[self.id] = value
|
|
|
|
|
2024-07-15 14:44:13 +00:00
|
|
|
def as_dict(self, service_options):
|
|
|
|
return {
|
|
|
|
"id": self.id,
|
|
|
|
"type": self.type,
|
|
|
|
"description": self.description,
|
|
|
|
"widget": self.widget,
|
|
|
|
"value": self.get_value(service_options),
|
|
|
|
"default_value": self.default_value,
|
|
|
|
"regex": self.regex.pattern if self.regex else None,
|
|
|
|
}
|
|
|
|
|
2024-07-23 14:05:29 +00:00
|
|
|
def validate_value(self, value):
|
|
|
|
if not isinstance(value, str):
|
|
|
|
return False
|
|
|
|
if not self.allow_empty and not value:
|
|
|
|
return False
|
|
|
|
if self.regex and not self.regex.match(value):
|
|
|
|
return False
|
|
|
|
if self.widget == "subdomain":
|
|
|
|
if check_if_subdomain_is_taken(value):
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
2024-07-02 19:35:28 +00:00
|
|
|
|
2024-07-03 13:35:40 +00:00
|
|
|
class BoolServiceConfigItem(ServiceConfigItem):
|
2024-07-02 19:35:28 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
id: str,
|
|
|
|
default_value: bool,
|
|
|
|
description: str,
|
|
|
|
widget: Optional[str] = None,
|
|
|
|
):
|
|
|
|
self.id = id
|
|
|
|
self.type = "bool"
|
|
|
|
self.default_value = default_value
|
|
|
|
self.description = description
|
|
|
|
self.widget = widget if widget else "switch"
|
|
|
|
|
|
|
|
def get_value(self, service_options):
|
|
|
|
return service_options.get(self.id, self.default_value)
|
|
|
|
|
|
|
|
def set_value(self, value, service_options):
|
2024-07-23 14:05:29 +00:00
|
|
|
if not isinstance(value, bool):
|
|
|
|
raise ValueError(f"Value {value} is not a boolean")
|
2024-07-02 19:35:28 +00:00
|
|
|
service_options[self.id] = value
|
2024-07-03 10:26:26 +00:00
|
|
|
|
2024-07-15 14:44:13 +00:00
|
|
|
def as_dict(self, service_options):
|
|
|
|
return {
|
|
|
|
"id": self.id,
|
|
|
|
"type": self.type,
|
|
|
|
"description": self.description,
|
|
|
|
"widget": self.widget,
|
|
|
|
"value": self.get_value(service_options),
|
|
|
|
"default_value": self.default_value,
|
|
|
|
}
|
|
|
|
|
2024-07-23 14:05:29 +00:00
|
|
|
def validate_value(self, value):
|
|
|
|
return isinstance(value, bool)
|
|
|
|
|
2024-07-03 10:26:26 +00:00
|
|
|
|
2024-07-03 13:35:40 +00:00
|
|
|
class EnumServiceConfigItem(ServiceConfigItem):
|
2024-07-03 10:26:26 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
id: str,
|
|
|
|
default_value: str,
|
|
|
|
description: str,
|
|
|
|
options: list[str],
|
|
|
|
widget: Optional[str] = None,
|
|
|
|
):
|
|
|
|
self.id = id
|
|
|
|
self.type = "enum"
|
|
|
|
self.default_value = default_value
|
|
|
|
self.description = description
|
|
|
|
self.options = options
|
|
|
|
self.widget = widget if widget else "select"
|
|
|
|
|
|
|
|
def get_value(self, service_options):
|
|
|
|
return service_options.get(self.id, self.default_value)
|
|
|
|
|
|
|
|
def set_value(self, value, service_options):
|
2024-07-23 14:05:29 +00:00
|
|
|
if not self.validate_value(value):
|
|
|
|
raise ValueError(f"Value {value} is not valid")
|
2024-07-03 10:26:26 +00:00
|
|
|
if value not in self.options:
|
|
|
|
raise ValueError(f"Value {value} not in options {self.options}")
|
|
|
|
service_options[self.id] = value
|
2024-07-15 14:44:13 +00:00
|
|
|
|
|
|
|
def as_dict(self, service_options):
|
|
|
|
return {
|
|
|
|
"id": self.id,
|
|
|
|
"type": self.type,
|
|
|
|
"description": self.description,
|
|
|
|
"widget": self.widget,
|
|
|
|
"value": self.get_value(service_options),
|
|
|
|
"default_value": self.default_value,
|
|
|
|
"options": self.options,
|
|
|
|
}
|
2024-07-15 16:00:31 +00:00
|
|
|
|
2024-07-23 14:05:29 +00:00
|
|
|
def validate_value(self, value):
|
|
|
|
if not isinstance(value, str):
|
|
|
|
return False
|
|
|
|
return value in self.options
|
|
|
|
|
2024-07-15 16:00:31 +00:00
|
|
|
|
|
|
|
# TODO: unused for now
|
|
|
|
class IntServiceConfigItem(ServiceConfigItem):
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
id: str,
|
|
|
|
default_value: int,
|
|
|
|
description: str,
|
|
|
|
widget: Optional[str] = None,
|
|
|
|
min_value: Optional[int] = None,
|
|
|
|
max_value: Optional[int] = None,
|
|
|
|
) -> None:
|
|
|
|
self.id = id
|
|
|
|
self.type = "int"
|
|
|
|
self.default_value = default_value
|
|
|
|
self.description = description
|
|
|
|
self.widget = widget if widget else "number"
|
|
|
|
self.min_value = min_value
|
|
|
|
self.max_value = max_value
|
|
|
|
|
|
|
|
def get_value(self, service_options):
|
|
|
|
return service_options.get(self.id, self.default_value)
|
|
|
|
|
|
|
|
def set_value(self, value, service_options):
|
|
|
|
if self.min_value is not None and value < self.min_value:
|
|
|
|
raise ValueError(f"Value {value} is less than min_value {self.min_value}")
|
|
|
|
if self.max_value is not None and value > self.max_value:
|
2024-07-23 14:05:29 +00:00
|
|
|
raise ValueError(
|
|
|
|
f"Value {value} is greater than max_value {self.max_value}"
|
|
|
|
)
|
2024-07-15 16:00:31 +00:00
|
|
|
service_options[self.id] = value
|
|
|
|
|
|
|
|
def as_dict(self, service_options):
|
|
|
|
return {
|
|
|
|
"id": self.id,
|
|
|
|
"type": self.type,
|
|
|
|
"description": self.description,
|
|
|
|
"widget": self.widget,
|
|
|
|
"value": self.get_value(service_options),
|
|
|
|
"default_value": self.default_value,
|
|
|
|
"min_value": self.min_value,
|
|
|
|
"max_value": self.max_value,
|
|
|
|
}
|
2024-07-23 14:05:29 +00:00
|
|
|
|
|
|
|
def validate_value(self, value):
|
|
|
|
if not isinstance(value, int):
|
|
|
|
return False
|
|
|
|
return (self.min_value is None or value >= self.min_value) and (
|
|
|
|
self.max_value is None or value <= self.max_value
|
|
|
|
)
|