-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathPuzzleExecutor.cs
105 lines (86 loc) · 3.61 KB
/
PuzzleExecutor.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
using System.Diagnostics;
using System.Reflection;
namespace Axnetg.AdventOfCode;
public static class PuzzleExecutor
{
public static void Execute(int year, int day)
{
try
{
var puzzle = GetPuzzleForYearAndDay(year, day);
using var reader = GetInputReader(puzzle);
var puzzleInstance = GetInstanceOfPuzzle(puzzle.PuzzleType, reader);
// ---- Puzzle solver start ----
Console.WriteLine("========================================");
Console.WriteLine(" > {0} Day {1}", puzzle.Year, puzzle.Day.ToString("00"));
Console.WriteLine(" > {0}", puzzle.Title);
// Time measurements
var timestamp = Stopwatch.GetTimestamp();
var resultPartOne = CatchError(puzzleInstance.SolvePartOne);
Console.WriteLine(" > Part 1: {0}", resultPartOne);
var resultPartTwo = CatchError(puzzleInstance.SolvePartTwo);
Console.WriteLine(" > Part 2: {0}", resultPartTwo);
// ---- Puzzle solver end ----
Console.WriteLine(" > Time: {0} ms", Stopwatch.GetElapsedTime(timestamp).TotalMilliseconds);
Console.WriteLine();
}
catch (InvalidOperationException ex)
{
Console.Error.WriteLine("[!] Invalid operation: {0}", ex.Message);
return;
}
catch (Exception ex)
{
Console.Error.WriteLine("[!] General error occurred: {0}", ex);
return;
}
}
private static PuzzleMetadata GetPuzzleForYearAndDay(int year, int day)
{
var puzzle = GetAllPuzzleTypes().FirstOrDefault(x => x.Year == year && x.Day == day);
if (puzzle == null)
throw new InvalidOperationException($"No puzzle found for year = {year} and day = {day}.");
return puzzle;
}
private static IEnumerable<PuzzleMetadata> GetAllPuzzleTypes()
{
return from type in Assembly.GetEntryAssembly()!.GetTypes()
where typeof(IPuzzle).IsAssignableFrom(type) && !type.IsInterface && !type.IsAbstract
let attribute = type.GetCustomAttribute<PuzzleAttribute>()
where attribute != null
orderby attribute.Year, attribute.Day
select new PuzzleMetadata(type, attribute.Title, attribute.Year, attribute.Day);
}
private static TextReader GetInputReader(PuzzleMetadata puzzle)
{
var inputFilePath = $"Inputs/{puzzle.Year}{puzzle.Day:00}.in";
if (!File.Exists(inputFilePath))
throw new InvalidOperationException($"Input file [{inputFilePath}] not found.");
return new StreamReader(inputFilePath);
}
private static IPuzzle GetInstanceOfPuzzle(Type type, TextReader reader)
{
var constructor = type.GetConstructor([typeof(TextReader)]);
if (constructor == null)
throw new InvalidOperationException($"Type [{type.FullName}] must have a constructor with a parameter of type TextReader");
if (constructor.Invoke([reader]) is not IPuzzle instance)
throw new InvalidOperationException($"Could not create instance of type [{type.FullName}].");
return instance;
}
private static PuzzleResult CatchError(Func<PuzzleResult> func)
{
try
{
return func.Invoke();
}
catch (NotImplementedException)
{
return "[NOT IMPLEMENTED]";
}
catch (Exception ex)
{
return $"{ex.Message}\n{ex.StackTrace}";
}
}
private sealed record PuzzleMetadata(Type PuzzleType, string Title, int Year, int Day);
}