/eng/common/scripts/git-branch-push.ps1

http://github.com/WindowsAzure/azure-sdk-for-java · Powershell · 191 lines · 147 code · 19 blank · 25 comment · 40 complexity · 078e3df33a253a3100d42e39fbea1356 MD5 · raw file

  1. #!/usr/bin/env pwsh -c
  2. <#
  3. .DESCRIPTION
  4. Create local branch of the given repo and attempt to push changes. The push may fail if
  5. there has been other changes pushed to the same branch, if so, fetch, rebase and try again.
  6. .PARAMETER PRBranchName
  7. The name of the github branch the changes are being put into
  8. .PARAMETER CommitMsg
  9. The message for this particular commit
  10. .PARAMETER GitUrl
  11. The GitHub repository URL
  12. .PARAMETER PushArgs
  13. Optional arguments to the push command
  14. #>
  15. [CmdletBinding(SupportsShouldProcess = $true)]
  16. param(
  17. [Parameter(Mandatory = $true)]
  18. [string] $PRBranchName,
  19. [Parameter(Mandatory = $true)]
  20. [string] $CommitMsg,
  21. [Parameter(Mandatory = $true)]
  22. [string] $GitUrl,
  23. [Parameter(Mandatory = $false)]
  24. [string] $PushArgs = "",
  25. [Parameter(Mandatory = $false)]
  26. [string] $RemoteName = "azure-sdk-fork",
  27. [Parameter(Mandatory = $false)]
  28. [boolean] $SkipCommit = $false,
  29. [Parameter(Mandatory = $false)]
  30. [boolean] $AmendCommit = $false
  31. )
  32. # This is necessary because of the git command output writing to stderr.
  33. # Without explicitly setting the ErrorActionPreference to continue the script
  34. # would fail the first time git wrote command output.
  35. $ErrorActionPreference = "Continue"
  36. if ((git remote) -contains $RemoteName)
  37. {
  38. Write-Host "git remote get-url $RemoteName"
  39. $remoteUrl = git remote get-url $RemoteName
  40. if ($remoteUrl -ne $GitUrl)
  41. {
  42. Write-Error "Remote with name $RemoteName already exists with an incompatible url [$remoteUrl] which should be [$GitUrl]."
  43. exit 1
  44. }
  45. }
  46. else
  47. {
  48. Write-Host "git remote add $RemoteName $GitUrl"
  49. git remote add $RemoteName $GitUrl
  50. if ($LASTEXITCODE -ne 0)
  51. {
  52. Write-Error "Unable to add remote LASTEXITCODE=$($LASTEXITCODE), see command output above."
  53. exit $LASTEXITCODE
  54. }
  55. }
  56. # Checkout to $PRBranch, create new one if not exists.
  57. git show-ref --verify --quiet refs/heads/$PRBranchName
  58. if ($LASTEXITCODE -eq 0) {
  59. Write-Host "git checkout $PRBranchName."
  60. git checkout $PRBranchName
  61. }
  62. else {
  63. Write-Host "git checkout -b $PRBranchName."
  64. git checkout -b $PRBranchName
  65. }
  66. if ($LASTEXITCODE -ne 0)
  67. {
  68. Write-Error "Unable to create branch LASTEXITCODE=$($LASTEXITCODE), see command output above."
  69. exit $LASTEXITCODE
  70. }
  71. if (!$SkipCommit) {
  72. if ($AmendCommit) {
  73. $amendOption = "--amend"
  74. }
  75. else {
  76. $amendOption = ""
  77. }
  78. Write-Host "git -c user.name=`"azure-sdk`" -c user.email=`"azuresdk@microsoft.com`" commit $amendOption -am `"$($CommitMsg)`""
  79. git -c user.name="azure-sdk" -c user.email="azuresdk@microsoft.com" commit $amendOption -am "$($CommitMsg)"
  80. if ($LASTEXITCODE -ne 0)
  81. {
  82. Write-Error "Unable to add files and create commit LASTEXITCODE=$($LASTEXITCODE), see command output above."
  83. exit $LASTEXITCODE
  84. }
  85. }
  86. else {
  87. Write-Host "Skipped applying commit"
  88. }
  89. # The number of retries can be increased if necessary. In theory, the number of retries
  90. # should be the max number of libraries in the largest pipeline -1 as everything except
  91. # the first commit could hit issues and need to rebase. The reason this isn't set to that
  92. # is because the largest pipeline is cognitive services which has 18 libraries in its
  93. # pipeline and that just seemed a bit too large and 10 seemed like a good starting value.
  94. $numberOfRetries = 10
  95. $needsRetry = $false
  96. $tryNumber = 0
  97. do
  98. {
  99. $needsRetry = $false
  100. Write-Host "git push $RemoteName $PRBranchName $PushArgs"
  101. git push $RemoteName $PRBranchName $PushArgs
  102. $tryNumber++
  103. if ($LASTEXITCODE -ne 0)
  104. {
  105. $needsRetry = $true
  106. Write-Host "Git push failed with LASTEXITCODE=$($LASTEXITCODE) Need to fetch and rebase: attempt number=$($tryNumber)"
  107. Write-Host "git fetch $RemoteName $PRBranchName"
  108. # Full fetch will fail when the repo is in a sparse-checkout state, and single branch fetch is faster anyway.
  109. git fetch $RemoteName $PRBranchName
  110. if ($LASTEXITCODE -ne 0)
  111. {
  112. Write-Error "Unable to fetch remote LASTEXITCODE=$($LASTEXITCODE), see command output above."
  113. exit $LASTEXITCODE
  114. }
  115. try
  116. {
  117. $TempPatchFile = New-TemporaryFile
  118. Write-Host "git diff ${PRBranchName}~ ${PRBranchName} --output $TempPatchFile"
  119. git diff ${PRBranchName}~ ${PRBranchName} --output $TempPatchFile
  120. if ($LASTEXITCODE -ne 0)
  121. {
  122. Write-Error "Unable to create diff file LASTEXITCODE=$($LASTEXITCODE), see command output above."
  123. continue
  124. }
  125. Write-Host "git reset --hard $RemoteName/${PRBranchName}"
  126. git reset --hard $RemoteName/${PRBranchName}
  127. if ($LASTEXITCODE -ne 0)
  128. {
  129. Write-Error "Unable to hard reset branch LASTEXITCODE=$($LASTEXITCODE), see command output above."
  130. continue
  131. }
  132. # -C0 means to use no extra before or after lines of context to enable us to avoid adjacent line merge conflicts
  133. Write-Host "git apply -C0 $TempPatchFile"
  134. git apply -C0 $TempPatchFile
  135. if ($LASTEXITCODE -ne 0)
  136. {
  137. Write-Error "Unable to apply diff file LASTEXITCODE=$($LASTEXITCODE), see command output above."
  138. exit $LASTEXITCODE
  139. }
  140. Write-Host "git add -A"
  141. git add -A
  142. if ($LASTEXITCODE -ne 0)
  143. {
  144. Write-Error "Unable to git add LASTEXITCODE=$($LASTEXITCODE), see command output above."
  145. continue
  146. }
  147. Write-Host "git -c user.name=`"azure-sdk`" -c user.email=`"azuresdk@microsoft.com`" commit -m `"$($CommitMsg)`""
  148. git -c user.name="azure-sdk" -c user.email="azuresdk@microsoft.com" commit -m "$($CommitMsg)"
  149. if ($LASTEXITCODE -ne 0)
  150. {
  151. Write-Error "Unable to commit LASTEXITCODE=$($LASTEXITCODE), see command output above."
  152. continue
  153. }
  154. }
  155. finally
  156. {
  157. if ( Test-Path $TempPatchFile )
  158. {
  159. Remove-Item $TempPatchFile
  160. }
  161. }
  162. }
  163. } while($needsRetry -and $tryNumber -le $numberOfRetries)
  164. if ($LASTEXITCODE -ne 0 -or $tryNumber -gt $numberOfRetries)
  165. {
  166. Write-Error "Unable to push commit after $($tryNumber) retries LASTEXITCODE=$($LASTEXITCODE), see command output above."
  167. if (0 -eq $LASTEXITCODE)
  168. {
  169. exit 1
  170. }
  171. exit $LASTEXITCODE
  172. }