Skip to content

Commit d8b42ff

Browse files
authored
chore: align MSTest & NUnit Workers implementation (#3042)
1 parent a732283 commit d8b42ff

9 files changed

+121
-167
lines changed

src/Playwright.MSTest/Services/BrowserService.cs src/Playwright.MSTest/BrowserService.cs

+14-4
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@
2828
using System.Threading.Tasks;
2929
using Microsoft.Playwright.TestAdapter;
3030

31-
namespace Microsoft.Playwright.MSTest.Services;
31+
namespace Microsoft.Playwright.MSTest;
3232

3333
internal class BrowserService : IWorkerService
3434
{
@@ -38,14 +38,24 @@ internal class BrowserService : IWorkerService
3838

3939
public Task DisposeAsync() => Browser?.CloseAsync() ?? Task.CompletedTask;
4040

41-
public async Task BuildAsync(PlaywrightTest parentTest)
41+
private BrowserService(IBrowser browser)
42+
{
43+
Browser = browser;
44+
}
45+
46+
public static Task<BrowserService> Register(WorkerAwareTest test, IBrowserType browserType)
47+
{
48+
return test.RegisterService("Browser", async () => new BrowserService(await CreateBrowser(browserType).ConfigureAwait(false)));
49+
}
50+
51+
private static async Task<IBrowser> CreateBrowser(IBrowserType browserType)
4252
{
4353
var accessToken = Environment.GetEnvironmentVariable("PLAYWRIGHT_SERVICE_ACCESS_TOKEN");
4454
var serviceUrl = Environment.GetEnvironmentVariable("PLAYWRIGHT_SERVICE_URL");
4555

4656
if (string.IsNullOrEmpty(accessToken) || string.IsNullOrEmpty(serviceUrl))
4757
{
48-
Browser = await parentTest!.BrowserType!.LaunchAsync(PlaywrightSettingsProvider.LaunchOptions).ConfigureAwait(false);
58+
return await browserType.LaunchAsync(PlaywrightSettingsProvider.LaunchOptions).ConfigureAwait(false);
4959
}
5060
else
5161
{
@@ -64,7 +74,7 @@ public async Task BuildAsync(PlaywrightTest parentTest)
6474
}
6575
};
6676

67-
Browser = await parentTest!.BrowserType!.ConnectAsync(wsEndpoint, connectOptions).ConfigureAwait(false);
77+
return await browserType.ConnectAsync(wsEndpoint, connectOptions).ConfigureAwait(false);
6878
}
6979
}
7080
}

src/Playwright.MSTest/BrowserTest.cs

+3-3
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,6 @@
2424

2525
using System.Collections.Generic;
2626
using System.Threading.Tasks;
27-
using Microsoft.Playwright.MSTest.Services;
2827
using Microsoft.VisualStudio.TestTools.UnitTesting;
2928

3029
namespace Microsoft.Playwright.MSTest;
@@ -45,13 +44,14 @@ public async Task<IBrowserContext> NewContextAsync(BrowserNewContextOptions? opt
4544
[TestInitialize]
4645
public async Task BrowserSetup()
4746
{
48-
Browser = (await GetService<BrowserService>().ConfigureAwait(false)).Browser;
47+
var service = await BrowserService.Register(this, BrowserType).ConfigureAwait(false);
48+
Browser = service.Browser;
4949
}
5050

5151
[TestCleanup]
5252
public async Task BrowserTearDown()
5353
{
54-
if (TestOK)
54+
if (TestOK())
5555
{
5656
foreach (var context in _contexts)
5757
{

src/Playwright.MSTest/IWorkerService.cs

-34
This file was deleted.

src/Playwright.MSTest/PageTest.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,6 @@ public class PageTest : ContextTest
3535
[TestInitialize]
3636
public async Task PageSetup()
3737
{
38-
Page = await Context!.NewPageAsync().ConfigureAwait(false);
38+
Page = await Context.NewPageAsync().ConfigureAwait(false);
3939
}
4040
}

src/Playwright.MSTest/PlaywrightTest.cs

+3-81
Original file line numberDiff line numberDiff line change
@@ -22,108 +22,30 @@
2222
* SOFTWARE.
2323
*/
2424

25-
using System;
26-
using System.Collections.Concurrent;
27-
using System.Collections.Generic;
28-
using System.Linq;
29-
using System.Threading;
3025
using System.Threading.Tasks;
31-
using Microsoft.Playwright.Core;
3226
using Microsoft.Playwright.TestAdapter;
3327
using Microsoft.VisualStudio.TestTools.UnitTesting;
3428

3529
namespace Microsoft.Playwright.MSTest;
3630

3731
[TestClass]
38-
public class PlaywrightTest
32+
public class PlaywrightTest : WorkerAwareTest
3933
{
40-
private static int _workerCount = 0;
41-
private static readonly ConcurrentStack<Worker> _allWorkers = new();
42-
private Worker? _currentWorker;
4334
private static readonly Task<IPlaywright> _playwrightTask = Microsoft.Playwright.Playwright.CreateAsync();
4435

4536
public string BrowserName { get; private set; } = null!;
4637
public IPlaywright Playwright { get; private set; } = null!;
47-
4838
public IBrowserType BrowserType { get; private set; } = null!;
4939

50-
public int WorkerIndex => _currentWorker!.WorkerIndex;
51-
5240
[TestInitialize]
53-
public async Task Setup()
41+
public async Task PlaywrightSetup()
5442
{
55-
if (PlaywrightSettingsProvider.ExpectTimeout.HasValue)
56-
{
57-
AssertionsBase.SetDefaultTimeout(PlaywrightSettingsProvider.ExpectTimeout.Value);
58-
}
59-
try
60-
{
61-
Playwright = await _playwrightTask.ConfigureAwait(false);
62-
}
63-
catch (Exception e)
64-
{
65-
Assert.Fail(e.Message, e.StackTrace);
66-
}
67-
Assert.IsNotNull(Playwright, "Playwright could not be instantiated.");
43+
Playwright = await _playwrightTask.ConfigureAwait(false);
6844
BrowserName = PlaywrightSettingsProvider.BrowserName;
6945
BrowserType = Playwright[BrowserName];
70-
71-
// get worker
72-
if (!_allWorkers.TryPop(out _currentWorker))
73-
{
74-
_currentWorker = new();
75-
}
7646
Playwright.Selectors.SetTestIdAttribute("data-testid");
7747
}
7848

79-
[TestCleanup]
80-
public async Task Teardown()
81-
{
82-
if (TestOK)
83-
{
84-
await Task.WhenAll(_currentWorker!.InstantiatedServices.Select(x => x.ResetAsync())).ConfigureAwait(false);
85-
_allWorkers.Push(_currentWorker);
86-
}
87-
else
88-
{
89-
await Task.WhenAll(_currentWorker!.InstantiatedServices.Select(x => x.DisposeAsync())).ConfigureAwait(false);
90-
_currentWorker.InstantiatedServices.Clear();
91-
}
92-
}
93-
94-
public async Task<T> GetService<T>(Func<T>? factory = null) where T : class, IWorkerService, new()
95-
{
96-
factory ??= () => new T();
97-
var serviceType = typeof(T);
98-
99-
var instance = _currentWorker!.InstantiatedServices.SingleOrDefault(x => serviceType.IsInstanceOfType(x));
100-
if (instance == null)
101-
{
102-
instance = factory();
103-
await instance.BuildAsync(this).ConfigureAwait(false);
104-
_currentWorker.InstantiatedServices.Add(instance);
105-
}
106-
107-
if (instance is not T)
108-
throw new Exception("There was a problem instantiating the service.");
109-
110-
return (T)instance;
111-
}
112-
113-
private class Worker
114-
{
115-
public int WorkerIndex { get; } = Interlocked.Increment(ref _workerCount);
116-
public List<IWorkerService> InstantiatedServices { get; } = new();
117-
}
118-
119-
protected bool TestOK
120-
{
121-
get => TestContext!.CurrentTestOutcome == UnitTestOutcome.Passed
122-
|| TestContext!.CurrentTestOutcome == UnitTestOutcome.NotRunnable;
123-
}
124-
125-
public TestContext TestContext { get; set; } = null!;
126-
12749
public static void SetDefaultExpectTimeout(float timeout) => Assertions.SetDefaultExpectTimeout(timeout);
12850

12951
public ILocatorAssertions Expect(ILocator locator) => Assertions.Expect(locator);
+83
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,83 @@
1+
using System;
2+
using System.Collections.Concurrent;
3+
using System.Collections.Generic;
4+
using System.Threading;
5+
using System.Threading.Tasks;
6+
using Microsoft.Playwright.Core;
7+
using Microsoft.Playwright.TestAdapter;
8+
using Microsoft.VisualStudio.TestTools.UnitTesting;
9+
namespace Microsoft.Playwright.MSTest;
10+
11+
public class WorkerAwareTest
12+
{
13+
public TestContext TestContext { get; set; } = null!;
14+
15+
private static readonly ConcurrentStack<Worker> _allWorkers = new();
16+
private Worker _currentWorker = null!;
17+
18+
private class Worker
19+
{
20+
private static int _lastWorkedIndex = 0;
21+
public int WorkerIndex { get; } = Interlocked.Increment(ref _lastWorkedIndex);
22+
public Dictionary<string, IWorkerService> Services = [];
23+
}
24+
25+
public int WorkerIndex => _currentWorker!.WorkerIndex;
26+
27+
public async Task<T> RegisterService<T>(string name, Func<Task<T>> factory) where T : class, IWorkerService
28+
{
29+
if (!_currentWorker.Services.ContainsKey(name))
30+
{
31+
_currentWorker.Services[name] = await factory().ConfigureAwait(false);
32+
}
33+
34+
return (_currentWorker.Services[name] as T)!;
35+
}
36+
37+
[TestInitialize]
38+
public void WorkerSetup()
39+
{
40+
if (PlaywrightSettingsProvider.ExpectTimeout.HasValue)
41+
{
42+
AssertionsBase.SetDefaultTimeout(PlaywrightSettingsProvider.ExpectTimeout.Value);
43+
}
44+
45+
if (!_allWorkers.TryPop(out _currentWorker))
46+
{
47+
_currentWorker = new();
48+
}
49+
}
50+
51+
[TestCleanup]
52+
public async Task WorkerTeardown()
53+
{
54+
if (TestOK())
55+
{
56+
foreach (var kv in _currentWorker.Services)
57+
{
58+
await kv.Value.ResetAsync().ConfigureAwait(false);
59+
}
60+
_allWorkers.Push(_currentWorker);
61+
}
62+
else
63+
{
64+
foreach (var kv in _currentWorker.Services)
65+
{
66+
await kv.Value.DisposeAsync().ConfigureAwait(false);
67+
}
68+
_currentWorker.Services.Clear();
69+
}
70+
}
71+
72+
protected bool TestOK()
73+
{
74+
return TestContext!.CurrentTestOutcome == UnitTestOutcome.Passed
75+
|| TestContext!.CurrentTestOutcome == UnitTestOutcome.NotRunnable;
76+
}
77+
}
78+
79+
public interface IWorkerService
80+
{
81+
public Task ResetAsync();
82+
public Task DisposeAsync();
83+
}

src/Playwright.NUnit/ContextTest.cs

+6-6
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,12 @@ public class ContextTest : BrowserTest
3131
{
3232
public IBrowserContext Context { get; private set; } = null!;
3333

34+
[SetUp]
35+
public async Task ContextSetup()
36+
{
37+
Context = await NewContext(ContextOptions()).ConfigureAwait(false);
38+
}
39+
3440
public virtual BrowserNewContextOptions ContextOptions()
3541
{
3642
return new()
@@ -39,10 +45,4 @@ public virtual BrowserNewContextOptions ContextOptions()
3945
ColorScheme = ColorScheme.Light,
4046
};
4147
}
42-
43-
[SetUp]
44-
public async Task ContextSetup()
45-
{
46-
Context = await NewContext(ContextOptions()).ConfigureAwait(false);
47-
}
4848
}

src/Playwright.NUnit/IWorkerService.cs

-33
This file was deleted.

0 commit comments

Comments
 (0)