elasticsearch_test.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. import pytest
  2. import json
  3. from openshift_checks.logging.elasticsearch import Elasticsearch, OpenShiftCheckExceptionList
  4. task_vars_config_base = dict(openshift=dict(common=dict(config_base='/etc/origin')))
  5. def canned_elasticsearch(task_vars=None, exec_oc=None):
  6. """Create an Elasticsearch check object with stubbed exec_oc method"""
  7. check = Elasticsearch(None, task_vars or {})
  8. if exec_oc:
  9. check.exec_oc = exec_oc
  10. return check
  11. def assert_error_in_list(expect_err, errorlist):
  12. assert any(err.name == expect_err for err in errorlist), "{} in {}".format(str(expect_err), str(errorlist))
  13. def pods_by_name(pods):
  14. return {pod['metadata']['name']: pod for pod in pods}
  15. plain_es_pod = {
  16. "metadata": {
  17. "labels": {"component": "es", "deploymentconfig": "logging-es"},
  18. "name": "logging-es",
  19. },
  20. "spec": {},
  21. "status": {
  22. "conditions": [{"status": "True", "type": "Ready"}],
  23. "containerStatuses": [{"ready": True}],
  24. "podIP": "10.10.10.10",
  25. },
  26. "_test_master_name_str": "name logging-es",
  27. }
  28. split_es_pod = {
  29. "metadata": {
  30. "labels": {"component": "es", "deploymentconfig": "logging-es-2"},
  31. "name": "logging-es-2",
  32. },
  33. "spec": {},
  34. "status": {
  35. "conditions": [{"status": "True", "type": "Ready"}],
  36. "containerStatuses": [{"ready": True}],
  37. "podIP": "10.10.10.10",
  38. },
  39. "_test_master_name_str": "name logging-es-2",
  40. }
  41. unready_es_pod = {
  42. "metadata": {
  43. "labels": {"component": "es", "deploymentconfig": "logging-es-3"},
  44. "name": "logging-es-3",
  45. },
  46. "spec": {},
  47. "status": {
  48. "conditions": [{"status": "False", "type": "Ready"}],
  49. "containerStatuses": [{"ready": False}],
  50. "podIP": "10.10.10.10",
  51. },
  52. "_test_master_name_str": "BAD_NAME_RESPONSE",
  53. }
  54. def test_check_elasticsearch():
  55. with pytest.raises(OpenShiftCheckExceptionList) as excinfo:
  56. canned_elasticsearch().check_elasticsearch([])
  57. assert_error_in_list('NoRunningPods', excinfo.value)
  58. # canned oc responses to match so all the checks pass
  59. def exec_oc(cmd, args, **_):
  60. if '_cat/master' in cmd:
  61. return 'name logging-es'
  62. elif '/_nodes' in cmd:
  63. return json.dumps(es_node_list)
  64. elif '_cluster/health' in cmd:
  65. return '{"status": "green"}'
  66. elif ' df ' in cmd:
  67. return 'IUse% Use%\n 3% 4%\n'
  68. else:
  69. raise Exception(cmd)
  70. check = canned_elasticsearch({}, exec_oc)
  71. check.get_pods_for_component = lambda *_: [plain_es_pod]
  72. assert {} == check.run()
  73. def test_check_running_es_pods():
  74. pods, errors = Elasticsearch().running_elasticsearch_pods([plain_es_pod, unready_es_pod])
  75. assert plain_es_pod in pods
  76. assert_error_in_list('PodNotRunning', errors)
  77. def test_check_elasticsearch_masters():
  78. pods = [plain_es_pod]
  79. check = canned_elasticsearch(task_vars_config_base, lambda *args, **_: plain_es_pod['_test_master_name_str'])
  80. assert not check.check_elasticsearch_masters(pods_by_name(pods))
  81. @pytest.mark.parametrize('pods, expect_error', [
  82. (
  83. [],
  84. 'NoMasterFound',
  85. ),
  86. (
  87. [unready_es_pod],
  88. 'NoMasterName',
  89. ),
  90. (
  91. [plain_es_pod, split_es_pod],
  92. 'SplitBrainMasters',
  93. ),
  94. ])
  95. def test_check_elasticsearch_masters_error(pods, expect_error):
  96. test_pods = list(pods)
  97. check = canned_elasticsearch(task_vars_config_base, lambda *args, **_: test_pods.pop(0)['_test_master_name_str'])
  98. assert_error_in_list(expect_error, check.check_elasticsearch_masters(pods_by_name(pods)))
  99. es_node_list = {
  100. 'nodes': {
  101. 'random-es-name': {
  102. 'host': 'logging-es',
  103. }}}
  104. def test_check_elasticsearch_node_list():
  105. check = canned_elasticsearch(task_vars_config_base, lambda *args, **_: json.dumps(es_node_list))
  106. assert not check.check_elasticsearch_node_list(pods_by_name([plain_es_pod]))
  107. @pytest.mark.parametrize('pods, node_list, expect_error', [
  108. (
  109. [],
  110. {},
  111. 'MissingComponentPods',
  112. ),
  113. (
  114. [plain_es_pod],
  115. {}, # empty list of nodes triggers KeyError
  116. 'MissingNodeList',
  117. ),
  118. (
  119. [split_es_pod],
  120. es_node_list,
  121. 'EsPodNodeMismatch',
  122. ),
  123. ])
  124. def test_check_elasticsearch_node_list_errors(pods, node_list, expect_error):
  125. check = canned_elasticsearch(task_vars_config_base, lambda cmd, args, **_: json.dumps(node_list))
  126. assert_error_in_list(expect_error, check.check_elasticsearch_node_list(pods_by_name(pods)))
  127. def test_check_elasticsearch_cluster_health():
  128. test_health_data = [{"status": "green"}]
  129. check = canned_elasticsearch(exec_oc=lambda *args, **_: json.dumps(test_health_data.pop(0)))
  130. assert not check.check_es_cluster_health(pods_by_name([plain_es_pod]))
  131. @pytest.mark.parametrize('pods, health_data, expect_error', [
  132. (
  133. [plain_es_pod],
  134. [{"no-status": "should bomb"}],
  135. 'BadEsResponse',
  136. ),
  137. (
  138. [plain_es_pod, split_es_pod],
  139. [{"status": "green"}, {"status": "red"}],
  140. 'EsClusterHealthRed',
  141. ),
  142. ])
  143. def test_check_elasticsearch_cluster_health_errors(pods, health_data, expect_error):
  144. test_health_data = list(health_data)
  145. check = canned_elasticsearch(exec_oc=lambda *args, **_: json.dumps(test_health_data.pop(0)))
  146. assert_error_in_list(expect_error, check.check_es_cluster_health(pods_by_name(pods)))
  147. def test_check_elasticsearch_diskspace():
  148. check = canned_elasticsearch(exec_oc=lambda *args, **_: 'IUse% Use%\n 3% 4%\n')
  149. assert not check.check_elasticsearch_diskspace(pods_by_name([plain_es_pod]))
  150. @pytest.mark.parametrize('disk_data, expect_error', [
  151. (
  152. 'df: /elasticsearch/persistent: No such file or directory\n',
  153. 'BadDfResponse',
  154. ),
  155. (
  156. 'IUse% Use%\n 95% 40%\n',
  157. 'InodeUsageTooHigh',
  158. ),
  159. (
  160. 'IUse% Use%\n 3% 94%\n',
  161. 'DiskUsageTooHigh',
  162. ),
  163. ])
  164. def test_check_elasticsearch_diskspace_errors(disk_data, expect_error):
  165. check = canned_elasticsearch(exec_oc=lambda *args, **_: disk_data)
  166. assert_error_in_list(expect_error, check.check_elasticsearch_diskspace(pods_by_name([plain_es_pod])))