Statistics
| Branch: | Tag: | Revision:

root / test / ganeti.rapi.client_unittest.py @ 2004e673

History | View | Annotate | Download (14.3 kB)

1
#!/usr/bin/python
2
#
3

    
4
# Copyright (C) 2010 Google Inc.
5
#
6
# This program is free software; you can redistribute it and/or modify
7
# it under the terms of the GNU General Public License as published by
8
# the Free Software Foundation; either version 2 of the License, or
9
# (at your option) any later version.
10
#
11
# This program is distributed in the hope that it will be useful, but
12
# WITHOUT ANY WARRANTY; without even the implied warranty of
13
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14
# General Public License for more details.
15
#
16
# You should have received a copy of the GNU General Public License
17
# along with this program; if not, write to the Free Software
18
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
19
# 02110-1301, USA.
20

    
21

    
22
"""Script for unittesting the RAPI client module"""
23

    
24

    
25
try:
26
  import httplib2
27
  BaseHttp = httplib2.Http
28
  from ganeti.rapi import client
29
except ImportError:
30
  httplib2 = None
31
  BaseHttp = object
32

    
33
import re
34
import unittest
35
import warnings
36

    
37
from ganeti import http
38

    
39
from ganeti.rapi import connector
40
from ganeti.rapi import rlib2
41

    
42
import testutils
43

    
44

    
45
_URI_RE = re.compile(r"https://(?P<host>.*):(?P<port>\d+)(?P<path>/.*)")
46

    
47

    
48
def _GetPathFromUri(uri):
49
  """Gets the path and query from a URI.
50

51
  """
52
  match = _URI_RE.match(uri)
53
  if match:
54
    return match.groupdict()["path"]
55
  else:
56
    return None
57

    
58

    
59
class HttpResponseMock(dict):
60
  """Dumb mock of httplib2.Response.
61

62
  """
63

    
64
  def __init__(self, status):
65
    self.status = status
66
    self['status'] = status
67

    
68

    
69
class HttpMock(BaseHttp):
70
  """Mock for httplib.Http.
71

72
  """
73

    
74
  def __init__(self, rapi):
75
    self._rapi = rapi
76
    self._last_request = None
77

    
78
  last_request_url = property(lambda self: self._last_request[0])
79
  last_request_method = property(lambda self: self._last_request[1])
80
  last_request_body = property(lambda self: self._last_request[2])
81

    
82
  def request(self, url, method, body, headers):
83
    self._last_request = (url, method, body)
84
    code, resp_body = self._rapi.FetchResponse(_GetPathFromUri(url), method)
85
    return HttpResponseMock(code), resp_body
86

    
87

    
88
class RapiMock(object):
89

    
90
  def __init__(self):
91
    self._mapper = connector.Mapper()
92
    self._responses = []
93
    self._last_handler = None
94

    
95
  def AddResponse(self, response):
96
    self._responses.insert(0, response)
97

    
98
  def PopResponse(self):
99
    if len(self._responses) > 0:
100
      return self._responses.pop()
101
    else:
102
      return None
103

    
104
  def GetLastHandler(self):
105
    return self._last_handler
106

    
107
  def FetchResponse(self, path, method):
108
    code = 200
109
    response = None
110

    
111
    try:
112
      HandlerClass, items, args = self._mapper.getController(path)
113
      self._last_handler = HandlerClass(items, args, None)
114
      if not hasattr(self._last_handler, method.upper()):
115
        code = 400
116
        response = "Bad request"
117
    except http.HttpException, ex:
118
      code = ex.code
119
      response = ex.message
120

    
121
    if not response:
122
      response = self.PopResponse()
123

    
124
    return code, response
125

    
126

    
127
class RapiMockTest(unittest.TestCase):
128

    
129
  def test(self):
130
    rapi = RapiMock()
131
    path = "/version"
132
    self.assertEqual((404, None), rapi.FetchResponse("/foo", "GET"))
133
    self.assertEqual((400, "Bad request"),
134
                     rapi.FetchResponse("/version", "POST"))
135
    rapi.AddResponse("2")
136
    code, response = rapi.FetchResponse("/version", "GET")
137
    self.assertEqual(200, code)
138
    self.assertEqual("2", response)
139
    self.failUnless(isinstance(rapi.GetLastHandler(), rlib2.R_version))
140

    
141

    
142
class GanetiRapiClientTests(unittest.TestCase):
143
  """Tests for remote API client.
144

145
  """
146

    
147
  def setUp(self):
148
    # Monkey-patch a fake VerifyCertificate function
149
    self._verify_certificate = client._VerifyCertificate
150
    client._VerifyCertificate = lambda x, y, z: True
151

    
152
    self.rapi = RapiMock()
153
    self.http = HttpMock(self.rapi)
154
    self.client = client.GanetiRapiClient('master.foo.com')
155
    self.client._http = self.http
156
    # Hard-code the version for easier testing.
157
    self.client._version = 2
158

    
159
  def tearDown(self):
160
    # Un-do the monkey-patch
161
    client._VerifyCertificate = self._verify_certificate
162

    
163
  def assertHandler(self, handler_cls):
164
    self.failUnless(isinstance(self.rapi.GetLastHandler(), handler_cls))
165

    
166
  def assertQuery(self, key, value):
167
    self.assertEqual(value, self.rapi.GetLastHandler().queryargs.get(key, None))
168

    
169
  def assertItems(self, items):
170
    self.assertEqual(items, self.rapi.GetLastHandler().items)
171

    
172
  def assertBulk(self):
173
    self.assertTrue(self.rapi.GetLastHandler().useBulk())
174

    
175
  def assertDryRun(self):
176
    self.assertTrue(self.rapi.GetLastHandler().dryRun())
177

    
178
  def testGetVersion(self):
179
    self.client._version = None
180
    self.rapi.AddResponse("2")
181
    self.assertEqual(2, self.client.GetVersion())
182
    self.assertHandler(rlib2.R_version)
183

    
184
  def testGetOperatingSystems(self):
185
    self.rapi.AddResponse("[\"beos\"]")
186
    self.assertEqual(["beos"], self.client.GetOperatingSystems())
187
    self.assertHandler(rlib2.R_2_os)
188

    
189
  def testGetClusterTags(self):
190
    self.rapi.AddResponse("[\"tag\"]")
191
    self.assertEqual(["tag"], self.client.GetClusterTags())
192
    self.assertHandler(rlib2.R_2_tags)
193

    
194
  def testAddClusterTags(self):
195
    self.client.AddClusterTags(["awesome"], dry_run=True)
196
    self.assertHandler(rlib2.R_2_tags)
197
    self.assertDryRun()
198
    self.assertQuery("tag", ["awesome"])
199

    
200
  def testDeleteClusterTags(self):
201
    self.client.DeleteClusterTags(["awesome"], dry_run=True)
202
    self.assertHandler(rlib2.R_2_tags)
203
    self.assertDryRun()
204
    self.assertQuery("tag", ["awesome"])
205

    
206
  def testGetInfo(self):
207
    self.rapi.AddResponse("{}")
208
    self.assertEqual({}, self.client.GetInfo())
209
    self.assertHandler(rlib2.R_2_info)
210

    
211
  def testGetInstances(self):
212
    self.rapi.AddResponse("[]")
213
    self.assertEqual([], self.client.GetInstances(bulk=True))
214
    self.assertHandler(rlib2.R_2_instances)
215
    self.assertBulk()
216

    
217
  def testGetInstanceInfo(self):
218
    self.rapi.AddResponse("[]")
219
    self.assertEqual([], self.client.GetInstanceInfo("instance"))
220
    self.assertHandler(rlib2.R_2_instances_name)
221
    self.assertItems(["instance"])
222

    
223
  def testCreateInstance(self):
224
    self.rapi.AddResponse("1234")
225
    self.assertEqual(1234, self.client.CreateInstance(dry_run=True))
226
    self.assertHandler(rlib2.R_2_instances)
227
    self.assertDryRun()
228

    
229
  def testDeleteInstance(self):
230
    self.client.DeleteInstance("instance", dry_run=True)
231
    self.assertHandler(rlib2.R_2_instances_name)
232
    self.assertItems(["instance"])
233
    self.assertDryRun()
234

    
235
  def testGetInstanceTags(self):
236
    self.rapi.AddResponse("[]")
237
    self.assertEqual([], self.client.GetInstanceTags("fooinstance"))
238
    self.assertHandler(rlib2.R_2_instances_name_tags)
239
    self.assertItems(["fooinstance"])
240

    
241
  def testAddInstanceTags(self):
242
    self.client.AddInstanceTags("fooinstance", ["awesome"], dry_run=True)
243
    self.assertHandler(rlib2.R_2_instances_name_tags)
244
    self.assertItems(["fooinstance"])
245
    self.assertDryRun()
246
    self.assertQuery("tag", ["awesome"])
247

    
248
  def testDeleteInstanceTags(self):
249
    self.client.DeleteInstanceTags("foo", ["awesome"], dry_run=True)
250
    self.assertHandler(rlib2.R_2_instances_name_tags)
251
    self.assertItems(["foo"])
252
    self.assertDryRun()
253
    self.assertQuery("tag", ["awesome"])
254

    
255
  def testRebootInstance(self):
256
    self.client.RebootInstance("i-bar", reboot_type="hard",
257
                               ignore_secondaries=True, dry_run=True)
258
    self.assertHandler(rlib2.R_2_instances_name_reboot)
259
    self.assertItems(["i-bar"])
260
    self.assertDryRun()
261
    self.assertQuery("type", ["hard"])
262
    self.assertQuery("ignore_secondaries", ["True"])
263

    
264
  def testShutdownInstance(self):
265
    self.client.ShutdownInstance("foo-instance", dry_run=True)
266
    self.assertHandler(rlib2.R_2_instances_name_shutdown)
267
    self.assertItems(["foo-instance"])
268
    self.assertDryRun()
269

    
270
  def testStartupInstance(self):
271
    self.client.StartupInstance("bar-instance", dry_run=True)
272
    self.assertHandler(rlib2.R_2_instances_name_startup)
273
    self.assertItems(["bar-instance"])
274
    self.assertDryRun()
275

    
276
  def testReinstallInstance(self):
277
    self.client.ReinstallInstance("baz-instance", "DOS", no_startup=True)
278
    self.assertHandler(rlib2.R_2_instances_name_reinstall)
279
    self.assertItems(["baz-instance"])
280
    self.assertQuery("os", ["DOS"])
281
    self.assertQuery("nostartup", ["1"])
282

    
283
  def testReplaceInstanceDisks(self):
284
    self.rapi.AddResponse("999")
285
    job_id = self.client.ReplaceInstanceDisks("instance-name",
286
        ["hda", "hdc"], dry_run=True)
287
    self.assertEqual(999, job_id)
288
    self.assertHandler(rlib2.R_2_instances_name_replace_disks)
289
    self.assertItems(["instance-name"])
290
    self.assertQuery("disks", ["hda,hdc"])
291
    self.assertQuery("mode", ["replace_auto"])
292
    self.assertQuery("iallocator", ["hail"])
293
    self.assertDryRun()
294

    
295
    self.assertRaises(client.InvalidReplacementMode,
296
                      self.client.ReplaceInstanceDisks,
297
                      "instance_a", ["hda"], mode="invalid_mode")
298
    self.assertRaises(client.GanetiApiError,
299
                      self.client.ReplaceInstanceDisks,
300
                      "instance-foo", ["hda"], mode="replace_on_secondary")
301

    
302
    self.rapi.AddResponse("1000")
303
    job_id = self.client.ReplaceInstanceDisks("instance-bar",
304
        ["hda"], mode="replace_on_secondary", remote_node="foo-node",
305
        dry_run=True)
306
    self.assertEqual(1000, job_id)
307
    self.assertItems(["instance-bar"])
308
    self.assertQuery("disks", ["hda"])
309
    self.assertQuery("remote_node", ["foo-node"])
310
    self.assertDryRun()
311

    
312
  def testGetJobs(self):
313
    self.rapi.AddResponse('[ { "id": "123", "uri": "\\/2\\/jobs\\/123" },'
314
                          '  { "id": "124", "uri": "\\/2\\/jobs\\/124" } ]')
315
    self.assertEqual([123, 124], self.client.GetJobs())
316
    self.assertHandler(rlib2.R_2_jobs)
317

    
318
  def testGetJobStatus(self):
319
    self.rapi.AddResponse("{\"foo\": \"bar\"}")
320
    self.assertEqual({"foo": "bar"}, self.client.GetJobStatus(1234))
321
    self.assertHandler(rlib2.R_2_jobs_id)
322
    self.assertItems(["1234"])
323

    
324
  def testDeleteJob(self):
325
    self.client.DeleteJob(999, dry_run=True)
326
    self.assertHandler(rlib2.R_2_jobs_id)
327
    self.assertItems(["999"])
328
    self.assertDryRun()
329

    
330
  def testGetNodes(self):
331
    self.rapi.AddResponse("[ { \"id\": \"node1\", \"uri\": \"uri1\" },"
332
                          " { \"id\": \"node2\", \"uri\": \"uri2\" } ]")
333
    self.assertEqual(["node1", "node2"], self.client.GetNodes())
334
    self.assertHandler(rlib2.R_2_nodes)
335

    
336
    self.rapi.AddResponse("[ { \"id\": \"node1\", \"uri\": \"uri1\" },"
337
                          " { \"id\": \"node2\", \"uri\": \"uri2\" } ]")
338
    self.assertEqual([{"id": "node1", "uri": "uri1"},
339
                      {"id": "node2", "uri": "uri2"}],
340
                     self.client.GetNodes(bulk=True))
341
    self.assertHandler(rlib2.R_2_nodes)
342
    self.assertBulk()
343

    
344
  def testGetNodeInfo(self):
345
    self.rapi.AddResponse("{}")
346
    self.assertEqual({}, self.client.GetNodeInfo("node-foo"))
347
    self.assertHandler(rlib2.R_2_nodes_name)
348
    self.assertItems(["node-foo"])
349

    
350
  def testEvacuateNode(self):
351
    self.rapi.AddResponse("9876")
352
    job_id = self.client.EvacuateNode("node-1", remote_node="node-2")
353
    self.assertEqual(9876, job_id)
354
    self.assertHandler(rlib2.R_2_nodes_name_evacuate)
355
    self.assertItems(["node-1"])
356
    self.assertQuery("remote_node", ["node-2"])
357

    
358
    self.rapi.AddResponse("8888")
359
    job_id = self.client.EvacuateNode("node-3", iallocator="hail", dry_run=True)
360
    self.assertEqual(8888, job_id)
361
    self.assertItems(["node-3"])
362
    self.assertQuery("iallocator", ["hail"])
363
    self.assertDryRun()
364

    
365
    self.assertRaises(client.GanetiApiError,
366
                      self.client.EvacuateNode,
367
                      "node-4", iallocator="hail", remote_node="node-5")
368

    
369
  def testMigrateNode(self):
370
    self.rapi.AddResponse("1111")
371
    self.assertEqual(1111, self.client.MigrateNode("node-a", dry_run=True))
372
    self.assertHandler(rlib2.R_2_nodes_name_migrate)
373
    self.assertItems(["node-a"])
374
    self.assertQuery("live", ["1"])
375
    self.assertDryRun()
376

    
377
  def testGetNodeRole(self):
378
    self.rapi.AddResponse("\"master\"")
379
    self.assertEqual("master", self.client.GetNodeRole("node-a"))
380
    self.assertHandler(rlib2.R_2_nodes_name_role)
381
    self.assertItems(["node-a"])
382

    
383
  def testSetNodeRole(self):
384
    self.rapi.AddResponse("789")
385
    self.assertEqual(789,
386
        self.client.SetNodeRole("node-foo", "master-candidate", force=True))
387
    self.assertHandler(rlib2.R_2_nodes_name_role)
388
    self.assertItems(["node-foo"])
389
    self.assertQuery("force", ["True"])
390
    self.assertEqual("master-candidate", self.http.last_request_body)
391

    
392
    self.assertRaises(client.InvalidNodeRole,
393
                      self.client.SetNodeRole, "node-bar", "fake-role")
394

    
395
  def testGetNodeStorageUnits(self):
396
    self.rapi.AddResponse("42")
397
    self.assertEqual(42,
398
        self.client.GetNodeStorageUnits("node-x", "lvm-pv", "fields"))
399
    self.assertHandler(rlib2.R_2_nodes_name_storage)
400
    self.assertItems(["node-x"])
401
    self.assertQuery("storage_type", ["lvm-pv"])
402
    self.assertQuery("output_fields", ["fields"])
403

    
404
    self.assertRaises(client.InvalidStorageType,
405
                      self.client.GetNodeStorageUnits,
406
                      "node-y", "floppy-disk", "fields")
407

    
408
  def testModifyNodeStorageUnits(self):
409
    self.rapi.AddResponse("14")
410
    self.assertEqual(14,
411
        self.client.ModifyNodeStorageUnits("node-z", "lvm-pv", "hda"))
412
    self.assertHandler(rlib2.R_2_nodes_name_storage_modify)
413
    self.assertItems(["node-z"])
414
    self.assertQuery("storage_type", ["lvm-pv"])
415
    self.assertQuery("name", ["hda"])
416

    
417
    self.assertRaises(client.InvalidStorageType,
418
                      self.client.ModifyNodeStorageUnits,
419
                      "node-n", "floppy-disk", "hdc")
420

    
421
  def testGetNodeTags(self):
422
    self.rapi.AddResponse("[\"fry\", \"bender\"]")
423
    self.assertEqual(["fry", "bender"], self.client.GetNodeTags("node-k"))
424
    self.assertHandler(rlib2.R_2_nodes_name_tags)
425
    self.assertItems(["node-k"])
426

    
427
  def testAddNodeTags(self):
428
    self.client.AddNodeTags("node-v", ["awesome"], dry_run=True)
429
    self.assertHandler(rlib2.R_2_nodes_name_tags)
430
    self.assertItems(["node-v"])
431
    self.assertDryRun()
432
    self.assertQuery("tag", ["awesome"])
433

    
434
  def testDeleteNodeTags(self):
435
    self.client.DeleteNodeTags("node-w", ["awesome"], dry_run=True)
436
    self.assertHandler(rlib2.R_2_nodes_name_tags)
437
    self.assertItems(["node-w"])
438
    self.assertDryRun()
439
    self.assertQuery("tag", ["awesome"])
440

    
441

    
442
if __name__ == '__main__':
443
  if httplib2 is None:
444
    warnings.warn("These tests require the httplib2 library")
445
  else:
446
    testutils.GanetiTestProgram()