Add iallocator to create vm
[ganeti_webmgr.git] / ganeti_web / forms / virtual_machine.py
blob2a21a7d989577ee15f965c3f9316ceb9f23ad1c9
1 # Copyright (C) 2010 Oregon State University et al.
3 # This program is free software; you can redistribute it and/or
4 # modify it under the terms of the GNU General Public License
5 # as published by the Free Software Foundation; either version 2
6 # of the License, or (at your option) any later version.
8 # This program is distributed in the hope that it will be useful,
9 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # GNU General Public License for more details.
13 # You should have received a copy of the GNU General Public License
14 # along with this program; if not, write to the Free Software
15 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
16 # USA.
18 from django import forms
19 from django.contrib.formtools.wizard.views import CookieWizardView
20 from django.conf import settings
21 from django.core.urlresolvers import reverse
22 from django.core.validators import MaxValueValidator, MinValueValidator
23 from django.db.models import Q
24 from django.forms import (Form, BooleanField, CharField, ChoiceField,
25 IntegerField, ModelChoiceField, ValidationError,
26 MultipleChoiceField, CheckboxSelectMultiple)
27 from django.http import HttpResponseRedirect
28 # Per #6579, do not change this import without discussion.
29 from django.utils import simplejson as json
30 from django.utils.translation import ugettext_lazy as _
32 from object_log.models import LogItem
33 log_action = LogItem.objects.log_action
35 from ganeti_web.backend.queries import (cluster_qs_for_user,
36 owner_qs_for_cluster)
37 from ganeti_web.backend.templates import template_to_instance
38 from ganeti_web.caps import has_cdrom2, has_balloonmem, has_sharedfile
39 from ganeti_web.constants import (EMPTY_CHOICE_FIELD, HV_DISK_TEMPLATES,
40 HV_NIC_MODES, KVM_CHOICES, HV_USB_MICE,
41 HV_SECURITY_MODELS, KVM_FLAGS,
42 HV_DISK_CACHES, MODE_CHOICES, HVM_CHOICES,
43 VM_HELP, VM_CREATE_HELP, VM_RENAME_HELP,
44 KVM_BOOT_ORDER, HVM_BOOT_ORDER)
45 from ganeti_web.fields import DataVolumeField, MACAddressField
46 from ganeti_web.models import (Cluster, ClusterUser, Node,
47 VirtualMachineTemplate, VirtualMachine)
48 from ganeti_web.utilities import (cluster_default_info, cluster_os_list,
49 get_hypervisor, hv_prettify)
50 from ganeti_web.util.client import (REPLACE_DISK_AUTO, REPLACE_DISK_PRI,
51 REPLACE_DISK_CHG, REPLACE_DISK_SECONDARY)
52 from ganeti_web.views.generic import LoginRequiredMixin
54 username_or_mtime = Q(username='') | Q(mtime__isnull=True)
57 class VirtualMachineForm(forms.ModelForm):
58 """
59 Parent class that holds all vm clean methods
60 and shared form fields.
61 """
62 memory = DataVolumeField(label=_('Memory'), min_value=100)
63 minmem = DataVolumeField(label=_('Minimum RAM (MiB)'),
64 required=True, min_value=100)
65 maxmem = DataVolumeField(label=_('Maximum RAM (MiB)'),
66 required=True, min_value=100)
68 class Meta:
69 model = VirtualMachineTemplate
71 def create_disk_fields(self, count):
72 """
73 dynamically add fields for disks
74 """
75 self.disk_fields = range(count)
76 for i in range(count):
77 disk_size = DataVolumeField(min_value=100, required=True,
78 label=_("Disk/%s Size" % i))
79 self.fields['disk_size_%s' % i] = disk_size
81 def create_nic_fields(self, count, defaults=None):
82 """
83 dynamically add fields for nics
84 """
85 self.nic_fields = range(count)
86 for i in range(count):
87 nic_mode = forms.ChoiceField(label=_('NIC/%s Mode' % i),
88 choices=HV_NIC_MODES)
89 nic_link = forms.CharField(label=_('NIC/%s Link' % i),
90 max_length=255)
91 if defaults is not None:
92 nic_link.initial = defaults['nic_link']
93 self.fields['nic_mode_%s' % i] = nic_mode
94 self.fields['nic_link_%s' % i] = nic_link
96 def clean_hostname(self):
97 data = self.cleaned_data
98 hostname = data.get('hostname')
99 cluster = data.get('cluster')
100 if hostname and cluster:
101 # Verify that this hostname is not in use for this cluster. It can
102 # only be reused when recovering a VM that failed to deploy.
104 # Recoveries are only allowed when the user is the owner of the VM
105 try:
106 vm = VirtualMachine.objects.get(cluster=cluster,
107 hostname=hostname)
109 # detect vm that failed to deploy
110 if not vm.pending_delete and vm.template is not None:
111 current_owner = vm.owner.cast()
112 if current_owner == self.owner:
113 data['vm_recovery'] = vm
114 else:
115 msg = _("Owner cannot be changed when recovering a "
116 "failed deployment")
117 self._errors["owner"] = self.error_class([msg])
118 else:
119 raise ValidationError(_("Hostname is already in use for "
120 "this cluster"))
122 except VirtualMachine.DoesNotExist:
123 # doesn't exist, no further checks needed
124 pass
126 # Spaces in hostname will always break things.
127 if ' ' in hostname:
128 self._errors["hostname"] = self.error_class(
129 ["Hostname contains illegal character"])
130 return hostname
132 def clean_vcpus(self):
133 vcpus = self.cleaned_data.get("vcpus", None)
135 if vcpus is not None and vcpus < 1:
136 self._errors["vcpus"] = self.error_class(
137 ["At least one CPU must be present"])
138 else:
139 return vcpus
141 def clean_initrd_path(self):
142 data = self.cleaned_data['initrd_path']
143 if data and not data.startswith('/') and data != 'no_initrd_path':
144 msg = u"%s." % _('This field must start with a "/"')
145 self._errors['initrd_path'] = self.error_class([msg])
146 return data
148 def clean_security_domain(self):
149 data = self.cleaned_data['security_domain']
150 security_model = self.cleaned_data['security_model']
151 msg = None
153 if data and security_model != 'user':
154 msg = u'%s.' % _(
155 'This field can not be set if Security '
156 'Mode is not set to User')
157 elif security_model == 'user':
158 if not data:
159 msg = u'%s.' % _('This field is required')
160 elif not data[0].isalpha():
161 msg = u'%s.' % _('This field must being with '
162 'an alpha character')
164 if msg:
165 self._errors['security_domain'] = self.error_class([msg])
166 return data
168 def clean_vnc_x509_path(self):
169 data = self.cleaned_data['vnc_x509_path']
170 if data and not data.startswith('/'):
171 msg = u'%s,' % _('This field must start with a "/"')
172 self._errors['vnc_x509_path'] = self.error_class([msg])
173 return data
176 def check_quota_modify(form):
177 """ method for validating user is within their quota when modifying """
178 data = form.cleaned_data
179 cluster = form.cluster
180 owner = form.owner
181 vm = form.vm
183 # check quota
184 if owner is not None:
185 start = data['start']
186 quota = cluster.get_quota(owner)
187 if quota.values():
188 used = owner.used_resources(cluster, only_running=True)
190 if (start and quota['ram'] is not None and
191 (used['ram'] + data['memory']-vm.ram) > quota['ram']):
192 del data['memory']
193 q_msg = u"%s" % _("Owner does not have enough ram "
194 "remaining on this cluster. You must "
195 "reduce the amount of ram.")
196 form._errors["ram"] = form.error_class([q_msg])
198 if 'disk_size' in data and data['disk_size']:
199 if quota['disk'] and used['disk'] + data['disk_size'] > \
200 quota['disk']:
201 del data['disk_size']
202 q_msg = u"%s" % _("Owner does not have enough diskspace "
203 "remaining on this cluster.")
204 form._errors["disk_size"] = form.error_class([q_msg])
206 if (start and quota['virtual_cpus'] is not None and
207 (used['virtual_cpus'] + data['vcpus']
208 - vm.virtual_cpus) >
209 quota['virtual_cpus']):
210 del data['vcpus']
211 q_msg = u"%s" % _("Owner does not have enough virtual "
212 "cpus remaining on this cluster. You "
213 "must reduce the amount of virtual "
214 "cpus.")
215 form._errors["vcpus"] = form.error_class([q_msg])
218 class ModifyVirtualMachineForm(VirtualMachineForm):
220 Base modify class.
221 If hvparam_fields (itirable) set on child, then
222 each field on the form will be initialized to the
223 value in vm.info.hvparams
225 always_required = ('vcpus', 'memory')
226 empty_field = EMPTY_CHOICE_FIELD
228 nic_count = forms.IntegerField(initial=1, widget=forms.HiddenInput())
229 os = forms.ChoiceField(label=_('Operating System'), choices=[empty_field])
231 class Meta:
232 model = VirtualMachineTemplate
233 exclude = ('start', 'owner', 'cluster', 'hostname', 'name_check',
234 'iallocator', 'iallocator_hostname', 'disk_template',
235 'pnode', 'nics', 'snode', 'disk_size', 'nic_mode',
236 'template_name', 'hypervisor', 'disks', 'description',
237 'no_install', 'ip_check', 'temporary')
239 def __init__(self, vm, initial=None, *args, **kwargs):
240 super(VirtualMachineForm, self).__init__(initial, *args, **kwargs)
242 if has_balloonmem(vm.cluster):
243 self.always_required = ('vcpus', 'memory', 'minmem')
245 # Set owner on form
246 try:
247 self.owner
248 except AttributeError:
249 self.owner = vm.owner
251 # Setup os choices
252 os_list = cluster_os_list(vm.cluster)
253 self.fields['os'].choices = os_list
255 for field in self.always_required:
256 self.fields[field].required = True
257 # If the required property is set on a child class,
258 # require those form fields
259 try:
260 if self.required:
261 for field in self.required:
262 self.fields[field].required = True
263 except AttributeError:
264 pass
266 # Need to set initial values from vm.info as these are not saved
267 # per the vm model.
268 if vm.info:
269 info = vm.info
270 hvparam = info['hvparams']
271 # XXX Convert ram string since it comes out
272 # from ganeti as an int and the DataVolumeField does not like
273 # ints.
274 self.fields['vcpus'].initial = info['beparams']['vcpus']
275 if has_balloonmem(vm.cluster):
276 del self.fields['memory']
277 self.fields['minmem'].initial = info['beparams']['minmem']
278 self.fields['maxmem'].initial = info['beparams']['maxmem']
279 else:
280 del self.fields['minmem']
281 del self.fields['maxmem']
282 self.fields['memory'].initial = str(info['beparams']['memory'])
284 # always take the larger nic count. this ensures that if nics are
285 # being removed that they will be in the form as Nones
286 self.nics = len(info['nic.links'])
287 nic_count = int(initial.get('nic_count', 1)) if initial else 1
288 nic_count = self.nics if self.nics > nic_count else nic_count
289 self.fields['nic_count'].initial = nic_count
290 self.nic_fields = xrange(nic_count)
291 for i in xrange(nic_count):
292 link = forms.CharField(label=_('NIC/%s Link' % i),
293 max_length=255, required=True)
294 self.fields['nic_link_%s' % i] = link
295 mac = MACAddressField(label=_('NIC/%s Mac' % i), required=True)
296 self.fields['nic_mac_%s' % i] = mac
297 if i < self.nics:
298 mac.initial = info['nic.macs'][i]
299 link.initial = info['nic.links'][i]
301 self.fields['os'].initial = info['os']
303 try:
304 if self.hvparam_fields:
305 for field in self.hvparam_fields:
306 self.fields[field].initial = hvparam.get(field)
307 except AttributeError:
308 pass
310 def clean(self):
311 data = self.cleaned_data
312 kernel_path = data.get('kernel_path')
313 initrd_path = data.get('initrd_path')
315 # Make sure if initrd_path is set, kernel_path is aswell
316 if initrd_path and not kernel_path:
317 msg = u"%s." % _("Kernel Path must be specified along "
318 "with Initrd Path")
319 self._errors['kernel_path'] = self.error_class([msg])
320 self._errors['initrd_path'] = self.error_class([msg])
321 del data['initrd_path']
323 vnc_tls = data.get('vnc_tls')
324 vnc_x509_path = data.get('vnc_x509_path')
325 vnc_x509_verify = data.get('vnc_x509_verify')
327 if not vnc_tls and vnc_x509_path:
328 msg = u'%s.' % _("This field can not be set without "
329 "VNC TLS enabled")
330 self._errors['vnc_x509_path'] = self.error_class([msg])
331 if vnc_x509_verify and not vnc_x509_path:
332 msg = u'%s.' % _('This field is required')
333 self._errors['vnc_x509_path'] = self.error_class([msg])
335 if self.owner:
336 data['start'] = 'reboot' in self.data or self.vm.is_running
337 check_quota_modify(self)
338 del data['start']
340 for i in xrange(data['nic_count']):
341 mac_field = 'nic_mac_%s' % i
342 link_field = 'nic_link_%s' % i
343 mac = data[mac_field] if mac_field in data else None
344 link = data[link_field] if link_field in data else None
345 if mac and not link:
346 self._errors[link_field] = self.error_class([_('This field is'
347 ' required')])
348 elif link and not mac:
349 self._errors[mac_field] = self.error_class([_('This field is '
350 'required')])
351 data['nic_count_original'] = self.nics
353 return data
356 class HvmModifyVirtualMachineForm(ModifyVirtualMachineForm):
357 hvparam_fields = ('boot_order', 'cdrom_image_path', 'nic_type',
358 'disk_type', 'vnc_bind_address', 'acpi',
359 'use_localtime')
360 required = ('disk_type', 'boot_order', 'nic_type')
361 empty_field = EMPTY_CHOICE_FIELD
362 disk_types = HVM_CHOICES['disk_type']
363 nic_types = HVM_CHOICES['nic_type']
364 boot_devices = HVM_CHOICES['boot_order']
366 acpi = forms.BooleanField(label='ACPI', required=False)
367 use_localtime = forms.BooleanField(label='Use Localtime', required=False)
368 vnc_bind_address = forms.IPAddressField(label='VNC Bind Address',
369 required=False)
370 disk_type = forms.ChoiceField(label=_('Disk Type'), choices=disk_types)
371 nic_type = forms.ChoiceField(label=_('NIC Type'), choices=nic_types)
372 boot_order = forms.ChoiceField(label=_('Boot Device'),
373 choices=boot_devices)
375 class Meta(ModifyVirtualMachineForm.Meta):
376 exclude = ModifyVirtualMachineForm.Meta.exclude + \
377 ('kernel_path', 'root_path', 'kernel_args',
378 'serial_console', 'cdrom2_image_path')
380 def __init__(self, vm, *args, **kwargs):
381 super(HvmModifyVirtualMachineForm, self).__init__(vm, *args, **kwargs)
384 class PvmModifyVirtualMachineForm(ModifyVirtualMachineForm):
385 hvparam_fields = ('root_path', 'kernel_path', 'kernel_args',
386 'initrd_path')
388 initrd_path = forms.CharField(label='initrd Path', required=False)
389 kernel_args = forms.CharField(label='Kernel Args', required=False)
391 class Meta(ModifyVirtualMachineForm.Meta):
392 exclude = ModifyVirtualMachineForm.Meta.exclude + \
393 ('disk_type', 'nic_type', 'boot_order',
394 'cdrom_image_path', 'serial_console',
395 'cdrom2_image_path')
397 def __init__(self, vm, *args, **kwargs):
398 super(PvmModifyVirtualMachineForm, self).__init__(vm, *args, **kwargs)
401 class KvmModifyVirtualMachineForm(PvmModifyVirtualMachineForm,
402 HvmModifyVirtualMachineForm):
403 hvparam_fields = (
404 'acpi', 'disk_cache', 'initrd_path',
405 'kernel_args', 'kvm_flag', 'mem_path',
406 'migration_downtime', 'security_domain',
407 'security_model', 'usb_mouse', 'use_chroot',
408 'use_localtime', 'vnc_bind_address', 'vnc_tls',
409 'vnc_x509_path', 'vnc_x509_verify', 'disk_type',
410 'boot_order', 'nic_type', 'root_path',
411 'kernel_path', 'serial_console',
412 'cdrom_image_path',
413 'cdrom2_image_path',
415 disk_caches = HV_DISK_CACHES
416 kvm_flags = KVM_FLAGS
417 security_models = HV_SECURITY_MODELS
418 usb_mice = HV_USB_MICE
419 disk_types = KVM_CHOICES['disk_type']
420 nic_types = KVM_CHOICES['nic_type']
421 boot_devices = KVM_CHOICES['boot_order']
423 disk_cache = forms.ChoiceField(label='Disk Cache', required=False,
424 choices=disk_caches)
425 kvm_flag = forms.ChoiceField(label='KVM Flag', required=False,
426 choices=kvm_flags)
427 mem_path = forms.CharField(label='Mem Path', required=False)
428 migration_downtime = forms.IntegerField(label='Migration Downtime',
429 required=False)
430 security_model = forms.ChoiceField(label='Security Model',
431 required=False, choices=security_models)
432 security_domain = forms.CharField(label='Security Domain', required=False)
433 usb_mouse = forms.ChoiceField(label='USB Mouse', required=False,
434 choices=usb_mice)
435 use_chroot = forms.BooleanField(label='Use Chroot', required=False)
436 vnc_tls = forms.BooleanField(label='VNC TLS', required=False)
437 vnc_x509_path = forms.CharField(label='VNC x509 Path', required=False)
438 vnc_x509_verify = forms.BooleanField(label='VNC x509 Verify',
439 required=False)
441 class Meta(ModifyVirtualMachineForm.Meta):
442 pass
444 def __init__(self, vm, *args, **kwargs):
445 super(KvmModifyVirtualMachineForm, self).__init__(vm, *args, **kwargs)
446 self.fields['disk_type'].choices = self.disk_types
447 self.fields['nic_type'].choices = self.nic_types
448 self.fields['boot_order'].choices = self.boot_devices
451 class ModifyConfirmForm(forms.Form):
453 def clean(self):
454 raw = self.data['rapi_dict']
455 data = json.loads(raw)
457 cleaned = self.cleaned_data
458 cleaned['rapi_dict'] = data
460 # XXX copy properties into cleaned data so that check_quota_modify can
461 # be used
462 if data.get('maxmem'):
463 cleaned['maxmem'] = data['maxmem']
464 cleaned['minmem'] = data['minmem']
465 else:
466 cleaned['memory'] = data['memory']
467 cleaned['vcpus'] = data['vcpus']
468 cleaned['start'] = 'reboot' in data or self.vm.is_running
469 check_quota_modify(self)
471 # Build NICs dicts. Add changes for existing nics and mark new or
472 # removed nics
474 # XXX Ganeti only allows a single remove or add but this code will
475 # format properly for unlimited adds or removes in the hope that this
476 # limitation is removed sometime in the future.
477 nics = []
478 nic_count_original = data.pop('nic_count_original')
479 nic_count = data.pop('nic_count')
480 for i in xrange(nic_count):
481 nic = dict(link=data.pop('nic_link_%s' % i))
482 if 'nic_mac_%s' % i in data:
483 nic['mac'] = data.pop('nic_mac_%s' % i)
484 index = i if i < nic_count_original else 'add'
485 nics.append((index, nic))
486 for i in xrange(nic_count_original-nic_count):
487 nics.append(('remove', {}))
488 try:
489 del data['nic_mac_%s' % (nic_count+i)]
490 except KeyError:
491 pass
492 del data['nic_link_%s' % (nic_count+i)]
494 data['nics'] = nics
495 return cleaned
498 class MigrateForm(forms.Form):
499 """ Form used for migrating a Virtual Machine """
500 mode = forms.ChoiceField(choices=MODE_CHOICES)
501 cleanup = forms.BooleanField(initial=False, required=False,
502 label=_("Attempt recovery from failed "
503 "migration"))
506 class RenameForm(forms.Form):
507 """ form used for renaming a Virtual Machine """
508 hostname = forms.CharField(label=_('Instance Name'), max_length=255,
509 required=True)
510 ip_check = forms.BooleanField(initial=True, required=False,
511 label=_('IP Check'))
512 name_check = forms.BooleanField(initial=True, required=False,
513 label=_('DNS Name Check'))
515 def __init__(self, vm, *args, **kwargs):
516 self.vm = vm
517 super(RenameForm, self).__init__(*args, **kwargs)
519 def clean_hostname(self):
520 data = self.cleaned_data
521 hostname = data.get('hostname', None)
522 if hostname and hostname == self.vm.hostname:
523 raise ValidationError(_("The new hostname must be different than "
524 "the current hostname"))
525 return hostname
528 class ChangeOwnerForm(forms.Form):
529 """ Form used when modifying the owner of a virtual machine """
530 owner = forms.ModelChoiceField(queryset=ClusterUser.objects.all(),
531 label=_('Owner'))
534 class ReplaceDisksForm(forms.Form):
536 Form used when replacing disks for a virtual machine
538 empty_field = EMPTY_CHOICE_FIELD
540 MODE_CHOICES = (
541 (REPLACE_DISK_AUTO, _('Automatic')),
542 (REPLACE_DISK_PRI, _('Replace disks on primary')),
543 (REPLACE_DISK_SECONDARY, _('Replace disks secondary')),
544 (REPLACE_DISK_CHG, _('Replace secondary with new disk')),
547 mode = forms.ChoiceField(choices=MODE_CHOICES, label=_('Mode'))
548 disks = forms.MultipleChoiceField(label=_('Disks'), required=False)
549 node = forms.ChoiceField(label=_('Node'), choices=[empty_field],
550 required=False)
551 iallocator = forms.BooleanField(initial=False, label=_('Iallocator'),
552 required=False,
553 help_text=_(VM_CREATE_HELP['iallocator']))
555 def __init__(self, instance, *args, **kwargs):
556 super(ReplaceDisksForm, self).__init__(*args, **kwargs)
557 self.instance = instance
559 # set disk choices based on the instance
560 disk_choices = [(i, 'disk/%s' % i) for i, v in
561 enumerate(instance.info['disk.sizes'])]
562 self.fields['disks'].choices = disk_choices
564 # set choices based on the instances cluster
565 cluster = instance.cluster
566 nodelist = [str(h) for h in
567 cluster.nodes.values_list('hostname', flat=True)]
568 nodes = zip(nodelist, nodelist)
569 nodes.insert(0, self.empty_field)
570 self.fields['node'].choices = nodes
572 defaults = cluster_default_info(cluster, get_hypervisor(instance))
573 if defaults['iallocator'] != '':
574 self.fields['iallocator'].initial = True
575 self.fields['iallocator_hostname'] = forms.CharField(
576 initial=defaults['iallocator'],
577 required=False,
578 widget=forms.HiddenInput())
580 def clean(self):
581 data = self.cleaned_data
582 mode = data.get('mode')
583 if mode == REPLACE_DISK_CHG:
584 iallocator = data.get('iallocator')
585 node = data.get('node')
586 if not (iallocator or node):
587 msg = _('Node or iallocator is required when '
588 'replacing secondary with new disk')
589 self._errors['mode'] = self.error_class([msg])
591 elif iallocator and node:
592 msg = _('Choose either node or iallocator')
593 self._errors['mode'] = self.error_class([msg])
595 return data
597 def clean_disks(self):
598 """ format disks into a comma delimited string """
599 disks = self.cleaned_data.get('disks')
600 if disks is not None:
601 disks = ','.join(disks)
602 return disks
604 def clean_node(self):
605 node = self.cleaned_data.get('node')
606 return node if node else None
608 def save(self):
610 Start a replace disks job using the data in this form.
612 data = self.cleaned_data
613 mode = data['mode']
614 disks = data['disks']
615 node = data['node']
616 if data['iallocator']:
617 iallocator = data['iallocator_hostname']
618 else:
619 iallocator = None
620 return self.instance.replace_disks(mode, disks, node, iallocator)
623 class VMWizardClusterForm(Form):
624 cluster = ModelChoiceField(label=_('Cluster'),
625 queryset=Cluster.objects.all(),
626 empty_label=None)
628 class Media:
629 css = {
630 # I'm not quite sure if this is the proper way to use static
631 'all': ('/static/css/vm_wizard/cluster_form.css',)
634 def __init__(self, options=None, *args, **kwargs):
635 super(VMWizardClusterForm, self).__init__(*args, **kwargs)
636 if options:
637 self.fields['choices'] = MultipleChoiceField(
638 widget=CheckboxSelectMultiple,
639 choices=options,
640 initial=self.initial,
641 label=_('What would you '
642 'like to create?'),
643 help_text=_(VM_CREATE_HELP['choices']))
645 def _configure_for_user(self, user):
646 self.fields["cluster"].queryset = cluster_qs_for_user(user)
648 def clean_cluster(self):
650 Ensure that the cluster is available.
653 cluster = self.cleaned_data.get('cluster', None)
654 if not getattr(cluster, "info", None):
655 msg = _("This cluster is currently unavailable. Please check"
656 " for Errors on the cluster detail page.")
657 self._errors['cluster'] = self.error_class([msg])
659 return cluster
662 class VMWizardOwnerForm(Form):
663 owner = ModelChoiceField(label=_('Owner'),
664 queryset=ClusterUser.objects.all(),
665 empty_label=None,
666 help_text=_(VM_CREATE_HELP['owner']))
667 template_name = CharField(label=_("Template Name"), max_length=255,
668 required=False,
669 help_text=_(VM_HELP['template_name']))
670 hostname = CharField(label=_('Instance Name'), max_length=255,
671 required=False,
672 help_text=_(VM_CREATE_HELP['hostname']))
674 def _configure_for_cluster(self, cluster):
675 if not cluster:
676 return
678 self.cluster = cluster
680 qs = owner_qs_for_cluster(cluster)
681 self.fields["owner"].queryset = qs
683 def _configure_for_template(self, template, choices=None):
684 # for each option not checked on step 0
685 if choices:
686 for field in choices:
687 # Hide it
688 self.fields[field].widget = forms.HiddenInput()
690 if not template:
691 return
693 self.fields["template_name"].initial = template.template_name
695 def clean_hostname(self):
696 hostname = self.cleaned_data.get('hostname')
697 if hostname:
698 # Confirm that the hostname is not already in use.
699 try:
700 vm = VirtualMachine.objects.get(cluster=self.cluster,
701 hostname=hostname)
702 except VirtualMachine.DoesNotExist:
703 # Well, *I'm* convinced.
704 pass
705 else:
706 raise ValidationError(
707 _("Hostname is already in use for this cluster"))
709 # Spaces in hostname will always break things.
710 if ' ' in hostname:
711 self._errors["hostname"] = self.error_class(
712 ["Hostnames cannot contain spaces."])
713 return hostname
715 def clean(self):
716 if (not self.cleaned_data.get("template_name") and
717 not self.cleaned_data.get("hostname")):
718 raise ValidationError("What should be created?")
719 return self.cleaned_data
722 class VMWizardBasicsForm(Form):
723 hv = ChoiceField(label=_("Hypervisor"), choices=[],
724 help_text=_(VM_CREATE_HELP['hypervisor']))
725 os = ChoiceField(label=_('Operating System'), choices=[],
726 help_text=_(VM_CREATE_HELP['os']))
727 no_install = BooleanField(label=_('Do not install the OS'), required=False)
728 iallocator = BooleanField(label=_("Automatic Allocation"),
729 initial=True, required=False,
730 help_text=_(VM_CREATE_HELP['iallocator']))
731 vcpus = IntegerField(label=_("Virtual CPU Count"), initial=1, min_value=1,
732 help_text=_(VM_HELP['vcpus']))
733 minram = DataVolumeField(label=_('Minimum RAM (MiB)'),
734 help_text=_(VM_HELP['memory']))
735 memory = DataVolumeField(label=_('Maximum RAM (MiB)'),
736 help_text=_(VM_HELP['memory']))
737 disk_template = ChoiceField(label=_('Disk Template'),
738 choices=HV_DISK_TEMPLATES,
739 help_text=_(VM_CREATE_HELP['disk_template']))
741 def __init__(self, *args, **kwargs):
742 super(VMWizardBasicsForm, self).__init__(*args, **kwargs)
744 # Create disk and nic fields based on value in settings
745 disk_count = settings.MAX_DISKS_ADD
746 self.create_disk_fields(disk_count)
748 nic_count = settings.MAX_NICS_ADD
749 self.create_nic_fields(nic_count)
751 def create_disk_fields(self, count):
753 dynamically add fields for disks
755 for i in range(count):
756 disk_size = DataVolumeField(
757 label=_("Disk/%s Size (MB)" % i), required=False,
758 help_text=_(VM_CREATE_HELP['disk_size']))
760 disk_size.widget.attrs['class'] = 'multi disk'
761 disk_size.widget.attrs['data-group'] = i
762 self.fields['disk_size_%s' % i] = disk_size
764 def create_nic_fields(self, count):
766 dynamically add fields for nics
768 self.nic_fields = range(count)
769 for i in range(count):
770 nic_mode = forms.ChoiceField(
771 label=_('NIC/%s Mode' % i), choices=HV_NIC_MODES, initial='',
772 required=False, help_text=_(VM_CREATE_HELP['nic_mode']))
774 nic_link = forms.CharField(
775 label=_('NIC/%s Link' % i), max_length=255, initial='',
776 required=False, help_text=_(VM_HELP['nic_link']))
778 # used for front end
779 nic_mode.widget.attrs['class'] = 'multi nic mode'
780 nic_mode.widget.attrs['data-group'] = i
781 nic_link.widget.attrs['class'] = 'multi nic link'
782 nic_link.widget.attrs['data-group'] = i
784 self.fields['nic_mode_%s' % i] = nic_mode
785 self.fields['nic_link_%s' % i] = nic_link
787 def _configure_for_cluster(self, cluster):
788 if not cluster:
789 return
791 self.cluster = cluster
793 # Get a look at the list of available hypervisors, and set the initial
794 # hypervisor appropriately.
795 hvs = cluster.info["enabled_hypervisors"]
796 prettified = [hv_prettify(hv) for hv in hvs]
797 hv = cluster.info["default_hypervisor"]
798 self.fields["hv"].choices = zip(hvs, prettified)
799 self.fields["hv"].initial = hv
801 if not has_sharedfile(cluster):
802 self.fields["disk_template"].choices.remove((u'sharedfile',
803 u'Sharedfile'))
805 # Get the OS list.
806 self.fields["os"].choices = cluster_os_list(cluster)
808 # Set the default CPU count based on the backend parameters.
809 beparams = cluster.info["beparams"]["default"]
810 self.fields["vcpus"].initial = beparams["vcpus"]
812 # Check for memory based on ganeti version
813 if has_balloonmem(cluster):
814 self.fields["memory"].initial = beparams["maxmem"]
815 self.fields["minram"].initial = beparams["minmem"]
816 else:
817 self.fields["memory"].initial = beparams["memory"]
819 # If there are ipolicy limits in place, add validators for them.
820 if "ipolicy" in cluster.info:
821 if "max" in cluster.info["ipolicy"]:
822 # disk maximums
823 v = cluster.info["ipolicy"]["max"]["disk-size"]
824 for disk in xrange(settings.MAX_DISKS_ADD):
825 self.fields["disk_size_%s" % disk].validators.append(
826 MaxValueValidator(v))
827 # ram minimums
828 v = cluster.info["ipolicy"]["max"]["memory-size"]
829 self.fields["memory"].validators.append(MaxValueValidator(v))
830 if has_balloonmem(cluster):
831 self.fields["minram"].validators.append(
832 MaxValueValidator(v))
834 if "min" in cluster.info["ipolicy"]:
835 # disk minimums
836 v = cluster.info["ipolicy"]["min"]["disk-size"]
837 for disk in xrange(settings.MAX_DISKS_ADD):
838 disk_field = self.fields["disk_size_%s" % disk]
839 disk_field.validators.append(MinValueValidator(v))
840 # if its the first disk, add the min value as a default
841 if disk == 0:
842 disk_field.initial = v
843 # memory minimums
844 v = cluster.info["ipolicy"]["min"]["memory-size"]
845 self.fields["memory"].validators.append(MinValueValidator(v))
846 if has_balloonmem(cluster):
847 self.fields["minram"].validators.append(
848 MinValueValidator(v))
850 # configure cluster defaults for nics
851 nic_defaults = cluster.info['nicparams']['default']
852 self.fields['nic_mode_0'].initial = nic_defaults['mode']
853 self.fields['nic_link_0'].initial = nic_defaults['link']
855 def _configure_for_template(self, template):
856 if not template:
857 return
859 self.fields["os"].initial = template.os
860 self.fields["vcpus"].initial = template.vcpus
861 self.fields["memory"].initial = template.memory
862 if has_balloonmem(template.cluster):
863 self.fields["minram"].initial = template.minmem
864 self.fields["disk_template"].initial = template.disk_template
865 for num, disk in enumerate(template.disks):
866 self.fields["disk_size_%s" % num].initial = disk["size"]
867 for num, nic in enumerate(template.nics):
868 self.fields["nic_link_%s" % num].initial = nic['link']
869 self.fields["nic_mode_%s" % num].initial = nic['mode']
871 def clean(self):
872 data = self.cleaned_data
873 # get disk sizes after validation (after 1.5G -> 1500)
874 # and filter empty fields.
875 disks = []
876 for disk_num in xrange(settings.MAX_DISKS_ADD):
877 disk = data.get("disk_size_%s" % disk_num, None)
878 if disk:
879 disks.append(disk)
880 # if disks validated (no errors), but none of them contain data, then
881 # they were all left empty
882 if not disks and not self._errors:
883 msg = _("You need to add at least 1 disk!")
884 self._errors["disk_size_0"] = self.error_class([msg])
886 # Store disks as an array of dicts for use in template.
887 data["disks"] = [{"size": disk} for disk in disks]
889 nics = []
890 for nic in xrange(settings.MAX_NICS_ADD):
891 link = data.get('nic_link_%s' % nic, None)
892 mode = data.get('nic_mode_%s' % nic, None)
893 # if both the mode and link for a NIC are filled out, add it to the
894 # nic list.
895 if link and mode:
896 nics.append({'link': link, 'mode': mode})
897 elif link or mode:
898 raise ValidationError(_("Please input both a link and mode."))
900 data['nics'] = nics
902 if data.get('minram') > data.get('memory'):
903 msg = _("The minimum ram cannot be larger than the maximum ram.")
904 self._errors["minram"] = self.error_class([msg])
906 return data
909 class VMWizardAdvancedForm(Form):
910 ip_check = BooleanField(label=_('Verify IP'), initial=False,
911 required=False,
912 help_text=_(VM_RENAME_HELP['ip_check']))
913 name_check = BooleanField(label=_('Verify hostname through DNS'),
914 initial=False, required=False,
915 help_text=_(VM_RENAME_HELP['name_check']))
916 no_start = BooleanField(label=_('Do not boot the VM'), required=False)
917 pnode = ModelChoiceField(label=_("Primary Node"),
918 queryset=Node.objects.all(), empty_label=None,
919 help_text=_(VM_CREATE_HELP['pnode']))
920 snode = ModelChoiceField(label=_("Secondary Node"),
921 queryset=Node.objects.all(), empty_label=None,
922 help_text=_(VM_CREATE_HELP['snode']))
924 def _configure_for_cluster(self, cluster):
925 if not cluster:
926 return
928 self.cluster = cluster
930 qs = Node.objects.filter(cluster=cluster)
931 self.fields["pnode"].queryset = qs
932 self.fields["snode"].queryset = qs
934 def _configure_for_template(self, template):
935 if not template:
936 return
938 self.fields["ip_check"].initial = template.ip_check
939 self.fields["name_check"].initial = template.name_check
940 self.fields["pnode"].initial = template.pnode
941 self.fields["snode"].initial = template.snode
943 def _configure_for_iallocator(self, use_iallocator):
944 if use_iallocator:
945 del self.fields["pnode"]
946 del self.fields["snode"]
947 self.use_iallocator = use_iallocator
949 def _configure_for_disk_template(self, template, use_iallocator=False):
950 # If its not drdb, we dont use the secondary node.
951 # If we're using the iallocator then this field
952 # will already be deleted.
953 if template != "drbd" and not use_iallocator:
954 del self.fields["snode"]
957 def clean(self):
958 # Ganeti will error on VM creation if an IP address check is requested
959 # but a name check is not.
960 data = self.cleaned_data
961 if (data.get("ip_check") and not data.get("name_check")):
962 msg = ["Cannot perform IP check without name check"]
963 self._errors["ip_check"] = self.error_class(msg)
965 if not self.use_iallocator and data.get('pnode') == data.get('snode'):
966 raise forms.ValidationError("The secondary node cannot be the "
967 "primary node.")
969 return data
972 class VMWizardPVMForm(Form):
973 kernel_path = CharField(label=_("Kernel path"), max_length=255)
974 root_path = CharField(label=_("Root path"), max_length=255)
976 def _configure_for_cluster(self, cluster):
977 if not cluster:
978 return
980 self.cluster = cluster
981 params = cluster.info["hvparams"]["xen-pvm"]
983 self.fields["kernel_path"].initial = params["kernel_path"]
984 self.fields["root_path"].initial = params["root_path"]
986 def _configure_for_template(self, template):
987 if not template:
988 return
990 self.fields["kernel_path"].initial = template.kernel_path
991 self.fields["root_path"].initial = template.root_path
994 class VMWizardHVMForm(Form):
995 boot_order = ChoiceField(label=_("Preferred boot device"),
996 required=False, choices=HVM_BOOT_ORDER,
997 help_text=_(VM_CREATE_HELP['boot_order']))
998 cdrom_image_path = CharField(label=_("CD-ROM image path"), max_length=512,
999 required=False,
1000 help_text=_(
1001 VM_CREATE_HELP['cdrom_image_path']))
1002 disk_type = ChoiceField(label=_("Disk type"),
1003 choices=HVM_CHOICES["disk_type"],
1004 help_text=_(VM_CREATE_HELP['disk_type']))
1005 nic_type = ChoiceField(label=_("NIC type"),
1006 choices=HVM_CHOICES["nic_type"],
1007 help_text=_(VM_CREATE_HELP['nic_type']))
1009 def _configure_for_cluster(self, cluster):
1010 if not cluster:
1011 return
1013 self.cluster = cluster
1014 params = cluster.info["hvparams"]["xen-pvm"]
1016 self.fields["boot_order"].initial = params["boot_order"]
1017 self.fields["disk_type"].initial = params["disk_type"]
1018 self.fields["nic_type"].initial = params["nic_type"]
1020 def _configure_for_template(self, template):
1021 if not template:
1022 return
1024 self.fields["boot_order"].initial = template.boot_order
1025 self.fields["cdrom_image_path"].initial = template.cdrom_image_path
1026 self.fields["disk_type"].initial = template.disk_type
1027 self.fields["nic_type"].initial = template.nic_type
1030 class VMWizardKVMForm(Form):
1031 kernel_path = CharField(label=_("Kernel path"), max_length=255,
1032 required=False,
1033 help_text=_(VM_CREATE_HELP['kernel_path']))
1034 root_path = CharField(label=_("Root path"), max_length=255,
1035 help_text=_(VM_CREATE_HELP['root_path']))
1036 serial_console = BooleanField(label=_("Enable serial console"),
1037 required=False,
1038 help_text=_(
1039 VM_CREATE_HELP['serial_console']))
1040 boot_order = ChoiceField(label=_("Preferred boot device"),
1041 required=False, choices=KVM_BOOT_ORDER,
1042 help_text=_(VM_CREATE_HELP['boot_order']))
1043 cdrom_image_path = CharField(label=_("CD-ROM image path"), max_length=512,
1044 required=False,
1045 help_text=_(
1046 VM_CREATE_HELP['cdrom_image_path']))
1047 cdrom2_image_path = CharField(label=_("Second CD-ROM image path"),
1048 max_length=512, required=False,
1049 help_text=_(
1050 VM_CREATE_HELP['cdrom2_image_path']))
1051 disk_type = ChoiceField(label=_("Disk type"),
1052 choices=KVM_CHOICES["disk_type"],
1053 help_text=_(VM_CREATE_HELP['disk_type']))
1054 nic_type = ChoiceField(label=_("NIC type"),
1055 choices=KVM_CHOICES["nic_type"],
1056 help_text=_(VM_CREATE_HELP['nic_type']))
1058 def _configure_for_cluster(self, cluster):
1059 if not cluster:
1060 return
1062 self.cluster = cluster
1063 params = cluster.info["hvparams"]["kvm"]
1065 self.fields["boot_order"].initial = params["boot_order"]
1066 self.fields["disk_type"].initial = params["disk_type"]
1067 self.fields["kernel_path"].initial = params["kernel_path"]
1068 self.fields["nic_type"].initial = params["nic_type"]
1069 self.fields["root_path"].initial = params["root_path"]
1070 self.fields["serial_console"].initial = params["serial_console"]
1072 # Remove cdrom2 if the cluster doesn't have it; see #11655.
1073 if not has_cdrom2(cluster):
1074 del self.fields["cdrom2_image_path"]
1076 def _configure_for_template(self, template):
1077 if not template:
1078 return
1080 self.fields["kernel_path"].initial = template.kernel_path
1081 self.fields["root_path"].initial = template.root_path
1082 self.fields["serial_console"].initial = template.serial_console
1083 self.fields["boot_order"].initial = template.boot_order
1084 self.fields["cdrom_image_path"].initial = template.cdrom_image_path
1085 self.fields["cdrom2_image_path"].initial = template.cdrom2_image_path
1086 self.fields["disk_type"].initial = template.disk_type
1087 self.fields["nic_type"].initial = template.nic_type
1089 def clean(self):
1090 data = super(VMWizardKVMForm, self).clean()
1092 # Force cdrom disk type to IDE; see #9297.
1093 data['cdrom_disk_type'] = 'ide'
1095 # If booting from CD-ROM, require the first CD-ROM image to be
1096 # present.
1097 if (data.get("boot_order") == "cdrom" and
1098 not data.get("cdrom_image_path")):
1099 msg = u"%s." % _("Image path required if boot device is CD-ROM")
1100 self._errors["cdrom_image_path"] = self.error_class([msg])
1102 return data
1105 class VMWizardView(LoginRequiredMixin, CookieWizardView):
1106 template_name = "ganeti/forms/vm_wizard.html"
1108 OPTIONS = (
1109 # value, display value
1110 # value corresponds to VMWizardOwnerForm's fields
1111 ('template_name', 'Template'),
1112 ('hostname', 'Virtual Machine'),
1115 def _get_vm_or_template(self):
1116 """Returns items that were not checked in step0"""
1117 data = self.get_cleaned_data_for_step('0')
1118 if data:
1119 options = [option[0] for option in self.OPTIONS]
1120 choices = data.get('choices', None)
1121 # which boxes weren't checked
1122 unchecked = set(options) - set(choices)
1123 return unchecked
1125 return None
1127 def _get_template(self):
1128 name = self.kwargs.get("template")
1129 if name:
1130 return VirtualMachineTemplate.objects.get(template_name=name)
1131 return None
1133 def _get_cluster(self):
1134 data = self.get_cleaned_data_for_step("0")
1135 if data:
1136 return data["cluster"]
1137 return None
1139 def _get_hv(self):
1140 data = self.get_cleaned_data_for_step("2")
1141 if data:
1142 return data["hv"]
1143 return None
1145 def _get_disk_template(self):
1146 data = self.get_cleaned_data_for_step("2")
1147 if data:
1148 return data["disk_template"]
1149 return None
1151 def _get_iallocator(self):
1152 data = self.get_cleaned_data_for_step("2")
1153 if data:
1154 return data["iallocator"]
1155 return False
1157 def get_form(self, step=None, data=None, files=None):
1158 s = int(self.steps.current) if step is None else int(step)
1159 initial = self.get_form_initial(s)
1161 if s == 0:
1162 form = VMWizardClusterForm(data=data, options=self.OPTIONS,
1163 initial=initial)
1164 form._configure_for_user(self.request.user)
1165 # XXX this should somehow become totally invalid if the user
1166 # doesn't have perms on the template.
1167 elif s == 1:
1168 form = VMWizardOwnerForm(data=data)
1169 form._configure_for_cluster(self._get_cluster())
1170 form._configure_for_template(self._get_template(),
1171 choices=self._get_vm_or_template())
1172 elif s == 2:
1173 form = VMWizardBasicsForm(data=data)
1174 form._configure_for_cluster(self._get_cluster())
1175 form._configure_for_template(self._get_template())
1176 elif s == 3:
1177 form = VMWizardAdvancedForm(data=data)
1178 form._configure_for_cluster(self._get_cluster())
1179 form._configure_for_template(self._get_template())
1180 using_iallocator = self._get_iallocator()
1181 form._configure_for_iallocator(using_iallocator)
1182 form._configure_for_disk_template(self._get_disk_template(),
1183 using_iallocator)
1184 elif s == 4:
1185 cluster = self._get_cluster()
1186 hv = self._get_hv()
1187 form = None
1189 if cluster and hv:
1190 if hv == "kvm":
1191 form = VMWizardKVMForm(data=data)
1192 elif hv == "xen-pvm":
1193 form = VMWizardPVMForm(data=data)
1194 elif hv == "xen-hvm":
1195 form = VMWizardHVMForm(data=data)
1197 if form:
1198 form._configure_for_cluster(cluster)
1199 form._configure_for_template(self._get_template())
1200 else:
1201 form = Form()
1202 else:
1203 form = super(VMWizardView, self).get_form(step, data, files)
1205 return form
1207 def get_context_data(self, form, **kwargs):
1208 context = super(VMWizardView, self).get_context_data(form=form,
1209 **kwargs)
1210 summary = {
1211 "cluster_form": self.get_cleaned_data_for_step("0"),
1212 "owner_form": self.get_cleaned_data_for_step("1"),
1213 "basics_form": self.get_cleaned_data_for_step("2"),
1214 "advanced_form": self.get_cleaned_data_for_step("3"),
1215 "hv_form": self.get_cleaned_data_for_step("4"),
1217 context["summary"] = summary
1219 return context
1221 def done(self, forms, template=None, **kwargs):
1223 Create a template. Optionally, bind a template to a VM instance
1224 created from the template. Optionally, name the template and save it.
1225 One or both of those is done depending on what the user has requested.
1228 # Hack: accepting kwargs in order to be able to work in several
1229 # different spots.
1231 if template is None:
1232 template = VirtualMachineTemplate()
1233 else:
1234 template = self._get_template()
1236 user = self.request.user
1238 cluster = forms[0].cleaned_data["cluster"]
1239 owner = forms[1].cleaned_data["owner"]
1241 template_name = forms[1].cleaned_data["template_name"]
1242 hostname = forms[1].cleaned_data["hostname"]
1244 # choice_data are the options that were not checked
1245 # if unchecked, than we should make sure that this is not submitted.
1246 # this fixes cases where the user checked a box in the beginning, put
1247 # data into the input, and went back and unchecked that box later.
1248 unchecked_options = self._get_vm_or_template()
1249 for unchecked in unchecked_options:
1250 if 'template_name' == unchecked:
1251 template_name = ''
1252 if 'hostname' == unchecked:
1253 hostname = ''
1255 template.cluster = cluster
1256 template.memory = forms[2].cleaned_data["memory"]
1257 if has_balloonmem(cluster):
1258 template.minmem = forms[2].cleaned_data["minram"]
1259 template.vcpus = forms[2].cleaned_data["vcpus"]
1260 template.disk_template = forms[2].cleaned_data["disk_template"]
1262 template.disks = forms[2].cleaned_data["disks"]
1264 nics = forms[2].cleaned_data["nics"]
1265 # default
1266 if not nics:
1267 nics = [{"link": "br0", "mode": "bridged"}]
1268 template.nics = nics
1270 template.hypervisor = forms[2].cleaned_data["hv"]
1272 template.os = forms[2].cleaned_data["os"]
1273 template.no_install = forms[2].cleaned_data["no_install"]
1274 template.iallocator = forms[2].cleaned_data["iallocator"]
1275 template.ip_check = forms[3].cleaned_data["ip_check"]
1276 template.name_check = forms[3].cleaned_data["name_check"]
1277 template.no_start = forms[3].cleaned_data["no_start"]
1279 if not template.iallocator:
1280 template.pnode = forms[3].cleaned_data["pnode"].hostname
1281 if "snode" in forms[3].cleaned_data:
1282 template.snode = forms[3].cleaned_data["snode"].hostname
1284 hvparams = forms[4].cleaned_data
1286 template.boot_order = hvparams.get("boot_order")
1287 template.cdrom2_image_path = hvparams.get("cdrom2_image_path")
1288 template.cdrom_image_path = hvparams.get("cdrom_image_path")
1289 template.kernel_path = hvparams.get("kernel_path")
1290 template.root_path = hvparams.get("root_path")
1291 template.serial_console = hvparams.get("serial_console")
1292 template.nic_type = hvparams.get('nic_type')
1293 template.disk_type = hvparams.get('disk_type')
1295 template.set_name(template_name)
1296 # only save the template to the database if its not temporary
1297 if not template.temporary:
1298 template.save()
1300 if hostname:
1301 vm = template_to_instance(template, hostname, owner)
1302 log_action('CREATE', user, vm)
1303 return HttpResponseRedirect(reverse('instance-detail',
1304 args=[cluster.slug,
1305 vm.hostname]))
1306 else:
1307 return HttpResponseRedirect(reverse("template-detail",
1308 args=[cluster.slug,
1309 template]))
1312 def vm_wizard(*args, **kwargs):
1313 forms = (
1314 VMWizardClusterForm,
1315 VMWizardOwnerForm,
1316 VMWizardBasicsForm,
1317 VMWizardAdvancedForm,
1318 Form,
1320 initial = kwargs.get('initial_dict', None)
1321 return VMWizardView.as_view(forms, initial_dict=initial)