/**
* 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 <http://www.gnu.org/licenses/>.
*/
package org.graylog2.dashboards.widgets.strategies;
import com.google.common.collect.ImmutableMap;
import com.google.inject.assistedinject.Assisted;
import com.google.inject.assistedinject.AssistedInject;
import org.graylog2.dashboards.widgets.InvalidWidgetConfigurationException;
import org.graylog2.indexer.FieldTypeException;
import org.graylog2.indexer.results.HistogramResult;
import org.graylog2.indexer.searches.Searches;
import org.graylog2.plugin.dashboards.widgets.ComputationResult;
import org.graylog2.plugin.dashboards.widgets.WidgetStrategy;
import org.graylog2.plugin.indexer.searches.timeranges.AbsoluteRange;
import org.graylog2.plugin.indexer.searches.timeranges.TimeRange;
import org.joda.time.DateTime;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.util.ArrayList;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import static com.google.common.base.Strings.isNullOrEmpty;
public class StackedChartWidgetStrategy extends ChartWidgetStrategy {
public interface Factory extends WidgetStrategy.Factory<StackedChartWidgetStrategy> {
@Override
StackedChartWidgetStrategy create(Map<String, Object> config, TimeRange timeRange, String widgetId);
}
private static final Logger LOG = LoggerFactory.getLogger(StackedChartWidgetStrategy.class);
private final List<Series> chartSeries;
private final Searches searches;
private final TimeRange timeRange;
private final String widgetId;
@AssistedInject
public StackedChartWidgetStrategy(Searches searches, @Assisted Map<String, Object> config, @Assisted TimeRange timeRange, @Assisted String widgetId) throws InvalidWidgetConfigurationException {
super(config);
this.searches = searches;
this.timeRange = timeRange;
this.widgetId = widgetId;
if (!checkConfig(config)) {
throw new InvalidWidgetConfigurationException("Missing or invalid widget configuration. Provided config was: " + config.toString());
}
final Object persistedSeries = config.get("series");
if (persistedSeries instanceof List) {
final List chartSeries = (List) persistedSeries;
this.chartSeries = new ArrayList<>(chartSeries.size());
for (final Object series : chartSeries) {
this.chartSeries.add(Series.fromMap((Map<String, Object>) series));
}
} else {
throw new InvalidWidgetConfigurationException("Invalid widget configuration, 'series' should be a list: " + config.toString());
}
}
@Override
public ComputationResult compute() {
String filter = null;
if (!isNullOrEmpty(streamId)) {
filter = "streams:" + streamId;
}
final List<Map> results = new ArrayList<>(chartSeries.size());
DateTime from = null;
DateTime to = null;
long tookMs = 0;
for (Series series : chartSeries) {
try {
final HistogramResult histogramResult = searches.fieldHistogram(
series.query,
series.field,
Searches.DateHistogramInterval.valueOf(interval.toString().toUpperCase(Locale.ENGLISH)),
filter,
this.timeRange,
"cardinality".equalsIgnoreCase(series.statisticalFunction));
if (from == null) {
from = histogramResult.getHistogramBoundaries().getFrom();
}
to = histogramResult.getHistogramBoundaries().getTo();
results.add(histogramResult.getResults());
tookMs += histogramResult.tookMs();
} catch (FieldTypeException e) {
String msg = "Could not calculate [" + this.getClass().getCanonicalName() + "] widget <" + widgetId + ">. Not a numeric field? The field was [" + series.field + "]";
LOG.error(msg, e);
throw new RuntimeException(msg, e);
}
}
final AbsoluteRange computationTimeRange = AbsoluteRange.create(from, to);
return new ComputationResult(results, tookMs, computationTimeRange);
}
private boolean checkConfig(Map<String, Object> config) {
return config.containsKey("renderer")
&& config.containsKey("interpolation")
&& config.containsKey("interval")
&& config.containsKey("series");
}
private static class Series {
final String field;
final String query;
final String statisticalFunction;
public static Series fromMap(Map<String, Object> fields) {
return new Series((String) fields.get("query"), (String) fields.get("field"), (String) fields.get("statistical_function"));
};
Series(String query, String field, String statisticalFunction) {
if (query == null || query.trim().isEmpty()) {
this.query = "*";
} else {
this.query = query;
}
this.field = field;
this.statisticalFunction = statisticalFunction;
}
public Map<String, Object> toMap() {
ImmutableMap.Builder<String, Object> mapBuilder = ImmutableMap.<String, Object>builder()
.put("query", query)
.put("field", field)
.put("statistical_function", statisticalFunction);
return mapBuilder.build();
}
}
}