From f6027e9a6bca477770ecfe4d370cde4c96738218 Mon Sep 17 00:00:00 2001 From: Brian Murphy Date: Wed, 26 Nov 2014 13:32:47 +0000 Subject: [PATCH] Alert History : Make alert history index time based. This commit makes the alert history index a time based index. The alert history now is a timebased index prefixed with .alert_history_ with the time fomat YYYY-MM-dd. This commit makes the alert history index a time based index. Original commit: elastic/x-pack-elasticsearch@df6d6dee2965e51806ede28446aae999a7c45c26 --- .../java/org/elasticsearch/alerts/Alert.java | 19 ++--- .../elasticsearch/alerts/AlertManager.java | 9 ++- .../org/elasticsearch/alerts/AlertsStore.java | 8 +-- .../alerts/actions/AlertActionManager.java | 71 +++++++++++++------ .../actions/IndexAlertActionFactory.java | 2 +- .../actions/ack/TransportAckAlertAction.java | 10 ++- .../delete/TransportDeleteAlertAction.java | 10 ++- .../actions/put/TransportPutAlertAction.java | 10 ++- src/main/resources/alerthistory.json | 2 +- src/main/resources/alerts.json | 2 +- .../alerts/AbstractAlertingTests.java | 33 +++------ .../alerts/AlertSerializationTest.java | 2 +- .../alerts/AlertThrottleTests.java | 10 +-- .../elasticsearch/alerts/BootStrapTest.java | 11 ++- .../alerts/actions/AlertActionsTest.java | 4 +- 15 files changed, 123 insertions(+), 80 deletions(-) diff --git a/src/main/java/org/elasticsearch/alerts/Alert.java b/src/main/java/org/elasticsearch/alerts/Alert.java index 4f4d14b88cd..a3c28dc4ddf 100644 --- a/src/main/java/org/elasticsearch/alerts/Alert.java +++ b/src/main/java/org/elasticsearch/alerts/Alert.java @@ -24,7 +24,7 @@ public class Alert implements ToXContent { private AlertTrigger trigger; private List actions; private String schedule; - private DateTime lastActionFire; + private DateTime lastExecuteTime; private long version; private TimeValue throttlePeriod = new TimeValue(0); private DateTime timeLastActionExecuted = null; @@ -34,13 +34,14 @@ public class Alert implements ToXContent { actions = new ArrayList<>(); } - public Alert(String alertName, SearchRequest searchRequest, AlertTrigger trigger, List actions, String schedule, DateTime lastActionFire, long version, TimeValue throttlePeriod, AlertAckState ackState) { + + public Alert(String alertName, SearchRequest searchRequest, AlertTrigger trigger, List actions, String schedule, DateTime lastExecuteTime, long version, TimeValue throttlePeriod, AlertAckState ackState) { this.alertName = alertName; this.searchRequest = searchRequest; this.trigger = trigger; this.actions = actions; this.schedule = schedule; - this.lastActionFire = lastActionFire; + this.lastExecuteTime = lastExecuteTime; this.version = version; this.throttlePeriod = throttlePeriod; this.ackState = ackState; @@ -59,8 +60,8 @@ public class Alert implements ToXContent { builder.field(AlertsStore.LAST_ACTION_EXECUTED_FIELD.getPreferredName(), timeLastActionExecuted); } - if (lastActionFire != null) { - builder.field(AlertsStore.LAST_ACTION_FIRE.getPreferredName(), lastActionFire); + if (lastExecuteTime != null) { + builder.field(AlertsStore.LAST_ACTION_FIRE.getPreferredName(), lastExecuteTime); } if (actions != null && !actions.isEmpty()) { @@ -85,12 +86,12 @@ public class Alert implements ToXContent { /** * @return The last time this alert ran. */ - public DateTime lastActionFire() { - return lastActionFire; + public DateTime lastExecuteTime() { + return lastExecuteTime; } - public void lastActionFire(DateTime lastActionFire) { - this.lastActionFire = lastActionFire; + public void lastExecuteTime(DateTime lastActionFire) { + this.lastExecuteTime = lastActionFire; } /** diff --git a/src/main/java/org/elasticsearch/alerts/AlertManager.java b/src/main/java/org/elasticsearch/alerts/AlertManager.java index 088cc849f78..6cb354e4482 100644 --- a/src/main/java/org/elasticsearch/alerts/AlertManager.java +++ b/src/main/java/org/elasticsearch/alerts/AlertManager.java @@ -142,24 +142,23 @@ public class AlertManager extends AbstractComponent { } TriggerResult triggerResult = triggerManager.isTriggered(alert, entry.getScheduledTime(), entry.getFireTime()); entry.setSearchResponse(triggerResult.getResponse()); - if (triggerResult.isTriggered()) { entry.setTriggered(true); if (!isActionThrottled(alert)) { actionRegistry.doAction(alert, triggerResult); - alert.setTimeLastActionExecuted(entry.getFireTime()); + alert.setTimeLastActionExecuted(entry.getScheduledTime()); if (alert.getAckState() == AlertAckState.NOT_TRIGGERED) { alert.setAckState(AlertAckState.NEEDS_ACK); } } else { entry.setState(AlertActionState.THROTTLED); } - alert.lastActionFire(entry.getFireTime()); - alertsStore.updateAlert(alert); + } else if (alert.getAckState() == AlertAckState.ACKED) { alert.setAckState(AlertAckState.NOT_TRIGGERED); - alertsStore.updateAlert(alert); } + alert.lastExecuteTime(entry.getFireTime()); + alertsStore.updateAlert(alert); return triggerResult; } finally { alertLock.release(entry.getAlertName()); diff --git a/src/main/java/org/elasticsearch/alerts/AlertsStore.java b/src/main/java/org/elasticsearch/alerts/AlertsStore.java index a204adb09f5..7d9e5cf85b7 100644 --- a/src/main/java/org/elasticsearch/alerts/AlertsStore.java +++ b/src/main/java/org/elasticsearch/alerts/AlertsStore.java @@ -51,7 +51,7 @@ public class AlertsStore extends AbstractComponent { public static final ParseField SCHEDULE_FIELD = new ParseField("schedule"); public static final ParseField TRIGGER_FIELD = new ParseField("trigger"); public static final ParseField ACTION_FIELD = new ParseField("actions"); - public static final ParseField LAST_ACTION_FIRE = new ParseField("last_action_fire"); + public static final ParseField LAST_ACTION_FIRE = new ParseField("last_alert_executed"); public static final ParseField REQUEST_FIELD = new ParseField("request"); public static final ParseField THROTTLE_PERIOD_FIELD = new ParseField("throttle_period"); public static final ParseField LAST_ACTION_EXECUTED_FIELD = new ParseField("last_action_executed"); @@ -250,7 +250,7 @@ public class AlertsStore extends AbstractComponent { if (SCHEDULE_FIELD.match(currentFieldName)) { alert.schedule(parser.textOrNull()); } else if (LAST_ACTION_FIRE.match(currentFieldName)) { - alert.lastActionFire(DateTime.parse(parser.textOrNull())); + alert.lastExecuteTime(DateTime.parse(parser.textOrNull())); } else if (LAST_ACTION_EXECUTED_FIELD.match(currentFieldName)) { alert.setTimeLastActionExecuted(DateTime.parse(parser.textOrNull())); } else if (THROTTLE_PERIOD_FIELD.match(currentFieldName)) { @@ -268,8 +268,8 @@ public class AlertsStore extends AbstractComponent { throw new ElasticsearchException("Error during parsing alert", e); } - if (alert.lastActionFire() == null) { - alert.lastActionFire(new DateTime(0)); + if (alert.lastExecuteTime() == null) { + alert.lastExecuteTime(new DateTime(0)); } if (alert.schedule() == null) { diff --git a/src/main/java/org/elasticsearch/alerts/actions/AlertActionManager.java b/src/main/java/org/elasticsearch/alerts/actions/AlertActionManager.java index 9d7f9ecaa5b..fc734665daf 100644 --- a/src/main/java/org/elasticsearch/alerts/actions/AlertActionManager.java +++ b/src/main/java/org/elasticsearch/alerts/actions/AlertActionManager.java @@ -13,6 +13,7 @@ import org.elasticsearch.action.index.IndexRequest; import org.elasticsearch.action.index.IndexResponse; import org.elasticsearch.action.search.SearchResponse; import org.elasticsearch.action.search.SearchType; +import org.elasticsearch.action.support.IndicesOptions; import org.elasticsearch.alerts.*; import org.elasticsearch.alerts.plugin.AlertsPlugin; import org.elasticsearch.alerts.triggers.TriggerManager; @@ -21,9 +22,13 @@ import org.elasticsearch.client.Client; import org.elasticsearch.cluster.ClusterState; import org.elasticsearch.cluster.metadata.IndexMetaData; import org.elasticsearch.common.bytes.BytesReference; +import org.elasticsearch.common.collect.ImmutableOpenMap; import org.elasticsearch.common.component.AbstractComponent; +import org.elasticsearch.common.hppc.cursors.ObjectCursor; import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.joda.time.DateTime; +import org.elasticsearch.common.joda.time.format.DateTimeFormat; +import org.elasticsearch.common.joda.time.format.DateTimeFormatter; import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.unit.TimeValue; import org.elasticsearch.common.xcontent.XContentFactory; @@ -54,7 +59,8 @@ public class AlertActionManager extends AbstractComponent { public static final String ACTIONS_FIELD = "actions"; public static final String STATE = "state"; - public static final String ALERT_HISTORY_INDEX = ".alert_history"; + public static final String ALERT_HISTORY_INDEX_PREFIX = ".alert_history_"; + public static final DateTimeFormatter alertHistoryIndexTimeFormat = DateTimeFormat.forPattern("YYYY-MM-dd"); public static final String ALERT_HISTORY_TYPE = "alerthistory"; private static AlertActionEntry END_ENTRY = new AlertActionEntry(); @@ -98,29 +104,34 @@ public class AlertActionManager extends AbstractComponent { if (started.get()) { return true; } - - IndexMetaData indexMetaData = state.getMetaData().index(ALERT_HISTORY_INDEX); - if (indexMetaData != null) { - if (state.routingTable().index(ALERT_HISTORY_INDEX).allPrimaryShardsActive()) { - try { - loadQueue(); - } catch (Exception e) { - logger.error("Unable to load unfinished jobs into the job queue", e); - actionsToBeProcessed.clear(); - } - templateHelper.checkAndUploadIndexTemplate(state, "alerthistory"); - doStart(); - return true; - } else { - logger.info("Not all primary shards of the .alertshistory index are started"); - return false; - } - } else { + String[] indices = state.metaData().concreteIndices(IndicesOptions.lenientExpandOpen(), ALERT_HISTORY_INDEX_PREFIX + "*"); + if (indices.length == 0) { logger.info("No previous .alerthistory index, skip loading of alert actions"); templateHelper.checkAndUploadIndexTemplate(state, "alerthistory"); doStart(); return true; } + + + for (String index : indices) { + IndexMetaData indexMetaData = state.getMetaData().index(index); + if (indexMetaData != null) { + if (!state.routingTable().index(index).allPrimaryShardsActive()) { + logger.info("Not all primary shards of the [{}] index are started", index); + return false; + } + } + } + + try { + loadQueue(); + } catch (Exception e) { + logger.error("Unable to load unfinished jobs into the job queue", e); + actionsToBeProcessed.clear(); + } + templateHelper.checkAndUploadIndexTemplate(state, "alerthistory"); + doStart(); + return true; } public void stop() { @@ -135,6 +146,20 @@ public class AlertActionManager extends AbstractComponent { return started.get(); } + /** + * Calculates the correct alert history index name for a given time using alertHistoryIndexTimeFormat +<<<<<<< HEAD +======= + * @param time + * @return +>>>>>>> 7462f14f6d6a8c1529fc4f4203184f30c83057d7 + */ + public static String getAlertHistoryIndexNameForTime(DateTime time) { + StringBuffer sb = new StringBuffer(ALERT_HISTORY_INDEX_PREFIX); + alertHistoryIndexTimeFormat.printTo(sb, time); + return sb.toString(); + } + private void doStart() { logger.info("Starting job queue"); if (started.compareAndSet(false, true)) { @@ -143,9 +168,9 @@ public class AlertActionManager extends AbstractComponent { } public void loadQueue() { - client.admin().indices().refresh(new RefreshRequest(ALERT_HISTORY_INDEX)).actionGet(); + client.admin().indices().refresh(new RefreshRequest(ALERT_HISTORY_INDEX_PREFIX + "*")).actionGet(); - SearchResponse response = client.prepareSearch(ALERT_HISTORY_INDEX) + SearchResponse response = client.prepareSearch(ALERT_HISTORY_INDEX_PREFIX + "*") .setQuery(QueryBuilders.termQuery(STATE, AlertActionState.SEARCH_NEEDED.toString())) .setSearchType(SearchType.SCAN) .setScroll(scrollTimeout) @@ -241,7 +266,7 @@ public class AlertActionManager extends AbstractComponent { public void addAlertAction(Alert alert, DateTime scheduledFireTime, DateTime fireTime) throws IOException { ensureStarted(); AlertActionEntry entry = new AlertActionEntry(alert, scheduledFireTime, fireTime, AlertActionState.SEARCH_NEEDED); - IndexResponse response = client.prepareIndex(ALERT_HISTORY_INDEX, ALERT_HISTORY_TYPE, entry.getId()) + IndexResponse response = client.prepareIndex(getAlertHistoryIndexNameForTime(scheduledFireTime), ALERT_HISTORY_TYPE, entry.getId()) .setSource(XContentFactory.jsonBuilder().value(entry)) .setOpType(IndexRequest.OpType.CREATE) .get(); @@ -264,7 +289,7 @@ public class AlertActionManager extends AbstractComponent { private void updateHistoryEntry(AlertActionEntry entry) throws IOException { ensureStarted(); - IndexResponse response = client.prepareIndex(ALERT_HISTORY_INDEX, ALERT_HISTORY_TYPE, entry.getId()) + IndexResponse response = client.prepareIndex(getAlertHistoryIndexNameForTime(entry.getScheduledTime()), ALERT_HISTORY_TYPE, entry.getId()) .setSource(XContentFactory.jsonBuilder().value(entry)) .get(); entry.setVersion(response.getVersion()); diff --git a/src/main/java/org/elasticsearch/alerts/actions/IndexAlertActionFactory.java b/src/main/java/org/elasticsearch/alerts/actions/IndexAlertActionFactory.java index 3562fac5e14..40633b638ce 100644 --- a/src/main/java/org/elasticsearch/alerts/actions/IndexAlertActionFactory.java +++ b/src/main/java/org/elasticsearch/alerts/actions/IndexAlertActionFactory.java @@ -71,7 +71,7 @@ public class IndexAlertActionFactory implements AlertActionFactory { XContentBuilder resultBuilder = XContentFactory.jsonBuilder().prettyPrint(); resultBuilder.startObject(); resultBuilder.field("response", result.getResponse()); - resultBuilder.field("timestamp", alert.lastActionFire()); ///@TODO FIXME the firetime should be in the result ? + resultBuilder.field("timestamp", alert.lastExecuteTime()); ///@TODO FIXME the firetime should be in the result ? resultBuilder.endObject(); indexRequest.source(resultBuilder); } catch (IOException ie) { diff --git a/src/main/java/org/elasticsearch/alerts/transport/actions/ack/TransportAckAlertAction.java b/src/main/java/org/elasticsearch/alerts/transport/actions/ack/TransportAckAlertAction.java index 0b21e97251f..8959391d89b 100644 --- a/src/main/java/org/elasticsearch/alerts/transport/actions/ack/TransportAckAlertAction.java +++ b/src/main/java/org/elasticsearch/alerts/transport/actions/ack/TransportAckAlertAction.java @@ -8,6 +8,7 @@ package org.elasticsearch.alerts.transport.actions.ack; import org.elasticsearch.ElasticsearchException; import org.elasticsearch.action.ActionListener; import org.elasticsearch.action.support.ActionFilters; +import org.elasticsearch.action.support.IndicesOptions; import org.elasticsearch.action.support.master.TransportMasterNodeOperationAction; import org.elasticsearch.alerts.AlertManager; import org.elasticsearch.alerts.AlertsStore; @@ -21,6 +22,10 @@ import org.elasticsearch.common.settings.Settings; import org.elasticsearch.threadpool.ThreadPool; import org.elasticsearch.transport.TransportService; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + /** * Performs the delete operation. */ @@ -62,7 +67,10 @@ public class TransportAckAlertAction extends TransportMasterNodeOperationAction< @Override protected ClusterBlockException checkBlock(AckAlertRequest request, ClusterState state) { - return state.blocks().indicesBlockedException(ClusterBlockLevel.WRITE, new String[]{AlertsStore.ALERT_INDEX, AlertActionManager.ALERT_HISTORY_INDEX}); + String[] indices = state.metaData().concreteIndices(IndicesOptions.lenientExpandOpen(), AlertActionManager.ALERT_HISTORY_INDEX_PREFIX + "*"); + List indicesToCheck = new ArrayList<>(Arrays.asList(indices)); + indicesToCheck.add(AlertsStore.ALERT_INDEX); + return state.blocks().indicesBlockedException(ClusterBlockLevel.WRITE, indicesToCheck.toArray(new String[indicesToCheck.size()])); } diff --git a/src/main/java/org/elasticsearch/alerts/transport/actions/delete/TransportDeleteAlertAction.java b/src/main/java/org/elasticsearch/alerts/transport/actions/delete/TransportDeleteAlertAction.java index 2e81e4665fa..0aa5a24ffd1 100644 --- a/src/main/java/org/elasticsearch/alerts/transport/actions/delete/TransportDeleteAlertAction.java +++ b/src/main/java/org/elasticsearch/alerts/transport/actions/delete/TransportDeleteAlertAction.java @@ -8,6 +8,7 @@ package org.elasticsearch.alerts.transport.actions.delete; import org.elasticsearch.ElasticsearchException; import org.elasticsearch.action.ActionListener; import org.elasticsearch.action.support.ActionFilters; +import org.elasticsearch.action.support.IndicesOptions; import org.elasticsearch.action.support.master.TransportMasterNodeOperationAction; import org.elasticsearch.alerts.AlertManager; import org.elasticsearch.alerts.AlertsStore; @@ -21,6 +22,10 @@ import org.elasticsearch.common.settings.Settings; import org.elasticsearch.threadpool.ThreadPool; import org.elasticsearch.transport.TransportService; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + /** * Performs the delete operation. */ @@ -62,7 +67,10 @@ public class TransportDeleteAlertAction extends TransportMasterNodeOperationActi @Override protected ClusterBlockException checkBlock(DeleteAlertRequest request, ClusterState state) { - return state.blocks().indicesBlockedException(ClusterBlockLevel.WRITE, new String[]{AlertsStore.ALERT_INDEX, AlertActionManager.ALERT_HISTORY_INDEX}); + String[] indices = state.metaData().concreteIndices(IndicesOptions.lenientExpandOpen(), AlertActionManager.ALERT_HISTORY_INDEX_PREFIX + "*"); + List indicesToCheck = new ArrayList<>(Arrays.asList(indices)); + indicesToCheck.add(AlertsStore.ALERT_INDEX); + return state.blocks().indicesBlockedException(ClusterBlockLevel.WRITE, indicesToCheck.toArray(new String[indicesToCheck.size()])); } diff --git a/src/main/java/org/elasticsearch/alerts/transport/actions/put/TransportPutAlertAction.java b/src/main/java/org/elasticsearch/alerts/transport/actions/put/TransportPutAlertAction.java index 262408f3097..ad0a92cc150 100644 --- a/src/main/java/org/elasticsearch/alerts/transport/actions/put/TransportPutAlertAction.java +++ b/src/main/java/org/elasticsearch/alerts/transport/actions/put/TransportPutAlertAction.java @@ -9,6 +9,7 @@ import org.elasticsearch.ElasticsearchException; import org.elasticsearch.action.ActionListener; import org.elasticsearch.action.index.IndexResponse; import org.elasticsearch.action.support.ActionFilters; +import org.elasticsearch.action.support.IndicesOptions; import org.elasticsearch.action.support.master.TransportMasterNodeOperationAction; import org.elasticsearch.alerts.AlertManager; import org.elasticsearch.alerts.AlertsStore; @@ -22,6 +23,10 @@ import org.elasticsearch.common.settings.Settings; import org.elasticsearch.threadpool.ThreadPool; import org.elasticsearch.transport.TransportService; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + /** */ public class TransportPutAlertAction extends TransportMasterNodeOperationAction { @@ -63,7 +68,10 @@ public class TransportPutAlertAction extends TransportMasterNodeOperationAction< @Override protected ClusterBlockException checkBlock(PutAlertRequest request, ClusterState state) { request.beforeLocalFork(); // This is the best place to make the alert source safe - return state.blocks().indicesBlockedException(ClusterBlockLevel.WRITE, new String[]{AlertsStore.ALERT_INDEX, AlertActionManager.ALERT_HISTORY_INDEX}); + String[] indices = state.metaData().concreteIndices(IndicesOptions.lenientExpandOpen(), AlertActionManager.ALERT_HISTORY_INDEX_PREFIX + "*"); + List indicesToCheck = new ArrayList<>(Arrays.asList(indices)); + indicesToCheck.add(AlertsStore.ALERT_INDEX); + return state.blocks().indicesBlockedException(ClusterBlockLevel.WRITE, indicesToCheck.toArray(new String[indicesToCheck.size()])); } diff --git a/src/main/resources/alerthistory.json b/src/main/resources/alerthistory.json index c050777b19c..60f16eea76f 100644 --- a/src/main/resources/alerthistory.json +++ b/src/main/resources/alerthistory.json @@ -1,5 +1,5 @@ { - "template": ".alert_history", + "template": ".alert_history*", "order": 0, "settings": { "number_of_shards": 1, diff --git a/src/main/resources/alerts.json b/src/main/resources/alerts.json index 0c6f157bbb8..d095a0ad208 100644 --- a/src/main/resources/alerts.json +++ b/src/main/resources/alerts.json @@ -16,7 +16,7 @@ "last_fire_time": { "type": "date" }, - "last_action_fire": { + "last_alert_executed": { "type": "date" }, "last_action_executed": { diff --git a/src/test/java/org/elasticsearch/alerts/AbstractAlertingTests.java b/src/test/java/org/elasticsearch/alerts/AbstractAlertingTests.java index d8eb1c491d0..7d43244091a 100644 --- a/src/test/java/org/elasticsearch/alerts/AbstractAlertingTests.java +++ b/src/test/java/org/elasticsearch/alerts/AbstractAlertingTests.java @@ -5,7 +5,6 @@ */ package org.elasticsearch.alerts; -import org.elasticsearch.action.admin.indices.exists.indices.IndicesExistsResponse; import org.elasticsearch.action.search.SearchRequest; import org.elasticsearch.action.search.SearchResponse; import org.elasticsearch.action.support.IndicesOptions; @@ -17,7 +16,6 @@ import org.elasticsearch.alerts.transport.actions.stats.AlertsStatsResponse; import org.elasticsearch.client.Client; import org.elasticsearch.cluster.ClusterState; import org.elasticsearch.cluster.metadata.IndexTemplateMetaData; -import org.elasticsearch.cluster.routing.IndexRoutingTable; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.hppc.cursors.ObjectObjectCursor; import org.elasticsearch.common.settings.ImmutableSettings; @@ -38,8 +36,10 @@ import java.util.*; import static org.elasticsearch.common.xcontent.XContentFactory.jsonBuilder; import static org.elasticsearch.index.query.QueryBuilders.boolQuery; import static org.elasticsearch.index.query.QueryBuilders.matchQuery; -import static org.hamcrest.Matchers.*; +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.greaterThanOrEqualTo; import static org.hamcrest.core.Is.is; +import static org.hamcrest.core.IsNot.not; /** */ @@ -83,6 +83,10 @@ public abstract class AbstractAlertingTests extends ElasticsearchIntegrationTest // Clear all internal alerting state for the next test method: logger.info("[{}#{}]: clearing alerts", getTestClass().getSimpleName(), getTestName()); stopAlerting(); + client().admin().indices().prepareDelete(AlertsStore.ALERT_INDEX, AlertActionManager.ALERT_HISTORY_INDEX_PREFIX + "*") + .setIndicesOptions(IndicesOptions.lenientExpandOpen()) + .get(); + startAlerting(); } protected BytesReference createAlertSource(String cron, SearchRequest request, String scriptTrigger) throws IOException { @@ -118,6 +122,7 @@ public abstract class AbstractAlertingTests extends ElasticsearchIntegrationTest return internalTestCluster().getInstance(AlertsClient.class); } + protected void assertAlertTriggered(final String alertName, final long minimumExpectedAlertActionsWithActionPerformed) throws Exception { assertAlertTriggered(alertName, minimumExpectedAlertActionsWithActionPerformed, true); } @@ -126,15 +131,7 @@ public abstract class AbstractAlertingTests extends ElasticsearchIntegrationTest assertBusy(new Runnable() { @Override public void run() { - // The alerthistory index gets created in the background when the first alert fires, so we to check first is this index is created and shards are started - IndicesExistsResponse indicesExistsResponse = client().admin().indices().prepareExists(AlertActionManager.ALERT_HISTORY_INDEX).get(); - assertThat(indicesExistsResponse.isExists(), is(true)); - ClusterState state = client().admin().cluster().prepareState().get().getState(); - IndexRoutingTable routingTable = state.getRoutingTable().index(AlertActionManager.ALERT_HISTORY_INDEX); - assertThat(routingTable, notNullValue()); - assertThat(routingTable.allPrimaryShardsActive(), is(true)); - - SearchResponse searchResponse = client().prepareSearch(AlertActionManager.ALERT_HISTORY_INDEX) + SearchResponse searchResponse = client().prepareSearch(AlertActionManager.ALERT_HISTORY_INDEX_PREFIX + "*") .setIndicesOptions(IndicesOptions.lenientExpandOpen()) .setQuery(boolQuery().must(matchQuery("alert_name", alertName)).must(matchQuery("state", AlertActionState.ACTION_PERFORMED.toString()))) .get(); @@ -147,7 +144,7 @@ public abstract class AbstractAlertingTests extends ElasticsearchIntegrationTest } protected long findNumberOfPerformedActions(String alertName) { - SearchResponse searchResponse = client().prepareSearch(AlertActionManager.ALERT_HISTORY_INDEX) + SearchResponse searchResponse = client().prepareSearch(AlertActionManager.ALERT_HISTORY_INDEX_PREFIX + "*") .setIndicesOptions(IndicesOptions.lenientExpandOpen()) .setQuery(boolQuery().must(matchQuery("alert_name", alertName)).must(matchQuery("state", AlertActionState.ACTION_PERFORMED.toString()))) .get(); @@ -158,15 +155,7 @@ public abstract class AbstractAlertingTests extends ElasticsearchIntegrationTest assertBusy(new Runnable() { @Override public void run() { - // The alerthistory index gets created in the background when the first alert fires, so we to check first is this index is created and shards are started - IndicesExistsResponse indicesExistsResponse = client().admin().indices().prepareExists(AlertActionManager.ALERT_HISTORY_INDEX).get(); - assertThat(indicesExistsResponse.isExists(), is(true)); - ClusterState state = client().admin().cluster().prepareState().get().getState(); - IndexRoutingTable routingTable = state.getRoutingTable().index(AlertActionManager.ALERT_HISTORY_INDEX); - assertThat(routingTable, notNullValue()); - assertThat(routingTable.allPrimaryShardsActive(), is(true)); - - SearchResponse searchResponse = client().prepareSearch(AlertActionManager.ALERT_HISTORY_INDEX) + SearchResponse searchResponse = client().prepareSearch(AlertActionManager.ALERT_HISTORY_INDEX_PREFIX + "*") .setIndicesOptions(IndicesOptions.lenientExpandOpen()) .setQuery(boolQuery().must(matchQuery("alert_name", alertName)).must(matchQuery("state", AlertActionState.NO_ACTION_NEEDED.toString()))) .get(); diff --git a/src/test/java/org/elasticsearch/alerts/AlertSerializationTest.java b/src/test/java/org/elasticsearch/alerts/AlertSerializationTest.java index 460bb15d633..28c674c42d7 100644 --- a/src/test/java/org/elasticsearch/alerts/AlertSerializationTest.java +++ b/src/test/java/org/elasticsearch/alerts/AlertSerializationTest.java @@ -50,7 +50,7 @@ public class AlertSerializationTest extends ElasticsearchIntegrationTest { Alert parsedAlert = alertsStore.parseAlert("test-serialization", jsonBuilder.bytes()); assertEquals(parsedAlert.version(), alert.version()); assertEquals(parsedAlert.actions(), alert.actions()); - assertEquals(parsedAlert.lastActionFire().getMillis(), alert.lastActionFire().getMillis()); + assertEquals(parsedAlert.lastExecuteTime().getMillis(), alert.lastExecuteTime().getMillis()); assertEquals(parsedAlert.schedule(), alert.schedule()); assertEquals(parsedAlert.getSearchRequest().source(), alert.getSearchRequest().source()); assertEquals(parsedAlert.trigger(), alert.trigger()); diff --git a/src/test/java/org/elasticsearch/alerts/AlertThrottleTests.java b/src/test/java/org/elasticsearch/alerts/AlertThrottleTests.java index 8507c016d2c..0a2e1447f86 100644 --- a/src/test/java/org/elasticsearch/alerts/AlertThrottleTests.java +++ b/src/test/java/org/elasticsearch/alerts/AlertThrottleTests.java @@ -56,8 +56,8 @@ public class AlertThrottleTests extends AbstractAlertingTests { alert.trigger(new ScriptedTrigger("hits.total > 0", ScriptService.ScriptType.INLINE, "groovy")); alert.actions().add(new IndexAlertAction("action-index", "action-type")); alert.schedule( "0/5 * * * * ? *"); - alert.lastActionFire(new DateTime()); + alert.lastExecuteTime(new DateTime()); XContentBuilder jsonBuilder = XContentFactory.jsonBuilder(); alert.toXContent(jsonBuilder, ToXContent.EMPTY_PARAMS); @@ -105,7 +105,7 @@ public class AlertThrottleTests extends AbstractAlertingTests { assertThat(parsedAlert.getAckState(), equalTo(AlertAckState.NOT_TRIGGERED)); CountResponse countOfThrottledActions = client() - .prepareCount(AlertActionManager.ALERT_HISTORY_INDEX) + .prepareCount(AlertActionManager.ALERT_HISTORY_INDEX_PREFIX + "*") .setQuery(QueryBuilders.matchQuery(AlertActionManager.STATE, AlertActionState.THROTTLED.toString())) .get(); assertThat(countOfThrottledActions.getCount(), greaterThan(0L)); @@ -128,7 +128,7 @@ public class AlertThrottleTests extends AbstractAlertingTests { alert.trigger(new ScriptedTrigger("hits.total > 0", ScriptService.ScriptType.INLINE, "groovy")); alert.actions().add(new IndexAlertAction("action-index", "action-type")); alert.schedule("0/5 * * * * ? *"); - alert.lastActionFire(new DateTime()); + alert.lastExecuteTime(new DateTime()); alert.setThrottlePeriod(new TimeValue(10, TimeUnit.SECONDS)); XContentBuilder jsonBuilder = XContentFactory.jsonBuilder(); @@ -149,7 +149,7 @@ public class AlertThrottleTests extends AbstractAlertingTests { .get(); if (countResponse.getCount() != 1){ - SearchResponse actionResponse = client().prepareSearch(AlertActionManager.ALERT_HISTORY_INDEX) + SearchResponse actionResponse = client().prepareSearch(AlertActionManager.ALERT_HISTORY_INDEX_PREFIX + "*") .setQuery(matchAllQuery()) .get(); for (SearchHit hit : actionResponse.getHits()) { @@ -176,7 +176,7 @@ public class AlertThrottleTests extends AbstractAlertingTests { }); CountResponse countOfThrottledActions = client() - .prepareCount(AlertActionManager.ALERT_HISTORY_INDEX) + .prepareCount(AlertActionManager.ALERT_HISTORY_INDEX_PREFIX + "*") .setQuery(QueryBuilders.matchQuery(AlertActionManager.STATE, AlertActionState.THROTTLED.toString())) .get(); assertThat(countOfThrottledActions.getCount(), greaterThan(0L)); diff --git a/src/test/java/org/elasticsearch/alerts/BootStrapTest.java b/src/test/java/org/elasticsearch/alerts/BootStrapTest.java index b209a76efaf..17b14567f5b 100644 --- a/src/test/java/org/elasticsearch/alerts/BootStrapTest.java +++ b/src/test/java/org/elasticsearch/alerts/BootStrapTest.java @@ -73,12 +73,19 @@ public class BootStrapTest extends AbstractAlertingTests { new TimeValue(0), AlertAckState.NOT_ACKABLE); - AlertActionEntry entry = new AlertActionEntry(alert, new DateTime(), new DateTime(), AlertActionState.SEARCH_NEEDED); - IndexResponse indexResponse = client().prepareIndex(AlertActionManager.ALERT_HISTORY_INDEX, AlertActionManager.ALERT_HISTORY_TYPE, entry.getId()) + DateTime scheduledFireTime = new DateTime(); + AlertActionEntry entry = new AlertActionEntry(alert, scheduledFireTime, scheduledFireTime, AlertActionState.SEARCH_NEEDED); + String actionHistoryIndex = AlertActionManager.getAlertHistoryIndexNameForTime(scheduledFireTime); + + createIndex(actionHistoryIndex); + ensureGreen(actionHistoryIndex); + + IndexResponse indexResponse = client().prepareIndex(actionHistoryIndex, AlertActionManager.ALERT_HISTORY_TYPE, entry.getId()) .setConsistencyLevel(WriteConsistencyLevel.ALL) .setSource(XContentFactory.jsonBuilder().value(entry)) .get(); assertTrue(indexResponse.isCreated()); + client().admin().indices().prepareRefresh(actionHistoryIndex).get(); stopAlerting(); startAlerting(); diff --git a/src/test/java/org/elasticsearch/alerts/actions/AlertActionsTest.java b/src/test/java/org/elasticsearch/alerts/actions/AlertActionsTest.java index 5501fe20cbd..b772d2aa48c 100644 --- a/src/test/java/org/elasticsearch/alerts/actions/AlertActionsTest.java +++ b/src/test/java/org/elasticsearch/alerts/actions/AlertActionsTest.java @@ -102,10 +102,8 @@ public class AlertActionsTest extends AbstractAlertingTests { @Test public void testAlertActions() throws Exception { createIndex("my-index"); - createIndex(AlertsStore.ALERT_INDEX); - createIndex(AlertActionManager.ALERT_HISTORY_INDEX); - ensureGreen("my-index", AlertsStore.ALERT_INDEX, AlertActionManager.ALERT_HISTORY_INDEX); + ensureGreen("my-index"); client().preparePutIndexedScript() .setScriptLang("mustache")