From 8b0420583e15ce85d6781052903045248811ea0d Mon Sep 17 00:00:00 2001 From: Ben Keith Date: Fri, 6 Dec 2019 09:36:50 -0500 Subject: [PATCH] Add Python 3 compatibility (#64) --- elasticsearch_collectd.py | 1286 ++++++++---------- elasticsearch_collectd_test.py | 2 +- tests/integration/20-elasticsearch-test.conf | 7 - tests/integration/Dockerfile.sink | 2 +- tests/integration/Dockerfile.test | 2 +- tests/integration/docker-compose.yml | 6 - tests/integration/make-es-dockerfiles | 4 +- tests/integration/sink.py | 23 +- tests/integration/test.py | 36 +- tests/integration/wait_for_es | 2 +- tests/simulate.py | 14 +- 11 files changed, 582 insertions(+), 802 deletions(-) diff --git a/elasticsearch_collectd.py b/elasticsearch_collectd.py index 60ea0cd..f173259 100755 --- a/elasticsearch_collectd.py +++ b/elasticsearch_collectd.py @@ -14,619 +14,404 @@ # limitations under the License. +import base64 import collections import json -import urllib2 -import base64 import logging import ssl +from functools import reduce + +from six.moves import urllib PREFIX = "elasticsearch" -DEFAULTS = set([ - # AUTOMATICALLY GENERATED METRIC NAMES - # TO INCLUDE BY DEFAULT - "indices.total.docs.deleted", - "indices.total.fielddata.memory-size", - "indices.merges.total", - "cluster.number-of-nodes", - "process.open_file_descriptors", - "indices.total.merges.total", - "indices.total.store.size", - "indices.segments.count", - "indices.merges.current", - "jvm.mem.heap-used", - "indices.search.query-time", - "cluster.number-of-data_nodes", - "cluster.active-shards", - "indices.indexing.index-total", - "indices.get.total", - "cluster.unassigned-shards", - "indices.cache.field.size", - "jvm.gc.time", - "indices.store.size", - "thread_pool.get.rejected", - "indices.total.search.query-time", - "indices.search.query-total", - "indices.total.merges.total-time", - "cluster.active-primary-shards", - "indices.docs.deleted", - "indices.cache.filter.size", - "indices.total.search.query-total", - "indices.docs.count", - "indices.total.indexing.index-time", - "indices.total.indexing.index-total", - "jvm.mem.heap-committed", - "indices.total.docs.count", - "cluster.relocating-shards", - "thread_pool.rejected", - "jvm.uptime", - "indices.total.filter-cache.memory-size", -]) - -DEFAULTS.update([ - # ADD ADDITIONAL METRIC NAMES - # TO INCLUDE BY DEFAULT - "cluster.status", - "indices.indexing.index-time", - "indices.merges.time", - "indices.store.throttle-time", -]) - -CLUSTER_STATUS = {'green': 0, 'yellow': 1, 'red': 2} - -Stat = collections.namedtuple('Stat', ('type', 'path')) +DEFAULTS = set( + [ + # AUTOMATICALLY GENERATED METRIC NAMES + # TO INCLUDE BY DEFAULT + "indices.total.docs.deleted", + "indices.total.fielddata.memory-size", + "indices.merges.total", + "cluster.number-of-nodes", + "process.open_file_descriptors", + "indices.total.merges.total", + "indices.total.store.size", + "indices.segments.count", + "indices.merges.current", + "jvm.mem.heap-used", + "indices.search.query-time", + "cluster.number-of-data_nodes", + "cluster.active-shards", + "indices.indexing.index-total", + "indices.get.total", + "cluster.unassigned-shards", + "indices.cache.field.size", + "jvm.gc.time", + "indices.store.size", + "thread_pool.get.rejected", + "indices.total.search.query-time", + "indices.search.query-total", + "indices.total.merges.total-time", + "cluster.active-primary-shards", + "indices.docs.deleted", + "indices.cache.filter.size", + "indices.total.search.query-total", + "indices.docs.count", + "indices.total.indexing.index-time", + "indices.total.indexing.index-total", + "jvm.mem.heap-committed", + "indices.total.docs.count", + "cluster.relocating-shards", + "thread_pool.rejected", + "jvm.uptime", + "indices.total.filter-cache.memory-size", + ] +) + +DEFAULTS.update( + [ + # ADD ADDITIONAL METRIC NAMES + # TO INCLUDE BY DEFAULT + "cluster.status", + "indices.indexing.index-time", + "indices.merges.time", + "indices.store.throttle-time", + ] +) + +CLUSTER_STATUS = {"green": 0, "yellow": 1, "red": 2} + +Stat = collections.namedtuple("Stat", ("type", "path")) # DICT: ElasticSearch 1.0.0 NODE_STATS = { # STORE - 'indices.store.throttle-time': - Stat("counter", "nodes.%s.indices.store.throttle_time_in_millis"), - + "indices.store.throttle-time": Stat("counter", "nodes.%s.indices.store.throttle_time_in_millis"), # SEARCH - 'indices.search.open-contexts': - Stat("gauge", "nodes.%s.indices.search.open_contexts"), - + "indices.search.open-contexts": Stat("gauge", "nodes.%s.indices.search.open_contexts"), # CACHE - 'indices.cache.field.eviction': - Stat("counter", "nodes.%s.indices.fielddata.evictions"), - 'indices.cache.field.size': - Stat("gauge", "nodes.%s.indices.fielddata.memory_size_in_bytes"), - 'indices.cache.filter.evictions': - Stat("counter", "nodes.%s.indices.filter_cache.evictions"), - 'indices.cache.filter.size': - Stat("gauge", "nodes.%s.indices.filter_cache.memory_size_in_bytes"), - + "indices.cache.field.eviction": Stat("counter", "nodes.%s.indices.fielddata.evictions"), + "indices.cache.field.size": Stat("gauge", "nodes.%s.indices.fielddata.memory_size_in_bytes"), + "indices.cache.filter.evictions": Stat("counter", "nodes.%s.indices.filter_cache.evictions"), + "indices.cache.filter.size": Stat("gauge", "nodes.%s.indices.filter_cache.memory_size_in_bytes"), # GC - 'jvm.gc.time': - Stat("counter", - "nodes.%s.jvm.gc.collectors.young.collection_time_in_millis"), - 'jvm.gc.count': - Stat("counter", "nodes.%s.jvm.gc.collectors.young.collection_count"), - 'jvm.gc.old-time': - Stat("counter", - "nodes.%s.jvm.gc.collectors.old.collection_time_in_millis"), - 'jvm.gc.old-count': - Stat("counter", "nodes.%s.jvm.gc.collectors.old.collection_count"), - + "jvm.gc.time": Stat("counter", "nodes.%s.jvm.gc.collectors.young.collection_time_in_millis"), + "jvm.gc.count": Stat("counter", "nodes.%s.jvm.gc.collectors.young.collection_count"), + "jvm.gc.old-time": Stat("counter", "nodes.%s.jvm.gc.collectors.old.collection_time_in_millis"), + "jvm.gc.old-count": Stat("counter", "nodes.%s.jvm.gc.collectors.old.collection_count"), # FLUSH - 'indices.flush.total': - Stat("counter", "nodes.%s.indices.flush.total"), - 'indices.flush.time': - Stat("counter", "nodes.%s.indices.flush.total_time_in_millis"), - + "indices.flush.total": Stat("counter", "nodes.%s.indices.flush.total"), + "indices.flush.time": Stat("counter", "nodes.%s.indices.flush.total_time_in_millis"), # MERGES - 'indices.merges.current': - Stat("gauge", "nodes.%s.indices.merges.current"), - 'indices.merges.current-docs': - Stat("gauge", "nodes.%s.indices.merges.current_docs"), - 'indices.merges.current-size': - Stat("gauge", "nodes.%s.indices.merges.current_size_in_bytes"), - 'indices.merges.total': - Stat("counter", "nodes.%s.indices.merges.total"), - 'indices.merges.total-docs': - Stat("gauge", "nodes.%s.indices.merges.total_docs"), - 'indices.merges.total-size': - Stat("counter", "nodes.%s.indices.merges.total_size_in_bytes"), - 'indices.merges.time': - Stat("counter", "nodes.%s.indices.merges.total_time_in_millis"), - + "indices.merges.current": Stat("gauge", "nodes.%s.indices.merges.current"), + "indices.merges.current-docs": Stat("gauge", "nodes.%s.indices.merges.current_docs"), + "indices.merges.current-size": Stat("gauge", "nodes.%s.indices.merges.current_size_in_bytes"), + "indices.merges.total": Stat("counter", "nodes.%s.indices.merges.total"), + "indices.merges.total-docs": Stat("gauge", "nodes.%s.indices.merges.total_docs"), + "indices.merges.total-size": Stat("counter", "nodes.%s.indices.merges.total_size_in_bytes"), + "indices.merges.time": Stat("counter", "nodes.%s.indices.merges.total_time_in_millis"), # REFRESH - 'indices.refresh.total': - Stat("counter", "nodes.%s.indices.refresh.total"), - 'indices.refresh.time': - Stat("counter", "nodes.%s.indices.refresh.total_time_in_millis"), - + "indices.refresh.total": Stat("counter", "nodes.%s.indices.refresh.total"), + "indices.refresh.time": Stat("counter", "nodes.%s.indices.refresh.total_time_in_millis"), # SEGMENTS - 'indices.segments.count': - Stat("gauge", "nodes.%s.indices.segments.count"), - 'indices.segments.size': - Stat("gauge", "nodes.%s.indices.segments.memory_in_bytes"), - 'indices.segments.index-writer-max-size': - Stat("gauge", - "nodes.%s.indices.segments.index_writer_max_memory_in_bytes"), - 'indices.segments.index-writer-size': - Stat("gauge", - "nodes.%s.indices.segments.index_writer_memory_in_bytes"), - + "indices.segments.count": Stat("gauge", "nodes.%s.indices.segments.count"), + "indices.segments.size": Stat("gauge", "nodes.%s.indices.segments.memory_in_bytes"), + "indices.segments.index-writer-max-size": Stat( + "gauge", "nodes.%s.indices.segments.index_writer_max_memory_in_bytes" + ), + "indices.segments.index-writer-size": Stat("gauge", "nodes.%s.indices.segments.index_writer_memory_in_bytes"), # DOCS - 'indices.docs.count': - Stat("gauge", "nodes.%s.indices.docs.count"), - 'indices.docs.deleted': - Stat("gauge", "nodes.%s.indices.docs.deleted"), - + "indices.docs.count": Stat("gauge", "nodes.%s.indices.docs.count"), + "indices.docs.deleted": Stat("gauge", "nodes.%s.indices.docs.deleted"), # STORE - 'indices.store.size': - Stat("gauge", "nodes.%s.indices.store.size_in_bytes"), - + "indices.store.size": Stat("gauge", "nodes.%s.indices.store.size_in_bytes"), # INDEXING - 'indices.indexing.index-total': - Stat("counter", "nodes.%s.indices.indexing.index_total"), - 'indices.indexing.index-time': - Stat("counter", "nodes.%s.indices.indexing.index_time_in_millis"), - 'indices.indexing.delete-total': - Stat("counter", "nodes.%s.indices.indexing.delete_total"), - 'indices.indexing.delete-time': - Stat("counter", "nodes.%s.indices.indexing.delete_time_in_millis"), - 'indices.indexing.index-current': - Stat("gauge", "nodes.%s.indices.indexing.index_current"), - 'indices.indexing.delete-current': - Stat("gauge", "nodes.%s.indices.indexing.delete_current"), - + "indices.indexing.index-total": Stat("counter", "nodes.%s.indices.indexing.index_total"), + "indices.indexing.index-time": Stat("counter", "nodes.%s.indices.indexing.index_time_in_millis"), + "indices.indexing.delete-total": Stat("counter", "nodes.%s.indices.indexing.delete_total"), + "indices.indexing.delete-time": Stat("counter", "nodes.%s.indices.indexing.delete_time_in_millis"), + "indices.indexing.index-current": Stat("gauge", "nodes.%s.indices.indexing.index_current"), + "indices.indexing.delete-current": Stat("gauge", "nodes.%s.indices.indexing.delete_current"), # GET - 'indices.get.total': - Stat("counter", "nodes.%s.indices.get.total"), - 'indices.get.time': - Stat("counter", "nodes.%s.indices.get.time_in_millis"), - 'indices.get.exists-total': - Stat("counter", "nodes.%s.indices.get.exists_total"), - 'indices.get.exists-time': - Stat("counter", "nodes.%s.indices.get.exists_time_in_millis"), - 'indices.get.missing-total': - Stat("counter", "nodes.%s.indices.get.missing_total"), - 'indices.get.missing-time': - Stat("counter", "nodes.%s.indices.get.missing_time_in_millis"), - 'indices.get.current': - Stat("gauge", "nodes.%s.indices.get.current"), - + "indices.get.total": Stat("counter", "nodes.%s.indices.get.total"), + "indices.get.time": Stat("counter", "nodes.%s.indices.get.time_in_millis"), + "indices.get.exists-total": Stat("counter", "nodes.%s.indices.get.exists_total"), + "indices.get.exists-time": Stat("counter", "nodes.%s.indices.get.exists_time_in_millis"), + "indices.get.missing-total": Stat("counter", "nodes.%s.indices.get.missing_total"), + "indices.get.missing-time": Stat("counter", "nodes.%s.indices.get.missing_time_in_millis"), + "indices.get.current": Stat("gauge", "nodes.%s.indices.get.current"), # SEARCH - 'indices.search.query-current': - Stat("gauge", "nodes.%s.indices.search.query_current"), - 'indices.search.query-total': - Stat("counter", "nodes.%s.indices.search.query_total"), - 'indices.search.query-time': - Stat("counter", "nodes.%s.indices.search.query_time_in_millis"), - 'indices.search.fetch-current': - Stat("gauge", "nodes.%s.indices.search.fetch_current"), - 'indices.search.fetch-total': - Stat("counter", "nodes.%s.indices.search.fetch_total"), - 'indices.search.fetch-time': - Stat("counter", "nodes.%s.indices.search.fetch_time_in_millis"), - + "indices.search.query-current": Stat("gauge", "nodes.%s.indices.search.query_current"), + "indices.search.query-total": Stat("counter", "nodes.%s.indices.search.query_total"), + "indices.search.query-time": Stat("counter", "nodes.%s.indices.search.query_time_in_millis"), + "indices.search.fetch-current": Stat("gauge", "nodes.%s.indices.search.fetch_current"), + "indices.search.fetch-total": Stat("counter", "nodes.%s.indices.search.fetch_total"), + "indices.search.fetch-time": Stat("counter", "nodes.%s.indices.search.fetch_time_in_millis"), # JVM METRICS # # MEM - 'jvm.mem.heap-committed': - Stat("gauge", "nodes.%s.jvm.mem.heap_committed_in_bytes"), - 'jvm.mem.heap-used': - Stat("gauge", "nodes.%s.jvm.mem.heap_used_in_bytes"), - 'jvm.mem.heap-used-percent': - Stat("percent", "nodes.%s.jvm.mem.heap_used_percent"), - 'jvm.mem.non-heap-committed': - Stat("gauge", "nodes.%s.jvm.mem.non_heap_committed_in_bytes"), - 'jvm.mem.non-heap-used': - Stat("gauge", "nodes.%s.jvm.mem.non_heap_used_in_bytes"), - 'jvm.mem.pools.young.max_in_bytes': - Stat("gauge", "nodes.%s.jvm.mem.pools.young.max_in_bytes"), - 'jvm.mem.pools.young.used_in_bytes': - Stat("gauge", "nodes.%s.jvm.mem.pools.young.used_in_bytes"), - 'jvm.mem.pools.old.max_in_bytes': - Stat("gauge", "nodes.%s.jvm.mem.pools.old.max_in_bytes"), - 'jvm.mem.pools.old.used_in_bytes': - Stat("gauge", "nodes.%s.jvm.mem.pools.old.used_in_bytes"), - + "jvm.mem.heap-committed": Stat("gauge", "nodes.%s.jvm.mem.heap_committed_in_bytes"), + "jvm.mem.heap-used": Stat("gauge", "nodes.%s.jvm.mem.heap_used_in_bytes"), + "jvm.mem.heap-used-percent": Stat("percent", "nodes.%s.jvm.mem.heap_used_percent"), + "jvm.mem.non-heap-committed": Stat("gauge", "nodes.%s.jvm.mem.non_heap_committed_in_bytes"), + "jvm.mem.non-heap-used": Stat("gauge", "nodes.%s.jvm.mem.non_heap_used_in_bytes"), + "jvm.mem.pools.young.max_in_bytes": Stat("gauge", "nodes.%s.jvm.mem.pools.young.max_in_bytes"), + "jvm.mem.pools.young.used_in_bytes": Stat("gauge", "nodes.%s.jvm.mem.pools.young.used_in_bytes"), + "jvm.mem.pools.old.max_in_bytes": Stat("gauge", "nodes.%s.jvm.mem.pools.old.max_in_bytes"), + "jvm.mem.pools.old.used_in_bytes": Stat("gauge", "nodes.%s.jvm.mem.pools.old.used_in_bytes"), # UPTIME - 'jvm.uptime': - Stat("counter", "nodes.%s.jvm.uptime_in_millis"), - + "jvm.uptime": Stat("counter", "nodes.%s.jvm.uptime_in_millis"), # THREADS - 'jvm.threads.count': - Stat("gauge", "nodes.%s.jvm.threads.count"), - 'jvm.threads.peak': - Stat("gauge", "nodes.%s.jvm.threads.peak_count"), - + "jvm.threads.count": Stat("gauge", "nodes.%s.jvm.threads.count"), + "jvm.threads.peak": Stat("gauge", "nodes.%s.jvm.threads.peak_count"), # TRANSPORT METRICS # - 'transport.server_open': - Stat("gauge", "nodes.%s.transport.server_open"), - 'transport.rx.count': - Stat("counter", "nodes.%s.transport.rx_count"), - 'transport.rx.size': - Stat("counter", "nodes.%s.transport.rx_size_in_bytes"), - 'transport.tx.count': - Stat("counter", "nodes.%s.transport.tx_count"), - 'transport.tx.size': - Stat("counter", "nodes.%s.transport.tx_size_in_bytes"), - + "transport.server_open": Stat("gauge", "nodes.%s.transport.server_open"), + "transport.rx.count": Stat("counter", "nodes.%s.transport.rx_count"), + "transport.rx.size": Stat("counter", "nodes.%s.transport.rx_size_in_bytes"), + "transport.tx.count": Stat("counter", "nodes.%s.transport.tx_count"), + "transport.tx.size": Stat("counter", "nodes.%s.transport.tx_size_in_bytes"), # HTTP METRICS # - 'http.current_open': - Stat("gauge", "nodes.%s.http.current_open"), - 'http.total_open': - Stat("counter", "nodes.%s.http.total_opened"), - + "http.current_open": Stat("gauge", "nodes.%s.http.current_open"), + "http.total_open": Stat("counter", "nodes.%s.http.total_opened"), # PROCESS METRICS # - 'process.open_file_descriptors': - Stat("gauge", "nodes.%s.process.open_file_descriptors"), - 'process.cpu.percent': - Stat("gauge", "nodes.%s.process.cpu.percent"), - 'process.mem.share_in_bytes': - Stat("gauge", "nodes.%s.process.mem.share_in_bytes"), + "process.open_file_descriptors": Stat("gauge", "nodes.%s.process.open_file_descriptors"), + "process.cpu.percent": Stat("gauge", "nodes.%s.process.cpu.percent"), + "process.mem.share_in_bytes": Stat("gauge", "nodes.%s.process.mem.share_in_bytes"), } # Deprecated node stats by first version in which they were removed DEPRECATED_NODE_STATS = [ - { - 'major': 2, - 'minor': 0, - 'revision': 0, - 'keys': ['process.mem.share_in_bytes'] - }, - { - 'major': 5, - 'minor': 0, - 'revision': 0, - 'keys': ['indices.segments.index-writer-max-size'] - }, - { - 'major': 6, - 'minor': 0, - 'revision': 0, - 'keys': ['indices.store.throttle-time'] - } + {"major": 2, "minor": 0, "revision": 0, "keys": ["process.mem.share_in_bytes"]}, + {"major": 5, "minor": 0, "revision": 0, "keys": ["indices.segments.index-writer-max-size"]}, + {"major": 6, "minor": 0, "revision": 0, "keys": ["indices.store.throttle-time"]}, ] # Deprecated thread pools by first version in which they were removed DEPRECATED_THREAD_POOLS = [ - { - 'major': 2, - 'minor': 0, - 'revision': 0, - 'keys': ['merge', 'optimize'] - }, - { - 'major': 5, - 'minor': 0, - 'revision': 0, - 'keys': ['suggest', 'percolate'] - }, - { - 'major': 6, - 'minor': 3, - 'revision': 2, - 'keys': ['bulk'] - } + {"major": 2, "minor": 0, "revision": 0, "keys": ["merge", "optimize"]}, + {"major": 5, "minor": 0, "revision": 0, "keys": ["suggest", "percolate"]}, + {"major": 6, "minor": 3, "revision": 2, "keys": ["bulk"]}, ] NODE_STATS_ES_2 = { - 'indices.cache.filter.evictions': - Stat("counter", "nodes.%s.indices.query_cache.evictions"), - 'indices.cache.filter.size': - Stat("gauge", "nodes.%s.indices.query_cache.cache_size"), - 'indices.cache.filter.hit-count': - Stat("counter", "nodes.%s.indices.query_cache.hit_count"), - 'indices.cache.filter.miss-count': - Stat("counter", "nodes.%s.indices.query_cache.miss_count"), - 'indices.cache.filter.cache-count': - Stat("counter", "nodes.%s.indices.query_cache.cache_count"), - 'indices.cache.filter.total-count': - Stat("counter", "nodes.%s.indices.query_cache.total_count"), - 'indices.search.scroll-time': - Stat("counter", "nodes.%s.indices.search.scroll_time_in_millis"), - 'indices.search.scroll.total': - Stat("counter", "nodes.%s.indices.search.scroll_total"), - 'indices.search.scroll.current': - Stat("gauge", "nodes.%s.indices.search.scroll_current"), + "indices.cache.filter.evictions": Stat("counter", "nodes.%s.indices.query_cache.evictions"), + "indices.cache.filter.size": Stat("gauge", "nodes.%s.indices.query_cache.cache_size"), + "indices.cache.filter.hit-count": Stat("counter", "nodes.%s.indices.query_cache.hit_count"), + "indices.cache.filter.miss-count": Stat("counter", "nodes.%s.indices.query_cache.miss_count"), + "indices.cache.filter.cache-count": Stat("counter", "nodes.%s.indices.query_cache.cache_count"), + "indices.cache.filter.total-count": Stat("counter", "nodes.%s.indices.query_cache.total_count"), + "indices.search.scroll-time": Stat("counter", "nodes.%s.indices.search.scroll_time_in_millis"), + "indices.search.scroll.total": Stat("counter", "nodes.%s.indices.search.scroll_total"), + "indices.search.scroll.current": Stat("gauge", "nodes.%s.indices.search.scroll_current"), } # ElasticSearch 6.0.0 NODE_STATS_ES_6 = { - "indices.translog.uncommitted_operations": - Stat("gauge", "nodes.%s.indices.translog.uncommitted_operations"), - "indices.translog.uncommitted_size_in_bytes": - Stat("gauge", "nodes.%s.indices.translog.uncommitted_size_in_bytes") + "indices.translog.uncommitted_operations": Stat("gauge", "nodes.%s.indices.translog.uncommitted_operations"), + "indices.translog.uncommitted_size_in_bytes": Stat("gauge", "nodes.%s.indices.translog.uncommitted_size_in_bytes"), } # ElasticSearch 6.3.0 NODE_STATS_ES_6_3 = { - "indices.flush.periodic": - Stat("gauge", "nodes.%s.indices.flush.periodic"), - "indices.translog.earliest_last_modified_age": - Stat("gauge", "nodes.%s.indices.translog.earliest_last_modified_age"), + "indices.flush.periodic": Stat("gauge", "nodes.%s.indices.flush.periodic"), + "indices.translog.earliest_last_modified_age": Stat( + "gauge", "nodes.%s.indices.translog.earliest_last_modified_age" + ), } # ElasticSearch 1.3.0 INDEX_STATS_ES_1_3 = { # SEGMENTS - "indices[index={index_name}].primaries.segments.index-writer-memory": - Stat("gauge", "primaries.segments.index_writer_memory_in_bytes"), - "indices[index={index_name}].primaries.segments.version-map-memory": - Stat("gauge", "primaries.segments.version_map_memory_in_bytes"), + "indices[index={index_name}].primaries.segments.index-writer-memory": Stat( + "gauge", "primaries.segments.index_writer_memory_in_bytes" + ), + "indices[index={index_name}].primaries.segments.version-map-memory": Stat( + "gauge", "primaries.segments.version_map_memory_in_bytes" + ), } # ElasticSearch 1.1.0 INDEX_STATS_ES_1_1 = { # SUGGEST - "indices[index={index_name}].primaries.suggest.total": - Stat("counter", "primaries.suggest.total"), - "indices[index={index_name}].primaries.suggest.time": - Stat("counter", "primaries.suggest.time_in_millis"), - "indices[index={index_name}].primaries.suggest.current": - Stat("gauge", "primaries.suggest.current"), + "indices[index={index_name}].primaries.suggest.total": Stat("counter", "primaries.suggest.total"), + "indices[index={index_name}].primaries.suggest.time": Stat("counter", "primaries.suggest.time_in_millis"), + "indices[index={index_name}].primaries.suggest.current": Stat("gauge", "primaries.suggest.current"), } # ElasticSearch 6.0.0 INDEX_STATS_ES_6 = { # SUGGEST - "indices[index={index_name}].primaries.flush.periodic": - Stat("gauge", "primaries.flush.periodic"), - "indices[index={index_name}].primaries.translog.earliest_last_modified_age": - Stat("gauge", "primaries.translog.earliest_last_modified_age"), - "indices[index={index_name}].primaries.translog.uncommitted_operations": - Stat("gauge", "primaries.translog.uncommitted_operations"), - "indices[index={index_name}].primaries.translog.uncommitted_size_in_bytes": - Stat("gauge", "primaries.translog.uncommitted_size_in_bytes"), - "indices[index={index_name}].total.flush.periodic": - Stat("counter", "total.flush.periodic"), - "indices[index={index_name}].total.translog.earliest_last_modified_age": - Stat("counter", "total.translog.earliest_last_modified_age"), - "indices[index={index_name}].total.translog.uncommitted_operations": - Stat("counter", "total.translog.uncommitted_operations"), - "indices[index={index_name}].total.translog.uncommitted_size_in_bytes": - Stat("counter", "total.translog.uncommitted_size_in_bytes"), + "indices[index={index_name}].primaries.flush.periodic": Stat("gauge", "primaries.flush.periodic"), + "indices[index={index_name}].primaries.translog.earliest_last_modified_age": Stat( + "gauge", "primaries.translog.earliest_last_modified_age" + ), + "indices[index={index_name}].primaries.translog.uncommitted_operations": Stat( + "gauge", "primaries.translog.uncommitted_operations" + ), + "indices[index={index_name}].primaries.translog.uncommitted_size_in_bytes": Stat( + "gauge", "primaries.translog.uncommitted_size_in_bytes" + ), + "indices[index={index_name}].total.flush.periodic": Stat("counter", "total.flush.periodic"), + "indices[index={index_name}].total.translog.earliest_last_modified_age": Stat( + "counter", "total.translog.earliest_last_modified_age" + ), + "indices[index={index_name}].total.translog.uncommitted_operations": Stat( + "counter", "total.translog.uncommitted_operations" + ), + "indices[index={index_name}].total.translog.uncommitted_size_in_bytes": Stat( + "counter", "total.translog.uncommitted_size_in_bytes" + ), } # ElasticSearch 1.0.0 INDEX_STATS = { # PRIMARIES # TRANSLOG - "indices[index={index_name}].primaries.translog.size": - Stat("gauge", "primaries.translog.size_in_bytes"), - "indices[index={index_name}].primaries.translog.operations": - Stat("counter", "primaries.translog.operations"), - + "indices[index={index_name}].primaries.translog.size": Stat("gauge", "primaries.translog.size_in_bytes"), + "indices[index={index_name}].primaries.translog.operations": Stat("counter", "primaries.translog.operations"), # SEGMENTS - "indices[index={index_name}].primaries.segments.memory": - Stat("gauge", "primaries.segments.memory_in_bytes"), - "indices[index={index_name}].primaries.segments.count": - Stat("counter", "primaries.segments.count"), - + "indices[index={index_name}].primaries.segments.memory": Stat("gauge", "primaries.segments.memory_in_bytes"), + "indices[index={index_name}].primaries.segments.count": Stat("counter", "primaries.segments.count"), # ID_CACHE - "indices[index={index_name}].primaries.id-cache.memory-size": - Stat("gauge", "primaries.id_cache.memory_size_in_bytes"), - + "indices[index={index_name}].primaries.id-cache.memory-size": Stat( + "gauge", "primaries.id_cache.memory_size_in_bytes" + ), # FLUSH - "indices[index={index_name}].primaries.flush.total": - Stat("counter", "primaries.flush.total"), - "indices[index={index_name}].primaries.flush.total-time": - Stat("counter", "primaries.flush.total_time_in_millis"), - + "indices[index={index_name}].primaries.flush.total": Stat("counter", "primaries.flush.total"), + "indices[index={index_name}].primaries.flush.total-time": Stat("counter", "primaries.flush.total_time_in_millis"), # WARMER "indices[index={index_name}].primaries.warmer.total.primaries.warmer" - ".total-time": Stat( - "counter", "primaries.warmer.total_time_in_millis"), - "indices[index={index_name}].primaries.warmer.total": - Stat("counter", "primaries.warmer.total"), - "indices[index={index_name}].primaries.warmer.current": - Stat("gauge", "primaries.warmer.current"), - + ".total-time": Stat("counter", "primaries.warmer.total_time_in_millis"), + "indices[index={index_name}].primaries.warmer.total": Stat("counter", "primaries.warmer.total"), + "indices[index={index_name}].primaries.warmer.current": Stat("gauge", "primaries.warmer.current"), # FIELDDATA "indices[index={index_name}].primaries.fielddata.memory-size": Stat( - "gauge", - "primaries.fielddata.memory_size_in_bytes"), - "indices[index={index_name}].primaries.fielddata.evictions": Stat( - "counter", - "primaries.fielddata.evictions"), - + "gauge", "primaries.fielddata.memory_size_in_bytes" + ), + "indices[index={index_name}].primaries.fielddata.evictions": Stat("counter", "primaries.fielddata.evictions"), # REFRESH - "indices[index={index_name}].primaries.refresh.total-time": - Stat("counter", "primaries.refresh.total_time_in_millis"), - "indices[index={index_name}].primaries.refresh.total": - Stat("counter", "primaries.refresh.total"), - + "indices[index={index_name}].primaries.refresh.total-time": Stat( + "counter", "primaries.refresh.total_time_in_millis" + ), + "indices[index={index_name}].primaries.refresh.total": Stat("counter", "primaries.refresh.total"), # MERGES - "indices[index={index_name}].primaries.merges.total-docs": - Stat("counter", "primaries.merges.total_docs"), - "indices[index={index_name}].primaries.merges.total-size": - Stat("bytes", "primaries.merges.total_size_in_bytes"), - "indices[index={index_name}].primaries.merges.current": - Stat("gauge", "primaries.merges.current"), - "indices[index={index_name}].primaries.merges.total": - Stat("counter", "primaries.merges.total"), - "indices[index={index_name}].primaries.merges.current-docs": - Stat("gauge", "primaries.merges.current_docs"), - "indices[index={index_name}].primaries.merges.total-time": - Stat("counter", "primaries.merges.total_time_in_millis"), - "indices[index={index_name}].primaries.merges.current-size": - Stat("gauge", "primaries.merges.current_size_in_bytes"), - + "indices[index={index_name}].primaries.merges.total-docs": Stat("counter", "primaries.merges.total_docs"), + "indices[index={index_name}].primaries.merges.total-size": Stat("bytes", "primaries.merges.total_size_in_bytes"), + "indices[index={index_name}].primaries.merges.current": Stat("gauge", "primaries.merges.current"), + "indices[index={index_name}].primaries.merges.total": Stat("counter", "primaries.merges.total"), + "indices[index={index_name}].primaries.merges.current-docs": Stat("gauge", "primaries.merges.current_docs"), + "indices[index={index_name}].primaries.merges.total-time": Stat("counter", "primaries.merges.total_time_in_millis"), + "indices[index={index_name}].primaries.merges.current-size": Stat( + "gauge", "primaries.merges.current_size_in_bytes" + ), # COMPLETION - "indices[index={index_name}].primaries.completion.size": - Stat("gauge", "primaries.completion.size_in_bytes"), - + "indices[index={index_name}].primaries.completion.size": Stat("gauge", "primaries.completion.size_in_bytes"), # PERCOLATE - "indices[index={index_name}].primaries.percolate.total": - Stat("counter", "primaries.percolate.total"), - "indices[index={index_name}].primaries.percolate.memory-size": - Stat("gauge", "primaries.percolate.memory_size_in_bytes"), - "indices[index={index_name}].primaries.percolate.queries": - Stat("counter", "primaries.percolate.queries"), - "indices[index={index_name}].primaries.percolate.time": - Stat("counter", "primaries.percolate.time_in_millis"), - "indices[index={index_name}].primaries.percolate.current": - Stat("gauge", "primaries.percolate.current"), - + "indices[index={index_name}].primaries.percolate.total": Stat("counter", "primaries.percolate.total"), + "indices[index={index_name}].primaries.percolate.memory-size": Stat( + "gauge", "primaries.percolate.memory_size_in_bytes" + ), + "indices[index={index_name}].primaries.percolate.queries": Stat("counter", "primaries.percolate.queries"), + "indices[index={index_name}].primaries.percolate.time": Stat("counter", "primaries.percolate.time_in_millis"), + "indices[index={index_name}].primaries.percolate.current": Stat("gauge", "primaries.percolate.current"), # FILTER_CACHE - "indices[index={index_name}].primaries.filter-cache.evictions": - Stat("counter", "primaries.filter_cache.evictions"), - "indices[index={index_name}].primaries.filter-cache.memory-size": - Stat("gauge", "primaries.filter_cache.memory_size_in_bytes"), - + "indices[index={index_name}].primaries.filter-cache.evictions": Stat("counter", "primaries.filter_cache.evictions"), + "indices[index={index_name}].primaries.filter-cache.memory-size": Stat( + "gauge", "primaries.filter_cache.memory_size_in_bytes" + ), # DOCS - "indices[index={index_name}].primaries.docs.count": - Stat("gauge", "primaries.docs.count"), - "indices[index={index_name}].primaries.docs.deleted": - Stat("gauge", "primaries.docs.deleted"), - + "indices[index={index_name}].primaries.docs.count": Stat("gauge", "primaries.docs.count"), + "indices[index={index_name}].primaries.docs.deleted": Stat("gauge", "primaries.docs.deleted"), # STORE - "indices[index={index_name}].primaries.store.size": - Stat("gauge", "primaries.store.size_in_bytes"), - "indices[index={index_name}].primaries.store.throttle-time": - Stat("counter", "primaries.store.throttle_time_in_millis"), - + "indices[index={index_name}].primaries.store.size": Stat("gauge", "primaries.store.size_in_bytes"), + "indices[index={index_name}].primaries.store.throttle-time": Stat( + "counter", "primaries.store.throttle_time_in_millis" + ), # INDEXING - "indices[index={index_name}].primaries.indexing.index-total": - Stat("counter", "primaries.indexing.index_total"), - "indices[index={index_name}].primaries.indexing.index-time": - Stat("counter", "primaries.indexing.index_time_in_millis"), - "indices[index={index_name}].primaries.indexing.index-current": - Stat("gauge", "primaries.indexing.index_current"), - "indices[index={index_name}].primaries.indexing.delete-total": - Stat("counter", "primaries.indexing.delete_total"), - "indices[index={index_name}].primaries.indexing.delete-time": - Stat("counter", "primaries.indexing.delete_time_in_millis"), - "indices[index={index_name}].primaries.indexing.delete-current": - Stat("gauge", "primaries.indexing.delete_current"), - + "indices[index={index_name}].primaries.indexing.index-total": Stat("counter", "primaries.indexing.index_total"), + "indices[index={index_name}].primaries.indexing.index-time": Stat( + "counter", "primaries.indexing.index_time_in_millis" + ), + "indices[index={index_name}].primaries.indexing.index-current": Stat("gauge", "primaries.indexing.index_current"), + "indices[index={index_name}].primaries.indexing.delete-total": Stat("counter", "primaries.indexing.delete_total"), + "indices[index={index_name}].primaries.indexing.delete-time": Stat( + "counter", "primaries.indexing.delete_time_in_millis" + ), + "indices[index={index_name}].primaries.indexing.delete-current": Stat("gauge", "primaries.indexing.delete_current"), # GET - "indices[index={index_name}].primaries.get.time": - Stat("counter", "primaries.get.time_in_millis"), - "indices[index={index_name}].primaries.get.exists-total": - Stat("counter", "primaries.get.exists_total"), - "indices[index={index_name}].primaries.get.exists-time": - Stat("counter", "primaries.get.exists_time_in_millis"), - "indices[index={index_name}].primaries.get.missing-total": - Stat("counter", "primaries.get.missing_total"), - "indices[index={index_name}].primaries.get.missing-time": - Stat("counter", "primaries.get.missing_time_in_millis"), - "indices[index={index_name}].primaries.get.current": - Stat("gauge", "primaries.get.current"), - + "indices[index={index_name}].primaries.get.time": Stat("counter", "primaries.get.time_in_millis"), + "indices[index={index_name}].primaries.get.exists-total": Stat("counter", "primaries.get.exists_total"), + "indices[index={index_name}].primaries.get.exists-time": Stat("counter", "primaries.get.exists_time_in_millis"), + "indices[index={index_name}].primaries.get.missing-total": Stat("counter", "primaries.get.missing_total"), + "indices[index={index_name}].primaries.get.missing-time": Stat("counter", "primaries.get.missing_time_in_millis"), + "indices[index={index_name}].primaries.get.current": Stat("gauge", "primaries.get.current"), # SEARCH - "indices[index={index_name}].primaries.search.open-contexts": - Stat("gauge", "primaries.search.open_contexts"), - "indices[index={index_name}].primaries.search.query-total": - Stat("counter", "primaries.search.query_total"), - "indices[index={index_name}].primaries.search.query-time": - Stat("counter", "primaries.search.query_time_in_millis"), - "indices[index={index_name}].primaries.search.query-current": - Stat("gauge", "primaries.search.query_current"), - "indices[index={index_name}].primaries.search.fetch-total": - Stat("counter", "primaries.search.fetch_total"), - "indices[index={index_name}].primaries.search.fetch-time": - Stat("counter", "primaries.search.fetch_time_in_millis"), - "indices[index={index_name}].primaries.search.fetch-current": - Stat("gauge", "primaries.search.fetch_current"), - + "indices[index={index_name}].primaries.search.open-contexts": Stat("gauge", "primaries.search.open_contexts"), + "indices[index={index_name}].primaries.search.query-total": Stat("counter", "primaries.search.query_total"), + "indices[index={index_name}].primaries.search.query-time": Stat("counter", "primaries.search.query_time_in_millis"), + "indices[index={index_name}].primaries.search.query-current": Stat("gauge", "primaries.search.query_current"), + "indices[index={index_name}].primaries.search.fetch-total": Stat("counter", "primaries.search.fetch_total"), + "indices[index={index_name}].primaries.search.fetch-time": Stat("counter", "primaries.search.fetch_time_in_millis"), + "indices[index={index_name}].primaries.search.fetch-current": Stat("gauge", "primaries.search.fetch_current"), # TOTAL # # DOCS - "indices[index={index_name}].total.docs.count": - Stat("gauge", "total.docs.count"), - "indices[index={index_name}].total.docs.deleted": - Stat("gauge", "total.docs.deleted"), - + "indices[index={index_name}].total.docs.count": Stat("gauge", "total.docs.count"), + "indices[index={index_name}].total.docs.deleted": Stat("gauge", "total.docs.deleted"), # STORE - "indices[index={index_name}].total.store.size": - Stat("gauge", "total.store.size_in_bytes"), - "indices[index={index_name}].total.store.throttle-time": - Stat("counter", "total.store.throttle_time_in_millis"), - + "indices[index={index_name}].total.store.size": Stat("gauge", "total.store.size_in_bytes"), + "indices[index={index_name}].total.store.throttle-time": Stat("counter", "total.store.throttle_time_in_millis"), # INDEXING - "indices[index={index_name}].total.indexing.index-total": - Stat("counter", "total.indexing.index_total"), - "indices[index={index_name}].total.indexing.index-time": - Stat("counter", "total.indexing.index_time_in_millis"), - "indices[index={index_name}].total.indexing.index-current": - Stat("gauge", "total.indexing.index_current"), - "indices[index={index_name}].total.indexing.delete-total": - Stat("counter", "total.indexing.delete_total"), - "indices[index={index_name}].total.indexing.delete-time": - Stat("counter", "total.indexing.delete_time_in_millis"), - "indices[index={index_name}].total.indexing.delete-current": - Stat("gauge", "total.indexing.delete_current"), - + "indices[index={index_name}].total.indexing.index-total": Stat("counter", "total.indexing.index_total"), + "indices[index={index_name}].total.indexing.index-time": Stat("counter", "total.indexing.index_time_in_millis"), + "indices[index={index_name}].total.indexing.index-current": Stat("gauge", "total.indexing.index_current"), + "indices[index={index_name}].total.indexing.delete-total": Stat("counter", "total.indexing.delete_total"), + "indices[index={index_name}].total.indexing.delete-time": Stat("counter", "total.indexing.delete_time_in_millis"), + "indices[index={index_name}].total.indexing.delete-current": Stat("gauge", "total.indexing.delete_current"), # GET - "indices[index={index_name}].total.get.total": - Stat("counter", "total.get.total"), - "indices[index={index_name}].total.get.time": - Stat("counter", "total.get.time_in_millis"), - "indices[index={index_name}].total.get.exists-total": - Stat("counter", "total.get.exists_total"), - "indices[index={index_name}].total.get.exists-time": - Stat("counter", "total.get.exists_time_in_millis"), - "indices[index={index_name}].total.get.missing-total": - Stat("counter", "total.get.missing_total"), - "indices[index={index_name}].total.get.missing-time": - Stat("counter", "total.get.missing_time_in_millis"), - "indices[index={index_name}].total.get.current": - Stat("gauge", "total.get.current"), - + "indices[index={index_name}].total.get.total": Stat("counter", "total.get.total"), + "indices[index={index_name}].total.get.time": Stat("counter", "total.get.time_in_millis"), + "indices[index={index_name}].total.get.exists-total": Stat("counter", "total.get.exists_total"), + "indices[index={index_name}].total.get.exists-time": Stat("counter", "total.get.exists_time_in_millis"), + "indices[index={index_name}].total.get.missing-total": Stat("counter", "total.get.missing_total"), + "indices[index={index_name}].total.get.missing-time": Stat("counter", "total.get.missing_time_in_millis"), + "indices[index={index_name}].total.get.current": Stat("gauge", "total.get.current"), # SEARCH - "indices[index={index_name}].total.search.open-contexts": - Stat("gauge", "total.search.open_contexts"), - "indices[index={index_name}].total.search.query-total": - Stat("counter", "total.search.query_total"), - "indices[index={index_name}].total.search.query-time": - Stat("counter", "total.search.query_time_in_millis"), - "indices[index={index_name}].total.search.query-current": - Stat("gauge", "total.search.query_current"), - "indices[index={index_name}].total.search.fetch-total": - Stat("counter", "total.search.fetch_total"), - + "indices[index={index_name}].total.search.open-contexts": Stat("gauge", "total.search.open_contexts"), + "indices[index={index_name}].total.search.query-total": Stat("counter", "total.search.query_total"), + "indices[index={index_name}].total.search.query-time": Stat("counter", "total.search.query_time_in_millis"), + "indices[index={index_name}].total.search.query-current": Stat("gauge", "total.search.query_current"), + "indices[index={index_name}].total.search.fetch-total": Stat("counter", "total.search.fetch_total"), # MERGES - "indices[index={index_name}].total.merges.total-docs": - Stat("counter", "total.merges.total_docs"), - "indices[index={index_name}].total.merges.total-size": - Stat("bytes", "total.merges.total_size_in_bytes"), - "indices[index={index_name}].total.merges.current": - Stat("gauge", "total.merges.current"), - "indices[index={index_name}].total.merges.total": - Stat("counter", "total.merges.total"), - "indices[index={index_name}].total.merges.current-docs": - Stat("gauge", "total.merges.current_docs"), - "indices[index={index_name}].total.merges.total-time": - Stat("counter", "total.merges.total_time_in_millis"), - "indices[index={index_name}].total.merges.current-size": - Stat("gauge", "total.merges.current_size_in_bytes"), - + "indices[index={index_name}].total.merges.total-docs": Stat("counter", "total.merges.total_docs"), + "indices[index={index_name}].total.merges.total-size": Stat("bytes", "total.merges.total_size_in_bytes"), + "indices[index={index_name}].total.merges.current": Stat("gauge", "total.merges.current"), + "indices[index={index_name}].total.merges.total": Stat("counter", "total.merges.total"), + "indices[index={index_name}].total.merges.current-docs": Stat("gauge", "total.merges.current_docs"), + "indices[index={index_name}].total.merges.total-time": Stat("counter", "total.merges.total_time_in_millis"), + "indices[index={index_name}].total.merges.current-size": Stat("gauge", "total.merges.current_size_in_bytes"), # FILTER_CACHE - "indices[index={index_name}].total.filter-cache.evictions": - Stat("counter", "total.filter_cache.evictions"), - "indices[index={index_name}].total.filter-cache.memory-size": - Stat("gauge", "total.filter_cache.memory_size_in_bytes"), - + "indices[index={index_name}].total.filter-cache.evictions": Stat("counter", "total.filter_cache.evictions"), + "indices[index={index_name}].total.filter-cache.memory-size": Stat( + "gauge", "total.filter_cache.memory_size_in_bytes" + ), # FIELDDATA - "indices[index={index_name}].total.fielddata.memory-size": - Stat("gauge", "total.fielddata.memory_size_in_bytes"), - "indices[index={index_name}].total.fielddata.evictions": - Stat("counter", "total.fielddata.evictions"), - + "indices[index={index_name}].total.fielddata.memory-size": Stat("gauge", "total.fielddata.memory_size_in_bytes"), + "indices[index={index_name}].total.fielddata.evictions": Stat("counter", "total.fielddata.evictions"), } # ElasticSearch cluster stats (1.0.0 and later) CLUSTER_STATS = { - 'cluster.active-primary-shards': Stat("gauge", "active_primary_shards"), - 'cluster.active-shards': Stat("gauge", "active_shards"), - 'cluster.initializing-shards': Stat("gauge", "initializing_shards"), - 'cluster.number-of-data_nodes': Stat("gauge", "number_of_data_nodes"), - 'cluster.number-of-nodes': Stat("gauge", "number_of_nodes"), - 'cluster.relocating-shards': Stat("gauge", "relocating_shards"), - 'cluster.unassigned-shards': Stat("gauge", "unassigned_shards"), - 'cluster.status': Stat("gauge", "status"), + "cluster.active-primary-shards": Stat("gauge", "active_primary_shards"), + "cluster.active-shards": Stat("gauge", "active_shards"), + "cluster.initializing-shards": Stat("gauge", "initializing_shards"), + "cluster.number-of-data_nodes": Stat("gauge", "number_of_data_nodes"), + "cluster.number-of-nodes": Stat("gauge", "number_of_nodes"), + "cluster.relocating-shards": Stat("gauge", "relocating_shards"), + "cluster.unassigned-shards": Stat("gauge", "unassigned_shards"), + "cluster.status": Stat("gauge", "status"), } # Thread pool metrics -THREAD_POOL_METRICS = { - "gauge": ['threads', 'queue', 'active', 'largest'], - "counter": ['completed', 'rejected'], -} +THREAD_POOL_METRICS = {"gauge": ["threads", "queue", "active", "largest"], "counter": ["completed", "rejected"]} # collectd callbacks @@ -635,7 +420,7 @@ def read_callback(cluster): interval. If this method throws, the plugin will be skipped for an increasing amount of time until it returns normally again""" - log.info('Read callback called') + log.info("Read callback called") # determine node information cluster.load_es_info() @@ -667,37 +452,34 @@ def configure_callback(conf): c = Cluster() for node in conf.children: - if node.key == 'Host': + if node.key == "Host": c.es_host = node.values[0] - elif node.key == 'Port': + elif node.key == "Port": c.es_port = int(node.values[0]) - elif node.key == 'Protocol': + elif node.key == "Protocol": c.es_url_scheme = node.values[0] - log.notice( - 'overriding elasticsearch url scheme to %s' % c.es_url_scheme) - elif node.key == 'Username': + log.notice("overriding elasticsearch url scheme to %s" % c.es_url_scheme) + elif node.key == "Username": c.es_username = node.values[0] - elif node.key == 'Password': + elif node.key == "Password": c.es_password = node.values[0] - elif node.key == 'Verbose': + elif node.key == "Verbose": handle.verbose = str_to_bool(node.values[0]) - elif node.key == 'Cluster': + elif node.key == "Cluster": c.es_cluster_from_config = node.values[0] - log.notice( - 'overriding elasticsearch cluster name to %s' % c.es_cluster_from_config) - elif node.key == 'Version': + log.notice("overriding elasticsearch cluster name to %s" % c.es_cluster_from_config) + elif node.key == "Version": c.es_version = node.values[0] - log.notice( - 'overriding elasticsearch version number to %s' % c.es_version) - elif node.key == 'Indexes': + log.notice("overriding elasticsearch version number to %s" % c.es_version) + elif node.key == "Indexes": c.es_index = node.values - elif node.key == 'EnableIndexStats': + elif node.key == "EnableIndexStats": c.enable_index_stats = str_to_bool(node.values[0]) - elif node.key == 'EnableClusterHealth': + elif node.key == "EnableClusterHealth": c.enable_cluster_stats = str_to_bool(node.values[0]) - elif node.key == 'Interval': + elif node.key == "Interval": c.collection_interval = int(node.values[0]) - elif node.key == 'IndexInterval': + elif node.key == "IndexInterval": c.index_interval = int(node.values[0]) elif node.key == "DetailedMetrics": c.detailed_metrics = str_to_bool(node.values[0]) @@ -707,8 +489,8 @@ def configure_callback(conf): for thread_pool in node.values: c.configured_thread_pools.add(thread_pool) # Include required thread pools (search and index) - c.configured_thread_pools.add('search') - c.configured_thread_pools.add('index') + c.configured_thread_pools.add("search") + c.configured_thread_pools.add("index") elif node.key == "AdditionalMetrics": for metric_name in node.values: c.defaults.add(metric_name) @@ -717,67 +499,63 @@ def configure_callback(conf): elif node.key == "Dimensions": c.extra_dimensions = node.values[0] else: - log.warning('Unknown config key: %s.' % node.key) - - log.info('host: %s' % c.es_host) - log.info('port: %s' % c.es_port) - log.info('es_index: %s' % c.es_index) - log.info('enable_index_stats: %s' % c.enable_index_stats) - log.info('enable_cluster_stats: %s' % c.enable_cluster_stats) - log.info('self.collection_interval: %s' % c.collection_interval) - log.info('index_interval: %s' % c.index_interval) - log.info('detailed_metrics: %s' % c.detailed_metrics) - log.info('index_summary_only: %s' % c.index_summary_only) - log.info('configured_thread_pools: %s' % c.configured_thread_pools) - log.info('metrics to collect: %s' % c.defaults) - log.info('master_only: %s' % c.master_only) + log.warning("Unknown config key: %s." % node.key) + + log.info("host: %s" % c.es_host) + log.info("port: %s" % c.es_port) + log.info("es_index: %s" % c.es_index) + log.info("enable_index_stats: %s" % c.enable_index_stats) + log.info("enable_cluster_stats: %s" % c.enable_cluster_stats) + log.info("self.collection_interval: %s" % c.collection_interval) + log.info("index_interval: %s" % c.index_interval) + log.info("detailed_metrics: %s" % c.detailed_metrics) + log.info("index_summary_only: %s" % c.index_summary_only) + log.info("configured_thread_pools: %s" % c.configured_thread_pools) + log.info("metrics to collect: %s" % c.defaults) + log.info("master_only: %s" % c.master_only) # register the read callback now that we have the complete config - collectd.register_read(read_callback, - interval=c.collection_interval, - name=get_unique_name(c.es_host, - c.es_port, c.es_index), - data=c) - log.notice( - 'started elasticsearch plugin with interval = %d seconds' % - c.collection_interval) + collectd.register_read( + read_callback, interval=c.collection_interval, name=get_unique_name(c.es_host, c.es_port, c.es_index), data=c + ) + log.notice("started elasticsearch plugin with interval = %d seconds" % c.collection_interval) def get_unique_name(host, port, index): if index: - return ('%s:%s:%s' % (host, port, index)) + return "%s:%s:%s" % (host, port, index) else: - return ('%s:%s' % (host, port)) + return "%s:%s" % (host, port) def remove_deprecated_elements(deprecated, elements, version): """Remove deprecated items from a list or dictionary""" # Attempt to parse the major, minor, and revision - (major, minor, revision) = version.split('.') + (major, minor, revision) = version.split(".") # Strings to int and sanitize alphas and betas from revision number - (major, minor, revision) = (int(major), int(minor), int(revision.split('-')[0])) + (major, minor, revision) = (int(major), int(minor), int(revision.split("-")[0])) # Iterate over deprecation lists and remove any keys that were deprecated # prior to the current version for dep in deprecated: - if (major > dep['major']) \ - or (major == dep['major'] and minor > dep['minor']) \ - or (major == dep['major'] and minor == dep['minor'] - and revision >= dep['revision']): + if ( + (major > dep["major"]) + or (major == dep["major"] and minor > dep["minor"]) + or (major == dep["major"] and minor == dep["minor"] and revision >= dep["revision"]) + ): if type(elements) is list: - for key in dep['keys']: + for key in dep["keys"]: if key in elements: elements.remove(key) if type(elements) is dict: - for key in dep['keys']: + for key in dep["keys"]: if key in elements: del elements[key] return elements class Cluster(object): - def __init__(self): self.collection_interval = 10 self.es_host = "localhost" @@ -816,7 +594,7 @@ def __init__(self): self.es_current_master = False self.node_id = None - self.extra_dimensions = '' + self.extra_dimensions = "" def sanitize_intervals(self): """Sanitizes the index interval to be greater or equal to and divisible @@ -829,54 +607,61 @@ def sanitize_intervals(self): # ? self.index_interval % self.collection_interval > 0: # round the self.index_interval up to a compatible value if self.index_interval % self.collection_interval > 0: - self.index_interval = self.index_interval + \ - self.collection_interval - \ - (self.index_interval % - self.collection_interval) - log.warning(('The Elasticsearch Index Interval must be ' - 'greater or equal to than and divisible by the ' - 'collection Interval. The Elasticsearch Index ' - 'Interval has been rounded to: %s') % - self.index_interval) + self.index_interval = ( + self.index_interval + self.collection_interval - (self.index_interval % self.collection_interval) + ) + log.warning( + ( + "The Elasticsearch Index Interval must be " + "greater or equal to than and divisible by the " + "collection Interval. The Elasticsearch Index " + "Interval has been rounded to: %s" + ) + % self.index_interval + ) # ? self.index_interval < self.collection_interval : # Set self.index_interval = self.collection_interval elif self.index_interval < self.collection_interval: self.index_interval = self.collection_interval - log.warning(('WARN: The Elasticsearch Index Interval must be ' - 'greater or equal to than and divisible by the ' - 'collection Interval. The Elasticsearch Index ' - 'Interval has been rounded to: %s') % - self.index_interval) + log.warning( + ( + "WARN: The Elasticsearch Index Interval must be " + "greater or equal to than and divisible by the " + "collection Interval. The Elasticsearch Index " + "Interval has been rounded to: %s" + ) + % self.index_interval + ) # self.index_skip = self.index_interval / self.collection_interval - self.index_skip = (self.index_interval / self.collection_interval) + self.index_skip = self.index_interval / self.collection_interval # ENSURE INDEX IS COLLECTED ON THE FIRST COLLECTION self.skip_count = self.index_skip def remove_deprecated_node_stats(self): """Remove deprecated node stats from the list of stats to collect""" - self.node_stats_cur = remove_deprecated_elements(DEPRECATED_NODE_STATS, - self.node_stats_cur, - self.es_version) + self.node_stats_cur = remove_deprecated_elements(DEPRECATED_NODE_STATS, self.node_stats_cur, self.es_version) def remove_deprecated_threads(self): """Remove deprecated thread_pools from the list of stats to collect""" - self.thread_pools = remove_deprecated_elements(DEPRECATED_THREAD_POOLS, - self.thread_pools, - self.es_version) + self.thread_pools = remove_deprecated_elements(DEPRECATED_THREAD_POOLS, self.thread_pools, self.es_version) # helper methods def init_stats(self): self.sanitize_intervals() - self.es_node_url = self.es_url_scheme + "://" + self.es_host + ":" + \ - str(self.es_port) + \ - ("/_nodes/_local/stats/transport,http,process,jvm,indices," - "thread_pool") - self.node_stats_cur = dict(NODE_STATS.items()) - self.index_stats_cur = dict(INDEX_STATS.items()) + self.es_node_url = ( + self.es_url_scheme + + "://" + + self.es_host + + ":" + + str(self.es_port) + + ("/_nodes/_local/stats/transport,http,process,jvm,indices," "thread_pool") + ) + self.node_stats_cur = dict(list(NODE_STATS.items())) + self.index_stats_cur = dict(list(INDEX_STATS.items())) if not self.es_version.startswith("1."): self.node_stats_cur.update(NODE_STATS_ES_2) @@ -889,8 +674,7 @@ def init_stats(self): self.remove_deprecated_node_stats() - if self.es_version.startswith("1.1") \ - or self.es_version.startswith("1.2"): + if self.es_version.startswith("1.1") or self.es_version.startswith("1.2"): self.index_stats_cur.update(INDEX_STATS_ES_1_1) else: # 1.3 and higher @@ -900,64 +684,66 @@ def init_stats(self): # version agnostic settings if not self.es_index: # get all index stats - self.es_index_url = self.es_url_scheme + "://" + self.es_host + \ - ":" + str(self.es_port) + "/_all/_stats" + self.es_index_url = self.es_url_scheme + "://" + self.es_host + ":" + str(self.es_port) + "/_all/_stats" else: - self.es_index_url = self.es_url_scheme + "://" + self.es_host + \ - ":" + str(self.es_port) + "/" + \ - ",".join(self.es_index) + "/_stats" + self.es_index_url = ( + self.es_url_scheme + + "://" + + self.es_host + + ":" + + str(self.es_port) + + "/" + + ",".join(self.es_index) + + "/_stats" + ) # common thread pools for all ES versions - thread_pools = ['generic', 'index', 'get', 'snapshot', 'bulk', - 'warmer', 'flush', 'search', 'refresh'] + thread_pools = ["generic", "index", "get", "snapshot", "bulk", "warmer", "flush", "search", "refresh"] # Add the 1.0 metrics if not self.es_version.startswith("0."): - thread_pools.extend(['merge', 'optimize']) + thread_pools.extend(["merge", "optimize"]) # Add the 2.0 metrics if not self.es_version.startswith("1."): - thread_pools.extend(['suggest', 'percolate', 'management', - 'listener', 'fetch_shard_store', - 'fetch_shard_started']) + thread_pools.extend( + ["suggest", "percolate", "management", "listener", "fetch_shard_store", "fetch_shard_started"] + ) # Add the 2.1 metrics - if not self.es_version.startswith("1.") \ - and not self.es_version.startswith("2.0"): - thread_pools.extend(['force_merge']) + if not self.es_version.startswith("1.") and not self.es_version.startswith("2.0"): + thread_pools.extend(["force_merge"]) # Add the 6.3 metrics if self.es_version.startswith("6."): if self.es_version[2:].find(".") != -1: - minor_release_number = self.es_version[2: 2 + self.es_version[2:].find(".")] + minor_release_number = self.es_version[2 : 2 + self.es_version[2:].find(".")] else: minor_release_number = self.es_version[2:] if int(minor_release_number) > 2: - thread_pools.extend(['write']) - elif int(self.es_version[:self.es_version.find(".")]) > 6: - thread_pools.extend(['write']) + thread_pools.extend(["write"]) + elif int(self.es_version[: self.es_version.find(".")]) > 6: + thread_pools.extend(["write"]) # Legacy support for old configurations without Thread Pools config if len(self.configured_thread_pools) == 0: self.thread_pools = list(self.configured_thread_pools) else: # Filter out the thread pools that aren't specified by user - self.thread_pools = filter(lambda pool: pool in - self.configured_thread_pools, - thread_pools) + self.thread_pools = [pool for pool in thread_pools if pool in self.configured_thread_pools] self.remove_deprecated_threads() - self.es_cluster_url = self.es_url_scheme + "://" + self.es_host + \ - ":" + str(self.es_port) + "/_cluster/health" + self.es_cluster_url = self.es_url_scheme + "://" + self.es_host + ":" + str(self.es_port) + "/_cluster/health" - log.notice('Initialized with version=%s, host=%s, port=%s, url=%s' % - (self.es_version, self.es_host, self.es_port, - self.es_node_url)) + log.notice( + "Initialized with version=%s, host=%s, port=%s, url=%s" + % (self.es_version, self.es_host, self.es_port, self.es_node_url) + ) # FUNCTION: Collect node stats from JSON result def lookup_node_stat(self, stat, json): - node = json['nodes'].keys()[0] + node = list(json["nodes"].keys())[0] val = dig_it_up(json, self.node_stats_cur[stat].path % node) return val @@ -972,13 +758,13 @@ def fetch_stats(self): # Only if Cluster name is not provided as a config option, use the # value returned from the ES endpoint if self.es_cluster_from_config is None: - self.es_cluster = node_json_stats['cluster_name'] + self.es_cluster = node_json_stats["cluster_name"] else: self.es_cluster = self.es_cluster_from_config - log.info('Configured with cluster_name=%s' % self.es_cluster) - log.info('Parsing node_json_stats') + log.info("Configured with cluster_name=%s" % self.es_cluster) + log.info("Parsing node_json_stats") self.parse_node_stats(node_json_stats, self.node_stats_cur) - log.info('Parsing thread pool stats') + log.info("Parsing thread pool stats") self.parse_thread_pool_stats(node_json_stats, self.thread_pools) # check the current master @@ -988,77 +774,70 @@ def fetch_stats(self): # avoids collecting too many metrics if the cluster has a lot of nodes if self.enable_cluster_stats and self.es_master_eligible: cluster_json_stats = self.fetch_url(self.es_cluster_url) - log.info('Parsing cluster stats') + log.info("Parsing cluster stats") self.parse_cluster_stats(cluster_json_stats, CLUSTER_STATS) - if (self.enable_index_stats and self.es_master_eligible and - self.skip_count >= self.index_skip) \ - and ((self.master_only and self.es_current_master) - or (not self.master_only)): + if (self.enable_index_stats and self.es_master_eligible and self.skip_count >= self.index_skip) and ( + (self.master_only and self.es_current_master) or (not self.master_only) + ): # Reset skip count self.skip_count = 0 indices = self.fetch_url(self.es_index_url) if indices: if self.index_summary_only: - log.info('Parsing index stats for _all summary') - self.parse_index_stats(indices['_all'], '_all') + log.info("Parsing index stats for _all summary") + self.parse_index_stats(indices["_all"], "_all") else: - indexes_json_stats = indices['indices'] - for index_name in indexes_json_stats.keys(): - log.info('Parsing index stats for index: %s' % - index_name) - self.parse_index_stats(indexes_json_stats[index_name], - index_name) + indexes_json_stats = indices["indices"] + for index_name in list(indexes_json_stats.keys()): + log.info("Parsing index stats for index: %s" % index_name) + self.parse_index_stats(indexes_json_stats[index_name], index_name) # Increment skip count self.skip_count += 1 def fetch_url(self, url): response = None try: - log.info('Fetching api information from: %s' % url) - request = urllib2.Request(url) + log.info("Fetching api information from: %s" % url) + request = urllib.request.Request(url) if self.es_username: - authheader = base64.encodestring('%s:%s' % - (self.es_username, - self.es_password) - ).replace('\n', '') + authheader = base64.encodebytes( + ("%s:%s" % (self.es_username, self.es_password)).encode("utf-8") + ).replace(b"\n", b"") request.add_header("Authorization", "Basic %s" % authheader) ctx = None if self.es_url_scheme == "https": ctx = ssl._create_unverified_context() - response = urllib2.urlopen(request, context=ctx, timeout=10) + response = urllib.request.urlopen(request, context=ctx, timeout=10) else: - response = urllib2.urlopen(request, timeout=10) - log.info('Raw api response: %s' % response) + response = urllib.request.urlopen(request, timeout=10) + log.info("Raw api response: %s" % response) return json.load(response) - except (urllib2.URLError, urllib2.HTTPError), e: - log.error('Error connecting to %s - %r : %s' % - (url, e, e)) + except (urllib.error.URLError, urllib.error.HTTPError) as e: + log.error("Error connecting to %s - %r : %s" % (url, e, e)) return None finally: if response is not None: response.close() def load_es_info(self): - json = self.fetch_url(self.es_url_scheme + "://" + self.es_host + ":" + - str(self.es_port) + "/_nodes/_local") + json = self.fetch_url(self.es_url_scheme + "://" + self.es_host + ":" + str(self.es_port) + "/_nodes/_local") if json is None: return # Identify the current node - self.node_id = json['nodes'].keys()[0] - log.notice('current node id: %s' % self.node_id) + self.node_id = list(json["nodes"].keys())[0] + log.notice("current node id: %s" % self.node_id) # we should have only one entry with the current node information - node_info = json['nodes'].itervalues().next() - version = node_info['version'] + node_info = next(iter(json["nodes"].values())) + version = node_info["version"] # a node is master eligible by default unless it's configured otherwise # Note: settings is deprecated from json starting ES 5.5 master_eligible = True try: - if 'node' in node_info['settings'] and \ - 'master' in node_info['settings']['node']: - master_eligible = node_info['settings']['node']['master'] == 'true' + if "node" in node_info["settings"] and "master" in node_info["settings"]["node"]: + master_eligible = node_info["settings"]["node"]["master"] == "true" except KeyError: pass @@ -1066,34 +845,34 @@ def load_es_info(self): self.es_master_eligible = master_eligible self.es_version = version - log.notice('version: %s, cluster: %s, master eligible: %s' % - (self.es_version, self.es_cluster, self.es_master_eligible)) + log.notice( + "version: %s, cluster: %s, master eligible: %s" + % (self.es_version, self.es_cluster, self.es_master_eligible) + ) def detect_es_master(self): """Determines if this is the current master. This method sets self.es_current_master""" # determine current master - cluster_state = self.fetch_url(self.es_url_scheme + "://" + - self.es_host + ":" + str(self.es_port) + - "/_cluster/state/master_node") + cluster_state = self.fetch_url( + self.es_url_scheme + "://" + self.es_host + ":" + str(self.es_port) + "/_cluster/state/master_node" + ) if cluster_state is None: self.es_current_master = False return - if self.es_current_master is False \ - and cluster_state['master_node'] == self.node_id: + if self.es_current_master is False and cluster_state["master_node"] == self.node_id: self.es_current_master = True - log.notice('current master: %s' % self.es_current_master) - elif self.es_current_master is True \ - and cluster_state['master_node'] != self.node_id: + log.notice("current master: %s" % self.es_current_master) + elif self.es_current_master is True and cluster_state["master_node"] != self.node_id: self.es_current_master = False - log.notice('current master: %s' % self.es_current_master) + log.notice("current master: %s" % self.es_current_master) else: - log.debug('current master: %s' % self.es_current_master) + log.debug("current master: %s" % self.es_current_master) def parse_node_stats(self, json, stats): """Parse node stats response from ElasticSearch""" - for name, key in stats.iteritems(): + for name, key in stats.items(): if self.detailed_metrics is True or name in self.defaults: result = self.lookup_node_stat(name, json) self.dispatch_stat(result, name, key) @@ -1101,93 +880,93 @@ def parse_node_stats(self, json, stats): def parse_thread_pool_stats(self, json, stats): """Parse thread pool stats response from ElasticSearch""" for pool in self.thread_pools: - for metric_type, value in THREAD_POOL_METRICS.iteritems(): + for metric_type, value in THREAD_POOL_METRICS.items(): for attr in value: - name = 'thread_pool.{0}'.format(attr) - key = Stat(metric_type, 'nodes.%s.thread_pool.{0}.{1}'. - format(pool, attr)) + name = "thread_pool.{0}".format(attr) + key = Stat(metric_type, "nodes.%s.thread_pool.{0}.{1}".format(pool, attr)) if self.detailed_metrics is True or name in self.defaults: - node = json['nodes'].keys()[0] + node = list(json["nodes"].keys())[0] result = dig_it_up(json, key.path % node) - self.dispatch_stat(result, name, key, - {'thread_pool': pool}) + self.dispatch_stat(result, name, key, {"thread_pool": pool}) def parse_cluster_stats(self, json, stats): """Parse cluster stats response from ElasticSearch""" # convert the status color into a number - json['status'] = CLUSTER_STATUS[json['status']] - for name, key in stats.iteritems(): + json["status"] = CLUSTER_STATUS[json["status"]] + for name, key in stats.items(): if self.detailed_metrics is True or name in self.defaults: result = dig_it_up(json, key.path) self.dispatch_stat(result, name, key) def parse_index_stats(self, json, index_name): """Parse index stats response from ElasticSearch""" - for name, key in self.index_stats_cur.iteritems(): + for name, key in self.index_stats_cur.items(): # filter default metrics - if self.detailed_metrics is True or \ - name.replace("[index={index_name}]", "") in self.defaults: + if self.detailed_metrics is True or name.replace("[index={index_name}]", "") in self.defaults: result = dig_it_up(json, key.path) # update the index name in the type_instance to include # the index as a dimensions - name = name.format( - index_name=sanitize_type_instance(index_name)) + name = name.format(index_name=sanitize_type_instance(index_name)) self.dispatch_stat(result, name, key) def dispatch_stat(self, result, name, key, dimensions=None): """Read a key from info response data and dispatch a value""" - log.info(('Parameters to be emitted:\n name: {n}\n key: {k}' - '\n dimensions: {d}\n result: {r}').format(n=name, - k=key, - d=dimensions, - r=result)) + log.info( + ("Parameters to be emitted:\n name: {n}\n key: {k}" "\n dimensions: {d}\n result: {r}").format( + n=name, k=key, d=dimensions, r=result + ) + ) if result is None: - log.debug('Value not found for %s' % name) + log.debug("Value not found for %s" % name) return # If we failed to get the cluster name and do not have a config # option set for it, do not emit data if self.es_cluster is None: - log.warning('Failed to determine Cluster name in read_callback' + - 'and no Cluster config option specified. ' + - 'Will not emit datapoints since plugin_instance ' + - 'which is the cluster name could not be determined') + log.warning( + "Failed to determine Cluster name in read_callback" + + "and no Cluster config option specified. " + + "Will not emit datapoints since plugin_instance " + + "which is the cluster name could not be determined" + ) return if self.es_version is None: - log.warning('Failed to determine ES version in read_callback. ' + - 'Will not emit datapoints since plugin_instance ' + - 'this interval. Will attempt to fetch version in the ' + - 'next interval.') + log.warning( + "Failed to determine ES version in read_callback. " + + "Will not emit datapoints since plugin_instance " + + "this interval. Will attempt to fetch version in the " + + "next interval." + ) return estype = key.type value = int(result) - log.info('Sending value[%s]: %s=%s' % (estype, name, value)) + log.info("Sending value[%s]: %s=%s" % (estype, name, value)) - val = collectd.Values(plugin='elasticsearch') + val = collectd.Values(plugin="elasticsearch") val.plugin_instance = self.es_cluster # If dimensions are provided, format them and append # them to the plugin_instance dim_str = self.get_dimension_string(dimensions) if dim_str: - val.plugin_instance += '[{dims}]'.format(dims=dim_str) + val.plugin_instance += "[{dims}]".format(dims=dim_str) val.type = estype val.type_instance = name val.values = [value] - val.meta = {'0': True} - log.info('Emitting value: %s' % val) + val.meta = {"0": True} + log.info("Emitting value: %s" % val) val.dispatch() def get_dimension_string(self, dimensions): - dim_str = '' + dim_str = "" if dimensions: - dim_str = ','.join(['='.join(d) for d in dimensions.items()]) + dim_str = ",".join(["=".join(d) for d in list(dimensions.items())]) if self.extra_dimensions: - dim_str += "%s%s" % (',' if dim_str else '', self.extra_dimensions) + dim_str += "%s%s" % ("," if dim_str else "", self.extra_dimensions) return dim_str @@ -1198,15 +977,14 @@ def sanitize_type_instance(index_name): the '/' character. This method does a lossy conversion to ascii and replaces the reserved character with '_' """ - ascii_index_name = index_name.encode('ascii', 'ignore') # '/' is reserved, so we substitute it with '_' instead - return ascii_index_name.replace('/', '_') + return index_name.replace("/", "_").encode("ascii", "ignore") def dig_it_up(obj, path): try: - if type(path) in (str, unicode): - path = path.split('.') + if hasattr(path, "split"): + path = path.split(".") return reduce(lambda x, y: x[y], path, obj) except: return None @@ -1221,35 +999,35 @@ def __init__(self): self.value_mock = CollectdValuesMock def debug(self, msg): - print 'DEBUG: {0}'.format(msg) + print("DEBUG: {0}".format(msg)) def info(self, msg): - print 'INFO: {0}'.format(msg) + print("INFO: {0}".format(msg)) def notice(self, msg): - print 'NOTICE: {0}'.format(msg) + print("NOTICE: {0}".format(msg)) def warning(self, msg): - print 'WARN: {0}'.format(msg) + print("WARN: {0}".format(msg)) def error(self, msg): - print 'ERROR: {0}'.format(msg) + print("ERROR: {0}".format(msg)) sys.exit(1) - def Values(self, plugin='elasticsearch'): + def Values(self, plugin="elasticsearch"): return (self.value_mock)() class CollectdValuesMock(object): def dispatch(self): - print self + print(self) def __str__(self): attrs = [] for name in dir(self): - if not name.startswith('_') and name is not 'dispatch': + if not name.startswith("_") and name != "dispatch": attrs.append("{0}={1}".format(name, getattr(self, name))) - return "".format(' '.join(attrs)) + return "".format(" ".join(attrs)) class CollectdLogHandler(logging.Handler): @@ -1265,6 +1043,7 @@ class CollectdLogHandler(logging.Handler): plugin -- name of the plugin (default 'unknown') verbose -- enable/disable verbose messages (default False) """ + def __init__(self, plugin="elasticsearch", verbose=False): """Initializes CollectdLogHandler Arguments @@ -1284,21 +1063,18 @@ def emit(self, record): """ try: if record.msg is not None: - if record.levelname == 'ERROR': - collectd.error('%s : %s' % (self.plugin, record.msg)) - elif record.levelname == 'WARNING': - collectd.warning('%s : %s' % (self.plugin, record.msg)) - elif record.levelname == 'NOTICE': - collectd.notice('%s : %s' % (self.plugin, record.msg)) - elif record.levelname == 'INFO' and self.verbose is True: - collectd.info('%s : %s' % (self.plugin, record.msg)) - elif record.levelname == 'DEBUG' and self.verbose is True: - collectd.debug('%s : %s' % (self.plugin, record.msg)) + if record.levelname == "ERROR": + collectd.error("%s : %s" % (self.plugin, record.msg)) + elif record.levelname == "WARNING": + collectd.warning("%s : %s" % (self.plugin, record.msg)) + elif record.levelname == "NOTICE": + collectd.notice("%s : %s" % (self.plugin, record.msg)) + elif record.levelname == "INFO" and self.verbose is True: + collectd.info("%s : %s" % (self.plugin, record.msg)) + elif record.levelname == "DEBUG" and self.verbose is True: + collectd.debug("%s : %s" % (self.plugin, record.msg)) except Exception as e: - collectd.warning(('{p} [ERROR]: Failed to write log statement due ' - 'to: {e}').format(p=self.plugin, - e=e - )) + collectd.warning(("{p} [ERROR]: Failed to write log statement due " "to: {e}").format(p=self.plugin, e=e)) class CollectdLogger(logging.Logger): @@ -1312,6 +1088,7 @@ class CollectdLogger(logging.Logger): name -- name of the logger level -- log level to filter by """ + def __init__(self, name, level=logging.NOTSET): """Initializes CollectdLogger @@ -1320,7 +1097,7 @@ def __init__(self, name, level=logging.NOTSET): level -- log level to filter by """ logging.Logger.__init__(self, name, level) - logging.addLevelName(25, 'NOTICE') + logging.addLevelName(25, "NOTICE") def notice(self, msg): """Logs a 'NOTICE' level statement at level 25 @@ -1344,15 +1121,28 @@ def configure_test(cluster): """Configure the plugin for testing""" # Ensure all possible threadpools are eligible for collection - cluster.configured_thread_pools = set(['generic', 'index', 'get', - 'snapshot', 'bulk', 'warmer', - 'flush', 'search', 'refresh', - 'suggest', 'percolate', - 'management', 'listener', - 'fetch_shard_store', - 'fetch_shard_started', - 'force_merge', 'merge', - 'optimize', ]) + cluster.configured_thread_pools = set( + [ + "generic", + "index", + "get", + "snapshot", + "bulk", + "warmer", + "flush", + "search", + "refresh", + "suggest", + "percolate", + "management", + "listener", + "fetch_shard_store", + "fetch_shard_started", + "force_merge", + "merge", + "optimize", + ] + ) cluster.detailed_metrics = True cluster.index_interval = 10 cluster.enable_index_stats = True @@ -1360,8 +1150,9 @@ def configure_test(cluster): cluster.es_master_eligible = True -if __name__ == '__main__': +if __name__ == "__main__": import sys + c = Cluster() # allow user to override ES host name for easier testing if len(sys.argv) > 1: @@ -1374,4 +1165,5 @@ def configure_test(cluster): c.fetch_stats() else: import collectd + collectd.register_config(configure_callback) diff --git a/elasticsearch_collectd_test.py b/elasticsearch_collectd_test.py index e7c4a2b..fb274b6 100644 --- a/elasticsearch_collectd_test.py +++ b/elasticsearch_collectd_test.py @@ -23,7 +23,7 @@ class MockCollectd(mock.MagicMock): ) ]) def test_remove_deprecated_elements(deprecated_elements, input_elements, version, expected_elements): - print deprecated_elements, input_elements, version, expected_elements + print(deprecated_elements, input_elements, version, expected_elements) elements = remove_deprecated_elements(deprecated_elements, input_elements, version) assert len(elements) == len(expected_elements) for i in range(len(elements)): diff --git a/tests/integration/20-elasticsearch-test.conf b/tests/integration/20-elasticsearch-test.conf index ba5ceb8..86558ee 100644 --- a/tests/integration/20-elasticsearch-test.conf +++ b/tests/integration/20-elasticsearch-test.conf @@ -7,13 +7,6 @@ Import "elasticsearch_collectd" - - Interval 3 - IndexInterval 3 - Host es17 - Dimensions "testdim=5" - - Interval 3 IndexInterval 3 diff --git a/tests/integration/Dockerfile.sink b/tests/integration/Dockerfile.sink index 9696480..b0b61e0 100644 --- a/tests/integration/Dockerfile.sink +++ b/tests/integration/Dockerfile.sink @@ -1,4 +1,4 @@ -FROM python:2 +FROM python:3 EXPOSE 80 8080 diff --git a/tests/integration/Dockerfile.test b/tests/integration/Dockerfile.test index 8823264..f67174c 100644 --- a/tests/integration/Dockerfile.test +++ b/tests/integration/Dockerfile.test @@ -1,4 +1,4 @@ -FROM python:2 +FROM python:3 ADD test.py /opt/test.py CMD python -u /opt/test.py diff --git a/tests/integration/docker-compose.yml b/tests/integration/docker-compose.yml index 44b97df..ace8450 100644 --- a/tests/integration/docker-compose.yml +++ b/tests/integration/docker-compose.yml @@ -10,17 +10,11 @@ services: SF_INGEST_HOST: fake_sfx depends_on: - fake_sfx - - es17 - es24 - es53 - es56 - es63 - es17: - build: - context: . - dockerfile: Dockerfile.es.1.7.6 - es24: build: context: . diff --git a/tests/integration/make-es-dockerfiles b/tests/integration/make-es-dockerfiles index 74b6d59..6f1e924 100755 --- a/tests/integration/make-es-dockerfiles +++ b/tests/integration/make-es-dockerfiles @@ -1,6 +1,6 @@ #!/bin/bash -for version in 1.7.6 2.4.5 5.3.2 5.6.3 +for version in 2.4.5 5.3.2 5.6.3 do sed -e "s/ES_VERSION/$version/" Dockerfile.es > Dockerfile.es.$version done @@ -8,4 +8,4 @@ done for version in 6.3.2 do sed -e "s/ES_VERSION/$version/" Dockerfile2.es > Dockerfile.es.$version -done \ No newline at end of file +done diff --git a/tests/integration/sink.py b/tests/integration/sink.py index 8a28047..00837e9 100644 --- a/tests/integration/sink.py +++ b/tests/integration/sink.py @@ -1,7 +1,7 @@ -from BaseHTTPServer import HTTPServer, BaseHTTPRequestHandler import json import signal import threading +from http.server import BaseHTTPRequestHandler, HTTPServer from time import time # This module collects metrics from collectd and can echo them back out for @@ -13,7 +13,7 @@ def run_fake_ingest(metric_data): class FakeCollectdIngest(BaseHTTPRequestHandler): def do_POST(self): - body = self.rfile.read(int(self.headers.getheader('Content-Length'))) + body = self.rfile.read(int(self.headers.get("Content-Length"))) metric_data.extend(json.loads(body)) @@ -21,12 +21,12 @@ def do_POST(self): self.send_header("Content-Type", "text/ascii") self.send_header("Content-Length", "2") self.end_headers() - self.wfile.write("OK") + self.wfile.write("OK".encode("utf-8")) - print 'Starting ingest server on port 80' - httpd = HTTPServer(('', 80), FakeCollectdIngest) + print("Starting ingest server on port 80") + httpd = HTTPServer(("", 80), FakeCollectdIngest) httpd.serve_forever() - print 'Ingest server shutting down' + print("Ingest server shutting down") # Dumps all of the collected metrics back out as JSON upon request @@ -38,13 +38,13 @@ def do_GET(self): self.send_header("Content-Type", "application/json") self.send_header("Content-Length", str(len(data))) self.end_headers() - print data - self.rfile.write(data) + print(data) + self.wfile.write(data.encode("utf-8")) - print 'Starting metric spewer on port 8080' - httpd = HTTPServer(('', 8080), MetricDataSpewer) + print("Starting metric spewer on port 8080") + httpd = HTTPServer(("", 8080), MetricDataSpewer) httpd.serve_forever() - print 'Metric spewer shutting down' + print("Metric spewer shutting down") if __name__ == "__main__": @@ -58,4 +58,3 @@ def do_GET(self): t1.join() t2.join() - diff --git a/tests/integration/test.py b/tests/integration/test.py index df445d4..1783012 100644 --- a/tests/integration/test.py +++ b/tests/integration/test.py @@ -1,8 +1,8 @@ #!/usr/bin/env python -import httplib +import http.client import json -from time import time, sleep +from time import sleep, time # Quick and dirty integration test for multi-cluster support in one collectd # instance. This test script is intended to be run with docker-compose with the @@ -12,18 +12,17 @@ # integration tests if so desired. VERSIONS_TESTED_WITH_METRICS = { - '1.7.6': ['indices.indexing.index-total'], - '2.4.5': ['indices.cache.filter.evictions'], - '5.3.2': ['indices.cache.filter.evictions'], - '5.6.3': ['indices.cache.filter.evictions'], - '6.3.2': ['indices.cache.filter.evictions'] + "2.4.5": ["indices.cache.filter.evictions"], + "5.3.2": ["indices.cache.filter.evictions"], + "5.6.3": ["indices.cache.filter.evictions"], + "6.3.2": ["indices.cache.filter.evictions"], } TIMEOUT_SECS = 60 def get_metric_data(): # Use httplib instead of requests so we don't have to install stuff with pip - conn = httplib.HTTPConnection("fake_sfx", 8080) + conn = http.client.HTTPConnection("fake_sfx", 8080) conn.request("GET", "/") resp = conn.getresponse() conn.close() @@ -33,16 +32,19 @@ def get_metric_data(): def wait_for_metrics_from_each_cluster(): start = time() for cluster in VERSIONS_TESTED_WITH_METRICS: - c = 'es-%s' % (cluster,) - print 'Waiting for metrics from cluster %s...' % (c,) - eventually_true(lambda: any([c in m.get('plugin_instance') for m in get_metric_data()]), - TIMEOUT_SECS - (time() - start)) - print 'plugin_instance Found!' + c = "es-%s" % (cluster,) + print("Waiting for metrics from cluster %s..." % (c,)) + eventually_true( + lambda: any([c in m.get("plugin_instance") for m in get_metric_data()]), TIMEOUT_SECS - (time() - start) + ) + print("plugin_instance Found!") for metric in VERSIONS_TESTED_WITH_METRICS.get(cluster): - print 'Waiting for metric: %s from cluster %s...' % (metric, c) - eventually_true(lambda: any([metric in str(m.get('type_instance')) for m in get_metric_data()]), - TIMEOUT_SECS - (time() - start)) - print 'metric: %s Found! from cluster: %s' % (metric, c) + print("Waiting for metric: %s from cluster %s..." % (metric, c)) + eventually_true( + lambda: any([metric in str(m.get("type_instance")) for m in get_metric_data()]), + TIMEOUT_SECS - (time() - start), + ) + print("metric: %s Found! from cluster: %s" % (metric, c)) def eventually_true(f, timeout_secs): diff --git a/tests/integration/wait_for_es b/tests/integration/wait_for_es index 3b656c9..1c9c09e 100755 --- a/tests/integration/wait_for_es +++ b/tests/integration/wait_for_es @@ -8,7 +8,7 @@ wait_for () { done } -for host in es17 es24 es53 es56 es63 +for host in es24 es53 es56 es63 do wait_for $host done diff --git a/tests/simulate.py b/tests/simulate.py index 6ac7b6a..93179bf 100755 --- a/tests/simulate.py +++ b/tests/simulate.py @@ -2,20 +2,20 @@ import sys import os -from BaseHTTPServer import BaseHTTPRequestHandler, HTTPServer +from http.server import BaseHTTPRequestHandler, HTTPServer PORT_NUMBER = 9200 if len(sys.argv) == 2: base_path = sys.argv[1] if os.path.isdir(base_path): - print "simulating ES server from %s" % base_path + print("simulating ES server from %s" % base_path) else: - print "invalid or missing directory %s" % base_path + print("invalid or missing directory %s" % base_path) sys.exit(1) else: - print "usage: ./simulate.py directory_path. Example ./simulate.py \ -data/1.7.2" + print("usage: ./simulate.py directory_path. Example ./simulate.py \ +data/1.7.2") sys.exit(1) @@ -38,9 +38,9 @@ def serve_path(self, path): try: server = HTTPServer(('', PORT_NUMBER), EsSimulatorHandler) - print 'Started ES simulator on port ', PORT_NUMBER + print('Started ES simulator on port ', PORT_NUMBER) server.serve_forever() except KeyboardInterrupt: - print 'ctrl-c received, shutting down' + print('ctrl-c received, shutting down') server.socket.close()