From fb1f43554d88936707f133ec845d94e09006c863 Mon Sep 17 00:00:00 2001 From: Gerald Versluis Date: Thu, 23 May 2024 16:42:34 +0200 Subject: [PATCH 1/6] Run tests by category groups --- eng/devices/android.cake | 11 ++- eng/devices/catalyst.cake | 23 +++-- eng/devices/devices-shared.cake | 2 +- eng/devices/ios.cake | 11 ++- eng/devices/windows.cake | 19 +++- .../common/ui-tests-build-sample.yml | 69 ++++++++++++++ .../common/ui-tests-compatibility-steps.yml | 16 +++- eng/pipelines/common/ui-tests-steps.yml | 91 +++++++++---------- eng/pipelines/common/ui-tests.yml | 89 +++++++++++++++++- eng/pipelines/ui-tests.yml | 2 +- .../UITestCategories.cs | 2 + 11 files changed, 257 insertions(+), 78 deletions(-) create mode 100644 eng/pipelines/common/ui-tests-build-sample.yml diff --git a/eng/devices/android.cake b/eng/devices/android.cake index d91c5ff6198c..dd9eb4d31c6a 100644 --- a/eng/devices/android.cake +++ b/eng/devices/android.cake @@ -18,9 +18,6 @@ var testAppInstrumentation = Argument("instrumentation", EnvironmentVariable("AN var testResultsPath = Argument("results", EnvironmentVariable("ANDROID_TEST_RESULTS") ?? GetTestResultsDirectory()?.FullPath); var deviceCleanupEnabled = Argument("cleanup", true); -// Test where clause -string testWhere = Argument("where", EnvironmentVariable("NUNIT_TEST_WHERE") ?? ""); - // Device details var deviceSkin = Argument("skin", EnvironmentVariable("ANDROID_TEST_SKIN") ?? "Nexus 5X"); var androidAvd = "DEVICE_TESTS_EMULATOR"; @@ -96,7 +93,6 @@ Task("uitest-build") }); Task("uitest") - .IsDependentOn("uitest-build") .Does(() => { ExecuteUITests(projectPath, testAppProjectPath, testAppPackageName, testDevice, testResultsPath, binlogDirectory, configuration, targetFramework, "", androidVersion, dotnetToolPath, testAppInstrumentation); @@ -305,7 +301,12 @@ void ExecuteUITests(string project, string app, string appPackageName, string de var name = System.IO.Path.GetFileNameWithoutExtension(project); var binlog = $"{binDir}/{name}-{config}-{platform}.binlog"; var appiumLog = $"{binDir}/appium_{platform}.log"; - var resultsFileName = $"{name}-{config}-{platform}"; + var resultsFileName = $"{name}-{config}-{platform}-{testFilter.Replace("|", "_").Replace("TestCategory=", "")}"; + + if (resultsFileName.Contains("!~'.+'")) + { + resultsFileName = $"{name}-{config}-{platform}-notcategorized"; + } DotNetBuild(project, new DotNetBuildSettings { diff --git a/eng/devices/catalyst.cake b/eng/devices/catalyst.cake index 6b5edb096671..9818d77d52e2 100644 --- a/eng/devices/catalyst.cake +++ b/eng/devices/catalyst.cake @@ -58,7 +58,6 @@ Task("uitest-build") }); Task("uitest") - .IsDependentOn("uitest-build") .Does(() => { ExecuteUITests(projectPath, testAppProjectPath, testDevice, testResultsPath, binlogDirectory, configuration, targetFramework, runtimeIdentifier, dotnetToolPath); @@ -131,21 +130,25 @@ void ExecuteUITests(string project, string app, string device, string resultsDir } // Launch the app so it can be found by the test runner - DotNetBuild(app, new DotNetBuildSettings - { - Configuration = config, - Framework = tfm, - ToolPath = toolPath, - ArgumentCustomization = args => args - .Append("/t:Run") - }); + StartProcess("chmod", $"+x {testApp}/Contents/MacOS/Controls.TestCases.HostApp"); + + var p = new System.Diagnostics.Process(); + p.StartInfo.UseShellExecute = true; + p.StartInfo.FileName = "open"; + p.StartInfo.Arguments = testApp; + p.Start(); Information("Build UITests project {0}", project); var name = System.IO.Path.GetFileNameWithoutExtension(project); var binlog = $"{binDir}/{name}-{config}-mac.binlog"; var appiumLog = $"{binDir}/appium_mac.log"; - var resultsFileName = $"{name}-{config}-catalyst"; + var resultsFileName = $"{name}-{config}-catalyst-{testFilter.Replace("|", "_").Replace("TestCategory=", "")}"; + + if (resultsFileName.Contains("!~'.+'")) + { + resultsFileName = $"{name}-{config}-catalyst-notcategorized"; + } DotNetBuild(project, new DotNetBuildSettings { diff --git a/eng/devices/devices-shared.cake b/eng/devices/devices-shared.cake index dc2d61ab9de4..00e99facfaa7 100644 --- a/eng/devices/devices-shared.cake +++ b/eng/devices/devices-shared.cake @@ -45,7 +45,7 @@ IEnumerable GetTestApplications(string project, string device, string co const string artifactsDir = "../../artifacts/bin/"; bool isAndroid = tfm.Contains("android"); - var binDir = new DirectoryPath(project).Combine($"{binDirBase}/{config}/{tfm}/{rid}"); + var binDir = new DirectoryPath(project).Combine($"{config}/{tfm}/{rid}"); IEnumerable applications; if (isAndroid) diff --git a/eng/devices/ios.cake b/eng/devices/ios.cake index 2d83da80853d..9afe44a751b9 100644 --- a/eng/devices/ios.cake +++ b/eng/devices/ios.cake @@ -18,9 +18,6 @@ var platform = testDevice.ToLower().Contains("simulator") ? "iPhoneSimulator" : var runtimeIdentifier = Argument("rid", EnvironmentVariable("IOS_RUNTIME_IDENTIFIER") ?? GetDefaultRuntimeIdentifier(testDevice)); var deviceCleanupEnabled = Argument("cleanup", true); -// Test where clause -string testWhere = Argument("where", EnvironmentVariable("NUNIT_TEST_WHERE") ?? ""); - // Device details var udid = Argument("udid", EnvironmentVariable("IOS_SIMULATOR_UDID") ?? ""); var iosVersion = Argument("apiversion", EnvironmentVariable("IOS_PLATFORM_VERSION") ?? DefaultVersion); @@ -87,7 +84,6 @@ Task("uitest-build") }); Task("uitest") - .IsDependentOn("uitest-build") .Does(() => { ExecuteUITests(projectPath, testAppProjectPath, testDevice, testResultsPath, binlogDirectory, configuration, targetFramework, runtimeIdentifier, iosVersion, dotnetToolPath); @@ -212,7 +208,12 @@ void ExecuteUITests(string project, string app, string device, string resultsDir var name = System.IO.Path.GetFileNameWithoutExtension(project); var binlog = $"{binDir}/{name}-{config}-ios.binlog"; var appiumLog = $"{binDir}/appium_ios.log"; - var resultsFileName = $"{name}-{config}-ios"; + var resultsFileName = $"{name}-{config}-ios-{testFilter.Replace("|", "_").Replace("TestCategory=", "")}"; + + if (resultsFileName.Contains("!~'.+'")) + { + resultsFileName = $"{name}-{config}-ios-notcategorized"; + } DotNetBuild(project, new DotNetBuildSettings { diff --git a/eng/devices/windows.cake b/eng/devices/windows.cake index 449926f74ffc..5b5ef64c4962 100644 --- a/eng/devices/windows.cake +++ b/eng/devices/windows.cake @@ -212,7 +212,20 @@ Task("Test") Information("Cleaned directories"); var testResultsPath = MakeAbsolute((DirectoryPath)TEST_RESULTS).FullPath.Replace("/", "\\"); - var testResultsFile = testResultsPath + $"\\TestResults-{PACKAGEID.Replace(".", "_")}.xml"; + var testResultsFile = testResultsPath + $"\\TestResults-{PACKAGEID.Replace(".", "_")}"; + + if (!string.IsNullOrWhiteSpace(testFilter)) + { + testResultsFile += $"-{testFilter.Replace("|", "_").Replace("TestCategory=", "")}"; + } + + if (testResultsFile.Contains("!~'.+'")) + { + testResultsFile = $"\\TestResults-{PACKAGEID.Replace(".", "_")}-notcategorized"; + } + + testResultsFile += ".xml"; + var testsToRunFile = MakeAbsolute((DirectoryPath)TEST_RESULTS).FullPath.Replace("/", "\\") + $"\\devicetestcategories.txt"; Information($"Test Results File: {testResultsFile}"); @@ -432,7 +445,7 @@ Task("SetupTestPaths") } var winVersion = $"{dotnetVersion}-windows{windowsVersion}"; - var binDir = TEST_APP_PROJECT.GetDirectory().Combine("bin").Combine(CONFIGURATION + "/" + winVersion).Combine(DOTNET_PLATFORM); + var binDir = TEST_APP_PROJECT.GetDirectory().Combine("Controls.TestCases.HostApp").Combine(CONFIGURATION + "/" + winVersion).Combine(DOTNET_PLATFORM); Information("BinDir: {0}", binDir); var apps = GetFiles(binDir + "/*.exe").Where(c => !c.FullPath.EndsWith("createdump.exe")); if (apps.Count() == 0) @@ -464,7 +477,7 @@ Task("SetupTestPaths") binDir = MakeAbsolute(new DirectoryPath(arcadeBin + "/Essentials.DeviceTests/" + CONFIGURATION + "/" + winVersion).Combine(DOTNET_PLATFORM)); } - Information("Looking for .app in arcade binDir {0}", binDir); + Information("Looking for .exe in arcade binDir {0}", binDir); apps = GetFiles(binDir + "/*.exe").Where(c => !c.FullPath.EndsWith("createdump.exe")); if(apps.Count() == 0 ) { diff --git a/eng/pipelines/common/ui-tests-build-sample.yml b/eng/pipelines/common/ui-tests-build-sample.yml new file mode 100644 index 000000000000..c9d411062eda --- /dev/null +++ b/eng/pipelines/common/ui-tests-build-sample.yml @@ -0,0 +1,69 @@ +parameters: + platform: '' # [ android, ios, windows, catalyst ] + path: '' # path to csproj + device: '' # the xharness device to use + cakeArgs: '' # additional cake args + app: '' #path to app to test + version: '' #the iOS version' + provisionatorChannel: 'latest' + agentPoolAccessToken: '' + configuration : "Release" + testFilter: '' + +steps: + - ${{ if eq(parameters.platform, 'ios')}}: + - bash: | + chmod +x $(System.DefaultWorkingDirectory)/eng/scripts/clean-bot.sh + $(System.DefaultWorkingDirectory)/eng/scripts/clean-bot.sh + displayName: 'Clean bot' + continueOnError: true + timeoutInMinutes: 60 + + - template: provision.yml + parameters: + skipProvisioning: ${{ eq(parameters.platform, 'windows') }} + skipAndroidSdks: ${{ ne(parameters.platform, 'android') }} + skipXcode: ${{ or(eq(parameters.platform, 'android'), eq(parameters.platform, 'windows')) }} + provisionatorChannel: ${{ parameters.provisionatorChannel }} + + - task: PowerShell@2 + condition: ne('${{ parameters.platform }}' , 'windows') + inputs: + targetType: 'inline' + script: | + defaults write -g NSAutomaticCapitalizationEnabled -bool false + defaults write -g NSAutomaticTextCompletionEnabled -bool false + defaults write -g NSAutomaticSpellingCorrectionEnabled -bool false + displayName: "Modify defaults" + continueOnError: true + + - pwsh: ./build.ps1 --target=dotnet --configuration="${{ parameters.configuration }}" --verbosity=diagnostic + displayName: 'Install .NET' + retryCountOnTaskFailure: 2 + env: + DOTNET_TOKEN: $(dotnetbuilds-internal-container-read-token) + PRIVATE_BUILD: $(PrivateBuild) + + - pwsh: echo "##vso[task.prependpath]$(DotNet.Dir)" + displayName: 'Add .NET to PATH' + + - pwsh: ./build.ps1 --target=dotnet-buildtasks --configuration="${{ parameters.configuration }}" + displayName: 'Build the MSBuild Tasks' + + - pwsh: ./build.ps1 --target=dotnet-samples --configuration="${{ parameters.configuration }}" --${{ parameters.platform }} --verbosity=diagnostic --usenuget=false + displayName: 'Build the samples' + + - bash: | + if [ -f "$HOME/Library/Logs/CoreSimulator/*" ]; then rm -r $HOME/Library/Logs/CoreSimulator/*; fi + if [ -f "$HOME/Library/Logs/DiagnosticReports/*" ]; then rm -r $HOME/Library/Logs/DiagnosticReports/*; fi + displayName: Delete Old Simulator Logs + condition: ${{ eq(parameters.platform, 'ios') }} + continueOnError: true + + - publish: $(System.DefaultWorkingDirectory)/artifacts/bin + condition: ne('${{ parameters.platform }}' , 'windows') + artifact: ui-tests-samples + + - publish: $(System.DefaultWorkingDirectory)/artifacts/bin + condition: eq('${{ parameters.platform }}' , 'windows') + artifact: ui-tests-samples-windows \ No newline at end of file diff --git a/eng/pipelines/common/ui-tests-compatibility-steps.yml b/eng/pipelines/common/ui-tests-compatibility-steps.yml index 8885cb1466fe..d234ef26e0eb 100644 --- a/eng/pipelines/common/ui-tests-compatibility-steps.yml +++ b/eng/pipelines/common/ui-tests-compatibility-steps.yml @@ -8,9 +8,9 @@ parameters: provisionatorChannel: 'latest' agentPoolAccessToken: '' configuration : "Release" - where : "cat==Issues" targetSample: "dotnet-legacy-controlgallery" provisionPlatform: "windows" + testFilter: '' steps: - bash: | @@ -56,7 +56,19 @@ steps: condition: ${{ eq(parameters.platform, 'ios') }} continueOnError: true - - pwsh: ./build.ps1 -Script eng/devices/${{ parameters.platform }}.cake --target=cg-uitest --project="${{ parameters.path }}" --appproject="${{ parameters.app }}" --device="${{ parameters.device }}" --apiversion="${{ parameters.version }}" --configuration="${{ parameters.configuration }}" --results="$(TestResultsDirectory)" --binlog="$(LogDirectory)" ${{ parameters.cakeArgs }} --verbosity=diagnostic --where="${{ parameters.where }}" + - pwsh: | + $command = "./build.ps1 -Script eng/devices/${{ parameters.platform }}.cake --target=cg-uitest --project=""${{ parameters.path }}""" + $command += " --appproject=""${{ parameters.app }}"" --device=""${{ parameters.device }}"" --apiversion=""${{ parameters.version }}"" --configuration=""${{ parameters.configuration }}""" + $command += " --results=""$(TestResultsDirectory)"" --binlog=""$(LogDirectory)"" ${{ parameters.cakeArgs }} --verbosity=diagnostic" + + $testFilter = "${{ parameters.testFilter }}" + + # Cake does not allow empty parameters, so check if our filter is empty before adding it + if ($testFilter) { + $command += " --test-filter ""${{ parameters.testFilter }}""" + } + + Invoke-Expression $command displayName: $(Agent.JobName) # retryCountOnTaskFailure: 2 diff --git a/eng/pipelines/common/ui-tests-steps.yml b/eng/pipelines/common/ui-tests-steps.yml index 37edc0526431..c802cb73ab24 100644 --- a/eng/pipelines/common/ui-tests-steps.yml +++ b/eng/pipelines/common/ui-tests-steps.yml @@ -8,33 +8,28 @@ parameters: provisionatorChannel: 'latest' agentPoolAccessToken: '' configuration : "Release" + testFilter: '' steps: - - ${{ if eq(parameters.platform, 'ios')}}: - - bash: | - chmod +x $(System.DefaultWorkingDirectory)/eng/scripts/clean-bot.sh - $(System.DefaultWorkingDirectory)/eng/scripts/clean-bot.sh - displayName: 'Clean bot' - continueOnError: true - timeoutInMinutes: 60 - - - template: provision.yml - parameters: - skipProvisioning: ${{ eq(parameters.platform, 'windows') }} - skipAndroidSdks: ${{ ne(parameters.platform, 'android') }} - skipXcode: ${{ or(eq(parameters.platform, 'android'), eq(parameters.platform, 'windows')) }} - provisionatorChannel: ${{ parameters.provisionatorChannel }} - - - task: PowerShell@2 + - task: DownloadPipelineArtifact@2 condition: ne('${{ parameters.platform }}' , 'windows') inputs: - targetType: 'inline' - script: | - defaults write -g NSAutomaticCapitalizationEnabled -bool false - defaults write -g NSAutomaticTextCompletionEnabled -bool false - defaults write -g NSAutomaticSpellingCorrectionEnabled -bool false - displayName: "Modify defaults" - continueOnError: true + artifact: ui-tests-samples + + - task: DownloadPipelineArtifact@2 + condition: eq('${{ parameters.platform }}' , 'windows') + inputs: + artifact: ui-tests-samples-windows + + - pwsh: ./build.ps1 --target=dotnet --configuration="${{ parameters.configuration }}" --verbosity=diagnostic + displayName: 'Install .NET' + retryCountOnTaskFailure: 2 + env: + DOTNET_TOKEN: $(dotnetbuilds-internal-container-read-token) + PRIVATE_BUILD: $(PrivateBuild) + + - pwsh: echo "##vso[task.prependpath]$(DotNet.Dir)" + displayName: 'Add .NET to PATH' # AzDO hosted agents default to 1024x768; set something bigger for Windows UI tests - task: ScreenResolutionUtility@1 @@ -60,30 +55,32 @@ steps: env: APPIUM_HOME: $(APPIUM_HOME) - - pwsh: ./build.ps1 --target=dotnet --configuration="${{ parameters.configuration }}" --verbosity=diagnostic - displayName: 'Install .NET' - retryCountOnTaskFailure: 2 - env: - DOTNET_TOKEN: $(dotnetbuilds-internal-container-read-token) - PRIVATE_BUILD: $(PrivateBuild) - - - pwsh: echo "##vso[task.prependpath]$(DotNet.Dir)" - displayName: 'Add .NET to PATH' - - - pwsh: ./build.ps1 --target=dotnet-buildtasks --configuration="${{ parameters.configuration }}" - displayName: 'Build the MSBuild Tasks' - - - pwsh: ./build.ps1 --target=dotnet-samples --configuration="${{ parameters.configuration }}" --${{ parameters.platform }} --verbosity=diagnostic --usenuget=false - displayName: 'Build the samples' - - - bash: | - if [ -f "$HOME/Library/Logs/CoreSimulator/*" ]; then rm -r $HOME/Library/Logs/CoreSimulator/*; fi - if [ -f "$HOME/Library/Logs/DiagnosticReports/*" ]; then rm -r $HOME/Library/Logs/DiagnosticReports/*; fi - displayName: Delete Old Simulator Logs - condition: ${{ eq(parameters.platform, 'ios') }} - continueOnError: true - - - pwsh: ./build.ps1 -Script eng/devices/${{ parameters.platform }}.cake --target=uitest --project="${{ parameters.path }}" --appproject="${{ parameters.app }}" --device="${{ parameters.device }}" --apiversion="${{ parameters.version }}" --configuration="${{ parameters.configuration }}" --results="$(TestResultsDirectory)" --binlog="$(LogDirectory)" ${{ parameters.cakeArgs }} --verbosity=diagnostic + - pwsh: | + $command = "./build.ps1 -Script eng/devices/${{ parameters.platform }}.cake --target=uitest --project=""${{ parameters.path }}""" + $command += " --appproject=""${{ parameters.app }}"" --device=""${{ parameters.device }}"" --apiversion=""${{ parameters.version }}"" --configuration=""${{ parameters.configuration }}""" + $command += " --results=""$(TestResultsDirectory)"" --binlog=""$(LogDirectory)"" ${{ parameters.cakeArgs }} --verbosity=diagnostic" + + $testFilter = "" + + if ("${{ parameters.testFilter }}" -eq "zzEmptyCategoryzz") + { + $testFilter = "TestCategory\!~'.+'" + } + else + { + "${{ parameters.testFilter }}".Split(",") | ForEach { + $testFilter += "TestCategory=" + $_ + "|" + } + + $testFilter = $testFilter.TrimEnd("|") + } + + # Cake does not allow empty parameters, so check if our filter is empty before adding it + if ($testFilter) { + $command += " --test-filter ""$testFilter""" + } + + Invoke-Expression $command displayName: $(Agent.JobName) ${{ if ne(parameters.platform, 'android')}}: retryCountOnTaskFailure: 1 diff --git a/eng/pipelines/common/ui-tests.yml b/eng/pipelines/common/ui-tests.yml index 9ae31afb61cb..08575c0db8e7 100644 --- a/eng/pipelines/common/ui-tests.yml +++ b/eng/pipelines/common/ui-tests.yml @@ -10,6 +10,24 @@ parameters: provisionatorChannel: 'latest' agentPoolAccessToken: '' runCompatibilityTests: false + categoryGroupsToTest: + # Make sure that this list is always up-to-date with src/Controls/tests/TestCases.Shared.Tests/UITestCategories.cs + # we might want to improve this somehow depending on how much the categories change over time + - 'Accessibility,ActionSheet,ActivityIndicator,Animation,AppLinks' + - 'Border,BoxView,Brush,Button' + - 'CarouselView,Cells,CheckBox,CollectionView,ContextActions,CustomRenderers' + - 'DatePicker,Dispatcher,DisplayAlert,DisplayPrompt,DragAndDrop' + - 'Editor,Effects,Entry,FlyoutPage,Focus,Frame,Gestures' + - 'Image,ImageButton,IndicatorView,InputTransparent,IsEnabled,IsVisible' + - 'Label,Layout,Lifecycle,ListView' + - 'ManualReview,Maps,Navigation' + - 'Page,Performance,Picker,ProgressBar' + - 'RadioButton,RefreshView' + - 'ScrollView,SearchBar,Shape,Shell,Slider,SoftInput,Stepper,Switch,SwipeView' + - 'TabbedPage,TableView,TimePicker,TitleView,ToolbarItem' + - 'ViewBaseTests,Visual,WebView,Window' + #- 'zzEmptyCategoryzz' # reserved keyword to make sure we run tests with no category specified + projects: - name: name desc: Human Description @@ -24,16 +42,48 @@ parameters: compatibilityiOSApp: /optional/path/to/app.csproj stages: + - stage: build_ui_tests + displayName: Build UITests Sample App + dependsOn: [] + jobs: + - job: build_ui_tests + displayName: Build Sample App + pool: ${{ parameters.androidPool }} + variables: + REQUIRED_XCODE: $(DEVICETESTS_REQUIRED_XCODE) + APPIUM_HOME: $(System.DefaultWorkingDirectory)/.appium/ + steps: + - template: ui-tests-build-sample.yml + + - stage: build_ui_tests_windows + displayName: Build UITests Windows Sample App + dependsOn: [] + jobs: + - job: build_ui_tests + displayName: Build Sample App (Windows) + pool: ${{ parameters.windowsPool }} + variables: + REQUIRED_XCODE: $(DEVICETESTS_REQUIRED_XCODE) + APPIUM_HOME: $(System.DefaultWorkingDirectory)/.appium/ + steps: + - template: ui-tests-build-sample.yml + parameters: + platform: windows - stage: android_ui_tests displayName: Android UITests - dependsOn: [] + dependsOn: build_ui_tests jobs: - ${{ each project in parameters.projects }}: - ${{ if ne(project.android, '') }}: - ${{ each api in parameters.androidApiLevels }}: - ${{ if not(containsValue(project.androidApiLevelsExclude, api)) }}: - job: android_ui_tests_${{ project.name }}_${{ api }} + strategy: + matrix: + ${{ each categoryGroup in parameters.categoryGroupsToTest }}: + ${{ categoryGroup }}: + CATEGORYGROUP: ${{ categoryGroup }} timeoutInMinutes: 240 # how long to run the job before automatically cancelling workspace: clean: all @@ -55,16 +105,22 @@ stages: device: android-emulator-64_${{ api }} provisionatorChannel: ${{ parameters.provisionatorChannel }} agentPoolAccessToken: ${{ parameters.agentPoolAccessToken }} + testFilter: $(CATEGORYGROUP) - stage: ios_ui_tests displayName: iOS UITests - dependsOn: [] + dependsOn: build_ui_tests jobs: - ${{ each project in parameters.projects }}: - ${{ if ne(project.ios, '') }}: - ${{ each version in parameters.iosVersions }}: - ${{ if not(containsValue(project.iosVersionsExclude, version)) }}: - job: ios_ui_tests_${{ project.name }}_${{ replace(version, '.', '_') }} + strategy: + matrix: + ${{ each categoryGroup in parameters.categoryGroupsToTest }}: + ${{ categoryGroup }}: + CATEGORYGROUP: ${{ categoryGroup }} timeoutInMinutes: 240 # how long to run the job before automatically cancelling workspace: clean: all @@ -89,14 +145,20 @@ stages: device: ios-simulator-64_${{ version }} provisionatorChannel: ${{ parameters.provisionatorChannel }} agentPoolAccessToken: ${{ parameters.agentPoolAccessToken }} + testFilter: $(CATEGORYGROUP) - stage: winui_ui_tests displayName: WinUI UITests - dependsOn: [] + dependsOn: build_ui_tests_windows jobs: - ${{ each project in parameters.projects }}: - ${{ if ne(project.winui, '') }}: - job: winui_ui_tests_${{ project.name }} + strategy: + matrix: + ${{ each categoryGroup in parameters.categoryGroupsToTest }}: + ${{ categoryGroup }}: + CATEGORYGROUP: ${{ categoryGroup }} timeoutInMinutes: 240 # how long to run the job before automatically cancelling workspace: clean: all @@ -114,14 +176,20 @@ stages: app: ${{ project.app }} provisionatorChannel: ${{ parameters.provisionatorChannel }} agentPoolAccessToken: ${{ parameters.agentPoolAccessToken }} + testFilter: $(CATEGORYGROUP) - stage: mac_ui_tests displayName: macOS UITests - dependsOn: [] + dependsOn: build_ui_tests jobs: - ${{ each project in parameters.projects }}: - ${{ if ne(project.mac, '') }}: - job: mac_ui_tests_${{ project.name }} + strategy: + matrix: + ${{ each categoryGroup in parameters.categoryGroupsToTest }}: + ${{ categoryGroup }}: + CATEGORYGROUP: ${{ categoryGroup }} timeoutInMinutes: 240 # how long to run the job before automatically cancelling workspace: clean: all @@ -140,6 +208,7 @@ stages: app: ${{ project.app }} provisionatorChannel: ${{ parameters.provisionatorChannel }} agentPoolAccessToken: ${{ parameters.agentPoolAccessToken }} + testFilter: $(CATEGORYGROUP) - ${{ if eq(parameters.runCompatibilityTests, true) }}: - stage: android_compatibility_ui_tests @@ -151,6 +220,11 @@ stages: - ${{ each api in parameters.androidApiLevels }}: - ${{ if not(containsValue(project.androidApiLevelsExclude, api)) }}: - job: android_compatibility_ui_tests_${{ project.name }}_${{ api }} + strategy: + matrix: + ${{ each categoryGroup in parameters.categoryGroupsToTest }}: + ${{ categoryGroup }}: + CATEGORYGROUP: ${{ categoryGroup }} timeoutInMinutes: 240 workspace: clean: all @@ -174,6 +248,7 @@ stages: DEVICE: android-emulator-64_${{ api }} provisionatorChannel: ${{ parameters.provisionatorChannel }} agentPoolAccessToken: ${{ parameters.agentPoolAccessToken }} + testFilter: $(CATEGORYGROUP) - stage: ios_compatibility_ui_tests displayName: iOS Compatibility UITests @@ -184,6 +259,11 @@ stages: - ${{ each version in parameters.iosVersions }}: - ${{ if not(containsValue(project.iosVersionsExclude, version)) }}: - job: ios_compatibility_ui_tests_${{ project.name }}_${{ replace(version, '.', '_') }} + strategy: + matrix: + ${{ each categoryGroup in parameters.categoryGroupsToTest }}: + ${{ categoryGroup }}: + CATEGORYGROUP: ${{ categoryGroup }} timeoutInMinutes: 240 workspace: clean: all @@ -210,3 +290,4 @@ stages: device: ios-simulator-64_${{ version }} provisionatorChannel: ${{ parameters.provisionatorChannel }} agentPoolAccessToken: ${{ parameters.agentPoolAccessToken }} + testFilter: $(CATEGORYGROUP) \ No newline at end of file diff --git a/eng/pipelines/ui-tests.yml b/eng/pipelines/ui-tests.yml index 217c21551e6c..c7e09bfb3ffc 100644 --- a/eng/pipelines/ui-tests.yml +++ b/eng/pipelines/ui-tests.yml @@ -143,7 +143,7 @@ stages: desc: Controls androidApiLevelsExclude: [25] # Ignore for now API25 since the runs's are not stable android: $(System.DefaultWorkingDirectory)/src/Controls/tests/TestCases.Android.Tests/Controls.TestCases.Android.Tests.csproj - app: $(System.DefaultWorkingDirectory)/src/Controls/tests/TestCases.HostApp/Controls.TestCases.HostApp.csproj + app: $(Pipeline.Workspace)/Controls.TestCases.HostApp/ iosVersionsExclude: [ '12.4'] # Ignore iOS 12.4 while we can't make it work on CI ios: $(System.DefaultWorkingDirectory)/src/Controls/tests/TestCases.iOS.Tests/Controls.TestCases.iOS.Tests.csproj winui: $(System.DefaultWorkingDirectory)/src/Controls/tests/TestCases.WinUI.Tests/Controls.TestCases.WinUI.Tests.csproj diff --git a/src/Controls/tests/TestCases.Shared.Tests/UITestCategories.cs b/src/Controls/tests/TestCases.Shared.Tests/UITestCategories.cs index 5d7f205d1fde..c4de089ef989 100644 --- a/src/Controls/tests/TestCases.Shared.Tests/UITestCategories.cs +++ b/src/Controls/tests/TestCases.Shared.Tests/UITestCategories.cs @@ -1,5 +1,7 @@ namespace Microsoft.Maui.TestCases.Tests { + // Make sure that this list is always up-to-date with eng/pipelines/common/ui-tests.yml + // we might want to improve this somehow depending on how much the categories change over time internal static class UITestCategories { public const string ViewBaseTests = "ViewBaseTests"; From 164feaff4ee4add1786c9487356a5920098c1eb3 Mon Sep 17 00:00:00 2001 From: Gerald Versluis Date: Fri, 5 Jul 2024 10:47:43 +0200 Subject: [PATCH 2/6] Some changes --- eng/devices/android.cake | 7 +------ eng/devices/catalyst.cake | 7 +------ eng/devices/devices-shared.cake | 7 +++++++ eng/devices/ios.cake | 7 +------ eng/devices/windows.cake | 7 +------ eng/pipelines/common/ui-tests-steps.yml | 13 +++---------- eng/pipelines/common/ui-tests.yml | 1 - .../Tests/ScrollViewUITests.cs | 4 ---- 8 files changed, 14 insertions(+), 39 deletions(-) diff --git a/eng/devices/android.cake b/eng/devices/android.cake index dd9eb4d31c6a..4ddbcab4aaf9 100644 --- a/eng/devices/android.cake +++ b/eng/devices/android.cake @@ -301,12 +301,7 @@ void ExecuteUITests(string project, string app, string appPackageName, string de var name = System.IO.Path.GetFileNameWithoutExtension(project); var binlog = $"{binDir}/{name}-{config}-{platform}.binlog"; var appiumLog = $"{binDir}/appium_{platform}.log"; - var resultsFileName = $"{name}-{config}-{platform}-{testFilter.Replace("|", "_").Replace("TestCategory=", "")}"; - - if (resultsFileName.Contains("!~'.+'")) - { - resultsFileName = $"{name}-{config}-{platform}-notcategorized"; - } + var resultsFileName = SanitizeTestResultsFilename($"{name}-{config}-{platform}-{testFilter}"); DotNetBuild(project, new DotNetBuildSettings { diff --git a/eng/devices/catalyst.cake b/eng/devices/catalyst.cake index 9818d77d52e2..5fe0fb1a2a2b 100644 --- a/eng/devices/catalyst.cake +++ b/eng/devices/catalyst.cake @@ -143,12 +143,7 @@ void ExecuteUITests(string project, string app, string device, string resultsDir var name = System.IO.Path.GetFileNameWithoutExtension(project); var binlog = $"{binDir}/{name}-{config}-mac.binlog"; var appiumLog = $"{binDir}/appium_mac.log"; - var resultsFileName = $"{name}-{config}-catalyst-{testFilter.Replace("|", "_").Replace("TestCategory=", "")}"; - - if (resultsFileName.Contains("!~'.+'")) - { - resultsFileName = $"{name}-{config}-catalyst-notcategorized"; - } + var resultsFileName = SanitizeTestResultsFilename($"{name}-{config}-catalyst-{testFilter}"); DotNetBuild(project, new DotNetBuildSettings { diff --git a/eng/devices/devices-shared.cake b/eng/devices/devices-shared.cake index 00e99facfaa7..6b99a2d483b7 100644 --- a/eng/devices/devices-shared.cake +++ b/eng/devices/devices-shared.cake @@ -222,3 +222,10 @@ void ExecuteWithRetries(Func action, int retries) System.Threading.Thread.Sleep(1000); } } + +string SanitizeTestResultsFilename(string input) +{ + string resultFilename = input.Replace("|", "_").Replace("TestCategory=", ""); + + return resultFilename; +} diff --git a/eng/devices/ios.cake b/eng/devices/ios.cake index 9afe44a751b9..8d94e17c0d87 100644 --- a/eng/devices/ios.cake +++ b/eng/devices/ios.cake @@ -208,12 +208,7 @@ void ExecuteUITests(string project, string app, string device, string resultsDir var name = System.IO.Path.GetFileNameWithoutExtension(project); var binlog = $"{binDir}/{name}-{config}-ios.binlog"; var appiumLog = $"{binDir}/appium_ios.log"; - var resultsFileName = $"{name}-{config}-ios-{testFilter.Replace("|", "_").Replace("TestCategory=", "")}"; - - if (resultsFileName.Contains("!~'.+'")) - { - resultsFileName = $"{name}-{config}-ios-notcategorized"; - } + var resultsFileName = SanitizeTestResultsFilename($"{name}-{config}-ios-{testFilter)}"); DotNetBuild(project, new DotNetBuildSettings { diff --git a/eng/devices/windows.cake b/eng/devices/windows.cake index 5b5ef64c4962..17a3aba88b34 100644 --- a/eng/devices/windows.cake +++ b/eng/devices/windows.cake @@ -216,12 +216,7 @@ Task("Test") if (!string.IsNullOrWhiteSpace(testFilter)) { - testResultsFile += $"-{testFilter.Replace("|", "_").Replace("TestCategory=", "")}"; - } - - if (testResultsFile.Contains("!~'.+'")) - { - testResultsFile = $"\\TestResults-{PACKAGEID.Replace(".", "_")}-notcategorized"; + testResultsFile += SanitizeTestResultsFilename($"-{testFilter}"); } testResultsFile += ".xml"; diff --git a/eng/pipelines/common/ui-tests-steps.yml b/eng/pipelines/common/ui-tests-steps.yml index c802cb73ab24..9e28c5579b6e 100644 --- a/eng/pipelines/common/ui-tests-steps.yml +++ b/eng/pipelines/common/ui-tests-steps.yml @@ -62,18 +62,11 @@ steps: $testFilter = "" - if ("${{ parameters.testFilter }}" -eq "zzEmptyCategoryzz") - { - $testFilter = "TestCategory\!~'.+'" + "${{ parameters.testFilter }}".Split(",") | ForEach { + $testFilter += "TestCategory=" + $_ + "|" } - else - { - "${{ parameters.testFilter }}".Split(",") | ForEach { - $testFilter += "TestCategory=" + $_ + "|" - } - $testFilter = $testFilter.TrimEnd("|") - } + $testFilter = $testFilter.TrimEnd("|") # Cake does not allow empty parameters, so check if our filter is empty before adding it if ($testFilter) { diff --git a/eng/pipelines/common/ui-tests.yml b/eng/pipelines/common/ui-tests.yml index 08575c0db8e7..bacbef8475cd 100644 --- a/eng/pipelines/common/ui-tests.yml +++ b/eng/pipelines/common/ui-tests.yml @@ -26,7 +26,6 @@ parameters: - 'ScrollView,SearchBar,Shape,Shell,Slider,SoftInput,Stepper,Switch,SwipeView' - 'TabbedPage,TableView,TimePicker,TitleView,ToolbarItem' - 'ViewBaseTests,Visual,WebView,Window' - #- 'zzEmptyCategoryzz' # reserved keyword to make sure we run tests with no category specified projects: - name: name diff --git a/src/Controls/tests/TestCases.Shared.Tests/Tests/ScrollViewUITests.cs b/src/Controls/tests/TestCases.Shared.Tests/Tests/ScrollViewUITests.cs index 6e41a367f975..e159d6989467 100644 --- a/src/Controls/tests/TestCases.Shared.Tests/Tests/ScrollViewUITests.cs +++ b/src/Controls/tests/TestCases.Shared.Tests/Tests/ScrollViewUITests.cs @@ -21,7 +21,6 @@ protected override void FixtureSetup() } [Test] - [Category(UITestCategories.ScrollView)] [Description("Scroll element to the start")] public void ScrollToElement1Start() { @@ -43,7 +42,6 @@ public void ScrollToElement1Start() } [Test] - [Category(UITestCategories.ScrollView)] [Description("Scroll element to the center")] public void ScrollToElement2Center() { @@ -68,7 +66,6 @@ public void ScrollToElement2Center() } [Test] - [Category(UITestCategories.ScrollView)] [Description("Scroll element to the end")] public void ScrollToElement3End() { @@ -130,7 +127,6 @@ public void ScrollToYTwice() } #if ANDROID || IOS [Test] - [Category(UITestCategories.ScrollView)] [Description("Scroll down the ScrollView using a gesture")] public void ScrollUpAndDownWithGestures() { From e3e4f87fe235913e3f3847828b8b503698407120 Mon Sep 17 00:00:00 2001 From: Gerald Versluis Date: Fri, 5 Jul 2024 11:31:45 +0200 Subject: [PATCH 3/6] Fix --- eng/devices/ios.cake | 2 +- eng/pipelines/common/ui-tests-compatibility-steps.yml | 4 +++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/eng/devices/ios.cake b/eng/devices/ios.cake index 8d94e17c0d87..9e9871fcc5ae 100644 --- a/eng/devices/ios.cake +++ b/eng/devices/ios.cake @@ -208,7 +208,7 @@ void ExecuteUITests(string project, string app, string device, string resultsDir var name = System.IO.Path.GetFileNameWithoutExtension(project); var binlog = $"{binDir}/{name}-{config}-ios.binlog"; var appiumLog = $"{binDir}/appium_ios.log"; - var resultsFileName = SanitizeTestResultsFilename($"{name}-{config}-ios-{testFilter)}"); + var resultsFileName = SanitizeTestResultsFilename($"{name}-{config}-ios-{testFilter}"); DotNetBuild(project, new DotNetBuildSettings { diff --git a/eng/pipelines/common/ui-tests-compatibility-steps.yml b/eng/pipelines/common/ui-tests-compatibility-steps.yml index d234ef26e0eb..449f6078eeb0 100644 --- a/eng/pipelines/common/ui-tests-compatibility-steps.yml +++ b/eng/pipelines/common/ui-tests-compatibility-steps.yml @@ -61,7 +61,9 @@ steps: $command += " --appproject=""${{ parameters.app }}"" --device=""${{ parameters.device }}"" --apiversion=""${{ parameters.version }}"" --configuration=""${{ parameters.configuration }}""" $command += " --results=""$(TestResultsDirectory)"" --binlog=""$(LogDirectory)"" ${{ parameters.cakeArgs }} --verbosity=diagnostic" - $testFilter = "${{ parameters.testFilter }}" + "${{ parameters.testFilter }}".Split(",") | ForEach { + $testFilter += "TestCategory=" + $_ + "|" + } # Cake does not allow empty parameters, so check if our filter is empty before adding it if ($testFilter) { From 57216ca1533a6d4000a0c27cbcf4ecba448e9e58 Mon Sep 17 00:00:00 2001 From: Gerald Versluis Date: Fri, 5 Jul 2024 15:01:08 +0200 Subject: [PATCH 4/6] Update ui-tests-compatibility-steps.yml --- eng/pipelines/common/ui-tests-compatibility-steps.yml | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/eng/pipelines/common/ui-tests-compatibility-steps.yml b/eng/pipelines/common/ui-tests-compatibility-steps.yml index 449f6078eeb0..030e3ec6f14d 100644 --- a/eng/pipelines/common/ui-tests-compatibility-steps.yml +++ b/eng/pipelines/common/ui-tests-compatibility-steps.yml @@ -61,13 +61,17 @@ steps: $command += " --appproject=""${{ parameters.app }}"" --device=""${{ parameters.device }}"" --apiversion=""${{ parameters.version }}"" --configuration=""${{ parameters.configuration }}""" $command += " --results=""$(TestResultsDirectory)"" --binlog=""$(LogDirectory)"" ${{ parameters.cakeArgs }} --verbosity=diagnostic" + $testFilter = "" + "${{ parameters.testFilter }}".Split(",") | ForEach { $testFilter += "TestCategory=" + $_ + "|" } + $testFilter = $testFilter.TrimEnd("|") + # Cake does not allow empty parameters, so check if our filter is empty before adding it if ($testFilter) { - $command += " --test-filter ""${{ parameters.testFilter }}""" + $command += " --test-filter ""$testFilter""" } Invoke-Expression $command From fed6951ced02b56691fa89ddbb8ee0fc99bd2f01 Mon Sep 17 00:00:00 2001 From: Gerald Versluis Date: Fri, 5 Jul 2024 16:54:17 +0200 Subject: [PATCH 5/6] No categories for compatibility project --- .../common/ui-tests-compatibility-steps.yml | 20 +------------------ eng/pipelines/common/ui-tests.yml | 14 +------------ 2 files changed, 2 insertions(+), 32 deletions(-) diff --git a/eng/pipelines/common/ui-tests-compatibility-steps.yml b/eng/pipelines/common/ui-tests-compatibility-steps.yml index 030e3ec6f14d..7d494f53cbea 100644 --- a/eng/pipelines/common/ui-tests-compatibility-steps.yml +++ b/eng/pipelines/common/ui-tests-compatibility-steps.yml @@ -56,25 +56,7 @@ steps: condition: ${{ eq(parameters.platform, 'ios') }} continueOnError: true - - pwsh: | - $command = "./build.ps1 -Script eng/devices/${{ parameters.platform }}.cake --target=cg-uitest --project=""${{ parameters.path }}""" - $command += " --appproject=""${{ parameters.app }}"" --device=""${{ parameters.device }}"" --apiversion=""${{ parameters.version }}"" --configuration=""${{ parameters.configuration }}""" - $command += " --results=""$(TestResultsDirectory)"" --binlog=""$(LogDirectory)"" ${{ parameters.cakeArgs }} --verbosity=diagnostic" - - $testFilter = "" - - "${{ parameters.testFilter }}".Split(",") | ForEach { - $testFilter += "TestCategory=" + $_ + "|" - } - - $testFilter = $testFilter.TrimEnd("|") - - # Cake does not allow empty parameters, so check if our filter is empty before adding it - if ($testFilter) { - $command += " --test-filter ""$testFilter""" - } - - Invoke-Expression $command + - pwsh: ./build.ps1 -Script eng/devices/${{ parameters.platform }}.cake --target=cg-uitest --project="${{ parameters.path }}" --appproject="${{ parameters.app }}" --device="${{ parameters.device }}" --apiversion="${{ parameters.version }}" --configuration="${{ parameters.configuration }}" --results="$(TestResultsDirectory)" --binlog="$(LogDirectory)" ${{ parameters.cakeArgs }} --verbosity=diagnostic --where="${{ parameters.where }}" displayName: $(Agent.JobName) # retryCountOnTaskFailure: 2 diff --git a/eng/pipelines/common/ui-tests.yml b/eng/pipelines/common/ui-tests.yml index bacbef8475cd..2cc19a42d6ac 100644 --- a/eng/pipelines/common/ui-tests.yml +++ b/eng/pipelines/common/ui-tests.yml @@ -219,11 +219,6 @@ stages: - ${{ each api in parameters.androidApiLevels }}: - ${{ if not(containsValue(project.androidApiLevelsExclude, api)) }}: - job: android_compatibility_ui_tests_${{ project.name }}_${{ api }} - strategy: - matrix: - ${{ each categoryGroup in parameters.categoryGroupsToTest }}: - ${{ categoryGroup }}: - CATEGORYGROUP: ${{ categoryGroup }} timeoutInMinutes: 240 workspace: clean: all @@ -247,7 +242,6 @@ stages: DEVICE: android-emulator-64_${{ api }} provisionatorChannel: ${{ parameters.provisionatorChannel }} agentPoolAccessToken: ${{ parameters.agentPoolAccessToken }} - testFilter: $(CATEGORYGROUP) - stage: ios_compatibility_ui_tests displayName: iOS Compatibility UITests @@ -258,11 +252,6 @@ stages: - ${{ each version in parameters.iosVersions }}: - ${{ if not(containsValue(project.iosVersionsExclude, version)) }}: - job: ios_compatibility_ui_tests_${{ project.name }}_${{ replace(version, '.', '_') }} - strategy: - matrix: - ${{ each categoryGroup in parameters.categoryGroupsToTest }}: - ${{ categoryGroup }}: - CATEGORYGROUP: ${{ categoryGroup }} timeoutInMinutes: 240 workspace: clean: all @@ -288,5 +277,4 @@ stages: ${{ if ne(version, 'latest') }}: device: ios-simulator-64_${{ version }} provisionatorChannel: ${{ parameters.provisionatorChannel }} - agentPoolAccessToken: ${{ parameters.agentPoolAccessToken }} - testFilter: $(CATEGORYGROUP) \ No newline at end of file + agentPoolAccessToken: ${{ parameters.agentPoolAccessToken }} \ No newline at end of file From 3b58662ea409f22ce7f9ff967cbc71e041aa9860 Mon Sep 17 00:00:00 2001 From: Gerald Versluis Date: Fri, 5 Jul 2024 17:09:36 +0200 Subject: [PATCH 6/6] Update ui-tests-compatibility-steps.yml --- eng/pipelines/common/ui-tests-compatibility-steps.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/eng/pipelines/common/ui-tests-compatibility-steps.yml b/eng/pipelines/common/ui-tests-compatibility-steps.yml index 7d494f53cbea..28f29d1eaa6e 100644 --- a/eng/pipelines/common/ui-tests-compatibility-steps.yml +++ b/eng/pipelines/common/ui-tests-compatibility-steps.yml @@ -56,7 +56,7 @@ steps: condition: ${{ eq(parameters.platform, 'ios') }} continueOnError: true - - pwsh: ./build.ps1 -Script eng/devices/${{ parameters.platform }}.cake --target=cg-uitest --project="${{ parameters.path }}" --appproject="${{ parameters.app }}" --device="${{ parameters.device }}" --apiversion="${{ parameters.version }}" --configuration="${{ parameters.configuration }}" --results="$(TestResultsDirectory)" --binlog="$(LogDirectory)" ${{ parameters.cakeArgs }} --verbosity=diagnostic --where="${{ parameters.where }}" + - pwsh: ./build.ps1 -Script eng/devices/${{ parameters.platform }}.cake --target=cg-uitest --project="${{ parameters.path }}" --appproject="${{ parameters.app }}" --device="${{ parameters.device }}" --apiversion="${{ parameters.version }}" --configuration="${{ parameters.configuration }}" --results="$(TestResultsDirectory)" --binlog="$(LogDirectory)" ${{ parameters.cakeArgs }} --verbosity=diagnostic displayName: $(Agent.JobName) # retryCountOnTaskFailure: 2