Autopsy 4.23.0
Graphical digital forensics platform for The Sleuth Kit and other tools.
ExtractActionHelper.java
Go to the documentation of this file.
1/*
2 * Autopsy Forensic Browser
3 *
4 * Copyright 2013-2021 Basis Technology Corp.
5 * Contact: carrier <at> sleuthkit <dot> org
6 *
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this content except in compliance with the License.
9 * You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
18 */
19package org.sleuthkit.autopsy.directorytree.actionhelpers;
20
21import java.awt.Component;
22import java.awt.event.ActionEvent;
23import java.io.File;
24import java.io.IOException;
25import java.text.MessageFormat;
26import java.util.ArrayList;
27import java.util.Collection;
28import java.util.HashSet;
29import java.util.Iterator;
30import java.util.List;
31import java.util.Set;
32import java.util.concurrent.ExecutionException;
33import java.util.logging.Level;
34import javax.swing.JFileChooser;
35import javax.swing.JOptionPane;
36import javax.swing.SwingWorker;
37import org.netbeans.api.progress.ProgressHandle;
38import org.openide.util.Cancellable;
39import org.openide.util.NbBundle;
40import org.openide.util.NbBundle.Messages;
41import org.sleuthkit.autopsy.casemodule.Case;
42import org.sleuthkit.autopsy.casemodule.NoCurrentCaseException;
43import org.sleuthkit.autopsy.coreutils.FileUtil;
44import org.sleuthkit.autopsy.coreutils.Logger;
45import org.sleuthkit.autopsy.coreutils.MessageNotifyUtil;
46import org.sleuthkit.autopsy.datamodel.ContentUtils;
47import org.sleuthkit.autopsy.datamodel.ContentUtils.ExtractFscContentVisitor;
48import org.sleuthkit.autopsy.guiutils.JFileChooserFactory;
49import org.sleuthkit.datamodel.AbstractFile;
50import org.sleuthkit.datamodel.Content;
51
55public class ExtractActionHelper {
56
57 private final Logger logger = Logger.getLogger(ExtractActionHelper.class.getName());
58 private String userDefinedExportPath;
59
62
71 public void extract(ActionEvent event, Collection<? extends AbstractFile> selectedFiles) {
72 if (selectedFiles.size() > 1) {
73 extractFiles(event, selectedFiles);
74 } else if (selectedFiles.size() == 1) {
75 extractFile(event, selectedFiles.iterator().next());
76 }
77 }
78
85 @NbBundle.Messages({"ExtractActionHelper.noOpenCase.errMsg=No open case available."
86 })
87 private void extractFile(ActionEvent event, AbstractFile selectedFile) {
88 Case openCase;
89 try {
90 openCase = Case.getCurrentCaseThrows();
91 } catch (NoCurrentCaseException ex) {
92 JOptionPane.showMessageDialog((Component) event.getSource(), Bundle.ExtractActionHelper_noOpenCase_errMsg());
93 logger.log(Level.INFO, "Exception while getting open case.", ex); //NON-NLS
94 return;
95 }
96 JFileChooser fileChooser = extractFileHelper.getChooser();
97 fileChooser.setCurrentDirectory(new File(getExportDirectory(openCase)));
98 // If there is an attribute name, change the ":". Otherwise the extracted file will be hidden
99 fileChooser.setSelectedFile(new File(FileUtil.escapeFileName(selectedFile.getName())));
100 if (fileChooser.showSaveDialog((Component) event.getSource()) == JFileChooser.APPROVE_OPTION) {
101 updateExportDirectory(fileChooser.getSelectedFile().getParent(), openCase);
102
103 ArrayList<FileExtractionTask> fileExtractionTasks = new ArrayList<>();
104 fileExtractionTasks.add(new FileExtractionTask(selectedFile, fileChooser.getSelectedFile()));
105 runExtractionTasks(event, fileExtractionTasks, fileChooser.getSelectedFile().getName());
106 }
107 }
108
115 private void extractFiles(ActionEvent event, Collection<? extends AbstractFile> selectedFiles) {
116 Case openCase;
117 try {
118 openCase = Case.getCurrentCaseThrows();
119 } catch (NoCurrentCaseException ex) {
120 JOptionPane.showMessageDialog((Component) event.getSource(), Bundle.ExtractActionHelper_noOpenCase_errMsg());
121 logger.log(Level.INFO, "Exception while getting open case.", ex); //NON-NLS
122 return;
123 }
124 JFileChooser folderChooser = extractFilesHelper.getChooser();
125 folderChooser.setFileSelectionMode(JFileChooser.DIRECTORIES_ONLY);
126 folderChooser.setCurrentDirectory(new File(getExportDirectory(openCase)));
127 if (folderChooser.showSaveDialog((Component) event.getSource()) == JFileChooser.APPROVE_OPTION) {
128 File destinationFolder = folderChooser.getSelectedFile();
129 if (!destinationFolder.exists()) {
130 try {
131 destinationFolder.mkdirs();
132 } catch (Exception ex) {
133 JOptionPane.showMessageDialog((Component) event.getSource(), NbBundle.getMessage(this.getClass(),
134 "ExtractAction.extractFiles.cantCreateFolderErr.msg"));
135 logger.log(Level.INFO, "Unable to create folder(s) for user " + destinationFolder.getAbsolutePath(), ex); //NON-NLS
136 return;
137 }
138 }
139 updateExportDirectory(destinationFolder.getPath(), openCase);
140
141 /*
142 * get the unique set of files from the list. A user once reported
143 * extraction taking days because it was extracting the same PST
144 * file 20k times. They selected 20k email messages in the tree and
145 * chose to extract them.
146 */
147 Set<AbstractFile> uniqueFiles = new HashSet<>(selectedFiles);
148
149 // make a task for each file
150 ArrayList<FileExtractionTask> fileExtractionTasks = new ArrayList<>();
151 for (AbstractFile source : uniqueFiles) {
152 // If there is an attribute name, change the ":". Otherwise the extracted file will be hidden
153 fileExtractionTasks.add(new FileExtractionTask(source, new File(destinationFolder, source.getId() + "-" + FileUtil.escapeFileName(source.getName()))));
154 }
155 runExtractionTasks(event, fileExtractionTasks, destinationFolder.getName());
156 }
157 }
158
166 private String getExportDirectory(Case openCase) {
167 String caseExportPath = openCase.getExportDirectory();
168
169 if (userDefinedExportPath == null) {
170 return caseExportPath;
171 }
172
173 File file = new File(userDefinedExportPath);
174 if (file.exists() == false || file.isDirectory() == false) {
175 return caseExportPath;
176 }
177
179 }
180
190 private void updateExportDirectory(String exportPath, Case openCase) {
191 if (exportPath.equalsIgnoreCase(openCase.getExportDirectory())) {
193 } else {
194 userDefinedExportPath = exportPath;
195 }
196 }
197
207 private void runExtractionTasks(ActionEvent event, List<FileExtractionTask> fileExtractionTasks, String destName) {
208
209 // verify all of the sources and destinations are OK
210 for (Iterator<FileExtractionTask> it = fileExtractionTasks.iterator(); it.hasNext();) {
211 FileExtractionTask task = it.next();
212
213 if (ContentUtils.isDotDirectory(task.source)) {
214 it.remove();
215 continue;
216 }
217
218 /*
219 * This code assumes that each destination is unique. We previously
220 * satisfied that by adding the unique ID.
221 */
222 if (task.destination.exists()) {
223 if (JOptionPane.showConfirmDialog((Component) event.getSource(),
224 NbBundle.getMessage(this.getClass(), "ExtractActionHelper.confDlg.destFileExist.msg", task.destination.getAbsolutePath()),
225 NbBundle.getMessage(this.getClass(), "ExtractActionHelper.confDlg.destFileExist.title"),
226 JOptionPane.YES_NO_OPTION) == JOptionPane.YES_OPTION) {
227 if (!FileUtil.deleteFileDir(task.destination)) {
228 JOptionPane.showMessageDialog((Component) event.getSource(),
229 NbBundle.getMessage(this.getClass(), "ExtractActionHelper.msgDlg.cantOverwriteFile.msg", task.destination.getAbsolutePath()));
230 it.remove();
231 }
232 } else {
233 it.remove();
234 }
235 }
236 }
237
238 // launch a thread to do the work
239 if (!fileExtractionTasks.isEmpty()) {
240 try {
241 FileExtracter extracter = new FileExtracter(fileExtractionTasks, destName);
242 extracter.execute();
243 } catch (Exception ex) {
244 logger.log(Level.WARNING, "Unable to start background file extraction thread", ex); //NON-NLS
245 }
246 } else {
248 NbBundle.getMessage(this.getClass(), "ExtractActionHelper.notifyDlg.noFileToExtr.msg"));
249 }
250 }
251
255 private class FileExtractionTask {
256
257 AbstractFile source;
258 File destination;
259
266 FileExtractionTask(AbstractFile source, File destination) {
267 this.source = source;
268 this.destination = destination;
269 }
270 }
271
276 private static class UIExtractionVisitor<T, V> extends ExtractFscContentVisitor<T, V> {
277
288 UIExtractionVisitor(File dest, ProgressHandle progress, SwingWorker<T, V> worker, boolean source) {
289 super(dest, progress, worker, source);
290 }
291
305 static <T,V> void writeContent(Content content, File dest, ProgressHandle progress, SwingWorker<T, V> worker) {
306 content.accept(new UIExtractionVisitor<>(dest, progress, worker, true));
307 }
308
309
310 @Override
311 protected void writeFile(Content file, File dest, ProgressHandle progress, SwingWorker<T, V> worker, boolean source) throws IOException {
312 File destFile;
313 if (dest.exists()) {
314 String parent = dest.getParent();
315 String fileName = dest.getName();
316 String objIdFileName = MessageFormat.format("{0}-{1}", file.getId(), fileName);
317 destFile = new File(parent, objIdFileName);
318 } else {
319 destFile = dest;
320 }
321
322 super.writeFile(file, destFile, progress, worker, source);
323 }
324
325 @Override
326 protected ExtractFscContentVisitor<T, V> getChildVisitor(File childFile, ProgressHandle progress, SwingWorker<T, V> worker) {
327 return new UIExtractionVisitor<>(childFile, progress, worker, false);
328 }
329
330
331
332 }
333
337 private class FileExtracter extends SwingWorker<Object, Void> {
338
339 private final Logger logger = Logger.getLogger(FileExtracter.class.getName());
340 private final String destName;
341 private ProgressHandle progress;
342 private final List<FileExtractionTask> extractionTasks;
343
351 FileExtracter(List<FileExtractionTask> extractionTasks, String destName) {
352 this.extractionTasks = extractionTasks;
353 this.destName = destName;
354 }
355
356 @Override
357 @Messages({
358 "# {0} - outputFolderName",
359 "ExtractActionHelper.progress.extracting=Extracting to {0}",
360 "# {0} - fileName",
361 "ExtractActionHelper.progress.fileExtracting=Extracting file: {0}"
362 })
363 protected Object doInBackground() throws Exception {
364 if (extractionTasks.isEmpty()) {
365 return null;
366 }
367
368 // Setup progress bar.
369 final String displayName = Bundle.ExtractActionHelper_progress_extracting(destName);
370 progress = ProgressHandle.createHandle(displayName, new Cancellable() {
371 @Override
372 public boolean cancel() {
373 if (progress != null) {
374 progress.setDisplayName(
375 NbBundle.getMessage(this.getClass(), "ExtractActionHelper.progress.cancellingExtraction", displayName));
376 }
377 return ExtractActionHelper.FileExtracter.this.cancel(true);
378 }
379 });
380 progress.start();
381 progress.switchToIndeterminate();
382
383 // Do the extraction tasks.
384 for (FileExtractionTask task : this.extractionTasks) {
385 progress.progress(Bundle.ExtractActionHelper_progress_fileExtracting(task.destination.getName()));
386 UIExtractionVisitor.writeContent(task.source, task.destination, null, this);
387 }
388
389 return null;
390 }
391
392 @Override
393 protected void done() {
394 boolean msgDisplayed = false;
395 try {
396 super.get();
397 } catch (InterruptedException | ExecutionException ex) {
398 logger.log(Level.SEVERE, "Fatal error during file extraction", ex); //NON-NLS
400 NbBundle.getMessage(this.getClass(), "ExtractActionHelper.done.notifyMsg.extractErr", ex.getMessage()));
401 msgDisplayed = true;
402 } finally {
403 progress.finish();
404 if (!this.isCancelled() && !msgDisplayed) {
406 NbBundle.getMessage(this.getClass(), "ExtractActionHelper.done.notifyMsg.fileExtr.text"));
407 }
408 }
409 }
410
419 /*
420 * private int calculateProgressBarWorkUnits(AbstractFile file) { int
421 * workUnits = 0; if (file.isFile()) { workUnits += file.getSize(); }
422 * else { try { for (Content child : file.getChildren()) { if (child
423 * instanceof AbstractFile) { workUnits +=
424 * calculateProgressBarWorkUnits((AbstractFile) child); } } } catch
425 * (TskCoreException ex) { logger.log(Level.SEVERE, "Could not get
426 * children of content", ex); //NON-NLS } } return workUnits; }
427 */
428 }
429}
static String escapeFileName(String fileName)
static boolean deleteFileDir(File path)
Definition FileUtil.java:87
synchronized static Logger getLogger(String name)
Definition Logger.java:124
ExtractFscContentVisitor(java.io.File dest, ProgressHandle progress, SwingWorker< T, V > worker, boolean source)
static boolean isDotDirectory(AbstractFile dir)
ExtractFscContentVisitor< T, V > getChildVisitor(File childFile, ProgressHandle progress, SwingWorker< T, V > worker)
void writeFile(Content file, File dest, ProgressHandle progress, SwingWorker< T, V > worker, boolean source)
void extract(ActionEvent event, Collection<? extends AbstractFile > selectedFiles)
void extractFile(ActionEvent event, AbstractFile selectedFile)
void runExtractionTasks(ActionEvent event, List< FileExtractionTask > fileExtractionTasks, String destName)
void extractFiles(ActionEvent event, Collection<? extends AbstractFile > selectedFiles)

Copyright © 2012-2024 Sleuth Kit Labs. Generated on:
This work is licensed under a Creative Commons Attribution-Share Alike 3.0 United States License.