diff --git a/src/main/java/org/opensearch/securityanalytics/threatIntel/service/SATIFSourceConfigManagementService.java b/src/main/java/org/opensearch/securityanalytics/threatIntel/service/SATIFSourceConfigManagementService.java index f7869c1df..e390abe47 100644 --- a/src/main/java/org/opensearch/securityanalytics/threatIntel/service/SATIFSourceConfigManagementService.java +++ b/src/main/java/org/opensearch/securityanalytics/threatIntel/service/SATIFSourceConfigManagementService.java @@ -27,7 +27,6 @@ import org.opensearch.rest.RestRequest; import org.opensearch.search.SearchHit; import org.opensearch.search.builder.SearchSourceBuilder; -import org.opensearch.securityanalytics.commons.model.IOCType; import org.opensearch.securityanalytics.model.STIX2IOC; import org.opensearch.securityanalytics.model.STIX2IOCDto; import org.opensearch.securityanalytics.services.STIX2IOCFetchService; @@ -162,16 +161,17 @@ public void createIocAndTIFSourceConfig( )); }, e -> { - log.error("Failed to download and save IOCs for threat intel source config [{}]", indexSaTifSourceConfigResponse.getId()); - saTifSourceConfigService.deleteTIFSourceConfig(indexSaTifSourceConfigResponse, ActionListener.wrap( + log.error("Failed to download and save IOCs for threat intel source config [{}]", indexSaTifSourceConfigResponse.getId(), e); + // set isDeleted as true because we want to delete failed source configs regardless if threat intel monitor exists + deleteAllIocsAndSourceConfig(ActionListener.wrap( deleteResponse -> { log.debug("Successfully deleted threat intel source config [{}]", indexSaTifSourceConfigResponse.getId()); listener.onFailure(e); }, ex -> { - log.error("Failed to delete threat intel source config [{}]", indexSaTifSourceConfigResponse.getId()); + log.error("Failed to delete threat intel source config [{}]", indexSaTifSourceConfigResponse.getId(), ex); listener.onFailure(ex); } - )); + ), indexSaTifSourceConfigResponse, true); }) ); }, e -> { @@ -497,7 +497,7 @@ public void deleteTIFSourceConfig( // Check if all threat intel monitors are deleted saTifSourceConfigService.checkAndEnsureThreatIntelMonitorsDeleted(ActionListener.wrap( isDeleted -> { - deleteAllIocsAndSourceConfig(saTifSourceConfigId, listener, saTifSourceConfig, isDeleted); + deleteAllIocsAndSourceConfig(listener, saTifSourceConfig, isDeleted); }, e -> { log.error("Failed to check if all threat intel monitors are deleted or if multiple threat intel source configs exist"); listener.onFailure(e); @@ -654,11 +654,11 @@ private Integer numOfIndicesToDelete(Integer totalNumIndices, Integer totalNumIn return 0; } - private void deleteAllIocsAndSourceConfig(String saTifSourceConfigId, ActionListener listener, SATIFSourceConfig saTifSourceConfig, Boolean isDeleted) { + private void deleteAllIocsAndSourceConfig(ActionListener listener, SATIFSourceConfig saTifSourceConfig, Boolean isDeleted) { if (isDeleted == false) { listener.onFailure(new IllegalArgumentException("All threat intel monitors need to be deleted before deleting last threat intel source config")); } else { - log.debug("All threat intel monitors are deleted or multiple threat intel source configs exist, can delete threat intel source config [{}]", saTifSourceConfigId); + log.debug("All threat intel monitors are deleted or multiple threat intel source configs exist, can delete threat intel source config [{}]", saTifSourceConfig.getId()); markSourceConfigAsAction( saTifSourceConfig, TIFJobState.DELETING, @@ -672,16 +672,28 @@ private void deleteAllIocsAndSourceConfig(String saTifSourceConfigId, ActionList } saTifSourceConfigService.getClusterState(ActionListener.wrap( clusterStateResponse -> { - Set concreteIndices = SATIFSourceConfigService.getConcreteIndices(clusterStateResponse); + Set concreteIndices; + if (false == iocIndexPatterns.isEmpty()) { + concreteIndices = SATIFSourceConfigService.getConcreteIndices(clusterStateResponse); + } else { + concreteIndices = new HashSet<>(); + } saTifSourceConfigService.deleteAllIocIndices(concreteIndices, false, ActionListener.wrap( r -> { log.debug("Successfully deleted all ioc indices"); - saTifSourceConfigService.deleteTIFSourceConfig(updateSaTifSourceConfigResponse, ActionListener.wrap( - deleteResponse -> { - log.debug("Successfully deleted threat intel source config [{}]", updateSaTifSourceConfigResponse.getId()); - listener.onResponse(deleteResponse); + saTifSourceConfigService.deleteJobSchedulerLockIfNeeded(updateSaTifSourceConfigResponse, ActionListener.wrap( + deleteLockResponse -> { + saTifSourceConfigService.deleteTIFSourceConfig(updateSaTifSourceConfigResponse, ActionListener.wrap( + deleteResponse -> { + log.debug("Successfully deleted threat intel source config [{}]", updateSaTifSourceConfigResponse.getId()); + listener.onResponse(deleteResponse); + }, e -> { + log.error("Failed to delete threat intel source config [{}]", saTifSourceConfig.getId()); + listener.onFailure(e); + } + )); }, e -> { - log.error("Failed to delete threat intel source config [{}]", saTifSourceConfigId); + log.error("Failed to delete threat intel job scheduler lock [{}]", saTifSourceConfig.getId()); listener.onFailure(e); } )); diff --git a/src/main/java/org/opensearch/securityanalytics/threatIntel/service/SATIFSourceConfigService.java b/src/main/java/org/opensearch/securityanalytics/threatIntel/service/SATIFSourceConfigService.java index bf5722a0f..beeffca93 100644 --- a/src/main/java/org/opensearch/securityanalytics/threatIntel/service/SATIFSourceConfigService.java +++ b/src/main/java/org/opensearch/securityanalytics/threatIntel/service/SATIFSourceConfigService.java @@ -50,6 +50,7 @@ import org.opensearch.securityanalytics.commons.model.IOCType; import org.opensearch.securityanalytics.threatIntel.action.monitor.SearchThreatIntelMonitorAction; import org.opensearch.securityanalytics.threatIntel.action.monitor.request.SearchThreatIntelMonitorRequest; +import org.opensearch.securityanalytics.threatIntel.common.SourceConfigType; import org.opensearch.securityanalytics.threatIntel.common.StashedThreadContext; import org.opensearch.securityanalytics.threatIntel.common.TIFLockService; import org.opensearch.securityanalytics.threatIntel.model.DefaultIocStoreConfig; @@ -72,6 +73,7 @@ import java.util.Set; import java.util.stream.Collectors; +import static org.opensearch.jobscheduler.spi.utils.LockService.LOCK_INDEX_NAME; import static org.opensearch.securityanalytics.settings.SecurityAnalyticsSettings.INDEX_TIMEOUT; import static org.opensearch.securityanalytics.threatIntel.common.TIFJobState.AVAILABLE; import static org.opensearch.securityanalytics.threatIntel.common.TIFJobState.REFRESHING; @@ -362,7 +364,7 @@ public void deleteTIFSourceConfig( client.delete(request, ActionListener.wrap( deleteResponse -> { if (deleteResponse.status().equals(RestStatus.OK)) { - log.debug("Deleted threat intel source config [{}] successfully", saTifSourceConfig.getId()); + log.info("Deleted threat intel source config [{}] successfully", saTifSourceConfig.getId()); actionListener.onResponse(deleteResponse); } else if (deleteResponse.status().equals(RestStatus.NOT_FOUND)) { actionListener.onFailure(SecurityAnalyticsException.wrap(new OpenSearchStatusException(String.format(Locale.getDefault(), "Threat intel source config with id [{%s}] not found", saTifSourceConfig.getId()), RestStatus.NOT_FOUND))); @@ -376,6 +378,44 @@ public void deleteTIFSourceConfig( )); } + // Manually delete threat intel job scheduler lock if IOC_UPLOAD type + public void deleteJobSchedulerLockIfNeeded( + SATIFSourceConfig saTifSourceConfig, + final ActionListener actionListener + ) { + if (saTifSourceConfig.getType() != SourceConfigType.IOC_UPLOAD) { + actionListener.onResponse(null); + return; + } + + // check to make sure the job scheduler lock index exists + if (clusterService.state().metadata().hasIndex(LOCK_INDEX_NAME) == false) { + actionListener.onFailure(SecurityAnalyticsException.wrap(new OpenSearchStatusException("Threat intel job scheduler lock index does not exist", RestStatus.BAD_REQUEST))); + return; + } + + String id = SecurityAnalyticsPlugin.JOB_INDEX_NAME + "-" + saTifSourceConfig.getId(); + DeleteRequest request = new DeleteRequest(LOCK_INDEX_NAME, id) + .setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE) + .timeout(clusterSettings.get(INDEX_TIMEOUT)); + + client.delete(request, ActionListener.wrap( + deleteResponse -> { + if (deleteResponse.status().equals(RestStatus.OK)) { + log.info("Deleted threat intel job scheduler lock [{}] successfully", id); + actionListener.onResponse(deleteResponse); + } else if (deleteResponse.status().equals(RestStatus.NOT_FOUND)) { + actionListener.onFailure(SecurityAnalyticsException.wrap(new OpenSearchStatusException(String.format(Locale.getDefault(), "Threat intel job scheduler lock with id [{%s}] not found", id), RestStatus.NOT_FOUND))); + } else { + actionListener.onFailure(SecurityAnalyticsException.wrap(new OpenSearchStatusException(String.format(Locale.getDefault(), "Failed to delete threat intel job scheduler lock with id [{%s}]", id), deleteResponse.status()))); + } + }, e -> { + log.error("Failed to delete threat intel job scheduler lock with id [{}]", id); + actionListener.onFailure(e); + } + )); + } + public void deleteAllIocIndices(Set indicesToDelete, Boolean backgroundJob, ActionListener listener) { if (indicesToDelete.isEmpty() == false) { DeleteIndexRequest deleteIndexRequest = new DeleteIndexRequest(indicesToDelete.toArray(new String[0])); @@ -398,6 +438,8 @@ public void deleteAllIocIndices(Set indicesToDelete, Boolean backgroundJ } ) ); + } else if (listener != null) { + listener.onResponse(new AcknowledgedResponse(true)); } } diff --git a/src/test/java/org/opensearch/securityanalytics/resthandler/SATIFSourceConfigRestApiIT.java b/src/test/java/org/opensearch/securityanalytics/resthandler/SATIFSourceConfigRestApiIT.java index 484709001..e3460e561 100644 --- a/src/test/java/org/opensearch/securityanalytics/resthandler/SATIFSourceConfigRestApiIT.java +++ b/src/test/java/org/opensearch/securityanalytics/resthandler/SATIFSourceConfigRestApiIT.java @@ -14,6 +14,7 @@ import org.junit.Before; import org.opensearch.client.Response; import org.opensearch.client.ResponseException; +import org.opensearch.core.rest.RestStatus; import org.opensearch.jobscheduler.spi.schedule.IntervalSchedule; import org.opensearch.search.SearchHit; import org.opensearch.securityanalytics.SecurityAnalyticsPlugin; @@ -734,7 +735,7 @@ public void testWithNoIOCsToDownload() { } } - public void testWhenBucketObjectDoesNotExist() { + public void testWhenBucketObjectDoesNotExist() throws IOException { // Only run tests when required system params are provided if (!canRunTests) return; @@ -779,13 +780,34 @@ public void testWhenBucketObjectDoesNotExist() { true ); - Exception exception = assertThrows(ResponseException.class, () -> - makeRequest(client(), "POST", SecurityAnalyticsPlugin.THREAT_INTEL_SOURCE_URI, Collections.emptyMap(), toHttpEntity(saTifSourceConfigDto)) - ); - - String expectedError = "{\"error\":{\"root_cause\":[{\"type\":\"no_such_key_exception\",\"reason\":\"The specified key does not exist."; - assertTrue("Exception contains unexpected message: " + exception.getMessage(), exception.getMessage().contains(expectedError)); + try { + makeRequest(client(), "POST", SecurityAnalyticsPlugin.THREAT_INTEL_SOURCE_URI, Collections.emptyMap(), toHttpEntity(saTifSourceConfigDto)); + } catch (ResponseException exception) { + assertEquals(RestStatus.NOT_FOUND, restStatus(exception.getResponse())); + String expectedError = "The specified key does not exist."; + assertTrue("Exception contains unexpected message: " + exception.getMessage(), exception.getMessage().contains(expectedError)); + } } + + // ensure that source config was deleted + String request = "{\n" + + " \"query\" : {\n" + + " \"match_all\":{\n" + + " }\n" + + " }\n" + + "}"; + List hits = executeSearch(JOB_INDEX_NAME, request); + Assert.assertEquals(0, hits.size()); + + // ensure that ioc indices were deleted + request = "{\n" + + " \"query\" : {\n" + + " \"match_all\":{\n" + + " }\n" + + " }\n" + + "}"; + hits = executeSearch(JOB_INDEX_NAME, request); + Assert.assertEquals(0, hits.size()); } public void testWhenRoleArnIsEmpty() throws IOException { diff --git a/src/test/java/org/opensearch/securityanalytics/resthandler/SourceConfigWithoutS3RestApiIT.java b/src/test/java/org/opensearch/securityanalytics/resthandler/SourceConfigWithoutS3RestApiIT.java index 2d9612f4f..4e7f89904 100644 --- a/src/test/java/org/opensearch/securityanalytics/resthandler/SourceConfigWithoutS3RestApiIT.java +++ b/src/test/java/org/opensearch/securityanalytics/resthandler/SourceConfigWithoutS3RestApiIT.java @@ -40,6 +40,7 @@ import java.util.Locale; import java.util.Map; +import static org.opensearch.jobscheduler.spi.utils.LockService.LOCK_INDEX_NAME; import static org.opensearch.securityanalytics.SecurityAnalyticsPlugin.JOB_INDEX_NAME; import static org.opensearch.securityanalytics.services.STIX2IOCFeedStore.IOC_ALL_INDEX_PATTERN; import static org.opensearch.securityanalytics.services.STIX2IOCFeedStore.getAllIocIndexPatternById; @@ -718,6 +719,10 @@ public void testDeleteIocUploadSourceConfigAndAllIocs() throws IOException { // ensure all iocs are deleted hits = executeSearch(IOC_ALL_INDEX_PATTERN, request); Assert.assertEquals(0, hits.size()); + + // ensure that lock is deleted + hits = executeSearch(LOCK_INDEX_NAME,request); + Assert.assertEquals(0, hits.size()); } public void testRefreshIocUploadSourceConfigFailure() throws IOException {