-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathHttpRequestExtensionsGenerator.cs
More file actions
192 lines (169 loc) · 8.19 KB
/
HttpRequestExtensionsGenerator.cs
File metadata and controls
192 lines (169 loc) · 8.19 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
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
using Microsoft.OpenApi;
using OpenAPI.WebApiGenerator.OpenApi;
namespace OpenAPI.WebApiGenerator.CodeGeneration;
internal sealed class HttpRequestExtensionsGenerator(
OpenApiSpecVersion openApiVersion,
string @namespace)
{
private const string HttpRequestExtensionsClassName = "HttpRequestExtensions";
internal string CreateBindParameterInvocation(
string requestVariableName,
string bindingTypeName,
IOpenApiParameter parameter) =>
$""""
{@namespace}.{HttpRequestExtensionsClassName}.Bind<{bindingTypeName}>(
{requestVariableName},
"""
{parameter.Serialize(openApiVersion)}
""")
"""";
internal string CreateBindBodyInvocation(
string requestVariableName,
string bindingTypeName)
{
return
$"""
await {@namespace}.{HttpRequestExtensionsClassName}.BindBodyAsync<{bindingTypeName}>(
{requestVariableName}, cancellationToken)
.ConfigureAwait(false)
""";
}
internal SourceCode GenerateHttpRequestExtensionsClass() =>
new($"{HttpRequestExtensionsClassName}.g.cs",
$$$""""
#nullable enable
using System.Collections.Concurrent;
using System.Diagnostics.CodeAnalysis;
using System.Text.Json;
using Corvus.Json;
using Microsoft.AspNetCore.Http;
using Microsoft.Extensions.Primitives;
using Microsoft.Net.Http.Headers;
using OpenAPI.ParameterStyleParsers;
namespace {{{@namespace}}};
/// <summary>
/// Extension methods for http request objects
/// </summary>
internal static class {{{HttpRequestExtensionsClassName}}}
{
private const string ParameterValueParserVersion = "{{{openApiVersion.GetParameterVersion()}}}";
private static readonly ConcurrentDictionary<IParameter, IParameterValueParser> ParserCache = new();
private static IParameterValueParser GetParser(IParameter parameter) =>
ParserCache.GetOrAdd(parameter, _ =>
parameter.CreateParameterValueParser());
private static readonly ConcurrentDictionary<string, IParameter> ParameterCache = new();
private static IParameter GetParameter(string parameterSpecificationAsJson) =>
ParameterCache.GetOrAdd(parameterSpecificationAsJson, _ =>
ParameterFactory.OpenApi(ParameterValueParserVersion, parameterSpecificationAsJson));
/// <summary>
/// Binds an http parameter to a json type
/// </summary>
/// <param name="request">Request to bind from</param>
/// <param name="parameterSpecificationAsJson">OpenAPI parameter specification formatted as json</param>
/// <typeparam name="T">The type to bind</typeparam>
/// <returns>The bound instance</returns>
internal static T Bind<T>(this HttpRequest request,
string parameterSpecificationAsJson)
where T : struct, IJsonValue<T>
{
var parameter = GetParameter(parameterSpecificationAsJson);
return parameter switch
{
_ when parameter.InBody => T.Parse(request.BodyReader.AsStream()),
_ when TryParse<T>(request, parameter, out var value) => value.Value,
_ => T.Undefined
};
}
/// <summary>
/// Binds an http body to a json type
/// </summary>
/// <param name="request">Request to bind from</param>
/// <param name="cancellationToken">Cancellation token</param>
/// <typeparam name="T">The type to bind</typeparam>
/// <returns>An awaitable task to the bound instance</returns>
internal static async Task<T> BindBodyAsync<T>(this HttpRequest request,
CancellationToken cancellationToken)
where T : struct, IJsonValue<T>
{
var document = await JsonDocument.ParseAsync(request.Body,
cancellationToken: cancellationToken)
.ConfigureAwait(false);
return T.FromJson(document.RootElement.Clone());
}
private static bool TryParse<T>(this HttpRequest request, IParameter parameter, [NotNullWhen(true)] out T? value)
where T : struct, IJsonValue<T> =>
parameter switch
{
_ when parameter.InHeader => TryParseHeader<T>(request.Headers, parameter, out value),
_ when parameter.InFormData => TryParseForm<T>(request.Form, parameter, out value),
_ when parameter.InPath => TryParsePath<T>(request.RouteValues, parameter, out value),
_ when parameter.InQuery => TryParseQuery<T>(request.Query, parameter, out value),
_ => throw new InvalidOperationException($"Parameter {parameter.Name} has an unknown location")
};
private static bool TryParseQuery<T>(IQueryCollection query, IParameter parameter, [NotNullWhen(true)] out T? value)
where T : struct, IJsonValue<T>
{
value = null;
return query.TryGetValue(parameter.Name, out var values) &&
TryParse<T>(values, parameter, out value);
}
private static bool TryParsePath<T>(RouteValueDictionary requestPath, IParameter parameter, [NotNullWhen(true)] out T? value)
where T : struct, IJsonValue<T>
{
if (!requestPath.TryGetValue(parameter.Name, out var objValue))
{
value = default;
return false;
}
var stringValue = objValue switch
{
null => null,
string strValue => strValue,
_ => throw new InvalidOperationException(
$"Route value of '{objValue}' with type '{objValue.GetType()}' is not supported")
};
var parser = GetParser(parameter);
value = Parse<T>(parser, stringValue);
return true;
}
private static bool TryParseForm<T>(IFormCollection requestForm, IParameter parameter, [NotNullWhen(true)] out T? value)
where T : struct, IJsonValue<T>
{
value = default;
return requestForm.TryGetValue(parameter.Name, out var values) && TryParse<T>(values, parameter, out value);
}
private static bool TryParseHeader<T>(IHeaderDictionary headers, IParameter parameter, [NotNullWhen(true)] out T? value)
where T : struct, IJsonValue<T>
{
value = default;
return headers.TryGetValue(parameter.Name, out var values) &&
TryParse<T>(values, parameter, out value);
}
private static bool TryParse<T>(StringValues values, IParameter parameter, [NotNullWhen(true)] out T? value)
where T : struct, IJsonValue<T>
{
if (values.Count == 0)
{
value = default;
return false;
}
var parser = GetParser(parameter);
var stringValue = parser.ValueIncludesParameterName
? string.Join('&', values.Select(value => $"{parameter.Name}={value}"))
: values.Single();
value = Parse<T>(parser, stringValue);
return true;
}
private static T Parse<T>(IParameterValueParser parser, string? value)
where T : struct, IJsonValue<T>
{
if (!parser.TryParse(value, out var instance, out var error))
{
throw new BadHttpRequestException(error);
}
return instance == null ? T.Null : T.Parse(instance.ToJsonString());
}
}
#nullable restore
"""");
}