fixture.py 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. # pylint: disable=missing-docstring
  2. import os
  3. import yaml
  4. import ooinstall.cli_installer as cli
  5. from test.oo_config_tests import OOInstallFixture
  6. from click.testing import CliRunner
  7. # Substitute in a product name before use:
  8. SAMPLE_CONFIG = """
  9. variant: %s
  10. variant_version: 3.3
  11. master_routingconfig_subdomain: example.com
  12. version: v2
  13. deployment:
  14. ansible_ssh_user: root
  15. hosts:
  16. - connect_to: 10.0.0.1
  17. ip: 10.0.0.1
  18. hostname: master-private.example.com
  19. public_ip: 24.222.0.1
  20. public_hostname: master.example.com
  21. roles:
  22. - master
  23. - node
  24. - connect_to: 10.0.0.2
  25. ip: 10.0.0.2
  26. hostname: node1-private.example.com
  27. public_ip: 24.222.0.2
  28. public_hostname: node1.example.com
  29. roles:
  30. - node
  31. - connect_to: 10.0.0.3
  32. ip: 10.0.0.3
  33. hostname: node2-private.example.com
  34. public_ip: 24.222.0.3
  35. public_hostname: node2.example.com
  36. roles:
  37. - node
  38. roles:
  39. master:
  40. node:
  41. """
  42. def read_yaml(config_file_path):
  43. cfg_f = open(config_file_path, 'r')
  44. config = yaml.safe_load(cfg_f.read())
  45. cfg_f.close()
  46. return config
  47. class OOCliFixture(OOInstallFixture):
  48. def setUp(self):
  49. OOInstallFixture.setUp(self)
  50. self.runner = CliRunner()
  51. # Add any arguments you would like to test here, the defaults ensure
  52. # we only do unattended invocations here, and using temporary files/dirs.
  53. self.cli_args = ["-a", self.work_dir]
  54. def run_cli(self):
  55. return self.runner.invoke(cli.cli, self.cli_args)
  56. def assert_result(self, result, exit_code):
  57. if result.exit_code != exit_code:
  58. print "Unexpected result from CLI execution"
  59. print "Exit code: %s" % result.exit_code
  60. print "Exception: %s" % result.exception
  61. print result.exc_info
  62. import traceback
  63. traceback.print_exception(*result.exc_info)
  64. print "Output:\n%s" % result.output
  65. self.fail("Exception during CLI execution")
  66. def _verify_load_facts(self, load_facts_mock):
  67. """ Check that we ran load facts with expected inputs. """
  68. load_facts_args = load_facts_mock.call_args[0]
  69. self.assertEquals(os.path.join(self.work_dir, "hosts"),
  70. load_facts_args[0])
  71. self.assertEquals(os.path.join(self.work_dir,
  72. "playbooks/byo/openshift_facts.yml"),
  73. load_facts_args[1])
  74. env_vars = load_facts_args[2]
  75. self.assertEquals(os.path.join(self.work_dir,
  76. '.ansible/callback_facts.yaml'),
  77. env_vars['OO_INSTALL_CALLBACK_FACTS_YAML'])
  78. self.assertEqual('/tmp/ansible.log', env_vars['ANSIBLE_LOG_PATH'])
  79. def _verify_run_playbook(self, run_playbook_mock, exp_hosts_len, exp_hosts_to_run_on_len):
  80. """ Check that we ran playbook with expected inputs. """
  81. hosts = run_playbook_mock.call_args[0][1]
  82. hosts_to_run_on = run_playbook_mock.call_args[0][2]
  83. self.assertEquals(exp_hosts_len, len(hosts))
  84. self.assertEquals(exp_hosts_to_run_on_len, len(hosts_to_run_on))
  85. def _verify_config_hosts(self, written_config, host_count):
  86. self.assertEquals(host_count, len(written_config['deployment']['hosts']))
  87. for host in written_config['deployment']['hosts']:
  88. self.assertTrue('hostname' in host)
  89. self.assertTrue('public_hostname' in host)
  90. if 'preconfigured' not in host:
  91. if 'roles' in host:
  92. self.assertTrue('node' in host['roles'] or 'storage' in host['roles'])
  93. self.assertTrue('ip' in host)
  94. self.assertTrue('public_ip' in host)
  95. #pylint: disable=too-many-arguments
  96. def _verify_get_hosts_to_run_on(self, mock_facts, load_facts_mock,
  97. run_playbook_mock, cli_input,
  98. exp_hosts_len=None, exp_hosts_to_run_on_len=None,
  99. force=None):
  100. """
  101. Tests cli_installer.py:get_hosts_to_run_on. That method has quite a
  102. few subtle branches in the logic. The goal with this method is simply
  103. to handle all the messy stuff here and allow the main test cases to be
  104. easily read. The basic idea is to modify mock_facts to return a
  105. version indicating OpenShift is already installed on particular hosts.
  106. """
  107. load_facts_mock.return_value = (mock_facts, 0)
  108. run_playbook_mock.return_value = 0
  109. if cli_input:
  110. self.cli_args.append("install")
  111. result = self.runner.invoke(cli.cli,
  112. self.cli_args,
  113. input=cli_input)
  114. else:
  115. config_file = self.write_config(
  116. os.path.join(self.work_dir,
  117. 'ooinstall.conf'), SAMPLE_CONFIG % 'openshift-enterprise')
  118. self.cli_args.extend(["-c", config_file, "install"])
  119. if force:
  120. self.cli_args.append("--force")
  121. result = self.runner.invoke(cli.cli, self.cli_args)
  122. written_config = read_yaml(config_file)
  123. self._verify_config_hosts(written_config, exp_hosts_len)
  124. if "Uninstalled" in result.output:
  125. # verify we exited on seeing uninstalled hosts
  126. self.assertEqual(result.exit_code, 1)
  127. else:
  128. self.assert_result(result, 0)
  129. self._verify_load_facts(load_facts_mock)
  130. self._verify_run_playbook(run_playbook_mock, exp_hosts_len, exp_hosts_to_run_on_len)
  131. # Make sure we ran on the expected masters and nodes:
  132. hosts = run_playbook_mock.call_args[0][1]
  133. hosts_to_run_on = run_playbook_mock.call_args[0][2]
  134. self.assertEquals(exp_hosts_len, len(hosts))
  135. self.assertEquals(exp_hosts_to_run_on_len, len(hosts_to_run_on))
  136. #pylint: disable=too-many-arguments,too-many-branches,too-many-statements
  137. def build_input(ssh_user=None, hosts=None, variant_num=None,
  138. add_nodes=None, confirm_facts=None, schedulable_masters_ok=None,
  139. master_lb=None, storage=None):
  140. """
  141. Build an input string simulating a user entering values in an interactive
  142. attended install.
  143. This is intended to give us one place to update when the CLI prompts change.
  144. We should aim to keep this dependent on optional keyword arguments with
  145. sensible defaults to keep things from getting too fragile.
  146. """
  147. inputs = [
  148. 'y', # let's proceed
  149. ]
  150. if ssh_user:
  151. inputs.append(ssh_user)
  152. if variant_num:
  153. inputs.append(str(variant_num)) # Choose variant + version
  154. num_masters = 0
  155. if hosts:
  156. i = 0
  157. for (host, is_master, is_containerized) in hosts:
  158. inputs.append(host)
  159. if is_master:
  160. inputs.append('y')
  161. num_masters += 1
  162. else:
  163. inputs.append('n')
  164. if is_containerized:
  165. inputs.append('container')
  166. else:
  167. inputs.append('rpm')
  168. #inputs.append('rpm')
  169. # We should not be prompted to add more hosts if we're currently at
  170. # 2 masters, this is an invalid HA configuration, so this question
  171. # will not be asked, and the user must enter the next host:
  172. if num_masters != 2:
  173. if i < len(hosts) - 1:
  174. if num_masters >= 1:
  175. inputs.append('y') # Add more hosts
  176. else:
  177. inputs.append('n') # Done adding hosts
  178. i += 1
  179. # You can pass a single master_lb or a list if you intend for one to get rejected:
  180. if master_lb:
  181. if isinstance(master_lb[0], list) or isinstance(master_lb[0], tuple):
  182. inputs.extend(master_lb[0])
  183. else:
  184. inputs.append(master_lb[0])
  185. inputs.append('y' if master_lb[1] else 'n')
  186. if storage:
  187. inputs.append(storage)
  188. inputs.append('subdomain.example.com')
  189. inputs.append('proxy.example.com')
  190. inputs.append('proxy-private.example.com')
  191. inputs.append('exclude.example.com')
  192. # TODO: support option 2, fresh install
  193. if add_nodes:
  194. if schedulable_masters_ok:
  195. inputs.append('y')
  196. inputs.append('1') # Add more nodes
  197. i = 0
  198. for (host, is_master, is_containerized) in add_nodes:
  199. inputs.append(host)
  200. if is_containerized:
  201. inputs.append('container')
  202. else:
  203. inputs.append('rpm')
  204. #inputs.append('rpm')
  205. if i < len(add_nodes) - 1:
  206. inputs.append('y') # Add more hosts
  207. else:
  208. inputs.append('n') # Done adding hosts
  209. i += 1
  210. if add_nodes is None:
  211. total_hosts = hosts
  212. else:
  213. total_hosts = hosts + add_nodes
  214. if total_hosts is not None and num_masters == len(total_hosts):
  215. inputs.append('y')
  216. inputs.extend([
  217. confirm_facts,
  218. 'y', # lets do this
  219. ])
  220. return '\n'.join(inputs)