Autopsy  4.17.0
Graphical digital forensics platform for The Sleuth Kit and other tools.
ExtractRecycleBin.java
Go to the documentation of this file.
1 /*
2  *
3  * Autopsy Forensic Browser
4  *
5  * Copyright 2019 Basis Technology Corp.
6  *
7  * Copyright 2012 42six Solutions.
8  * Contact: aebadirad <at> 42six <dot> com
9  * Project Contact/Architect: carrier <at> sleuthkit <dot> org
10  *
11  * Licensed under the Apache License, Version 2.0 (the "License");
12  * you may not use this file except in compliance with the License.
13  * You may obtain a copy of the License at
14  *
15  * http://www.apache.org/licenses/LICENSE-2.0
16  *
17  * Unless required by applicable law or agreed to in writing, software
18  * distributed under the License is distributed on an "AS IS" BASIS,
19  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
20  * See the License for the specific language governing permissions and
21  * limitations under the License.
22  */
23 package org.sleuthkit.autopsy.recentactivity;
24 
25 import java.io.File;
26 import java.io.IOException;
27 import java.nio.ByteBuffer;
28 import java.nio.ByteOrder;
29 import java.nio.BufferUnderflowException;
30 import java.nio.file.Files;
31 import java.nio.file.Path;
32 import java.nio.file.Paths;
33 import java.util.ArrayList;
34 import java.util.Arrays;
35 import java.util.HashMap;
36 import java.util.List;
37 import java.util.Map;
38 import java.util.logging.Level;
39 import org.joda.time.Instant;
40 import org.openide.util.NbBundle.Messages;
47 import org.sleuthkit.datamodel.AbstractFile;
48 import org.sleuthkit.datamodel.Blackboard.BlackboardException;
49 import org.sleuthkit.datamodel.BlackboardArtifact;
50 import static org.sleuthkit.datamodel.BlackboardArtifact.ARTIFACT_TYPE.TSK_OS_ACCOUNT;
51 import org.sleuthkit.datamodel.BlackboardAttribute;
52 import static org.sleuthkit.datamodel.BlackboardAttribute.ATTRIBUTE_TYPE.TSK_DATETIME_DELETED;
53 import static org.sleuthkit.datamodel.BlackboardAttribute.ATTRIBUTE_TYPE.TSK_PATH;
54 import static org.sleuthkit.datamodel.BlackboardAttribute.ATTRIBUTE_TYPE.TSK_USER_ID;
55 import static org.sleuthkit.datamodel.BlackboardAttribute.ATTRIBUTE_TYPE.TSK_USER_NAME;
56 import org.sleuthkit.datamodel.Content;
57 import org.sleuthkit.datamodel.FsContent;
58 import org.sleuthkit.datamodel.SleuthkitCase;
59 import org.sleuthkit.datamodel.TskCoreException;
60 import org.sleuthkit.datamodel.TskData;
61 import org.sleuthkit.datamodel.TskDataException;
62 
70 final class ExtractRecycleBin extends Extract {
71 
72  private static final Logger logger = Logger.getLogger(ExtractRecycleBin.class.getName());
73 
74  private static final String RECYCLE_BIN_ARTIFACT_NAME = "TSK_RECYCLE_BIN"; //NON-NLS
75 
76  private static final String RECYCLE_BIN_DIR_NAME = "$RECYCLE.BIN"; //NON-NLS
77 
78  private static final int V1_FILE_NAME_OFFSET = 24;
79  private static final int V2_FILE_NAME_OFFSET = 28;
80 
81  @Messages({
82  "ExtractRecycleBin_module_name=Recycle Bin"
83  })
84  ExtractRecycleBin() {
85  this.moduleName = Bundle.ExtractRecycleBin_module_name();
86  }
87 
88  @Override
89  void process(Content dataSource, IngestJobContext context, DataSourceIngestModuleProgress progressBar) {
90  // At this time it was decided that we would not include TSK_RECYCLE_BIN
91  // in the default list of BlackboardArtifact types.
92  try {
93  createRecycleBinArtifactType();
94  } catch (TskCoreException ex) {
95  logger.log(Level.WARNING, String.format("%s may not have been created.", RECYCLE_BIN_ARTIFACT_NAME), ex);
96  }
97 
98  BlackboardArtifact.Type recycleBinArtifactType;
99 
100  try {
101  recycleBinArtifactType = tskCase.getArtifactType(RECYCLE_BIN_ARTIFACT_NAME);
102  } catch (TskCoreException ex) {
103  logger.log(Level.WARNING, String.format("Unable to retrive custom artifact type %s", RECYCLE_BIN_ARTIFACT_NAME), ex); // NON-NLS
104  // If this doesn't work bail.
105  return;
106  }
107 
108  // map SIDs to user names so that we can include that in the artifact
109  Map<String, String> userNameMap;
110  try {
111  userNameMap = makeUserNameMap(dataSource);
112  } catch (TskCoreException ex) {
113  logger.log(Level.WARNING, "Unable to create OS Account user name map", ex);
114  // This is not the end of the world we will just continue without
115  // user names
116  userNameMap = new HashMap<>();
117  }
118 
119  FileManager fileManager = Case.getCurrentCase().getServices().getFileManager();
120 
121  // Collect all of the $R files so that we can later easily map them to corresponding $I file
122  Map<String, List<AbstractFile>> rFileMap;
123  try {
124  rFileMap = makeRFileMap(dataSource);
125  } catch (TskCoreException ex) {
126  logger.log(Level.WARNING, String.format("Unable to create $R file map for dataSource: %s", dataSource.getName()), ex);
127  return; // No $R files, no need to continue;
128  }
129 
130  // Get the $I files
131  List<AbstractFile> iFiles;
132  try {
133  iFiles = fileManager.findFiles(dataSource, "$I%", RECYCLE_BIN_DIR_NAME); //NON-NLS
134  } catch (TskCoreException ex) {
135  logger.log(Level.WARNING, "Unable to find recycle bin I files.", ex); //NON-NLS
136  return; // No need to continue
137  }
138 
139  String tempRARecycleBinPath = RAImageIngestModule.getRATempPath(Case.getCurrentCase(), "recyclebin"); //NON-NLS
140 
141  // cycle through the $I files and process each.
142  for (AbstractFile iFile : iFiles) {
143 
144  if (context.dataSourceIngestIsCancelled()) {
145  return;
146  }
147 
148  processIFile(context, recycleBinArtifactType, iFile, userNameMap, rFileMap, tempRARecycleBinPath);
149  }
150 
151  (new File(tempRARecycleBinPath)).delete();
152  }
153 
164  private void processIFile(IngestJobContext context, BlackboardArtifact.Type recycleBinArtifactType, AbstractFile iFile, Map<String, String> userNameMap, Map<String, List<AbstractFile>> rFileMap, String tempRARecycleBinPath) {
165  String tempFilePath = tempRARecycleBinPath + File.separator + Instant.now().getMillis() + iFile.getName();
166  try {
167  try {
168  ContentUtils.writeToFile(iFile, new File(tempFilePath));
169  } catch (IOException ex) {
170  logger.log(Level.WARNING, String.format("Unable to write %s to temp directory. File name: %s", iFile.getName(), tempFilePath), ex); //NON-NLS
171  // if we cannot make a copy of the $I file for later processing
172  // move onto the next file
173  return;
174  }
175 
176  // get the original name, dates, etc. from the $I file
177  RecycledFileMetaData metaData;
178  try {
179  metaData = parseIFile(tempFilePath);
180  } catch (IOException ex) {
181  logger.log(Level.WARNING, String.format("Unable to parse iFile %s", iFile.getParentPath() + iFile.getName()), ex); //NON-NLS
182  // Unable to parse the $I file move onto the next file
183  return;
184  }
185 
186  // each user has its own Recyle Bin folder. Figure out the user name based on its name .
187  String userID = getUserIDFromPath(iFile.getParentPath());
188  String userName = "";
189  if (!userID.isEmpty()) {
190  userName = userNameMap.get(userID);
191  } else {
192  // If the iFile doesn't have a user ID in its parent
193  // directory structure then it is not from the recyle bin
194  return;
195  }
196 
197  // get the corresponding $R file, which is in the same folder and has the file content
198  String rFileName = iFile.getName().replace("$I", "$R"); //NON-NLS
199  List<AbstractFile> rFiles = rFileMap.get(rFileName);
200  if (rFiles == null) {
201  return;
202  }
203  SleuthkitCase skCase = Case.getCurrentCase().getSleuthkitCase();
204  for (AbstractFile rFile : rFiles) {
205  if (context.dataSourceIngestIsCancelled()) {
206  return;
207  }
208 
209  if (iFile.getParentPath().equals(rFile.getParentPath())
210  && iFile.getMetaFlagsAsString().equals(rFile.getMetaFlagsAsString())) {
211  try {
212  postArtifact(createArtifact(rFile, recycleBinArtifactType, metaData.getFullWindowsPath(), userName, metaData.getDeletedTimeStamp()));
213 
214  // If we are processing a disk image, we will also make a deleted file entry so that the user
215  // sees the deleted file in its original folder. We re-use the metadata address so that the user
216  // can see the content.
217  if (rFile instanceof FsContent) {
218  // if the user deleted a folder, then we need to recusively go into it. Note the contents of the $R folder
219  // do not have corresponding $I files anymore. Only the $R folder does.
220  if (rFile.isDir()) {
221  AbstractFile directory = getOrMakeFolder(Case.getCurrentCase().getSleuthkitCase(), (FsContent) rFile, metaData.getFullWindowsPath());
222  popuplateDeletedDirectory(Case.getCurrentCase().getSleuthkitCase(), directory, rFile.getChildren(), metaData.getFullWindowsPath(), metaData.getDeletedTimeStamp());
223 
224  } else {
225  AbstractFile folder = getOrMakeFolder(Case.getCurrentCase().getSleuthkitCase(), (FsContent) rFile.getParent(), Paths.get(metaData.getFullWindowsPath()).getParent().toString());
226  addFileSystemFile(skCase, (FsContent)rFile, folder, Paths.get(metaData.getFullWindowsPath()).getFileName().toString(), metaData.getDeletedTimeStamp());
227  }
228  }
229  } catch (TskCoreException ex) {
230  logger.log(Level.WARNING, String.format("Unable to add attributes to artifact %s", rFile.getName()), ex); //NON-NLS
231  }
232  }
233  }
234  } finally {
235  (new File(tempFilePath)).delete();
236  }
237  }
238 
253  private void popuplateDeletedDirectory(SleuthkitCase skCase, AbstractFile parentFolder, List<Content> recycledChildren, String parentPath, long deletedTimeStamp) throws TskCoreException {
254  if (recycledChildren == null) {
255  return;
256  }
257 
258  for (Content child : recycledChildren) {
259  if (child instanceof FsContent) {
260  FsContent fsContent = (FsContent) child;
261  if (fsContent.isFile()) {
262  addFileSystemFile(skCase, fsContent, parentFolder, fsContent.getName(), deletedTimeStamp);
263  } else if (fsContent.isDir()) {
264  String newPath = parentPath + "\\" + fsContent.getName();
265  AbstractFile childFolder = getOrMakeFolder(skCase, fsContent, parentPath);
266  popuplateDeletedDirectory(skCase, childFolder, fsContent.getChildren(), newPath, deletedTimeStamp);
267  }
268  }
269  }
270  }
271 
297  private RecycledFileMetaData parseIFile(String iFilePath) throws IOException {
298  try {
299  byte[] allBytes = Files.readAllBytes(Paths.get(iFilePath));
300 
301 
302  ByteBuffer byteBuffer = ByteBuffer.wrap(allBytes);
303  byteBuffer.order(ByteOrder.LITTLE_ENDIAN);
304 
305  long version = byteBuffer.getLong();
306  long fileSize = byteBuffer.getLong();
307  long timestamp = byteBuffer.getLong();
308 
309  // Convert from windows FILETIME to Unix Epoch seconds
310  timestamp = Util.filetimeToMillis(timestamp) / 1000;
311 
312  byte[] stringBytes;
313 
314  if (version == 1) {
315  stringBytes = Arrays.copyOfRange(allBytes, V1_FILE_NAME_OFFSET, allBytes.length);
316  } else {
317  int fileNameLength = byteBuffer.getInt() * 2; //Twice the bytes for unicode
318  stringBytes = Arrays.copyOfRange(allBytes, V2_FILE_NAME_OFFSET, V2_FILE_NAME_OFFSET + fileNameLength);
319  }
320 
321  String fileName = new String(stringBytes, "UTF-16LE"); //NON-NLS
322 
323  return new RecycledFileMetaData(fileSize, timestamp, fileName);
324  } catch (IOException | BufferUnderflowException | IllegalArgumentException | ArrayIndexOutOfBoundsException ex) {
325  throw new IOException("Error parsing $I File, file is corrupt or not a valid I$ file", ex);
326  }
327  }
328 
338  private Map<String, String> makeUserNameMap(Content dataSource) throws TskCoreException {
339  Map<String, String> userNameMap = new HashMap<>();
340 
341  List<BlackboardArtifact> accounts = blackboard.getArtifacts(TSK_OS_ACCOUNT.getTypeID(), dataSource.getId());
342 
343  for (BlackboardArtifact account : accounts) {
344  BlackboardAttribute nameAttribute = getAttributeForArtifact(account, TSK_USER_NAME);
345  BlackboardAttribute idAttribute = getAttributeForArtifact(account, TSK_USER_ID);
346 
347  String userName = nameAttribute != null ? nameAttribute.getDisplayString() : "";
348  String userID = idAttribute != null ? idAttribute.getDisplayString() : "";
349 
350  if (!userID.isEmpty()) {
351  userNameMap.put(userID, userName);
352  }
353  }
354 
355  return userNameMap;
356  }
357 
368  private Map<String, List<AbstractFile>> makeRFileMap(Content dataSource) throws TskCoreException {
369  FileManager fileManager = Case.getCurrentCase().getServices().getFileManager();
370  List<AbstractFile> rFiles = fileManager.findFiles(dataSource, "$R%");
371  Map<String, List<AbstractFile>> fileMap = new HashMap<>();
372 
373  for (AbstractFile rFile : rFiles) {
374  String fileName = rFile.getName();
375  List<AbstractFile> fileList = fileMap.get(fileName);
376 
377  if (fileList == null) {
378  fileList = new ArrayList<>();
379  fileMap.put(fileName, fileList);
380  }
381 
382  fileList.add(rFile);
383  }
384 
385  return fileMap;
386  }
387 
396  private String getUserIDFromPath(String iFileParentPath) {
397  int index = iFileParentPath.indexOf('-') - 1;
398  if (index >= 0) {
399  return (iFileParentPath.substring(index)).replace("/", "");
400  } else {
401  return "";
402  }
403  }
404 
415  private BlackboardAttribute getAttributeForArtifact(BlackboardArtifact artifact, BlackboardAttribute.ATTRIBUTE_TYPE type) throws TskCoreException {
416  return artifact.getAttribute(new BlackboardAttribute.Type(BlackboardAttribute.ATTRIBUTE_TYPE.fromID(type.getTypeID())));
417  }
418 
419  @Messages({
420  "ExtractRecycleBin_Recyle_Bin_Display_Name=Recycle Bin"
421  })
427  private void createRecycleBinArtifactType() throws TskCoreException {
428  try {
429  tskCase.getBlackboard().getOrAddArtifactType(RECYCLE_BIN_ARTIFACT_NAME, Bundle.ExtractRecycleBin_Recyle_Bin_Display_Name()); //NON-NLS
430  } catch (BlackboardException ex) {
431  throw new TskCoreException(String.format("An exception was thrown while defining artifact type %s", RECYCLE_BIN_ARTIFACT_NAME), ex);
432  }
433 
434  }
435 
449  private BlackboardArtifact createArtifact(AbstractFile rFile, BlackboardArtifact.Type type, String fileName, String userName, long dateTime) throws TskCoreException {
450  BlackboardArtifact bba = rFile.newArtifact(type.getTypeID());
451  bba.addAttribute(new BlackboardAttribute(TSK_PATH, getName(), fileName));
452  bba.addAttribute(new BlackboardAttribute(TSK_DATETIME_DELETED, getName(), dateTime));
453  bba.addAttribute(new BlackboardAttribute(TSK_USER_NAME, getName(), userName == null || userName.isEmpty() ? "" : userName));
454  return bba;
455  }
456 
469  private AbstractFile getOrMakeFolder(SleuthkitCase skCase, FsContent dataSource, String path) throws TskCoreException {
470 
471  String parentPath = getParentPath(path);
472  String folderName = getFileName(path);
473 
474  List<AbstractFile> files = null;
475  if (parentPath != null) {
476  if (!parentPath.equals("/")) {
477  parentPath = parentPath + "/";
478  }
479 
480  files = skCase.findAllFilesWhere(String.format("fs_obj_id=%s AND parent_path='%s' AND name='%s'",
481  dataSource.getFileSystemId(), SleuthkitCase.escapeSingleQuotes(parentPath), folderName != null ? SleuthkitCase.escapeSingleQuotes(folderName) : ""));
482  } else {
483  files = skCase.findAllFilesWhere(String.format("fs_obj_id=%s AND parent_path='/' AND name=''", dataSource.getFileSystemId()));
484  }
485 
486  if (files == null || files.isEmpty()) {
487  AbstractFile parent = getOrMakeFolder(skCase, dataSource, parentPath);
488  return skCase.addVirtualDirectory(parent.getId(), folderName);
489  } else {
490  return files.get(0);
491  }
492  }
493 
506  private void addFileSystemFile(SleuthkitCase skCase, FsContent recycleBinFile, Content parentDir, String fileName, long deletedTime) throws TskCoreException {
507  skCase.addFileSystemFile(
508  recycleBinFile.getDataSourceObjectId(),
509  recycleBinFile.getFileSystemId(),
510  fileName,
511  recycleBinFile.getMetaAddr(),
512  (int) recycleBinFile.getMetaSeq(),
513  recycleBinFile.getAttrType(),
514  recycleBinFile.getAttributeId(),
515  TskData.TSK_FS_NAME_FLAG_ENUM.UNALLOC,
516  (short) (TskData.TSK_FS_META_FLAG_ENUM.UNALLOC.getValue() | TskData.TSK_FS_META_FLAG_ENUM.USED.getValue()),
517  recycleBinFile.getSize(),
518  recycleBinFile.getCtime(), recycleBinFile.getCrtime(), recycleBinFile.getAtime(), deletedTime,
519  true, parentDir);
520  }
521 
530  String normalizeFilePath(String pathString) {
531  if (pathString == null || pathString.isEmpty()) {
532  return null;
533  }
534 
535  Path path = Paths.get(pathString);
536  int nameCount = path.getNameCount();
537  if(nameCount > 0) {
538  String rootless = "/" + path.subpath(0, nameCount);
539  return rootless.replace("\\", "/");
540  } else {
541  return "/";
542  }
543  }
544 
554  String getFileName(String filePath) {
555  Path fileNamePath = Paths.get(filePath).getFileName();
556  if (fileNamePath != null) {
557  return fileNamePath.toString();
558  }
559  return filePath;
560  }
561 
569  String getParentPath(String path) {
570  Path parentPath = Paths.get(path).getParent();
571  if (parentPath != null) {
572  return normalizeFilePath(parentPath.toString());
573  }
574  return null;
575  }
576 
580  final class RecycledFileMetaData {
581 
582  private final long fileSize;
583  private final long deletedTimeStamp;
584  private final String fileName;
585 
593  RecycledFileMetaData(Long fileSize, long deletedTimeStamp, String fileName) {
594  this.fileSize = fileSize;
595  this.deletedTimeStamp = deletedTimeStamp;
596  this.fileName = fileName;
597  }
598 
604  long getFileSize() {
605  return fileSize;
606  }
607 
613  long getDeletedTimeStamp() {
614  return deletedTimeStamp;
615  }
616 
623  String getFullWindowsPath() {
624  return fileName.trim();
625  }
626  }
627 }

Copyright © 2012-2021 Basis Technology. Generated on: Tue Jan 19 2021
This work is licensed under a Creative Commons Attribution-Share Alike 3.0 United States License.