Merge upstream
This commit is contained in:
		
				commit
				
					
						06ef940b01
					
				
			
		
					 23 changed files with 289 additions and 286 deletions
				
			
		
							
								
								
									
										1
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
										
									
									
										vendored
									
									
								
							|  | @ -36,3 +36,4 @@ secret-key | |||
| .env | ||||
| *.mo | ||||
| *.log | ||||
| *.sql | ||||
|  |  | |||
|  | @ -1,3 +1,10 @@ | |||
| Next: | ||||
|     * #3764: [hosting] Show cancelled VMs' invoices  | ||||
|     * #3736: [dcl] Refactor the place where we compute the VM price | ||||
|     * #3730: [dcl] Refactor price parameter passed in the DCL flow | ||||
|     * #3807: [dcl] Remove PricingView as it is no more used | ||||
|     * #3813: [hosting] JS error in create ssh key page | ||||
| 
 | ||||
| 1.2.3: 2017-09-25 | ||||
|     * #3484: [dcl, hosting] Refactored account activation, password reset, VM order and cancellation email | ||||
|     * #3731: [dcl, hosting] Added cdist ssh key handler  | ||||
|  | @ -8,6 +15,7 @@ | |||
|     * #3777: [hosting] Create new VM calculator added like dcl landing | ||||
|     * #3781: [hosting] Resend activation mail | ||||
|     * #3806: [hosting] Fix can not create VMs after password reset | ||||
|     * #3812: [hosting] Modal check icon made thin and font-size fixed | ||||
|     * Feature: [cms, blog] Added /cms prefix for all the django-cms generated urls | ||||
|     * Bugfix: [dcl, hosting] added host to celery error mails | ||||
|     * Bugfix: [ungleich] Fixed wrong subdomain digitalglarus.ungleich.ch | ||||
|  |  | |||
|  | @ -1653,3 +1653,20 @@ a.list-group-item-danger.active:focus { | |||
| .panel-danger > .panel-heading .badge { | ||||
|     background-color: #eb4d5c; | ||||
| } | ||||
| 
 | ||||
| .checkmark { | ||||
|   display: inline-block; | ||||
| } | ||||
| .checkmark:after { | ||||
|   /*Add another block-level blank space*/ | ||||
|   content: ''; | ||||
|   display: block; | ||||
|   /*Make it a small rectangle so the border will create an L-shape*/ | ||||
|   width: 25px; | ||||
|   height: 60px; | ||||
|   /*Add a white border on the bottom and left, creating that 'L' */ | ||||
|   border: solid #777; | ||||
|   border-width: 0 3px 3px 0; | ||||
|   /*Rotate the L 45 degrees to turn it into a checkmark*/ | ||||
|   transform: rotate(45deg); | ||||
| } | ||||
|  |  | |||
|  | @ -155,9 +155,7 @@ | |||
|     function _calcPricing() { | ||||
|         var total = (cardPricing['cpu'].value * 5) + (2 * cardPricing['ram'].value) + (0.6 * cardPricing['storage'].value); | ||||
|         total = parseFloat(total.toFixed(2)); | ||||
| 
 | ||||
|         $("#total").text(total); | ||||
|         $('input[name=total]').val(total); | ||||
|     } | ||||
| 
 | ||||
|     function form_success() { | ||||
|  |  | |||
|  | @ -13,7 +13,7 @@ from hosting.models import HostingOrder, HostingBill | |||
| from membership.models import StripeCustomer, CustomUser | ||||
| from opennebula_api.models import OpenNebulaManager | ||||
| from opennebula_api.serializers import VirtualMachineSerializer | ||||
| from utils.hosting_utils import get_all_public_keys | ||||
| from utils.hosting_utils import get_all_public_keys, get_or_create_vm_detail | ||||
| from utils.forms import UserBillingAddressForm | ||||
| from utils.mailer import BaseEmail | ||||
| from utils.models import BillingAddress | ||||
|  | @ -52,7 +52,8 @@ def create_vm_task(self, vm_template_id, user, specs, template, | |||
|                    stripe_customer_id, billing_address_data, | ||||
|                    billing_address_id, | ||||
|                    charge, cc_details): | ||||
|     logger.debug("Running create_vm_task on {}".format(current_task.request.hostname)) | ||||
|     logger.debug( | ||||
|         "Running create_vm_task on {}".format(current_task.request.hostname)) | ||||
|     vm_id = None | ||||
|     try: | ||||
|         final_price = specs.get('price') | ||||
|  | @ -144,7 +145,8 @@ def create_vm_task(self, vm_template_id, user, specs, template, | |||
|         if 'pass' in user: | ||||
|             lang = 'en-us' | ||||
|             if user.get('language') is not None: | ||||
|                 logger.debug("Language is set to {}".format(user.get('language'))) | ||||
|                 logger.debug( | ||||
|                     "Language is set to {}".format(user.get('language'))) | ||||
|                 lang = user.get('language') | ||||
|             translation.activate(lang) | ||||
|             # Send notification to the user as soon as VM has been booked | ||||
|  | @ -174,6 +176,7 @@ def create_vm_task(self, vm_template_id, user, specs, template, | |||
|             logger.debug("New VM ID is {vm_id}".format(vm_id=vm_id)) | ||||
|             if new_host is not None: | ||||
|                 custom_user = CustomUser.objects.get(email=user.get('email')) | ||||
|                 get_or_create_vm_detail(custom_user, manager, vm_id) | ||||
|                 if custom_user is not None: | ||||
|                     public_keys = get_all_public_keys(custom_user) | ||||
|                     keys = [{'value': key, 'state': True} for key in | ||||
|  |  | |||
|  | @ -8,7 +8,7 @@ | |||
|               <button type="button" class="close" data-dismiss="modal" aria-label="Close"><span aria-hidden="true">×</span></button> | ||||
|             </div> | ||||
|             <div class="modal-body"> | ||||
|               <div class="modal-icon"><i class="fa fa-check" aria-hidden="true"></i></div> | ||||
|               <div class="modal-icon"><i class="checkmark" aria-hidden="true"></i></div> | ||||
|               <h4 class="modal-title">{% trans "Request Sent" %}</h4> | ||||
|               <p class="modal-text">{% trans "Thank you for your subscription! You will receive a confirmation mail from our team" %}</p> | ||||
|             </div> | ||||
|  |  | |||
|  | @ -77,7 +77,6 @@ | |||
|                 {% endfor %} | ||||
|             </select> | ||||
|         </div> | ||||
|         <input type="hidden" name="total"> | ||||
|         <!--<div class="description check-ip"> | ||||
|             <input type="checkbox" name="ipv6"> Ipv6 Only<br> | ||||
|         </div>--> | ||||
|  |  | |||
|  | @ -1,96 +0,0 @@ | |||
| {% extends "datacenterlight/base.html" %} | ||||
| {% load staticfiles i18n%} | ||||
| {% get_current_language as LANGUAGE_CODE %} | ||||
| 
 | ||||
| {% block content %} | ||||
| 	<div class="intro-pricing"> | ||||
| 
 | ||||
| 		<div class="intro-message"> | ||||
| 			<h2 class="section-heading">{% trans "We are cutting down the costs significantly!" %}</h2> | ||||
| 		</div> | ||||
| 		 | ||||
| 	</div> | ||||
| 
 | ||||
| 	<div class="price-calc-section"> | ||||
| 		<div class="card"> | ||||
|             <img class="img-beta" src="{% static 'datacenterlight/img/beta-img.png' %}" alt=""> | ||||
|             <div class="caption"> | ||||
|                 <form method="POST" action=""> | ||||
|                     {% csrf_token %} | ||||
| 
 | ||||
|                     <div class="title"> | ||||
|                        <h3>{% trans "VM hosting" %} </h3> | ||||
|                     </div> | ||||
|                   <div class="price"> | ||||
|                       <span id="total">15</span> | ||||
|                       <span>CHF</span> | ||||
|                       <div class="price-text"> | ||||
|                           <p>{% trans "VAT included" %}</p> | ||||
|                       </div> | ||||
|                   </div> | ||||
|                   <div class="descriptions"> | ||||
|                       <div class="description"> | ||||
|                         <p>{% trans "Hosted in Switzerland" %}</p> | ||||
|                       </div> | ||||
|                       <div class="description"> | ||||
|                          <i class="fa fa-minus-circle left" data-minus="cpu" aria-hidden="true"></i> | ||||
|                          <input class="input-price" type="number"  min="1" max="42" id="coreValue" name="cpu"> | ||||
|                          <span> Core</span> | ||||
|                          <i class="fa fa-plus-circle right" data-plus="cpu"  aria-hidden="true"></i> | ||||
|                       </div> | ||||
|                       <div class="description"> | ||||
|                        <i class="fa fa-minus-circle left" data-minus="ram" aria-hidden="true"></i> | ||||
|                         <input id="ramValue" class="input-price" type="number"  min="2" max="200"  name="ram"> | ||||
|                         <span> GB RAM</span> | ||||
|                         <i class="fa fa-plus-circle right" data-plus="ram"  aria-hidden="true"></i> | ||||
|                       </div> | ||||
|                       <div class="description"> | ||||
|                         <i class="fa fa-minus-circle left" data-minus="storage" aria-hidden="true"></i> | ||||
|                         <input id="storageValue" class="input-price" type="number"  min="10" max="500" step="10" name="storage"> | ||||
|                         <span>{% trans "GB Storage (SSD)" %}</span> | ||||
|                          <i class="fa fa-plus-circle right" data-plus="storage"  aria-hidden="true"></i> | ||||
|                       </div> | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|                       <div class="description select-configuration input"> | ||||
|                         <label for="name">OS</label> | ||||
|                         <select name="config" id=""> | ||||
|                             {% for template in templates %} | ||||
|                                     <option value="{{template.id}}">{{template.name}} </option> | ||||
|                             {% endfor %} | ||||
|                         </select> | ||||
|                       </div> | ||||
|                       <input type="hidden" name="total"> | ||||
| 
 | ||||
|                      <!--  <div class="description input"> | ||||
|                         <label for="name">Name</label> | ||||
|                         <input type="text" name="name" placeholder="Your Name"> | ||||
|                       </div> | ||||
|                       <div class="description input"> | ||||
|                         <label for="email">Email</label> | ||||
|                         <input type="email" name="email" placeholder="Your Email"> | ||||
|                       </div> --> | ||||
| 
 | ||||
|                        <!--<div class="description check-ip"> | ||||
|                         <input type="checkbox" name="ipv6"> Ipv6 Only<br> | ||||
|                        </div>--> | ||||
|                   </div> | ||||
|                   <input type="submit" class="btn btn-primary" value="{% trans 'Order Now!' %}"></input> | ||||
| 
 | ||||
|                 </form> | ||||
|             </div> | ||||
| 		</div> | ||||
| 
 | ||||
| 		<div class="text"> | ||||
| 			<h2 class="section-heading">{% trans "Simple and affordable: Try our virtual machine with featherlight price." %}</h2> | ||||
| 
 | ||||
| 			<div class="description"> | ||||
| 				<p>{% trans "Our VMs are hosted in Glarus, Switzerland, and our website is currently running in BETA mode. If you want more information that you did not find on our website, or if your order is more detailed, or if you encounter any technical hiccups, please contact us at support@datacenterlight.ch, our team will get in touch with you asap." %}</p> | ||||
| 			</div> | ||||
| 		</div> | ||||
| 	</div> | ||||
| {% endblock %} | ||||
| 
 | ||||
| 
 | ||||
|   | ||||
|  | @ -12,6 +12,7 @@ from datacenterlight.models import VMTemplate | |||
| from datacenterlight.tasks import create_vm_task | ||||
| from membership.models import StripeCustomer | ||||
| from opennebula_api.serializers import VMTemplateSerializer | ||||
| from utils.hosting_utils import get_vm_price | ||||
| from utils.models import BillingAddress | ||||
| from utils.stripe_utils import StripeUtils | ||||
| 
 | ||||
|  | @ -94,12 +95,11 @@ class CeleryTaskTestCase(TestCase): | |||
|         cpu = specs.get('cpu') | ||||
|         memory = specs.get('memory') | ||||
|         disk_size = specs.get('disk_size') | ||||
|         amount_to_be_charged = (cpu * 5) + (memory * 2) + (disk_size * 0.6) | ||||
|         plan_name = "{cpu} Cores, {memory} GB RAM, {disk_size} GB SSD".format( | ||||
|             cpu=cpu, | ||||
|         amount_to_be_charged = get_vm_price(cpu=cpu, memory=memory, | ||||
|                                             disk_size=disk_size) | ||||
|         plan_name = StripeUtils.get_stripe_plan_name(cpu=cpu, | ||||
|                                                      memory=memory, | ||||
|                                                      disk_size=disk_size) | ||||
| 
 | ||||
|         stripe_plan_id = StripeUtils.get_stripe_plan_id(cpu=cpu, | ||||
|                                                         ram=memory, | ||||
|                                                         ssd=disk_size, | ||||
|  |  | |||
|  | @ -1,7 +1,7 @@ | |||
| from django.conf.urls import url | ||||
| 
 | ||||
| from .views import IndexView, BetaProgramView, LandingProgramView, \ | ||||
|     BetaAccessView, PricingView, SuccessView, \ | ||||
|     BetaAccessView, SuccessView, \ | ||||
|     PaymentOrderView, OrderConfirmationView, \ | ||||
|     WhyDataCenterLightView, ContactUsView | ||||
| 
 | ||||
|  | @ -15,7 +15,6 @@ urlpatterns = [ | |||
|         name='whydatacenterlight'), | ||||
|     url(r'^beta-program/?$', BetaProgramView.as_view(), name='beta'), | ||||
|     url(r'^landing/?$', LandingProgramView.as_view(), name='landing'), | ||||
|     url(r'^pricing/?$', PricingView.as_view(), name='pricing'), | ||||
|     url(r'^payment/?$', PaymentOrderView.as_view(), name='payment'), | ||||
|     url(r'^order-confirmation/?$', OrderConfirmationView.as_view(), | ||||
|         name='order_confirmation'), | ||||
|  |  | |||
|  | @ -18,13 +18,9 @@ from datacenterlight.tasks import create_vm_task | |||
| from hosting.models import HostingOrder | ||||
| from hosting.forms import HostingUserLoginForm | ||||
| from membership.models import CustomUser, StripeCustomer | ||||
| from opennebula_api.models import OpenNebulaManager | ||||
| from opennebula_api.serializers import ( | ||||
|     VirtualMachineTemplateSerializer, VMTemplateSerializer | ||||
| ) | ||||
| from utils.forms import ( | ||||
|     BillingAddressForm, BillingAddressFormSignup, UserBillingAddressForm | ||||
| ) | ||||
| from opennebula_api.serializers import VMTemplateSerializer | ||||
| from utils.forms import BillingAddressForm | ||||
| from utils.hosting_utils import get_vm_price | ||||
| from utils.mailer import BaseEmail | ||||
| from utils.stripe_utils import StripeUtils | ||||
| from utils.tasks import send_plain_email_task | ||||
|  | @ -84,7 +80,7 @@ class SuccessView(TemplateView): | |||
|     def get(self, request, *args, **kwargs): | ||||
|         if 'specs' not in request.session or 'user' not in request.session: | ||||
|             return HttpResponseRedirect(reverse('datacenterlight:index')) | ||||
|         if 'token' not in request.session: | ||||
|         elif 'token' not in request.session: | ||||
|             return HttpResponseRedirect(reverse('datacenterlight:payment')) | ||||
|         elif 'order_confirmation' not in request.session: | ||||
|             return HttpResponseRedirect( | ||||
|  | @ -98,56 +94,6 @@ class SuccessView(TemplateView): | |||
|         return render(request, self.template_name) | ||||
| 
 | ||||
| 
 | ||||
| class PricingView(TemplateView): | ||||
|     template_name = "datacenterlight/pricing.html" | ||||
| 
 | ||||
|     def get(self, request, *args, **kwargs): | ||||
|         try: | ||||
|             manager = OpenNebulaManager() | ||||
|             templates = manager.get_templates() | ||||
| 
 | ||||
|             context = { | ||||
|                 'templates': VirtualMachineTemplateSerializer(templates, | ||||
|                                                               many=True).data, | ||||
|             } | ||||
|         except Exception: | ||||
|             messages.error(request, | ||||
|                            'We have a temporary problem to connect to our backend. \ | ||||
|                            Please try again in a few minutes' | ||||
|                            ) | ||||
|             context = { | ||||
|                 'error': 'connection' | ||||
|             } | ||||
| 
 | ||||
|         return render(request, self.template_name, context) | ||||
| 
 | ||||
|     def post(self, request): | ||||
| 
 | ||||
|         cores = request.POST.get('cpu') | ||||
|         memory = request.POST.get('ram') | ||||
|         storage = request.POST.get('storage') | ||||
|         price = request.POST.get('total') | ||||
| 
 | ||||
|         template_id = int(request.POST.get('config')) | ||||
|         manager = OpenNebulaManager() | ||||
|         template = manager.get_template(template_id) | ||||
| 
 | ||||
|         request.session['template'] = VirtualMachineTemplateSerializer( | ||||
|             template).data | ||||
| 
 | ||||
|         if not request.user.is_authenticated(): | ||||
|             request.session['next'] = reverse('hosting:payment') | ||||
| 
 | ||||
|         request.session['specs'] = { | ||||
|             'cpu': cores, | ||||
|             'memory': memory, | ||||
|             'disk_size': storage, | ||||
|             'price': price, | ||||
|         } | ||||
| 
 | ||||
|         return redirect(reverse('hosting:payment')) | ||||
| 
 | ||||
| 
 | ||||
| class BetaAccessView(FormView): | ||||
|     template_name = "datacenterlight/beta_access.html" | ||||
|     form_class = BetaAccessForm | ||||
|  | @ -284,7 +230,6 @@ class IndexView(CreateView): | |||
|         memory_field = forms.IntegerField(validators=[self.validate_memory]) | ||||
|         storage = request.POST.get('storage') | ||||
|         storage_field = forms.IntegerField(validators=[self.validate_storage]) | ||||
|         price = request.POST.get('total') | ||||
|         template_id = int(request.POST.get('config')) | ||||
|         template = VMTemplate.objects.filter( | ||||
|             opennebula_vm_template_id=template_id).first() | ||||
|  | @ -321,7 +266,6 @@ class IndexView(CreateView): | |||
|             'cpu': cores, | ||||
|             'memory': memory, | ||||
|             'disk_size': storage, | ||||
|             'price': price | ||||
|         } | ||||
|         request.session['specs'] = specs | ||||
|         request.session['template'] = template_data | ||||
|  | @ -532,9 +476,10 @@ class OrderConfirmationView(DetailView): | |||
|         cpu = specs.get('cpu') | ||||
|         memory = specs.get('memory') | ||||
|         disk_size = specs.get('disk_size') | ||||
|         amount_to_be_charged = (cpu * 5) + (memory * 2) + (disk_size * 0.6) | ||||
|         plan_name = "{cpu} Cores, {memory} GB RAM, {disk_size} GB SSD".format( | ||||
|             cpu=cpu, | ||||
|         amount_to_be_charged = get_vm_price(cpu=cpu, memory=memory, | ||||
|                                             disk_size=disk_size) | ||||
|         specs['price'] = amount_to_be_charged | ||||
|         plan_name = StripeUtils.get_stripe_plan_name(cpu=cpu, | ||||
|                                                      memory=memory, | ||||
|                                                      disk_size=disk_size) | ||||
|         stripe_plan_id = StripeUtils.get_stripe_plan_id(cpu=cpu, | ||||
|  | @ -552,8 +497,8 @@ class OrderConfirmationView(DetailView): | |||
|                 'response_object').stripe_plan_id}]) | ||||
|         stripe_subscription_obj = subscription_result.get('response_object') | ||||
|         # Check if the subscription was approved and is active | ||||
|         if stripe_subscription_obj is None \ | ||||
|                 or stripe_subscription_obj.status != 'active': | ||||
|         if stripe_subscription_obj is None or \ | ||||
|                 stripe_subscription_obj.status != 'active': | ||||
|             msg = subscription_result.get('error') | ||||
|             messages.add_message(self.request, messages.ERROR, msg, | ||||
|                                  extra_tags='failed_payment') | ||||
|  |  | |||
|  | @ -388,7 +388,7 @@ msgid "Processing..." | |||
| msgstr "Abarbeitung..." | ||||
| 
 | ||||
| msgid "Hold tight, we are processing your request" | ||||
| msgstr "Bitte warten - wir verbeiten Deine Anfrage gerade" | ||||
| msgstr "Bitte warten - wir bearbeiten Deine Anfrage gerade" | ||||
| 
 | ||||
| msgid "Some problem encountered. Please try again later." | ||||
| msgstr "Ein Problem ist aufgetreten. Bitte versuche es später noch einmal." | ||||
|  |  | |||
							
								
								
									
										34
									
								
								hosting/migrations/0043_vmdetail.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								hosting/migrations/0043_vmdetail.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,34 @@ | |||
| # -*- coding: utf-8 -*- | ||||
| # Generated by Django 1.9.4 on 2017-09-24 18:12 | ||||
| from __future__ import unicode_literals | ||||
| 
 | ||||
| from django.conf import settings | ||||
| from django.db import migrations, models | ||||
| import django.db.models.deletion | ||||
| 
 | ||||
| 
 | ||||
| class Migration(migrations.Migration): | ||||
| 
 | ||||
|     dependencies = [ | ||||
|         migrations.swappable_dependency(settings.AUTH_USER_MODEL), | ||||
|         ('hosting', '0042_hostingorder_subscription_id'), | ||||
|     ] | ||||
| 
 | ||||
|     operations = [ | ||||
|         migrations.CreateModel( | ||||
|             name='VMDetail', | ||||
|             fields=[ | ||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||
|                 ('vm_id', models.IntegerField(default=0)), | ||||
|                 ('disk_size', models.FloatField(default=0.0)), | ||||
|                 ('cores', models.FloatField(default=0.0)), | ||||
|                 ('memory', models.FloatField(default=0.0)), | ||||
|                 ('configuration', models.CharField(default='', max_length=25)), | ||||
|                 ('ipv4', models.TextField(default='')), | ||||
|                 ('ipv6', models.TextField(default='')), | ||||
|                 ('created_at', models.DateTimeField(auto_now_add=True)), | ||||
|                 ('terminated_at', models.DateTimeField(null=True)), | ||||
|                 ('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)), | ||||
|             ], | ||||
|         ), | ||||
|     ] | ||||
|  | @ -159,3 +159,16 @@ class HostingBill(AssignPermissionsMixin, models.Model): | |||
|         instance = cls.objects.create(customer=customer, | ||||
|                                       billing_address=billing_address) | ||||
|         return instance | ||||
| 
 | ||||
| 
 | ||||
| class VMDetail(models.Model): | ||||
|     user = models.ForeignKey(CustomUser) | ||||
|     vm_id = models.IntegerField(default=0) | ||||
|     disk_size = models.FloatField(default=0.0) | ||||
|     cores = models.FloatField(default=0.0) | ||||
|     memory = models.FloatField(default=0.0) | ||||
|     configuration = models.CharField(default='', max_length=25) | ||||
|     ipv4 = models.TextField(default='') | ||||
|     ipv6 = models.TextField(default='') | ||||
|     created_at = models.DateTimeField(auto_now_add=True) | ||||
|     terminated_at = models.DateTimeField(null=True) | ||||
|  |  | |||
|  | @ -586,7 +586,7 @@ a.unlink:hover { | |||
| } | ||||
| 
 | ||||
| .dcl-order-table-total .tbl-total { | ||||
|     text-align: right; | ||||
|     text-align: center; | ||||
|     color: #000; | ||||
|     padding-left: 44px; | ||||
| } | ||||
|  | @ -614,8 +614,8 @@ a.unlink:hover { | |||
|     font-weight: 300; | ||||
|     border: 1px solid #a1a1a1; | ||||
|     border-radius: 3px; | ||||
|     padding: 8px 10px; | ||||
|     margin-bottom: 20px; | ||||
|     padding: 5px; | ||||
|     margin-bottom: 15px; | ||||
| } | ||||
| .card-warning-error { | ||||
|     border: 1px solid #EB4D5C; | ||||
|  | @ -656,7 +656,7 @@ a.unlink:hover { | |||
| 
 | ||||
| .card-element { | ||||
|     margin-bottom: 10px; | ||||
|     /* padding: 0; */ | ||||
|     padding: 0; | ||||
| } | ||||
| 
 | ||||
| .card-element label{ | ||||
|  | @ -674,12 +674,12 @@ a.unlink:hover { | |||
| 
 | ||||
| #card-errors { | ||||
|     clear: both; | ||||
|     padding: 0 5px 10px; | ||||
|     padding: 0 0 10px; | ||||
|     color: #eb4d5c; | ||||
| } | ||||
| 
 | ||||
| .credit-card-goup{ | ||||
|     padding: 0 5px; | ||||
|     padding: 0; | ||||
| } | ||||
| 
 | ||||
| .order-confirm-date{ | ||||
|  | @ -880,3 +880,40 @@ a.list-group-item-danger.active:focus { | |||
| .has-success .form-control:active { | ||||
|     box-shadow: inset 0 0 1px rgba(0,0,0,0.25); | ||||
| } | ||||
| .checkmark { | ||||
|   display: inline-block; | ||||
| } | ||||
| .checkmark:after { | ||||
|   /*Add another block-level blank space*/ | ||||
|   content: ''; | ||||
|   display: block; | ||||
|   /*Make it a small rectangle so the border will create an L-shape*/ | ||||
|   width: 25px; | ||||
|   height: 60px; | ||||
|   /*Add a white border on the bottom and left, creating that 'L' */ | ||||
|   border: solid #777; | ||||
|   border-width: 0 3px 3px 0; | ||||
|   /*Rotate the L 45 degrees to turn it into a checkmark*/ | ||||
|   transform: rotate(45deg); | ||||
| } | ||||
| 
 | ||||
| .closemark { | ||||
|     display: inline-block; | ||||
|     width: 50px; | ||||
|     height: 50px; | ||||
|     position: relative; | ||||
| } | ||||
| .closemark:before, .closemark:after { | ||||
|   position: absolute; | ||||
|   left: 25px; | ||||
|   content: ' '; | ||||
|   height: 50px; | ||||
|   width: 2px; | ||||
|   background-color: #777; | ||||
| } | ||||
| .closemark:before { | ||||
|   transform: rotate(45deg); | ||||
| } | ||||
| .closemark:after { | ||||
|   transform: rotate(-45deg); | ||||
| } | ||||
|  |  | |||
|  | @ -79,7 +79,6 @@ $(document).ready(function() { | |||
|         $('html,body').scrollTop(scrollmem); | ||||
|     }); | ||||
| 
 | ||||
|     $('.modal-text').removeClass('hide'); | ||||
|     var create_vm_form = $('#virtual_machine_create_form'); | ||||
|     create_vm_form.submit(function () { | ||||
|         $('#btn-create-vm').prop('disabled', true); | ||||
|  | @ -90,26 +89,28 @@ $(document).ready(function() { | |||
|             success: function (data) { | ||||
|                 if (data.status === true) { | ||||
|                     fa_icon = $('.modal-icon > .fa'); | ||||
|                     fa_icon.attr('class', 'fa fa-check'); | ||||
|                     $('.modal-header > .close').attr('class', 'close'); | ||||
|                     fa_icon.attr('class', 'checkmark'); | ||||
|                     // $('.modal-header > .close').removeClass('hidden');
 | ||||
|                     $('#createvm-modal-title').text(data.msg_title); | ||||
|                     $('#createvm-modal-body').text(data.msg_body); | ||||
|                     $('#createvm-modal').on('hidden.bs.modal', function () { | ||||
|                         window.location = data.redirect; | ||||
|                     }) | ||||
|                     $('#createvm-modal-done-btn') | ||||
|                         .attr('href', data.redirect) | ||||
|                         .removeClass('hide'); | ||||
|                 } | ||||
|             }, | ||||
|             error: function (xmlhttprequest, textstatus, message) { | ||||
|                     fa_icon = $('.modal-icon > .fa'); | ||||
|                     fa_icon.attr('class', 'fa fa-times'); | ||||
|                     $('.modal-header > .close').attr('class', 'close'); | ||||
|                     $('.modal-text').addClass('hide'); | ||||
|                     fa_icon.attr('class', 'fa fa-close'); | ||||
|                     if (typeof(create_vm_error_message) !== 'undefined') { | ||||
|                         $('#createvm-modal-title').text(create_vm_error_message); | ||||
|                         $('#createvm-modal-text').text(create_vm_error_message); | ||||
|                     } | ||||
|                     $('#btn-create-vm').prop('disabled', false); | ||||
|                     $('#createvm-modal-close-btn').removeClass('hide'); | ||||
|             } | ||||
|         }); | ||||
|         return false; | ||||
|     }); | ||||
|     $('#createvm-modal').on('hidden.bs.modal', function () { | ||||
|         $(this).find('.modal-footer .btn').addClass('hide'); | ||||
|     }) | ||||
| }); | ||||
|  |  | |||
|  | @ -47,20 +47,5 @@ | |||
|         window.location.href = '{{next_url}}'; | ||||
|     </script> | ||||
| {% endif %} | ||||
| 
 | ||||
| 
 | ||||
| <script type="text/javascript"> | ||||
| 
 | ||||
|     window.onload = function () { | ||||
|         {% for user_key in keys %} | ||||
|             var locale_date = moment.utc(document.getElementById("ssh-created_at-{{user_key.id}}").textContent,'YYYY-MM-DD HH:mm').toDate(); | ||||
|             locale_date =  moment(locale_date).format("YYYY-MM-DD h:mm:ss a"); | ||||
|             document.getElementById('ssh-created_at-{{user_key.id}}').innerHTML = locale_date; | ||||
|         {% endfor %} | ||||
|     }; | ||||
| 
 | ||||
| </script> | ||||
| 
 | ||||
| 
 | ||||
| {%endblock%} | ||||
| 
 | ||||
|  |  | |||
|  | @ -42,7 +42,9 @@ | |||
|                 <p> | ||||
|                     <strong>{% trans "Status" %}: </strong> | ||||
|                     <strong> | ||||
|                         {% if order.status == 'Approved' %} | ||||
|                         {% if vm.terminated_at %} | ||||
|                             <span class="vm-color-failed">{% trans "Terminated" %}</span> | ||||
|                         {% elif order.status == 'Approved' %} | ||||
|                             <span class="vm-color-online">{% trans "Approved" %}</span> | ||||
|                         {% else %} | ||||
|                             <span class="vm-status-failed">{% trans "Declined" %}</span> | ||||
|  | @ -96,14 +98,14 @@ | |||
|                 </p> | ||||
|                 <div class="row"> | ||||
|                     <div class="col-sm-6"> | ||||
|                         {% comment %} | ||||
|                         {% if vm.created_at %} | ||||
|                             <p> | ||||
|                             <span>{% trans "Period" %}</span> | ||||
|                             <span class="pull-right">{{}}</span> | ||||
|                                 <span>{% trans "Period" %}: </span> | ||||
|                                 <span>{{ vm.created_at|date:'Y/m/d' }} - {% if vm.terminated_at %}{{ vm.terminated_at|date:'Y/m/d' }}{% else %}{% now 'Y/m/d' %}{% endif %}</span> | ||||
|                             </p> | ||||
|                         {% endcomment %} | ||||
|                         {% endif %} | ||||
|                         <p> | ||||
|                             <span>{% trans "Cores" %}</span> | ||||
|                             <span>{% trans "Cores" %}: </span> | ||||
|                             {% if vm.cores %} | ||||
|                                 <span class="pull-right">{{vm.cores|floatformat}}</span> | ||||
|                             {% else %} | ||||
|  | @ -111,11 +113,11 @@ | |||
|                             {% endif %} | ||||
|                         </p> | ||||
|                         <p> | ||||
|                             <span>{% trans "Memory" %}</span> | ||||
|                             <span>{% trans "Memory" %}: </span> | ||||
|                             <span class="pull-right">{{vm.memory}} GB</span> | ||||
|                         </p> | ||||
|                         <p> | ||||
|                             <span>{% trans "Disk space" %}</span> | ||||
|                             <span>{% trans "Disk space" %}: </span> | ||||
|                             <span class="pull-right">{{vm.disk_size}} GB</span> | ||||
|                         </p> | ||||
|                         <p> | ||||
|  | @ -160,22 +162,19 @@ | |||
|         <div class="modal-dialog"> | ||||
|             <div class="modal-content"> | ||||
|                 <div class="modal-header"> | ||||
|                     <button type="button" class="close hidden" data-dismiss="modal" | ||||
|                             aria-label="create-vm-close"> | ||||
|                         <span aria-hidden="true">×</span> | ||||
|                     </button> | ||||
|                 </div> | ||||
|                 <div class="modal-body"> | ||||
|                     <div class="modal-icon"> | ||||
|                         <i class="fa fa-cog fa-spin fa-3x fa-fw"></i> | ||||
|                         <span class="sr-only">{% trans "Processing..." %}</span> | ||||
|                     </div> | ||||
|                     <h4 class="modal-title" id="createvm-modal-title"> | ||||
|                     </h4> | ||||
|                     <h4 class="modal-title" id="createvm-modal-title"></h4> | ||||
|                     <div class="modal-text" id="createvm-modal-body"> | ||||
|                         {% trans "Hold tight, we are processing your request" %} | ||||
|                     </div> | ||||
|                     <div class="modal-footer"> | ||||
|                         <a id="createvm-modal-done-btn" class="btn btn-success btn-ok btn-wide hide" href="{% url 'hosting:virtual_machines' %}">{% trans "OK" %}</a> | ||||
|                         <button id="createvm-modal-close-btn" type="button" class="btn btn-danger btn-ok btn-wide hide" data-dismiss="modal" aria-label="create-vm-close">{% trans "Close" %}</button> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|  |  | |||
|  | @ -101,21 +101,5 @@ | |||
|         window.location.href = '{{next_url}}'; | ||||
|     </script> | ||||
| {% endif %} | ||||
| 
 | ||||
| 
 | ||||
| <script type="text/javascript"> | ||||
| 
 | ||||
|     window.onload = function () { | ||||
|         {% for user_key in keys %} | ||||
|             var locale_date = moment.utc(document.getElementById("ssh-created_at-{{user_key.id}}").textContent,'YYYY-MM-DD HH:mm').toDate(); | ||||
|             locale_date =  moment(locale_date).format("YYYY-MM-DD h:mm:ss a"); | ||||
|             document.getElementById('ssh-created_at-{{user_key.id}}').innerHTML = locale_date; | ||||
| 
 | ||||
|         {% endfor %} | ||||
|     }; | ||||
| 
 | ||||
| </script> | ||||
| 
 | ||||
| 
 | ||||
| {%endblock%} | ||||
| 
 | ||||
|  |  | |||
|  | @ -123,8 +123,9 @@ | |||
| 				<div class="modal-header"> | ||||
| 				</div> | ||||
|         <div class="modal-body"> | ||||
| 					<div class="modal-icon"><i class="fa fa-check" aria-hidden="true"></i></div> | ||||
| 					<h4 class="modal-title" id="ModalLabel">{% blocktrans with machine_name=virtual_machine.name %}Your Virtual Machine <strong>{{machine_name}}</strong> is successfully terminated!{% endblocktrans %}</h4> | ||||
| 					<div class="modal-icon"><i class="checkmark" aria-hidden="true"></i></div> | ||||
| 					<h4 class="modal-title"></h4> | ||||
| 					<div class="modal-text" id="ModalLabel">{% blocktrans with machine_name=virtual_machine.name %}Your Virtual Machine <strong>{{machine_name}}</strong> is successfully terminated!{% endblocktrans %}</div> | ||||
|           <div class="modal-footer"> | ||||
|             <a href="{% url 'hosting:virtual_machines' %}"	class="btn btn-success btn-wide">{% trans "OK" %}</a> | ||||
|           </div> | ||||
|  |  | |||
|  | @ -1,6 +1,7 @@ | |||
| import json | ||||
| import logging | ||||
| import uuid | ||||
| from datetime import datetime | ||||
| from time import sleep | ||||
| 
 | ||||
| from django import forms | ||||
|  | @ -37,6 +38,7 @@ from utils.forms import ( | |||
|     BillingAddressForm, PasswordResetRequestForm, UserBillingAddressForm, | ||||
|     ResendActivationEmailForm | ||||
| ) | ||||
| from utils.hosting_utils import get_vm_price | ||||
| from utils.mailer import BaseEmail | ||||
| from utils.stripe_utils import StripeUtils | ||||
| from utils.views import ( | ||||
|  | @ -46,10 +48,11 @@ from utils.views import ( | |||
| from .forms import HostingUserSignupForm, HostingUserLoginForm, \ | ||||
|     UserHostingKeyForm, generate_ssh_key_name | ||||
| from .mixins import ProcessVMSelectionMixin | ||||
| from .models import HostingOrder, HostingBill, HostingPlan, UserHostingKey | ||||
| from .models import ( | ||||
|     HostingOrder, HostingBill, HostingPlan, UserHostingKey, VMDetail | ||||
| ) | ||||
| from datacenterlight.models import VMTemplate | ||||
| 
 | ||||
| 
 | ||||
| logger = logging.getLogger(__name__) | ||||
| 
 | ||||
| CONNECTION_ERROR = "Your VMs cannot be displayed at the moment due to a \ | ||||
|  | @ -681,6 +684,11 @@ class OrdersHostingDetailView(LoginRequiredMixin, | |||
| 
 | ||||
|         if obj is not None: | ||||
|             # invoice for previous order | ||||
|             try: | ||||
|                 vm_detail = VMDetail.objects.get(vm_id=obj.vm_id) | ||||
|                 context['vm'] = vm_detail.__dict__ | ||||
|                 context['vm']['name'] = '{}-{}'.format(context['vm']['configuration'], context['vm']['vm_id']) | ||||
|             except VMDetail.DoesNotExist: | ||||
|                 try: | ||||
|                     manager = OpenNebulaManager( | ||||
|                         email=owner.email, password=owner.password | ||||
|  | @ -760,12 +768,11 @@ class OrdersHostingDetailView(LoginRequiredMixin, | |||
|         cpu = specs.get('cpu') | ||||
|         memory = specs.get('memory') | ||||
|         disk_size = specs.get('disk_size') | ||||
|         amount_to_be_charged = (cpu * 5) + (memory * 2) + (disk_size * 0.6) | ||||
|         plan_name = "{cpu} Cores, {memory} GB RAM, {disk_size} GB SSD".format( | ||||
|             cpu=cpu, | ||||
|         amount_to_be_charged = get_vm_price(cpu=cpu, memory=memory, | ||||
|                                             disk_size=disk_size) | ||||
|         plan_name = StripeUtils.get_stripe_plan_name(cpu=cpu, | ||||
|                                                      memory=memory, | ||||
|                                                      disk_size=disk_size) | ||||
| 
 | ||||
|         stripe_plan_id = StripeUtils.get_stripe_plan_id(cpu=cpu, | ||||
|                                                         ram=memory, | ||||
|                                                         ssd=disk_size, | ||||
|  | @ -810,7 +817,8 @@ class OrdersHostingDetailView(LoginRequiredMixin, | |||
|             'status': True, | ||||
|             'redirect': reverse('hosting:virtual_machines'), | ||||
|             'msg_title': str(_('Thank you for the order.')), | ||||
|             'msg_body': str(_('Your VM will be up and running in a few moments.' | ||||
|             'msg_body': str( | ||||
|                 _('Your VM will be up and running in a few moments.' | ||||
|                   ' We will send you a confirmation email as soon as' | ||||
|                   ' it is ready.')) | ||||
|         } | ||||
|  | @ -904,7 +912,6 @@ class CreateVirtualMachinesView(LoginRequiredMixin, View): | |||
|         memory_field = forms.IntegerField(validators=[self.validate_memory]) | ||||
|         storage = request.POST.get('storage') | ||||
|         storage_field = forms.IntegerField(validators=[self.validate_storage]) | ||||
|         price = request.POST.get('total') | ||||
|         template_id = int(request.POST.get('config')) | ||||
|         template = VMTemplate.objects.filter( | ||||
|             opennebula_vm_template_id=template_id).first() | ||||
|  | @ -936,7 +943,8 @@ class CreateVirtualMachinesView(LoginRequiredMixin, View): | |||
|                                  extra_tags='storage') | ||||
|             return HttpResponseRedirect( | ||||
|                 reverse('datacenterlight:index') + "#order_form") | ||||
| 
 | ||||
|         price = get_vm_price(cpu=cores, memory=memory, | ||||
|                              disk_size=storage) | ||||
|         specs = { | ||||
|             'cpu': cores, | ||||
|             'memory': memory, | ||||
|  | @ -1042,6 +1050,10 @@ class VirtualMachineView(LoginRequiredMixin, View): | |||
|                 except WrongIdError: | ||||
|                     response['status'] = True | ||||
|                     response['text'] = ugettext('Terminated') | ||||
|                     vm_detail_obj = VMDetail.objects.filter( | ||||
|                         vm_id=opennebula_vm_id).first() | ||||
|                     vm_detail_obj.terminated_at = datetime.utcnow() | ||||
|                     vm_detail_obj.save() | ||||
|                     break | ||||
|                 except BaseException: | ||||
|                     break | ||||
|  |  | |||
|  | @ -1,4 +1,10 @@ | |||
| from hosting.models import UserHostingKey | ||||
| import logging | ||||
| from oca.pool import WrongIdError | ||||
| 
 | ||||
| from hosting.models import UserHostingKey, VMDetail | ||||
| from opennebula_api.serializers import VirtualMachineSerializer | ||||
| 
 | ||||
| logger = logging.getLogger(__name__) | ||||
| 
 | ||||
| 
 | ||||
| def get_all_public_keys(customer): | ||||
|  | @ -9,3 +15,48 @@ def get_all_public_keys(customer): | |||
|     """ | ||||
|     return UserHostingKey.objects.filter(user_id=customer.id).values_list( | ||||
|         "public_key", flat=True) | ||||
| 
 | ||||
| 
 | ||||
| def get_or_create_vm_detail(user, manager, vm_id): | ||||
|     """ | ||||
|     Returns VMDetail object related to given vm_id. Creates the object | ||||
|      if it does not exist | ||||
| 
 | ||||
|     :param vm_id: The ID of the VM which should be greater than 0. | ||||
|     :param user: The CustomUser object that owns this VM | ||||
|     :param manager: The OpenNebulaManager object | ||||
|     :return: The VMDetail object. None if vm_id is less than or equal to 0. | ||||
|     Also, for the cases where the VMDetail does not exist and we can not | ||||
|     fetch data about the VM from OpenNebula, the function returns None | ||||
|     """ | ||||
|     if vm_id <= 0: | ||||
|         return None | ||||
|     try: | ||||
|         vm_detail_obj = VMDetail.objects.get(vm_id=vm_id) | ||||
|     except VMDetail.DoesNotExist: | ||||
|         try: | ||||
|             vm_obj = manager.get_vm(vm_id) | ||||
|         except (WrongIdError, ConnectionRefusedError) as e: | ||||
|             logger.error(str(e)) | ||||
|             return None | ||||
|         vm = VirtualMachineSerializer(vm_obj).data | ||||
|         vm_detail_obj = VMDetail.objects.create( | ||||
|             user=user, vm_id=vm_id, disk_size=vm['disk_size'], | ||||
|             cores=vm['cores'], memory=vm['memory'], | ||||
|             configuration=vm['configuration'], ipv4=vm['ipv4'], | ||||
|             ipv6=vm['ipv6'] | ||||
|         ) | ||||
|     return vm_detail_obj | ||||
| 
 | ||||
| 
 | ||||
| def get_vm_price(cpu, memory, disk_size): | ||||
|     """ | ||||
|     A helper function that computes price of a VM from given cpu, ram and | ||||
|     ssd parameters | ||||
| 
 | ||||
|     :param cpu: Number of cores of the VM | ||||
|     :param memory: RAM of the VM | ||||
|     :param disk_size: Disk space of the VM | ||||
|     :return: The price of the VM | ||||
|     """ | ||||
|     return (cpu * 5) + (memory * 2) + (disk_size * 0.6) | ||||
|  |  | |||
|  | @ -238,7 +238,7 @@ class StripeUtils(object): | |||
|     @staticmethod | ||||
|     def get_stripe_plan_id(cpu, ram, ssd, version, app='dcl', hdd=None): | ||||
|         """ | ||||
|         Returns the stripe plan id string of the form | ||||
|         Returns the Stripe plan id string of the form | ||||
|         `dcl-v1-cpu-2-ram-5gb-ssd-10gb` based on the input parameters | ||||
| 
 | ||||
|         :param cpu: The number of cores | ||||
|  | @ -256,7 +256,19 @@ class StripeUtils(object): | |||
|         if hdd is not None: | ||||
|             dcl_plan_string = '{dcl_plan_string}-hdd-{hdd}gb'.format( | ||||
|                 dcl_plan_string=dcl_plan_string, hdd=hdd) | ||||
|         stripe_plan_id_string = '{app}-v{version}-{plan}'.format(app=app, | ||||
|         stripe_plan_id_string = '{app}-v{version}-{plan}'.format( | ||||
|             app=app, | ||||
|             version=version, | ||||
|             plan=dcl_plan_string) | ||||
|         return stripe_plan_id_string | ||||
| 
 | ||||
|     @staticmethod | ||||
|     def get_stripe_plan_name(cpu, memory, disk_size): | ||||
|         """ | ||||
|         Returns the Stripe plan name | ||||
|         :return: | ||||
|         """ | ||||
|         return "{cpu} Cores, {memory} GB RAM, {disk_size} GB SSD".format( | ||||
|             cpu=cpu, | ||||
|             memory=memory, | ||||
|             disk_size=disk_size) | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue