Autopsy  4.15.0
Graphical digital forensics platform for The Sleuth Kit and other tools.
IngestTasksScheduler.java
Go to the documentation of this file.
1 /*
2  * Autopsy Forensic Browser
3  *
4  * Copyright 2012-2018 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 file 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  */
19 package org.sleuthkit.autopsy.ingest;
20 
21 import java.io.Serializable;
22 import java.util.ArrayList;
23 import java.util.Collection;
24 import java.util.Collections;
25 import java.util.Comparator;
26 import java.util.Deque;
27 import java.util.Iterator;
28 import java.util.LinkedList;
29 import java.util.List;
30 import java.util.TreeSet;
31 import java.util.concurrent.BlockingDeque;
32 import java.util.concurrent.LinkedBlockingDeque;
33 import java.util.logging.Level;
34 import java.util.regex.Matcher;
35 import java.util.regex.Pattern;
36 import javax.annotation.concurrent.GuardedBy;
37 import javax.annotation.concurrent.ThreadSafe;
39 import org.sleuthkit.datamodel.AbstractFile;
40 import org.sleuthkit.datamodel.Content;
41 import org.sleuthkit.datamodel.FileSystem;
42 import org.sleuthkit.datamodel.TskCoreException;
43 import org.sleuthkit.datamodel.TskData;
44 
49 @ThreadSafe
50 final class IngestTasksScheduler {
51 
52  private static final int FAT_NTFS_FLAGS = TskData.TSK_FS_TYPE_ENUM.TSK_FS_TYPE_FAT12.getValue() | TskData.TSK_FS_TYPE_ENUM.TSK_FS_TYPE_FAT16.getValue() | TskData.TSK_FS_TYPE_ENUM.TSK_FS_TYPE_FAT32.getValue() | TskData.TSK_FS_TYPE_ENUM.TSK_FS_TYPE_NTFS.getValue();
53  private static final Logger logger = Logger.getLogger(IngestTasksScheduler.class.getName());
54  @GuardedBy("IngestTasksScheduler.this")
55  private static IngestTasksScheduler instance;
56  private final IngestTaskTrackingQueue dataSourceIngestThreadQueue;
57  @GuardedBy("this")
58  private final TreeSet<FileIngestTask> rootFileTaskQueue;
59  @GuardedBy("this")
60  private final Deque<FileIngestTask> pendingFileTaskQueue;
61  private final IngestTaskTrackingQueue fileIngestThreadsQueue;
62 
68  synchronized static IngestTasksScheduler getInstance() {
69  if (IngestTasksScheduler.instance == null) {
70  IngestTasksScheduler.instance = new IngestTasksScheduler();
71  }
72  return IngestTasksScheduler.instance;
73  }
74 
80  private IngestTasksScheduler() {
81  this.dataSourceIngestThreadQueue = new IngestTaskTrackingQueue();
82  this.rootFileTaskQueue = new TreeSet<>(new RootDirectoryTaskComparator());
83  this.pendingFileTaskQueue = new LinkedList<>();
84  this.fileIngestThreadsQueue = new IngestTaskTrackingQueue();
85  }
86 
93  BlockingIngestTaskQueue getDataSourceIngestTaskQueue() {
94  return this.dataSourceIngestThreadQueue;
95  }
96 
103  BlockingIngestTaskQueue getFileIngestTaskQueue() {
104  return this.fileIngestThreadsQueue;
105  }
106 
113  synchronized void scheduleIngestTasks(DataSourceIngestJob job) {
114  if (!job.isCancelled()) {
115  /*
116  * Scheduling of both the data source ingest task and the initial
117  * file ingest tasks for a job must be an atomic operation.
118  * Otherwise, the data source task might be completed before the
119  * file tasks are scheduled, resulting in a potential false positive
120  * when another thread checks whether or not all the tasks for the
121  * job are completed.
122  */
123  this.scheduleDataSourceIngestTask(job);
124  this.scheduleFileIngestTasks(job, Collections.emptyList());
125  }
126  }
127 
133  synchronized void scheduleDataSourceIngestTask(DataSourceIngestJob job) {
134  if (!job.isCancelled()) {
135  DataSourceIngestTask task = new DataSourceIngestTask(job);
136  try {
137  this.dataSourceIngestThreadQueue.putLast(task);
138  } catch (InterruptedException ex) {
139  IngestTasksScheduler.logger.log(Level.INFO, String.format("Ingest tasks scheduler interrupted while blocked adding a task to the data source level ingest task queue (jobId={%d)", job.getId()), ex);
140  Thread.currentThread().interrupt();
141  }
142  }
143  }
144 
153  synchronized void scheduleFileIngestTasks(DataSourceIngestJob job, Collection<AbstractFile> files) {
154  if (!job.isCancelled()) {
155  Collection<AbstractFile> candidateFiles;
156  if (files.isEmpty()) {
157  candidateFiles = getTopLevelFiles(job.getDataSource());
158  } else {
159  candidateFiles = files;
160  }
161  for (AbstractFile file : candidateFiles) {
162  FileIngestTask task = new FileIngestTask(job, file);
163  if (IngestTasksScheduler.shouldEnqueueFileTask(task)) {
164  this.rootFileTaskQueue.add(task);
165  }
166  }
167  shuffleFileTaskQueues();
168  }
169  }
170 
179  synchronized void fastTrackFileIngestTasks(DataSourceIngestJob job, Collection<AbstractFile> files) {
180  if (!job.isCancelled()) {
181  /*
182  * Put the files directly into the queue for the file ingest
183  * threads, if they pass the file filter for the job. The files are
184  * added to the queue for the ingest threads BEFORE the other queued
185  * tasks because the use case for this method is scheduling new
186  * carved or derived files from a higher priority task that is
187  * already in progress.
188  */
189  for (AbstractFile file : files) {
190  FileIngestTask fileTask = new FileIngestTask(job, file);
191  if (shouldEnqueueFileTask(fileTask)) {
192  try {
193  this.fileIngestThreadsQueue.putFirst(fileTask);
194  } catch (InterruptedException ex) {
195  IngestTasksScheduler.logger.log(Level.INFO, String.format("Ingest tasks scheduler interrupted while scheduling file level ingest tasks (jobId={%d)", job.getId()), ex);
196  Thread.currentThread().interrupt();
197  return;
198  }
199  }
200  }
201  }
202  }
203 
210  synchronized void notifyTaskCompleted(DataSourceIngestTask task) {
211  this.dataSourceIngestThreadQueue.taskCompleted(task);
212  }
213 
220  synchronized void notifyTaskCompleted(FileIngestTask task) {
221  this.fileIngestThreadsQueue.taskCompleted(task);
222  shuffleFileTaskQueues();
223  }
224 
233  synchronized boolean tasksForJobAreCompleted(DataSourceIngestJob job) {
234  long jobId = job.getId();
235  return !(this.dataSourceIngestThreadQueue.hasTasksForJob(jobId)
236  || hasTasksForJob(this.rootFileTaskQueue, jobId)
237  || hasTasksForJob(this.pendingFileTaskQueue, jobId)
238  || this.fileIngestThreadsQueue.hasTasksForJob(jobId));
239  }
240 
248  synchronized void cancelPendingTasksForIngestJob(DataSourceIngestJob job) {
249  long jobId = job.getId();
250  IngestTasksScheduler.removeTasksForJob(this.rootFileTaskQueue, jobId);
251  IngestTasksScheduler.removeTasksForJob(this.pendingFileTaskQueue, jobId);
252  }
253 
263  private static List<AbstractFile> getTopLevelFiles(Content dataSource) {
264  List<AbstractFile> topLevelFiles = new ArrayList<>();
265  Collection<AbstractFile> rootObjects = dataSource.accept(new GetRootDirectoryVisitor());
266  if (rootObjects.isEmpty() && dataSource instanceof AbstractFile) {
267  // The data source is itself a file to be processed.
268  topLevelFiles.add((AbstractFile) dataSource);
269  } else {
270  for (AbstractFile root : rootObjects) {
271  List<Content> children;
272  try {
273  children = root.getChildren();
274  if (children.isEmpty()) {
275  // Add the root object itself, it could be an unallocated
276  // space file, or a child of a volume or an image.
277  topLevelFiles.add(root);
278  } else {
279  // The root object is a file system root directory, get
280  // the files within it.
281  for (Content child : children) {
282  if (child instanceof AbstractFile) {
283  topLevelFiles.add((AbstractFile) child);
284  }
285  }
286  }
287  } catch (TskCoreException ex) {
288  logger.log(Level.WARNING, "Could not get children of root to enqueue: " + root.getId() + ": " + root.getName(), ex); //NON-NLS
289  }
290  }
291  }
292  return topLevelFiles;
293  }
294 
325  synchronized private void shuffleFileTaskQueues() {
326  while (this.fileIngestThreadsQueue.isEmpty()) {
327  /*
328  * If the pending file task queue is empty, move the highest
329  * priority root file task, if there is one, into it.
330  */
331  if (this.pendingFileTaskQueue.isEmpty()) {
332  final FileIngestTask rootTask = this.rootFileTaskQueue.pollFirst();
333  if (rootTask != null) {
334  this.pendingFileTaskQueue.addLast(rootTask);
335  }
336  }
337 
338  /*
339  * Try to move the next task from the pending task queue into the
340  * queue for the file ingest threads, if it passes the filter for
341  * the job.
342  */
343  final FileIngestTask pendingTask = this.pendingFileTaskQueue.pollFirst();
344  if (pendingTask == null) {
345  return;
346  }
347  if (shouldEnqueueFileTask(pendingTask)) {
348  try {
349  /*
350  * The task is added to the queue for the ingest threads
351  * AFTER the higher priority tasks that preceded it.
352  */
353  this.fileIngestThreadsQueue.putLast(pendingTask);
354  } catch (InterruptedException ex) {
355  IngestTasksScheduler.logger.log(Level.INFO, "Ingest tasks scheduler interrupted while blocked adding a task to the file level ingest task queue", ex);
356  Thread.currentThread().interrupt();
357  return;
358  }
359  }
360 
361  /*
362  * If the task that was just queued for the file ingest threads has
363  * children, try to queue tasks for the children. Each child task
364  * will go into either the directory queue if it has children of its
365  * own, or into the queue for the file ingest threads, if it passes
366  * the filter for the job.
367  */
368  final AbstractFile file = pendingTask.getFile();
369  try {
370  for (Content child : file.getChildren()) {
371  if (child instanceof AbstractFile) {
372  AbstractFile childFile = (AbstractFile) child;
373  FileIngestTask childTask = new FileIngestTask(pendingTask.getIngestJob(), childFile);
374  if (childFile.hasChildren()) {
375  this.pendingFileTaskQueue.add(childTask);
376  } else if (shouldEnqueueFileTask(childTask)) {
377  try {
378  this.fileIngestThreadsQueue.putLast(childTask);
379  } catch (InterruptedException ex) {
380  IngestTasksScheduler.logger.log(Level.INFO, "Ingest tasks scheduler interrupted while blocked adding a task to the file level ingest task queue", ex);
381  Thread.currentThread().interrupt();
382  return;
383  }
384  }
385  }
386  }
387  } catch (TskCoreException ex) {
388  logger.log(Level.SEVERE, String.format("Error getting the children of %s (objId=%d)", file.getName(), file.getId()), ex); //NON-NLS
389  }
390  }
391  }
392 
402  private static boolean shouldEnqueueFileTask(final FileIngestTask task) {
403  final AbstractFile file = task.getFile();
404 
405  // Skip the task if the file is actually the pseudo-file for the parent
406  // or current directory.
407  String fileName = file.getName();
408 
409  if (fileName.equals(".") || fileName.equals("..")) {
410  return false;
411  }
412 
413  /*
414  * Ensures that all directories, files which are members of the ingest
415  * file filter, and unallocated blocks (when processUnallocated is
416  * enabled) all continue to be processed. AbstractFiles which do not
417  * meet one of these criteria will be skipped.
418  *
419  * An additional check to see if unallocated space should be processed
420  * is part of the FilesSet.fileIsMemberOf() method.
421  *
422  * This code may need to be updated when
423  * TSK_DB_FILES_TYPE_ENUM.UNUSED_BLOCKS comes into use by Autopsy.
424  */
425  if (!file.isDir() && !shouldBeCarved(task) && !fileAcceptedByFilter(task)) {
426  return false;
427  }
428 
429  // Skip the task if the file is one of a select group of special, large
430  // NTFS or FAT file system files.
431  if (file instanceof org.sleuthkit.datamodel.File) {
432  final org.sleuthkit.datamodel.File f = (org.sleuthkit.datamodel.File) file;
433 
434  // Get the type of the file system, if any, that owns the file.
435  TskData.TSK_FS_TYPE_ENUM fsType = TskData.TSK_FS_TYPE_ENUM.TSK_FS_TYPE_UNSUPP;
436  try {
437  FileSystem fs = f.getFileSystem();
438  if (fs != null) {
439  fsType = fs.getFsType();
440  }
441  } catch (TskCoreException ex) {
442  logger.log(Level.SEVERE, "Error querying file system for " + f, ex); //NON-NLS
443  }
444 
445  // If the file system is not NTFS or FAT, don't skip the file.
446  if ((fsType.getValue() & FAT_NTFS_FLAGS) == 0) {
447  return true;
448  }
449 
450  // Find out whether the file is in a root directory.
451  boolean isInRootDir = false;
452  try {
453  AbstractFile parent = f.getParentDirectory();
454  if (parent == null) {
455  isInRootDir = true;
456  } else {
457  isInRootDir = parent.isRoot();
458  }
459  } catch (TskCoreException ex) {
460  logger.log(Level.WARNING, "Error querying parent directory for" + f.getName(), ex); //NON-NLS
461  }
462 
463  // If the file is in the root directory of an NTFS or FAT file
464  // system, check its meta-address and check its name for the '$'
465  // character and a ':' character (not a default attribute).
466  if (isInRootDir && f.getMetaAddr() < 32) {
467  String name = f.getName();
468  if (name.length() > 0 && name.charAt(0) == '$' && name.contains(":")) {
469  return false;
470  }
471  }
472  }
473 
474  return true;
475  }
476 
485  private static boolean shouldBeCarved(final FileIngestTask task) {
486  return task.getIngestJob().shouldProcessUnallocatedSpace() && task.getFile().getType().equals(TskData.TSK_DB_FILES_TYPE_ENUM.UNALLOC_BLOCKS);
487  }
488 
497  private static boolean fileAcceptedByFilter(final FileIngestTask task) {
498  return !(task.getIngestJob().getFileIngestFilter().fileIsMemberOf(task.getFile()) == null);
499  }
500 
510  synchronized private static boolean hasTasksForJob(Collection<? extends IngestTask> tasks, long jobId) {
511  for (IngestTask task : tasks) {
512  if (task.getIngestJob().getId() == jobId) {
513  return true;
514  }
515  }
516  return false;
517  }
518 
526  private static void removeTasksForJob(Collection<? extends IngestTask> tasks, long jobId) {
527  Iterator<? extends IngestTask> iterator = tasks.iterator();
528  while (iterator.hasNext()) {
529  IngestTask task = iterator.next();
530  if (task.getIngestJob().getId() == jobId) {
531  iterator.remove();
532  }
533  }
534  }
535 
544  private static int countTasksForJob(Collection<? extends IngestTask> queue, long jobId) {
545  int count = 0;
546  for (IngestTask task : queue) {
547  if (task.getIngestJob().getId() == jobId) {
548  count++;
549  }
550  }
551  return count;
552  }
553 
562  synchronized IngestJobTasksSnapshot getTasksSnapshotForJob(long jobId) {
563  return new IngestJobTasksSnapshot(jobId, this.dataSourceIngestThreadQueue.countQueuedTasksForJob(jobId),
564  countTasksForJob(this.rootFileTaskQueue, jobId),
565  countTasksForJob(this.pendingFileTaskQueue, jobId),
566  this.fileIngestThreadsQueue.countQueuedTasksForJob(jobId),
567  this.dataSourceIngestThreadQueue.countRunningTasksForJob(jobId) + this.fileIngestThreadsQueue.countRunningTasksForJob(jobId));
568  }
569 
574  private static class RootDirectoryTaskComparator implements Comparator<FileIngestTask> {
575 
576  @Override
577  public int compare(FileIngestTask q1, FileIngestTask q2) {
578  AbstractFilePriority.Priority p1 = AbstractFilePriority.getPriority(q1.getFile());
579  AbstractFilePriority.Priority p2 = AbstractFilePriority.getPriority(q2.getFile());
580  if (p1 == p2) {
581  return (int) (q2.getFile().getId() - q1.getFile().getId());
582  } else {
583  return p2.ordinal() - p1.ordinal();
584  }
585  }
586 
591  private static class AbstractFilePriority {
592 
594  }
595 
596  enum Priority {
597 
598  LAST, LOW, MEDIUM, HIGH
599  }
600 
601  static final List<Pattern> LAST_PRI_PATHS = new ArrayList<>();
602 
603  static final List<Pattern> LOW_PRI_PATHS = new ArrayList<>();
604 
605  static final List<Pattern> MEDIUM_PRI_PATHS = new ArrayList<>();
606 
607  static final List<Pattern> HIGH_PRI_PATHS = new ArrayList<>();
608 
609  /*
610  * prioritize root directory folders based on the assumption that we
611  * are looking for user content. Other types of investigations may
612  * want different priorities.
613  */
614  static /*
615  * prioritize root directory folders based on the assumption that we
616  * are looking for user content. Other types of investigations may
617  * want different priorities.
618  */ {
619  // these files have no structure, so they go last
620  //unalloc files are handled as virtual files in getPriority()
621  //LAST_PRI_PATHS.schedule(Pattern.compile("^\\$Unalloc", Pattern.CASE_INSENSITIVE));
622  //LAST_PRI_PATHS.schedule(Pattern.compile("^\\Unalloc", Pattern.CASE_INSENSITIVE));
623  LAST_PRI_PATHS.add(Pattern.compile("^pagefile", Pattern.CASE_INSENSITIVE));
624  LAST_PRI_PATHS.add(Pattern.compile("^hiberfil", Pattern.CASE_INSENSITIVE));
625  // orphan files are often corrupt and windows does not typically have
626  // user content, so put them towards the bottom
627  LOW_PRI_PATHS.add(Pattern.compile("^\\$OrphanFiles", Pattern.CASE_INSENSITIVE));
628  LOW_PRI_PATHS.add(Pattern.compile("^Windows", Pattern.CASE_INSENSITIVE));
629  // all other files go into the medium category too
630  MEDIUM_PRI_PATHS.add(Pattern.compile("^Program Files", Pattern.CASE_INSENSITIVE));
631  // user content is top priority
632  HIGH_PRI_PATHS.add(Pattern.compile("^Users", Pattern.CASE_INSENSITIVE));
633  HIGH_PRI_PATHS.add(Pattern.compile("^Documents and Settings", Pattern.CASE_INSENSITIVE));
634  HIGH_PRI_PATHS.add(Pattern.compile("^home", Pattern.CASE_INSENSITIVE));
635  HIGH_PRI_PATHS.add(Pattern.compile("^ProgramData", Pattern.CASE_INSENSITIVE));
636  }
637 
645  static AbstractFilePriority.Priority getPriority(final AbstractFile abstractFile) {
646  if (!abstractFile.getType().equals(TskData.TSK_DB_FILES_TYPE_ENUM.FS)) {
647  //quickly filter out unstructured content
648  //non-fs virtual files and dirs, such as representing unalloc space
649  return AbstractFilePriority.Priority.LAST;
650  }
651  //determine the fs files priority by name
652  final String path = abstractFile.getName();
653  if (path == null) {
654  return AbstractFilePriority.Priority.MEDIUM;
655  }
656  for (Pattern p : HIGH_PRI_PATHS) {
657  Matcher m = p.matcher(path);
658  if (m.find()) {
659  return AbstractFilePriority.Priority.HIGH;
660  }
661  }
662  for (Pattern p : MEDIUM_PRI_PATHS) {
663  Matcher m = p.matcher(path);
664  if (m.find()) {
665  return AbstractFilePriority.Priority.MEDIUM;
666  }
667  }
668  for (Pattern p : LOW_PRI_PATHS) {
669  Matcher m = p.matcher(path);
670  if (m.find()) {
671  return AbstractFilePriority.Priority.LOW;
672  }
673  }
674  for (Pattern p : LAST_PRI_PATHS) {
675  Matcher m = p.matcher(path);
676  if (m.find()) {
677  return AbstractFilePriority.Priority.LAST;
678  }
679  }
680  //default is medium
681  return AbstractFilePriority.Priority.MEDIUM;
682  }
683  }
684  }
685 
690  @ThreadSafe
691  private class IngestTaskTrackingQueue implements BlockingIngestTaskQueue {
692 
693  private final BlockingDeque<IngestTask> taskQueue = new LinkedBlockingDeque<>();
694  @GuardedBy("this")
695  private final List<IngestTask> queuedTasks = new LinkedList<>();
696  @GuardedBy("this")
697  private final List<IngestTask> tasksInProgress = new LinkedList<>();
698 
709  void putFirst(IngestTask task) throws InterruptedException {
710  synchronized (this) {
711  this.queuedTasks.add(task);
712  }
713  try {
714  this.taskQueue.putFirst(task);
715  } catch (InterruptedException ex) {
716  synchronized (this) {
717  this.queuedTasks.remove(task);
718  }
719  throw ex;
720  }
721  }
722 
733  void putLast(IngestTask task) throws InterruptedException {
734  synchronized (this) {
735  this.queuedTasks.add(task);
736  }
737  try {
738  this.taskQueue.putLast(task);
739  } catch (InterruptedException ex) {
740  synchronized (this) {
741  this.queuedTasks.remove(task);
742  }
743  throw ex;
744  }
745  }
746 
757  @Override
758  public IngestTask getNextTask() throws InterruptedException {
759  IngestTask task = taskQueue.takeFirst();
760  synchronized (this) {
761  this.queuedTasks.remove(task);
762  this.tasksInProgress.add(task);
763  }
764  return task;
765  }
766 
772  boolean isEmpty() {
773  synchronized (this) {
774  return this.queuedTasks.isEmpty();
775  }
776  }
777 
784  void taskCompleted(IngestTask task) {
785  synchronized (this) {
786  this.tasksInProgress.remove(task);
787  }
788  }
789 
798  boolean hasTasksForJob(long jobId) {
799  synchronized (this) {
800  return IngestTasksScheduler.hasTasksForJob(this.queuedTasks, jobId) || IngestTasksScheduler.hasTasksForJob(this.tasksInProgress, jobId);
801  }
802  }
803 
812  int countQueuedTasksForJob(long jobId) {
813  synchronized (this) {
814  return IngestTasksScheduler.countTasksForJob(this.queuedTasks, jobId);
815  }
816  }
817 
826  int countRunningTasksForJob(long jobId) {
827  synchronized (this) {
828  return IngestTasksScheduler.countTasksForJob(this.tasksInProgress, jobId);
829  }
830  }
831 
832  }
833 
837  static final class IngestJobTasksSnapshot implements Serializable {
838 
839  private static final long serialVersionUID = 1L;
840  private final long jobId;
841  private final long dsQueueSize;
842  private final long rootQueueSize;
843  private final long dirQueueSize;
844  private final long fileQueueSize;
845  private final long runningListSize;
846 
852  IngestJobTasksSnapshot(long jobId, long dsQueueSize, long rootQueueSize, long dirQueueSize, long fileQueueSize, long runningListSize) {
853  this.jobId = jobId;
854  this.dsQueueSize = dsQueueSize;
855  this.rootQueueSize = rootQueueSize;
856  this.dirQueueSize = dirQueueSize;
857  this.fileQueueSize = fileQueueSize;
858  this.runningListSize = runningListSize;
859  }
860 
867  long getJobId() {
868  return jobId;
869  }
870 
877  long getRootQueueSize() {
878  return rootQueueSize;
879  }
880 
887  long getDirectoryTasksQueueSize() {
888  return dirQueueSize;
889  }
890 
891  long getFileQueueSize() {
892  return fileQueueSize;
893  }
894 
895  long getDsQueueSize() {
896  return dsQueueSize;
897  }
898 
899  long getRunningListSize() {
900  return runningListSize;
901  }
902 
903  }
904 
905 }

Copyright © 2012-2020 Basis Technology. Generated on: Mon Jul 6 2020
This work is licensed under a Creative Commons Attribution-Share Alike 3.0 United States License.