mirror of
https://github.com/freedombox/FreedomBox.git
synced 2026-01-28 08:03:36 +00:00
- Try to mark class variables in component classes. - Leave typing hints generic, such as 'list' and 'dict' where content is usually not filled, too complex, or context is unimportant. - backups: Handle failure for tarfile extraction so that methods are not called on potentially None valued variables. - backups: Prevent potentially passing a keyword argument twice. - dynamicdns: Deal properly with outcome of urlparsing. - ejabberd: Deal with failed regex match - email: Fix a mypy compliant when iterating a filtered list. - tor: Don't reuse variables for different typed values. - tor: Don't reuse variables for different typed values. - operation: Return None explicitly. - operation: Ensure that keyword argument is not repeated. Tests: - Where only typing hints were modified and no syntax error came up, additional testing was not done. - `mypy --ignore-missing-imports .` run successfully. - Generate developer documentation. - Service runs without errors upon start up. - backups: Listing and restoring specific apps from a backup works. - backups: Mounting a remote backup repository works. - NOT TESTED: dynamicdns: Migrating from old style configuration works. - ejabberd: Verify that setting coturn configuration works. - email: Test that showing configuration from postfix works. - tor: Orport value is properly shown. - transmission: Configuration values are properly set. - users: Running unit tests as root works. - operation: Operation status messages are show properly during app install. - ./setup.py install runs Signed-off-by: Sunil Mohan Adapa <sunil@medhas.org> Reviewed-by: James Valleroy <jvalleroy@mailbox.org>
222 lines
7.1 KiB
Python
222 lines
7.1 KiB
Python
# SPDX-License-Identifier: AGPL-3.0-or-later
|
|
"""
|
|
App component to introduce a new domain type.
|
|
"""
|
|
|
|
from typing import ClassVar
|
|
|
|
from django.utils.translation import gettext_lazy as _
|
|
|
|
from plinth import app
|
|
|
|
_SERVICES = {
|
|
'__all__': {
|
|
'display_name': _('All'),
|
|
'port': None
|
|
},
|
|
'http': {
|
|
'display_name': _('All web apps'),
|
|
'port': 80
|
|
},
|
|
'https': {
|
|
'display_name': _('All web apps'),
|
|
'port': 443
|
|
},
|
|
'ssh': {
|
|
'display_name': _('Secure Shell'),
|
|
'port': 22
|
|
},
|
|
}
|
|
|
|
|
|
class DomainType(app.FollowerComponent):
|
|
"""Component to create a new type of domain.
|
|
|
|
It serves the primary purpose of showing a entry in the domain names page.
|
|
This will allow users to discover the new type of domain and use the
|
|
providing app to create that type of domain.
|
|
|
|
Similar to a menu entry, domain type information is available to the user
|
|
even when the corresponding app is disabled.
|
|
|
|
"""
|
|
|
|
_all: ClassVar[dict[str, 'DomainType']] = {}
|
|
|
|
def __init__(self, component_id, display_name, configuration_url,
|
|
can_have_certificate=True):
|
|
"""Initialize the domain type component.
|
|
|
|
component_id should be a unique ID across all components of an app and
|
|
across all components. This will also act as the 'type' parameter for
|
|
each created domain.
|
|
|
|
display_name is the type of domain displayed to the user in the
|
|
interface.
|
|
|
|
configuration_url is the Django URL to which a user is redirected to in
|
|
order to create or manage a domain of this type.
|
|
|
|
can_have_certificate indicates if this type of domain can have a TLS
|
|
certificate that can be validated by a typical browser.
|
|
|
|
"""
|
|
super().__init__(component_id)
|
|
|
|
self.display_name = display_name
|
|
self.configuration_url = configuration_url
|
|
self.can_have_certificate = can_have_certificate
|
|
|
|
self._all[component_id] = self
|
|
|
|
@classmethod
|
|
def get(cls, component_id):
|
|
"""Return a component of given ID."""
|
|
return cls._all[component_id]
|
|
|
|
@classmethod
|
|
def list(cls):
|
|
"""Return a list of all domain types."""
|
|
return dict(cls._all)
|
|
|
|
|
|
class DomainName(app.FollowerComponent):
|
|
"""Component to represent a domain name and its properties.
|
|
|
|
Each domain name is necessarily attached to a domain type component that
|
|
must be created prior to creating the domain name.
|
|
|
|
When an application providing or managing a domain name is disabled, the
|
|
corresponding domain name should become unavailable for others apps and
|
|
they must de-configure the domain name from app configuration. This is the
|
|
primary reason for making a domain name available as a component.
|
|
|
|
"""
|
|
_all: ClassVar[dict[str, 'DomainName']] = {}
|
|
|
|
def __init__(self, component_id, name, domain_type, services):
|
|
"""Initialize a domain name.
|
|
|
|
component_id should be a unique ID across all components of an app and
|
|
across all components. The value is typically 'domain-{app}-{domain}'.
|
|
This ensures that if the same domain is managed by multiple apps, it is
|
|
available as multiple instances. When one instance is removed, say by
|
|
disabling that app, the other instance will still provide that domain.
|
|
|
|
name is the domain name that is being represented by the component.
|
|
This should be fully qualified domain name.
|
|
|
|
domain_type should be a string representing the type of the domain.
|
|
This is the component ID of the domain type earlier registered by the
|
|
app that is creating the domain name component.
|
|
|
|
services is a list of string identifiers for services potentially
|
|
provided by the domain. For example, 'ssh' for secure shell service
|
|
provided on port 22. It is used for showing information to the user and
|
|
to retrieve a list of a domains that an app may use.
|
|
|
|
A service value can also be an integer in which case it will be
|
|
converted to a string by looking up a list of known services. This
|
|
process is not perfect and may cause problems when filtering domains
|
|
that could potentially provide a service.
|
|
|
|
The most common value of a services parameter is the string '__all__'
|
|
indicating that the domain can potentially provide any service without
|
|
limitations due to the nature of the domain name.
|
|
|
|
"""
|
|
super().__init__(component_id)
|
|
|
|
self.name = name
|
|
self.domain_type = DomainType.get(domain_type)
|
|
self._services = DomainName._normalize_services(services)
|
|
|
|
self._all[component_id] = self
|
|
|
|
@property
|
|
def services(self):
|
|
"""Read-only property to get the list of services."""
|
|
return self._services
|
|
|
|
@staticmethod
|
|
def _normalize_services(services):
|
|
"""If ports numbers are provided convert them to service IDs."""
|
|
if services == '__all__':
|
|
return services
|
|
|
|
return [DomainName._normalize_service(service) for service in services]
|
|
|
|
@staticmethod
|
|
def _normalize_service(service):
|
|
"""Return the service ID for a given port number.
|
|
|
|
XXX: Eliminate this and use a generalized approach eventually.
|
|
|
|
"""
|
|
if isinstance(service, str):
|
|
return service
|
|
|
|
if not isinstance(service, int):
|
|
raise ValueError('Invalid service')
|
|
|
|
for service_id, description in _SERVICES.items():
|
|
if description['port'] == service:
|
|
return service_id
|
|
|
|
return str(service)
|
|
|
|
def get_readable_services(self):
|
|
"""Return list of unique service strings that can be shown to user."""
|
|
services = self.services
|
|
if self.services == '__all__':
|
|
services = [services]
|
|
|
|
return {
|
|
_SERVICES.get(service, {'display_name': service})['display_name']
|
|
for service in services
|
|
}
|
|
|
|
def has_service(self, service):
|
|
"""Return whether a service is available for this domain name."""
|
|
return (service is None or self.services == '__all__'
|
|
or service in self.services)
|
|
|
|
def remove(self):
|
|
"""Remove the domain name from global list of domains.
|
|
|
|
It is acceptable to call remove() multiple times.
|
|
|
|
"""
|
|
try:
|
|
del self._all[self.component_id]
|
|
except KeyError:
|
|
pass
|
|
|
|
@classmethod
|
|
def get(cls, component_id):
|
|
"""Return the domain name object given name and app."""
|
|
return cls._all[component_id]
|
|
|
|
@classmethod
|
|
def list(cls, filter_for_service=None):
|
|
"""Return list of domains."""
|
|
return [
|
|
domain for domain in cls._all.values()
|
|
if domain.has_service(filter_for_service)
|
|
]
|
|
|
|
@classmethod
|
|
def list_names(cls, filter_for_service=None):
|
|
"""Return a set of unique domain names.
|
|
|
|
Multiple different components may provide the same domain name. This
|
|
method could be used to retrieve a list of all domain names without
|
|
duplication.
|
|
|
|
"""
|
|
return {
|
|
domain.name
|
|
for domain in cls._all.values()
|
|
if domain.has_service(filter_for_service)
|
|
}
|