19package org.sleuthkit.autopsy.timeline.ui.detailview.datamodel;
21import com.google.common.cache.CacheBuilder;
22import com.google.common.cache.LoadingCache;
23import com.google.common.collect.HashMultimap;
24import com.google.common.collect.SetMultimap;
25import com.google.common.eventbus.Subscribe;
26import java.sql.ResultSet;
27import java.sql.SQLException;
28import java.util.ArrayList;
29import java.util.Collection;
30import java.util.Comparator;
31import java.util.HashMap;
32import java.util.Iterator;
36import java.util.SortedSet;
37import java.util.TreeSet;
38import java.util.concurrent.ExecutionException;
39import java.util.concurrent.TimeUnit;
40import java.util.function.Consumer;
41import java.util.logging.Level;
42import java.util.stream.Collectors;
43import org.apache.commons.lang3.tuple.ImmutablePair;
44import org.joda.time.DateTimeZone;
45import org.joda.time.Interval;
46import org.joda.time.Period;
47import org.sleuthkit.autopsy.coreutils.Logger;
48import org.sleuthkit.autopsy.timeline.EventsModel;
49import org.sleuthkit.autopsy.timeline.TimeLineController;
50import org.sleuthkit.autopsy.timeline.ui.filtering.datamodel.UIFilter;
51import org.sleuthkit.autopsy.timeline.utils.CacheLoaderImpl;
52import org.sleuthkit.autopsy.timeline.utils.RangeDivision;
53import org.sleuthkit.autopsy.timeline.zooming.TimeUnits;
54import org.sleuthkit.autopsy.timeline.zooming.EventsModelParams;
55import org.sleuthkit.datamodel.SleuthkitCase;
56import org.sleuthkit.datamodel.TimelineManager;
57import org.sleuthkit.datamodel.TskCoreException;
58import org.sleuthkit.datamodel.TimelineEventType;
59import org.sleuthkit.datamodel.TimelineEvent;
60import org.sleuthkit.datamodel.TimelineFilter;
61import org.sleuthkit.datamodel.TimelineLevelOfDetail;
72 private final LoadingCache<EventsModelParams, List<TimelineEvent>>
eventCache;
79 this.sleuthkitCase =
eventsModel.getSleuthkitCase();
82 .expireAfterAccess(10, TimeUnit.MINUTES)
119 Interval timeRange = zoom.getTimeRange();
120 TimelineLevelOfDetail descriptionLOD = zoom.getTimelineLOD();
123 Map<TimelineEventType, SetMultimap< String, EventCluster>> eventClusters =
new HashMap<>();
127 .forEach(
new Consumer<TimelineEvent>() {
129 public void accept(TimelineEvent event) {
130 TimelineEventType clusterType =
event.getEventType().getCategory();
131 eventClusters.computeIfAbsent(clusterType, eventType -> HashMultimap.create())
132 .put(event.getDescription(descriptionLOD),
new EventCluster(event, clusterType, descriptionLOD));
139 }
catch (ExecutionException ex) {
140 throw new TskCoreException(
"Failed to load Event Stripes from cache for " + zoom.toString(), ex);
161 Interval timeRange = zoom.getTimeRange();
162 TimelineFilter.RootFilter activeFilter = zoom.getEventFilterState().getActiveFilter();
179 static private List<EventStripe>
mergeClustersToStripes(Period timeUnitLength, Map<TimelineEventType, SetMultimap< String, EventCluster>> eventClusters) {
182 ArrayList<EventCluster> mergedClusters =
new ArrayList<>();
185 for (Map.Entry<TimelineEventType, SetMultimap<String, EventCluster>> typeMapEntry : eventClusters.entrySet()) {
186 TimelineEventType type = typeMapEntry.getKey();
187 SetMultimap<String, EventCluster> descrMap = typeMapEntry.getValue();
189 for (String descr : descrMap.keySet()) {
190 Set<EventCluster>
events = descrMap.get(descr);
192 Iterator<EventCluster> iterator =
events.stream()
198 while (iterator.hasNext()) {
204 if (gap ==
null || gap.toDuration().getMillis() <= timeUnitLength.toDurationFrom(gap.getStart()).getMillis() / 4) {
209 mergedClusters.add(current);
213 mergedClusters.add(current);
218 Map<ImmutablePair<TimelineEventType, String>,
EventStripe> stripeDescMap =
new HashMap<>();
221 stripeDescMap.merge(ImmutablePair.of(eventCluster.getEventType(), eventCluster.getDescription()),
225 return stripeDescMap.values().stream()
227 .collect(Collectors.toList());
239 static <X> SortedSet<X> copyAsSortedSet(Collection<X> setA, Comparator<X> comparator) {
240 TreeSet<X> treeSet =
new TreeSet<>(comparator);
241 treeSet.addAll(setA);
synchronized static Logger getLogger(String name)
static DateTimeZone getJodaTimeZone()
List< TimelineEvent > getEvents(EventsModelParams zoom, DateTimeZone timeZone)
DetailsViewModel(EventsModel eventsModel)
List< EventStripe > getEventStripes(UIFilter uiFilter, EventsModelParams zoom)
static final Logger logger
final SleuthkitCase sleuthkitCase
final TimelineManager eventManager
List< EventStripe > getEventStripes(EventsModelParams zoom)
final EventsModel eventsModel
static List< EventStripe > mergeClustersToStripes(Period timeUnitLength, Map< TimelineEventType, SetMultimap< String, EventCluster > > eventClusters)
final LoadingCache< EventsModelParams, List< TimelineEvent > > eventCache
static EventCluster merge(EventCluster cluster1, EventCluster cluster2)
static EventStripe merge(EventStripe stripeA, EventStripe stripeB)
TimeUnits getPeriodSize()
static RangeDivision getRangeDivision(Interval timeRange, DateTimeZone timeZone)
static UIFilter getAllPassFilter()