ddaf10204a
Replace django.core.urlresolves with django.urls (In Django 2.0) The django.core.urlresolvers module is removed in favor of its new location, django.urls. It was deprecated in Django 1.10: https://docs.djangoproject.com/en/2.0/releases/1.10/#id3 The arguments of include() has been changed in Django 1.9 and the older style was dropped in Django 2.0. https://docs.djangoproject.com/en/2.0/releases/1.9/#passing-a-3-tuple-or-an-app-name-to-include Add py35dj20 job to test Django 2.0 integration. Also drops older Django unit tests from tox.ini as horizon dropped Django <=1.10 support in Rocky. Depends-On: https://review.openstack.org/#/c/571061/ Change-Id: I122f8ad81807386517149f37aa8d63c76daac533
286 lines
12 KiB
Python
286 lines
12 KiB
Python
# Copyright 2014 Tesora Inc.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
from django.forms import ValidationError # noqa
|
|
from django.urls import reverse
|
|
from django.utils.translation import ugettext_lazy as _
|
|
|
|
import six
|
|
|
|
from horizon import exceptions
|
|
from horizon import forms
|
|
from horizon import messages
|
|
from horizon.utils import validators
|
|
|
|
from trove_dashboard import api
|
|
|
|
|
|
class CreateDatabaseForm(forms.SelfHandlingForm):
|
|
instance_id = forms.CharField(widget=forms.HiddenInput())
|
|
name = forms.CharField(label=_("Name"))
|
|
character_set = forms.CharField(
|
|
label=_("Character Set"), required=False,
|
|
help_text=_("Optional character set for the database."))
|
|
collation = forms.CharField(
|
|
label=_("Collation"), required=False,
|
|
help_text=_("Optional collation type for the database."))
|
|
|
|
def handle(self, request, data):
|
|
instance = data.get('instance_id')
|
|
try:
|
|
api.trove.database_create(request, instance, data['name'],
|
|
character_set=data['character_set'],
|
|
collation=data['collation'])
|
|
|
|
messages.success(request,
|
|
_('Created database "%s".') % data['name'])
|
|
except Exception as e:
|
|
redirect = reverse("horizon:project:databases:detail",
|
|
args=(instance,))
|
|
exceptions.handle(request, _('Unable to create database. %s') %
|
|
six.text_type(e), redirect=redirect)
|
|
return True
|
|
|
|
|
|
class ResizeVolumeForm(forms.SelfHandlingForm):
|
|
instance_id = forms.CharField(widget=forms.HiddenInput())
|
|
orig_size = forms.IntegerField(
|
|
label=_("Current Size (GB)"),
|
|
widget=forms.TextInput(attrs={'readonly': 'readonly'}),
|
|
required=False,
|
|
)
|
|
new_size = forms.IntegerField(label=_("New Size (GB)"))
|
|
|
|
def __init__(self, request, *args, **kwargs):
|
|
super(ResizeVolumeForm, self).__init__(request, *args, **kwargs)
|
|
|
|
self.fields['instance_id'].initial = (kwargs.get('initial', {}).
|
|
get('instance_id'))
|
|
self.fields['orig_size'].initial = (kwargs.get('initial', {}).
|
|
get('orig_size'))
|
|
|
|
def clean(self):
|
|
cleaned_data = super(ResizeVolumeForm, self).clean()
|
|
new_size = cleaned_data.get('new_size')
|
|
if new_size <= self.initial['orig_size']:
|
|
raise ValidationError(
|
|
_("New size for volume must be greater than current size."))
|
|
|
|
return cleaned_data
|
|
|
|
def handle(self, request, data):
|
|
instance = data.get('instance_id')
|
|
try:
|
|
api.trove.instance_resize_volume(request,
|
|
instance,
|
|
data['new_size'])
|
|
|
|
messages.success(request, _('Resizing volume "%s"') % instance)
|
|
except Exception as e:
|
|
redirect = reverse("horizon:project:databases:index")
|
|
exceptions.handle(request, _('Unable to resize volume. %s') %
|
|
six.text_type(e), redirect=redirect)
|
|
return True
|
|
|
|
|
|
class ResizeInstanceForm(forms.SelfHandlingForm):
|
|
instance_id = forms.CharField(widget=forms.HiddenInput())
|
|
old_flavor_name = forms.CharField(label=_("Old Flavor"),
|
|
required=False,
|
|
widget=forms.TextInput(
|
|
attrs={'readonly': 'readonly'}))
|
|
new_flavor = forms.ChoiceField(label=_("New Flavor"),
|
|
help_text=_("Choose a new instance "
|
|
"flavor."))
|
|
|
|
def __init__(self, request, *args, **kwargs):
|
|
super(ResizeInstanceForm, self).__init__(request, *args, **kwargs)
|
|
|
|
old_flavor_id = kwargs.get('initial', {}).get('old_flavor_id')
|
|
choices = kwargs.get('initial', {}).get('flavors')
|
|
# Remove current flavor from the list of flavor choices
|
|
choices = [(flavor_id, name) for (flavor_id, name) in choices
|
|
if flavor_id != old_flavor_id]
|
|
if choices:
|
|
choices.insert(0, ("", _("Select a new flavor")))
|
|
else:
|
|
choices.insert(0, ("", _("No flavors available")))
|
|
self.fields['new_flavor'].choices = choices
|
|
|
|
def handle(self, request, data):
|
|
instance = data.get('instance_id')
|
|
flavor = data.get('new_flavor')
|
|
try:
|
|
api.trove.instance_resize(request, instance, flavor)
|
|
|
|
messages.success(request, _('Resizing instance "%s"') % instance)
|
|
except Exception as e:
|
|
redirect = reverse("horizon:project:databases:index")
|
|
exceptions.handle(request, _('Unable to resize instance. %s') %
|
|
six.text_type(e), redirect=redirect)
|
|
return True
|
|
|
|
|
|
class PromoteToReplicaSourceForm(forms.SelfHandlingForm):
|
|
instance_id = forms.CharField(widget=forms.HiddenInput())
|
|
|
|
def handle(self, request, data):
|
|
instance_id = data.get('instance_id')
|
|
name = self.initial['replica'].name
|
|
try:
|
|
api.trove.promote_to_replica_source(request, instance_id)
|
|
messages.success(
|
|
request,
|
|
_('Promoted replica "%s" as the new replica source.') % name)
|
|
except Exception as e:
|
|
redirect = reverse("horizon:project:databases:index")
|
|
exceptions.handle(
|
|
request,
|
|
_('Unable to promote replica as the new replica source. "%s"')
|
|
% six.text_type(e), redirect=redirect)
|
|
return True
|
|
|
|
|
|
class CreateUserForm(forms.SelfHandlingForm):
|
|
instance_id = forms.CharField(widget=forms.HiddenInput())
|
|
name = forms.CharField(label=_("Name"))
|
|
password = forms.RegexField(
|
|
label=_("Password"),
|
|
widget=forms.PasswordInput(render_value=False),
|
|
regex=validators.password_validator(),
|
|
error_messages={'invalid': validators.password_validator_msg()})
|
|
host = forms.CharField(
|
|
label=_("Host"), required=False, help_text=_("Optional host of user."))
|
|
databases = forms.CharField(
|
|
label=_('Initial Databases'), required=False,
|
|
help_text=_('Optional comma separated list of databases user has '
|
|
'access to.'))
|
|
|
|
def handle(self, request, data):
|
|
instance = data.get('instance_id')
|
|
try:
|
|
api.trove.user_create(
|
|
request,
|
|
instance,
|
|
data['name'],
|
|
data['password'],
|
|
host=data['host'],
|
|
databases=self._get_databases(data))
|
|
|
|
messages.success(request,
|
|
_('Created user "%s".') % data['name'])
|
|
except Exception as e:
|
|
redirect = reverse("horizon:project:databases:detail",
|
|
args=(instance,))
|
|
exceptions.handle(request, _('Unable to create user. %s') %
|
|
six.text_type(e), redirect=redirect)
|
|
return True
|
|
|
|
def _get_databases(self, data):
|
|
databases = []
|
|
db_value = data['databases']
|
|
if db_value and db_value != u'':
|
|
dbs = data['databases']
|
|
databases = [{'name': d.strip()} for d in dbs.split(',')]
|
|
return databases
|
|
|
|
|
|
class EditUserForm(forms.SelfHandlingForm):
|
|
instance_id = forms.CharField(widget=forms.HiddenInput())
|
|
user_name = forms.CharField(
|
|
label=_("Name"),
|
|
widget=forms.TextInput(attrs={'readonly': 'readonly'}))
|
|
user_host = forms.CharField(
|
|
label=_("Host"), required=False,
|
|
widget=forms.TextInput(attrs={'readonly': 'readonly'}))
|
|
new_name = forms.CharField(label=_("New Name"), required=False)
|
|
new_password = forms.RegexField(
|
|
label=_("New Password"), required=False,
|
|
widget=forms.PasswordInput(render_value=False),
|
|
regex=validators.password_validator(),
|
|
error_messages={'invalid': validators.password_validator_msg()})
|
|
new_host = forms.CharField(label=_("New Host"), required=False)
|
|
|
|
validation_error_message = _('A new name or new password or '
|
|
'new host must be specified.')
|
|
|
|
def handle(self, request, data):
|
|
instance = data.get('instance_id')
|
|
try:
|
|
api.trove.user_update_attributes(
|
|
request,
|
|
instance,
|
|
data['user_name'],
|
|
host=data['user_host'],
|
|
new_name=data['new_name'],
|
|
new_password=data['new_password'],
|
|
new_host=data['new_host'])
|
|
|
|
messages.success(request,
|
|
_('Updated user "%s".') % data['user_name'])
|
|
except Exception as e:
|
|
redirect = reverse("horizon:project:databases:detail",
|
|
args=(instance,))
|
|
exceptions.handle(request, _('Unable to update user. %s') %
|
|
six.text_type(e), redirect=redirect)
|
|
return True
|
|
|
|
def clean(self):
|
|
cleaned_data = super(EditUserForm, self).clean()
|
|
|
|
if (not (cleaned_data['new_name'] or
|
|
cleaned_data['new_password'] or
|
|
cleaned_data['new_host'])):
|
|
raise ValidationError(self.validation_error_message)
|
|
|
|
return cleaned_data
|
|
|
|
|
|
class AttachConfigurationForm(forms.SelfHandlingForm):
|
|
instance_id = forms.CharField(widget=forms.HiddenInput())
|
|
configuration = forms.ChoiceField(label=_("Configuration Group"))
|
|
|
|
def __init__(self, request, *args, **kwargs):
|
|
super(AttachConfigurationForm, self).__init__(request, *args, **kwargs)
|
|
instance_id = kwargs.get('initial', {}).get('instance_id')
|
|
datastore = kwargs.get('initial', {}).get('datastore')
|
|
datastore_version = kwargs.get('initial', {}).get('datastore_version')
|
|
self.fields['instance_id'].initial = instance_id
|
|
|
|
configurations = api.trove.configuration_list(request)
|
|
choices = [(c.id, c.name) for c in configurations
|
|
if (c.datastore_name == datastore and
|
|
c.datastore_version_name == datastore_version)]
|
|
if choices:
|
|
choices.insert(0, ("", _("Select configuration group")))
|
|
else:
|
|
choices.insert(0, ("", _("No configuration groups available")))
|
|
self.fields['configuration'].choices = choices
|
|
|
|
def handle(self, request, data):
|
|
instance_id = data.get('instance_id')
|
|
try:
|
|
api.trove.instance_attach_configuration(request,
|
|
instance_id,
|
|
data['configuration'])
|
|
|
|
messages.success(request, _('Attaching Configuration group "%s"')
|
|
% instance_id)
|
|
except Exception as e:
|
|
redirect = reverse("horizon:project:databases:index")
|
|
exceptions.handle(request, _('Unable to attach configuration '
|
|
'group. %s')
|
|
% six.text_type(e), redirect=redirect)
|
|
return True
|