| Oleksii Zhurba | 020fab4 | 2017-11-01 20:13:28 +0000 | [diff] [blame] | 1 | import json | 
 | 2 | import requests | 
| Oleksii Zhurba | 84ce7fe | 2018-01-16 21:34:01 +0000 | [diff] [blame] | 3 | import datetime | 
| Oleksii Zhurba | 9848e21 | 2018-09-05 10:53:51 -0500 | [diff] [blame] | 4 | import pytest | 
| Oleksii Zhurba | 020fab4 | 2017-11-01 20:13:28 +0000 | [diff] [blame] | 5 |  | 
| Oleksii Zhurba | 23c1833 | 2019-05-09 18:53:40 -0500 | [diff] [blame] | 6 | @pytest.mark.sl_dup | 
 | 7 | #ElasticsearchClusterHealthStatusMajor or stacklight-pytest | 
 | 8 | @pytest.mark.full | 
| Oleksii Zhurba | 8ce9fcf | 2018-10-05 18:38:22 +0300 | [diff] [blame] | 9 | @pytest.mark.usefixtures('check_kibana') | 
| Oleksii Zhurba | 020fab4 | 2017-11-01 20:13:28 +0000 | [diff] [blame] | 10 | def test_elasticsearch_cluster(local_salt_client): | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 11 |     salt_output = local_salt_client.pillar_get( | 
 | 12 |         tgt='kibana:server', | 
 | 13 |         param='_param:haproxy_elasticsearch_bind_host') | 
| Oleksii Zhurba | 7d4f07e | 2019-06-10 17:30:53 -0500 | [diff] [blame] | 14 |     ssl = local_salt_client.pillar_get( | 
 | 15 |         tgt='elasticsearch:server', | 
 | 16 |         param='haproxy:proxy:listen:elasticsearch:binds:ssl:enabled') | 
 | 17 |     proto = "https" if ssl == "True" else "http" | 
| Oleksii Zhurba | 9848e21 | 2018-09-05 10:53:51 -0500 | [diff] [blame] | 18 |  | 
| Oleksii Zhurba | e592ed1 | 2018-06-21 18:01:09 -0500 | [diff] [blame] | 19 |     proxies = {"http": None, "https": None} | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 20 |     IP = salt_output | 
| Oleksii Zhurba | 7d4f07e | 2019-06-10 17:30:53 -0500 | [diff] [blame] | 21 |     assert requests.get('{0}://{1}:9200/'.format(proto, IP), | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 22 |                         proxies=proxies).status_code == 200, \ | 
 | 23 |         'Cannot check elasticsearch url on {}.'.format(IP) | 
| Oleksii Zhurba | 7d4f07e | 2019-06-10 17:30:53 -0500 | [diff] [blame] | 24 |     resp = requests.get('{0}://{1}:9200/_cat/health'.format(proto, IP), | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 25 |                         proxies=proxies).content | 
 | 26 |     assert resp.split()[3] == 'green', \ | 
 | 27 |         'elasticsearch status is not good {}'.format( | 
 | 28 |         json.dumps(resp, indent=4)) | 
 | 29 |     assert resp.split()[4] == '3', \ | 
 | 30 |         'elasticsearch status is not good {}'.format( | 
 | 31 |         json.dumps(resp, indent=4)) | 
 | 32 |     assert resp.split()[5] == '3', \ | 
 | 33 |         'elasticsearch status is not good {}'.format( | 
 | 34 |         json.dumps(resp, indent=4)) | 
 | 35 |     assert resp.split()[10] == '0', \ | 
 | 36 |         'elasticsearch status is not good {}'.format( | 
 | 37 |         json.dumps(resp, indent=4)) | 
 | 38 |     assert resp.split()[13] == '100.0%', \ | 
 | 39 |         'elasticsearch status is not good {}'.format( | 
 | 40 |         json.dumps(resp, indent=4)) | 
| Oleksii Zhurba | 020fab4 | 2017-11-01 20:13:28 +0000 | [diff] [blame] | 41 |  | 
 | 42 |  | 
| Oleksii Zhurba | 23c1833 | 2019-05-09 18:53:40 -0500 | [diff] [blame] | 43 | @pytest.mark.sl_dup | 
 | 44 | #stacklight-pytest | 
 | 45 | @pytest.mark.full | 
| Oleksii Zhurba | 8ce9fcf | 2018-10-05 18:38:22 +0300 | [diff] [blame] | 46 | @pytest.mark.usefixtures('check_kibana') | 
| Ievgeniia Zadorozhna | 511f0ce | 2018-11-08 17:43:10 +0300 | [diff] [blame] | 47 | def test_kibana_status(local_salt_client): | 
 | 48 |     proxies = {"http": None, "https": None} | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 49 |     IP = local_salt_client.pillar_get(param='_param:stacklight_log_address') | 
| Oleksii Zhurba | 7d4f07e | 2019-06-10 17:30:53 -0500 | [diff] [blame] | 50 |     ssl = local_salt_client.pillar_get( | 
 | 51 |         tgt='kibana:server', | 
 | 52 |         param='haproxy:proxy:listen:kibana:binds:ssl:enabled') | 
 | 53 |     proto = "https" if ssl == "True" else "http" | 
 | 54 |  | 
 | 55 |     resp = requests.get('{0}://{1}:5601/api/status'.format(proto, IP), | 
| Ievgeniia Zadorozhna | 511f0ce | 2018-11-08 17:43:10 +0300 | [diff] [blame] | 56 |                         proxies=proxies).content | 
 | 57 |     body = json.loads(resp) | 
 | 58 |     assert body['status']['overall']['state'] == "green", \ | 
 | 59 |         "Kibana status is not expected: {}".format( | 
 | 60 |         body['status']['overall']) | 
 | 61 |     for i in body['status']['statuses']: | 
 | 62 |         assert i['state'] == "green", \ | 
 | 63 |             "Kibana statuses are unexpected: {}".format(i) | 
 | 64 |  | 
 | 65 |  | 
| Oleksii Zhurba | 23c1833 | 2019-05-09 18:53:40 -0500 | [diff] [blame] | 66 | @pytest.mark.smoke | 
 | 67 | #TODO: recheck | 
| Ievgeniia Zadorozhna | 511f0ce | 2018-11-08 17:43:10 +0300 | [diff] [blame] | 68 | @pytest.mark.usefixtures('check_kibana') | 
| Oleksii Zhurba | 84ce7fe | 2018-01-16 21:34:01 +0000 | [diff] [blame] | 69 | def test_elasticsearch_node_count(local_salt_client): | 
 | 70 |     now = datetime.datetime.now() | 
 | 71 |     today = now.strftime("%Y.%m.%d") | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 72 |     salt_output = local_salt_client.pillar_get( | 
 | 73 |         tgt='kibana:server', | 
 | 74 |         param='_param:haproxy_elasticsearch_bind_host') | 
| Oleksii Zhurba | 9848e21 | 2018-09-05 10:53:51 -0500 | [diff] [blame] | 75 |  | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 76 |     IP = salt_output | 
| Oleksii Zhurba | 7d4f07e | 2019-06-10 17:30:53 -0500 | [diff] [blame] | 77 |     ssl = local_salt_client.pillar_get( | 
 | 78 |         tgt='elasticsearch:server', | 
 | 79 |         param='haproxy:proxy:listen:elasticsearch:binds:ssl:enabled') | 
 | 80 |     proto = "https" if ssl == "True" else "http" | 
 | 81 |  | 
| Tatyana Leontovich | ed57cb0 | 2019-01-11 16:26:32 +0200 | [diff] [blame] | 82 |     headers = {'Content-type': 'application/json', 'Accept': 'text/plain'} | 
| Oleksii Zhurba | e592ed1 | 2018-06-21 18:01:09 -0500 | [diff] [blame] | 83 |     proxies = {"http": None, "https": None} | 
| Tatyana Leontovich | ed57cb0 | 2019-01-11 16:26:32 +0200 | [diff] [blame] | 84 |     data = ('{"size": 0, "aggs": ' | 
 | 85 |             '{"uniq_hostname": ' | 
| Oleksii Zhurba | 23c1833 | 2019-05-09 18:53:40 -0500 | [diff] [blame] | 86 |             '{"terms": {"size": 500, ' | 
| Tatyana Leontovich | ed57cb0 | 2019-01-11 16:26:32 +0200 | [diff] [blame] | 87 |             '"field": "Hostname.keyword"}}}}') | 
 | 88 |     response = requests.post( | 
| Oleksii Zhurba | 7d4f07e | 2019-06-10 17:30:53 -0500 | [diff] [blame] | 89 |         '{0}://{1}:9200/log-{2}/_search?pretty'.format(proto, IP, today), | 
| Tatyana Leontovich | ed57cb0 | 2019-01-11 16:26:32 +0200 | [diff] [blame] | 90 |         proxies=proxies, | 
 | 91 |         headers=headers, | 
 | 92 |         data=data) | 
 | 93 |     assert 200 == response.status_code, 'Unexpected code {}'.format( | 
 | 94 |         response.text) | 
 | 95 |     resp = json.loads(response.text) | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 96 |     cluster_domain = local_salt_client.pillar_get(param='_param:cluster_domain') | 
| Oleksii Zhurba | d2847dc | 2018-02-16 15:13:09 -0600 | [diff] [blame] | 97 |     monitored_nodes = [] | 
| Oleksii Zhurba | 7f46341 | 2018-03-21 16:32:44 -0500 | [diff] [blame] | 98 |     for item_ in resp['aggregations']['uniq_hostname']['buckets']: | 
| Oleksii Zhurba | d2847dc | 2018-02-16 15:13:09 -0600 | [diff] [blame] | 99 |         node_name = item_['key'] | 
 | 100 |         monitored_nodes.append(node_name + '.' + cluster_domain) | 
 | 101 |     missing_nodes = [] | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 102 |     all_nodes = local_salt_client.test_ping(tgt='*').keys() | 
 | 103 |     for node in all_nodes: | 
| Oleksii Zhurba | d2847dc | 2018-02-16 15:13:09 -0600 | [diff] [blame] | 104 |         if node not in monitored_nodes: | 
 | 105 |             missing_nodes.append(node) | 
 | 106 |     assert len(missing_nodes) == 0, \ | 
| Oleksii Zhurba | 84ce7fe | 2018-01-16 21:34:01 +0000 | [diff] [blame] | 107 |         'Not all nodes are in Elasticsearch. Found {0} keys, ' \ | 
| Oleksii Zhurba | d2847dc | 2018-02-16 15:13:09 -0600 | [diff] [blame] | 108 |         'expected {1}. Missing nodes: \n{2}'. \ | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 109 |             format(len(monitored_nodes), len(all_nodes), missing_nodes) | 
| Oleksii Zhurba | 84ce7fe | 2018-01-16 21:34:01 +0000 | [diff] [blame] | 110 |  | 
 | 111 |  | 
| Oleksii Zhurba | 23c1833 | 2019-05-09 18:53:40 -0500 | [diff] [blame] | 112 | @pytest.mark.sl_dup | 
 | 113 | #DockerServiceMonitoring* | 
 | 114 | @pytest.mark.full | 
| Oleksii Zhurba | 020fab4 | 2017-11-01 20:13:28 +0000 | [diff] [blame] | 115 | def test_stacklight_services_replicas(local_salt_client): | 
| Oleksii Zhurba | 8ce9fcf | 2018-10-05 18:38:22 +0300 | [diff] [blame] | 116 |     # TODO | 
 | 117 |     # change to docker:swarm:role:master ? | 
| Oleksii Zhurba | 020fab4 | 2017-11-01 20:13:28 +0000 | [diff] [blame] | 118 |     salt_output = local_salt_client.cmd( | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 119 |         tgt='I@docker:client:stack:monitoring and I@prometheus:server', | 
 | 120 |         param='docker service ls', | 
| Oleksii Zhurba | 8ce9fcf | 2018-10-05 18:38:22 +0300 | [diff] [blame] | 121 |         expr_form='compound') | 
| Oleksii Zhurba | 9848e21 | 2018-09-05 10:53:51 -0500 | [diff] [blame] | 122 |  | 
 | 123 |     if not salt_output: | 
| Oleksii Zhurba | 8ce9fcf | 2018-10-05 18:38:22 +0300 | [diff] [blame] | 124 |         pytest.skip("docker:client:stack:monitoring or \ | 
 | 125 |         prometheus:server pillars are not found on this environment.") | 
| Oleksii Zhurba | 9848e21 | 2018-09-05 10:53:51 -0500 | [diff] [blame] | 126 |  | 
| Oleksii Zhurba | 020fab4 | 2017-11-01 20:13:28 +0000 | [diff] [blame] | 127 |     wrong_items = [] | 
 | 128 |     for line in salt_output[salt_output.keys()[0]].split('\n'): | 
 | 129 |         if line[line.find('/') - 1] != line[line.find('/') + 1] \ | 
 | 130 |            and 'replicated' in line: | 
 | 131 |             wrong_items.append(line) | 
 | 132 |     assert len(wrong_items) == 0, \ | 
 | 133 |         '''Some monitoring services doesn't have expected number of replicas: | 
 | 134 |               {}'''.format(json.dumps(wrong_items, indent=4)) | 
 | 135 |  | 
 | 136 |  | 
| Oleksii Zhurba | 23c1833 | 2019-05-09 18:53:40 -0500 | [diff] [blame] | 137 | @pytest.mark.smoke | 
| Oleksii Zhurba | 8ce9fcf | 2018-10-05 18:38:22 +0300 | [diff] [blame] | 138 | @pytest.mark.usefixtures('check_prometheus') | 
| Oleksii Zhurba | 2c2dc94 | 2019-01-31 16:35:57 -0600 | [diff] [blame] | 139 | def test_prometheus_alert_count(local_salt_client, ctl_nodes_pillar): | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 140 |     IP = local_salt_client.pillar_get(param='_param:cluster_public_host') | 
| Oleksii Zhurba | 7d4f07e | 2019-06-10 17:30:53 -0500 | [diff] [blame] | 141 |     proto = local_salt_client.pillar_get( | 
 | 142 |         param='_param:cluster_public_protocol') | 
| Oleksii Zhurba | 468e6c7 | 2018-01-16 17:43:15 +0000 | [diff] [blame] | 143 |     # keystone:server can return 3 nodes instead of 1 | 
 | 144 |     # this will be fixed later | 
 | 145 |     # TODO | 
| Oleksii Zhurba | 0c039ee | 2018-01-16 19:44:53 +0000 | [diff] [blame] | 146 |     nodes_info = local_salt_client.cmd( | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 147 |         tgt=ctl_nodes_pillar, | 
| Oleksii Zhurba | 7d4f07e | 2019-06-10 17:30:53 -0500 | [diff] [blame] | 148 |         param='curl -s {0}://{1}:15010/alerts | grep icon-chevron-down | ' | 
 | 149 |               'grep -v "0 active"'.format(proto, IP), | 
| Oleksii Zhurba | 468e6c7 | 2018-01-16 17:43:15 +0000 | [diff] [blame] | 150 |         expr_form='pillar') | 
| Oleksii Zhurba | 9848e21 | 2018-09-05 10:53:51 -0500 | [diff] [blame] | 151 |  | 
| Oleksii Zhurba | 0c039ee | 2018-01-16 19:44:53 +0000 | [diff] [blame] | 152 |     result = nodes_info[nodes_info.keys()[0]].replace('</td>', '').replace( | 
 | 153 |         '<td><i class="icon-chevron-down"></i> <b>', '').replace('</b>', '') | 
 | 154 |     assert result == '', 'AlertManager page has some alerts! {}'.format( | 
 | 155 |                          json.dumps(result), indent=4) | 
| Oleksii Zhurba | 468e6c7 | 2018-01-16 17:43:15 +0000 | [diff] [blame] | 156 |  | 
 | 157 |  | 
| Oleksii Zhurba | 23c1833 | 2019-05-09 18:53:40 -0500 | [diff] [blame] | 158 | @pytest.mark.sl_dup | 
 | 159 | #DockerServiceMonitoring* ?? | 
 | 160 | @pytest.mark.full | 
| Oleksii Zhurba | 020fab4 | 2017-11-01 20:13:28 +0000 | [diff] [blame] | 161 | def test_stacklight_containers_status(local_salt_client): | 
 | 162 |     salt_output = local_salt_client.cmd( | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 163 |         tgt='I@docker:swarm:role:master and I@prometheus:server', | 
 | 164 |         param='docker service ps $(docker stack services -q monitoring)', | 
| Oleksii Zhurba | 8ce9fcf | 2018-10-05 18:38:22 +0300 | [diff] [blame] | 165 |         expr_form='compound') | 
| Oleksii Zhurba | 9848e21 | 2018-09-05 10:53:51 -0500 | [diff] [blame] | 166 |  | 
 | 167 |     if not salt_output: | 
| Oleksii Zhurba | 8ce9fcf | 2018-10-05 18:38:22 +0300 | [diff] [blame] | 168 |         pytest.skip("docker:swarm:role:master or prometheus:server \ | 
 | 169 |         pillars are not found on this environment.") | 
| Oleksii Zhurba | 9848e21 | 2018-09-05 10:53:51 -0500 | [diff] [blame] | 170 |  | 
| Oleksii Zhurba | 020fab4 | 2017-11-01 20:13:28 +0000 | [diff] [blame] | 171 |     result = {} | 
| Oleksii Zhurba | 468e6c7 | 2018-01-16 17:43:15 +0000 | [diff] [blame] | 172 |     # for old reclass models, docker:swarm:role:master can return | 
 | 173 |     # 2 nodes instead of one. Here is temporary fix. | 
 | 174 |     # TODO | 
 | 175 |     if len(salt_output.keys()) > 1: | 
 | 176 |         if 'CURRENT STATE' not in salt_output[salt_output.keys()[0]]: | 
 | 177 |             del salt_output[salt_output.keys()[0]] | 
| Oleksii Zhurba | f2af637 | 2017-11-01 22:53:03 +0000 | [diff] [blame] | 178 |     for line in salt_output[salt_output.keys()[0]].split('\n')[1:]: | 
| Oleksii Zhurba | 020fab4 | 2017-11-01 20:13:28 +0000 | [diff] [blame] | 179 |         shift = 0 | 
| Oleksii Zhurba | 020fab4 | 2017-11-01 20:13:28 +0000 | [diff] [blame] | 180 |         if line.split()[1] == '\\_': | 
 | 181 |             shift = 1 | 
 | 182 |         if line.split()[1 + shift] not in result.keys(): | 
 | 183 |             result[line.split()[1]] = 'NOT OK' | 
 | 184 |         if line.split()[4 + shift] == 'Running' \ | 
 | 185 |            or line.split()[4 + shift] == 'Ready': | 
 | 186 |             result[line.split()[1 + shift]] = 'OK' | 
 | 187 |     assert 'NOT OK' not in result.values(), \ | 
 | 188 |         '''Some containers are in incorrect state: | 
 | 189 |               {}'''.format(json.dumps(result, indent=4)) | 
| Oleksii Zhurba | e592ed1 | 2018-06-21 18:01:09 -0500 | [diff] [blame] | 190 |  | 
 | 191 |  | 
| Oleksii Zhurba | 23c1833 | 2019-05-09 18:53:40 -0500 | [diff] [blame] | 192 | @pytest.mark.sl_dup | 
 | 193 | #PrometheusTargetDown | 
 | 194 | @pytest.mark.full | 
| Oleksii Zhurba | e592ed1 | 2018-06-21 18:01:09 -0500 | [diff] [blame] | 195 | def test_running_telegraf_services(local_salt_client): | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 196 |     salt_output = local_salt_client.cmd(tgt='telegraf:agent', | 
 | 197 |                                         fun='service.status', | 
 | 198 |                                         param='telegraf', | 
 | 199 |                                         expr_form='pillar',) | 
| Oleksii Zhurba | 9848e21 | 2018-09-05 10:53:51 -0500 | [diff] [blame] | 200 |  | 
 | 201 |     if not salt_output: | 
 | 202 |         pytest.skip("Telegraf or telegraf:agent \ | 
 | 203 |         pillar are not found on this environment.") | 
 | 204 |  | 
| Oleksii Zhurba | e592ed1 | 2018-06-21 18:01:09 -0500 | [diff] [blame] | 205 |     result = [{node: status} for node, status | 
 | 206 |               in salt_output.items() | 
 | 207 |               if status is False] | 
 | 208 |     assert result == [], 'Telegraf service is not running ' \ | 
| Oleksii Zhurba | 9848e21 | 2018-09-05 10:53:51 -0500 | [diff] [blame] | 209 |                          'on following nodes: {}'.format(result) | 
| Ievgeniia Zadorozhna | 6775eb7 | 2018-11-09 19:50:04 +0300 | [diff] [blame] | 210 |  | 
 | 211 |  | 
| Oleksii Zhurba | 23c1833 | 2019-05-09 18:53:40 -0500 | [diff] [blame] | 212 | @pytest.mark.sl_dup | 
 | 213 | #PrometheusTargetDown | 
 | 214 | @pytest.mark.full | 
| Ievgeniia Zadorozhna | 6775eb7 | 2018-11-09 19:50:04 +0300 | [diff] [blame] | 215 | def test_running_fluentd_services(local_salt_client): | 
| Hanna Arhipova | e6ed8e4 | 2019-05-15 16:27:08 +0300 | [diff] [blame] | 216 |     salt_output = local_salt_client.cmd(tgt='fluentd:agent', | 
 | 217 |                                         fun='service.status', | 
 | 218 |                                         param='td-agent', | 
| Ievgeniia Zadorozhna | 6775eb7 | 2018-11-09 19:50:04 +0300 | [diff] [blame] | 219 |                                         expr_form='pillar') | 
 | 220 |     result = [{node: status} for node, status | 
 | 221 |               in salt_output.items() | 
 | 222 |               if status is False] | 
 | 223 |     assert result == [], 'Fluentd check failed: td-agent service is not ' \ | 
| Oleksii Zhurba | 7d4f07e | 2019-06-10 17:30:53 -0500 | [diff] [blame] | 224 |                          'running on following nodes:'.format(result) |