From 66ebdff447cb455463d3c3c59011e029028795b5 Mon Sep 17 00:00:00 2001 From: Tim Sullivan Date: Fri, 7 Jul 2017 09:55:45 -0700 Subject: [PATCH 1/5] [Monitoring] Email actions for Cluster Alerts (elastic/x-pack-elasticsearch#1879) * [Monitoring] Email actions for Cluster Alerts * fix quotations in email fields * move email vars to transform, and rename for snake_case * add state to email subject for cluster status alert * remove types field in kibana_settings search * simplify email action condition script * uppercase the state for the email subject * only append state to email subject if alert is new * show state in email subject even when alert is resolved Original commit: elastic/x-pack-elasticsearch@e6fdd8d6205bda9999ff15001d2bccbacdfeb4df --- .../watches/elasticsearch_cluster_status.json | 52 +++++++++++++++++-- .../elasticsearch_version_mismatch.json | 48 ++++++++++++++++- .../watches/kibana_version_mismatch.json | 48 ++++++++++++++++- .../watches/logstash_version_mismatch.json | 48 ++++++++++++++++- 4 files changed, 185 insertions(+), 11 deletions(-) diff --git a/plugin/src/main/resources/monitoring/watches/elasticsearch_cluster_status.json b/plugin/src/main/resources/monitoring/watches/elasticsearch_cluster_status.json index f8aebb8c1b1..d7b362ff8f7 100644 --- a/plugin/src/main/resources/monitoring/watches/elasticsearch_cluster_status.json +++ b/plugin/src/main/resources/monitoring/watches/elasticsearch_cluster_status.json @@ -97,20 +97,49 @@ } } } + }, + { + "kibana_settings": { + "search": { + "request": { + "search_type": "query_then_fetch", + "indices": [ + ".monitoring-kibana-6-*" + ], + "body": { + "size": 1, + "query": { + "bool": { + "filter": { + "term": { + "type": "kibana_settings" + } + } + } + }, + "sort": [ + { + "timestamp": { + "order": "desc" + } + } + ] + } + } + } + } } ] } }, "condition": { "script": { - "source": - "ctx.vars.fails_check = ctx.payload.check.hits.total != 0 && ctx.payload.check.hits.hits[0]._source.cluster_state.status != 'green';ctx.vars.not_resolved = ctx.payload.alert.hits.total == 1 && ctx.payload.alert.hits.hits[0]._source.resolved_timestamp == null;return ctx.vars.fails_check || ctx.vars.not_resolved" + "source": "ctx.vars.fails_check = ctx.payload.check.hits.total != 0 && ctx.payload.check.hits.hits[0]._source.cluster_state.status != 'green';ctx.vars.not_resolved = ctx.payload.alert.hits.total == 1 && ctx.payload.alert.hits.hits[0]._source.resolved_timestamp == null;return ctx.vars.fails_check || ctx.vars.not_resolved" } }, "transform": { "script": { - "source": - "def state = 'red';if (ctx.vars.fails_check){state = ctx.payload.check.hits.hits[0]._source.cluster_state.status;}if (ctx.vars.not_resolved){ctx.payload = ctx.payload.alert.hits.hits[0]._source;if (ctx.vars.fails_check == false) {ctx.payload.resolved_timestamp = ctx.execution_time;}} else {ctx.payload = ['timestamp': ctx.execution_time, 'metadata': ctx.metadata.xpack];}if (ctx.vars.fails_check) {ctx.payload.prefix = 'Elasticsearch cluster status is ' + state + '.';if (state == 'red') {ctx.payload.message = 'Allocate missing primary shards and replica shards.';ctx.payload.metadata.severity = 2100;} else {ctx.payload.message = 'Allocate missing replica shards.';ctx.payload.metadata.severity = 1100;}}ctx.payload.update_timestamp = ctx.execution_time;return ctx.payload;" + "source": "ctx.vars.email_recipient = (ctx.payload.kibana_settings.hits.total > 0) ? ctx.payload.kibana_settings.hits.hits[0]._source.kibana_settings.xpack.defaultAdminEmail : null;ctx.vars.is_new = ctx.vars.fails_check && !ctx.vars.not_resolved;ctx.vars.is_resolved = !ctx.vars.fails_check && ctx.vars.not_resolved;def state = ctx.payload.check.hits.hits[0]._source.cluster_state.status;if (ctx.vars.not_resolved){ctx.payload = ctx.payload.alert.hits.hits[0]._source;if (ctx.vars.fails_check == false) {ctx.payload.resolved_timestamp = ctx.execution_time;}} else {ctx.payload = ['timestamp': ctx.execution_time, 'metadata': ctx.metadata.xpack];}if (ctx.vars.fails_check) {ctx.payload.prefix = 'Elasticsearch cluster status is ' + state + '.';if (state == 'red') {ctx.payload.message = 'Allocate missing primary shards and replica shards.';ctx.payload.metadata.severity = 2100;} else {ctx.payload.message = 'Allocate missing replica shards.';ctx.payload.metadata.severity = 1100;}}ctx.vars.state = state.toUpperCase();ctx.payload.update_timestamp = ctx.execution_time;return ctx.payload;" } }, "actions": { @@ -120,6 +149,19 @@ "doc_type": "doc", "doc_id": "${monitoring.watch.unique_id}" } + }, + "send_email_to_admin": { + "condition": { + "script": "return ctx.vars.email_recipient != null && (ctx.vars.is_new || ctx.vars.is_resolved)" + }, + "email": { + "to": "X-Pack Admin <{{ctx.vars.email_recipient}}>", + "from": "X-Pack Admin <{{ctx.vars.email_recipient}}>", + "subject": "[{{#ctx.vars.is_new}}NEW{{/ctx.vars.is_new}}{{#ctx.vars.is_resolved}}RESOLVED{{/ctx.vars.is_resolved}}] {{ctx.metadata.name}} [{{ctx.vars.state}}]", + "body": { + "text": "{{#ctx.vars.is_resolved}}This cluster alert has been resolved: {{/ctx.vars.is_resolved}}{{ctx.payload.prefix}} {{ctx.payload.message}}" + } + } } } -} \ No newline at end of file +} diff --git a/plugin/src/main/resources/monitoring/watches/elasticsearch_version_mismatch.json b/plugin/src/main/resources/monitoring/watches/elasticsearch_version_mismatch.json index 3119ff46db7..0991999476c 100644 --- a/plugin/src/main/resources/monitoring/watches/elasticsearch_version_mismatch.json +++ b/plugin/src/main/resources/monitoring/watches/elasticsearch_version_mismatch.json @@ -93,6 +93,37 @@ } } } + }, + { + "kibana_settings": { + "search": { + "request": { + "search_type": "query_then_fetch", + "indices": [ + ".monitoring-kibana-6-*" + ], + "body": { + "size": 1, + "query": { + "bool": { + "filter": { + "term": { + "type": "kibana_settings" + } + } + } + }, + "sort": [ + { + "timestamp": { + "order": "desc" + } + } + ] + } + } + } + } } ] } @@ -104,7 +135,7 @@ }, "transform": { "script": { - "source": "def versionMessage = null;if (ctx.vars.fails_check) {def versions = new ArrayList(ctx.payload.check.hits.hits[0]._source.cluster_stats.nodes.versions);Collections.sort(versions);versionMessage = 'Versions: [' + String.join(', ', versions) + '].';}if (ctx.vars.not_resolved) {ctx.payload = ctx.payload.alert.hits.hits[0]._source;if (ctx.vars.fails_check) {ctx.payload.message = versionMessage;} else {ctx.payload.resolved_timestamp = ctx.execution_time;}} else {ctx.payload = [ 'timestamp': ctx.execution_time, 'prefix': 'This cluster is running with multiple versions of Elasticsearch.', 'message': versionMessage, 'metadata': ctx.metadata.xpack ];}ctx.payload.update_timestamp = ctx.execution_time;return ctx.payload;" + "source": "ctx.vars.email_recipient = (ctx.payload.kibana_settings.hits.total > 0) ? ctx.payload.kibana_settings.hits.hits[0]._source.kibana_settings.xpack.defaultAdminEmail : null;ctx.vars.is_new = ctx.vars.fails_check && !ctx.vars.not_resolved;ctx.vars.is_resolved = !ctx.vars.fails_check && ctx.vars.not_resolved;def versionMessage = null;if (ctx.vars.fails_check) {def versions = new ArrayList(ctx.payload.check.hits.hits[0]._source.cluster_stats.nodes.versions);Collections.sort(versions);versionMessage = 'Versions: [' + String.join(', ', versions) + '].';}if (ctx.vars.not_resolved) {ctx.payload = ctx.payload.alert.hits.hits[0]._source;if (ctx.vars.fails_check) {ctx.payload.message = versionMessage;} else {ctx.payload.resolved_timestamp = ctx.execution_time;}} else {ctx.payload = [ 'timestamp': ctx.execution_time, 'prefix': 'This cluster is running with multiple versions of Elasticsearch.', 'message': versionMessage, 'metadata': ctx.metadata.xpack ];}ctx.payload.update_timestamp = ctx.execution_time;return ctx.payload;" } }, "actions": { @@ -114,6 +145,19 @@ "doc_type": "doc", "doc_id": "${monitoring.watch.unique_id}" } + }, + "send_email_to_admin": { + "condition": { + "script": "return ctx.vars.email_recipient != null && (ctx.vars.is_new || ctx.vars.is_resolved)" + }, + "email": { + "to": "X-Pack Admin <{{ctx.vars.email_recipient}}>", + "from": "X-Pack Admin <{{ctx.vars.email_recipient}}>", + "subject": "[{{#ctx.vars.is_new}}NEW{{/ctx.vars.is_new}}{{#ctx.vars.is_resolved}}RESOLVED{{/ctx.vars.is_resolved}}] {{ctx.metadata.name}}", + "body": { + "text": "{{#ctx.vars.is_resolved}}This cluster alert has been resolved: {{/ctx.vars.is_resolved}}{{ctx.payload.prefix}} {{ctx.payload.message}}" + } + } } } -} \ No newline at end of file +} diff --git a/plugin/src/main/resources/monitoring/watches/kibana_version_mismatch.json b/plugin/src/main/resources/monitoring/watches/kibana_version_mismatch.json index dbaa489ddbe..4f0d036f538 100644 --- a/plugin/src/main/resources/monitoring/watches/kibana_version_mismatch.json +++ b/plugin/src/main/resources/monitoring/watches/kibana_version_mismatch.json @@ -120,6 +120,37 @@ } } } + }, + { + "kibana_settings": { + "search": { + "request": { + "search_type": "query_then_fetch", + "indices": [ + ".monitoring-kibana-6-*" + ], + "body": { + "size": 1, + "query": { + "bool": { + "filter": { + "term": { + "type": "kibana_settings" + } + } + } + }, + "sort": [ + { + "timestamp": { + "order": "desc" + } + } + ] + } + } + } + } } ] } @@ -131,7 +162,7 @@ }, "transform": { "script": { - "source": "def versionMessage = null;if (ctx.vars.fails_check) {versionMessage = 'Versions: [' + String.join(', ', ctx.vars.versions) + '].';}if (ctx.vars.not_resolved) {ctx.payload = ctx.payload.alert.hits.hits[0]._source;if (ctx.vars.fails_check) {ctx.payload.message = versionMessage;} else {ctx.payload.resolved_timestamp = ctx.execution_time;}} else {ctx.payload = [ 'timestamp': ctx.execution_time, 'prefix': 'This cluster is running with multiple versions of Kibana.', 'message': versionMessage, 'metadata': ctx.metadata.xpack ];}ctx.payload.update_timestamp = ctx.execution_time;return ctx.payload;" + "source": "ctx.vars.email_recipient = (ctx.payload.kibana_settings.hits.total > 0) ? ctx.payload.kibana_settings.hits.hits[0]._source.kibana_settings.xpack.defaultAdminEmail : null;ctx.vars.is_new = ctx.vars.fails_check && !ctx.vars.not_resolved;ctx.vars.is_resolved = !ctx.vars.fails_check && ctx.vars.not_resolved;def versionMessage = null;if (ctx.vars.fails_check) {versionMessage = 'Versions: [' + String.join(', ', ctx.vars.versions) + '].';}if (ctx.vars.not_resolved) {ctx.payload = ctx.payload.alert.hits.hits[0]._source;if (ctx.vars.fails_check) {ctx.payload.message = versionMessage;} else {ctx.payload.resolved_timestamp = ctx.execution_time;}} else {ctx.payload = [ 'timestamp': ctx.execution_time, 'prefix': 'This cluster is running with multiple versions of Kibana.', 'message': versionMessage, 'metadata': ctx.metadata.xpack ];}ctx.payload.update_timestamp = ctx.execution_time;return ctx.payload;" } }, "actions": { @@ -141,6 +172,19 @@ "doc_type": "doc", "doc_id": "${monitoring.watch.unique_id}" } + }, + "send_email_to_admin": { + "condition": { + "script": "return ctx.vars.email_recipient != null && (ctx.vars.is_new || ctx.vars.is_resolved)" + }, + "email": { + "to": "X-Pack Admin <{{ctx.vars.email_recipient}}>", + "from": "X-Pack Admin <{{ctx.vars.email_recipient}}>", + "subject": "[{{#ctx.vars.is_new}}NEW{{/ctx.vars.is_new}}{{#ctx.vars.is_resolved}}RESOLVED{{/ctx.vars.is_resolved}}] {{ctx.metadata.name}}", + "body": { + "text": "{{#ctx.vars.is_resolved}}This cluster alert has been resolved: {{/ctx.vars.is_resolved}}{{ctx.payload.prefix}} {{ctx.payload.message}}" + } + } } } -} \ No newline at end of file +} diff --git a/plugin/src/main/resources/monitoring/watches/logstash_version_mismatch.json b/plugin/src/main/resources/monitoring/watches/logstash_version_mismatch.json index 66be0d68511..cd1007339b0 100644 --- a/plugin/src/main/resources/monitoring/watches/logstash_version_mismatch.json +++ b/plugin/src/main/resources/monitoring/watches/logstash_version_mismatch.json @@ -120,6 +120,37 @@ } } } + }, + { + "kibana_settings": { + "search": { + "request": { + "search_type": "query_then_fetch", + "indices": [ + ".monitoring-kibana-6-*" + ], + "body": { + "size": 1, + "query": { + "bool": { + "filter": { + "term": { + "type": "kibana_settings" + } + } + } + }, + "sort": [ + { + "timestamp": { + "order": "desc" + } + } + ] + } + } + } + } } ] } @@ -131,7 +162,7 @@ }, "transform": { "script": { - "source": "def versionMessage = null;if (ctx.vars.fails_check) {versionMessage = 'Versions: [' + String.join(', ', ctx.vars.versions) + '].';}if (ctx.vars.not_resolved) {ctx.payload = ctx.payload.alert.hits.hits[0]._source;if (ctx.vars.fails_check) {ctx.payload.message = versionMessage;} else {ctx.payload.resolved_timestamp = ctx.execution_time;}} else {ctx.payload = [ 'timestamp': ctx.execution_time, 'prefix': 'This cluster is running with multiple versions of Logstash.', 'message': versionMessage, 'metadata': ctx.metadata.xpack ];}ctx.payload.update_timestamp = ctx.execution_time;return ctx.payload;" + "source": "ctx.vars.email_recipient = (ctx.payload.kibana_settings.hits.total > 0) ? ctx.payload.kibana_settings.hits.hits[0]._source.kibana_settings.xpack.defaultAdminEmail : null;ctx.vars.is_new = ctx.vars.fails_check && !ctx.vars.not_resolved;ctx.vars.is_resolved = !ctx.vars.fails_check && ctx.vars.not_resolved;def versionMessage = null;if (ctx.vars.fails_check) {versionMessage = 'Versions: [' + String.join(', ', ctx.vars.versions) + '].';}if (ctx.vars.not_resolved) {ctx.payload = ctx.payload.alert.hits.hits[0]._source;if (ctx.vars.fails_check) {ctx.payload.message = versionMessage;} else {ctx.payload.resolved_timestamp = ctx.execution_time;}} else {ctx.payload = [ 'timestamp': ctx.execution_time, 'prefix': 'This cluster is running with multiple versions of Logstash.', 'message': versionMessage, 'metadata': ctx.metadata.xpack ];}ctx.payload.update_timestamp = ctx.execution_time;return ctx.payload;" } }, "actions": { @@ -141,6 +172,19 @@ "doc_type": "doc", "doc_id": "${monitoring.watch.unique_id}" } + }, + "send_email_to_admin": { + "condition": { + "script": "return ctx.vars.email_recipient != null && (ctx.vars.is_new || ctx.vars.is_resolved)" + }, + "email": { + "to": "X-Pack Admin <{{ctx.vars.email_recipient}}>", + "from": "X-Pack Admin <{{ctx.vars.email_recipient}}>", + "subject": "[{{#ctx.vars.is_new}}NEW{{/ctx.vars.is_new}}{{#ctx.vars.is_resolved}}RESOLVED{{/ctx.vars.is_resolved}}] {{ctx.metadata.name}}", + "body": { + "text": "{{#ctx.vars.is_resolved}}This cluster alert has been resolved: {{/ctx.vars.is_resolved}}{{ctx.payload.prefix}} {{ctx.payload.message}}" + } + } } } -} \ No newline at end of file +} From 2aa53f253f9899b58fcb9e1d1eb2f6d9003b99a1 Mon Sep 17 00:00:00 2001 From: Lisa Cawley Date: Fri, 7 Jul 2017 10:11:06 -0700 Subject: [PATCH 2/5] [DOCS] Update doc links for graph API (elastic/x-pack-elasticsearch#1884) Original commit: elastic/x-pack-elasticsearch@9eebb6c9a678d31ee366cc5d5c3d200d4f172dbc --- .../test/resources/rest-api-spec/api/xpack.graph.explore.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/plugin/src/test/resources/rest-api-spec/api/xpack.graph.explore.json b/plugin/src/test/resources/rest-api-spec/api/xpack.graph.explore.json index c80cf8b2336..2879ce182cb 100644 --- a/plugin/src/test/resources/rest-api-spec/api/xpack.graph.explore.json +++ b/plugin/src/test/resources/rest-api-spec/api/xpack.graph.explore.json @@ -1,6 +1,6 @@ { "xpack.graph.explore": { - "documentation": "https://www.elastic.co/guide/en/x-pack/current/graph-api-explore.html", + "documentation": "https://www.elastic.co/guide/en/elasticsearch/reference/current/graph-explore-api.html", "methods": ["GET", "POST"], "url": { "path": "/{index}/_xpack/graph/_explore", From 2e58164b96b98c7eed33153d11132878325a0e15 Mon Sep 17 00:00:00 2001 From: Lisa Cawley Date: Fri, 7 Jul 2017 10:17:55 -0700 Subject: [PATCH 3/5] [DOCS] Add ML limitation (elastic/x-pack-elasticsearch#1916) Original commit: elastic/x-pack-elasticsearch@7730810f980a2e3dc6b0eeaf91487ac92410fe2a --- docs/en/ml/limitations.asciidoc | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/docs/en/ml/limitations.asciidoc b/docs/en/ml/limitations.asciidoc index 604a01240c7..12a4e52dfa3 100644 --- a/docs/en/ml/limitations.asciidoc +++ b/docs/en/ml/limitations.asciidoc @@ -101,12 +101,22 @@ analyzed. [float] === Time-based index patterns are not supported +//See x-pack-elasticsearch/#1910 It is not possible to create an {xpackml} analysis job that uses time-based -index patterns, for example `[logstash-]YYYY.MM.DD`. +index patterns, for example `[logstash-]YYYY.MM.DD`. This applies to the single metric or multi metric job creation wizards in {kib}. +[float] +=== Fields named "by", "count", or "over" cannot be used to split data +//See x-pack-elasticsearch/#858 + +You cannot use the following field names in the `by_field_name` or +`over_field_name` properties in a job: `by`; `count`; `over`. This limitation +also applies to those properties when you create advanced jobs in {kib}. + + [float] === Jobs created in {kib} use model plot config and pre-aggregated data //See x-pack-elasticsearch/#844 From 1af9f56834305f7190e94eb5a7ee89d679dc206a Mon Sep 17 00:00:00 2001 From: Nik Everett Date: Fri, 7 Jul 2017 11:36:56 -0400 Subject: [PATCH 4/5] Remove `IndexResponse.created` Core removed this in elastic/elasticsearch#25516. Original commit: elastic/x-pack-elasticsearch@b1e7040fa20439cd2fd2ae1f77ee9e7ac125dbab --- .../xpack/ml/job/persistence/JobResultsPersister.java | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/plugin/src/main/java/org/elasticsearch/xpack/ml/job/persistence/JobResultsPersister.java b/plugin/src/main/java/org/elasticsearch/xpack/ml/job/persistence/JobResultsPersister.java index 45cc6cb5b8a..e059c2debf4 100644 --- a/plugin/src/main/java/org/elasticsearch/xpack/ml/job/persistence/JobResultsPersister.java +++ b/plugin/src/main/java/org/elasticsearch/xpack/ml/job/persistence/JobResultsPersister.java @@ -8,6 +8,7 @@ package org.elasticsearch.xpack.ml.job.persistence; import org.apache.logging.log4j.message.ParameterizedMessage; import org.elasticsearch.action.ActionFuture; import org.elasticsearch.action.ActionListener; +import org.elasticsearch.action.DocWriteResponse.Result; import org.elasticsearch.action.admin.indices.refresh.RefreshRequest; import org.elasticsearch.action.bulk.BulkRequest; import org.elasticsearch.action.bulk.BulkResponse; @@ -328,7 +329,7 @@ public class JobResultsPersister extends AbstractComponent { } catch (IOException e) { logger.error(new ParameterizedMessage("[{}] Error writing [{}]", jobId, (id == null) ? "auto-generated ID" : id), e); IndexResponse.Builder notCreatedResponse = new IndexResponse.Builder(); - notCreatedResponse.setCreated(false); + notCreatedResponse.setResult(Result.NOOP); listener.onResponse(notCreatedResponse.build()); } } From 65a0d44b2f12116adac40c00baeb07723a8ccc7c Mon Sep 17 00:00:00 2001 From: Nik Everett Date: Fri, 9 Jun 2017 23:29:07 -0400 Subject: [PATCH 5/5] Handle "current" being dropped from the list of released versions Original commit: elastic/x-pack-elasticsearch@46668ce672b440c9aac7806a2acf32ed842922c8 --- ...ckIndicesBackwardsCompatibilityTestCase.java | 17 ++++------------- 1 file changed, 4 insertions(+), 13 deletions(-) diff --git a/plugin/src/test/java/org/elasticsearch/AbstractOldXPackIndicesBackwardsCompatibilityTestCase.java b/plugin/src/test/java/org/elasticsearch/AbstractOldXPackIndicesBackwardsCompatibilityTestCase.java index 9c56bc3e45a..4f7cc043422 100644 --- a/plugin/src/test/java/org/elasticsearch/AbstractOldXPackIndicesBackwardsCompatibilityTestCase.java +++ b/plugin/src/test/java/org/elasticsearch/AbstractOldXPackIndicesBackwardsCompatibilityTestCase.java @@ -100,11 +100,10 @@ public abstract class AbstractOldXPackIndicesBackwardsCompatibilityTestCase exte public void testAllVersionsTested() throws Exception { SortedSet expectedVersions = new TreeSet<>(); for (Version v : VersionUtils.allReleasedVersions()) { - if (false == shouldTestVersion(v)) continue; - if (v.before(Version.CURRENT.minimumIndexCompatibilityVersion())) continue; // we can only support one major version backward - if (v.equals(Version.CURRENT)) continue; // the current version is always compatible with itself - if (v.isBeta() == true || v.isAlpha() == true || v.isRC() == true) continue; // don't check alphas etc - expectedVersions.add("x-pack-" + v.toString() + ".zip"); + if (v.isRelease()) { + // no guarantees for prereleases + expectedVersions.add("x-pack-" + v.toString() + ".zip"); + } } expectedVersions.removeAll(dataFiles); if (expectedVersions.isEmpty() == false) { @@ -121,7 +120,6 @@ public abstract class AbstractOldXPackIndicesBackwardsCompatibilityTestCase exte Collections.shuffle(dataFiles, random()); for (String dataFile : dataFiles) { Version version = Version.fromString(dataFile.replace("x-pack-", "").replace(".zip", "")); - if (false == shouldTestVersion(version)) continue; long clusterStartTime = System.nanoTime(); setupCluster(dataFile); ensureYellow(); @@ -137,13 +135,6 @@ public abstract class AbstractOldXPackIndicesBackwardsCompatibilityTestCase exte } } - /** - * Should we test this version at all? Called before loading the data directory. Return false to skip it entirely. - */ - protected boolean shouldTestVersion(Version version) { - return true; - } - /** * Actually test this version. */