Dmitry Tantsur d2d80fa9ef Manage image ID's for node profile
Kernel and ramdisk image ID's are stored for each node profile
in nova keys and should be managed via UI.

Only image ID is displayed in node profile list. Though
somewhat unfriendly, this is better for now than making
2 calls to Glance for every row. Separate node details page
will be introduced in the next patch to fix this issue.

Also this patch introduce abstraction over flavor to ease
writing code specific to node profiles.

Change-Id: Ie76651536e2d8b90e14c9000da0226743db91e88
2014-02-27 06:29:27 -05:00

107 lines
4.1 KiB
Python

# -*- coding: utf8 -*-
#
# 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 fields
from django.utils.translation import ugettext_lazy as _
from horizon import exceptions
from horizon import workflows
from openstack_dashboard.api import glance
from openstack_dashboard.dashboards.admin.flavors \
import workflows as flavor_workflows
from tuskar_ui import api
class CreateNodeProfileAction(flavor_workflows.CreateFlavorInfoAction):
arch = fields.ChoiceField(choices=(('i386', 'i386'), ('amd64', 'amd64')),
label=_("Architecture"))
kernel_image_id = fields.ChoiceField(choices=(),
label=_("Deploy Kernel Image"))
ramdisk_image_id = fields.ChoiceField(choices=(),
label=_("Deploy Ramdisk Image"))
def __init__(self, *args, **kwrds):
super(CreateNodeProfileAction, self).__init__(*args, **kwrds)
try:
kernel_images = glance.image_list_detailed(
self.request,
filters={'disk_format': 'aki'}
)[0]
ramdisk_images = glance.image_list_detailed(
self.request,
filters={'disk_format': 'ari'}
)[0]
except Exception:
exceptions.handle(self.request,
_('Unable to retrieve images list.'))
kernel_images = []
ramdisk_images = []
self.fields['kernel_image_id'].choices = [(img.id, img.name)
for img in kernel_images]
self.fields['ramdisk_image_id'].choices = [(img.id, img.name)
for img in ramdisk_images]
# Delete what is not applicable to hardware
del self.fields['eph_gb']
del self.fields['swap_mb']
# Alter user-visible strings
self.fields['vcpus'].label = _("CPUs")
self.fields['disk_gb'].label = _("Disk GB")
# No idea why Horizon exposes this database detail
del self.fields['flavor_id']
class Meta:
name = _("Node Profile")
# FIXME(dtantsur): maybe better help text?
help_text = _("From here you can create a new "
"node profile to organize instance resources.")
class CreateNodeProfileStep(workflows.Step):
action_class = CreateNodeProfileAction
contributes = ("name",
"vcpus",
"memory_mb",
"disk_gb",
"arch",
"kernel_image_id",
"ramdisk_image_id")
class CreateNodeProfile(flavor_workflows.CreateFlavor):
slug = "create_node_profile"
name = _("Create Node Profile")
finalize_button_name = _("Create Node Profile")
success_message = _('Created new node profile "%s".')
failure_message = _('Unable to create node profile "%s".')
success_url = "horizon:infrastructure:node_profiles:index"
default_steps = (CreateNodeProfileStep,)
def handle(self, request, data):
try:
self.object = api.NodeProfile.create(
request,
name=data['name'],
memory=data['memory_mb'],
vcpus=data['vcpus'],
disk=data['disk_gb'],
cpu_arch=data['arch'],
kernel_image_id=data['kernel_image_id'],
ramdisk_image_id=data['ramdisk_image_id']
)
except Exception:
exceptions.handle(request, _("Unable to create node profile"))
return False
return True