Statistics
| Branch: | Tag: | Revision:

root / snf-cyclades-app / synnefo / logic / tests / servers.py @ 2e19e7dc

History | View | Annotate | Download (10.9 kB)

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

    
31
# Provides automated tests for logic module
32
from django.test import TransactionTestCase
33
#from snf_django.utils.testing import mocked_quotaholder
34
from synnefo.logic import servers
35
from synnefo import quotas
36
from synnefo.db import models_factory as mfactory, models
37
from mock import patch
38

    
39
from snf_django.lib.api import faults
40
from snf_django.utils.testing import mocked_quotaholder, override_settings
41
from django.conf import settings
42
from copy import deepcopy
43

    
44

    
45
@patch("synnefo.logic.rapi_pool.GanetiRapiClient")
46
class ServerCreationTest(TransactionTestCase):
47
    def test_create(self, mrapi):
48
        flavor = mfactory.FlavorFactory()
49
        kwargs = {
50
            "userid": "test",
51
            "name": "test_vm",
52
            "password": "1234",
53
            "flavor": flavor,
54
            "image": {"id": "foo", "backend_id": "foo", "format": "diskdump",
55
                      "metadata": "{}"},
56
            "networks": [],
57
            "metadata": {"foo": "bar"},
58
            "personality": [],
59
        }
60
        # no backend!
61
        mfactory.BackendFactory(offline=True)
62
        self.assertRaises(faults.ServiceUnavailable, servers.create, **kwargs)
63
        self.assertEqual(models.VirtualMachine.objects.count(), 0)
64

    
65
        mfactory.IPv4SubnetFactory(network__public=True)
66
        mfactory.IPv6SubnetFactory(network__public=True)
67
        mfactory.BackendFactory()
68

    
69
        # error in nics
70
        req = deepcopy(kwargs)
71
        req["networks"] = [{"uuid": 42}]
72
        self.assertRaises(faults.ItemNotFound, servers.create, **req)
73
        self.assertEqual(models.VirtualMachine.objects.count(), 0)
74

    
75
        # error in enqueue. check the vm is deleted and resources released
76
        mrapi().CreateInstance.side_effect = Exception("ganeti is down")
77
        with mocked_quotaholder():
78
            servers.create(**kwargs)
79
        vm = models.VirtualMachine.objects.get()
80
        self.assertFalse(vm.deleted)
81
        self.assertEqual(vm.operstate, "ERROR")
82
        for nic in vm.nics.all():
83
            self.assertEqual(nic.state, "ERROR")
84

    
85

    
86
@patch("synnefo.logic.rapi_pool.GanetiRapiClient")
87
class ServerTest(TransactionTestCase):
88
    def test_connect_network(self, mrapi):
89
        # Common connect
90
        for dhcp in [True, False]:
91
            subnet = mfactory.IPv4SubnetFactory(network__flavor="CUSTOM",
92
                                                cidr="192.168.2.0/24",
93
                                                gateway="192.168.2.1",
94
                                                dhcp=dhcp)
95
            net = subnet.network
96
            vm = mfactory.VirtualMachineFactory(operstate="STARTED")
97
            mfactory.BackendNetworkFactory(network=net, backend=vm.backend)
98
            mrapi().ModifyInstance.return_value = 42
99
            with override_settings(settings, GANETI_USE_HOTPLUG=True):
100
                servers.connect(vm, net)
101
            pool = net.get_ip_pools(locked=False)[0]
102
            self.assertFalse(pool.is_available("192.168.2.2"))
103
            args, kwargs = mrapi().ModifyInstance.call_args
104
            nics = kwargs["nics"][0]
105
            self.assertEqual(kwargs["instance"], vm.backend_vm_id)
106
            self.assertEqual(nics[0], "add")
107
            self.assertEqual(nics[1], "-1")
108
            self.assertEqual(nics[2]["ip"], "192.168.2.2")
109
            self.assertEqual(nics[2]["network"], net.backend_id)
110

    
111
        # Test connect to IPv6 only network
112
        vm = mfactory.VirtualMachineFactory(operstate="STARTED")
113
        subnet = mfactory.IPv6SubnetFactory(cidr="2000::/64",
114
                                            gateway="2000::1")
115
        net = subnet.network
116
        mfactory.BackendNetworkFactory(network=net, backend=vm.backend)
117
        with override_settings(settings, GANETI_USE_HOTPLUG=True):
118
            servers.connect(vm, net)
119
        args, kwargs = mrapi().ModifyInstance.call_args
120
        nics = kwargs["nics"][0]
121
        self.assertEqual(kwargs["instance"], vm.backend_vm_id)
122
        self.assertEqual(nics[0], "add")
123
        self.assertEqual(nics[1], "-1")
124
        self.assertEqual(nics[2]["ip"], None)
125
        self.assertEqual(nics[2]["network"], net.backend_id)
126

    
127

    
128
@patch("synnefo.logic.rapi_pool.GanetiRapiClient")
129
class ServerCommandTest(TransactionTestCase):
130
    def test_pending_task(self, mrapi):
131
        vm = mfactory.VirtualMachineFactory(task="REBOOT", task_job_id=1)
132
        self.assertRaises(faults.BadRequest, servers.start, vm)
133
        vm = mfactory.VirtualMachineFactory(task="BUILD", task_job_id=1)
134
        self.assertRaises(faults.BuildInProgress, servers.start, vm)
135
        # Assert always succeeds
136
        vm = mfactory.VirtualMachineFactory(task="BUILD", task_job_id=1)
137
        mrapi().DeleteInstance.return_value = 1
138
        with mocked_quotaholder():
139
            servers.destroy(vm)
140
        vm = mfactory.VirtualMachineFactory(task="REBOOT", task_job_id=1)
141
        with mocked_quotaholder():
142
            servers.destroy(vm)
143

    
144
    def test_deleted_vm(self, mrapi):
145
        vm = mfactory.VirtualMachineFactory(deleted=True)
146
        self.assertRaises(faults.BadRequest, servers.start, vm)
147

    
148
    def test_invalid_operstate_for_action(self, mrapi):
149
        vm = mfactory.VirtualMachineFactory(operstate="STARTED")
150
        self.assertRaises(faults.BadRequest, servers.start, vm)
151
        vm = mfactory.VirtualMachineFactory(operstate="STOPPED")
152
        self.assertRaises(faults.BadRequest, servers.stop, vm)
153
        vm = mfactory.VirtualMachineFactory(operstate="STARTED")
154
        flavor = mfactory.FlavorFactory()
155
        self.assertRaises(faults.BadRequest, servers.resize, vm, flavor)
156
        # Check that connect/disconnect is allowed only in STOPPED vms
157
        # if hotplug is disabled.
158
        vm = mfactory.VirtualMachineFactory(operstate="STARTED")
159
        network = mfactory.NetworkFactory(state="ACTIVE")
160
        with override_settings(settings, GANETI_USE_HOTPLUG=False):
161
            self.assertRaises(faults.BadRequest, servers.connect, vm, network)
162
            self.assertRaises(faults.BadRequest, servers.disconnect, vm,
163
                              network)
164
        #test valid
165
        vm = mfactory.VirtualMachineFactory(operstate="STOPPED")
166
        mrapi().StartupInstance.return_value = 1
167
        with mocked_quotaholder():
168
            servers.start(vm)
169
        vm.task = None
170
        vm.task_job_id = None
171
        vm.save()
172
        with mocked_quotaholder():
173
            quotas.accept_serial(vm.serial)
174
        mrapi().RebootInstance.return_value = 1
175
        with mocked_quotaholder():
176
            servers.reboot(vm, "HARD")
177

    
178
    def test_commission(self, mrapi):
179
        vm = mfactory.VirtualMachineFactory(operstate="STOPPED")
180
        # Still pending
181
        vm.serial = mfactory.QuotaHolderSerialFactory(serial=200,
182
                                                      resolved=False,
183
                                                      pending=True)
184
        serial = vm.serial
185
        mrapi().StartupInstance.return_value = 1
186
        with mocked_quotaholder() as m:
187
            with self.assertRaises(quotas.ResolveError):
188
                servers.start(vm)
189
        # Not pending, rejct
190
        vm.task = None
191
        vm.serial = mfactory.QuotaHolderSerialFactory(serial=400,
192
                                                      resolved=False,
193
                                                      pending=False,
194
                                                      accept=False)
195
        serial = vm.serial
196
        mrapi().StartupInstance.return_value = 1
197
        with mocked_quotaholder() as m:
198
            servers.start(vm)
199
            m.resolve_commissions.assert_called_once_with([],
200
                                                          [serial.serial])
201
            self.assertTrue(m.issue_one_commission.called)
202
        # Not pending, accept
203
        vm.task = None
204
        vm.serial = mfactory.QuotaHolderSerialFactory(serial=600,
205
                                                      resolved=False,
206
                                                      pending=False,
207
                                                      accept=True)
208
        serial = vm.serial
209
        mrapi().StartupInstance.return_value = 1
210
        with mocked_quotaholder() as m:
211
            servers.start(vm)
212
            m.resolve_commissions.assert_called_once_with([serial.serial],
213
                                                          [])
214
            self.assertTrue(m.issue_one_commission.called)
215

    
216
        mrapi().StartupInstance.side_effect = ValueError
217
        vm.task = None
218
        vm.serial = None
219
        # Test reject if Ganeti erro
220
        with mocked_quotaholder() as m:
221
            try:
222
                servers.start(vm)
223
            except:
224
                m.resolve_commissions\
225
                 .assert_called_once_with([], [vm.serial.serial])
226

    
227
    def test_task_after(self, mrapi):
228
        return
229
        vm = mfactory.VirtualMachineFactory()
230
        mrapi().StartupInstance.return_value = 1
231
        mrapi().ShutdownInstance.return_value = 2
232
        mrapi().RebootInstance.return_value = 2
233
        with mocked_quotaholder():
234
            vm.task = None
235
            vm.operstate = "STOPPED"
236
            servers.start(vm)
237
            self.assertEqual(vm.task, "START")
238
            self.assertEqual(vm.task_job_id, 1)
239
        with mocked_quotaholder():
240
            vm.task = None
241
            vm.operstate = "STARTED"
242
            servers.stop(vm)
243
            self.assertEqual(vm.task, "STOP")
244
            self.assertEqual(vm.task_job_id, 2)
245
        with mocked_quotaholder():
246
            vm.task = None
247
            servers.reboot(vm)
248
            self.assertEqual(vm.task, "REBOOT")
249
            self.assertEqual(vm.task_job_id, 3)