Skip to content

Commit 0a58448

Browse files
authored
Add IUserContextBuilder<TSchema> (#60)
* Add IUserContextBuilder<TSchema> * Update
1 parent bc1ba12 commit 0a58448

File tree

7 files changed

+61
-6
lines changed

7 files changed

+61
-6
lines changed

Directory.Build.props

+1-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@
2121
<Nullable>enable</Nullable>
2222
<IsPackable>false</IsPackable>
2323
<GraphQLVersion>7.0.0</GraphQLVersion>
24-
<NoWarn>$(NoWarn);IDE0056;IDE0057</NoWarn>
24+
<NoWarn>$(NoWarn);IDE0056;IDE0057;NU1902;NU1903</NoWarn>
2525
</PropertyGroup>
2626

2727
<ItemGroup Condition="'$(IsPackable)' == 'true'">

README.md

+4
Original file line numberDiff line numberDiff line change
@@ -741,6 +741,10 @@ are long-lived, using scoped services within a user context builder will result
741741
scoped services having a matching long lifetime. You may wish to alleviate this by
742742
creating a service scope temporarily within your user context builder.
743743

744+
For applications that service multiple schemas, you may register `IUserContextBuilder<TSchema>`
745+
to create a user context for a specific schema. This is useful when you need to create
746+
a user context that is specific to a particular schema.
747+
744748
### Mutations within GET request
745749

746750
For security reasons and pursuant to current recommendations, mutation GraphQL requests

src/GraphQL.AspNetCore3/GraphQLHttpMiddleware.cs

+15-5
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,15 @@ public GraphQLHttpMiddleware(
3232
: base(next, serializer, documentExecuter, serviceScopeFactory, options, hostApplicationLifetime)
3333
{
3434
}
35+
36+
/// <inheritdoc/>
37+
protected override ValueTask<IDictionary<string, object?>> BuildUserContextAsync(HttpContext context, object? payload)
38+
{
39+
var userContextBuilder = context.RequestServices.GetService<IUserContextBuilder<TSchema>>();
40+
return userContextBuilder == null
41+
? base.BuildUserContextAsync(context, payload)
42+
: userContextBuilder.BuildUserContextAsync(context, payload);
43+
}
3544
}
3645

3746
/// <summary>
@@ -405,17 +414,18 @@ protected virtual async Task<ExecutionResult> ExecuteRequestAsync(HttpContext co
405414
/// <see cref="IHttpContextAccessor"/> via <see cref="ExecutionOptions.RequestServices"/>
406415
/// if needed.
407416
/// <br/><br/>
408-
/// By default this method pulls the registered <see cref="IUserContextBuilder"/>,
409-
/// if any, within the service scope and executes it to build the user context.
417+
/// By default this method pulls the registered <see cref="IUserContextBuilder{TSchema}"/>
418+
/// or <see cref="IUserContextBuilder"/> instance, if any, within the service scope
419+
/// and executes it to build the user context.
410420
/// In this manner, both scoped and singleton <see cref="IUserContextBuilder"/>
411421
/// instances are supported, although singleton instances are recommended.
412422
/// </summary>
413-
protected virtual async ValueTask<IDictionary<string, object?>> BuildUserContextAsync(HttpContext context, object? payload)
423+
protected virtual ValueTask<IDictionary<string, object?>> BuildUserContextAsync(HttpContext context, object? payload)
414424
{
415425
var userContextBuilder = context.RequestServices.GetService<IUserContextBuilder>();
416426
var userContext = userContextBuilder == null
417-
? new Dictionary<string, object?>()
418-
: await userContextBuilder.BuildUserContextAsync(context, payload);
427+
? new(new Dictionary<string, object?>())
428+
: userContextBuilder.BuildUserContextAsync(context, payload);
419429
return userContext;
420430
}
421431

src/GraphQL.AspNetCore3/IUserContextBuilder.cs

+6
Original file line numberDiff line numberDiff line change
@@ -23,3 +23,9 @@ public interface IUserContextBuilder
2323
/// </param>
2424
ValueTask<IDictionary<string, object?>> BuildUserContextAsync(HttpContext context, object? payload);
2525
}
26+
27+
/// <inheritdoc cref="IUserContextBuilder"/>
28+
public interface IUserContextBuilder<TSchema> : IUserContextBuilder
29+
where TSchema : ISchema
30+
{
31+
}

src/Tests.ApiApprovals/GraphQL.AspNetCore3.approved.txt

+3
Original file line numberDiff line numberDiff line change
@@ -160,6 +160,7 @@ namespace GraphQL.AspNetCore3
160160
where TSchema : GraphQL.Types.ISchema
161161
{
162162
public GraphQLHttpMiddleware(Microsoft.AspNetCore.Http.RequestDelegate next, GraphQL.IGraphQLTextSerializer serializer, GraphQL.IDocumentExecuter<TSchema> documentExecuter, Microsoft.Extensions.DependencyInjection.IServiceScopeFactory serviceScopeFactory, GraphQL.AspNetCore3.GraphQLHttpMiddlewareOptions options, Microsoft.Extensions.Hosting.IHostApplicationLifetime hostApplicationLifetime) { }
163+
protected override System.Threading.Tasks.ValueTask<System.Collections.Generic.IDictionary<string, object?>> BuildUserContextAsync(Microsoft.AspNetCore.Http.HttpContext context, object? payload) { }
163164
}
164165
public sealed class HttpGetValidationRule : GraphQL.Validation.IValidationRule
165166
{
@@ -181,6 +182,8 @@ namespace GraphQL.AspNetCore3
181182
{
182183
System.Threading.Tasks.ValueTask<System.Collections.Generic.IDictionary<string, object?>> BuildUserContextAsync(Microsoft.AspNetCore.Http.HttpContext context, object? payload);
183184
}
185+
public interface IUserContextBuilder<TSchema> : GraphQL.AspNetCore3.IUserContextBuilder
186+
where TSchema : GraphQL.Types.ISchema { }
184187
public class UserContextBuilder<TUserContext> : GraphQL.AspNetCore3.IUserContextBuilder
185188
where TUserContext : System.Collections.Generic.IDictionary<string, object?>
186189
{

src/Tests/Middleware/MiscTests.cs

+31
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,37 @@ public void Constructors()
3434

3535
}
3636

37+
[Fact]
38+
public async Task SchemaSpecificBuilders()
39+
{
40+
var middleware = new MyMiddleware<Schema>(
41+
Mock.Of<RequestDelegate>(MockBehavior.Strict),
42+
Mock.Of<IGraphQLTextSerializer>(MockBehavior.Strict),
43+
Mock.Of<IDocumentExecuter<Schema>>(MockBehavior.Strict),
44+
Mock.Of<IServiceScopeFactory>(MockBehavior.Strict),
45+
new GraphQLHttpMiddlewareOptions(),
46+
Mock.Of<IHostApplicationLifetime>(MockBehavior.Strict));
47+
var builder = new Mock<IUserContextBuilder<Schema>>(MockBehavior.Strict);
48+
var d = new Dictionary<string, object?> { { "test", "test" } };
49+
builder.Setup(x => x.BuildUserContextAsync(It.IsAny<HttpContext>(), It.IsAny<object?>())).ReturnsAsync(d);
50+
var serviceProviderMock = new Mock<IServiceProvider>(MockBehavior.Strict);
51+
serviceProviderMock.Setup(x => x.GetService(typeof(IUserContextBuilder<Schema>))).Returns(builder.Object);
52+
var contextMock = new Mock<HttpContext>(MockBehavior.Strict);
53+
contextMock.Setup(x => x.RequestServices).Returns(serviceProviderMock.Object);
54+
(await middleware.MyBuildUserContextAsync(contextMock.Object, null)).ShouldBe(d);
55+
}
56+
57+
private class MyMiddleware<TSchema> : GraphQLHttpMiddleware<TSchema>
58+
where TSchema : ISchema
59+
{
60+
public MyMiddleware(RequestDelegate next, IGraphQLTextSerializer serializer, IDocumentExecuter<TSchema> executer, IServiceScopeFactory scopeFactory, GraphQLHttpMiddlewareOptions options, IHostApplicationLifetime hostApplicationLifetime)
61+
: base(next, serializer, executer, scopeFactory, options, hostApplicationLifetime)
62+
{
63+
}
64+
65+
public ValueTask<IDictionary<string, object?>> MyBuildUserContextAsync(HttpContext context, object? payload) => base.BuildUserContextAsync(context, payload);
66+
}
67+
3768
[Fact]
3869
public async Task WriteErrorResponseString()
3970
{

src/Tests/Tests.csproj

+1
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@
4646
<Using Include="GraphQL.AspNetCore3" />
4747
<Using Include="GraphQL.AspNetCore3.WebSockets" />
4848
<Using Include="Shouldly" />
49+
<Using Include="System.Net.Http"/>
4950
<Using Include="System.Text" />
5051
<Using Include="System.Threading" />
5152
<Using Include="System.Threading.Tasks" />

0 commit comments

Comments
 (0)