19 package org.sleuthkit.autopsy.datamodel;
21 import java.beans.PropertyChangeEvent;
22 import java.beans.PropertyChangeListener;
23 import java.util.ArrayList;
24 import java.util.Arrays;
25 import java.util.EnumSet;
26 import java.util.List;
27 import java.util.Observable;
28 import java.util.Observer;
30 import java.util.logging.Level;
31 import javax.swing.JOptionPane;
32 import javax.swing.SwingUtilities;
33 import org.openide.nodes.AbstractNode;
34 import org.openide.nodes.ChildFactory;
35 import org.openide.nodes.Children;
36 import org.openide.nodes.Node;
37 import org.openide.nodes.Sheet;
38 import org.openide.util.NbBundle;
39 import org.openide.util.lookup.Lookups;
40 import org.openide.windows.WindowManager;
65 @NbBundle.Messages({
"DeletedContent.fsDelFilter.text=File System",
66 "DeletedContent.allDelFilter.text=All"})
69 FS_DELETED_FILTER(0,
"FS_DELETED_FILTER",
70 Bundle.DeletedContent_fsDelFilter_text()),
71 ALL_DELETED_FILTER(1,
"ALL_DELETED_FILTER",
72 Bundle.DeletedContent_allDelFilter_text());
81 this.displayName = displayName;
94 return this.displayName;
99 return visitor.
visit(
this);
109 return visitor.
visit(
this);
118 @NbBundle.Messages(
"DeletedContent.deletedContentsNode.name=Deleted Files")
119 private static final String
NAME = Bundle.DeletedContent_deletedContentsNode_name();
124 super.setDisplayName(NAME);
125 this.setIconBaseWithExtension(
"org/sleuthkit/autopsy/images/file-icon-deleted.png");
135 return visitor.
visit(
this);
140 "DeletedContent.createSheet.name.displayName=Name",
141 "DeletedContent.createSheet.name.desc=no description"})
143 Sheet sheet = super.createSheet();
144 Sheet.Set sheetSet = sheet.get(Sheet.PROPERTIES);
145 if (sheetSet == null) {
146 sheetSet = Sheet.createPropertiesSet();
151 Bundle.DeletedContent_createSheet_name_displayName(),
152 Bundle.DeletedContent_createSheet_name_desc(),
159 return getClass().getName();
198 private final PropertyChangeListener
pcl = (PropertyChangeEvent evt) -> {
199 String eventType = evt.getPropertyName();
238 if (evt.getNewValue() == null) {
241 maxFilesDialogShown =
false;
253 protected boolean createKeys(List<DeletedContent.DeletedContentFilter> list) {
270 super(Children.create(
new DeletedContentChildren(
filter, skCase, null),
true), Lookups.singleton(
filter.getDisplayName()));
276 super(Children.create(
new DeletedContentChildren(
filter, skCase, o),
true), Lookups.singleton(
filter.getDisplayName()));
279 o.addObserver(
new DeletedContentNodeObserver());
283 super.setName(
filter.getName());
285 String tooltip =
filter.getDisplayName();
286 this.setShortDescription(tooltip);
287 this.setIconBaseWithExtension(
"org/sleuthkit/autopsy/images/file-icon-deleted.png");
295 public void update(Observable o, Object arg) {
302 final long count = DeletedContentChildren.calculateItems(skCase,
filter);
304 super.setDisplayName(
filter.getDisplayName() +
" (" + count +
")");
309 return visitor.
visit(
this);
314 "DeletedContent.createSheet.filterType.displayName=Type",
315 "DeletedContent.createSheet.filterType.desc=no description"})
317 Sheet sheet = super.createSheet();
318 Sheet.Set sheetSet = sheet.get(Sheet.PROPERTIES);
319 if (sheetSet == null) {
320 sheetSet = Sheet.createPropertiesSet();
325 Bundle.DeletedContent_createSheet_filterType_displayName(),
326 Bundle.DeletedContent_createSheet_filterType_desc(),
327 filter.getDisplayName()));
347 static class DeletedContentChildren
extends ChildFactory.Detachable<AbstractFile> {
349 private final SleuthkitCase
skCase;
352 private static final int MAX_OBJECTS = 10001;
357 this.filter = filter;
361 private final Observer observer =
new DeletedContentChildrenObserver();
367 public void update(Observable o, Object arg) {
373 protected void addNotify() {
374 if (notifier != null) {
375 notifier.addObserver(observer);
380 protected void removeNotify() {
381 if (notifier != null) {
382 notifier.deleteObserver(observer);
387 @NbBundle.Messages({
"# {0} - The deleted files threshold",
388 "DeletedContent.createKeys.maxObjects.msg="
389 +
"There are more Deleted Files than can be displayed."
390 +
" Only the first {0} Deleted Files will be shown."})
391 protected boolean createKeys(List<AbstractFile> list) {
392 List<AbstractFile> queryList = runFsQuery();
393 if (queryList.size() == MAX_OBJECTS) {
394 queryList.remove(queryList.size() - 1);
396 if (maxFilesDialogShown ==
false) {
397 maxFilesDialogShown =
true;
398 SwingUtilities.invokeLater(()
399 -> JOptionPane.showMessageDialog(WindowManager.getDefault().getMainWindow(),
400 DeletedContent_createKeys_maxObjects_msg(MAX_OBJECTS - 1))
404 list.addAll(queryList);
408 static private String makeQuery(
DeletedContent.DeletedContentFilter filter) {
411 case FS_DELETED_FILTER:
412 query =
"dir_flags = " + TskData.TSK_FS_NAME_FLAG_ENUM.UNALLOC.getValue()
413 +
" AND meta_flags != " + TskData.TSK_FS_META_FLAG_ENUM.ORPHAN.getValue()
414 +
" AND type = " + TskData.TSK_DB_FILES_TYPE_ENUM.FS.getFileType();
417 case ALL_DELETED_FILTER:
420 +
"(dir_flags = " + TskData.TSK_FS_NAME_FLAG_ENUM.UNALLOC.getValue()
422 +
"meta_flags = " + TskData.TSK_FS_META_FLAG_ENUM.ORPHAN.getValue()
424 +
" AND type = " + TskData.TSK_DB_FILES_TYPE_ENUM.FS.getFileType()
426 +
" OR type = " + TskData.TSK_DB_FILES_TYPE_ENUM.CARVED.getFileType()
437 logger.log(Level.SEVERE,
"Unsupported filter type to get deleted content: {0}", filter);
441 if (UserPreferences.hideKnownFilesInViewsTree()) {
442 query +=
" AND (known != " + TskData.FileKnown.KNOWN.getFileKnownValue()
443 +
" OR known IS NULL)";
446 query +=
" LIMIT " + MAX_OBJECTS;
450 private List<AbstractFile> runFsQuery() {
451 List<AbstractFile> ret =
new ArrayList<>();
453 String query = makeQuery(filter);
455 ret = skCase.findAllFilesWhere(query);
456 }
catch (TskCoreException e) {
457 logger.log(Level.SEVERE,
"Error getting files for the deleted content view using: " + query, e);
472 static long calculateItems(SleuthkitCase sleuthkitCase,
DeletedContent.DeletedContentFilter filter) {
474 return sleuthkitCase.countFilesWhere(makeQuery(filter));
475 }
catch (TskCoreException ex) {
476 logger.log(Level.SEVERE,
"Error getting deleted files search view count", ex);
483 return key.accept(
new ContentVisitor.Default<AbstractNode>() {
484 public FileNode visit(AbstractFile f) {
485 return new FileNode(f, false);
488 public FileNode visit(FsContent f) {
489 return new FileNode(f,
false);
493 public FileNode visit(LayoutFile f) {
494 return new FileNode(f,
false);
498 public FileNode visit(File f) {
499 return new FileNode(f,
false);
503 public FileNode visit(Directory f) {
504 return new FileNode(f,
false);
508 protected AbstractNode defaultVisit(Content di) {
509 throw new UnsupportedOperationException(
"Not supported for this type of Displayable Item: " + di.toString());
void removeIngestModuleEventListener(final PropertyChangeListener listener)
static synchronized IngestManager getInstance()
public< T > T accept(AutopsyItemVisitor< T > visitor)
void update(Observable o, Object arg)
static volatile boolean maxFilesDialogShown
static final Set< Case.Events > CASE_EVENTS_OF_INTEREST
T visit(DataSourcesNode in)
final DeletedContent.DeletedContentFilter filter
void removeIngestJobEventListener(final PropertyChangeListener listener)
DeletedContentFilter(int id, String name, String displayName)
boolean createKeys(List< DeletedContent.DeletedContentFilter > list)
SleuthkitCase getSleuthkitCase()
void addIngestJobEventListener(final PropertyChangeListener listener)
final PropertyChangeListener pcl
DeletedContentsChildren(SleuthkitCase skCase)
void addIngestModuleEventListener(final PropertyChangeListener listener)
synchronized static Logger getLogger(String name)
static Case getCurrentCaseThrows()
static void addEventTypeSubscriber(Set< Events > eventTypes, PropertyChangeListener subscriber)
Node createNodeForKey(DeletedContent.DeletedContentFilter key)
DeletedContent(SleuthkitCase skCase)
void update(Observable o, Object arg)
static void removeEventTypeSubscriber(Set< Events > eventTypes, PropertyChangeListener subscriber)