Statistics
| Branch: | Tag: | Revision:

root / snf-cyclades-app / synnefo / api / tests / servers.py @ 09b76b7e

History | View | Annotate | Download (37.7 kB)

1
# Copyright 2012 GRNET S.A. All rights reserved.
2
#
3
# Redistribution and use in source and binary forms, with or
4
# without modification, are permitted provided that the following
5
# conditions are met:
6
#
7
#   1. Redistributions of source code must retain the above
8
#      copyright notice, this list of conditions and the following
9
#      disclaimer.
10
#
11
#   2. Redistributions in binary form must reproduce the above
12
#      copyright notice, this list of conditions and the following
13
#      disclaimer in the documentation and/or other materials
14
#      provided with the distribution.
15
#
16
# THIS SOFTWARE IS PROVIDED BY GRNET S.A. ``AS IS'' AND ANY EXPRESS
17
# OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL GRNET S.A OR
20
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
23
# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
24
# AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
26
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27
# POSSIBILITY OF SUCH DAMAGE.
28
#
29
# The views and conclusions contained in the software and
30
# documentation are those of the authors and should not be
31
# interpreted as representing official policies, either expressed
32
# or implied, of GRNET S.A.
33

    
34
import json
35
from copy import deepcopy
36

    
37
from snf_django.utils.testing import (BaseAPITest, mocked_quotaholder,
38
                                      override_settings)
39
from synnefo.db.models import (VirtualMachine, VirtualMachineMetadata,
40
                               IPAddress, NetworkInterface)
41
from synnefo.db import models_factory as mfactory
42
from synnefo.logic.utils import get_rsapi_state
43
from synnefo.cyclades_settings import cyclades_services
44
from synnefo.lib.services import get_service_path
45
from synnefo.lib import join_urls
46
from django.conf import settings
47
from synnefo.logic.rapi import GanetiApiError
48

    
49
from mock import patch, Mock
50

    
51

    
52
class ComputeAPITest(BaseAPITest):
53
    def __init__(self, *args, **kwargs):
54
        super(ComputeAPITest, self).__init__(*args, **kwargs)
55
        self.compute_path = get_service_path(cyclades_services, 'compute',
56
                                             version='v2.0')
57

    
58
    def myget(self, path, *args, **kwargs):
59
        path = join_urls(self.compute_path, path)
60
        return self.get(path, *args, **kwargs)
61

    
62
    def myput(self, path, *args, **kwargs):
63
        path = join_urls(self.compute_path, path)
64
        return self.put(path, *args, **kwargs)
65

    
66
    def mypost(self, path, *args, **kwargs):
67
        path = join_urls(self.compute_path, path)
68
        return self.post(path, *args, **kwargs)
69

    
70
    def mydelete(self, path, *args, **kwargs):
71
        path = join_urls(self.compute_path, path)
72
        return self.delete(path, *args, **kwargs)
73

    
74

    
75
class ServerAPITest(ComputeAPITest):
76
    def setUp(self):
77
        self.user1 = 'user1'
78
        self.user2 = 'user2'
79
        self.vm1 = mfactory.VirtualMachineFactory(userid=self.user1)
80
        self.vm2 = mfactory.VirtualMachineFactory(userid=self.user2)
81
        self.vm3 = mfactory.VirtualMachineFactory(deleted=True,
82
                                                  userid=self.user1)
83
        self.vm4 = mfactory.VirtualMachineFactory(userid=self.user2)
84

    
85
    def test_server_list_1(self):
86
        """Test if the expected list of servers is returned."""
87
        response = self.myget('servers')
88
        self.assertSuccess(response)
89
        servers = json.loads(response.content)['servers']
90
        self.assertEqual(servers, [])
91

    
92
    def test_server_list_2(self):
93
        """Test if the expected list of servers is returned."""
94
        response = self.myget('servers', self.user1)
95
        self.assertSuccess(response)
96
        servers = json.loads(response.content)['servers']
97
        db_server = self.vm1
98
        server = servers[0]
99
        self.assertEqual(server["name"], db_server.name)
100
        self.assertEqual(server["id"], db_server.id)
101

    
102
    def test_server_list_detail(self):
103
        """Test if the servers list details are returned."""
104
        user = self.user2
105
        user_vms = {self.vm2.id: self.vm2,
106
                    self.vm4.id: self.vm4}
107

    
108
        response = self.myget('servers/detail', user)
109
        servers = json.loads(response.content)['servers']
110
        self.assertEqual(len(servers), len(user_vms))
111
        for api_vm in servers:
112
            db_vm = user_vms[api_vm['id']]
113
            self.assertEqual(api_vm['flavor']["id"], db_vm.flavor.id)
114
            self.assertEqual(api_vm['hostId'], db_vm.hostid)
115
            self.assertEqual(api_vm['id'], db_vm.id)
116
            self.assertEqual(api_vm['image']["id"], db_vm.imageid)
117
            self.assertEqual(api_vm['name'], db_vm.name)
118
            self.assertEqual(api_vm['status'], get_rsapi_state(db_vm))
119
            self.assertSuccess(response)
120

    
121
    def test_server_detail(self):
122
        """Test if a server details are returned."""
123
        db_vm = self.vm2
124
        user = self.vm2.userid
125
        ip4 = mfactory.IPv4AddressFactory(nic__machine=self.vm2)
126
        nic = ip4.nic
127
        net = ip4.network
128
        ip6 = mfactory.IPv6AddressFactory(nic=nic, network=net)
129
        nic.mac = "aa:00:11:22:33:44"
130
        nic.save()
131

    
132
        db_vm_meta = mfactory.VirtualMachineMetadataFactory(vm=db_vm)
133

    
134
        response = self.myget('servers/%d' % db_vm.id, user)
135
        server = json.loads(response.content)['server']
136

    
137
        self.assertEqual(server['flavor']["id"], db_vm.flavor.id)
138
        self.assertEqual(server['hostId'], db_vm.hostid)
139
        self.assertEqual(server['id'], db_vm.id)
140
        self.assertEqual(server['image']["id"], db_vm.imageid)
141
        self.assertEqual(server['name'], db_vm.name)
142
        self.assertEqual(server['status'], get_rsapi_state(db_vm))
143
        api_nic = server['attachments'][0]
144
        self.assertEqual(api_nic['network_id'], str(net.id))
145
        self.assertEqual(api_nic['mac_address'], nic.mac)
146
        self.assertEqual(api_nic['firewallProfile'], nic.firewall_profile)
147
        self.assertEqual(api_nic['ipv4'], ip4.address)
148
        self.assertEqual(api_nic['ipv6'], ip6.address)
149
        self.assertEqual(api_nic['OS-EXT-IPS:type'], "fixed")
150
        self.assertEqual(api_nic['id'], nic.id)
151
        api_address = server["addresses"]
152
        self.assertEqual(api_address[str(net.id)], [
153
            {"version": 4, "addr": ip4.address, "OS-EXT-IPS:type": "fixed"},
154
            {"version": 6, "addr": ip6.address, "OS-EXT-IPS:type": "fixed"}
155
        ])
156

    
157
        metadata = server['metadata']
158
        self.assertEqual(len(metadata), 1)
159
        self.assertEqual(metadata[db_vm_meta.meta_key], db_vm_meta.meta_value)
160
        self.assertSuccess(response)
161

    
162
    def test_server_fqdn(self):
163
        vm = mfactory.VirtualMachineFactory()
164
        # test no public ip
165
        with override_settings(settings,
166
                               CYCLADES_SERVERS_FQDN="vm.example.org"):
167
            response = self.myget("servers/%d" % vm.id, vm.userid)
168
            server = json.loads(response.content)['server']
169
            self.assertEqual(server["SNF:fqdn"], "")
170
        mfactory.IPv4AddressFactory(nic__machine=vm, network__public=True)
171
        with override_settings(settings,
172
                               CYCLADES_SERVERS_FQDN="vm.example.org"):
173
            response = self.myget("servers/%d" % vm.id, vm.userid)
174
            server = json.loads(response.content)['server']
175
            self.assertEqual(server["SNF:fqdn"], "vm.example.org")
176
        with override_settings(settings, CYCLADES_SERVERS_FQDN=
177
                               "snf-%(id)s.vm.example.org"):
178
            response = self.myget("servers/%d" % vm.id, vm.userid)
179
            server = json.loads(response.content)['server']
180
            self.assertEqual(server["SNF:fqdn"],
181
                             "snf-%d.vm.example.org" % vm.id)
182
        with override_settings(settings,
183
                               CYCLADES_SERVERS_FQDN=
184
                               "snf-%(id)s.vm-%(id)s.example.org"):
185
            response = self.myget("servers/%d" % vm.id, vm.userid)
186
            server = json.loads(response.content)['server']
187
            self.assertEqual(server["SNF:fqdn"], "snf-%d.vm-%d.example.org" %
188
                             (vm.id, vm.id))
189

    
190
        vm = mfactory.VirtualMachineFactory()
191
        # No setting, no NICs
192
        with override_settings(settings,
193
                               CYCLADES_SERVERS_FQDN=None):
194
            response = self.myget("servers/%d" % vm.id, vm.userid)
195
            server = json.loads(response.content)['server']
196
            self.assertEqual(server["SNF:fqdn"], "")
197

    
198
        # IPv6 NIC
199
        ipv6_address = mfactory.IPv6AddressFactory(nic__machine=vm,
200
                                                   network__public=True)
201
        with override_settings(settings,
202
                               CYCLADES_SERVERS_FQDN=None):
203
            response = self.myget("servers/%d" % vm.id, vm.userid)
204
            server = json.loads(response.content)['server']
205
            self.assertEqual(server["SNF:fqdn"], ipv6_address.address)
206

    
207
        # IPv4 NIC
208
        ipv4_address = mfactory.IPv4AddressFactory(nic__machine=vm,
209
                                                   network__public=True)
210
        with override_settings(settings,
211
                               CYCLADES_SERVERS_FQDN=None):
212
            response = self.myget("servers/%d" % vm.id, vm.userid)
213
            server = json.loads(response.content)['server']
214
            self.assertEqual(server["SNF:fqdn"], ipv4_address.address)
215

    
216
    def test_server_port_forwarding(self):
217
        vm = mfactory.VirtualMachineFactory()
218
        # test None if the server has no public IP
219
        ports = {
220
            22: ("foo", 61000),
221
            80: lambda ip, id, fqdn, user: ("bar", 61001)}
222
        with override_settings(settings,
223
                               CYCLADES_PORT_FORWARDING=ports):
224
            response = self.myget("servers/%d" % vm.id, vm.userid)
225
        server = json.loads(response.content)['server']
226
        self.assertEqual(server["SNF:port_forwarding"], {})
227

    
228
        # Add with public IP
229
        mfactory.IPv4AddressFactory(nic__machine=vm, network__public=True)
230
        with override_settings(settings,
231
                               CYCLADES_PORT_FORWARDING=ports):
232
            response = self.myget("servers/%d" % vm.id, vm.userid)
233
        server = json.loads(response.content)['server']
234
        self.assertEqual(server["SNF:port_forwarding"],
235
                         {"22": {"host": "foo", "port": "61000"},
236
                          "80": {"host": "bar", "port": "61001"}})
237

    
238
        def _port_from_ip(ip, base):
239
            fields = ip.split('.', 4)
240
            return (base + 256*int(fields[2]) + int(fields[3]))
241

    
242
        ports = {
243
            22: lambda ip, id, fqdn, user:
244
            ip and ("gate", _port_from_ip(ip, 10000)) or None}
245
        vm = mfactory.VirtualMachineFactory()
246
        with override_settings(settings,
247
                               CYCLADES_PORT_FORWARDING=ports):
248
            response = self.myget("servers/%d" % vm.id, vm.userid)
249
            server = json.loads(response.content)['server']
250
            self.assertEqual(server["SNF:port_forwarding"], {})
251

    
252
        mfactory.IPv4AddressFactory(nic__machine=vm,
253
                                    network__public=True,
254
                                    address="192.168.2.2")
255
        with override_settings(settings,
256
                               CYCLADES_PORT_FORWARDING=ports):
257
            response = self.myget("servers/%d" % vm.id, vm.userid)
258
            server = json.loads(response.content)['server']
259
            self.assertEqual(server["SNF:port_forwarding"],
260
                             {"22": {"host": "gate", "port": "10514"}})
261

    
262
    def test_server_building_nics(self):
263
        db_vm = self.vm2
264
        user = self.vm2.userid
265
        net1 = mfactory.NetworkFactory()
266
        net2 = mfactory.NetworkFactory()
267
        net3 = mfactory.NetworkFactory()
268
        mfactory.NetworkInterfaceFactory(machine=self.vm2, network=net1,
269
                                         state="BUILD")
270
        nic2 = mfactory.NetworkInterfaceFactory(machine=self.vm2, network=net2,
271
                                                state="ACTIVE")
272
        mfactory.NetworkInterfaceFactory(machine=self.vm2, network=net3,
273
                                         state="BUILD")
274

    
275
        response = self.myget('servers/%d' % db_vm.id, user)
276
        server = json.loads(response.content)['server']
277
        nics = server["attachments"]
278
        self.assertEqual(len(nics), 1)
279
        self.assertEqual(nics[0]["network_id"], str(nic2.network_id))
280

    
281
    def test_noauthorized(self):
282
        """Test 404 for detail of other user vm"""
283
        db_vm = self.vm2
284

    
285
        response = self.myget('servers/%d' % db_vm.id, 'wrong_user')
286
        self.assertItemNotFound(response)
287

    
288
    def test_wrong_server(self):
289
        """Test 404 response if server does not exist."""
290
        response = self.myget('servers/%d' % 5000)
291
        self.assertItemNotFound(response)
292

    
293
    def test_create_server_empty(self):
294
        """Test if the create server call returns a 400 badRequest if
295
           no attributes are specified."""
296

    
297
        response = self.mypost('servers', params={})
298
        self.assertBadRequest(response)
299

    
300
    def test_rename_server(self):
301
        vm = self.vm2
302
        request = {'server': {'name': 'new_name'}}
303
        response = self.myput('servers/%d' % vm.id, vm.userid,
304
                              json.dumps(request), 'json')
305
        self.assertSuccess(response)
306
        self.assertEqual(VirtualMachine.objects.get(id=vm.id).name, "new_name")
307

    
308
    def test_catch_wrong_api_paths(self):
309
        response = self.myget('nonexistent')
310
        self.assertEqual(response.status_code, 400)
311
        try:
312
            json.loads(response.content)
313
        except ValueError:
314
            self.assertTrue(False)
315

    
316
    def test_method_not_allowed(self, *args):
317
        # /servers/ allows only POST, GET
318
        response = self.myput('servers', '', '')
319
        self.assertMethodNotAllowed(response)
320
        response = self.mydelete('servers')
321
        self.assertMethodNotAllowed(response)
322

    
323
        # /servers/<srvid>/ allows only GET, PUT, DELETE
324
        response = self.mypost("servers/42")
325
        self.assertMethodNotAllowed(response)
326

    
327
        # /imags/<srvid>/metadata/ allows only POST, GET
328
        response = self.myput('servers/42/metadata', '', '')
329
        self.assertMethodNotAllowed(response)
330
        response = self.mydelete('servers/42/metadata')
331
        self.assertMethodNotAllowed(response)
332

    
333
        # /imags/<srvid>/metadata/ allows only POST, GET
334
        response = self.myput('servers/42/metadata', '', '')
335
        self.assertMethodNotAllowed(response)
336
        response = self.mydelete('servers/42/metadata')
337
        self.assertMethodNotAllowed(response)
338

    
339
        # /imags/<srvid>/metadata/<key> allows only PUT, GET, DELETE
340
        response = self.mypost('servers/42/metadata/foo')
341
        self.assertMethodNotAllowed(response)
342

    
343

    
344
fixed_image = Mock()
345
fixed_image.return_value = {'location': 'pithos://foo',
346
                            'checksum': '1234',
347
                            "id": 1,
348
                            "name": "test_image",
349
                            "size": "41242",
350
                            'disk_format': 'diskdump'}
351

    
352

    
353
@patch('synnefo.api.util.get_image', fixed_image)
354
@patch('synnefo.logic.rapi_pool.GanetiRapiClient')
355
class ServerCreateAPITest(ComputeAPITest):
356
    def setUp(self):
357
        self.flavor = mfactory.FlavorFactory()
358
        # Create public network and backend
359
        subnet = mfactory.IPv4SubnetFactory(network__public=True)
360
        self.network = subnet.network
361
        self.backend = mfactory.BackendFactory()
362
        mfactory.BackendNetworkFactory(network=self.network,
363
                                       backend=self.backend,
364
                                       operstate="ACTIVE")
365
        self.request = {
366
            "server": {
367
                "name": "new-server-test",
368
                "userid": "test_user",
369
                "imageRef": 1,
370
                "flavorRef": self.flavor.id,
371
                "metadata": {
372
                    "My Server Name": "Apache1"
373
                },
374
                "personality": []
375
            }
376
        }
377

    
378
    def test_create_server(self, mrapi):
379
        """Test if the create server call returns the expected response
380
           if a valid request has been speficied."""
381

    
382
        mrapi().CreateInstance.return_value = 12
383
        with override_settings(settings, DEFAULT_INSTANCE_NETWORKS=[]):
384
            with mocked_quotaholder():
385
                response = self.mypost('servers', 'test_user',
386
                                       json.dumps(self.request), 'json')
387
        self.assertEqual(response.status_code, 202)
388
        mrapi().CreateInstance.assert_called_once()
389

    
390
        api_server = json.loads(response.content)['server']
391
        self.assertEqual(api_server['status'], "BUILD")
392
        self.assertEqual(api_server['progress'], 0)
393
        self.assertEqual(api_server['metadata'],
394
                         {"My Server Name":  "Apache1"})
395
        self.assertTrue('adminPass' in api_server)
396

    
397
        db_vm = VirtualMachine.objects.get(userid='test_user')
398
        self.assertEqual(api_server['name'], db_vm.name)
399
        self.assertEqual(api_server['status'], db_vm.operstate)
400

    
401
        # Test drained flag in Network:
402
        self.network.drained = True
403
        self.network.save()
404
        with mocked_quotaholder():
405
            response = self.mypost('servers', 'test_user',
406
                                   json.dumps(self.request), 'json')
407
        self.assertEqual(response.status_code, 503, "serviceUnavailable")
408

    
409
    def test_create_server_with_port(self, mrapi):
410
        mrapi().CreateInstance.return_value = 42
411
        ip = mfactory.IPv4AddressFactory(nic__machine=None)
412
        port1 = ip.nic
413
        request = deepcopy(self.request)
414
        request["server"]["networks"] = [{"port": port1.id}]
415
        with mocked_quotaholder():
416
            response = self.mypost("servers", port1.userid,
417
                                   json.dumps(request), 'json')
418
        self.assertEqual(response.status_code, 202)
419
        vm_id = json.loads(response.content)["server"]["id"]
420
        port1 = NetworkInterface.objects.get(id=port1.id)
421
        self.assertEqual(port1.machine_id, vm_id)
422
        # 409 if already used
423
        with mocked_quotaholder():
424
            response = self.mypost("servers", port1.userid,
425
                                   json.dumps(request), 'json')
426
        self.assertConflict(response)
427
        # Test permissions
428
        ip = mfactory.IPv4AddressFactory(userid="user1", nic__userid="user1")
429
        port2 = ip.nic
430
        request["server"]["networks"] = [{"port": port2.id}]
431
        with mocked_quotaholder():
432
            response = self.mypost("servers", "user2",
433
                                   json.dumps(request), 'json')
434
        self.assertEqual(response.status_code, 404)
435

    
436
    def test_create_network_settings(self, mrapi):
437
        mrapi().CreateInstance.return_value = 12
438
        # Create public network and backend
439
        subnet1 = mfactory.IPv4SubnetFactory(network__userid="test_user")
440
        bnet1 = mfactory.BackendNetworkFactory(network=subnet1.network,
441
                                               backend=self.backend,
442
                                               operstate="ACTIVE")
443
        subnet2 = mfactory.IPv4SubnetFactory(network__userid="test_user")
444
        bnet2 = mfactory.BackendNetworkFactory(network=subnet2.network,
445
                                               backend=self.backend,
446
                                               operstate="ACTIVE")
447
        # User requested private networks
448
        request = deepcopy(self.request)
449
        request["server"]["networks"] = [{"uuid": bnet1.network.id},
450
                                         {"uuid": bnet2.network.id}]
451
        with override_settings(settings,
452
                               DEFAULT_INSTANCE_NETWORKS=[
453
                                   "SNF:ANY_PUBLIC"]):
454
            with mocked_quotaholder():
455
                response = self.mypost('servers', 'test_user',
456
                                       json.dumps(request), 'json')
457
        self.assertEqual(response.status_code, 202)
458
        name, args, kwargs = mrapi().CreateInstance.mock_calls[0]
459
        self.assertEqual(len(kwargs["nics"]), 3)
460
        self.assertEqual(kwargs["nics"][0]["network"],
461
                         self.network.backend_id)
462
        self.assertEqual(kwargs["nics"][1]["network"],
463
                         bnet1.network.backend_id)
464
        self.assertEqual(kwargs["nics"][2]["network"],
465
                         bnet2.network.backend_id)
466

    
467
        subnet3 = mfactory.IPv4SubnetFactory(network__public=True,
468
                                             network__floating_ip_pool=True)
469
        bnet3 = mfactory.BackendNetworkFactory(network=subnet3.network,
470
                                               backend=self.backend,
471
                                               operstate="ACTIVE")
472
        request["server"]["floating_ips"] = []
473
        with override_settings(settings,
474
                               DEFAULT_INSTANCE_NETWORKS=[bnet3.network.id]):
475
            with mocked_quotaholder():
476
                response = self.mypost('servers', 'test_user',
477
                                       json.dumps(request), 'json')
478
        self.assertEqual(response.status_code, 202)
479
        name, args, kwargs = mrapi().CreateInstance.mock_calls[1]
480
        self.assertEqual(len(kwargs["nics"]), 3)
481
        self.assertEqual(kwargs["nics"][0]["network"],
482
                         bnet3.network.backend_id)
483
        self.assertEqual(kwargs["nics"][1]["network"],
484
                         bnet1.network.backend_id)
485
        self.assertEqual(kwargs["nics"][2]["network"],
486
                         bnet2.network.backend_id)
487

    
488
        # test invalid network in DEFAULT_INSTANCE_NETWORKS
489
        with override_settings(settings, DEFAULT_INSTANCE_NETWORKS=[42]):
490
            response = self.mypost('servers', 'test_user',
491
                                   json.dumps(request), 'json')
492
        self.assertFault(response, 500, "internalServerError")
493

    
494
        # test connect to public netwok
495
        request = deepcopy(self.request)
496
        request["server"]["networks"] = [{"uuid": self.network.id}]
497
        with override_settings(settings,
498
                               DEFAULT_INSTANCE_NETWORKS=["SNF:ANY_PUBLIC"]):
499
            response = self.mypost('servers', 'test_user',
500
                                   json.dumps(request), 'json')
501
        self.assertFault(response, 403, "forbidden")
502

    
503
        # test wrong user
504
        request = deepcopy(self.request)
505
        request["server"]["networks"] = [{"uuid": bnet1.network.id}]
506
        with override_settings(settings,
507
                               DEFAULT_INSTANCE_NETWORKS=["SNF:ANY_PUBLIC"]):
508
            with mocked_quotaholder():
509
                response = self.mypost('servers', 'dummy_user',
510
                                       json.dumps(request), 'json')
511
        self.assertItemNotFound(response)
512

    
513
        # Test floating IPs
514
        request = deepcopy(self.request)
515
        request["server"]["networks"] = [{"uuid": bnet1.network.id}]
516
        fp1 = mfactory.FloatingIPFactory(address="10.0.0.2",
517
                                         userid="test_user",
518
                                         network=self.network,
519
                                         nic=None)
520
        fp2 = mfactory.FloatingIPFactory(address="10.0.0.3",
521
                                         userid="test_user",
522
                                         network=self.network,
523
                                         nic=None)
524
        request["server"]["floating_ips"] = [fp1.id, fp2.id]
525
        with override_settings(settings,
526
                               DEFAULT_INSTANCE_NETWORKS=[bnet3.network.id]):
527
            with mocked_quotaholder():
528
                response = self.mypost('servers', 'test_user',
529
                                       json.dumps(request), 'json')
530
        self.assertEqual(response.status_code, 202)
531
        api_server = json.loads(response.content)['server']
532
        vm = VirtualMachine.objects.get(id=api_server["id"])
533
        fp1 = IPAddress.objects.get(floating_ip=True, id=fp1.id)
534
        fp2 = IPAddress.objects.get(floating_ip=True, id=fp2.id)
535
        self.assertEqual(fp1.nic.machine, vm)
536
        self.assertEqual(fp2.nic.machine, vm)
537
        name, args, kwargs = mrapi().CreateInstance.mock_calls[2]
538
        self.assertEqual(len(kwargs["nics"]), 4)
539
        self.assertEqual(kwargs["nics"][0]["network"],
540
                         bnet3.network.backend_id)
541
        self.assertEqual(kwargs["nics"][1]["network"], fp1.network.backend_id)
542
        self.assertEqual(kwargs["nics"][1]["ip"], fp1.address)
543
        self.assertEqual(kwargs["nics"][2]["network"], fp2.network.backend_id)
544
        self.assertEqual(kwargs["nics"][2]["ip"], fp2.address)
545
        self.assertEqual(kwargs["nics"][3]["network"],
546
                         bnet1.network.backend_id)
547

    
548
    def test_create_server_no_flavor(self, mrapi):
549
        request = deepcopy(self.request)
550
        request["server"]["flavorRef"] = 42
551
        with mocked_quotaholder():
552
            response = self.mypost('servers', 'test_user',
553
                                   json.dumps(request), 'json')
554
        self.assertItemNotFound(response)
555

    
556
    def test_create_server_error(self, mrapi):
557
        """Test if the create server call returns the expected response
558
           if a valid request has been speficied."""
559
        mrapi().CreateInstance.side_effect = GanetiApiError("..ganeti is down")
560

    
561
        request = self.request
562
        with mocked_quotaholder():
563
            response = self.mypost('servers', 'test_user',
564
                                   json.dumps(request), 'json')
565
        self.assertEqual(response.status_code, 202)
566
        mrapi().CreateInstance.assert_called_once()
567
        vm = VirtualMachine.objects.get()
568
        # The VM has not been deleted
569
        self.assertFalse(vm.deleted)
570
        # but is in "ERROR" operstate
571
        self.assertEqual(vm.operstate, "ERROR")
572

    
573

    
574
@patch('synnefo.logic.rapi_pool.GanetiRapiClient')
575
class ServerDestroyAPITest(ComputeAPITest):
576
    def test_delete_server(self, mrapi):
577
        vm = mfactory.VirtualMachineFactory()
578
        mrapi().DeleteInstance.return_value = 12
579
        response = self.mydelete('servers/%d' % vm.id, vm.userid)
580
        self.assertEqual(response.status_code, 204)
581
        mrapi().DeleteInstance.assert_called_once()
582

    
583
    def test_non_existing_delete_server(self, mrapi):
584
        vm = mfactory.VirtualMachineFactory()
585
        response = self.mydelete('servers/%d' % 42, vm.userid)
586
        self.assertItemNotFound(response)
587
        self.assertFalse(mrapi.mock_calls)
588

    
589

    
590
class ServerMetadataAPITest(ComputeAPITest):
591
    def setUp(self):
592
        self.vm = mfactory.VirtualMachineFactory()
593
        self.metadata = mfactory.VirtualMachineMetadataFactory(vm=self.vm)
594
        super(ServerMetadataAPITest, self).setUp()
595

    
596
    def test_get_metadata(self):
597
        vm = self.vm
598
        create_meta = lambda: mfactory.VirtualMachineMetadataFactory(vm=vm)
599
        metadata = [create_meta(), create_meta(), create_meta()]
600
        response = self.myget('servers/%d/metadata' % vm.id, vm.userid)
601
        self.assertTrue(response.status_code in [200, 203])
602
        api_metadata = json.loads(response.content)['metadata']
603
        self.assertEqual(len(api_metadata), len(metadata) + 1)
604
        for db_m in metadata:
605
            self.assertEqual(api_metadata[db_m.meta_key], db_m.meta_value)
606

    
607
        request = {
608
            'metadata': {
609
                'foo': 'bar'
610
            },
611
            metadata[0].meta_key: 'bar2'
612
        }
613
        response = self.mypost('servers/%d/metadata' % vm.id,
614
                               vm.userid, json.dumps(request), 'json')
615
        metadata2 = VirtualMachineMetadata.objects.filter(vm=vm)
616
        response = self.myget('servers/%d/metadata' % vm.id, vm.userid)
617
        self.assertTrue(response.status_code in [200, 203])
618
        api_metadata2 = json.loads(response.content)['metadata']
619
        self.assertTrue('foo' in api_metadata2.keys())
620
        self.assertTrue(api_metadata2[metadata[0].meta_key], 'bar2')
621
        self.assertEqual(len(api_metadata2), len(metadata2))
622
        for db_m in metadata2:
623
            self.assertEqual(api_metadata2[db_m.meta_key], db_m.meta_value)
624

    
625
        # Create new meta
626
        request = {'meta': {'foo2': 'bar2'}}
627
        response = self.myput('servers/%d/metadata/foo2' % vm.id,
628
                              vm.userid, json.dumps(request), 'json')
629

    
630
        # Get the new meta
631
        response = self.myget('servers/%d/metadata/foo2' % vm.id, vm.userid)
632
        meta = json.loads(response.content)['meta']
633
        self.assertEqual(meta['foo2'], 'bar2')
634

    
635
        # Delete the new meta
636
        response = self.mydelete('servers/%d/metadata/foo2' % vm.id, vm.userid)
637
        self.assertEqual(response.status_code, 204)
638

    
639
        # Try to get the deleted meta: should raise 404
640
        response = self.myget('servers/%d/metadata/foo2' % vm.id, vm.userid)
641
        self.assertEqual(response.status_code, 404)
642

    
643
    def test_invalid_metadata(self):
644
        vm = self.vm
645
        response = self.mypost('servers/%d/metadata' % vm.id, vm.userid)
646
        self.assertBadRequest(response)
647
        self.assertEqual(len(vm.metadata.all()), 1)
648

    
649
    def test_invalid_metadata_server(self):
650
        response = self.mypost('servers/42/metadata', 'user')
651
        self.assertItemNotFound(response)
652

    
653
    def test_get_meta_invalid_key(self):
654
        vm = self.vm
655
        response = self.myget('servers/%d/metadata/foo2' % vm.id, vm.userid)
656
        self.assertItemNotFound(response)
657

    
658

    
659
@patch('synnefo.api.util.get_image')
660
@patch('synnefo.logic.rapi_pool.GanetiRapiClient')
661
class ServerActionAPITest(ComputeAPITest):
662
    def test_actions(self, mrapi, mimage):
663
        actions = ['start', 'shutdown', 'reboot']
664
        vm = mfactory.VirtualMachineFactory()
665
        vm.operstate = "STOPPED"
666
        vm.save()
667
        mrapi().StartupInstance.return_value = 0
668
        mrapi().ShutdownInstance.return_value = 1
669
        mrapi().RebootInstance.return_value = 2
670
        for jobId, action in enumerate(actions):
671
            if action in ["shutdown", "reboot"]:
672
                vm.operstate = "STARTED"
673
            else:
674
                vm.operstate = "STOPPED"
675
            vm.task = None
676
            vm.task_job_id = None
677
            vm.save()
678
            val = {'type': 'HARD'} if action == 'reboot' else {}
679
            request = {action: val}
680
            response = self.mypost('servers/%d/action' % vm.id,
681
                                   vm.userid, json.dumps(request), 'json')
682
            self.assertEqual(response.status_code, 202)
683
            if action == 'shutdown':
684
                self.assertEqual(VirtualMachine.objects.get(id=vm.id).task,
685
                                 "STOP")
686
            else:
687
                self.assertEqual(VirtualMachine.objects.get(id=vm.id).task,
688
                                 action.upper())
689
            self.assertEqual(VirtualMachine.objects.get(id=vm.id).task_job_id,
690
                             jobId)
691

    
692
    def test_action_in_building_vm(self, mrapi, mimage):
693
        """Test building in progress"""
694
        vm = mfactory.VirtualMachineFactory(operstate="BUILD")
695
        request = {'start': {}}
696
        with mocked_quotaholder():
697
            response = self.mypost('servers/%d/action' % vm.id,
698
                                   vm.userid, json.dumps(request), 'json')
699
        self.assertEqual(response.status_code, 409)
700
        self.assertFalse(mrapi.mock_calls)
701

    
702
    def test_destroy_build_vm(self, mrapi, mimage):
703
        """Test building in progress"""
704
        vm = mfactory.VirtualMachineFactory()
705
        mrapi().DeleteInstance.return_value = 2
706
        response = self.mydelete('servers/%d' % vm.id,
707
                                 vm.userid)
708
        self.assertSuccess(response)
709
        mrapi().RemoveInstance.assert_called_once()
710

    
711
    def test_firewall(self, mrapi, mimage):
712
        vm = mfactory.VirtualMachineFactory()
713
        vm.operstate = "STOPPED"
714
        vm.save()
715
        request = {'firewallProfile': {'profile': 'PROTECTED'}}
716
        response = self.mypost('servers/%d/action' % vm.id,
717
                               vm.userid, json.dumps(request), 'json')
718
        self.assertBadRequest(response)
719
        request = {'firewallProfile': {'profile': 'PROTECTED', "nic": "10"}}
720
        response = self.mypost('servers/%d/action' % vm.id,
721
                               vm.userid, json.dumps(request), 'json')
722
        self.assertItemNotFound(response)
723
        nic = mfactory.NetworkInterfaceFactory(machine=vm)
724
        request = {'firewallProfile': {'profile': 'PROTECTED', "nic": nic.id}}
725
        response = self.mypost('servers/%d/action' % vm.id,
726
                               vm.userid, json.dumps(request), 'json')
727
        self.assertSuccess(response)
728
        mrapi().ModifyInstance.assert_called_once()
729

    
730
    def test_unsupported_firewall(self, mrapi, mimage):
731
        vm = mfactory.VirtualMachineFactory()
732
        vm.operstate = "STOPPED"
733
        vm.save()
734
        request = {'firewallProfile': {'profile': 'FOO'}}
735
        response = self.mypost('servers/%d/action' % vm.id,
736
                               vm.userid, json.dumps(request), 'json')
737
        self.assertBadRequest(response)
738
        self.assertFalse(mrapi.mock_calls)
739

    
740
    def test_resize_vm(self, mrapi, mimage):
741
        flavor = mfactory.FlavorFactory(cpu=1, ram=1024)
742
        # Check building VM
743
        vm = self.get_vm(flavor=flavor, operstate="BUILD")
744
        request = {'resize': {'flavorRef': flavor.id}}
745
        response = self.mypost('servers/%d/action' % vm.id,
746
                               vm.userid, json.dumps(request), 'json')
747
        self.assertFault(response, 409, "buildInProgress")
748
        # Check same Flavor
749
        vm = self.get_vm(flavor=flavor, operstate="STOPPED")
750
        request = {'resize': {'flavorRef': flavor.id}}
751
        response = self.mypost('servers/%d/action' % vm.id,
752
                               vm.userid, json.dumps(request), 'json')
753
        self.assertBadRequest(response)
754
        # Check flavor with different disk
755
        flavor2 = mfactory.FlavorFactory(disk=1024)
756
        flavor3 = mfactory.FlavorFactory(disk=2048)
757
        vm = self.get_vm(flavor=flavor2, operstate="STOPPED")
758
        request = {'resize': {'flavorRef': flavor3.id}}
759
        response = self.mypost('servers/%d/action' % vm.id,
760
                               vm.userid, json.dumps(request), 'json')
761
        self.assertBadRequest(response)
762
        flavor2 = mfactory.FlavorFactory(disk_template="foo")
763
        flavor3 = mfactory.FlavorFactory(disk_template="baz")
764
        vm = self.get_vm(flavor=flavor2, operstate="STOPPED")
765
        request = {'resize': {'flavorRef': flavor3.id}}
766
        response = self.mypost('servers/%d/action' % vm.id,
767
                               vm.userid, json.dumps(request), 'json')
768
        self.assertBadRequest(response)
769
        # Check success
770
        vm = self.get_vm(flavor=flavor, operstate="STOPPED")
771
        flavor4 = mfactory.FlavorFactory(disk_template=flavor.disk_template,
772
                                         disk=flavor.disk,
773
                                         cpu=4, ram=2048)
774
        request = {'resize': {'flavorRef': flavor4.id}}
775
        mrapi().ModifyInstance.return_value = 42
776
        response = self.mypost('servers/%d/action' % vm.id,
777
                               vm.userid, json.dumps(request), 'json')
778
        self.assertEqual(response.status_code, 202)
779
        vm = VirtualMachine.objects.get(id=vm.id)
780
        self.assertEqual(vm.task_job_id, 42)
781
        name, args, kwargs = mrapi().ModifyInstance.mock_calls[0]
782
        self.assertEqual(kwargs["beparams"]["vcpus"], 4)
783
        self.assertEqual(kwargs["beparams"]["minmem"], 2048)
784
        self.assertEqual(kwargs["beparams"]["maxmem"], 2048)
785

    
786
    def test_action_on_resizing_vm(self, mrapi, mimage):
787
        vm = mfactory.VirtualMachineFactory()
788
        vm.operstate = "RESIZE"
789
        vm.save()
790
        for action in VirtualMachine.ACTIONS:
791
            request = {action[0]: ""}
792
            response = self.mypost('servers/%d/action' % vm.id,
793
                                   vm.userid, json.dumps(request), 'json')
794
            self.assertBadRequest(response)
795
        # however you can destroy
796
        mrapi().DeleteInstance.return_value = 42
797
        response = self.mydelete('servers/%d' % vm.id,
798
                                 vm.userid)
799
        self.assertSuccess(response)
800

    
801
    def get_vm(self, flavor, operstate):
802
        vm = mfactory.VirtualMachineFactory(flavor=flavor)
803
        vm.operstate = operstate
804
        vm.backendjobstatus = "success"
805
        vm.save()
806
        return vm
807

    
808

    
809
class ServerVNCConsole(ComputeAPITest):
810
    def test_not_active_server(self):
811
        """Test console req for not ACTIVE server returns badRequest"""
812
        vm = mfactory.VirtualMachineFactory(operstate="BUILD")
813
        data = json.dumps({'console': {'type': 'vnc'}})
814
        response = self.mypost('servers/%d/action' % vm.id,
815
                               vm.userid, data, 'json')
816
        self.assertBadRequest(response)
817

    
818
    def test_active_server(self):
819
        """Test console req for ACTIVE server"""
820
        vm = mfactory.VirtualMachineFactory()
821
        vm.operstate = 'STARTED'
822
        vm.save()
823

    
824
        data = json.dumps({'console': {'type': 'vnc'}})
825
        with override_settings(settings, TEST=True):
826
            response = self.mypost('servers/%d/action' % vm.id,
827
                                   vm.userid, data, 'json')
828
        self.assertEqual(response.status_code, 200)
829
        reply = json.loads(response.content)
830
        self.assertEqual(reply.keys(), ['console'])
831
        console = reply['console']
832
        self.assertEqual(console['type'], 'vnc')
833
        self.assertEqual(set(console.keys()),
834
                         set(['type', 'host', 'port', 'password']))
835

    
836
    def test_wrong_console_type(self):
837
        """Test console req for ACTIVE server"""
838
        vm = mfactory.VirtualMachineFactory()
839
        vm.operstate = 'STARTED'
840
        vm.save()
841

    
842
        data = json.dumps({'console': {'type': 'foo'}})
843
        response = self.mypost('servers/%d/action' % vm.id,
844
                               vm.userid, data, 'json')
845
        self.assertBadRequest(response)