2019-10-30 12:05:46 +00:00
|
|
|
import requests
|
2019-10-29 09:47:50 +00:00
|
|
|
import urllib.parse
|
|
|
|
|
|
|
|
from django.conf import settings
|
2019-12-17 09:25:10 +00:00
|
|
|
from django.http import Http404
|
2019-10-29 09:47:50 +00:00
|
|
|
from django.urls.exceptions import NoReverseMatch
|
2019-12-17 09:25:10 +00:00
|
|
|
from django.utils.translation import gettext_lazy as _
|
2019-10-29 09:47:50 +00:00
|
|
|
|
2019-12-12 13:18:29 +00:00
|
|
|
from .models import Domain, DatabaseService, MailService, SaasService, UserAccount
|
2019-12-10 12:04:04 +00:00
|
|
|
|
|
|
|
|
2019-10-29 09:47:50 +00:00
|
|
|
DOMAINS_PATH = 'domains/'
|
|
|
|
TOKEN_PATH = '/api-token-auth/'
|
|
|
|
|
|
|
|
API_PATHS = {
|
|
|
|
# auth
|
|
|
|
'token-auth': '/api-token-auth/',
|
2019-10-30 13:00:12 +00:00
|
|
|
'my-account': 'accounts/',
|
2019-10-29 09:47:50 +00:00
|
|
|
|
|
|
|
# services
|
2019-11-13 11:27:25 +00:00
|
|
|
'database-list': 'databases/',
|
2019-10-29 09:47:50 +00:00
|
|
|
'domain-list': 'domains/',
|
2019-12-13 14:08:01 +00:00
|
|
|
'domain-detail': 'domains/{pk}/',
|
2019-11-13 10:08:19 +00:00
|
|
|
'address-list': 'addresses/',
|
|
|
|
'mailbox-list': 'mailboxes/',
|
2019-10-31 13:08:49 +00:00
|
|
|
'mailinglist-list': 'lists/',
|
2019-12-06 09:27:18 +00:00
|
|
|
'saas-list': 'saas/',
|
2019-11-20 19:07:35 +00:00
|
|
|
|
|
|
|
# other
|
2019-12-17 13:48:21 +00:00
|
|
|
'bill-list': 'bills/',
|
2019-12-17 14:15:58 +00:00
|
|
|
'bill-document': 'bills/{pk}/document/',
|
2019-11-20 19:07:35 +00:00
|
|
|
'payment-source-list': 'payment-sources/',
|
2019-10-29 09:47:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-10-30 12:05:46 +00:00
|
|
|
class Orchestra(object):
|
|
|
|
def __init__(self, *args, username=None, password=None, **kwargs):
|
|
|
|
self.base_url = kwargs.pop('base_url', settings.API_BASE_URL)
|
|
|
|
self.username = username
|
|
|
|
self.session = requests.Session()
|
|
|
|
self.auth_token = kwargs.pop("auth_token", None)
|
2019-10-29 09:47:50 +00:00
|
|
|
|
2019-10-30 12:05:46 +00:00
|
|
|
if self.auth_token is None:
|
|
|
|
self.auth_token = self.authenticate(self.username, password)
|
|
|
|
|
|
|
|
def build_absolute_uri(self, path_name):
|
|
|
|
path = API_PATHS.get(path_name, None)
|
|
|
|
if path is None:
|
|
|
|
raise NoReverseMatch(
|
|
|
|
"Not found API path name '{}'".format(path_name))
|
|
|
|
|
|
|
|
return urllib.parse.urljoin(self.base_url, path)
|
|
|
|
|
|
|
|
def authenticate(self, username, password):
|
|
|
|
url = self.build_absolute_uri('token-auth')
|
|
|
|
response = self.session.post(
|
|
|
|
url,
|
|
|
|
data={"username": username, "password": password},
|
|
|
|
)
|
|
|
|
|
|
|
|
return response.json().get("token", None)
|
|
|
|
|
2019-12-17 14:15:58 +00:00
|
|
|
def request(self, verb, resource=None, url=None, render_as="json", querystring=None, raise_exception=True):
|
2019-10-30 12:05:46 +00:00
|
|
|
assert verb in ["HEAD", "GET", "POST", "PATCH", "PUT", "DELETE"]
|
2019-12-13 14:08:01 +00:00
|
|
|
if resource is not None:
|
|
|
|
url = self.build_absolute_uri(resource)
|
|
|
|
elif url is None:
|
|
|
|
raise AttributeError("Provide `resource` or `url` params")
|
|
|
|
|
2019-12-12 13:18:29 +00:00
|
|
|
if querystring is not None:
|
|
|
|
url = "{}?{}".format(url, querystring)
|
2019-10-30 12:05:46 +00:00
|
|
|
|
|
|
|
verb = getattr(self.session, verb.lower())
|
|
|
|
response = verb(url, headers={"Authorization": "Token {}".format(
|
|
|
|
self.auth_token)}, allow_redirects=False)
|
|
|
|
|
2019-10-30 13:00:12 +00:00
|
|
|
if raise_exception:
|
|
|
|
response.raise_for_status()
|
2019-10-30 12:05:46 +00:00
|
|
|
|
|
|
|
status = response.status_code
|
2019-12-17 14:15:58 +00:00
|
|
|
if render_as == "json":
|
|
|
|
output = response.json()
|
|
|
|
else:
|
|
|
|
output = response.content
|
2019-10-30 12:05:46 +00:00
|
|
|
|
|
|
|
return status, output
|
|
|
|
|
2019-12-12 13:18:29 +00:00
|
|
|
def retrieve_service_list(self, service_name, querystring=None):
|
2019-10-31 13:08:49 +00:00
|
|
|
pattern_name = '{}-list'.format(service_name)
|
|
|
|
if pattern_name not in API_PATHS:
|
|
|
|
raise ValueError("Unknown service {}".format(service_name))
|
2019-12-12 13:18:29 +00:00
|
|
|
_, output = self.request("GET", pattern_name, querystring=querystring)
|
2019-10-30 12:05:46 +00:00
|
|
|
return output
|
2019-10-30 13:00:12 +00:00
|
|
|
|
2019-12-10 12:04:04 +00:00
|
|
|
def retrieve_profile(self):
|
|
|
|
status, output = self.request("GET", 'my-account')
|
|
|
|
if status >= 400:
|
|
|
|
raise PermissionError("Cannot retrieve profile of an anonymous user.")
|
|
|
|
return UserAccount.new_from_json(output[0])
|
|
|
|
|
2019-12-17 14:15:58 +00:00
|
|
|
def retrieve_bill_document(self, pk):
|
|
|
|
path = API_PATHS.get('bill-document').format_map({'pk': pk})
|
|
|
|
|
|
|
|
url = urllib.parse.urljoin(self.base_url, path)
|
|
|
|
status, bill_pdf = self.request("GET", render_as="html", url=url, raise_exception=False)
|
|
|
|
if status == 404:
|
|
|
|
raise Http404(_("No domain found matching the query"))
|
|
|
|
return bill_pdf
|
|
|
|
|
2019-12-13 14:08:01 +00:00
|
|
|
def retrieve_domain(self, pk):
|
|
|
|
path = API_PATHS.get('domain-detail').format_map({'pk': pk})
|
|
|
|
|
|
|
|
url = urllib.parse.urljoin(self.base_url, path)
|
2019-12-17 09:25:10 +00:00
|
|
|
status, domain_json = self.request("GET", url=url, raise_exception=False)
|
|
|
|
if status == 404:
|
|
|
|
raise Http404(_("No domain found matching the query"))
|
2019-12-13 14:08:01 +00:00
|
|
|
return Domain.new_from_json(domain_json)
|
|
|
|
|
2019-12-12 13:18:29 +00:00
|
|
|
def retrieve_domain_list(self):
|
|
|
|
output = self.retrieve_service_list(Domain.api_name)
|
|
|
|
domains = []
|
|
|
|
for domain_json in output:
|
|
|
|
# filter querystring
|
|
|
|
querystring = "domain={}".format(domain_json['id'])
|
|
|
|
|
|
|
|
# retrieve services associated to a domain
|
|
|
|
domain_json['mails'] = self.retrieve_service_list(
|
|
|
|
MailService.api_name, querystring)
|
|
|
|
# TODO(@slamora): databases and sass are not related to a domain, so cannot be filtered
|
|
|
|
# domain_json['databases'] = self.retrieve_service_list(DatabaseService.api_name, querystring)
|
|
|
|
# domain_json['saas'] = self.retrieve_service_list(SaasService.api_name, querystring)
|
|
|
|
|
|
|
|
# TODO(@slamora): update when backend provides resource disk usage data
|
|
|
|
domain_json['usage'] = {
|
|
|
|
'usage': 300,
|
|
|
|
'total': 650,
|
|
|
|
'unit': 'MB',
|
|
|
|
'percent': 50,
|
|
|
|
}
|
|
|
|
|
|
|
|
# append to list a Domain object
|
|
|
|
domains.append(Domain.new_from_json(domain_json))
|
|
|
|
|
|
|
|
return domains
|
2019-10-30 13:00:12 +00:00
|
|
|
|
|
|
|
def verify_credentials(self):
|
|
|
|
"""
|
|
|
|
Returns:
|
|
|
|
A user profile info if the
|
|
|
|
credentials are valid, None otherwise.
|
|
|
|
"""
|
|
|
|
status, output = self.request("GET", 'my-account', raise_exception=False)
|
|
|
|
|
|
|
|
if status < 400:
|
|
|
|
return output
|
|
|
|
|
|
|
|
return None
|