mirror of
https://github.com/retspen/webvirtcloud
synced 2025-01-12 08:25:18 +00:00
Merge pull request #321 from catborise/master
Translation and locale file updates
This commit is contained in:
commit
ad4b417695
57 changed files with 13088 additions and 5644 deletions
|
@ -7,7 +7,7 @@ from django.conf import settings
|
||||||
|
|
||||||
class UserAddForm(forms.Form):
|
class UserAddForm(forms.Form):
|
||||||
name = forms.CharField(label="Name",
|
name = forms.CharField(label="Name",
|
||||||
error_messages={'required': _('No User name has been entered')},
|
error_messages={'required': _('No username has been entered')},
|
||||||
max_length=20)
|
max_length=20)
|
||||||
password = forms.CharField(required=not settings.ALLOW_EMPTY_PASSWORD,
|
password = forms.CharField(required=not settings.ALLOW_EMPTY_PASSWORD,
|
||||||
error_messages={'required': _('No password has been entered')},)
|
error_messages={'required': _('No password has been entered')},)
|
||||||
|
|
|
@ -19,8 +19,8 @@ class UserInstance(models.Model):
|
||||||
|
|
||||||
class UserSSHKey(models.Model):
|
class UserSSHKey(models.Model):
|
||||||
user = models.ForeignKey(User, on_delete=models.DO_NOTHING)
|
user = models.ForeignKey(User, on_delete=models.DO_NOTHING)
|
||||||
keyname = models.CharField(max_length=25)
|
keyname = models.CharField(_('key name'), max_length=25)
|
||||||
keypublic = models.CharField(max_length=500)
|
keypublic = models.CharField(_('public key'), max_length=500)
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return self.keyname
|
return self.keyname
|
||||||
|
@ -29,24 +29,28 @@ class UserSSHKey(models.Model):
|
||||||
class UserAttributes(models.Model):
|
class UserAttributes(models.Model):
|
||||||
user = models.OneToOneField(User, on_delete=models.CASCADE)
|
user = models.OneToOneField(User, on_delete=models.CASCADE)
|
||||||
can_clone_instances = models.BooleanField(default=True)
|
can_clone_instances = models.BooleanField(default=True)
|
||||||
max_instances = models.IntegerField(default=2,
|
max_instances = models.IntegerField(_('max instances'),
|
||||||
help_text="-1 for unlimited. Any integer value",
|
default=2,
|
||||||
|
help_text=_("-1 for unlimited. Any integer value"),
|
||||||
validators=[
|
validators=[
|
||||||
MinValueValidator(-1),
|
MinValueValidator(-1),
|
||||||
])
|
])
|
||||||
max_cpus = models.IntegerField(
|
max_cpus = models.IntegerField(
|
||||||
|
_('max CPUs'),
|
||||||
default=2,
|
default=2,
|
||||||
help_text="-1 for unlimited. Any integer value",
|
help_text=_("-1 for unlimited. Any integer value"),
|
||||||
validators=[MinValueValidator(-1)],
|
validators=[MinValueValidator(-1)],
|
||||||
)
|
)
|
||||||
max_memory = models.IntegerField(
|
max_memory = models.IntegerField(
|
||||||
|
_('max memory'),
|
||||||
default=2048,
|
default=2048,
|
||||||
help_text="-1 for unlimited. Any integer value",
|
help_text=_("-1 for unlimited. Any integer value"),
|
||||||
validators=[MinValueValidator(-1)],
|
validators=[MinValueValidator(-1)],
|
||||||
)
|
)
|
||||||
max_disk_size = models.IntegerField(
|
max_disk_size = models.IntegerField(
|
||||||
|
_('max disk size'),
|
||||||
default=20,
|
default=20,
|
||||||
help_text="-1 for unlimited. Any integer value",
|
help_text=_("-1 for unlimited. Any integer value"),
|
||||||
validators=[MinValueValidator(-1)],
|
validators=[MinValueValidator(-1)],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -8,7 +8,7 @@
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
{% include 'create_user_block.html' %}
|
{% include 'create_user_block.html' %}
|
||||||
<div class="float-right search">
|
<div class="float-right search">
|
||||||
<input id="filter" class="form-control" type="text" placeholder="Search">
|
<input id="filter" class="form-control" type="text" placeholder="{% trans 'Search' %}">
|
||||||
</div>
|
</div>
|
||||||
<h2 class="page-header">{% trans "Users" %}</h1>
|
<h2 class="page-header">{% trans "Users" %}</h1>
|
||||||
</div>
|
</div>
|
||||||
|
@ -22,7 +22,7 @@
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
<div class="alert alert-warning alert-dismissable">
|
<div class="alert alert-warning alert-dismissable">
|
||||||
<button type="button" class="close" data-dismiss="alert" aria-hidden="true">×</button>
|
<button type="button" class="close" data-dismiss="alert" aria-hidden="true">×</button>
|
||||||
<i class="fa fa-exclamation-triangle"></i> <strong>{% trans "Warning" %}:</strong> {% trans "You don't have any User" %}
|
<i class="fa fa-exclamation-triangle"></i> <strong>{% trans "Warning" %}:</strong> {% trans "You don't have any user" %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% else %}
|
{% else %}
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
<div class="alert alert-warning alert-dismissable">
|
<div class="alert alert-warning alert-dismissable">
|
||||||
<button type="button" class="close" data-dismiss="alert" aria-hidden="true">×</button>
|
<button type="button" class="close" data-dismiss="alert" aria-hidden="true">×</button>
|
||||||
<i class="fa fa-exclamation-triangle"></i> <strong>{% trans "Warning" %}:</strong> {% trans "You don't have any User" %}
|
<i class="fa fa-exclamation-triangle"></i> <strong>{% trans "Warning" %}:</strong> {% trans "You don't have any user" %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% else %}
|
{% else %}
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Name" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Name" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="text" class="form-control" name="name" placeholder="john" required pattern="[a-z0-9]+">
|
<input type="text" class="form-control" name="name" placeholder="{% trans "john" %}" required pattern="[a-z0-9]+">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
|
|
|
@ -9,7 +9,6 @@ class Migration(migrations.Migration):
|
||||||
initial = True
|
initial = True
|
||||||
|
|
||||||
dependencies = [
|
dependencies = [
|
||||||
('auth', '0011_update_proxy_permissions'),
|
|
||||||
]
|
]
|
||||||
|
|
||||||
operations = [
|
operations = [
|
||||||
|
|
14
admin/migrations/0002_auto_20200609_0830.py
Normal file
14
admin/migrations/0002_auto_20200609_0830.py
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
# Generated by Django 2.2.12 on 2020-06-09 08:30
|
||||||
|
|
||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('admin', '0001_initial'),
|
||||||
|
('auth', '0011_update_proxy_permissions'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
]
|
|
@ -22,7 +22,7 @@
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
<div class="alert alert-warning alert-dismissable">
|
<div class="alert alert-warning alert-dismissable">
|
||||||
<button type="button" class="close" data-dismiss="alert" aria-hidden="true">×</button>
|
<button type="button" class="close" data-dismiss="alert" aria-hidden="true">×</button>
|
||||||
{% icon 'exclamation-triangle '%} <strong>{% trans "Warning" %}:</strong> {% trans "You don't have any users" %}
|
{% icon 'exclamation-triangle '%} <strong>{% trans "Warning" %}:</strong> {% trans "You don't have any user" %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% else %}
|
{% else %}
|
||||||
|
@ -35,7 +35,7 @@
|
||||||
<th span="col">{% trans "Staff" %}</th>
|
<th span="col">{% trans "Staff" %}</th>
|
||||||
<th span="col">{% trans "Superuser" %}</th>
|
<th span="col">{% trans "Superuser" %}</th>
|
||||||
<th span="col">{% trans "Can Clone" %}</th>
|
<th span="col">{% trans "Can Clone" %}</th>
|
||||||
<th span="col">{% trans "" %}</th>
|
<th span="col">{% trans "Actions" %}</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody class="searchable">
|
<tbody class="searchable">
|
||||||
|
|
|
@ -1,65 +1,38 @@
|
||||||
# Generated by Django 2.2.12 on 2020-05-23 12:05
|
# Generated by Django 2.2.12 on 2020-05-23 12:05
|
||||||
|
|
||||||
from django.db import migrations
|
from django.db import migrations
|
||||||
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
def add_default_settings(apps, schema_editor):
|
def add_default_settings(apps, schema_editor):
|
||||||
|
|
||||||
setting = apps.get_model("appsettings", "AppSettings")
|
setting = apps.get_model("appsettings", "AppSettings")
|
||||||
db_alias = schema_editor.connection.alias
|
db_alias = schema_editor.connection.alias
|
||||||
setting.objects.using(db_alias).bulk_create([
|
setting.objects.using(db_alias).bulk_create([
|
||||||
setting(1, "Theme", "BOOTSTRAP_THEME", "flaty",
|
setting(1, _("Theme"), "BOOTSTRAP_THEME", "flaty", "", _("Bootstrap CSS & Bootswatch Theme")),
|
||||||
"", "Bootstrap CSS & Bootswatch Theme"),
|
setting(2, _("Theme SASS Path"), "SASS_DIR", "dev/scss/", "", _("Bootstrap SASS & Bootswatch SASS Directory")),
|
||||||
setting(2, "Theme SASS Path", "SASS_DIR", "dev/scss/",
|
setting(3, _("All Instances View Style"), "VIEW_INSTANCES_LIST_STYLE", "grouped", "grouped,nongrouped", _("All instances list style")),
|
||||||
"", "Bootstrap SASS & Bootswatch SASS Directory"),
|
setting(4, _("Logs per Page"), "LOGS_PER_PAGE", "100", "", _("Pagination for logs")),
|
||||||
setting(3, "All Instances View Style", "VIEW_INSTANCES_LIST_STYLE",
|
setting(5, _("Multiple Owner for VM"), "ALLOW_INSTANCE_MULTIPLE_OWNER", "True", "True,False", _("Allow to have multiple owner for instance")),
|
||||||
"grouped", "grouped,nongrouped", "All instances list style"),
|
setting(6, _("Quota Debug"), "QUOTA_DEBUG", "True", "True,False", _("Debug for user quotas")),
|
||||||
setting(4, "Logs per Page", "LOGS_PER_PAGE",
|
setting(7, _("Disk Format"), "INSTANCE_VOLUME_DEFAULT_FORMAT", "qcow2", "raw,qcow,qcow2", _("Instance disk format")),
|
||||||
"100", "", "Pagination for logs"),
|
setting(8, _("Disk Bus"), "INSTANCE_VOLUME_DEFAULT_BUS", "virtio", "virtio,scsi,ide,usb,sata", _("Instance disk bus type")),
|
||||||
setting(5, "Multiple Owner for VM ", "ALLOW_INSTANCE_MULTIPLE_OWNER",
|
setting(9, _("Disk SCSI Controller"), "INSTANCE_VOLUME_DEFAULT_SCSI_CONTROLLER", "virtio-scsi", "virtio-scsi, lsilogic, virtio-blk", _("SCSI controller type")),
|
||||||
"True", "True,False", "Allow to have multiple owner for instance"),
|
setting(10, _("Disk Cache"), "INSTANCE_VOLUME_DEFAULT_CACHE", "directsync", "default,directsync,none,unsafe,writeback,writethrough", _("Disk volume cache type")),
|
||||||
setting(6, "Quota Debug", "QUOTA_DEBUG", "True",
|
setting(11, _("Disk IO Type"), "INSTANCE_VOLUME_DEFAULT_IO", "default", "default,native,threads", _("Volume io modes")),
|
||||||
"True,False", "Debug for user quotas"),
|
setting(12, _("Disk Detect Zeroes"), "INSTANCE_VOLUME_DEFAULT_DETECT_ZEROES", "default", "default,on,off,unmap", _("Volume detect zeroes mode")),
|
||||||
setting(7, "Disk Format", "INSTANCE_VOLUME_DEFAULT_FORMAT",
|
setting(13, _("Disk Discard"), "INSTANCE_VOLUME_DEFAULT_DISCARD", "default", "default,unmap,ignore", _("Volume discard mode")),
|
||||||
"qcow2", "raw,qcow,qcow2", "Instance disk format"),
|
setting(14, _("Disk Owner UID"), "INSTANCE_VOLUME_DEFAULT_OWNER_UID", "0", "", _("Owner UID: up to os, 0=root, 107=qemu or libvirt-bin(for ubuntu)")),
|
||||||
setting(8, "Disk Bus", "INSTANCE_VOLUME_DEFAULT_BUS", "virtio",
|
setting(15, _("Disk Owner GID"), "INSTANCE_VOLUME_DEFAULT_OWNER_GID", "0", "", _("Owner GID: up to os, 0=root, 107=qemu or libvirt-bin(for ubuntu)")),
|
||||||
"virtio,scsi,ide,usb,sata", "Instance disk bus type"),
|
setting(16, _("VM CPU Mode"), "INSTANCE_CPU_DEFAULT_MODE", "host-model", "no-model,host-model,host-passthrough,custom", _("Cpu modes")),
|
||||||
setting(9, "Disk SCSI Controller", "INSTANCE_VOLUME_DEFAULT_SCSI_CONTROLLER",
|
setting(17, _("VM Machine Type"), "INSTANCE_MACHINE_DEFAULT_TYPE", "q35", "q35,x86_64", _("Chipset/Machine type")),
|
||||||
"virtio-scsi", "virtio-scsi, lsilogic, virtio-blk", "SCSI controller type"),
|
setting(18, _("VM Firmware Type"), "INSTANCE_FIRMWARE_DEFAULT_TYPE", "BIOS", "BIOS,UEFI", _("Firmware type for x86_64")),
|
||||||
setting(10, "Disk Cache", "INSTANCE_VOLUME_DEFAULT_CACHE", "directsync",
|
setting(19, _("VM Architecture Type"), "INSTANCE_ARCH_DEFAULT_TYPE", "x86_64", "x86_64,i686", _("Architecture type: x86_64, i686, etc")),
|
||||||
"default,directsync,none,unsafe,writeback,writethrough", "Disk volume cache type"),
|
setting(20, _("VM Console Type"), "QEMU_CONSOLE_DEFAULT_TYPE", "vnc", "vnc,spice", _("Default console type")),
|
||||||
setting(11, "Disk IO Type", "INSTANCE_VOLUME_DEFAULT_IO",
|
setting(21, _("VM Clone Name Prefix"), "CLONE_INSTANCE_DEFAULT_PREFIX", "instance", "True,False", _("Prefix for cloned instance name")),
|
||||||
"default", "default,native,threads", "Volume io modes"),
|
setting(22, _("VM Clone Auto Name"), "CLONE_INSTANCE_AUTO_NAME", "False", "True,False", _("Generated name for cloned instance")),
|
||||||
setting(12, "Disk Detect Zeroes", "INSTANCE_VOLUME_DEFAULT_DETECT_ZEROES",
|
setting(23, _("VM Clone Auto Migrate"), "CLONE_INSTANCE_AUTO_MIGRATE", "False", "True,False", _("Auto migrate instance after clone")),
|
||||||
"default", "default,on,off,unmap", "Volume detect zeroes mode"),
|
setting(24, _("VM Bottom Bar"), "VIEW_INSTANCE_DETAIL_BOTTOM_BAR", "True", "True,False", _("Bottom navbar for instance details")),
|
||||||
setting(13, "Disk Discard", "INSTANCE_VOLUME_DEFAULT_DISCARD",
|
setting(25, _("Show Access Root Pass"), "SHOW_ACCESS_ROOT_PASSWORD", "False", "True,False", _("Show access root password")),
|
||||||
"default", "default,unmap,ignore", "Volume discard mode"),
|
setting(26, _("Show Access SSH Keys"), "SHOW_ACCESS_SSH_KEYS", "False", "True,False", _("Show access ssh keys")),
|
||||||
setting(14, "Disk Owner UID", "INSTANCE_VOLUME_DEFAULT_OWNER_UID", "0",
|
|
||||||
"", "Owner UID: up to os, 0=root, 107=qemu or libvirt-bin(for ubuntu)"),
|
|
||||||
setting(15, "Disk Owner GID", "INSTANCE_VOLUME_DEFAULT_OWNER_GID", "0",
|
|
||||||
"", "Owner GID: up to os, 0=root, 107=qemu or libvirt-bin(for ubuntu)"),
|
|
||||||
setting(16, "VM CPU Mode", "INSTANCE_CPU_DEFAULT_MODE", "host-model",
|
|
||||||
"no-model,host-model,host-passthrough,custom", "Cpu modes"),
|
|
||||||
setting(17, "VM Machine Type", "INSTANCE_MACHINE_DEFAULT_TYPE",
|
|
||||||
"q35", "q35,x86_64", "Chipset/Machine type"),
|
|
||||||
setting(18, "VM Firmware Type", "INSTANCE_FIRMWARE_DEFAULT_TYPE",
|
|
||||||
"BIOS", "BIOS,UEFI", "Firmware type for x86_64"),
|
|
||||||
setting(19, "VM Architecture Type", "INSTANCE_ARCH_DEFAULT_TYPE",
|
|
||||||
"x86_64", "x86_64,i686", "Architecture type: x86_64, i686, etc"),
|
|
||||||
setting(20, "VM Console Type", "QEMU_CONSOLE_DEFAULT_TYPE",
|
|
||||||
"vnc", "vnc,spice", "Default console type"),
|
|
||||||
setting(21, "VM Clone Name Prefix", "CLONE_INSTANCE_DEFAULT_PREFIX",
|
|
||||||
"instance", "True,False", "Prefix for cloned instance name"),
|
|
||||||
setting(22, "VM Clone Auto Name", "CLONE_INSTANCE_AUTO_NAME",
|
|
||||||
"False", "True,False", "Generated name for cloned instance"),
|
|
||||||
setting(23, "VM Clone Auto Migrate", "CLONE_INSTANCE_AUTO_MIGRATE",
|
|
||||||
"False", "True,False", "Auto migrate instance after clone"),
|
|
||||||
setting(24, "VM Bottom Bar", "VIEW_INSTANCE_DETAIL_BOTTOM_BAR",
|
|
||||||
"True", "True,False", "Bottom navbar for instance details"),
|
|
||||||
setting(25, "Show Access Root Pass", "SHOW_ACCESS_ROOT_PASSWORD",
|
|
||||||
"False", "True,False", "Show access root password"),
|
|
||||||
setting(26, "Show Access SSH Keys", "SHOW_ACCESS_SSH_KEYS",
|
|
||||||
"False", "True,False", "Show access ssh keys"),
|
|
||||||
])
|
])
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,13 +1,13 @@
|
||||||
from django.db import models
|
from django.db import models
|
||||||
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
class AppSettings(models.Model):
|
class AppSettings(models.Model):
|
||||||
|
|
||||||
def choices_as_list(self):
|
def choices_as_list(self):
|
||||||
return self.choices.split(',')
|
return self.choices.split(',')
|
||||||
|
|
||||||
name = models.CharField(max_length=25, null=False)
|
name = models.CharField(_('name'), max_length=25, null=False)
|
||||||
key = models.CharField(db_index=True, max_length=50, unique=True)
|
key = models.CharField(_('key'), db_index=True, max_length=50, unique=True)
|
||||||
value = models.CharField(max_length=25)
|
value = models.CharField(_('value'), max_length=25)
|
||||||
choices = models.CharField(max_length=50)
|
choices = models.CharField(_('choices'), max_length=50)
|
||||||
description = models.CharField(max_length=100, null=True)
|
description = models.CharField(_('description'), max_length=100, null=True)
|
||||||
|
|
|
@ -37,9 +37,9 @@
|
||||||
{% if request.user.is_superuser %}
|
{% if request.user.is_superuser %}
|
||||||
<form method="post" action="" role="form" aria-label="Edit sass directory settings form">{% csrf_token %}
|
<form method="post" action="" role="form" aria-label="Edit sass directory settings form">{% csrf_token %}
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{{ sass_dir.name }}</label>
|
<label class="col-sm-3 col-form-label">{% trans sass_dir.name %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input class="form-control" name="{{ sass_dir.key }}" value="{{ sass_dir.value }}" onchange="this.form.submit()"/>
|
<input class="form-control" name="{{ sass_dir.key }}" value="{{ sass_dir.value }}" onchange="this.form.submit()" title="{% trans sass_dir.description %}"/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</form>
|
</form>
|
||||||
|
@ -47,7 +47,7 @@
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans bootstrap_theme.name %}</label>
|
<label class="col-sm-3 col-form-label">{% trans bootstrap_theme.name %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<select class="form-control" name="{{ bootstrap_theme.key }}" onchange="this.form.submit()">
|
<select class="form-control" name="{{ bootstrap_theme.key }}" onchange="this.form.submit()" title="{% trans bootstrap_theme.description %}">
|
||||||
{% for theme in themes_list %}
|
{% for theme in themes_list %}
|
||||||
<option {% if bootstrap_theme.value == theme %}selected{% endif %} value="{{ theme }}">{{ theme }}</option>
|
<option {% if bootstrap_theme.value == theme %}selected{% endif %} value="{{ theme }}">{{ theme }}</option>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
@ -61,16 +61,16 @@
|
||||||
{% for setting in appsettings %}
|
{% for setting in appsettings %}
|
||||||
<form method="post" action="" role="form" aria-label="{{setting.name}} form">{% csrf_token %}
|
<form method="post" action="" role="form" aria-label="{{setting.name}} form">{% csrf_token %}
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{{ setting.name }}</label>
|
<label class="col-sm-3 col-form-label">{% trans setting.name %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
{% if setting.choices %}
|
{% if setting.choices %}
|
||||||
<select class="form-control" name="{{ setting.key }}" onchange="this.form.submit()" title="{{ setting.description }}">
|
<select class="form-control" name="{{ setting.key }}" onchange="this.form.submit()" title="{% trans setting.description %}">
|
||||||
{% for choice in setting.choices_as_list %}
|
{% for choice in setting.choices_as_list %}
|
||||||
<option {% if setting.value == choice %} selected {% endif %} value={{ choice }}>{% trans choice %}</option>
|
<option {% if setting.value == choice %} selected {% endif %} value={{ choice }}>{% trans choice %}</option>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</select>
|
</select>
|
||||||
{% else %}
|
{% else %}
|
||||||
<input class="form-control" name="{{ setting.key }}" value="{{ setting.value }}" title="{{ setting.description }}" onchange="this.form.submit()"/>
|
<input class="form-control" name="{{ setting.key }}" value="{{ setting.value }}" title="{% trans setting.description %}" onchange="this.form.submit()"/>
|
||||||
{% endif%}
|
{% endif%}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -20,7 +20,7 @@ def appsettings(request):
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
error_messages = []
|
error_messages = []
|
||||||
|
main_css = "wvc-main.min.css"
|
||||||
sass_dir = AppSettings.objects.get(key="SASS_DIR")
|
sass_dir = AppSettings.objects.get(key="SASS_DIR")
|
||||||
bootstrap_theme = AppSettings.objects.get(key="BOOTSTRAP_THEME")
|
bootstrap_theme = AppSettings.objects.get(key="BOOTSTRAP_THEME")
|
||||||
try:
|
try:
|
||||||
|
@ -59,7 +59,7 @@ def appsettings(request):
|
||||||
main.write(scss_var + "\n" + scss_boot + "\n" + scss_bootswatch + "\n")
|
main.write(scss_var + "\n" + scss_boot + "\n" + scss_bootswatch + "\n")
|
||||||
|
|
||||||
css_compressed = sass.compile(string=scss_var + "\n"+ scss_boot + "\n" + scss_bootswatch, output_style='compressed')
|
css_compressed = sass.compile(string=scss_var + "\n"+ scss_boot + "\n" + scss_bootswatch, output_style='compressed')
|
||||||
with open("static/" + "css/wvc-main.min.css", "w") as css:
|
with open("static/css/" + main_css, "w") as css:
|
||||||
css.write(css_compressed)
|
css.write(css_compressed)
|
||||||
|
|
||||||
bootstrap_theme.value = theme
|
bootstrap_theme.value = theme
|
||||||
|
|
|
@ -1,12 +1,12 @@
|
||||||
from django.db.models import Model, CharField, IntegerField
|
from django.db.models import Model, CharField, IntegerField
|
||||||
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
class Compute(Model):
|
class Compute(Model):
|
||||||
name = CharField(max_length=64, unique=True)
|
name = CharField(_('name'), max_length=64, unique=True)
|
||||||
hostname = CharField(max_length=64)
|
hostname = CharField(_('hostname'), max_length=64)
|
||||||
login = CharField(max_length=20)
|
login = CharField(_('login'), max_length=20)
|
||||||
password = CharField(max_length=14, blank=True, null=True)
|
password = CharField(_('password'), max_length=14, blank=True, null=True)
|
||||||
details = CharField(max_length=64, null=True, blank=True)
|
details = CharField(_('details'), max_length=64, null=True, blank=True)
|
||||||
type = IntegerField()
|
type = IntegerField()
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
<div class="row">
|
<div class="row">
|
||||||
{% if computes_info %}
|
{% if computes_info %}
|
||||||
{% for compute in computes_info %}
|
{% for compute in computes_info %}
|
||||||
<div id="{{ compute.name }}" class="col-12 col-sm-4">
|
<div id="{{ compute.name }}" class="mb-3 col-12 col-sm-4">
|
||||||
{% if compute.status is True %}
|
{% if compute.status is True %}
|
||||||
<div class="card border-success shadow h-100">
|
<div class="card border-success shadow h-100">
|
||||||
<div class="card-header bg-success">
|
<div class="card-header bg-success">
|
||||||
|
@ -37,17 +37,17 @@
|
||||||
</div>
|
</div>
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<dl class="row">
|
<dl class="row">
|
||||||
<dt class="col-4">{% trans "Status" %}</dt>
|
<dt class="col-5">{% trans "Status" %}</dt>
|
||||||
{% if compute.status %}
|
{% if compute.status %}
|
||||||
<dd class="col-8">{% trans "Connected" %}</dd>
|
<dd class="col-7">{% trans "Connected" %}</dd>
|
||||||
{% else %}
|
{% else %}
|
||||||
<dd class="col-8">{% trans "Not Connected" %}</dd>
|
<dd class="col-7">{% trans "Not Connected" %}</dd>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<dt class="col-4">{% trans "Details" %}</dt>
|
<dt class="col-5">{% trans "Details" %}</dt>
|
||||||
{% if compute.details %}
|
{% if compute.details %}
|
||||||
<dd class="col-8">{% trans compute.details %}</dd>
|
<dd class="col-7">{% trans compute.details %}</dd>
|
||||||
{% else %}
|
{% else %}
|
||||||
<dd class="col-8">{% trans "No details available" %}</dd>
|
<dd class="col-7">{% trans "No details available" %}</dd>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</dl>
|
</dl>
|
||||||
|
|
||||||
|
@ -63,7 +63,7 @@
|
||||||
<form method="post" role="form" aria-label="Edit tcp host form">{% csrf_token %}
|
<form method="post" role="form" aria-label="Edit tcp host form">{% csrf_token %}
|
||||||
<div class="modal-body">
|
<div class="modal-body">
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Label" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Name" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="hidden" name="host_id" value="{{ compute.id }}">
|
<input type="hidden" name="host_id" value="{{ compute.id }}">
|
||||||
<input type="text" name="name" class="form-control" value="{{ compute.name }}" maxlength="20" required pattern="[a-zA-Z0-9\.\-_]+">
|
<input type="text" name="name" class="form-control" value="{{ compute.name }}" maxlength="20" required pattern="[a-zA-Z0-9\.\-_]+">
|
||||||
|
@ -90,7 +90,7 @@
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Details" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Details" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="text" name="details" class="form-control" placeholder="Details" value="{{ compute.details }}">
|
<input type="text" name="details" class="form-control" placeholder="{% trans "Details" %}" value="{{ compute.details }}">
|
||||||
</div>
|
</div>
|
||||||
</div></div>
|
</div></div>
|
||||||
<div class="modal-footer">
|
<div class="modal-footer">
|
||||||
|
@ -111,7 +111,7 @@
|
||||||
<div class="modal-body">
|
<div class="modal-body">
|
||||||
<p class="modal-body">{% trans "Need create ssh <a href='https://github.com/retspen/webvirtmgr/wiki/Setup-SSH-Authorization'>authorization key</a>. If you have another SSH port on your server, you can add IP:PORT like '192.168.1.1:2222'." %}</p>
|
<p class="modal-body">{% trans "Need create ssh <a href='https://github.com/retspen/webvirtmgr/wiki/Setup-SSH-Authorization'>authorization key</a>. If you have another SSH port on your server, you can add IP:PORT like '192.168.1.1:2222'." %}</p>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Label" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Name" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="hidden" name="host_id" value="{{ compute.id }}">
|
<input type="hidden" name="host_id" value="{{ compute.id }}">
|
||||||
<input type="text" name="name" class="form-control" value="{{ compute.name }}" maxlength="20" required pattern="[a-z0-9\.\-_]+">
|
<input type="text" name="name" class="form-control" value="{{ compute.name }}" maxlength="20" required pattern="[a-z0-9\.\-_]+">
|
||||||
|
@ -133,7 +133,7 @@
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Details" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Details" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="text" name="details" class="form-control" placeholder="Details" value="{{ compute.details }}">
|
<input type="text" name="details" class="form-control" placeholder="{% trans "Details" %}" value="{{ compute.details }}">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -154,7 +154,7 @@
|
||||||
<form method="post" role="form" aria-label="Edit tls host form">{% csrf_token %}
|
<form method="post" role="form" aria-label="Edit tls host form">{% csrf_token %}
|
||||||
<div class="modal-body">
|
<div class="modal-body">
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Label" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Name" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="hidden" name="host_id" value="{{ compute.id }}">
|
<input type="hidden" name="host_id" value="{{ compute.id }}">
|
||||||
<input type="text" name="name" class="form-control" value="{{ compute.name }}" maxlength="20" required pattern="[a-z0-9\.\-_]+">
|
<input type="text" name="name" class="form-control" value="{{ compute.name }}" maxlength="20" required pattern="[a-z0-9\.\-_]+">
|
||||||
|
@ -181,7 +181,7 @@
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Details" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Details" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="text" name="details" class="form-control" placeholder="Details" value="{{ compute.details }}">
|
<input type="text" name="details" class="form-control" placeholder="{% trans "Details" %}" value="{{ compute.details }}">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -202,7 +202,7 @@
|
||||||
<form method="post" role="form" aria-label="Edit/delete host form">{% csrf_token %}
|
<form method="post" role="form" aria-label="Edit/delete host form">{% csrf_token %}
|
||||||
<div class="modal-body">
|
<div class="modal-body">
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Label" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Name" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="hidden" name="host_id" value="{{ compute.id }}">
|
<input type="hidden" name="host_id" value="{{ compute.id }}">
|
||||||
<input type="text" name="name" class="form-control" value="{{ compute.name }}" maxlength="20" required pattern="[a-z0-9\.\-_]+">
|
<input type="text" name="name" class="form-control" value="{{ compute.name }}" maxlength="20" required pattern="[a-z0-9\.\-_]+">
|
||||||
|
@ -211,7 +211,7 @@
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Details" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Details" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="text" name="details" class="form-control" placeholder="Details" value="{{ compute.details }}">
|
<input type="text" name="details" class="form-control" placeholder="{% trans 'Details' %}" value="{{ compute.details }}">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -21,4 +21,4 @@ def validate_hostname(value):
|
||||||
def validate_name(value):
|
def validate_name(value):
|
||||||
have_symbol = wrong_name.match('[^a-zA-Z0-9._-]+')
|
have_symbol = wrong_name.match('[^a-zA-Z0-9._-]+')
|
||||||
if have_symbol:
|
if have_symbol:
|
||||||
raise ValidationError(_('The host name must not contain any special characters'))
|
raise ValidationError(_('The hostname must not contain any special characters'))
|
||||||
|
|
|
@ -111,7 +111,7 @@
|
||||||
var path = spice_query_var('path', 'websockify');
|
var path = spice_query_var('path', 'websockify');
|
||||||
|
|
||||||
if ((!host) || (!port)) {
|
if ((!host) || (!port)) {
|
||||||
console.log("must specify host and port in URL");
|
console.log(_("must specify host and port in URL"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -222,4 +222,3 @@
|
||||||
<!-- If DUMPXXX is turned on, dumped images will go here -->
|
<!-- If DUMPXXX is turned on, dumped images will go here -->
|
||||||
</div>
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
|
|
@ -45,7 +45,7 @@ def console(request):
|
||||||
if console_type == 'vnc' or console_type == 'spice':
|
if console_type == 'vnc' or console_type == 'spice':
|
||||||
response = render(request, console_page, locals())
|
response = render(request, console_page, locals())
|
||||||
else:
|
else:
|
||||||
console_error = "Console type: %s no support" % console_type
|
console_error = f"Console type: {console_type} no support"
|
||||||
response = render(request, 'console-vnc-lite.html', locals())
|
response = render(request, 'console-vnc-lite.html', locals())
|
||||||
|
|
||||||
response.set_cookie('token', token)
|
response.set_cookie('token', token)
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
from django.db.models import Model, CharField, IntegerField
|
from django.db.models import Model, CharField, IntegerField
|
||||||
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
class Flavor(Model):
|
class Flavor(Model):
|
||||||
label = CharField(max_length=12)
|
label = CharField(_('label'), max_length=12)
|
||||||
memory = IntegerField()
|
memory = IntegerField(_('memory'))
|
||||||
vcpu = IntegerField()
|
vcpu = IntegerField(_('vcpu'))
|
||||||
disk = IntegerField()
|
disk = IntegerField(_('disk'))
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Name" %}</label>
|
<label class="col-sm-3 col-form-label">{% trans "Name" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="text" name="label" class="form-control" placeholder="Micro" maxlength="20"
|
<input type="text" name="label" class="form-control" placeholder="{% trans "Micro" %}" maxlength="20"
|
||||||
required pattern="[a-zA-Z0-9]+">
|
required pattern="[a-zA-Z0-9]+">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -1,13 +1,15 @@
|
||||||
{% extends "base.html" %}
|
{% extends "base.html" %}
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
{% load staticfiles %}
|
{% load staticfiles %}
|
||||||
{% block title %}{% trans "Create new instanc" %} - {% trans "Select Type" %}{% endblock %}
|
{% block title %}{% trans "Create new instance" %} - {% trans "Select Type" %}{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<!-- Page Heading -->
|
<!-- Page Heading -->
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
<h3 class="page-header">{% trans "New instance on" %} {{ compute.name }}</h3>
|
<h3 class="page-header">
|
||||||
|
{% blocktrans with host=compute.name %}New instance on {{ host }} {% endblocktrans %}
|
||||||
|
</h3>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<!-- /.row -->
|
<!-- /.row -->
|
||||||
|
|
|
@ -10,7 +10,7 @@
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
<h3 class="page-header">
|
<h3 class="page-header">
|
||||||
{% trans "New instance on" %} {{ compute.name }}
|
{% blocktrans with host=compute.name %}New instance on {{ host }} {% endblocktrans %}</h3>
|
||||||
</h3>
|
</h3>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -104,8 +104,8 @@
|
||||||
<div class="modal-body">
|
<div class="modal-body">
|
||||||
<form method="post" role="form" aria-label="Create instance from flavor form">{% csrf_token %}
|
<form method="post" role="form" aria-label="Create instance from flavor form">{% csrf_token %}
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Name" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Name" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<input type="text" class="form-control" name="name"
|
<input type="text" class="form-control" name="name"
|
||||||
placeholder="{% trans "Name" %}" maxlength="64" required
|
placeholder="{% trans "Name" %}" maxlength="64" required
|
||||||
pattern="[a-zA-Z0-9\.\-_]+">
|
pattern="[a-zA-Z0-9\.\-_]+">
|
||||||
|
@ -116,9 +116,9 @@
|
||||||
</div>
|
</div>
|
||||||
{% if firmwares %}
|
{% if firmwares %}
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Firmware" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Firmware" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<select class="custom-select" id="select_firmware" name="firmware">
|
<select class="form-control" id="select_firmware" name="firmware">
|
||||||
{% for frm in firmwares %}
|
{% for frm in firmwares %}
|
||||||
<option value="{{ frm }}" {% if frm in default_firmware %}selected{% endif %}>{{ frm }}</option>
|
<option value="{{ frm }}" {% if frm in default_firmware %}selected{% endif %}>{{ frm }}</option>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
@ -128,9 +128,9 @@
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if dom_caps.cpu_modes %}
|
{% if dom_caps.cpu_modes %}
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "VCPU Config" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "VCPU Config" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<select id="vcpu_mode" name="vcpu_mode" class="custom-select">
|
<select class="form-control" id="vcpu_mode" name="vcpu_mode">
|
||||||
<option value=""> {% trans 'no-mode' %}</option>
|
<option value=""> {% trans 'no-mode' %}</option>
|
||||||
{% for mode in dom_caps.cpu_modes %}
|
{% for mode in dom_caps.cpu_modes %}
|
||||||
{% if mode == 'custom' %}
|
{% if mode == 'custom' %}
|
||||||
|
@ -150,10 +150,10 @@
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Storage" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Storage" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<input type="hidden" name="cache_mode" value="default">
|
<input type="hidden" name="cache_mode" value="default">
|
||||||
<select name="storage" class="custom-select">
|
<select class="form-control" name="storage">
|
||||||
{% if storages %}
|
{% if storages %}
|
||||||
{% for storage in storages %}
|
{% for storage in storages %}
|
||||||
<option value="{{ storage }}">{{ storage }}</option>
|
<option value="{{ storage }}">{{ storage }}</option>
|
||||||
|
@ -165,9 +165,9 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Network" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Network" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<select name="networks" class="custom-select">
|
<select class="form-control" name="networks">
|
||||||
{% for network in networks %}
|
{% for network in networks %}
|
||||||
<option value="{{ network }}">{{ network }}</option>
|
<option value="{{ network }}">{{ network }}</option>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
@ -175,8 +175,8 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Advanced" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Advanced" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<button class="btn btn-block btn-outline-light" type="button" data-toggle="collapse" data-target="#flavAddCollapse" aria-expanded="false" aria-controls="collapseExample">
|
<button class="btn btn-block btn-outline-light" type="button" data-toggle="collapse" data-target="#flavAddCollapse" aria-expanded="false" aria-controls="collapseExample">
|
||||||
<span class="fa fa-caret-down"></span>
|
<span class="fa fa-caret-down"></span>
|
||||||
</button>
|
</button>
|
||||||
|
@ -184,9 +184,9 @@
|
||||||
</div>
|
</div>
|
||||||
<div class="collapse" id="flavAddCollapse">
|
<div class="collapse" id="flavAddCollapse">
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "NWFilter" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "NWFilter" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<select name="nwfilter" class="custom-select">
|
<select class="form-control" name="nwfilter">
|
||||||
<option value="">{% trans "None" %}</option>
|
<option value="">{% trans "None" %}</option>
|
||||||
{% for nwfilter in nwfilters %}
|
{% for nwfilter in nwfilters %}
|
||||||
<option value="{{ nwfilter }}">{{ nwfilter }}</option>
|
<option value="{{ nwfilter }}">{{ nwfilter }}</option>
|
||||||
|
@ -195,9 +195,9 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "HDD cache mode" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "HDD cache mode" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<select id="cache_mode" name="cache_mode" class="custom-select">
|
<select class="form-control" id="cache_mode" name="cache_mode">
|
||||||
{% for mode, name in cache_modes %}
|
{% for mode, name in cache_modes %}
|
||||||
<option value="{{ mode }}" {% if mode == default_cache %}selected {% endif %}>
|
<option value="{{ mode }}" {% if mode == default_cache %}selected {% endif %}>
|
||||||
{% trans name %}</option>
|
{% trans name %}</option>
|
||||||
|
@ -206,16 +206,16 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "MAC" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "MAC" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<input type="text" class="form-control" name="mac" maxlength="17" value="{{ mac_auto }}" required pattern="[a-zA-Z0-9:]+">
|
<input type="text" class="form-control" name="mac" maxlength="17" value="{{ mac_auto }}" required pattern="[a-zA-Z0-9:]+">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% if dom_caps.graphics_support == 'yes' %}
|
{% if dom_caps.graphics_support == 'yes' %}
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Graphics" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Graphics" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<select name="graphics" class="custom-select">
|
<select class="form-control" name="graphics">
|
||||||
{% for graphics in dom_caps.graphics_types %}
|
{% for graphics in dom_caps.graphics_types %}
|
||||||
<option value="{{ graphics }}" {% if default_graphics == graphics %}selected{% endif %}>{{ graphics }}</option>
|
<option value="{{ graphics }}" {% if default_graphics == graphics %}selected{% endif %}>{{ graphics }}</option>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
@ -224,9 +224,9 @@
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Video" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Video" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<select name="video" class="custom-select">
|
<select class="form-control" name="video">
|
||||||
{% if not videos %}
|
{% if not videos %}
|
||||||
<option value="vga">vga</option>
|
<option value="vga">vga</option>
|
||||||
<option value="cirrus">cirrus</option>
|
<option value="cirrus">cirrus</option>
|
||||||
|
@ -238,9 +238,9 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Console Access" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Console Access" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<select name="listener_addr" class="custom-select">
|
<select class="custom-select" name="listener_addr">
|
||||||
{% for addr, label in listener_addr %}
|
{% for addr, label in listener_addr %}
|
||||||
<option value="{{ addr }}" {% if addr == "0.0.0.0" %} selected {% endif %}>{{ label }}</option>
|
<option value="{{ addr }}" {% if addr == "0.0.0.0" %} selected {% endif %}>{{ label }}</option>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
@ -248,21 +248,21 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Console Password" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Console Password" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<input type="password" class="form-control" name="console_pass" placeholder="{% trans "Console Password" %}" maxlength="64">
|
<input class="form-control" type="password" name="console_pass" placeholder="{% trans "Console Password" %}" maxlength="64">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Guest Agent" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Guest Agent" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<input type="checkbox" name="qemu_ga" value="true" checked>
|
<input type="checkbox" name="qemu_ga" value="true" checked>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% if virtio_support %}
|
{% if virtio_support %}
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "VirtIO" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "VirtIO" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-7">
|
||||||
<input type="checkbox" name="virtio" value="{{ virtio_support }}" checked>
|
<input type="checkbox" name="virtio" value="{{ virtio_support }}" checked>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -414,7 +414,7 @@
|
||||||
<div class="form-group row meta-prealloc">
|
<div class="form-group row meta-prealloc">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Disk Metadata" %}</label>
|
<label class="col-sm-3 col-form-label">{% trans "Disk Metadata" %}</label>
|
||||||
<div class="col-sm-7">
|
<div class="col-sm-7">
|
||||||
<input type="checkbox" name="meta_prealloc" title="{% trans "Metadata Preallocation" %}" value="true">
|
<input type="checkbox" name="meta_prealloc" title="{% trans "Metadata preallocation" %}" value="true">
|
||||||
</div>
|
</div>
|
||||||
<label class="col-lg-1 col-form-label">{% trans "Image" %}</label>
|
<label class="col-lg-1 col-form-label">{% trans "Image" %}</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -34,7 +34,7 @@ def os_metadata_json(request, version):
|
||||||
response = {'uuid': OS_UUID, 'hostname': hostname}
|
response = {'uuid': OS_UUID, 'hostname': hostname}
|
||||||
return HttpResponse(json.dumps(response))
|
return HttpResponse(json.dumps(response))
|
||||||
else:
|
else:
|
||||||
err = 'Invalid version: {}'.format(version)
|
err = f"Invalid version: {version}"
|
||||||
raise Http404(err)
|
raise Http404(err)
|
||||||
|
|
||||||
|
|
||||||
|
@ -59,7 +59,7 @@ def os_userdata(request, version):
|
||||||
|
|
||||||
return render(request, 'user_data', locals())
|
return render(request, 'user_data', locals())
|
||||||
else:
|
else:
|
||||||
err = 'Invalid version: {}'.format(version)
|
err = f"Invalid version: {version}"
|
||||||
raise Http404(err)
|
raise Http404(err)
|
||||||
|
|
||||||
|
|
||||||
|
@ -108,5 +108,5 @@ def get_vdi_url(request, compute_id, vname):
|
||||||
response = url
|
response = url
|
||||||
return HttpResponse(response)
|
return HttpResponse(response)
|
||||||
except libvirtError as lib_err:
|
except libvirtError as lib_err:
|
||||||
err = f"Error getting vdi url for {vname}"
|
err = f"Error getting VDI URL for {vname}"
|
||||||
raise Http404(err)
|
raise Http404(err)
|
||||||
|
|
|
@ -7,10 +7,10 @@ from computes.models import Compute
|
||||||
|
|
||||||
class Instance(Model):
|
class Instance(Model):
|
||||||
compute = ForeignKey(Compute, on_delete=CASCADE)
|
compute = ForeignKey(Compute, on_delete=CASCADE)
|
||||||
name = CharField(max_length=120)
|
name = CharField(_('name'), max_length=120)
|
||||||
uuid = CharField(max_length=36)
|
uuid = CharField(_('uuid'), max_length=36)
|
||||||
is_template = BooleanField(default=False)
|
is_template = BooleanField(_('is template'), default=False)
|
||||||
created = DateField(auto_now_add=True)
|
created = DateField(_('created'), auto_now_add=True)
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return self.name
|
return self.name
|
||||||
|
|
|
@ -14,7 +14,7 @@
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if all_host_vms or all_user_vms %}
|
{% if all_host_vms or all_user_vms %}
|
||||||
<div class="float-right search">
|
<div class="float-right search">
|
||||||
<input id="filter" class="form-control" type="text" placeholder="Search">
|
<input id="filter" class="form-control" type="text" placeholder="{% trans 'Search' %}">
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<h2 class="page-header">{% trans "Instances" %}</h2>
|
<h2 class="page-header">{% trans "Instances" %}</h2>
|
||||||
|
|
|
@ -229,7 +229,7 @@
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if status == 5 %}
|
{% if status == 5 %}
|
||||||
<div role="tabpanel" class="tab-pane tab-pane-bordered active" id="boot">
|
<div role="tabpanel" class="tab-pane tab-pane-bordered active" id="boot">
|
||||||
<p>{% trans "Click on Boot button to start this instance." %}</p>
|
<p>{% trans "Click on Power On button to start this instance." %}</p>
|
||||||
<form action="" method="post" role="form" aria-label="Start instance form">{% csrf_token %}
|
<form action="" method="post" role="form" aria-label="Start instance form">{% csrf_token %}
|
||||||
{% if instance.is_template %}
|
{% if instance.is_template %}
|
||||||
<p>{% trans "Template instance cannot be started." %}</p>
|
<p>{% trans "Template instance cannot be started." %}</p>
|
||||||
|
@ -447,7 +447,7 @@
|
||||||
<form method="post" role="form" aria-label="Resize instance memory form">{% csrf_token %}
|
<form method="post" role="form" aria-label="Resize instance memory form">{% csrf_token %}
|
||||||
<p class="font-weight-bold">{% trans "Total host memory" %}: {{ memory_host|filesizeformat }}</p>
|
<p class="font-weight-bold">{% trans "Total host memory" %}: {{ memory_host|filesizeformat }}</p>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Current allocation" %} ({% trans "MB" %})</label>
|
<label class="col-sm-4 col-form-label">{% trans "Current Allocation" %} ({% trans "MB" %})</label>
|
||||||
<div class="col-sm-4 js-custom__container">
|
<div class="col-sm-4 js-custom__container">
|
||||||
<select name="cur_memory" class="custom-select js-custom__toggle">
|
<select name="cur_memory" class="custom-select js-custom__toggle">
|
||||||
{% for mem in memory_range %}
|
{% for mem in memory_range %}
|
||||||
|
@ -460,7 +460,7 @@
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">
|
<label class="col-sm-4 col-form-label">
|
||||||
{% trans "Maximum allocation" %} ({% trans "MB" %})
|
{% trans "Maximum Allocation" %} ({% trans "MB" %})
|
||||||
</label>
|
</label>
|
||||||
<div class="col-sm-4 js-custom__container">
|
<div class="col-sm-4 js-custom__container">
|
||||||
<select name="memory" class="form-control js-custom__toggle">
|
<select name="memory" class="form-control js-custom__toggle">
|
||||||
|
@ -487,7 +487,7 @@
|
||||||
<p class="font-weight-bold">{% trans "Disk allocation (GB)" %}:</p>
|
<p class="font-weight-bold">{% trans "Disk allocation (GB)" %}:</p>
|
||||||
{% for disk in disks %}
|
{% for disk in disks %}
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Current allocation" %} ({{ disk.dev }})</label>
|
<label class="col-sm-4 col-form-label">{% trans "Current Allocation" %} ({{ disk.dev }})</label>
|
||||||
<div class="col-sm-4 js-custom__container">
|
<div class="col-sm-4 js-custom__container">
|
||||||
<input type="text" name="disk_size_{{ disk.dev }}" class="form-control" value="{{ disk.size|filesizeformat }}" />
|
<input type="text" name="disk_size_{{ disk.dev }}" class="form-control" value="{{ disk.size|filesizeformat }}" />
|
||||||
</div>
|
</div>
|
||||||
|
@ -761,7 +761,7 @@
|
||||||
data-toggle="popover"
|
data-toggle="popover"
|
||||||
data-html="true"
|
data-html="true"
|
||||||
data-content="<strong>{% trans 'Bus' %}:</strong> {{ cd.bus }} <br/>
|
data-content="<strong>{% trans 'Bus' %}:</strong> {{ cd.bus }} <br/>
|
||||||
<strong>{% trans 'Dev' %}:</strong> {{ cd.dev }}">
|
<strong>{% trans 'Device' %}:</strong> {{ cd.dev }}">
|
||||||
{% trans "CD-ROM" %} {{ forloop.counter }}
|
{% trans "CD-ROM" %} {{ forloop.counter }}
|
||||||
</a>
|
</a>
|
||||||
{% if not cd.image %}
|
{% if not cd.image %}
|
||||||
|
@ -1036,55 +1036,57 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{% if qos %}
|
{% if qos %}
|
||||||
<div class="col-10 col-sm-10">
|
<div class="row mt-3">
|
||||||
<p><strong>{% trans "QoS Configuration" %}</strong></p>
|
<div class="col-sm-10">
|
||||||
</div>
|
<p><strong>{% trans "QoS Configuration" %}</strong></p>
|
||||||
<div class="col-12 col-sm-12">
|
</div>
|
||||||
<table class="table table-hover">
|
<div class="col-12 col-sm-12">
|
||||||
<thead>
|
<table class="table table-hover">
|
||||||
<tr>
|
<thead>
|
||||||
<th scope="col">{% trans "MAC" %}/{% trans "Direction" %}</th>
|
<tr class="d-flex">
|
||||||
<th scope="col">{% trans "Average" %}</th>
|
<th scope="col" class="col-2">{% trans "MAC" %}/{% trans "Direction" %}</th>
|
||||||
<th scope="col">{% trans "Peak" %}</th>
|
<th scope="col" class="col-2">{% trans "Average" %}</th>
|
||||||
<th scope="col">{% trans "Burst" %}</th>
|
<th scope="col" class="col-3">{% trans "Peak" %}</th>
|
||||||
<th scope="col">{% trans "Actions" %}</th>
|
<th scope="col" class="col-3">{% trans "Burst" %}</th>
|
||||||
</tr>
|
<th scope="col" class="col-2">{% trans "Actions" %}</th>
|
||||||
</thead>
|
</tr>
|
||||||
<tbody>
|
</thead>
|
||||||
{% for q, attrs in qos.items %}
|
<tbody>
|
||||||
{% for att in attrs %}
|
{% for q, attrs in qos.items %}
|
||||||
<form method="post" role="form" aria-label="Instance Qos configuration form">{% csrf_token %}
|
{% for att in attrs %}
|
||||||
<tr>
|
<form method="post" role="form" aria-label="Instance QoS configuration form">{% csrf_token %}
|
||||||
<td><label class="col-form-label">{{ q }} {{ att.direction | capfirst }}</label></td>
|
<tr class="d-flex">
|
||||||
<td><input id="qos_average" class="form-control" name="qos_average"
|
<td class="col-2"><label class="col-form-label">{{ q }} {{ att.direction | capfirst }}</label></td>
|
||||||
value="{{ att.average|default:'' }}"/>
|
<td class="col-2"><input id="qos_average" class="form-control" name="qos_average"
|
||||||
</td>
|
value="{{ att.average|default:'' }}"/>
|
||||||
<td><input id="qos_peak" class="form-control" name="qos_peak"
|
</td>
|
||||||
value="{{ att.peak|default:'' }}"/>
|
<td class="col-3"><input id="qos_peak" class="form-control" name="qos_peak"
|
||||||
</td>
|
value="{{ att.peak|default:'' }}"/>
|
||||||
<td><input id="qos_burst" class="form-control" name="qos_burst"
|
</td>
|
||||||
value="{{ att.burst|default:'' }}"/>
|
<td class="col-3"><input id="qos_burst" class="form-control" name="qos_burst"
|
||||||
</td>
|
value="{{ att.burst|default:'' }}"/>
|
||||||
<td class="col-sm-2">
|
</td>
|
||||||
<input name="qos_direction" value="{{ att.direction }}" hidden/>
|
<td class="col-sm-2">
|
||||||
<input name="net-mac" value="{{ q }}" hidden/>
|
<input name="qos_direction" value="{{ att.direction }}" hidden/>
|
||||||
<button type="submit" class="btn btn-sm btn-primary"
|
<input name="net-mac" value="{{ q }}" hidden/>
|
||||||
name="set_qos" data-toggle="modal"
|
<button type="submit" class="btn btn-sm btn-primary"
|
||||||
title="{% trans "Edit QoS" %}" onclick="return confirm('{% trans "Are you sure?" %}')">
|
name="set_qos" data-toggle="modal"
|
||||||
<i class="fa fa-save"></i>
|
title="{% trans "Edit QoS" %}" onclick="return confirm('{% trans "Are you sure?" %}')">
|
||||||
</button>
|
<i class="fa fa-save"></i>
|
||||||
<button type="submit" class="btn btn-sm btn-danger"
|
</button>
|
||||||
name="unset_qos"
|
<button type="submit" class="btn btn-sm btn-danger"
|
||||||
title="{% trans "Delete QoS" %}" onclick="return confirm('{% trans "Are you sure?" %}')">
|
name="unset_qos"
|
||||||
<i class="fa fa-trash"></i>
|
title="{% trans "Delete QoS" %}" onclick="return confirm('{% trans "Are you sure?" %}')">
|
||||||
</button>
|
<i class="fa fa-trash"></i>
|
||||||
</td>
|
</button>
|
||||||
</tr>
|
</td>
|
||||||
</form>
|
</tr>
|
||||||
{% endfor %}
|
</form>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
{% endfor %}
|
||||||
</table>
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<div class="clearfix"></div>
|
<div class="clearfix"></div>
|
||||||
|
@ -1177,7 +1179,7 @@
|
||||||
<div class="clearfix"></div>
|
<div class="clearfix"></div>
|
||||||
</div>
|
</div>
|
||||||
<div role="tabpanel" class="tab-pane tab-pane-bordered" id="xmledit">
|
<div role="tabpanel" class="tab-pane tab-pane-bordered" id="xmledit">
|
||||||
<p>{% trans "If you need to edit xml please Power Off the instance" %}</p>
|
<p>{% trans "If you need to edit XML please Power Off the instance" %}</p>
|
||||||
<form method="post" role="form" aria-label="Edit instance XML form">{% csrf_token %}
|
<form method="post" role="form" aria-label="Edit instance XML form">{% csrf_token %}
|
||||||
<div class="col-sm-12" id="xmlheight">
|
<div class="col-sm-12" id="xmlheight">
|
||||||
<textarea id="editor">{{ inst_xml }}</textarea>
|
<textarea id="editor">{{ inst_xml }}</textarea>
|
||||||
|
@ -1620,7 +1622,7 @@
|
||||||
{% for disk in disks %}
|
{% for disk in disks %}
|
||||||
<div class="mb-1 card border-warning">
|
<div class="mb-1 card border-warning">
|
||||||
<div class="card-header">
|
<div class="card-header">
|
||||||
<h3 class="card-title"><i class="fa fa-long-arrow-right"></i> {% trans "Disk I/O device" %}: {{ disk.dev }}</h3>
|
<h5 class="card-title"><i class="fa fa-long-arrow-right"></i> {% trans "Disk I/O device" %}: {{ disk.dev }}</h5>
|
||||||
</div>
|
</div>
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<div class="flot-chart">
|
<div class="flot-chart">
|
||||||
|
@ -1682,7 +1684,7 @@
|
||||||
<div class="ml-3 form-row">
|
<div class="ml-3 form-row">
|
||||||
<div class="custom-control custom-switch">
|
<div class="custom-control custom-switch">
|
||||||
<input class="custom-control-input" type="checkbox" name="delete_nvram" value="true" id="delete_nvram" checked>
|
<input class="custom-control-input" type="checkbox" name="delete_nvram" value="true" id="delete_nvram" checked>
|
||||||
<label class="custom-control-label font-weight-bold" for="delete_nvram" {% trans "Remove Instance's NVRAM" %}</label>
|
<label class="custom-control-label font-weight-bold" for="delete_nvram">{% trans "Remove Instance's NVRAM" %}</label>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if all_host_vms or all_user_vms %}
|
{% if all_host_vms or all_user_vms %}
|
||||||
<div class="float-right search">
|
<div class="float-right search">
|
||||||
<input id="filter" class="form-control" type="text" placeholder="Search">
|
<input id="filter" class="form-control" type="text" placeholder="{% trans 'Search' %}">
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<h2 class="page-header">{{ compute.name }}</h2>
|
<h2 class="page-header">{{ compute.name }}</h2>
|
||||||
|
|
|
@ -257,7 +257,7 @@ def instance(request, compute_id, vname):
|
||||||
if autostart:
|
if autostart:
|
||||||
conn_new.set_autostart(1)
|
conn_new.set_autostart(1)
|
||||||
conn_new.close()
|
conn_new.close()
|
||||||
msg = _("Migrate to %s" % new_compute.hostname)
|
msg = _(f"Migrate to {new_compute.hostname}")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
@ -337,12 +337,12 @@ def instance(request, compute_id, vname):
|
||||||
if instance.uuid != uuid:
|
if instance.uuid != uuid:
|
||||||
instance.uuid = uuid
|
instance.uuid = uuid
|
||||||
instance.save()
|
instance.save()
|
||||||
msg = _("Fixing uuid %s" % uuid)
|
msg = _(f"Fixing UUID {uuid}")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
except Instance.DoesNotExist:
|
except Instance.DoesNotExist:
|
||||||
instance = Instance(compute_id=compute_id, name=vname, uuid=uuid)
|
instance = Instance(compute_id=compute_id, name=vname, uuid=uuid)
|
||||||
instance.save()
|
instance.save()
|
||||||
msg = _("Instance.DoesNotExist: Creating new instance")
|
msg = _("Instance does not exist: Creating new instance")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
|
|
||||||
userinstances = UserInstance.objects.filter(instance=instance).order_by('user__username')
|
userinstances = UserInstance.objects.filter(instance=instance).order_by('user__username')
|
||||||
|
@ -456,7 +456,7 @@ def instance(request, compute_id, vname):
|
||||||
s.send(json.dumps(data))
|
s.send(json.dumps(data))
|
||||||
result = json.loads(s.recv(1024))
|
result = json.loads(s.recv(1024))
|
||||||
s.close()
|
s.close()
|
||||||
msg = _("Installed new ssh public key %s" % publickey.keyname)
|
msg = _(f"Installed new SSH public key {publickey.keyname}")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
|
|
||||||
if result['return'] == 'success':
|
if result['return'] == 'success':
|
||||||
|
@ -565,10 +565,10 @@ def instance(request, compute_id, vname):
|
||||||
driver_type = conn_create.get_volume_type(name)
|
driver_type = conn_create.get_volume_type(name)
|
||||||
path = conn_create.get_target_path()
|
path = conn_create.get_target_path()
|
||||||
target_dev = get_new_disk_dev(media, disks, bus)
|
target_dev = get_new_disk_dev(media, disks, bus)
|
||||||
source = path + "/" + name
|
source = f"{path}/{name}"
|
||||||
|
|
||||||
conn.attach_disk(target_dev, source, target_bus=bus, driver_type=driver_type, cache_mode=cache)
|
conn.attach_disk(target_dev, source, target_bus=bus, driver_type=driver_type, cache_mode=cache)
|
||||||
msg = _('Attach Existing disk: ' + target_dev)
|
msg = _(f"Attach Existing disk: {target_dev}")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
||||||
|
|
||||||
|
@ -600,11 +600,11 @@ def instance(request, compute_id, vname):
|
||||||
cache, io, discard, zeroes)
|
cache, io, discard, zeroes)
|
||||||
|
|
||||||
if not conn.get_status() == 5:
|
if not conn.get_status() == 5:
|
||||||
messages.success(request, _("Disk changes changes are applied. " +
|
messages.success(request, _("Volume changes are applied. " +
|
||||||
"But it will be activated after shutdown"))
|
"But it will be activated after shutdown"))
|
||||||
else:
|
else:
|
||||||
messages.success(request, _("Disk is changed successfully."))
|
messages.success(request, _("Volume is changed successfully."))
|
||||||
msg = _('Edit disk: ' + target_dev)
|
msg = _(f"Edit disk: {target_dev}")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
|
|
||||||
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
||||||
|
@ -620,12 +620,12 @@ def instance(request, compute_id, vname):
|
||||||
path = request.POST.get('path', '')
|
path = request.POST.get('path', '')
|
||||||
name = request.POST.get('name', '')
|
name = request.POST.get('name', '')
|
||||||
|
|
||||||
msg = _('Delete disk: ' + dev)
|
msg = _(f"Delete disk: {dev}")
|
||||||
conn.detach_disk(dev)
|
conn.detach_disk(dev)
|
||||||
try:
|
try:
|
||||||
conn_delete.del_volume(name)
|
conn_delete.del_volume(name)
|
||||||
except libvirtError as err:
|
except libvirtError as err:
|
||||||
msg = _('The disk: ' + dev + ' is detached but not deleted. ' + err)
|
msg = _(f"The disk: {dev} is detached but not deleted. Error: {err}")
|
||||||
messages.warning(request, msg)
|
messages.warning(request, msg)
|
||||||
|
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
|
@ -635,7 +635,7 @@ def instance(request, compute_id, vname):
|
||||||
dev = request.POST.get('detach_vol', '')
|
dev = request.POST.get('detach_vol', '')
|
||||||
path = request.POST.get('path', '')
|
path = request.POST.get('path', '')
|
||||||
conn.detach_disk(dev)
|
conn.detach_disk(dev)
|
||||||
msg = _('Detach disk: ' + dev)
|
msg = _(f"Detach disk: {dev}")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
||||||
|
|
||||||
|
@ -643,14 +643,14 @@ def instance(request, compute_id, vname):
|
||||||
bus = request.POST.get('bus', 'ide' if machine == 'pc' else 'sata')
|
bus = request.POST.get('bus', 'ide' if machine == 'pc' else 'sata')
|
||||||
target = get_new_disk_dev(media, disks, bus)
|
target = get_new_disk_dev(media, disks, bus)
|
||||||
conn.attach_disk(target, "", disk_device='cdrom', cache_mode='none', target_bus=bus, readonly=True)
|
conn.attach_disk(target, "", disk_device='cdrom', cache_mode='none', target_bus=bus, readonly=True)
|
||||||
msg = _('Add CD-ROM: ' + target)
|
msg = _(f"Add CD-ROM: {target}")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
||||||
|
|
||||||
if 'detach_cdrom' in request.POST and allow_admin_or_not_template:
|
if 'detach_cdrom' in request.POST and allow_admin_or_not_template:
|
||||||
dev = request.POST.get('detach_cdrom', '')
|
dev = request.POST.get('detach_cdrom', '')
|
||||||
conn.detach_disk(dev)
|
conn.detach_disk(dev)
|
||||||
msg = _('Detach CD-ROM: ' + dev)
|
msg = _(f'Detach CD-ROM: {dev}')
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
||||||
|
|
||||||
|
@ -658,7 +658,7 @@ def instance(request, compute_id, vname):
|
||||||
image = request.POST.get('path', '')
|
image = request.POST.get('path', '')
|
||||||
dev = request.POST.get('umount_iso', '')
|
dev = request.POST.get('umount_iso', '')
|
||||||
conn.umount_iso(dev, image)
|
conn.umount_iso(dev, image)
|
||||||
msg = _("Mount media: " + dev)
|
msg = _(f"Mount media: {dev}")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
||||||
|
|
||||||
|
@ -666,21 +666,21 @@ def instance(request, compute_id, vname):
|
||||||
image = request.POST.get('media', '')
|
image = request.POST.get('media', '')
|
||||||
dev = request.POST.get('mount_iso', '')
|
dev = request.POST.get('mount_iso', '')
|
||||||
conn.mount_iso(dev, image)
|
conn.mount_iso(dev, image)
|
||||||
msg = _("Umount media: " + dev)
|
msg = _(f"Umount media: {dev}")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
return HttpResponseRedirect(request.get_full_path() + '#disks')
|
||||||
|
|
||||||
if 'snapshot' in request.POST and allow_admin_or_not_template:
|
if 'snapshot' in request.POST and allow_admin_or_not_template:
|
||||||
name = request.POST.get('name', '')
|
name = request.POST.get('name', '')
|
||||||
conn.create_snapshot(name)
|
conn.create_snapshot(name)
|
||||||
msg = _("New snapshot :" + name)
|
msg = _(f"New snapshot : {name}")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
return HttpResponseRedirect(request.get_full_path() + '#managesnapshot')
|
return HttpResponseRedirect(request.get_full_path() + '#managesnapshot')
|
||||||
|
|
||||||
if 'delete_snapshot' in request.POST and allow_admin_or_not_template:
|
if 'delete_snapshot' in request.POST and allow_admin_or_not_template:
|
||||||
snap_name = request.POST.get('name', '')
|
snap_name = request.POST.get('name', '')
|
||||||
conn.snapshot_delete(snap_name)
|
conn.snapshot_delete(snap_name)
|
||||||
msg = _("Delete snapshot :" + snap_name)
|
msg = _(f"Delete snapshot : {snap_name}")
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
return HttpResponseRedirect(request.get_full_path() + '#managesnapshot')
|
return HttpResponseRedirect(request.get_full_path() + '#managesnapshot')
|
||||||
|
|
||||||
|
@ -713,14 +713,14 @@ def instance(request, compute_id, vname):
|
||||||
conn.set_vcpu(id, 1)
|
conn.set_vcpu(id, 1)
|
||||||
else:
|
else:
|
||||||
conn.set_vcpu(id, 0)
|
conn.set_vcpu(id, 0)
|
||||||
msg = _("vCPU {} is enabled={}".format(id, enabled))
|
msg = _(f"VCPU {id} is enabled={enabled}")
|
||||||
messages.success(request, msg)
|
messages.success(request, msg)
|
||||||
addlogmsg(request.user.username, instance.name, msg)
|
addlogmsg(request.user.username, instance.name, msg)
|
||||||
return HttpResponseRedirect(request.get_full_path() + '#resize')
|
return HttpResponseRedirect(request.get_full_path() + '#resize')
|
||||||
|
|
||||||
if 'set_vcpu_hotplug' in request.POST:
|
if 'set_vcpu_hotplug' in request.POST:
|
||||||
status = request.POST.get('vcpu_hotplug', '')
|
status = request.POST.get('vcpu_hotplug', '')
|
||||||
msg = _("VCPU Hot-plug is enabled={}".format(status))
|
msg = _(f"VCPU Hot-plug is enabled={status}")
|
||||||
try:
|
try:
|
||||||
conn.set_vcpu_hotplug(eval(status))
|
conn.set_vcpu_hotplug(eval(status))
|
||||||
except libvirtError as lib_err:
|
except libvirtError as lib_err:
|
||||||
|
@ -943,8 +943,8 @@ def instance(request, compute_id, vname):
|
||||||
messages.success(request, _(f"{qos_dir.capitalize()} QoS is deleted"))
|
messages.success(request, _(f"{qos_dir.capitalize()} QoS is deleted"))
|
||||||
else:
|
else:
|
||||||
messages.success(request,
|
messages.success(request,
|
||||||
f"{qos_dir.capitalize()} QoS is deleted. Network XML is changed. " +
|
_(f"{qos_dir.capitalize()} QoS is deleted. Network XML is changed. ") +
|
||||||
"Stop and start network to activate new config.")
|
_("Stop and start network to activate new config."))
|
||||||
return HttpResponseRedirect(request.get_full_path() + '#network')
|
return HttpResponseRedirect(request.get_full_path() + '#network')
|
||||||
|
|
||||||
if 'add_owner' in request.POST:
|
if 'add_owner' in request.POST:
|
||||||
|
@ -956,7 +956,7 @@ def instance(request, compute_id, vname):
|
||||||
check_inst = UserInstance.objects.filter(instance=instance)
|
check_inst = UserInstance.objects.filter(instance=instance)
|
||||||
|
|
||||||
if check_inst:
|
if check_inst:
|
||||||
msg = _("One owner is allowed and owner already added")
|
msg = _("Only one owner is allowed and the one already added")
|
||||||
error_messages.append(msg)
|
error_messages.append(msg)
|
||||||
else:
|
else:
|
||||||
add_user_inst = UserInstance(instance=instance, user_id=user_id)
|
add_user_inst = UserInstance(instance=instance, user_id=user_id)
|
||||||
|
@ -993,7 +993,7 @@ def instance(request, compute_id, vname):
|
||||||
clone_data['name'] = auto_vname
|
clone_data['name'] = auto_vname
|
||||||
clone_data['clone-net-mac-0'] = _get_dhcp_mac_address(auto_vname)
|
clone_data['clone-net-mac-0'] = _get_dhcp_mac_address(auto_vname)
|
||||||
for disk in disks:
|
for disk in disks:
|
||||||
disk_dev = "disk-{}".format(disk['dev'])
|
disk_dev = f"disk-{disk['dev']}"
|
||||||
disk_name = get_clone_disk_name(disk, vname, auto_vname)
|
disk_name = get_clone_disk_name(disk, vname, auto_vname)
|
||||||
clone_data[disk_dev] = disk_name
|
clone_data[disk_dev] = disk_name
|
||||||
|
|
||||||
|
@ -1024,7 +1024,7 @@ def instance(request, compute_id, vname):
|
||||||
user_instance = UserInstance(instance_id=new_instance.id, user_id=request.user.id, is_delete=True)
|
user_instance = UserInstance(instance_id=new_instance.id, user_id=request.user.id, is_delete=True)
|
||||||
user_instance.save()
|
user_instance.save()
|
||||||
|
|
||||||
msg = _("Clone of '%s'" % instance.name)
|
msg = _(f"Clone of '{instance.name}'")
|
||||||
addlogmsg(request.user.username, new_instance.name, msg)
|
addlogmsg(request.user.username, new_instance.name, msg)
|
||||||
|
|
||||||
if appsettings.get(key="CLONE_INSTANCE_AUTO_MIGRATE").value == 'True':
|
if appsettings.get(key="CLONE_INSTANCE_AUTO_MIGRATE").value == 'True':
|
||||||
|
@ -1157,7 +1157,7 @@ def get_host_instances(request, comp):
|
||||||
|
|
||||||
conn.close()
|
conn.close()
|
||||||
else:
|
else:
|
||||||
raise libvirtError("Problem occurred with host: {} - {}".format(comp.name, status))
|
raise libvirtError(_(f"Problem occurred with host: {comp.name} - {status}"))
|
||||||
return all_host_vms
|
return all_host_vms
|
||||||
|
|
||||||
|
|
||||||
|
@ -1343,7 +1343,7 @@ def guess_clone_name(request):
|
||||||
with open(dhcp_file, 'r') as f:
|
with open(dhcp_file, 'r') as f:
|
||||||
for line in f:
|
for line in f:
|
||||||
line = line.strip()
|
line = line.strip()
|
||||||
if "host %s" % prefix in line:
|
if f"host {prefix}" in line:
|
||||||
fqdn = line.split(' ')[1]
|
fqdn = line.split(' ')[1]
|
||||||
hostname = fqdn.split('.')[0]
|
hostname = fqdn.split('.')[0]
|
||||||
if hostname.startswith(prefix) and hostname not in instance_names:
|
if hostname.startswith(prefix) and hostname not in instance_names:
|
||||||
|
@ -1364,12 +1364,12 @@ def get_clone_disk_name(disk, prefix, clone_name=''):
|
||||||
return None
|
return None
|
||||||
if disk['image'].startswith(prefix) and clone_name:
|
if disk['image'].startswith(prefix) and clone_name:
|
||||||
suffix = disk['image'][len(prefix):]
|
suffix = disk['image'][len(prefix):]
|
||||||
image = "{}{}".format(clone_name, suffix)
|
image = f"{clone_name}{suffix}"
|
||||||
elif "." in disk['image'] and len(disk['image'].rsplit(".", 1)[1]) <= 7:
|
elif "." in disk['image'] and len(disk['image'].rsplit(".", 1)[1]) <= 7:
|
||||||
name, suffix = disk['image'].rsplit(".", 1)
|
name, suffix = disk['image'].rsplit(".", 1)
|
||||||
image = "{}-clone.{}".format(name, suffix)
|
image = f"{name}-clone.{suffix}"
|
||||||
else:
|
else:
|
||||||
image = "{}-clone".format(disk['image'])
|
image = f"{disk['image']}-clone"
|
||||||
return image
|
return image
|
||||||
|
|
||||||
|
|
||||||
|
@ -1441,7 +1441,7 @@ def delete_instance(instance, delete_disk=False):
|
||||||
|
|
||||||
del_userinstance = UserInstance.objects.filter(instance=instance)
|
del_userinstance = UserInstance.objects.filter(instance=instance)
|
||||||
if del_userinstance:
|
if del_userinstance:
|
||||||
print("Deleting UserInstances")
|
print("Deleting user instances")
|
||||||
print(del_userinstance)
|
print(del_userinstance)
|
||||||
del_userinstance.delete()
|
del_userinstance.delete()
|
||||||
|
|
||||||
|
@ -1459,8 +1459,8 @@ def delete_instance(instance, delete_disk=False):
|
||||||
conn.delete()
|
conn.delete()
|
||||||
instance.delete()
|
instance.delete()
|
||||||
|
|
||||||
print("Instance {} on compute {} successfully deleted".format(instance_name, compute.hostname))
|
print(f"Instance {instance_name} on compute {compute.hostname} successfully deleted")
|
||||||
|
|
||||||
except libvirtError as lib_err:
|
except libvirtError as lib_err:
|
||||||
print("Error removing instance {} on compute {}".format(instance_name, compute.hostname))
|
print(f"Error removing instance {instance_name} on compute {compute.hostname}")
|
||||||
raise lib_err
|
raise lib_err
|
||||||
|
|
|
@ -22,36 +22,36 @@ class AddInterface(forms.Form):
|
||||||
ipv4_addr = self.cleaned_data['ipv4_addr']
|
ipv4_addr = self.cleaned_data['ipv4_addr']
|
||||||
have_symbol = re.match('^[0-9./]+$', ipv4_addr)
|
have_symbol = re.match('^[0-9./]+$', ipv4_addr)
|
||||||
if not have_symbol:
|
if not have_symbol:
|
||||||
raise forms.ValidationError(_('The ipv4 must not contain any special characters'))
|
raise forms.ValidationError(_('The IPv4 address must not contain any special characters'))
|
||||||
elif len(ipv4_addr) > 20:
|
elif len(ipv4_addr) > 20:
|
||||||
raise forms.ValidationError(_('The ipv4 must not exceed 20 characters'))
|
raise forms.ValidationError(_('The IPv4 address must not exceed 20 characters'))
|
||||||
return ipv4_addr
|
return ipv4_addr
|
||||||
|
|
||||||
def clean_ipv4_gw(self):
|
def clean_ipv4_gw(self):
|
||||||
ipv4_gw = self.cleaned_data['ipv4_gw']
|
ipv4_gw = self.cleaned_data['ipv4_gw']
|
||||||
have_symbol = re.match('^[0-9.]+$', ipv4_gw)
|
have_symbol = re.match('^[0-9.]+$', ipv4_gw)
|
||||||
if not have_symbol:
|
if not have_symbol:
|
||||||
raise forms.ValidationError(_('The ipv4 gateway must not contain any special characters'))
|
raise forms.ValidationError(_('The IPv4 gateway must not contain any special characters'))
|
||||||
elif len(ipv4_gw) > 20:
|
elif len(ipv4_gw) > 20:
|
||||||
raise forms.ValidationError(_('The ipv4 gateway must not exceed 20 characters'))
|
raise forms.ValidationError(_('The IPv4 gateway must not exceed 20 characters'))
|
||||||
return ipv4_gw
|
return ipv4_gw
|
||||||
|
|
||||||
def clean_ipv6_addr(self):
|
def clean_ipv6_addr(self):
|
||||||
ipv6_addr = self.cleaned_data['ipv6_addr']
|
ipv6_addr = self.cleaned_data['ipv6_addr']
|
||||||
have_symbol = re.match('^[0-9a-f./:]+$', ipv6_addr)
|
have_symbol = re.match('^[0-9a-f./:]+$', ipv6_addr)
|
||||||
if not have_symbol:
|
if not have_symbol:
|
||||||
raise forms.ValidationError(_('The ipv6 must not contain any special characters'))
|
raise forms.ValidationError(_('The IPv6 address must not contain any special characters'))
|
||||||
elif len(ipv6_addr) > 100:
|
elif len(ipv6_addr) > 100:
|
||||||
raise forms.ValidationError(_('The ipv6 must not exceed 100 characters'))
|
raise forms.ValidationError(_('The IPv6 address must not exceed 100 characters'))
|
||||||
return ipv6_addr
|
return ipv6_addr
|
||||||
|
|
||||||
def clean_ipv6_gw(self):
|
def clean_ipv6_gw(self):
|
||||||
ipv6_gw = self.cleaned_data['ipv6_gw']
|
ipv6_gw = self.cleaned_data['ipv6_gw']
|
||||||
have_symbol = re.match('^[0-9.]+$', ipv6_gw)
|
have_symbol = re.match('^[0-9.]+$', ipv6_gw)
|
||||||
if not have_symbol:
|
if not have_symbol:
|
||||||
raise forms.ValidationError(_('The ipv6 gateway must not contain any special characters'))
|
raise forms.ValidationError(_('The IPv6 gateway must not contain any special characters'))
|
||||||
elif len(ipv6_gw) > 100:
|
elif len(ipv6_gw) > 100:
|
||||||
raise forms.ValidationError(_('The ipv6 gateway must not exceed 100 characters'))
|
raise forms.ValidationError(_('The IPv6 gateway must not exceed 100 characters'))
|
||||||
return ipv6_gw
|
return ipv6_gw
|
||||||
|
|
||||||
def clean_name(self):
|
def clean_name(self):
|
||||||
|
|
BIN
locale/de/LC_MESSAGES/django.mo
Normal file
BIN
locale/de/LC_MESSAGES/django.mo
Normal file
Binary file not shown.
File diff suppressed because it is too large
Load diff
BIN
locale/en/LC_MESSAGES/django.mo
Normal file
BIN
locale/en/LC_MESSAGES/django.mo
Normal file
Binary file not shown.
4836
locale/en/LC_MESSAGES/django.po
Normal file
4836
locale/en/LC_MESSAGES/django.po
Normal file
File diff suppressed because it is too large
Load diff
BIN
locale/es/LC_MESSAGES/django.mo
Normal file
BIN
locale/es/LC_MESSAGES/django.mo
Normal file
Binary file not shown.
File diff suppressed because it is too large
Load diff
BIN
locale/fr/LC_MESSAGES/django.mo
Normal file
BIN
locale/fr/LC_MESSAGES/django.mo
Normal file
Binary file not shown.
File diff suppressed because it is too large
Load diff
BIN
locale/nl/LC_MESSAGES/django.mo
Normal file
BIN
locale/nl/LC_MESSAGES/django.mo
Normal file
Binary file not shown.
File diff suppressed because it is too large
Load diff
BIN
locale/ru/LC_MESSAGES/django.mo
Normal file
BIN
locale/ru/LC_MESSAGES/django.mo
Normal file
Binary file not shown.
File diff suppressed because it is too large
Load diff
BIN
locale/tr/LC_MESSAGES/django.mo
Normal file
BIN
locale/tr/LC_MESSAGES/django.mo
Normal file
Binary file not shown.
File diff suppressed because it is too large
Load diff
BIN
locale/uk/LC_MESSAGES/django.mo
Normal file
BIN
locale/uk/LC_MESSAGES/django.mo
Normal file
Binary file not shown.
File diff suppressed because it is too large
Load diff
|
@ -1,11 +1,11 @@
|
||||||
from django.db.models import Model, CharField, DateTimeField
|
from django.db.models import Model, CharField, DateTimeField
|
||||||
|
from django.utils.translation import ugettext_lazy as _
|
||||||
|
|
||||||
class Logs(Model):
|
class Logs(Model):
|
||||||
user = CharField(max_length=50)
|
user = CharField(_('user'), max_length=50)
|
||||||
instance = CharField(max_length=50)
|
instance = CharField(_('instance'), max_length=50)
|
||||||
message = CharField(max_length=255)
|
message = CharField(_('message'), max_length=255)
|
||||||
date = DateTimeField(auto_now=True)
|
date = DateTimeField(_('date'), auto_now=True)
|
||||||
|
|
||||||
def __unicode__(self):
|
def __unicode__(self):
|
||||||
return self.instance
|
return self.instance
|
||||||
|
|
|
@ -30,18 +30,18 @@ class AddNetPool(forms.Form):
|
||||||
subnet = self.cleaned_data['subnet']
|
subnet = self.cleaned_data['subnet']
|
||||||
have_symbol = re.match('^[0-9./]+$', subnet if subnet else ".")
|
have_symbol = re.match('^[0-9./]+$', subnet if subnet else ".")
|
||||||
if not have_symbol:
|
if not have_symbol:
|
||||||
raise forms.ValidationError(_('The pool subnet must not contain any special characters'))
|
raise forms.ValidationError(_('The IPv4 subnet must not contain any special characters'))
|
||||||
elif len(subnet) > 20:
|
elif len(subnet) > 20:
|
||||||
raise forms.ValidationError(_('The pool subnet must not exceed 20 characters'))
|
raise forms.ValidationError(_('The IPv4 subnet must not exceed 20 characters'))
|
||||||
return subnet
|
return subnet
|
||||||
|
|
||||||
def clean_subnet6(self):
|
def clean_subnet6(self):
|
||||||
subnet = self.cleaned_data['subnet6']
|
subnet = self.cleaned_data['subnet6']
|
||||||
have_symbol = re.match('^[0-9a-fA-F:/]+$', subnet if subnet else ":")
|
have_symbol = re.match('^[0-9a-fA-F:/]+$', subnet if subnet else ":")
|
||||||
if not have_symbol:
|
if not have_symbol:
|
||||||
raise forms.ValidationError(_('The pool subnet must not contain any special characters'))
|
raise forms.ValidationError(_('The IPv6 subnet must not contain any special characters'))
|
||||||
elif len(subnet) > 42:
|
elif len(subnet) > 42:
|
||||||
raise forms.ValidationError(_('The pool subnet must not exceed 42 characters'))
|
raise forms.ValidationError(_('The IPv6 subnet must not exceed 42 characters'))
|
||||||
return subnet
|
return subnet
|
||||||
|
|
||||||
def clean_bridge_name(self):
|
def clean_bridge_name(self):
|
||||||
|
|
|
@ -26,21 +26,21 @@
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Average" %}:</label>
|
<label class="col-sm-4 col-form-label">{% trans "Average" %}:</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input class="form-control" name="qos_average" placeholder="kilobytes"
|
<input class="form-control" name="qos_average" placeholder="{% trans "kilobytes" %}"
|
||||||
required pattern="[0-9]+"/>
|
required pattern="[0-9]+"/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Peak" %}:</label>
|
<label class="col-sm-4 col-form-label">{% trans "Peak" %}:</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input class="form-control" name="qos_peak" placeholder="kilobytes"
|
<input class="form-control" name="qos_peak" placeholder="{% trans "kilobytes" %}"
|
||||||
pattern="[0-9]+"/>
|
pattern="[0-9]+"/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Burst" %}:</label>
|
<label class="col-sm-4 col-form-label">{% trans "Burst" %}:</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input class="form-control" name="qos_burst" placeholder="kilobytes"
|
<input class="form-control" name="qos_burst" placeholder="{% trans "kilobytes" %}"
|
||||||
pattern="[0-9]+"/>
|
pattern="[0-9]+"/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -17,7 +17,7 @@
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Name" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Name" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="text" class="form-control" name="name" placeholder="default" required pattern="[a-zA-Z0-9_]+">
|
<input type="text" class="form-control" name="name" placeholder="{% trans "default" %}" required pattern="[a-zA-Z0-9_]+">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
|
@ -44,7 +44,7 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row bridge_name_form_group_dhcp">
|
<div class="form-group row bridge_name_form_group_dhcp">
|
||||||
<label class="col-sm-4 col-form-label">{% trans "Fixed Address" %}</label>
|
<label class="col-sm-4 col-form-label">{% trans "Fixed Addresses" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="checkbox" name="fixed" value="true">
|
<input type="checkbox" name="fixed" value="true">
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -149,7 +149,7 @@
|
||||||
{% include 'modify_ipv4_fixed_address.html' %}
|
{% include 'modify_ipv4_fixed_address.html' %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
<h5 class="page-header">{% trans "IPv4 Fixed Address" %}</h5>
|
<h5 class="page-header">{% trans "IPv4 Fixed Addresses" %}</h5>
|
||||||
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
|
@ -261,7 +261,7 @@
|
||||||
{% if state %}
|
{% if state %}
|
||||||
{% include 'modify_ipv6_fixed_address.html' %}
|
{% include 'modify_ipv6_fixed_address.html' %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
<h5 class="page-header">{% trans "IPv6 Fixed Address" %}</h5>
|
<h5 class="page-header">{% trans "IPv6 Fixed Addresses" %}</h5>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% if ipv6_fixed_address %}
|
{% if ipv6_fixed_address %}
|
||||||
|
|
|
@ -8,7 +8,7 @@
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
{% include 'create_nwfilter_block.html' %}
|
{% include 'create_nwfilter_block.html' %}
|
||||||
<div class="float-right search">
|
<div class="float-right search">
|
||||||
<input id="filter" class="form-control" type="text" placeholder="Search">
|
<input id="filter" class="form-control" type="text" placeholder="{% trans 'Search' %}">
|
||||||
</div>
|
</div>
|
||||||
<h2 class="page-header">{{ compute.name }} - {% trans "NWFilters" %}</h2>
|
<h2 class="page-header">{{ compute.name }} - {% trans "NWFilters" %}</h2>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -43,7 +43,7 @@ def nwfilters(request, compute_id):
|
||||||
error_msg = _("A network filter with this name already exists")
|
error_msg = _("A network filter with this name already exists")
|
||||||
raise Exception(error_msg)
|
raise Exception(error_msg)
|
||||||
if uuid == nwf.UUIDString():
|
if uuid == nwf.UUIDString():
|
||||||
error_msg = _("A network filter with this uuid already exists")
|
error_msg = _("A network filter with this UUID already exists")
|
||||||
raise Exception(error_msg)
|
raise Exception(error_msg)
|
||||||
else:
|
else:
|
||||||
try:
|
try:
|
||||||
|
|
|
@ -45,9 +45,9 @@ class AddStgPool(forms.Form):
|
||||||
have_symbol = re.match('^[a-zA-Z0-9\/]+$', source)
|
have_symbol = re.match('^[a-zA-Z0-9\/]+$', source)
|
||||||
if storage_type == 'logical' or storage_type == 'netfs':
|
if storage_type == 'logical' or storage_type == 'netfs':
|
||||||
if not source:
|
if not source:
|
||||||
raise forms.ValidationError(_('No device has been entered'))
|
raise forms.ValidationError(_('No device or path has been entered'))
|
||||||
if not have_symbol:
|
if not have_symbol:
|
||||||
raise forms.ValidationError(_('The source must not contain any special characters'))
|
raise forms.ValidationError(_('The disk source must not contain any special characters'))
|
||||||
return source
|
return source
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -41,7 +41,7 @@
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Name" %}</label>
|
<label class="col-sm-3 col-form-label">{% trans "Name" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="text" class="form-control" name="name" placeholder="default" maxlength="20" required pattern="[a-zA-Z0-9\.\-_]+">
|
<input type="text" class="form-control" name="name" placeholder="{% trans "default" %}" maxlength="20" required pattern="[a-zA-Z0-9\.\-_]+">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
|
@ -70,7 +70,7 @@
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label class="col-sm-3 col-form-label">{% trans "Name" %}</label>
|
<label class="col-sm-3 col-form-label">{% trans "Name" %}</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<input type="text" class="form-control" name="name" placeholder="default" maxlength="20" required pattern="[a-zA-Z0-9\.\-_]+">
|
<input type="text" class="form-control" name="name" placeholder="{% trans "default" %}" maxlength="20" required pattern="[a-zA-Z0-9\.\-_]+">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
|
|
|
@ -86,7 +86,7 @@
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<h5 class="page-header mr-auto">{% trans "Volumes" %}</h5>
|
<h5 class="page-header mr-auto">{% trans "Volumes" %}</h5>
|
||||||
<div class="col-3 ">
|
<div class="col-3 ">
|
||||||
<input id="filter" class="form-control" type="text" placeholder="Search">
|
<input id="filter" class="form-control" type="text" placeholder="{% trans 'Search' %}">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
|
@ -162,7 +162,7 @@ def storage(request, compute_id, pool):
|
||||||
try:
|
try:
|
||||||
vol = conn.get_volume(volname)
|
vol = conn.get_volume(volname)
|
||||||
vol.delete(0)
|
vol.delete(0)
|
||||||
messages.success(request, _('Volume: {} is deleted.'.format(volname)))
|
messages.success(request, _(f"Volume: {volname} is deleted."))
|
||||||
return HttpResponseRedirect(request.get_full_path())
|
return HttpResponseRedirect(request.get_full_path())
|
||||||
except libvirtError as lib_err:
|
except libvirtError as lib_err:
|
||||||
error_messages.append(lib_err)
|
error_messages.append(lib_err)
|
||||||
|
@ -172,7 +172,7 @@ def storage(request, compute_id, pool):
|
||||||
error_messages.append(error_msg)
|
error_messages.append(error_msg)
|
||||||
else:
|
else:
|
||||||
handle_uploaded_file(path, request.FILES['file'])
|
handle_uploaded_file(path, request.FILES['file'])
|
||||||
messages.success(request, _('ISO: {} is uploaded.'.format(request.FILES['file'])))
|
messages.success(request, _(f"ISO: {request.FILES['file']} is uploaded."))
|
||||||
return HttpResponseRedirect(request.get_full_path())
|
return HttpResponseRedirect(request.get_full_path())
|
||||||
if 'cln_volume' in request.POST:
|
if 'cln_volume' in request.POST:
|
||||||
form = CloneImage(request.POST)
|
form = CloneImage(request.POST)
|
||||||
|
@ -192,7 +192,7 @@ def storage(request, compute_id, pool):
|
||||||
format = None
|
format = None
|
||||||
try:
|
try:
|
||||||
name = conn.clone_volume(data['image'], data['name'], format, meta_prealloc)
|
name = conn.clone_volume(data['image'], data['name'], format, meta_prealloc)
|
||||||
messages.success(request, _("{} image cloned as {} successfully".format(data['image'], name)))
|
messages.success(request, _(f"{data['image']} image cloned as {name} successfully"))
|
||||||
return HttpResponseRedirect(request.get_full_path())
|
return HttpResponseRedirect(request.get_full_path())
|
||||||
except libvirtError as lib_err:
|
except libvirtError as lib_err:
|
||||||
error_messages.append(lib_err)
|
error_messages.append(lib_err)
|
||||||
|
|
|
@ -38,6 +38,10 @@
|
||||||
<i class="fa fa-fw fa-user"></i> {{ request.user.username }}
|
<i class="fa fa-fw fa-user"></i> {{ request.user.username }}
|
||||||
</a>
|
</a>
|
||||||
<div class="dropdown-menu" aria-labelledby="navbarUserDropdown">
|
<div class="dropdown-menu" aria-labelledby="navbarUserDropdown">
|
||||||
|
{% get_current_language as LANGUAGE_CODE %}
|
||||||
|
<a class="dropdown-item disabled" href="#">
|
||||||
|
{% trans "Language" %}: <span class="badge badge-secondary">{{ LANGUAGE_CODE }}</span>
|
||||||
|
</a>
|
||||||
<a class="dropdown-item {% view_active request 'profile' %}" href="{% url 'profile' %}"><i class="fa fa-fw fa-pencil-square-o"></i> {% trans "Profile" %}</a>
|
<a class="dropdown-item {% view_active request 'profile' %}" href="{% url 'profile' %}"><i class="fa fa-fw fa-pencil-square-o"></i> {% trans "Profile" %}</a>
|
||||||
<div class="dropdown-divider"></div>
|
<div class="dropdown-divider"></div>
|
||||||
<a class="dropdown-item" href="{% url 'logout' %}"><i class="fa fa-fw fa-power-off"></i> {% trans "Log Out" %}</a>
|
<a class="dropdown-item" href="{% url 'logout' %}"><i class="fa fa-fw fa-power-off"></i> {% trans "Log Out" %}</a>
|
||||||
|
|
|
@ -42,6 +42,7 @@ INSTALLED_APPS = [
|
||||||
MIDDLEWARE = [
|
MIDDLEWARE = [
|
||||||
'django.middleware.security.SecurityMiddleware',
|
'django.middleware.security.SecurityMiddleware',
|
||||||
'django.contrib.sessions.middleware.SessionMiddleware',
|
'django.contrib.sessions.middleware.SessionMiddleware',
|
||||||
|
'django.middleware.locale.LocaleMiddleware',
|
||||||
'django.middleware.common.CommonMiddleware',
|
'django.middleware.common.CommonMiddleware',
|
||||||
'django.middleware.csrf.CsrfViewMiddleware',
|
'django.middleware.csrf.CsrfViewMiddleware',
|
||||||
'django.contrib.auth.middleware.AuthenticationMiddleware',
|
'django.contrib.auth.middleware.AuthenticationMiddleware',
|
||||||
|
@ -49,7 +50,6 @@ MIDDLEWARE = [
|
||||||
'django.contrib.auth.middleware.RemoteUserMiddleware',
|
'django.contrib.auth.middleware.RemoteUserMiddleware',
|
||||||
'django.contrib.messages.middleware.MessageMiddleware',
|
'django.contrib.messages.middleware.MessageMiddleware',
|
||||||
'django.middleware.clickjacking.XFrameOptionsMiddleware',
|
'django.middleware.clickjacking.XFrameOptionsMiddleware',
|
||||||
'django.middleware.locale.LocaleMiddleware',
|
|
||||||
]
|
]
|
||||||
|
|
||||||
ROOT_URLCONF = 'webvirtcloud.urls'
|
ROOT_URLCONF = 'webvirtcloud.urls'
|
||||||
|
@ -111,6 +111,10 @@ STATICFILES_DIRS = [
|
||||||
os.path.join(BASE_DIR, "static"),
|
os.path.join(BASE_DIR, "static"),
|
||||||
]
|
]
|
||||||
|
|
||||||
|
LOCALE_PATHS = [
|
||||||
|
'locale/',
|
||||||
|
]
|
||||||
|
|
||||||
LOGGING = {
|
LOGGING = {
|
||||||
"version": 1,
|
"version": 1,
|
||||||
"disable_existing_loggers": False,
|
"disable_existing_loggers": False,
|
||||||
|
|
Loading…
Reference in a new issue