From d77647d0915ef8712ed97e6e39c7df17715e828f Mon Sep 17 00:00:00 2001 From: Pavel Vostretsov Date: Sun, 14 Jan 2024 21:07:27 +0500 Subject: [PATCH] playwright playground --- .github/workflows/actions.yml | 12 +- DbViewer.Tests/DbViewer.Tests.csproj | 1 + .../FrontTests/AutoFill/PwAutoFill.cs | 60 ++++ .../BusinessObjectsChangeValuesTest.cs | 334 +++++++++--------- .../FrontTests/BusinessObjectsDeleteTest.cs | 73 ++-- .../FrontTests/BusinessObjectsSearchTests.cs | 91 ++--- .../FrontTests/Controls/Accordion.cs | 31 ++ .../Controls/AccordionFieldValue.cs | 33 ++ .../FrontTests/Controls/AccordionRow.cs | 25 ++ .../FrontTests/Controls/AccordionToggle.cs | 16 + .../Controls/BusinessObjectFilter.cs | 28 ++ .../Controls/BusinessObjectGroup.cs | 21 ++ .../FrontTests/Controls/BusinessObjectItem.cs | 16 + .../Controls/BusinessObjectTableRow.cs | 24 ++ .../Controls/ConfirmDeleteObjectModal.cs | 17 + .../FrontTests/Controls/FilterModal.cs | 30 ++ .../Pages/PwBusinessObjectDetailsPage.cs | 25 ++ .../Pages/PwBusinessObjectTablePage.cs | 50 +++ .../FrontTests/Pages/PwBusinessObjectsPage.cs | 32 ++ .../Playwright/AssertionExtensions.cs | 146 ++++++++ .../FrontTests/Playwright/Browser.cs | 117 ++++++ .../FrontTests/Playwright/PwButton.cs | 12 + .../FrontTests/Playwright/PwCheckbox.cs | 12 + .../FrontTests/Playwright/PwControlBase.cs | 17 + .../FrontTests/Playwright/PwControlList.cs | 48 +++ .../FrontTests/Playwright/PwDatePicker.cs | 12 + .../FrontTests/Playwright/PwInput.cs | 12 + .../FrontTests/Playwright/PwLabel.cs | 12 + .../FrontTests/Playwright/PwLink.cs | 12 + .../FrontTests/Playwright/PwPageBase.cs | 17 + .../FrontTests/Playwright/PwSelect.cs | 17 + DbViewer.Tests/FrontTests/PlaywrightSetup.cs | 66 ++++ global.json | 2 +- 33 files changed, 1170 insertions(+), 251 deletions(-) create mode 100644 DbViewer.Tests/FrontTests/AutoFill/PwAutoFill.cs create mode 100644 DbViewer.Tests/FrontTests/Controls/Accordion.cs create mode 100644 DbViewer.Tests/FrontTests/Controls/AccordionFieldValue.cs create mode 100644 DbViewer.Tests/FrontTests/Controls/AccordionRow.cs create mode 100644 DbViewer.Tests/FrontTests/Controls/AccordionToggle.cs create mode 100644 DbViewer.Tests/FrontTests/Controls/BusinessObjectFilter.cs create mode 100644 DbViewer.Tests/FrontTests/Controls/BusinessObjectGroup.cs create mode 100644 DbViewer.Tests/FrontTests/Controls/BusinessObjectItem.cs create mode 100644 DbViewer.Tests/FrontTests/Controls/BusinessObjectTableRow.cs create mode 100644 DbViewer.Tests/FrontTests/Controls/ConfirmDeleteObjectModal.cs create mode 100644 DbViewer.Tests/FrontTests/Controls/FilterModal.cs create mode 100644 DbViewer.Tests/FrontTests/Pages/PwBusinessObjectDetailsPage.cs create mode 100644 DbViewer.Tests/FrontTests/Pages/PwBusinessObjectTablePage.cs create mode 100644 DbViewer.Tests/FrontTests/Pages/PwBusinessObjectsPage.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/AssertionExtensions.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/Browser.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/PwButton.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/PwCheckbox.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/PwControlBase.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/PwControlList.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/PwDatePicker.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/PwInput.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/PwLabel.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/PwLink.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/PwPageBase.cs create mode 100644 DbViewer.Tests/FrontTests/Playwright/PwSelect.cs create mode 100644 DbViewer.Tests/FrontTests/PlaywrightSetup.cs diff --git a/.github/workflows/actions.yml b/.github/workflows/actions.yml index bc2b8772..5d55737b 100644 --- a/.github/workflows/actions.yml +++ b/.github/workflows/actions.yml @@ -46,11 +46,14 @@ jobs: - name: Pack dotnet run: dotnet pack --no-build --configuration Release ./DbViewer.sln + - name: Install Playwright + run: pwsh ./DbViewer.Tests/bin/Release/net6.0/playwright.ps1 install chromium --with-deps + - name: Build front run: yarn --cwd db-viewer-ui build - name: Check C# code style - run: dotnet jb cleanupcode DbViewer.sln --profile=CatalogueCleanup --exclude=./DbViewer.TestApi/Migrations/*.cs --verbosity=WARN && git diff --exit-code -- ':!./db-viewer-ui/.yarn' + run: dotnet jb cleanupcode DbViewer.sln --profile=CatalogueCleanup --exclude=./DbViewer.TestApi/Migrations/*.cs --verbosity=WARN --no-build && git diff --exit-code -- ':!./db-viewer-ui/.yarn' - name: Check front code run: yarn --cwd db-viewer-ui lint @@ -79,6 +82,13 @@ jobs: **/*.nupkg **/*.tgz if-no-files-found: error + + - name: Upload Playwright traces + if: failure() + uses: actions/upload-artifact@v3 + with: + name: traces + path: DbViewer.Tests/bin/Release/net6.0/out/*.zip publish: runs-on: ubuntu-20.04 diff --git a/DbViewer.Tests/DbViewer.Tests.csproj b/DbViewer.Tests/DbViewer.Tests.csproj index bc0b37ad..7025c767 100644 --- a/DbViewer.Tests/DbViewer.Tests.csproj +++ b/DbViewer.Tests/DbViewer.Tests.csproj @@ -10,6 +10,7 @@ + diff --git a/DbViewer.Tests/FrontTests/AutoFill/PwAutoFill.cs b/DbViewer.Tests/FrontTests/AutoFill/PwAutoFill.cs new file mode 100644 index 00000000..643b9add --- /dev/null +++ b/DbViewer.Tests/FrontTests/AutoFill/PwAutoFill.cs @@ -0,0 +1,60 @@ +using System; +using System.Linq; +using System.Reflection; + +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.AutoFill +{ + public class PwAutoFill + { + public static TPage InitializePage(IPage page) + where TPage : PwPageBase + { + var newPage = (TPage)Activator.CreateInstance(typeof(TPage), page)!; + InitializeControls(newPage, newPage.Page, null); + return newPage; + } + + public static void InitializeControls(object instance, IPage page, ILocator? parent) + { + var properties = instance + .GetType() + .GetProperties(BindingFlags.Instance | BindingFlags.Public) + .Where(p => p.CanWrite && typeof(PwControlBase).IsAssignableFrom(p.PropertyType) && !p.GetCustomAttributes().Any()); + + foreach (var property in properties) + { + var locator = LocatorForProperty(property, page, parent); + var value = Activator.CreateInstance(property.PropertyType, locator)!; + InitializeControls(value, page, locator); + property.SetValue(instance, value); + } + } + + public static ILocator LocatorForProperty(PropertyInfo property, IPage page, ILocator? parent) + { + var selector = property + .GetCustomAttributes() + .Select(x => x.Selector.ToString()) + .FirstOrDefault(); + + if (string.IsNullOrEmpty(selector)) + return GetByTestId(page, parent, property.Name); + + var selectors = selector.Split(" ", StringSplitOptions.TrimEntries | StringSplitOptions.RemoveEmptyEntries); + return selectors + .Aggregate( + parent, + (current, s) => s.StartsWith("##") + ? GetByTestId(page, current, s[2..]) + : GetLocator(page, current, s) + )!; + } + + private static ILocator GetByTestId(IPage page, ILocator? parent, string tid) => parent == null ? page.GetByTestId(tid) : parent.GetByTestId(tid); + private static ILocator GetLocator(IPage page, ILocator? parent, string loc) => parent == null ? page.Locator(loc) : parent.Locator(loc); + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/BusinessObjectsChangeValuesTest.cs b/DbViewer.Tests/FrontTests/BusinessObjectsChangeValuesTest.cs index 0d28ab74..1137c27d 100644 --- a/DbViewer.Tests/FrontTests/BusinessObjectsChangeValuesTest.cs +++ b/DbViewer.Tests/FrontTests/BusinessObjectsChangeValuesTest.cs @@ -1,5 +1,6 @@ using System; using System.Linq; +using System.Threading.Tasks; using Cassandra.Data.Linq; @@ -9,13 +10,11 @@ using NUnit.Framework; -using OpenQA.Selenium; - using SkbKontur.DbViewer.TestApi.Cql; using SkbKontur.DbViewer.TestApi.EntityFramework; using SkbKontur.DbViewer.TestApi.Impl.Document; -using SkbKontur.DbViewer.Tests.FrontTests.Helpers; using SkbKontur.DbViewer.Tests.FrontTests.Pages; +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; namespace SkbKontur.DbViewer.Tests.FrontTests { @@ -33,38 +32,38 @@ public BusinessObjectsChangeValuesTest(string documentName) /// Проверяем, что заданные значения сохраняются /// [Test] - public void TestChangeOrganizationName() + public async Task TestChangeOrganizationName() { var documentId = CreateDocument(documentName); - using var browser = new BrowserForTests(); - var businessObjectEditingPage = browser.LoginAsSuperUser().SwitchTo(documentName, $"Id={documentId}"); + await using var browser = new Browser(); + var businessObjectEditingPage = await (await browser.LoginAsSuperUser()).SwitchTo(documentName, $"Id={documentId}"); var filledNumberRow = businessObjectEditingPage.RootAccordion.FindField("DocumentNumber"); - filledNumberRow.FieldValue.WaitText("123"); - filledNumberRow.Edit.Click(); - filledNumberRow.FieldValue.Input.ClearAndInputText("2qwe123QWE2"); - filledNumberRow.Save.Click(); - filledNumberRow.FieldValue.WaitText("2qwe123QWE2"); + await filledNumberRow.FieldValue.WaitText("123"); + await filledNumberRow.Edit.Click(); + await filledNumberRow.FieldValue.Input.ClearAndInputText("2qwe123QWE2"); + await filledNumberRow.Save.Click(); + await filledNumberRow.FieldValue.WaitText("2qwe123QWE2"); - browser.WebDriver.Navigate().Refresh(); + await businessObjectEditingPage.Page.ReloadAsync(); filledNumberRow = businessObjectEditingPage.RootAccordion.FindField("DocumentNumber"); - filledNumberRow.FieldValue.WaitText("2qwe123QWE2"); + await filledNumberRow.FieldValue.WaitText("2qwe123QWE2"); GetDocument(documentName, documentId).DocumentNumber.Should().Be("2qwe123QWE2"); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); var unfilledNumberRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_OrdersNumber"); - unfilledNumberRow.FieldValue.WaitText("null"); - unfilledNumberRow.Edit.Click(); - unfilledNumberRow.FieldValue.Input.ClearAndInputText("123"); - unfilledNumberRow.Save.Click(); - unfilledNumberRow.FieldValue.WaitText("123"); - - browser.WebDriver.Navigate().Refresh(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await unfilledNumberRow.FieldValue.WaitText("null"); + await unfilledNumberRow.Edit.Click(); + await unfilledNumberRow.FieldValue.Input.ClearAndInputText("123"); + await unfilledNumberRow.Save.Click(); + await unfilledNumberRow.FieldValue.WaitText("123"); + + await businessObjectEditingPage.Page.ReloadAsync(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); unfilledNumberRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_OrdersNumber"); - unfilledNumberRow.FieldValue.WaitText("123"); + await unfilledNumberRow.FieldValue.WaitText("123"); GetDocument(documentName, documentId).DocumentContent.OrdersNumber.Should().Be("123"); } @@ -74,46 +73,44 @@ public void TestChangeOrganizationName() /// Проверяем, что заданные значения сохраняются /// [Test] - public void TestChangeDocumentDates() + public async Task TestChangeDocumentDates() { var documentId = CreateDocument(documentName); - using var browser = new BrowserForTests(); - var businessObjectEditingPage = browser.LoginAsSuperUser().SwitchTo(documentName, $"Id={documentId}"); + await using var browser = new Browser(); + var businessObjectEditingPage = await (await browser.LoginAsSuperUser()).SwitchTo(documentName, $"Id={documentId}"); var filledDateRow = businessObjectEditingPage.RootAccordion.FindField("DocumentDate"); - filledDateRow.FieldValue.WaitTextContains("2014-12-11"); - filledDateRow.Edit.Click(); - - var expectedOffset = documentName == "CqlDocument" ? TimeSpan.Zero : DateTimeOffset.Now.Offset; - var expectedOffsetStr = $"+{expectedOffset:hh':'mm}"; - filledDateRow.FieldValue.Date.ClearAndInputText("13.12.2014"); - filledDateRow.FieldValue.Time.ClearAndInputText("10:18:13.567"); - filledDateRow.FieldValue.TimeOffsetLabel.WaitText(expectedOffsetStr); - filledDateRow.Save.Click(); - var expectedStr = $"2014-12-13T10:18:13.567{expectedOffsetStr}"; - filledDateRow.FieldValue.WaitTextContains(expectedStr); - - browser.WebDriver.Navigate().Refresh(); + await filledDateRow.FieldValue.WaitTextContains("2014-12-11"); + await filledDateRow.Edit.Click(); + + await filledDateRow.FieldValue.Date.ClearAndInputText("13.10.2016"); + await filledDateRow.FieldValue.Time.ClearAndInputText("10:18:13.567"); + await filledDateRow.FieldValue.TimeOffsetLabel.WaitText("+00:00"); + await filledDateRow.Save.Click(); + var expectedStr = "2016-10-13T10:18:13.567+00:00"; + await filledDateRow.FieldValue.WaitTextContains(expectedStr); + + await businessObjectEditingPage.Page.ReloadAsync(); filledDateRow = businessObjectEditingPage.RootAccordion.FindField("DocumentDate"); - filledDateRow.FieldValue.WaitTextContains(expectedStr); + await filledDateRow.FieldValue.WaitTextContains(expectedStr); - GetDocument(documentName, documentId).DocumentDate.Should().Be(new DateTimeOffset(2014, 12, 13, 10, 18, 13, 567, expectedOffset)); + GetDocument(documentName, documentId).DocumentDate.Should().Be(new DateTimeOffset(2016, 10, 13, 10, 18, 13, 567, TimeSpan.Zero)); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); var unfilledDateRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_DeliveryDate"); - unfilledDateRow.FieldValue.WaitText("null"); - unfilledDateRow.Edit.Click(); - unfilledDateRow.FieldValue.Date.ClearAndInputText("14.12.2014"); - unfilledDateRow.FieldValue.Time.ClearAndInputText("20:19"); - unfilledDateRow.FieldValue.TimeZoneSelect.SelectValueByText("UTC"); - unfilledDateRow.Save.Click(); - unfilledDateRow.FieldValue.WaitText("2014-12-14T20:19:00Z"); - - browser.WebDriver.Navigate().Refresh(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await unfilledDateRow.FieldValue.WaitText("null"); + await unfilledDateRow.Edit.Click(); + await unfilledDateRow.FieldValue.Date.ClearAndInputText("14.12.2014"); + await unfilledDateRow.FieldValue.Time.ClearAndInputText("20:19"); + await unfilledDateRow.FieldValue.TimeZoneSelect.SelectValueByText("UTC"); + await unfilledDateRow.Save.Click(); + await unfilledDateRow.FieldValue.WaitText("2014-12-14T20:19:00Z"); + + await businessObjectEditingPage.Page.ReloadAsync(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); unfilledDateRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_DeliveryDate"); - unfilledDateRow.FieldValue.WaitText("2014-12-14T20:19:00Z"); + await unfilledDateRow.FieldValue.WaitText("2014-12-14T20:19:00Z"); GetDocument(documentName, documentId).DocumentContent.DeliveryDate.Should().Be(new DateTime(2014, 12, 14, 20, 19, 00, 00, DateTimeKind.Utc)); } @@ -124,40 +121,40 @@ public void TestChangeDocumentDates() /// Меняем на пустое значение, ждем, что засетится null. /// [Test] - public void TestChangeEnumValue() + public async Task TestChangeEnumValue() { var documentId = CreateDocument(documentName); - using var browser = new BrowserForTests(); - var businessObjectEditingPage = browser.LoginAsSuperUser().SwitchTo(documentName, $"Id={documentId}"); + await using var browser = new Browser(); + var businessObjectEditingPage = await (await browser.LoginAsSuperUser()).SwitchTo(documentName, $"Id={documentId}"); var filledEnumRow = businessObjectEditingPage.RootAccordion.FindField("DocumentType"); - filledEnumRow.FieldValue.WaitText("Orders"); - filledEnumRow.Edit.Click(); - filledEnumRow.FieldValue.EnumSelect.WaitItems(new[] {"Orders", "Desadv"}); - filledEnumRow.FieldValue.EnumSelect.SelectValueByText("Desadv"); - filledEnumRow.Save.Click(); - filledEnumRow.FieldValue.WaitText("Desadv"); - - browser.WebDriver.Navigate().Refresh(); + await filledEnumRow.FieldValue.WaitText("Orders"); + await filledEnumRow.Edit.Click(); + await filledEnumRow.FieldValue.EnumSelect.WaitItems(new[] {"Orders", "Desadv"}); + await filledEnumRow.FieldValue.EnumSelect.SelectValueByText("Desadv"); + await filledEnumRow.Save.Click(); + await filledEnumRow.FieldValue.WaitText("Desadv"); + + await businessObjectEditingPage.Page.ReloadAsync(); filledEnumRow = businessObjectEditingPage.RootAccordion.FindField("DocumentType"); - filledEnumRow.FieldValue.WaitText("Desadv"); + await filledEnumRow.FieldValue.WaitText("Desadv"); GetDocument(documentName, documentId).DocumentType.Should().Be(DocumentType.Desadv); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); var nullableEnumRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_OrderStatus"); - nullableEnumRow.FieldValue.WaitText("Processed"); - nullableEnumRow.Edit.Click(); - nullableEnumRow.FieldValue.EnumSelect.WaitItems(new[] {"null", "Processed", "Failed"}); - nullableEnumRow.FieldValue.EnumSelect.SelectValueByText("null"); - nullableEnumRow.Save.Click(); - nullableEnumRow.FieldValue.WaitText("null"); - - browser.WebDriver.Navigate().Refresh(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await nullableEnumRow.FieldValue.WaitText("Processed"); + await nullableEnumRow.Edit.Click(); + await nullableEnumRow.FieldValue.EnumSelect.WaitItems(new[] {"null", "Processed", "Failed"}); + await nullableEnumRow.FieldValue.EnumSelect.SelectValueByText("null"); + await nullableEnumRow.Save.Click(); + await nullableEnumRow.FieldValue.WaitText("null"); + + await businessObjectEditingPage.Page.ReloadAsync(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); nullableEnumRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_OrderStatus"); - nullableEnumRow.FieldValue.WaitText("null"); + await nullableEnumRow.FieldValue.WaitText("null"); GetDocument(documentName, documentId).DocumentContent.OrderStatus.Should().BeNull(); } @@ -166,40 +163,40 @@ public void TestChangeEnumValue() /// Меняем значение с false на true и обратно, во всех случаях проверяем, что значение сохранится /// [Test] - public void TestChangeBooleanValue() + public async Task TestChangeBooleanValue() { var documentId = CreateDocument(documentName); - using var browser = new BrowserForTests(); - var businessObjectEditingPage = browser.LoginAsSuperUser().SwitchTo(documentName, $"Id={documentId}"); + await using var browser = new Browser(); + var businessObjectEditingPage = await (await browser.LoginAsSuperUser()).SwitchTo(documentName, $"Id={documentId}"); var booleanRow = businessObjectEditingPage.RootAccordion.FindField("IsLargeDocument"); - booleanRow.FieldValue.WaitText("false"); - booleanRow.Edit.Click(); - booleanRow.FieldValue.BooleanSelect.WaitItems(new[] {"false", "true"}); - booleanRow.FieldValue.BooleanSelect.SelectValueByText("true"); - booleanRow.Save.Click(); - booleanRow.FieldValue.WaitText("true"); - - browser.WebDriver.Navigate().Refresh(); + await booleanRow.FieldValue.WaitText("false"); + await booleanRow.Edit.Click(); + await booleanRow.FieldValue.BooleanSelect.WaitItems(new[] {"true", "false"}); + await booleanRow.FieldValue.BooleanSelect.SelectValueByText("true"); + await booleanRow.Save.Click(); + await booleanRow.FieldValue.WaitText("true"); + + await businessObjectEditingPage.Page.ReloadAsync(); booleanRow = businessObjectEditingPage.RootAccordion.FindField("IsLargeDocument"); - booleanRow.FieldValue.WaitText("true"); + await booleanRow.FieldValue.WaitText("true"); GetDocument(documentName, documentId).IsLargeDocument.Should().BeTrue(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); var nullableBoolRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_WasRead"); - nullableBoolRow.FieldValue.WaitText("true"); - nullableBoolRow.Edit.Click(); - nullableBoolRow.FieldValue.BooleanSelect.WaitItems(new[] {"null", "false", "true"}); - nullableBoolRow.FieldValue.BooleanSelect.SelectValueByText("null"); - nullableBoolRow.Save.Click(); - nullableBoolRow.FieldValue.WaitText("null"); - - browser.WebDriver.Navigate().Refresh(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await nullableBoolRow.FieldValue.WaitText("true"); + await nullableBoolRow.Edit.Click(); + await nullableBoolRow.FieldValue.BooleanSelect.WaitItems(new[] {"null", "true", "false"}); + await nullableBoolRow.FieldValue.BooleanSelect.SelectValueByText("null"); + await nullableBoolRow.Save.Click(); + await nullableBoolRow.FieldValue.WaitText("null"); + + await businessObjectEditingPage.Page.ReloadAsync(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); nullableBoolRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_WasRead"); - nullableBoolRow.FieldValue.WaitText("null"); + await nullableBoolRow.FieldValue.WaitText("null"); GetDocument(documentName, documentId).DocumentContent.WasRead.Should().BeNull(); } @@ -208,58 +205,55 @@ public void TestChangeBooleanValue() /// Меняем значение числового поля, ждем, что значение сохранится /// [Test] - public void TestChangeNumericValue() + public async Task TestChangeNumericValue() { var documentId = CreateDocument(documentName); - using var browser = new BrowserForTests(); - var businessObjectEditingPage = browser.LoginAsSuperUser().SwitchTo(documentName, $"Id={documentId}"); + await using var browser = new Browser(); + var businessObjectEditingPage = await (await browser.LoginAsSuperUser()).SwitchTo(documentName, $"Id={documentId}"); var integerRow = businessObjectEditingPage.RootAccordion.FindField("ShardNumber"); - integerRow.FieldValue.WaitText("0"); - integerRow.Edit.Click(); - integerRow.FieldValue.Input.AppendText($"{Keys.Control}a"); - integerRow.FieldValue.Input.AppendText("12.,45"); - integerRow.Save.Click(); - integerRow.FieldValue.WaitText("1245"); - integerRow.Edit.Click(); - integerRow.FieldValue.Input.AppendText($"{Keys.Control}a"); - integerRow.FieldValue.Input.AppendText("14ab"); - integerRow.Save.Click(); - integerRow.FieldValue.WaitText("14"); - - browser.WebDriver.Navigate().Refresh(); + await integerRow.FieldValue.WaitText("0"); + await integerRow.Edit.Click(); + await integerRow.FieldValue.Input.SelectAndInputText("12.,45"); + await integerRow.Save.Click(); + await integerRow.FieldValue.WaitText("1245"); + await integerRow.Edit.Click(); + await integerRow.FieldValue.Input.SelectAndInputText("14ab"); + await integerRow.Save.Click(); + await integerRow.FieldValue.WaitText("14"); + + await businessObjectEditingPage.Page.ReloadAsync(); integerRow = businessObjectEditingPage.RootAccordion.FindField("ShardNumber"); - integerRow.FieldValue.WaitText("14"); + await integerRow.FieldValue.WaitText("14"); GetDocument(documentName, documentId).ShardNumber.Should().Be(14); var decimalRow = businessObjectEditingPage.RootAccordion.FindField("DocumentPrice"); - decimalRow.FieldValue.WaitText("10.1"); - decimalRow.Edit.Click(); - decimalRow.FieldValue.Input.AppendText($"{Keys.Control}a"); - decimalRow.FieldValue.Input.AppendText("12,45"); - decimalRow.Save.Click(); - decimalRow.FieldValue.WaitText("12.45"); - - browser.WebDriver.Navigate().Refresh(); + await decimalRow.FieldValue.WaitText("10.1"); + await decimalRow.Edit.Click(); + await decimalRow.FieldValue.Input.SelectAndInputText("12,45"); + await decimalRow.Save.Click(); + await decimalRow.FieldValue.WaitText("12.45"); + + await businessObjectEditingPage.Page.ReloadAsync(); decimalRow = businessObjectEditingPage.RootAccordion.FindField("DocumentPrice"); - decimalRow.FieldValue.WaitText("12.45"); + await decimalRow.FieldValue.WaitText("12.45"); GetDocument(documentName, documentId).DocumentPrice.Should().Be(12.45m); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); var nullableDecimalRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_TotalAmount"); - nullableDecimalRow.FieldValue.WaitText("10.2"); - nullableDecimalRow.Edit.Click(); - nullableDecimalRow.FieldValue.Input.AppendText($"{Keys.Control}a{Keys.Delete}"); - nullableDecimalRow.Save.Click(); - nullableDecimalRow.FieldValue.WaitText("null"); - - browser.WebDriver.Navigate().Refresh(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await nullableDecimalRow.FieldValue.WaitText("10.2"); + await nullableDecimalRow.Edit.Click(); + await nullableDecimalRow.FieldValue.Input.Clear(); + await nullableDecimalRow.Save.Click(); + await nullableDecimalRow.FieldValue.WaitText("null"); + + await businessObjectEditingPage.Page.ReloadAsync(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); nullableDecimalRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_TotalAmount"); - nullableDecimalRow.FieldValue.WaitText("null"); + await nullableDecimalRow.FieldValue.WaitText("null"); GetDocument(documentName, documentId).DocumentContent.TotalAmount.Should().BeNull(); } @@ -269,48 +263,48 @@ public void TestChangeNumericValue() /// Ждем, что изменения сохранятся /// [Test] - public void TestChangeValueInArrayElement() + public async Task TestChangeValueInArrayElement() { var documentId = CreateDocument(documentName); - using var browser = new BrowserForTests(); - var businessObjectEditingPage = browser.LoginAsSuperUser().SwitchTo(documentName, $"Id={documentId}"); + await using var browser = new Browser(); + var businessObjectEditingPage = await (await browser.LoginAsSuperUser()).SwitchTo(documentName, $"Id={documentId}"); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems").ToggleButton.Click(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_0").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_0").ToggleButton.Click(); var inArrayFieldRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_GoodItems_0_Name"); - inArrayFieldRow.FieldValue.WaitText("Name of lintem"); - inArrayFieldRow.Edit.Click(); - inArrayFieldRow.FieldValue.Input.ClearAndInputText("Changed name"); - inArrayFieldRow.Save.Click(); - inArrayFieldRow.FieldValue.WaitText("Changed name"); - - browser.WebDriver.Navigate().Refresh(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems").ToggleButton.Click(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_0").ToggleButton.Click(); + await inArrayFieldRow.FieldValue.WaitText("Name of lintem"); + await inArrayFieldRow.Edit.Click(); + await inArrayFieldRow.FieldValue.Input.ClearAndInputText("Changed name"); + await inArrayFieldRow.Save.Click(); + await inArrayFieldRow.FieldValue.WaitText("Changed name"); + + await businessObjectEditingPage.Page.ReloadAsync(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_0").ToggleButton.Click(); inArrayFieldRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_GoodItems_0_Name"); - inArrayFieldRow.FieldValue.WaitText("Changed name"); + await inArrayFieldRow.FieldValue.WaitText("Changed name"); GetDocument(documentName, documentId).DocumentContent.GoodItems[0].Name.Should().Be("Changed name"); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_1").ToggleButton.Click(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_1_CustomDeclarationNumbers").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_1").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_1_CustomDeclarationNumbers").ToggleButton.Click(); var inArrayInArrayFieldRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_GoodItems_1_CustomDeclarationNumbers_0"); - inArrayInArrayFieldRow.FieldValue.WaitText("224"); - inArrayInArrayFieldRow.Edit.Click(); - inArrayInArrayFieldRow.FieldValue.Input.ClearAndInputText("225a"); - inArrayInArrayFieldRow.Save.Click(); - inArrayInArrayFieldRow.FieldValue.WaitText("225a"); - - browser.WebDriver.Navigate().Refresh(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems").ToggleButton.Click(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_1").ToggleButton.Click(); - businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_1_CustomDeclarationNumbers").ToggleButton.Click(); + await inArrayInArrayFieldRow.FieldValue.WaitText("224"); + await inArrayInArrayFieldRow.Edit.Click(); + await inArrayInArrayFieldRow.FieldValue.Input.ClearAndInputText("225a"); + await inArrayInArrayFieldRow.Save.Click(); + await inArrayInArrayFieldRow.FieldValue.WaitText("225a"); + + await businessObjectEditingPage.Page.ReloadAsync(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_1").ToggleButton.Click(); + await businessObjectEditingPage.RootAccordion.FindAccordionToggle("DocumentContent_GoodItems_1_CustomDeclarationNumbers").ToggleButton.Click(); inArrayInArrayFieldRow = businessObjectEditingPage.RootAccordion.FindField("DocumentContent_GoodItems_1_CustomDeclarationNumbers_0"); - inArrayInArrayFieldRow.FieldValue.WaitText("225a"); + await inArrayInArrayFieldRow.FieldValue.WaitText("225a"); GetDocument(documentName, documentId).DocumentContent.GoodItems[1].CustomDeclarationNumbers[0].Should().Be("225a"); } @@ -319,15 +313,15 @@ public void TestChangeValueInArrayElement() /// Проверяем, что поля ScopeId и Id нельзя изменить /// [Test] - public void TestCanNotEditScopeIdAndId() + public async Task TestCanNotEditScopeIdAndId() { var documentId = CreateDocument(documentName); - using var browser = new BrowserForTests(); - var businessObjectEditingPage = browser.LoginAsSuperUser().SwitchTo(documentName, $"Id={documentId}"); + await using var browser = new Browser(); + var businessObjectEditingPage = await (await browser.LoginAsSuperUser()).SwitchTo(documentName, $"Id={documentId}"); var row = businessObjectEditingPage.RootAccordion.FindField("id"); - row.Edit.WaitAbsence(); + await row.Edit.WaitAbsence(); } private static Guid CreateDocument(string documentName) diff --git a/DbViewer.Tests/FrontTests/BusinessObjectsDeleteTest.cs b/DbViewer.Tests/FrontTests/BusinessObjectsDeleteTest.cs index e1fc132a..8ead9166 100644 --- a/DbViewer.Tests/FrontTests/BusinessObjectsDeleteTest.cs +++ b/DbViewer.Tests/FrontTests/BusinessObjectsDeleteTest.cs @@ -1,12 +1,16 @@ using System; using System.Linq; using System.Threading; +using System.Threading.Tasks; using NUnit.Framework; using SkbKontur.DbViewer.TestApi.EntityFramework; -using SkbKontur.DbViewer.Tests.FrontTests.Helpers; using SkbKontur.DbViewer.Tests.FrontTests.Pages; +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +using BusinessObjectFilter = SkbKontur.DbViewer.Tests.FrontTests.Controls.BusinessObjectFilter; +using ConfirmDeleteObjectModal = SkbKontur.DbViewer.Tests.FrontTests.Controls.ConfirmDeleteObjectModal; namespace SkbKontur.DbViewer.Tests.FrontTests { @@ -16,20 +20,19 @@ public class BusinessObjectsDeleteTest /// Проверяем, что кнопка с удалением бизнес объекта на странице с таблицей доступна только с SuperUserAccessLevel.God /// [Test] - public void DeleteViaSearchPageRequiresGodAccess() + public async Task DeleteViaSearchPageRequiresGodAccess() { var ftpUser = CreateFtpUser(); - using var browser = new BrowserForTests(); - var businessObjectPage = browser.SwitchTo("FtpUser"); + await using var browser = new Browser(); + var businessObjectPage = await browser.SwitchTo("FtpUser"); - businessObjectPage.OpenFilter.Click(); - businessObjectPage.FilterModal.GetFilter("Login").Input.ClearAndInputText(ftpUser.Login); - businessObjectPage.FilterModal.Apply.Click(); + await businessObjectPage.OpenFilter.Click(); + await (await businessObjectPage.FilterModal.GetFilter("Login")).Input.ClearAndInputText(ftpUser.Login); + await businessObjectPage.FilterModal.Apply.Click(); - businessObjectPage = browser.RefreshUntil(businessObjectPage, x => x.BusinessObjectItems.IsPresent.Get()); - businessObjectPage.BusinessObjectItems.WaitCount(1); - businessObjectPage.BusinessObjectItems[0].Delete.IsPresent.Wait().That(Is.False, "Delete link should only be present for gods"); + await businessObjectPage.BusinessObjectItems.WaitCount(1); + await businessObjectPage.BusinessObjectItems[0].Delete.WaitAbsence(); } /// @@ -39,24 +42,23 @@ public void DeleteViaSearchPageRequiresGodAccess() /// [TestCase(true)] [TestCase(false)] - public void DeleteViaSearchPage(bool confirmDeletion) + public async Task DeleteViaSearchPage(bool confirmDeletion) { var ftpUser = CreateFtpUser(); - using var browser = new BrowserForTests(); - var businessObjectPage = browser.LoginAsSuperUser().SwitchTo("FtpUser"); + await using var browser = new Browser(); + var businessObjectPage = await (await browser.LoginAsSuperUser()).SwitchTo("FtpUser"); - businessObjectPage.OpenFilter.Click(); - businessObjectPage.FilterModal.GetFilter("Login").Input.ClearAndInputText(ftpUser.Login); - businessObjectPage.FilterModal.Apply.Click(); + await businessObjectPage.OpenFilter.Click(); + await (await businessObjectPage.FilterModal.GetFilter("Login")).Input.ClearAndInputText(ftpUser.Login); + await businessObjectPage.FilterModal.Apply.Click(); - businessObjectPage = browser.RefreshUntil(businessObjectPage, x => x.BusinessObjectItems.IsPresent.Get()); - businessObjectPage.BusinessObjectItems.WaitCount(1); - businessObjectPage.BusinessObjectItems[0].Delete.Click(); - ConfirmDeletion(businessObjectPage.ConfirmDeleteObjectModal, confirmDeletion); + await businessObjectPage.BusinessObjectItems.WaitCount(1); + await businessObjectPage.BusinessObjectItems[0].Delete.Click(); + await ConfirmDeletion(businessObjectPage.ConfirmDeleteObjectModal, confirmDeletion); if (confirmDeletion) - businessObjectPage.BusinessObjectItems.WaitAbsence(); + await businessObjectPage.BusinessObjectItems.WaitAbsence(); AssertFtpUserExistence(ftpUser.Id, confirmDeletion); } @@ -65,13 +67,13 @@ public void DeleteViaSearchPage(bool confirmDeletion) /// Проверяем, что кнопка с удалением бизнес объекта на странице с конкретным объектом доступна только с SuperUserAccessLevel.God /// [Test] - public void DeleteViaDetailsPageRequiresGodAccess() + public async Task DeleteViaDetailsPageRequiresGodAccess() { var ftpUser = CreateFtpUser(); - using var browser = new BrowserForTests(); - var detailsPage = browser.SwitchTo("FtpUser", $"Id={ftpUser.Id}"); - detailsPage.Delete.IsPresent.Wait().That(Is.False, "Delete link should only be present for gods"); + await using var browser = new Browser(); + var detailsPage = await browser.SwitchTo("FtpUser", $"Id={ftpUser.Id}"); + await detailsPage.Delete.WaitAbsence(); } /// @@ -81,27 +83,26 @@ public void DeleteViaDetailsPageRequiresGodAccess() /// [TestCase(true)] [TestCase(false)] - public void DeleteViaDetailsPage(bool confirmDeletion) + public async Task DeleteViaDetailsPage(bool confirmDeletion) { var ftpUser = CreateFtpUser(); - using var browser = new BrowserForTests(); - var detailsPage = browser.LoginAsSuperUser().SwitchTo("FtpUser", $"Id={ftpUser.Id}"); - detailsPage.Delete.Click(); - ConfirmDeletion(detailsPage.ConfirmDeleteObjectModal, confirmDeletion); + await using var browser = new Browser(); + var detailsPage = await (await browser.LoginAsSuperUser()).SwitchTo("FtpUser", $"Id={ftpUser.Id}"); + await detailsPage.Delete.Click(); + await ConfirmDeletion(detailsPage.ConfirmDeleteObjectModal, confirmDeletion); if (confirmDeletion) - detailsPage.GoTo(); + detailsPage.GoTo(); AssertFtpUserExistence(ftpUser.Id, confirmDeletion); } - private static void ConfirmDeletion(ConfirmDeleteObjectModal modal, bool confirmDeletion) + private static Task ConfirmDeletion(ConfirmDeleteObjectModal modal, bool confirmDeletion) { - if (confirmDeletion) - modal.Delete.Click(); - else - modal.Cancel.Click(); + return confirmDeletion + ? modal.Delete.Click() + : modal.Cancel.Click(); } private static void AssertFtpUserExistence(Guid userId, bool deletionConfirmed) diff --git a/DbViewer.Tests/FrontTests/BusinessObjectsSearchTests.cs b/DbViewer.Tests/FrontTests/BusinessObjectsSearchTests.cs index db4f32b9..479d8d13 100644 --- a/DbViewer.Tests/FrontTests/BusinessObjectsSearchTests.cs +++ b/DbViewer.Tests/FrontTests/BusinessObjectsSearchTests.cs @@ -1,7 +1,9 @@ -using NUnit.Framework; +using System.Threading.Tasks; + +using NUnit.Framework; -using SkbKontur.DbViewer.Tests.FrontTests.Helpers; using SkbKontur.DbViewer.Tests.FrontTests.Pages; +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; namespace SkbKontur.DbViewer.Tests.FrontTests { @@ -12,46 +14,49 @@ public class BusinessObjectsSearchTests /// Вводим в строке поиска запрос в R# стиле, проверяем ожидаемое количество и название каждого объекта /// [Test] - public void TestSearch() + public async Task TestSearch() { - using var browser = new BrowserForTests(); + await using var browser = new Browser(); + + var businessObjectsPage = await browser.SwitchTo(); + await businessObjectsPage.FilterInput.ClearAndInputText("CI"); + await businessObjectsPage.ObjectGroups.WaitCount(1); + + var pagedObjects = await businessObjectsPage.ObjectGroups.GetItemWithText(x => x.Name, "CQL Paged Objects"); + await pagedObjects.ObjectsList + .Select(x => x.ObjectLink) + .WaitText("CqlOrganizationInfo", "CqlUserInfo"); + + await businessObjectsPage.FilterInput.ClearAndInputText("DocMe"); + await businessObjectsPage.ObjectGroups.WaitCount(2); - var businessObjectsPage = browser.SwitchTo(); - businessObjectsPage.FilterInput.ClearAndInputText("CI"); - businessObjectsPage.ObjectGroups.WaitCount(1); - businessObjectsPage.ObjectGroups - .GetItemWithText(x => x.Name.Text, "CQL Paged Objects").ObjectsList - .Wait(x => x.ObjectLink.Text) - .That(Is.EqualTo(new[] {"CqlOrganizationInfo", "CqlUserInfo"})); + var cqlObjects = await businessObjectsPage.ObjectGroups.GetItemWithText(x => x.Name, "CQL Objects"); + await cqlObjects.ObjectsList + .Select(x => x.ObjectLink) + .WaitText(new[] {"DocumentBindingsMeta"}); - businessObjectsPage.FilterInput.ClearAndInputText("DocMe"); - businessObjectsPage.ObjectGroups.WaitCount(2); - businessObjectsPage.ObjectGroups - .GetItemWithText(x => x.Name.Text, "CQL Objects").ObjectsList - .Wait(x => x.ObjectLink.Text) - .That(Is.EqualTo(new[] {"DocumentBindingsMeta"})); - businessObjectsPage.ObjectGroups - .GetItemWithText(x => x.Name.Text, "CQL Paged Objects").ObjectsList - .Wait(x => x.ObjectLink.Text) - .That(Is.EqualTo(new[] {"CqlDocumentMeta"})); + var cqlPagedObjects = await businessObjectsPage.ObjectGroups.GetItemWithText(x => x.Name, "CQL Paged Objects"); + await cqlPagedObjects.ObjectsList + .Select(x => x.ObjectLink) + .WaitText(new[] {"CqlDocumentMeta"}); } /// /// Проверяем, что у таблицы FtpUser есть плашка indexed /// [Test] - public void TestSearchIndexedField() + public async Task TestSearchIndexedField() { - using var browser = new BrowserForTests(); + await using var browser = new Browser(); - var businessObjects = browser.SwitchTo(); - businessObjects.FilterInput.ClearAndInputText("UsersTable"); - businessObjects.ObjectGroups.WaitCount(1); + var businessObjects = await browser.SwitchTo(); + await businessObjects.FilterInput.ClearAndInputText("UsersTable"); + await businessObjects.ObjectGroups.WaitCount(1); - var objects = businessObjects.ObjectGroups.GetItemWithText(x => x.Name.Text, "Postgres Objects"); - objects.IndexedLabel.WaitPresence(); - objects.ObjectsList.WaitCount(1); - objects.ObjectsList[0].ObjectLink.WaitText("UsersTable"); + var objects = await businessObjects.ObjectGroups.GetItemWithText(x => x.Name, "Postgres Objects"); + await objects.IndexedLabel.WaitPresence(); + await objects.ObjectsList.WaitCount(1); + await objects.ObjectsList[0].ObjectLink.WaitText("UsersTable"); } /// @@ -59,15 +64,15 @@ public void TestSearchIndexedField() /// Проверяем что нам выдает ровно одну ссылку /// [Test] - public void TestSearchLink() + public async Task TestSearchLink() { - using var browser = new BrowserForTests(); + await using var browser = new Browser(); - var businessObjectsPage = browser.SwitchTo(); - businessObjectsPage.FilterInput.ClearAndInputText("DocumentPrintingInfo"); - businessObjectsPage.ObjectGroups.WaitCount(1); - businessObjectsPage.ObjectGroups[0].ObjectsList.WaitCount(1); - businessObjectsPage.FindBusinessObjectLink("CQL Objects", "DocumentPrintingInfo").WaitPresence(); + var businessObjectsPage = await browser.SwitchTo(); + await businessObjectsPage.FilterInput.ClearAndInputText("DocumentPrintingInfo"); + await businessObjectsPage.ObjectGroups.WaitCount(1); + await businessObjectsPage.ObjectGroups[0].ObjectsList.WaitCount(1); + await businessObjectsPage.FindBusinessObjectLink("CQL Objects", "DocumentPrintingInfo"); } /// @@ -75,14 +80,14 @@ public void TestSearchLink() /// Проверяем, что ссылка ведет туда, куда нам нужно /// [Test] - public void TestLinkShouldReferToShowTablePage() + public async Task TestLinkShouldReferToShowTablePage() { - using var browser = new BrowserForTests(); + await using var browser = new Browser(); - var businessObjectsPage = browser.SwitchTo(); - var link = businessObjectsPage.FindBusinessObjectLink("CQL Objects", "DocumentPrintingInfo"); - var page = link.ClickAndGoTo(); - page.Header.WaitText("DocumentPrintingInfo"); + var businessObjectsPage = await browser.SwitchTo(); + var link = await businessObjectsPage.FindBusinessObjectLink("CQL Objects", "DocumentPrintingInfo"); + var page = await link.ClickAndGoTo(); + await page.Header.WaitText("DocumentPrintingInfo"); } } } \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Controls/Accordion.cs b/DbViewer.Tests/FrontTests/Controls/Accordion.cs new file mode 100644 index 00000000..6a272427 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Controls/Accordion.cs @@ -0,0 +1,31 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.AutoFill; +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Controls +{ + public class Accordion : PwControlBase + { + public Accordion(ILocator locator) + : base(locator) + { + } + + public AccordionRow FindField(string id) + { + var row = new AccordionRow(Locator.GetByTestId(id)); + PwAutoFill.InitializeControls(row, Locator.Page, row.Locator); + + return row; + } + + public AccordionToggle FindAccordionToggle(string id) + { + var toggle = new AccordionToggle(Locator.GetByTestId(id)); + PwAutoFill.InitializeControls(toggle, Locator.Page, toggle.Locator); + + return toggle; + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Controls/AccordionFieldValue.cs b/DbViewer.Tests/FrontTests/Controls/AccordionFieldValue.cs new file mode 100644 index 00000000..4fcfa653 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Controls/AccordionFieldValue.cs @@ -0,0 +1,33 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.AutoFill; +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Controls +{ + public class AccordionFieldValue : PwControlBase + { + public AccordionFieldValue(ILocator locator) + : base(locator) + { + } + + public PwLink GoToLink { get; set; } + + [Selector("Input")] + public PwInput Input { get; set; } + + [Selector("Checkbox")] + public PwCheckbox Checkbox { get; set; } + + public PwSelect EnumSelect { get; set; } + public PwSelect BooleanSelect { get; set; } + + public PwLink DownloadLink { get; set; } + + public PwDatePicker Date { get; set; } + public PwInput Time { get; set; } + public PwSelect TimeZoneSelect { get; set; } + public PwLabel TimeOffsetLabel { get; set; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Controls/AccordionRow.cs b/DbViewer.Tests/FrontTests/Controls/AccordionRow.cs new file mode 100644 index 00000000..0e38d33d --- /dev/null +++ b/DbViewer.Tests/FrontTests/Controls/AccordionRow.cs @@ -0,0 +1,25 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Controls +{ + public class AccordionRow : PwControlBase + { + public AccordionRow(ILocator locator) + : base(locator) + { + } + + public PwLabel Key { get; set; } + + public PwLink GoToLink { get; set; } + + public AccordionFieldValue FieldValue { get; set; } + public PwLabel Value { get; set; } + + public PwButton Edit { get; set; } + public PwButton Save { get; set; } + public PwButton Cancel { get; set; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Controls/AccordionToggle.cs b/DbViewer.Tests/FrontTests/Controls/AccordionToggle.cs new file mode 100644 index 00000000..02f94d01 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Controls/AccordionToggle.cs @@ -0,0 +1,16 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Controls +{ + public class AccordionToggle : PwControlBase + { + public AccordionToggle(ILocator locator) + : base(locator) + { + } + + public PwButton ToggleButton { get; set; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Controls/BusinessObjectFilter.cs b/DbViewer.Tests/FrontTests/Controls/BusinessObjectFilter.cs new file mode 100644 index 00000000..8ade8c77 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Controls/BusinessObjectFilter.cs @@ -0,0 +1,28 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.Pages; +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +using SKBKontur.SeleniumTesting.Controls; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Controls +{ + public class BusinessObjectFilter : PwControlBase + { + public BusinessObjectFilter(ILocator locator) + : base(locator) + { + } + + public PwLabel FormCaption { get; set; } + public Select OperatorSelect { get; set; } + public Select EnumSelect { get; set; } + public Select BooleanSelect { get; set; } + public PwInput Input { get; set; } + public Validation InputValidation { get; set; } + public Validation DateTimeValidation { get; set; } + public DatePicker Date { get; set; } + public PwInput Time { get; set; } + public PwInput DateTimeInTicks { get; set; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Controls/BusinessObjectGroup.cs b/DbViewer.Tests/FrontTests/Controls/BusinessObjectGroup.cs new file mode 100644 index 00000000..fffd567e --- /dev/null +++ b/DbViewer.Tests/FrontTests/Controls/BusinessObjectGroup.cs @@ -0,0 +1,21 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.AutoFill; +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Controls +{ + public class BusinessObjectGroup : PwControlBase + { + public BusinessObjectGroup(ILocator locator) + : base(locator) + { + } + + public PwLabel Name { get; set; } + public PwLabel IndexedLabel { get; set; } + + [Selector("##ObjectsList ##ObjectItem")] + public PwControlList ObjectsList { get; set; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Controls/BusinessObjectItem.cs b/DbViewer.Tests/FrontTests/Controls/BusinessObjectItem.cs new file mode 100644 index 00000000..8f7478dc --- /dev/null +++ b/DbViewer.Tests/FrontTests/Controls/BusinessObjectItem.cs @@ -0,0 +1,16 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Controls +{ + public class BusinessObjectItem : PwControlBase + { + public BusinessObjectItem(ILocator locator) + : base(locator) + { + } + + public PwLink ObjectLink { get; set; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Controls/BusinessObjectTableRow.cs b/DbViewer.Tests/FrontTests/Controls/BusinessObjectTableRow.cs new file mode 100644 index 00000000..63980be1 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Controls/BusinessObjectTableRow.cs @@ -0,0 +1,24 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Controls +{ + public class BusinessObjectTableRow : PwControlBase + { + public BusinessObjectTableRow(ILocator locator) + : base(locator) + { + } + + public PwLink Details { get; set; } + public PwLink Delete { get; set; } + public PwLabel Id { get; set; } + public PwLabel ScopeId { get; set; } + + public PwLabel FindColumn(string tid) + { + return new PwLabel(Locator.GetByTestId(tid)); + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Controls/ConfirmDeleteObjectModal.cs b/DbViewer.Tests/FrontTests/Controls/ConfirmDeleteObjectModal.cs new file mode 100644 index 00000000..a22c4af1 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Controls/ConfirmDeleteObjectModal.cs @@ -0,0 +1,17 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Controls +{ + public class ConfirmDeleteObjectModal : PwControlBase + { + public ConfirmDeleteObjectModal(ILocator locator) + : base(locator) + { + } + + public PwButton Delete { get; set; } + public PwButton Cancel { get; set; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Controls/FilterModal.cs b/DbViewer.Tests/FrontTests/Controls/FilterModal.cs new file mode 100644 index 00000000..39580649 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Controls/FilterModal.cs @@ -0,0 +1,30 @@ +using System.Threading.Tasks; + +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.AutoFill; +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Controls +{ + public class FilterModal : PwControlBase + { + public FilterModal(ILocator locator) + : base(locator) + { + } + + public Task GetFilter(string name) + { + return ObjectFilters.Where(x => x.Locator.Page.GetByTestId(name)).Single(); + } + + [Selector("##ObjectFilters ##Filter")] + public PwControlList ObjectFilters { get; set; } + + public PwButton Apply { get; set; } + public PwLink Clear { get; set; } + public PwInput Id { get; set; } + public PwInput ScopeId { get; set; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Pages/PwBusinessObjectDetailsPage.cs b/DbViewer.Tests/FrontTests/Pages/PwBusinessObjectDetailsPage.cs new file mode 100644 index 00000000..6981580f --- /dev/null +++ b/DbViewer.Tests/FrontTests/Pages/PwBusinessObjectDetailsPage.cs @@ -0,0 +1,25 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.Helpers; +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Pages +{ + [PageRoute("BusinessObjects/{scopeId}/details?{id}")] + public class PwBusinessObjectDetailsPage : PwPageBase + { + public PwBusinessObjectDetailsPage(IPage page) + : base(page) + { + } + + public PwLabel Header { get; set; } + + public PwLink Copy { get; set; } + public PwLink Delete { get; set; } + + public Controls.ConfirmDeleteObjectModal ConfirmDeleteObjectModal { get; set; } + + public Controls.Accordion RootAccordion { get; set; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Pages/PwBusinessObjectTablePage.cs b/DbViewer.Tests/FrontTests/Pages/PwBusinessObjectTablePage.cs new file mode 100644 index 00000000..4c7d337c --- /dev/null +++ b/DbViewer.Tests/FrontTests/Pages/PwBusinessObjectTablePage.cs @@ -0,0 +1,50 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.AutoFill; +using SkbKontur.DbViewer.Tests.FrontTests.Helpers; +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +using SKBKontur.SeleniumTesting.Controls; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Pages +{ + [PageRoute("BusinessObjects/{id}")] + public class PwBusinessObjectTablePage : PwPageBase + { + public PwBusinessObjectTablePage(IPage page) + : base(page) + { + } + + public PwLabel Header { get; set; } + + public PwLink GoBack { get; set; } + public PwLabel NothingFound { get; set; } + + public PwLabel ItemsCountInfo { get; set; } + + public PwLink OpenFilter { get; set; } + + [Selector("portal=FilterModal")] + public Controls.FilterModal FilterModal { get; set; } + + public DownloadLimitModal DownloadLimitModal { get; set; } + + public CountDropdown CountDropdown { get; set; } + public PwLink ClearFilter { get; set; } + public PwLink FieldSettings { get; set; } + public PwLink DownloadLink { get; set; } + + [Selector("Portal:portal ##ColumnSelector")] + public ColumnSelector ColumnSelector { get; set; } + + public Controls.ConfirmDeleteObjectModal ConfirmDeleteObjectModal { get; set; } + + public BusinessObjectTableHeader TableHeader { get; set; } + + [Selector("##Body ##Row")] + public PwControlList BusinessObjectItems { get; set; } + + public Paging Paging { get; set; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Pages/PwBusinessObjectsPage.cs b/DbViewer.Tests/FrontTests/Pages/PwBusinessObjectsPage.cs new file mode 100644 index 00000000..adca1f70 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Pages/PwBusinessObjectsPage.cs @@ -0,0 +1,32 @@ +using System.Threading.Tasks; + +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.AutoFill; +using SkbKontur.DbViewer.Tests.FrontTests.Helpers; +using SkbKontur.DbViewer.Tests.FrontTests.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Pages +{ + [PageRoute("BusinessObjects")] + public class PwBusinessObjectsPage : PwPageBase + { + public PwBusinessObjectsPage(IPage page) + : base(page) + { + } + + public PwLabel Header { get; set; } + public PwLink BackLink { get; set; } + public PwInput FilterInput { get; set; } + + [Selector("##ObjectGroups ##ObjectGroup")] + public PwControlList ObjectGroups { get; set; } + + public async Task FindBusinessObjectLink(string groupName, string objectName) + { + var businessObjectsList = (await ObjectGroups.GetItemWithText(x => x.Name, groupName)).ObjectsList; + return (await businessObjectsList.GetItemWithText(x => x.ObjectLink, objectName)).ObjectLink; + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/AssertionExtensions.cs b/DbViewer.Tests/FrontTests/Playwright/AssertionExtensions.cs new file mode 100644 index 00000000..a9e7056f --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/AssertionExtensions.cs @@ -0,0 +1,146 @@ +using System; +using System.Diagnostics; +using System.Threading; +using System.Threading.Tasks; + +using NUnit.Framework; + +using SkbKontur.DbViewer.Tests.FrontTests.AutoFill; +using SkbKontur.DbViewer.Tests.FrontTests.Pages; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + public static class AssertionExtensions + { + public static Task GetItemWithText(this PwControlList list, Func f, string text) + where T : PwControlBase + { + return list.Where(x => f(x).Locator.GetByText(text)).Single(); + } + + public static async Task ClickAndGoTo(this PwControlBase control) + where T : PwPageBase + { + await control.Locator.ClickAsync(); + return PwAutoFill.InitializePage(control.Locator.Page); + } + + public static T GoTo(this PwPageBase page) + where T : PwPageBase + { + return PwAutoFill.InitializePage(page.Page); + } + + public static async Task Clear(this PwInput input) + { + await input.Click(); + await input.Locator.Page.Keyboard.PressAsync("Control+A+Delete"); + } + + public static Task ClearAndInputText(this PwInput input, string value) + { + return input.Locator.FillAsync(value); + } + + public static async Task SelectAndInputText(this PwInput input, string value) + { + await input.Click(); + await input.Locator.Page.Keyboard.PressAsync("Control+A"); + await input.Locator.PressSequentiallyAsync(value); + } + + public static async Task ClearAndInputText(this PwDatePicker datePicker, string value) + { + await datePicker.Locator.ClickAsync(); + var input = datePicker.Locator.GetByTestId("InputLikeText__input"); + await input.Page.Keyboard.PressAsync("Home"); + await input.PressSequentiallyAsync(value); + } + + public static Task WaitCount(this PwControlList list, int count) + where T : PwControlBase + { + return list.Expect().ToHaveCountAsync(count); + } + + public static async Task Single(this PwControlList list) + where T : PwControlBase + { + await list.WaitCount(1); + return list[0]; + } + + public static Task WaitPresence(this PwControlBase control) + { + return control.Expect().ToBeVisibleAsync(); + } + + public static Task WaitAbsence(this PwControlBase control) + { + return control.Expect().Not.ToBeVisibleAsync(); + } + + public static Task WaitText(this PwControlBase control, string value) + { + return control.Expect().ToHaveTextAsync(value); + } + + public static Task WaitText(this PwControlBase control, params string[] values) + { + return control.Expect().ToHaveTextAsync(values); + } + + public static Task WaitTextContains(this PwControlBase control, string value) + { + return control.Expect().ToContainTextAsync(value); + } + + public static async Task WaitItems(this PwSelect select, string[] items) + { + await select.Click(); + await select.Items.Expect().ToHaveTextAsync(items); + await select.Click(); + } + + public static async Task SelectValueByText(this PwSelect select, string value) + { + await select.Click(); + await select.Items.Locator.GetByText(value).ClickAsync(); + } + + public static Task Click(this PwControlBase control) + { + return control.Locator.ClickAsync(); + } + + public static async Task LoginAsSuperUser(this Browser browser) + { + await browser.SwitchToUri(new Uri("Admin", UriKind.Relative)); + return browser; + } + + public static async Task Refresh(this Browser browser, TPage page) + where TPage : PwPageBase + { + await browser.Page.ReloadAsync(); + return browser.GoTo(); + } + + public static async Task RefreshUntil(this Browser browser, TPage page, Func> conditionFunc, string cause = null, int timeout = 30000, int waitTimeout = 100) + where TPage : PwPageBase + { + var w = Stopwatch.StartNew(); + if (await conditionFunc(page)) + return page; + do + { + page = await browser.Refresh(page); + if (await conditionFunc(page)) + return page; + Thread.Sleep(waitTimeout); + } while (w.ElapsedMilliseconds < timeout); + Assert.Fail(cause ?? $"Не смогли дождаться страницу за {timeout} мс"); + return default; + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/Browser.cs b/DbViewer.Tests/FrontTests/Playwright/Browser.cs new file mode 100644 index 00000000..ceaca567 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/Browser.cs @@ -0,0 +1,117 @@ +using System; +using System.Reflection; +using System.Threading.Tasks; + +using Microsoft.Playwright; + +using NUnit.Framework; + +using SkbKontur.DbViewer.Tests.FrontTests.AutoFill; +using SkbKontur.DbViewer.Tests.FrontTests.Helpers; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + public class Browser : IAsyncDisposable + { + public async Task SwitchToUri(Uri uri) + where TPage : PwPageBase + { + await Page.GotoAsync(MakeAbsolute(uri).ToString()); + return PwAutoFill.InitializePage(Page); + } + + public TPage GoTo() + where TPage : PwPageBase + { + var newPage = PwAutoFill.InitializePage(Page); + return newPage; + } + + public async ValueTask DisposeAsync() + { + if (context != null) + { + await context.Tracing.StopAsync(new TracingStopOptions + { + Path = $"{TestContext.CurrentContext.TestDirectory}/out/{TestContext.CurrentContext.Test.Name}.zip" + }); + } + + if (page != null) + await page.CloseAsync(); + + if (context != null) + { + await context.CloseAsync(); + } + } + + private Uri MakeAbsolute(Uri uri) + { + return uri.IsAbsoluteUri ? uri : new Uri(new Uri(baseUrl), uri); + } + + public IPage Page + { + get + { + if (page != null) + return page; + + context = PlaywrightSetup.Browser.NewContextAsync(new BrowserNewContextOptions + { + Locale = "en-US", + TimezoneId = "Europe/London" + }).GetAwaiter().GetResult(); + context.Tracing.StartAsync(new TracingStartOptions + { + Screenshots = true, + Snapshots = true, + Sources = true, + Name = TestContext.CurrentContext.Test.Name + }).GetAwaiter().GetResult(); + + return page = context.NewPageAsync().GetAwaiter().GetResult(); + } + } + + private const string baseUrl = "http://localhost:5000/"; + + private IPage? page; + private IBrowserContext? context; + } + + public static class AttributeNavigationExtensions + { + public static ILocatorAssertions Expect(this PwControlBase controlBase) + { + return Assertions.Expect(controlBase.Locator); + } + + public static Task SwitchTo(this Browser browser, params string[] templateParameters) + where TPage : PwPageBase + { + return browser.SwitchToUri(new Uri(GetPageRoute(templateParameters), UriKind.Relative)); + } + + private static string GetPageRoute(string[] templateParameters) + { + foreach (var attribute in typeof(TPage).GetCustomAttributes()) + { + var route = attribute.Route; + if (templateParameters.Length == 0 && !route.Contains("{id}") && !route.Contains("{scopeId}")) + return route; + if (templateParameters.Length == 1 && route.Contains("{id}") && !route.Contains("{scopeId}")) + return route.Replace("{id}", templateParameters[0]); + if (templateParameters.Length == 2 && route.Contains("{id}") && route.Contains("{scopeId}")) + return route.Replace("{scopeId}", templateParameters[0]).Replace("{id}", templateParameters[1]); + if (templateParameters.Length == 3 && route.Contains("{scopeId}") && route.Contains("{id}") && route.Contains("{versionId}")) + return route + .Replace("{scopeId}", templateParameters[0]) + .Replace("{id}", templateParameters[1]) + .Replace("{versionId}", templateParameters[2]); + } + throw new InvalidOperationException($"Corresponding PageRoute attribute with {templateParameters.Length} arguments not found for {typeof(TPage).Name}"); + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/PwButton.cs b/DbViewer.Tests/FrontTests/Playwright/PwButton.cs new file mode 100644 index 00000000..2796df9a --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/PwButton.cs @@ -0,0 +1,12 @@ +using Microsoft.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + public class PwButton : PwControlBase + { + public PwButton(ILocator locator) + : base(locator) + { + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/PwCheckbox.cs b/DbViewer.Tests/FrontTests/Playwright/PwCheckbox.cs new file mode 100644 index 00000000..a5906e36 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/PwCheckbox.cs @@ -0,0 +1,12 @@ +using Microsoft.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + public class PwCheckbox : PwControlBase + { + public PwCheckbox(ILocator locator) + : base(locator) + { + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/PwControlBase.cs b/DbViewer.Tests/FrontTests/Playwright/PwControlBase.cs new file mode 100644 index 00000000..9ab31a04 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/PwControlBase.cs @@ -0,0 +1,17 @@ +using JetBrains.Annotations; + +using Microsoft.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + [MeansImplicitUse(ImplicitUseTargetFlags.Members | ImplicitUseTargetFlags.WithInheritors)] + public class PwControlBase + { + public PwControlBase(ILocator locator) + { + Locator = locator; + } + + public ILocator Locator { get; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/PwControlList.cs b/DbViewer.Tests/FrontTests/Playwright/PwControlList.cs new file mode 100644 index 00000000..59fdea15 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/PwControlList.cs @@ -0,0 +1,48 @@ +using System; + +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.AutoFill; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + public class PwControlList : PwControlBase + where TControl : PwControlBase + { + public PwControlList(ILocator locator) + : base(locator) + { + } + + public PwControlList Select(Func selector) + where T : PwControlBase + { + var dummyControl = (TControl)Activator.CreateInstance(typeof(TControl), Locator)!; + PwAutoFill.InitializeControls(dummyControl, Locator.Page, Locator); + + return new PwControlList(selector(dummyControl).Locator); + } + + public PwControlList Where(Func predicate) + { + var dummyControl = (TControl)Activator.CreateInstance(typeof(TControl), Locator)!; + PwAutoFill.InitializeControls(dummyControl, Locator.Page, null); + + return new PwControlList(Locator.Filter(new LocatorFilterOptions + { + Has = predicate(dummyControl) + })); + } + + public TControl this[int index] + { + get + { + var controlLocator = Locator.Nth(index); + var control = (TControl)Activator.CreateInstance(typeof(TControl), controlLocator)!; + PwAutoFill.InitializeControls(control, Locator.Page, controlLocator); + return control; + } + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/PwDatePicker.cs b/DbViewer.Tests/FrontTests/Playwright/PwDatePicker.cs new file mode 100644 index 00000000..af82a9fa --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/PwDatePicker.cs @@ -0,0 +1,12 @@ +using Microsoft.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + public class PwDatePicker : PwControlBase + { + public PwDatePicker(ILocator locator) + : base(locator) + { + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/PwInput.cs b/DbViewer.Tests/FrontTests/Playwright/PwInput.cs new file mode 100644 index 00000000..c539da44 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/PwInput.cs @@ -0,0 +1,12 @@ +using Microsoft.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + public class PwInput : PwControlBase + { + public PwInput(ILocator locator) + : base(locator) + { + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/PwLabel.cs b/DbViewer.Tests/FrontTests/Playwright/PwLabel.cs new file mode 100644 index 00000000..eb0079a5 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/PwLabel.cs @@ -0,0 +1,12 @@ +using Microsoft.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + public class PwLabel : PwControlBase + { + public PwLabel(ILocator locator) + : base(locator) + { + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/PwLink.cs b/DbViewer.Tests/FrontTests/Playwright/PwLink.cs new file mode 100644 index 00000000..8881b3a0 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/PwLink.cs @@ -0,0 +1,12 @@ +using Microsoft.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + public class PwLink : PwControlBase + { + public PwLink(ILocator locator) + : base(locator) + { + } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/PwPageBase.cs b/DbViewer.Tests/FrontTests/Playwright/PwPageBase.cs new file mode 100644 index 00000000..373b8621 --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/PwPageBase.cs @@ -0,0 +1,17 @@ +using JetBrains.Annotations; + +using Microsoft.Playwright; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + [MeansImplicitUse(ImplicitUseTargetFlags.Members | ImplicitUseTargetFlags.WithInheritors)] + public class PwPageBase + { + public PwPageBase(IPage page) + { + Page = page; + } + + public IPage Page { get; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/Playwright/PwSelect.cs b/DbViewer.Tests/FrontTests/Playwright/PwSelect.cs new file mode 100644 index 00000000..7e07cc0a --- /dev/null +++ b/DbViewer.Tests/FrontTests/Playwright/PwSelect.cs @@ -0,0 +1,17 @@ +using Microsoft.Playwright; + +using SkbKontur.DbViewer.Tests.FrontTests.AutoFill; + +namespace SkbKontur.DbViewer.Tests.FrontTests.Playwright +{ + public class PwSelect : PwControlBase + { + public PwSelect(ILocator locator) + : base(locator) + { + } + + [Selector("portal=Select__menu ##MenuItem__root")] + public PwControlList Items { get; set; } + } +} \ No newline at end of file diff --git a/DbViewer.Tests/FrontTests/PlaywrightSetup.cs b/DbViewer.Tests/FrontTests/PlaywrightSetup.cs new file mode 100644 index 00000000..ff5f6185 --- /dev/null +++ b/DbViewer.Tests/FrontTests/PlaywrightSetup.cs @@ -0,0 +1,66 @@ +using System.Threading.Tasks; + +using Microsoft.Playwright; + +using NUnit.Framework; + +namespace SkbKontur.DbViewer.Tests.FrontTests +{ + [SetUpFixture] + public class PlaywrightSetup + { + public static IPlaywright Playwright { get; private set; } + public static IBrowser Browser { get; private set; } + + [OneTimeSetUp] + public async Task SetUp() + { + Playwright = await Microsoft.Playwright.Playwright.CreateAsync(); + Playwright.Selectors.SetTestIdAttribute("data-tid"); +// await Playwright.Selectors.RegisterAsync("portal", new SelectorsRegisterOptions +// { +// Script = @" +// // Must evaluate to a selector engine instance. +// { +// // Returns the first element matching given selector in the root's subtree. +// query(root, selector) { +// console.info(root); +// console.info(selector); +// return root.querySelector(selector); +// }, +// +// // Returns all elements matching given selector in the root's subtree. +// queryAll(root, selector) { +// return Array.from(root.querySelectorAll(selector)); +// } +// }" +// }); + + await Playwright.Selectors.RegisterAsync("portal", new SelectorsRegisterOptions + { + Script = @"{ + query(root, selector) { + const portal = root.querySelector(`[data-tid='${selector}']`); + const id = portal.getAttribute('data-render-container-id'); + return document.querySelector(`[data-rendered-container-id='${id}']`); + }, + + queryAll(root, selector) { + const portal = root.querySelector(`[data-tid='${selector}']`); + const id = portal.getAttribute('data-render-container-id'); + return [document.querySelector(`[data-rendered-container-id='${id}']`)]; + } + }" + }); + + Browser = await Playwright.Chromium.LaunchAsync(); + } + + [OneTimeTearDown] + public async Task TearDown() + { + await Browser.DisposeAsync(); + Playwright.Dispose(); + } + } +} \ No newline at end of file diff --git a/global.json b/global.json index 447ef432..501e79a8 100644 --- a/global.json +++ b/global.json @@ -1,6 +1,6 @@ { "sdk": { - "version": "6.0.400", + "version": "8.0.100", "rollForward": "latestFeature" } } \ No newline at end of file