diff --git a/DiscImageChef/Commands/Ls.cs b/DiscImageChef/Commands/Ls.cs
new file mode 100644
index 00000000..7ca47ce9
--- /dev/null
+++ b/DiscImageChef/Commands/Ls.cs
@@ -0,0 +1,320 @@
+// /***************************************************************************
+// The Disc Image Chef
+// ----------------------------------------------------------------------------
+//
+// Filename : Ls.cs
+// Version : 1.0
+// Author(s) : Natalia Portillo
+//
+// Component : Component
+//
+// Revision : $Revision$
+// Last change by : $Author$
+// Date : $Date$
+//
+// --[ Description ] ----------------------------------------------------------
+//
+// Description
+//
+// --[ License ] --------------------------------------------------------------
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as
+// published by the Free Software Foundation, either version 3 of the
+// License, or (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program. If not, see .
+//
+// ----------------------------------------------------------------------------
+// Copyright (C) 2011-2015 Claunia.com
+// ****************************************************************************/
+// //$Id$
+using System;
+using System.Collections.Generic;
+using DiscImageChef.Console;
+using DiscImageChef.Filesystems;
+using DiscImageChef.ImagePlugins;
+using DiscImageChef.PartPlugins;
+
+namespace DiscImageChef.Commands
+{
+ public class Ls
+ {
+ public static void doLs(LsOptions options)
+ {
+ DicConsole.DebugWriteLine("Ls command", "--debug={0}", options.Debug);
+ DicConsole.DebugWriteLine("Ls command", "--verbose={0}", options.Verbose);
+ DicConsole.DebugWriteLine("Ls command", "--input={0}", options.InputFile);
+
+ if(!System.IO.File.Exists(options.InputFile))
+ {
+ DicConsole.ErrorWriteLine("Specified file does not exist.");
+ return;
+ }
+
+ PluginBase plugins = new PluginBase();
+ plugins.RegisterAllPlugins();
+
+ List id_plugins;
+ Filesystem _plugin;
+ ImagePlugin _imageFormat;
+ Errno error;
+
+ try
+ {
+ _imageFormat = ImageFormat.Detect(options.InputFile);
+
+ if(_imageFormat == null)
+ {
+ DicConsole.WriteLine("Image format not identified, not proceeding with analysis.");
+ return;
+ }
+ else
+ {
+ if(options.Verbose)
+ DicConsole.VerboseWriteLine("Image format identified by {0} ({1}).", _imageFormat.Name, _imageFormat.PluginUUID);
+ else
+ DicConsole.WriteLine("Image format identified by {0}.", _imageFormat.Name);
+ }
+
+ try
+ {
+ if(!_imageFormat.OpenImage(options.InputFile))
+ {
+ DicConsole.WriteLine("Unable to open image format");
+ DicConsole.WriteLine("No error given");
+ return;
+ }
+
+ DicConsole.DebugWriteLine("Ls command", "Correctly opened image file.");
+ DicConsole.DebugWriteLine("Ls command", "Image without headers is {0} bytes.", _imageFormat.GetImageSize());
+ DicConsole.DebugWriteLine("Ls command", "Image has {0} sectors.", _imageFormat.GetSectors());
+ DicConsole.DebugWriteLine("Ls command", "Image identifies disk type as {0}.", _imageFormat.GetMediaType());
+
+ Core.Statistics.AddMediaFormat(_imageFormat.GetImageFormat());
+ Core.Statistics.AddMedia(_imageFormat.ImageInfo.mediaType, false);
+ }
+ catch(Exception ex)
+ {
+ DicConsole.ErrorWriteLine("Unable to open image format");
+ DicConsole.ErrorWriteLine("Error: {0}", ex.Message);
+ return;
+ }
+
+ List partitions = new List();
+ string partition_scheme = "";
+
+ // TODO: Solve possibility of multiple partition schemes (CUE + MBR, MBR + RDB, CUE + APM, etc)
+ foreach(PartPlugin _partplugin in plugins.PartPluginsList.Values)
+ {
+ List _partitions;
+ if(_partplugin.GetInformation(_imageFormat, out _partitions))
+ {
+ partition_scheme = _partplugin.Name;
+ partitions.AddRange(_partitions);
+ Core.Statistics.AddPartition(_partplugin.Name);
+ }
+ }
+
+ if(_imageFormat.ImageHasPartitions())
+ {
+ partition_scheme = _imageFormat.GetImageFormat();
+ partitions.AddRange(_imageFormat.GetPartitions());
+ }
+
+ if(partition_scheme == "")
+ DicConsole.DebugWriteLine("Ls command", "No partitions found");
+ else
+ {
+ DicConsole.WriteLine("Partition scheme identified as {0}", partition_scheme);
+ DicConsole.WriteLine("{0} partitions found.", partitions.Count);
+
+ for(int i = 0; i < partitions.Count; i++)
+ {
+ DicConsole.WriteLine();
+ DicConsole.WriteLine("Partition {0}:", partitions[i].PartitionSequence);
+
+ DicConsole.WriteLine("Identifying filesystem on partition");
+
+ IdentifyFilesystems(_imageFormat, out id_plugins, partitions[i].PartitionStartSector, partitions[i].PartitionStartSector + partitions[i].PartitionSectors);
+ if(id_plugins.Count == 0)
+ DicConsole.WriteLine("Filesystem not identified");
+ else if(id_plugins.Count > 1)
+ {
+ DicConsole.WriteLine(String.Format("Identified by {0} plugins", id_plugins.Count));
+
+ foreach(string plugin_name in id_plugins)
+ {
+ if(plugins.PluginsList.TryGetValue(plugin_name, out _plugin))
+ {
+ DicConsole.WriteLine(String.Format("As identified by {0}.", _plugin.Name));
+ Filesystem fs = (Filesystem)_plugin.GetType().GetConstructor(new Type[] { typeof(ImagePlugin), typeof(ulong), typeof(ulong) }).Invoke(new object[] { _imageFormat, partitions[i].PartitionStartSector, partitions[i].PartitionStartSector + partitions[i].PartitionSectors });
+
+ error = fs.Mount(options.Debug);
+ if(error == Errno.NoError)
+ {
+ List rootDir = new List();
+ error = fs.ReadDir("/", ref rootDir);
+ if(error == Errno.NoError)
+ {
+ foreach(string entry in rootDir)
+ DicConsole.WriteLine("{0}", entry);
+ }
+ else
+ DicConsole.ErrorWriteLine("Error {0} reading root directory {0}", error.ToString());
+
+ Core.Statistics.AddFilesystem(fs.XmlFSType.Type);
+ }
+ else
+ DicConsole.ErrorWriteLine("Unable to mount device, error {0}", error.ToString());
+ }
+ }
+ }
+ else
+ {
+ plugins.PluginsList.TryGetValue(id_plugins[0], out _plugin);
+ DicConsole.WriteLine(String.Format("Identified by {0}.", _plugin.Name));
+ Filesystem fs = (Filesystem)_plugin.GetType().GetConstructor(new Type[] { typeof(ImagePlugin), typeof(ulong), typeof(ulong) }).Invoke(new object[] { _imageFormat, partitions[i].PartitionStartSector, partitions[i].PartitionStartSector + partitions[i].PartitionSectors });
+ error = fs.Mount(options.Debug);
+ if(error == Errno.NoError)
+ {
+ List rootDir = new List();
+ error = fs.ReadDir("/", ref rootDir);
+ if(error == Errno.NoError)
+ {
+ foreach(string entry in rootDir)
+ DicConsole.WriteLine("{0}", entry);
+ }
+ else
+ DicConsole.ErrorWriteLine("Error {0} reading root directory {0}", error.ToString());
+
+ Core.Statistics.AddFilesystem(fs.XmlFSType.Type);
+ }
+ else
+ DicConsole.ErrorWriteLine("Unable to mount device, error {0}", error.ToString());
+ }
+ }
+ }
+
+ IdentifyFilesystems(_imageFormat, out id_plugins, 0, _imageFormat.GetSectors() - 1);
+ if(id_plugins.Count == 0)
+ DicConsole.WriteLine("Filesystem not identified");
+ else if(id_plugins.Count > 1)
+ {
+ DicConsole.WriteLine(String.Format("Identified by {0} plugins", id_plugins.Count));
+
+ foreach(string plugin_name in id_plugins)
+ {
+ if(plugins.PluginsList.TryGetValue(plugin_name, out _plugin))
+ {
+ DicConsole.WriteLine(String.Format("As identified by {0}.", _plugin.Name));
+ Filesystem fs = (Filesystem)_plugin.GetType().GetConstructor(new Type[] { typeof(ImagePlugin), typeof(ulong), typeof(ulong) }).Invoke(new object[] { _imageFormat, (ulong)0, (ulong)(_imageFormat.GetSectors() - 1) });
+ error = fs.Mount(options.Debug);
+ if(error == Errno.NoError)
+ {
+ List rootDir = new List();
+ error = fs.ReadDir("/", ref rootDir);
+ if(error == Errno.NoError)
+ {
+ foreach(string entry in rootDir)
+ DicConsole.WriteLine("{0}", entry);
+ }
+ else
+ DicConsole.ErrorWriteLine("Error {0} reading root directory {0}", error.ToString());
+
+ Core.Statistics.AddFilesystem(fs.XmlFSType.Type);
+ }
+ else
+ DicConsole.ErrorWriteLine("Unable to mount device, error {0}", error.ToString());
+ }
+ }
+ }
+ else
+ {
+ plugins.PluginsList.TryGetValue(id_plugins[0], out _plugin);
+ DicConsole.WriteLine(String.Format("Identified by {0}.", _plugin.Name));
+ Filesystem fs = (Filesystem)_plugin.GetType().GetConstructor(new Type[] { typeof(ImagePlugin), typeof(ulong), typeof(ulong) }).Invoke(new object[] { _imageFormat, (ulong)0, (ulong)(_imageFormat.GetSectors() - 1) });
+ error = fs.Mount(options.Debug);
+ if(error == Errno.NoError)
+ {
+ List rootDir = new List();
+ error = fs.ReadDir("/", ref rootDir);
+ if(error == Errno.NoError)
+ {
+ foreach(string entry in rootDir)
+ {
+ if(options.Long)
+ {
+ FileEntryInfo stat = new FileEntryInfo();
+ List xattrs = new List();
+
+ error = fs.Stat(entry, ref stat);
+ if(error == Errno.NoError)
+ {
+ DicConsole.WriteLine("{0}\t{1}\t{2} bytes\t{3}",
+ stat.CreationTimeUtc,
+ stat.Inode,
+ stat.Length,
+ entry);
+
+ error = fs.ListXAttr(entry, ref xattrs);
+ if(error == Errno.NoError)
+ {
+ foreach(string xattr in xattrs)
+ {
+ byte[] xattrBuf = new byte[0];
+ error = fs.GetXattr(entry, xattr, ref xattrBuf);
+ if(error == Errno.NoError)
+ {
+ DicConsole.WriteLine("\t\t{0}\t{1} bytes",
+ xattr, xattrBuf.Length);
+ }
+ }
+ }
+ }
+ else
+ DicConsole.WriteLine("{0}", entry);
+ }
+ else
+ DicConsole.WriteLine("{0}", entry);
+ }
+ }
+ else
+ DicConsole.ErrorWriteLine("Error {0} reading root directory {0}", error.ToString());
+
+ Core.Statistics.AddFilesystem(fs.XmlFSType.Type);
+ }
+ else
+ DicConsole.ErrorWriteLine("Unable to mount device, error {0}", error.ToString());
+ }
+ }
+ catch(Exception ex)
+ {
+ DicConsole.ErrorWriteLine(String.Format("Error reading file: {0}", ex.Message));
+ DicConsole.DebugWriteLine("Ls command", ex.StackTrace);
+ }
+
+ Core.Statistics.AddCommand("ls");
+ }
+
+ static void IdentifyFilesystems(ImagePlugin imagePlugin, out List id_plugins, ulong partitionStart, ulong partitionEnd)
+ {
+ id_plugins = new List();
+ PluginBase plugins = new PluginBase();
+ plugins.RegisterAllPlugins();
+
+ foreach(Filesystem _plugin in plugins.PluginsList.Values)
+ {
+ if(_plugin.Identify(imagePlugin, partitionStart, partitionEnd))
+ id_plugins.Add(_plugin.Name.ToLower());
+ }
+ }
+ }
+}
+
diff --git a/DiscImageChef/DiscImageChef.csproj b/DiscImageChef/DiscImageChef.csproj
index 4693df69..4fb3e940 100644
--- a/DiscImageChef/DiscImageChef.csproj
+++ b/DiscImageChef/DiscImageChef.csproj
@@ -66,6 +66,7 @@
+
diff --git a/DiscImageChef/Main.cs b/DiscImageChef/Main.cs
index f62a91fa..39f927ff 100644
--- a/DiscImageChef/Main.cs
+++ b/DiscImageChef/Main.cs
@@ -57,7 +57,7 @@ namespace DiscImageChef
Parser.Default.ParseArguments(args)
+ DumpMediaOptions, DeviceReportOptions, ConfigureOptions, StatsOptions, LsOptions>(args)
.WithParsed(opts =>
{
if(opts.Debug)
@@ -206,6 +206,17 @@ namespace DiscImageChef
Commands.DeviceReport.doDeviceReport(opts);
})
+ .WithParsed(opts =>
+ {
+ if(opts.Debug)
+ DicConsole.DebugWriteLineEvent += System.Console.Error.WriteLine;
+ if(opts.Verbose)
+ DicConsole.VerboseWriteLineEvent += System.Console.WriteLine;
+ PrintCopyright();
+ Commands.Ls.doLs(opts);
+ })
+
+
.WithParsed(opts => { PrintCopyright(); Commands.Configure.doConfigure(); })
.WithParsed(opts => { PrintCopyright(); Commands.Statistics.showStats(); })
.WithNotParsed(errs => Environment.Exit(1));
diff --git a/DiscImageChef/Options.cs b/DiscImageChef/Options.cs
index a9cb49fd..3479b2c5 100644
--- a/DiscImageChef/Options.cs
+++ b/DiscImageChef/Options.cs
@@ -322,5 +322,15 @@ namespace DiscImageChef
public class StatsOptions
{
}
+
+ [Verb("ls", HelpText = "Lists files in disc image.")]
+ public class LsOptions : CommonOptions
+ {
+ [Option('i', "input", Required = true, HelpText = "Disc image.")]
+ public string InputFile { get; set; }
+
+ [Option('l', "long", Default = false, HelpText = "Uses long format.")]
+ public bool Long { get; set; }
+ }
}