-
-
Notifications
You must be signed in to change notification settings - Fork 6
Expand file tree
/
Copy pathGenericLoggerExtensionsEmitter.cs
More file actions
131 lines (103 loc) · 5.47 KB
/
GenericLoggerExtensionsEmitter.cs
File metadata and controls
131 lines (103 loc) · 5.47 KB
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
using System.CodeDom.Compiler;
using Microsoft.Extensions.Logging;
using static AutoLoggerMessageGenerator.Constants;
namespace AutoLoggerMessageGenerator.Emitters;
/// Generates the <see cref="LoggerExtensions"/> class.
/// Provides type-specific logging method overrides to avoid boxing.
/// The <see cref="LoggerExtensions"/> class is pre-generated and saved in the solution
/// to prevent excessive code duplication when multiple projects use the same library
/// To run this emitter you need to run LoggerExtensionEmitterTests and take it from the snapshot/>
internal static class GenericLoggerExtensionsEmitter
{
public const string ClassName = "GenericLoggerExtensions";
public static string Emit()
{
using var sb = new IndentedTextWriter(new StringWriter());
sb.WriteLine(GeneratedFileHeader);
sb.WriteLine(JetBrainsAnnotationsImport);
sb.WriteLine();
sb.WriteLine($"namespace {DefaultLoggingNamespace}");
sb.WriteLine('{');
sb.Indent++;
sb.WriteLine(Constants.GeneratedCodeAttribute);
sb.WriteLine(EditorNotBrowsableAttribute);
sb.WriteLine(DebuggerStepThroughAttribute);
sb.WriteLine(ExcludeFromCoverageAttribute);
sb.WriteLine($"public static class {ClassName}");
sb.WriteLine('{');
sb.Indent++;
string[] logLevels = ["Trace", "Debug", "Information", "Warning", "Error", "Critical"];
var messageParameter = ($"{MessageTemplateDecorator} string", MessageParameterName);
var exceptionParameter = ("Exception?", ExceptionParameterName);
var eventIdParameter = ("EventId", EventIdParameterName);
(string Type, string Name)[][] fixedParametersOverloads =
[
[messageParameter],
[exceptionParameter, messageParameter],
[eventIdParameter, messageParameter],
[eventIdParameter, exceptionParameter, messageParameter]
];
foreach (var fixedParametersOverload in fixedParametersOverloads)
{
var fixedParametersDefinition =
string.Join(", ", fixedParametersOverload.Select(o => $"{o.Type} {o.Name}"));
var fixedParameters = string.Join(", ", fixedParametersOverload.Select(o => o.Name));
for (int i = 0; i <= MaxLogParameters; i++)
{
var parameters = Enumerable.Range(0, i).ToArray();
var genericTypesDefinition = string.Join(", ", parameters.Select(ix => $"T{ix}"));
genericTypesDefinition = string.IsNullOrEmpty(genericTypesDefinition)
? string.Empty
: $"<{genericTypesDefinition}>";
var genericParametersDefinition =
string.Join(", ", parameters.Select(ix => $"T{ix} {ParameterName}{ix}"));
genericParametersDefinition = string.IsNullOrEmpty(genericParametersDefinition)
? string.Empty
: $", {genericParametersDefinition}";
var objectParameters = string.Join(", ", parameters.Select(ix => $"{ParameterName}{ix}"));
objectParameters = string.IsNullOrEmpty(objectParameters)
? string.Empty
: $", new object?[] {{ {objectParameters} }}";
foreach (var logLevel in logLevels)
{
GenerateLogMethodWithLogLevelInName(sb, logLevel, genericTypesDefinition,
fixedParametersDefinition, genericParametersDefinition, fixedParameters, objectParameters);
sb.WriteLine();
}
GenerateLogMethodWithLogLevelParameter(sb, genericTypesDefinition, fixedParametersDefinition,
genericParametersDefinition, fixedParameters, objectParameters);
sb.WriteLine();
}
sb.WriteLine();
}
sb.Indent--;
sb.WriteLine('}');
sb.Indent--;
sb.WriteLine('}');
return sb.InnerWriter.ToString()!;
}
private static void GenerateLogMethodWithLogLevelInName(IndentedTextWriter sb, string logLevel,
string genericTypesDefinition, string fixedParametersDefinition, string genericParametersDefinition,
string fixedParameters, string objectParameters)
{
sb.WriteLine(
$"public static void Log{logLevel}{genericTypesDefinition}(this ILogger {LoggerParameterName}, {fixedParametersDefinition}{genericParametersDefinition})");
sb.WriteLine('{');
sb.Indent++;
sb.WriteLine($"{DefaultLoggingNamespace}.{nameof(LoggerExtensions)}.Log{logLevel}({LoggerParameterName}, {fixedParameters}{objectParameters});");
sb.Indent--;
sb.WriteLine('}');
}
private static void GenerateLogMethodWithLogLevelParameter(IndentedTextWriter sb, string genericTypesDefinition,
string fixedParametersDefinition, string genericParametersDefinition, string fixedParameters,
string objectParameters)
{
sb.WriteLine(
$"public static void Log{genericTypesDefinition}(this ILogger {LoggerParameterName}, {DefaultLoggingNamespace}.LogLevel {LogLevelParameterName}, {fixedParametersDefinition}{genericParametersDefinition})");
sb.WriteLine('{');
sb.Indent++;
sb.WriteLine($"{DefaultLoggingNamespace}.{nameof(LoggerExtensions)}.{nameof(LoggerExtensions.Log)}({LoggerParameterName}, {LogLevelParameterName}, {fixedParameters}{objectParameters});");
sb.Indent--;
sb.WriteLine('}');
}
}