org.graylog.plugins.views.search.export.es.Scroll Maven / Gradle / Ivy
/**
* This file is part of Graylog.
*
* Graylog is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Graylog is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Graylog. If not, see .
*/
package org.graylog.plugins.views.search.export.es;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import io.searchbox.client.JestResult;
import io.searchbox.core.ClearScroll;
import io.searchbox.core.Search;
import io.searchbox.core.SearchResult;
import io.searchbox.core.SearchScroll;
import io.searchbox.core.search.sort.Sort;
import io.searchbox.params.Parameters;
import org.graylog.plugins.views.search.export.ExportMessagesCommand;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.inject.Inject;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;
import static com.google.common.collect.Lists.newArrayList;
public class Scroll implements RequestStrategy {
private static final Logger LOG = LoggerFactory.getLogger(Scroll.class);
private static final String SCROLL_TIME = "1m";
private final ObjectMapper objectMapper;
private final JestWrapper jestWrapper;
private String currentScrollId;
@Inject
public Scroll(ObjectMapper objectMapper, JestWrapper jestWrapper) {
this.objectMapper = objectMapper;
this.jestWrapper = jestWrapper;
}
@SuppressWarnings("rawtypes")
@Override
public List> nextChunk(Search.Builder search, ExportMessagesCommand command) {
List> hits = retrieveHits(search);
if (hits.isEmpty()) {
cancelScroll();
}
return hits;
}
@SuppressWarnings("rawtypes")
private List> retrieveHits(Search.Builder search) {
if (isFirstRequest()) {
SearchResult result = search(search);
currentScrollId = scrollIdFrom(result);
return result.getHits(Map.class, false);
} else {
JestResult result = continueScroll(currentScrollId);
currentScrollId = scrollIdFrom(result);
return hitsFrom(result);
}
}
@SuppressWarnings("rawtypes")
private List> hitsFrom(JestResult result) {
return StreamSupport.stream(result.getJsonObject().path("hits").path("hits").spliterator(), false)
.map(this::hitFromSource)
.collect(Collectors.toList());
}
private boolean isFirstRequest() {
return currentScrollId == null;
}
private void cancelScroll() {
ClearScroll.Builder clearScrollBuilder = new ClearScroll.Builder().addScrollId(currentScrollId);
JestResult result = jestWrapper.execute(clearScrollBuilder.build(), () -> "Failed to cancel scroll " + currentScrollId);
if (!result.isSucceeded()) {
LOG.error("Failed to cancel scroll with id " + currentScrollId);
}
}
private SearchResult search(Search.Builder search) {
Search.Builder modified = search
.setParameter(Parameters.SCROLL, "1m")
.addSort(unsorted());
return jestWrapper.execute(modified.build(), () -> "Failed to execute initial Scroll request");
}
private Sort unsorted() {
return new Sort("_doc", Sort.Sorting.ASC);
}
private JestResult continueScroll(String scrollId) {
SearchScroll scroll = new SearchScroll.Builder(scrollId, SCROLL_TIME).build();
return jestWrapper.execute(scroll, () -> "Failed to execute Scroll request with scroll id " + currentScrollId);
}
private String scrollIdFrom(JestResult result) {
return result.getJsonObject().path("_scroll_id").asText();
}
@SuppressWarnings("rawtypes")
private SearchResult.Hit