Add authentication
This commit is contained in:
		
					parent
					
						
							
								ef00d676f5
							
						
					
				
			
			
				commit
				
					
						358d70570a
					
				
			
		
					 7 changed files with 149 additions and 30 deletions
				
			
		
							
								
								
									
										20
									
								
								opennebula_api/migrations/0002_auto_20170511_0246.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								opennebula_api/migrations/0002_auto_20170511_0246.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,20 @@ | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | # Generated by Django 1.9.4 on 2017-05-11 02:46 | ||||||
|  | from __future__ import unicode_literals | ||||||
|  | 
 | ||||||
|  | from django.db import migrations | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  | 
 | ||||||
|  |     dependencies = [ | ||||||
|  |         ('opennebula_api', '0001_initial'), | ||||||
|  |     ] | ||||||
|  | 
 | ||||||
|  |     operations = [ | ||||||
|  |         migrations.RenameField( | ||||||
|  |             model_name='virtualmachine', | ||||||
|  |             old_name='template', | ||||||
|  |             new_name='vm_template', | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
							
								
								
									
										24
									
								
								opennebula_api/migrations/0003_virtualmachine_owner.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								opennebula_api/migrations/0003_virtualmachine_owner.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,24 @@ | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | # Generated by Django 1.9.4 on 2017-05-11 09:06 | ||||||
|  | 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), | ||||||
|  |         ('opennebula_api', '0002_auto_20170511_0246'), | ||||||
|  |     ] | ||||||
|  | 
 | ||||||
|  |     operations = [ | ||||||
|  |         migrations.AddField( | ||||||
|  |             model_name='virtualmachine', | ||||||
|  |             name='owner', | ||||||
|  |             field=models.ForeignKey(default=1, on_delete=django.db.models.deletion.CASCADE, related_name='virtual_machines', to=settings.AUTH_USER_MODEL), | ||||||
|  |             preserve_default=False, | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
|  | @ -3,6 +3,9 @@ import socket | ||||||
| 
 | 
 | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|  | from django.utils.functional import cached_property | ||||||
|  | 
 | ||||||
|  | from membership.models import CustomUser | ||||||
| 
 | 
 | ||||||
| from oca.pool import WrongNameError | from oca.pool import WrongNameError | ||||||
| 
 | 
 | ||||||
|  | @ -14,8 +17,13 @@ class VirtualMachineTemplate(models.Model): | ||||||
|     core_price = models.FloatField() |     core_price = models.FloatField() | ||||||
|     disk_size_price = models.FloatField() |     disk_size_price = models.FloatField() | ||||||
| 
 | 
 | ||||||
|  |     @cached_property | ||||||
|  |     def template(self): | ||||||
|  |         template = OpenNebulaManager()._get_template(self.opennebula_id) | ||||||
|  |         return template | ||||||
|  | 
 | ||||||
|     def calculate_price(self): |     def calculate_price(self): | ||||||
|         template = OpenNebulaManager()._get_template(self.opennebula_id).template |         template = self.template.template | ||||||
| 
 | 
 | ||||||
|         price = int(template.vcpu) * self.core_price |         price = int(template.vcpu) * self.core_price | ||||||
|         price += int(template.memory) / 1024 * self.memory_price |         price += int(template.memory) / 1024 * self.memory_price | ||||||
|  | @ -28,17 +36,17 @@ class VirtualMachineTemplate(models.Model): | ||||||
| 
 | 
 | ||||||
|     def get_name(self): |     def get_name(self): | ||||||
| 
 | 
 | ||||||
|         template = OpenNebulaManager()._get_template(template_id=self.opennebula_id) |         template = self.template | ||||||
|         return template.name |         return template.name | ||||||
| 
 | 
 | ||||||
|     def get_cores(self): |     def get_cores(self): | ||||||
| 
 | 
 | ||||||
|         template = OpenNebulaManager()._get_template(template_id=self.opennebula_id).template |         template = self.template.template | ||||||
|         return int(template.vcpu) |         return int(template.vcpu) | ||||||
|      |      | ||||||
|     def get_disk_size(self): |     def get_disk_size(self): | ||||||
| 
 | 
 | ||||||
|         template = OpenNebulaManager()._get_template(template_id=self.opennebula_id).template |         template = self.template.template | ||||||
|         disk_size = 0 |         disk_size = 0 | ||||||
|         for disk in template.disks: |         for disk in template.disks: | ||||||
|             disk_size += int(disk.size) |             disk_size += int(disk.size) | ||||||
|  | @ -46,12 +54,13 @@ class VirtualMachineTemplate(models.Model): | ||||||
| 
 | 
 | ||||||
|     def get_memory(self): |     def get_memory(self): | ||||||
| 
 | 
 | ||||||
|         template = OpenNebulaManager()._get_template(template_id=self.opennebula_id).template |         template = self.template.template | ||||||
|         return int(template.memory) / 1024 |         return int(template.memory) / 1024 | ||||||
| 
 | 
 | ||||||
| class VirtualMachine(models.Model): | class VirtualMachine(models.Model): | ||||||
|     """This class represents an opennebula virtual machine.""" |     """This class represents an opennebula virtual machine.""" | ||||||
|     opennebula_id = models.IntegerField() |     opennebula_id = models.IntegerField() | ||||||
|  |     owner = models.ForeignKey(CustomUser, related_name='virtual_machines') | ||||||
|     vm_template = models.ForeignKey(VirtualMachineTemplate) |     vm_template = models.ForeignKey(VirtualMachineTemplate) | ||||||
| 
 | 
 | ||||||
|     VM_STATE = { |     VM_STATE = { | ||||||
|  | @ -68,10 +77,17 @@ class VirtualMachine(models.Model): | ||||||
|         '11': 'CLONING_FAILURE', |         '11': 'CLONING_FAILURE', | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     @cached_property | ||||||
|  |     def vm(self): | ||||||
|  |         manager = OpenNebulaManager(email=self.owner.email, | ||||||
|  |                 password=self.owner.password[0:20],  | ||||||
|  |                                     create_user=True) | ||||||
|  |         vm = manager._get_vm(self.opennebula_id) | ||||||
|  |         return vm | ||||||
|  | 
 | ||||||
|     def get_name(self): |     def get_name(self): | ||||||
|              |              | ||||||
|         vm = OpenNebulaManager()._get_vm(vm_id=self.opennebula_id) |         return self.vm.name | ||||||
|         return vm.name |  | ||||||
| 
 | 
 | ||||||
|     def get_cores(self): |     def get_cores(self): | ||||||
| 
 | 
 | ||||||
|  | @ -87,21 +103,18 @@ class VirtualMachine(models.Model): | ||||||
| 
 | 
 | ||||||
|     def get_id(self): |     def get_id(self): | ||||||
| 
 | 
 | ||||||
|         vm = OpenNebulaManager()._get_vm(vm_id=self.opennebula_id) |         return self.vm.id | ||||||
|         return vm.id |  | ||||||
| 
 | 
 | ||||||
|     def get_ip(self): |     def get_ip(self): | ||||||
| 
 | 
 | ||||||
|         vm = OpenNebulaManager()._get_vm(vm_id=self.opennebula_id) |  | ||||||
|         try: |         try: | ||||||
|             return vm.user_template.ungleich_public_ip |             return self.vm.user_template.ungleich_public_ip | ||||||
|         except AttributeError: |         except AttributeError: | ||||||
|             return '-' |             return '-' | ||||||
| 
 | 
 | ||||||
|     def get_state(self): |     def get_state(self): | ||||||
| 
 | 
 | ||||||
|         vm = OpenNebulaManager()._get_vm(vm_id=self.opennebula_id) |         return self.VM_STATE.get(str(self.vm.state)) | ||||||
|         return self.VM_STATE.get(str(vm.state)) |  | ||||||
| 
 | 
 | ||||||
|     def get_price(self): |     def get_price(self): | ||||||
|         return self.vm_template.calculate_price() |         return self.vm_template.calculate_price() | ||||||
|  | @ -175,8 +188,13 @@ class OpenNebulaManager(): | ||||||
| 
 | 
 | ||||||
|     def _get_vm_pool(self): |     def _get_vm_pool(self): | ||||||
|         try: |         try: | ||||||
|            vm_pool = oca.VirtualMachinePool(self.oneadmin_client) |             vm_pool = oca.VirtualMachinePool(self.client) | ||||||
|             vm_pool.info() |             vm_pool.info() | ||||||
|  |         except AttributeError: | ||||||
|  |             print('Could not connect via client, using oneadmin instead')  | ||||||
|  |             vm_pool = oca.VirtualMachinePool(self.oneadmin_client) | ||||||
|  |             vm_pool.info(filter=-2) | ||||||
|  | 
 | ||||||
|         #TODO: Replace with logger |         #TODO: Replace with logger | ||||||
|         except ConnectionRefusedError: |         except ConnectionRefusedError: | ||||||
|             print('Could not connect to host: {host} via protocol {protocol}'.format( |             print('Could not connect to host: {host} via protocol {protocol}'.format( | ||||||
|  | @ -189,16 +207,13 @@ class OpenNebulaManager(): | ||||||
|     |     | ||||||
|     def _get_vm(self, vm_id): |     def _get_vm(self, vm_id): | ||||||
|         vm_pool = self._get_vm_pool() |         vm_pool = self._get_vm_pool() | ||||||
|         # Get virtual machines from all users  |  | ||||||
|         vm_pool.info(filter=-2) |  | ||||||
|         return vm_pool.get_by_id(vm_id) |         return vm_pool.get_by_id(vm_id) | ||||||
| 
 | 
 | ||||||
|     def create_vm(self, template_id): |     def create_vm(self, template_id): | ||||||
|         template_pool = self._get_template_pool() |         vm_id = self.oneadmin_client.call( | ||||||
| 
 |                     oca.VmTemplate.METHODS['instantiate'], | ||||||
|         template = template_pool.get_by_id(template_id) |                     template_id, | ||||||
| 
 |                 ) | ||||||
|         vm_id = template.instantiate() |  | ||||||
|         try: |         try: | ||||||
|             self.oneadmin_client.call( |             self.oneadmin_client.call( | ||||||
|                 oca.VirtualMachine.METHODS['chown'], |                 oca.VirtualMachine.METHODS['chown'], | ||||||
|  | @ -207,7 +222,7 @@ class OpenNebulaManager(): | ||||||
|                 self.opennebula_user.group_ids[0] |                 self.opennebula_user.group_ids[0] | ||||||
|             ) |             ) | ||||||
|         except AttributeError: |         except AttributeError: | ||||||
|             pass |             print('Could not change owner, opennebula_user is not set.') | ||||||
|         return vm_id |         return vm_id | ||||||
| 
 | 
 | ||||||
|     def delete_vm(self, vm_id): |     def delete_vm(self, vm_id): | ||||||
|  | @ -269,5 +284,3 @@ class OpenNebulaManager(): | ||||||
|     def delete_template(self, template_id): |     def delete_template(self, template_id): | ||||||
|         self.oneadmin_client.call(oca.VmTemplate.METHODS['delete'], template_id, False) |         self.oneadmin_client.call(oca.VmTemplate.METHODS['delete'], template_id, False) | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     |  | ||||||
|  |  | ||||||
							
								
								
									
										10
									
								
								opennebula_api/permissions.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								opennebula_api/permissions.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | ||||||
|  | from rest_framework.permissions import BasePermission | ||||||
|  | 
 | ||||||
|  | from .models import VirtualMachine | ||||||
|  | 
 | ||||||
|  | class IsOwner(BasePermission): | ||||||
|  | 
 | ||||||
|  |     def has_object_permission(self, request, view, obj): | ||||||
|  |         if isinstance(obj, VirtualMachine): | ||||||
|  |             return obj.owner == request.user | ||||||
|  |         return obj.owner == request.user | ||||||
|  | @ -60,6 +60,8 @@ class VirtualMachineSerializer(serializers.ModelSerializer): | ||||||
|     state       = serializers.CharField(read_only=True, source='get_state') |     state       = serializers.CharField(read_only=True, source='get_state') | ||||||
|     price       = serializers.FloatField(read_only=True, source='get_price') |     price       = serializers.FloatField(read_only=True, source='get_price') | ||||||
| 
 | 
 | ||||||
|  |     owner = serializers.ReadOnlyField(source='owner.name') | ||||||
|  | 
 | ||||||
|     vm_template = VirtualMachineTemplateSerializer(read_only=True) |     vm_template = VirtualMachineTemplateSerializer(read_only=True) | ||||||
| 
 | 
 | ||||||
|     vm_template_id = TemplatePrimaryKeyRelatedField( |     vm_template_id = TemplatePrimaryKeyRelatedField( | ||||||
|  | @ -71,15 +73,18 @@ class VirtualMachineSerializer(serializers.ModelSerializer): | ||||||
|         model = VirtualMachine |         model = VirtualMachine | ||||||
|         fields = ('id', 'opennebula_id', 'vm_template', 'vm_template_id', 'cores', 'name', |         fields = ('id', 'opennebula_id', 'vm_template', 'vm_template_id', 'cores', 'name', | ||||||
|                 'disk_size', 'memory', 'ip', 'deploy_id', 'state', 'vm_id', |                 'disk_size', 'memory', 'ip', 'deploy_id', 'state', 'vm_id', | ||||||
|                 'price') |                 'price', 'owner') | ||||||
|         read_only_fields = ('opennebula_id', ) |         read_only_fields = ('opennebula_id', ) | ||||||
| 
 | 
 | ||||||
|     def validate(self, data): |     def validate(self, data): | ||||||
|         # Create the opennebula model |         # Create the opennebula model | ||||||
|         manager = OpenNebulaManager(create_user = False) |  | ||||||
|          |          | ||||||
|         try: |         try: | ||||||
|             template_id = data['vm_template'].opennebula_id |             template_id = data['vm_template'].opennebula_id | ||||||
|  |             owner = self.context.get('request').user | ||||||
|  |             manager = OpenNebulaManager(email=owner.email, | ||||||
|  |                                         password=owner.password[0:20], | ||||||
|  |                                         create_user = True) | ||||||
|             opennebula_id = manager.create_vm(template_id) |             opennebula_id = manager.create_vm(template_id) | ||||||
|             data.update({'opennebula_id':opennebula_id}) |             data.update({'opennebula_id':opennebula_id}) | ||||||
|         except OpenNebulaException as err: |         except OpenNebulaException as err: | ||||||
|  | @ -90,4 +95,17 @@ class VirtualMachineSerializer(serializers.ModelSerializer): | ||||||
|     def create(self, validated_data): |     def create(self, validated_data): | ||||||
|         return VirtualMachine.objects.create(**validated_data) |         return VirtualMachine.objects.create(**validated_data) | ||||||
| 
 | 
 | ||||||
|  |     def update(self, instance, validated_data): | ||||||
|  |         pass | ||||||
|  | 
 | ||||||
|  |     def delete(self, instance, validated_data): | ||||||
|  |         try: | ||||||
|  |             owner = instance.owner | ||||||
|  |             manager = OpenNebulaManager(email=owner.email, | ||||||
|  |                                         password=owner.password[0:20], | ||||||
|  |                                         create_user = True) | ||||||
|  |             manager.delete_vm(template_id) | ||||||
|  |         except OpenNebulaException as err: | ||||||
|  |             raise serializers.ValidationError("OpenNebulaException occured. {0}".format(err)) | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										18
									
								
								opennebula_api/urls.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								opennebula_api/urls.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,18 @@ | ||||||
|  | from django.conf.urls import url, include | ||||||
|  | from rest_framework.urlpatterns import format_suffix_patterns | ||||||
|  | from .views import TemplateCreateView, TemplateDetailsView,\ | ||||||
|  |                    VmCreateView, VmDetailsView | ||||||
|  | 
 | ||||||
|  | urlpatterns = { | ||||||
|  |     url(r'^auth/', include('rest_framework.urls', namespace='rest_framework')), | ||||||
|  | 
 | ||||||
|  |     url(r'^templates/$', TemplateCreateView.as_view(), name="template_create"), | ||||||
|  |     url(r'^templates/(?P<pk>[0-9]+)/$', TemplateDetailsView.as_view(), | ||||||
|  |         name="templates_details"), | ||||||
|  |      | ||||||
|  |     url(r'^vms/$', VmCreateView.as_view(), name="vm_create"), | ||||||
|  |     url(r'^vms/(?P<pk>[0-9]+)/$', VmDetailsView.as_view(), | ||||||
|  |         name="vm_details"), | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | urlpatterns = format_suffix_patterns(urlpatterns) | ||||||
|  | @ -1,13 +1,26 @@ | ||||||
| from rest_framework import generics | from rest_framework import generics | ||||||
|  | from rest_framework import permissions | ||||||
|  | 
 | ||||||
|  | from django.contrib.auth.mixins import LoginRequiredMixin | ||||||
|  | from django.contrib.auth import authenticate, login | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | from utils.views import LoginViewMixin | ||||||
|  | from membership.models import CustomUser, StripeCustomer | ||||||
|  | from guardian.mixins import PermissionRequiredMixin | ||||||
| 
 | 
 | ||||||
| from .serializers import VirtualMachineTemplateSerializer, \ | from .serializers import VirtualMachineTemplateSerializer, \ | ||||||
|                          VirtualMachineSerializer |                          VirtualMachineSerializer | ||||||
| from .models import VirtualMachineTemplate, VirtualMachine, OpenNebulaManager | from .models import VirtualMachineTemplate, VirtualMachine, OpenNebulaManager | ||||||
|  | from .permissions import IsOwner | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| class TemplateCreateView(generics.ListCreateAPIView): | class TemplateCreateView(generics.ListCreateAPIView): | ||||||
|     """This class defines the create behavior of our rest api.""" |     """This class handles the GET and POST requests.""" | ||||||
|  | 
 | ||||||
|     queryset = VirtualMachineTemplate.objects.all() |     queryset = VirtualMachineTemplate.objects.all() | ||||||
|     serializer_class = VirtualMachineTemplateSerializer |     serializer_class = VirtualMachineTemplateSerializer | ||||||
|  |     permission_classes = (permissions.IsAuthenticated, permissions.IsAdminUser) | ||||||
| 
 | 
 | ||||||
|     def perform_create(self, serializer): |     def perform_create(self, serializer): | ||||||
|         """Save the post data when creating a new template.""" |         """Save the post data when creating a new template.""" | ||||||
|  | @ -18,18 +31,21 @@ class TemplateDetailsView(generics.RetrieveUpdateDestroyAPIView): | ||||||
| 
 | 
 | ||||||
|     queryset = VirtualMachineTemplate.objects.all() |     queryset = VirtualMachineTemplate.objects.all() | ||||||
|     serializer_class = VirtualMachineTemplateSerializer |     serializer_class = VirtualMachineTemplateSerializer | ||||||
|  |     permission_classes = (permissions.IsAuthenticated) | ||||||
| 
 | 
 | ||||||
| class VmCreateView(generics.ListCreateAPIView): | class VmCreateView(generics.ListCreateAPIView): | ||||||
|     """This class defines the create behavior of our rest api.""" |     """This class handles the GET and POST requests.""" | ||||||
|     queryset = VirtualMachine.objects.all() |     queryset = VirtualMachine.objects.all() | ||||||
|     serializer_class = VirtualMachineSerializer |     serializer_class = VirtualMachineSerializer | ||||||
|  |     permission_classes = (permissions.IsAuthenticated, IsOwner) | ||||||
| 
 | 
 | ||||||
|     def perform_create(self, serializer): |     def perform_create(self, serializer): | ||||||
|         """Save the post data when creating a new template.""" |         """Save the post data when creating a new template.""" | ||||||
|         serializer.save() |         serializer.save(owner=self.request.user) | ||||||
| 
 | 
 | ||||||
| class VmDetailsView(generics.RetrieveUpdateDestroyAPIView): | class VmDetailsView(generics.RetrieveUpdateDestroyAPIView): | ||||||
|     """This class handles the http GET, PUT and DELETE requests.""" |     """This class handles the http GET, PUT and DELETE requests.""" | ||||||
|  |     permission_classes = (permissions.IsAuthenticated, IsOwner) | ||||||
| 
 | 
 | ||||||
|     queryset = VirtualMachine.objects.all() |     queryset = VirtualMachine.objects.all() | ||||||
|     serializer_class = VirtualMachineSerializer |     serializer_class = VirtualMachineSerializer | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue