Skip to content

Commit 55d726a

Browse files
authored
fix: restrict access to _activeInvocations in concurrent RouteHandler methods … (#2887)
1 parent 04431af commit 55d726a

File tree

2 files changed

+42
-3
lines changed

2 files changed

+42
-3
lines changed

src/Playwright.Tests/BrowserContextRouteTests.cs

+38
Original file line numberDiff line numberDiff line change
@@ -177,6 +177,44 @@ await page.RouteAsync("**/non-empty.html", (route) =>
177177
Assert.AreEqual("context", await response.TextAsync());
178178
}
179179

180+
181+
[PlaywrightTest("browsercontext-route.spec.ts", "should not throw on concurrent requests")]
182+
public async Task ShouldNotThrowOnConcurrentRequest()
183+
{
184+
await using var context = await Browser.NewContextAsync();
185+
186+
// provide an html page which performs a high load of requests
187+
await context.RouteAsync("**/empty.html", async (route) =>
188+
{
189+
await route.FulfillAsync(new()
190+
{
191+
Status = (int)HttpStatusCode.OK,
192+
Body = @"<script>document.addEventListener('DOMContentLoaded', async () => { while (true) { await fetch('test.json') }})</script>"
193+
});
194+
});
195+
196+
// provide a simple json to fulfill the requests
197+
await context.RouteAsync("**/test.json", async (route) =>
198+
{
199+
await route.FulfillAsync(new()
200+
{
201+
Status = (int)HttpStatusCode.OK,
202+
Body = @"[]"
203+
});
204+
});
205+
206+
207+
// open 10 pages to generate load on the RouteHandler
208+
foreach (int pageNr in Enumerable.Range(0, 10))
209+
await (await context.NewPageAsync())
210+
.GotoAsync(Server.EmptyPage);
211+
212+
// let the test run for 5 second
213+
await Task.Delay(5000);
214+
215+
// unobserved task exceptions are automatically collected by the PlaywrightTest attribute
216+
}
217+
180218
[PlaywrightTest("browsercontext-route.spec.ts", "should support the times parameter with route matching")]
181219
public async Task ShouldSupportTheTimesParameterWithRouteMatching()
182220
{

src/Playwright/Core/RouteHandler.cs

+4-3
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@
2323
*/
2424

2525
using System;
26+
using System.Collections.Concurrent;
2627
using System.Collections.Generic;
2728
using System.Text.RegularExpressions;
2829
using System.Threading.Tasks;
@@ -32,7 +33,7 @@ namespace Microsoft.Playwright.Core;
3233

3334
internal class RouteHandler
3435
{
35-
private readonly HashSet<HandlerInvocation> _activeInvocations = new HashSet<HandlerInvocation>();
36+
private readonly IDictionary<HandlerInvocation, bool> _activeInvocations = new ConcurrentDictionary<HandlerInvocation, bool>();
3637

3738
private bool _ignoreException;
3839

@@ -88,7 +89,7 @@ public async Task<bool> HandleAsync(Route route)
8889
Complete = new TaskCompletionSource<bool>(),
8990
Route = route,
9091
};
91-
_activeInvocations.Add(handlerInvocation);
92+
_activeInvocations[handlerInvocation] = false;
9293
try
9394
{
9495
return await HandleInternalAsync(route).ConfigureAwait(false);
@@ -122,7 +123,7 @@ public async Task StopAsync(UnrouteBehavior behavior)
122123
else
123124
{
124125
var tasks = new List<Task>();
125-
foreach (var activation in _activeInvocations)
126+
foreach (var activation in _activeInvocations.Keys)
126127
{
127128
if (!activation.Route.DidThrow)
128129
{

0 commit comments

Comments
 (0)