From b6c29e7c1dcb123b3b9ba00b689ba51e87adb65a Mon Sep 17 00:00:00 2001 From: Steven Kirk Date: Thu, 2 May 2024 15:15:51 +0200 Subject: [PATCH 1/4] Make sure we don't close apps we didn't start. Tests currently failing. --- src/FlaUI.WebDriver.UITests/SessionTests.cs | 29 +++++++++++++++++++++ 1 file changed, 29 insertions(+) diff --git a/src/FlaUI.WebDriver.UITests/SessionTests.cs b/src/FlaUI.WebDriver.UITests/SessionTests.cs index f7f8cdd..2e46c23 100644 --- a/src/FlaUI.WebDriver.UITests/SessionTests.cs +++ b/src/FlaUI.WebDriver.UITests/SessionTests.cs @@ -315,6 +315,35 @@ public void NewCommandTimeout_NotExpired_DoesNotEndSession() Assert.That(() => driver.Title, Throws.Nothing); } + [Test, Explicit(("Sometimes multiple processes are left open"))] + public void NewCommandTimeout_Expired_DoesNot_CloseApp_AppTopLevelWindowTitleMatch() + { + using var testAppProcess = new TestAppProcess(); + var driverOptions = FlaUIDriverOptions.AppTopLevelWindowTitleMatch("FlaUI WPF Test App"); + driverOptions.AddAdditionalOption("appium:newCommandTimeout", 1); + using var driver = new RemoteWebDriver(WebDriverFixture.WebDriverUrl, driverOptions); + + System.Threading.Thread.Sleep(TimeSpan.FromSeconds(1) + WebDriverFixture.SessionCleanupInterval * 2); + + Assert.That(testAppProcess.Process.HasExited, Is.False); + Assert.That(() => driver.Title, Throws.TypeOf().With.Message.Matches("No active session with ID '.*'")); + } + + [Test] + public void NewCommandTimeout_Expired_DoesNot_CloseApp_AppTopLevelWindow() + { + using var testAppProcess = new TestAppProcess(); + var windowHandle = string.Format("0x{0:x}", testAppProcess.Process.MainWindowHandle); + var driverOptions = FlaUIDriverOptions.AppTopLevelWindow(windowHandle); + driverOptions.AddAdditionalOption("appium:newCommandTimeout", 1); + using var driver = new RemoteWebDriver(WebDriverFixture.WebDriverUrl, driverOptions); + + System.Threading.Thread.Sleep(TimeSpan.FromSeconds(1) + WebDriverFixture.SessionCleanupInterval * 2); + + Assert.That(testAppProcess.Process.HasExited, Is.False); + Assert.That(() => driver.Title, Throws.TypeOf().With.Message.Matches("No active session with ID '.*'")); + } + [TestCase("123")] [TestCase(false)] [TestCase("not a number")] From bd110107b9e2f8dd36aa1dfe9d814d0c38484555 Mon Sep 17 00:00:00 2001 From: Steven Kirk Date: Thu, 2 May 2024 15:20:18 +0200 Subject: [PATCH 2/4] Don't close apps we didn't start. When a session times out, don't close the app if the session was attached to an existing app using `appTopLevelWindow` or `appTopLevelWindowTitleMatch`. --- src/FlaUI.WebDriver/Controllers/SessionController.cs | 5 ++++- src/FlaUI.WebDriver/Session.cs | 6 ++++-- 2 files changed, 8 insertions(+), 3 deletions(-) diff --git a/src/FlaUI.WebDriver/Controllers/SessionController.cs b/src/FlaUI.WebDriver/Controllers/SessionController.cs index 1b94db3..4759dda 100644 --- a/src/FlaUI.WebDriver/Controllers/SessionController.cs +++ b/src/FlaUI.WebDriver/Controllers/SessionController.cs @@ -30,6 +30,7 @@ public async Task CreateNewSession([FromBody] CreateSessionRequest .Select(capabillities => matchedCapabilities!); Core.Application? app; + var isAppOwnedBySession = false; var capabilities = matchingCapabilities.FirstOrDefault(); if (capabilities == null) { @@ -69,6 +70,8 @@ public async Task CreateNewSession([FromBody] CreateSessionRequest throw WebDriverResponseException.InvalidArgument($"Starting app '{appPath}' with arguments '{appArguments}' threw an exception: {e.Message}"); } } + + isAppOwnedBySession = true; } else if (TryGetStringCapability(capabilities, "appium:appTopLevelWindow", out var appTopLevelWindowString)) { @@ -84,7 +87,7 @@ public async Task CreateNewSession([FromBody] CreateSessionRequest { throw WebDriverResponseException.InvalidArgument("One of appium:app, appium:appTopLevelWindow or appium:appTopLevelWindowTitleMatch must be passed as a capability"); } - var session = new Session(app); + var session = new Session(app, isAppOwnedBySession); if(TryGetNumberCapability(capabilities, "appium:newCommandTimeout", out var newCommandTimeout)) { session.NewCommandTimeout = TimeSpan.FromSeconds(newCommandTimeout); diff --git a/src/FlaUI.WebDriver/Session.cs b/src/FlaUI.WebDriver/Session.cs index 75309b8..9991fd9 100644 --- a/src/FlaUI.WebDriver/Session.cs +++ b/src/FlaUI.WebDriver/Session.cs @@ -6,13 +6,14 @@ namespace FlaUI.WebDriver { public class Session : IDisposable { - public Session(Application? app) + public Session(Application? app, bool isAppOwnedBySession) { App = app; SessionId = Guid.NewGuid().ToString(); Automation = new UIA3Automation(); InputState = new InputState(); TimeoutsConfiguration = new TimeoutsConfiguration(); + IsAppOwnedBySession = isAppOwnedBySession; if (app != null) { @@ -30,6 +31,7 @@ public Session(Application? app) public TimeSpan ImplicitWaitTimeout => TimeSpan.FromMilliseconds(TimeoutsConfiguration.ImplicitWaitTimeoutMs); public TimeSpan PageLoadTimeout => TimeSpan.FromMilliseconds(TimeoutsConfiguration.PageLoadTimeoutMs); public TimeSpan? ScriptTimeout => TimeoutsConfiguration.ScriptTimeoutMs.HasValue ? TimeSpan.FromMilliseconds(TimeoutsConfiguration.ScriptTimeoutMs.Value) : null; + public bool IsAppOwnedBySession { get; } public TimeoutsConfiguration TimeoutsConfiguration { get; set; } @@ -124,7 +126,7 @@ public void RemoveKnownWindow(Window window) public void Dispose() { - if (App != null && !App.HasExited) + if (IsAppOwnedBySession && App != null && !App.HasExited) { App.Close(); } From 84a3af7b39593b0788acb5e803cb96bc61b65acb Mon Sep 17 00:00:00 2001 From: Steven Kirk Date: Thu, 2 May 2024 15:46:57 +0200 Subject: [PATCH 3/4] Make test names conform to convention. --- src/FlaUI.WebDriver.UITests/SessionTests.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/FlaUI.WebDriver.UITests/SessionTests.cs b/src/FlaUI.WebDriver.UITests/SessionTests.cs index 2e46c23..98d5dd2 100644 --- a/src/FlaUI.WebDriver.UITests/SessionTests.cs +++ b/src/FlaUI.WebDriver.UITests/SessionTests.cs @@ -316,7 +316,7 @@ public void NewCommandTimeout_NotExpired_DoesNotEndSession() } [Test, Explicit(("Sometimes multiple processes are left open"))] - public void NewCommandTimeout_Expired_DoesNot_CloseApp_AppTopLevelWindowTitleMatch() + public void NewCommandTimeout_SessionWithAppTopLevelWindowTitleMatch_ClosesSessionButDoesNotCloseApp() { using var testAppProcess = new TestAppProcess(); var driverOptions = FlaUIDriverOptions.AppTopLevelWindowTitleMatch("FlaUI WPF Test App"); @@ -330,7 +330,7 @@ public void NewCommandTimeout_Expired_DoesNot_CloseApp_AppTopLevelWindowTitleMat } [Test] - public void NewCommandTimeout_Expired_DoesNot_CloseApp_AppTopLevelWindow() + public void NewCommandTimeout_SessionWithAppTopLevelWindow_ClosesSessionButDoesNotCloseApp() { using var testAppProcess = new TestAppProcess(); var windowHandle = string.Format("0x{0:x}", testAppProcess.Process.MainWindowHandle); From 18193c232fb3527b30e8c7086e1c1725944f88aa Mon Sep 17 00:00:00 2001 From: Steven Kirk Date: Thu, 2 May 2024 16:43:05 +0200 Subject: [PATCH 4/4] Add extra checks that process has not exited. --- src/FlaUI.WebDriver.UITests/SessionTests.cs | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/src/FlaUI.WebDriver.UITests/SessionTests.cs b/src/FlaUI.WebDriver.UITests/SessionTests.cs index 98d5dd2..2d5940d 100644 --- a/src/FlaUI.WebDriver.UITests/SessionTests.cs +++ b/src/FlaUI.WebDriver.UITests/SessionTests.cs @@ -129,6 +129,10 @@ public void NewSession_AppTopLevelWindow_IsSupported() var title = driver.Title; Assert.That(title, Is.EqualTo("FlaUI WPF Test App")); + + driver.Quit(); + + Assert.That(testAppProcess.Process.HasExited, Is.False); } [Test] @@ -166,6 +170,10 @@ public void NewSession_AppTopLevelWindowTitleMatch_IsSupported(string match) var title = driver.Title; Assert.That(title, Is.EqualTo("FlaUI WPF Test App")); + + driver.Quit(); + + Assert.That(testAppProcess.Process.HasExited, Is.False); } [Test, Ignore("Sometimes multiple processes are left open")]