Skip to content

Commit

Permalink
chore: dispose APIRequestContext on BrowserContext close (#2968)
Browse files Browse the repository at this point in the history
  • Loading branch information
mxschmitt authored Jul 29, 2024
1 parent a5f629d commit 7494c1b
Show file tree
Hide file tree
Showing 3 changed files with 37 additions and 2 deletions.
8 changes: 8 additions & 0 deletions src/Playwright.Tests/BrowserContextFetchTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -884,6 +884,14 @@ public async Task ShouldSupportRepeatingNamesInMultipartFormData()
Assert.AreEqual(200, response.Status);
}

[PlaywrightTest("browsercontext-fetch.spec.ts", "should not work after context dispose")]
public async Task ShouldNotWorkAfterContextDispose()
{
await Context.CloseAsync(new() { Reason = "Test ended." });
var exception = await PlaywrightAssert.ThrowsAsync<PlaywrightException>(() => Context.APIRequest.GetAsync(Server.EmptyPage));
StringAssert.Contains("Test ended.", exception.Message);
}

private async Task ForAllMethods(IAPIRequestContext request, Func<Task<IAPIResponse>, Task> callback, string url, APIRequestContextOptions options = null)
{
var methodsToTest = new[] { "fetch", "delete", "get", "head", "patch", "post", "put" };
Expand Down
23 changes: 22 additions & 1 deletion src/Playwright/Core/APIRequestContext.cs
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ namespace Microsoft.Playwright.Core;
internal class APIRequestContext : ChannelOwner, IAPIRequestContext
{
internal readonly Tracing _tracing;
private string _closeReason;

internal APIRequest _request;

Expand All @@ -49,7 +50,23 @@ public APIRequestContext(ChannelOwner parent, string guid, APIRequestContextInit
[MethodImpl(MethodImplOptions.NoInlining)]
public async ValueTask DisposeAsync()
{
await SendMessageToServerAsync("dispose").ConfigureAwait(false);
await DisposeAsync(null).ConfigureAwait(false);
}

internal async ValueTask DisposeAsync(string reason)
{
_closeReason = reason;
try
{
await SendMessageToServerAsync("dispose", new Dictionary<string, object>
{
["reason"] = reason,
}).ConfigureAwait(false);
}
catch (Exception e) when (DriverMessages.IsTargetClosedError(e))
{
// Swallow exception
}
_tracing.ResetStackCounter();
}

Expand Down Expand Up @@ -79,6 +96,10 @@ internal async Task<IAPIResponse> InnerFetchAsync(IRequest request, string urlOv
[MethodImpl(MethodImplOptions.NoInlining)]
public async Task<IAPIResponse> FetchAsync(string url, APIRequestContextOptions options = null)
{
if (!string.IsNullOrEmpty(_closeReason))
{
throw new PlaywrightException(_closeReason);
}
options ??= new APIRequestContextOptions();

if (options.MaxRedirects != null && options.MaxRedirects < 0)
Expand Down
8 changes: 7 additions & 1 deletion src/Playwright/Core/BrowserContext.cs
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ internal class BrowserContext : ChannelOwner, IBrowserContext
private readonly Tracing _tracing;
private readonly Clock _clock;
internal readonly HashSet<IPage> _backgroundPages = new();
internal readonly IAPIRequestContext _request;
internal readonly APIRequestContext _request;
private readonly Dictionary<string, HarRecorder> _harRecorders = new();
internal readonly List<IWorker> _serviceWorkers = new();
private List<RouteHandler> _routes = new();
Expand Down Expand Up @@ -320,6 +320,12 @@ public async Task CloseAsync(BrowserContextCloseOptions options = default)
}
_closeReason = options?.Reason;
CloseWasCalled = true;
await WrapApiCallAsync(
async () =>
{
await _request.DisposeAsync(options?.Reason).ConfigureAwait(false);
},
true).ConfigureAwait(false);
await WrapApiCallAsync(
async () =>
{
Expand Down

0 comments on commit 7494c1b

Please sign in to comment.