Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Content replacement automation to all repos #4928

Merged
merged 25 commits into from
Jan 12, 2023
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
0b9fbaa
content replacement automation to all repos
sima-zhu Jan 10, 2023
213a1bb
Update agent-pool-migration.yml
sima-zhu Jan 10, 2023
c4828b5
Update Repo-File-Content-Replacements.ps1
sima-zhu Jan 10, 2023
18eea88
print
sima-zhu Jan 10, 2023
5160aa2
remove not used
sima-zhu Jan 10, 2023
2365b04
change assigness
sima-zhu Jan 10, 2023
c730cb7
change the default path
sima-zhu Jan 10, 2023
745945b
remove extra lines
sima-zhu Jan 10, 2023
7f2b87f
change to single quotes
sima-zhu Jan 10, 2023
f59538f
Update Repo-File-Content-Replacements.ps1
sima-zhu Jan 10, 2023
3d00e58
Update eng/pipelines/agent-pool-migration.yml
sima-zhu Jan 11, 2023
450d4e3
Update agent-pool-migration.yml
sima-zhu Jan 11, 2023
6da0b03
Address comments
sima-zhu Jan 12, 2023
cf94448
add default path
sima-zhu Jan 12, 2023
4b8f246
add test cases comment
sima-zhu Jan 12, 2023
df9a497
add more comments
sima-zhu Jan 12, 2023
50805eb
add more comments
sima-zhu Jan 12, 2023
e5f785d
add more comments
sima-zhu Jan 12, 2023
37ec2e4
Update eng/scripts/Repo-File-Content-Replacements.ps1
sima-zhu Jan 12, 2023
48c56b4
move the tests under scripts folder
sima-zhu Jan 12, 2023
030f95a
Merge branch 'main' of https://github.com/Azure/azure-sdk-tools into …
sima-zhu Jan 12, 2023
62b7306
move scripts test to scripts folder
sima-zhu Jan 12, 2023
f5e3734
Merge branch 'migrate_pool' of https://github.com/sima-zhu/azure-sdk-…
sima-zhu Jan 12, 2023
e50ab99
Fix the paht
sima-zhu Jan 12, 2023
f73cb44
add force remove
sima-zhu Jan 12, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
148 changes: 148 additions & 0 deletions eng/pipelines/agent-pool-migration.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,148 @@
# The pipeline is mainly used to migrate our agent pool from one version to another version.
sima-zhu marked this conversation as resolved.
Show resolved Hide resolved
sima-zhu marked this conversation as resolved.
Show resolved Hide resolved
# Here is how we use the pipeline:
# 1. What repos are you going to work on for your migration? Put your repos in Parameter 'Repos'. Follow the format in default.
# 2. The replacement map, e.g We will search all occurrences of the strings in Parameter 'MigrateFrom' and replace to the value in 'MigrateTo'. This is regex map, and you can define multiple replacement rules here.
# 3. The Parameter 'WhatIfPreference' boolean flag allows you to check your changed files before open a PR. Check the box if you don't want to create a PR.
# 4. The pipeline will generate PRs in specified repos, Parameter 'ShortForMigrateFrom' and 'ShortForMigrateTo' are the short descriptions of the PR purpose.
# The PR title will in the format of "Migrate from ${{parameters.ShortForMigrateFrom}} to ${{parameters.ShortForMigrateTo}}"
# You can also specify who triggers the changes in Paramter 'GithubAssigneeAlias'. Then your PR will request review from the specified person.
# If you want to create PR from your own fork, you can specify the Parameters 'PRForkOwner'
# 5. Go to the PR link in 'Create pull request' step. Review and merge the PR. Note, the pipeline intented to be aggressive, so take a closer look at the changes.
# The PR also has a link pointed back to the pipeline, so people will learn where it comes from and the purpose of the PR.
trigger: none
pr: none

# Notes on ExcludePaths and IncludePaths:
# 1. ExcludePaths is used to exclude the file pattern you don't want to scan against.
# 2. IncludePaths is used to add back the files you don't want to exclude. It is supposed to be a subset of ExcludePaths.
# 3. Both excludePaths and includePaths support standard powershell regex.
# 4. The paths here are relative to the repo folder.
# E.g.
# In win agent pool, the path is like ".\eng\scripts\Repo-File-Content-Replacements.ps1", so the regex path can be like "^.\\eng\\"
# In linux agent pool, the path is like "./eng/scripts/Repo-File-Content-Replacements.ps1", so the regex path can be like "^./eng/"
# To compitable with both cases, you can also specify your regex path like "^./|\\eng/|\\"
sima-zhu marked this conversation as resolved.
Show resolved Hide resolved
parameters:
- name: Repos
type: object
default:
- RepoOwner: Azure
RepoName: azure-sdk-tools
- RepoOwner: Azure
RepoName: azure-sdk
ExcludePaths: '^.\\eng\\common\\*'
sima-zhu marked this conversation as resolved.
Show resolved Hide resolved
- RepoOwner: Azure
RepoName: azure-sdk-for-android
ExcludePaths: '^.\\sdk\\|^.\\eng\\common\\|\\tests\\resources\\session-records\\'
IncludePaths: '^.\\sdk\\.*-matrix.*.json$|^.\\sdk\\.*\\.*yml$'
- RepoOwner: Azure
RepoName: azure-sdk-for-c
ExcludePaths: '^.\\sdk\\|^.\\eng\\common\\'
IncludePaths: '^.\\sdk\\.*-matrix.*.json$|^.\\sdk\\.*\\.*yml$'
- RepoOwner: Azure
RepoName: azure-sdk-for-cpp
ExcludePaths: '^.\\sdk\\|^.\\eng\\common\\'
IncludePaths: '^.\\sdk\\.*-matrix.*.json$|^.\\sdk\\.*\\.*yml$'
- RepoOwner: Azure
RepoName: azure-sdk-for-go
ExcludePaths: '^.\\sdk\\|^.\\eng\\common\\'
IncludePaths: '^.\\sdk\\.*-matrix.*.json$|^.\\sdk\\.*\\.*yml$'
- RepoOwner: Azure
RepoName: azure-sdk-for-ios
ExcludePaths: '^.\\sdk\\|^.\\eng\\common\\'
IncludePaths: '^.\\sdk\\.*-matrix.*.json$|^.\\sdk\\.*\\.*yml$'
- RepoOwner: Azure
RepoName: azure-sdk-for-java
ExcludePaths: '^.\\sdk\\|^.\\eng\\common\\|\\resources\\session-records\\'
IncludePaths: '^.\\sdk\\.*-matrix.*.json$|^.\\sdk\\.*\\.*yml$'
- RepoOwner: Azure
RepoName: azure-sdk-for-js
ExcludePaths: '^.\\sdk\\|^.\\eng\\common\\|\\recordings\\node\\'
IncludePaths: '^.\\sdk\\.*-matrix.*.json$|^.\\sdk\\.*\\.*yml$'
- RepoOwner: Azure
RepoName: azure-sdk-for-net
ExcludePaths: '^.\\sdk\\|^.\\eng\\common\\|\\tests\\SessionRecords\\'
IncludePaths: '^.\\sdk\\.*-matrix.*.json$|^.\\sdk\\.*\\.*yml$'
- RepoOwner: Azure
RepoName: azure-sdk-for-python
ExcludePaths: '^.\\sdk\\|^.\\eng\\common\\|\\tests\\recordings\\'
IncludePaths: '^.\\sdk\\.*-matrix.*.json$|^.\\sdk\\.*\\.*yml$'
# Default setting is an example of the map looks like:
- name: MigrationMap
type: object
default:
- MigrateFrom: azsdk-pool-mms-win-2019-general
MigrateTo: azsdk-pool-mms-win-2022-general
- MigrateFrom: windows-2019
MigrateTo: windows-2022
- MigrateFrom: windows2019
MigrateTo: windows2022
- MigrateFrom: MMS2019
MigrateTo: MMS2022
- name: ShortForMigrateFrom
type: string
default: ''
- name: ShortForMigrateTo
type: string
default: ''
# The value here is github alias: e.g. sima-zhu
# Users to assign to the PR after opening. Users should be a comma-separated list. e.g. "user1,usertwo,user3"
- name: GithubAssignessAlias
type: string
default: ''
- name: PRForkOwner
type: string
default: 'azure-sdk'
- name: WhatIfPreference
type: boolean
default: false

variables:
- template: /eng/pipelines/templates/variables/globals.yml
jobs:
- job: Migration
pool:
name: azsdk-pool-mms-win-2022-general
sima-zhu marked this conversation as resolved.
Show resolved Hide resolved
steps:
- ${{ each repo in parameters.Repos }}:
- template: /eng/common/pipelines/templates/steps/sparse-checkout.yml
parameters:
SkipCheckoutNone: true
Repositories:
- Name: ${{ repo.RepoOwner }}/${{ repo.RepoName }}
WorkingDirectory: $(Build.SourcesDirectory)/repos/${{ repo.RepoName }}
Commitish: ${{ repo.branch }}
Paths:
- /*
- template: /eng/common/pipelines/templates/steps/set-default-branch.yml
parameters:
WorkingDirectory: $(Build.SourcesDirectory)/repos/${{ repo.RepoName }}
- pwsh: |
$migrationMap = '${{ convertToJson(parameters.MigrationMap) }}'
$(Build.SourcesDirectory)/eng/scripts/Repo-File-Content-Replacements.ps1 `
-ExcludePathsRegex '${{ repo.ExcludePaths }}' `
-IncludePathsRegex '${{ repo.IncludePaths }}' `
-MigrationMap $migrationMap
displayName: Migration for "${{ repo.RepoName }}"
workingDirectory: $(Build.SourcesDirectory)/repos/${{ repo.RepoName }}
continueOnError: true
- ${{ if ne(parameters.WhatIfPreference, 'true')}}:
- pwsh: |
$requestBy = '${{ parameters.GithubAssignessAlias }}'
if ($requestBy) {
Write-Host "##vso[task.setvariable variable=Assigness]$requestBy"
}
displayName: Set Github reviewers variable
- template: /eng/common/pipelines/templates/steps/create-pull-request.yml
parameters:
BaseBranchName: $(DefaultBranch)
PRBranchName: migration
PROwner: ${{ parameters.PRForkOwner }}
CommitMsg: "Migrate from ${{parameters.ShortForMigrateFrom}} to ${{parameters.ShortForMigrateTo}}"
RepoOwner: ${{ repo.RepoOwner }}
RepoName: ${{ repo.RepoName }}
PRTitle: "Migrate from ${{parameters.ShortForMigrateFrom}} to ${{parameters.ShortForMigrateTo}}"
WorkingDirectory: $(Build.SourcesDirectory)/repos/${{ repo.RepoName }}
${{ if ne(parameters.GithubAssignessAlias, '')}}:
GHAssignessVariable: Assigness
PRBody: "This is auto PR generated by pipeline: https://dev.azure.com/azure-sdk/internal/_build/results?buildId=$(Build.BuildId)&view=results"
PushArgs: -f
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
<#
```
Invoke-Pester -Output Detailed $PSScriptRoot/Repo-File-Content-Replacements.Tests.ps1
```
#>

Import-Module Pester
Set-StrictMode -Version Latest

BeforeAll {
function Backup-Folder($targetFolder, $backupFolder) {
if (!(Test-Path $targetFolder)) {
return $null
}

if (!(Test-Path $backupFolder)) {
New-Item $backupFolder -ItemType Directory
}
Copy-Item -Path "$targetFolder/*" -Destination $backupFolder -Recurse -PassThru -Force
}
function Reset-Folder($backupFolder, $targetFolder) {
sima-zhu marked this conversation as resolved.
Show resolved Hide resolved
if (Test-Path $backupFolder) {
Copy-Item -Path "$backupFolder/*" -Destination $targetFolder -Recurse -Force
Remove-Item $backupFolder -Recurse
}
}
}
AfterAll {
if (Test-Path "$PSScriptRoot/inputs/backup") {
Remove-Item "$PSScriptRoot/inputs/backup" -Recurse
}
}
# Test plan:
# 1. Tests on the path does not match exclude nor include.
# 1. Tests on the path matching exclude paths.
# 2. Tests on the path matching include paths.
# 3. Tests on the path matching both exclude and include.
# 4. Tests on the file match the migration map.
# 5. Tests on the file does not match the migration map.
Describe "repo-file-content-replacement" -Tag "UnitTest" {
# Passed cases
It "Test on the files have matching content" -TestCases @(
@{
exludePaths = "(/|\\)excludes(/|\\)|(/|\\)eng(/|\\)scripts-tests(/|\\)Repo-File-Content-Replacements(/|\\)inputs(/|\\)test1(/|\\)jsonFiles(/|\\)1.json";
includePaths = "(/|\\)excludes(/|\\)includes(/|\\)";
migrationMapFile = "$PSScriptRoot/inputs/test1/jsonFiles/1.json";
scannedDirectory = "$PSScriptRoot/inputs/test1";
}
) {
$backupFolder = "$PSScriptRoot/inputs/backup"
Backup-Folder -targetFolder $ScannedDirectory -backupFolder $backupFolder
$migrationMap = Get-Content $migrationMapFile -Raw
. $PSScriptRoot/../../scripts/Repo-File-Content-Replacements.ps1 `
sima-zhu marked this conversation as resolved.
Show resolved Hide resolved
-ExcludePathsRegex $exludePaths `
sima-zhu marked this conversation as resolved.
Show resolved Hide resolved
-IncludePathsRegex $includePaths `
-MigrationMapJson $migrationMap `
-ScannedDirectory $ScannedDirectory
$files = Get-ChildItem "$scannedDirectory/*" -Recurse -File -Include *.txt
foreach ($file in $files) {
$expectPath = $file.FullName -replace 'inputs', 'expected'
(Get-Content $file) | Should -Be (Get-Content $expectPath)
}
Reset-Folder -backupFolder $backupFolder -targetFolder $ScannedDirectory
sima-zhu marked this conversation as resolved.
Show resolved Hide resolved
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
3.txt
This is a testing for azsdk-pool-mms-win-2022-general migration.
This is a testing for windows2022 migration.
Expect to have changes for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
1.txt
This is a testing for windows-2022 migration.
No change expected for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
2.txt
This is a testing for windows-2022 migration.
Expect to have changes for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
6.txt
This is a testing for azsdk-pool-mms-win-2022-general migration.
This is a testing for windows2022 migration.
Expect to have changes for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
5.txt
This is a testing for azsdk-pool-mms-win-2022-general migration.
This is a testing for windows2022 migration.
Expect to have changes for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
4.txt
This is a testing for azsdk-pool-mms-win-2022-general migration.
This is a testing for windows2022 migration.
Expect to have changes for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
5.txt
No changes expected.
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
3.txt
This is a testing for azsdk-pool-mms-win-2022-general migration.
This is a testing for windows2022 migration.
Expect to have changes for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
1.txt
This is a testing for windows-2022 migration.
No change expected for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
2.txt
This is a testing for windows-2022 migration.
Expect to have changes for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
6.txt
This is a testing for azsdk-pool-mms-win-2022-general migration.
This is a testing for windows2022 migration.
Expect to have changes for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
5.txt
This is a testing for azsdk-pool-mms-win-2022-general migration.
This is a testing for windows2022 migration.
Expect to have changes for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
4.txt
This is a testing for azsdk-pool-mms-win-2022-general migration.
This is a testing for windows2022 migration.
Expect to have changes for this file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
[
{
"MigrateFrom": "azsdk-pool-mms-win-2022-general",
"MigrateTo": "azsdk-pool-mms-win-2022-general"
},
{
"MigrateFrom": "windows-2022",
"MigrateTo": "windows-2022"
},
{
"MigrateFrom": "windows2022",
"MigrateTo": "windows2022"
}
]
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
5.txt
No changes expected.
95 changes: 95 additions & 0 deletions eng/scripts/Repo-File-Content-Replacements.ps1
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
<#
.SYNOPSIS
The script is to do file content replacements under current folder.

.DESCRIPTION
We are having the script to help search all occurrences of specified strings within scope and replace to the target strings.
The script will scan all files by default.

.PARAMETER ExcludePathsRegex
The regex of excluded file paths. It is standard regex applied to the file path of the agent. The file path is relative to repo folder.
In win agent, the path is like ".\eng\scripts\Repo-File-Content-Replacements.ps1", so the regex path can be like "^.\\eng\\"
In linux agent, the path is like "./eng/scripts/Repo-File-Content-Replacements.ps1", so the regex path can be like "^./eng/"
To compitable with both cases, you can also specify your regex path like "^./|\\eng/|\\"
e.g we can specified multiple paths using '(^./sdk/)|(^./eng/common/)'
The script will exclude the specified paths from scanned file list first.

.PARAMETER IncludePathsRegex
The regex of file pattern which is used to add back the files you don't want to exclude. The IncludePathsRegex is supposed to match a subset of ExcludePathsRegex
It is standard regex applied to the file path of the agent. The file path is relative to repo folder.
In win agent, the path is like ".\eng\scripts\Repo-File-Content-Replacements.ps1", so the regex path can be like "^.\\eng\\"
In linux agent, the path is like "./eng/scripts/Repo-File-Content-Replacements.ps1", so the regex path can be like "^./eng/"
To compitable with both cases, you can also specify your regex path like "^./|\\eng/|\\"
e.g we can specified multiple paths using "(^./sdk/.*/platform-matrix.*.json$)|(^./sdk/.*/.*yml$)"

.PARAMETER MigrationMapJson
The map includes information of from-to pair. E.g:
[
{
"MigrateFrom": "azsdk-pool-mms-win-2022-general",
"MigrateTo": "azsdk-pool-mms-win-2022-general"
},
{
"MigrateFrom": "windows-2022",
"MigrateTo": "windows-2022"
},
{
"MigrateFrom": "windows2022",
"MigrateTo": "windows2022"
}
]

.PARAMETER ScannedDirectory
The directory to scan against. Usually, it is repo root.

.EXAMPLE
./eng/scripts/check-spelling-in-changed-files.ps1

This will run spell check with changes in the current branch with respect to
`target_branch_name`

#>

[CmdletBinding()]
param (
[Parameter(Mandatory = $false)]
[string]$ExcludePathsRegex = '',
[Parameter(Mandatory = $false)]
[string]$IncludePathsRegex = '',
sima-zhu marked this conversation as resolved.
Show resolved Hide resolved
[Parameter(Mandatory = $true)]
[string]$MigrationMapJson,
[Parameter(Mandatory = $false)]
[string]$ScannedDirectory = '.'
)

$files = Get-ChildItem -LiteralPath $ScannedDirectory -Recurse -File
$newFileCollection = @()
foreach ($file in $files) {
$relativePath = Resolve-Path -LiteralPath $file.FullName -Relative
if ($relativePath -notmatch $ExcludePathsRegex) {
$newFileCollection += $file
}
elseif ($relativePath -match $ExcludePathsRegex -and $relativePath -match $IncludePathsRegex) {
sima-zhu marked this conversation as resolved.
Show resolved Hide resolved
$newFileCollection += $file
}
}

# Without -NoEnumerate, a single element array[T] gets unwrapped as a single item T.
$migrationMap = ConvertFrom-Json $MigrationMapJson -NoEnumerate

Write-Host "The number of matching files: $($newFileCollection.Count)"
# Scan all Files and check the match
foreach ($file in $newFileCollection) {
$newcontent = Get-Content -LiteralPath $file.FullName -Raw
$fileChanged = $False
foreach ($migrate in $MigrationMap) {
if ($newcontent -match $migrate.MigrateFrom) {
$fileChanged = $True
$newcontent = $newcontent -replace "$($migrate.MigrateFrom)", "$($migrate.MigrateTo)"
}
}
if ($fileChanged) {
Write-Host "File to update: $($file.FullName)"
Set-Content -Path $file.FullName -Value $newcontent -NoNewline
}
}