mirror of
https://github.com/cmderdev/cmder.git
synced 2026-04-15 06:44:31 +08:00
Compare commits
87 Commits
copilot/re
...
copilot/im
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
58ab7976d2 | ||
|
|
09b00758b1 | ||
|
|
00a140dac9 | ||
|
|
d3df508d8e | ||
|
|
426d9e6efe | ||
|
|
59ef29cbab | ||
|
|
7fdae8ef48 | ||
|
|
e4ab8fb2fd | ||
|
|
6924bcb8b5 | ||
|
|
d11c5641bd | ||
|
|
75cd20b7ff | ||
|
|
dc7e1c144f | ||
|
|
55251f7cc3 | ||
|
|
0983e9b763 | ||
|
|
cb59bb54b0 | ||
|
|
c882ade42d | ||
|
|
dcbb0e9203 | ||
|
|
f3fa589ac3 | ||
|
|
32cf8767f5 | ||
|
|
03deb9d859 | ||
|
|
01d01c6a57 | ||
|
|
25ea725338 | ||
|
|
cbc7f53cd3 | ||
|
|
31e73cce59 | ||
|
|
2de93d1d55 | ||
|
|
aa801c2c17 | ||
|
|
1418da1c18 | ||
|
|
05b24c2173 | ||
|
|
f69c1db205 | ||
|
|
2ebb4f2d5b | ||
|
|
165ca3b05b | ||
|
|
be3d66d8a3 | ||
|
|
35c1cd6f42 | ||
|
|
07a2c6e5cd | ||
|
|
7cfa69fe74 | ||
|
|
74b089662c | ||
|
|
ed5ebfe5ba | ||
|
|
86d1cbcfa0 | ||
|
|
ae3ed7718c | ||
|
|
1e2d8eb47d | ||
|
|
adbd6322f6 | ||
|
|
d495b06726 | ||
|
|
a6f0015b16 | ||
|
|
8c0e2c542c | ||
|
|
f2bde614d4 | ||
|
|
0c6b2d9900 | ||
|
|
66ab9ad6f7 | ||
|
|
6137073ba3 | ||
|
|
6858619614 | ||
|
|
002282636a | ||
|
|
48cc28a634 | ||
|
|
e71c7679f6 | ||
|
|
dc93fa5d1b | ||
|
|
684185ca8a | ||
|
|
7b86263b0a | ||
|
|
10a0675c96 | ||
|
|
400cdfed54 | ||
|
|
a475de2d4c | ||
|
|
1afbd4ae49 | ||
|
|
82f7fdd61a | ||
|
|
e598536485 | ||
|
|
9cfff38939 | ||
|
|
2cc1125f62 | ||
|
|
2780a43323 | ||
|
|
641ce7b1e8 | ||
|
|
a52b0b9e10 | ||
|
|
825df3aace | ||
|
|
a937901e82 | ||
|
|
27c7a0cdc6 | ||
|
|
838c1cfaa3 | ||
|
|
4911924259 | ||
|
|
bff3862ad4 | ||
|
|
cd2c4ec877 | ||
|
|
2057ee58b0 | ||
|
|
0f6584fa02 | ||
|
|
570b1d6043 | ||
|
|
fa2a34a4c0 | ||
|
|
0bc7fd3aaa | ||
|
|
1e04711a21 | ||
|
|
3810b8d258 | ||
|
|
e6fea0bbab | ||
|
|
bf90303c96 | ||
|
|
e64c0b110d | ||
|
|
f2e8ae5189 | ||
|
|
4d21982f26 | ||
|
|
be60888258 | ||
|
|
d61a4f7f36 |
7
.github/dependabot.yml
vendored
7
.github/dependabot.yml
vendored
@@ -1,7 +1,12 @@
|
|||||||
|
# To get started with Dependabot version updates, you'll need to specify which
|
||||||
|
# package ecosystems to update and where the package manifests are located.
|
||||||
|
# Please see the documentation for all configuration options:
|
||||||
|
# https://docs.github.com/github/administering-a-repository/configuration-options-for-dependency-updates
|
||||||
|
|
||||||
version: 2
|
version: 2
|
||||||
updates:
|
updates:
|
||||||
# Enable version updates for GitHub Actions
|
# Enable version updates for GitHub Actions
|
||||||
- package-ecosystem: "github-actions"
|
- package-ecosystem: "github-actions" # See documentation for possible values
|
||||||
directory: "/" # Location of package manifests
|
directory: "/" # Location of package manifests
|
||||||
schedule:
|
schedule:
|
||||||
interval: "weekly"
|
interval: "weekly"
|
||||||
|
|||||||
229
.github/workflows/build.yml
vendored
229
.github/workflows/build.yml
vendored
@@ -12,6 +12,7 @@ on:
|
|||||||
- "v*"
|
- "v*"
|
||||||
pull_request:
|
pull_request:
|
||||||
branches: [ "master", "development" ]
|
branches: [ "master", "development" ]
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
#---------------------------------#
|
#---------------------------------#
|
||||||
# environment configuration #
|
# environment configuration #
|
||||||
@@ -42,20 +43,97 @@ jobs:
|
|||||||
- name: Summary - Repository checkout
|
- name: Summary - Repository checkout
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
echo "## 📦 Build Cmder - Workflow Summary" >> $env:GITHUB_STEP_SUMMARY
|
# Get Cmder version
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
. scripts/utils.ps1
|
||||||
echo "### Repository Information" >> $env:GITHUB_STEP_SUMMARY
|
$cmderVersion = Get-VersionStr
|
||||||
echo "| Property | Value |" >> $env:GITHUB_STEP_SUMMARY
|
$buildTime = (Get-Date).ToUniversalTime().ToString("yyyy-MM-ddTHH:mm:ssZ")
|
||||||
echo "| --- | --- |" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "| Repository | \`${{ github.repository }}\` |" >> $env:GITHUB_STEP_SUMMARY
|
# Determine branch and PR information
|
||||||
echo "| Branch | \`${{ github.ref_name }}\` |" >> $env:GITHUB_STEP_SUMMARY
|
$refName = "${{ github.ref_name }}"
|
||||||
echo "| Commit | \`${{ github.sha }}\` |" >> $env:GITHUB_STEP_SUMMARY
|
$headRef = "${{ github.head_ref }}"
|
||||||
echo "| Actor | @${{ github.actor }} |" >> $env:GITHUB_STEP_SUMMARY
|
$eventName = "${{ github.event_name }}"
|
||||||
echo "| Workflow | \`${{ github.workflow }}\` |" >> $env:GITHUB_STEP_SUMMARY
|
$prNumber = $null
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
$actualBranchName = $refName
|
||||||
|
$branchLink = ""
|
||||||
|
$prLink = ""
|
||||||
|
|
||||||
|
# Check if this is a PR merge ref (e.g., "3061/merge")
|
||||||
|
if ($refName -match '^(\d+)/(merge|head)$') {
|
||||||
|
$prNumber = $Matches[1]
|
||||||
|
# Use head_ref for the actual branch name if available
|
||||||
|
if ($headRef) {
|
||||||
|
$actualBranchName = $headRef
|
||||||
|
}
|
||||||
|
$branchLink = "https://github.com/${{ github.repository }}/tree/$actualBranchName"
|
||||||
|
$prLink = "https://github.com/${{ github.repository }}/pull/$prNumber"
|
||||||
|
} elseif ($eventName -eq "pull_request") {
|
||||||
|
# This is a pull request event
|
||||||
|
$prNumber = "${{ github.event.pull_request.number }}"
|
||||||
|
if ($headRef) {
|
||||||
|
$actualBranchName = $headRef
|
||||||
|
}
|
||||||
|
$branchLink = "https://github.com/${{ github.repository }}/tree/$actualBranchName"
|
||||||
|
$prLink = "https://github.com/${{ github.repository }}/pull/$prNumber"
|
||||||
|
} else {
|
||||||
|
# Regular branch, link to the branch tree
|
||||||
|
$branchLink = "https://github.com/${{ github.repository }}/tree/$refName"
|
||||||
|
}
|
||||||
|
|
||||||
|
$summary = @"
|
||||||
|
## 📦 Build Cmder - Workflow Summary
|
||||||
|
|
||||||
|
<small>Build started: ``$buildTime``</small>
|
||||||
|
|
||||||
|
### Repository Information
|
||||||
|
| Property | Value |
|
||||||
|
| --- | --- |
|
||||||
|
| Repository | [``${{ github.repository }}``](https://github.com/${{ github.repository }}) |
|
||||||
|
| Branch | [``$actualBranchName``]($branchLink) |
|
||||||
|
$(if ($prNumber) { "| Pull Request | [#$prNumber]($prLink) |" })
|
||||||
|
| Commit | [``${{ github.sha }}``](https://github.com/${{ github.repository }}/commit/${{ github.sha }}) |
|
||||||
|
| Actor | [@${{ github.actor }}](https://github.com/${{ github.actor }}) |
|
||||||
|
| Workflow | ``${{ github.workflow }}`` |
|
||||||
|
| Cmder Version | **$cmderVersion** |
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### 🗃️ Vendor Packages ([sources.json](vendor/sources.json))
|
||||||
|
| Package | Version |
|
||||||
|
| --- | --- |
|
||||||
|
"@
|
||||||
|
|
||||||
|
# Read vendor sources.json and add to summary
|
||||||
|
$vendorSources = Get-Content -Raw "vendor/sources.json" | ConvertFrom-Json
|
||||||
|
if ($vendorSources.Count -eq 0) {
|
||||||
|
$summary += "`n| _No vendor packages found_ | |"
|
||||||
|
} else {
|
||||||
|
foreach ($vendor in $vendorSources) {
|
||||||
|
# Create release link based on vendor package
|
||||||
|
$versionLink = "$($vendor.version)"
|
||||||
|
if ($vendor.url) {
|
||||||
|
# Extract owner/repo/tag from the URL and create release link
|
||||||
|
# Handle both /releases/download/ and /archive/ URLs
|
||||||
|
if ($vendor.url -match 'github\.com/([^/]+)/([^/]+)/(releases/download|archive)/([^/]+)') {
|
||||||
|
$owner = $Matches[1]
|
||||||
|
$repo = $Matches[2]
|
||||||
|
$pathType = $Matches[3]
|
||||||
|
$tag = $Matches[4]
|
||||||
|
if ($pathType -eq 'archive') {
|
||||||
|
$tag = $tag -replace '\.(?:tar\.gz|tgz|zip)$', ''
|
||||||
|
}
|
||||||
|
$versionLink = "[$($vendor.version)](https://github.com/$owner/$repo/releases/tag/$tag)"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$summary += "`n| ``$($vendor.name)`` | $versionLink |"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$summary += "`n"
|
||||||
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|
||||||
- name: Add MSBuild to PATH
|
- name: Add MSBuild to PATH
|
||||||
uses: microsoft/setup-msbuild@v2
|
uses: microsoft/setup-msbuild@v3
|
||||||
|
|
||||||
- name: Build Cmder Launcher
|
- name: Build Cmder Launcher
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
@@ -66,73 +144,102 @@ jobs:
|
|||||||
if: success()
|
if: success()
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
echo "### ✅ Build Status" >> $env:GITHUB_STEP_SUMMARY
|
$summary = @"
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "Cmder launcher successfully compiled." >> $env:GITHUB_STEP_SUMMARY
|
---
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
|
✅ Cmder built successfully.
|
||||||
|
|
||||||
|
"@
|
||||||
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|
||||||
- name: Pack the built files
|
- name: Pack the built files
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
working-directory: scripts
|
working-directory: scripts
|
||||||
run: .\pack.ps1 -verbose
|
run: .\pack.ps1 -verbose
|
||||||
|
|
||||||
- name: Summary - Package artifacts
|
|
||||||
if: success()
|
|
||||||
shell: pwsh
|
|
||||||
run: |
|
|
||||||
echo "### 📦 Artifacts Created" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "| Artifact | Size | Hash (SHA256) |" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "| --- | --- | --- |" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
$artifacts = @("cmder.zip", "cmder.7z", "cmder_mini.zip")
|
|
||||||
foreach ($artifact in $artifacts) {
|
|
||||||
$path = "build/$artifact"
|
|
||||||
if (Test-Path $path) {
|
|
||||||
$size = (Get-Item $path).Length / 1MB
|
|
||||||
# Truncate hash to first 16 chars for summary readability (full hash in hashes.txt)
|
|
||||||
$hash = (Get-FileHash $path -Algorithm SHA256).Hash.Substring(0, 16)
|
|
||||||
echo "| \`$artifact\` | $([math]::Round($size, 2)) MB | \`$hash...\` |" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
}
|
|
||||||
}
|
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
|
|
||||||
- name: Upload artifact (cmder.zip)
|
- name: Upload artifact (cmder.zip)
|
||||||
uses: actions/upload-artifact@v6
|
uses: actions/upload-artifact@v7
|
||||||
with:
|
with:
|
||||||
path: build/cmder.zip
|
path: build/cmder.zip
|
||||||
name: cmder.zip
|
name: cmder.zip
|
||||||
|
archive: false
|
||||||
if-no-files-found: error
|
if-no-files-found: error
|
||||||
|
|
||||||
- name: Upload artifact (cmder.7z)
|
- name: Upload artifact (cmder.7z)
|
||||||
uses: actions/upload-artifact@v6
|
uses: actions/upload-artifact@v7
|
||||||
with:
|
with:
|
||||||
path: build/cmder.7z
|
path: build/cmder.7z
|
||||||
name: cmder.7z
|
name: cmder.7z
|
||||||
|
archive: false
|
||||||
|
|
||||||
- name: Upload artifact (cmder_mini.zip)
|
- name: Upload artifact (cmder_mini.zip)
|
||||||
uses: actions/upload-artifact@v6
|
uses: actions/upload-artifact@v7
|
||||||
with:
|
with:
|
||||||
path: build/cmder_mini.zip
|
path: build/cmder_mini.zip
|
||||||
name: cmder_mini.zip
|
name: cmder_mini.zip
|
||||||
|
archive: false
|
||||||
|
|
||||||
- name: Upload artifact (hashes.txt)
|
- name: Upload artifact (hashes.txt)
|
||||||
uses: actions/upload-artifact@v6
|
uses: actions/upload-artifact@v7
|
||||||
with:
|
with:
|
||||||
path: build/hashes.txt
|
path: build/hashes.txt
|
||||||
name: hashes.txt
|
name: hashes.txt
|
||||||
|
archive: false
|
||||||
|
|
||||||
- name: Summary - Artifacts uploaded
|
- name: Summary - Artifacts uploaded
|
||||||
if: success()
|
if: success()
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
|
env:
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
run: |
|
run: |
|
||||||
echo "### ☁️ Upload Status" >> $env:GITHUB_STEP_SUMMARY
|
# Source utility functions
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
. scripts/utils.ps1
|
||||||
echo "All artifacts successfully uploaded to GitHub Actions:" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo '- ✅ `cmder.zip`' >> $env:GITHUB_STEP_SUMMARY
|
$summary = @"
|
||||||
echo '- ✅ `cmder.7z`' >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo '- ✅ `cmder_mini.zip`' >> $env:GITHUB_STEP_SUMMARY
|
### 🗃️ Artifacts
|
||||||
echo '- ✅ `hashes.txt`' >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
| Artifact | Size | Hash (SHA256) |
|
||||||
|
| --- | --- | --- |
|
||||||
|
"@
|
||||||
|
|
||||||
|
# Get all files from the build directory (excluding directories and hidden files)
|
||||||
|
if (Test-Path "build") {
|
||||||
|
$buildFiles = Get-ChildItem -Path "build" -File | Where-Object { -not $_.Name.StartsWith('.') } | Sort-Object Name
|
||||||
|
|
||||||
|
foreach ($file in $buildFiles) {
|
||||||
|
$artifact = $file.Name
|
||||||
|
$path = $file.FullName
|
||||||
|
$sizeFormatted = Format-FileSize -Bytes $file.Length
|
||||||
|
$hash = (Get-FileHash $path -Algorithm SHA256).Hash
|
||||||
|
|
||||||
|
# Try to get the actual artifact download URL
|
||||||
|
$downloadUrl = Get-ArtifactDownloadUrl -ArtifactName $artifact -Repository "${{ github.repository }}" -RunId "${{ github.run_id }}"
|
||||||
|
$warning = ""
|
||||||
|
|
||||||
|
if (-not $downloadUrl) {
|
||||||
|
# Fallback to workflow run page if artifact URL fetch fails
|
||||||
|
$downloadUrl = "https://github.com/${{ github.repository }}/actions/runs/${{ github.run_id }}"
|
||||||
|
$warning = " ⚠️"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Determine emoji based on file type
|
||||||
|
if ($artifact -match '\.txt$') {
|
||||||
|
$emoji = "📄"
|
||||||
|
} elseif ($artifact -match '\.(zip|rar|7z)$') {
|
||||||
|
$emoji = "🗄️"
|
||||||
|
} else {
|
||||||
|
$emoji = "📦"
|
||||||
|
}
|
||||||
|
|
||||||
|
$summary += "`n| $emoji [``$artifact``$warning]($downloadUrl) | $sizeFormatted | ``$hash`` |"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$summary += "`n"
|
||||||
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|
||||||
- name: Create Release
|
- name: Create Release
|
||||||
uses: softprops/action-gh-release@v2
|
uses: softprops/action-gh-release@v2
|
||||||
@@ -150,14 +257,20 @@ jobs:
|
|||||||
if: startsWith(github.ref, 'refs/tags/')
|
if: startsWith(github.ref, 'refs/tags/')
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
echo "### 🚀 Release Information" >> $env:GITHUB_STEP_SUMMARY
|
$summary = @"
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "Draft release created for tag: **\`${{ github.ref_name }}\`**" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "Release includes:" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "- Full version (\`cmder.zip\`, \`cmder.7z\`)" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "- Mini version (\`cmder_mini.zip\`)" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "- File hashes (\`hashes.txt\`)" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "> ⚠️ Release is in **draft** mode. Please review and publish manually." >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
### Release Information
|
||||||
|
|
||||||
|
🚀 Draft release created for tag: **``${{ github.ref_name }}``**
|
||||||
|
|
||||||
|
Release includes:
|
||||||
|
- Full version (``cmder.zip``, ``cmder.7z``)
|
||||||
|
- Mini version (``cmder_mini.zip``)
|
||||||
|
- File hashes (``hashes.txt``)
|
||||||
|
|
||||||
|
> ⚠️ Release is in **draft** mode. Please review and publish manually.
|
||||||
|
"@
|
||||||
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|||||||
67
.github/workflows/codeql.yml
vendored
67
.github/workflows/codeql.yml
vendored
@@ -50,20 +50,21 @@ jobs:
|
|||||||
- name: Summary - CodeQL analysis started
|
- name: Summary - CodeQL analysis started
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
$summary = @(
|
$summary = @"
|
||||||
'## 🔒 CodeQL Security Analysis - Workflow Summary'
|
## 🔒 CodeQL Security Analysis - Workflow Summary
|
||||||
''
|
|
||||||
'### Analysis Configuration'
|
### Analysis Configuration
|
||||||
''
|
|
||||||
'| Property | Value |'
|
| Property | Value |
|
||||||
'| --- | --- |'
|
| --- | --- |
|
||||||
'| Repository | `${{ github.repository }}` |'
|
| Repository | ``${{ github.repository }}`` |
|
||||||
'| Branch | `${{ github.ref_name }}` |'
|
| Branch | ``${{ github.ref_name }}`` |
|
||||||
'| Language | `${{ matrix.language }}` |'
|
| Language | ``${{ matrix.language }}`` |
|
||||||
'| Commit | `${{ github.sha }}` |'
|
| Commit | ``${{ github.sha }}`` |
|
||||||
''
|
|
||||||
)
|
"@
|
||||||
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|
||||||
# Initializes the CodeQL tools for scanning.
|
# Initializes the CodeQL tools for scanning.
|
||||||
- name: Initialize CodeQL
|
- name: Initialize CodeQL
|
||||||
@@ -78,7 +79,7 @@ jobs:
|
|||||||
# queries: security-extended,security-and-quality
|
# queries: security-extended,security-and-quality
|
||||||
|
|
||||||
- name: Add MSBuild to PATH
|
- name: Add MSBuild to PATH
|
||||||
uses: microsoft/setup-msbuild@v2
|
uses: microsoft/setup-msbuild@v3
|
||||||
|
|
||||||
- name: Build Cmder Launcher
|
- name: Build Cmder Launcher
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
@@ -89,13 +90,14 @@ jobs:
|
|||||||
if: success()
|
if: success()
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
$summary = @(
|
$summary = @"
|
||||||
'### ✅ Build Completed'
|
### ✅ Build Completed
|
||||||
''
|
|
||||||
'Cmder launcher built successfully for CodeQL analysis.'
|
Cmder launcher built successfully for CodeQL analysis.
|
||||||
''
|
|
||||||
)
|
"@
|
||||||
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|
||||||
- name: Perform CodeQL Analysis
|
- name: Perform CodeQL Analysis
|
||||||
uses: github/codeql-action/analyze@v4
|
uses: github/codeql-action/analyze@v4
|
||||||
@@ -106,13 +108,14 @@ jobs:
|
|||||||
if: success()
|
if: success()
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
$summary = @(
|
$summary = @"
|
||||||
'### 🔍 CodeQL Analysis Results'
|
### 🔍 CodeQL Analysis Results
|
||||||
''
|
|
||||||
'✅ CodeQL security analysis completed successfully.'
|
✅ CodeQL security analysis completed successfully.
|
||||||
''
|
|
||||||
'**Language analyzed:** `${{ matrix.language }}`'
|
**Language analyzed:** ``${{ matrix.language }}``
|
||||||
''
|
|
||||||
'> Check the Security tab for detailed findings and recommendations.'
|
> Check the Security tab for detailed findings and recommendations.
|
||||||
)
|
"@
|
||||||
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|||||||
137
.github/workflows/tests.yml
vendored
137
.github/workflows/tests.yml
vendored
@@ -39,20 +39,33 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v6
|
- uses: actions/checkout@v6
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
- name: Summary - Test execution started
|
- name: Summary - Test execution started
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
echo "## 🧪 Run Tests - Workflow Summary" >> $env:GITHUB_STEP_SUMMARY
|
# Get Cmder version
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
. scripts/utils.ps1
|
||||||
echo "### Test Environment" >> $env:GITHUB_STEP_SUMMARY
|
$cmderVersion = Get-VersionStr
|
||||||
echo "| Property | Value |" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "| --- | --- |" >> $env:GITHUB_STEP_SUMMARY
|
$summary = @"
|
||||||
echo "| Repository | \`${{ github.repository }}\` |" >> $env:GITHUB_STEP_SUMMARY
|
## ✅ Run Tests - Workflow Summary
|
||||||
echo "| Branch | \`${{ github.ref_name }}\` |" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "| Commit | \`${{ github.sha }}\` |" >> $env:GITHUB_STEP_SUMMARY
|
### Test Environment
|
||||||
echo "| Runner OS | \`${{ runner.os }}\` |" >> $env:GITHUB_STEP_SUMMARY
|
| Property | Value |
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
| --- | --- |
|
||||||
|
| Repository | ``${{ github.repository }}`` |
|
||||||
|
| Branch | ``${{ github.ref_name }}`` |
|
||||||
|
| Commit | ``${{ github.sha }}`` |
|
||||||
|
| Runner OS | ``${{ runner.os }}`` |
|
||||||
|
| Cmder Version | **$cmderVersion** |
|
||||||
|
| PowerShell Version | **$($PSVersionTable.PSVersion)** |
|
||||||
|
| Event | ``${{ github.event_name }}`` |
|
||||||
|
|
||||||
|
"@
|
||||||
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|
||||||
- name: Initialize vendors
|
- name: Initialize vendors
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
@@ -63,53 +76,127 @@ jobs:
|
|||||||
if: success()
|
if: success()
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
echo "### ⚙️ Vendor Initialization" >> $env:GITHUB_STEP_SUMMARY
|
# Get vendor versions from sources.json
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
$vendorInfo = @()
|
||||||
echo "✅ Vendor dependencies initialized successfully." >> $env:GITHUB_STEP_SUMMARY
|
$sources = Get-Content "vendor\sources.json" -Raw | ConvertFrom-Json
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
foreach ($source in $sources) {
|
||||||
|
$dir = $source.name
|
||||||
|
if (-not $dir) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
$versionFile = "vendor/$dir/.cmderver"
|
||||||
|
if (Test-Path $versionFile) {
|
||||||
|
$version = Get-Content $versionFile -Raw
|
||||||
|
$vendorInfo += "- **$dir**: $($version.Trim())"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$summary = @"
|
||||||
|
### ⚙️ Vendor Initialization
|
||||||
|
|
||||||
|
✅ Vendor dependencies initialized successfully.
|
||||||
|
|
||||||
|
**Vendor Versions:**
|
||||||
|
"@
|
||||||
|
|
||||||
|
$(
|
||||||
|
if ($vendorInfo.Count -eq 0) {
|
||||||
|
$summary += "_No vendor version information available._"
|
||||||
|
} else {
|
||||||
|
$summary += $vendorInfo -join "`n"
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|
||||||
- name: Summary - Test results table header
|
- name: Summary - Test results table header
|
||||||
if: success()
|
if: success()
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
echo "### 📋 Test Results" >> $env:GITHUB_STEP_SUMMARY
|
$summary = @"
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
### 📋 Test Results
|
||||||
echo "| Test | Status |" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "| --- | --- |" >> $env:GITHUB_STEP_SUMMARY
|
| Test | Status | Duration |
|
||||||
|
| --- | --- | --- |
|
||||||
|
"@
|
||||||
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|
||||||
- name: Testing Clink Shell
|
- name: Testing Clink Shell
|
||||||
|
id: test-clink
|
||||||
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
|
$startTime = Get-Date
|
||||||
cmd /c vendor\init.bat /v /d /t
|
cmd /c vendor\init.bat /v /d /t
|
||||||
|
$duration = [math]::Round(((Get-Date) - $startTime).TotalSeconds, 2)
|
||||||
|
echo "duration=$duration" >> $env:GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Summary - Clink Shell test
|
- name: Summary - Clink Shell test
|
||||||
if: success()
|
if: success()
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
echo "| Clink Shell | ✅ Passed |" >> $env:GITHUB_STEP_SUMMARY
|
$duration = "${{ steps.test-clink.outputs.duration }}"
|
||||||
|
if ($duration) {
|
||||||
|
$duration = "$duration s"
|
||||||
|
} else {
|
||||||
|
$duration = "N/A"
|
||||||
|
}
|
||||||
|
"| Clink Shell | ✅ Passed | $duration |" | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
- name: Testing PowerShell
|
- name: Testing PowerShell
|
||||||
|
id: test-powershell
|
||||||
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
|
$startTime = Get-Date
|
||||||
PowerShell.exe -ExecutionPolicy Bypass -NoLogo -NoProfile -Command "$env:CMDER_DEBUG='1'; . 'vendor\profile.ps1'"
|
PowerShell.exe -ExecutionPolicy Bypass -NoLogo -NoProfile -Command "$env:CMDER_DEBUG='1'; . 'vendor\profile.ps1'"
|
||||||
|
$duration = [math]::Round(((Get-Date) - $startTime).TotalSeconds, 2)
|
||||||
|
echo "duration=$duration" >> $env:GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Summary - PowerShell test
|
- name: Summary - PowerShell test
|
||||||
if: success()
|
if: success()
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
echo "| PowerShell | ✅ Passed |" >> $env:GITHUB_STEP_SUMMARY
|
$duration = "${{ steps.test-powershell.outputs.duration }}"
|
||||||
|
if ($duration) {
|
||||||
|
$duration = "$duration s"
|
||||||
|
} else {
|
||||||
|
$duration = "N/A"
|
||||||
|
}
|
||||||
|
"| PowerShell | ✅ Passed | $duration |" | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
- name: Testing Bash
|
- name: Testing Bash
|
||||||
|
id: test-bash
|
||||||
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
|
$startTime = Get-Date
|
||||||
bash vendor/cmder.sh
|
bash vendor/cmder.sh
|
||||||
|
$duration = [math]::Round(((Get-Date) - $startTime).TotalSeconds, 2)
|
||||||
|
echo "duration=$duration" >> $env:GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Summary - Bash test
|
- name: Summary - Bash test
|
||||||
if: success()
|
if: success()
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
echo "| Bash | ✅ Passed |" >> $env:GITHUB_STEP_SUMMARY
|
$duration = "${{ steps.test-bash.outputs.duration }}"
|
||||||
|
if ($duration) {
|
||||||
|
$duration = "$duration s"
|
||||||
|
} else {
|
||||||
|
$duration = "N/A"
|
||||||
|
}
|
||||||
|
"| Bash | ✅ Passed | $duration |" | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|
||||||
- name: Summary - All tests completed
|
- name: Summary - All tests completed
|
||||||
if: success()
|
if: success()
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
$summary = @"
|
||||||
echo "### ✅ All Tests Completed" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
echo "" >> $env:GITHUB_STEP_SUMMARY
|
### ✅ All Tests Completed
|
||||||
echo "All shell environments tested successfully!" >> $env:GITHUB_STEP_SUMMARY
|
|
||||||
|
All shell environments tested successfully!
|
||||||
|
|
||||||
|
**Test Coverage:**
|
||||||
|
- ✅ Clink shell environment (Windows cmd.exe with Clink)
|
||||||
|
- ✅ PowerShell environment (with Cmder profile)
|
||||||
|
- ✅ Bash environment (Git Bash integration)
|
||||||
|
"@
|
||||||
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|||||||
264
.github/workflows/vendor.yml
vendored
264
.github/workflows/vendor.yml
vendored
@@ -6,6 +6,10 @@ on:
|
|||||||
# At 13:37 UTC every day.
|
# At 13:37 UTC every day.
|
||||||
- cron: '37 13 * * *'
|
- cron: '37 13 * * *'
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: vendor-update
|
||||||
|
cancel-in-progress: false
|
||||||
|
|
||||||
defaults:
|
defaults:
|
||||||
run:
|
run:
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
@@ -31,87 +35,259 @@ jobs:
|
|||||||
- name: Summary - Workflow started
|
- name: Summary - Workflow started
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
$summary = @(
|
$summary = @"
|
||||||
'## 📦 Update Vendor - Workflow Summary'
|
## 📦 Update Vendor - Workflow Summary
|
||||||
''
|
|
||||||
'Checking for vendor dependency updates...'
|
Checking for vendor dependency updates...
|
||||||
''
|
|
||||||
)
|
"@
|
||||||
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|
||||||
- id: make-changes
|
- id: make-changes
|
||||||
name: Checking for updates
|
name: Checking for updates
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
run: |
|
run: |
|
||||||
$currentVersion = (Get-Content .\vendor\sources.json | ConvertFrom-Json)
|
$currentVersion = (Get-Content -Raw .\vendor\sources.json | ConvertFrom-Json)
|
||||||
. .\scripts\update.ps1 -verbose
|
. .\scripts\update.ps1 -verbose
|
||||||
|
|
||||||
|
# Export count of updated packages (update.ps1 is expected to set $count)
|
||||||
|
if (-not ($count)) { $count = 0 }
|
||||||
Set-GHVariable -Name COUNT_UPDATED -Value $count
|
Set-GHVariable -Name COUNT_UPDATED -Value $count
|
||||||
$newVersion = (Get-Content .\vendor\sources.json | ConvertFrom-Json)
|
|
||||||
|
$newVersion = (Get-Content -Raw .\vendor\sources.json | ConvertFrom-Json)
|
||||||
|
# Source utility functions
|
||||||
|
. scripts/utils.ps1
|
||||||
|
|
||||||
$listUpdated = ""
|
$listUpdated = ""
|
||||||
$updateMessage = "| Name | Old Version | New Version |`n| :--- | ---- | ---- |`n"
|
$updateMessage = "| Name | Old Version | New Version |`n| :--- | :---: | :---: |`n"
|
||||||
|
$majorUpdates = @()
|
||||||
|
$singleDepName = ""
|
||||||
|
$singleDepOldVersion = ""
|
||||||
|
$singleDepNewVersion = ""
|
||||||
foreach ($s in $newVersion) {
|
foreach ($s in $newVersion) {
|
||||||
$oldVersion = ($currentVersion | Where-Object {$_.name -eq $s.name}).version
|
$oldVersion = ($currentVersion | Where-Object {$_.name -eq $s.name}).version
|
||||||
if ($s.version -ne $oldVersion) {
|
if ($s.version -ne $oldVersion) {
|
||||||
$repoUrl = ($repoUrl = $s.Url.Replace("/archive/", "/releases/")).Substring(0, $repoUrl.IndexOf("/releases/")) + "/releases"
|
$repoUrl = ($repoUrl = $s.Url.Replace("/archive/", "/releases/")).Substring(0, $repoUrl.IndexOf("/releases/")) + "/releases"
|
||||||
|
|
||||||
|
# Store single dependency info for messages (only if this is the only update)
|
||||||
|
if ($count -eq 1) {
|
||||||
|
$singleDepName = $s.name
|
||||||
|
$singleDepOldVersion = $oldVersion
|
||||||
|
$singleDepNewVersion = $s.version
|
||||||
|
}
|
||||||
|
|
||||||
|
# Determine change type and emoji using shared function
|
||||||
|
$result = Get-VersionChangeType -OldVersion $oldVersion -NewVersion $s.version
|
||||||
|
$changeType = $result.ChangeType
|
||||||
|
$emoji = $result.Emoji
|
||||||
|
$isMajor = $result.IsMajor
|
||||||
|
|
||||||
|
# Track major updates for changelog section
|
||||||
|
if ($isMajor) {
|
||||||
|
$compareUrl = "$repoUrl/compare/v$oldVersion...v$($s.version)"
|
||||||
|
$majorUpdates += @{
|
||||||
|
name = $s.name
|
||||||
|
oldVersion = $oldVersion
|
||||||
|
newVersion = $s.version
|
||||||
|
compareUrl = $compareUrl
|
||||||
|
repoUrl = $repoUrl
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$listUpdated += "$($s.name) v$($s.version), "
|
$listUpdated += "$($s.name) v$($s.version), "
|
||||||
$updateMessage += "| **[$($s.name)]($repoUrl)** | $oldVersion | **$($s.version)** |`n"
|
$updateMessage += "| $emoji **[$($s.name)]($repoUrl)** | \`$oldVersion\` | **\`$($s.version)\`** |`n"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($count -eq 0) { return }
|
if ($count -eq 0) { return }
|
||||||
|
|
||||||
Set-GHVariable -Name LIST_UPDATED -Value $listUpdated.Trim(', ')
|
Set-GHVariable -Name LIST_UPDATED -Value $listUpdated.Trim(', ')
|
||||||
echo "UPDATE_MESSAGE<<<EOF`n$updateMessage`n<EOF" | Out-File -FilePath $env:GITHUB_ENV -Append -Encoding utf8
|
# Set single dependency variables (they will only be used if COUNT_UPDATED is 1)
|
||||||
|
# Use safe fallback values in case variables weren't set (shouldn't happen but prevents errors)
|
||||||
|
if ([string]::IsNullOrEmpty($singleDepName) -and $count -eq 1) {
|
||||||
|
# This shouldn't happen, but if it does, log a warning
|
||||||
|
Write-Warning "Single dependency name not set despite count being 1"
|
||||||
|
$singleDepName = "unknown-package"
|
||||||
|
$singleDepOldVersion = "unknown"
|
||||||
|
$singleDepNewVersion = "unknown"
|
||||||
|
} elseif ([string]::IsNullOrEmpty($singleDepName)) {
|
||||||
|
# For multiple dependencies, set placeholder values (won't be used)
|
||||||
|
$singleDepName = ""
|
||||||
|
$singleDepOldVersion = ""
|
||||||
|
$singleDepNewVersion = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
Set-GHVariable -Name SINGLE_DEP_NAME -Value $singleDepName
|
||||||
|
Set-GHVariable -Name SINGLE_DEP_OLD_VERSION -Value $singleDepOldVersion
|
||||||
|
Set-GHVariable -Name SINGLE_DEP_NEW_VERSION -Value $singleDepNewVersion
|
||||||
|
|
||||||
|
# Write multiline UPDATE_MESSAGE to GITHUB_ENV
|
||||||
|
## echo "UPDATE_MESSAGE<<EOF`n$updateMessage`nEOF" | Out-File -FilePath $env:GITHUB_ENV -Append -Encoding utf8
|
||||||
|
Add-Content -Path $env:GITHUB_ENV -Value "UPDATE_MESSAGE<<EOF"
|
||||||
|
Add-Content -Path $env:GITHUB_ENV -Value $updateMessage
|
||||||
|
Add-Content -Path $env:GITHUB_ENV -Value "EOF"
|
||||||
|
|
||||||
|
# Generate major updates changelog section and export
|
||||||
|
if ($majorUpdates.Count -gt 0) {
|
||||||
|
$changelogSection = "`n<details>`n<summary>🔥 Major version updates - View changelog</summary>`n`n"
|
||||||
|
foreach ($update in $majorUpdates) {
|
||||||
|
$changelogSection += "### [$($update.name)]($($update.repoUrl))`n"
|
||||||
|
$changelogSection += "**$($update.oldVersion)** → **$($update.newVersion)**`n`n"
|
||||||
|
$changelogSection += "- [View full changelog]($($update.compareUrl))`n"
|
||||||
|
$changelogSection += "- [Release notes]($($update.repoUrl)/tag/v$($update.newVersion))`n`n"
|
||||||
|
}
|
||||||
|
$changelogSection += "</details>`n"
|
||||||
|
|
||||||
|
Add-Content -Path $env:GITHUB_ENV -Value "CHANGELOG_SECTION<<EOF"
|
||||||
|
Add-Content -Path $env:GITHUB_ENV -Value $changelogSection
|
||||||
|
Add-Content -Path $env:GITHUB_ENV -Value "EOF"
|
||||||
|
Add-Content -Path $env:GITHUB_ENV -Value "HAS_BREAKING_CHANGES=True"
|
||||||
|
} else {
|
||||||
|
Add-Content -Path $env:GITHUB_ENV -Value "CHANGELOG_SECTION="
|
||||||
|
Add-Content -Path $env:GITHUB_ENV -Value "HAS_BREAKING_CHANGES=False"
|
||||||
|
}
|
||||||
|
|
||||||
- name: Summary - Update check results
|
- name: Summary - Update check results
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
$count = [int]$env:COUNT_UPDATED
|
$count = [int]$env:COUNT_UPDATED
|
||||||
|
|
||||||
if ($count -eq 0) {
|
if ($count -eq 0) {
|
||||||
$summary = @(
|
$summary = @"
|
||||||
'### ✅ No Updates Available'
|
### ✅ No Updates Available
|
||||||
''
|
|
||||||
'All vendor dependencies are up to date.'
|
All vendor dependencies are up to date! 🎉
|
||||||
)
|
"@
|
||||||
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY
|
|
||||||
} else {
|
} else {
|
||||||
$word = if ($count -eq 1) { 'dependency' } else { 'dependencies' }
|
$word = if ($count -eq 1) { 'dependency' } else { 'dependencies' }
|
||||||
$summary = @(
|
$summary = @"
|
||||||
'### 🔄 Updates Found'
|
### 🔄 Updates Found
|
||||||
''
|
|
||||||
"**$count** vendor $word updated:"
|
"@
|
||||||
''
|
$summary += '📦 **' + $env:SINGLE_DEP_NAME + '** updated from `' + $env:SINGLE_DEP_OLD_VERSION + '` to `' + $env:SINGLE_DEP_NEW_VERSION + '`' + "`n" + "`n"
|
||||||
$env:UPDATE_MESSAGE
|
$summary += '📦 **' + $count + '** vendor ' + $word + ' updated:' + "`n" + "`n"
|
||||||
''
|
}
|
||||||
)
|
|
||||||
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY
|
$summary += $env:UPDATE_MESSAGE + "`n"
|
||||||
|
|
||||||
|
# Check if we can auto-merge (only minor/patch changes)
|
||||||
|
$hasBreaking = $env:HAS_BREAKING_CHANGES -eq 'True'
|
||||||
|
if ($hasBreaking) {
|
||||||
|
$summary += '> ⚠️ **Note:** This update contains major version changes that may include breaking changes.'
|
||||||
|
} else {
|
||||||
|
$summary += '> ℹ️ **Note:** This update only contains minor or patch changes.'
|
||||||
|
}
|
||||||
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|
||||||
|
- name: Auto-merge minor updates
|
||||||
|
if: env.COUNT_UPDATED > 0 && env.HAS_BREAKING_CHANGES != 'True'
|
||||||
|
shell: pwsh
|
||||||
|
run: |
|
||||||
|
try {
|
||||||
|
echo "### 🚀 Auto-merging Updates" >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
echo "" >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
echo "Attempting to automatically merge non-breaking changes to master..." >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
|
||||||
|
git config --global user.name "github-actions[bot]"
|
||||||
|
git config --global user.email "github-actions[bot]@users.noreply.github.com"
|
||||||
|
|
||||||
|
# Commit the changes
|
||||||
|
git add vendor/sources.json
|
||||||
|
$commitResult = git commit -m "⬆️ Update dependencies ($env:LIST_UPDATED)"
|
||||||
|
$commitSuccess = $LASTEXITCODE -eq 0
|
||||||
|
|
||||||
|
if ($commitSuccess) {
|
||||||
|
# Push directly to master
|
||||||
|
git push origin HEAD:master
|
||||||
|
$pushSuccess = $LASTEXITCODE -eq 0
|
||||||
|
|
||||||
|
if ($pushSuccess) {
|
||||||
|
echo "" >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
echo "✅ **Success!** Updates have been automatically merged to master." >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
echo "" >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
echo "**Updated dependencies:** $env:LIST_UPDATED" >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
|
||||||
|
# Set a flag to skip PR creation
|
||||||
|
echo "AUTO_MERGED=true" | Out-File -FilePath $env:GITHUB_ENV -Append -Encoding utf8
|
||||||
|
} else {
|
||||||
|
throw "Failed to push to master (exit code: $LASTEXITCODE)"
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
throw "Failed to commit changes (exit code: $LASTEXITCODE)"
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
echo "" >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
echo "⚠️ **Warning:** Unable to automatically merge updates." >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
echo "" >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
echo "**Error:** $($_.Exception.Message)" >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
echo "" >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
echo "Falling back to creating a pull request..." >> $env:GITHUB_STEP_SUMMARY
|
||||||
|
|
||||||
|
Write-Warning "Failed to auto-merge: $($_.Exception.Message)"
|
||||||
|
|
||||||
|
# Only reset if commit was successful but push failed
|
||||||
|
if ($commitSuccess -and -not $pushSuccess) {
|
||||||
|
try {
|
||||||
|
git reset --hard HEAD~1
|
||||||
|
if ($LASTEXITCODE -ne 0) {
|
||||||
|
Write-Warning "Failed to reset commit (exit code: $LASTEXITCODE), continuing with PR creation"
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
Write-Warning "Failed to reset commit: $($_.Exception.Message), continuing with PR creation"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
# Set flag to create PR instead
|
||||||
|
echo "AUTO_MERGED=false" | Out-File -FilePath $env:GITHUB_ENV -Append -Encoding utf8
|
||||||
}
|
}
|
||||||
|
|
||||||
- uses: peter-evans/create-pull-request@v8
|
- uses: peter-evans/create-pull-request@v8
|
||||||
if: env.COUNT_UPDATED > 0
|
if: fromJSON(env.COUNT_UPDATED) > 0 && (env.HAS_BREAKING_CHANGES == 'True' || env.AUTO_MERGED == 'false')
|
||||||
with:
|
with:
|
||||||
title: 'Updates to `${{ env.COUNT_UPDATED }}` vendored dependencies'
|
title: ${{ env.COUNT_UPDATED == 1 && format('⬆️ Update {0}', env.LIST_UPDATED) || format('⬆️ Update {0} vendored dependencies', env.COUNT_UPDATED) }}
|
||||||
body: |
|
body: |
|
||||||
### Automatically updated `${{ env.COUNT_UPDATED }}` dependencies:
|
### ${{ env.COUNT_UPDATED == 1 && format('📦 Updated {0} from `{1}` to `{2}`', env.SINGLE_DEP_NAME, env.SINGLE_DEP_OLD_VERSION, env.SINGLE_DEP_NEW_VERSION) || format('📦 Automatically updated `{0}` dependencies', env.COUNT_UPDATED) }}
|
||||||
|
|
||||||
${{ env.UPDATE_MESSAGE }}
|
${{ env.UPDATE_MESSAGE }}
|
||||||
|
|
||||||
|
${{ env.CHANGELOG_SECTION }}
|
||||||
|
|
||||||
---
|
---
|
||||||
Please verify and then **Merge** the pull request to update.
|
|
||||||
|
${{ env.HAS_BREAKING_CHANGES == 'True' && '⚠️ **This update contains major version changes that may include breaking changes.**' || 'ℹ️ This update only contains minor or patch changes.' }}
|
||||||
|
|
||||||
|
Please verify and then **Merge** the pull request to apply the updates.
|
||||||
commit-message: '⬆️ Update dependencies (${{ env.LIST_UPDATED }})'
|
commit-message: '⬆️ Update dependencies (${{ env.LIST_UPDATED }})'
|
||||||
branch: update-vendor
|
branch: update-vendor
|
||||||
base: master
|
base: master
|
||||||
|
|
||||||
- name: Summary - Pull request created
|
- name: Summary - Pull request created
|
||||||
if: env.COUNT_UPDATED > 0
|
if: env.COUNT_UPDATED > 0 && (env.HAS_BREAKING_CHANGES == 'True' || env.AUTO_MERGED == 'false')
|
||||||
shell: pwsh
|
shell: pwsh
|
||||||
run: |
|
run: |
|
||||||
$summary = @(
|
$summary = @"
|
||||||
'### 🎉 Pull Request Created'
|
### 🎉 Pull Request Created
|
||||||
''
|
|
||||||
'A pull request has been created to update the vendor dependencies.'
|
A pull request has been created to update the vendor dependencies.
|
||||||
''
|
|
||||||
'**Branch:** `update-vendor`'
|
**Branch:** ``update-vendor``
|
||||||
''
|
|
||||||
$env:LIST_UPDATED -and "**Updated dependencies:** $env:LIST_UPDATED" -or "**Updated dependencies:** "
|
$(if (-not [string]::IsNullOrEmpty($env:LIST_UPDATED)) { "**Updated dependencies:** $env:LIST_UPDATED" } else { "**Updated dependencies:** " })
|
||||||
''
|
|
||||||
'> Please review and merge the pull request to apply the updates.'
|
"@
|
||||||
)
|
|
||||||
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY
|
if ($env:HAS_BREAKING_CHANGES -eq 'True') {
|
||||||
|
$summary += "> ⚠️ **Manual review required:** This update contains major version changes."
|
||||||
|
} else {
|
||||||
|
$summary += "> ℹ️ **Note:** Auto-merge failed, manual review required."
|
||||||
|
}
|
||||||
|
|
||||||
|
$summary += @"
|
||||||
|
> Please review and merge the pull request to apply the updates.
|
||||||
|
|
||||||
|
"@
|
||||||
|
|
||||||
|
$summary | Add-Content -Path $env:GITHUB_STEP_SUMMARY -Encoding utf8
|
||||||
|
|||||||
@@ -70,7 +70,7 @@ if ($PSCmdlet.MyInvocation.BoundParameters["Verbose"].IsPresent) {
|
|||||||
foreach ($t in $targets.GetEnumerator()) {
|
foreach ($t in $targets.GetEnumerator()) {
|
||||||
Create-Archive "$cmderRoot" "$saveTo\$($t.Name)" $t.Value
|
Create-Archive "$cmderRoot" "$saveTo\$($t.Name)" $t.Value
|
||||||
$hash = (Digest-Hash "$saveTo\$($t.Name)")
|
$hash = (Digest-Hash "$saveTo\$($t.Name)")
|
||||||
Add-Content -path "$saveTo\hashes.txt" -value ($t.Name + ' ' + $hash)
|
Add-Content -path "$saveTo\hashes.txt" -value ($t.Name + "`t" + $hash)
|
||||||
}
|
}
|
||||||
|
|
||||||
Pop-Location
|
Pop-Location
|
||||||
|
|||||||
@@ -259,6 +259,8 @@ function Fetch-DownloadUrl {
|
|||||||
}
|
}
|
||||||
|
|
||||||
$count = 0
|
$count = 0
|
||||||
|
$hasBreakingChanges = $false
|
||||||
|
$updateDetails = @()
|
||||||
|
|
||||||
# Read the current sources content
|
# Read the current sources content
|
||||||
$sources = Get-Content $sourcesPath | Out-String | ConvertFrom-Json
|
$sources = Get-Content $sourcesPath | Out-String | ConvertFrom-Json
|
||||||
@@ -301,6 +303,26 @@ foreach ($s in $sources) {
|
|||||||
# }
|
# }
|
||||||
|
|
||||||
$count++
|
$count++
|
||||||
|
|
||||||
|
# Analyze version change type using shared function
|
||||||
|
$result = Get-VersionChangeType -OldVersion $s.version -NewVersion $version
|
||||||
|
$changeType = $result.ChangeType
|
||||||
|
|
||||||
|
# Determine if this is a breaking change
|
||||||
|
if ($changeType -eq "downgrade" -or $changeType -eq "major") {
|
||||||
|
$hasBreakingChanges = $true
|
||||||
|
} elseif ($changeType -eq "unknown") {
|
||||||
|
# If version parsing failed, treat as potentially breaking
|
||||||
|
$hasBreakingChanges = $true
|
||||||
|
Write-Verbose "Could not parse version as semantic version for dependency '$($s.name)' (old: '$($s.version)', new: '$version'), treating as potentially breaking"
|
||||||
|
}
|
||||||
|
|
||||||
|
$updateDetails += @{
|
||||||
|
name = $s.name
|
||||||
|
oldVersion = $s.version
|
||||||
|
newVersion = $version
|
||||||
|
changeType = $changeType
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$s.url = $downloadUrl
|
$s.url = $downloadUrl
|
||||||
@@ -314,12 +336,16 @@ if ($count -eq 0) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Export update details for GitHub Actions
|
||||||
|
if ($Env:GITHUB_ACTIONS -eq 'true') {
|
||||||
|
$updateDetailsJson = $updateDetails | ConvertTo-Json -Compress
|
||||||
|
Write-Output "UPDATE_DETAILS=$updateDetailsJson" | Out-File -FilePath $env:GITHUB_ENV -Append -Encoding utf8
|
||||||
|
Write-Output "HAS_BREAKING_CHANGES=$hasBreakingChanges" | Out-File -FilePath $env:GITHUB_ENV -Append -Encoding utf8
|
||||||
|
Write-Output "::notice title=Task Complete::Successfully updated $count dependencies."
|
||||||
|
}
|
||||||
|
|
||||||
if ($Env:APPVEYOR -eq 'True') {
|
if ($Env:APPVEYOR -eq 'True') {
|
||||||
Add-AppveyorMessage -Message "Successfully updated $count dependencies." -Category Information
|
Add-AppveyorMessage -Message "Successfully updated $count dependencies." -Category Information
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($Env:GITHUB_ACTIONS -eq 'true') {
|
|
||||||
Write-Output "::notice title=Task Complete::Successfully updated $count dependencies."
|
|
||||||
}
|
|
||||||
|
|
||||||
Write-Host -ForegroundColor green "Successfully updated $count dependencies."
|
Write-Host -ForegroundColor green "Successfully updated $count dependencies."
|
||||||
|
|||||||
@@ -249,3 +249,201 @@ function Download-File {
|
|||||||
$wc.Proxy.Credentials = [System.Net.CredentialCache]::DefaultNetworkCredentials;
|
$wc.Proxy.Credentials = [System.Net.CredentialCache]::DefaultNetworkCredentials;
|
||||||
$wc.DownloadFile($Url, $File)
|
$wc.DownloadFile($Url, $File)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function Format-FileSize {
|
||||||
|
<#
|
||||||
|
.SYNOPSIS
|
||||||
|
Formats a file size in bytes to a human-readable string using binary units.
|
||||||
|
|
||||||
|
.DESCRIPTION
|
||||||
|
Converts file sizes to appropriate binary units (B, KiB, MiB, GiB) for better readability.
|
||||||
|
|
||||||
|
.PARAMETER Bytes
|
||||||
|
The file size in bytes to format.
|
||||||
|
|
||||||
|
.EXAMPLE
|
||||||
|
Format-FileSize -Bytes 1024
|
||||||
|
Returns "1.00 KiB"
|
||||||
|
|
||||||
|
.EXAMPLE
|
||||||
|
Format-FileSize -Bytes 15728640
|
||||||
|
Returns "15.00 MiB"
|
||||||
|
#>
|
||||||
|
param(
|
||||||
|
[Parameter(Mandatory = $true)]
|
||||||
|
[double]$Bytes
|
||||||
|
)
|
||||||
|
|
||||||
|
if ($Bytes -ge 1GB) {
|
||||||
|
return "{0:N2} GiB" -f ($Bytes / 1GB)
|
||||||
|
} elseif ($Bytes -ge 1MB) {
|
||||||
|
return "{0:N2} MiB" -f ($Bytes / 1MB)
|
||||||
|
} elseif ($Bytes -ge 1KB) {
|
||||||
|
return "{0:N2} KiB" -f ($Bytes / 1KB)
|
||||||
|
} else {
|
||||||
|
return "{0:N0} B" -f $Bytes
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function Get-VersionChangeType {
|
||||||
|
<#
|
||||||
|
.SYNOPSIS
|
||||||
|
Analyzes version changes using semantic versioning to determine the type of update and appropriate emoji.
|
||||||
|
|
||||||
|
.DESCRIPTION
|
||||||
|
Parses old and new version strings, compares them using semantic versioning rules,
|
||||||
|
and returns information about the change type, emoji indicator, and whether it's a major update.
|
||||||
|
|
||||||
|
.PARAMETER OldVersion
|
||||||
|
The previous version string (e.g., "1.2.3" or "2.52.0.windows.1").
|
||||||
|
|
||||||
|
.PARAMETER NewVersion
|
||||||
|
The new version string (e.g., "1.3.0" or "3.0.0.windows.1").
|
||||||
|
|
||||||
|
.OUTPUTS
|
||||||
|
Returns a hashtable with the following keys:
|
||||||
|
- ChangeType: "major", "minor", "patch", "downgrade", or "unknown"
|
||||||
|
- Emoji: The corresponding emoji indicator (🔥, 🚀, ⬆️, or 🔄)
|
||||||
|
- IsMajor: Boolean indicating if this is a major version update
|
||||||
|
|
||||||
|
.EXAMPLE
|
||||||
|
$result = Get-VersionChangeType -OldVersion "1.2.3" -NewVersion "2.0.0"
|
||||||
|
# Returns: @{ ChangeType = "major"; Emoji = "🔥"; IsMajor = $true }
|
||||||
|
|
||||||
|
.EXAMPLE
|
||||||
|
$result = Get-VersionChangeType -OldVersion "1.2.3" -NewVersion "1.3.0"
|
||||||
|
# Returns: @{ ChangeType = "minor"; Emoji = "🚀"; IsMajor = $false }
|
||||||
|
#>
|
||||||
|
param(
|
||||||
|
[Parameter(Mandatory = $true)]
|
||||||
|
[string]$OldVersion,
|
||||||
|
|
||||||
|
[Parameter(Mandatory = $true)]
|
||||||
|
[string]$NewVersion
|
||||||
|
)
|
||||||
|
|
||||||
|
$changeType = "unknown"
|
||||||
|
$emoji = "🔄"
|
||||||
|
$isMajor = $false
|
||||||
|
|
||||||
|
try {
|
||||||
|
# Handle versions with more than 4 parts and strip pre-release identifiers
|
||||||
|
$oldVerStr = $OldVersion.Split('-')[0]
|
||||||
|
$newVerStr = $NewVersion.Split('-')[0]
|
||||||
|
|
||||||
|
# Split by dots and take only numeric parts, first 4 max
|
||||||
|
$oldParts = $oldVerStr.Split('.') | Where-Object { $_ -match '^\d+$' } | Select-Object -First 4
|
||||||
|
$newParts = $newVerStr.Split('.') | Where-Object { $_ -match '^\d+$' } | Select-Object -First 4
|
||||||
|
|
||||||
|
# Ensure we have at least 2 parts (major.minor) for semantic versioning
|
||||||
|
if ($oldParts.Count -ge 2 -and $newParts.Count -ge 2) {
|
||||||
|
$oldVerParseable = $oldParts -join '.'
|
||||||
|
$newVerParseable = $newParts -join '.'
|
||||||
|
|
||||||
|
$oldVer = [System.Version]::Parse($oldVerParseable)
|
||||||
|
$newVer = [System.Version]::Parse($newVerParseable)
|
||||||
|
|
||||||
|
if ($newVer -lt $oldVer) {
|
||||||
|
$changeType = "downgrade"
|
||||||
|
# Don't set emoji for downgrades in this function - caller handles it
|
||||||
|
} elseif ($newVer.Major -gt $oldVer.Major) {
|
||||||
|
$changeType = "major"
|
||||||
|
$emoji = "🔥"
|
||||||
|
$isMajor = $true
|
||||||
|
} elseif ($newVer.Minor -gt $oldVer.Minor) {
|
||||||
|
$changeType = "minor"
|
||||||
|
$emoji = "🚀"
|
||||||
|
} elseif ($newVer -gt $oldVer) {
|
||||||
|
$changeType = "patch"
|
||||||
|
$emoji = "⬆️"
|
||||||
|
} else {
|
||||||
|
# No version increase detected (versions are equal)
|
||||||
|
$changeType = "unknown"
|
||||||
|
$emoji = "🔄"
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
# Not enough numeric parts for semantic versioning
|
||||||
|
throw "Not enough numeric version parts"
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
# If semantic versioning fails, return unknown
|
||||||
|
$changeType = "unknown"
|
||||||
|
$emoji = "🔄"
|
||||||
|
$isMajor = $false
|
||||||
|
}
|
||||||
|
|
||||||
|
return @{
|
||||||
|
ChangeType = $changeType
|
||||||
|
Emoji = $emoji
|
||||||
|
IsMajor = $isMajor
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function Get-ArtifactDownloadUrl {
|
||||||
|
<#
|
||||||
|
.SYNOPSIS
|
||||||
|
Retrieves the download URL for a GitHub Actions artifact with retry logic.
|
||||||
|
|
||||||
|
.DESCRIPTION
|
||||||
|
Uses the GitHub CLI to fetch artifact information from the GitHub API with automatic retries.
|
||||||
|
Falls back to returning $null if all attempts fail.
|
||||||
|
|
||||||
|
.PARAMETER ArtifactName
|
||||||
|
The name of the artifact to retrieve the download URL for.
|
||||||
|
|
||||||
|
.PARAMETER Repository
|
||||||
|
The GitHub repository in the format "owner/repo".
|
||||||
|
|
||||||
|
.PARAMETER RunId
|
||||||
|
The GitHub Actions workflow run ID.
|
||||||
|
|
||||||
|
.PARAMETER MaxRetries
|
||||||
|
Maximum number of retry attempts. Default is 3.
|
||||||
|
|
||||||
|
.PARAMETER DelaySeconds
|
||||||
|
Delay in seconds between retry attempts. Default is 2.
|
||||||
|
|
||||||
|
.EXAMPLE
|
||||||
|
Get-ArtifactDownloadUrl -ArtifactName "cmder.zip" -Repository "cmderdev/cmder" -RunId "123456789"
|
||||||
|
|
||||||
|
.EXAMPLE
|
||||||
|
Get-ArtifactDownloadUrl -ArtifactName "build-output" -Repository "owner/repo" -RunId "987654321" -MaxRetries 5 -DelaySeconds 3
|
||||||
|
#>
|
||||||
|
param(
|
||||||
|
[Parameter(Mandatory = $true)]
|
||||||
|
[string]$ArtifactName,
|
||||||
|
|
||||||
|
[Parameter(Mandatory = $true)]
|
||||||
|
[string]$Repository,
|
||||||
|
|
||||||
|
[Parameter(Mandatory = $true)]
|
||||||
|
[string]$RunId,
|
||||||
|
|
||||||
|
[int]$MaxRetries = 3,
|
||||||
|
[int]$DelaySeconds = 2
|
||||||
|
)
|
||||||
|
|
||||||
|
for ($i = 0; $i -lt $MaxRetries; $i++) {
|
||||||
|
try {
|
||||||
|
# Use GitHub CLI to get artifact information
|
||||||
|
$artifactsJson = gh api "repos/$Repository/actions/runs/$RunId/artifacts" --jq ".artifacts[] | select(.name == `"$ArtifactName`")"
|
||||||
|
|
||||||
|
if ($artifactsJson) {
|
||||||
|
$artifact = $artifactsJson | ConvertFrom-Json
|
||||||
|
if ($artifact.id) {
|
||||||
|
# Construct browser-accessible GitHub Actions artifact download URL
|
||||||
|
# Format: https://github.com/owner/repo/actions/runs/{run_id}/artifacts/{artifact_id}
|
||||||
|
return "https://github.com/$Repository/actions/runs/$RunId/artifacts/$($artifact.id)"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
Write-Host "Attempt $($i + 1) failed to get artifact URL for $ArtifactName : $_"
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($i -lt ($MaxRetries - 1)) {
|
||||||
|
Start-Sleep -Seconds $DelaySeconds
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $null
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user