Used to re-organized folders
This commit is contained in:
		| @ -4,6 +4,7 @@ using System.Text.Json; | |||||||
| using View_by_Distance.Metadata.Models.Stateless; | using View_by_Distance.Metadata.Models.Stateless; | ||||||
| using View_by_Distance.Metadata.Models.Stateless.Methods; | using View_by_Distance.Metadata.Models.Stateless.Methods; | ||||||
| using View_by_Distance.Shared.Models; | using View_by_Distance.Shared.Models; | ||||||
|  | using View_by_Distance.Shared.Models.Properties; | ||||||
| using View_by_Distance.Shared.Models.Stateless.Methods; | using View_by_Distance.Shared.Models.Stateless.Methods; | ||||||
|  |  | ||||||
| namespace View_by_Distance.Metadata.Models; | namespace View_by_Distance.Metadata.Models; | ||||||
| @ -26,21 +27,50 @@ public class A_Metadata | |||||||
|         _FileGroups = IPath.GetKeyValuePairs(metadataConfiguration.ResultConfiguration, bResultsFullGroupDirectory, [metadataConfiguration.ResultConfiguration.ResultSingleton]); |         _FileGroups = IPath.GetKeyValuePairs(metadataConfiguration.ResultConfiguration, bResultsFullGroupDirectory, [metadataConfiguration.ResultConfiguration.ResultSingleton]); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private FileInfo GetFileInfo(ResultConfiguration resultConfiguration, FilePath filePath) |     private (int, FileInfo) GetFileInfo(ResultConfiguration resultConfiguration, FilePath filePath) | ||||||
|     { |     { | ||||||
|         FileInfo result; |         FileInfo result; | ||||||
|         FileInfo fileInfo = new(filePath.FullName); |         FileInfo fileInfo = new(filePath.FullName); | ||||||
|         (_, int directoryIndex) = IPath.GetDirectoryNameAndIndex(resultConfiguration, filePath); |         (_, int directoryIndex) = IPath.GetDirectoryNameAndIndex(resultConfiguration, filePath); | ||||||
|         DateTime minimumDateTime = fileInfo.CreationTime < fileInfo.LastWriteTime ? fileInfo.CreationTime : fileInfo.LastWriteTime; |         DateTime minimumDateTime = fileInfo.CreationTime < fileInfo.LastWriteTime ? fileInfo.CreationTime : fileInfo.LastWriteTime; | ||||||
|         int minimumYear = minimumDateTime.Year < resultConfiguration.EpicYear ? resultConfiguration.EpicYear : minimumDateTime.Year; |         int fileInfoMinimumYear = minimumDateTime.Year < resultConfiguration.EpicYear ? resultConfiguration.EpicYear : minimumDateTime.Year; | ||||||
|         result = new(Path.Combine(_FileGroups[minimumYear][_MetadataConfiguration.ResultConfiguration.ResultSingleton][directoryIndex], $"{filePath.NameWithoutExtension}{filePath.ExtensionLowered}.json")); |         result = new(Path.Combine(_FileGroups[fileInfoMinimumYear][_MetadataConfiguration.ResultConfiguration.ResultSingleton][directoryIndex], $"{filePath.NameWithoutExtension}{filePath.ExtensionLowered}.json")); | ||||||
|         return result; |         return (fileInfoMinimumYear, result); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private (int, string) GetJsonFile(ResultConfiguration resultConfiguration, FilePath filePath, ExifDirectory exifDirectory) | ||||||
|  |     { | ||||||
|  |         string? result; | ||||||
|  |         DateTime? dateTime; | ||||||
|  |         dateTime = IDate.GetDateTimeOriginal(exifDirectory); | ||||||
|  |         dateTime ??= IDate.GetMinimum(exifDirectory); | ||||||
|  |         (_, int directoryIndex) = IPath.GetDirectoryNameAndIndex(resultConfiguration, filePath); | ||||||
|  |         int exifYear = dateTime.Value.Year < resultConfiguration.EpicYear ? resultConfiguration.EpicYear : dateTime.Value.Year; | ||||||
|  |         result = Path.Combine(_FileGroups[exifYear][_MetadataConfiguration.ResultConfiguration.ResultSingleton][directoryIndex], $"{filePath.NameWithoutExtension}{filePath.ExtensionLowered}.json"); | ||||||
|  |         return (exifYear, result); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static (string, ExifDirectory?) Get(string jsonFile) | ||||||
|  |     { | ||||||
|  |         ExifDirectory? result; | ||||||
|  |         string json = File.ReadAllText(jsonFile); | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             result = JsonSerializer.Deserialize(json, ExifDirectorySourceGenerationContext.Default.ExifDirectory); | ||||||
|  |             if (result is null) | ||||||
|  |                 throw new Exception(); | ||||||
|  |         } | ||||||
|  |         catch (Exception) | ||||||
|  |         { | ||||||
|  |             result = null; | ||||||
|  |         } | ||||||
|  |         return (json, result); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public (FileInfo, ExifDirectory) GetMetadataCollection(MetadataConfiguration metadataConfiguration, FilePath filePath, DeterministicHashCode deterministicHashCode) |     public (FileInfo, ExifDirectory) GetMetadataCollection(MetadataConfiguration metadataConfiguration, FilePath filePath, DeterministicHashCode deterministicHashCode) | ||||||
|     { |     { | ||||||
|         ExifDirectory? results; |         ExifDirectory? result; | ||||||
|         FileInfo fileInfo = GetFileInfo(metadataConfiguration.ResultConfiguration, filePath); |         (int fileInfoMinimumYear, FileInfo fileInfo) = GetFileInfo(metadataConfiguration.ResultConfiguration, filePath); | ||||||
|         if (_MetadataConfiguration.ForceMetadataLastWriteTimeToCreationTime && !fileInfo.Exists && File.Exists(Path.ChangeExtension(fileInfo.FullName, ".delete"))) |         if (_MetadataConfiguration.ForceMetadataLastWriteTimeToCreationTime && !fileInfo.Exists && File.Exists(Path.ChangeExtension(fileInfo.FullName, ".delete"))) | ||||||
|         { |         { | ||||||
|             File.Move(Path.ChangeExtension(fileInfo.FullName, ".delete"), fileInfo.FullName); |             File.Move(Path.ChangeExtension(fileInfo.FullName, ".delete"), fileInfo.FullName); | ||||||
| @ -52,9 +82,9 @@ public class A_Metadata | |||||||
|             fileInfo.Refresh(); |             fileInfo.Refresh(); | ||||||
|         } |         } | ||||||
|         if (_MetadataConfiguration.PropertiesChangedForMetadata) |         if (_MetadataConfiguration.PropertiesChangedForMetadata) | ||||||
|             results = null; |             result = null; | ||||||
|         else if (!fileInfo.Exists) |         else if (!fileInfo.Exists) | ||||||
|             results = null; |             result = null; | ||||||
|         else if (!fileInfo.FullName.EndsWith(".json") && !fileInfo.FullName.EndsWith(".old")) |         else if (!fileInfo.FullName.EndsWith(".json") && !fileInfo.FullName.EndsWith(".old")) | ||||||
|             throw new ArgumentException("must be a *.json file"); |             throw new ArgumentException("must be a *.json file"); | ||||||
|         else |         else | ||||||
| @ -62,59 +92,74 @@ public class A_Metadata | |||||||
|             string json = File.ReadAllText(fileInfo.FullName); |             string json = File.ReadAllText(fileInfo.FullName); | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 results = JsonSerializer.Deserialize(json, ExifDirectorySourceGenerationContext.Default.ExifDirectory); |                 result = JsonSerializer.Deserialize(json, ExifDirectorySourceGenerationContext.Default.ExifDirectory); | ||||||
|                 if (results is null) |                 if (result is null) | ||||||
|                     throw new Exception(); |                     throw new Exception(); | ||||||
|             } |             } | ||||||
|             catch (Exception) |             catch (Exception) | ||||||
|             { |             { | ||||||
|                 results = null; |                 result = null; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         if (results is null) |         if (result is null) | ||||||
|         { |         { | ||||||
|             System.Drawing.Size? size; |             string json; | ||||||
|             try |             System.Drawing.Size? size = Dimensions.GetDimensions(filePath.FullName); | ||||||
|             { size = Dimensions.GetDimensions(filePath.FullName); } |  | ||||||
|             catch (Exception) { size = null; } |  | ||||||
|             IReadOnlyList<MetadataExtractor.Directory> directories = ImageMetadataReader.ReadMetadata(filePath.FullName); |             IReadOnlyList<MetadataExtractor.Directory> directories = ImageMetadataReader.ReadMetadata(filePath.FullName); | ||||||
|             results = Exif.Covert(filePath, deterministicHashCode, size, directories); |             result = Exif.Covert(filePath, deterministicHashCode, size, directories); | ||||||
|             string json = JsonSerializer.Serialize(results, ExifDirectorySourceGenerationContext.Default.ExifDirectory); |             (int exifYear, string jsonFile) = GetJsonFile(metadataConfiguration.ResultConfiguration, filePath, result); | ||||||
|  |             if (exifYear == fileInfoMinimumYear) | ||||||
|  |                 json = JsonSerializer.Serialize(result, ExifDirectorySourceGenerationContext.Default.ExifDirectory); | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 fileInfo = new(jsonFile); | ||||||
|  |                 if (!File.Exists(jsonFile)) | ||||||
|  |                     json = JsonSerializer.Serialize(result, ExifDirectorySourceGenerationContext.Default.ExifDirectory); | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     (string checkJson, ExifDirectory? exifDirectory) = Get(jsonFile); | ||||||
|  |                     if (exifDirectory is null) | ||||||
|  |                         json = JsonSerializer.Serialize(result, ExifDirectorySourceGenerationContext.Default.ExifDirectory); | ||||||
|  |                     else | ||||||
|  |                     { | ||||||
|  |                         json = checkJson; | ||||||
|  |                         result = exifDirectory; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|             if (IPath.WriteAllText(fileInfo.FullName, json, updateDateWhenMatches: false, compareBeforeWrite: true, updateToWhenMatches: null) && _MetadataConfiguration.ForceMetadataLastWriteTimeToCreationTime) |             if (IPath.WriteAllText(fileInfo.FullName, json, updateDateWhenMatches: false, compareBeforeWrite: true, updateToWhenMatches: null) && _MetadataConfiguration.ForceMetadataLastWriteTimeToCreationTime) | ||||||
|             { |             { | ||||||
|                 File.SetLastWriteTime(fileInfo.FullName, fileInfo.CreationTime); |                 File.SetLastWriteTime(fileInfo.FullName, fileInfo.CreationTime); | ||||||
|                 fileInfo.Refresh(); |                 fileInfo.Refresh(); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         return (fileInfo, results); |         return (fileInfo, result); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static Action<string> SetExifDirectoryCollection(IRename rename, MetadataConfiguration metadataConfiguration, A_Metadata metadata, List<(string, FileInfo, ExifDirectory)> exifDirectories, Action tick) |     public static Action<string> SetExifDirectoryCollection(IRename rename, IRenameConfiguration renameConfiguration, A_Metadata metadata, List<(string, FileInfo, ExifDirectory)> exifDirectories) | ||||||
|     { |     { | ||||||
|         return file => |         return file => | ||||||
|         { |         { | ||||||
|             tick.Invoke(); |             rename.Tick(); | ||||||
|             FileInfo fileInfo; |             FileInfo fileInfo; | ||||||
|             FilePath? ffmpegFilePath; |             FilePath? ffmpegFilePath; | ||||||
|             ExifDirectory exifDirectory; |             ExifDirectory exifDirectory; | ||||||
|             ReadOnlyCollection<string> ffmpegFiles; |             ReadOnlyCollection<string> ffmpegFiles; | ||||||
|             DeterministicHashCode deterministicHashCode; |             DeterministicHashCode deterministicHashCode; | ||||||
|             FilePath filePath = IId.GetFilePath(metadataConfiguration, file); |             FilePath filePath = IId.GetFilePath(renameConfiguration.MetadataConfiguration, file); | ||||||
|             if (filePath.ExtensionLowered is not ".paddedId" and not ".lsv") |             if (!renameConfiguration.SkipIdFiles || filePath.Id is null || !filePath.IsIdFormat && !filePath.IsPaddedIdFormat) | ||||||
|             { |             { | ||||||
|                 if (filePath.Id is null || (!filePath.IsIdFormat && !filePath.IsPaddedIdFormat)) |                 (ffmpegFiles, ffmpegFilePath) = rename.ConvertAndGetFfmpegFiles(renameConfiguration, filePath); | ||||||
|                 { |  | ||||||
|                     (ffmpegFiles, ffmpegFilePath) = rename.ConvertAndGetFfmpegFiles(filePath); |  | ||||||
|                 if (ffmpegFilePath is not null) |                 if (ffmpegFilePath is not null) | ||||||
|                     filePath = ffmpegFilePath; |                     filePath = ffmpegFilePath; | ||||||
|                 deterministicHashCode = filePath.Id is not null ? deterministicHashCode = new(null, filePath.Id, null) : deterministicHashCode = rename.GetDeterministicHashCode(filePath); |                 deterministicHashCode = filePath.Id is not null ? deterministicHashCode = new(null, filePath.Id, null) : deterministicHashCode = rename.GetDeterministicHashCode(filePath); | ||||||
|                     (fileInfo, exifDirectory) = metadata.GetMetadataCollection(metadataConfiguration, filePath, deterministicHashCode); |                 (fileInfo, exifDirectory) = metadata.GetMetadataCollection(renameConfiguration.MetadataConfiguration, filePath, deterministicHashCode); | ||||||
|                 lock (exifDirectories) |                 lock (exifDirectories) | ||||||
|                     exifDirectories.Add(new(file, fileInfo, exifDirectory)); |                     exifDirectories.Add(new(file, fileInfo, exifDirectory)); | ||||||
|                 foreach (string ffmpegFile in ffmpegFiles) |                 foreach (string ffmpegFile in ffmpegFiles) | ||||||
|                     File.Delete(ffmpegFile); |                     File.Delete(ffmpegFile); | ||||||
|             } |             } | ||||||
|             } |  | ||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | |||||||
| @ -32,6 +32,7 @@ public class MetadataConfiguration | |||||||
|                     continue; |                     continue; | ||||||
|                 throw new NotSupportedException(physicalFileProvider.Root); |                 throw new NotSupportedException(physicalFileProvider.Root); | ||||||
|             } |             } | ||||||
|  |             throw new NotSupportedException("Not Found!"); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | |||||||
| @ -39,6 +39,7 @@ public class ResultConfiguration | |||||||
|                     continue; |                     continue; | ||||||
|                 throw new NotSupportedException(physicalFileProvider.Root); |                 throw new NotSupportedException(physicalFileProvider.Root); | ||||||
|             } |             } | ||||||
|  |             throw new NotSupportedException("Not Found!"); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | |||||||
| @ -5,10 +5,8 @@ namespace View_by_Distance.Metadata.Models.Stateless.Methods; | |||||||
| internal static class Dimensions | internal static class Dimensions | ||||||
| { | { | ||||||
|  |  | ||||||
|     const string _ErrorMessage = "Could not recognize image format."; |  | ||||||
|  |  | ||||||
| #pragma warning disable IDE0230 | #pragma warning disable IDE0230 | ||||||
|     private static readonly Dictionary<byte[], Func<BinaryReader, Size>> _ImageFormatDecoders = new() |     private static readonly Dictionary<byte[], Func<BinaryReader, Size?>> _ImageFormatDecoders = new() | ||||||
|     { |     { | ||||||
|         { new byte[] { 0x42, 0x4D }, DecodeBitmap }, |         { new byte[] { 0x42, 0x4D }, DecodeBitmap }, | ||||||
|         { new byte[] { 0x47, 0x49, 0x46, 0x38, 0x37, 0x61 }, DecodeGif }, |         { new byte[] { 0x47, 0x49, 0x46, 0x38, 0x37, 0x61 }, DecodeGif }, | ||||||
| @ -23,11 +21,10 @@ internal static class Dimensions | |||||||
|     { |     { | ||||||
|         for (int i = 0; i < thatBytes.Length; i += 1) |         for (int i = 0; i < thatBytes.Length; i += 1) | ||||||
|         { |         { | ||||||
|             if (thisBytes[i] != thatBytes[i]) |             if (thisBytes[i] == thatBytes[i]) | ||||||
|             { |                 continue; | ||||||
|             return false; |             return false; | ||||||
|         } |         } | ||||||
|         } |  | ||||||
|         return true; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -35,9 +32,7 @@ internal static class Dimensions | |||||||
|     { |     { | ||||||
|         byte[] bytes = new byte[sizeof(short)]; |         byte[] bytes = new byte[sizeof(short)]; | ||||||
|         for (int i = 0; i < sizeof(short); i += 1) |         for (int i = 0; i < sizeof(short); i += 1) | ||||||
|         { |  | ||||||
|             bytes[sizeof(short) - 1 - i] = binaryReader.ReadByte(); |             bytes[sizeof(short) - 1 - i] = binaryReader.ReadByte(); | ||||||
|         } |  | ||||||
|         return BitConverter.ToInt16(bytes, 0); |         return BitConverter.ToInt16(bytes, 0); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -45,13 +40,11 @@ internal static class Dimensions | |||||||
|     { |     { | ||||||
|         byte[] bytes = new byte[sizeof(int)]; |         byte[] bytes = new byte[sizeof(int)]; | ||||||
|         for (int i = 0; i < sizeof(int); i += 1) |         for (int i = 0; i < sizeof(int); i += 1) | ||||||
|         { |  | ||||||
|             bytes[sizeof(int) - 1 - i] = binaryReader.ReadByte(); |             bytes[sizeof(int) - 1 - i] = binaryReader.ReadByte(); | ||||||
|         } |  | ||||||
|         return BitConverter.ToInt32(bytes, 0); |         return BitConverter.ToInt32(bytes, 0); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static Size DecodeBitmap(BinaryReader binaryReader) |     private static Size? DecodeBitmap(BinaryReader binaryReader) | ||||||
|     { |     { | ||||||
|         _ = binaryReader.ReadBytes(16); |         _ = binaryReader.ReadBytes(16); | ||||||
|         int width = binaryReader.ReadInt32(); |         int width = binaryReader.ReadInt32(); | ||||||
| @ -59,14 +52,14 @@ internal static class Dimensions | |||||||
|         return new Size(width, height); |         return new Size(width, height); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static Size DecodeGif(BinaryReader binaryReader) |     private static Size? DecodeGif(BinaryReader binaryReader) | ||||||
|     { |     { | ||||||
|         int width = binaryReader.ReadInt16(); |         int width = binaryReader.ReadInt16(); | ||||||
|         int height = binaryReader.ReadInt16(); |         int height = binaryReader.ReadInt16(); | ||||||
|         return new Size(width, height); |         return new Size(width, height); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static Size DecodePng(BinaryReader binaryReader) |     private static Size? DecodePng(BinaryReader binaryReader) | ||||||
|     { |     { | ||||||
|         _ = binaryReader.ReadBytes(8); |         _ = binaryReader.ReadBytes(8); | ||||||
|         int width = ReadLittleEndianInt32(binaryReader); |         int width = ReadLittleEndianInt32(binaryReader); | ||||||
| @ -74,100 +67,60 @@ internal static class Dimensions | |||||||
|         return new Size(width, height); |         return new Size(width, height); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static Size DecodeJfif(BinaryReader binaryReader) |     private static Size? DecodeJfif(BinaryReader binaryReader) | ||||||
|     { |     { | ||||||
|         while (binaryReader.ReadByte() == 0xff) |         while (binaryReader.ReadByte() == 0xff) | ||||||
|         { |         { | ||||||
|             byte marker = binaryReader.ReadByte(); |             byte marker = binaryReader.ReadByte(); | ||||||
|             short chunkLength = ReadLittleEndianInt16(binaryReader); |             short chunkLength = ReadLittleEndianInt16(binaryReader); | ||||||
|  |  | ||||||
|             if (marker == 0xc0) |             if (marker == 0xc0) | ||||||
|             { |             { | ||||||
|                 _ = binaryReader.ReadByte(); |                 _ = binaryReader.ReadByte(); | ||||||
|  |  | ||||||
|                 int height = ReadLittleEndianInt16(binaryReader); |                 int height = ReadLittleEndianInt16(binaryReader); | ||||||
|                 int width = ReadLittleEndianInt16(binaryReader); |                 int width = ReadLittleEndianInt16(binaryReader); | ||||||
|                 return new Size(width, height); |                 return new Size(width, height); | ||||||
|             } |             } | ||||||
|  |             if (chunkLength >= 0) | ||||||
|             if (chunkLength < 0) |                 _ = binaryReader.ReadBytes(chunkLength - 2); | ||||||
|  |             else | ||||||
|             { |             { | ||||||
|                 ushort uChunkLength = (ushort)chunkLength; |                 ushort uChunkLength = (ushort)chunkLength; | ||||||
|                 _ = binaryReader.ReadBytes(uChunkLength - 2); |                 _ = binaryReader.ReadBytes(uChunkLength - 2); | ||||||
|             } |             } | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 _ = binaryReader.ReadBytes(chunkLength - 2); |  | ||||||
|         } |         } | ||||||
|  |         return null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|         throw new ArgumentException(_ErrorMessage); |     private static Size? DecodeWebP(BinaryReader binaryReader) | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static Size DecodeWebP(BinaryReader binaryReader) |  | ||||||
|     { |     { | ||||||
|         _ = binaryReader.ReadUInt32(); // Size |         _ = binaryReader.ReadUInt32(); // Size | ||||||
|         _ = binaryReader.ReadBytes(15); // WEBP, VP8 + more |         _ = binaryReader.ReadBytes(15); // WEBP, VP8 + more | ||||||
|         _ = binaryReader.ReadBytes(3); // SYNC |         _ = binaryReader.ReadBytes(3); // SYNC | ||||||
|  |  | ||||||
|         int width = binaryReader.ReadUInt16() & 0b00_11111111111111; // 14 bits width |         int width = binaryReader.ReadUInt16() & 0b00_11111111111111; // 14 bits width | ||||||
|         int height = binaryReader.ReadUInt16() & 0b00_11111111111111; // 14 bits height |         int height = binaryReader.ReadUInt16() & 0b00_11111111111111; // 14 bits height | ||||||
|  |  | ||||||
|         return new Size(width, height); |         return new Size(width, height); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /// <summary> |     internal static Size? GetDimensions(BinaryReader binaryReader) | ||||||
|     /// Gets the dimensions of an image. |  | ||||||
|     /// </summary> |  | ||||||
|     /// <param name="path">The path of the image to get the dimensions of.</param> |  | ||||||
|     /// <returns>The dimensions of the specified image.</returns> |  | ||||||
|     /// <exception cref="ArgumentException">The image was of an unrecognized format.</exception>     |  | ||||||
|     internal static Size GetDimensions(BinaryReader binaryReader) |  | ||||||
|     { |     { | ||||||
|         int maxMagicBytesLength = _ImageFormatDecoders.Keys.OrderByDescending(x => x.Length).First().Length; |         int maxMagicBytesLength = _ImageFormatDecoders.Keys.OrderByDescending(x => x.Length).First().Length; | ||||||
|  |  | ||||||
|         byte[] magicBytes = new byte[maxMagicBytesLength]; |         byte[] magicBytes = new byte[maxMagicBytesLength]; | ||||||
|  |  | ||||||
|         for (int i = 0; i < maxMagicBytesLength; i += 1) |         for (int i = 0; i < maxMagicBytesLength; i += 1) | ||||||
|         { |         { | ||||||
|             magicBytes[i] = binaryReader.ReadByte(); |             magicBytes[i] = binaryReader.ReadByte(); | ||||||
|  |             foreach (KeyValuePair<byte[], Func<BinaryReader, Size?>> kvPair in _ImageFormatDecoders) | ||||||
|             foreach (KeyValuePair<byte[], Func<BinaryReader, Size>> kvPair in _ImageFormatDecoders) |  | ||||||
|             { |             { | ||||||
|                 if (StartsWith(magicBytes, kvPair.Key)) |                 if (StartsWith(magicBytes, kvPair.Key)) | ||||||
|                 { |  | ||||||
|                     return kvPair.Value(binaryReader); |                     return kvPair.Value(binaryReader); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |         return null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|         throw new ArgumentException(_ErrorMessage, nameof(binaryReader)); |     internal static Size? GetDimensions(string path) | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /// <summary> |  | ||||||
|     /// Gets the dimensions of an image. |  | ||||||
|     ///internal </summary> |  | ||||||
|     /// <param name="path">The path of the image to get the dimensions of.</param> |  | ||||||
|     /// <returns>The dimensions of the specified image.</returns> |  | ||||||
|     /// <exception cref="ArgumentException">The image was of an unrecognized format.</exception> |  | ||||||
|     internal static Size GetDimensions(string path) |  | ||||||
|     { |     { | ||||||
|         using BinaryReader binaryReader = new(File.OpenRead(path)); |         using BinaryReader binaryReader = new(File.OpenRead(path)); | ||||||
|         try |  | ||||||
|         { |  | ||||||
|         return GetDimensions(binaryReader); |         return GetDimensions(binaryReader); | ||||||
|     } |     } | ||||||
|         catch (ArgumentException e) |  | ||||||
|         { |  | ||||||
|             if (e.Message.StartsWith(_ErrorMessage)) |  | ||||||
|             { |  | ||||||
|                 throw new ArgumentException(_ErrorMessage, nameof(path), e); |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 throw; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
| } | } | ||||||
| @ -357,10 +357,10 @@ internal abstract class Exif | |||||||
|     internal static string GetMaker(Shared.Models.ExifDirectoryBase exifDirectoryBase) |     internal static string GetMaker(Shared.Models.ExifDirectoryBase exifDirectoryBase) | ||||||
|     { |     { | ||||||
|         string result; |         string result; | ||||||
|         if (string.IsNullOrEmpty(exifDirectoryBase.Make)) |         if (string.IsNullOrEmpty(exifDirectoryBase.Make?.ToString().Trim())) | ||||||
|             result = "Unknown"; |             result = "Unknown"; | ||||||
|         else |         else | ||||||
|             result = $"{exifDirectoryBase.Make[0].ToString().ToUpper()}{exifDirectoryBase.Make[1..]}".Trim(); |             result = $"{exifDirectoryBase.Make[0].ToString().ToUpper()}{exifDirectoryBase.Make[1..].ToLower()}".Trim(); | ||||||
|         return result; |         return result; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | |||||||
| @ -31,6 +31,7 @@ public class AppSettings | |||||||
|                     continue; |                     continue; | ||||||
|                 throw new NotSupportedException(physicalFileProvider.Root); |                 throw new NotSupportedException(physicalFileProvider.Root); | ||||||
|             } |             } | ||||||
|  |             throw new NotSupportedException("Not Found!"); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | |||||||
| @ -8,6 +8,8 @@ public class RenameConfiguration | |||||||
| { | { | ||||||
|  |  | ||||||
|     public string[]? IgnoreExtensions { get; set; } |     public string[]? IgnoreExtensions { get; set; } | ||||||
|  |     public bool? MoveFilesToRoot { get; set; } | ||||||
|  |     public bool? SkipIdFiles { get; set; } | ||||||
|     public string[]? ValidImageFormatExtensions { get; set; } |     public string[]? ValidImageFormatExtensions { get; set; } | ||||||
|  |  | ||||||
|     public override string ToString() |     public override string ToString() | ||||||
| @ -30,6 +32,7 @@ public class RenameConfiguration | |||||||
|                     continue; |                     continue; | ||||||
|                 throw new NotSupportedException(physicalFileProvider.Root); |                 throw new NotSupportedException(physicalFileProvider.Root); | ||||||
|             } |             } | ||||||
|  |             throw new NotSupportedException("Not Found!"); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -44,10 +47,14 @@ public class RenameConfiguration | |||||||
|         Models.RenameConfiguration result; |         Models.RenameConfiguration result; | ||||||
|         if (configuration is null) throw new NullReferenceException(nameof(configuration)); |         if (configuration is null) throw new NullReferenceException(nameof(configuration)); | ||||||
|         if (configuration.IgnoreExtensions is null) throw new NullReferenceException(nameof(configuration.IgnoreExtensions)); |         if (configuration.IgnoreExtensions is null) throw new NullReferenceException(nameof(configuration.IgnoreExtensions)); | ||||||
|  |         if (configuration.MoveFilesToRoot is null) throw new NullReferenceException(nameof(configuration.MoveFilesToRoot)); | ||||||
|  |         if (configuration.SkipIdFiles is null) throw new NullReferenceException(nameof(configuration.SkipIdFiles)); | ||||||
|         if (configuration.ValidImageFormatExtensions is null) throw new NullReferenceException(nameof(configuration.ValidImageFormatExtensions)); |         if (configuration.ValidImageFormatExtensions is null) throw new NullReferenceException(nameof(configuration.ValidImageFormatExtensions)); | ||||||
|         Verify(configuration); |         Verify(configuration); | ||||||
|         result = new(metadataConfiguration, |         result = new(metadataConfiguration, | ||||||
|                      configuration.IgnoreExtensions, |                      configuration.IgnoreExtensions, | ||||||
|  |                      configuration.MoveFilesToRoot.Value, | ||||||
|  |                      configuration.SkipIdFiles.Value, | ||||||
|                      configuration.ValidImageFormatExtensions); |                      configuration.ValidImageFormatExtensions); | ||||||
|         return result; |         return result; | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -6,7 +6,9 @@ namespace View_by_Distance.Rename.Models; | |||||||
|  |  | ||||||
| public record RenameConfiguration(Shared.Models.MetadataConfiguration MetadataConfiguration, | public record RenameConfiguration(Shared.Models.MetadataConfiguration MetadataConfiguration, | ||||||
|                                   string[] IgnoreExtensions, |                                   string[] IgnoreExtensions, | ||||||
|                                   string[] ValidImageFormatExtensions) |                                   bool MoveFilesToRoot, | ||||||
|  |                                   bool SkipIdFiles, | ||||||
|  |                                   string[] ValidImageFormatExtensions) : Shared.Models.Properties.IRenameConfiguration | ||||||
| { | { | ||||||
|  |  | ||||||
|     public override string ToString() |     public override string ToString() | ||||||
|  | |||||||
							
								
								
									
										113
									
								
								Rename/Rename.cs
									
									
									
									
									
								
							
							
						
						
									
										113
									
								
								Rename/Rename.cs
									
									
									
									
									
								
							| @ -10,6 +10,7 @@ using View_by_Distance.Metadata.Models; | |||||||
| using View_by_Distance.Metadata.Models.Stateless.Methods; | using View_by_Distance.Metadata.Models.Stateless.Methods; | ||||||
| using View_by_Distance.Rename.Models; | using View_by_Distance.Rename.Models; | ||||||
| using View_by_Distance.Shared.Models; | using View_by_Distance.Shared.Models; | ||||||
|  | using View_by_Distance.Shared.Models.Properties; | ||||||
| using View_by_Distance.Shared.Models.Stateless.Methods; | using View_by_Distance.Shared.Models.Stateless.Methods; | ||||||
|  |  | ||||||
| namespace View_by_Distance.Rename; | namespace View_by_Distance.Rename; | ||||||
| @ -20,8 +21,7 @@ public class Rename : IRename | |||||||
|     private record ToDo(string? Directory, FileHolder FileHolder, string File, bool JsonFile); |     private record ToDo(string? Directory, FileHolder FileHolder, string File, bool JsonFile); | ||||||
|     private record Record(DateTime DateTime, ExifDirectory ExifDirectory, string File, string JsonFile); |     private record Record(DateTime DateTime, ExifDirectory ExifDirectory, string File, string JsonFile); | ||||||
|  |  | ||||||
|     private readonly AppSettings _AppSettings; |     private ProgressBar? _ProgressBar; | ||||||
|     private readonly RenameConfiguration _RenameConfiguration; |  | ||||||
|  |  | ||||||
|     public Rename(List<string> args, ILogger<Program>? logger, IConfigurationRoot configurationRoot, AppSettings appSettings, bool isSilent, IConsole console) |     public Rename(List<string> args, ILogger<Program>? logger, IConfigurationRoot configurationRoot, AppSettings appSettings, bool isSilent, IConsole console) | ||||||
|     { |     { | ||||||
| @ -31,31 +31,24 @@ public class Rename : IRename | |||||||
|             throw new NullReferenceException(nameof(args)); |             throw new NullReferenceException(nameof(args)); | ||||||
|         if (console is null) |         if (console is null) | ||||||
|             throw new NullReferenceException(nameof(console)); |             throw new NullReferenceException(nameof(console)); | ||||||
|         _AppSettings = appSettings; |         IRename rename = this; | ||||||
|         long ticks = DateTime.Now.Ticks; |         long ticks = DateTime.Now.Ticks; | ||||||
|         ResultConfiguration resultConfiguration = Metadata.Models.Binder.ResultConfiguration.Get(configurationRoot, appSettings.RequireRootDirectoryExists); |         ResultConfiguration resultConfiguration = Metadata.Models.Binder.ResultConfiguration.Get(configurationRoot, appSettings.RequireRootDirectoryExists); | ||||||
|         MetadataConfiguration metadataConfiguration = Metadata.Models.Binder.MetadataConfiguration.Get(configurationRoot, resultConfiguration); |         MetadataConfiguration metadataConfiguration = Metadata.Models.Binder.MetadataConfiguration.Get(configurationRoot, resultConfiguration); | ||||||
|         RenameConfiguration renameConfiguration = Models.Binder.RenameConfiguration.Get(configurationRoot, metadataConfiguration); |         RenameConfiguration renameConfiguration = Models.Binder.RenameConfiguration.Get(configurationRoot, metadataConfiguration); | ||||||
|         _RenameConfiguration = renameConfiguration; |         RenameWork(logger, appSettings, rename, ticks, renameConfiguration); | ||||||
|         DirectoryInfo directoryInfo = new(Path.GetFullPath(resultConfiguration.RootDirectory)); |  | ||||||
|         logger?.LogInformation("{RootDirectory}", directoryInfo.FullName); |  | ||||||
|         ReadOnlyCollection<Record> exifDirectories = GetExifDirectoryCollection(directoryInfo); |  | ||||||
|         ReadOnlyCollection<ToDo> toDoCollection = GetToDoCollection(logger, ticks, exifDirectories); |  | ||||||
|         ReadOnlyCollection<string> lines = RenameFilesInDirectories(toDoCollection); |  | ||||||
|         if (lines.Count != 0) |  | ||||||
|         { |  | ||||||
|             File.WriteAllLines($"D:/Tmp/Phares/{DateTime.Now.Ticks}.tsv", lines); |  | ||||||
|             _ = IPath.DeleteEmptyDirectories(directoryInfo.FullName); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     (ReadOnlyCollection<string>, FilePath?) IRename.ConvertAndGetFfmpegFiles(FilePath filePath) |     void IRename.Tick() => | ||||||
|  |         _ProgressBar?.Tick(); | ||||||
|  |  | ||||||
|  |     (ReadOnlyCollection<string>, FilePath?) IRename.ConvertAndGetFfmpegFiles(IRenameConfiguration renameConfiguration, FilePath filePath) | ||||||
|     { |     { | ||||||
|         List<string> results = []; |         List<string> results = []; | ||||||
|         FilePath? result; |         FilePath? result; | ||||||
|         bool isIgnoreExtension; |         bool isIgnoreExtension; | ||||||
|         bool isValidImageFormatExtension = _RenameConfiguration.ValidImageFormatExtensions.Contains(filePath.ExtensionLowered); |         bool isValidImageFormatExtension = renameConfiguration.ValidImageFormatExtensions.Contains(filePath.ExtensionLowered); | ||||||
|         isIgnoreExtension = isValidImageFormatExtension && _RenameConfiguration.IgnoreExtensions.Contains(filePath.ExtensionLowered); |         isIgnoreExtension = isValidImageFormatExtension && renameConfiguration.IgnoreExtensions.Contains(filePath.ExtensionLowered); | ||||||
|         if (!isIgnoreExtension && isValidImageFormatExtension) |         if (!isIgnoreExtension && isValidImageFormatExtension) | ||||||
|             result = null; |             result = null; | ||||||
|         else |         else | ||||||
| @ -69,11 +62,11 @@ public class Rename : IRename | |||||||
|             results.AddRange(Directory.GetFiles(filePath.DirectoryName, $"{filePath.Name}-*.jpg", SearchOption.TopDirectoryOnly)); |             results.AddRange(Directory.GetFiles(filePath.DirectoryName, $"{filePath.Name}-*.jpg", SearchOption.TopDirectoryOnly)); | ||||||
|             if (results.Count == 0) |             if (results.Count == 0) | ||||||
|                 throw new Exception(); |                 throw new Exception(); | ||||||
|             result = IId.GetFilePath(_RenameConfiguration.MetadataConfiguration, results[0]); |             result = IId.GetFilePath(renameConfiguration.MetadataConfiguration, results[0]); | ||||||
|             if (!result.Name.EndsWith("-0001.jpg")) |             if (!result.Name.EndsWith("-0001.jpg")) | ||||||
|                 throw new Exception(); |                 throw new Exception(); | ||||||
|             isValidImageFormatExtension = _RenameConfiguration.ValidImageFormatExtensions.Contains(result.ExtensionLowered); |             isValidImageFormatExtension = renameConfiguration.ValidImageFormatExtensions.Contains(result.ExtensionLowered); | ||||||
|             isIgnoreExtension = isValidImageFormatExtension && _RenameConfiguration.IgnoreExtensions.Contains(result.ExtensionLowered); |             isIgnoreExtension = isValidImageFormatExtension && renameConfiguration.IgnoreExtensions.Contains(result.ExtensionLowered); | ||||||
|             if (isIgnoreExtension || !isValidImageFormatExtension) |             if (isIgnoreExtension || !isValidImageFormatExtension) | ||||||
|                 throw new Exception(); |                 throw new Exception(); | ||||||
|             if (result.DirectoryName is null) |             if (result.DirectoryName is null) | ||||||
| @ -117,7 +110,7 @@ public class Rename : IRename | |||||||
|  |  | ||||||
| #pragma warning restore CA1416 | #pragma warning restore CA1416 | ||||||
|  |  | ||||||
|     private void GetExifDirectoryCollection(IRename rename, List<(string, FileInfo, ExifDirectory)> exifDirectories, IEnumerable<string> files, A_Metadata metadata) |     private static void GetExifDirectoryCollection(IRename rename, RenameConfiguration renameConfiguration, List<(string, FileInfo, ExifDirectory)> exifDirectories, IEnumerable<string> files, A_Metadata metadata) | ||||||
|     { |     { | ||||||
|         FileInfo fileInfo; |         FileInfo fileInfo; | ||||||
|         FilePath filePath; |         FilePath filePath; | ||||||
| @ -127,21 +120,18 @@ public class Rename : IRename | |||||||
|         DeterministicHashCode deterministicHashCode; |         DeterministicHashCode deterministicHashCode; | ||||||
|         foreach (string file in files) |         foreach (string file in files) | ||||||
|         { |         { | ||||||
|             filePath = IId.GetFilePath(_RenameConfiguration.MetadataConfiguration, file); |             rename.Tick(); | ||||||
|             if (filePath.ExtensionLowered is ".paddedId" or ".lsv") |             filePath = IId.GetFilePath(renameConfiguration.MetadataConfiguration, file); | ||||||
|  |             if (renameConfiguration.SkipIdFiles && filePath.Id is not null && (filePath.IsIdFormat || filePath.IsPaddedIdFormat)) | ||||||
|                 continue; |                 continue; | ||||||
|             if (files.Contains($"{filePath.FullName}.paddedId")) |             (ffmpegFiles, ffmpegFilePath) = rename.ConvertAndGetFfmpegFiles(renameConfiguration, filePath); | ||||||
|                 continue; |  | ||||||
|             if (filePath.Id is not null && (filePath.IsIdFormat || filePath.IsPaddedIdFormat)) |  | ||||||
|                 continue; |  | ||||||
|             (ffmpegFiles, ffmpegFilePath) = rename.ConvertAndGetFfmpegFiles(filePath); |  | ||||||
|             if (ffmpegFilePath is not null) |             if (ffmpegFilePath is not null) | ||||||
|                 filePath = ffmpegFilePath; |                 filePath = ffmpegFilePath; | ||||||
|             if (filePath.Id is not null) |             if (filePath.Id is not null) | ||||||
|                 deterministicHashCode = new(null, filePath.Id, null); |                 deterministicHashCode = new(null, filePath.Id, null); | ||||||
|             else |             else | ||||||
|                 deterministicHashCode = rename.GetDeterministicHashCode(filePath); |                 deterministicHashCode = rename.GetDeterministicHashCode(filePath); | ||||||
|             (fileInfo, exifDirectory) = metadata.GetMetadataCollection(_RenameConfiguration.MetadataConfiguration, filePath, deterministicHashCode); |             (fileInfo, exifDirectory) = metadata.GetMetadataCollection(renameConfiguration.MetadataConfiguration, filePath, deterministicHashCode); | ||||||
|             exifDirectories.Add(new(file, fileInfo, exifDirectory)); |             exifDirectories.Add(new(file, fileInfo, exifDirectory)); | ||||||
|             foreach (string ffmpegFile in ffmpegFiles) |             foreach (string ffmpegFile in ffmpegFiles) | ||||||
|                 File.Delete(ffmpegFile); |                 File.Delete(ffmpegFile); | ||||||
| @ -161,24 +151,24 @@ public class Rename : IRename | |||||||
|         return new(results); |         return new(results); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private ReadOnlyCollection<Record> GetExifDirectoryCollection(DirectoryInfo directoryInfo) |     private ReadOnlyCollection<Record> GetExifDirectoryCollection(IRename rename, AppSettings appSettings, RenameConfiguration renameConfiguration, DirectoryInfo directoryInfo) | ||||||
|     { |     { | ||||||
|         ReadOnlyCollection<Record> results; |         ReadOnlyCollection<Record> results; | ||||||
|         IRename rename = this; |  | ||||||
|         List<(string, FileInfo, ExifDirectory)> exifDirectories = []; |         List<(string, FileInfo, ExifDirectory)> exifDirectories = []; | ||||||
|         int appSettingsMaxDegreeOfParallelism = _AppSettings.MaxDegreeOfParallelism; |         int appSettingsMaxDegreeOfParallelism = appSettings.MaxDegreeOfParallelism; | ||||||
|         IEnumerable<string> files = Directory.EnumerateFiles(directoryInfo.FullName, "*", SearchOption.AllDirectories); |         IEnumerable<string> files = Directory.EnumerateFiles(directoryInfo.FullName, "*", SearchOption.AllDirectories); | ||||||
|         A_Metadata metadata = new(_RenameConfiguration.MetadataConfiguration); |         A_Metadata metadata = new(renameConfiguration.MetadataConfiguration); | ||||||
|  |         _ProgressBar = new(123000, "EnumerateFiles load", new ProgressBarOptions() { ProgressCharacter = '─', ProgressBarOnBottom = true, DisableBottomPercentage = true }); | ||||||
|         if (appSettingsMaxDegreeOfParallelism == 1) |         if (appSettingsMaxDegreeOfParallelism == 1) | ||||||
|             GetExifDirectoryCollection(rename, exifDirectories, files, metadata); |             GetExifDirectoryCollection(rename, renameConfiguration, exifDirectories, files, metadata); | ||||||
|         else |         else | ||||||
|         { |         { | ||||||
|             ParallelOptions parallelOptions = new() { MaxDegreeOfParallelism = appSettingsMaxDegreeOfParallelism }; |             ParallelOptions parallelOptions = new() { MaxDegreeOfParallelism = appSettingsMaxDegreeOfParallelism }; | ||||||
|             ProgressBar progressBar = new(123000, "EnumerateFiles load", new ProgressBarOptions() { ProgressCharacter = '─', ProgressBarOnBottom = true, DisableBottomPercentage = true }); |             files.AsParallel().ForAll(A_Metadata.SetExifDirectoryCollection(rename, renameConfiguration, metadata, exifDirectories)); | ||||||
|             files.AsParallel().ForAll(A_Metadata.SetExifDirectoryCollection(rename, _RenameConfiguration.MetadataConfiguration, metadata, exifDirectories, () => progressBar.Tick())); |             if (_ProgressBar.CurrentTick != exifDirectories.Count) | ||||||
|             if (progressBar.CurrentTick != exifDirectories.Count) |  | ||||||
|                 throw new NotSupportedException(); |                 throw new NotSupportedException(); | ||||||
|         } |         } | ||||||
|  |         _ProgressBar.Dispose(); | ||||||
|         results = GetExifDirectoryCollection(exifDirectories); |         results = GetExifDirectoryCollection(exifDirectories); | ||||||
|         return results; |         return results; | ||||||
|     } |     } | ||||||
| @ -194,16 +184,26 @@ public class Rename : IRename | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private ReadOnlyCollection<ToDo> GetToDoCollection(ILogger<Program>? logger, long ticks, ReadOnlyCollection<Record> exifDirectories) |     private static string GetCheckDirectory(RenameConfiguration renameConfiguration, Record record, FileHolder fileHolder) | ||||||
|  |     { | ||||||
|  |         string? checkDirectory; | ||||||
|  |         if (fileHolder.DirectoryName is null) | ||||||
|  |             throw new NullReferenceException(nameof(fileHolder.DirectoryName)); | ||||||
|  |         (int season, string seasonName) = IDate.GetSeason(record.DateTime.DayOfYear); | ||||||
|  |         string maker = IMetadata.GetMaker(record.ExifDirectory.ExifDirectoryBase); | ||||||
|  |         string splat = fileHolder.DirectoryName[^3..][1] == '!' ? fileHolder.DirectoryName[^3..] : string.Empty; | ||||||
|  |         string rootDirectory = renameConfiguration.MoveFilesToRoot ? renameConfiguration.MetadataConfiguration.ResultConfiguration.RootDirectory : fileHolder.DirectoryName; | ||||||
|  |         checkDirectory = Path.Combine(rootDirectory, "_ Destination _", $"{record.DateTime.Year}.{season} {seasonName} {maker.Split(' ')[0]}{splat}"); | ||||||
|  |         return checkDirectory; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static ReadOnlyCollection<ToDo> GetToDoCollection(ILogger<Program>? logger, RenameConfiguration renameConfiguration, long ticks, ReadOnlyCollection<Record> exifDirectories) | ||||||
|     { |     { | ||||||
|         List<ToDo> results = []; |         List<ToDo> results = []; | ||||||
|         int season; |  | ||||||
|         string maker; |  | ||||||
|         Record record; |         Record record; | ||||||
|         string jsonFile; |         string jsonFile; | ||||||
|         string paddedId; |         string paddedId; | ||||||
|         string checkFile; |         string checkFile; | ||||||
|         string seasonName; |  | ||||||
|         string directoryName; |         string directoryName; | ||||||
|         FileHolder fileHolder; |         FileHolder fileHolder; | ||||||
|         string? checkDirectory; |         string? checkDirectory; | ||||||
| @ -214,6 +214,7 @@ public class Rename : IRename | |||||||
|         string jsonFileSubDirectory; |         string jsonFileSubDirectory; | ||||||
|         int intMinValueLength = int.MinValue.ToString().Length; |         int intMinValueLength = int.MinValue.ToString().Length; | ||||||
|         VerifyIntMinValueLength(exifDirectories, intMinValueLength); |         VerifyIntMinValueLength(exifDirectories, intMinValueLength); | ||||||
|  |         ResultConfiguration resultConfiguration = renameConfiguration.MetadataConfiguration.ResultConfiguration; | ||||||
|         ReadOnlyCollection<Record> records = new((from l in exifDirectories orderby l.DateTime select l).ToArray()); |         ReadOnlyCollection<Record> records = new((from l in exifDirectories orderby l.DateTime select l).ToArray()); | ||||||
|         for (int i = 0; i < records.Count; i++) |         for (int i = 0; i < records.Count; i++) | ||||||
|         { |         { | ||||||
| @ -223,12 +224,10 @@ public class Rename : IRename | |||||||
|             fileHolder = new(record.File); |             fileHolder = new(record.File); | ||||||
|             if (fileHolder.DirectoryName is null) |             if (fileHolder.DirectoryName is null) | ||||||
|                 continue; |                 continue; | ||||||
|             maker = IMetadata.GetMaker(record.ExifDirectory.ExifDirectoryBase); |             checkDirectory = GetCheckDirectory(renameConfiguration, record, fileHolder); | ||||||
|             (season, seasonName) = IDate.GetSeason(record.DateTime.DayOfYear); |  | ||||||
|             checkFileExtension = fileHolder.ExtensionLowered == jpeg ? jpg : fileHolder.ExtensionLowered; |             checkFileExtension = fileHolder.ExtensionLowered == jpeg ? jpg : fileHolder.ExtensionLowered; | ||||||
|             checkDirectory = Path.Combine(fileHolder.DirectoryName, $"{record.DateTime.Year}.{season} {seasonName}{maker}"); |  | ||||||
|             jsonFileSubDirectory = Path.GetDirectoryName(Path.GetDirectoryName(record.JsonFile)) ?? throw new Exception(); |             jsonFileSubDirectory = Path.GetDirectoryName(Path.GetDirectoryName(record.JsonFile)) ?? throw new Exception(); | ||||||
|             paddedId = IId.GetPaddedId(intMinValueLength, _RenameConfiguration.MetadataConfiguration.Offset + i, record.ExifDirectory.Id.Value); |             paddedId = IId.GetPaddedId(intMinValueLength, renameConfiguration.MetadataConfiguration.Offset + i, record.ExifDirectory.Id.Value); | ||||||
|             checkFile = Path.Combine(checkDirectory, $"{paddedId}{checkFileExtension}"); |             checkFile = Path.Combine(checkDirectory, $"{paddedId}{checkFileExtension}"); | ||||||
|             if (checkFile == fileHolder.FullName) |             if (checkFile == fileHolder.FullName) | ||||||
|                 continue; |                 continue; | ||||||
| @ -238,7 +237,7 @@ public class Rename : IRename | |||||||
|                 if (File.Exists(checkFile)) |                 if (File.Exists(checkFile)) | ||||||
|                     continue; |                     continue; | ||||||
|             } |             } | ||||||
|             (directoryName, _) = IPath.GetDirectoryNameAndIndex(_RenameConfiguration.MetadataConfiguration.ResultConfiguration, record.ExifDirectory.Id.Value); |             (directoryName, _) = IPath.GetDirectoryNameAndIndex(resultConfiguration, record.ExifDirectory.Id.Value); | ||||||
|             jsonFile = Path.Combine(jsonFileSubDirectory, directoryName, $"{record.ExifDirectory.Id.Value}{checkFileExtension}.json"); |             jsonFile = Path.Combine(jsonFileSubDirectory, directoryName, $"{record.ExifDirectory.Id.Value}{checkFileExtension}.json"); | ||||||
|             if (record.JsonFile != jsonFile) |             if (record.JsonFile != jsonFile) | ||||||
|                 results.Add(new(null, new(record.JsonFile), jsonFile, JsonFile: true)); |                 results.Add(new(null, new(record.JsonFile), jsonFile, JsonFile: true)); | ||||||
| @ -267,8 +266,10 @@ public class Rename : IRename | |||||||
|     { |     { | ||||||
|         List<string> results = []; |         List<string> results = []; | ||||||
|         VerifyDirectories(toDoCollection); |         VerifyDirectories(toDoCollection); | ||||||
|  |         _ProgressBar = new(toDoCollection.Count, "Move Files", new ProgressBarOptions() { ProgressCharacter = '─', ProgressBarOnBottom = true, DisableBottomPercentage = true }); | ||||||
|         foreach (ToDo toDo in toDoCollection) |         foreach (ToDo toDo in toDoCollection) | ||||||
|         { |         { | ||||||
|  |             _ProgressBar.Tick(); | ||||||
|             if (toDo.JsonFile) |             if (toDo.JsonFile) | ||||||
|             { |             { | ||||||
|                 if (File.Exists(toDo.File)) |                 if (File.Exists(toDo.File)) | ||||||
| @ -281,11 +282,29 @@ public class Rename : IRename | |||||||
|             { |             { | ||||||
|                 if (File.Exists(toDo.File)) |                 if (File.Exists(toDo.File)) | ||||||
|                     File.Delete(toDo.File); |                     File.Delete(toDo.File); | ||||||
|                 File.Move(toDo.FileHolder.FullName, toDo.File); |                 try | ||||||
|  |                 { File.Move(toDo.FileHolder.FullName, toDo.File); } | ||||||
|  |                 catch (Exception) | ||||||
|  |                 { continue; } | ||||||
|                 results.Add($"{toDo.FileHolder.FullName}\t{toDo.File}"); |                 results.Add($"{toDo.FileHolder.FullName}\t{toDo.File}"); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |         _ProgressBar.Dispose(); | ||||||
|         return new(results); |         return new(results); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private void RenameWork(ILogger<Program>? logger, AppSettings appSettings, IRename rename, long ticks, RenameConfiguration renameConfiguration) | ||||||
|  |     { | ||||||
|  |         DirectoryInfo directoryInfo = new(Path.GetFullPath(renameConfiguration.MetadataConfiguration.ResultConfiguration.RootDirectory)); | ||||||
|  |         logger?.LogInformation("{RootDirectory}", directoryInfo.FullName); | ||||||
|  |         ReadOnlyCollection<Record> exifDirectories = GetExifDirectoryCollection(rename, appSettings, renameConfiguration, directoryInfo); | ||||||
|  |         ReadOnlyCollection<ToDo> toDoCollection = GetToDoCollection(logger, renameConfiguration, ticks, exifDirectories); | ||||||
|  |         ReadOnlyCollection<string> lines = RenameFilesInDirectories(toDoCollection); | ||||||
|  |         if (lines.Count != 0) | ||||||
|  |         { | ||||||
|  |             File.WriteAllLines($"D:/Tmp/Phares/{DateTime.Now.Ticks}.tsv", lines); | ||||||
|  |             _ = IPath.DeleteEmptyDirectories(directoryInfo.FullName); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
							
								
								
									
										11
									
								
								Shared/Models/Properties/IRenameConfiguration.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								Shared/Models/Properties/IRenameConfiguration.cs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | namespace View_by_Distance.Shared.Models.Properties; | ||||||
|  |  | ||||||
|  | public interface IRenameConfiguration | ||||||
|  | { | ||||||
|  |  | ||||||
|  |     public MetadataConfiguration MetadataConfiguration { init; get; } | ||||||
|  |     public string[] IgnoreExtensions { init; get; } | ||||||
|  |     public bool SkipIdFiles { init; get; } | ||||||
|  |     public string[] ValidImageFormatExtensions { init; get; } | ||||||
|  |  | ||||||
|  | } | ||||||
| @ -1,11 +1,13 @@ | |||||||
| using System.Collections.ObjectModel; | using System.Collections.ObjectModel; | ||||||
|  | using View_by_Distance.Shared.Models.Properties; | ||||||
|  |  | ||||||
| namespace View_by_Distance.Shared.Models.Stateless.Methods; | namespace View_by_Distance.Shared.Models.Stateless.Methods; | ||||||
|  |  | ||||||
| public interface IRename | public interface IRename | ||||||
| { | { | ||||||
|  |  | ||||||
|     (ReadOnlyCollection<string>, FilePath?) ConvertAndGetFfmpegFiles(FilePath filePath); |     (ReadOnlyCollection<string>, FilePath?) ConvertAndGetFfmpegFiles(IRenameConfiguration renameConfiguration, FilePath filePath); | ||||||
|     DeterministicHashCode GetDeterministicHashCode(FilePath filePath); |     DeterministicHashCode GetDeterministicHashCode(FilePath filePath); | ||||||
|  |     void Tick(); | ||||||
|  |  | ||||||
| } | } | ||||||
		Reference in New Issue
	
	Block a user