merged master
This commit is contained in:
		
				commit
				
					
						6802e2fdbc
					
				
			
		
					 26 changed files with 294 additions and 395 deletions
				
			
		|  | @ -1,10 +1,15 @@ | |||
| Next: | ||||
| 1.2.4: 2017-10-02 | ||||
|     * #3780: [hosting] Store VM details locally | ||||
|     * #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 | ||||
|     * #3756: [dcl] Update landing calculator and billing info page | ||||
|     * Bugfix: Fix PR 493 bug that creates a new StripeCustomer for each buying of VM with the same email id | ||||
|     * #3835: [all] Forbidden (403) CSRF verification failed issue. | ||||
|     * Bugfix: [hosting] Dashboard strictly available after login | ||||
|     * #3808: [dcl] Order confirmation page redesign | ||||
| 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,7 +8,7 @@ msgid "" | |||
| msgstr "" | ||||
| "Project-Id-Version: PACKAGE VERSION\n" | ||||
| "Report-Msgid-Bugs-To: \n" | ||||
| "POT-Creation-Date: 2017-09-28 18:36+0530\n" | ||||
| "POT-Creation-Date: 2017-10-03 18:54+0530\n" | ||||
| "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" | ||||
| "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n" | ||||
| "Language-Team: LANGUAGE <LL@li.org>\n" | ||||
|  | @ -374,24 +374,6 @@ msgstr "Weiter" | |||
| msgid "Enter your credit card number" | ||||
| msgstr "Deine Kreditkartennummer" | ||||
| 
 | ||||
| msgid "Confirm Order" | ||||
| msgstr "Bestellung Bestätigen" | ||||
| 
 | ||||
| msgid "Date" | ||||
| msgstr "Datum" | ||||
| 
 | ||||
| msgid "Billed To:" | ||||
| msgstr "Rechnungsadresse" | ||||
| 
 | ||||
| msgid "Payment Method:" | ||||
| msgstr "Bezahlmethode" | ||||
| 
 | ||||
| msgid "ending in" | ||||
| msgstr "endend in" | ||||
| 
 | ||||
| msgid "Order summary" | ||||
| msgstr "Bestellungsübersicht" | ||||
| 
 | ||||
| #, python-format | ||||
| msgid "" | ||||
| "By clicking \"Place order\" this plan will charge your credit card account " | ||||
|  | @ -403,21 +385,6 @@ msgstr "" | |||
| msgid "Place order" | ||||
| msgstr "Bestellen" | ||||
| 
 | ||||
| msgid "Processing..." | ||||
| msgstr "Abarbeitung..." | ||||
| 
 | ||||
| msgid "Hold tight, we are processing your request" | ||||
| msgstr "Bitte warten - wir verbeiten Deine Anfrage gerade" | ||||
| 
 | ||||
| msgid "OK" | ||||
| msgstr "" | ||||
| 
 | ||||
| msgid "Close" | ||||
| msgstr "" | ||||
| 
 | ||||
| msgid "Some problem encountered. Please try again later." | ||||
| msgstr "Ein Problem ist aufgetreten. Bitte versuche es später noch einmal." | ||||
| 
 | ||||
| msgid "Thank you for order! Our team will contact you via email" | ||||
| msgstr "" | ||||
| "Vielen Dank für die Bestellung. Unser Team setzt sich sobald wie möglich mit " | ||||
|  | @ -511,6 +478,9 @@ msgstr "Ungültige RAM-Grösse" | |||
| msgid "Invalid storage size" | ||||
| msgstr "Ungültige Speicher-Grösse" | ||||
| 
 | ||||
| msgid "Confirm Order" | ||||
| msgstr "Bestellung Bestätigen" | ||||
| 
 | ||||
| msgid "Error." | ||||
| msgstr "" | ||||
| 
 | ||||
|  | @ -518,6 +488,8 @@ msgid "" | |||
| "There was a payment related error. On close of this popup, you will be " | ||||
| "redirected back to the payment page." | ||||
| msgstr "" | ||||
| "Es ist ein Fehler bei der Zahlung betreten. Du wirst nach dem Schliessen vom " | ||||
| "Popup zur Bezahlseite weitergeleitet." | ||||
| 
 | ||||
| msgid "Thank you for the order." | ||||
| msgstr "Danke für Deine Bestellung." | ||||
|  | @ -529,6 +501,30 @@ msgstr "" | |||
| "Deine VM ist gleich bereit. Wir senden Dir eine Bestätigungsemail, sobald Du " | ||||
| "auf sie zugreifen kannst." | ||||
| 
 | ||||
| #~ msgid "Processing..." | ||||
| #~ msgstr "Abarbeitung..." | ||||
| 
 | ||||
| #~ msgid "Hold tight, we are processing your request" | ||||
| #~ msgstr "Bitte warten - wir verbeiten Deine Anfrage gerade" | ||||
| 
 | ||||
| #~ msgid "Some problem encountered. Please try again later." | ||||
| #~ msgstr "Ein Problem ist aufgetreten. Bitte versuche es später noch einmal." | ||||
| 
 | ||||
| #~ msgid "Date" | ||||
| #~ msgstr "Datum" | ||||
| 
 | ||||
| #~ msgid "Billed To:" | ||||
| #~ msgstr "Rechnungsadresse" | ||||
| 
 | ||||
| #~ msgid "Payment Method:" | ||||
| #~ msgstr "Bezahlmethode" | ||||
| 
 | ||||
| #~ msgid "ending in" | ||||
| #~ msgstr "endend in" | ||||
| 
 | ||||
| #~ msgid "Order summary" | ||||
| #~ msgstr "Bestellungsübersicht" | ||||
| 
 | ||||
| #~ msgid "We are cutting down the costs significantly!" | ||||
| #~ msgstr "Wir sorgen dafür, dass die Kosten für Dich signifikant abnehmen" | ||||
| 
 | ||||
|  |  | |||
|  | @ -1501,6 +1501,8 @@ tech-sub-sec h2 { | |||
|     font-size: 14px; | ||||
|     font-weight: 300; | ||||
|     letter-spacing: 2px; | ||||
|     line-height: 24px; | ||||
|     display: block; | ||||
| } | ||||
| 
 | ||||
| .content-section-a { | ||||
|  |  | |||
|  | @ -51,15 +51,20 @@ def retry_task(task, exception=None): | |||
| @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, cc_details): | ||||
|                    stripe_subscription_id, cc_details): | ||||
|     logger.debug( | ||||
|         "Running create_vm_task on {}".format(current_task.request.hostname)) | ||||
|     vm_id = None | ||||
|     try: | ||||
|         final_price = specs.get('price') | ||||
|         billing_address = BillingAddress.objects.filter( | ||||
|             id=billing_address_id).first() | ||||
|         billing_address = BillingAddress( | ||||
|             cardholder_name=billing_address_data['cardholder_name'], | ||||
|             street_address=billing_address_data['street_address'], | ||||
|             city=billing_address_data['city'], | ||||
|             postal_code=billing_address_data['postal_code'], | ||||
|             country=billing_address_data['country'] | ||||
|         ) | ||||
|         billing_address.save() | ||||
|         customer = StripeCustomer.objects.filter(id=stripe_customer_id).first() | ||||
| 
 | ||||
|         if 'pass' in user: | ||||
|  | @ -112,8 +117,7 @@ def create_vm_task(self, vm_template_id, user, specs, template, | |||
|             billing_address_user_form.save() | ||||
| 
 | ||||
|         # Associate an order with a stripe subscription | ||||
|         charge_object = DictDotLookup(charge) | ||||
|         order.set_subscription_id(charge_object, cc_details) | ||||
|         order.set_subscription_id(stripe_subscription_id, cc_details) | ||||
| 
 | ||||
|         # If the Stripe payment succeeds, set order status approved | ||||
|         order.set_approved() | ||||
|  | @ -185,7 +189,8 @@ def create_vm_task(self, vm_template_id, user, specs, template, | |||
|                             public_keys] | ||||
|                     if len(keys) > 0: | ||||
|                         logger.debug( | ||||
|                             "Calling configure on {host} for {num_keys} keys".format( | ||||
|                             "Calling configure on {host} for " | ||||
|                             "{num_keys} keys".format( | ||||
|                                 host=new_host, num_keys=len(keys))) | ||||
|                         # Let's delay the task by 75 seconds to be sure | ||||
|                         # that we run the cdist configure after the host | ||||
|  | @ -214,32 +219,3 @@ def create_vm_task(self, vm_template_id, user, specs, template, | |||
|             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()) | ||||
|  |  | |||
|  | @ -16,7 +16,7 @@ | |||
|                         <hr class="intro-divider"> | ||||
|                         <ul class="list-inline intro-social-buttons"> | ||||
|                             <li> | ||||
|                                 <a class="btn btn-default btn-lg btn-transparent url" href="#how"><i class="#Services"></i> <span class="network-name">{% trans "Highlights" %}</span></a> | ||||
|                                 <a class="btn btn-default btn-lg btn-transparent url" href="#how"><span class="network-name">{% trans "Highlights" %}</span></a> | ||||
|                             </li> | ||||
|                             <li> | ||||
|                                 <a class="btn btn-primary btn-lg page-scroll url" href="#price"><span class="network-name">{% trans "I want it!" %}</span></a> | ||||
|  |  | |||
|  | @ -1,143 +1,24 @@ | |||
| {% extends "hosting/base_short.html" %} | ||||
| {% load staticfiles bootstrap3 %} | ||||
| {% extends "hosting/order_detail.html" %} | ||||
| {% load i18n %} | ||||
| {% load custom_tags %} | ||||
| 
 | ||||
| {% block navbar %} | ||||
|     {% include "datacenterlight/includes/_navbar.html" %} | ||||
| {% endblock navbar %} | ||||
| 
 | ||||
| {% block content %} | ||||
| {% block submit_btn %} | ||||
|   <form id="virtual_machine_create_form" action="" method="POST"> | ||||
|       {% csrf_token %} | ||||
|       <div class="row"> | ||||
|           <div class="col-sm-8"> | ||||
|               <div class="dcl-place-order-text">{% blocktrans with vm_price=request.session.specs.price %}By clicking "Place order" this plan will charge your credit card account with the fee of {{ vm_price }}CHF/month{% endblocktrans %}.</div> | ||||
|           </div> | ||||
|           <div class="col-sm-4 order-confirm-btn text-right"> | ||||
|               <button class="btn choice-btn" id="btn-create-vm" | ||||
|                       data-toggle="modal" data-target="#createvm-modal"> | ||||
| 				{% trans "Place order" %} | ||||
|               </button> | ||||
|           </div> | ||||
|       </div> | ||||
|   </form> | ||||
| {% endblock submit_btn %} | ||||
| 
 | ||||
| <div class="order-detail-container"> | ||||
|    {% if messages %} | ||||
|         <div class="row"> | ||||
|             <div class="col-xs-12 col-md-8 col-md-offset-2"> | ||||
|                 <br/> | ||||
|                 <div class="alert alert-warning"> | ||||
|                     {% for message in messages %} | ||||
|                     <span>{{ message }}</span> | ||||
|                     {% endfor %} | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
|     {% endif %} | ||||
|     {% if not error %} | ||||
|         <div class="row"> | ||||
|             <div class="col-xs-12 col-md-8 col-md-offset-2"> | ||||
|                 <div class="invoice-title"> | ||||
|                     <h2>{% trans "Confirm Order"%}</h2> | ||||
|                 </div> | ||||
|                 <hr> | ||||
|                 <div class="row"> | ||||
|                     <div class="col-xs-12 col-sm-6 pull-right order-confirm-date"> | ||||
|                         <address> | ||||
|                             <strong>{% trans "Date"%}:</strong><br> | ||||
|                             <span id="order-created_at">{% now "Y-m-d H:i" %}</span><br><br> | ||||
|                         </address> | ||||
|                     </div> | ||||
|                     <div class="col-xs-12 col-sm-6"> | ||||
|                         <address> | ||||
|                         <h3><b>{% trans "Billed To:"%}</b></h3> | ||||
|                         {% with billing_address_data as billing_address %} | ||||
|                             {{billing_address.cardholder_name}}<br> {{billing_address.street_address}}, {{billing_address.postal_code}}<br> | ||||
|                             {{billing_address.city}}, {{billing_address.country}}. | ||||
|                         {% endwith %} | ||||
|                         </address> | ||||
|                     </div> | ||||
|                 </div> | ||||
|                 <div class="row"> | ||||
|                     <div class="col-xs-6"> | ||||
|                         <address> | ||||
|                             <strong>{% trans "Payment Method:"%}</strong><br> | ||||
|                                 {{cc_brand}} {% trans "ending in" %} **** {{cc_last4}}<br> | ||||
|                                 {{request.session.user.email}} | ||||
|                         </address> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div> | ||||
| 
 | ||||
|         <div class="row"> | ||||
|             <div class="col-md-8 col-md-offset-2"> | ||||
|                 <h3><b>{% trans "Order summary"%}</b></h3> | ||||
|                 <hr> | ||||
|                 <div class="content"> | ||||
|                     {% with request.session.specs as vm %} | ||||
|                         <p><b>{% trans "Cores"%}</b> <span class="pull-right">{{vm.cpu}}</span></p> | ||||
|                         <hr> | ||||
|                         <p><b>{% trans "Memory"%}</b> <span class="pull-right">{{vm.memory}} GB</span></p> | ||||
|                         <hr> | ||||
|                         <p><b>{% trans "Disk space"%}</b> <span class="pull-right">{{vm.disk_size}} GB</span></p> | ||||
|                         <hr> | ||||
|                         <p><b>{% trans "Configuration"%}</b> <span class="pull-right">{{request.session.template.name}}</span></p> | ||||
|                         <hr> | ||||
|                         <h4>{% trans "Total"%}<p class="pull-right"><b>{{vm.price}} CHF</b><span class="dcl-price-month"> /{% trans "Month" %}</span></p></h4> | ||||
|                     {% endwith %} | ||||
|                 </div> | ||||
|                 <br/> | ||||
|                 <form method="post" id="virtual_machine_create_form"> | ||||
|                     {% csrf_token %} | ||||
|                     <div class="row"> | ||||
|                         <div class="col-sm-8"> | ||||
|                             <p class="dcl-place-order-text">{% blocktrans with vm_price=request.session.specs.price %}By clicking "Place order" this plan will charge your credit card account with the fee of {{ vm_price }}CHF/month{% endblocktrans %}.</p> | ||||
|                         </div> | ||||
|                         <div class="col-sm-4 content"> | ||||
|                             <button class="btn btn-info pull-right" | ||||
|                                     id="btn-create-vm" | ||||
|                                     data-href="{% url 'hosting:order-confirmation' %}" | ||||
|                                     data-toggle="modal" | ||||
|                                     data-target="#createvm-modal"> | ||||
|                                 {% trans "Place order"%} | ||||
|                             </button> | ||||
|                         </div> | ||||
|                     </div> | ||||
|                 </form> | ||||
|             </div> | ||||
|         </div> | ||||
|     {% endif %} | ||||
| </div> | ||||
| <!-- Create VM Modal --> | ||||
| <div class="modal fade" id="createvm-modal" tabindex="-1" role="dialog" | ||||
|      aria-hidden="true" data-backdrop="static" data-keyboard="false"> | ||||
|     <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> | ||||
|                 <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="">{% 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> | ||||
|     </div> | ||||
| </div> | ||||
| <!-- / Create VM Modal --> | ||||
| 
 | ||||
| 
 | ||||
| <script type="text/javascript"> | ||||
|     {% trans "Some problem encountered. Please try again later." as err_msg %} | ||||
|     var create_vm_error_message = '{{err_msg|safe}}.'; | ||||
| 
 | ||||
|     window.onload = function () { | ||||
|         var locale_date = moment.utc(document.getElementById("order-created_at").textContent,'YYYY-MM-DD HH:mm').toDate(); | ||||
|         locale_date =  moment(locale_date).format("YYYY-MM-DD h:mm:ss a"); | ||||
|         document.getElementById('order-created_at').innerHTML = locale_date; | ||||
|     }; | ||||
| 
 | ||||
| </script> | ||||
| {%endblock%} | ||||
|  |  | |||
|  | @ -13,7 +13,6 @@ 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 | ||||
| 
 | ||||
| 
 | ||||
|  | @ -75,21 +74,12 @@ class CeleryTaskTestCase(TestCase): | |||
|             stripe_customer.stripe_id, | ||||
|             self.token) | ||||
|         card_details_dict = card_details.get('response_object') | ||||
|         billing_address = BillingAddress( | ||||
|             cardholder_name=self.customer_name, | ||||
|             postal_code='1232', | ||||
|             country='CH', | ||||
|             street_address='Monty\'s Street', | ||||
|             city='Hollywood') | ||||
|         billing_address.save() | ||||
|         billing_address_data = {'cardholder_name': self.customer_name, | ||||
|                                 'postal_code': '1231', | ||||
|                                 'country': 'CH', | ||||
|                                 'token': self.token, | ||||
|                                 'street_address': 'Monty\'s Street', | ||||
|                                 'city': 'Hollywood'} | ||||
| 
 | ||||
|         billing_address_id = billing_address.id | ||||
|         vm_template_id = template_data.get('id', 1) | ||||
| 
 | ||||
|         cpu = specs.get('cpu') | ||||
|  | @ -125,8 +115,7 @@ class CeleryTaskTestCase(TestCase): | |||
|                                           template_data, | ||||
|                                           stripe_customer.id, | ||||
|                                           billing_address_data, | ||||
|                                           billing_address_id, | ||||
|                                           stripe_subscription_obj, | ||||
|                                           stripe_subscription_obj.id, | ||||
|                                           card_details_dict) | ||||
|         new_vm_id = 0 | ||||
|         res = None | ||||
|  |  | |||
|  | @ -19,7 +19,7 @@ from hosting.forms import HostingUserLoginForm | |||
| from membership.models import CustomUser, StripeCustomer | ||||
| from opennebula_api.serializers import VMTemplateSerializer | ||||
| from utils.forms import ( | ||||
|     BillingAddressForm, BillingAddressFormSignup, UserBillingAddressForm | ||||
|     BillingAddressForm, BillingAddressFormSignup | ||||
| ) | ||||
| from utils.hosting_utils import get_vm_price | ||||
| from utils.mailer import BaseEmail | ||||
|  | @ -348,6 +348,11 @@ class PaymentOrderView(FormView): | |||
|             form_kwargs.update({ | ||||
|                 'instance': self.request.user.billing_addresses.first() | ||||
|             }) | ||||
|         if 'billing_address_data' in self.request.session: | ||||
|             billing_address_data = self.request.session['billing_address_data'] | ||||
|             form_kwargs.update({ | ||||
|                 'initial': billing_address_data | ||||
|             }) | ||||
|         return form_kwargs | ||||
| 
 | ||||
|     def get_context_data(self, **kwargs): | ||||
|  | @ -379,14 +384,40 @@ class PaymentOrderView(FormView): | |||
|                     email=this_user.get('email'), | ||||
|                     token=token) | ||||
|             else: | ||||
|                 user_email = form.cleaned_data.get('email') | ||||
|                 user_name = form.cleaned_data.get('name') | ||||
|                 this_user = { | ||||
|                     'email': form.cleaned_data.get('email'), | ||||
|                     'name': form.cleaned_data.get('name') | ||||
|                     'email': user_email, | ||||
|                     'name': user_name | ||||
|                 } | ||||
|                 customer = StripeCustomer.create_stripe_api_customer( | ||||
|                     email=this_user.get('email'), | ||||
|                     token=token, | ||||
|                     customer_name=form.cleaned_data.get('name')) | ||||
|                 try: | ||||
|                     custom_user = CustomUser.objects.get(email=user_email) | ||||
|                     customer = StripeCustomer.objects.filter( | ||||
|                         user_id=custom_user.id).first() | ||||
|                     if customer is None: | ||||
|                         logger.debug( | ||||
|                             ("User {email} is already registered with us." | ||||
|                              "But, StripeCustomer does not exist for {email}." | ||||
|                              "Hence, creating a new StripeCustomer.").format( | ||||
|                                 email=user_email | ||||
|                             ) | ||||
|                         ) | ||||
|                         customer = StripeCustomer.create_stripe_api_customer( | ||||
|                             email=user_email, | ||||
|                             token=token, | ||||
|                             customer_name=user_name) | ||||
|                 except CustomUser.DoesNotExist: | ||||
|                     logger.debug( | ||||
|                         ("StripeCustomer does not exist for {email}." | ||||
|                          "Hence, creating a new StripeCustomer.").format( | ||||
|                             email=user_email | ||||
|                         ) | ||||
|                     ) | ||||
|                     customer = StripeCustomer.create_stripe_api_customer( | ||||
|                         email=user_email, | ||||
|                         token=token, | ||||
|                         customer_name=user_name) | ||||
| 
 | ||||
|             request.session['billing_address_data'] = form.cleaned_data | ||||
|             request.session['user'] = this_user | ||||
|             # Get or create stripe customer | ||||
|  | @ -395,8 +426,10 @@ class PaymentOrderView(FormView): | |||
|                 return self.render_to_response( | ||||
|                     self.get_context_data(form=form)) | ||||
|             request.session['token'] = token | ||||
|             request.session[ | ||||
|                 'customer'] = customer.id if request.user.is_authenticated() else customer | ||||
|             if type(customer) is StripeCustomer: | ||||
|                 request.session['customer'] = customer.stripe_id | ||||
|             else: | ||||
|                 request.session['customer'] = customer | ||||
|             return HttpResponseRedirect( | ||||
|                 reverse('datacenterlight:order_confirmation')) | ||||
|         else: | ||||
|  | @ -415,14 +448,7 @@ class OrderConfirmationView(DetailView): | |||
|             return HttpResponseRedirect(reverse('datacenterlight:index')) | ||||
|         if 'token' not in request.session: | ||||
|             return HttpResponseRedirect(reverse('datacenterlight:payment')) | ||||
|         stripe_customer_id = request.session.get('customer') | ||||
|         if request.user.is_authenticated(): | ||||
|             customer = StripeCustomer.objects.filter( | ||||
|                 id=stripe_customer_id).first() | ||||
|             stripe_api_cus_id = customer.stripe_id | ||||
|         else: | ||||
|             stripe_api_cus_id = stripe_customer_id | ||||
| 
 | ||||
|         stripe_api_cus_id = request.session.get('customer') | ||||
|         stripe_utils = StripeUtils() | ||||
|         card_details = stripe_utils.get_card_details(stripe_api_cus_id, | ||||
|                                                      request.session.get( | ||||
|  | @ -437,6 +463,8 @@ class OrderConfirmationView(DetailView): | |||
|             'site_url': reverse('datacenterlight:index'), | ||||
|             'cc_last4': card_details.get('response_object').get('last4'), | ||||
|             'cc_brand': card_details.get('response_object').get('brand'), | ||||
|             'vm': request.session.get('specs'), | ||||
|             'page_header_text': _('Confirm Order'), | ||||
|             'billing_address_data': request.session.get('billing_address_data') | ||||
|         } | ||||
|         return render(request, self.template_name, context) | ||||
|  | @ -445,15 +473,8 @@ class OrderConfirmationView(DetailView): | |||
|         template = request.session.get('template') | ||||
|         specs = request.session.get('specs') | ||||
|         user = request.session.get('user') | ||||
|         stripe_customer_id = request.session.get('customer') | ||||
|         if request.user.is_authenticated(): | ||||
|             customer = StripeCustomer.objects.filter( | ||||
|                 id=stripe_customer_id).first() | ||||
|             stripe_api_cus_id = customer.stripe_id | ||||
|         else: | ||||
|             stripe_api_cus_id = stripe_customer_id | ||||
|         stripe_api_cus_id = request.session.get('customer') | ||||
|         vm_template_id = template.get('id', 1) | ||||
| 
 | ||||
|         stripe_utils = StripeUtils() | ||||
|         card_details = stripe_utils.get_card_details(stripe_api_cus_id, | ||||
|                                                      request.session.get( | ||||
|  | @ -498,8 +519,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') | ||||
|  | @ -546,10 +567,10 @@ class OrderConfirmationView(DetailView): | |||
|                                         password=password) | ||||
|                 login(request, new_user) | ||||
|         else: | ||||
|             customer = StripeCustomer.objects.filter( | ||||
|                 id=stripe_customer_id).first() | ||||
|             custom_user = customer.user | ||||
|             stripe_customer_id = customer.id | ||||
|             # We assume that if the user is here, his/her StripeCustomer | ||||
|             # object already exists | ||||
|             stripe_customer_id = request.user.stripecustomer.id | ||||
|             custom_user = request.user | ||||
| 
 | ||||
|         # Save billing address | ||||
|         billing_address_data = request.session.get('billing_address_data') | ||||
|  | @ -557,12 +578,6 @@ class OrderConfirmationView(DetailView): | |||
|         billing_address_data.update({ | ||||
|             'user': custom_user.id | ||||
|         }) | ||||
|         billing_address_user_form = UserBillingAddressForm( | ||||
|             instance=custom_user.billing_addresses.first(), | ||||
|             data=billing_address_data) | ||||
|         billing_address = billing_address_user_form.save() | ||||
|         billing_address_id = billing_address.id | ||||
|         logger.debug("billing address id = {}".format(billing_address_id)) | ||||
|         user = { | ||||
|             'name': custom_user.name, | ||||
|             'email': custom_user.email, | ||||
|  | @ -574,8 +589,7 @@ class OrderConfirmationView(DetailView): | |||
| 
 | ||||
|         create_vm_task.delay(vm_template_id, user, specs, template, | ||||
|                              stripe_customer_id, billing_address_data, | ||||
|                              billing_address_id, | ||||
|                              stripe_subscription_obj, card_details_dict) | ||||
|                              stripe_subscription_obj.id, card_details_dict) | ||||
|         for session_var in ['specs', 'template', 'billing_address', | ||||
|                             'billing_address_data', | ||||
|                             'token', 'customer']: | ||||
|  |  | |||
|  | @ -161,6 +161,8 @@ MIDDLEWARE_CLASSES = ( | |||
|     'cms.middleware.language.LanguageCookieMiddleware', | ||||
| ) | ||||
| 
 | ||||
| CSRF_FAILURE_VIEW = 'hosting.views.forbidden_view' | ||||
| 
 | ||||
| ROOT_URLCONF = 'dynamicweb.urls' | ||||
| 
 | ||||
| TEMPLATES = [ | ||||
|  |  | |||
|  | @ -8,7 +8,7 @@ msgid "" | |||
| msgstr "" | ||||
| "Project-Id-Version: PACKAGE VERSION\n" | ||||
| "Report-Msgid-Bugs-To: \n" | ||||
| "POT-Creation-Date: 2017-09-28 18:36+0530\n" | ||||
| "POT-Creation-Date: 2017-10-03 18:54+0530\n" | ||||
| "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" | ||||
| "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n" | ||||
| "Language-Team: LANGUAGE <LL@li.org>\n" | ||||
|  | @ -308,8 +308,8 @@ msgstr "Alle Benachrichtigungen" | |||
| msgid "%(page_header_text)s" | ||||
| msgstr "" | ||||
| 
 | ||||
| msgid "Invoice Date" | ||||
| msgstr "Rechnung Datum" | ||||
| msgid "Date" | ||||
| msgstr "Datum" | ||||
| 
 | ||||
| msgid "Status" | ||||
| msgstr "" | ||||
|  | @ -332,6 +332,9 @@ msgstr "Bezahlmethode" | |||
| msgid "ending in" | ||||
| msgstr "endend in" | ||||
| 
 | ||||
| msgid "Credit Card" | ||||
| msgstr "Kreditkarte" | ||||
| 
 | ||||
| msgid "Order summary" | ||||
| msgstr "Bestellungsübersicht" | ||||
| 
 | ||||
|  | @ -339,7 +342,7 @@ msgid "Product" | |||
| msgstr "Produkt" | ||||
| 
 | ||||
| msgid "Period" | ||||
| msgstr "" | ||||
| msgstr "Periode" | ||||
| 
 | ||||
| msgid "Cores" | ||||
| msgstr "Prozessorkerne" | ||||
|  | @ -385,9 +388,6 @@ msgstr "Ein Problem ist aufgetreten. Bitte versuche es später noch einmal." | |||
| msgid "Order Nr." | ||||
| msgstr "Bestellung Nr." | ||||
| 
 | ||||
| msgid "Date" | ||||
| msgstr "Datum" | ||||
| 
 | ||||
| msgid "Amount" | ||||
| msgstr "Betrag" | ||||
| 
 | ||||
|  | @ -412,9 +412,6 @@ msgstr "inkl. Mehrwertsteuer" | |||
| msgid "Billing Address" | ||||
| msgstr "Rechnungsadresse" | ||||
| 
 | ||||
| msgid "Credit Card" | ||||
| msgstr "Kreditkarte" | ||||
| 
 | ||||
| msgid "" | ||||
| "\n" | ||||
| "                                Please fill in your credit card information " | ||||
|  | @ -604,8 +601,10 @@ msgstr "Um eine neue VM zu erzeugen, klicke \"Neue VM erzeugen\"" | |||
| 
 | ||||
| #, python-format | ||||
| msgid "" | ||||
| "To access your VM, add your SSH key <a href=\"%(create_ssh_url)s\">here</a>" | ||||
| "To access your VM, <a href=\"%(create_ssh_url)s\">add your SSH key here</a>" | ||||
| msgstr "" | ||||
| "Um auf Deine VM zuzugreifen, <a href=\"%(create_ssh_url)s\">füge Deinen SSH-" | ||||
| "Key hinzu</a>" | ||||
| 
 | ||||
| msgid "login" | ||||
| msgstr "anmelden" | ||||
|  | @ -661,6 +660,16 @@ msgid "In order to create a VM, you need to create/upload your SSH KEY first." | |||
| msgstr "" | ||||
| "Um eine VM zu erstellen musst du zuerst einen SSH-Key erstellen / hochladen." | ||||
| 
 | ||||
| msgid "Error." | ||||
| msgstr "Fehler" | ||||
| 
 | ||||
| msgid "" | ||||
| "There was a payment related error. On close of this popup, you will be " | ||||
| "redirected back to the payment page." | ||||
| msgstr "" | ||||
| "Es ist ein Fehler bei der Zahlung betreten. Du wirst nach dem Schliessen vom " | ||||
| "Popup zur Bezahlseite weitergeleitet" | ||||
| 
 | ||||
| msgid "Thank you for the order." | ||||
| msgstr "Danke für Deine Bestellung." | ||||
| 
 | ||||
|  | @ -685,65 +694,29 @@ msgid "" | |||
| "contact Data Center Light Support." | ||||
| msgstr "Kontaktiere den Data Center Light Support." | ||||
| 
 | ||||
| msgid "" | ||||
| "We could not find the requested VM. Please contact Data Center Light Support." | ||||
| msgstr "" | ||||
| "Wir konnten die gesucht VM nicht finden. Kontaktiere den Data Center Light " | ||||
| "Support." | ||||
| 
 | ||||
| msgid "Error terminating VM" | ||||
| msgstr "Fehler beenden VM" | ||||
| 
 | ||||
| #, fuzzy, python-format | ||||
| #| msgid "Virtual Machine Cancellation" | ||||
| #, python-format | ||||
| msgid "Virtual Machine %(vm_name)s Cancelled" | ||||
| msgstr "VM Kündigung" | ||||
| msgstr "Virtuelle Maschine %(vm_name)s Kündigung" | ||||
| 
 | ||||
| msgid "There was an error processing your request. Please try again." | ||||
| msgstr "" | ||||
| "Es gab einen Fehler bei der Bearbeitung Deine Anfrage. Bitte versuche es " | ||||
| "noch einmal." | ||||
| 
 | ||||
| #~ msgid "My VM page" | ||||
| #~ msgstr "Meine VM page" | ||||
| 
 | ||||
| #~ msgid "" | ||||
| #~ "\n" | ||||
| #~ "You're receiving this email because you requested a password reset for " | ||||
| #~ "your user account at %(site_name)s.<br/>\n" | ||||
| #~ "Please go to the following page and choose a new password: %(base_url)s" | ||||
| #~ "%(password_reset_url)s<br/>\n" | ||||
| #~ "If you didn't request a new password, ignore this e-mail.<br/>\n" | ||||
| #~ "Thank you!\n" | ||||
| #~ msgstr "" | ||||
| #~ "\n" | ||||
| #~ "Du erhälst diese E-Mail da Du Dein Passwort für Deinen Account bei " | ||||
| #~ "%(site_name)s zurücksetzen möchtest.<br/>\n" | ||||
| #~ "Bitte folge diesem Link und wähle ein neues Passwort: %(base_url)s" | ||||
| #~ "%(password_reset_url)s Solltest Du kein neues Passwort angefordert haben, " | ||||
| #~ "dann ignoriere diese E-Mail.<br/>\n" | ||||
| #~ "Dankeschön!\n" | ||||
| 
 | ||||
| #~ msgid "" | ||||
| #~ "You're receiving this email because you requested a password reset for " | ||||
| #~ "your user account at %(site_name)s.\n" | ||||
| #~ "Please go to the following page and choose a new password: %(base_url)s" | ||||
| #~ "%(password_reset_url)s\n" | ||||
| #~ "If you didn't request a new password, ignore this e-mail.\n" | ||||
| #~ "Thank you!\n" | ||||
| #~ msgstr "" | ||||
| #~ "Du erhälst diese E-Mail da Du Dein Passwort für Deinen Account bei " | ||||
| #~ "%(site_name)s zurücksetzen möchtest.\n" | ||||
| #~ "Bitte folge diesem Link und wähle ein neues Passwort: %(base_url)s" | ||||
| #~ "%(password_reset_url)s Solltest Du kein neues Passwort angefordert haben, " | ||||
| #~ "dann ignoriere diese E-Mail.\n" | ||||
| #~ "Dankeschön!\n" | ||||
| 
 | ||||
| #~ msgid "" | ||||
| #~ "You're receiving this mail because your virtual machine [%(vm_name)s] has " | ||||
| #~ "been cancelled.\n" | ||||
| #~ "You can see your order status by clicking here\n" | ||||
| #~ "%(base_url)s%(vm_order_url)s\n" | ||||
| #~ "If you want to order a new virtual machine, you can do it by clicking " | ||||
| #~ "this link.\n" | ||||
| #~ "%(base_url)s%(my_virtual_machines_url)s\n" | ||||
| #~ msgstr "" | ||||
| #~ "Du erhälst diese E-Mail, da Deine virtuelle Maschine [%(vm_name)s] " | ||||
| #~ "gekündigt wurde.\n" | ||||
| #~ "Um Deinen Auftragsstatus zu sehen, klicke hier.\n" | ||||
| #~ "%(base_url)s%(vm_order_url)s\n" | ||||
| #~ "Falls Du eine neue virtuelle Maschine bestellen möchtest, kannst Du dies " | ||||
| #~ "tun, indem Du diesen Link klickst.\n" | ||||
| #~ "%(base_url)s%(my_virtual_machines_url)s\n" | ||||
| #~ msgid "Invoice Date" | ||||
| #~ msgstr "Rechnung Datum" | ||||
| 
 | ||||
| #~ msgid "VM %(VM_ID)s terminated successfully" | ||||
| #~ msgstr "VM %(VM_ID)s erfolgreich beendet" | ||||
|  |  | |||
|  | @ -88,19 +88,19 @@ class HostingOrder(AssignPermissionsMixin, models.Model): | |||
|         self.cc_brand = stripe_charge.source.brand | ||||
|         self.save() | ||||
| 
 | ||||
|     def set_subscription_id(self, subscription_object, cc_details): | ||||
|     def set_subscription_id(self, subscription_id, cc_details): | ||||
|         """ | ||||
|         When creating a Stripe subscription, we have subscription id. | ||||
|         We store this in the subscription_id field. | ||||
|         This method sets the subscription id from subscription_object | ||||
|         and also the last4 and credit card brands used for this order. | ||||
|         This method sets the subscription id | ||||
|         and the last4 and credit card brands used for this order. | ||||
| 
 | ||||
|         :param subscription_object: Stripe's subscription object | ||||
|         :param subscription_id: Stripe's subscription id | ||||
|         :param cc_details: A dict containing card details | ||||
|         {last4, brand} | ||||
|         :return: | ||||
|         """ | ||||
|         self.subscription_id = subscription_object.id | ||||
|         self.subscription_id = subscription_id | ||||
|         self.last4 = cc_details.get('last4') | ||||
|         self.cc_brand = cc_details.get('brand') | ||||
|         self.save() | ||||
|  |  | |||
|  | @ -656,7 +656,6 @@ a.unlink:hover { | |||
| 
 | ||||
| .card-element { | ||||
|     margin-bottom: 10px; | ||||
|     padding: 0; | ||||
| } | ||||
| 
 | ||||
| .card-element label{ | ||||
|  |  | |||
|  | @ -86,23 +86,31 @@ $(document).ready(function() { | |||
|             url: create_vm_form.attr('action'), | ||||
|             type: 'POST', | ||||
|             data: create_vm_form.serialize(), | ||||
|             init: function(){ | ||||
|                 ok_btn = $('#createvm-modal-done-btn'); | ||||
|                 close_btn = $('#createvm-modal-close-btn'); | ||||
|                 ok_btn.addClass('btn btn-success btn-ok btn-wide hide'); | ||||
|                 close_btn.addClass('btn btn-danger btn-ok btn-wide hide'); | ||||
|             }, | ||||
|             success: function (data) { | ||||
|                 fa_icon = $('.modal-icon > .fa'); | ||||
|                 modal_btn = $('#createvm-modal-done-btn'); | ||||
|                 $('#createvm-modal-title').text(data.msg_title); | ||||
|                 $('#createvm-modal-body').html(data.msg_body); | ||||
|                 modal_btn.attr('href', data.redirect) | ||||
|                     .removeClass('hide'); | ||||
|                 if (data.status === true) { | ||||
|                     fa_icon = $('.modal-icon > .fa'); | ||||
|                     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-done-btn') | ||||
|                         .attr('href', data.redirect) | ||||
|                         .removeClass('hide'); | ||||
|                 } else { | ||||
|                     fa_icon.attr('class', 'fa fa-close'); | ||||
|                     modal_btn.attr('class', '').addClass('btn btn-danger btn-ok btn-wide'); | ||||
|                 } | ||||
|             }, | ||||
|             error: function (xmlhttprequest, textstatus, message) { | ||||
|                     fa_icon = $('.modal-icon > .fa'); | ||||
|                     fa_icon.attr('class', 'fa fa-close'); | ||||
|                     if (typeof(create_vm_error_message) !== 'undefined') { | ||||
|                         $('#createvm-modal-text').text(create_vm_error_message); | ||||
|                         $('#createvm-modal-body').text(create_vm_error_message); | ||||
|                     } | ||||
|                     $('#btn-create-vm').prop('disabled', false); | ||||
|                     $('#createvm-modal-close-btn').removeClass('hide'); | ||||
|  |  | |||
							
								
								
									
										8
									
								
								hosting/templates/hosting/includes/_messages.html
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								hosting/templates/hosting/includes/_messages.html
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,8 @@ | |||
| {% if messages %} | ||||
|     <ul class="list-unstyled msg-list"> | ||||
|     {% for message in messages %} | ||||
|         <div | ||||
|                 class="alert {% if message.tags and message.tags == 'error' %} alert-danger {% else %} alert-{{message.tags}} {% endif %}">{{ message|safe }}</div> | ||||
|     {% endfor %} | ||||
|     </ul> | ||||
| {% endif %} | ||||
|  | @ -16,13 +16,7 @@ | |||
|         <div class="auth-content"> | ||||
|             <div class="intro-message auth-box"> | ||||
|                 <h2 class="section-heading">{% trans "Login"%}</h2> | ||||
|                 {% if messages %} | ||||
|                     <ul class="list-unstyled msg-list"> | ||||
|                     {% for message in messages %} | ||||
|                         <li>{{ message }}</li> | ||||
|                     {% endfor %} | ||||
|                     </ul> | ||||
|                 {% endif %} | ||||
|                 {% include  'hosting/includes/_messages.html' %} | ||||
|                 <form action="{% url 'hosting:login' %}" method="post" class="form" novalidated> | ||||
|                     {% csrf_token %} | ||||
|                     {% for field in form %} | ||||
|  |  | |||
|  | @ -17,10 +17,12 @@ | |||
|             <h1 class="dashboard-title-thin"> | ||||
|                 <img src="{% static 'hosting/img/billing.svg' %}" class="un-icon">{% blocktrans with page_header_text=page_header_text|default:"Invoice" %}{{page_header_text}}{% endblocktrans %} | ||||
|             </h1> | ||||
|             <div class="dashboard-container-options"> | ||||
|                 <button type="button" class="btn-plain btn-pdf" data-target="#order-detail{{order.pk}}"><img src="{% static 'hosting/img/icon-pdf.svg' %}" class="svg-img"></button> | ||||
|                 <button type="button" class="btn-plain btn-print"><img src="{% static 'hosting/img/icon-print.svg' %}" class="svg-img"></button> | ||||
|             </div> | ||||
|             {% if order %} | ||||
|                 <div class="dashboard-container-options"> | ||||
|                     <button type="button" class="btn-plain btn-pdf" data-target="#order-detail{{order.pk}}"><img src="{% static 'hosting/img/icon-pdf.svg' %}" class="svg-img"></button> | ||||
|                     <button type="button" class="btn-plain btn-print"><img src="{% static 'hosting/img/icon-print.svg' %}" class="svg-img"></button> | ||||
|                 </div> | ||||
|             {% endif %} | ||||
|         </div> | ||||
|         <div class="order-details"> | ||||
|             {% if order %} | ||||
|  | @ -29,7 +31,7 @@ | |||
|                 </p> | ||||
|             {% endif %} | ||||
|             <p> | ||||
|                 <strong>{% trans "Invoice Date" %}:</strong> | ||||
|                 <strong>{% trans "Date" %}:</strong> | ||||
|                 <span id="order-created_at"> | ||||
|                     {% if order %} | ||||
|                         {{order.created_at|date:'Y-m-d H:i'}} | ||||
|  | @ -80,8 +82,7 @@ | |||
|                         {{order.last4}}<br> | ||||
|                         {{user.email}} | ||||
|                     {% else %} | ||||
|                         {{cc_brand|default:'Card'}} {% trans "ending in" %} **** | ||||
|                         {{cc_last4}}<br> | ||||
|                         {{cc_brand|default:_('Credit Card')}} {% trans "ending in" %} ****{{cc_last4}}<br> | ||||
|                         {% if request.user.is_authenticated %} | ||||
|                             {{request.user.email}} | ||||
|                         {% else %} | ||||
|  | @ -94,7 +95,12 @@ | |||
|             <div> | ||||
|                 <h4>{% trans "Order summary" %}</h4> | ||||
|                 <p> | ||||
|                     <strong>{% trans "Product" %}:</strong> {{vm.name}} | ||||
|                     <strong>{% trans "Product" %}:</strong>  | ||||
|                     {% if vm.name %} | ||||
|                         {{ vm.name }} | ||||
|                     {% else %} | ||||
|                         {{ request.session.template.name }} | ||||
|                     {% endif %} | ||||
|                 </p> | ||||
|                 <div class="row"> | ||||
|                     <div class="col-sm-6"> | ||||
|  | @ -132,19 +138,21 @@ | |||
|             {% endif %} | ||||
|         </div> | ||||
|         {% if not order %} | ||||
|             <form method="post" id="virtual_machine_create_form"> | ||||
|                 {% csrf_token %} | ||||
|                 <div class="row"> | ||||
|                     <div class="col-sm-8"> | ||||
|                         <div class="dcl-place-order-text">{% blocktrans with vm_price=request.session.specs.price %}By clicking "Place order" this plan will charge your credit card account with the fee of {{ vm_price }}CHF/month{% endblocktrans %}.</div> | ||||
|             {% block submit_btn %} | ||||
|                 <form method="post" id="virtual_machine_create_form"> | ||||
|                     {% csrf_token %} | ||||
|                     <div class="row"> | ||||
|                         <div class="col-sm-8"> | ||||
|                             <div class="dcl-place-order-text">{% blocktrans with vm_price=request.session.specs.price %}By clicking "Place order" this plan will charge your credit card account with the fee of {{ vm_price }}CHF/month{% endblocktrans %}.</div> | ||||
|                         </div> | ||||
|                         <div class="col-sm-4 order-confirm-btn text-right"> | ||||
|                             <button class="btn choice-btn" id="btn-create-vm" data-href="{% url 'hosting:order-confirmation' %}" data-toggle="modal" data-target="#createvm-modal"> | ||||
|                                 {% trans "Place order" %} | ||||
|                             </button> | ||||
|                         </div> | ||||
|                     </div> | ||||
|                     <div class="col-sm-4 order-confirm-btn text-right"> | ||||
|                         <button class="btn choice-btn" id="btn-create-vm" data-href="{% url 'hosting:order-confirmation' %}" data-toggle="modal" data-target="#createvm-modal"> | ||||
|                             {% trans "Place order" %} | ||||
|                         </button> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </form> | ||||
|                 </form> | ||||
|             {% endblock submit_btn %} | ||||
|         {% endif %} | ||||
|     {% endif %} | ||||
| </div> | ||||
|  | @ -197,8 +205,10 @@ | |||
| {%endblock%} | ||||
| 
 | ||||
| {% block js_extra %} | ||||
|     <script src="https://cdnjs.cloudflare.com/ajax/libs/jspdf/1.3.5/jspdf.min.js"></script> | ||||
|     <script src="https://cdnjs.cloudflare.com/ajax/libs/html2canvas/0.4.1/html2canvas.min.js"></script> | ||||
|     <script src="{% static 'hosting/js/html2pdf.js' %}"></script> | ||||
|     <script src="{% static 'hosting/js/order.js' %}"></script> | ||||
|     {% if order %} | ||||
|         <script src="https://cdnjs.cloudflare.com/ajax/libs/jspdf/1.3.5/jspdf.min.js"></script> | ||||
|         <script src="https://cdnjs.cloudflare.com/ajax/libs/html2canvas/0.4.1/html2canvas.min.js"></script> | ||||
|         <script src="{% static 'hosting/js/html2pdf.js' %}"></script> | ||||
|         <script src="{% static 'hosting/js/order.js' %}"></script> | ||||
|     {% endif %} | ||||
| {% endblock js_extra %} | ||||
|  |  | |||
|  | @ -17,6 +17,7 @@ | |||
|             <div class="auth-content"> | ||||
|                 <div class="intro-message auth-box sign-up"> | ||||
|                     <h2  class="section-heading">{% trans "Resend activation link"%}</h2> | ||||
|                     {% include  'hosting/includes/_messages.html' %} | ||||
|                     <form action="{% url 'hosting:resend_activation_link' %}" method="post" class="form" novalidate> | ||||
|                         {% csrf_token %} | ||||
|                         {% for field in form %} | ||||
|  |  | |||
|  | @ -17,6 +17,7 @@ | |||
|             <div class="auth-content"> | ||||
|                 <div class="intro-message auth-box sign-up"> | ||||
|                     <h2  class="section-heading">{% trans "Reset your password"%}</h2> | ||||
|                     {% include  'hosting/includes/_messages.html' %} | ||||
|                     <form action="{% url 'hosting:reset_password' %}" method="post" class="form" novalidate> | ||||
|                         {% csrf_token %} | ||||
|                         {% for field in form %} | ||||
|  |  | |||
|  | @ -15,6 +15,7 @@ | |||
|         <div class="auth-content"> | ||||
|             <div class="intro-message auth-box sign-up"> | ||||
|                 <h2  class="section-heading">{% trans "Sign up"%}</h2> | ||||
|                 {% include  'hosting/includes/_messages.html' %} | ||||
|                 <form action="{% url 'hosting:signup' %}" method="post" class="form" novalidate> | ||||
|                     {% csrf_token %} | ||||
|                     {% for field in form %} | ||||
|  |  | |||
|  | @ -11,6 +11,7 @@ | |||
|     <div class="auth-center"> | ||||
|             <div class="auth-title"> | ||||
|                 <h2>{% trans "Your VM hosted in Switzerland"%}</h2> | ||||
|                 {% include  'hosting/includes/_messages.html' %} | ||||
|             </div> | ||||
|             <div class="auth-content"> | ||||
|                 <div class="intro-message auth-box sign-up"> | ||||
|  |  | |||
|  | @ -16,7 +16,7 @@ | |||
|                 <p>{% trans 'To create a new virtual machine, click "Create VM"' %} | ||||
|                 {% if show_create_ssh_key_msg %} | ||||
|                      {% url 'hosting:create_ssh_key' as create_ssh_url %} | ||||
|                      <br/>{% blocktrans %}To access your VM, add your SSH key <a href="{{create_ssh_url}}">here</a>{% endblocktrans %} | ||||
|                      <br/>{% blocktrans %}To access your VM, <a href="{{create_ssh_url}}">add your SSH key here</a>{% endblocktrans %} | ||||
|                 {% endif %} | ||||
|                 </p> | ||||
|                 <div class="text-right"> | ||||
|  |  | |||
|  | @ -60,8 +60,9 @@ CONNECTION_ERROR = "Your VMs cannot be displayed at the moment due to a \ | |||
|                     minutes." | ||||
| 
 | ||||
| 
 | ||||
| class DashboardView(View): | ||||
| class DashboardView(LoginRequiredMixin, View): | ||||
|     template_name = "hosting/dashboard.html" | ||||
|     login_url = reverse_lazy('hosting:login') | ||||
| 
 | ||||
|     def get_context_data(self, **kwargs): | ||||
|         context = {} | ||||
|  | @ -80,8 +81,6 @@ class DjangoHostingView(ProcessVMSelectionMixin, View): | |||
|         templates = OpenNebulaManager().get_templates() | ||||
|         data = VirtualMachineTemplateSerializer(templates, many=True).data | ||||
|         configuration_options = HostingPlan.get_serialized_configs() | ||||
| 
 | ||||
|         # configuration_detail = dict(VirtualMachinePlan.VM_CONFIGURATION).get(HOSTING) | ||||
|         context = { | ||||
|             'hosting': HOSTING, | ||||
|             'hosting_long': "Django", | ||||
|  | @ -134,7 +133,6 @@ class NodeJSHostingView(ProcessVMSelectionMixin, View): | |||
| 
 | ||||
|     def get_context_data(self, **kwargs): | ||||
|         HOSTING = 'nodejs' | ||||
|         # configuration_detail = dict(VirtualMachinePlan.VM_CONFIGURATION).get(HOSTING) | ||||
|         templates = OpenNebulaManager().get_templates() | ||||
|         configuration_options = HostingPlan.get_serialized_configs() | ||||
| 
 | ||||
|  | @ -249,7 +247,8 @@ class SignupValidateView(TemplateView): | |||
|                  <br />{go_back} {hurl}.'.format( | ||||
|             signup_success_message=_( | ||||
|                 'Thank you for signing up. We have sent an email to you. ' | ||||
|                 'Please follow the instructions in it to activate your account. Once activated, you can login using'), | ||||
|                 'Please follow the instructions in it to activate your ' | ||||
|                 'account. Once activated, you can login using'), | ||||
|             go_back=_('Go back to'), | ||||
|             lurl=login_url, | ||||
|             hurl=home_url | ||||
|  | @ -271,7 +270,8 @@ class SignupValidatedView(SignupValidateView): | |||
|         user = CustomUser.objects.filter( | ||||
|             validation_slug=self.kwargs['validate_slug']).first() | ||||
|         if validated: | ||||
|             message = '{account_activation_string} <br /> {login_string} {lurl}.'.format( | ||||
|             message = ('{account_activation_string} <br />' | ||||
|                        ' {login_string} {lurl}.').format( | ||||
|                 account_activation_string=_( | ||||
|                     "Your account has been activated."), | ||||
|                 login_string=_("You can now"), | ||||
|  | @ -653,10 +653,7 @@ class PaymentVMView(LoginRequiredMixin, FormView): | |||
|                 return HttpResponseRedirect( | ||||
|                     reverse('hosting:payment') + '#payment_error') | ||||
| 
 | ||||
|             # Create Billing Address | ||||
|             billing_address = form.save() | ||||
|             request.session['billing_address_data'] = billing_address_data | ||||
|             request.session['billing_address'] = billing_address.id | ||||
|             request.session['token'] = token | ||||
|             request.session['customer'] = customer.id | ||||
|             return HttpResponseRedirect("{url}?{query_params}".format( | ||||
|  | @ -704,7 +701,13 @@ class OrdersHostingDetailView(LoginRequiredMixin, | |||
|             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']) | ||||
|                 context['vm']['name'] = '{}-{}'.format( | ||||
|                     context['vm']['configuration'], context['vm']['vm_id']) | ||||
|                 context['vm']['price'] = get_vm_price( | ||||
|                     cpu=context['vm']['cores'], | ||||
|                     disk_size=context['vm']['disk_size'], | ||||
|                     memory=context['vm']['memory'] | ||||
|                 ) | ||||
|             except VMDetail.DoesNotExist: | ||||
|                 try: | ||||
|                     manager = OpenNebulaManager( | ||||
|  | @ -767,7 +770,6 @@ class OrdersHostingDetailView(LoginRequiredMixin, | |||
|         stripe_customer_id = request.session.get('customer') | ||||
|         customer = StripeCustomer.objects.filter(id=stripe_customer_id).first() | ||||
|         billing_address_data = request.session.get('billing_address_data') | ||||
|         billing_address_id = request.session.get('billing_address') | ||||
|         vm_template_id = template.get('id', 1) | ||||
| 
 | ||||
|         # Make stripe charge to a customer | ||||
|  | @ -785,8 +787,7 @@ class OrdersHostingDetailView(LoginRequiredMixin, | |||
|         cpu = specs.get('cpu') | ||||
|         memory = specs.get('memory') | ||||
|         disk_size = specs.get('disk_size') | ||||
|         amount_to_be_charged = get_vm_price(cpu=cpu, memory=memory, | ||||
|                                             disk_size=disk_size) | ||||
|         amount_to_be_charged = specs.get('price') | ||||
|         plan_name = StripeUtils.get_stripe_plan_name(cpu=cpu, | ||||
|                                                      memory=memory, | ||||
|                                                      disk_size=disk_size) | ||||
|  | @ -805,12 +806,24 @@ class OrdersHostingDetailView(LoginRequiredMixin, | |||
|                 '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') | ||||
|             return HttpResponseRedirect( | ||||
|                 reverse('hosting:payment') + '#payment_error') | ||||
|             response = { | ||||
|                 'status': False, | ||||
|                 'redirect': "{url}#{section}".format( | ||||
|                     url=reverse('hosting:payment'), | ||||
|                     section='payment_error'), | ||||
|                 'msg_title': str(_('Error.')), | ||||
|                 'msg_body': str( | ||||
|                     _('There was a payment related error.' | ||||
|                       ' On close of this popup, you will be redirected back to' | ||||
|                       ' the payment page.')) | ||||
|             } | ||||
|             return HttpResponse(json.dumps(response), | ||||
|                                 content_type="application/json") | ||||
|         user = { | ||||
|             'name': self.request.user.name, | ||||
|             'email': self.request.user.email, | ||||
|  | @ -821,8 +834,7 @@ class OrdersHostingDetailView(LoginRequiredMixin, | |||
|         } | ||||
|         create_vm_task.delay(vm_template_id, user, specs, template, | ||||
|                              stripe_customer_id, billing_address_data, | ||||
|                              billing_address_id, | ||||
|                              stripe_subscription_obj, card_details_dict) | ||||
|                              stripe_subscription_obj.id, card_details_dict) | ||||
| 
 | ||||
|         for session_var in ['specs', 'template', 'billing_address', | ||||
|                             'billing_address_data', | ||||
|  | @ -1025,6 +1037,7 @@ class VirtualMachineView(LoginRequiredMixin, View): | |||
|                 return redirect(reverse('hosting:virtual_machines')) | ||||
|         elif self.request.is_ajax(): | ||||
|             return HttpResponse() | ||||
|         context = None | ||||
|         try: | ||||
|             serializer = VirtualMachineSerializer(vm) | ||||
|             context = { | ||||
|  | @ -1034,7 +1047,11 @@ class VirtualMachineView(LoginRequiredMixin, View): | |||
|             } | ||||
|         except Exception as ex: | ||||
|             logger.debug("Exception generated {}".format(str(ex))) | ||||
|             pass | ||||
|             messages.error(self.request, | ||||
|                            _('We could not find the requested VM. Please ' | ||||
|                              'contact Data Center Light Support.') | ||||
|                            ) | ||||
|             return redirect(reverse('hosting:virtual_machines')) | ||||
| 
 | ||||
|         return render(request, self.template_name, context) | ||||
| 
 | ||||
|  | @ -1144,3 +1161,15 @@ class HostingBillDetailView(PermissionRequiredMixin, LoginRequiredMixin, | |||
|             bill.total_price += vm['price'] | ||||
|         context['vms'] = vms | ||||
|         return context | ||||
| 
 | ||||
| 
 | ||||
| def forbidden_view(request, exception=None, reason=''): | ||||
|     """ | ||||
|     Handle 403 error | ||||
|     """ | ||||
|     logger.error(str(exception) if exception else None) | ||||
|     logger.error('Reason = {reason}'.format(reason=reason)) | ||||
|     err_msg = _('There was an error processing your request. Please try ' | ||||
|                 'again.') | ||||
|     messages.add_message(request, messages.ERROR, err_msg) | ||||
|     return HttpResponseRedirect(request.get_full_path()) | ||||
|  |  | |||
|  | @ -8,7 +8,7 @@ msgid "" | |||
| msgstr "" | ||||
| "Project-Id-Version: PACKAGE VERSION\n" | ||||
| "Report-Msgid-Bugs-To: \n" | ||||
| "POT-Creation-Date: 2017-09-25 20:11+0000\n" | ||||
| "POT-Creation-Date: 2017-09-29 20:33+0000\n" | ||||
| "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" | ||||
| "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n" | ||||
| "Language-Team: LANGUAGE <LL@li.org>\n" | ||||
|  | @ -771,10 +771,13 @@ msgstr "" | |||
| msgid "Country" | ||||
| msgstr "" | ||||
| 
 | ||||
| msgid "Street Building" | ||||
| msgid "Name" | ||||
| msgstr "" | ||||
| 
 | ||||
| msgid "Name" | ||||
| msgid "Email Address" | ||||
| msgstr "" | ||||
| 
 | ||||
| msgid "Street Building" | ||||
| msgstr "" | ||||
| 
 | ||||
| msgid "Email" | ||||
|  | @ -786,9 +789,9 @@ msgstr "Telefon" | |||
| msgid "Message" | ||||
| msgstr "Nachricht" | ||||
| 
 | ||||
| msgid "An email with the activation link has been sent to your email" | ||||
| msgid "An email with the activation link has been sent to you" | ||||
| msgstr "" | ||||
| "Der Link zum Zurücksetzen deines Passwortes wurde an deine E-Mail gesendet" | ||||
| "Es wurde eine E-Mail mit dem Aktivierungslink an Dich gesendet." | ||||
| 
 | ||||
| msgid "Account Activation" | ||||
| msgstr "Accountaktivierung" | ||||
|  |  | |||
|  | @ -28,28 +28,34 @@ def handleStripeError(f): | |||
|             body = e.json_body | ||||
|             err = body['error'] | ||||
|             response.update({'error': err['message']}) | ||||
|             logger.error(str(e)) | ||||
|             return response | ||||
|         except stripe.error.RateLimitError as e: | ||||
|             response.update( | ||||
|                 {'error': "Too many requests made to the API too quickly"}) | ||||
|             return response | ||||
|         except stripe.error.InvalidRequestError as e: | ||||
|             logger.error(str(e)) | ||||
|             response.update({'error': "Invalid parameters"}) | ||||
|             return response | ||||
|         except stripe.error.AuthenticationError as e: | ||||
|             # Authentication with Stripe's API failed | ||||
|             # (maybe you changed API keys recently) | ||||
|             logger.error(str(e)) | ||||
|             response.update({'error': common_message}) | ||||
|             return response | ||||
|         except stripe.error.APIConnectionError as e: | ||||
|             logger.error(str(e)) | ||||
|             response.update({'error': common_message}) | ||||
|             return response | ||||
|         except stripe.error.StripeError as e: | ||||
|             # maybe send email | ||||
|             logger.error(str(e)) | ||||
|             response.update({'error': common_message}) | ||||
|             return response | ||||
|         except Exception as e: | ||||
|             # maybe send email | ||||
|             logger.error(str(e)) | ||||
|             response.update({'error': common_message}) | ||||
|             return response | ||||
| 
 | ||||
|  |  | |||
|  | @ -66,7 +66,7 @@ class LoginViewMixin(FormView): | |||
| 
 | ||||
| class ResendActivationLinkViewMixin(FormView): | ||||
|     success_message = _( | ||||
|         "An email with the activation link has been sent to your email") | ||||
|         "An email with the activation link has been sent to you") | ||||
| 
 | ||||
|     def generate_email_context(self, user): | ||||
|         context = { | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue