19 package org.sleuthkit.autopsy.timeline.ui.detailview.datamodel;
21 import com.google.common.cache.CacheBuilder;
22 import com.google.common.cache.LoadingCache;
23 import com.google.common.collect.HashMultimap;
24 import com.google.common.collect.SetMultimap;
25 import com.google.common.eventbus.Subscribe;
26 import java.sql.ResultSet;
27 import java.sql.SQLException;
28 import java.util.ArrayList;
29 import java.util.Collection;
30 import java.util.Comparator;
31 import java.util.HashMap;
32 import java.util.Iterator;
33 import java.util.List;
36 import java.util.SortedSet;
37 import java.util.TreeSet;
38 import java.util.concurrent.ExecutionException;
39 import java.util.concurrent.TimeUnit;
40 import java.util.logging.Level;
41 import java.util.stream.Collectors;
42 import org.apache.commons.lang3.tuple.ImmutablePair;
43 import org.joda.time.DateTimeZone;
44 import org.joda.time.Interval;
45 import org.joda.time.Period;
70 private final LoadingCache<ZoomState, List<TimelineEvent>>
eventCache;
78 eventCache = CacheBuilder.newBuilder()
80 .expireAfterAccess(10, TimeUnit.MINUTES)
88 eventCache.invalidateAll();
116 Interval timeRange = zoom.getTimeRange();
117 TimelineEvent.DescriptionLevel descriptionLOD = zoom.getDescriptionLOD();
118 TimelineEventType.TypeLevel typeZoomLevel = zoom.getTypeZoomLevel();
121 Map<TimelineEventType, SetMultimap< String, EventCluster>> eventClusters =
new HashMap<>();
123 eventCache.get(zoom).stream()
126 TimelineEventType clusterType =
event.getEventType(typeZoomLevel);
127 eventClusters.computeIfAbsent(clusterType, eventType -> HashMultimap.create())
128 .put(event.getDescription(descriptionLOD),
new EventCluster(event, clusterType, descriptionLOD));
134 }
catch (ExecutionException ex) {
135 throw new TskCoreException(
"Failed to load Event Stripes from cache for " + zoom.toString(), ex);
154 private List<TimelineEvent>
getEvents(
ZoomState zoom, DateTimeZone timeZone)
throws TskCoreException {
156 Interval timeRange = zoom.getTimeRange();
157 TimelineFilter.RootFilter activeFilter = zoom.getFilterState().getActiveFilter();
158 return eventManager.getEvents(timeRange, activeFilter);
174 static private List<EventStripe>
mergeClustersToStripes(Period timeUnitLength, Map<TimelineEventType, SetMultimap< String, EventCluster>> eventClusters) {
177 ArrayList<EventCluster> mergedClusters =
new ArrayList<>();
180 for (Map.Entry<TimelineEventType, SetMultimap<String, EventCluster>> typeMapEntry : eventClusters.entrySet()) {
181 TimelineEventType type = typeMapEntry.getKey();
182 SetMultimap<String, EventCluster> descrMap = typeMapEntry.getValue();
184 for (String descr : descrMap.keySet()) {
185 Set<EventCluster> events = descrMap.get(descr);
187 Iterator<EventCluster> iterator = events.stream()
193 while (iterator.hasNext()) {
199 if (gap == null || gap.toDuration().getMillis() <= timeUnitLength.toDurationFrom(gap.getStart()).getMillis() / 4) {
204 mergedClusters.add(current);
208 mergedClusters.add(current);
213 Map<ImmutablePair<TimelineEventType, String>,
EventStripe> stripeDescMap =
new HashMap<>();
216 stripeDescMap.
merge(ImmutablePair.of(eventCluster.getEventType(), eventCluster.getDescription()),
220 return stripeDescMap.values().stream()
222 .collect(Collectors.toList());
234 static <X> SortedSet<X> copyAsSortedSet(Collection<X> setA, Comparator<X> comparator) {
235 TreeSet<X> treeSet =
new TreeSet<>(comparator);
236 treeSet.addAll(setA);
List< EventStripe > getEventStripes(UIFilter uiFilter, ZoomState zoom)
List< TimelineEvent > getEvents(ZoomState zoom, DateTimeZone timeZone)
final LoadingCache< ZoomState, List< TimelineEvent > > eventCache
SleuthkitCase getSleuthkitCase()
final TimelineManager eventManager
static UIFilter getAllPassFilter()
static final Logger logger
TimelineManager getEventManager()
final SleuthkitCase sleuthkitCase
List< EventStripe > getEventStripes(ZoomState zoom)
static EventStripe merge(EventStripe stripeA, EventStripe stripeB)
static EventCluster merge(EventCluster cluster1, EventCluster cluster2)
DetailsViewModel(FilteredEventsModel eventsModel)
static DateTimeZone getJodaTimeZone()
static List< EventStripe > mergeClustersToStripes(Period timeUnitLength, Map< TimelineEventType, SetMultimap< String, EventCluster >> eventClusters)
TimeUnits getPeriodSize()
synchronized static Logger getLogger(String name)
final FilteredEventsModel eventsModel
static RangeDivision getRangeDivision(Interval timeRange, DateTimeZone timeZone)
synchronized void registerForEvents(Object subscriber)