|
@@ -41,6 +41,41 @@ MOCK_FACTS = {
|
|
|
},
|
|
|
}
|
|
|
|
|
|
+MOCK_FACTS_QUICKHA = {
|
|
|
+ '10.0.0.1': {
|
|
|
+ 'common': {
|
|
|
+ 'ip': '10.0.0.1',
|
|
|
+ 'public_ip': '10.0.0.1',
|
|
|
+ 'hostname': 'master-private.example.com',
|
|
|
+ 'public_hostname': 'master.example.com'
|
|
|
+ }
|
|
|
+ },
|
|
|
+ '10.0.0.2': {
|
|
|
+ 'common': {
|
|
|
+ 'ip': '10.0.0.2',
|
|
|
+ 'public_ip': '10.0.0.2',
|
|
|
+ 'hostname': 'node1-private.example.com',
|
|
|
+ 'public_hostname': 'node1.example.com'
|
|
|
+ }
|
|
|
+ },
|
|
|
+ '10.0.0.3': {
|
|
|
+ 'common': {
|
|
|
+ 'ip': '10.0.0.3',
|
|
|
+ 'public_ip': '10.0.0.3',
|
|
|
+ 'hostname': 'node2-private.example.com',
|
|
|
+ 'public_hostname': 'node2.example.com'
|
|
|
+ }
|
|
|
+ },
|
|
|
+ '10.0.0.4': {
|
|
|
+ 'common': {
|
|
|
+ 'ip': '10.0.0.4',
|
|
|
+ 'public_ip': '10.0.0.4',
|
|
|
+ 'hostname': 'proxy-private.example.com',
|
|
|
+ 'public_hostname': 'proxy.example.com'
|
|
|
+ }
|
|
|
+ },
|
|
|
+}
|
|
|
+
|
|
|
# Substitute in a product name before use:
|
|
|
SAMPLE_CONFIG = """
|
|
|
variant: %s
|
|
@@ -91,6 +126,38 @@ hosts:
|
|
|
node: true
|
|
|
"""
|
|
|
|
|
|
+QUICKHA_CONFIG = """
|
|
|
+variant: %s
|
|
|
+ansible_ssh_user: root
|
|
|
+hosts:
|
|
|
+ - connect_to: 10.0.0.1
|
|
|
+ ip: 10.0.0.1
|
|
|
+ hostname: master-private.example.com
|
|
|
+ public_ip: 24.222.0.1
|
|
|
+ public_hostname: master.example.com
|
|
|
+ master: true
|
|
|
+ node: true
|
|
|
+ - connect_to: 10.0.0.2
|
|
|
+ ip: 10.0.0.2
|
|
|
+ hostname: node1-private.example.com
|
|
|
+ public_ip: 24.222.0.2
|
|
|
+ public_hostname: node1.example.com
|
|
|
+ master: true
|
|
|
+ node: true
|
|
|
+ - connect_to: 10.0.0.3
|
|
|
+ ip: 10.0.0.3
|
|
|
+ hostname: node2-private.example.com
|
|
|
+ public_ip: 24.222.0.3
|
|
|
+ public_hostname: node2.example.com
|
|
|
+ node: true
|
|
|
+ - connect_to: 10.0.0.4
|
|
|
+ ip: 10.0.0.4
|
|
|
+ hostname: proxy-private.example.com
|
|
|
+ public_ip: 24.222.0.4
|
|
|
+ public_hostname: proxy.example.com
|
|
|
+ master_lb: true
|
|
|
+"""
|
|
|
+
|
|
|
class OOCliFixture(OOInstallFixture):
|
|
|
|
|
|
def setUp(self):
|
|
@@ -145,11 +212,12 @@ class OOCliFixture(OOInstallFixture):
|
|
|
print written_config['hosts']
|
|
|
self.assertEquals(host_count, len(written_config['hosts']))
|
|
|
for h in written_config['hosts']:
|
|
|
- self.assertTrue(h['node'])
|
|
|
- self.assertTrue('ip' in h)
|
|
|
self.assertTrue('hostname' in h)
|
|
|
- self.assertTrue('public_ip' in h)
|
|
|
self.assertTrue('public_hostname' in h)
|
|
|
+ if 'preconfigured' not in h:
|
|
|
+ self.assertTrue(h['node'])
|
|
|
+ self.assertTrue('ip' in h)
|
|
|
+ self.assertTrue('public_ip' in h)
|
|
|
|
|
|
#pylint: disable=too-many-arguments
|
|
|
def _verify_get_hosts_to_run_on(self, mock_facts, load_facts_mock,
|
|
@@ -504,6 +572,40 @@ class UnattendedCliTests(OOCliFixture):
|
|
|
assert result.exit_code == 1
|
|
|
assert result.output == "You must specify either and 'ip' or 'hostname' to connect to.\n"
|
|
|
|
|
|
+ #unattended with two masters, one node, and haproxy
|
|
|
+ @patch('ooinstall.openshift_ansible.run_main_playbook')
|
|
|
+ @patch('ooinstall.openshift_ansible.load_system_facts')
|
|
|
+ def test_quick_ha_full_run(self, load_facts_mock, run_playbook_mock):
|
|
|
+ load_facts_mock.return_value = (MOCK_FACTS_QUICKHA, 0)
|
|
|
+ run_playbook_mock.return_value = 0
|
|
|
+
|
|
|
+ config_file = self.write_config(os.path.join(self.work_dir,
|
|
|
+ 'ooinstall.conf'), QUICKHA_CONFIG % 'openshift-enterprise')
|
|
|
+
|
|
|
+ self.cli_args.extend(["-c", config_file, "install"])
|
|
|
+ result = self.runner.invoke(cli.cli, self.cli_args)
|
|
|
+ self.assert_result(result, 0)
|
|
|
+
|
|
|
+ load_facts_args = load_facts_mock.call_args[0]
|
|
|
+ self.assertEquals(os.path.join(self.work_dir, ".ansible/hosts"),
|
|
|
+ load_facts_args[0])
|
|
|
+ self.assertEquals(os.path.join(self.work_dir,
|
|
|
+ "playbooks/byo/openshift_facts.yml"), load_facts_args[1])
|
|
|
+ env_vars = load_facts_args[2]
|
|
|
+ self.assertEquals(os.path.join(self.work_dir,
|
|
|
+ '.ansible/callback_facts.yaml'),
|
|
|
+ env_vars['OO_INSTALL_CALLBACK_FACTS_YAML'])
|
|
|
+ self.assertEqual('/tmp/ansible.log', env_vars['ANSIBLE_LOG_PATH'])
|
|
|
+ # If user running test has rpm installed, this might be set to default:
|
|
|
+ self.assertTrue('ANSIBLE_CONFIG' not in env_vars or
|
|
|
+ env_vars['ANSIBLE_CONFIG'] == cli.DEFAULT_ANSIBLE_CONFIG)
|
|
|
+
|
|
|
+ # Make sure we ran on the expected masters and nodes:
|
|
|
+ hosts = run_playbook_mock.call_args[0][0]
|
|
|
+ hosts_to_run_on = run_playbook_mock.call_args[0][1]
|
|
|
+ self.assertEquals(4, len(hosts))
|
|
|
+ self.assertEquals(4, len(hosts_to_run_on))
|
|
|
+
|
|
|
class AttendedCliTests(OOCliFixture):
|
|
|
|
|
|
def setUp(self):
|
|
@@ -512,9 +614,10 @@ class AttendedCliTests(OOCliFixture):
|
|
|
self.config_file = os.path.join(self.work_dir, 'config.yml')
|
|
|
self.cli_args.extend(["-c", self.config_file])
|
|
|
|
|
|
- #pylint: disable=too-many-arguments
|
|
|
+ #pylint: disable=too-many-arguments,too-many-branches
|
|
|
def _build_input(self, ssh_user=None, hosts=None, variant_num=None,
|
|
|
- add_nodes=None, confirm_facts=None):
|
|
|
+ add_nodes=None, confirm_facts=None, scheduleable_masters_ok=None,
|
|
|
+ master_lb=None):
|
|
|
"""
|
|
|
Builds a CLI input string with newline characters to simulate
|
|
|
the full run.
|
|
@@ -527,28 +630,40 @@ class AttendedCliTests(OOCliFixture):
|
|
|
if ssh_user:
|
|
|
inputs.append(ssh_user)
|
|
|
|
|
|
+ if variant_num:
|
|
|
+ inputs.append(str(variant_num)) # Choose variant + version
|
|
|
+
|
|
|
+ num_masters = 0
|
|
|
if hosts:
|
|
|
i = 0
|
|
|
+ min_masters_for_ha = 3
|
|
|
for (host, is_master) in hosts:
|
|
|
inputs.append(host)
|
|
|
- inputs.append('y' if is_master else 'n')
|
|
|
+ if is_master:
|
|
|
+ inputs.append('y')
|
|
|
+ num_masters += 1
|
|
|
+ else:
|
|
|
+ inputs.append('n')
|
|
|
#inputs.append('rpm')
|
|
|
if i < len(hosts) - 1:
|
|
|
- inputs.append('y') # Add more hosts
|
|
|
+ if num_masters <= 1 or num_masters >= min_masters_for_ha:
|
|
|
+ inputs.append('y') # Add more hosts
|
|
|
else:
|
|
|
inputs.append('n') # Done adding hosts
|
|
|
i += 1
|
|
|
|
|
|
- if variant_num:
|
|
|
- inputs.append(str(variant_num)) # Choose variant + version
|
|
|
+ if master_lb:
|
|
|
+ inputs.append(master_lb[0])
|
|
|
+ inputs.append('y' if master_lb[1] else 'n')
|
|
|
|
|
|
# TODO: support option 2, fresh install
|
|
|
if add_nodes:
|
|
|
+ if scheduleable_masters_ok:
|
|
|
+ inputs.append('y')
|
|
|
inputs.append('1') # Add more nodes
|
|
|
i = 0
|
|
|
for (host, is_master) in add_nodes:
|
|
|
inputs.append(host)
|
|
|
- inputs.append('y' if is_master else 'n')
|
|
|
#inputs.append('rpm')
|
|
|
if i < len(add_nodes) - 1:
|
|
|
inputs.append('y') # Add more hosts
|
|
@@ -556,6 +671,13 @@ class AttendedCliTests(OOCliFixture):
|
|
|
inputs.append('n') # Done adding hosts
|
|
|
i += 1
|
|
|
|
|
|
+ if add_nodes is None:
|
|
|
+ total_hosts = hosts
|
|
|
+ else:
|
|
|
+ total_hosts = hosts + add_nodes
|
|
|
+ if total_hosts is not None and num_masters == len(total_hosts):
|
|
|
+ inputs.append('y')
|
|
|
+
|
|
|
inputs.extend([
|
|
|
confirm_facts,
|
|
|
'y', # lets do this
|
|
@@ -587,6 +709,15 @@ class AttendedCliTests(OOCliFixture):
|
|
|
written_config = self._read_yaml(self.config_file)
|
|
|
self._verify_config_hosts(written_config, 3)
|
|
|
|
|
|
+ inventory = ConfigParser.ConfigParser(allow_no_value=True)
|
|
|
+ inventory.read(os.path.join(self.work_dir, '.ansible/hosts'))
|
|
|
+ self.assertEquals('False',
|
|
|
+ inventory.get('nodes', '10.0.0.1 openshift_scheduleable'))
|
|
|
+ self.assertEquals(None,
|
|
|
+ inventory.get('nodes', '10.0.0.2'))
|
|
|
+ self.assertEquals(None,
|
|
|
+ inventory.get('nodes', '10.0.0.3'))
|
|
|
+
|
|
|
# interactive with config file and some installed some uninstalled hosts
|
|
|
@patch('ooinstall.openshift_ansible.run_main_playbook')
|
|
|
@patch('ooinstall.openshift_ansible.load_system_facts')
|
|
@@ -613,6 +744,7 @@ class AttendedCliTests(OOCliFixture):
|
|
|
result = self.runner.invoke(cli.cli,
|
|
|
self.cli_args,
|
|
|
input=cli_input)
|
|
|
+ print result
|
|
|
self.assert_result(result, 0)
|
|
|
|
|
|
self._verify_load_facts(load_facts_mock)
|
|
@@ -658,6 +790,7 @@ class AttendedCliTests(OOCliFixture):
|
|
|
add_nodes=[('10.0.0.2', False)],
|
|
|
ssh_user='root',
|
|
|
variant_num=1,
|
|
|
+ scheduleable_masters_ok=True,
|
|
|
confirm_facts='y')
|
|
|
|
|
|
self._verify_get_hosts_to_run_on(mock_facts, load_facts_mock,
|
|
@@ -667,6 +800,113 @@ class AttendedCliTests(OOCliFixture):
|
|
|
exp_hosts_to_run_on_len=2,
|
|
|
force=False)
|
|
|
|
|
|
+ #interactive multimaster: one more node than master
|
|
|
+ @patch('ooinstall.openshift_ansible.run_main_playbook')
|
|
|
+ @patch('ooinstall.openshift_ansible.load_system_facts')
|
|
|
+ def test_quick_ha1(self, load_facts_mock, run_playbook_mock):
|
|
|
+ load_facts_mock.return_value = (MOCK_FACTS_QUICKHA, 0)
|
|
|
+ run_playbook_mock.return_value = 0
|
|
|
+
|
|
|
+ cli_input = self._build_input(hosts=[
|
|
|
+ ('10.0.0.1', True),
|
|
|
+ ('10.0.0.2', True),
|
|
|
+ ('10.0.0.3', False),
|
|
|
+ ('10.0.0.4', True)],
|
|
|
+ ssh_user='root',
|
|
|
+ variant_num=1,
|
|
|
+ confirm_facts='y',
|
|
|
+ master_lb=('10.0.0.5', False))
|
|
|
+ self.cli_args.append("install")
|
|
|
+ result = self.runner.invoke(cli.cli, self.cli_args,
|
|
|
+ input=cli_input)
|
|
|
+ self.assert_result(result, 0)
|
|
|
+
|
|
|
+ self._verify_load_facts(load_facts_mock)
|
|
|
+ self._verify_run_playbook(run_playbook_mock, 5, 5)
|
|
|
+
|
|
|
+ written_config = self._read_yaml(self.config_file)
|
|
|
+ self._verify_config_hosts(written_config, 5)
|
|
|
+
|
|
|
+ inventory = ConfigParser.ConfigParser(allow_no_value=True)
|
|
|
+ inventory.read(os.path.join(self.work_dir, '.ansible/hosts'))
|
|
|
+ self.assertEquals('False',
|
|
|
+ inventory.get('nodes', '10.0.0.1 openshift_scheduleable'))
|
|
|
+ self.assertEquals('False',
|
|
|
+ inventory.get('nodes', '10.0.0.2 openshift_scheduleable'))
|
|
|
+ self.assertEquals(None,
|
|
|
+ inventory.get('nodes', '10.0.0.3'))
|
|
|
+ self.assertEquals('False',
|
|
|
+ inventory.get('nodes', '10.0.0.4 openshift_scheduleable'))
|
|
|
+
|
|
|
+ return
|
|
|
+
|
|
|
+ #interactive multimaster: equal number masters and nodes
|
|
|
+ @patch('ooinstall.openshift_ansible.run_main_playbook')
|
|
|
+ @patch('ooinstall.openshift_ansible.load_system_facts')
|
|
|
+ def test_quick_ha2(self, load_facts_mock, run_playbook_mock):
|
|
|
+ load_facts_mock.return_value = (MOCK_FACTS_QUICKHA, 0)
|
|
|
+ run_playbook_mock.return_value = 0
|
|
|
+
|
|
|
+ cli_input = self._build_input(hosts=[
|
|
|
+ ('10.0.0.1', True),
|
|
|
+ ('10.0.0.2', True),
|
|
|
+ ('10.0.0.3', True)],
|
|
|
+ ssh_user='root',
|
|
|
+ variant_num=1,
|
|
|
+ confirm_facts='y',
|
|
|
+ master_lb=('10.0.0.5', False))
|
|
|
+ self.cli_args.append("install")
|
|
|
+ result = self.runner.invoke(cli.cli, self.cli_args,
|
|
|
+ input=cli_input)
|
|
|
+ self.assert_result(result, 0)
|
|
|
+
|
|
|
+ self._verify_load_facts(load_facts_mock)
|
|
|
+ self._verify_run_playbook(run_playbook_mock, 4, 4)
|
|
|
+
|
|
|
+ written_config = self._read_yaml(self.config_file)
|
|
|
+ self._verify_config_hosts(written_config, 4)
|
|
|
+
|
|
|
+ inventory = ConfigParser.ConfigParser(allow_no_value=True)
|
|
|
+ inventory.read(os.path.join(self.work_dir, '.ansible/hosts'))
|
|
|
+ self.assertEquals(None,
|
|
|
+ inventory.get('nodes', '10.0.0.1'))
|
|
|
+ self.assertEquals(None,
|
|
|
+ inventory.get('nodes', '10.0.0.2'))
|
|
|
+ self.assertEquals(None,
|
|
|
+ inventory.get('nodes', '10.0.0.3'))
|
|
|
+
|
|
|
+ return
|
|
|
+
|
|
|
+ #interactive all-in-one
|
|
|
+ @patch('ooinstall.openshift_ansible.run_main_playbook')
|
|
|
+ @patch('ooinstall.openshift_ansible.load_system_facts')
|
|
|
+ def test_all_in_one(self, load_facts_mock, run_playbook_mock):
|
|
|
+ load_facts_mock.return_value = (MOCK_FACTS, 0)
|
|
|
+ run_playbook_mock.return_value = 0
|
|
|
+
|
|
|
+ cli_input = self._build_input(hosts=[
|
|
|
+ ('10.0.0.1', True)],
|
|
|
+ ssh_user='root',
|
|
|
+ variant_num=1,
|
|
|
+ confirm_facts='y')
|
|
|
+ self.cli_args.append("install")
|
|
|
+ result = self.runner.invoke(cli.cli, self.cli_args,
|
|
|
+ input=cli_input)
|
|
|
+ self.assert_result(result, 0)
|
|
|
+
|
|
|
+ self._verify_load_facts(load_facts_mock)
|
|
|
+ self._verify_run_playbook(run_playbook_mock, 1, 1)
|
|
|
+
|
|
|
+ written_config = self._read_yaml(self.config_file)
|
|
|
+ self._verify_config_hosts(written_config, 1)
|
|
|
+
|
|
|
+ inventory = ConfigParser.ConfigParser(allow_no_value=True)
|
|
|
+ inventory.read(os.path.join(self.work_dir, '.ansible/hosts'))
|
|
|
+ self.assertEquals(None,
|
|
|
+ inventory.get('nodes', '10.0.0.1'))
|
|
|
+
|
|
|
+ return
|
|
|
+
|
|
|
# TODO: test with config file, attended add node
|
|
|
# TODO: test with config file, attended new node already in config file
|
|
|
# TODO: test with config file, attended new node already in config file, plus manually added nodes
|