diff --git a/doc/development/powershell.md b/doc/development/powershell.md index f313dc40c5a..57132f5ff4b 100644 --- a/doc/development/powershell.md +++ b/doc/development/powershell.md @@ -3,14 +3,15 @@ This page contains guidelines for developing or updating powershell scripts used Table of Contents ================= -* [TLDR](#tldr) -* [Structure](#structure) -* [Functionality](#functionality) -* [Style](#style) -* [Testing](#testing) - * [Unit Testing](#unit-testing) - * [Running Pester Tests](#running-pester-tests) - * [Local/Pipeline Functional Testing](#localpipeline-functional-testing) +- [Table of Contents](#table-of-contents) + - [TLDR](#tldr) + - [Structure](#structure) + - [Functionality](#functionality) + - [Style](#style) + - [Testing](#testing) + - [Unit Testing](#unit-testing) + - [Running Pester Tests](#running-pester-tests) + - [Local/Pipeline Functional Testing](#localpipeline-functional-testing) ## TLDR @@ -110,13 +111,13 @@ Powershell scripts should be testable, via one or more methods: Unit tests should be written for all scripts, and should utilize [Pester](https://pester.dev/). - Tests can be located alongside scripts in a directory called `tests`. -- Example pester test suites: [job matrix tests](https://github.com/Azure/azure-sdk-tools/tree/main/eng/common/scripts/job-matrix/tests), [asset sync tests](https://github.com/Azure/azure-sdk-tools/blob/main/tools/asset-sync/assets.Tests.ps1) +- Example pester test suites: [job matrix tests](https://github.com/Azure/azure-sdk-tools/tree/main/eng/common/scripts/job-matrix/tests), [asset sync tests](https://github.com/Azure/azure-sdk-tools/blob/main/tools/assets-automation/asset-sync/assets.Tests.ps1) - A CI pipeline should be defined to run scripts unit tests at the very least. See [archetype-sdk-tool-pwsh](https://github.com/Azure/azure-sdk-tools/blob/main/eng/common/pipelines/templates/stages/archetype-sdk-tool-pwsh.yml) for how to do this. - Script code should always be written so as much of the surface area as possible can be run via unit tests. Move code that calls out to external dependencies into modular functions, and simplify context/data structures passed to functions as much as possible to it can be easily mocked. #### Running Pester Tests -(stolen from https://github.com/Azure/azure-sdk-tools/blob/main/tools/asset-sync/contributing.md). +(stolen from https://github.com/Azure/azure-sdk-tools/blob/main/tools/assets-automation/asset-sync/contributing.md). > **First, ensure you have `pester` installed:** > diff --git a/eng/common/scripts/TypeSpec-Project-Process.ps1 b/eng/common/scripts/TypeSpec-Project-Process.ps1 index a27036a80a4..71c7e003ce5 100644 --- a/eng/common/scripts/TypeSpec-Project-Process.ps1 +++ b/eng/common/scripts/TypeSpec-Project-Process.ps1 @@ -8,7 +8,8 @@ param ( [Parameter(Position = 1)] [string] $CommitHash, [Parameter(Position = 2)] - [string] $RepoUrl + [string] $RepoUrl, + [switch] $SkipSyncAndGenerate ) . $PSScriptRoot/common.ps1 @@ -111,7 +112,7 @@ $specRepoRoot = "" $generateFromLocalTypeSpec = $false # remote url scenario # example url of tspconfig.yaml: https://github.com/Azure/azure-rest-api-specs-pr/blob/724ccc4d7ef7655c0b4d5c5ac4a5513f19bbef35/specification/containerservice/Fleet.Management/tspconfig.yaml -if ($TypeSpecProjectDirectory -match '^https://github.com/(?Azure/azure-rest-api-specs(-pr)?)/blob/(?[0-9a-f]{40})/(?.*)/tspconfig.yaml$') { +if ($TypeSpecProjectDirectory -match '^https://github.com/(?[^/]*/azure-rest-api-specs(-pr)?)/blob/(?[0-9a-f]{40})/(?.*)/tspconfig.yaml$') { try { $TypeSpecProjectDirectory = $TypeSpecProjectDirectory -replace "https://github.com/(.*)/(tree|blob)", "https://raw.githubusercontent.com/`$1" Invoke-WebRequest $TypeSpecProjectDirectory -OutFile $tspConfigPath -MaximumRetryCount 3 @@ -199,14 +200,19 @@ if ($generateFromLocalTypeSpec) { $sdkProjectFolder = CreateUpdate-TspLocation $tspConfigYaml $TypeSpecProjectDirectory $CommitHash $repo $sdkRepoRootPath } -# call TypeSpec-Project-Sync.ps1 -$syncScript = Join-Path $PSScriptRoot TypeSpec-Project-Sync.ps1 -& $syncScript $sdkProjectFolder $specRepoRoot -if ($LASTEXITCODE) { exit $LASTEXITCODE } - -# call TypeSpec-Project-Generate.ps1 -$generateScript = Join-Path $PSScriptRoot TypeSpec-Project-Generate.ps1 -& $generateScript $sdkProjectFolder -if ($LASTEXITCODE) { exit $LASTEXITCODE } +# checking skip switch +if ($SkipSyncAndGenerate) { + Write-Host "Skip calling TypeSpec-Project-Sync.ps1 and TypeSpec-Project-Generate.ps1." +} else { + # call TypeSpec-Project-Sync.ps1 + $syncScript = Join-Path $PSScriptRoot TypeSpec-Project-Sync.ps1 + & $syncScript $sdkProjectFolder $specRepoRoot + if ($LASTEXITCODE) { exit $LASTEXITCODE } + + # call TypeSpec-Project-Generate.ps1 + $generateScript = Join-Path $PSScriptRoot TypeSpec-Project-Generate.ps1 + & $generateScript $sdkProjectFolder + if ($LASTEXITCODE) { exit $LASTEXITCODE } +} return $sdkProjectFolder \ No newline at end of file diff --git a/src/dotnet/APIView/APIViewWeb/Managers/OpenSourceRequestManager.cs b/src/dotnet/APIView/APIViewWeb/Managers/OpenSourceRequestManager.cs index c5b6c715287..6baab8f1f79 100644 --- a/src/dotnet/APIView/APIViewWeb/Managers/OpenSourceRequestManager.cs +++ b/src/dotnet/APIView/APIViewWeb/Managers/OpenSourceRequestManager.cs @@ -37,23 +37,36 @@ public OpenSourceRequestManager(IConfiguration configuration) } public async Task GetUserInfo(string githubUserId) - { - try - { - var ossClient = new HttpClient(); - await SetHeaders(ossClient); - var response = await ossClient.GetAsync($"https://repos.opensource.microsoft.com/api/people/links/github/{githubUserId}"); - response.EnsureSuccessStatusCode(); - var userDetailsJson = await response.Content.ReadAsStringAsync(); - return JsonConvert.DeserializeObject(userDetailsJson); - } - catch (HttpRequestException ex) when (ex.StatusCode == HttpStatusCode.NotFound) - { - _telemetryClient.TrackTrace($"Github username {githubUserId} is not found"); - } - catch (Exception ex) - { - _telemetryClient.TrackException(ex); + { + int retryCount = 0; + bool authCheckCompleted = false; + while (!authCheckCompleted && retryCount < 3) + { + try + { + retryCount++; + var ossClient = new HttpClient(); + await SetHeaders(ossClient); + var response = await ossClient.GetAsync($"https://repos.opensource.microsoft.com/api/people/links/github/{githubUserId}"); + response.EnsureSuccessStatusCode(); + var userDetailsJson = await response.Content.ReadAsStringAsync(); + return JsonConvert.DeserializeObject(userDetailsJson); + } + catch (HttpRequestException ex) when (ex.StatusCode == HttpStatusCode.NotFound) + { + _telemetryClient.TrackTrace($"GitHub username {githubUserId} is not found"); + authCheckCompleted = true; + } + catch (Exception ex) + { + _telemetryClient.TrackException(ex); + } + + if(!authCheckCompleted && retryCount < 3) + { + await Task.Delay(2000); + _telemetryClient.TrackTrace($"Retrying to check user authorization for user Id {githubUserId}"); + } } return null; } @@ -63,7 +76,7 @@ public async Task IsAuthorizedUser(string githubUserId) var resp = await GetUserInfo(githubUserId); if (resp == null) return false; - // For now we only need to check if user info is availableon MS OSS + // For now we only need to check if user info is available on MS OSS return true; } diff --git a/tools/assets-automation/README.md b/tools/assets-automation/README.md new file mode 100644 index 00000000000..709667d5866 --- /dev/null +++ b/tools/assets-automation/README.md @@ -0,0 +1,9 @@ +# Assets automation tooling + +This directory contains tooling pertaining to the _support_ of externalized assets created by the [azure-sdk test-proxy](../test-proxy/Azure.Sdk.Tools.TestProxy/README.md). + +| Directory | Description | +|---|---| +| [assets-maintenance-tool](./assets-maintenance-tool/README.md) | CLI tool used to scan, backup, and clean azure-sdk assets across all repositories. | +| [assets-reporting](./assets-reporting/README.md) | CLI tool used to audit current repositories and find status of test-proxy adoption on a per-package basis. Used to generate weekly reporting. | +| [asset-sync](./assets-sync/README.md) | Deprecated initial version of `asset-sync` implementation. The current implementation ended up integrated directly into the `test-proxy` codebase, rather than existing as an external powershell script. | diff --git a/tools/asset-sync/README.md b/tools/assets-automation/asset-sync/README.md similarity index 100% rename from tools/asset-sync/README.md rename to tools/assets-automation/asset-sync/README.md diff --git a/tools/asset-sync/assets.Tests.Helpers.ps1 b/tools/assets-automation/asset-sync/assets.Tests.Helpers.ps1 similarity index 100% rename from tools/asset-sync/assets.Tests.Helpers.ps1 rename to tools/assets-automation/asset-sync/assets.Tests.Helpers.ps1 diff --git a/tools/asset-sync/assets.Tests.ps1 b/tools/assets-automation/asset-sync/assets.Tests.ps1 similarity index 100% rename from tools/asset-sync/assets.Tests.ps1 rename to tools/assets-automation/asset-sync/assets.Tests.ps1 diff --git a/tools/asset-sync/assets.json b/tools/assets-automation/asset-sync/assets.json similarity index 100% rename from tools/asset-sync/assets.json rename to tools/assets-automation/asset-sync/assets.json diff --git a/tools/asset-sync/assets.ps1 b/tools/assets-automation/asset-sync/assets.ps1 similarity index 100% rename from tools/asset-sync/assets.ps1 rename to tools/assets-automation/asset-sync/assets.ps1 diff --git a/tools/asset-sync/contributing.md b/tools/assets-automation/asset-sync/contributing.md similarity index 100% rename from tools/asset-sync/contributing.md rename to tools/assets-automation/asset-sync/contributing.md diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests.csproj b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests.csproj new file mode 100644 index 00000000000..000175aeb79 --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests.csproj @@ -0,0 +1,36 @@ + + + + net6.0 + enable + enable + + false + + + + + + + + + + + + + + + + + + + + + Always + + + Always + + + + diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/CLITests.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/CLITests.cs new file mode 100644 index 00000000000..ad8e35cd420 --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/CLITests.cs @@ -0,0 +1,59 @@ +global using NUnit.Framework; +using Azure.Sdk.Tools.Assets.MaintenanceTool.Model; +using Azure.Sdk.Tools.Assets.MaintenanceTool.Scan; +using Microsoft.AspNetCore.Http; +using Newtonsoft.Json; + +namespace Azure.Sdk.Tools.Assets.MaintenanceTool.Tests +{ + public class CLITests + { + public string TestDirectory { get; protected set; } + + [SetUp] + public void Setup() + { + var workingDirectory = Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString()); + + if (!Directory.Exists(workingDirectory)) + { + Directory.CreateDirectory(workingDirectory); + } + + // copy our static test files there + var source = Path.Combine(Directory.GetCurrentDirectory(), "TestResources"); + var target = Path.Combine(workingDirectory, "TestResources"); + + Microsoft.VisualBasic.FileIO.FileSystem.CopyDirectory(source, target); + + TestDirectory = workingDirectory; + } + + [TearDown] + public void TearDown() + { + Directory.Delete(TestDirectory, true); + } + + [Test] + [TestCase("scan", "-c", "")] + [TestCase("scan", "--config", "")] + public void TestScanOptions(params string[] args) + { + } + + [Test] + [TestCase("scan")] + [TestCase("scan", "--config")] + public void TestInvalidScanOptions(params string[] args) + { + var obj = new object(); + + var rootCommand = Program.InitializeCommandOptions((DefaultOptions) => + { + obj = DefaultOptions; + }); + + } + } +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/GitTokenSkipAttribute.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/GitTokenSkipAttribute.cs new file mode 100644 index 00000000000..7a10ea1d80c --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/GitTokenSkipAttribute.cs @@ -0,0 +1,23 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; +using System.Threading.Tasks; +using NUnit.Framework.Interfaces; +using NUnit.Framework.Internal; + +namespace Azure.Sdk.Tools.Assets.MaintenanceTool.Tests +{ + public class GitTokenSkipAttribute : NUnitAttribute, IApplyToTest + { + public GitTokenSkipAttribute() { } + + public void ApplyToTest(Test test) + { + if (!string.IsNullOrEmpty(Environment.GetEnvironmentVariable("TF_BUILD")) && string.IsNullOrEmpty(Environment.GetEnvironmentVariable("GIT_TOKEN"))) + { + new IgnoreAttribute("Skipping this test. Within a CI run, and GIT_TOKEN is not set.").ApplyToTest(test); + } + } + } +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/ScanTests.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/ScanTests.cs new file mode 100644 index 00000000000..a5d3344ee96 --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/ScanTests.cs @@ -0,0 +1,341 @@ +global using NUnit.Framework; +using System.Text; +using System.Text.Json; +using Azure.Sdk.Tools.Assets.MaintenanceTool.Model; +using Azure.Sdk.Tools.Assets.MaintenanceTool.Scan; + +namespace Azure.Sdk.Tools.Assets.MaintenanceTool.Tests +{ + public class ScanTests + { + // These tests assume the presence of three integration branches for the purposes of testing. + // Azure/azure-sdk-tools + // https://github.com/Azure/azure-sdk-tools/tree/integration/assets-test-branch + // Azure/azure-sdk-assets-integration + // This repo is non-public, and as such a valid GIT_TOKEN must be set or the default git account + // on the invoking machine should have permissions to Azure/azure-sdk-assets-integration. + // + // https://github.com/Azure/azure-sdk-assets-integration/tree/integration/assets-branch-1 + // https://github.com/Azure/azure-sdk-assets-integration/tree/integration/assets-branch-2 + private RunConfiguration RunConfiguration + { + get + { + return new RunConfiguration() + { + LanguageRepos = new List + { + new RepoConfiguration("azure/azure-sdk-tools") + { + Branches = new List() { "integration/assets-test-branch" } + }, + new RepoConfiguration("azure/azure-sdk-assets-integration") + { + Branches = new List() + { + "integration/assets-branch-1", + "integration/assets-branch-2" + } + } + } + }; + } + } + + public string TestDirectory { get; protected set; } + + [SetUp] + public void Setup() + { + var workingDirectory = Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString()); + + if (!Directory.Exists(workingDirectory)) + { + Directory.CreateDirectory(workingDirectory); + } + + // copy our static test files there + var source = Path.Combine(Directory.GetCurrentDirectory(), "TestResources"); + var target = Path.Combine(workingDirectory, "TestResources"); + + Microsoft.VisualBasic.FileIO.FileSystem.CopyDirectory(source, target); + + TestDirectory = workingDirectory; + } + + [TearDown] + public void TearDown() + { + Directory.Delete(TestDirectory, true); + } + + [Test] + [GitTokenSkip] + public void TestBasicScanSingleBranch() + { + var scanner = new AssetsScanner(TestDirectory); + var config = RunConfiguration; + config.LanguageRepos.RemoveAt(0); + config.LanguageRepos.First().Branches.RemoveAt(1); + var results = scanner.Scan(config); + + Assert.IsNotNull(results); + Assert.That(results.Results.Count(), Is.EqualTo(3)); + + var payload0 = results.Results[0]; + var payload1 = results.Results[1]; + var payload2 = results.Results[2]; + + Assert.That(payload0.AssetsLocation, Is.EqualTo("sdk/agrifood/arm-agrifood/assets.json")); + Assert.That(payload0.Commit, Is.EqualTo("48bca526a2a9972e4219ec87d29a7aa31438581a")); + Assert.That(payload0.LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(payload0.Tag, Is.EqualTo("js/agrifood/arm-agrifood_4f244d09c7")); + Assert.That(payload0.AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + + Assert.That(payload1.AssetsLocation, Is.EqualTo("sdk/storage/storage-blob/assets.json")); + Assert.That(payload1.Commit, Is.EqualTo("48bca526a2a9972e4219ec87d29a7aa31438581a")); + Assert.That(payload1.LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(payload1.Tag, Is.EqualTo("js/storage/storage-blob_5d5a32b74a")); + Assert.That(payload1.AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + + Assert.That(payload2.AssetsLocation, Is.EqualTo("sdk/agrifood/arm-agrifood/assets.json")); + Assert.That(payload2.Commit, Is.EqualTo("f139c4ddf7aaa4d637282ae7da4466b473044281")); + Assert.That(payload2.LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(payload2.Tag, Is.EqualTo("js/agrifood/arm-agrifood_4f244d09c7")); + Assert.That(payload2.AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + } + + [Test] + [GitTokenSkip] + public void TestBasicScanMultipleBranches() + { + var scanner = new AssetsScanner(TestDirectory); + var config = RunConfiguration; + config.LanguageRepos.RemoveAt(0); + var results = scanner.Scan(config); + + Assert.IsNotNull(results); + Assert.That(results.Results.Count(), Is.EqualTo(6)); + + Assert.That(results.Results[0].AssetsLocation, Is.EqualTo("sdk/agrifood/arm-agrifood/assets.json")); + Assert.That(results.Results[0].Commit, Is.EqualTo("48bca526a2a9972e4219ec87d29a7aa31438581a")); + Assert.That(results.Results[0].LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(results.Results[0].Tag, Is.EqualTo("js/agrifood/arm-agrifood_4f244d09c7")); + Assert.That(results.Results[0].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + + Assert.That(results.Results[1].AssetsLocation, Is.EqualTo("sdk/storage/storage-blob/assets.json")); + Assert.That(results.Results[1].Commit, Is.EqualTo("48bca526a2a9972e4219ec87d29a7aa31438581a")); + Assert.That(results.Results[1].LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(results.Results[1].Tag, Is.EqualTo("js/storage/storage-blob_5d5a32b74a")); + Assert.That(results.Results[1].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + + Assert.That(results.Results[2].AssetsLocation, Is.EqualTo("sdk/appconfiguration/app-configuration/assets.json")); + Assert.That(results.Results[2].Commit, Is.EqualTo("4b6ee6ea00af2384c0dcc0558e9b96d8051aa8cf")); + Assert.That(results.Results[2].LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(results.Results[2].Tag, Is.EqualTo("js/appconfiguration/app-configuration_61261605e2")); + Assert.That(results.Results[2].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + + Assert.That(results.Results[3].AssetsLocation, Is.EqualTo("sdk/keyvault/keyvault-certificates/assets.json")); + Assert.That(results.Results[3].Commit, Is.EqualTo("4b6ee6ea00af2384c0dcc0558e9b96d8051aa8cf")); + Assert.That(results.Results[3].LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(results.Results[3].Tag, Is.EqualTo("js/keyvault/keyvault-certificates_43821e21b3")); + Assert.That(results.Results[3].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + + Assert.That(results.Results[4].AssetsLocation, Is.EqualTo("sdk/keyvault/keyvault-keys/assets.json")); + Assert.That(results.Results[4].Commit, Is.EqualTo("4b6ee6ea00af2384c0dcc0558e9b96d8051aa8cf")); + Assert.That(results.Results[4].LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(results.Results[4].Tag, Is.EqualTo("js/keyvault/keyvault-keys_b69a5239e9")); + Assert.That(results.Results[4].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + + Assert.That(results.Results[5].AssetsLocation, Is.EqualTo("sdk/agrifood/arm-agrifood/assets.json")); + Assert.That(results.Results[5].Commit, Is.EqualTo("f139c4ddf7aaa4d637282ae7da4466b473044281")); + Assert.That(results.Results[5].LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(results.Results[5].Tag, Is.EqualTo("js/agrifood/arm-agrifood_4f244d09c7")); + Assert.That(results.Results[5].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + } + + [Test] + [GitTokenSkip] + public void TestBasicScanMultipleBranchesMultipleRepos() + { + var scanner = new AssetsScanner(TestDirectory); + var config = RunConfiguration; + var results = scanner.Scan(config); + + Assert.IsNotNull(results); + Assert.That(results.Results.Count(), Is.EqualTo(8)); + + Assert.That(results.ByLanguageRepo.Keys.Count(), Is.EqualTo(2)); + + Assert.That(results.ByLanguageRepo["azure/azure-sdk-tools"][0].AssetsLocation, Is.EqualTo("sdk/formrecognizer/azure-ai-formrecognizer/assets.json")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-tools"][0].Commit, Is.EqualTo("eeeee9e00cc0d0111edf7471962b0da826d9a5cc")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-tools"][0].LanguageRepo, Is.EqualTo("azure/azure-sdk-tools")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-tools"][0].Tag, Is.EqualTo("python/formrecognizer/azure-ai-formrecognizer_f60081bf10")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-tools"][0].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + + Assert.That(results.ByLanguageRepo["azure/azure-sdk-tools"][1].AssetsLocation, Is.EqualTo("sdk/keyvault/assets.json")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-tools"][1].Commit, Is.EqualTo("eeeee9e00cc0d0111edf7471962b0da826d9a5cc")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-tools"][1].LanguageRepo, Is.EqualTo("azure/azure-sdk-tools")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-tools"][1].Tag, Is.EqualTo("python/keyvault/azure-keyvault-administration_f6e776f55f")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-tools"][1].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + + Assert.That(results.ByLanguageRepo["azure/azure-sdk-assets-integration"][5].AssetsLocation, Is.EqualTo("sdk/agrifood/arm-agrifood/assets.json")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-assets-integration"][5].Commit, Is.EqualTo("f139c4ddf7aaa4d637282ae7da4466b473044281")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-assets-integration"][5].LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-assets-integration"][5].Tag, Is.EqualTo("js/agrifood/arm-agrifood_4f244d09c7")); + Assert.That(results.ByLanguageRepo["azure/azure-sdk-assets-integration"][5].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + } + + [Test] + [GitTokenSkip] + public void TestScanHonorsPreviousResults() + { + var specificDirectory = Path.Combine(TestDirectory, "TestResources", "basic_output"); + var scanner = new AssetsScanner(specificDirectory); + + var config = RunConfiguration; + var resultSet = scanner.Scan(config); + var scanDate = DateTime.UtcNow.AddSeconds(-10); + Assert.That(resultSet, Is.Not.Null); + + // The only elements that _could_ change (given that git is immutable per commit) + // are the BackupURIs and the ScanDates. If we have properly loaded previous results, + // these should align. + var resultsForCommit48b = resultSet.ByOriginSHA["48bca526a2a9972e4219ec87d29a7aa31438581a"]; + var resultsForf1 = resultSet.ByOriginSHA["f139c4ddf7aaa4d637282ae7da4466b473044281"]; + var resultsFore6 = resultSet.ByOriginSHA["eeeee9e00cc0d0111edf7471962b0da826d9a5cc"]; + + Assert.That(resultsForCommit48b[0].BackupUri, Is.Null); + Assert.That(resultsForCommit48b[0].CreationDate, Is.EqualTo(DateTime.Parse("2023-05-11T11:05:59"))); + Assert.That(resultsForCommit48b[1].BackupUri, Is.Null); + Assert.That(resultsForCommit48b[1].CreationDate, Is.EqualTo(DateTime.Parse("2023-05-10T00:00:00"))); + Assert.That(resultsForCommit48b.Count(), Is.EqualTo(2)); + + Assert.That(resultsForf1[0].BackupUri, Is.EqualTo("https://github.com/azure-sdk/azure-sdk-assets/tree/backup_js/agrifood/arm-agrifood_4f244d09c7")); + Assert.That(resultsForf1[0].CreationDate, Is.EqualTo(DateTime.Parse("2023-05-08T00:00:00"))); + Assert.That(resultsForf1.Count(), Is.EqualTo(1)); + + Assert.That(resultsFore6[0].AssetsLocation, Is.EqualTo("sdk/formrecognizer/azure-ai-formrecognizer/assets.json")); + Assert.That(resultsFore6[0].Commit, Is.EqualTo("eeeee9e00cc0d0111edf7471962b0da826d9a5cc")); + Assert.That(resultsFore6[0].LanguageRepo, Is.EqualTo("azure/azure-sdk-tools")); + Assert.That(resultsFore6[0].Tag, Is.EqualTo("python/formrecognizer/azure-ai-formrecognizer_f60081bf10")); + Assert.That(resultsFore6[0].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + Assert.That(resultsFore6[0].CreationDate, Is.GreaterThanOrEqualTo(scanDate)); + + Assert.That(resultsFore6[1].AssetsLocation, Is.EqualTo("sdk/keyvault/assets.json")); + Assert.That(resultsFore6[1].Commit, Is.EqualTo("eeeee9e00cc0d0111edf7471962b0da826d9a5cc")); + Assert.That(resultsFore6[1].LanguageRepo, Is.EqualTo("azure/azure-sdk-tools")); + Assert.That(resultsFore6[1].Tag, Is.EqualTo("python/keyvault/azure-keyvault-administration_f6e776f55f")); + Assert.That(resultsFore6[1].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + Assert.That(resultsFore6[1].CreationDate, Is.GreaterThanOrEqualTo(scanDate)); + Assert.That(resultsFore6.Count(), Is.EqualTo(2)); + } + + [Test] + [GitTokenSkip] + public void TestParsePreviouslyOutputResults() + { + var specificDirectory = Path.Combine(TestDirectory, "TestResources", "basic_output"); + var scanner = new AssetsScanner(specificDirectory); + + // ensure that we can parse a default set of existing results + var resultSet = scanner.ParseExistingResults(); + + Assert.That(resultSet, Is.Not.Null); + + Assert.That(resultSet.Results[0].AssetsLocation, Is.EqualTo("sdk/agrifood/arm-agrifood/assets.json")); + Assert.That(resultSet.Results[0].Commit, Is.EqualTo("48bca526a2a9972e4219ec87d29a7aa31438581a")); + Assert.That(resultSet.Results[0].LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(resultSet.Results[0].Tag, Is.EqualTo("js/agrifood/arm-agrifood_4f244d09c7")); + Assert.That(resultSet.Results[0].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + Assert.That(resultSet.Results[0].BackupUri, Is.Null); + Assert.That(resultSet.Results[0].CreationDate, Is.EqualTo(DateTime.Parse("2023-05-11T11:05:59"))); + + Assert.That(resultSet.Results[1].AssetsLocation, Is.EqualTo("sdk/storage/storage-blob/assets.json")); + Assert.That(resultSet.Results[1].Commit, Is.EqualTo("48bca526a2a9972e4219ec87d29a7aa31438581a")); + Assert.That(resultSet.Results[1].LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(resultSet.Results[1].Tag, Is.EqualTo("js/storage/storage-blob_5d5a32b74a")); + Assert.That(resultSet.Results[1].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + Assert.That(resultSet.Results[1].BackupUri, Is.Null); + Assert.That(resultSet.Results[1].CreationDate, Is.EqualTo(DateTime.Parse("2023-05-10T00:00:00"))); + + Assert.That(resultSet.Results[2].AssetsLocation, Is.EqualTo("sdk/agrifood/arm-agrifood/assets.json")); + Assert.That(resultSet.Results[2].Commit, Is.EqualTo("f139c4ddf7aaa4d637282ae7da4466b473044281")); + Assert.That(resultSet.Results[2].LanguageRepo, Is.EqualTo("azure/azure-sdk-assets-integration")); + Assert.That(resultSet.Results[2].Tag, Is.EqualTo("js/agrifood/arm-agrifood_4f244d09c7")); + Assert.That(resultSet.Results[2].AssetsRepo, Is.EqualTo("Azure/azure-sdk-assets")); + Assert.That(resultSet.Results[2].BackupUri, Is.EqualTo("https://github.com/azure-sdk/azure-sdk-assets/tree/backup_js/agrifood/arm-agrifood_4f244d09c7")); + Assert.That(resultSet.Results[2].CreationDate, Is.EqualTo(DateTime.Parse("2023-05-08T00:00:00"))); + } + + [Test] + [GitTokenSkip] + public void TestLoadConfiguration() + { + var scanner = new AssetsScanner(TestDirectory); + var runConfigurationPath = Path.Combine(TestDirectory, "TestResources", "configurations", "sample-repo-configuration.json"); + var config = new RunConfiguration(runConfigurationPath); + + var results = scanner.Scan(config); + + Assert.That(results.Results.Count, Is.EqualTo(8)); + } + + [Test] + [GitTokenSkip] + public void TestScanOutputsResults() + { + var scanner = new AssetsScanner(TestDirectory); + var config = RunConfiguration; + config.LanguageRepos.RemoveAt(0); + config.LanguageRepos.First().Branches.RemoveAt(1); + var resultSet = scanner.Scan(config); + scanner.Save(resultSet); + + var fileThatShouldExist = Path.Combine(TestDirectory, "output.json"); + var newOutFile = Path.Combine(TestDirectory, "test_output"); + + Assert.That(File.Exists(fileThatShouldExist), Is.EqualTo(true)); + + var parsedNewResults = scanner.ParseExistingResults(); + + using (var stream = System.IO.File.OpenWrite(newOutFile)) + { + stream.Write(Encoding.UTF8.GetBytes(JsonSerializer.Serialize(RunConfiguration))); + } + + if (parsedNewResults != null) + { + AssertResultsSame(resultSet, parsedNewResults); + } + else + { + Assert.NotNull(parsedNewResults); + } + } + + private bool AssertResultsSame(AssetsResultSet a, AssetsResultSet b) + { + if (a.Results.Count() != b.Results.Count()) + { + return false; + } + + for (int i = 0; i < a.Results.Count(); i++) + { + AssetsResult aResult = a.Results[i]; + AssetsResult bResult = b.Results[i]; + + Assert.That(bResult.CreationDate, Is.EqualTo(aResult.CreationDate)); + Assert.That(bResult.AssetsLocation, Is.EqualTo(aResult.AssetsLocation)); + Assert.That(bResult.Tag, Is.EqualTo(aResult.Tag)); + Assert.That(bResult.AssetsRepo, Is.EqualTo(aResult.AssetsRepo)); + Assert.That(bResult.Commit, Is.EqualTo(aResult.Commit)); + Assert.That(bResult.LanguageRepo, Is.EqualTo(aResult.LanguageRepo)); + Assert.That(bResult.BackupUri, Is.EqualTo(aResult.BackupUri)); + } + + return true; + } + } +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/TestResources/basic_output/output.json b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/TestResources/basic_output/output.json new file mode 100644 index 00000000000..78c699fc8c1 --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/TestResources/basic_output/output.json @@ -0,0 +1,29 @@ +[ + { + "LanguageRepo": "azure/azure-sdk-assets-integration", + "Commit": "48bca526a2a9972e4219ec87d29a7aa31438581a", + "AssetsLocation": "sdk/agrifood/arm-agrifood/assets.json", + "Tag": "js/agrifood/arm-agrifood_4f244d09c7", + "AssetsRepo": "Azure/azure-sdk-assets", + "BackupUri": null, + "CreationDate": "2023-05-11T11:05:59" + }, + { + "LanguageRepo": "azure/azure-sdk-assets-integration", + "Commit": "48bca526a2a9972e4219ec87d29a7aa31438581a", + "AssetsLocation": "sdk/storage/storage-blob/assets.json", + "Tag": "js/storage/storage-blob_5d5a32b74a", + "AssetsRepo": "Azure/azure-sdk-assets", + "BackupUri": null, + "CreationDate": "2023-05-10T00:00:00" + }, + { + "LanguageRepo": "azure/azure-sdk-assets-integration", + "Commit": "f139c4ddf7aaa4d637282ae7da4466b473044281", + "AssetsLocation": "sdk/agrifood/arm-agrifood/assets.json", + "Tag": "js/agrifood/arm-agrifood_4f244d09c7", + "AssetsRepo": "Azure/azure-sdk-assets", + "BackupUri": "https://github.com/azure-sdk/azure-sdk-assets/tree/backup_js/agrifood/arm-agrifood_4f244d09c7", + "CreationDate": "2023-05-08T00:00:00" + } +] \ No newline at end of file diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/TestResources/configurations/sample-repo-configuration.json b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/TestResources/configurations/sample-repo-configuration.json new file mode 100644 index 00000000000..693238f9c67 --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool.Tests/TestResources/configurations/sample-repo-configuration.json @@ -0,0 +1,17 @@ +{ + "LanguageRepos": [ + { + "LanguageRepo": "azure/azure-sdk-tools", + "Branches": [ + "integration/assets-test-branch" + ] + }, + { + "LanguageRepo": "azure/azure-sdk-assets-integration", + "Branches": [ + "integration/assets-branch-1", + "integration/assets-branch-2" + ] + } + ] +} \ No newline at end of file diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Azure.Sdk.Tools.Assets.MaintenanceTool.csproj b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Azure.Sdk.Tools.Assets.MaintenanceTool.csproj new file mode 100644 index 00000000000..4e636649756 --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Azure.Sdk.Tools.Assets.MaintenanceTool.csproj @@ -0,0 +1,19 @@ + + + + Exe + net6.0 + enable + enable + nullable + + + + + + + + + + + diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Cleanup/AssetsCleanupClient.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Cleanup/AssetsCleanupClient.cs new file mode 100644 index 00000000000..3679b300026 --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Cleanup/AssetsCleanupClient.cs @@ -0,0 +1,13 @@ +using Azure.Sdk.Tools.Assets.MaintenanceTool.Model; + +namespace Azure.Sdk.Tools.Assets.MaintenanceTool.Cleanup; + +public class AssetsCleanupClient +{ + public AssetsCleanupClient() { } + + public AssetsResultSet Cleanup(RunConfiguration config, AssetsResultSet backupResult) + { + return new AssetsResultSet(new List()); ; + } +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/AssetsResult.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/AssetsResult.cs new file mode 100644 index 00000000000..c7f60a4f7e3 --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/AssetsResult.cs @@ -0,0 +1,69 @@ +namespace Azure.Sdk.Tools.Assets.MaintenanceTool.Model +{ + /// + /// This class represents a single reference TO the assets repo FROM a language repo. It is populated by combining + /// an assets's json content with metadata about which language repo it was discovered in. + /// + /// The "language" repo is the repo from which recordings are being generated. The "assets" repo is the storage medium for a set + /// of recordings that are generated by invoking tests in record mode FROM the language repo. + /// + public class AssetsResult + { + public AssetsResult(string repo, string repoCommit, string assetsLocation, string tag, string tagRepo, string? backupUri = null, DateTime? scanDate = null) + { + LanguageRepo = repo; + Commit = repoCommit; + AssetsLocation = assetsLocation; + Tag = tag; + AssetsRepo = tagRepo; + BackupUri = backupUri; + CreationDate = scanDate??DateTime.UtcNow; + } + + public AssetsResult() + { + LanguageRepo = string.Empty; + Commit = string.Empty; + AssetsLocation = string.Empty; + Tag = string.Empty; + AssetsRepo = string.Empty; + BackupUri = string.Empty; + CreationDate = DateTime.UtcNow; + } + + /// + /// The containing language repo from within which this result was generated. + /// + public string LanguageRepo { get; set; } + + /// + /// The SHA of the language repo from which this result was generated. + /// + public string Commit { get; set; } + + /// + /// The location of the assets.json within the language repo from which this result was generated. + /// + public string AssetsLocation { get; set; } + + /// + /// What tag in the assets repo is this reference pointed at? + /// + public string Tag { get; set; } + + /// + /// Which git repo is being used to store the assets? In other words, the targeted assets repo. + /// + public string AssetsRepo { get; set; } + + /// + /// The URI of where we backed this item up. Null when we haven't yet backed up this instance. + /// + public string? BackupUri { get; set; } + + /// + /// Used as a datapoint for debugging and testing of the scanner. Creation datetime of this result. + /// + public DateTime CreationDate { get; set; } + } +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/AssetsResultSet.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/AssetsResultSet.cs new file mode 100644 index 00000000000..a0760376177 --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/AssetsResultSet.cs @@ -0,0 +1,52 @@ +namespace Azure.Sdk.Tools.Assets.MaintenanceTool.Model; + +/// +/// This class abstracts some common query patterns of a set of scanned results. +/// +public class AssetsResultSet +{ + + public AssetsResultSet(List input) + { + Results = input; + CalculateObjects(); + } + public List Results { get; set; } = new List(); + + public Dictionary> ByLanguageRepo { get; private set; } = new(); + + public Dictionary> ByTargetTag { get; private set; } = new(); + + public Dictionary> ByOriginSHA { get; private set; } = new(); + + private void CalculateObjects() + { + ByLanguageRepo = new Dictionary>(); + ByTargetTag = new Dictionary>(); + ByOriginSHA = new Dictionary>(); + + // sort to ensure that orderings are always the same + Results = Results.OrderBy(asset => asset.AssetsRepo).ThenBy(asset => asset.Commit).ThenBy(asset => asset.Tag).ToList(); + + foreach (var result in Results) + { + if (!ByLanguageRepo.ContainsKey(result.LanguageRepo)) + { + ByLanguageRepo.Add(result.LanguageRepo, new List()); + } + ByLanguageRepo[result.LanguageRepo].Add(result); + + if (!ByTargetTag.ContainsKey(result.Tag)) + { + ByTargetTag.Add(result.Tag, new List()); + } + ByTargetTag[result.Tag].Add(result); + + if (!ByOriginSHA.ContainsKey(result.Commit)) + { + ByOriginSHA.Add(result.Commit, new List()); + } + ByOriginSHA[result.Commit].Add(result); + } + } +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/RepoConfiguration.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/RepoConfiguration.cs new file mode 100644 index 00000000000..04d062e7005 --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/RepoConfiguration.cs @@ -0,0 +1,34 @@ +namespace Azure.Sdk.Tools.Assets.MaintenanceTool.Model; + +/// +/// Configuration class describing options available while targeting a repository for scanning. +/// +public class RepoConfiguration +{ + public RepoConfiguration(string repo) + { + LanguageRepo = repo; + } + + public RepoConfiguration() { + LanguageRepo = string.Empty; + } + + /// + /// The full orgname/repo-id identifier to access a repo on github. EG: "azure/azure-sdk-for-net" + /// + public string LanguageRepo { get; set; } + + /// + /// The time from which we will search for commits that contain assets.jsons. The current default was chosen + /// almost arbitrarily. Official test-proxy began supported external assets in late November of 2022, so we don't + /// need to go further back then that when examining the SHAs in the language repos. There is no possibility of an + /// assets.json past this date! + /// + public DateTime ScanStartDate { get; set; } = DateTime.Parse("2022-12-01"); + + /// + /// The set of branches that we will examine. Defaults to just 'main'. + /// + public List Branches { get; set; } = new List { "main" }; +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/RunConfiguration.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/RunConfiguration.cs new file mode 100644 index 00000000000..9e978149b9b --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Model/RunConfiguration.cs @@ -0,0 +1,40 @@ +using System.Text.Json; + +namespace Azure.Sdk.Tools.Assets.MaintenanceTool.Model; + +/// +/// A RunConfiguration is the basic unit of configuration for this maintenance tool. It contains a set of targeted language +/// repositories that should be scanned for for assets.jsons. +/// +public class RunConfiguration +{ + + + public RunConfiguration() { + LanguageRepos = new List(); + } + + public RunConfiguration(string configPath) + { + if (File.Exists(configPath)) + { + LanguageRepos = new List(); + + using var stream = System.IO.File.OpenRead(configPath); + using var doc = JsonDocument.Parse(stream); + + var results = JsonSerializer.Deserialize(doc); + + if (results != null) + { + LanguageRepos = results.LanguageRepos; + } + } + else + { + throw new ArgumentException($"The configuration file path \"{configPath}\" does not exist."); + } + } + + public List LanguageRepos { get; set; } +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Options/BaseOptions.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Options/BaseOptions.cs new file mode 100644 index 00000000000..82012ff2ff1 --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Options/BaseOptions.cs @@ -0,0 +1,39 @@ +using System.CommandLine; +using System.CommandLine.Binding; + +namespace Azure.Sdk.Tools.Assets.MaintenanceTool.Options; + +public class BaseOptions +{ + public string ConfigLocation { get; set; } = string.Empty; +} + +public class BaseOptionsBinder : BinderBase +{ + private readonly Option _configLocationOption; + + public BaseOptionsBinder(Option configLocationOption) + { + _configLocationOption = configLocationOption; + } + + protected override BaseOptions GetBoundValue(BindingContext bindingContext) + { + var result = bindingContext.ParseResult.GetValueForOption(_configLocationOption); + + if (result != null) + { + return new BaseOptions + { + ConfigLocation = result + }; + } + else + { + return new BaseOptions + { + ConfigLocation = string.Empty + }; + } + } +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Program.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Program.cs new file mode 100644 index 00000000000..6e71d3af60b --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Program.cs @@ -0,0 +1,86 @@ +using System.CommandLine; +using Azure.Sdk.Tools.Assets.MaintenanceTool.Model; +using Azure.Sdk.Tools.Assets.MaintenanceTool.Options; +using Azure.Sdk.Tools.Assets.MaintenanceTool.Scan; + +namespace Azure.Sdk.Tools.Assets.MaintenanceTool; + +public class Program +{ + public static string[] StoredArgs = new string[] { }; + + // By default, we will only search under the CWD for the previous output. This may change + // in the future, but we are keeping it simple to start. + + // all should honor + // --scan-data <-- results from previous scans (todo, currently checks working directory) + + // SCAN + // --configuration: -> path to file + + // BACKUP + // --configuration provided? + // SCAN + // BACKUP + // as each tag is backed up, it is saved with suffix _backup + + // RESTORE + // --input-tag + + // CLEANUP + // --configuration provided? + // SCAN + // BACKUP + // CLEANUP + // each tag as found by configuration + + // --input-tag ? + // SCAN, BACKUP, and CLEANUP individual tag + + public static void Main(string[] args) + { + StoredArgs = args; + + var rootCommand = InitializeCommandOptions(Run); + var resultCode = rootCommand.Invoke(args); + Environment.Exit(resultCode); + } + + public static void Run(object commandObj) + { + switch (commandObj) + { + case BaseOptions configOptions: + AssetsScanner scanner = new AssetsScanner(); + var runConfig = new RunConfiguration(configOptions.ConfigLocation); + AssetsResultSet results = scanner.Scan(runConfig); + scanner.Save(results); + + break; + default: + throw new ArgumentException($"Unable to parse the argument set: {string.Join(" ", StoredArgs)}"); + } + } + + public static RootCommand InitializeCommandOptions(Action action) + { + var root = new RootCommand(); + var configOption = new Option( + name: "--config", + description: "The path to the json file containing the repo configuration. A sample repo configuration can be seen under /integration-test-repo-configuration.yml." + ) { + IsRequired = true + }; + configOption.AddAlias("-c"); + + var scanCommand = new Command("scan", "Scan the repositories as configured within the file provided to input argument --config ."); + scanCommand.AddOption(configOption); + scanCommand.SetHandler( + (configOpts) => action(configOpts), + new BaseOptionsBinder(configOption) + ); + root.Add(scanCommand); + + return root; + } +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Scan/AssetsScanner.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Scan/AssetsScanner.cs new file mode 100644 index 00000000000..99656ab335a --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Scan/AssetsScanner.cs @@ -0,0 +1,325 @@ +using System.Text; +using System.Text.Json; +using Azure.Sdk.Tools.Assets.MaintenanceTool.Model; +using Azure.Sdk.Tools.TestProxy.Common.Exceptions; +using Azure.Sdk.Tools.TestProxy.Store; +using Microsoft.Extensions.FileSystemGlobbing; + +namespace Azure.Sdk.Tools.Assets.MaintenanceTool.Scan; + +/// +/// Used to walk through repo configurations and locate all assets. +/// +public class AssetsScanner +{ + public string WorkingDirectory { get; set; } + public static readonly string GitTokenEnvVar = "GIT_TOKEN"; + + private string ResultsFile + => Path.Combine(WorkingDirectory, "output.json"); + + public GitProcessHandler handler { get; set; } = new GitProcessHandler(); + + public AssetsScanner(string? workingDirectory = null) + { + WorkingDirectory = workingDirectory ?? Directory.GetCurrentDirectory(); + } + + /// + /// Walk a run configuration and create a resultSet of all found assets.json references. + /// + /// This function automatically takes previous output into account by checking in the current + /// working directory for an "output.json" file that contains the output of a previously run Scan. + /// + /// + /// A set of results which combines any previous output with a new scan. + public AssetsResultSet Scan(RunConfiguration config) + { + var resultSet = new List(); + AssetsResultSet? existingResults = ParseExistingResults(); + + Parallel.ForEach(config.LanguageRepos, repoConfig => + { + resultSet.AddRange(ScanRepo(repoConfig, existingResults)); + }); + + return new AssetsResultSet(resultSet); + } + + /// + /// If the tool is invoked in a directory containing an "output.json" file, that file will be parsed + /// for its results. The file itself is merely a List of type AssetsResult serialized to disk. + /// + /// + public AssetsResultSet? ParseExistingResults() + { + if (File.Exists(ResultsFile)) + { + using var stream = System.IO.File.OpenRead(ResultsFile); + using var doc = JsonDocument.Parse(stream); + + var results = JsonSerializer.Deserialize>(doc); + + if (results != null) + { + return new AssetsResultSet(results); + } + } + + return null; + } + + /// + /// Given a repo configuration, scan the repo and return an AssetsResult list from all targeted branches. + /// + /// + /// + /// + private List ScanRepo(RepoConfiguration config, AssetsResultSet? previousOutput) + { + string? envOverride = Environment.GetEnvironmentVariable(GitTokenEnvVar); + var authString = string.Empty; + if (!string.IsNullOrWhiteSpace(envOverride)) + { + authString = $"{envOverride}@"; + } + + var targetRepoUri = $"https://{authString}github.com/{config.LanguageRepo}.git"; + var workingDirectory = Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString()); + var results = new List(); + + try + { + if (!Directory.Exists(workingDirectory)) + { + Directory.CreateDirectory(workingDirectory); + } + + foreach (var branch in config.Branches) + { + var commitsOnBranch = GetBranchCommits(targetRepoUri, branch, config.ScanStartDate, workingDirectory); + var unretrievedCommits = ResolveUnhandledCommits(commitsOnBranch, previousOutput); + + results.AddRange(GetAssetsResults(config.LanguageRepo, unretrievedCommits, workingDirectory)); + + if (previousOutput != null) + { + foreach (var commit in commitsOnBranch.Where(commit => !unretrievedCommits.Contains(commit))) + { + results.AddRange(previousOutput.ByOriginSHA[commit]); + } + } + } + } + finally + { + CleanupWorkingDirectory(workingDirectory); + } + + return results; + } + + /// + /// Clones a specific branch, then returns all commit shas newer than our targeted date. + /// + /// A list of commits (limited to after a startdate) from the targeted branch. + private List GetBranchCommits(string uri, string branch, DateTime since, string workingDirectory) + { + var commitSHAs = new List(); + try + { + // if git is already initialized, we just need to checkout a specific branch + if (!Directory.Exists(Path.Combine(workingDirectory, ".git"))) + { + handler.Run($"clone {uri} --branch {branch} --single-branch .", workingDirectory); + } + else + { + handler.Run($"fetch origin {branch}", workingDirectory); + handler.Run($"branch {branch} FETCH_HEAD", workingDirectory); + handler.Run($"checkout {branch}", workingDirectory); + Cleanup(workingDirectory); + } + + var tagResult = handler.Run($"log --since={since.ToString("yyyy-MM-dd")} --format=format:%H", workingDirectory); + commitSHAs.AddRange(tagResult.StdOut.Split(Environment.NewLine).Select(x => x.Trim()).Where(x => !string.IsNullOrWhiteSpace(x))); + } + catch (GitProcessException gitException) + { + // special case handling here? + Console.WriteLine(gitException.ToString()); + Environment.Exit(1); + } + catch (Exception e) + { + Console.WriteLine(e.ToString()); + Environment.Exit(1); + } + + return commitSHAs; + } + + /// + /// We only need to process each commit _once_, as commit SHAs are immutable in git. Given that, once we have + /// a list of commits from a targeted branch, we need to check against the previous results to ensure we don't + /// reprocess those and emit duplicate assetsResults. + /// + /// The set of unprocessed commit SHAs. + private List ResolveUnhandledCommits(List commits, AssetsResultSet? previousResults) + { + if (previousResults == null) + { + return commits; + } + else + { + return commits.Where(x => !previousResults.ByOriginSHA.ContainsKey(x)).ToList(); + } + } + + /// + /// Used to easily parse an assets.json and grab only the properties that this tool cares about. + /// + private class Assets + { + public Assets() + { + AssetsRepo = string.Empty; + Tag = string.Empty; + } + + public string AssetsRepo { get; set; } + + public string Tag { get; set; } + } + + /// + /// Deserialize an assets.json from disk into a class instance to retrieve the targeted Tag and Assets Repository. + /// + /// + /// A class instance containing the assets.json details. + private Assets? ExtractAssetsData(string assetsJson) + { + return JsonSerializer.Deserialize(File.ReadAllText(assetsJson)); + } + + /// + /// Find all assets.jsons beneath a targeted folder. + /// + /// AssetsResults for each discovered assets.json, populating other metadata as necessary. + private List ScanDirectory(string repo, string commit, string workingDirectory) + { + Matcher matcher = new(); + List locatedAssets = new List(); + matcher.AddIncludePatterns(new[] { "**/assets.json" }); + IEnumerable assetsJsons = matcher.GetResultsInFullPath(workingDirectory); + + foreach (var assetsJson in assetsJsons) + { + var path = Path.GetRelativePath(workingDirectory, assetsJson).Replace("\\", "/"); + var assetsData = ExtractAssetsData(assetsJson); + + if (assetsData != null) + { + var newResult = new AssetsResult(repo, commit, path, assetsData.Tag, assetsData.AssetsRepo, null); + locatedAssets.Add(newResult); + } + } + + return locatedAssets; + } + + /// + /// Walks a set of targeted commits, extracting all available assets.jsons from each. + /// + /// A list of AssetsResults reflecting all discovered assets.jsons from each targeted commit. + private List GetAssetsResults(string repo, List commits, string workingDirectory) + { + var allResults = new List(); + foreach (var commit in commits) + { + handler.Run($"checkout {commit}", workingDirectory); + Cleanup(workingDirectory); + allResults.AddRange(ScanDirectory(repo, commit, workingDirectory)); + } + + return allResults; + } + + /// + /// Cleans up a git repo. When swapping between commits, we don't want to accidentally include assets.jsons that are + /// present simply because a folder didn't auto delete itself when we switched commits. + /// + private void Cleanup(string workingDirectory) + { + try + { + handler.Run("clean -xdf", workingDirectory); + } + catch (GitProcessException gitException) + { + Console.WriteLine(gitException.ToString()); + Environment.Exit(1); + } + catch (Exception e) + { + Console.WriteLine(e.ToString()); + Environment.Exit(1); + } + } + + /// + /// Intended to be aimed at a specific .git folder. Walks every file and ensures that + /// any wonky permissions that could prevent deletion are removed. + /// + /// This is necessary because certain `.pack` files created by git cannot be deleted without + /// adjusting these permissions. + /// + private void SetPermissionsAndDelete(string gitfolder) + { + File.SetAttributes(gitfolder, FileAttributes.Normal); + + string[] files = Directory.GetFiles(gitfolder); + string[] dirs = Directory.GetDirectories(gitfolder); + + foreach (string file in files) + { + File.SetAttributes(file, FileAttributes.Normal); + File.Delete(file); + } + + foreach (string dir in dirs) + { + SetPermissionsAndDelete(dir); + } + + Directory.Delete(gitfolder, false); + } + + /// + /// The .git folder's .pack files can be super finicky to delete from code. + /// This function abstracts the necessary permissions update and cleans that folder for us. + /// + private void CleanupWorkingDirectory(string workingDirectory) + { + var gitDir = Path.Combine(workingDirectory, ".git"); + + if (Directory.Exists(gitDir)) + { + SetPermissionsAndDelete(gitDir); + } + + Directory.Delete(workingDirectory, true); + } + + /// + /// Writes a resultSet to disk. + /// + public void Save(AssetsResultSet newResults) + { + using (var stream = System.IO.File.OpenWrite(ResultsFile)) + { + stream.Write(Encoding.UTF8.GetBytes(JsonSerializer.Serialize(newResults.Results))); + } + } +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Store/AssetsStorageClient.cs b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Store/AssetsStorageClient.cs new file mode 100644 index 00000000000..b2d0344cc6f --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenanceTool/Store/AssetsStorageClient.cs @@ -0,0 +1,9 @@ +namespace Azure.Sdk.Tools.Assets.MaintenanceTool.Store; + +/// +/// Used to write our backup entries. +/// +public class AssetsStorageClient +{ + // placeholder for now. +} diff --git a/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenceTool.sln b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenceTool.sln new file mode 100644 index 00000000000..4ebb4d7340c --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/Azure.Sdk.Tools.Assets.MaintenceTool.sln @@ -0,0 +1,37 @@ + +Microsoft Visual Studio Solution File, Format Version 12.00 +# Visual Studio Version 17 +VisualStudioVersion = 17.5.33530.505 +MinimumVisualStudioVersion = 10.0.40219.1 +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.Sdk.Tools.Assets.MaintenanceTool", "Azure.Sdk.Tools.Assets.MaintenanceTool\Azure.Sdk.Tools.Assets.MaintenanceTool.csproj", "{8ECBAB4E-1209-454D-B0AA-F93A61C7C1B9}" +EndProject +Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Azure.Sdk.Tools.TestProxy", "..\..\test-proxy\Azure.Sdk.Tools.TestProxy\Azure.Sdk.Tools.TestProxy.csproj", "{531B1A4C-B213-4486-B965-330585816BAA}" +EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Azure.Sdk.Tools.Assets.MaintenanceTool.Tests", "Azure.Sdk.Tools.Assets.MaintenanceTool.Tests\Azure.Sdk.Tools.Assets.MaintenanceTool.Tests.csproj", "{968EC25C-861D-4C10-967D-A3E0137D35C2}" +EndProject +Global + GlobalSection(SolutionConfigurationPlatforms) = preSolution + Debug|Any CPU = Debug|Any CPU + Release|Any CPU = Release|Any CPU + EndGlobalSection + GlobalSection(ProjectConfigurationPlatforms) = postSolution + {8ECBAB4E-1209-454D-B0AA-F93A61C7C1B9}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {8ECBAB4E-1209-454D-B0AA-F93A61C7C1B9}.Debug|Any CPU.Build.0 = Debug|Any CPU + {8ECBAB4E-1209-454D-B0AA-F93A61C7C1B9}.Release|Any CPU.ActiveCfg = Release|Any CPU + {8ECBAB4E-1209-454D-B0AA-F93A61C7C1B9}.Release|Any CPU.Build.0 = Release|Any CPU + {531B1A4C-B213-4486-B965-330585816BAA}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {531B1A4C-B213-4486-B965-330585816BAA}.Debug|Any CPU.Build.0 = Debug|Any CPU + {531B1A4C-B213-4486-B965-330585816BAA}.Release|Any CPU.ActiveCfg = Release|Any CPU + {531B1A4C-B213-4486-B965-330585816BAA}.Release|Any CPU.Build.0 = Release|Any CPU + {968EC25C-861D-4C10-967D-A3E0137D35C2}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {968EC25C-861D-4C10-967D-A3E0137D35C2}.Debug|Any CPU.Build.0 = Debug|Any CPU + {968EC25C-861D-4C10-967D-A3E0137D35C2}.Release|Any CPU.ActiveCfg = Release|Any CPU + {968EC25C-861D-4C10-967D-A3E0137D35C2}.Release|Any CPU.Build.0 = Release|Any CPU + EndGlobalSection + GlobalSection(SolutionProperties) = preSolution + HideSolutionNode = FALSE + EndGlobalSection + GlobalSection(ExtensibilityGlobals) = postSolution + SolutionGuid = {A9C5822D-B16B-4043-A8FD-E7F8B5536855} + EndGlobalSection +EndGlobal diff --git a/tools/assets-automation/assets-maintenance-tool/README.md b/tools/assets-automation/assets-maintenance-tool/README.md new file mode 100644 index 00000000000..310db7c697d --- /dev/null +++ b/tools/assets-automation/assets-maintenance-tool/README.md @@ -0,0 +1,22 @@ +# Assets Maintenance Tooling + +The tool contained within this directory is intended two fulfill three main tasks. + +1. Scan the commits of a set of targeted repos and branches to create a representation of all assets.jsons, their targeted tags, and their origin SHAs. + * This is just a map of the complete footprint of `test-proxy` assets that are referenced in azure-sdk repositories. +2. Use the map of data created in step 1 to individually backup tags from the assets repository to a target that we can retrieve later. +3. Use the map of data created in step 1 to clean up _unnecessary_ tags in the assets repository. + +## Usage + +### Installation + +`` + +### Scanning + +`` + +## What does the process look like? + +![example processing layout](processing_layout.png) \ No newline at end of file diff --git a/tools/assets-automation/assets-maintenance-tool/integration-test-repo-configuration.yml b/tools/assets-automation/assets-maintenance-tool/integration-test-repo-configuration.yml new file mode 100644 index 00000000000..e69de29bb2d diff --git a/tools/assets-automation/assets-maintenance-tool/processing_layout.png b/tools/assets-automation/assets-maintenance-tool/processing_layout.png new file mode 100644 index 00000000000..449e9e01f2f Binary files /dev/null and b/tools/assets-automation/assets-maintenance-tool/processing_layout.png differ diff --git a/eng/scripts/python/assets-automation/README.md b/tools/assets-automation/assets-reporting/README.md similarity index 100% rename from eng/scripts/python/assets-automation/README.md rename to tools/assets-automation/assets-reporting/README.md diff --git a/eng/scripts/python/assets-automation/generate_assets_report.py b/tools/assets-automation/assets-reporting/generate_assets_report.py similarity index 100% rename from eng/scripts/python/assets-automation/generate_assets_report.py rename to tools/assets-automation/assets-reporting/generate_assets_report.py diff --git a/eng/scripts/python/assets-automation/requirements.txt b/tools/assets-automation/assets-reporting/requirements.txt similarity index 100% rename from eng/scripts/python/assets-automation/requirements.txt rename to tools/assets-automation/assets-reporting/requirements.txt diff --git a/tools/asset-sync/ci.yml b/tools/assets-automation/ci.yml similarity index 62% rename from tools/asset-sync/ci.yml rename to tools/assets-automation/ci.yml index 18d17781de9..67836b2d062 100644 --- a/tools/asset-sync/ci.yml +++ b/tools/assets-automation/ci.yml @@ -8,7 +8,7 @@ trigger: - hotfix/* paths: include: - - tools/asset-sync + - tools/assets-automation pr: branches: @@ -19,10 +19,9 @@ pr: - hotfix/* paths: include: - - tools/asset-sync + - tools/assets-automation extends: - template: /eng/common/pipelines/templates/stages/archetype-sdk-tool-pwsh.yml + template: /eng/pipelines/templates/stages/archetype-sdk-tool-dotnet.yml parameters: - TargetDirectory: tools/asset-sync/ - TargetTags: 'Unit' + ToolDirectory: tools/assets-automation/assets-maintenance-tool/ diff --git a/tools/assets-automation/tests.yml b/tools/assets-automation/tests.yml new file mode 100644 index 00000000000..346d874507b --- /dev/null +++ b/tools/assets-automation/tests.yml @@ -0,0 +1,44 @@ +trigger: none + +variables: + - template: /eng/pipelines/templates/variables/globals.yml + +stages: + - stage: IntegrationTests + displayName: "Integration Tests" + jobs: + - job: Solution_Integration_Test + displayName: Run Solution Tests + + strategy: + matrix: + Windows: + Pool: 'azsdk-pool-mms-win-2022-general' + OS: 'Windows' + Linux: + Pool: azsdk-pool-mms-ubuntu-2204-general + OS: 'Linux' + + pool: + name: $(Pool) + + steps: + - template: /eng/pipelines/templates/steps/install-dotnet.yml + + - script: 'dotnet test /p:ArtifactsPackagesDir=$(Build.ArtifactStagingDirectory) --logger trx $(Build.SourcesDirectory)/tools/assets-automation/assets-maintenance-tool/' + displayName: 'Test' + env: + DOTNET_SKIP_FIRST_TIME_EXPERIENCE: 1 + DOTNET_CLI_TELEMETRY_OPTOUT: 1 + DOTNET_MULTILEVEL_LOOKUP: 0 + GIT_TOKEN: $(azuresdk-github-pat) + GIT_COMMIT_OWNER: azure-sdk + GIT_COMMIT_EMAIL: azuresdk@microsoft.com + + - task: PublishTestResults@2 + condition: succeededOrFailed() + inputs: + testResultsFiles: '**/*.trx' + testRunTitle: '$(OS) Maintenance tool tests against .NET' + testResultsFormat: 'VSTest' + mergeTestResults: true diff --git a/tools/test-proxy/documentation/asset-sync/README.md b/tools/test-proxy/documentation/asset-sync/README.md index 31aae84e7a8..a92c52f5615 100644 --- a/tools/test-proxy/documentation/asset-sync/README.md +++ b/tools/test-proxy/documentation/asset-sync/README.md @@ -2,6 +2,8 @@ The `test-proxy` optionally offers integration with other git repositories for **storing** and **retrieving** recordings. This enables the proxy to work against repositories that do not emplace their test recordings directly alongside their test implementations. +Colloquially, any file that is stored externally using the `asset-sync` feature of the `test-proxy` is called an `asset`. + ![asset sync block diagram](../_images/asset_sync_block_diagram.png) In the context of a `monorepo`, this means that we store FAR less data per feature. To update recordings, the only change alongside the source code is to update the targeted tag.