merged master
This commit is contained in:
		
				commit
				
					
						30551e5ac5
					
				
			
		
					 18 changed files with 880 additions and 336 deletions
				
			
		
							
								
								
									
										169
									
								
								datacenterlight/tasks.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										169
									
								
								datacenterlight/tasks.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,169 @@ | ||||||
|  | from dynamicweb.celery import app | ||||||
|  | from celery.utils.log import get_task_logger | ||||||
|  | from django.conf import settings | ||||||
|  | from opennebula_api.models import OpenNebulaManager | ||||||
|  | from opennebula_api.serializers import VirtualMachineSerializer | ||||||
|  | from hosting.models import HostingOrder, HostingBill | ||||||
|  | from utils.forms import UserBillingAddressForm | ||||||
|  | from datetime import datetime | ||||||
|  | from membership.models import StripeCustomer | ||||||
|  | from django.core.mail import EmailMessage | ||||||
|  | from utils.models import BillingAddress | ||||||
|  | from celery.exceptions import MaxRetriesExceededError | ||||||
|  | 
 | ||||||
|  | logger = get_task_logger(__name__) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def retry_task(task, exception=None): | ||||||
|  |     """Retries the specified task using a "backing off countdown", | ||||||
|  |     meaning that the interval between retries grows exponentially | ||||||
|  |     with every retry. | ||||||
|  | 
 | ||||||
|  |     Arguments: | ||||||
|  |         task: | ||||||
|  |             The task to retry. | ||||||
|  | 
 | ||||||
|  |         exception: | ||||||
|  |             Optionally, the exception that caused the retry. | ||||||
|  |     """ | ||||||
|  | 
 | ||||||
|  |     def backoff(attempts): | ||||||
|  |         return 2 ** attempts | ||||||
|  | 
 | ||||||
|  |     kwargs = { | ||||||
|  |         'countdown': backoff(task.request.retries), | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if exception: | ||||||
|  |         kwargs['exc'] = exception | ||||||
|  | 
 | ||||||
|  |     raise task.retry(**kwargs) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @app.task(bind=True, max_retries=settings.CELERY_MAX_RETRIES) | ||||||
|  | def create_vm_task(self, vm_template_id, user, specs, template, stripe_customer_id, billing_address_data, | ||||||
|  |                    billing_address_id, | ||||||
|  |                    charge): | ||||||
|  |     vm_id = None | ||||||
|  |     try: | ||||||
|  |         final_price = specs.get('price') | ||||||
|  |         billing_address = BillingAddress.objects.filter(id=billing_address_id).first() | ||||||
|  |         customer = StripeCustomer.objects.filter(id=stripe_customer_id).first() | ||||||
|  |         # Create OpenNebulaManager | ||||||
|  |         manager = OpenNebulaManager(email=settings.OPENNEBULA_USERNAME, | ||||||
|  |                                     password=settings.OPENNEBULA_PASSWORD) | ||||||
|  | 
 | ||||||
|  |         # Create a vm using oneadmin, also specify the name | ||||||
|  |         vm_id = manager.create_vm( | ||||||
|  |             template_id=vm_template_id, | ||||||
|  |             specs=specs, | ||||||
|  |             ssh_key=settings.ONEADMIN_USER_SSH_PUBLIC_KEY, | ||||||
|  |             vm_name="{email}-{template_name}-{date}".format( | ||||||
|  |                 email=user.get('email'), | ||||||
|  |                 template_name=template.get('name'), | ||||||
|  |                 date=int(datetime.now().strftime("%s"))) | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |         if vm_id is None: | ||||||
|  |             raise Exception("Could not create VM") | ||||||
|  | 
 | ||||||
|  |         # Create a Hosting Order | ||||||
|  |         order = HostingOrder.create( | ||||||
|  |             price=final_price, | ||||||
|  |             vm_id=vm_id, | ||||||
|  |             customer=customer, | ||||||
|  |             billing_address=billing_address | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |         # Create a Hosting Bill | ||||||
|  |         HostingBill.create( | ||||||
|  |             customer=customer, billing_address=billing_address) | ||||||
|  | 
 | ||||||
|  |         # Create Billing Address for User if he does not have one | ||||||
|  |         if not customer.user.billing_addresses.count(): | ||||||
|  |             billing_address_data.update({ | ||||||
|  |                 'user': customer.user.id | ||||||
|  |             }) | ||||||
|  |             billing_address_user_form = UserBillingAddressForm( | ||||||
|  |                 billing_address_data) | ||||||
|  |             billing_address_user_form.is_valid() | ||||||
|  |             billing_address_user_form.save() | ||||||
|  | 
 | ||||||
|  |         # Associate an order with a stripe payment | ||||||
|  |         charge_object = DictDotLookup(charge) | ||||||
|  |         order.set_stripe_charge(charge_object) | ||||||
|  | 
 | ||||||
|  |         # If the Stripe payment succeeds, set order status approved | ||||||
|  |         order.set_approved() | ||||||
|  | 
 | ||||||
|  |         vm = VirtualMachineSerializer(manager.get_vm(vm_id)).data | ||||||
|  | 
 | ||||||
|  |         context = { | ||||||
|  |             'name': user.get('name'), | ||||||
|  |             'email': user.get('email'), | ||||||
|  |             'cores': specs.get('cpu'), | ||||||
|  |             'memory': specs.get('memory'), | ||||||
|  |             'storage': specs.get('disk_size'), | ||||||
|  |             'price': specs.get('price'), | ||||||
|  |             'template': template.get('name'), | ||||||
|  |             'vm.name': vm['name'], | ||||||
|  |             'vm.id': vm['vm_id'], | ||||||
|  |             'order.id': order.id | ||||||
|  |         } | ||||||
|  |         email_data = { | ||||||
|  |             'subject': settings.DCL_TEXT + " Order from %s" % context['email'], | ||||||
|  |             'from_email': settings.DCL_SUPPORT_FROM_ADDRESS, | ||||||
|  |             'to': ['info@ungleich.ch'], | ||||||
|  |             'body': "\n".join(["%s=%s" % (k, v) for (k, v) in context.items()]), | ||||||
|  |             'reply_to': [context['email']], | ||||||
|  |         } | ||||||
|  |         email = EmailMessage(**email_data) | ||||||
|  |         email.send() | ||||||
|  |     except Exception as e: | ||||||
|  |         logger.error(str(e)) | ||||||
|  |         try: | ||||||
|  |             retry_task(self) | ||||||
|  |         except MaxRetriesExceededError: | ||||||
|  |             msg_text = 'Finished {} retries for create_vm_task'.format(self.request.retries) | ||||||
|  |             logger.error(msg_text) | ||||||
|  |             # Try sending email and stop | ||||||
|  |             email_data = { | ||||||
|  |                 'subject': '{} CELERY TASK ERROR: {}'.format(settings.DCL_TEXT, msg_text), | ||||||
|  |                 'from_email': settings.DCL_SUPPORT_FROM_ADDRESS, | ||||||
|  |                 'to': ['info@ungleich.ch'], | ||||||
|  |                 'body': ',\n'.join(str(i) for i in self.request.args) | ||||||
|  |             } | ||||||
|  |             email = EmailMessage(**email_data) | ||||||
|  |             email.send() | ||||||
|  |             return | ||||||
|  | 
 | ||||||
|  |     return vm_id | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class DictDotLookup(object): | ||||||
|  |     """ | ||||||
|  |     Creates objects that behave much like a dictionaries, but allow nested | ||||||
|  |     key access using object '.' (dot) lookups. | ||||||
|  |     """ | ||||||
|  | 
 | ||||||
|  |     def __init__(self, d): | ||||||
|  |         for k in d: | ||||||
|  |             if isinstance(d[k], dict): | ||||||
|  |                 self.__dict__[k] = DictDotLookup(d[k]) | ||||||
|  |             elif isinstance(d[k], (list, tuple)): | ||||||
|  |                 l = [] | ||||||
|  |                 for v in d[k]: | ||||||
|  |                     if isinstance(v, dict): | ||||||
|  |                         l.append(DictDotLookup(v)) | ||||||
|  |                     else: | ||||||
|  |                         l.append(v) | ||||||
|  |                 self.__dict__[k] = l | ||||||
|  |             else: | ||||||
|  |                 self.__dict__[k] = d[k] | ||||||
|  | 
 | ||||||
|  |     def __getitem__(self, name): | ||||||
|  |         if name in self.__dict__: | ||||||
|  |             return self.__dict__[name] | ||||||
|  | 
 | ||||||
|  |     def __iter__(self): | ||||||
|  |         return iter(self.__dict__.keys()) | ||||||
|  | @ -4,7 +4,6 @@ from .forms import BetaAccessForm | ||||||
| from .models import BetaAccess, BetaAccessVMType, BetaAccessVM, VMTemplate | from .models import BetaAccess, BetaAccessVMType, BetaAccessVM, VMTemplate | ||||||
| from django.contrib import messages | from django.contrib import messages | ||||||
| from django.core.urlresolvers import reverse | from django.core.urlresolvers import reverse | ||||||
| from django.core.mail import EmailMessage |  | ||||||
| from utils.mailer import BaseEmail | from utils.mailer import BaseEmail | ||||||
| from django.shortcuts import render | from django.shortcuts import render | ||||||
| from django.shortcuts import redirect | from django.shortcuts import redirect | ||||||
|  | @ -13,14 +12,14 @@ from django.core.exceptions import ValidationError | ||||||
| from django.views.decorators.cache import cache_control | from django.views.decorators.cache import cache_control | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.utils.translation import ugettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
| from utils.forms import BillingAddressForm, UserBillingAddressForm | from utils.forms import BillingAddressForm | ||||||
| from utils.models import BillingAddress | from utils.models import BillingAddress | ||||||
| from hosting.models import HostingOrder, HostingBill | from hosting.models import HostingOrder | ||||||
| from utils.stripe_utils import StripeUtils | from utils.stripe_utils import StripeUtils | ||||||
| from datetime import datetime |  | ||||||
| from membership.models import CustomUser, StripeCustomer | from membership.models import CustomUser, StripeCustomer | ||||||
| from opennebula_api.models import OpenNebulaManager | from opennebula_api.models import OpenNebulaManager | ||||||
| from opennebula_api.serializers import VirtualMachineTemplateSerializer, VirtualMachineSerializer, VMTemplateSerializer | from opennebula_api.serializers import VirtualMachineTemplateSerializer, VMTemplateSerializer | ||||||
|  | from datacenterlight.tasks import create_vm_task | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class LandingProgramView(TemplateView): | class LandingProgramView(TemplateView): | ||||||
|  | @ -33,7 +32,6 @@ class SuccessView(TemplateView): | ||||||
|     def get(self, request, *args, **kwargs): |     def get(self, request, *args, **kwargs): | ||||||
|         if 'specs' not in request.session or 'user' not in request.session: |         if 'specs' not in request.session or 'user' not in request.session: | ||||||
|             return HttpResponseRedirect(reverse('datacenterlight:index')) |             return HttpResponseRedirect(reverse('datacenterlight:index')) | ||||||
| 
 |  | ||||||
|         elif 'token' not in request.session: |         elif 'token' not in request.session: | ||||||
|             return HttpResponseRedirect(reverse('datacenterlight:payment')) |             return HttpResponseRedirect(reverse('datacenterlight:payment')) | ||||||
|         elif 'order_confirmation' not in request.session: |         elif 'order_confirmation' not in request.session: | ||||||
|  | @ -79,8 +77,7 @@ class PricingView(TemplateView): | ||||||
|         manager = OpenNebulaManager() |         manager = OpenNebulaManager() | ||||||
|         template = manager.get_template(template_id) |         template = manager.get_template(template_id) | ||||||
| 
 | 
 | ||||||
|         request.session['template'] = VirtualMachineTemplateSerializer( |         request.session['template'] = VirtualMachineTemplateSerializer(template).data | ||||||
|             template).data |  | ||||||
| 
 | 
 | ||||||
|         if not request.user.is_authenticated(): |         if not request.user.is_authenticated(): | ||||||
|             request.session['next'] = reverse('hosting:payment') |             request.session['next'] = reverse('hosting:payment') | ||||||
|  | @ -132,8 +129,7 @@ class BetaAccessView(FormView): | ||||||
|         email = BaseEmail(**email_data) |         email = BaseEmail(**email_data) | ||||||
|         email.send() |         email.send() | ||||||
| 
 | 
 | ||||||
|         messages.add_message( |         messages.add_message(self.request, messages.SUCCESS, self.success_message) | ||||||
|             self.request, messages.SUCCESS, self.success_message) |  | ||||||
|         return render(self.request, 'datacenterlight/beta_success.html', {}) |         return render(self.request, 'datacenterlight/beta_success.html', {}) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -185,8 +181,7 @@ class BetaProgramView(CreateView): | ||||||
|         email = BaseEmail(**email_data) |         email = BaseEmail(**email_data) | ||||||
|         email.send() |         email.send() | ||||||
| 
 | 
 | ||||||
|         messages.add_message( |         messages.add_message(self.request, messages.SUCCESS, self.success_message) | ||||||
|             self.request, messages.SUCCESS, self.success_message) |  | ||||||
|         return HttpResponseRedirect(self.get_success_url()) |         return HttpResponseRedirect(self.get_success_url()) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -230,8 +225,7 @@ class IndexView(CreateView): | ||||||
|         storage_field = forms.IntegerField(validators=[self.validate_storage]) |         storage_field = forms.IntegerField(validators=[self.validate_storage]) | ||||||
|         price = request.POST.get('total') |         price = request.POST.get('total') | ||||||
|         template_id = int(request.POST.get('config')) |         template_id = int(request.POST.get('config')) | ||||||
|         template = VMTemplate.objects.filter( |         template = VMTemplate.objects.filter(opennebula_vm_template_id=template_id).first() | ||||||
|             opennebula_vm_template_id=template_id).first() |  | ||||||
|         template_data = VMTemplateSerializer(template).data |         template_data = VMTemplateSerializer(template).data | ||||||
| 
 | 
 | ||||||
|         name = request.POST.get('name') |         name = request.POST.get('name') | ||||||
|  | @ -243,40 +237,35 @@ class IndexView(CreateView): | ||||||
|             cores = cores_field.clean(cores) |             cores = cores_field.clean(cores) | ||||||
|         except ValidationError as err: |         except ValidationError as err: | ||||||
|             msg = '{} : {}.'.format(cores, str(err)) |             msg = '{} : {}.'.format(cores, str(err)) | ||||||
|             messages.add_message( |             messages.add_message(self.request, messages.ERROR, msg, extra_tags='cores') | ||||||
|                 self.request, messages.ERROR, msg, extra_tags='cores') |  | ||||||
|             return HttpResponseRedirect(reverse('datacenterlight:index') + "#order_form") |             return HttpResponseRedirect(reverse('datacenterlight:index') + "#order_form") | ||||||
| 
 | 
 | ||||||
|         try: |         try: | ||||||
|             memory = memory_field.clean(memory) |             memory = memory_field.clean(memory) | ||||||
|         except ValidationError as err: |         except ValidationError as err: | ||||||
|             msg = '{} : {}.'.format(memory, str(err)) |             msg = '{} : {}.'.format(memory, str(err)) | ||||||
|             messages.add_message( |             messages.add_message(self.request, messages.ERROR, msg, extra_tags='memory') | ||||||
|                 self.request, messages.ERROR, msg, extra_tags='memory') |  | ||||||
|             return HttpResponseRedirect(reverse('datacenterlight:index') + "#order_form") |             return HttpResponseRedirect(reverse('datacenterlight:index') + "#order_form") | ||||||
| 
 | 
 | ||||||
|         try: |         try: | ||||||
|             storage = storage_field.clean(storage) |             storage = storage_field.clean(storage) | ||||||
|         except ValidationError as err: |         except ValidationError as err: | ||||||
|             msg = '{} : {}.'.format(storage, str(err)) |             msg = '{} : {}.'.format(storage, str(err)) | ||||||
|             messages.add_message( |             messages.add_message(self.request, messages.ERROR, msg, extra_tags='storage') | ||||||
|                 self.request, messages.ERROR, msg, extra_tags='storage') |  | ||||||
|             return HttpResponseRedirect(reverse('datacenterlight:index') + "#order_form") |             return HttpResponseRedirect(reverse('datacenterlight:index') + "#order_form") | ||||||
| 
 | 
 | ||||||
|         try: |         try: | ||||||
|             name = name_field.clean(name) |             name = name_field.clean(name) | ||||||
|         except ValidationError as err: |         except ValidationError as err: | ||||||
|             msg = '{} {}.'.format(name, _('is not a proper name')) |             msg = '{} {}.'.format(name, _('is not a proper name')) | ||||||
|             messages.add_message( |             messages.add_message(self.request, messages.ERROR, msg, extra_tags='name') | ||||||
|                 self.request, messages.ERROR, msg, extra_tags='name') |  | ||||||
|             return HttpResponseRedirect(reverse('datacenterlight:index') + "#order_form") |             return HttpResponseRedirect(reverse('datacenterlight:index') + "#order_form") | ||||||
| 
 | 
 | ||||||
|         try: |         try: | ||||||
|             email = email_field.clean(email) |             email = email_field.clean(email) | ||||||
|         except ValidationError as err: |         except ValidationError as err: | ||||||
|             msg = '{} {}.'.format(email, _('is not a proper email')) |             msg = '{} {}.'.format(email, _('is not a proper email')) | ||||||
|             messages.add_message( |             messages.add_message(self.request, messages.ERROR, msg, extra_tags='email') | ||||||
|                 self.request, messages.ERROR, msg, extra_tags='email') |  | ||||||
|             return HttpResponseRedirect(reverse('datacenterlight:index') + "#order_form") |             return HttpResponseRedirect(reverse('datacenterlight:index') + "#order_form") | ||||||
| 
 | 
 | ||||||
|         specs = { |         specs = { | ||||||
|  | @ -341,8 +330,7 @@ class IndexView(CreateView): | ||||||
|         email = BaseEmail(**email_data) |         email = BaseEmail(**email_data) | ||||||
|         email.send() |         email.send() | ||||||
| 
 | 
 | ||||||
|         messages.add_message( |         messages.add_message(self.request, messages.SUCCESS, self.success_message) | ||||||
|             self.request, messages.SUCCESS, self.success_message) |  | ||||||
|         return super(IndexView, self).form_valid(form) |         return super(IndexView, self).form_valid(form) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -411,7 +399,6 @@ class PaymentOrderView(FormView): | ||||||
| 
 | 
 | ||||||
|             # Create Billing Address |             # Create Billing Address | ||||||
|             billing_address = form.save() |             billing_address = form.save() | ||||||
| 
 |  | ||||||
|             request.session['billing_address_data'] = billing_address_data |             request.session['billing_address_data'] = billing_address_data | ||||||
|             request.session['billing_address'] = billing_address.id |             request.session['billing_address'] = billing_address.id | ||||||
|             request.session['token'] = token |             request.session['token'] = token | ||||||
|  | @ -436,13 +423,11 @@ class OrderConfirmationView(DetailView): | ||||||
|         stripe_customer_id = request.session.get('customer') |         stripe_customer_id = request.session.get('customer') | ||||||
|         customer = StripeCustomer.objects.filter(id=stripe_customer_id).first() |         customer = StripeCustomer.objects.filter(id=stripe_customer_id).first() | ||||||
|         stripe_utils = StripeUtils() |         stripe_utils = StripeUtils() | ||||||
|         card_details = stripe_utils.get_card_details( |         card_details = stripe_utils.get_card_details(customer.stripe_id, request.session.get('token')) | ||||||
|             customer.stripe_id, request.session.get('token')) |  | ||||||
|         if not card_details.get('response_object') and not card_details.get('paid'): |         if not card_details.get('response_object') and not card_details.get('paid'): | ||||||
|             msg = card_details.get('error') |             msg = card_details.get('error') | ||||||
|             messages.add_message(self.request, messages.ERROR, msg, extra_tags='failed_payment') |             messages.add_message(self.request, messages.ERROR, msg, extra_tags='failed_payment') | ||||||
|             return HttpResponseRedirect(reverse('datacenterlight:payment') + '#payment_error') |             return HttpResponseRedirect(reverse('datacenterlight:payment') + '#payment_error') | ||||||
| 
 |  | ||||||
|         context = { |         context = { | ||||||
|             'site_url': reverse('datacenterlight:index'), |             'site_url': reverse('datacenterlight:index'), | ||||||
|             'cc_last4': card_details.get('response_object').get('last4'), |             'cc_last4': card_details.get('response_object').get('last4'), | ||||||
|  | @ -458,8 +443,6 @@ class OrderConfirmationView(DetailView): | ||||||
|         customer = StripeCustomer.objects.filter(id=stripe_customer_id).first() |         customer = StripeCustomer.objects.filter(id=stripe_customer_id).first() | ||||||
|         billing_address_data = request.session.get('billing_address_data') |         billing_address_data = request.session.get('billing_address_data') | ||||||
|         billing_address_id = request.session.get('billing_address') |         billing_address_id = request.session.get('billing_address') | ||||||
|         billing_address = BillingAddress.objects.filter( |  | ||||||
|             id=billing_address_id).first() |  | ||||||
|         vm_template_id = template.get('id', 1) |         vm_template_id = template.get('id', 1) | ||||||
|         final_price = specs.get('price') |         final_price = specs.get('price') | ||||||
| 
 | 
 | ||||||
|  | @ -475,72 +458,8 @@ class OrderConfirmationView(DetailView): | ||||||
|             return HttpResponseRedirect(reverse('datacenterlight:payment') + '#payment_error') |             return HttpResponseRedirect(reverse('datacenterlight:payment') + '#payment_error') | ||||||
| 
 | 
 | ||||||
|         charge = charge_response.get('response_object') |         charge = charge_response.get('response_object') | ||||||
| 
 |         create_vm_task.delay(vm_template_id, user, specs, template, stripe_customer_id, billing_address_data, | ||||||
|         # Create OpenNebulaManager |                              billing_address_id, | ||||||
|         manager = OpenNebulaManager(email=settings.OPENNEBULA_USERNAME, |                              charge) | ||||||
|                                     password=settings.OPENNEBULA_PASSWORD) |  | ||||||
| 
 |  | ||||||
|         # Create a vm using oneadmin, also specify the name |  | ||||||
|         vm_id = manager.create_vm( |  | ||||||
|             template_id=vm_template_id, |  | ||||||
|             specs=specs, |  | ||||||
|             ssh_key=settings.ONEADMIN_USER_SSH_PUBLIC_KEY, |  | ||||||
|             vm_name="{email}-{template_name}-{date}".format( |  | ||||||
|                 email=user.get('email'), |  | ||||||
|                 template_name=template.get('name'), |  | ||||||
|                 date=int(datetime.now().strftime("%s"))) |  | ||||||
|         ) |  | ||||||
| 
 |  | ||||||
|         # Create a Hosting Order |  | ||||||
|         order = HostingOrder.create( |  | ||||||
|             price=final_price, |  | ||||||
|             vm_id=vm_id, |  | ||||||
|             customer=customer, |  | ||||||
|             billing_address=billing_address |  | ||||||
|         ) |  | ||||||
| 
 |  | ||||||
|         # Create a Hosting Bill |  | ||||||
|         HostingBill.create( |  | ||||||
|             customer=customer, billing_address=billing_address) |  | ||||||
| 
 |  | ||||||
|         # Create Billing Address for User if he does not have one |  | ||||||
|         if not customer.user.billing_addresses.count(): |  | ||||||
|             billing_address_data.update({ |  | ||||||
|                 'user': customer.user.id |  | ||||||
|             }) |  | ||||||
|             billing_address_user_form = UserBillingAddressForm( |  | ||||||
|                 billing_address_data) |  | ||||||
|             billing_address_user_form.is_valid() |  | ||||||
|             billing_address_user_form.save() |  | ||||||
| 
 |  | ||||||
|         # Associate an order with a stripe payment |  | ||||||
|         order.set_stripe_charge(charge) |  | ||||||
| 
 |  | ||||||
|         # If the Stripe payment was successed, set order status approved |  | ||||||
|         order.set_approved() |  | ||||||
| 
 |  | ||||||
|         vm = VirtualMachineSerializer(manager.get_vm(vm_id)).data |  | ||||||
| 
 |  | ||||||
|         context = { |  | ||||||
|             'name': user.get('name'), |  | ||||||
|             'email': user.get('email'), |  | ||||||
|             'cores': specs.get('cpu'), |  | ||||||
|             'memory': specs.get('memory'), |  | ||||||
|             'storage': specs.get('disk_size'), |  | ||||||
|             'price': specs.get('price'), |  | ||||||
|             'template': template.get('name'), |  | ||||||
|             'vm.name': vm['name'], |  | ||||||
|             'vm.id': vm['vm_id'], |  | ||||||
|             'order.id': order.id |  | ||||||
|         } |  | ||||||
|         email_data = { |  | ||||||
|             'subject': settings.DCL_TEXT + " Order from %s" % context['email'], |  | ||||||
|             'from_email': settings.DCL_SUPPORT_FROM_ADDRESS, |  | ||||||
|             'to': ['info@ungleich.ch'], |  | ||||||
|             'body': "\n".join(["%s=%s" % (k, v) for (k, v) in context.items()]), |  | ||||||
|             'reply_to': [context['email']], |  | ||||||
|         } |  | ||||||
|         email = EmailMessage(**email_data) |  | ||||||
|         email.send() |  | ||||||
|         request.session['order_confirmation'] = True |         request.session['order_confirmation'] = True | ||||||
|         return HttpResponseRedirect(reverse('datacenterlight:order_success')) |         return HttpResponseRedirect(reverse('datacenterlight:order_success')) | ||||||
|  |  | ||||||
							
								
								
									
										20
									
								
								deploy.sh
									
										
									
									
									
								
							
							
						
						
									
										20
									
								
								deploy.sh
									
										
									
									
									
								
							|  | @ -13,6 +13,7 @@ while true; do | ||||||
|   case "$1" in |   case "$1" in | ||||||
|     -h | --help ) HELP=true; shift ;; |     -h | --help ) HELP=true; shift ;; | ||||||
|     -v | --verbose ) VERBOSE=true; shift ;; |     -v | --verbose ) VERBOSE=true; shift ;; | ||||||
|  |     -D | --dbmakemigrations ) DB_MAKE_MIGRATIONS=true; shift ;; | ||||||
|     -d | --dbmigrate ) DB_MIGRATE=true; shift ;; |     -d | --dbmigrate ) DB_MIGRATE=true; shift ;; | ||||||
|     -n | --nogit ) NO_GIT=true; shift ;; |     -n | --nogit ) NO_GIT=true; shift ;; | ||||||
|     -b | --branch ) BRANCH="$2"; shift 2 ;; |     -b | --branch ) BRANCH="$2"; shift 2 ;; | ||||||
|  | @ -31,13 +32,15 @@ if [ "$HELP" == "true" ]; then | ||||||
|     echo "options are : " |     echo "options are : " | ||||||
|     echo "        -h, --help: Print this help message" |     echo "        -h, --help: Print this help message" | ||||||
|     echo "        -v, --verbose: Show verbose output to stdout. Without this a deploy.log is written to ~/app folder" |     echo "        -v, --verbose: Show verbose output to stdout. Without this a deploy.log is written to ~/app folder" | ||||||
|     echo "        -d, --dbmigrate: Do DB migrate" |     echo "        -D, --dbmakemigrations: Do DB makemigrations" | ||||||
|     echo "        -n, --nogit: Don't execute git commands. With this --branch has no effect." |     echo "        -d, --dbmigrate: Do DB migrate. To do both makemigrations and migrate, supply both switches -D and -d" | ||||||
|  |     echo "        -n, --nogit: Don't execute git commands. This is used to deploy the current code in the project repo. With this --branch has no effect." | ||||||
|     echo "        -b, --branch: The branch to pull from origin repo." |     echo "        -b, --branch: The branch to pull from origin repo." | ||||||
|     exit |     exit | ||||||
| fi | fi | ||||||
| 
 | 
 | ||||||
| echo "BRANCH="$BRANCH | echo "BRANCH="$BRANCH | ||||||
|  | echo "DB_MAKE_MIGRATIONS="$DB_MAKE_MIGRATIONS | ||||||
| echo "DB_MIGRATE="$DB_MIGRATE | echo "DB_MIGRATE="$DB_MIGRATE | ||||||
| echo "NO_GIT="$NO_GIT | echo "NO_GIT="$NO_GIT | ||||||
| echo "VERBOSE="$VERBOSE | echo "VERBOSE="$VERBOSE | ||||||
|  | @ -45,7 +48,7 @@ echo "VERBOSE="$VERBOSE | ||||||
| # The project directory exists, we pull the specified branch | # The project directory exists, we pull the specified branch | ||||||
| cd $APP_HOME_DIR | cd $APP_HOME_DIR | ||||||
| if [ -z "$NO_GIT" ]; then | if [ -z "$NO_GIT" ]; then | ||||||
|     echo 'We are executing default git commands. Please -no_git to not use this.' |     echo 'We are executing default git commands. Please add --nogit to not do this.' | ||||||
|     # Save any modified changes before git pulling |     # Save any modified changes before git pulling | ||||||
|     git stash |     git stash | ||||||
|     # Fetch all branches/tags |     # Fetch all branches/tags | ||||||
|  | @ -59,16 +62,23 @@ fi | ||||||
| source ~/pyvenv/bin/activate | source ~/pyvenv/bin/activate | ||||||
| pip install -r requirements.txt > deploy.log 2>&1 | pip install -r requirements.txt > deploy.log 2>&1 | ||||||
| echo "###" >> deploy.log | echo "###" >> deploy.log | ||||||
| if [ -z "$DB_MIGRATE" ]; then | if [ -z "$DB_MAKE_MIGRATIONS" ]; then | ||||||
|     echo 'We are not doing DB migration' |     echo 'We are not doing DB makemigrations' | ||||||
| else | else | ||||||
|  |     echo 'Doing DB makemigrations' | ||||||
|     ./manage.py makemigrations >> deploy.log 2>&1 |     ./manage.py makemigrations >> deploy.log 2>&1 | ||||||
|     echo "###" >> deploy.log |     echo "###" >> deploy.log | ||||||
|  | fi | ||||||
|  | if [ -z "$DB_MIGRATE" ]; then | ||||||
|  |     echo 'We are not doing DB migrate' | ||||||
|  | else | ||||||
|  |     echo 'Doing DB migrate' | ||||||
|     ./manage.py migrate >> deploy.log 2>&1 |     ./manage.py migrate >> deploy.log 2>&1 | ||||||
|     echo "###" >> deploy.log |     echo "###" >> deploy.log | ||||||
| fi | fi | ||||||
| printf 'yes' | ./manage.py collectstatic >> deploy.log 2>&1 | printf 'yes' | ./manage.py collectstatic >> deploy.log 2>&1 | ||||||
| echo "###" >> deploy.log | echo "###" >> deploy.log | ||||||
| django-admin compilemessages | django-admin compilemessages | ||||||
|  | sudo systemctl restart celery.service | ||||||
| sudo systemctl restart uwsgi | sudo systemctl restart uwsgi | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -0,0 +1,5 @@ | ||||||
|  | # This will make sure the app is always imported when | ||||||
|  | # Django starts so that shared_task will use this app. | ||||||
|  | from .celery import app as celery_app | ||||||
|  | 
 | ||||||
|  | __all__ = ['celery_app'] | ||||||
							
								
								
									
										21
									
								
								dynamicweb/celery.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								dynamicweb/celery.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,21 @@ | ||||||
|  | import os | ||||||
|  | from celery import Celery | ||||||
|  | 
 | ||||||
|  | # set the default Django settings module for the 'celery' program. | ||||||
|  | os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'dynamicweb.settings') | ||||||
|  | 
 | ||||||
|  | app = Celery('dynamicweb') | ||||||
|  | 
 | ||||||
|  | # Using a string here means the worker don't have to serialize | ||||||
|  | # the configuration object to child processes. | ||||||
|  | # - namespace='CELERY' means all celery-related configuration keys | ||||||
|  | #   should have a `CELERY_` prefix. | ||||||
|  | app.config_from_object('django.conf:settings', namespace='CELERY') | ||||||
|  | 
 | ||||||
|  | # Load task modules from all registered Django app configs. | ||||||
|  | app.autodiscover_tasks() | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @app.task(bind=True) | ||||||
|  | def debug_task(self): | ||||||
|  |     print('Request: {0!r}'.format(self.request)) | ||||||
|  | @ -10,6 +10,9 @@ from django.utils.translation import ugettext_lazy as _ | ||||||
| 
 | 
 | ||||||
| # dotenv | # dotenv | ||||||
| import dotenv | import dotenv | ||||||
|  | import logging | ||||||
|  | 
 | ||||||
|  | logger = logging.getLogger(__name__) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def gettext(s): | def gettext(s): | ||||||
|  | @ -25,6 +28,21 @@ def bool_env(val): | ||||||
|     return True if os.environ.get(val, False) == 'True' else False |     return True if os.environ.get(val, False) == 'True' else False | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | def int_env(val, default_value=0): | ||||||
|  |     """Replaces string based environment values with Python integers | ||||||
|  |     Return default_value if val is not set or cannot be parsed, otherwise | ||||||
|  |     returns the python integer equal to the passed val | ||||||
|  |     """ | ||||||
|  |     return_value = default_value | ||||||
|  |     try: | ||||||
|  |         return_value = int(os.environ.get(val)) | ||||||
|  |     except Exception as e: | ||||||
|  |         logger.error("Encountered exception trying to get env value for {}\nException details: {}".format( | ||||||
|  |             val, str(e))) | ||||||
|  | 
 | ||||||
|  |     return return_value | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) | BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) | ||||||
| 
 | 
 | ||||||
| PROJECT_DIR = os.path.abspath( | PROJECT_DIR = os.path.abspath( | ||||||
|  | @ -120,7 +138,8 @@ INSTALLED_APPS = ( | ||||||
|     'datacenterlight.templatetags', |     'datacenterlight.templatetags', | ||||||
|     'alplora', |     'alplora', | ||||||
|     'rest_framework', |     'rest_framework', | ||||||
|     'opennebula_api' |     'opennebula_api', | ||||||
|  |     'django_celery_results', | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| MIDDLEWARE_CLASSES = ( | MIDDLEWARE_CLASSES = ( | ||||||
|  | @ -524,6 +543,15 @@ GOOGLE_ANALYTICS_PROPERTY_IDS = { | ||||||
|     'dynamicweb-staging.ungleich.ch': 'staging' |     'dynamicweb-staging.ungleich.ch': 'staging' | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | # CELERY Settings | ||||||
|  | CELERY_BROKER_URL = env('CELERY_BROKER_URL') | ||||||
|  | CELERY_RESULT_BACKEND = env('CELERY_RESULT_BACKEND') | ||||||
|  | CELERY_ACCEPT_CONTENT = ['application/json'] | ||||||
|  | CELERY_TASK_SERIALIZER = 'json' | ||||||
|  | CELERY_RESULT_SERIALIZER = 'json' | ||||||
|  | CELERY_TIMEZONE = 'Europe/Zurich' | ||||||
|  | CELERY_MAX_RETRIES = int_env('CELERY_MAX_RETRIES', 5) | ||||||
|  | 
 | ||||||
| ENABLE_DEBUG_LOGGING = bool_env('ENABLE_DEBUG_LOGGING') | ENABLE_DEBUG_LOGGING = bool_env('ENABLE_DEBUG_LOGGING') | ||||||
| 
 | 
 | ||||||
| if ENABLE_DEBUG_LOGGING: | if ENABLE_DEBUG_LOGGING: | ||||||
|  |  | ||||||
|  | @ -392,38 +392,76 @@ msgstr "Möchtest Du den Schlüssel löschen?" | ||||||
| msgid "Show" | msgid "Show" | ||||||
| msgstr "Anzeigen" | msgstr "Anzeigen" | ||||||
| 
 | 
 | ||||||
| msgid "Public SSH Key" | #, fuzzy | ||||||
|  | #| msgid "Public SSH Key" | ||||||
|  | msgid "Public SSH key" | ||||||
| msgstr "Public SSH Key" | msgstr "Public SSH Key" | ||||||
| 
 | 
 | ||||||
| msgid "Download" | msgid "Download" | ||||||
| msgstr "" | msgstr "" | ||||||
| 
 | 
 | ||||||
| msgid "Settings" | msgid "Your Virtual Machine Detail" | ||||||
| msgstr "Einstellungen" | msgstr "Virtuelle Maschinen Detail" | ||||||
| 
 | 
 | ||||||
| msgid "Billing" | msgid "VM Settings" | ||||||
| msgstr "Abrechnungen" | msgstr "VM Einstellungen" | ||||||
| 
 | 
 | ||||||
| msgid "Ip not assigned yet" | msgid "Copied" | ||||||
| msgstr "Ip nicht zugewiesen" | msgstr "Kopiert" | ||||||
| 
 | 
 | ||||||
| msgid "Disk" | msgid "Disk" | ||||||
| msgstr "Festplatte" | msgstr "Festplatte" | ||||||
| 
 | 
 | ||||||
| msgid "Current pricing" | msgid "Billing" | ||||||
|  | msgstr "Abrechnungen" | ||||||
|  | 
 | ||||||
|  | msgid "Current Pricing" | ||||||
| msgstr "Aktueller Preis" | msgstr "Aktueller Preis" | ||||||
| 
 | 
 | ||||||
| msgid "Current status" | msgid "Month" | ||||||
| msgstr "Aktueller Status" | msgstr "Monat" | ||||||
| 
 | 
 | ||||||
| msgid "Terminate Virtual Machine" | msgid "See Invoice" | ||||||
| msgstr "Virtuelle Maschine beenden" | msgstr "Rechnung" | ||||||
|  | 
 | ||||||
|  | msgid "Your VM is" | ||||||
|  | msgstr "Deine VM ist" | ||||||
|  | 
 | ||||||
|  | msgid "Pending" | ||||||
|  | msgstr "In Vorbereitung" | ||||||
|  | 
 | ||||||
|  | msgid "Online" | ||||||
|  | msgstr "" | ||||||
|  | 
 | ||||||
|  | msgid "Failed" | ||||||
|  | msgstr "Fehlgeschlagen" | ||||||
|  | 
 | ||||||
|  | msgid "Terminate VM" | ||||||
|  | msgstr "VM Beenden" | ||||||
|  | 
 | ||||||
|  | msgid "Support / Contact" | ||||||
|  | msgstr "Support / Kontakt" | ||||||
|  | 
 | ||||||
|  | msgid "Something doesn't work?" | ||||||
|  | msgstr "Etwas funktioniert nicht?" | ||||||
|  | 
 | ||||||
|  | msgid "We are here to help you!" | ||||||
|  | msgstr "Wir sind hier, um Dir zu helfen!" | ||||||
|  | 
 | ||||||
|  | msgid "CONTACT" | ||||||
|  | msgstr "KONTACT" | ||||||
|  | 
 | ||||||
|  | msgid "BACK TO LIST" | ||||||
|  | msgstr "ZURÜCK ZUR LISTE" | ||||||
| 
 | 
 | ||||||
| msgid "Terminate your Virtual Machine" | msgid "Terminate your Virtual Machine" | ||||||
| msgstr "Ihre virtuelle Maschine beenden" | msgstr "Deine Virtuelle Maschine beenden" | ||||||
| 
 | 
 | ||||||
| msgid "Are you sure do you want to cancel your Virtual Machine " | msgid "Do you want to cancel your Virtual Machine" | ||||||
| msgstr "Sind Sie sicher, dass Sie ihre virtuelle Maschine beenden wollen " | msgstr "Bist Du sicher, dass Du Deine virtuelle Maschine beenden willst" | ||||||
|  | 
 | ||||||
|  | msgid "OK" | ||||||
|  | msgstr "" | ||||||
| 
 | 
 | ||||||
| msgid "Virtual Machines" | msgid "Virtual Machines" | ||||||
| msgstr "Virtuelle Maschinen" | msgstr "Virtuelle Maschinen" | ||||||
|  | @ -499,6 +537,15 @@ msgstr "" | ||||||
| #~ "Kreditkateninformationen wirst du auf die Bestellbestätigungsseite " | #~ "Kreditkateninformationen wirst du auf die Bestellbestätigungsseite " | ||||||
| #~ "weitergeleitet." | #~ "weitergeleitet." | ||||||
| 
 | 
 | ||||||
|  | #~ msgid "Ip not assigned yet" | ||||||
|  | #~ msgstr "Ip nicht zugewiesen" | ||||||
|  | 
 | ||||||
|  | #~ msgid "Current status" | ||||||
|  | #~ msgstr "Aktueller Status" | ||||||
|  | 
 | ||||||
|  | #~ msgid "Terminate Virtual Machine" | ||||||
|  | #~ msgstr "Virtuelle Maschine beenden" | ||||||
|  | 
 | ||||||
| #~ msgid "Ipv4" | #~ msgid "Ipv4" | ||||||
| #~ msgstr "IPv4" | #~ msgstr "IPv4" | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,13 +1,9 @@ | ||||||
| import os | import os | ||||||
| import logging | import logging | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.utils.functional import cached_property | from django.utils.functional import cached_property | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| from Crypto.PublicKey import RSA | from Crypto.PublicKey import RSA | ||||||
| 
 |  | ||||||
| from membership.models import StripeCustomer, CustomUser | from membership.models import StripeCustomer, CustomUser | ||||||
| from utils.models import BillingAddress | from utils.models import BillingAddress | ||||||
| from utils.mixins import AssignPermissionsMixin | from utils.mixins import AssignPermissionsMixin | ||||||
|  | @ -42,7 +38,6 @@ class HostingPlan(models.Model): | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class HostingOrder(AssignPermissionsMixin, models.Model): | class HostingOrder(AssignPermissionsMixin, models.Model): | ||||||
| 
 |  | ||||||
|     ORDER_APPROVED_STATUS = 'Approved' |     ORDER_APPROVED_STATUS = 'Approved' | ||||||
|     ORDER_DECLINED_STATUS = 'Declined' |     ORDER_DECLINED_STATUS = 'Declined' | ||||||
| 
 | 
 | ||||||
|  | @ -101,7 +96,7 @@ class HostingOrder(AssignPermissionsMixin, models.Model): | ||||||
| class UserHostingKey(models.Model): | class UserHostingKey(models.Model): | ||||||
|     user = models.ForeignKey(CustomUser) |     user = models.ForeignKey(CustomUser) | ||||||
|     public_key = models.TextField() |     public_key = models.TextField() | ||||||
|     private_key = models.FileField(upload_to='private_keys',  blank=True) |     private_key = models.FileField(upload_to='private_keys', blank=True) | ||||||
|     created_at = models.DateTimeField(auto_now_add=True) |     created_at = models.DateTimeField(auto_now_add=True) | ||||||
|     name = models.CharField(max_length=100) |     name = models.CharField(max_length=100) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,3 +1,6 @@ | ||||||
|  | .virtual-machine-container { | ||||||
|  |   max-width: 900px; | ||||||
|  | } | ||||||
| .virtual-machine-container .tabs-left, .virtual-machine-container .tabs-right { | .virtual-machine-container .tabs-left, .virtual-machine-container .tabs-right { | ||||||
|   border-bottom: none; |   border-bottom: none; | ||||||
|   padding-top: 2px; |   padding-top: 2px; | ||||||
|  | @ -229,6 +232,204 @@ | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /* Vm Details */ | ||||||
|  | 
 | ||||||
|  | .vm-detail-item, .vm-contact-us { | ||||||
|  |     overflow: hidden; | ||||||
|  |     border: 1px solid #ccc; | ||||||
|  |     padding: 15px; | ||||||
|  |     color: #555; | ||||||
|  |     font-weight: 300; | ||||||
|  |     margin-bottom: 15px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-detail-title { | ||||||
|  |   margin-top: 0; | ||||||
|  |   font-size: 20px; | ||||||
|  |   font-weight: 300; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-detail-title .un-icon { | ||||||
|  |   float: right; | ||||||
|  |   height: 24px; | ||||||
|  |   width: 21px; | ||||||
|  |   margin-top: 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-detail-item .vm-name { | ||||||
|  |   font-size: 16px; | ||||||
|  |   margin-bottom: 15px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-detail-item p { | ||||||
|  |   margin-bottom: 5px; | ||||||
|  |   position: relative; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-detail-ip { | ||||||
|  |   padding-bottom: 5px; | ||||||
|  |   border-bottom: 1px solid #ddd; | ||||||
|  |   margin-bottom: 10px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-detail-ip .un-icon { | ||||||
|  |   height: 14px; | ||||||
|  |   width: 14px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-detail-ip .to_copy { | ||||||
|  |   position: absolute; | ||||||
|  |   right: 0; | ||||||
|  |   top: 1px; | ||||||
|  |   padding: 0; | ||||||
|  |   line-height: 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-vmid { | ||||||
|  |   padding: 50px 0 70px; | ||||||
|  |   text-align: center; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-item-lg { | ||||||
|  |   font-size: 22px; | ||||||
|  |   margin-top: 5px; | ||||||
|  |   margin-bottom: 15px; | ||||||
|  |   letter-spacing: 0.6px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-color-online { | ||||||
|  |   color: #37B07B; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-color-pending { | ||||||
|  |   color: #e47f2f; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-detail-item .value{ | ||||||
|  |   font-weight: 400; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-detail-config .value { | ||||||
|  |   float: right; | ||||||
|  |   font-weight: 600; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-detail-contain { | ||||||
|  |   margin-top: 25px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-contact-us { | ||||||
|  |   margin: 25px 0 30px; | ||||||
|  |   /* text-align: center; */ | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @media(min-width: 768px) { | ||||||
|  |   .vm-detail-contain { | ||||||
|  |     display: flex; | ||||||
|  |     margin-left: -15px; | ||||||
|  |     margin-right: -15px; | ||||||
|  |   } | ||||||
|  |   .vm-detail-item { | ||||||
|  |     width: 33.333333%; | ||||||
|  |     margin: 0 15px; | ||||||
|  |   } | ||||||
|  |   .vm-contact-us { | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |     justify-content: space-between; | ||||||
|  |   } | ||||||
|  |   .vm-contact-us .vm-detail-title { | ||||||
|  |     margin-bottom: 0; | ||||||
|  |   } | ||||||
|  |   .vm-contact-us .un-icon { | ||||||
|  |     width: 22px; | ||||||
|  |     height: 22px; | ||||||
|  |     margin-right: 5px; | ||||||
|  |   } | ||||||
|  |   .vm-contact-us div { | ||||||
|  |     padding: 0 15px; | ||||||
|  |     position: relative; | ||||||
|  |   } | ||||||
|  |   .vm-contact-us-text { | ||||||
|  |     display: flex; | ||||||
|  |     align-items: center; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .value-sm-block { | ||||||
|  |   display: block; | ||||||
|  |   padding-top: 2px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @media(max-width: 767px) { | ||||||
|  |   .vm-contact-us div { | ||||||
|  |     margin-bottom: 30px; | ||||||
|  |   } | ||||||
|  |   .vm-contact-us div span { | ||||||
|  |     display: block; | ||||||
|  |     margin-bottom: 3px; | ||||||
|  |   } | ||||||
|  |   .dashboard-title-thin { | ||||||
|  |     font-size: 22px; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .btn-vm-invoice { | ||||||
|  |   color: #87B6EA; | ||||||
|  |   border: 2px solid #87B6EA; | ||||||
|  |   padding: 4px 18px; | ||||||
|  |   letter-spacing: 0.6px; | ||||||
|  | } | ||||||
|  | .btn-vm-invoice:hover, .btn-vm-invoice:focus { | ||||||
|  |   color : #fff; | ||||||
|  |   background: #87B6EA; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | .btn-vm-term { | ||||||
|  |   color: #aaa; | ||||||
|  |   border: 2px solid #ccc; | ||||||
|  |   background: #fff; | ||||||
|  |   padding: 4px 18px; | ||||||
|  |   letter-spacing: 0.6px; | ||||||
|  | } | ||||||
|  | .btn-vm-term:hover, .btn-vm-term:focus, .btn-vm-term:active { | ||||||
|  |   color: #eb4d5c; | ||||||
|  |   border-color: #eb4d5c; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .btn-vm-contact { | ||||||
|  |   color: #fff; | ||||||
|  |   background: #A3C0E2; | ||||||
|  |   border: 2px solid #A3C0E2; | ||||||
|  |   padding: 5px 25px; | ||||||
|  |   font-size: 12px; | ||||||
|  |   letter-spacing: 1.3px; | ||||||
|  | } | ||||||
|  | .btn-vm-contact:hover, .btn-vm-contact:focus { | ||||||
|  |   background: #fff; | ||||||
|  |   color: #a3c0e2; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .btn-vm-back { | ||||||
|  |   color: #fff; | ||||||
|  |   background: #C4CEDA; | ||||||
|  |   border: 2px solid #C4CEDA; | ||||||
|  |   padding: 5px 25px; | ||||||
|  |   font-size: 12px; | ||||||
|  |   letter-spacing: 1.3px; | ||||||
|  | } | ||||||
|  | .btn-vm-back:hover, .btn-vm-back:focus { | ||||||
|  |   color: #fff; | ||||||
|  |   background: #8da4c0; | ||||||
|  |   border-color: #8da4c0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .vm-contact-us-text { | ||||||
|  |   letter-spacing: 0.4px; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| /* New styles */ | /* New styles */ | ||||||
| .dashboard-container-head { | .dashboard-container-head { | ||||||
|   padding: 0 8px; |   padding: 0 8px; | ||||||
|  |  | ||||||
							
								
								
									
										61
									
								
								hosting/static/hosting/img/24-hours-support.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								hosting/static/hosting/img/24-hours-support.svg
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,61 @@ | ||||||
|  | <?xml version="1.0" encoding="iso-8859-1"?> | ||||||
|  | <!-- Generator: Adobe Illustrator 18.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||||
|  | <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||||
|  | <svg version="1.1" id="Capa_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||||
|  | 	 viewBox="0 0 279.525 279.525" style="enable-background:new 0 0 279.525 279.525;" xml:space="preserve"> | ||||||
|  | <g> | ||||||
|  | 	<path d="M165.066,1.544c-29.272,0-56.007,11.05-76.268,29.191c4.494,7.146,7.047,15.46,7.287,24.042l0.001,0.025l0.001,0.025 | ||||||
|  | 		c0.102,3.867,0.333,7.735,0.664,11.597c15.368-21.117,40.258-34.88,68.315-34.88c46.571,0,84.459,37.888,84.459,84.459 | ||||||
|  | 		c0,46.08-37.098,83.634-82.994,84.422c4.191,3.502,8.518,6.84,12.976,9.974l0.02,0.015l0.021,0.014 | ||||||
|  | 		c6.07,4.282,11.014,9.896,14.483,16.317c49.133-12.861,85.493-57.633,85.493-110.742C279.525,52.89,228.18,1.544,165.066,1.544z"/> | ||||||
|  | 	<path d="M162.256,234.942c-13.076-10.438-21.234-17.389-32.909-28.204c-3.435-3.182-7.633-5.164-11.944-5.164 | ||||||
|  | 		c-3.299,0-6.557,1.051-9.239,3.252c-2.768,2.33-5.536,4.66-8.305,6.989c-22.499-26.738-39.206-57.895-49.027-91.431 | ||||||
|  | 		c3.472-1.016,6.945-2.033,10.417-3.049c7.652-2.343,11.252-10.512,10.129-18.701c-2.443-17.824-3.77-26.679-5.282-43.018 | ||||||
|  | 		c-0.775-8.375-6.349-15.65-14.338-16.085c-1.246-0.121-2.491-0.181-3.726-0.181c-29.71,0-55.578,34.436-46.009,76.564 | ||||||
|  | 		c11.907,52.172,37.684,100.243,74.551,139.031c15.102,15.856,33.603,23.036,50.312,23.036c17.627,0,33.261-7.984,40.833-22.195 | ||||||
|  | 		C171.778,248.891,168.83,240.19,162.256,234.942z"/> | ||||||
|  | 	<path d="M130.645,118.121c-7.912,7.341-13.089,13.113-15.823,17.643c-1.93,3.195-3.338,6.573-4.187,10.04 | ||||||
|  | 		c-0.399,1.632-0.032,3.326,1.007,4.649c1.038,1.321,2.596,2.079,4.276,2.079h37.758c4.626,0,8.39-3.764,8.39-8.39 | ||||||
|  | 		c0-4.626-3.764-8.39-8.39-8.39h-17.051c0.139-0.164,0.282-0.328,0.428-0.493c1.114-1.254,3.842-3.874,8.107-7.785 | ||||||
|  | 		c4.473-4.105,7.493-7.179,9.232-9.398c2.621-3.336,4.571-6.593,5.794-9.679c1.247-3.145,1.88-6.498,1.88-9.967 | ||||||
|  | 		c0-6.224-2.254-11.507-6.699-15.705c-4.416-4.164-10.495-6.274-18.071-6.274c-6.884,0-12.731,1.802-17.377,5.356 | ||||||
|  | 		c-2.803,2.146-4.961,5.119-6.415,8.839c-0.982,2.513-0.728,5.388,0.68,7.689c1.408,2.302,3.852,3.837,6.537,4.105 | ||||||
|  | 		c0.299,0.03,0.597,0.045,0.891,0.045c3.779,0,7.149-2.403,8.387-5.979c0.388-1.121,0.901-2.012,1.527-2.65 | ||||||
|  | 		c1.318-1.343,3.093-1.997,5.428-1.997c2.373,0,4.146,0.618,5.418,1.889c1.269,1.269,1.886,3.12,1.886,5.66 | ||||||
|  | 		c0,2.359-0.843,4.819-2.505,7.314C140.862,108.028,138.199,111.083,130.645,118.121z"/> | ||||||
|  | 	<path d="M206.235,76.451h-6.307c-1.797,0-3.475,0.886-4.489,2.37l-29.168,42.698c-0.851,1.246-1.301,2.703-1.301,4.212v6.919 | ||||||
|  | 		c0,2.997,2.439,5.436,5.436,5.436h23.945v5.787c0,4.775,3.885,8.66,8.66,8.66c4.775,0,8.66-3.885,8.66-8.66v-5.787h0.865 | ||||||
|  | 		c4.437,0,8.047-3.61,8.047-8.047c0-4.437-3.61-8.047-8.047-8.047h-0.865V81.887C211.671,78.89,209.232,76.451,206.235,76.451z | ||||||
|  | 		 M194.352,121.992h-10.748l10.748-15.978V121.992z"/> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 3 KiB | 
							
								
								
									
										1
									
								
								hosting/static/hosting/img/billing.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								hosting/static/hosting/img/billing.svg
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1 @@ | ||||||
|  | <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 16 16" class="symbol symbol-billing" aria-labelledby="title" role="img"><title id="title">billing icon</title><g data-name="Layer 1"><path class="cls-1" d="M.37.023v15.954l2.775-1.387 2.775 1.387L8 14.59l2.775 1.387 2.081-1.387 2.775 1.387V.023zm13.873 13.709l-1.487-.744-2.081 1.387L7.9 12.989l-2.08 1.387-2.675-1.337-1.387.694V1.41h12.485z" role="presentation"/><path class="cls-1" d="M4.206 3.617h7.741v1.348H4.206zm0 2.697h7.741v1.349H4.206zm0 2.697h7.741v1.349H4.206z" role="presentation"/></g></svg> | ||||||
| After Width: | Height: | Size: 558 B | 
							
								
								
									
										45
									
								
								hosting/static/hosting/img/connected.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								hosting/static/hosting/img/connected.svg
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,45 @@ | ||||||
|  | <?xml version="1.0" encoding="iso-8859-1"?> | ||||||
|  | <!-- Generator: Adobe Illustrator 18.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||||
|  | <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||||
|  | <svg version="1.1" id="Capa_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||||
|  | 	 viewBox="0 0 278.898 278.898" style="enable-background:new 0 0 278.898 278.898;" xml:space="preserve"> | ||||||
|  | <g> | ||||||
|  | 	<path d="M269.898,175.773h-20.373V64.751c0-4.971-4.029-9-9-9h-62.702V35.377c0-4.971-4.029-9-9-9h-58.748c-4.971,0-9,4.029-9,9 | ||||||
|  | 		v20.374H38.373c-4.971,0-9,4.029-9,9v111.022H9c-4.971,0-9,4.029-9,9v58.748c0,4.971,4.029,9,9,9h58.747c4.971,0,9-4.029,9-9 | ||||||
|  | 		v-58.748c0-4.971-4.029-9-9-9H47.373V73.751h53.702v20.374c0,4.971,4.029,9,9,9h20.374v72.648h-20.374c-4.971,0-9,4.029-9,9v58.748 | ||||||
|  | 		c0,4.971,4.029,9,9,9h58.748c4.971,0,9-4.029,9-9v-58.748c0-4.971-4.029-9-9-9h-20.374v-72.648h20.374c4.971,0,9-4.029,9-9V73.751 | ||||||
|  | 		h53.702v102.022h-20.374c-4.971,0-9,4.029-9,9v58.748c0,4.971,4.029,9,9,9h58.747c4.971,0,9-4.029,9-9v-58.748 | ||||||
|  | 		C278.898,179.803,274.869,175.773,269.898,175.773z M58.747,234.521H18v-40.748h40.747V234.521z M159.823,234.521h-40.748v-40.748 | ||||||
|  | 		h40.748V234.521z M159.823,85.125h-40.748V44.377h40.748V85.125z M260.898,234.521h-40.747v-40.748h40.747V234.521z"/> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 1.4 KiB | 
							
								
								
									
										53
									
								
								hosting/static/hosting/img/settings.svg
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								hosting/static/hosting/img/settings.svg
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,53 @@ | ||||||
|  | <?xml version="1.0" encoding="iso-8859-1"?> | ||||||
|  | <!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  --> | ||||||
|  | <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||||
|  | <svg version="1.1" id="Capa_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | ||||||
|  | 	 width="340.274px" height="340.274px" viewBox="0 0 340.274 340.274" style="enable-background:new 0 0 340.274 340.274;" | ||||||
|  | 	 xml:space="preserve"> | ||||||
|  | <g> | ||||||
|  | 	<g> | ||||||
|  | 		<g> | ||||||
|  | 			<path d="M293.629,127.806l-5.795-13.739c19.846-44.856,18.53-46.189,14.676-50.08l-25.353-24.77l-2.516-2.12h-2.937 | ||||||
|  | 				c-1.549,0-6.173,0-44.712,17.48l-14.184-5.719c-18.332-45.444-20.212-45.444-25.58-45.444h-35.765 | ||||||
|  | 				c-5.362,0-7.446-0.006-24.448,45.606l-14.123,5.734C86.848,43.757,71.574,38.19,67.452,38.19l-3.381,0.105L36.801,65.032 | ||||||
|  | 				c-4.138,3.891-5.582,5.263,15.402,49.425l-5.774,13.691C0,146.097,0,147.838,0,153.33v35.068c0,5.501,0,7.44,46.585,24.127 | ||||||
|  | 				l5.773,13.667c-19.843,44.832-18.51,46.178-14.655,50.032l25.353,24.8l2.522,2.168h2.951c1.525,0,6.092,0,44.685-17.516 | ||||||
|  | 				l14.159,5.758c18.335,45.438,20.218,45.427,25.598,45.427h35.771c5.47,0,7.41,0,24.463-45.589l14.195-5.74 | ||||||
|  | 				c26.014,11,41.253,16.585,45.349,16.585l3.404-0.096l27.479-26.901c3.909-3.945,5.278-5.309-15.589-49.288l5.734-13.702 | ||||||
|  | 				c46.496-17.967,46.496-19.853,46.496-25.221v-35.029C340.268,146.361,340.268,144.434,293.629,127.806z M170.128,228.474 | ||||||
|  | 				c-32.798,0-59.504-26.187-59.504-58.364c0-32.153,26.707-58.315,59.504-58.315c32.78,0,59.43,26.168,59.43,58.315 | ||||||
|  | 				C229.552,202.287,202.902,228.474,170.128,228.474z"/> | ||||||
|  | 		</g> | ||||||
|  | 	</g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | <g> | ||||||
|  | </g> | ||||||
|  | </svg> | ||||||
| After Width: | Height: | Size: 1.7 KiB | 
|  | @ -22,3 +22,53 @@ $( document ).ready(function() { | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
| }); | }); | ||||||
|  | 
 | ||||||
|  | function getScrollbarWidth() { | ||||||
|  |     var outer = document.createElement("div"); | ||||||
|  |     outer.style.visibility = "hidden"; | ||||||
|  |     outer.style.width = "100px"; | ||||||
|  |     outer.style.msOverflowStyle = "scrollbar"; // needed for WinJS apps
 | ||||||
|  | 
 | ||||||
|  |     document.body.appendChild(outer); | ||||||
|  | 
 | ||||||
|  |     var widthNoScroll = outer.offsetWidth; | ||||||
|  |     // force scrollbars
 | ||||||
|  |     outer.style.overflow = "scroll"; | ||||||
|  | 
 | ||||||
|  |     // add innerdiv
 | ||||||
|  |     var inner = document.createElement("div"); | ||||||
|  |     inner.style.width = "100%"; | ||||||
|  |     outer.appendChild(inner); | ||||||
|  | 
 | ||||||
|  |     var widthWithScroll = inner.offsetWidth; | ||||||
|  | 
 | ||||||
|  |     // remove divs
 | ||||||
|  |     outer.parentNode.removeChild(outer); | ||||||
|  | 
 | ||||||
|  |     return widthNoScroll - widthWithScroll; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // globally stores the width of scrollbar
 | ||||||
|  | var scrollbarWidth = getScrollbarWidth(); | ||||||
|  | var paddingAdjusted = false; | ||||||
|  | 
 | ||||||
|  | $( document ).ready(function() { | ||||||
|  |     // add proper padding to fixed topnav on modal show
 | ||||||
|  |     $('body').on('click', '[data-toggle=modal]', function(){ | ||||||
|  |         var $body = $('body'); | ||||||
|  |         if ($body[0].scrollHeight > $body.height()) { | ||||||
|  |             scrollbarWidth = getScrollbarWidth(); | ||||||
|  |             var topnavPadding = parseInt($('.navbar-fixed-top.topnav').css('padding-right')); | ||||||
|  |             $('.navbar-fixed-top.topnav').css('padding-right', topnavPadding+scrollbarWidth); | ||||||
|  |             paddingAdjusted = true; | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | 
 | ||||||
|  |     // remove added padding on modal hide
 | ||||||
|  |     $('body').on('hidden.bs.modal', function(){ | ||||||
|  |         if (paddingAdjusted) { | ||||||
|  |             var topnavPadding = parseInt($('.navbar-fixed-top.topnav').css('padding-right')); | ||||||
|  |             $('.navbar-fixed-top.topnav').css('padding-right', topnavPadding-scrollbarWidth); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | }); | ||||||
|  | @ -3,193 +3,110 @@ | ||||||
| {% load i18n %} | {% load i18n %} | ||||||
| 
 | 
 | ||||||
| {% block content %} | {% block content %} | ||||||
| <div> | 	{% if messages %} | ||||||
| 	<div class="virtual-machine-container dashboard-container "> | 	    <div class="alert alert-warning"> | ||||||
| 		<div class="row"> | 	        {% for message in messages %} | ||||||
| 			<div class="col-md-9 col-md-offset-2"> | 	        <span>{{ message }}</span> | ||||||
| 				 <div  class="col-sm-12"> | 	        {% endfor %} | ||||||
| 				        <h3><i class="fa fa-cloud fa-separate" aria-hidden="true"></i> {{virtual_machine.name}}</h3> | 	    </div> | ||||||
| 				        <hr/> | 	{% endif %} | ||||||
| 				        <div class="col-md-3"> <!-- required for floating --> | 	<div class="virtual-machine-container dashboard-container"> | ||||||
| 				          <!-- Nav tabs --> |     <h1 class="dashboard-title-thin">{% trans "Your Virtual Machine Detail" %}</h1> | ||||||
| 				          <ul class="nav nav-tabs tabs-left sideways"> | 		<div class="vm-detail-contain"> | ||||||
| 				            <li class="active"> | 			<div class="vm-detail-item"> | ||||||
| 				            	<a href="#settings-v" data-toggle="tab"> | 				<h2 class="vm-detail-title">{% trans "VM Settings" %} <img src="{% static 'hosting/img/settings.svg' %}" class="un-icon"></h2> | ||||||
| 				            		<i class="fa fa-cogs" aria-hidden="true"></i> | 				<h3 class="vm-name">{{virtual_machine.name}}</h3> | ||||||
| 				            		{% trans "Settings"%} | 				{% if virtual_machine.ipv6 %} | ||||||
| 				            	</a> | 					<div class="vm-detail-ip"> | ||||||
| 				            </li> | 						<p> | ||||||
| 				            <li> | 							<span>IPv4:</span> | ||||||
| 				            	<a href="#billing-v" data-toggle="tab"> | 							<span class="value">{{virtual_machine.ipv4}}</span> | ||||||
| 				            		<i class="fa fa-money" aria-hidden="true"></i> | 							<button data-clipboard-text="{{virtual_machine.ipv4}}" class="to_copy btn btn-link" data-toggle="tooltip" data-placement="left" title="{% trans 'Copied' %}" data-trigger="click"> | ||||||
| 				            		{% trans "Billing"%} | 								<img class="un-icon" src="{% static 'hosting/img/copy.svg' %}"> | ||||||
| 				            	</a> | 							</button> | ||||||
| 				            </li> | 						</p> | ||||||
| 				            <li> | 						<p> | ||||||
| 				            	<a href="#status-v" data-toggle="tab"> | 							<span>IPv6:</span> | ||||||
| 				            		<i class="fa fa-signal" aria-hidden="true"></i> {% trans "Status"%} | 							<span class="value value-sm-block">{{virtual_machine.ipv6}}</span> | ||||||
| 				            	</a> | 							<button data-clipboard-text="{{virtual_machine.ipv6}}" class="to_copy btn btn-link" data-toggle="tooltip" data-placement="left" title="{% trans 'Copied' %}" data-trigger="click"> | ||||||
| 				            </li> | 								<img class="un-icon" src="{% static 'hosting/img/copy.svg' %}"> | ||||||
| 				          </ul> | 							</button> | ||||||
| 				        </div> | 						</p> | ||||||
| 
 | 					</div> | ||||||
| 				        <div class="col-md-9"> | 				{% endif %} | ||||||
| 				          <!-- Tab panes --> | 				<div class="vm-detail-config"> | ||||||
| 				          <div class="tab-content"> | 					<p><span>{% trans "Cores" %}:</span><span class="value">{{virtual_machine.cores}}</span></p> | ||||||
| 				            <div class="tab-pane active" id="settings-v"> | 					<p><span>{% trans "Memory" %}:</span><span class="value">{{virtual_machine.memory}} GB</span></p> | ||||||
| 				            	<div class="row"> | 					<p><span>{% trans "Disk" %}:</span><span class="value">{{virtual_machine.disk_size|floatformat:2}} GB</span></p> | ||||||
| 									<div class="col-md-12 inline-headers"> | 					<p><span>{% trans "Configuration" %}:</span><span class="value">{{virtual_machine.configuration}}</span></p> | ||||||
| 									<h3>{{virtual_machine.hosting_company_name}}</h3> |  | ||||||
| 
 |  | ||||||
| 										{% if virtual_machine.ipv6 %} |  | ||||||
| 											<div class="pull-right right-place"> |  | ||||||
| 												<button type="link" |  | ||||||
| 					data-clipboard-text="{{virtual_machine.ipv4}}" id="copy_vm_id" class="to_copy btn btn-link" |  | ||||||
| 													data-toggle="tooltip"  data-placement="bottom" title="Copied"  data-trigger="click"> |  | ||||||
| 														Ipv4: {{virtual_machine.ipv4}} <i class="fa fa-files-o" aria-hidden="true"></i> |  | ||||||
| 												</button> |  | ||||||
| 												<button type="link" |  | ||||||
| 					data-clipboard-text="{{virtual_machine.ipv6}}" id="copy_vm_id" class="to_copy btn btn-link" |  | ||||||
| 													data-toggle="tooltip"  data-placement="bottom" title="Copied"  data-trigger="click"> |  | ||||||
| 														Ipv6: {{virtual_machine.ipv6}} <i class="fa fa-files-o" aria-hidden="true"></i> |  | ||||||
| 												</button> |  | ||||||
| 											</div> |  | ||||||
| 										{% else %} |  | ||||||
| 
 |  | ||||||
| 											<div class="pull-right right-place"> |  | ||||||
| 												<span class="label label-warning"><strong>{% trans "Ip not assigned yet"%}</strong></span> |  | ||||||
| 												<i data-toggle="tooltip"  title="Your ip will be assigned soon" class="fa fa-info-circle" aria-hidden="true"></i> |  | ||||||
| 											</div> |  | ||||||
| 
 |  | ||||||
| 										{% endif %} |  | ||||||
| 
 |  | ||||||
| 									<hr> |  | ||||||
| 
 |  | ||||||
| 									</div> |  | ||||||
| 				            	</div> |  | ||||||
| 								<div class="row"> |  | ||||||
| 								  <div class="col-md-12"> |  | ||||||
| 								    <div class="row"> |  | ||||||
| 								      <div class="col-md-3"> |  | ||||||
| 								        <div class="well text-center box-setting"> |  | ||||||
| 								        	<i class="fa fa-cubes" aria-hidden="true"></i> |  | ||||||
| 								        	<span>{% trans "Cores"%}</span> |  | ||||||
| 								        	<span class="label label-success">{{virtual_machine.cores}}</span> |  | ||||||
| 								        </div> |  | ||||||
| 								      </div> |  | ||||||
| 								      <div class="col-md-3"> |  | ||||||
| 								        <div class="well text-center box-setting"> |  | ||||||
| 								        	<i class="fa fa-tachometer" aria-hidden="true"></i> {% trans "Memory"%} <br/> |  | ||||||
| 								        	<span class="label label-success">{{virtual_machine.memory}} GB</span> |  | ||||||
| 								        </div> |  | ||||||
| 								      </div> |  | ||||||
| 								      <div class="col-md-3"> |  | ||||||
| 								        <div class="well text-center box-setting"> |  | ||||||
| 								        	<i class="fa fa-hdd-o" aria-hidden="true"></i> |  | ||||||
| 								        	<span>{% trans "Disk"%}</span> |  | ||||||
| 								        	<span class="label label-success">{{virtual_machine.disk_size|floatformat:2}} GB</span> |  | ||||||
| 								        </div> |  | ||||||
| 								      </div> |  | ||||||
| 								    </div><!--/row--> |  | ||||||
| 								  </div><!--/col-12--> |  | ||||||
| 								</div><!--/row--> |  | ||||||
| 								<div class="row"> |  | ||||||
| 									<div class="col-md-12"> |  | ||||||
| 										{% trans "Configuration"%}: {{virtual_machine.configuration}} |  | ||||||
| 									</div> |  | ||||||
| 								</div> |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 				            </div> |  | ||||||
| 				            <div class="tab-pane" id="billing-v"> |  | ||||||
| 
 |  | ||||||
| 				            	<div class="row "> |  | ||||||
| 									<div class="col-md-12 inline-headers"> |  | ||||||
| 										<h3>{% trans "Current pricing"%}</h3> |  | ||||||
| 										<span class="h3 pull-right"><strong>{{virtual_machine.price|floatformat}} CHF</strong>/month</span> |  | ||||||
| 										<hr> |  | ||||||
| 									</div> |  | ||||||
| 				            	</div> |  | ||||||
| 				            </div> |  | ||||||
| 				            <div class="tab-pane" id="status-v"> |  | ||||||
| 				            	<div class="row "> |  | ||||||
| 									<div class="col-md-12 inline-headers"> |  | ||||||
| 										<h3>{% trans "Current status"%}</h3> |  | ||||||
| 
 |  | ||||||
| 										<div  class="pull-right space-above"> |  | ||||||
| 											{% if virtual_machine.state == 'PENDING' %} |  | ||||||
| 												<span class="label |  | ||||||
|                                                     label-warning"><strong>Pending</strong></span> |  | ||||||
| 											{% elif  virtual_machine.state == 'ACTIVE' %} |  | ||||||
| 												<span class="label |  | ||||||
|                                                     label-success"><strong>Online</strong></span> |  | ||||||
| 											{% elif  virtual_machine.state == 'FAILED'%} |  | ||||||
| 												<span class="label |  | ||||||
|                                                     label-danger"><strong>Failed</strong></span> |  | ||||||
| 											{% endif %} |  | ||||||
| 										</div> |  | ||||||
| 									</div> |  | ||||||
| 				            	</div> |  | ||||||
| 				            	{% if not virtual_machine.status == 'canceled' %} |  | ||||||
| 				            	<div class="row"> |  | ||||||
| 									<div class="col-md-12 separate-md"> |  | ||||||
| 										<div class="pull-right"> |  | ||||||
| 											<form method="POST" |  | ||||||
|                  id="virtual_machine_cancel_form" class="cancel-form" action="{% url 'hosting:virtual_machines' virtual_machine.vm_id %}"> |  | ||||||
| 											{% csrf_token %} |  | ||||||
| 											</form> |  | ||||||
| 
 |  | ||||||
| 												<button type="text" data-href="{% url 'hosting:virtual_machines' virtual_machine.vm_id %}" data-toggle="modal" data-target="#confirm-cancel" class="btn btn-danger">{% trans "Terminate Virtual Machine"%}</button> |  | ||||||
| 
 |  | ||||||
| 										</div> |  | ||||||
| 
 |  | ||||||
| 									</div> |  | ||||||
|                                     <div class="col-md-12"> |  | ||||||
|                                         <br/> |  | ||||||
|                                         {% if messages %} |  | ||||||
|                                             <div class="alert alert-warning"> |  | ||||||
|                                                 {% for message in messages %} |  | ||||||
|                                                 <span>{{ message }}</span> |  | ||||||
|                                                 {% endfor %} |  | ||||||
|                                             </div> |  | ||||||
|                                         {% endif %} |  | ||||||
|                                     </div> |  | ||||||
| 
 |  | ||||||
| 									<!-- Cancel Modal --> |  | ||||||
| 									<div class="modal fade" id="confirm-cancel" tabindex="-1" role="dialog" aria-labelledby="myModalLabel" aria-hidden="true"> |  | ||||||
| 									    <div class="modal-dialog"> |  | ||||||
| 									        <div class="modal-content"> |  | ||||||
| 												<div class="modal-header"> |  | ||||||
| 													<button type="button" class="close" data-dismiss="modal" |  | ||||||
| 															aria-label="Confirm"><span |  | ||||||
| 															aria-hidden="true">×</span> |  | ||||||
| 													</button> |  | ||||||
| 												</div> |  | ||||||
| 									            <div class="modal-body"> |  | ||||||
| 													<div class="modal-icon"><i class="fa fa-ban" aria-hidden="true"></i></div> |  | ||||||
| 													<h4 class="modal-title" id="ModalLabel">{% trans "Terminate your Virtual Machine"%}</h4> |  | ||||||
| 									                <p class="modal-text">{% trans "Are you sure do you want to cancel your Virtual Machine "%} {{virtual_machine.name}} ?</p> |  | ||||||
| 									            </div> |  | ||||||
| 									            <div class="modal-footer"> |  | ||||||
| 									                <a class="btn btn-danger btn-ok">OK</a> |  | ||||||
| 									            </div> |  | ||||||
| 									        </div> |  | ||||||
| 									    </div> |  | ||||||
| 									</div> |  | ||||||
| 									<!-- / Cancel Modal --> |  | ||||||
| 				            	</div> |  | ||||||
| 				            	{% endif %} |  | ||||||
| 				            </div> |  | ||||||
| 				          </div> |  | ||||||
| 				        </div> |  | ||||||
| 
 |  | ||||||
| 				        <div class="clearfix"></div> |  | ||||||
| 				</div> | 				</div> | ||||||
| 			</div> | 			</div> | ||||||
| 
 | 			<div class="vm-detail-item"> | ||||||
|  | 				<h2 class="vm-detail-title">{% trans "Billing" %} <img src="{% static 'hosting/img/billing.svg' %}" class="un-icon"></h2> | ||||||
|  | 				<div class="vm-vmid"> | ||||||
|  | 					<div class="vm-item-subtitle">{% trans "Current Pricing" %}</div> | ||||||
|  | 					<div class="vm-item-lg">{{virtual_machine.price|floatformat}} CHF/{% trans "Month" %}</div> | ||||||
|  | 					<a class="btn btn-vm-invoice" href="{% url 'hosting:orders' order.pk %}">{% trans "See Invoice" %}</a> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="vm-detail-item"> | ||||||
|  | 				<h2 class="vm-detail-title">{% trans "Status" %} <img src="{% static 'hosting/img/connected.svg' %}" class="un-icon"></h2> | ||||||
|  | 				<div class="vm-vmid"> | ||||||
|  | 					<div class="vm-item-subtitle">{% trans "Your VM is" %}</div> | ||||||
|  | 					{% if virtual_machine.state == 'PENDING' %} | ||||||
|  | 						<div class="vm-item-lg vm-color-pending">{% trans "Pending" %}</div> | ||||||
|  | 					{% elif  virtual_machine.state == 'ACTIVE' %} | ||||||
|  | 						<div class="vm-item-lg vm-color-online">{% trans "Online" %}</div> | ||||||
|  | 					{% elif  virtual_machine.state == 'FAILED'%} | ||||||
|  | 						<div class="vm-item-lg vm-color-failed">{% trans "Failed" %}</div> | ||||||
|  | 					{% endif %} | ||||||
|  | 					{% if not virtual_machine.status == 'canceled' %} | ||||||
|  | 						<form method="POST" id="virtual_machine_cancel_form" class="cancel-form" action="{% url 'hosting:virtual_machines' virtual_machine.vm_id %}"> | ||||||
|  | 							{% csrf_token %} | ||||||
|  | 						</form> | ||||||
|  | 						<button data-href="{% url 'hosting:virtual_machines' virtual_machine.vm_id %}" data-toggle="modal" data-target="#confirm-cancel" class="btn btn-vm-term">{% trans "Terminate VM" %}</button> | ||||||
|  | 					{% endif %} | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="vm-contact-us"> | ||||||
|  | 			<div> | ||||||
|  | 				<h2 class="vm-detail-title">{% trans "Support / Contact" %} <img class="un-icon visible-xs" src="{% static 'hosting/img/24-hours-support.svg' %}"></h2> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="vm-contact-us-text text-center"> | ||||||
|  | 				<img class="un-icon hidden-xs" src="{% static 'hosting/img/24-hours-support.svg' %}"> | ||||||
|  | 				<div> | ||||||
|  | 					<span>{% trans "Something doesn't work?" %}</span> <span>{% trans "We are here to help you!" %}</span> | ||||||
|  | 				</div> | ||||||
|  | 			</div> | ||||||
|  | 			<div class="text-center"> | ||||||
|  | 				<a class="btn btn-vm-contact" href="mailto:support@datacenterlight.ch">{% trans "CONTACT" %}</a> | ||||||
|  | 			</div> | ||||||
|  | 		</div> | ||||||
|  | 		<div class="text-center"> | ||||||
|  | 			<a class="btn btn-vm-back" href="{% url 'hosting:virtual_machines' %}">{% trans "BACK TO LIST" %}</a> | ||||||
|  | 		</div> | ||||||
|  | 	</div> | ||||||
|  | 	<!-- Cancel Modal --> | ||||||
|  | 	<div class="modal fade" id="confirm-cancel" tabindex="-1" role="dialog" aria-labelledby="myModalLabel" aria-hidden="true"> | ||||||
|  | 	    <div class="modal-dialog"> | ||||||
|  | 	        <div class="modal-content"> | ||||||
|  | 				<div class="modal-header"> | ||||||
|  | 					<button type="button" class="close" data-dismiss="modal" aria-label="Confirm"><span aria-hidden="true">×</span></button> | ||||||
|  | 				</div> | ||||||
|  | 	            <div class="modal-body"> | ||||||
|  | 					<div class="modal-icon"><i class="fa fa-ban" aria-hidden="true"></i></div> | ||||||
|  | 					<h4 class="modal-title" id="ModalLabel">{% trans "Terminate your Virtual Machine"%}</h4> | ||||||
|  | 					<div class="modal-text"> | ||||||
|  | 						<p>{% trans "Do you want to cancel your Virtual Machine" %} ?</p> | ||||||
|  | 						<p><strong>{{virtual_machine.name}}</strong></p> | ||||||
|  | 					</div> | ||||||
|  | 	            </div> | ||||||
|  | 	            <div class="modal-footer"> | ||||||
|  | 	                <a class="btn btn-danger btn-ok">{% trans "OK" %}</a> | ||||||
|  | 	            </div> | ||||||
|  | 	        </div> | ||||||
| 	    </div> | 	    </div> | ||||||
| 	</div> | 	</div> | ||||||
| 
 | 	<!-- / Cancel Modal --> | ||||||
| </div> |  | ||||||
| 
 |  | ||||||
| {%endblock%} | {%endblock%} | ||||||
|  |  | ||||||
|  | @ -20,9 +20,12 @@ urlpatterns = [ | ||||||
|     url(r'orders/(?P<pk>\d+)/?$', OrdersHostingDetailView.as_view(), name='orders'), |     url(r'orders/(?P<pk>\d+)/?$', OrdersHostingDetailView.as_view(), name='orders'), | ||||||
|     url(r'bills/?$', HostingBillListView.as_view(), name='bills'), |     url(r'bills/?$', HostingBillListView.as_view(), name='bills'), | ||||||
|     url(r'bills/(?P<pk>\d+)/?$', HostingBillDetailView.as_view(), name='bills'), |     url(r'bills/(?P<pk>\d+)/?$', HostingBillDetailView.as_view(), name='bills'), | ||||||
|     url(r'cancel_order/(?P<pk>\d+)/?$', OrdersHostingDeleteView.as_view(), name='delete_order'), |     url(r'cancel_order/(?P<pk>\d+)/?$', | ||||||
|     url(r'create_virtual_machine/?$', CreateVirtualMachinesView.as_view(), name='create_virtual_machine'), |         OrdersHostingDeleteView.as_view(), name='delete_order'), | ||||||
|     url(r'my-virtual-machines/?$', VirtualMachinesPlanListView.as_view(), name='virtual_machines'), |     url(r'create_virtual_machine/?$', CreateVirtualMachinesView.as_view(), | ||||||
|  |         name='create_virtual_machine'), | ||||||
|  |     url(r'my-virtual-machines/?$', | ||||||
|  |         VirtualMachinesPlanListView.as_view(), name='virtual_machines'), | ||||||
|     url(r'my-virtual-machines/(?P<pk>\d+)/?$', VirtualMachineView.as_view(), |     url(r'my-virtual-machines/(?P<pk>\d+)/?$', VirtualMachineView.as_view(), | ||||||
|         name='virtual_machines'), |         name='virtual_machines'), | ||||||
|     url(r'ssh_keys/?$', SSHKeyListView.as_view(), |     url(r'ssh_keys/?$', SSHKeyListView.as_view(), | ||||||
|  | @ -44,5 +47,6 @@ urlpatterns = [ | ||||||
|         PasswordResetConfirmView.as_view(), name='reset_password_confirm'), |         PasswordResetConfirmView.as_view(), name='reset_password_confirm'), | ||||||
|     url(r'^logout/?$', auth_views.logout, |     url(r'^logout/?$', auth_views.logout, | ||||||
|         {'next_page': '/hosting/login?logged_out=true'}, name='logout'), |         {'next_page': '/hosting/login?logged_out=true'}, name='logout'), | ||||||
|     url(r'^validate/(?P<validate_slug>.*)/$', SignupValidatedView.as_view(), name='validate') |     url(r'^validate/(?P<validate_slug>.*)/$', | ||||||
|  |         SignupValidatedView.as_view(), name='validate') | ||||||
| ] | ] | ||||||
|  |  | ||||||
|  | @ -244,7 +244,8 @@ class SignupValidatedView(SignupValidateView): | ||||||
|                 lurl=login_url) |                 lurl=login_url) | ||||||
|         else: |         else: | ||||||
|             home_url = '<a href="' + \ |             home_url = '<a href="' + \ | ||||||
|                        reverse('datacenterlight:index') + '">Data Center Light</a>' |                        reverse('datacenterlight:index') + \ | ||||||
|  |                        '">Data Center Light</a>' | ||||||
|             message = '{sorry_message} <br />{go_back_to} {hurl}'.format( |             message = '{sorry_message} <br />{go_back_to} {hurl}'.format( | ||||||
|                 sorry_message=_("Sorry. Your request is invalid."), |                 sorry_message=_("Sorry. Your request is invalid."), | ||||||
|                 go_back_to=_('Go back to'), |                 go_back_to=_('Go back to'), | ||||||
|  | @ -831,6 +832,7 @@ class VirtualMachineView(LoginRequiredMixin, View): | ||||||
|             serializer = VirtualMachineSerializer(vm) |             serializer = VirtualMachineSerializer(vm) | ||||||
|             context = { |             context = { | ||||||
|                 'virtual_machine': serializer.data, |                 'virtual_machine': serializer.data, | ||||||
|  |                 'order': HostingOrder.objects.get(vm_id=serializer.data['vm_id']) | ||||||
|             } |             } | ||||||
|         except: |         except: | ||||||
|             pass |             pass | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| import ipaddress | import ipaddress | ||||||
| 
 | 
 | ||||||
|  | from builtins import hasattr | ||||||
| from rest_framework import serializers | from rest_framework import serializers | ||||||
| 
 | 
 | ||||||
| from oca import OpenNebulaException | from oca import OpenNebulaException | ||||||
|  | @ -32,7 +33,7 @@ class VirtualMachineTemplateSerializer(serializers.Serializer): | ||||||
|             return 0 |             return 0 | ||||||
| 
 | 
 | ||||||
|     def get_memory(self, obj): |     def get_memory(self, obj): | ||||||
|         return int(obj.template.memory)/1024 |         return int(obj.template.memory) / 1024 | ||||||
| 
 | 
 | ||||||
|     def get_name(self, obj): |     def get_name(self, obj): | ||||||
|         return obj.name.strip('public-') |         return obj.name.strip('public-') | ||||||
|  | @ -57,13 +58,13 @@ class VirtualMachineSerializer(serializers.Serializer): | ||||||
|     configuration = serializers.SerializerMethodField() |     configuration = serializers.SerializerMethodField() | ||||||
| 
 | 
 | ||||||
|     template_id = serializers.ChoiceField( |     template_id = serializers.ChoiceField( | ||||||
|                 choices=[(key.id, key.name) for key in |         choices=[(key.id, key.name) for key in | ||||||
|                          OpenNebulaManager().try_get_templates() |                  OpenNebulaManager().try_get_templates() | ||||||
|                          ], |                  ], | ||||||
|                 source='template.template_id', |         source='template.template_id', | ||||||
|                 write_only=True, |         write_only=True, | ||||||
|                 default=[] |         default=[] | ||||||
|             ) |     ) | ||||||
| 
 | 
 | ||||||
|     def create(self, validated_data): |     def create(self, validated_data): | ||||||
|         owner = validated_data['owner'] |         owner = validated_data['owner'] | ||||||
|  | @ -74,10 +75,10 @@ class VirtualMachineSerializer(serializers.Serializer): | ||||||
| 
 | 
 | ||||||
|         template_id = validated_data['template']['template_id'] |         template_id = validated_data['template']['template_id'] | ||||||
|         specs = { |         specs = { | ||||||
|                     'cpu': cores, |             'cpu': cores, | ||||||
|                     'disk_size': disk, |             'disk_size': disk, | ||||||
|                     'memory': memory, |             'memory': memory, | ||||||
|                 } |         } | ||||||
| 
 | 
 | ||||||
|         try: |         try: | ||||||
|             manager = OpenNebulaManager(email=owner.email, |             manager = OpenNebulaManager(email=owner.email, | ||||||
|  | @ -92,7 +93,7 @@ class VirtualMachineSerializer(serializers.Serializer): | ||||||
|         return manager.get_vm(opennebula_id) |         return manager.get_vm(opennebula_id) | ||||||
| 
 | 
 | ||||||
|     def get_memory(self, obj): |     def get_memory(self, obj): | ||||||
|         return int(obj.template.memory)/1024 |         return int(obj.template.memory) / 1024 | ||||||
| 
 | 
 | ||||||
|     def get_disk_size(self, obj): |     def get_disk_size(self, obj): | ||||||
|         template = obj.template |         template = obj.template | ||||||
|  | @ -104,9 +105,9 @@ class VirtualMachineSerializer(serializers.Serializer): | ||||||
|     def get_price(self, obj): |     def get_price(self, obj): | ||||||
|         template = obj.template |         template = obj.template | ||||||
|         price = float(template.vcpu) * 5.0 |         price = float(template.vcpu) * 5.0 | ||||||
|         price += (int(template.memory)/1024 * 2.0) |         price += (int(template.memory) / 1024 * 2.0) | ||||||
|         for disk in template.disks: |         for disk in template.disks: | ||||||
|             price += int(disk.size)/1024 * 0.6 |             price += int(disk.size) / 1024 * 0.6 | ||||||
|         return price |         return price | ||||||
| 
 | 
 | ||||||
|     def get_configuration(self, obj): |     def get_configuration(self, obj): | ||||||
|  | @ -115,15 +116,30 @@ class VirtualMachineSerializer(serializers.Serializer): | ||||||
|         return template.name.strip('public-') |         return template.name.strip('public-') | ||||||
| 
 | 
 | ||||||
|     def get_ipv4(self, obj): |     def get_ipv4(self, obj): | ||||||
|         nic = obj.template.nics[0] |         """ | ||||||
|         if 'vm-ipv6-nat64-ipv4' in nic.network and is_in_v4_range(nic.mac): |         Get the IPv4s from the given VM | ||||||
|             return str(v4_from_mac(nic.mac)) | 
 | ||||||
|  |         :param obj: The VM in contention | ||||||
|  |         :return: Returns csv string of all IPv4s added to this VM otherwise returns "-" if no IPv4 is available | ||||||
|  |         """ | ||||||
|  |         ipv4 = [] | ||||||
|  |         for nic in obj.template.nics: | ||||||
|  |             if hasattr(nic, 'ip'): | ||||||
|  |                 ipv4.append(nic.ip) | ||||||
|  |         if len(ipv4) > 0: | ||||||
|  |             return ', '.join(ipv4) | ||||||
|         else: |         else: | ||||||
|             return '-' |             return '-' | ||||||
| 
 | 
 | ||||||
|     def get_ipv6(self, obj): |     def get_ipv6(self, obj): | ||||||
|         nic = obj.template.nics[0] |         ipv6 = [] | ||||||
|         return nic.ip6_global |         for nic in obj.template.nics: | ||||||
|  |             if hasattr(nic, 'ip6_global'): | ||||||
|  |                 ipv6.append(nic.ip6_global) | ||||||
|  |         if len(ipv6) > 0: | ||||||
|  |             return ', '.join(ipv6) | ||||||
|  |         else: | ||||||
|  |             return '-' | ||||||
| 
 | 
 | ||||||
|     def get_name(self, obj): |     def get_name(self, obj): | ||||||
|         return obj.name.strip('public-') |         return obj.name.strip('public-') | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue