-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathProgram.cs
150 lines (130 loc) · 4.82 KB
/
Program.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
using KeyValue3Updater;
using KeyValue3Updater.Updaters;
using System.Diagnostics;
using System.Reflection;
using System.Text;
using System.Text.RegularExpressions;
//Find all Updaters to be used
var currentAssembly = Assembly.GetCallingAssembly();
var updaterType = typeof(Updater);
var UpdaterTypes = currentAssembly.GetTypes().Where(x => !x.IsAbstract && x.IsSubclassOf(updaterType)).ToList();
Config config = Config.Load();
if(!config.Enable_InitFloat_4_RadToDeg)
{
Console.WriteLine("Disabling C_INIT_InitFloat_4_RadToDeg...");
var t = typeof(C_INIT_InitFloatRadiansUpdater);
UpdaterTypes.RemoveAll((x) => x == t);
}
Console.WriteLine("KeyValue3Updater by rob5300. (github.com/rob5300/KeyValue3Updater)");
Console.WriteLine("-= Input folder to update files in: (Press enter to begin. Leave blank to process the current folder)");
string targetFolder = Console.ReadLine();
string outputFolder;
if (string.IsNullOrEmpty(targetFolder))
{
targetFolder = Directory.GetCurrentDirectory();
}
else if (!Directory.Exists(targetFolder))
{
targetFolder = Path.Combine(Directory.GetCurrentDirectory(), targetFolder);
}
if(Directory.Exists(targetFolder))
{
Stopwatch stopwatch = new Stopwatch();
stopwatch.Start();
outputFolder = Path.Combine(targetFolder, "out");
if(!Directory.Exists(outputFolder))
{
Directory.CreateDirectory(outputFolder);
}
Log.WriteLine($"-= Will update files in directory '{targetFolder}'");
//Get all files in the target directory that we want to update
var files = new List<string>(Directory.EnumerateFiles(targetFolder, "*.vpcf", SearchOption.AllDirectories));
//Remove file paths that are in an existing "out" dir.
files.RemoveAll((x) => Path.GetDirectoryName(x).Contains(outputFolder));
Log.WriteLine($"-= Will process '{files.Count}' files.");
//Split into chunks to process concurrently
List<Task> tasks = new();
var chunks = files.Chunk(300);
int chunkNum = 1;
foreach (string[] fileList in chunks)
{
string[] _fileList = fileList;
int num = chunkNum++;
tasks.Add(
Task.Run(() =>
{
Log.WriteLine($"-= Started processing file thread ({num}/{chunks.Count()})");
UpdateFiles(_fileList);
Log.WriteLine($"-= Finished processing file thread ({num}/{chunks.Count()})");
})
);
}
//Wait for all tasks to complete or time out in 10 mins.
Task.WaitAll(tasks.ToArray(), 10 * 60 * 60 * 1000);
//Write log to program dir
Log.WriteToFile(Directory.GetCurrentDirectory());
stopwatch.Stop();
Console.WriteLine($"Done! Took {stopwatch.Elapsed}.");
}
else
{
Log.WriteLine("-= Directory doesnt exist, exiting...");
}
Console.ReadKey();
/// <summary>
/// Update a list of files
/// </summary>
void UpdateFiles(string[] fileList)
{
//Create a new set of updaters and log string builder.
List<Updater> updaters = new(UpdaterTypes.Count);
StringBuilder logBuilder = new();
foreach (Type type in UpdaterTypes)
{
Updater newUpdater = (Updater)Activator.CreateInstance(type);
//Set where to log to
newUpdater.SetLogBuilder(logBuilder);
updaters.Add(newUpdater);
}
foreach (string _file in fileList)
{
ProcessFile(_file, updaters, logBuilder);
Log.WriteLine(logBuilder.ToString());
logBuilder.Clear();
}
}
/// <summary>
/// Update a single file, using the provided updaters. Write log data to given StringBuilder.
/// </summary>
void ProcessFile(string file, IEnumerable<Updater> updaters, StringBuilder logBuilder)
{
logBuilder.AppendLine($"\n-= Processing File: '{file}'");
string text = File.ReadAllText(file);
//Remove tabs and carrage return chars (to make expressions work with new lines)
text = text.Replace("\t", "").Replace("\r", "");
//Remove all space blocks that are more than 1 in length (hopefully just visual formatting)
text = Regex.Replace(text, @" {2,}", "");
foreach (Updater updater in updaters)
{
try
{
string changed = updater.Process(ref text);
text = changed;
}
catch (RegexMatchTimeoutException timeout)
{
Log.WriteLine($"Updater '{updater.GetType().Name}' timed out for file '{file}'. Updater did not apply update. Is the file valid?");
}
}
string filename = Path.GetFileName(file);
string fileDir = Path.GetDirectoryName(file);
string relativeDir = Path.GetRelativePath(targetFolder, fileDir);
string newPath = Path.Combine(outputFolder, relativeDir, filename);
string newPathDir = Path.GetDirectoryName(newPath);
if (!Directory.Exists(newPathDir))
{
Directory.CreateDirectory(newPathDir);
}
File.WriteAllText(newPath, text);
logBuilder.AppendLine($"-= Updated '${filename}'");
}