QA: test using OS API v20
[ganeti-local] / qa / qa_os.py
1 #
2 #
3
4 # Copyright (C) 2007, 2008, 2009, 2010, 2011 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 """OS related QA tests.
23
24 """
25
26 import os
27 import os.path
28
29 from ganeti import utils
30 from ganeti import constants
31
32 import qa_config
33 import qa_utils
34 import qa_error
35
36 from qa_utils import AssertCommand, AssertIn, AssertNotIn
37
38
39 _TEMP_OS_NAME = "TEMP-Ganeti-QA-OS"
40 _TEMP_OS_PATH = os.path.join(constants.OS_SEARCH_PATH[0], _TEMP_OS_NAME)
41
42 (_ALL_VALID,
43  _ALL_INVALID,
44  _PARTIALLY_VALID) = range(1, 4)
45
46
47 def TestOsList():
48   """gnt-os list"""
49   AssertCommand(["gnt-os", "list"])
50
51
52 def TestOsDiagnose():
53   """gnt-os diagnose"""
54   AssertCommand(["gnt-os", "diagnose"])
55
56
57 def _TestOsModify(hvp_dict, fail=False):
58   """gnt-os modify"""
59   cmd = ["gnt-os", "modify"]
60
61   for hv_name, hv_params in hvp_dict.items():
62     cmd.append("-H")
63     options = []
64     for key, value in hv_params.items():
65       options.append("%s=%s" % (key, value))
66     cmd.append("%s:%s" % (hv_name, ",".join(options)))
67
68   cmd.append(_TEMP_OS_NAME)
69   AssertCommand(cmd, fail=fail)
70
71
72 def _TestOsStates(os_name):
73   """gnt-os modify, more stuff"""
74   cmd = ["gnt-os", "modify"]
75
76   for param in ["hidden", "blacklisted"]:
77     for val in ["yes", "no"]:
78       new_cmd = cmd + ["--%s" % param, val, os_name]
79       AssertCommand(new_cmd)
80       # check that double-running the command is OK
81       AssertCommand(new_cmd)
82
83
84 def _SetupTempOs(node, dirname, valid):
85   """Creates a temporary OS definition on the given node.
86
87   """
88   sq = utils.ShellQuoteArgs
89   parts = [
90     sq(["rm", "-rf", dirname]),
91     sq(["mkdir", "-p", dirname]),
92     sq(["cd", dirname]),
93     sq(["ln", "-fs", "/bin/true", "export"]),
94     sq(["ln", "-fs", "/bin/true", "import"]),
95     sq(["ln", "-fs", "/bin/true", "rename"]),
96     sq(["ln", "-fs", "/bin/true", "verify"]),
97     ]
98
99   if valid:
100     parts.append(sq(["ln", "-fs", "/bin/true", "create"]))
101
102   parts.append(sq(["echo", str(constants.OS_API_V20)]) +
103                " >ganeti_api_version")
104
105   parts.append(sq(["echo", "default"]) + " >variants.list")
106   parts.append(sq(["echo", "funny this is funny"]) + " >parameters.list")
107
108   cmd = " && ".join(parts)
109
110   print qa_utils.FormatInfo("Setting up %s with %s OS definition" %
111                             (node["primary"],
112                              ["an invalid", "a valid"][int(valid)]))
113
114   AssertCommand(cmd, node=node)
115
116
117 def _RemoveTempOs(node, dirname):
118   """Removes a temporary OS definition.
119
120   """
121   AssertCommand(["rm", "-rf", dirname], node=node)
122
123
124 def _TestOs(mode, rapi_cb):
125   """Generic function for OS definition testing
126
127   """
128   master = qa_config.GetMasterNode()
129
130   name = _TEMP_OS_NAME
131   dirname = _TEMP_OS_PATH
132
133   # Ensure OS is usable
134   cmd = ["gnt-os", "modify", "--hidden=no", "--blacklisted=no", name]
135   AssertCommand(cmd)
136
137   nodes = []
138   try:
139     for i, node in enumerate(qa_config.get("nodes")):
140       nodes.append(node)
141       if mode == _ALL_INVALID:
142         valid = False
143       elif mode == _ALL_VALID:
144         valid = True
145       elif mode == _PARTIALLY_VALID:
146         valid = bool(i % 2)
147       else:
148         raise AssertionError("Unknown mode %s" % mode)
149       _SetupTempOs(node, dirname, valid)
150
151     # TODO: Use Python 2.6's itertools.permutations
152     for (hidden, blacklisted) in [(False, False), (True, False),
153                                   (False, True), (True, True)]:
154       # Change OS' visibility
155       cmd = ["gnt-os", "modify", "--hidden", ["no", "yes"][int(hidden)],
156              "--blacklisted", ["no", "yes"][int(blacklisted)], name]
157       AssertCommand(cmd)
158
159       # Diagnose, checking exit status
160       AssertCommand(["gnt-os", "diagnose"], fail=(mode != _ALL_VALID))
161
162       # Diagnose again, ignoring exit status
163       output = qa_utils.GetCommandOutput(master["primary"],
164                                          "gnt-os diagnose || :")
165       for line in output.splitlines():
166         if line.startswith("OS: %s [global status:" % name):
167           break
168       else:
169         raise qa_error.Error("Didn't find OS '%s' in 'gnt-os diagnose'" % name)
170
171       # Check info for all
172       cmd = ["gnt-os", "info"]
173       output = qa_utils.GetCommandOutput(master["primary"],
174                                          utils.ShellQuoteArgs(cmd))
175       AssertIn("%s:" % name, output.splitlines())
176
177       # Check info for OS
178       cmd = ["gnt-os", "info", name]
179       output = qa_utils.GetCommandOutput(master["primary"],
180                                          utils.ShellQuoteArgs(cmd)).splitlines()
181       AssertIn("%s:" % name, output)
182       for (field, value) in [("valid", mode == _ALL_VALID),
183                              ("hidden", hidden),
184                              ("blacklisted", blacklisted)]:
185         AssertIn("  - %s: %s" % (field, value), output)
186
187       # Only valid OSes should be listed
188       cmd = ["gnt-os", "list", "--no-headers"]
189       output = qa_utils.GetCommandOutput(master["primary"],
190                                          utils.ShellQuoteArgs(cmd))
191       if mode == _ALL_VALID and not (hidden or blacklisted):
192         assert_fn = AssertIn
193       else:
194         assert_fn = AssertNotIn
195       assert_fn(name, output.splitlines())
196
197       # Check via RAPI
198       if rapi_cb:
199         assert_fn(name, rapi_cb())
200   finally:
201     for node in nodes:
202       _RemoveTempOs(node, dirname)
203
204
205 def TestOsValid(rapi_cb):
206   """Testing valid OS definition"""
207   return _TestOs(_ALL_VALID, rapi_cb)
208
209
210 def TestOsInvalid(rapi_cb):
211   """Testing invalid OS definition"""
212   return _TestOs(_ALL_INVALID, rapi_cb)
213
214
215 def TestOsPartiallyValid(rapi_cb):
216   """Testing partially valid OS definition"""
217   return _TestOs(_PARTIALLY_VALID, rapi_cb)
218
219
220 def TestOsModifyValid():
221   """Testing a valid os modify invocation"""
222   hv_dict = {
223     constants.HT_XEN_PVM: {
224       constants.HV_ROOT_PATH: "/dev/sda5",
225       },
226     constants.HT_XEN_HVM: {
227       constants.HV_ACPI: False,
228       constants.HV_PAE: True,
229       },
230     }
231
232   return _TestOsModify(hv_dict)
233
234
235 def TestOsModifyInvalid():
236   """Testing an invalid os modify invocation"""
237   hv_dict = {
238     "blahblahblubb": {"bar": ""},
239     }
240
241   return _TestOsModify(hv_dict, fail=True)
242
243
244 def TestOsStatesNonExisting():
245   """Testing OS states with non-existing OS"""
246   AssertCommand(["test", "-e", _TEMP_OS_PATH], fail=True)
247   return _TestOsStates(_TEMP_OS_NAME)