(Draft) Add view to update existing addresses

This commit is contained in:
Santiago L 2021-06-24 13:08:16 +02:00
parent f635721831
commit 7ff01d60ef
6 changed files with 75 additions and 8 deletions

View file

@ -23,6 +23,7 @@ API_PATHS = {
'domain-list': 'domains/', 'domain-list': 'domains/',
'domain-detail': 'domains/{pk}/', 'domain-detail': 'domains/{pk}/',
'address-list': 'addresses/', 'address-list': 'addresses/',
'address-detail': 'addresses/{pk}/',
'mailbox-list': 'mailboxes/', 'mailbox-list': 'mailboxes/',
'mailinglist-list': 'lists/', 'mailinglist-list': 'lists/',
'saas-list': 'saas/', 'saas-list': 'saas/',
@ -114,13 +115,22 @@ class Orchestra(object):
def create_mail_address(self, data): def create_mail_address(self, data):
resource = '{}-list'.format(MailService.api_name) resource = '{}-list'.format(MailService.api_name)
# transform form data to expected format
data["domain"] = {"url": data["domain"]}
data["mailboxes"] = [{"url": mbox} for mbox in data["mailboxes"]]
return self.request("POST", resource=resource, data=data) return self.request("POST", resource=resource, data=data)
def retrieve_mail_address(self, pk):
path = API_PATHS.get('address-detail').format_map({'pk': pk})
url = urllib.parse.urljoin(self.base_url, path)
status, data = self.request("GET", url=url, raise_exception=False)
if status == 404:
raise Http404(_("No object found matching the query"))
return MailService.new_from_json(data)
def update_mail_address(self, pk, data):
path = API_PATHS.get('address-detail').format_map({'pk': pk})
url = urllib.parse.urljoin(self.base_url, path)
return self.request("PUT", url=url, data=data)
def retrieve_mail_address_list(self, querystring=None): def retrieve_mail_address_list(self, querystring=None):
def get_mailbox_id(value): def get_mailbox_id(value):
mailboxes = value.get('mailboxes') mailboxes = value.get('mailboxes')

View file

@ -31,8 +31,13 @@ class MailForm(forms.Form):
forward = forms.EmailField(required=False) forward = forms.EmailField(required=False)
def __init__(self, *args, **kwargs): def __init__(self, *args, **kwargs):
instance = kwargs.pop('instance', None)
if instance is not None:
kwargs['initial'] = instance.deserialize()
domains = kwargs.pop('domains') domains = kwargs.pop('domains')
mailboxes = kwargs.pop('mailboxes') mailboxes = kwargs.pop('mailboxes')
super().__init__(*args, **kwargs) super().__init__(*args, **kwargs)
self.fields['domain'].choices = [(d.url, d.name) for d in domains] self.fields['domain'].choices = [(d.url, d.name) for d in domains]
self.fields['mailboxes'].choices = [(m['url'], m['name']) for m in mailboxes] self.fields['mailboxes'].choices = [(m['url'], m['name']) for m in mailboxes]
@ -42,3 +47,13 @@ class MailForm(forms.Form):
if not cleaned_data.get('mailboxes') and not cleaned_data.get('forward'): if not cleaned_data.get('mailboxes') and not cleaned_data.get('forward'):
raise ValidationError("A mailbox or forward address should be provided.") raise ValidationError("A mailbox or forward address should be provided.")
return cleaned_data return cleaned_data
def serialize(self):
assert hasattr(self, 'cleaned_data')
serialized_data = {
"name": self.cleaned_data["name"],
"domain": {"url": self.cleaned_data["domain"]},
"mailboxes": [{"url": mbox} for mbox in self.cleaned_data["mailboxes"]],
"forward": self.cleaned_data["forward"],
}
return serialized_data

View file

@ -225,12 +225,13 @@ class DomainRecord(OrchestraModel):
return '<%s: %s>' % (self.type, self.value) return '<%s: %s>' % (self.type, self.value)
# TODO(@slamora) rename to Address
class MailService(OrchestraModel): class MailService(OrchestraModel):
api_name = 'address' api_name = 'address'
verbose_name = _('Mail addresses') verbose_name = _('Mail addresses')
description = _('Description details for mail addresses page.') description = _('Description details for mail addresses page.')
fields = ('mail_address', 'aliases', 'type', 'type_detail') fields = ('mail_address', 'aliases', 'type', 'type_detail')
param_defaults = {} param_defaults = {"id": None,}
FORWARD = 'forward' FORWARD = 'forward'
MAILBOX = 'mailbox' MAILBOX = 'mailbox'
@ -239,6 +240,15 @@ class MailService(OrchestraModel):
self.data = kwargs self.data = kwargs
super().__init__(**kwargs) super().__init__(**kwargs)
def deserialize(self):
data = {
'name': self.data['name'],
'domain': self.data['domain']['url'],
'mailboxes': [mbox['url'] for mbox in self.data['mailboxes']],
'forward': self.data['forward'],
}
return data
@property @property
def aliases(self): def aliases(self):
return [ return [

View file

@ -26,7 +26,7 @@
<tbody> <tbody>
{% for obj in object_list %} {% for obj in object_list %}
<tr> <tr>
<td>{{ obj.mail_address }}</td> <td><a href="{% url 'musician:mail-update' obj.id %}">{{ obj.mail_address }}</a></td>
<td>{{ obj.aliases|join:" , " }}</td> <td>{{ obj.aliases|join:" , " }}</td>
<td>{{ obj.type|capfirst }}</td> <td>{{ obj.type|capfirst }}</td>
<td> <td>

View file

@ -21,6 +21,7 @@ urlpatterns = [
path('profile/', views.ProfileView.as_view(), name='profile'), path('profile/', views.ProfileView.as_view(), name='profile'),
path('mails/', views.MailView.as_view(), name='mails'), path('mails/', views.MailView.as_view(), name='mails'),
path('mails/new/', views.MailCreateView.as_view(), name='mail-create'), path('mails/new/', views.MailCreateView.as_view(), name='mail-create'),
path('mails/<int:pk>/', views.MailUpdateView.as_view(), name='mail-update'),
path('mailing-lists/', views.MailingListsView.as_view(), name='mailing-lists'), path('mailing-lists/', views.MailingListsView.as_view(), name='mailing-lists'),
path('databases/', views.DatabasesView.as_view(), name='databases'), path('databases/', views.DatabasesView.as_view(), name='databases'),
path('software-as-a-service/', views.SaasView.as_view(), name='saas'), path('software-as-a-service/', views.SaasView.as_view(), name='saas'),

View file

@ -217,7 +217,38 @@ class MailCreateView(CustomContextMixin, UserTokenRequiredMixin, FormView):
def form_valid(self, form): def form_valid(self, form):
# handle request errors e.g. 400 validation # handle request errors e.g. 400 validation
try: try:
self.orchestra.create_mail_address(form.cleaned_data) serialized_data = form.serialize()
self.orchestra.create_mail_address(serialized_data)
except HTTPError as e:
form.add_error(field='__all__', error=e)
return self.form_invalid(form)
return super().form_valid(form)
class MailUpdateView(CustomContextMixin, UserTokenRequiredMixin, FormView):
service_class = MailService
template_name = "musician/mail_form.html"
form_class = MailForm
success_url = reverse_lazy("musician:mails")
def get_form_kwargs(self):
kwargs = super().get_form_kwargs()
instance = self.orchestra.retrieve_mail_address(self.kwargs['pk'])
kwargs.update({
'instance': instance,
'domains': self.orchestra.retrieve_domain_list(),
'mailboxes': self.orchestra.retrieve_mailbox_list(),
})
return kwargs
def form_valid(self, form):
# handle request errors e.g. 400 validation
try:
serialized_data = form.serialize()
self.orchestra.update_mail_address(self.kwargs['pk'], serialized_data)
except HTTPError as e: except HTTPError as e:
form.add_error(field='__all__', error=e) form.add_error(field='__all__', error=e)
return self.form_invalid(form) return self.form_invalid(form)