146 lines
6.1 KiB
C#
146 lines
6.1 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Text.RegularExpressions;
|
|
using CRD.Utils.Structs;
|
|
|
|
namespace CRD.Utils.Files;
|
|
|
|
public class FileNameManager{
|
|
public static List<string> ParseFileName(string input, List<Variable> variables, int numbers, List<string> @override){
|
|
Regex varRegex = new Regex(@"\${[A-Za-z1-9]+}");
|
|
var matches = varRegex.Matches(input).Cast<Match>().Select(m => m.Value).ToList();
|
|
var overriddenVars = ParseOverride(variables, @override);
|
|
if (!matches.Any())
|
|
return new List<string>{
|
|
input
|
|
};
|
|
foreach (var match in matches){
|
|
string varName = match.Substring(2, match.Length - 3); // Removing ${ and }
|
|
var variable = overriddenVars.FirstOrDefault(v => v.Name == varName);
|
|
|
|
if (variable == null){
|
|
Console.Error.WriteLine($"[ERROR] Found variable '{match}' in fileName but no values was internally found!");
|
|
continue;
|
|
}
|
|
|
|
string replacement = variable.ReplaceWith.ToString();
|
|
if (variable.Type == "int32"){
|
|
int len = replacement.Length;
|
|
replacement = len < numbers ? new string('0', numbers - len) + replacement : replacement;
|
|
} else if (variable.Type == "double"){
|
|
string[] parts = replacement.Split(',');
|
|
string formattedIntegerPart = parts[0].PadLeft(numbers, '0');
|
|
replacement = formattedIntegerPart + (parts.Length > 1 ? "," + parts[1] : "");
|
|
replacement = replacement.Replace(",", ".");
|
|
} else if (variable.Sanitize){
|
|
replacement = CleanupFilename(replacement);
|
|
}
|
|
|
|
input = input.Replace(match, replacement);
|
|
}
|
|
|
|
return input.Split(Path.DirectorySeparatorChar).Select(CleanupFilename).ToList();
|
|
}
|
|
|
|
public static List<Variable> ParseOverride(List<Variable> variables, List<string>? overrides){
|
|
if (overrides == null){
|
|
return variables;
|
|
}
|
|
|
|
foreach (var item in overrides){
|
|
int index = item.IndexOf('=');
|
|
if (index == -1){
|
|
Console.Error.WriteLine($"Error: Invalid override format '{item}'");
|
|
continue;
|
|
}
|
|
|
|
string[] parts ={ item.Substring(0, index), item.Substring(index + 1) };
|
|
if (!(parts[1].StartsWith("'") && parts[1].EndsWith("'") && parts[1].Length >= 2)){
|
|
Console.Error.WriteLine($"Error: Invalid value format for '{item}'");
|
|
continue;
|
|
}
|
|
|
|
parts[1] = parts[1][1..^1]; // Removing the surrounding single quotes
|
|
int alreadyIndex = variables.FindIndex(a => a.Name == parts[0]);
|
|
|
|
if (alreadyIndex > -1){
|
|
if (variables[alreadyIndex].Type == "number"){
|
|
if (!float.TryParse(parts[1], NumberStyles.Any, CultureInfo.InvariantCulture, out float numberValue)){
|
|
Console.Error.WriteLine($"Error: Wrong type for '{item}'");
|
|
continue;
|
|
}
|
|
|
|
variables[alreadyIndex].ReplaceWith = numberValue;
|
|
} else{
|
|
variables[alreadyIndex].ReplaceWith = parts[1];
|
|
}
|
|
} else{
|
|
bool isNumber = float.TryParse(parts[1], NumberStyles.Any, CultureInfo.InvariantCulture, out float parsedNumber);
|
|
variables.Add(new Variable{
|
|
Name = parts[0],
|
|
ReplaceWith = isNumber ? parsedNumber : (object)parts[1],
|
|
Type = isNumber ? "number" : "string"
|
|
});
|
|
}
|
|
}
|
|
|
|
return variables;
|
|
}
|
|
|
|
public static string CleanupFilename(string filename){
|
|
string fixingChar = "";
|
|
Regex illegalRe = new Regex(@"[\/\?<>\\:\*\|"":]"); // Illegal Characters on most Operating Systems
|
|
Regex controlRe = new Regex(@"[\x00-\x1f\x80-\x9f]"); // Unicode Control codes: C0 and C1
|
|
Regex reservedRe = new Regex(@"^\.\.?$"); // Reserved filenames on Unix-based systems (".", "..")
|
|
Regex windowsReservedRe = new Regex(@"^(con|prn|aux|nul|com[0-9]|lpt[0-9])(\..*)?$", RegexOptions.IgnoreCase);
|
|
/* Reserved filenames in Windows ("CON", "PRN", "AUX", "NUL", "COM1"-"COM9", "LPT1"-"LPT9")
|
|
case-insensitively and with or without filename extensions. */
|
|
Regex windowsTrailingRe = new Regex(@"[\. ]+$");
|
|
|
|
filename = illegalRe.Replace(filename, fixingChar);
|
|
filename = controlRe.Replace(filename, fixingChar);
|
|
filename = reservedRe.Replace(filename, fixingChar);
|
|
filename = windowsReservedRe.Replace(filename, fixingChar);
|
|
filename = windowsTrailingRe.Replace(filename, fixingChar);
|
|
|
|
return filename;
|
|
}
|
|
|
|
|
|
public static void DeleteEmptyFolders(string rootFolderPath){
|
|
if (string.IsNullOrEmpty(rootFolderPath) || !Directory.Exists(rootFolderPath)){
|
|
Console.WriteLine("Invalid directory path.");
|
|
return;
|
|
}
|
|
|
|
DeleteEmptyFoldersRecursive(rootFolderPath, isRoot: true);
|
|
}
|
|
|
|
private static bool DeleteEmptyFoldersRecursive(string folderPath, bool isRoot = false){
|
|
bool isFolderEmpty = true;
|
|
|
|
try{
|
|
foreach (var directory in Directory.GetDirectories(folderPath)){
|
|
// Recursively delete empty subfolders
|
|
if (!DeleteEmptyFoldersRecursive(directory)){
|
|
isFolderEmpty = false;
|
|
}
|
|
}
|
|
|
|
// Check if the current folder is empty (no files and no non-deleted subfolders)
|
|
if (!isRoot && isFolderEmpty && Directory.GetFiles(folderPath).Length == 0){
|
|
Directory.Delete(folderPath);
|
|
Console.WriteLine($"Deleted empty folder: {folderPath}");
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
} catch (Exception ex){
|
|
Console.WriteLine($"An error occurred while deleting folder {folderPath}: {ex.Message}");
|
|
return false;
|
|
}
|
|
}
|
|
} |