mirror of
				https://github.com/velopack/velopack.git
				synced 2025-10-25 15:19:22 +00:00 
			
		
		
		
	Add cross-compiling support
This commit is contained in:
		
							
								
								
									
										104
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										104
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -134,6 +134,9 @@ jobs: | |||||||
|           lipo -create -output macos-universal/UpdateMac macos-x64/UpdateMac macos-arm64/UpdateMac |           lipo -create -output macos-universal/UpdateMac macos-x64/UpdateMac macos-arm64/UpdateMac | ||||||
|           file macos-universal/UpdateMac |           file macos-universal/UpdateMac | ||||||
|           lipo -archs macos-universal/UpdateMac |           lipo -archs macos-universal/UpdateMac | ||||||
|  |       - uses: geekyeggo/delete-artifact@v5 | ||||||
|  |         with: | ||||||
|  |           name: rust-macos-* | ||||||
|       - name: Upload Universal Binary |       - name: Upload Universal Binary | ||||||
|         uses: actions/upload-artifact@v4 |         uses: actions/upload-artifact@v4 | ||||||
|         with: |         with: | ||||||
| @@ -172,52 +175,75 @@ jobs: | |||||||
|           sudo add-apt-repository universe |           sudo add-apt-repository universe | ||||||
|           sudo apt install libfuse2 |           sudo apt install libfuse2 | ||||||
|         if: ${{ matrix.os == 'ubuntu-latest' }} |         if: ${{ matrix.os == 'ubuntu-latest' }} | ||||||
|  |       - name: Install squashfs-tools | ||||||
|  |         run: brew install squashfs | ||||||
|  |         if: ${{ matrix.os == 'macos-latest' }} | ||||||
|       - name: Install dotnet-coverage |       - name: Install dotnet-coverage | ||||||
|         run: dotnet tool install -g dotnet-coverage |         run: dotnet tool install -g dotnet-coverage | ||||||
|       - name: Build .NET |       - name: Build .NET | ||||||
|         run: dotnet build -c Release |         run: dotnet build -c Release | ||||||
|       - name: Wait for artifacts |       - uses: caesay/wait-artifact-action@494939e840383463b1686ce3624a8aab059c2c8b | ||||||
|         shell: pwsh |         with: | ||||||
|         env: |           token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|           GH_TOKEN: ${{ secrets.GITHUB_TOKEN }} # for gh cli |           max_wait_seconds: 900 | ||||||
|         run: | |           artifacts: rust-macos-latest,rust-windows-latest,rust-ubuntu-latest | ||||||
|           # Wait 15 minutes for the artifact to become available |           verbose: true | ||||||
|           $artifactName = "rust-${{ matrix.os }}" |  | ||||||
|           $maxAttempts = 90 |  | ||||||
|           $sleepSeconds = 10 |  | ||||||
|           $attempt = 0 |  | ||||||
|           $workflowRunId = $env:GITHUB_RUN_ID |  | ||||||
|  |  | ||||||
|           Write-Host "Waiting for artifact '$artifactName' to become available in workflow run $workflowRunId..." |  | ||||||
|           while ($attempt -lt $maxAttempts) { |  | ||||||
|             $artifactsJson = gh api repos/$env:GITHUB_REPOSITORY/actions/runs/$workflowRunId/artifacts |  | ||||||
|             $artifacts = $artifactsJson | ConvertFrom-Json |  | ||||||
|             if ($artifacts.artifacts | Where-Object { $_.name -eq $artifactName }) { |  | ||||||
|               Write-Host "Artifact '$artifactName' is available, continuing..." |  | ||||||
|               break |  | ||||||
|             } |  | ||||||
|             Write-Host "Artifact not available yet. Attempt $($attempt + 1)/$maxAttempts" |  | ||||||
|             Start-Sleep -Seconds $sleepSeconds |  | ||||||
|             $attempt++ |  | ||||||
|           } |  | ||||||
|    |  | ||||||
|           if ($attempt -ge $maxAttempts) { |  | ||||||
|             Write-Host "Error: Artifact '$artifactName' did not become available in time." |  | ||||||
|             exit 1 |  | ||||||
|           } |  | ||||||
|       - name: Download Rust Artifacts |       - name: Download Rust Artifacts | ||||||
|         uses: actions/download-artifact@v4 |         uses: actions/download-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: rust-${{ matrix.os }} |  | ||||||
|           path: src/Rust/target/release |           path: src/Rust/target/release | ||||||
|  |           pattern: rust-* | ||||||
|  |           merge-multiple: true | ||||||
|       - name: Test .NET |       - name: Test .NET | ||||||
|         run: dotnet test --no-build -c Release -l "console;verbosity=detailed;consoleLoggerParameters=ErrorsOnly" -l GithubActions -- RunConfiguration.CollectSourceInformation=true |         run: dotnet test --no-build -c Release -l "console;verbosity=detailed;consoleLoggerParameters=ErrorsOnly" -l GithubActions -- RunConfiguration.CollectSourceInformation=true | ||||||
|  |       - name: Upload Cross-Compile Artifacts | ||||||
|  |         uses: actions/upload-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           name: cross-${{ matrix.os }} | ||||||
|  |           path: test/artifacts/* | ||||||
|       - name: Upload Coverage Artifacts |       - name: Upload Coverage Artifacts | ||||||
|         uses: actions/upload-artifact@v4 |         uses: actions/upload-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: coverage-dotnet-${{ matrix.os }} |           name: coverage-dotnet-${{ matrix.os }} | ||||||
|           path: ./test/*.xml |           path: ./test/*.xml | ||||||
|  |  | ||||||
|  |   test-cross: | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         os: [windows-latest, ubuntu-latest] | ||||||
|  |     needs: [test-dotnet] | ||||||
|  |     runs-on: ${{ matrix.os }} | ||||||
|  |     env: | ||||||
|  |       NUGET_PACKAGES: ${{ github.workspace }}/.nuget/packages | ||||||
|  |     steps: | ||||||
|  |       - uses: actions/checkout@v4 | ||||||
|  |         with: | ||||||
|  |           fetch-depth: 0 | ||||||
|  |       - uses: actions/cache@v4 | ||||||
|  |         with: | ||||||
|  |           path: ${{ github.workspace }}/.nuget/packages | ||||||
|  |           key: ${{ runner.os }}-nuget-${{ hashFiles('**/packages.lock.json') }} | ||||||
|  |           restore-keys: | | ||||||
|  |             ${{ runner.os }}-nuget- | ||||||
|  |       - name: Install FUSE | ||||||
|  |         run: | | ||||||
|  |           sudo add-apt-repository universe | ||||||
|  |           sudo apt install libfuse2 | ||||||
|  |         if: ${{ matrix.os == 'ubuntu-latest' }} | ||||||
|  |       - name: Download Cross Artifacts | ||||||
|  |         uses: actions/download-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           path: test/artifacts | ||||||
|  |           pattern: cross-* | ||||||
|  |           merge-multiple: true | ||||||
|  |       - name: Test Cross-Compiled Apps | ||||||
|  |         env: | ||||||
|  |           VELOPACK_CROSS_ARTIFACTS: true | ||||||
|  |         run: dotnet test -c Release test/Velopack.Packaging.Tests/Velopack.Packaging.Tests.csproj --filter "FullyQualifiedName~RunCrossApp" -l "console;verbosity=detailed;consoleLoggerParameters=ErrorsOnly" -l GithubActions -- RunConfiguration.CollectSourceInformation=true | ||||||
|  |       - uses: geekyeggo/delete-artifact@v5 | ||||||
|  |         with: | ||||||
|  |           name: cross-* | ||||||
|  |  | ||||||
|   package: |   package: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     needs: [build-rust-windows, build-rust-linux, build-mac-universal] |     needs: [build-rust-windows, build-rust-linux, build-mac-universal] | ||||||
| @@ -227,21 +253,12 @@ jobs: | |||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v4 | ||||||
|         with: |         with: | ||||||
|           fetch-depth: 0 |           fetch-depth: 0 | ||||||
|       - name: Download Rust OSX |       - name: Download Rust Artifacts | ||||||
|         uses: actions/download-artifact@v4 |         uses: actions/download-artifact@v4 | ||||||
|         with: |         with: | ||||||
|           name: rust-macos-latest |  | ||||||
|           path: src/Rust/target/release |  | ||||||
|       - name: Download Rust Windows |  | ||||||
|         uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: rust-windows-latest |  | ||||||
|           path: src/Rust/target/release |  | ||||||
|       - name: Download Rust Linux |  | ||||||
|         uses: actions/download-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: rust-ubuntu-latest |  | ||||||
|           path: src/Rust/target/release |           path: src/Rust/target/release | ||||||
|  |           pattern: rust-* | ||||||
|  |           merge-multiple: true | ||||||
|       - name: Build .NET |       - name: Build .NET | ||||||
|         run: dotnet build -c Release /p:PackRustAssets=true /p:ContinuousIntegrationBuild=true |         run: dotnet build -c Release /p:PackRustAssets=true /p:ContinuousIntegrationBuild=true | ||||||
|       - name: Upload Package Artifacts |       - name: Upload Package Artifacts | ||||||
| @@ -267,6 +284,9 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           directory: ./test |           directory: ./test | ||||||
|           fail_ci_if_error: true |           fail_ci_if_error: true | ||||||
|  |       - uses: geekyeggo/delete-artifact@v5 | ||||||
|  |         with: | ||||||
|  |           name: coverage-* | ||||||
|      |      | ||||||
|  |  | ||||||
|       # - name: Publish to GitHub Packages |       # - name: Publish to GitHub Packages | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								Velopack.sln
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								Velopack.sln
									
									
									
									
									
								
							| @@ -54,10 +54,10 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "VeloWpfSample", "samples\Ve | |||||||
| EndProject | EndProject | ||||||
| Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Divergic.Logging.Xunit", "test\Divergic.Logging.Xunit\Divergic.Logging.Xunit.csproj", "{5ED2E9AF-101D-4D2D-B0B5-90A920EF692D}" | Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Divergic.Logging.Xunit", "test\Divergic.Logging.Xunit\Divergic.Logging.Xunit.csproj", "{5ED2E9AF-101D-4D2D-B0B5-90A920EF692D}" | ||||||
| EndProject | EndProject | ||||||
| Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Velopack.Packaging.HostModel", "src\Velopack.Packaging.HostModel\Velopack.Packaging.HostModel.csproj", "{E9A2620C-C638-446C-BA30-F62C05709365}" |  | ||||||
| EndProject |  | ||||||
| Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Velopack.Build", "src\Velopack.Build\Velopack.Build.csproj", "{97C9B2CF-877F-4C98-A513-058784A23697}" | Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Velopack.Build", "src\Velopack.Build\Velopack.Build.csproj", "{97C9B2CF-877F-4C98-A513-058784A23697}" | ||||||
| EndProject | EndProject | ||||||
|  | Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "Velopack.IcoLib", "src\Velopack.IcoLib\Velopack.IcoLib.csproj", "{8A0A980A-D51C-458E-8942-00BC900FD2D0}" | ||||||
|  | EndProject | ||||||
| Global | Global | ||||||
| 	GlobalSection(SolutionConfigurationPlatforms) = preSolution | 	GlobalSection(SolutionConfigurationPlatforms) = preSolution | ||||||
| 		Debug|Any CPU = Debug|Any CPU | 		Debug|Any CPU = Debug|Any CPU | ||||||
| @@ -120,14 +120,14 @@ Global | |||||||
| 		{5ED2E9AF-101D-4D2D-B0B5-90A920EF692D}.Debug|Any CPU.Build.0 = Debug|Any CPU | 		{5ED2E9AF-101D-4D2D-B0B5-90A920EF692D}.Debug|Any CPU.Build.0 = Debug|Any CPU | ||||||
| 		{5ED2E9AF-101D-4D2D-B0B5-90A920EF692D}.Release|Any CPU.ActiveCfg = Release|Any CPU | 		{5ED2E9AF-101D-4D2D-B0B5-90A920EF692D}.Release|Any CPU.ActiveCfg = Release|Any CPU | ||||||
| 		{5ED2E9AF-101D-4D2D-B0B5-90A920EF692D}.Release|Any CPU.Build.0 = Release|Any CPU | 		{5ED2E9AF-101D-4D2D-B0B5-90A920EF692D}.Release|Any CPU.Build.0 = Release|Any CPU | ||||||
| 		{E9A2620C-C638-446C-BA30-F62C05709365}.Debug|Any CPU.ActiveCfg = Debug|Any CPU |  | ||||||
| 		{E9A2620C-C638-446C-BA30-F62C05709365}.Debug|Any CPU.Build.0 = Debug|Any CPU |  | ||||||
| 		{E9A2620C-C638-446C-BA30-F62C05709365}.Release|Any CPU.ActiveCfg = Release|Any CPU |  | ||||||
| 		{E9A2620C-C638-446C-BA30-F62C05709365}.Release|Any CPU.Build.0 = Release|Any CPU |  | ||||||
| 		{97C9B2CF-877F-4C98-A513-058784A23697}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | 		{97C9B2CF-877F-4C98-A513-058784A23697}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | ||||||
| 		{97C9B2CF-877F-4C98-A513-058784A23697}.Debug|Any CPU.Build.0 = Debug|Any CPU | 		{97C9B2CF-877F-4C98-A513-058784A23697}.Debug|Any CPU.Build.0 = Debug|Any CPU | ||||||
| 		{97C9B2CF-877F-4C98-A513-058784A23697}.Release|Any CPU.ActiveCfg = Release|Any CPU | 		{97C9B2CF-877F-4C98-A513-058784A23697}.Release|Any CPU.ActiveCfg = Release|Any CPU | ||||||
| 		{97C9B2CF-877F-4C98-A513-058784A23697}.Release|Any CPU.Build.0 = Release|Any CPU | 		{97C9B2CF-877F-4C98-A513-058784A23697}.Release|Any CPU.Build.0 = Release|Any CPU | ||||||
|  | 		{8A0A980A-D51C-458E-8942-00BC900FD2D0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU | ||||||
|  | 		{8A0A980A-D51C-458E-8942-00BC900FD2D0}.Debug|Any CPU.Build.0 = Debug|Any CPU | ||||||
|  | 		{8A0A980A-D51C-458E-8942-00BC900FD2D0}.Release|Any CPU.ActiveCfg = Release|Any CPU | ||||||
|  | 		{8A0A980A-D51C-458E-8942-00BC900FD2D0}.Release|Any CPU.Build.0 = Release|Any CPU | ||||||
| 	EndGlobalSection | 	EndGlobalSection | ||||||
| 	GlobalSection(SolutionProperties) = preSolution | 	GlobalSection(SolutionProperties) = preSolution | ||||||
| 		HideSolutionNode = FALSE | 		HideSolutionNode = FALSE | ||||||
|   | |||||||
| @@ -10,7 +10,7 @@ | |||||||
|     <AppendTargetFrameworkToOutputPath>true</AppendTargetFrameworkToOutputPath> |     <AppendTargetFrameworkToOutputPath>true</AppendTargetFrameworkToOutputPath> | ||||||
|     <AppendRuntimeIdentifierToOutputPath>false</AppendRuntimeIdentifierToOutputPath> |     <AppendRuntimeIdentifierToOutputPath>false</AppendRuntimeIdentifierToOutputPath> | ||||||
|     <BeforeTargetFrameworkInferenceTargets>$(MSBuildThisFileDirectory)SelfContained.targets</BeforeTargetFrameworkInferenceTargets> |     <BeforeTargetFrameworkInferenceTargets>$(MSBuildThisFileDirectory)SelfContained.targets</BeforeTargetFrameworkInferenceTargets> | ||||||
|     <NoWarn>$(NoWarn);NETSDK1188</NoWarn> |     <NoWarn>$(NoWarn);NETSDK1188;NU5100</NoWarn> | ||||||
|  |  | ||||||
|     <SatelliteResourceLanguages>en</SatelliteResourceLanguages> |     <SatelliteResourceLanguages>en</SatelliteResourceLanguages> | ||||||
|     <PathMap>$([System.IO.Path]::GetFullPath('$(MSBuildThisFileDirectory)'))=./</PathMap> |     <PathMap>$([System.IO.Path]::GetFullPath('$(MSBuildThisFileDirectory)'))=./</PathMap> | ||||||
|   | |||||||
| @@ -35,10 +35,7 @@ | |||||||
|  |  | ||||||
|   <Target Name="AddNugetVendorLibs" BeforeTargets="Build" Condition=" '$(VelopackPackageVendorLibs)' == 'true' "> |   <Target Name="AddNugetVendorLibs" BeforeTargets="Build" Condition=" '$(VelopackPackageVendorLibs)' == 'true' "> | ||||||
|     <ItemGroup> |     <ItemGroup> | ||||||
|       <None Include="..\..\vendor\rcedit.exe" Pack="true" PackagePath="vendor" /> |       <None Include="..\..\vendor\**" Pack="true" PackagePath="vendor" /> | ||||||
|       <None Include="..\..\vendor\zstd.exe" Pack="true" PackagePath="vendor" /> |  | ||||||
|       <None Include="..\..\vendor\signtool.exe" Pack="true" PackagePath="vendor" /> |  | ||||||
|       <None Include="..\..\vendor\appimagetool-x86_64.AppImage" Pack="true" PackagePath="vendor" /> |  | ||||||
|       <None Include="..\..\Velopack.entitlements" Pack="true" PackagePath="vendor" /> |       <None Include="..\..\Velopack.entitlements" Pack="true" PackagePath="vendor" /> | ||||||
|       <None Include="..\..\artwork\DefaultApp.icns" Pack="true" PackagePath="vendor" /> |       <None Include="..\..\artwork\DefaultApp.icns" Pack="true" PackagePath="vendor" /> | ||||||
|       <None Include="..\..\artwork\DefaultApp.png" Pack="true" PackagePath="vendor" /> |       <None Include="..\..\artwork\DefaultApp.png" Pack="true" PackagePath="vendor" /> | ||||||
|   | |||||||
| @@ -39,7 +39,10 @@ public class PublishTask : MSBuildAsyncTask | |||||||
|             return true; |             return true; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         await client.UploadLatestReleaseAssetsAsync(Channel, ReleaseDirectory, ServiceUrl, cancellationToken) |         // todo: currently it's not possible to cross-compile for different OSes using Velopack.Build | ||||||
|  |         var targetOs = VelopackRuntimeInfo.SystemOs; | ||||||
|  | 
 | ||||||
|  |         await client.UploadLatestReleaseAssetsAsync(Channel, ReleaseDirectory, ServiceUrl, targetOs, cancellationToken) | ||||||
|             .ConfigureAwait(false); |             .ConfigureAwait(false); | ||||||
| 
 | 
 | ||||||
|         return true; |         return true; | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ | |||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageReference Include="Microsoft.Build.Utilities.Core" Version="17.9.5" /> |     <PackageReference Include="Microsoft.Build.Utilities.Core" Version="17.10.4" /> | ||||||
|     <PackageReference Include="Riok.Mapperly" Version="3.5.1" /> |     <PackageReference Include="Riok.Mapperly" Version="3.5.1" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,13 +4,13 @@ | |||||||
|     ".NETFramework,Version=v4.7.2": { |     ".NETFramework,Version=v4.7.2": { | ||||||
|       "Microsoft.Build.Utilities.Core": { |       "Microsoft.Build.Utilities.Core": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[17.9.5, )", |         "requested": "[17.10.4, )", | ||||||
|         "resolved": "17.9.5", |         "resolved": "17.10.4", | ||||||
|         "contentHash": "H2hpVdm7cX/uGJD1HOfab3RKgD5tlnvzQkFqvsrAqGHRi0sqb2w1+hRwERFm23witCjmERnqNgiQjYks6/ds8A==", |         "contentHash": "eEB/tcXkSV+nQgvoa/l53UPtn+KVtKZ8zBceDZsXVTrfE4fA+4+/olrx9W8n2tq4XiESsL9UuGJgCKzqBwQCoQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Build.Framework": "17.9.5", |           "Microsoft.Build.Framework": "17.10.4", | ||||||
|           "Microsoft.IO.Redist": "6.0.0", |           "Microsoft.IO.Redist": "6.0.0", | ||||||
|           "Microsoft.NET.StringTools": "17.9.5", |           "Microsoft.NET.StringTools": "17.10.4", | ||||||
|           "System.Collections.Immutable": "8.0.0", |           "System.Collections.Immutable": "8.0.0", | ||||||
|           "System.Configuration.ConfigurationManager": "8.0.0", |           "System.Configuration.ConfigurationManager": "8.0.0", | ||||||
|           "System.Memory": "4.5.5", |           "System.Memory": "4.5.5", | ||||||
| @@ -30,8 +30,8 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "Riok.Mapperly": { |       "Riok.Mapperly": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
| @@ -101,8 +101,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Build.Framework": { |       "Microsoft.Build.Framework": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "17.9.5", |         "resolved": "17.10.4", | ||||||
|         "contentHash": "CjRmqu9Wv2fyC1d7NKOuBDXcNMI8+GiXGM6izygB+skGGu4Vf0cBcoPq7AFqZCcMpn5DtZ+y7RpaLpB2qrzanQ==", |         "contentHash": "4qXCwNOXBR1dyCzuks9SwTwFJQO/xmf2wcMislotDWJu7MN/r3xDNoU8Ae5QmKIHPaLG1xmfDkYS7qBVzxmeKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "System.Runtime.CompilerServices.Unsafe": "6.0.0" |           "System.Runtime.CompilerServices.Unsafe": "6.0.0" | ||||||
|         } |         } | ||||||
| @@ -119,8 +119,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client": { |       "Microsoft.Identity.Client": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "3TDPEie+9t/NgBhoNifLFwM6nkypYUa8GUHfMnkYtovDDTQM8bsS2KEIP5tAh28BgfJZgof/KlCEKvGxz0H3Eg==", |         "contentHash": "zXPDXjTju1yOZQzi5PeNmQfBzlxv+VlCfgfZ70VqyFuhaPtFXwvbLSlwCOMoVALR/MzOR6xkiadoyVDmWDmqKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.IdentityModel.Abstractions": "6.35.0", |           "Microsoft.IdentityModel.Abstractions": "6.35.0", | ||||||
|           "System.Diagnostics.DiagnosticSource": "6.0.1" |           "System.Diagnostics.DiagnosticSource": "6.0.1" | ||||||
| @@ -128,27 +128,27 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Broker": { |       "Microsoft.Identity.Client.Broker": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "D5M7plyrXrQwhj/PILHLS7G3TqDg8rUO9lR+5awUSXueGxsdA2qHpeKiSR0sVfX6wxziWszOLDUR2vjFFKHPSg==", |         "contentHash": "+13pImNYfPGl04qZh3Rtq8wPfo55FScAUEio4ymPzeWJ/PE04jCsKH6ZIlrYbPCqe6t9u0lru1eZ+ho9Dsnvpw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "Microsoft.Identity.Client.NativeInterop": "0.16.0" |           "Microsoft.Identity.Client.NativeInterop": "0.16.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Extensions.Msal": { |       "Microsoft.Identity.Client.Extensions.Msal": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "tCX3c1bSkwH89AKjtB/oAOLlv661tOsEA8ULOCavOc6te/CKlqmMmFs8zMI58Qlv3XkbLOrswYr50IhDntiiPA==", |         "contentHash": "GCSBhzkr7/7DZDk2kYS9dfihB5ieM2CqwGAhtjuE6T1GtAqzuh62UJhMgO0dSx/PLgIYMzNHnh+pfHK8aWJQJQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "System.IO.FileSystem.AccessControl": "5.0.0", |           "System.IO.FileSystem.AccessControl": "5.0.0", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.5.0" |           "System.Security.Cryptography.ProtectedData": "4.5.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.NativeInterop": { |       "Microsoft.Identity.Client.NativeInterop": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "0.16.0", |         "resolved": "0.16.1", | ||||||
|         "contentHash": "riaIH9tT4kRG4xKFxx+wpuo+VAZVOr1ElbNubcvpFXb82gjEmZtJRzpdiS4tdRYTVh6CUKNz9AINAptActTtBQ==" |         "contentHash": "OvY/+/AHESi24f5zOCf9kL4HXwPxXVQ3A+tMQsJvFk2DmP+sc88FYWL49zlku5q0bvx5yFvBLNpHQeRT9a6A5g==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.IdentityModel.Abstractions": { |       "Microsoft.IdentityModel.Abstractions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -166,8 +166,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.NET.StringTools": { |       "Microsoft.NET.StringTools": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "17.9.5", |         "resolved": "17.10.4", | ||||||
|         "contentHash": "C/oPRnjcIZBRzcpl1V06R1eEMCxOGt6mIm+8ioyblELgJEXLM8XjUPuCwljMO52VetsHw54xMcYwU8UEeHEIEg==", |         "contentHash": "wyABaqY+IHCMMSTQmcc3Ca6vbmg5BaEPgicnEgpll+4xyWZWlkQqUwafweUd9VAhBb4jqplMl6voUHQ6yfdUcg==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "System.Memory": "4.5.5", |           "System.Memory": "4.5.5", | ||||||
|           "System.Runtime.CompilerServices.Unsafe": "6.0.0" |           "System.Runtime.CompilerServices.Unsafe": "6.0.0" | ||||||
| @@ -185,8 +185,29 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|  |       }, | ||||||
|  |       "SharpZipLib": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "1.4.2", | ||||||
|  |         "contentHash": "yjj+3zgz8zgXpiiC3ZdF/iyTBbz2fFvMxZFEBPUcwZjIvXOf37Ylm+K58hqMfIBt5JgU/Z2uoUS67JmTLe973A==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Memory": "4.5.4", | ||||||
|  |           "System.Threading.Tasks.Extensions": "4.5.2" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "SixLabors.ImageSharp": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "2.1.8", | ||||||
|  |         "contentHash": "ML1++vactR5xMW36wHA2nAX3vc6VpR62qYnpopQdLNGbP8BAJ/ckv1IrGIIFRUEQ0JS9EgWr1y1gEa/81f+HaA==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Buffers": "4.5.1", | ||||||
|  |           "System.Memory": "4.5.4", | ||||||
|  |           "System.Numerics.Vectors": "4.5.0", | ||||||
|  |           "System.Runtime.CompilerServices.Unsafe": "5.0.0", | ||||||
|  |           "System.Text.Encoding.CodePages": "5.0.0" | ||||||
|  |         } | ||||||
|       }, |       }, | ||||||
|       "System.Buffers": { |       "System.Buffers": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -261,15 +282,6 @@ | |||||||
|         "resolved": "4.5.0", |         "resolved": "4.5.0", | ||||||
|         "contentHash": "QQTlPTl06J/iiDbJCiepZ4H//BVraReU4O4EoRw1U02H5TLUIT7xn3GnDp9AXPSlJUDyFs4uWjWafNX6WrAojQ==" |         "contentHash": "QQTlPTl06J/iiDbJCiepZ4H//BVraReU4O4EoRw1U02H5TLUIT7xn3GnDp9AXPSlJUDyFs4uWjWafNX6WrAojQ==" | ||||||
|       }, |       }, | ||||||
|       "System.Reflection.Metadata": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "8.0.0", |  | ||||||
|         "contentHash": "ptvgrFh7PvWI8bcVqG5rsA/weWM09EnthFHR5SCnS6IN+P4mj6rE1lBDC4U8HL9/57htKAqy4KQ3bBj84cfYyQ==", |  | ||||||
|         "dependencies": { |  | ||||||
|           "System.Collections.Immutable": "8.0.0", |  | ||||||
|           "System.Memory": "4.5.5" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "System.Runtime": { |       "System.Runtime": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.3.0", |         "resolved": "4.3.0", | ||||||
| @@ -328,6 +340,14 @@ | |||||||
|         "resolved": "5.0.0", |         "resolved": "5.0.0", | ||||||
|         "contentHash": "t0MGLukB5WAVU9bO3MGzvlGnyJPgUlcwerXn1kzBRjwLKixT96XV0Uza41W49gVd8zEMFu9vQEFlv0IOrytICA==" |         "contentHash": "t0MGLukB5WAVU9bO3MGzvlGnyJPgUlcwerXn1kzBRjwLKixT96XV0Uza41W49gVd8zEMFu9vQEFlv0IOrytICA==" | ||||||
|       }, |       }, | ||||||
|  |       "System.Text.Encoding.CodePages": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "5.0.0", | ||||||
|  |         "contentHash": "NyscU59xX6Uo91qvhOs2Ccho3AR2TnZPomo1Z0K6YpyztBPM/A5VbkzOO19sy3A3i1TtEnTxA7bCe3Us+r5MWg==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Runtime.CompilerServices.Unsafe": "5.0.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "System.Text.Encodings.Web": { |       "System.Text.Encodings.Web": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.0.0", |         "resolved": "6.0.0", | ||||||
| @@ -371,31 +391,32 @@ | |||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Extensions.Logging.Abstractions": "[2.2.0, )", |           "Microsoft.Extensions.Logging.Abstractions": "[2.2.0, )", | ||||||
|           "Newtonsoft.Json": "[13.0.1, )", |           "Newtonsoft.Json": "[13.0.1, )", | ||||||
|           "NuGet.Versioning": "[6.9.1, )" |           "NuGet.Versioning": "[6.10.0, )" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "velopack.icolib": { | ||||||
|  |         "type": "Project", | ||||||
|  |         "dependencies": { | ||||||
|  |           "SixLabors.ImageSharp": "[2.1.8, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.packaging": { |       "velopack.packaging": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Markdig": "[0.37.0, )", |           "Markdig": "[0.37.0, )", | ||||||
|           "Microsoft.Identity.Client": "[4.61.0, )", |           "Microsoft.Identity.Client": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Broker": "[4.61.0, )", |           "Microsoft.Identity.Client.Broker": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.0, )", |           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.2, )", | ||||||
|           "System.Linq.Async": "[6.0.1, )", |           "System.Linq.Async": "[6.0.1, )", | ||||||
|           "System.Net.Http": "[4.3.4, )", |           "System.Net.Http": "[4.3.4, )", | ||||||
|           "Velopack": "[1.0.0, )" |           "Velopack": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.packaging.hostmodel": { |  | ||||||
|         "type": "Project", |  | ||||||
|         "dependencies": { |  | ||||||
|           "System.Reflection.Metadata": "[8.0.0, )" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "velopack.packaging.unix": { |       "velopack.packaging.unix": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "ELFSharp": "[2.17.3, )", |           "ELFSharp": "[2.17.3, )", | ||||||
|  |           "SharpZipLib": "[1.4.2, )", | ||||||
|           "Velopack.Packaging": "[1.0.0, )" |           "Velopack.Packaging": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
| @@ -404,20 +425,20 @@ | |||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "AsmResolver.DotNet": "[5.5.1, )", |           "AsmResolver.DotNet": "[5.5.1, )", | ||||||
|           "AsmResolver.PE.Win32Resources": "[5.5.1, )", |           "AsmResolver.PE.Win32Resources": "[5.5.1, )", | ||||||
|           "Velopack.Packaging": "[1.0.0, )", |           "Velopack.IcoLib": "[1.1.1, )", | ||||||
|           "Velopack.Packaging.HostModel": "[1.0.0, )" |           "Velopack.Packaging": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "net6.0": { |     "net6.0": { | ||||||
|       "Microsoft.Build.Utilities.Core": { |       "Microsoft.Build.Utilities.Core": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[17.9.5, )", |         "requested": "[17.10.4, )", | ||||||
|         "resolved": "17.9.5", |         "resolved": "17.10.4", | ||||||
|         "contentHash": "H2hpVdm7cX/uGJD1HOfab3RKgD5tlnvzQkFqvsrAqGHRi0sqb2w1+hRwERFm23witCjmERnqNgiQjYks6/ds8A==", |         "contentHash": "eEB/tcXkSV+nQgvoa/l53UPtn+KVtKZ8zBceDZsXVTrfE4fA+4+/olrx9W8n2tq4XiESsL9UuGJgCKzqBwQCoQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Build.Framework": "17.9.5", |           "Microsoft.Build.Framework": "17.10.4", | ||||||
|           "Microsoft.NET.StringTools": "17.9.5", |           "Microsoft.NET.StringTools": "17.10.4", | ||||||
|           "Microsoft.Win32.Registry": "5.0.0", |           "Microsoft.Win32.Registry": "5.0.0", | ||||||
|           "System.Collections.Immutable": "8.0.0", |           "System.Collections.Immutable": "8.0.0", | ||||||
|           "System.Configuration.ConfigurationManager": "8.0.0", |           "System.Configuration.ConfigurationManager": "8.0.0", | ||||||
| @@ -440,8 +461,8 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "Riok.Mapperly": { |       "Riok.Mapperly": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
| @@ -505,8 +526,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Build.Framework": { |       "Microsoft.Build.Framework": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "17.9.5", |         "resolved": "17.10.4", | ||||||
|         "contentHash": "CjRmqu9Wv2fyC1d7NKOuBDXcNMI8+GiXGM6izygB+skGGu4Vf0cBcoPq7AFqZCcMpn5DtZ+y7RpaLpB2qrzanQ==", |         "contentHash": "4qXCwNOXBR1dyCzuks9SwTwFJQO/xmf2wcMislotDWJu7MN/r3xDNoU8Ae5QmKIHPaLG1xmfDkYS7qBVzxmeKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Win32.Registry": "5.0.0", |           "Microsoft.Win32.Registry": "5.0.0", | ||||||
|           "System.Memory": "4.5.5", |           "System.Memory": "4.5.5", | ||||||
| @@ -526,8 +547,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client": { |       "Microsoft.Identity.Client": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "3TDPEie+9t/NgBhoNifLFwM6nkypYUa8GUHfMnkYtovDDTQM8bsS2KEIP5tAh28BgfJZgof/KlCEKvGxz0H3Eg==", |         "contentHash": "zXPDXjTju1yOZQzi5PeNmQfBzlxv+VlCfgfZ70VqyFuhaPtFXwvbLSlwCOMoVALR/MzOR6xkiadoyVDmWDmqKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.IdentityModel.Abstractions": "6.35.0", |           "Microsoft.IdentityModel.Abstractions": "6.35.0", | ||||||
|           "System.Diagnostics.DiagnosticSource": "6.0.1" |           "System.Diagnostics.DiagnosticSource": "6.0.1" | ||||||
| @@ -535,26 +556,26 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Broker": { |       "Microsoft.Identity.Client.Broker": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "D5M7plyrXrQwhj/PILHLS7G3TqDg8rUO9lR+5awUSXueGxsdA2qHpeKiSR0sVfX6wxziWszOLDUR2vjFFKHPSg==", |         "contentHash": "+13pImNYfPGl04qZh3Rtq8wPfo55FScAUEio4ymPzeWJ/PE04jCsKH6ZIlrYbPCqe6t9u0lru1eZ+ho9Dsnvpw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "Microsoft.Identity.Client.NativeInterop": "0.16.0" |           "Microsoft.Identity.Client.NativeInterop": "0.16.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Extensions.Msal": { |       "Microsoft.Identity.Client.Extensions.Msal": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "tCX3c1bSkwH89AKjtB/oAOLlv661tOsEA8ULOCavOc6te/CKlqmMmFs8zMI58Qlv3XkbLOrswYr50IhDntiiPA==", |         "contentHash": "GCSBhzkr7/7DZDk2kYS9dfihB5ieM2CqwGAhtjuE6T1GtAqzuh62UJhMgO0dSx/PLgIYMzNHnh+pfHK8aWJQJQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.5.0" |           "System.Security.Cryptography.ProtectedData": "4.5.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.NativeInterop": { |       "Microsoft.Identity.Client.NativeInterop": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "0.16.0", |         "resolved": "0.16.1", | ||||||
|         "contentHash": "riaIH9tT4kRG4xKFxx+wpuo+VAZVOr1ElbNubcvpFXb82gjEmZtJRzpdiS4tdRYTVh6CUKNz9AINAptActTtBQ==" |         "contentHash": "OvY/+/AHESi24f5zOCf9kL4HXwPxXVQ3A+tMQsJvFk2DmP+sc88FYWL49zlku5q0bvx5yFvBLNpHQeRT9a6A5g==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.IdentityModel.Abstractions": { |       "Microsoft.IdentityModel.Abstractions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -563,8 +584,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.NET.StringTools": { |       "Microsoft.NET.StringTools": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "17.9.5", |         "resolved": "17.10.4", | ||||||
|         "contentHash": "C/oPRnjcIZBRzcpl1V06R1eEMCxOGt6mIm+8ioyblELgJEXLM8XjUPuCwljMO52VetsHw54xMcYwU8UEeHEIEg==", |         "contentHash": "wyABaqY+IHCMMSTQmcc3Ca6vbmg5BaEPgicnEgpll+4xyWZWlkQqUwafweUd9VAhBb4jqplMl6voUHQ6yfdUcg==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "System.Memory": "4.5.5", |           "System.Memory": "4.5.5", | ||||||
|           "System.Runtime.CompilerServices.Unsafe": "6.0.0" |           "System.Runtime.CompilerServices.Unsafe": "6.0.0" | ||||||
| @@ -596,8 +617,8 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "runtime.debian.8-x64.runtime.native.System.Security.Cryptography.OpenSsl": { |       "runtime.debian.8-x64.runtime.native.System.Security.Cryptography.OpenSsl": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -697,6 +718,20 @@ | |||||||
|         "resolved": "4.3.2", |         "resolved": "4.3.2", | ||||||
|         "contentHash": "leXiwfiIkW7Gmn7cgnNcdtNAU70SjmKW3jxGj1iKHOvdn0zRWsgv/l2OJUO5zdGdiv2VRFnAsxxhDgMzofPdWg==" |         "contentHash": "leXiwfiIkW7Gmn7cgnNcdtNAU70SjmKW3jxGj1iKHOvdn0zRWsgv/l2OJUO5zdGdiv2VRFnAsxxhDgMzofPdWg==" | ||||||
|       }, |       }, | ||||||
|  |       "SharpZipLib": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "1.4.2", | ||||||
|  |         "contentHash": "yjj+3zgz8zgXpiiC3ZdF/iyTBbz2fFvMxZFEBPUcwZjIvXOf37Ylm+K58hqMfIBt5JgU/Z2uoUS67JmTLe973A==" | ||||||
|  |       }, | ||||||
|  |       "SixLabors.ImageSharp": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "2.1.8", | ||||||
|  |         "contentHash": "ML1++vactR5xMW36wHA2nAX3vc6VpR62qYnpopQdLNGbP8BAJ/ckv1IrGIIFRUEQ0JS9EgWr1y1gEa/81f+HaA==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Runtime.CompilerServices.Unsafe": "5.0.0", | ||||||
|  |           "System.Text.Encoding.CodePages": "5.0.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "System.Collections": { |       "System.Collections": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.3.0", |         "resolved": "4.3.0", | ||||||
| @@ -1219,28 +1254,32 @@ | |||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", |           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", | ||||||
|           "NuGet.Versioning": "[6.9.1, )" |           "NuGet.Versioning": "[6.10.0, )" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "velopack.icolib": { | ||||||
|  |         "type": "Project", | ||||||
|  |         "dependencies": { | ||||||
|  |           "SixLabors.ImageSharp": "[2.1.8, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.packaging": { |       "velopack.packaging": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Markdig": "[0.37.0, )", |           "Markdig": "[0.37.0, )", | ||||||
|           "Microsoft.Identity.Client": "[4.61.0, )", |           "Microsoft.Identity.Client": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Broker": "[4.61.0, )", |           "Microsoft.Identity.Client.Broker": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.0, )", |           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.2, )", | ||||||
|           "System.Linq.Async": "[6.0.1, )", |           "System.Linq.Async": "[6.0.1, )", | ||||||
|           "System.Net.Http": "[4.3.4, )", |           "System.Net.Http": "[4.3.4, )", | ||||||
|           "Velopack": "[1.0.0, )" |           "Velopack": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.packaging.hostmodel": { |  | ||||||
|         "type": "Project" |  | ||||||
|       }, |  | ||||||
|       "velopack.packaging.unix": { |       "velopack.packaging.unix": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "ELFSharp": "[2.17.3, )", |           "ELFSharp": "[2.17.3, )", | ||||||
|  |           "SharpZipLib": "[1.4.2, )", | ||||||
|           "Velopack.Packaging": "[1.0.0, )" |           "Velopack.Packaging": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
| @@ -1249,8 +1288,8 @@ | |||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "AsmResolver.DotNet": "[5.5.1, )", |           "AsmResolver.DotNet": "[5.5.1, )", | ||||||
|           "AsmResolver.PE.Win32Resources": "[5.5.1, )", |           "AsmResolver.PE.Win32Resources": "[5.5.1, )", | ||||||
|           "Velopack.Packaging": "[1.0.0, )", |           "Velopack.IcoLib": "[1.1.1, )", | ||||||
|           "Velopack.Packaging.HostModel": "[1.0.0, )" |           "Velopack.Packaging": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -56,7 +56,7 @@ public class GitHubRepository : SourceRepository<GitHubDownloadOptions, GithubSo | |||||||
|     public async Task UploadMissingAssetsAsync(GitHubUploadOptions options) |     public async Task UploadMissingAssetsAsync(GitHubUploadOptions options) | ||||||
|     { |     { | ||||||
|         var (repoOwner, repoName) = GetOwnerAndRepo(options.RepoUrl); |         var (repoOwner, repoName) = GetOwnerAndRepo(options.RepoUrl); | ||||||
|         var helper = new ReleaseEntryHelper(options.ReleaseDir.FullName, options.Channel, Log); |         var helper = new ReleaseEntryHelper(options.ReleaseDir.FullName, options.Channel, Log, options.TargetOs); | ||||||
|         var build = BuildAssets.Read(options.ReleaseDir.FullName, options.Channel); |         var build = BuildAssets.Read(options.ReleaseDir.FullName, options.Channel); | ||||||
|         var latest = helper.GetLatestFullRelease(); |         var latest = helper.GetLatestFullRelease(); | ||||||
|         var latestPath = Path.Combine(options.ReleaseDir.FullName, latest.FileName); |         var latestPath = Path.Combine(options.ReleaseDir.FullName, latest.FileName); | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ | |||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageReference Include="AWSSDK.S3" Version="3.7.308" /> |     <PackageReference Include="AWSSDK.S3" Version="3.7.308.8" /> | ||||||
|     <PackageReference Include="Azure.Storage.Blobs" Version="12.20.0" /> |     <PackageReference Include="Azure.Storage.Blobs" Version="12.20.0" /> | ||||||
|     <PackageReference Include="Octokit" Version="11.0.1" /> |     <PackageReference Include="Octokit" Version="11.0.1" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|   | |||||||
| @@ -7,7 +7,14 @@ namespace Velopack.Deployment; | |||||||
| 
 | 
 | ||||||
| public class RepositoryOptions : IOutputOptions | public class RepositoryOptions : IOutputOptions | ||||||
| { | { | ||||||
|     public string Channel { get; set; } = ReleaseEntryHelper.GetDefaultChannel(); |     private string _channel; | ||||||
|  | 
 | ||||||
|  |     public RuntimeOs TargetOs { get; set; } | ||||||
|  | 
 | ||||||
|  |     public string Channel { | ||||||
|  |         get => _channel ?? ReleaseEntryHelper.GetDefaultChannel(TargetOs); | ||||||
|  |         set => _channel = value; | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     public DirectoryInfo ReleaseDir { get; set; } |     public DirectoryInfo ReleaseDir { get; set; } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -4,11 +4,11 @@ | |||||||
|     "net6.0": { |     "net6.0": { | ||||||
|       "AWSSDK.S3": { |       "AWSSDK.S3": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.7.308, )", |         "requested": "[3.7.308.8, )", | ||||||
|         "resolved": "3.7.308", |         "resolved": "3.7.308.8", | ||||||
|         "contentHash": "U4LWi1yTKVK6IFHWQc4anKPJKXj9dPN+dT1oXto4ndXXlr+DVzm9dULkquyXBjAHlr6jf339ojuYlE6t9Mk6pQ==", |         "contentHash": "lUQgJsj9n/sJo1CRQFsiB2Gqh3rC21I4e5CWR1b+oYBnc9m8LIsGewctX6YQRzGzuv7Im7iIZo7XoirmYhhqEA==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "AWSSDK.Core": "[3.7.304.1, 4.0.0)" |           "AWSSDK.Core": "[3.7.304.10, 4.0.0)" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Azure.Storage.Blobs": { |       "Azure.Storage.Blobs": { | ||||||
| @@ -34,8 +34,8 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "Octokit": { |       "Octokit": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
| @@ -45,8 +45,8 @@ | |||||||
|       }, |       }, | ||||||
|       "AWSSDK.Core": { |       "AWSSDK.Core": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "3.7.304.1", |         "resolved": "3.7.304.10", | ||||||
|         "contentHash": "O/gyE5ptF5zEc3QDk3JI3+AOgBfRfmg6eLH5z3x7hUPmV1Wxu0V4Fm86FSyT6czRviMmQGY0q0SVMbWRbP4vDA==" |         "contentHash": "0CXnPzoM+KYXODm2bvRW8eYs7ic2VLop45sphL8FBfvxHfBK/3OJgpxEE2InSdWS1Iby+6KXpz3NplJJY5+Y1A==" | ||||||
|       }, |       }, | ||||||
|       "Azure.Core": { |       "Azure.Core": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -94,8 +94,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client": { |       "Microsoft.Identity.Client": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "3TDPEie+9t/NgBhoNifLFwM6nkypYUa8GUHfMnkYtovDDTQM8bsS2KEIP5tAh28BgfJZgof/KlCEKvGxz0H3Eg==", |         "contentHash": "zXPDXjTju1yOZQzi5PeNmQfBzlxv+VlCfgfZ70VqyFuhaPtFXwvbLSlwCOMoVALR/MzOR6xkiadoyVDmWDmqKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.IdentityModel.Abstractions": "6.35.0", |           "Microsoft.IdentityModel.Abstractions": "6.35.0", | ||||||
|           "System.Diagnostics.DiagnosticSource": "6.0.1" |           "System.Diagnostics.DiagnosticSource": "6.0.1" | ||||||
| @@ -103,26 +103,26 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Broker": { |       "Microsoft.Identity.Client.Broker": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "D5M7plyrXrQwhj/PILHLS7G3TqDg8rUO9lR+5awUSXueGxsdA2qHpeKiSR0sVfX6wxziWszOLDUR2vjFFKHPSg==", |         "contentHash": "+13pImNYfPGl04qZh3Rtq8wPfo55FScAUEio4ymPzeWJ/PE04jCsKH6ZIlrYbPCqe6t9u0lru1eZ+ho9Dsnvpw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "Microsoft.Identity.Client.NativeInterop": "0.16.0" |           "Microsoft.Identity.Client.NativeInterop": "0.16.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Extensions.Msal": { |       "Microsoft.Identity.Client.Extensions.Msal": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "tCX3c1bSkwH89AKjtB/oAOLlv661tOsEA8ULOCavOc6te/CKlqmMmFs8zMI58Qlv3XkbLOrswYr50IhDntiiPA==", |         "contentHash": "GCSBhzkr7/7DZDk2kYS9dfihB5ieM2CqwGAhtjuE6T1GtAqzuh62UJhMgO0dSx/PLgIYMzNHnh+pfHK8aWJQJQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.5.0" |           "System.Security.Cryptography.ProtectedData": "4.5.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.NativeInterop": { |       "Microsoft.Identity.Client.NativeInterop": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "0.16.0", |         "resolved": "0.16.1", | ||||||
|         "contentHash": "riaIH9tT4kRG4xKFxx+wpuo+VAZVOr1ElbNubcvpFXb82gjEmZtJRzpdiS4tdRYTVh6CUKNz9AINAptActTtBQ==" |         "contentHash": "OvY/+/AHESi24f5zOCf9kL4HXwPxXVQ3A+tMQsJvFk2DmP+sc88FYWL49zlku5q0bvx5yFvBLNpHQeRT9a6A5g==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.IdentityModel.Abstractions": { |       "Microsoft.IdentityModel.Abstractions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -146,8 +146,8 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "runtime.debian.8-x64.runtime.native.System.Security.Cryptography.OpenSsl": { |       "runtime.debian.8-x64.runtime.native.System.Security.Cryptography.OpenSsl": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -752,16 +752,16 @@ | |||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", |           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", | ||||||
|           "NuGet.Versioning": "[6.9.1, )" |           "NuGet.Versioning": "[6.10.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.packaging": { |       "velopack.packaging": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Markdig": "[0.37.0, )", |           "Markdig": "[0.37.0, )", | ||||||
|           "Microsoft.Identity.Client": "[4.61.0, )", |           "Microsoft.Identity.Client": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Broker": "[4.61.0, )", |           "Microsoft.Identity.Client.Broker": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.0, )", |           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.2, )", | ||||||
|           "System.Linq.Async": "[6.0.1, )", |           "System.Linq.Async": "[6.0.1, )", | ||||||
|           "System.Net.Http": "[4.3.4, )", |           "System.Net.Http": "[4.3.4, )", | ||||||
|           "Velopack": "[1.0.0, )" |           "Velopack": "[1.0.0, )" | ||||||
|   | |||||||
							
								
								
									
										61
									
								
								src/Velopack.IcoLib/Binary/BitReader.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								src/Velopack.IcoLib/Binary/BitReader.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | |||||||
|  | using System; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Binary | ||||||
|  | { | ||||||
|  |     internal ref struct BitReader | ||||||
|  |     { | ||||||
|  |         private ByteReader reader; | ||||||
|  |         private byte currentByte; | ||||||
|  |         private uint currentBitsRemaining; | ||||||
|  | 
 | ||||||
|  |         public BitReader(ByteReader reader) | ||||||
|  |         { | ||||||
|  |             this.reader = reader; | ||||||
|  |             this.currentByte = 0; | ||||||
|  |             this.currentBitsRemaining = 0; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public uint NextBit1() | ||||||
|  |         { | ||||||
|  |             return NextBit(1); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public uint NextBit2() | ||||||
|  |         { | ||||||
|  |             return NextBit(2); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public uint NextBit4() | ||||||
|  |         { | ||||||
|  |             return NextBit(4); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public uint NextBit(uint n) | ||||||
|  |         { | ||||||
|  |             EnsureBits(n); | ||||||
|  | 
 | ||||||
|  |             var shift = (byte)(currentBitsRemaining - n); | ||||||
|  |             var mask = (1u << (int)n) - 1; | ||||||
|  |             var result = (uint)(currentByte >> shift) & mask; | ||||||
|  | 
 | ||||||
|  |             currentBitsRemaining -= n; | ||||||
|  |             return result; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private void EnsureBits(uint numBitsNeeded) | ||||||
|  |         { | ||||||
|  |             if (0 != (currentBitsRemaining % numBitsNeeded)) | ||||||
|  |             { | ||||||
|  |                 throw new Exception("Cannot read unaligned value"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (currentBitsRemaining >= numBitsNeeded) | ||||||
|  |             { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             currentByte = reader.NextUint8(); | ||||||
|  |             currentBitsRemaining = 8; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										63
									
								
								src/Velopack.IcoLib/Binary/BitWriter.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								src/Velopack.IcoLib/Binary/BitWriter.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | |||||||
|  | using System; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Binary | ||||||
|  | { | ||||||
|  |     internal class BitWriter | ||||||
|  |     { | ||||||
|  |         private ByteWriter writer; | ||||||
|  |         private uint currentBitsRemaining; | ||||||
|  | 
 | ||||||
|  |         public BitWriter(ByteWriter writer) | ||||||
|  |         { | ||||||
|  |             this.writer = writer; | ||||||
|  |             this.currentBitsRemaining = 0; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public void AddBit1(byte value) | ||||||
|  |         { | ||||||
|  |             AddBits(1, value); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public void AddBit2(byte value) | ||||||
|  |         { | ||||||
|  |             AddBits(2, value); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public void AddBit4(byte value) | ||||||
|  |         { | ||||||
|  |             AddBits(4, value); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public void AddBits(uint numBits, byte value) | ||||||
|  |         { | ||||||
|  |             EnsureBits(numBits); | ||||||
|  | 
 | ||||||
|  |             if (value != 0) | ||||||
|  |             { | ||||||
|  |                 var shift = (int)(currentBitsRemaining - numBits); | ||||||
|  | 
 | ||||||
|  |                 var b = writer.Data[writer.Data.Count - 1]; | ||||||
|  |                 b = (byte)(b | (value << shift)); | ||||||
|  |                 writer.Data[writer.Data.Count - 1] = b; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             currentBitsRemaining -= numBits; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private void EnsureBits(uint numBitsNeeded) | ||||||
|  |         { | ||||||
|  |             if (0 != (currentBitsRemaining % numBitsNeeded)) | ||||||
|  |             { | ||||||
|  |                 throw new Exception("Cannot write unaligned value"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (currentBitsRemaining >= numBitsNeeded) | ||||||
|  |             { | ||||||
|  |                 return; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             writer.AddUint8(0); | ||||||
|  |             currentBitsRemaining = 8; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										39
									
								
								src/Velopack.IcoLib/Binary/ByteOrderConverter.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								src/Velopack.IcoLib/Binary/ByteOrderConverter.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Text; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Binary | ||||||
|  | { | ||||||
|  |     public enum ByteOrder | ||||||
|  |     { | ||||||
|  |         LittleEndian, | ||||||
|  |         BigEndian, | ||||||
|  |         NetworkEndian = BigEndian, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static class ByteOrderConverter | ||||||
|  |     { | ||||||
|  |         private static bool NeedsSwap(ByteOrder endian) => BitConverter.IsLittleEndian != (endian == ByteOrder.LittleEndian); | ||||||
|  | 
 | ||||||
|  |         public static ushort To(ByteOrder endian, ushort value) => NeedsSwap(endian) ? Swap(value) : value; | ||||||
|  | 
 | ||||||
|  |         public static uint To(ByteOrder endian, uint value) => NeedsSwap(endian) ? Swap(value) : value; | ||||||
|  | 
 | ||||||
|  |         public static ulong To(ByteOrder endian, ulong value) => NeedsSwap(endian) ? Swap(value) : value; | ||||||
|  | 
 | ||||||
|  |         public static int To(ByteOrder endian, int value) => NeedsSwap(endian) ? (int)Swap((uint)value) : value; | ||||||
|  | 
 | ||||||
|  |         public static ushort Swap(ushort value) => | ||||||
|  |                 (ushort)((value & 0xffu) << 8 | (value & 0xff00u) >> 8); | ||||||
|  | 
 | ||||||
|  |         private static uint Swap(uint value) => | ||||||
|  |                 (value & 0x000000FFU) << 24 | (value & 0x0000FF00U) << 8 | | ||||||
|  |                 (value & 0x00FF0000U) >> 8  | (value & 0xFF000000U) >> 24; | ||||||
|  | 
 | ||||||
|  |         private static ulong Swap(ulong value) => | ||||||
|  |                 (value & 0x00000000000000FFUL) << 56 | (value & 0x000000000000FF00UL) << 40 | | ||||||
|  |                 (value & 0x0000000000FF0000UL) << 24 | (value & 0x00000000FF000000UL) << 8  | | ||||||
|  |                 (value & 0x000000FF00000000UL) >> 8  | (value & 0x0000FF0000000000UL) >> 24 | | ||||||
|  |                 (value & 0x00FF000000000000UL) >> 40 | (value & 0xFF00000000000000UL) >> 56; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										53
									
								
								src/Velopack.IcoLib/Binary/ByteReader.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								src/Velopack.IcoLib/Binary/ByteReader.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | |||||||
|  | using System; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Binary | ||||||
|  | { | ||||||
|  |     public class ByteReader | ||||||
|  |     { | ||||||
|  |         public Memory<byte> Data { get; } | ||||||
|  |         public ByteOrder Endianness { get; } | ||||||
|  |         public int SeekOffset { get; set; } | ||||||
|  | 
 | ||||||
|  |         public ByteReader(Memory<byte> data, ByteOrder endianness) | ||||||
|  |         { | ||||||
|  |             Data = data; | ||||||
|  |             Endianness = endianness; | ||||||
|  |             SeekOffset = 0; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public byte NextUint8() | ||||||
|  |         { | ||||||
|  |             return Data.Span[SeekOffset++]; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public ushort NextUint16() | ||||||
|  |         { | ||||||
|  |             var result = BitConverter.ToUInt16(Data.Span.Slice(SeekOffset, 2).ToArray(), 0); | ||||||
|  |             SeekOffset += 2; | ||||||
|  |             return ByteOrderConverter.To(Endianness, result); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public uint NextUint32() | ||||||
|  |         { | ||||||
|  |             var result = BitConverter.ToUInt32(Data.Span.Slice(SeekOffset, 4).ToArray(), 0); | ||||||
|  |             SeekOffset += 4; | ||||||
|  |             return ByteOrderConverter.To(Endianness, result); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public int NextInt32() | ||||||
|  |         { | ||||||
|  |             var result = BitConverter.ToInt32(Data.Span.Slice(SeekOffset, 4).ToArray(), 0); | ||||||
|  |             SeekOffset += 4; | ||||||
|  |             return ByteOrderConverter.To(Endianness, result); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public ulong NextUint64() | ||||||
|  |         { | ||||||
|  |             var result = BitConverter.ToUInt64(Data.Span.Slice(SeekOffset, 8).ToArray(), 0); | ||||||
|  |             SeekOffset += 8; | ||||||
|  |             return ByteOrderConverter.To(Endianness, result); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public bool IsEof => SeekOffset == Data.Length; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										70
									
								
								src/Velopack.IcoLib/Binary/ByteWriter.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								src/Velopack.IcoLib/Binary/ByteWriter.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Binary | ||||||
|  | { | ||||||
|  |     internal class ByteWriter | ||||||
|  |     { | ||||||
|  |         public List<byte> Data { get; } = new List<byte>(); | ||||||
|  |         public ByteOrder Endianness { get; } | ||||||
|  |         public int SeekOffset { get; set; } = 0; | ||||||
|  | 
 | ||||||
|  |         public ByteWriter(ByteOrder endianness) | ||||||
|  |         { | ||||||
|  |             Endianness = endianness; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public void AddUint8(byte value) | ||||||
|  |         { | ||||||
|  |             EnsureCapacity(1); | ||||||
|  |             Data[SeekOffset++] = value; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public void AddUint16(ushort value) | ||||||
|  |         { | ||||||
|  |             EnsureCapacity(2); | ||||||
|  |             value = ByteOrderConverter.To(Endianness, value); | ||||||
|  |             foreach (var b in BitConverter.GetBytes(value)) | ||||||
|  |             { | ||||||
|  |                 Data[SeekOffset++] = b; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public void AddUint32(uint value) | ||||||
|  |         { | ||||||
|  |             EnsureCapacity(4); | ||||||
|  |             value = ByteOrderConverter.To(Endianness, value); | ||||||
|  |             foreach (var b in BitConverter.GetBytes(value)) | ||||||
|  |             { | ||||||
|  |                 Data[SeekOffset++] = b; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public void AddInt32(int value) | ||||||
|  |         { | ||||||
|  |             EnsureCapacity(4); | ||||||
|  |             value = ByteOrderConverter.To(Endianness, value); | ||||||
|  |             foreach (var b in BitConverter.GetBytes(value)) | ||||||
|  |             { | ||||||
|  |                 Data[SeekOffset++] = b; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private void EnsureCapacity(int additionalBytesNeeded) | ||||||
|  |         { | ||||||
|  |             while (Data.Count < SeekOffset + additionalBytesNeeded) | ||||||
|  |             { | ||||||
|  |                 Data.Add(0); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         internal void AddBlob(byte[] blob) | ||||||
|  |         { | ||||||
|  |             EnsureCapacity(blob.Length); | ||||||
|  |             foreach (var b in blob) | ||||||
|  |             { | ||||||
|  |                 Data[SeekOffset++] = b; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										302
									
								
								src/Velopack.IcoLib/Codecs/BmpDecoder.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										302
									
								
								src/Velopack.IcoLib/Codecs/BmpDecoder.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,302 @@ | |||||||
|  | using Ico.Binary; | ||||||
|  | using Ico.Model; | ||||||
|  | using Ico.Validation; | ||||||
|  | using SixLabors.ImageSharp; | ||||||
|  | using SixLabors.ImageSharp.PixelFormats; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Codecs | ||||||
|  | { | ||||||
|  |     public static class BmpDecoder | ||||||
|  |     { | ||||||
|  |         public static void DoBitmapEntry(ByteReader reader, ParseContext context, IcoFrame source) | ||||||
|  |         { | ||||||
|  |             var biSize = reader.NextUint32(); | ||||||
|  |             var biWidth = reader.NextInt32(); | ||||||
|  |             var biHeight = reader.NextInt32(); | ||||||
|  |             var biPlanes = reader.NextUint16(); | ||||||
|  |             var biBitCount = reader.NextUint16(); | ||||||
|  |             var biCompression = reader.NextUint32(); | ||||||
|  |             var biSizeImage = reader.NextUint32(); | ||||||
|  |             var biXPelsPerMeter = reader.NextInt32(); | ||||||
|  |             var biYPelsPerMeter = reader.NextInt32(); | ||||||
|  |             var biClrUsed = reader.NextUint32(); | ||||||
|  |             var biClrImportant = reader.NextUint32(); | ||||||
|  | 
 | ||||||
|  |             if (biSize != FileFormatConstants._bitmapInfoHeaderSize) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidIcoFileException(IcoErrorCode.InvalidBitapInfoHeader_ciSize, $"BITMAPINFOHEADER.ciSize should be {FileFormatConstants._bitmapInfoHeaderSize}, was {biSize}.", context); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (biXPelsPerMeter != 0) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.InvalidBitapInfoHeader_biXPelsPerMeter, $"BITMAPINFOHEADER.biXPelsPerMeter should be 0, was {biXPelsPerMeter}.", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (biYPelsPerMeter != 0) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.InvalidBitapInfoHeader_biYPelsPerMeter, $"BITMAPINFOHEADER.biYPelsPerMeter should be 0, was {biYPelsPerMeter}.", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (biCompression == FileFormatConstants.BI_BITFIELDS) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidIcoFileException(IcoErrorCode.BitfieldCompressionNotSupported, $"This tool does not implement icon bitmaps that use BI_BITFIELDS compression.  (The .ICO file may be okay, although it is certainly unusual.)", context); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (biCompression != FileFormatConstants.BI_RGB) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidIcoFileException(IcoErrorCode.BitmapCompressionNotSupported, $"BITMAPINFOHEADER.biCompression is unknown value ({biCompression}).", context); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (biHeight != source.Encoding.ClaimedHeight * 2) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.MismatchedHeight, $"BITMAPINFOHEADER.biHeight is not exactly double ICONDIRECTORY.bHeight ({biHeight} != 2 * {source.Encoding.ClaimedHeight}).", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (biWidth != source.Encoding.ClaimedWidth) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.MismatchedWidth, $"BITMAPINFOHEADER.biWidth is not exactly equal to ICONDIRECTORY.bWidth ({biWidth} != 2 * {source.Encoding.ClaimedWidth}).", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var height = biHeight / 2; | ||||||
|  |             var width = biWidth; | ||||||
|  | 
 | ||||||
|  |             source.Encoding.ActualHeight = (uint)height; | ||||||
|  |             source.Encoding.ActualWidth = (uint)width; | ||||||
|  |             source.Encoding.ActualBitDepth = biBitCount; | ||||||
|  |             source.Encoding.Type = IcoEncodingType.Bitmap; | ||||||
|  |             source.CookedData = new Image<Rgba32>(width, height); | ||||||
|  | 
 | ||||||
|  |             switch (biBitCount) | ||||||
|  |             { | ||||||
|  |                 case 1: | ||||||
|  |                 case 2: | ||||||
|  |                 case 4: | ||||||
|  |                 case 8: | ||||||
|  |                     ReadIndexedBitmap(reader, context, biBitCount, biClrUsed, height, width, source); | ||||||
|  |                     break; | ||||||
|  |                 case 16: | ||||||
|  |                     ReadBitmap16(reader, context, height, width, source); | ||||||
|  |                     break; | ||||||
|  |                 case 24: | ||||||
|  |                     ReadBitmap24(reader, context, biClrUsed, height, width, source); | ||||||
|  |                     break; | ||||||
|  |                 case 32: | ||||||
|  |                     ReadBitmap32(reader, context, height, width, source); | ||||||
|  |                     break; | ||||||
|  |                 default: | ||||||
|  |                     throw new InvalidIcoFileException(IcoErrorCode.InvalidBitapInfoHeader_biBitCount, $"BITMAPINFOHEADER.biBitCount is unknown value ({biBitCount}); expected 1, 4, 8, 16, or 32 bit depth.", context); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         private static void ReadIndexedBitmap(ByteReader reader, ParseContext context, uint bitDepth, uint colorTableSize, int height, int width, IcoFrame source) | ||||||
|  |         { | ||||||
|  |             var anyReservedChannel = false; | ||||||
|  |             var anyIndexOutOfBounds = false; | ||||||
|  | 
 | ||||||
|  |             if (colorTableSize == 0) | ||||||
|  |             { | ||||||
|  |                 colorTableSize = 1u << (int)bitDepth; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             source.Encoding.PaletteSize = colorTableSize; | ||||||
|  | 
 | ||||||
|  |             if (colorTableSize > 1u << (int)bitDepth) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidIcoFileException(IcoErrorCode.InvalidBitapInfoHeader_biClrUsed, $"BITMAPINFOHEADER.biClrUsed is greater than 2^biBitCount (biClrUsed == {colorTableSize}, biBitCount = {bitDepth}).", context); | ||||||
|  |             } | ||||||
|  |             else if (colorTableSize < 1u << (int)bitDepth) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.UndersizedColorTable, $"This bitmap uses a color table that is smaller than the bit depth ({colorTableSize} < 2^{bitDepth})", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var colorTable = new Rgba32[colorTableSize]; | ||||||
|  |             for (var i = 0; i < colorTableSize; i++) | ||||||
|  |             { | ||||||
|  |                 var c = new Bgra32 | ||||||
|  |                 { | ||||||
|  |                     PackedValue = reader.NextUint32() | ||||||
|  |                 }; | ||||||
|  | 
 | ||||||
|  |                 if (c.A != 0) | ||||||
|  |                 { | ||||||
|  |                     anyReservedChannel = true; | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 c.A = 255; | ||||||
|  |                 c.ToRgba32(ref colorTable[i]); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var padding = reader.SeekOffset % 4; | ||||||
|  | 
 | ||||||
|  |             for (var y = height - 1; y >= 0; y--) | ||||||
|  |             { | ||||||
|  |                 var bits = new BitReader(reader); | ||||||
|  | 
 | ||||||
|  |                 for (var x = 0; x < width; x++) | ||||||
|  |                 { | ||||||
|  |                     var colorIndex = bits.NextBit(bitDepth); | ||||||
|  | 
 | ||||||
|  |                     if (colorIndex >= colorTableSize) | ||||||
|  |                     { | ||||||
|  |                         anyIndexOutOfBounds = true; | ||||||
|  |                         source.CookedData[x, y] = Color.Black; | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                     { | ||||||
|  |                         source.CookedData[x, y] = colorTable[colorIndex]; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 while ((reader.SeekOffset % 4) != padding) | ||||||
|  |                 { | ||||||
|  |                     reader.SeekOffset += 1; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             switch (bitDepth) | ||||||
|  |             { | ||||||
|  |                 case 1: | ||||||
|  |                     source.Encoding.PixelFormat = BitmapEncoding.Pixel_indexed1; | ||||||
|  |                     break; | ||||||
|  |                 case 2: | ||||||
|  |                     source.Encoding.PixelFormat = BitmapEncoding.Pixel_indexed2; | ||||||
|  |                     break; | ||||||
|  |                 case 4: | ||||||
|  |                     source.Encoding.PixelFormat = BitmapEncoding.Pixel_indexed4; | ||||||
|  |                     break; | ||||||
|  |                 case 8: | ||||||
|  |                     source.Encoding.PixelFormat = BitmapEncoding.Pixel_indexed8; | ||||||
|  |                     break; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             ReadBitmapMask(reader, context, height, width, source); | ||||||
|  | 
 | ||||||
|  |             if (anyReservedChannel) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.NonzeroAlpha, $"Reserved Alpha channel used in color table.", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (anyIndexOutOfBounds) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.IndexedColorOutOfBounds, $"Bitmap uses color at illegal index; pixel filled with Black color.", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static void ReadBitmap16(ByteReader reader, ParseContext context, int height, int width, IcoFrame source) | ||||||
|  |         { | ||||||
|  |             for (var y = height - 1; y >= 0; y--) | ||||||
|  |             { | ||||||
|  |                 for (var x = 0; x < width; x++) | ||||||
|  |                 { | ||||||
|  |                     var colorValue = reader.NextUint16(); | ||||||
|  |                     source.CookedData[x, y] = new Rgba32( | ||||||
|  |                         _5To8[colorValue >> 10], | ||||||
|  |                         _5To8[(colorValue >> 5) & 0x1f], | ||||||
|  |                         _5To8[colorValue & 0x1f], | ||||||
|  |                         255); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             source.Encoding.PixelFormat = BitmapEncoding.Pixel_rgb15; | ||||||
|  |             ReadBitmapMask(reader, context, height, width, source); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static readonly byte[] _5To8 = new byte[] | ||||||
|  |         { | ||||||
|  |             0, 8, 16, 25, 33, 41, 49, 58, 66, 74, 82, 90, 99, 107, 115, 123, 132, 140, 148, 156, 165, 173, 181, 189, 197, 206, 214, 222, 230, 239, 247, 255, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         private static void ReadBitmap24(ByteReader reader, ParseContext context, uint colorTableSize, int height, int width, IcoFrame source) | ||||||
|  |         { | ||||||
|  |             reader.SeekOffset += (int)colorTableSize * 4; | ||||||
|  | 
 | ||||||
|  |             for (var y = height - 1; y >= 0; y--) | ||||||
|  |             { | ||||||
|  |                 for (var x = 0; x < width; x++) | ||||||
|  |                 { | ||||||
|  |                     var b = reader.NextUint8(); | ||||||
|  |                     var g = reader.NextUint8(); | ||||||
|  |                     var r = reader.NextUint8(); | ||||||
|  | 
 | ||||||
|  |                     source.CookedData[x, y] = new Rgba32(r, g, b, 255); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             source.Encoding.PixelFormat = BitmapEncoding.Pixel_rgb24; | ||||||
|  |             ReadBitmapMask(reader, context, height, width, source); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static void ReadBitmap32(ByteReader reader, ParseContext context, int height, int width, IcoFrame source) | ||||||
|  |         { | ||||||
|  |             for (var y = height - 1; y >= 0; y--) | ||||||
|  |             { | ||||||
|  |                 for (var x = 0; x < width; x++) | ||||||
|  |                 { | ||||||
|  |                     var colorValue = new Bgra32 { PackedValue = reader.NextUint32() }; | ||||||
|  |                     Rgba32 rgba32Value = source.CookedData[x, y]; // the ref keyword cannot be used on this indexer, so we need a temporary | ||||||
|  |                     colorValue.ToRgba32(ref rgba32Value); | ||||||
|  |                     source.CookedData[x, y] = rgba32Value; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             source.Encoding.PixelFormat = BmpUtil.IsAnyAlphaChannel(source.CookedData) | ||||||
|  |                 ? BitmapEncoding.Pixel_argb32 | ||||||
|  |                 : BitmapEncoding.Pixel_0rgb32; | ||||||
|  | 
 | ||||||
|  |             ReadBitmapMask(reader, context, height, width, source); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static void ReadBitmapMask(ByteReader reader, ParseContext context, int height, int width, IcoFrame source) | ||||||
|  |         { | ||||||
|  |             source.Mask = new bool[width, height]; | ||||||
|  | 
 | ||||||
|  |             var anyMask = false; | ||||||
|  |             var anyMaskedColors = false; | ||||||
|  | 
 | ||||||
|  |             var padding = reader.SeekOffset % 4; | ||||||
|  | 
 | ||||||
|  |             for (var y = height - 1; y >= 0; y--) | ||||||
|  |             { | ||||||
|  |                 var bits = new BitReader(reader); | ||||||
|  | 
 | ||||||
|  |                 for (var x = 0; x < width; x++) | ||||||
|  |                 { | ||||||
|  |                     var mask = bits.NextBit1(); | ||||||
|  | 
 | ||||||
|  |                     if (mask == 0) | ||||||
|  |                     { | ||||||
|  |                         continue; | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     source.Mask[x, y] = true; | ||||||
|  | 
 | ||||||
|  |                     anyMask = true; | ||||||
|  | 
 | ||||||
|  |                     if (source.CookedData[x, y].R != 0 || source.CookedData[x, y].G != 0 || source.CookedData[x, y].B != 0) | ||||||
|  |                     { | ||||||
|  |                         anyMaskedColors = true; | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     //source.CookedData[x, y] = new Rgba32(0, 0, 0, 0); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 while ((reader.SeekOffset % 4) != padding) | ||||||
|  |                 { | ||||||
|  |                     reader.SeekOffset += 1; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (!anyMask) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.NoMaskedPixels, $"No bitmap mask.", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (anyMaskedColors) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.MaskedPixelWithColor, $"Non-black image pixels masked out.", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
							
								
								
									
										328
									
								
								src/Velopack.IcoLib/Codecs/BmpEncoder.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										328
									
								
								src/Velopack.IcoLib/Codecs/BmpEncoder.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,328 @@ | |||||||
|  | using Ico.Binary; | ||||||
|  | using Ico.Model; | ||||||
|  | using Ico.Validation; | ||||||
|  | using SixLabors.ImageSharp.PixelFormats; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Linq; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Codecs | ||||||
|  | { | ||||||
|  |     public static class BmpEncoder | ||||||
|  |     { | ||||||
|  |         public enum Dialect | ||||||
|  |         { | ||||||
|  |             Ico, | ||||||
|  |             Bmp, | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static byte[] EncodeBitmap(ParseContext context, BitmapEncoding encoding, Dialect dialect, IcoFrame source) | ||||||
|  |         { | ||||||
|  |             context.LastEncodeError = IcoErrorCode.NoError; | ||||||
|  | 
 | ||||||
|  |             return (BmpUtil.GetBitDepthForPixelFormat(encoding) < 16) | ||||||
|  |                 ? EncodeIndexedBitmap(context, encoding, dialect, source) | ||||||
|  |                 : EncodeRgbBitmap(source, context, encoding, dialect); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static byte[] EncodeIndexedBitmap(ParseContext context, BitmapEncoding encoding, Dialect dialect, IcoFrame source) | ||||||
|  |         { | ||||||
|  |             var numBits = BmpUtil.GetBitDepthForPixelFormat(encoding); | ||||||
|  | 
 | ||||||
|  |             var colorTable = BuildColorTable(1u << numBits, context, source); | ||||||
|  |             if (colorTable == null) | ||||||
|  |             { | ||||||
|  |                 context.LastEncodeError = IcoErrorCode.TooManyColorsForBitDepth; | ||||||
|  |                 return null; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var writer = new ByteWriter(ByteOrder.LittleEndian); | ||||||
|  | 
 | ||||||
|  |             EncodeBitmapHeader(source, dialect, encoding, colorTable, writer, out var offsetToImageSize); | ||||||
|  | 
 | ||||||
|  |             var reverseTable = new Dictionary<Rgba32, int>(); | ||||||
|  |             for (var i = 0; i < colorTable.Length; i++) | ||||||
|  |             { | ||||||
|  |                 if (!reverseTable.ContainsKey(colorTable[i])) | ||||||
|  |                 { | ||||||
|  |                     reverseTable.Add(colorTable[i], i); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var offsetToData = (uint)writer.Data.Count; | ||||||
|  |             var padding = writer.Data.Count % 4; | ||||||
|  | 
 | ||||||
|  |             for (var y = source.CookedData.Height - 1; y >= 0; y--) | ||||||
|  |             { | ||||||
|  |                 var bits = new BitWriter(writer); | ||||||
|  | 
 | ||||||
|  |                 for (var x = 0; x < source.CookedData.Width; x++) | ||||||
|  |                 { | ||||||
|  |                     var color = source.CookedData[x, y]; | ||||||
|  | 
 | ||||||
|  |                     if (source.Mask[x, y]) | ||||||
|  |                     { | ||||||
|  |                         switch (context.MaskedImagePixelEmitOptions) | ||||||
|  |                         { | ||||||
|  |                             case StrictnessPolicy.Compliant: | ||||||
|  |                                 color = new Rgba32(0, 0, 0, 255); | ||||||
|  |                                 break; | ||||||
|  |                             case StrictnessPolicy.PreserveSource: | ||||||
|  |                                 // Pass through whatever the original pixel was. | ||||||
|  |                                 break; | ||||||
|  |                             case StrictnessPolicy.Loose: | ||||||
|  |                                 color = colorTable.First(); | ||||||
|  |                                 break; | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     color.A = 255; | ||||||
|  | 
 | ||||||
|  |                     var index = reverseTable[color]; | ||||||
|  |                     bits.AddBits((uint)numBits, (byte)index); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 while ((writer.Data.Count % 4) != padding) | ||||||
|  |                 { | ||||||
|  |                     writer.AddUint8(0); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return FinalizeBitmap(source, encoding, dialect, writer, offsetToData, offsetToImageSize); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static byte[] EncodeRgbBitmap(IcoFrame source, ParseContext context, BitmapEncoding encoding, Dialect dialect) | ||||||
|  |         { | ||||||
|  |             var writer = new ByteWriter(ByteOrder.LittleEndian); | ||||||
|  |             EncodeBitmapHeader(source, dialect, encoding, null, writer, out var offsetToImageSize); | ||||||
|  | 
 | ||||||
|  |             var offsetToData = (uint)writer.Data.Count; | ||||||
|  |             var padding = writer.Data.Count % 4; | ||||||
|  | 
 | ||||||
|  |             for (var y = source.CookedData.Height - 1; y >= 0; y--) | ||||||
|  |             { | ||||||
|  |                 var bits = new BitWriter(writer); | ||||||
|  | 
 | ||||||
|  |                 for (var x = 0; x < source.CookedData.Width; x++) | ||||||
|  |                 { | ||||||
|  |                     var color = source.CookedData[x, y]; | ||||||
|  | 
 | ||||||
|  |                     if (source.Mask[x, y]) | ||||||
|  |                     { | ||||||
|  |                         switch (context.MaskedImagePixelEmitOptions) | ||||||
|  |                         { | ||||||
|  |                             case StrictnessPolicy.Compliant: | ||||||
|  |                             case StrictnessPolicy.Loose: | ||||||
|  |                                 color = new Rgba32(0, 0, 0, 0); | ||||||
|  |                                 break; | ||||||
|  |                             case StrictnessPolicy.PreserveSource: | ||||||
|  |                                 // Pass through whatever the original pixel was. | ||||||
|  |                                 break; | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     switch (encoding) | ||||||
|  |                     { | ||||||
|  |                         case BitmapEncoding.Pixel_rgb15: | ||||||
|  |                             var value = X8To5(color.R) << 10 | X8To5(color.G) << 5 | X8To5(color.B); | ||||||
|  |                             writer.AddUint16((ushort)value); | ||||||
|  |                             break; | ||||||
|  |                         case BitmapEncoding.Pixel_rgb24: | ||||||
|  |                             writer.AddUint8(color.B); | ||||||
|  |                             writer.AddUint8(color.G); | ||||||
|  |                             writer.AddUint8(color.R); | ||||||
|  |                             break; | ||||||
|  |                         case BitmapEncoding.Pixel_0rgb32: | ||||||
|  |                             writer.AddUint8(color.B); | ||||||
|  |                             writer.AddUint8(color.G); | ||||||
|  |                             writer.AddUint8(color.R); | ||||||
|  |                             writer.AddUint8(0); | ||||||
|  |                             break; | ||||||
|  |                         case BitmapEncoding.Pixel_argb32: | ||||||
|  |                             writer.AddUint8(color.B); | ||||||
|  |                             writer.AddUint8(color.G); | ||||||
|  |                             writer.AddUint8(color.R); | ||||||
|  |                             writer.AddUint8(color.A); | ||||||
|  |                             break; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 while ((writer.Data.Count % 4) != padding) | ||||||
|  |                 { | ||||||
|  |                     writer.AddUint8(0); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return FinalizeBitmap(source, encoding, dialect, writer, offsetToData, offsetToImageSize); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static byte X8To5(uint b) | ||||||
|  |         { | ||||||
|  |             return (byte)(b * 32 / 256); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static void EncodeBitmapHeader(IcoFrame source, Dialect dialect, BitmapEncoding encoding, Rgba32[] colorTable, ByteWriter writer, out uint offsetToImageSize) | ||||||
|  |         { | ||||||
|  |             if (dialect != Dialect.Ico) | ||||||
|  |             { | ||||||
|  |                 writer.AddUint16(FileFormatConstants._bitmapFileMagic); | ||||||
|  |                 writer.AddUint32(0); // Size will be filled in later | ||||||
|  |                 writer.AddUint32(0); // Reserved | ||||||
|  |                 writer.AddUint32(0); // Offset will be filled in later | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             writer.AddUint32(FileFormatConstants._bitmapInfoHeaderSize); | ||||||
|  |             writer.AddUint32((uint)source.CookedData.Width); | ||||||
|  |             writer.AddUint32((uint)source.CookedData.Height * ((dialect == Dialect.Ico) ? 2u : 1u)); | ||||||
|  |             writer.AddUint16(1); // biPlanes | ||||||
|  |             writer.AddUint16((ushort)BmpUtil.GetBitDepthForPixelFormat(encoding)); // biBitCount | ||||||
|  |             writer.AddUint32(FileFormatConstants.BI_RGB); // biCompression | ||||||
|  |             offsetToImageSize = (uint)writer.SeekOffset; | ||||||
|  |             writer.AddUint32(0); // biSizeImage | ||||||
|  |             writer.AddUint32((dialect == Dialect.Ico) ? 0u : FileFormatConstants._72dpiInPixelsPerMeter); // biXPelsPerMeter | ||||||
|  |             writer.AddUint32((dialect == Dialect.Ico) ? 0u : FileFormatConstants._72dpiInPixelsPerMeter); // biYPelsPerMeter | ||||||
|  |             writer.AddUint32((uint)(colorTable?.Length ?? 0)); // biClrUsed | ||||||
|  |             writer.AddUint32(0); // biClrImportant | ||||||
|  | 
 | ||||||
|  |             if (colorTable != null) | ||||||
|  |             { | ||||||
|  |                 foreach (var color in colorTable) | ||||||
|  |                 { | ||||||
|  |                     writer.AddUint8(color.B); | ||||||
|  |                     writer.AddUint8(color.G); | ||||||
|  |                     writer.AddUint8(color.R); | ||||||
|  |                     writer.AddUint8(0); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (dialect != Dialect.Ico) | ||||||
|  |             { | ||||||
|  |                 while (writer.Data.Count % 4 != 0) | ||||||
|  |                 { | ||||||
|  |                     writer.AddUint8(0); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static byte[] FinalizeBitmap(IcoFrame source, BitmapEncoding encoding, Dialect dialect, ByteWriter writer, uint offsetToData, uint offsetToImageSize) | ||||||
|  |         { | ||||||
|  |             var offsetToEndOfData = writer.SeekOffset; | ||||||
|  | 
 | ||||||
|  |             if (dialect == Dialect.Ico) | ||||||
|  |             { | ||||||
|  |                 var padding = writer.Data.Count % 4; | ||||||
|  | 
 | ||||||
|  |                 var inferMaskFromAlpha = (source.Encoding.PixelFormat == BitmapEncoding.Pixel_argb32 && encoding != BitmapEncoding.Pixel_argb32); | ||||||
|  | 
 | ||||||
|  |                 for (var y = source.CookedData.Height - 1; y >= 0; y--) | ||||||
|  |                 { | ||||||
|  |                     var bits = new BitWriter(writer); | ||||||
|  | 
 | ||||||
|  |                     for (var x = 0; x < source.CookedData.Width; x++) | ||||||
|  |                     { | ||||||
|  |                         var mask = inferMaskFromAlpha | ||||||
|  |                             ? (source.CookedData[x, y].A == 0) | ||||||
|  |                             : source.Mask[x, y]; | ||||||
|  | 
 | ||||||
|  |                         bits.AddBit1((byte)(mask ? 1 : 0)); | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     while ((writer.Data.Count % 4) != padding) | ||||||
|  |                     { | ||||||
|  |                         writer.AddUint8(0); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (dialect != Dialect.Ico) | ||||||
|  |             { | ||||||
|  |                 writer.SeekOffset = 2; | ||||||
|  |                 writer.AddUint32((uint)writer.Data.Count); | ||||||
|  | 
 | ||||||
|  |                 writer.SeekOffset = 10; | ||||||
|  |                 writer.AddUint32(offsetToData); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             writer.SeekOffset = (int)offsetToImageSize; | ||||||
|  |             writer.AddUint32((uint)(offsetToEndOfData - offsetToData)); // biSizeImage | ||||||
|  | 
 | ||||||
|  |             return writer.Data.ToArray(); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static Rgba32[] BuildColorTable(uint maxColorTableSize, ParseContext context, IcoFrame source) | ||||||
|  |         { | ||||||
|  |             var colorTable = new Dictionary<Rgba32, uint>(); | ||||||
|  | 
 | ||||||
|  |             for (var y = source.CookedData.Height - 1; y >= 0; y--) | ||||||
|  |             { | ||||||
|  |                 for (var x = 0; x < source.CookedData.Width; x++) | ||||||
|  |                 { | ||||||
|  |                     var color = source.CookedData[x, y]; | ||||||
|  | 
 | ||||||
|  |                     if (source.Mask[x, y]) | ||||||
|  |                     { | ||||||
|  |                         switch (context.MaskedImagePixelEmitOptions) | ||||||
|  |                         { | ||||||
|  |                             case StrictnessPolicy.Compliant: | ||||||
|  |                                 // Ensure an entry is added for black. | ||||||
|  |                                 color = new Rgba32(0, 0, 0, 0); | ||||||
|  |                                 break; | ||||||
|  |                             case StrictnessPolicy.PreserveSource: | ||||||
|  |                                 // Pass through whatever the original pixel was. | ||||||
|  |                                 break; | ||||||
|  |                             case StrictnessPolicy.Loose: | ||||||
|  |                                 // Don't create a palette entry for this pixel. | ||||||
|  |                                 continue; | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  | 
 | ||||||
|  |                     color.A = 255; | ||||||
|  | 
 | ||||||
|  |                     if (colorTable.ContainsKey(color)) | ||||||
|  |                     { | ||||||
|  |                         colorTable[color] += 1; | ||||||
|  |                     } | ||||||
|  |                     else if (colorTable.Count == maxColorTableSize) | ||||||
|  |                     { | ||||||
|  |                         return null; | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                     { | ||||||
|  |                         colorTable.Add(color, 1); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (colorTable.Count == 0) | ||||||
|  |             { | ||||||
|  |                 colorTable.Add(new Rgba32(0, 0, 0, 255), 1); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var table = (from c in colorTable | ||||||
|  |                          orderby c.Value descending | ||||||
|  |                          select c.Key).ToList(); | ||||||
|  | 
 | ||||||
|  |             var targetPaletteSize = 0u; | ||||||
|  | 
 | ||||||
|  |             switch (context.AllowPaletteTruncation) | ||||||
|  |             { | ||||||
|  |                 case StrictnessPolicy.Compliant: | ||||||
|  |                     targetPaletteSize = maxColorTableSize; | ||||||
|  |                     break; | ||||||
|  |                 case StrictnessPolicy.PreserveSource: | ||||||
|  |                     targetPaletteSize = source.Encoding.PaletteSize; | ||||||
|  |                     break; | ||||||
|  |                 case StrictnessPolicy.Loose: | ||||||
|  |                     targetPaletteSize = (uint)table.Count; | ||||||
|  |                     break; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             while (table.Count < targetPaletteSize) | ||||||
|  |             { | ||||||
|  |                 table.Add(new Rgba32(0, 0, 0, 255)); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return table.ToArray(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										236
									
								
								src/Velopack.IcoLib/Codecs/BmpUtil.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										236
									
								
								src/Velopack.IcoLib/Codecs/BmpUtil.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,236 @@ | |||||||
|  | using Ico.Model; | ||||||
|  | using SixLabors.ImageSharp; | ||||||
|  | using SixLabors.ImageSharp.PixelFormats; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Codecs | ||||||
|  | { | ||||||
|  |     public static class BmpUtil | ||||||
|  |     { | ||||||
|  |         public static bool IsAlphaSignificant(IcoFrame source) | ||||||
|  |         { | ||||||
|  |             if (IsAnyPartialTransparency(source.CookedData)) | ||||||
|  |             { | ||||||
|  |                 return true; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return IsAnyPixel(source.CookedData, (x, y, pixel)  | ||||||
|  |                 => IsCompletelyTransparent(pixel) != source.Mask[x, y]); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static bool IsAnyPartialTransparency(Image<Rgba32> image) | ||||||
|  |         { | ||||||
|  |             return IsAnyPixel(image, (x, y, pixel)  | ||||||
|  |                 => !IsCompletelyOpaque(pixel) && !IsCompletelyTransparent(pixel)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static bool IsAnyAlphaChannel(Image<Rgba32> image) | ||||||
|  |         { | ||||||
|  |             return IsAnyPixel(image, (x, y, pixel) | ||||||
|  |                 => !IsCompletelyOpaque(pixel)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static ulong GetNumberOfDistinctColors(Image<Rgba32> image, bool includeAlpha) | ||||||
|  |         { | ||||||
|  |             var colors = new HashSet<uint>(); | ||||||
|  | 
 | ||||||
|  |             if (includeAlpha) | ||||||
|  |             { | ||||||
|  |                 ForeachPixel(image, (x, y, pixel) | ||||||
|  |                     => colors.Add(pixel.PackedValue)); | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 ForeachPixel(image, (x, y, pixel) | ||||||
|  |                     => colors.Add((uint)((pixel.R << 16) | (pixel.G << 8) | pixel.B))); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return (uint)colors.Count; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private class NumBitsPerChannel | ||||||
|  |         { | ||||||
|  |             public int R { get; set; } = 1; | ||||||
|  |             public int G { get; set; } = 1; | ||||||
|  |             public int B { get; set; } = 1; | ||||||
|  |             public int A { get; set; } = 1; | ||||||
|  | 
 | ||||||
|  |             public bool RgbLessThan(int depth) | ||||||
|  |             { | ||||||
|  |                 return R < depth && G < depth && B < depth; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static int GetMinimumColorDepthForDisplay(Image<Rgba32> image) | ||||||
|  |         { | ||||||
|  |             if (IsAnyPartialTransparency(image)) | ||||||
|  |             { | ||||||
|  |                 return 32; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (!IsAnyPixel(image, (x, y, pixel) => !IsBlack(pixel) && !IsWhite(pixel))) | ||||||
|  |             { | ||||||
|  |                 return 1; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var bpc = new NumBitsPerChannel(); | ||||||
|  |             ForeachPixel(image, (x, y, pixel) => UpdateNumBitsPerChannel(pixel, bpc)); | ||||||
|  | 
 | ||||||
|  |             if (bpc.RgbLessThan(2)) | ||||||
|  |                 return 3 * 1; | ||||||
|  |             if (bpc.RgbLessThan(3)) | ||||||
|  |                 return 3 * 2; | ||||||
|  |             if (bpc.RgbLessThan(5)) | ||||||
|  |                 return 3 * 3; | ||||||
|  | 
 | ||||||
|  |             return 3 * 4; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static BitmapEncoding GetIdealBitmapEncoding(Image<Rgba32> image, bool hasIcoMask) | ||||||
|  |         { | ||||||
|  |             if (IsAnyPartialTransparency(image)) | ||||||
|  |             { | ||||||
|  |                 return BitmapEncoding.Pixel_argb32; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var numColors = GetNumberOfDistinctColors(image, !hasIcoMask); | ||||||
|  |             if (numColors <= 2) | ||||||
|  |                 return BitmapEncoding.Pixel_indexed1; | ||||||
|  |             if (numColors <= 4) | ||||||
|  |                 return BitmapEncoding.Pixel_indexed2; | ||||||
|  |             if (numColors <= 16) | ||||||
|  |                 return BitmapEncoding.Pixel_indexed4; | ||||||
|  |             if (numColors <= 256) | ||||||
|  |                 return BitmapEncoding.Pixel_indexed8; | ||||||
|  | 
 | ||||||
|  |             var bpc = new NumBitsPerChannel(); | ||||||
|  |             ForeachPixel(image, (x, y, pixel) => UpdateNumBitsPerChannel(pixel, bpc)); | ||||||
|  | 
 | ||||||
|  |             if (hasIcoMask || !IsAnyAlphaChannel(image)) | ||||||
|  |             { | ||||||
|  |                 if (bpc.RgbLessThan(6)) | ||||||
|  |                     return BitmapEncoding.Pixel_rgb15; | ||||||
|  |                 return BitmapEncoding.Pixel_rgb24; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return BitmapEncoding.Pixel_argb32; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static void UpdateNumBitsPerChannel(Rgba32 pixel, NumBitsPerChannel bpc) | ||||||
|  |         { | ||||||
|  |             bpc.R = Math.Max(bpc.R, GetMinimumColorDepth(pixel.R)); | ||||||
|  |             bpc.G = Math.Max(bpc.G, GetMinimumColorDepth(pixel.G)); | ||||||
|  |             bpc.B = Math.Max(bpc.B, GetMinimumColorDepth(pixel.B)); | ||||||
|  |             bpc.A = Math.Max(bpc.A, GetMinimumColorDepth(pixel.A)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static int GetMinimumColorDepth(byte channel) | ||||||
|  |         { | ||||||
|  |             if (channel == 255) | ||||||
|  |                 return 1; | ||||||
|  | 
 | ||||||
|  |             var mask = 255; | ||||||
|  |             int depth; | ||||||
|  | 
 | ||||||
|  |             for (depth = 1; depth < 8; depth++) | ||||||
|  |             { | ||||||
|  |                 if ((channel & mask) == 0) | ||||||
|  |                     break; | ||||||
|  | 
 | ||||||
|  |                 mask >>= 1; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return depth; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static bool[,] CreateMaskFromImage(Image<Rgba32> image, bool blackIsTransparent) | ||||||
|  |         { | ||||||
|  |             var mask = new bool[image.Width, image.Height]; | ||||||
|  | 
 | ||||||
|  |             ForeachPixel(image, (x, y, pixel) => | ||||||
|  |             { | ||||||
|  |                 if (IsCompletelyTransparent(pixel) || | ||||||
|  |                    (blackIsTransparent && IsBlack(pixel))) | ||||||
|  |                 { | ||||||
|  |                     mask[x, y] = true; | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  | 
 | ||||||
|  |             return mask; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static int GetBitDepthForPixelFormat(BitmapEncoding pixelFormat) | ||||||
|  |         { | ||||||
|  |             switch (pixelFormat) | ||||||
|  |             { | ||||||
|  |                 case BitmapEncoding.Pixel_indexed1: | ||||||
|  |                     return 1; | ||||||
|  |                 case BitmapEncoding.Pixel_indexed2: | ||||||
|  |                     return 2; | ||||||
|  |                 case BitmapEncoding.Pixel_indexed4: | ||||||
|  |                     return 4; | ||||||
|  |                 case BitmapEncoding.Pixel_indexed8: | ||||||
|  |                     return 8; | ||||||
|  |                 case BitmapEncoding.Pixel_rgb15: | ||||||
|  |                     return 16; | ||||||
|  |                 case BitmapEncoding.Pixel_rgb24: | ||||||
|  |                     return 24; | ||||||
|  |                 case BitmapEncoding.Pixel_0rgb32: | ||||||
|  |                     return 32; | ||||||
|  |                 case BitmapEncoding.Pixel_argb32: | ||||||
|  |                     return 32; | ||||||
|  |             } | ||||||
|  |             throw new ArgumentException(nameof(pixelFormat)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static void ForeachPixel(Image<Rgba32> image, Action<int, int, Rgba32> action) | ||||||
|  |         { | ||||||
|  |             var mask = new bool[image.Width, image.Height]; | ||||||
|  | 
 | ||||||
|  |             for (int x = 0; x < image.Width; x++) | ||||||
|  |             { | ||||||
|  |                 for (int y = 0; y < image.Height; y++) | ||||||
|  |                 { | ||||||
|  |                     action(x, y, image[x,y]); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static bool IsAnyPixel(Image<Rgba32> image, Func<int, int, Rgba32, bool> predicate) | ||||||
|  |         { | ||||||
|  |             var mask = new bool[image.Width, image.Height]; | ||||||
|  | 
 | ||||||
|  |             for (int x = 0; x < image.Width; x++) | ||||||
|  |             { | ||||||
|  |                 for (int y = 0; y < image.Height; y++) | ||||||
|  |                 { | ||||||
|  |                     if (predicate(x, y, image[x, y])) | ||||||
|  |                         return true; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static bool IsCompletelyTransparent(Rgba32 pixel) | ||||||
|  |         { | ||||||
|  |             return pixel.A == 0; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static bool IsCompletelyOpaque(Rgba32 pixel) | ||||||
|  |         { | ||||||
|  |             return pixel.A == 255; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static bool IsBlack(Rgba32 pixel) | ||||||
|  |         { | ||||||
|  |             return pixel.R == 0 && pixel.G == 0 && pixel.B == 0; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static bool IsWhite(Rgba32 pixel) | ||||||
|  |         { | ||||||
|  |             return pixel.R == 255 && pixel.G == 255 && pixel.B == 255; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										26
									
								
								src/Velopack.IcoLib/Codecs/EncodingOptions.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								src/Velopack.IcoLib/Codecs/EncodingOptions.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | namespace Ico.Codecs | ||||||
|  | { | ||||||
|  |     public enum MaskedImagePixelEmitOptions | ||||||
|  |     { | ||||||
|  |         Compliant, | ||||||
|  |         PreserveSource, | ||||||
|  |         Loose, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public enum StrictnessPolicy | ||||||
|  |     { | ||||||
|  |         Compliant, | ||||||
|  |         PreserveSource, | ||||||
|  |         Loose, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public enum BestFormatPolicy | ||||||
|  |     { | ||||||
|  |         PreserveSource, | ||||||
|  |         MinimizeStorage, | ||||||
|  |         PngLargeImages, | ||||||
|  |         AlwaysPng, | ||||||
|  |         AlwaysBmp, | ||||||
|  |         Inherited, | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										20
									
								
								src/Velopack.IcoLib/Codecs/FileFormatConstants.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								src/Velopack.IcoLib/Codecs/FileFormatConstants.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | namespace Ico.Codecs | ||||||
|  | { | ||||||
|  |     public static class FileFormatConstants | ||||||
|  |     { | ||||||
|  |         public static int MaxIcoFileSize = 20 * 1024 * 1024; | ||||||
|  |         internal static ushort _iconMagicHeader = 0; | ||||||
|  |         internal static ushort _iconMagicType = 1; | ||||||
|  |         internal static ushort _iconMaxEntries = 256; | ||||||
|  |         internal static byte _iconEntryReserved = 0; | ||||||
|  |         internal static ulong _pngHeader = 0x89504e470d0a1a0a; | ||||||
|  | 
 | ||||||
|  |         internal static uint _bitmapInfoHeaderSize = 40; | ||||||
|  |         internal static uint BI_RGB = 0; | ||||||
|  |         internal static uint BI_BITFIELDS = 3; | ||||||
|  | 
 | ||||||
|  |         internal static ushort _bitmapFileMagic = 0x4d42; // 'BM' | ||||||
|  |         internal static ushort _bitmapFileHeaderSize = 14; | ||||||
|  |         internal static uint _72dpiInPixelsPerMeter = 2835u; | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										110
									
								
								src/Velopack.IcoLib/Codecs/IcoDecoder.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								src/Velopack.IcoLib/Codecs/IcoDecoder.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,110 @@ | |||||||
|  | using Ico.Binary; | ||||||
|  | using Ico.Model; | ||||||
|  | using Ico.Validation; | ||||||
|  | using System; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Codecs | ||||||
|  | { | ||||||
|  |     public static class IcoDecoder | ||||||
|  |     { | ||||||
|  |         public static void DoFile(byte[] data, ParseContext context, Action<IcoFrame> processFrame) | ||||||
|  |         { | ||||||
|  |             var reader = new ByteReader(data, ByteOrder.LittleEndian); | ||||||
|  | 
 | ||||||
|  |             var idReserved = reader.NextUint16(); | ||||||
|  |             var idType = reader.NextUint16(); | ||||||
|  |             var idCount = reader.NextUint16(); | ||||||
|  | 
 | ||||||
|  |             if (idReserved != FileFormatConstants._iconMagicHeader) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidIcoFileException(IcoErrorCode.InvalidIcoHeader_idReserved, $"ICONDIR.idReserved should be {FileFormatConstants._iconMagicHeader}, was {idReserved}.", context); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (idType != FileFormatConstants._iconMagicType) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidIcoFileException(IcoErrorCode.InvalidIconHeader_idType, $"ICONDIR.idType should be {FileFormatConstants._iconMagicType}, was {idType}.", context); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (idCount == 0 || idCount > FileFormatConstants._iconMaxEntries) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidIcoFileException(IcoErrorCode.TooManyFrames, $"ICONDIR.idCount is {idCount}, an implausible value for an ICO file.", context); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             for (var i = 0u; i < idCount; i++) | ||||||
|  |             { | ||||||
|  |                 context.ImageDirectoryIndex = i; | ||||||
|  |                 var source = ProcessIcoFrame(reader, context); | ||||||
|  |                 processFrame(source); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             context.ImageDirectoryIndex = null; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static IcoFrame ProcessIcoFrame(ByteReader reader, ParseContext context) | ||||||
|  |         { | ||||||
|  |             var bWidth = reader.NextUint8(); | ||||||
|  |             var bHeight = reader.NextUint8(); | ||||||
|  |             var bColorCount = reader.NextUint8(); | ||||||
|  |             var bReserved = reader.NextUint8(); | ||||||
|  |             var wPlanes = reader.NextUint16(); | ||||||
|  |             var wBitCount = reader.NextUint16(); | ||||||
|  |             var dwBytesInRes = reader.NextUint32(); | ||||||
|  |             var dwImageOffset = reader.NextUint32(); | ||||||
|  | 
 | ||||||
|  |             if (bWidth != bHeight) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.NotSquare, $"Icon is not square ({bWidth}x{bHeight}).", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (bReserved != FileFormatConstants._iconEntryReserved) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidIcoFileException(IcoErrorCode.InvalidFrameHeader_bReserved, $"ICONDIRECTORY.bReserved should be {FileFormatConstants._iconEntryReserved}, was {bReserved}.", context); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (wPlanes > 1) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidIcoFileException(IcoErrorCode.InvalidFrameHeader_wPlanes, $"ICONDIRECTORY.wPlanes is {wPlanes}.  Only single-plane bitmaps are supported.", context); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (dwBytesInRes > int.MaxValue) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidIcoFileException(IcoErrorCode.InvalidFrameHeader_dwBytesInRes, $"ICONDIRECTORY.dwBytesInRes == {dwBytesInRes}, which is unreasonably large.", context); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (dwImageOffset > int.MaxValue) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidIcoFileException(IcoErrorCode.InvalidFrameHeader_dwImageOffset, $"ICONDIRECTORY.dwImageOffset == {dwImageOffset}, which is unreasonably large.", context); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var source = new IcoFrame | ||||||
|  |             { | ||||||
|  |                 TotalDiskUsage = dwBytesInRes + /* sizeof(ICONDIRENTRY) */ 16, | ||||||
|  | 
 | ||||||
|  |                 Encoding = new IcoFrameEncoding | ||||||
|  |                 { | ||||||
|  |                     ClaimedBitDepth = wBitCount, | ||||||
|  |                     ClaimedHeight = bHeight > 0 ? bHeight : 256u, | ||||||
|  |                     ClaimedWidth = bWidth > 0 ? bWidth : 256u, | ||||||
|  |                 }, | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             source.RawData = reader.Data.Slice((int)dwImageOffset, (int)dwBytesInRes).ToArray(); | ||||||
|  |             var bitmapHeader = new ByteReader(source.RawData, ByteOrder.LittleEndian); | ||||||
|  | 
 | ||||||
|  |             var signature = bitmapHeader.NextUint64(); | ||||||
|  |             bitmapHeader.SeekOffset = 0; | ||||||
|  | 
 | ||||||
|  |             if (PngDecoder.IsProbablyPngFile(ByteOrderConverter.To(ByteOrder.NetworkEndian, signature))) | ||||||
|  |             { | ||||||
|  |                 PngDecoder.DoPngEntry(bitmapHeader, context, source); | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 BmpDecoder.DoBitmapEntry(bitmapHeader, context, source); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return source; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										53
									
								
								src/Velopack.IcoLib/Codecs/IcoEncoder.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								src/Velopack.IcoLib/Codecs/IcoEncoder.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | |||||||
|  | using Ico.Binary; | ||||||
|  | using Ico.Model; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.IO; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Codecs | ||||||
|  | { | ||||||
|  |     public static class IcoEncoder | ||||||
|  |     { | ||||||
|  |         public static void EmitIco(string outputPath, ParseContext context) | ||||||
|  |         { | ||||||
|  |             var writer = new ByteWriter(ByteOrder.LittleEndian); | ||||||
|  | 
 | ||||||
|  |             writer.AddUint16(FileFormatConstants._iconMagicHeader); | ||||||
|  |             writer.AddUint16(FileFormatConstants._iconMagicType); | ||||||
|  |             writer.AddUint16((ushort)context.GeneratedFrames.Count); | ||||||
|  | 
 | ||||||
|  |             var offsets = new Queue<uint>(); | ||||||
|  | 
 | ||||||
|  |             foreach (var frame in context.GeneratedFrames) | ||||||
|  |             { | ||||||
|  |                 var width = frame.Encoding.ClaimedWidth; | ||||||
|  |                 var height = frame.Encoding.ClaimedHeight; | ||||||
|  |                 var bitDepth = frame.Encoding.ClaimedBitDepth; | ||||||
|  | 
 | ||||||
|  |                 writer.AddUint8((byte)(width >= 256 ? 0 : width)); // bWidth | ||||||
|  |                 writer.AddUint8((byte)(height >= 256 ? 0 : height)); // bHeight | ||||||
|  |                 writer.AddUint8((byte)(bitDepth < 8 ? 1u << (int)bitDepth : 0)); // bColorCount | ||||||
|  |                 writer.AddUint8(0); // bReserved | ||||||
|  |                 writer.AddUint16(1); // wPlanes | ||||||
|  |                 writer.AddUint16((ushort)bitDepth); // wBitCount | ||||||
|  |                 writer.AddUint32((uint)frame.RawData.Length); // dwBytesInRes | ||||||
|  | 
 | ||||||
|  |                 offsets.Enqueue((uint)writer.SeekOffset); | ||||||
|  |                 writer.AddUint32(0); // dwImageOffset (will fix later) | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             foreach (var frame in context.GeneratedFrames) | ||||||
|  |             { | ||||||
|  |                 var currentOffset = writer.SeekOffset; | ||||||
|  |                 writer.SeekOffset = (int)offsets.Dequeue(); | ||||||
|  |                 writer.AddUint32((uint)currentOffset); // dwImageOffset | ||||||
|  |                 writer.SeekOffset = currentOffset; | ||||||
|  | 
 | ||||||
|  |                 writer.AddBlob(frame.RawData); | ||||||
|  | 
 | ||||||
|  |                 frame.TotalDiskUsage = (uint)frame.RawData.Length + /* sizeof(ICONDIRENTRY) */ 16; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             File.WriteAllBytes(outputPath, writer.Data.ToArray()); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										157
									
								
								src/Velopack.IcoLib/Codecs/PngDecoder.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										157
									
								
								src/Velopack.IcoLib/Codecs/PngDecoder.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,157 @@ | |||||||
|  | using Ico.Binary; | ||||||
|  | using Ico.Model; | ||||||
|  | using Ico.Validation; | ||||||
|  | using SixLabors.ImageSharp; | ||||||
|  | using SixLabors.ImageSharp.PixelFormats; | ||||||
|  | using System; | ||||||
|  | using System.IO; | ||||||
|  | using System.Threading; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Codecs | ||||||
|  | { | ||||||
|  |     public static class PngDecoder | ||||||
|  |     { | ||||||
|  |         private const int _ihdrChunkName = 0x49484452; // "IHDR" | ||||||
|  | 
 | ||||||
|  |         public static bool IsProbablyPngFile(ulong first8Bytes) | ||||||
|  |         { | ||||||
|  |             return FileFormatConstants._pngHeader == first8Bytes; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static void DoPngEntry(ByteReader bitmapHeader, ParseContext context, IcoFrame source) | ||||||
|  |         { | ||||||
|  |             if (source.Encoding.ClaimedBitDepth != 32) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.PngNot32Bit, $"PNG-encoded image with bit depth {source.Encoding.ClaimedBitDepth} (expected 32).", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             using (var stream = new MemoryStream(bitmapHeader.Data.ToArray())) | ||||||
|  |             { | ||||||
|  |                 var decoder = new SixLabors.ImageSharp.Formats.Png.PngDecoder(); | ||||||
|  |                 source.CookedData = decoder.Decode<Rgba32>(new Configuration(), stream, CancellationToken.None); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             source.Encoding.Type = IcoEncodingType.Png; | ||||||
|  |             source.Encoding.PixelFormat = BmpUtil.IsAnyPartialTransparency(source.CookedData) ? BitmapEncoding.Pixel_argb32 : BitmapEncoding.Pixel_0rgb32; | ||||||
|  |             source.Mask = GenerateMaskFromAlpha(source.CookedData); | ||||||
|  | 
 | ||||||
|  |             // Conservatively assume that the output wouldn't have used palette trimming, if it had been a bmp frame. | ||||||
|  |             if (source.Encoding.ClaimedBitDepth < 16) | ||||||
|  |             { | ||||||
|  |                 source.Encoding.PaletteSize = 1u << (int)source.Encoding.ClaimedBitDepth; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var encoding = GetPngFileEncoding(bitmapHeader.Data); | ||||||
|  |             if (encoding.ColorType != PngColorType.RGBA) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.PngNotRGBA32, $"ICO files require the embedded PNG image to be encoded in RGBA32 format; this is {encoding.ColorType}", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  |             else if (encoding.BitsPerChannel != 8) | ||||||
|  |             { | ||||||
|  |                 context.Reporter.WarnLine(IcoErrorCode.PngNotRGBA32, $"ICO files require the embedded PNG image to be encoded in RGBA32 format; this is RGBA{encoding.BitsPerChannel * 4}", context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             uint numChannels = 0; | ||||||
|  |             switch (encoding.ColorType) | ||||||
|  |             { | ||||||
|  |                 case PngColorType.Grayscale: | ||||||
|  |                     numChannels = 1; | ||||||
|  |                     break; | ||||||
|  |                 case PngColorType.RGB: | ||||||
|  |                     numChannels = 3; | ||||||
|  |                     break; | ||||||
|  |                 case PngColorType.GrayscaleAlpha: | ||||||
|  |                     numChannels = 2; | ||||||
|  |                     break; | ||||||
|  |                 case PngColorType.RGBA: | ||||||
|  |                     numChannels = 4; | ||||||
|  |                     break; | ||||||
|  |                 case PngColorType.Palette: | ||||||
|  |                 default: | ||||||
|  |                     break; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             source.Encoding.ActualHeight = encoding.Height; | ||||||
|  |             source.Encoding.ActualWidth = encoding.Width; | ||||||
|  |             source.Encoding.ActualBitDepth = encoding.BitsPerChannel * numChannels; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public static PngFileEncoding GetPngFileEncoding(Memory<byte> data) | ||||||
|  |         { | ||||||
|  |             var reader = new ByteReader(data, Ico.Binary.ByteOrder.NetworkEndian); | ||||||
|  |             if (FileFormatConstants._pngHeader != reader.NextUint64()) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidPngFileException(IcoErrorCode.NotPng, $"Data stream does not begin with the PNG magic constant"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var chunkLength = reader.NextUint32(); | ||||||
|  |             var chunkType = reader.NextUint32(); | ||||||
|  | 
 | ||||||
|  |             if (chunkType != _ihdrChunkName) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidPngFileException(IcoErrorCode.PngBadIHDR, $"PNG file should begin with IHDR chunk; found {chunkType} instead"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             if (chunkLength < 13) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidPngFileException(IcoErrorCode.PngBadIHDR, $"IHDR chunk is invalid length {chunkLength}; expected at least 13 bytes"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             var result = new PngFileEncoding | ||||||
|  |             { | ||||||
|  |                 Width = reader.NextUint32(), | ||||||
|  |                 Height = reader.NextUint32(), | ||||||
|  |                 BitsPerChannel = reader.NextUint8(), | ||||||
|  |                 ColorType = (PngColorType)reader.NextUint8(), | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             if (result.Width == 0 || result.Height == 0) | ||||||
|  |             { | ||||||
|  |                 throw new InvalidPngFileException(IcoErrorCode.PngIllegalInputDimensions, $"Illegal Width x Height of {result.Width} x {result.Height}"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             switch (result.BitsPerChannel) | ||||||
|  |             { | ||||||
|  |                 case 1: | ||||||
|  |                 case 2: | ||||||
|  |                 case 4: | ||||||
|  |                 case 8: | ||||||
|  |                 case 16: | ||||||
|  |                     break; | ||||||
|  |                 default: | ||||||
|  |                     throw new InvalidPngFileException(IcoErrorCode.PngIllegalInputDepth, $"Illegal bits per color channel / palette entry of {result.BitsPerChannel}"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             switch (result.ColorType) | ||||||
|  |             { | ||||||
|  |                 case PngColorType.Grayscale: | ||||||
|  |                 case PngColorType.RGB: | ||||||
|  |                 case PngColorType.Palette: | ||||||
|  |                 case PngColorType.GrayscaleAlpha: | ||||||
|  |                 case PngColorType.RGBA: | ||||||
|  |                     break; | ||||||
|  |                 default: | ||||||
|  |                     throw new InvalidPngFileException(IcoErrorCode.PngIllegalColorType, $"Illegal color type {result.ColorType}"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return result; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private static bool[,] GenerateMaskFromAlpha(Image<Rgba32> image) | ||||||
|  |         { | ||||||
|  |             var mask = new bool[image.Width, image.Height]; | ||||||
|  | 
 | ||||||
|  |             for (var x = 0; x < image.Width; x++) | ||||||
|  |             { | ||||||
|  |                 for (var y = 0; y < image.Height; y++) | ||||||
|  |                 { | ||||||
|  |                     var alpha = image[x, y].A; | ||||||
|  |                     mask[x, y] = alpha == 0; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             return mask; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										40
									
								
								src/Velopack.IcoLib/Host/ExceptionWrapper.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								src/Velopack.IcoLib/Host/ExceptionWrapper.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | using Ico.Model; | ||||||
|  | using Ico.Validation; | ||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Text; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Host | ||||||
|  | { | ||||||
|  |     public static class ExceptionWrapper | ||||||
|  |     { | ||||||
|  |         public static void Try(Action action, ParseContext context, IErrorReporter reporter) | ||||||
|  |         { | ||||||
|  |             try | ||||||
|  |             { | ||||||
|  |                 action(); | ||||||
|  |             } | ||||||
|  |             catch (InvalidIcoFileException e) | ||||||
|  |             { | ||||||
|  |                 var frame = (e.Context?.ImageDirectoryIndex); | ||||||
|  | 
 | ||||||
|  |                 if (frame != null) | ||||||
|  |                 { | ||||||
|  |                     reporter.ErrorLine(e.ErrorCode, e.Message, context.DisplayedPath, context.ImageDirectoryIndex.Value); | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     reporter.ErrorLine(e.ErrorCode, e.Message, context.DisplayedPath); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             catch (InvalidPngFileException e) when (e.ErrorCode != IcoErrorCode.NoError) | ||||||
|  |             { | ||||||
|  |                 reporter.ErrorLine(e.ErrorCode, e.Message); | ||||||
|  |             } | ||||||
|  |             catch (Exception e) | ||||||
|  |             { | ||||||
|  |                 reporter.ErrorLine(IcoErrorCode.NoError, e.ToString(), context.DisplayedPath); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										23
									
								
								src/Velopack.IcoLib/Host/IErrorReporter.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								src/Velopack.IcoLib/Host/IErrorReporter.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | using Ico.Validation; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Host | ||||||
|  | { | ||||||
|  |     public interface IErrorReporter | ||||||
|  |     { | ||||||
|  |         void ErrorLine(IcoErrorCode errorCode, string message); | ||||||
|  | 
 | ||||||
|  |         void ErrorLine(IcoErrorCode errorCode, string message, string fileName); | ||||||
|  | 
 | ||||||
|  |         void ErrorLine(IcoErrorCode errorCode, string message, string fileName, uint frameNumber); | ||||||
|  | 
 | ||||||
|  |         void WarnLine(IcoErrorCode errorCode, string message); | ||||||
|  | 
 | ||||||
|  |         void WarnLine(IcoErrorCode errorCode, string message, string fileName); | ||||||
|  | 
 | ||||||
|  |         void WarnLine(IcoErrorCode errorCode, string message, string fileName, uint frameNumber); | ||||||
|  | 
 | ||||||
|  |         void InfoLine(string message); | ||||||
|  | 
 | ||||||
|  |         void VerboseLine(string message); | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										18
									
								
								src/Velopack.IcoLib/Model/IcoFrame.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/Velopack.IcoLib/Model/IcoFrame.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | using SixLabors.ImageSharp; | ||||||
|  | using SixLabors.ImageSharp.PixelFormats; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Model | ||||||
|  | { | ||||||
|  |     public class IcoFrame | ||||||
|  |     { | ||||||
|  |         public IcoFrameEncoding Encoding { get; set; } | ||||||
|  | 
 | ||||||
|  |         public byte[] RawData { get; set; } | ||||||
|  | 
 | ||||||
|  |         public Image<Rgba32> CookedData { get; set; } | ||||||
|  | 
 | ||||||
|  |         public bool[,] Mask { get; set; } | ||||||
|  | 
 | ||||||
|  |         public uint TotalDiskUsage { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										45
									
								
								src/Velopack.IcoLib/Model/IcoFrameEncoding.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								src/Velopack.IcoLib/Model/IcoFrameEncoding.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | |||||||
|  | namespace Ico.Model | ||||||
|  | { | ||||||
|  |     public enum IcoEncodingType | ||||||
|  |     { | ||||||
|  |         Bitmap, | ||||||
|  |         Png, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public enum BitmapEncoding | ||||||
|  |     { | ||||||
|  |         Pixel_indexed1, | ||||||
|  |         Pixel_indexed2, | ||||||
|  |         Pixel_indexed4, | ||||||
|  |         Pixel_indexed8, | ||||||
|  |         Pixel_rgb15, | ||||||
|  |         Pixel_rgb24, | ||||||
|  |         Pixel_0rgb32, | ||||||
|  |         Pixel_argb32, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public class IcoFrameEncoding | ||||||
|  |     { | ||||||
|  |         public IcoEncodingType Type { get; set; } | ||||||
|  | 
 | ||||||
|  |         // In the ICO header | ||||||
|  | 
 | ||||||
|  |         public uint ClaimedBitDepth { get; set; } | ||||||
|  | 
 | ||||||
|  |         public uint ClaimedWidth { get; set; } | ||||||
|  | 
 | ||||||
|  |         public uint ClaimedHeight { get; set; } | ||||||
|  | 
 | ||||||
|  |         public uint ActualWidth { get; set; } | ||||||
|  | 
 | ||||||
|  |         public uint ActualHeight { get; set; } | ||||||
|  | 
 | ||||||
|  |         // Bitmap only | ||||||
|  | 
 | ||||||
|  |         public BitmapEncoding PixelFormat { get; set; } | ||||||
|  | 
 | ||||||
|  |         public uint ActualBitDepth { get; set; } | ||||||
|  | 
 | ||||||
|  |         public uint PaletteSize { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										29
									
								
								src/Velopack.IcoLib/Model/ParseContext.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								src/Velopack.IcoLib/Model/ParseContext.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | |||||||
|  | using Ico.Codecs; | ||||||
|  | using Ico.Host; | ||||||
|  | using Ico.Validation; | ||||||
|  | using SixLabors.ImageSharp.Formats.Png; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Model | ||||||
|  | { | ||||||
|  |     public class ParseContext | ||||||
|  |     { | ||||||
|  |         public string FullPath { get; set; } | ||||||
|  | 
 | ||||||
|  |         public string DisplayedPath { get; set; } | ||||||
|  | 
 | ||||||
|  |         public uint? ImageDirectoryIndex { get; set; } | ||||||
|  | 
 | ||||||
|  |         public List<IcoFrame> GeneratedFrames { get; set; } | ||||||
|  | 
 | ||||||
|  |         public PngEncoder PngEncoder { get; set; } | ||||||
|  | 
 | ||||||
|  |         public StrictnessPolicy MaskedImagePixelEmitOptions { get; set; } | ||||||
|  | 
 | ||||||
|  |         public StrictnessPolicy AllowPaletteTruncation { get; set; } | ||||||
|  | 
 | ||||||
|  |         public IErrorReporter Reporter { get; set; } | ||||||
|  | 
 | ||||||
|  |         public IcoErrorCode LastEncodeError { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										23
									
								
								src/Velopack.IcoLib/Model/PngFileEncoding.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								src/Velopack.IcoLib/Model/PngFileEncoding.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | namespace Ico.Model | ||||||
|  | { | ||||||
|  |     public enum PngColorType | ||||||
|  |     { | ||||||
|  |         Grayscale = 0, | ||||||
|  |         RGB = 2, | ||||||
|  |         Palette = 3, | ||||||
|  |         GrayscaleAlpha = 4, | ||||||
|  |         RGBA = 6, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public class PngFileEncoding | ||||||
|  |     { | ||||||
|  |         // 1, 2, 4, 8, or 16 | ||||||
|  |         public uint BitsPerChannel { get; set; } | ||||||
|  | 
 | ||||||
|  |         public PngColorType ColorType { get; set; } | ||||||
|  | 
 | ||||||
|  |         public uint Width { get; set; } | ||||||
|  | 
 | ||||||
|  |         public uint Height { get; set; } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										4
									
								
								src/Velopack.IcoLib/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								src/Velopack.IcoLib/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | # IcoLib | ||||||
|  | IcoLib is a library for creating, extracting, and managing ICO files. It is written in C# and is compatible with .NET Standard 2.0. | ||||||
|  |  | ||||||
|  | Find the original project here https://github.com/jtippet/IcoTools | ||||||
							
								
								
									
										60
									
								
								src/Velopack.IcoLib/Validation/IcoErrorCode.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								src/Velopack.IcoLib/Validation/IcoErrorCode.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | |||||||
|  | using System; | ||||||
|  | using System.Collections.Generic; | ||||||
|  | using System.Text; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Validation | ||||||
|  | { | ||||||
|  |     public enum IcoErrorCode | ||||||
|  |     { | ||||||
|  |         NoError = 0, | ||||||
|  | 
 | ||||||
|  |         // 100 - 199: invalid ICO file format | ||||||
|  |         InvalidIcoHeader_idReserved = 100, | ||||||
|  |         InvalidIconHeader_idType = 101, | ||||||
|  |         InvalidFrameHeader_bReserved = 110, | ||||||
|  |         InvalidFrameHeader_wPlanes = 111, | ||||||
|  |         InvalidFrameHeader_dwBytesInRes = 112, | ||||||
|  |         InvalidFrameHeader_dwImageOffset = 113, | ||||||
|  |         TooManyFrames = 120, | ||||||
|  |         InvalidBitapInfoHeader_ciSize = 130, | ||||||
|  |         InvalidBitapInfoHeader_biXPelsPerMeter = 131, | ||||||
|  |         InvalidBitapInfoHeader_biYPelsPerMeter = 132, | ||||||
|  |         InvalidBitapInfoHeader_biBitCount = 133, | ||||||
|  |         InvalidBitapInfoHeader_biClrUsed = 134, | ||||||
|  | 
 | ||||||
|  |         // 200 - 299: nonstandard or nonportable ICO file format | ||||||
|  |         ZeroFrames = 200, | ||||||
|  |         DuplicateFrameTypes = 201, | ||||||
|  |         MismatchedHeight = 210, | ||||||
|  |         MismatchedWidth = 211, | ||||||
|  |         NonzeroAlpha = 220, | ||||||
|  |         MaskedPixelWithColor = 221, | ||||||
|  |         NoMaskedPixels = 222, | ||||||
|  |         IndexedColorOutOfBounds = 230, | ||||||
|  |         UndersizedColorTable = 231, | ||||||
|  |         PngNot32Bit = 240, | ||||||
|  |         PngNotRGBA32 = 241, | ||||||
|  |         NotSquare = 256, | ||||||
|  | 
 | ||||||
|  |         // 300 - 399: tool limitations | ||||||
|  |         FileTooLarge = 300, | ||||||
|  |         BitfieldCompressionNotSupported = 310, | ||||||
|  |         BitmapCompressionNotSupported = 311, | ||||||
|  | 
 | ||||||
|  |         // 400 - 499: usage or environmental error | ||||||
|  |         FileExists = 403, | ||||||
|  |         FileNotFound = 404, | ||||||
|  |         UnsupportedCodec = 410, | ||||||
|  |         UnsupportedBitmapEncoding = 411, | ||||||
|  |         OnlySupportedOnBitmaps = 412, | ||||||
|  |         BitmapMaskWrongDimensions = 420, | ||||||
|  |         BitampMaskWrongColors = 421, | ||||||
|  |         InvalidFrameIndex = 430, | ||||||
|  |         TooManyColorsForBitDepth = 431, | ||||||
|  |         NotPng = 440, | ||||||
|  |         PngBadIHDR = 441, | ||||||
|  |         PngIllegalInputDimensions = 445, | ||||||
|  |         PngIllegalInputDepth = 446, | ||||||
|  |         PngIllegalColorType = 447, | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										56
									
								
								src/Velopack.IcoLib/Validation/InvalidIcoFileException.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								src/Velopack.IcoLib/Validation/InvalidIcoFileException.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | |||||||
|  | using Ico.Model; | ||||||
|  | using System; | ||||||
|  | using System.Runtime.Serialization; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Validation | ||||||
|  | { | ||||||
|  |     [Serializable] | ||||||
|  |     public class InvalidIcoFileException : Exception | ||||||
|  |     { | ||||||
|  |         public ParseContext Context { get; private set; } | ||||||
|  | 
 | ||||||
|  |         public IcoErrorCode ErrorCode { get; private set; } | ||||||
|  | 
 | ||||||
|  |         public InvalidIcoFileException() | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public InvalidIcoFileException(string message) : base(message) | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public InvalidIcoFileException(string message, Exception innerException) : base(message, innerException) | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public InvalidIcoFileException(IcoErrorCode errorCode, string message, ParseContext context) : this(message) | ||||||
|  |         { | ||||||
|  |             ErrorCode = errorCode; | ||||||
|  |             Context = context; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         protected InvalidIcoFileException(SerializationInfo info, StreamingContext context) : base(info, context) | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public override string ToString() | ||||||
|  |         { | ||||||
|  |             if (Context != null) | ||||||
|  |             { | ||||||
|  |                 if (Context.DisplayedPath != null && Context.ImageDirectoryIndex == null) | ||||||
|  |                 { | ||||||
|  |                     return base.ToString() + $"\nFile: \"{Context.DisplayedPath}\""; | ||||||
|  |                 } | ||||||
|  |                 else if (Context.DisplayedPath != null && Context.ImageDirectoryIndex == null) | ||||||
|  |                 { | ||||||
|  |                     return base.ToString() + $"\nFile: \"{Context.DisplayedPath}\""; | ||||||
|  |                 } | ||||||
|  |                 else if (Context.DisplayedPath != null && Context.ImageDirectoryIndex.HasValue) | ||||||
|  |                 { | ||||||
|  |                     return base.ToString() + $"\nFile: \"{Context.DisplayedPath}\"\nImage directory index: #{Context.ImageDirectoryIndex.Value}"; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             return base.ToString(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										32
									
								
								src/Velopack.IcoLib/Validation/InvalidPngFileException.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								src/Velopack.IcoLib/Validation/InvalidPngFileException.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | |||||||
|  | using System; | ||||||
|  | using System.Runtime.Serialization; | ||||||
|  | 
 | ||||||
|  | namespace Ico.Validation | ||||||
|  | { | ||||||
|  |     public class InvalidPngFileException : Exception | ||||||
|  |     { | ||||||
|  |         public IcoErrorCode ErrorCode { get; private set; } | ||||||
|  | 
 | ||||||
|  |         public InvalidPngFileException() | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public InvalidPngFileException(string message) : base(message) | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public InvalidPngFileException(IcoErrorCode errorCode, string message) : base(message) | ||||||
|  |         { | ||||||
|  |             ErrorCode = errorCode; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         public InvalidPngFileException(string message, Exception innerException) : base(message, innerException) | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         protected InvalidPngFileException(SerializationInfo info, StreamingContext context) : base(info, context) | ||||||
|  |         { | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										17
									
								
								src/Velopack.IcoLib/Velopack.IcoLib.csproj
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								src/Velopack.IcoLib/Velopack.IcoLib.csproj
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | <Project Sdk="Microsoft.NET.Sdk"> | ||||||
|  |  | ||||||
|  |   <PropertyGroup> | ||||||
|  |     <TargetFramework>netstandard2.0</TargetFramework> | ||||||
|  |     <RootNamespace>Ico</RootNamespace> | ||||||
|  |     <LangVersion>7.3</LangVersion> | ||||||
|  |     <Version>1.1.1</Version> | ||||||
|  |     <Authors>Jeffrey Tippet</Authors> | ||||||
|  |     <Product>IcoTools</Product> | ||||||
|  |     <Copyright>Copyright 2019, Jeffrey Tippet.  All rights reserved.</Copyright> | ||||||
|  |   </PropertyGroup> | ||||||
|  |  | ||||||
|  |   <ItemGroup> | ||||||
|  |     <PackageReference Include="SixLabors.ImageSharp" Version="2.1.8" /> | ||||||
|  |   </ItemGroup> | ||||||
|  |  | ||||||
|  | </Project> | ||||||
							
								
								
									
										93
									
								
								src/Velopack.IcoLib/packages.lock.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								src/Velopack.IcoLib/packages.lock.json
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,93 @@ | |||||||
|  | { | ||||||
|  |   "version": 1, | ||||||
|  |   "dependencies": { | ||||||
|  |     ".NETStandard,Version=v2.0": { | ||||||
|  |       "Microsoft.SourceLink.GitHub": { | ||||||
|  |         "type": "Direct", | ||||||
|  |         "requested": "[8.0.0, )", | ||||||
|  |         "resolved": "8.0.0", | ||||||
|  |         "contentHash": "G5q7OqtwIyGTkeIOAc3u2ZuV/kicQaec5EaRnc0pIeSnh9LUjj+PYQrJYBURvDt7twGl2PKA7nSN0kz1Zw5bnQ==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "Microsoft.Build.Tasks.Git": "8.0.0", | ||||||
|  |           "Microsoft.SourceLink.Common": "8.0.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "Nerdbank.GitVersioning": { | ||||||
|  |         "type": "Direct", | ||||||
|  |         "requested": "[3.6.*, )", | ||||||
|  |         "resolved": "3.6.139", | ||||||
|  |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|  |       }, | ||||||
|  |       "NETStandard.Library": { | ||||||
|  |         "type": "Direct", | ||||||
|  |         "requested": "[2.0.3, )", | ||||||
|  |         "resolved": "2.0.3", | ||||||
|  |         "contentHash": "st47PosZSHrjECdjeIzZQbzivYBJFv6P2nv4cj2ypdI204DO+vZ7l5raGMiX4eXMJ53RfOIg+/s4DHVZ54Nu2A==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "Microsoft.NETCore.Platforms": "1.1.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "SixLabors.ImageSharp": { | ||||||
|  |         "type": "Direct", | ||||||
|  |         "requested": "[2.1.8, )", | ||||||
|  |         "resolved": "2.1.8", | ||||||
|  |         "contentHash": "ML1++vactR5xMW36wHA2nAX3vc6VpR62qYnpopQdLNGbP8BAJ/ckv1IrGIIFRUEQ0JS9EgWr1y1gEa/81f+HaA==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Buffers": "4.5.1", | ||||||
|  |           "System.Memory": "4.5.4", | ||||||
|  |           "System.Numerics.Vectors": "4.5.0", | ||||||
|  |           "System.Runtime.CompilerServices.Unsafe": "5.0.0", | ||||||
|  |           "System.Text.Encoding.CodePages": "5.0.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "Microsoft.Build.Tasks.Git": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "8.0.0", | ||||||
|  |         "contentHash": "bZKfSIKJRXLTuSzLudMFte/8CempWjVamNUR5eHJizsy+iuOuO/k2gnh7W0dHJmYY0tBf+gUErfluCv5mySAOQ==" | ||||||
|  |       }, | ||||||
|  |       "Microsoft.NETCore.Platforms": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "1.1.0", | ||||||
|  |         "contentHash": "kz0PEW2lhqygehI/d6XsPCQzD7ff7gUJaVGPVETX611eadGsA3A877GdSlU0LRVMCTH/+P3o2iDTak+S08V2+A==" | ||||||
|  |       }, | ||||||
|  |       "Microsoft.SourceLink.Common": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "8.0.0", | ||||||
|  |         "contentHash": "dk9JPxTCIevS75HyEQ0E4OVAFhB2N+V9ShCXf8Q6FkUQZDkgLI12y679Nym1YqsiSysuQskT7Z+6nUf3yab6Vw==" | ||||||
|  |       }, | ||||||
|  |       "System.Buffers": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "4.5.1", | ||||||
|  |         "contentHash": "Rw7ijyl1qqRS0YQD/WycNst8hUUMgrMH4FCn1nNm27M4VxchZ1js3fVjQaANHO5f3sN4isvP4a+Met9Y4YomAg==" | ||||||
|  |       }, | ||||||
|  |       "System.Memory": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "4.5.4", | ||||||
|  |         "contentHash": "1MbJTHS1lZ4bS4FmsJjnuGJOu88ZzTT2rLvrhW7Ygic+pC0NWA+3hgAen0HRdsocuQXCkUTdFn9yHJJhsijDXw==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Buffers": "4.5.1", | ||||||
|  |           "System.Numerics.Vectors": "4.4.0", | ||||||
|  |           "System.Runtime.CompilerServices.Unsafe": "4.5.3" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "System.Numerics.Vectors": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "4.5.0", | ||||||
|  |         "contentHash": "QQTlPTl06J/iiDbJCiepZ4H//BVraReU4O4EoRw1U02H5TLUIT7xn3GnDp9AXPSlJUDyFs4uWjWafNX6WrAojQ==" | ||||||
|  |       }, | ||||||
|  |       "System.Runtime.CompilerServices.Unsafe": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "5.0.0", | ||||||
|  |         "contentHash": "ZD9TMpsmYJLrxbbmdvhwt9YEgG5WntEnZ/d1eH8JBX9LBp+Ju8BSBhUGbZMNVHHomWo2KVImJhTDl2hIgw/6MA==" | ||||||
|  |       }, | ||||||
|  |       "System.Text.Encoding.CodePages": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "5.0.0", | ||||||
|  |         "contentHash": "NyscU59xX6Uo91qvhOs2Ccho3AR2TnZPomo1Z0K6YpyztBPM/A5VbkzOO19sy3A3i1TtEnTxA7bCe3Us+r5MWg==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Runtime.CompilerServices.Unsafe": "5.0.0" | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,90 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.AppHost |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// An instance of this exception is thrown when an AppHost binary update |  | ||||||
|     /// fails due to known user errors. |  | ||||||
|     /// </summary> |  | ||||||
|     public class AppHostUpdateException : Exception |  | ||||||
|     { |  | ||||||
|         internal AppHostUpdateException(string message = null) |  | ||||||
|             : base(message) |  | ||||||
|         { |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// <summary> |  | ||||||
|     /// The application host executable cannot be customized because adding resources requires |  | ||||||
|     /// that the build be performed on Windows (excluding Nano Server). |  | ||||||
|     /// </summary> |  | ||||||
|     public sealed class AppHostCustomizationUnsupportedOSException : AppHostUpdateException |  | ||||||
|     { |  | ||||||
|         internal AppHostCustomizationUnsupportedOSException() |  | ||||||
|         { |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// <summary> |  | ||||||
|     /// The MachO application host executable cannot be customized because |  | ||||||
|     /// it was not in the expected format |  | ||||||
|     /// </summary> |  | ||||||
|     public sealed class AppHostMachOFormatException : AppHostUpdateException |  | ||||||
|     { |  | ||||||
|         public readonly MachOFormatError Error; |  | ||||||
| 
 |  | ||||||
|         internal AppHostMachOFormatException(MachOFormatError error) |  | ||||||
|         { |  | ||||||
|             Error = error; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// <summary> |  | ||||||
|     /// Unable to use the input file as application host executable because it's not a |  | ||||||
|     /// Windows executable for the CUI (Console) subsystem. |  | ||||||
|     /// </summary> |  | ||||||
|     public sealed class AppHostNotCUIException : AppHostUpdateException |  | ||||||
|     { |  | ||||||
|         internal AppHostNotCUIException() |  | ||||||
|         { |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// <summary> |  | ||||||
|     ///  Unable to use the input file as an application host executable |  | ||||||
|     ///  because it's not a Windows PE file |  | ||||||
|     /// </summary> |  | ||||||
|     public sealed class AppHostNotPEFileException : AppHostUpdateException |  | ||||||
|     { |  | ||||||
|         internal AppHostNotPEFileException() |  | ||||||
|         { |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// <summary> |  | ||||||
|     /// Unable to sign the apphost binary. |  | ||||||
|     /// </summary> |  | ||||||
|     public sealed class AppHostSigningException : AppHostUpdateException |  | ||||||
|     { |  | ||||||
|         public readonly int ExitCode; |  | ||||||
| 
 |  | ||||||
|         internal AppHostSigningException(int exitCode, string signingErrorMessage) |  | ||||||
|             : base(signingErrorMessage) |  | ||||||
|         { |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     /// <summary> |  | ||||||
|     /// Given app file name is longer than 1024 bytes |  | ||||||
|     /// </summary> |  | ||||||
|     public sealed class AppNameTooLongException : AppHostUpdateException |  | ||||||
|     { |  | ||||||
|         public string LongName { get; } |  | ||||||
| 
 |  | ||||||
|         internal AppNameTooLongException(string name) |  | ||||||
|         { |  | ||||||
|             LongName = name; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,168 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| using System.IO.MemoryMappedFiles; |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.AppHost |  | ||||||
| { |  | ||||||
|     public static class BinaryUtils |  | ||||||
|     { |  | ||||||
|         internal static unsafe void SearchAndReplace( |  | ||||||
|             MemoryMappedViewAccessor accessor, |  | ||||||
|             byte[] searchPattern, |  | ||||||
|             byte[] patternToReplace, |  | ||||||
|             bool pad0s = true) |  | ||||||
|         { |  | ||||||
|             byte* pointer = null; |  | ||||||
| 
 |  | ||||||
|             try { |  | ||||||
|                 accessor.SafeMemoryMappedViewHandle.AcquirePointer(ref pointer); |  | ||||||
|                 byte* bytes = pointer + accessor.PointerOffset; |  | ||||||
| 
 |  | ||||||
|                 int position = KMPSearch(searchPattern, bytes, accessor.Capacity); |  | ||||||
|                 if (position < 0) { |  | ||||||
|                     throw new PlaceHolderNotFoundInAppHostException(searchPattern); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 accessor.WriteArray( |  | ||||||
|                     position: position, |  | ||||||
|                     array: patternToReplace, |  | ||||||
|                     offset: 0, |  | ||||||
|                     count: patternToReplace.Length); |  | ||||||
| 
 |  | ||||||
|                 if (pad0s) { |  | ||||||
|                     Pad0(searchPattern, patternToReplace, bytes, position); |  | ||||||
|                 } |  | ||||||
|             } finally { |  | ||||||
|                 if (pointer != null) { |  | ||||||
|                     accessor.SafeMemoryMappedViewHandle.ReleasePointer(); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private static unsafe void Pad0(byte[] searchPattern, byte[] patternToReplace, byte* bytes, int offset) |  | ||||||
|         { |  | ||||||
|             if (patternToReplace.Length < searchPattern.Length) { |  | ||||||
|                 for (int i = patternToReplace.Length; i < searchPattern.Length; i++) { |  | ||||||
|                     bytes[i + offset] = 0x0; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static unsafe void SearchAndReplace( |  | ||||||
|             string filePath, |  | ||||||
|             byte[] searchPattern, |  | ||||||
|             byte[] patternToReplace, |  | ||||||
|             bool pad0s = true) |  | ||||||
|         { |  | ||||||
|             using (var mappedFile = MemoryMappedFile.CreateFromFile(filePath)) { |  | ||||||
|                 using (var accessor = mappedFile.CreateViewAccessor()) { |  | ||||||
|                     SearchAndReplace(accessor, searchPattern, patternToReplace, pad0s); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static unsafe int SearchInFile(MemoryMappedViewAccessor accessor, byte[] searchPattern) |  | ||||||
|         { |  | ||||||
|             var safeBuffer = accessor.SafeMemoryMappedViewHandle; |  | ||||||
|             return KMPSearch(searchPattern, (byte*) safeBuffer.DangerousGetHandle(), (int) safeBuffer.ByteLength); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static unsafe int SearchInFile(string filePath, byte[] searchPattern) |  | ||||||
|         { |  | ||||||
|             using (var mappedFile = MemoryMappedFile.CreateFromFile(filePath)) { |  | ||||||
|                 using (var accessor = mappedFile.CreateViewAccessor(0, 0, MemoryMappedFileAccess.Read)) { |  | ||||||
|                     return SearchInFile(accessor, searchPattern); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         // See: https://en.wikipedia.org/wiki/Knuth%E2%80%93Morris%E2%80%93Pratt_algorithm |  | ||||||
|         private static int[] ComputeKMPFailureFunction(byte[] pattern) |  | ||||||
|         { |  | ||||||
|             int[] table = new int[pattern.Length]; |  | ||||||
|             if (pattern.Length >= 1) { |  | ||||||
|                 table[0] = -1; |  | ||||||
|             } |  | ||||||
|             if (pattern.Length >= 2) { |  | ||||||
|                 table[1] = 0; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             int pos = 2; |  | ||||||
|             int cnd = 0; |  | ||||||
|             while (pos < pattern.Length) { |  | ||||||
|                 if (pattern[pos - 1] == pattern[cnd]) { |  | ||||||
|                     table[pos] = cnd + 1; |  | ||||||
|                     cnd++; |  | ||||||
|                     pos++; |  | ||||||
|                 } else if (cnd > 0) { |  | ||||||
|                     cnd = table[cnd]; |  | ||||||
|                 } else { |  | ||||||
|                     table[pos] = 0; |  | ||||||
|                     pos++; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             return table; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         // See: https://en.wikipedia.org/wiki/Knuth%E2%80%93Morris%E2%80%93Pratt_algorithm |  | ||||||
|         private static unsafe int KMPSearch(byte[] pattern, byte* bytes, long bytesLength) |  | ||||||
|         { |  | ||||||
|             int m = 0; |  | ||||||
|             int i = 0; |  | ||||||
|             int[] table = ComputeKMPFailureFunction(pattern); |  | ||||||
| 
 |  | ||||||
|             while (m + i < bytesLength) { |  | ||||||
|                 if (pattern[i] == bytes[m + i]) { |  | ||||||
|                     if (i == pattern.Length - 1) { |  | ||||||
|                         return m; |  | ||||||
|                     } |  | ||||||
|                     i++; |  | ||||||
|                 } else { |  | ||||||
|                     if (table[i] > -1) { |  | ||||||
|                         m = m + i - table[i]; |  | ||||||
|                         i = table[i]; |  | ||||||
|                     } else { |  | ||||||
|                         m++; |  | ||||||
|                         i = 0; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return -1; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static void CopyFile(string sourcePath, string destinationPath) |  | ||||||
|         { |  | ||||||
|             var destinationDirectory = new FileInfo(destinationPath).Directory.FullName; |  | ||||||
|             if (!Directory.Exists(destinationDirectory)) { |  | ||||||
|                 Directory.CreateDirectory(destinationDirectory); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             // Copy file to destination path so it inherits the same attributes/permissions. |  | ||||||
|             File.Copy(sourcePath, destinationPath, overwrite: true); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         internal static void WriteToStream(MemoryMappedViewAccessor sourceViewAccessor, FileStream fileStream, long length) |  | ||||||
|         { |  | ||||||
|             int pos = 0; |  | ||||||
|             int bufSize = 16384; //16K |  | ||||||
| 
 |  | ||||||
|             byte[] buf = new byte[bufSize]; |  | ||||||
|             length = Math.Min(length, sourceViewAccessor.Capacity); |  | ||||||
|             do { |  | ||||||
|                 int bytesRequested = Math.Min((int) length - pos, bufSize); |  | ||||||
|                 if (bytesRequested <= 0) { |  | ||||||
|                     break; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 int bytesRead = sourceViewAccessor.ReadArray(pos, buf, 0, bytesRequested); |  | ||||||
|                 if (bytesRead > 0) { |  | ||||||
|                     fileStream.Write(buf, 0, bytesRead); |  | ||||||
|                     pos += bytesRead; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             while (true); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,49 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.AppHost |  | ||||||
| { |  | ||||||
|     internal static class ElfUtils |  | ||||||
|     { |  | ||||||
|         // The Linux Headers are copied from elf.h |  | ||||||
| 
 |  | ||||||
| #pragma warning disable 0649 |  | ||||||
|         private struct ElfHeader |  | ||||||
|         { |  | ||||||
|             private byte EI_MAG0; |  | ||||||
|             private byte EI_MAG1; |  | ||||||
|             private byte EI_MAG2; |  | ||||||
|             private byte EI_MAG3; |  | ||||||
| 
 |  | ||||||
|             public bool IsValid() |  | ||||||
|             { |  | ||||||
|                 return EI_MAG0 == 0x7f && |  | ||||||
|                        EI_MAG1 == 0x45 && |  | ||||||
|                        EI_MAG2 == 0x4C && |  | ||||||
|                        EI_MAG3 == 0x46; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|         } |  | ||||||
| #pragma warning restore 0649 |  | ||||||
| 
 |  | ||||||
|         public static bool IsElfImage(string filePath) |  | ||||||
|         { |  | ||||||
|             using (BinaryReader reader = new BinaryReader(File.OpenRead(filePath))) |  | ||||||
|             { |  | ||||||
|                 if (reader.BaseStream.Length < 16) // EI_NIDENT = 16 |  | ||||||
|                 { |  | ||||||
|                     return false; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 byte[] eIdent = reader.ReadBytes(4); |  | ||||||
| 
 |  | ||||||
|                 // Check that the first four bytes are 0x7f, 'E', 'L', 'F' |  | ||||||
|                 return eIdent[0] == 0x7f && |  | ||||||
|                        eIdent[1] == 0x45 && |  | ||||||
|                        eIdent[2] == 0x4C && |  | ||||||
|                        eIdent[3] == 0x46; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Represents an exception thrown because of a Win32 error |  | ||||||
|     /// </summary> |  | ||||||
|     public class HResultException : Exception |  | ||||||
|     { |  | ||||||
|         public readonly int Win32HResult; |  | ||||||
|         public HResultException(int hResult) : base(hResult.ToString("X4")) |  | ||||||
|         { |  | ||||||
|             Win32HResult = hResult; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,277 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| using System.ComponentModel; |  | ||||||
| using System.IO.MemoryMappedFiles; |  | ||||||
| using System.Runtime.InteropServices; |  | ||||||
| using System.Text; |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.AppHost |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Embeds the App Name into the AppHost.exe |  | ||||||
|     /// If an apphost is a single-file bundle, updates the location of the bundle headers. |  | ||||||
|     /// </summary> |  | ||||||
|     public static class HostWriter |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// hash value embedded in default apphost executable in a place where the path to the app binary should be stored. |  | ||||||
|         /// </summary> |  | ||||||
|         private const string AppBinaryPathPlaceholder = "c3ab8ff13720e8ad9047dd39466b3c8974e592c2fa383d4a3960714caef0c4f2"; |  | ||||||
|         private static readonly byte[] AppBinaryPathPlaceholderSearchValue = Encoding.UTF8.GetBytes(AppBinaryPathPlaceholder); |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Create an AppHost with embedded configuration of app binary location |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="appHostSourceFilePath">The path of Apphost template, which has the place holder</param> |  | ||||||
|         /// <param name="appHostDestinationFilePath">The destination path for desired location to place, including the file name</param> |  | ||||||
|         /// <param name="appBinaryFilePath">Full path to app binary or relative path to the result apphost file</param> |  | ||||||
|         /// <param name="windowsGraphicalUserInterface">Specify whether to set the subsystem to GUI. Only valid for PE apphosts.</param> |  | ||||||
|         /// <param name="assemblyToCopyResorcesFrom">Path to the intermediate assembly, used for copying resources to PE apphosts.</param> |  | ||||||
|         /// <param name="enableMacOSCodeSign">Sign the app binary using codesign with an anonymous certificate.</param> |  | ||||||
|         public static void CreateAppHost( |  | ||||||
|             string appHostSourceFilePath, |  | ||||||
|             string appHostDestinationFilePath, |  | ||||||
|             string appBinaryFilePath, |  | ||||||
|             bool windowsGraphicalUserInterface = false, |  | ||||||
|             string assemblyToCopyResorcesFrom = null, |  | ||||||
|             bool enableMacOSCodeSign = false) |  | ||||||
|         { |  | ||||||
|             var bytesToWrite = Encoding.UTF8.GetBytes(appBinaryFilePath); |  | ||||||
|             if (bytesToWrite.Length > 1024) |  | ||||||
|             { |  | ||||||
|                 throw new AppNameTooLongException(appBinaryFilePath); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             bool appHostIsPEImage = false; |  | ||||||
| 
 |  | ||||||
|             void RewriteAppHost(MemoryMappedViewAccessor accessor) |  | ||||||
|             { |  | ||||||
|                 // Re-write the destination apphost with the proper contents. |  | ||||||
|                 BinaryUtils.SearchAndReplace(accessor, AppBinaryPathPlaceholderSearchValue, bytesToWrite); |  | ||||||
| 
 |  | ||||||
|                 appHostIsPEImage = PEUtils.IsPEImage(accessor); |  | ||||||
| 
 |  | ||||||
|                 if (windowsGraphicalUserInterface) |  | ||||||
|                 { |  | ||||||
|                     if (!appHostIsPEImage) |  | ||||||
|                     { |  | ||||||
|                         throw new AppHostNotPEFileException(); |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     PEUtils.SetWindowsGraphicalUserInterfaceBit(accessor); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             void UpdateResources() |  | ||||||
|             { |  | ||||||
|                 if (assemblyToCopyResorcesFrom != null && appHostIsPEImage) |  | ||||||
|                 { |  | ||||||
|                     if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows) && ResourceUpdater.IsSupportedOS()) |  | ||||||
|                     { |  | ||||||
|                         // Copy resources from managed dll to the apphost |  | ||||||
|                         new ResourceUpdater(appHostDestinationFilePath) |  | ||||||
|                             .AddResourcesFromPEImage(assemblyToCopyResorcesFrom) |  | ||||||
|                             .Update(); |  | ||||||
|                     } |  | ||||||
|                     else |  | ||||||
|                     { |  | ||||||
|                         throw new AppHostCustomizationUnsupportedOSException(); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 RetryUtil.RetryOnIOError(() => |  | ||||||
|                 { |  | ||||||
|                     FileStream appHostSourceStream = null; |  | ||||||
|                     MemoryMappedFile memoryMappedFile = null; |  | ||||||
|                     MemoryMappedViewAccessor memoryMappedViewAccessor = null; |  | ||||||
|                     try |  | ||||||
|                     { |  | ||||||
|                         // Open the source host file. |  | ||||||
|                         appHostSourceStream = new FileStream(appHostSourceFilePath, FileMode.Open, FileAccess.Read, FileShare.Read); |  | ||||||
|                         memoryMappedFile = MemoryMappedFile.CreateFromFile(appHostSourceStream, null, 0, MemoryMappedFileAccess.Read, HandleInheritability.None, true); |  | ||||||
|                         memoryMappedViewAccessor = memoryMappedFile.CreateViewAccessor(0, 0, MemoryMappedFileAccess.CopyOnWrite); |  | ||||||
| 
 |  | ||||||
|                         // Get the size of the source app host to ensure that we don't write extra data to the destination. |  | ||||||
|                         // On Windows, the size of the view accessor is rounded up to the next page boundary. |  | ||||||
|                         long sourceAppHostLength = appHostSourceStream.Length; |  | ||||||
| 
 |  | ||||||
|                         // Transform the host file in-memory. |  | ||||||
|                         RewriteAppHost(memoryMappedViewAccessor); |  | ||||||
| 
 |  | ||||||
|                         // Save the transformed host. |  | ||||||
|                         using (FileStream fileStream = new FileStream(appHostDestinationFilePath, FileMode.Create)) |  | ||||||
|                         { |  | ||||||
|                             BinaryUtils.WriteToStream(memoryMappedViewAccessor, fileStream, sourceAppHostLength); |  | ||||||
| 
 |  | ||||||
|                             // Remove the signature from MachO hosts. |  | ||||||
|                             if (!appHostIsPEImage) |  | ||||||
|                             { |  | ||||||
|                                 MachOUtils.RemoveSignature(fileStream); |  | ||||||
|                             } |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     finally |  | ||||||
|                     { |  | ||||||
|                         memoryMappedViewAccessor?.Dispose(); |  | ||||||
|                         memoryMappedFile?.Dispose(); |  | ||||||
|                         appHostSourceStream?.Dispose(); |  | ||||||
|                     } |  | ||||||
|                 }); |  | ||||||
| 
 |  | ||||||
|                 RetryUtil.RetryOnWin32Error(UpdateResources); |  | ||||||
| 
 |  | ||||||
|                 if (!RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) |  | ||||||
|                 { |  | ||||||
|                     var filePermissionOctal = Convert.ToInt32("755", 8); // -rwxr-xr-x |  | ||||||
|                     const int EINTR = 4; |  | ||||||
|                     int chmodReturnCode = 0; |  | ||||||
| 
 |  | ||||||
|                     do |  | ||||||
|                     { |  | ||||||
|                         chmodReturnCode = chmod(appHostDestinationFilePath, filePermissionOctal); |  | ||||||
|                     } |  | ||||||
|                     while (chmodReturnCode == -1 && Marshal.GetLastWin32Error() == EINTR); |  | ||||||
| 
 |  | ||||||
|                     if (chmodReturnCode == -1) |  | ||||||
|                     { |  | ||||||
|                         throw new Win32Exception(Marshal.GetLastWin32Error(), $"Could not set file permission {filePermissionOctal} for {appHostDestinationFilePath}."); |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     if (enableMacOSCodeSign && RuntimeInformation.IsOSPlatform(OSPlatform.OSX) && HostModelUtils.IsCodesignAvailable()) |  | ||||||
|                     { |  | ||||||
|                         (int exitCode, string stdErr) = HostModelUtils.RunCodesign("-s -", appHostDestinationFilePath); |  | ||||||
|                         if (exitCode != 0) |  | ||||||
|                         { |  | ||||||
|                             throw new AppHostSigningException(exitCode, stdErr); |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             catch (Exception ex) |  | ||||||
|             { |  | ||||||
|                 // Delete the destination file so we don't leave an unmodified apphost |  | ||||||
|                 try |  | ||||||
|                 { |  | ||||||
|                     File.Delete(appHostDestinationFilePath); |  | ||||||
|                 } |  | ||||||
|                 catch (Exception failedToDeleteEx) |  | ||||||
|                 { |  | ||||||
|                     throw new AggregateException(ex, failedToDeleteEx); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 throw; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Set the current AppHost as a single-file bundle. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="appHostPath">The path of Apphost template, which has the place holder</param> |  | ||||||
|         /// <param name="bundleHeaderOffset">The offset to the location of bundle header</param> |  | ||||||
|         public static void SetAsBundle( |  | ||||||
|             string appHostPath, |  | ||||||
|             long bundleHeaderOffset) |  | ||||||
|         { |  | ||||||
|             byte[] bundleHeaderPlaceholder = { |  | ||||||
|                 // 8 bytes represent the bundle header-offset |  | ||||||
|                 // Zero for non-bundle apphosts (default). |  | ||||||
|                 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, |  | ||||||
|                 // 32 bytes represent the bundle signature: SHA-256 for ".net core bundle" |  | ||||||
|                 0x8b, 0x12, 0x02, 0xb9, 0x6a, 0x61, 0x20, 0x38, |  | ||||||
|                 0x72, 0x7b, 0x93, 0x02, 0x14, 0xd7, 0xa0, 0x32, |  | ||||||
|                 0x13, 0xf5, 0xb9, 0xe6, 0xef, 0xae, 0x33, 0x18, |  | ||||||
|                 0xee, 0x3b, 0x2d, 0xce, 0x24, 0xb3, 0x6a, 0xae |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             // Re-write the destination apphost with the proper contents. |  | ||||||
|             RetryUtil.RetryOnIOError(() => |  | ||||||
|                 BinaryUtils.SearchAndReplace(appHostPath, |  | ||||||
|                                              bundleHeaderPlaceholder, |  | ||||||
|                                              BitConverter.GetBytes(bundleHeaderOffset), |  | ||||||
|                                              pad0s: false)); |  | ||||||
| 
 |  | ||||||
|             RetryUtil.RetryOnIOError(() => |  | ||||||
|                 MachOUtils.AdjustHeadersForBundle(appHostPath)); |  | ||||||
| 
 |  | ||||||
|             // Memory-mapped write does not updating last write time |  | ||||||
|             RetryUtil.RetryOnIOError(() => |  | ||||||
|                 File.SetLastWriteTimeUtc(appHostPath, DateTime.UtcNow)); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Check if the an AppHost is a single-file bundle |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="appHostFilePath">The path of Apphost to check</param> |  | ||||||
|         /// <returns>True if the AppHost is a single-file bundle, false otherwise</returns> |  | ||||||
|         public static void ResetBundle(string appHostFilePath) |  | ||||||
|         { |  | ||||||
|             byte[] bundleSignature = { |  | ||||||
|                 // 32 bytes represent the bundle signature: SHA-256 for ".net core bundle" |  | ||||||
|                 0x8b, 0x12, 0x02, 0xb9, 0x6a, 0x61, 0x20, 0x38, |  | ||||||
|                 0x72, 0x7b, 0x93, 0x02, 0x14, 0xd7, 0xa0, 0x32, |  | ||||||
|                 0x13, 0xf5, 0xb9, 0xe6, 0xef, 0xae, 0x33, 0x18, |  | ||||||
|                 0xee, 0x3b, 0x2d, 0xce, 0x24, 0xb3, 0x6a, 0xae |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             void ResetBundleHeader() |  | ||||||
|             { |  | ||||||
|                 using (var memoryMappedFile = MemoryMappedFile.CreateFromFile(appHostFilePath)) |  | ||||||
|                 { |  | ||||||
|                     using (MemoryMappedViewAccessor accessor = memoryMappedFile.CreateViewAccessor()) |  | ||||||
|                     { |  | ||||||
|                         int position = BinaryUtils.SearchInFile(accessor, bundleSignature); |  | ||||||
|                         if (position == -1) |  | ||||||
|                         { |  | ||||||
|                             throw new PlaceHolderNotFoundInAppHostException(bundleSignature); |  | ||||||
|                         } |  | ||||||
|                          |  | ||||||
|                         accessor.WriteArray(position - sizeof(long), new byte[sizeof(long)], 0, sizeof(long)); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             RetryUtil.RetryOnIOError(ResetBundleHeader); |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         public static bool IsBundle(string appHostFilePath, out long bundleHeaderOffset) |  | ||||||
|         { |  | ||||||
|             byte[] bundleSignature = { |  | ||||||
|                 // 32 bytes represent the bundle signature: SHA-256 for ".net core bundle" |  | ||||||
|                 0x8b, 0x12, 0x02, 0xb9, 0x6a, 0x61, 0x20, 0x38, |  | ||||||
|                 0x72, 0x7b, 0x93, 0x02, 0x14, 0xd7, 0xa0, 0x32, |  | ||||||
|                 0x13, 0xf5, 0xb9, 0xe6, 0xef, 0xae, 0x33, 0x18, |  | ||||||
|                 0xee, 0x3b, 0x2d, 0xce, 0x24, 0xb3, 0x6a, 0xae |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             long headerOffset = 0; |  | ||||||
|             void FindBundleHeader() |  | ||||||
|             { |  | ||||||
|                 using (var memoryMappedFile = MemoryMappedFile.CreateFromFile(appHostFilePath)) |  | ||||||
|                 { |  | ||||||
|                     using (MemoryMappedViewAccessor accessor = memoryMappedFile.CreateViewAccessor()) |  | ||||||
|                     { |  | ||||||
|                         int position = BinaryUtils.SearchInFile(accessor, bundleSignature); |  | ||||||
|                         if (position == -1) |  | ||||||
|                         { |  | ||||||
|                             throw new PlaceHolderNotFoundInAppHostException(bundleSignature); |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                         headerOffset = accessor.ReadInt64(position - sizeof(long)); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             RetryUtil.RetryOnIOError(FindBundleHeader); |  | ||||||
|             bundleHeaderOffset = headerOffset; |  | ||||||
| 
 |  | ||||||
|             return headerOffset != 0; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         [DllImport("libc", SetLastError = true)] |  | ||||||
|         private static extern int chmod(string pathname, int mode); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,26 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.AppHost |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Additional details about the failure with caused an AppHostMachOFormatException |  | ||||||
|     /// </summary> |  | ||||||
|     public enum MachOFormatError |  | ||||||
|     { |  | ||||||
|         Not64BitExe,            // Apphost is expected to be a 64-bit MachO executable |  | ||||||
|         DuplicateLinkEdit,      // Only one __LINKEDIT segment is expected in the apphost |  | ||||||
|         DuplicateSymtab,        // Only one SYMTAB is expected in the apphost |  | ||||||
|         MissingLinkEdit,        // CODE_SIGNATURE command must follow a Segment64 command named __LINKEDIT |  | ||||||
|         MissingSymtab,          // CODE_SIGNATURE command must follow the SYMTAB command |  | ||||||
|         LinkEditNotLast,        // __LINKEDIT must be the last segment in the binary layout |  | ||||||
|         SymtabNotInLinkEdit,    // SYMTAB must within the __LINKEDIT segment! |  | ||||||
|         SignNotInLinkEdit,      // Signature blob must be within the __LINKEDIT segment! |  | ||||||
|         SignCommandNotLast,     // CODE_SIGNATURE command must be the last command |  | ||||||
|         SignBlobNotLast,        // Signature blob must be at the very end of the file |  | ||||||
|         SignDoesntFollowSymtab, // Signature blob must immediately follow the Symtab |  | ||||||
|         MemoryMapAccessFault,   // Error reading the memory-mapped apphost |  | ||||||
|         InvalidUTF8,            // UTF8 decoding failed |  | ||||||
|         SignNotRemoved,         // Signature not removed from the host (while processing a single-file bundle) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,444 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| using System.IO.MemoryMappedFiles; |  | ||||||
| using System.Runtime.CompilerServices; |  | ||||||
| using System.Text; |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.AppHost |  | ||||||
| { |  | ||||||
|     internal static class MachOUtils |  | ||||||
|     { |  | ||||||
|         // The MachO Headers are copied from |  | ||||||
|         // https://opensource.apple.com/source/cctools/cctools-870/include/mach-o/loader.h |  | ||||||
|         // |  | ||||||
|         // The data fields and enumerations match the structure definitions in the above file, |  | ||||||
|         // and hence do not conform to C# CoreFx naming style. |  | ||||||
| 
 |  | ||||||
|         private enum Magic : uint |  | ||||||
|         { |  | ||||||
|             MH_MAGIC = 0xfeedface, |  | ||||||
|             MH_CIGAM = 0xcefaedfe, |  | ||||||
|             MH_MAGIC_64 = 0xfeedfacf, |  | ||||||
|             MH_CIGAM_64 = 0xcffaedfe |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private enum FileType : uint |  | ||||||
|         { |  | ||||||
|             MH_EXECUTE = 0x2 |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
| #pragma warning disable 0649 |  | ||||||
|         private struct MachHeader |  | ||||||
|         { |  | ||||||
|             public Magic magic; |  | ||||||
|             public int cputype; |  | ||||||
|             public int cpusubtype; |  | ||||||
|             public FileType filetype; |  | ||||||
|             public uint ncmds; |  | ||||||
|             public uint sizeofcmds; |  | ||||||
|             public uint flags; |  | ||||||
|             public uint reserved; |  | ||||||
| 
 |  | ||||||
|             public bool Is64BitExecutable() |  | ||||||
|             { |  | ||||||
|                 return magic == Magic.MH_MAGIC_64 && filetype == FileType.MH_EXECUTE; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             public bool IsValid() |  | ||||||
|             { |  | ||||||
|                 switch (magic) |  | ||||||
|                 { |  | ||||||
|                     case Magic.MH_CIGAM: |  | ||||||
|                     case Magic.MH_CIGAM_64: |  | ||||||
|                     case Magic.MH_MAGIC: |  | ||||||
|                     case Magic.MH_MAGIC_64: |  | ||||||
|                         return true; |  | ||||||
| 
 |  | ||||||
|                     default: |  | ||||||
|                         return false; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private enum Command : uint |  | ||||||
|         { |  | ||||||
|             LC_SYMTAB = 0x2, |  | ||||||
|             LC_SEGMENT_64 = 0x19, |  | ||||||
|             LC_CODE_SIGNATURE = 0x1d, |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private struct LoadCommand |  | ||||||
|         { |  | ||||||
|             public Command cmd; |  | ||||||
|             public uint cmdsize; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         // The linkedit_data_command contains the offsets and sizes of a blob |  | ||||||
|         // of data in the __LINKEDIT segment (including LC_CODE_SIGNATURE). |  | ||||||
|         private struct LinkEditDataCommand |  | ||||||
|         { |  | ||||||
|             public Command cmd; |  | ||||||
|             public uint cmdsize; |  | ||||||
|             public uint dataoff; |  | ||||||
|             public uint datasize; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private struct SymtabCommand |  | ||||||
|         { |  | ||||||
|             public uint cmd; |  | ||||||
|             public uint cmdsize; |  | ||||||
|             public uint symoff; |  | ||||||
|             public uint nsyms; |  | ||||||
|             public uint stroff; |  | ||||||
|             public uint strsize; |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         private unsafe struct SegmentCommand64 |  | ||||||
|         { |  | ||||||
|             public Command cmd; |  | ||||||
|             public uint cmdsize; |  | ||||||
|             public fixed byte segname[16]; |  | ||||||
|             public ulong vmaddr; |  | ||||||
|             public ulong vmsize; |  | ||||||
|             public ulong fileoff; |  | ||||||
|             public ulong filesize; |  | ||||||
|             public int maxprot; |  | ||||||
|             public int initprot; |  | ||||||
|             public uint nsects; |  | ||||||
|             public uint flags; |  | ||||||
| 
 |  | ||||||
|             public string SegName |  | ||||||
|             { |  | ||||||
|                 get |  | ||||||
|                 { |  | ||||||
|                     fixed (byte* p = segname) |  | ||||||
|                     { |  | ||||||
|                         int len = 0; |  | ||||||
|                         while (*(p + len) != 0 && len++ < 16) ; |  | ||||||
| 
 |  | ||||||
|                         try |  | ||||||
|                         { |  | ||||||
|                             return Encoding.UTF8.GetString(p, len); |  | ||||||
|                         } |  | ||||||
|                         catch (ArgumentException) |  | ||||||
|                         { |  | ||||||
|                             throw new AppHostMachOFormatException(MachOFormatError.InvalidUTF8); |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
| #pragma warning restore 0649 |  | ||||||
| 
 |  | ||||||
|         private static void Verify(bool condition, MachOFormatError error) |  | ||||||
|         { |  | ||||||
|             if (!condition) |  | ||||||
|             { |  | ||||||
|                 throw new AppHostMachOFormatException(error); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static bool IsMachOImage(string filePath) |  | ||||||
|         { |  | ||||||
|             using (BinaryReader reader = new BinaryReader(File.OpenRead(filePath))) |  | ||||||
|             { |  | ||||||
|                 if (reader.BaseStream.Length < 256) // Header size |  | ||||||
|                 { |  | ||||||
|                     return false; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 uint magic = reader.ReadUInt32(); |  | ||||||
|                 return Enum.IsDefined(typeof(Magic), magic); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// This Method is a utility to remove the code-signature (if any) |  | ||||||
|         /// from a MachO AppHost binary. |  | ||||||
|         /// |  | ||||||
|         /// The tool assumes the following layout of the executable: |  | ||||||
|         /// |  | ||||||
|         /// * MachoHeader (64-bit, executable, not swapped integers) |  | ||||||
|         /// * LoadCommands |  | ||||||
|         ///     LC_SEGMENT_64 (__PAGEZERO) |  | ||||||
|         ///     LC_SEGMENT_64 (__TEXT) |  | ||||||
|         ///     LC_SEGMENT_64 (__DATA) |  | ||||||
|         ///     LC_SEGMENT_64 (__LINKEDIT) |  | ||||||
|         ///     ... |  | ||||||
|         ///     LC_SYMTAB |  | ||||||
|         ///     ... |  | ||||||
|         ///     LC_CODE_SIGNATURE (last) |  | ||||||
|         /// |  | ||||||
|         ///  * ... Different Segments ... |  | ||||||
|         /// |  | ||||||
|         ///  * The __LINKEDIT Segment (last) |  | ||||||
|         ///      * ... Different sections ... |  | ||||||
|         ///      * SYMTAB |  | ||||||
|         ///      * (Some alignment bytes) |  | ||||||
|         ///      * The Code-signature |  | ||||||
|         /// |  | ||||||
|         /// In order to remove the signature, the method: |  | ||||||
|         /// - Removes (zeros out) the LC_CODE_SIGNATURE command |  | ||||||
|         /// - Adjusts the size and count of the load commands in the header |  | ||||||
|         /// - Truncates the size of the __LINKEDIT segment to the end of SYMTAB |  | ||||||
|         /// - Truncates the apphost file to the end of the __LINKEDIT segment |  | ||||||
|         /// |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="stream">Stream containing the AppHost</param> |  | ||||||
|         /// <returns> |  | ||||||
|         ///  True if |  | ||||||
|         ///    - The input is a MachO binary, and |  | ||||||
|         ///    - It is a signed binary, and |  | ||||||
|         ///    - The signature was successfully removed |  | ||||||
|         ///   False otherwise |  | ||||||
|         /// </returns> |  | ||||||
|         /// <exception cref="AppHostMachOFormatException"> |  | ||||||
|         /// The input is a MachO file, but doesn't match the expect format of the AppHost. |  | ||||||
|         /// </exception> |  | ||||||
|         public static unsafe bool RemoveSignature(FileStream stream) |  | ||||||
|         { |  | ||||||
|             uint signatureSize = 0; |  | ||||||
|             using (var mappedFile = MemoryMappedFile.CreateFromFile(stream, |  | ||||||
|                                                                     mapName: null, |  | ||||||
|                                                                     capacity: 0, |  | ||||||
|                                                                     MemoryMappedFileAccess.ReadWrite, |  | ||||||
|                                                                     HandleInheritability.None, |  | ||||||
|                                                                     leaveOpen: true)) |  | ||||||
|             { |  | ||||||
|                 using (var accessor = mappedFile.CreateViewAccessor()) |  | ||||||
|                 { |  | ||||||
|                     byte* file = null; |  | ||||||
|                     try |  | ||||||
|                     { |  | ||||||
|                         accessor.SafeMemoryMappedViewHandle.AcquirePointer(ref file); |  | ||||||
|                         Verify(file != null, MachOFormatError.MemoryMapAccessFault); |  | ||||||
| 
 |  | ||||||
|                         MachHeader* header = (MachHeader*)file; |  | ||||||
| 
 |  | ||||||
|                         if (!header->IsValid()) |  | ||||||
|                         { |  | ||||||
|                             // Not a MachO file. |  | ||||||
|                             return false; |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                         Verify(header->Is64BitExecutable(), MachOFormatError.Not64BitExe); |  | ||||||
| 
 |  | ||||||
|                         file += sizeof(MachHeader); |  | ||||||
|                         SegmentCommand64* linkEdit = null; |  | ||||||
|                         SymtabCommand* symtab = null; |  | ||||||
|                         LinkEditDataCommand* signature = null; |  | ||||||
| 
 |  | ||||||
|                         for (uint i = 0; i < header->ncmds; i++) |  | ||||||
|                         { |  | ||||||
|                             LoadCommand* command = (LoadCommand*)file; |  | ||||||
|                             if (command->cmd == Command.LC_SEGMENT_64) |  | ||||||
|                             { |  | ||||||
|                                 SegmentCommand64* segment = (SegmentCommand64*)file; |  | ||||||
|                                 if (segment->SegName.Equals("__LINKEDIT")) |  | ||||||
|                                 { |  | ||||||
|                                     Verify(linkEdit == null, MachOFormatError.DuplicateLinkEdit); |  | ||||||
|                                     linkEdit = segment; |  | ||||||
|                                 } |  | ||||||
|                             } |  | ||||||
|                             else if (command->cmd == Command.LC_SYMTAB) |  | ||||||
|                             { |  | ||||||
|                                 Verify(symtab == null, MachOFormatError.DuplicateSymtab); |  | ||||||
|                                 symtab = (SymtabCommand*)command; |  | ||||||
|                             } |  | ||||||
|                             else if (command->cmd == Command.LC_CODE_SIGNATURE) |  | ||||||
|                             { |  | ||||||
|                                 Verify(i == header->ncmds - 1, MachOFormatError.SignCommandNotLast); |  | ||||||
|                                 signature = (LinkEditDataCommand*)command; |  | ||||||
|                                 break; |  | ||||||
|                             } |  | ||||||
| 
 |  | ||||||
|                             file += command->cmdsize; |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                         if (signature != null) |  | ||||||
|                         { |  | ||||||
|                             Verify(linkEdit != null, MachOFormatError.MissingLinkEdit); |  | ||||||
|                             Verify(symtab != null, MachOFormatError.MissingSymtab); |  | ||||||
| 
 |  | ||||||
|                             var symtabEnd = symtab->stroff + symtab->strsize; |  | ||||||
|                             var linkEditEnd = linkEdit->fileoff + linkEdit->filesize; |  | ||||||
|                             var signatureEnd = signature->dataoff + signature->datasize; |  | ||||||
|                             var fileEnd = (ulong)stream.Length; |  | ||||||
| 
 |  | ||||||
|                             Verify(linkEditEnd == fileEnd, MachOFormatError.LinkEditNotLast); |  | ||||||
|                             Verify(signatureEnd == fileEnd, MachOFormatError.SignBlobNotLast); |  | ||||||
| 
 |  | ||||||
|                             Verify(symtab->symoff > linkEdit->fileoff, MachOFormatError.SymtabNotInLinkEdit); |  | ||||||
|                             Verify(signature->dataoff > linkEdit->fileoff, MachOFormatError.SignNotInLinkEdit); |  | ||||||
| 
 |  | ||||||
|                             // The signature blob immediately follows the symtab blob, |  | ||||||
|                             // except for a few bytes of padding. |  | ||||||
|                             Verify(signature->dataoff >= symtabEnd && signature->dataoff - symtabEnd < 32, MachOFormatError.SignBlobNotLast); |  | ||||||
| 
 |  | ||||||
|                             // Remove the signature command |  | ||||||
|                             header->ncmds--; |  | ||||||
|                             header->sizeofcmds -= signature->cmdsize; |  | ||||||
|                             Unsafe.InitBlock(signature, 0, signature->cmdsize); |  | ||||||
| 
 |  | ||||||
|                             // Remove the signature blob (note for truncation) |  | ||||||
|                             signatureSize = (uint)(fileEnd - symtabEnd); |  | ||||||
| 
 |  | ||||||
|                             // Adjust the __LINKEDIT segment load command |  | ||||||
|                             linkEdit->filesize -= signatureSize; |  | ||||||
| 
 |  | ||||||
|                             // codesign --remove-signature doesn't reset the vmsize. |  | ||||||
|                             // Setting the vmsize here makes the output bin-equal with the original |  | ||||||
|                             // unsigned apphost (and not bin-equal with a signed-unsigned-apphost). |  | ||||||
|                             linkEdit->vmsize = linkEdit->filesize; |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     finally |  | ||||||
|                     { |  | ||||||
|                         if (file != null) |  | ||||||
|                         { |  | ||||||
|                             accessor.SafeMemoryMappedViewHandle.ReleasePointer(); |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (signatureSize != 0) |  | ||||||
|             { |  | ||||||
|                 // The signature was removed, update the file length |  | ||||||
|                 stream.SetLength(stream.Length - signatureSize); |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// This Method is a utility to adjust the apphost MachO-header |  | ||||||
|         /// to include the bytes added by the single-file bundler at the end of the file. |  | ||||||
|         /// |  | ||||||
|         /// The tool assumes the following layout of the executable |  | ||||||
|         /// |  | ||||||
|         /// * MachoHeader (64-bit, executable, not swapped integers) |  | ||||||
|         /// * LoadCommands |  | ||||||
|         ///     LC_SEGMENT_64 (__PAGEZERO) |  | ||||||
|         ///     LC_SEGMENT_64 (__TEXT) |  | ||||||
|         ///     LC_SEGMENT_64 (__DATA) |  | ||||||
|         ///     LC_SEGMENT_64 (__LINKEDIT) |  | ||||||
|         ///     ... |  | ||||||
|         ///     LC_SYMTAB |  | ||||||
|         /// |  | ||||||
|         ///  * ... Different Segments |  | ||||||
|         /// |  | ||||||
|         ///  * The __LINKEDIT Segment (last) |  | ||||||
|         ///      * ... Different sections ... |  | ||||||
|         ///      * SYMTAB (last) |  | ||||||
|         /// |  | ||||||
|         /// The MAC codesign tool places several restrictions on the layout |  | ||||||
|         ///   * The __LINKEDIT segment must be the last one |  | ||||||
|         ///   * The __LINKEDIT segment must cover the end of the file |  | ||||||
|         ///   * All bytes in the __LINKEDIT segment are used by other linkage commands |  | ||||||
|         ///     (ex: symbol/string table, dynamic load information etc) |  | ||||||
|         /// |  | ||||||
|         /// In order to circumvent these restrictions, we: |  | ||||||
|         ///    * Extend the __LINKEDIT segment to include the bundle-data |  | ||||||
|         ///    * Extend the string table to include all the bundle-data |  | ||||||
|         ///      (that is, the bundle-data appear as strings to the loader/codesign tool). |  | ||||||
|         /// |  | ||||||
|         ///  This method has certain limitations: |  | ||||||
|         ///    * The bytes for the bundler may be unnecessarily loaded at startup |  | ||||||
|         ///    * Tools that process the string table may be confused (?) |  | ||||||
|         ///    * The string table size is limited to 4GB. Bundles larger than that size |  | ||||||
|         ///      cannot be accomodated by this utility. |  | ||||||
|         /// |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="filePath">Path to the AppHost</param> |  | ||||||
|         /// <returns> |  | ||||||
|         ///  True if |  | ||||||
|         ///    - The input is a MachO binary, and |  | ||||||
|         ///    - The additional bytes were successfully accomodated within the MachO segments. |  | ||||||
|         ///   False otherwise |  | ||||||
|         /// </returns> |  | ||||||
|         /// <exception cref="AppHostMachOFormatException"> |  | ||||||
|         /// The input is a MachO file, but doesn't match the expect format of the AppHost. |  | ||||||
|         /// </exception> |  | ||||||
|         public static unsafe bool AdjustHeadersForBundle(string filePath) |  | ||||||
|         { |  | ||||||
|             ulong fileLength = (ulong)new FileInfo(filePath).Length; |  | ||||||
|             using (var mappedFile = MemoryMappedFile.CreateFromFile(filePath)) |  | ||||||
|             { |  | ||||||
|                 using (var accessor = mappedFile.CreateViewAccessor()) |  | ||||||
|                 { |  | ||||||
|                     byte* file = null; |  | ||||||
|                     try |  | ||||||
|                     { |  | ||||||
|                         accessor.SafeMemoryMappedViewHandle.AcquirePointer(ref file); |  | ||||||
|                         Verify(file != null, MachOFormatError.MemoryMapAccessFault); |  | ||||||
| 
 |  | ||||||
|                         MachHeader* header = (MachHeader*)file; |  | ||||||
| 
 |  | ||||||
|                         if (!header->IsValid()) |  | ||||||
|                         { |  | ||||||
|                             // Not a MachO file. |  | ||||||
|                             return false; |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                         Verify(header->Is64BitExecutable(), MachOFormatError.Not64BitExe); |  | ||||||
| 
 |  | ||||||
|                         file += sizeof(MachHeader); |  | ||||||
|                         SegmentCommand64* linkEdit = null; |  | ||||||
|                         SymtabCommand* symtab = null; |  | ||||||
|                         LinkEditDataCommand* signature = null; |  | ||||||
| 
 |  | ||||||
|                         for (uint i = 0; i < header->ncmds; i++) |  | ||||||
|                         { |  | ||||||
|                             LoadCommand* command = (LoadCommand*)file; |  | ||||||
|                             if (command->cmd == Command.LC_SEGMENT_64) |  | ||||||
|                             { |  | ||||||
|                                 SegmentCommand64* segment = (SegmentCommand64*)file; |  | ||||||
|                                 if (segment->SegName.Equals("__LINKEDIT")) |  | ||||||
|                                 { |  | ||||||
|                                     Verify(linkEdit == null, MachOFormatError.DuplicateLinkEdit); |  | ||||||
|                                     linkEdit = segment; |  | ||||||
|                                 } |  | ||||||
|                             } |  | ||||||
|                             else if (command->cmd == Command.LC_SYMTAB) |  | ||||||
|                             { |  | ||||||
|                                 Verify(symtab == null, MachOFormatError.DuplicateSymtab); |  | ||||||
|                                 symtab = (SymtabCommand*)command; |  | ||||||
|                             } |  | ||||||
| 
 |  | ||||||
|                             file += command->cmdsize; |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                         Verify(linkEdit != null, MachOFormatError.MissingLinkEdit); |  | ||||||
|                         Verify(symtab != null, MachOFormatError.MissingSymtab); |  | ||||||
| 
 |  | ||||||
|                         // Update the string table to include bundle-data |  | ||||||
|                         ulong newStringTableSize = fileLength - symtab->stroff; |  | ||||||
|                         if (newStringTableSize > uint.MaxValue) |  | ||||||
|                         { |  | ||||||
|                             // Too big, too bad; |  | ||||||
|                             return false; |  | ||||||
|                         } |  | ||||||
|                         symtab->strsize = (uint)newStringTableSize; |  | ||||||
| 
 |  | ||||||
|                         // Update the __LINKEDIT segment to include bundle-data |  | ||||||
|                         linkEdit->filesize = fileLength - linkEdit->fileoff; |  | ||||||
|                         linkEdit->vmsize = linkEdit->filesize; |  | ||||||
|                     } |  | ||||||
|                     finally |  | ||||||
|                     { |  | ||||||
|                         if (file != null) |  | ||||||
|                         { |  | ||||||
|                             accessor.SafeMemoryMappedViewHandle.ReleasePointer(); |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,178 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| using System.IO.MemoryMappedFiles; |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.AppHost |  | ||||||
| { |  | ||||||
|     public static class PEUtils |  | ||||||
|     { |  | ||||||
|         /// <summary> |  | ||||||
|         /// The first two bytes of a PE file are a constant signature. |  | ||||||
|         /// </summary> |  | ||||||
|         private const ushort PEFileSignature = 0x5A4D; |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// The offset of the PE header pointer in the DOS header. |  | ||||||
|         /// </summary> |  | ||||||
|         private const int PEHeaderPointerOffset = 0x3C; |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// The offset of the Subsystem field in the PE header. |  | ||||||
|         /// </summary> |  | ||||||
|         private const int SubsystemOffset = 0x5C; |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// The value of the sybsystem field which indicates Windows GUI (Graphical UI) |  | ||||||
|         /// </summary> |  | ||||||
|         private const ushort WindowsGUISubsystem = 0x2; |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// The value of the subsystem field which indicates Windows CUI (Console) |  | ||||||
|         /// </summary> |  | ||||||
|         private const ushort WindowsCUISubsystem = 0x3; |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Check whether the apphost file is a windows PE image by looking at the first few bytes. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="accessor">The memory accessor which has the apphost file opened.</param> |  | ||||||
|         /// <returns>true if the accessor represents a PE image, false otherwise.</returns> |  | ||||||
|         internal static unsafe bool IsPEImage(MemoryMappedViewAccessor accessor) |  | ||||||
|         { |  | ||||||
|             byte* pointer = null; |  | ||||||
| 
 |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 accessor.SafeMemoryMappedViewHandle.AcquirePointer(ref pointer); |  | ||||||
|                 byte* bytes = pointer + accessor.PointerOffset; |  | ||||||
| 
 |  | ||||||
|                 // https://en.wikipedia.org/wiki/Portable_Executable |  | ||||||
|                 // Validate that we're looking at Windows PE file |  | ||||||
|                 if (((ushort*)bytes)[0] != PEFileSignature || accessor.Capacity < PEHeaderPointerOffset + sizeof(uint)) |  | ||||||
|                 { |  | ||||||
|                     return false; |  | ||||||
|                 } |  | ||||||
|                 return true; |  | ||||||
|             } |  | ||||||
|             finally |  | ||||||
|             { |  | ||||||
|                 if (pointer != null) |  | ||||||
|                 { |  | ||||||
|                     accessor.SafeMemoryMappedViewHandle.ReleasePointer(); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static bool IsPEImage(string filePath) |  | ||||||
|         { |  | ||||||
|             using (BinaryReader reader = new BinaryReader(File.OpenRead(filePath))) |  | ||||||
|             { |  | ||||||
|                 if (reader.BaseStream.Length < PEHeaderPointerOffset + sizeof(uint)) |  | ||||||
|                 { |  | ||||||
|                     return false; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 ushort signature = reader.ReadUInt16(); |  | ||||||
|                 return signature == PEFileSignature; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// This method will attempt to set the subsystem to GUI. The apphost file should be a windows PE file. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="accessor">The memory accessor which has the apphost file opened.</param> |  | ||||||
|         internal static unsafe void SetWindowsGraphicalUserInterfaceBit(MemoryMappedViewAccessor accessor) |  | ||||||
|         { |  | ||||||
|             byte* pointer = null; |  | ||||||
| 
 |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 accessor.SafeMemoryMappedViewHandle.AcquirePointer(ref pointer); |  | ||||||
|                 byte* bytes = pointer + accessor.PointerOffset; |  | ||||||
| 
 |  | ||||||
|                 // https://en.wikipedia.org/wiki/Portable_Executable |  | ||||||
|                 uint peHeaderOffset = ((uint*)(bytes + PEHeaderPointerOffset))[0]; |  | ||||||
| 
 |  | ||||||
|                 if (accessor.Capacity < peHeaderOffset + SubsystemOffset + sizeof(ushort)) |  | ||||||
|                 { |  | ||||||
|                     throw new AppHostNotPEFileException(); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 ushort* subsystem = ((ushort*)(bytes + peHeaderOffset + SubsystemOffset)); |  | ||||||
| 
 |  | ||||||
|                 // https://docs.microsoft.com/windows/desktop/Debug/pe-format#windows-subsystem |  | ||||||
|                 // The subsystem of the prebuilt apphost should be set to CUI |  | ||||||
|                 if (subsystem[0] != WindowsCUISubsystem) |  | ||||||
|                 { |  | ||||||
|                     throw new AppHostNotCUIException(); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 // Set the subsystem to GUI |  | ||||||
|                 subsystem[0] = WindowsGUISubsystem; |  | ||||||
|             } |  | ||||||
|             finally |  | ||||||
|             { |  | ||||||
|                 if (pointer != null) |  | ||||||
|                 { |  | ||||||
|                     accessor.SafeMemoryMappedViewHandle.ReleasePointer(); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static unsafe void SetWindowsGraphicalUserInterfaceBit(string filePath) |  | ||||||
|         { |  | ||||||
|             using (var mappedFile = MemoryMappedFile.CreateFromFile(filePath)) |  | ||||||
|             { |  | ||||||
|                 using (var accessor = mappedFile.CreateViewAccessor()) |  | ||||||
|                 { |  | ||||||
|                     SetWindowsGraphicalUserInterfaceBit(accessor); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// This method will return the subsystem CUI/GUI value. The apphost file should be a windows PE file. |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="accessor">The memory accessor which has the apphost file opened.</param> |  | ||||||
|         internal static unsafe ushort GetWindowsGraphicalUserInterfaceBit(MemoryMappedViewAccessor accessor) |  | ||||||
|         { |  | ||||||
|             byte* pointer = null; |  | ||||||
| 
 |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 accessor.SafeMemoryMappedViewHandle.AcquirePointer(ref pointer); |  | ||||||
|                 byte* bytes = pointer + accessor.PointerOffset; |  | ||||||
| 
 |  | ||||||
|                 // https://en.wikipedia.org/wiki/Portable_Executable |  | ||||||
|                 uint peHeaderOffset = ((uint*)(bytes + PEHeaderPointerOffset))[0]; |  | ||||||
| 
 |  | ||||||
|                 if (accessor.Capacity < peHeaderOffset + SubsystemOffset + sizeof(ushort)) |  | ||||||
|                 { |  | ||||||
|                     throw new AppHostNotPEFileException(); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 ushort* subsystem = ((ushort*)(bytes + peHeaderOffset + SubsystemOffset)); |  | ||||||
| 
 |  | ||||||
|                 return subsystem[0]; |  | ||||||
|             } |  | ||||||
|             finally |  | ||||||
|             { |  | ||||||
|                 if (pointer != null) |  | ||||||
|                 { |  | ||||||
|                     accessor.SafeMemoryMappedViewHandle.ReleasePointer(); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static unsafe ushort GetWindowsGraphicalUserInterfaceBit(string filePath) |  | ||||||
|         { |  | ||||||
|             using (var mappedFile = MemoryMappedFile.CreateFromFile(filePath)) |  | ||||||
|             { |  | ||||||
|                 using (var accessor = mappedFile.CreateViewAccessor()) |  | ||||||
|                 { |  | ||||||
|                     return GetWindowsGraphicalUserInterfaceBit(accessor); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,18 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.AppHost |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Unable to use input file as a valid application host executable, as it does not contain |  | ||||||
|     /// the expected placeholder byte sequence. |  | ||||||
|     /// </summary> |  | ||||||
|     public class PlaceHolderNotFoundInAppHostException : AppHostUpdateException |  | ||||||
|     { |  | ||||||
|         public byte[] MissingPattern { get; } |  | ||||||
|         public PlaceHolderNotFoundInAppHostException(byte[] pattern) |  | ||||||
|         { |  | ||||||
|             MissingPattern = pattern; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,92 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// HostModel library implements several services for updating the AppHost DLL. |  | ||||||
|     /// These updates involve multiple file open/close operations. |  | ||||||
|     /// An Antivirus scanner may intercept in-between and lock the file, |  | ||||||
|     /// causing the operations to fail with IO-Error. |  | ||||||
|     /// So, the operations are retried a few times on failures such as |  | ||||||
|     /// - IOException |  | ||||||
|     /// - Failure with Win32 errors indicating file-lock |  | ||||||
|     /// </summary> |  | ||||||
|     public static class RetryUtil |  | ||||||
|     { |  | ||||||
|         public const int NumberOfRetries = 500; |  | ||||||
|         public const int NumMilliSecondsToWait = 100; |  | ||||||
| 
 |  | ||||||
|         public static void RetryOnIOError(Action func) |  | ||||||
|         { |  | ||||||
|             for (int i = 1; i <= NumberOfRetries; i++) |  | ||||||
|             { |  | ||||||
|                 try |  | ||||||
|                 { |  | ||||||
|                     func(); |  | ||||||
|                     break; |  | ||||||
|                 } |  | ||||||
|                 catch (IOException) when (i < NumberOfRetries) |  | ||||||
|                 { |  | ||||||
|                     Thread.Sleep(NumMilliSecondsToWait); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public static void RetryOnWin32Error(Action func) |  | ||||||
|         { |  | ||||||
|             static bool IsKnownIrrecoverableError(int hresult) |  | ||||||
|             { |  | ||||||
|                 // Error codes are defined in winerror.h |  | ||||||
|                 // The error code is stored in the lowest 16 bits of the HResult |  | ||||||
| 
 |  | ||||||
|                 switch (hresult & 0xffff) |  | ||||||
|                 { |  | ||||||
|                     case 0x00000001: // ERROR_INVALID_FUNCTION |  | ||||||
|                     case 0x00000002: // ERROR_FILE_NOT_FOUND |  | ||||||
|                     case 0x00000003: // ERROR_PATH_NOT_FOUND |  | ||||||
|                     case 0x00000006: // ERROR_INVALID_HANDLE |  | ||||||
|                     case 0x00000008: // ERROR_NOT_ENOUGH_MEMORY |  | ||||||
|                     case 0x0000000B: // ERROR_BAD_FORMAT |  | ||||||
|                     case 0x0000000E: // ERROR_OUTOFMEMORY |  | ||||||
|                     case 0x0000000F: // ERROR_INVALID_DRIVE |  | ||||||
|                     case 0x00000012: // ERROR_NO_MORE_FILES |  | ||||||
|                     case 0x00000035: // ERROR_BAD_NETPATH |  | ||||||
|                     case 0x00000057: // ERROR_INVALID_PARAMETER |  | ||||||
|                     case 0x00000071: // ERROR_NO_MORE_SEARCH_HANDLES |  | ||||||
|                     case 0x00000072: // ERROR_INVALID_TARGET_HANDLE |  | ||||||
|                     case 0x00000078: // ERROR_CALL_NOT_IMPLEMENTED |  | ||||||
|                     case 0x0000007B: // ERROR_INVALID_NAME |  | ||||||
|                     case 0x0000007C: // ERROR_INVALID_LEVEL |  | ||||||
|                     case 0x0000007D: // ERROR_NO_VOLUME_LABEL |  | ||||||
|                     case 0x0000009A: // ERROR_LABEL_TOO_LONG |  | ||||||
|                     case 0x000000A0: // ERROR_BAD_ARGUMENTS |  | ||||||
|                     case 0x000000A1: // ERROR_BAD_PATHNAME |  | ||||||
|                     case 0x000000CE: // ERROR_FILENAME_EXCED_RANGE |  | ||||||
|                     case 0x000000DF: // ERROR_FILE_TOO_LARGE |  | ||||||
|                     case 0x000003ED: // ERROR_UNRECOGNIZED_VOLUME |  | ||||||
|                     case 0x000003EE: // ERROR_FILE_INVALID |  | ||||||
|                     case 0x00000651: // ERROR_DEVICE_REMOVED |  | ||||||
|                         return true; |  | ||||||
| 
 |  | ||||||
|                     default: |  | ||||||
|                         return false; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             for (int i = 1; i <= NumberOfRetries; i++) |  | ||||||
|             { |  | ||||||
|                 try |  | ||||||
|                 { |  | ||||||
|                     func(); |  | ||||||
|                     break; |  | ||||||
|                 } |  | ||||||
|                 catch (HResultException hrex) |  | ||||||
|                     when (i < NumberOfRetries && !IsKnownIrrecoverableError(hrex.Win32HResult)) |  | ||||||
|                 { |  | ||||||
|                     Thread.Sleep(NumMilliSecondsToWait); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.Bundle |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// BundleOptions: Optional settings for configuring the type of files |  | ||||||
|     ///                included in the single file bundle. |  | ||||||
|     /// </summary> |  | ||||||
|     [Flags] |  | ||||||
|     public enum BundleOptions |  | ||||||
|     { |  | ||||||
|         None = 0, |  | ||||||
|         BundleNativeBinaries = 1, |  | ||||||
|         BundleOtherFiles = 2, |  | ||||||
|         BundleSymbolFiles = 4, |  | ||||||
|         BundleAllContent = BundleNativeBinaries | BundleOtherFiles, |  | ||||||
|         EnableCompression = 8, |  | ||||||
|     }; |  | ||||||
| } |  | ||||||
| @@ -1,373 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| using System.Diagnostics; |  | ||||||
| using System.IO.Compression; |  | ||||||
| using System.Reflection.PortableExecutable; |  | ||||||
| using System.Runtime.InteropServices; |  | ||||||
| using Microsoft.NET.HostModel.AppHost; |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.Bundle |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Bundler: Functionality to embed the managed app and its dependencies |  | ||||||
|     /// into the host native binary. |  | ||||||
|     /// </summary> |  | ||||||
|     public class Bundler |  | ||||||
|     { |  | ||||||
|         public const uint BundlerMajorVersion = 6; |  | ||||||
|         public const uint BundlerMinorVersion = 0; |  | ||||||
|         public readonly Manifest BundleManifest; |  | ||||||
| 
 |  | ||||||
|         private readonly string _hostName; |  | ||||||
|         private readonly string _outputDir; |  | ||||||
|         private readonly string _depsJson; |  | ||||||
|         private readonly string _runtimeConfigJson; |  | ||||||
|         private readonly string _runtimeConfigDevJson; |  | ||||||
| 
 |  | ||||||
|         private readonly Trace _tracer; |  | ||||||
|         private readonly TargetInfo _target; |  | ||||||
|         private readonly BundleOptions _options; |  | ||||||
|         private readonly bool _macosCodesign; |  | ||||||
| 
 |  | ||||||
|         public Bundler(string hostName, |  | ||||||
|                        string outputDir, |  | ||||||
|                        BundleOptions options = BundleOptions.None, |  | ||||||
|                        OSPlatform? targetOS = null, |  | ||||||
|                        Architecture? targetArch = null, |  | ||||||
|                        Version targetFrameworkVersion = null, |  | ||||||
|                        bool diagnosticOutput = false, |  | ||||||
|                        string appAssemblyName = null, |  | ||||||
|                        bool macosCodesign = true) |  | ||||||
|         { |  | ||||||
|             _tracer = new Trace(diagnosticOutput); |  | ||||||
| 
 |  | ||||||
|             _hostName = hostName; |  | ||||||
|             _outputDir = Path.GetFullPath(string.IsNullOrEmpty(outputDir) ? Environment.CurrentDirectory : outputDir); |  | ||||||
|             _target = new TargetInfo(targetOS, targetArch, targetFrameworkVersion); |  | ||||||
| 
 |  | ||||||
|             if (_target.BundleMajorVersion < 6 && |  | ||||||
|                 (options & BundleOptions.EnableCompression) != 0) |  | ||||||
|             { |  | ||||||
|                 throw new ArgumentException("Compression requires framework version 6.0 or above", nameof(options)); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             appAssemblyName ??= _target.GetAssemblyName(hostName); |  | ||||||
|             _depsJson = appAssemblyName + ".deps.json"; |  | ||||||
|             _runtimeConfigJson = appAssemblyName + ".runtimeconfig.json"; |  | ||||||
|             _runtimeConfigDevJson = appAssemblyName + ".runtimeconfig.dev.json"; |  | ||||||
| 
 |  | ||||||
|             BundleManifest = new Manifest(_target.BundleMajorVersion, netcoreapp3CompatMode: options.HasFlag(BundleOptions.BundleAllContent)); |  | ||||||
|             _options = _target.DefaultOptions | options; |  | ||||||
|             _macosCodesign = macosCodesign; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private bool ShouldCompress(FileType type) |  | ||||||
|         { |  | ||||||
|             if (!_options.HasFlag(BundleOptions.EnableCompression)) |  | ||||||
|             { |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             switch (type) |  | ||||||
|             { |  | ||||||
|                 case FileType.DepsJson: |  | ||||||
|                 case FileType.RuntimeConfigJson: |  | ||||||
|                     return false; |  | ||||||
| 
 |  | ||||||
|                 default: |  | ||||||
|                     return true; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Embed 'file' into 'bundle' |  | ||||||
|         /// </summary> |  | ||||||
|         /// <returns> |  | ||||||
|         /// startOffset: offset of the start 'file' within 'bundle' |  | ||||||
|         /// compressedSize: size of the compressed data, if entry was compressed, otherwise 0 |  | ||||||
|         /// </returns> |  | ||||||
|         private (long startOffset, long compressedSize) AddToBundle(Stream bundle, Stream file, FileType type) |  | ||||||
|         { |  | ||||||
|             long startOffset = bundle.Position; |  | ||||||
|             if (ShouldCompress(type)) |  | ||||||
|             { |  | ||||||
|                 long fileLength = file.Length; |  | ||||||
|                 file.Position = 0; |  | ||||||
| 
 |  | ||||||
|                 // We use DeflateStream here. |  | ||||||
|                 // It uses GZip algorithm, but with a trivial header that does not contain file info. |  | ||||||
|                 using (DeflateStream compressionStream = new DeflateStream(bundle, CompressionLevel.Optimal, leaveOpen: true)) |  | ||||||
|                 { |  | ||||||
|                     file.CopyTo(compressionStream); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 long compressedSize = bundle.Position - startOffset; |  | ||||||
|                 if (compressedSize < fileLength * 0.75) |  | ||||||
|                 { |  | ||||||
|                     return (startOffset, compressedSize); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 // compression rate was not good enough |  | ||||||
|                 // roll back the bundle offset and let the uncompressed code path take care of the entry. |  | ||||||
|                 bundle.Seek(startOffset, SeekOrigin.Begin); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (type == FileType.Assembly) |  | ||||||
|             { |  | ||||||
|                 long misalignment = (bundle.Position % _target.AssemblyAlignment); |  | ||||||
| 
 |  | ||||||
|                 if (misalignment != 0) |  | ||||||
|                 { |  | ||||||
|                     long padding = _target.AssemblyAlignment - misalignment; |  | ||||||
|                     bundle.Position += padding; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             file.Position = 0; |  | ||||||
|             startOffset = bundle.Position; |  | ||||||
|             file.CopyTo(bundle); |  | ||||||
| 
 |  | ||||||
|             return (startOffset, 0); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private bool IsHost(string fileRelativePath) |  | ||||||
|         { |  | ||||||
|             return fileRelativePath.Equals(_hostName); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private bool ShouldIgnore(string fileRelativePath) |  | ||||||
|         { |  | ||||||
|             return fileRelativePath.Equals(_runtimeConfigDevJson); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private bool ShouldExclude(FileType type, string relativePath) |  | ||||||
|         { |  | ||||||
|             switch (type) |  | ||||||
|             { |  | ||||||
|                 case FileType.Assembly: |  | ||||||
|                 case FileType.DepsJson: |  | ||||||
|                 case FileType.RuntimeConfigJson: |  | ||||||
|                     return false; |  | ||||||
| 
 |  | ||||||
|                 case FileType.NativeBinary: |  | ||||||
|                     return !_options.HasFlag(BundleOptions.BundleNativeBinaries) || _target.ShouldExclude(relativePath); |  | ||||||
| 
 |  | ||||||
|                 case FileType.Symbols: |  | ||||||
|                     return !_options.HasFlag(BundleOptions.BundleSymbolFiles); |  | ||||||
| 
 |  | ||||||
|                 case FileType.Unknown: |  | ||||||
|                     return !_options.HasFlag(BundleOptions.BundleOtherFiles); |  | ||||||
| 
 |  | ||||||
|                 default: |  | ||||||
|                     Debug.Assert(false); |  | ||||||
|                     return false; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private bool IsAssembly(string path, out bool isPE) |  | ||||||
|         { |  | ||||||
|             isPE = false; |  | ||||||
| 
 |  | ||||||
|             using (FileStream file = File.OpenRead(path)) |  | ||||||
|             { |  | ||||||
|                 try |  | ||||||
|                 { |  | ||||||
|                     PEReader peReader = new PEReader(file); |  | ||||||
|                     CorHeader corHeader = peReader.PEHeaders.CorHeader; |  | ||||||
| 
 |  | ||||||
|                     isPE = true; // If peReader.PEHeaders doesn't throw, it is a valid PEImage |  | ||||||
|                     return corHeader != null; |  | ||||||
|                 } |  | ||||||
|                 catch (BadImageFormatException) |  | ||||||
|                 { |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private FileType InferType(FileSpec fileSpec) |  | ||||||
|         { |  | ||||||
|             if (fileSpec.BundleRelativePath.Equals(_depsJson)) |  | ||||||
|             { |  | ||||||
|                 return FileType.DepsJson; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (fileSpec.BundleRelativePath.Equals(_runtimeConfigJson)) |  | ||||||
|             { |  | ||||||
|                 return FileType.RuntimeConfigJson; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (Path.GetExtension(fileSpec.BundleRelativePath).ToLowerInvariant().Equals(".pdb")) |  | ||||||
|             { |  | ||||||
|                 return FileType.Symbols; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             bool isPE; |  | ||||||
|             if (IsAssembly(fileSpec.SourcePath, out isPE)) |  | ||||||
|             { |  | ||||||
|                 return FileType.Assembly; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             bool isNativeBinary = _target.IsWindows ? isPE : _target.IsNativeBinary(fileSpec.SourcePath); |  | ||||||
| 
 |  | ||||||
|             if (isNativeBinary) |  | ||||||
|             { |  | ||||||
|                 return FileType.NativeBinary; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return FileType.Unknown; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Generate a bundle, given the specification of embedded files |  | ||||||
|         /// </summary> |  | ||||||
|         /// <param name="fileSpecs"> |  | ||||||
|         /// An enumeration FileSpecs for the files to be embedded. |  | ||||||
|         /// |  | ||||||
|         /// Files in fileSpecs that are not bundled within the single file bundle, |  | ||||||
|         /// and should be published as separate files are marked as "IsExcluded" by this method. |  | ||||||
|         /// This doesn't include unbundled files that should be dropped, and not publised as output. |  | ||||||
|         /// </param> |  | ||||||
|         /// <returns> |  | ||||||
|         /// The full path the the generated bundle file |  | ||||||
|         /// </returns> |  | ||||||
|         /// <exceptions> |  | ||||||
|         /// ArgumentException if input is invalid |  | ||||||
|         /// IOExceptions and ArgumentExceptions from callees flow to the caller. |  | ||||||
|         /// </exceptions> |  | ||||||
|         public string GenerateBundle(IReadOnlyList<FileSpec> fileSpecs) |  | ||||||
|         { |  | ||||||
|             _tracer.Log($"Bundler Version: {BundlerMajorVersion}.{BundlerMinorVersion}"); |  | ||||||
|             _tracer.Log($"Bundle  Version: {BundleManifest.BundleVersion}"); |  | ||||||
|             _tracer.Log($"Target Runtime: {_target}"); |  | ||||||
|             _tracer.Log($"Bundler Options: {_options}"); |  | ||||||
| 
 |  | ||||||
|             if (fileSpecs.Any(x => !x.IsValid())) |  | ||||||
|             { |  | ||||||
|                 throw new ArgumentException("Invalid input specification: Found entry with empty source-path or bundle-relative-path."); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             string hostSource; |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 hostSource = fileSpecs.Where(x => x.BundleRelativePath.Equals(_hostName)).Single().SourcePath; |  | ||||||
|             } |  | ||||||
|             catch (InvalidOperationException) |  | ||||||
|             { |  | ||||||
|                 throw new ArgumentException("Invalid input specification: Must specify the host binary"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             string bundlePath = Path.Combine(_outputDir, _hostName); |  | ||||||
|             if (File.Exists(bundlePath)) |  | ||||||
|             { |  | ||||||
|                 _tracer.Log($"Ovewriting existing File {bundlePath}"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             BinaryUtils.CopyFile(hostSource, bundlePath); |  | ||||||
| 
 |  | ||||||
|             if (RuntimeInformation.IsOSPlatform(OSPlatform.OSX) && HostModelUtils.IsCodesignAvailable()) |  | ||||||
|             { |  | ||||||
|                 RemoveCodesignIfNecessary(bundlePath); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             // Note: We're comparing file paths both on the OS we're running on as well as on the target OS for the app |  | ||||||
|             // We can't really make assumptions about the file systems (even on Linux there can be case insensitive file systems |  | ||||||
|             // and vice versa for Windows). So it's safer to do case sensitive comparison everywhere. |  | ||||||
|             var relativePathToSpec = new Dictionary<string, FileSpec>(StringComparer.Ordinal); |  | ||||||
| 
 |  | ||||||
|             long headerOffset = 0; |  | ||||||
|             using (BinaryWriter writer = new BinaryWriter(File.OpenWrite(bundlePath))) |  | ||||||
|             { |  | ||||||
|                 Stream bundle = writer.BaseStream; |  | ||||||
|                 bundle.Position = bundle.Length; |  | ||||||
| 
 |  | ||||||
|                 foreach (var fileSpec in fileSpecs) |  | ||||||
|                 { |  | ||||||
|                     string relativePath = fileSpec.BundleRelativePath; |  | ||||||
| 
 |  | ||||||
|                     if (IsHost(relativePath)) |  | ||||||
|                     { |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     if (ShouldIgnore(relativePath)) |  | ||||||
|                     { |  | ||||||
|                         _tracer.Log($"Ignore: {relativePath}"); |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     FileType type = InferType(fileSpec); |  | ||||||
| 
 |  | ||||||
|                     if (ShouldExclude(type, relativePath)) |  | ||||||
|                     { |  | ||||||
|                         _tracer.Log($"Exclude [{type}]: {relativePath}"); |  | ||||||
|                         fileSpec.Excluded = true; |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     if (relativePathToSpec.TryGetValue(fileSpec.BundleRelativePath, out var existingFileSpec)) |  | ||||||
|                     { |  | ||||||
|                         if (!string.Equals(fileSpec.SourcePath, existingFileSpec.SourcePath, StringComparison.Ordinal)) |  | ||||||
|                         { |  | ||||||
|                             throw new ArgumentException($"Invalid input specification: Found entries '{fileSpec.SourcePath}' and '{existingFileSpec.SourcePath}' with the same BundleRelativePath '{fileSpec.BundleRelativePath}'"); |  | ||||||
|                         } |  | ||||||
| 
 |  | ||||||
|                         // Exact duplicate - intentionally skip and don't include a second copy in the bundle |  | ||||||
|                         continue; |  | ||||||
|                     } |  | ||||||
|                     else |  | ||||||
|                     { |  | ||||||
|                         relativePathToSpec.Add(fileSpec.BundleRelativePath, fileSpec); |  | ||||||
|                     } |  | ||||||
| 
 |  | ||||||
|                     using (FileStream file = File.OpenRead(fileSpec.SourcePath)) |  | ||||||
|                     { |  | ||||||
|                         FileType targetType = _target.TargetSpecificFileType(type); |  | ||||||
|                         (long startOffset, long compressedSize) = AddToBundle(bundle, file, targetType); |  | ||||||
|                         FileEntry entry = BundleManifest.AddEntry(targetType, file, relativePath, startOffset, compressedSize, _target.BundleMajorVersion); |  | ||||||
|                         _tracer.Log($"Embed: {entry}"); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 // Write the bundle manifest |  | ||||||
|                 headerOffset = BundleManifest.Write(writer); |  | ||||||
|                 _tracer.Log($"Header Offset={headerOffset}"); |  | ||||||
|                 _tracer.Log($"Meta-data Size={writer.BaseStream.Position - headerOffset}"); |  | ||||||
|                 _tracer.Log($"Bundle: Path={bundlePath}, Size={bundle.Length}"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             HostWriter.SetAsBundle(bundlePath, headerOffset); |  | ||||||
| 
 |  | ||||||
|             // Sign the bundle if requested |  | ||||||
|             if (_macosCodesign && RuntimeInformation.IsOSPlatform(OSPlatform.OSX) && HostModelUtils.IsCodesignAvailable()) |  | ||||||
|             { |  | ||||||
|                 var (exitCode, stdErr) = HostModelUtils.RunCodesign("-s -", bundlePath); |  | ||||||
|                 if (exitCode != 0) |  | ||||||
|                 { |  | ||||||
|                     throw new InvalidOperationException($"Failed to codesign '{bundlePath}': {stdErr}"); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return bundlePath; |  | ||||||
| 
 |  | ||||||
|             // Remove mac code signature if applied before bundling |  | ||||||
|             static void RemoveCodesignIfNecessary(string bundlePath) |  | ||||||
|             { |  | ||||||
|                 Debug.Assert(RuntimeInformation.IsOSPlatform(OSPlatform.OSX)); |  | ||||||
|                 Debug.Assert(HostModelUtils.IsCodesignAvailable()); |  | ||||||
| 
 |  | ||||||
|                 // `codesign -v` returns 0 if app is signed |  | ||||||
|                 if (HostModelUtils.RunCodesign("-v", bundlePath).ExitCode == 0) |  | ||||||
|                 { |  | ||||||
|                     var (exitCode, stdErr) = HostModelUtils.RunCodesign("--remove-signature", bundlePath); |  | ||||||
|                     if (exitCode != 0) |  | ||||||
|                     { |  | ||||||
|                         throw new InvalidOperationException($"Removing codesign from '{bundlePath}' failed: {stdErr}"); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,56 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.Bundle |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// FileEntry: Records information about embedded files. |  | ||||||
|     /// |  | ||||||
|     /// The bundle manifest records the following meta-data for each |  | ||||||
|     /// file embedded in the bundle: |  | ||||||
|     /// * Type       (1 byte) |  | ||||||
|     /// * NameLength (7-bit extension encoding, typically 1 byte) |  | ||||||
|     /// * Name       ("NameLength" Bytes) |  | ||||||
|     /// * Offset     (Int64) |  | ||||||
|     /// * Size       (Int64) |  | ||||||
|     /// === present only in bundle version 3+ |  | ||||||
|     /// * CompressedSize   (Int64)  0 indicates No Compression |  | ||||||
|     /// </summary> |  | ||||||
|     public class FileEntry |  | ||||||
|     { |  | ||||||
|         public readonly uint BundleMajorVersion; |  | ||||||
| 
 |  | ||||||
|         public readonly long Offset; |  | ||||||
|         public readonly long Size; |  | ||||||
|         public readonly long CompressedSize; |  | ||||||
|         public readonly FileType Type; |  | ||||||
|         public readonly string RelativePath; // Path of an embedded file, relative to the Bundle source-directory. |  | ||||||
| 
 |  | ||||||
|         public const char DirectorySeparatorChar = '/'; |  | ||||||
| 
 |  | ||||||
|         public FileEntry(FileType fileType, string relativePath, long offset, long size, long compressedSize, uint bundleMajorVersion) |  | ||||||
|         { |  | ||||||
|             BundleMajorVersion = bundleMajorVersion; |  | ||||||
|             Type = fileType; |  | ||||||
|             RelativePath = relativePath.Replace('\\', DirectorySeparatorChar); |  | ||||||
|             Offset = offset; |  | ||||||
|             Size = size; |  | ||||||
|             CompressedSize = compressedSize; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public void Write(BinaryWriter writer) |  | ||||||
|         { |  | ||||||
|             writer.Write(Offset); |  | ||||||
|             writer.Write(Size); |  | ||||||
|             // compression is used only in version 6.0+ |  | ||||||
|             if (BundleMajorVersion >= 6) |  | ||||||
|             { |  | ||||||
|                 writer.Write(CompressedSize); |  | ||||||
|             } |  | ||||||
|             writer.Write((byte)Type); |  | ||||||
|             writer.Write(RelativePath); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public override string ToString() => $"{RelativePath} [{Type}] @{Offset} Sz={Size} CompressedSz={CompressedSize}"; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,34 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.Bundle |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     ///  Information about files to embed into the Bundle (input to the Bundler). |  | ||||||
|     /// |  | ||||||
|     ///   SourcePath: path to the file to be bundled at compile time |  | ||||||
|     ///   BundleRelativePath: path where the file is expected at run time, |  | ||||||
|     ///                       relative to the app DLL. |  | ||||||
|     /// </summary> |  | ||||||
|     public class FileSpec |  | ||||||
|     { |  | ||||||
|         public readonly string SourcePath; |  | ||||||
|         public readonly string BundleRelativePath; |  | ||||||
|         public bool Excluded; |  | ||||||
| 
 |  | ||||||
|         public FileSpec(string sourcePath, string bundleRelativePath) |  | ||||||
|         { |  | ||||||
|             SourcePath = sourcePath; |  | ||||||
|             BundleRelativePath = bundleRelativePath; |  | ||||||
|             Excluded = false; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public bool IsValid() |  | ||||||
|         { |  | ||||||
|             return !string.IsNullOrWhiteSpace(SourcePath) && |  | ||||||
|                    !string.IsNullOrWhiteSpace(BundleRelativePath); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public override string ToString() => $"SourcePath: {SourcePath}, RelativePath: {BundleRelativePath} {(Excluded ? "[Excluded]" : "")}"; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,21 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.Bundle |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// FileType: Identifies the type of file embedded into the bundle. |  | ||||||
|     /// |  | ||||||
|     /// The bundler differentiates a few kinds of files via the manifest, |  | ||||||
|     /// with respect to the way in which they'll be used by the runtime. |  | ||||||
|     /// </summary> |  | ||||||
|     public enum FileType : byte |  | ||||||
|     { |  | ||||||
|         Unknown,           // Type not determined. |  | ||||||
|         Assembly,          // IL and R2R Assemblies |  | ||||||
|         NativeBinary,      // NativeBinaries |  | ||||||
|         DepsJson,          // .deps.json configuration file |  | ||||||
|         RuntimeConfigJson, // .runtimeconfig.json configuration file |  | ||||||
|         Symbols            // PDB Files |  | ||||||
|     }; |  | ||||||
| } |  | ||||||
| @@ -1,173 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| using System.Security.Cryptography; |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.Bundle |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     ///  BundleManifest is a description of the contents of a bundle file. |  | ||||||
|     ///  This class handles creation and consumption of bundle-manifests. |  | ||||||
|     /// |  | ||||||
|     ///  Here is the description of the Bundle Layout: |  | ||||||
|     ///  _______________________________________________ |  | ||||||
|     ///  AppHost |  | ||||||
|     /// |  | ||||||
|     /// |  | ||||||
|     /// ------------Embedded Files --------------------- |  | ||||||
|     /// The embedded files including the app, its |  | ||||||
|     /// configuration files, dependencies, and |  | ||||||
|     /// possibly the runtime. |  | ||||||
|     /// |  | ||||||
|     /// |  | ||||||
|     /// |  | ||||||
|     /// |  | ||||||
|     /// |  | ||||||
|     /// |  | ||||||
|     /// |  | ||||||
|     /// ------------ Bundle Header ------------- |  | ||||||
|     ///     MajorVersion |  | ||||||
|     ///     MinorVersion |  | ||||||
|     ///     NumEmbeddedFiles |  | ||||||
|     ///     ExtractionID |  | ||||||
|     ///     DepsJson Location [Version 2+] |  | ||||||
|     ///        Offset |  | ||||||
|     ///        Size |  | ||||||
|     ///     RuntimeConfigJson Location [Version 2+] |  | ||||||
|     ///        Offset |  | ||||||
|     ///        Size |  | ||||||
|     ///     Flags [Version 2+] |  | ||||||
|     /// - - - - - - Manifest Entries - - - - - - - - - - - |  | ||||||
|     ///     Series of FileEntries (for each embedded file) |  | ||||||
|     ///     [File Type, Name, Offset, Size information] |  | ||||||
|     /// |  | ||||||
|     /// |  | ||||||
|     /// |  | ||||||
|     /// _________________________________________________ |  | ||||||
|     /// </summary> |  | ||||||
|     public class Manifest |  | ||||||
|     { |  | ||||||
|         // NetcoreApp3CompatMode flag is set on a .net5 app, |  | ||||||
|         // which chooses to build single-file apps in .netcore3.x compat mode, |  | ||||||
|         // by constructing the bundler with BundleAllConent option. |  | ||||||
|         // This mode is expected to be deprecated in future versions of .NET. |  | ||||||
|         [Flags] |  | ||||||
|         private enum HeaderFlags : ulong |  | ||||||
|         { |  | ||||||
|             None = 0, |  | ||||||
|             NetcoreApp3CompatMode = 1 |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         // Bundle ID is a string that is used to uniquely |  | ||||||
|         // identify this bundle. It is choosen to be compatible |  | ||||||
|         // with path-names so that the AppHost can use it in |  | ||||||
|         // extraction path. |  | ||||||
|         public string BundleID { get; private set; } |  | ||||||
|         //Same as Path.GetRandomFileName |  | ||||||
|         private const int BundleIdLength = 12; |  | ||||||
|         private SHA256 bundleHash = SHA256.Create(); |  | ||||||
|         public readonly uint BundleMajorVersion; |  | ||||||
|         // The Minor version is currently unused, and is always zero |  | ||||||
|         public const uint BundleMinorVersion = 0; |  | ||||||
|         private FileEntry DepsJsonEntry; |  | ||||||
|         private FileEntry RuntimeConfigJsonEntry; |  | ||||||
|         private HeaderFlags Flags; |  | ||||||
|         public List<FileEntry> Files; |  | ||||||
|         public string BundleVersion => $"{BundleMajorVersion}.{BundleMinorVersion}"; |  | ||||||
| 
 |  | ||||||
|         public Manifest(uint bundleMajorVersion, bool netcoreapp3CompatMode = false) |  | ||||||
|         { |  | ||||||
|             BundleMajorVersion = bundleMajorVersion; |  | ||||||
|             Files = new List<FileEntry>(); |  | ||||||
|             Flags = (netcoreapp3CompatMode) ? HeaderFlags.NetcoreApp3CompatMode : HeaderFlags.None; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public FileEntry AddEntry(FileType type, FileStream fileContent, string relativePath, long offset, long compressedSize, uint bundleMajorVersion) |  | ||||||
|         { |  | ||||||
|             if (bundleHash == null) |  | ||||||
|             { |  | ||||||
|                 throw new InvalidOperationException("It is forbidden to change Manifest state after it was written or BundleId was obtained."); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             FileEntry entry = new FileEntry(type, relativePath, offset, fileContent.Length, compressedSize, bundleMajorVersion); |  | ||||||
|             Files.Add(entry); |  | ||||||
| 
 |  | ||||||
|             fileContent.Position = 0; |  | ||||||
|             byte[] hashBytes = ComputeSha256Hash(fileContent); |  | ||||||
|             bundleHash.TransformBlock(hashBytes, 0, hashBytes.Length, hashBytes, 0); |  | ||||||
| 
 |  | ||||||
|             switch (entry.Type) |  | ||||||
|             { |  | ||||||
|                 case FileType.DepsJson: |  | ||||||
|                     DepsJsonEntry = entry; |  | ||||||
|                     break; |  | ||||||
|                 case FileType.RuntimeConfigJson: |  | ||||||
|                     RuntimeConfigJsonEntry = entry; |  | ||||||
|                     break; |  | ||||||
| 
 |  | ||||||
|                 case FileType.Assembly: |  | ||||||
|                     break; |  | ||||||
| 
 |  | ||||||
|                 default: |  | ||||||
|                     break; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return entry; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private static byte[] ComputeSha256Hash(Stream stream) |  | ||||||
|         { |  | ||||||
|             using (SHA256 sha = SHA256.Create()) |  | ||||||
|             { |  | ||||||
|                 return sha.ComputeHash(stream); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private string GenerateDeterministicId() |  | ||||||
|         { |  | ||||||
|             bundleHash.TransformFinalBlock(Array.Empty<byte>(), 0, 0); |  | ||||||
|             byte[] manifestHash = bundleHash.Hash; |  | ||||||
|             bundleHash.Dispose(); |  | ||||||
|             bundleHash = null; |  | ||||||
| 
 |  | ||||||
|             return Convert.ToBase64String(manifestHash).Substring(BundleIdLength).Replace('/', '_'); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public long Write(BinaryWriter writer) |  | ||||||
|         { |  | ||||||
|             BundleID = BundleID ?? GenerateDeterministicId(); |  | ||||||
| 
 |  | ||||||
|             long startOffset = writer.BaseStream.Position; |  | ||||||
| 
 |  | ||||||
|             // Write the bundle header |  | ||||||
|             writer.Write(BundleMajorVersion); |  | ||||||
|             writer.Write(BundleMinorVersion); |  | ||||||
|             writer.Write(Files.Count); |  | ||||||
|             writer.Write(BundleID); |  | ||||||
| 
 |  | ||||||
|             if (BundleMajorVersion >= 2) |  | ||||||
|             { |  | ||||||
|                 writer.Write((DepsJsonEntry != null) ? DepsJsonEntry.Offset : 0); |  | ||||||
|                 writer.Write((DepsJsonEntry != null) ? DepsJsonEntry.Size : 0); |  | ||||||
| 
 |  | ||||||
|                 writer.Write((RuntimeConfigJsonEntry != null) ? RuntimeConfigJsonEntry.Offset : 0); |  | ||||||
|                 writer.Write((RuntimeConfigJsonEntry != null) ? RuntimeConfigJsonEntry.Size : 0); |  | ||||||
| 
 |  | ||||||
|                 writer.Write((ulong)Flags); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             // Write the manifest entries |  | ||||||
|             foreach (FileEntry entry in Files) |  | ||||||
|             { |  | ||||||
|                 entry.Write(writer); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return startOffset; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public bool Contains(string relativePath) |  | ||||||
|         { |  | ||||||
|             return Files.Any(entry => relativePath.Equals(entry.RelativePath)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,118 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| using Microsoft.NET.HostModel.AppHost; |  | ||||||
| using System.Diagnostics; |  | ||||||
| using System.Runtime.InteropServices; |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.Bundle |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// TargetInfo: Information about the target for which the single-file bundle is built. |  | ||||||
|     /// |  | ||||||
|     /// Currently the TargetInfo only tracks: |  | ||||||
|     ///   - the target operating system |  | ||||||
|     ///   - the target architecture |  | ||||||
|     ///   - the target framework |  | ||||||
|     ///   - the default options for this target |  | ||||||
|     ///   - the assembly alignment for this target |  | ||||||
|     /// </summary> |  | ||||||
| 
 |  | ||||||
|     public class TargetInfo |  | ||||||
|     { |  | ||||||
|         public readonly OSPlatform OS; |  | ||||||
|         public readonly Architecture Arch; |  | ||||||
|         public readonly Version FrameworkVersion; |  | ||||||
|         public readonly uint BundleMajorVersion; |  | ||||||
|         public readonly BundleOptions DefaultOptions; |  | ||||||
|         public readonly int AssemblyAlignment; |  | ||||||
| 
 |  | ||||||
|         public TargetInfo(OSPlatform? os, Architecture? arch, Version targetFrameworkVersion) |  | ||||||
|         { |  | ||||||
|             OS = os ?? HostOS; |  | ||||||
|             Arch = arch ?? RuntimeInformation.OSArchitecture; |  | ||||||
|             FrameworkVersion = targetFrameworkVersion ?? net60; |  | ||||||
| 
 |  | ||||||
|             Debug.Assert(IsLinux || IsOSX || IsWindows); |  | ||||||
| 
 |  | ||||||
|             if (FrameworkVersion.CompareTo(net60) >= 0) |  | ||||||
|             { |  | ||||||
|                 BundleMajorVersion = 6u; |  | ||||||
|                 DefaultOptions = BundleOptions.None; |  | ||||||
|             } |  | ||||||
|             else if (FrameworkVersion.CompareTo(net50) >= 0) |  | ||||||
|             { |  | ||||||
|                 BundleMajorVersion = 2u; |  | ||||||
|                 DefaultOptions = BundleOptions.None; |  | ||||||
|             } |  | ||||||
|             else if (FrameworkVersion.Major == 3 && (FrameworkVersion.Minor == 0 || FrameworkVersion.Minor == 1)) |  | ||||||
|             { |  | ||||||
|                 BundleMajorVersion = 1u; |  | ||||||
|                 DefaultOptions = BundleOptions.BundleAllContent; |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 throw new ArgumentException($"Invalid input: Unsupported Target Framework Version {targetFrameworkVersion}"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (IsLinux && Arch == Architecture.Arm64) |  | ||||||
|             { |  | ||||||
|                 // We align assemblies in the bundle at 4K so that we can use mmap on Linux without changing the page alignment of ARM64 R2R code. |  | ||||||
|                 // This is only necessary for R2R assemblies, but we do it for all assemblies for simplicity. |  | ||||||
|                 // See https://github.com/dotnet/runtime/issues/41832. |  | ||||||
|                 AssemblyAlignment = 4096; |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 // Otherwise, assemblies are 16 bytes aligned, so that their sections can be memory-mapped cache aligned. |  | ||||||
|                 AssemblyAlignment = 16; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public bool IsNativeBinary(string filePath) |  | ||||||
|         { |  | ||||||
|             return IsLinux ? ElfUtils.IsElfImage(filePath) : IsOSX ? MachOUtils.IsMachOImage(filePath) : PEUtils.IsPEImage(filePath); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public string GetAssemblyName(string hostName) |  | ||||||
|         { |  | ||||||
|             // This logic to calculate assembly name from hostName should be removed (and probably moved to test helpers) |  | ||||||
|             // once the SDK in the correct assembly name. |  | ||||||
|             return (IsWindows ? Path.GetFileNameWithoutExtension(hostName) : hostName); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public override string ToString() |  | ||||||
|         { |  | ||||||
|             string os = IsWindows ? "win" : IsLinux ? "linux" : "osx"; |  | ||||||
|             string arch = Arch.ToString().ToLowerInvariant(); |  | ||||||
|             return $"OS: {os} Arch: {arch} FrameworkVersion: {FrameworkVersion}"; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private static OSPlatform HostOS => RuntimeInformation.IsOSPlatform(OSPlatform.Linux) ? OSPlatform.Linux : |  | ||||||
|                                     RuntimeInformation.IsOSPlatform(OSPlatform.OSX) ? OSPlatform.OSX : OSPlatform.Windows; |  | ||||||
| 
 |  | ||||||
|         public bool IsLinux => OS.Equals(OSPlatform.Linux); |  | ||||||
|         public bool IsOSX => OS.Equals(OSPlatform.OSX); |  | ||||||
|         public bool IsWindows => OS.Equals(OSPlatform.Windows); |  | ||||||
| 
 |  | ||||||
|         // The .net core 3 apphost doesn't care about semantics of FileType -- all files are extracted at startup. |  | ||||||
|         // However, the apphost checks that the FileType value is within expected bounds, so set it to the first enumeration. |  | ||||||
|         public FileType TargetSpecificFileType(FileType fileType) => (BundleMajorVersion == 1) ? FileType.Unknown : fileType; |  | ||||||
| 
 |  | ||||||
|         // In .net core 3.x, bundle processing happens within the AppHost. |  | ||||||
|         // Therefore HostFxr and HostPolicy can be bundled within the single-file app. |  | ||||||
|         // In .net 5, bundle processing happens in HostFxr and HostPolicy libraries. |  | ||||||
|         // Therefore, these libraries themselves cannot be bundled into the single-file app. |  | ||||||
|         // This problem is mitigated by statically linking these host components with the AppHost. |  | ||||||
|         // https://github.com/dotnet/runtime/issues/32823 |  | ||||||
|         public bool ShouldExclude(string relativePath) => |  | ||||||
|             (FrameworkVersion.Major != 3) && (relativePath.Equals(HostFxr) || relativePath.Equals(HostPolicy)); |  | ||||||
| 
 |  | ||||||
|         private readonly Version net60 = new Version(6, 0); |  | ||||||
|         private readonly Version net50 = new Version(5, 0); |  | ||||||
|         private string HostFxr => IsWindows ? "hostfxr.dll" : IsLinux ? "libhostfxr.so" : "libhostfxr.dylib"; |  | ||||||
|         private string HostPolicy => IsWindows ? "hostpolicy.dll" : IsLinux ? "libhostpolicy.so" : "libhostpolicy.dylib"; |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,31 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel.Bundle |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     ///  Tracing utilities for diagnostic output |  | ||||||
|     /// </summary> |  | ||||||
|     public class Trace |  | ||||||
|     { |  | ||||||
|         private readonly bool Verbose; |  | ||||||
| 
 |  | ||||||
|         public Trace(bool verbose) |  | ||||||
|         { |  | ||||||
|             Verbose = verbose; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public void Log(string fmt, params object[] args) |  | ||||||
|         { |  | ||||||
|             if (Verbose) |  | ||||||
|             { |  | ||||||
|                 Console.WriteLine("LOG: " + fmt, args); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public void Error(string type, string message) |  | ||||||
|         { |  | ||||||
|             Console.Error.WriteLine($"ERROR: {message}"); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,33 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| using System.Diagnostics; |  | ||||||
| using System.Runtime.InteropServices; |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel |  | ||||||
| { |  | ||||||
|     internal static class HostModelUtils |  | ||||||
|     { |  | ||||||
|         private const string CodesignPath = @"/usr/bin/codesign"; |  | ||||||
| 
 |  | ||||||
|         public static bool IsCodesignAvailable() => File.Exists(CodesignPath); |  | ||||||
| 
 |  | ||||||
|         public static (int ExitCode, string StdErr) RunCodesign(string args, string appHostPath) |  | ||||||
|         { |  | ||||||
|             Debug.Assert(RuntimeInformation.IsOSPlatform(OSPlatform.OSX)); |  | ||||||
|             Debug.Assert(IsCodesignAvailable()); |  | ||||||
| 
 |  | ||||||
|             var psi = new ProcessStartInfo() { |  | ||||||
|                 Arguments = $"{args} \"{appHostPath}\"", |  | ||||||
|                 FileName = CodesignPath, |  | ||||||
|                 RedirectStandardError = true, |  | ||||||
|                 UseShellExecute = false, |  | ||||||
|             }; |  | ||||||
| 
 |  | ||||||
|             using (var p = Process.Start(psi)) { |  | ||||||
|                 p.WaitForExit(); |  | ||||||
|                 return (p.ExitCode, p.StandardError.ReadToEnd()); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| Host Model |  | ||||||
| =================================== |  | ||||||
|  |  | ||||||
| HostModel is a library used by the [SDK](https://github.com/dotnet/sdk) to perform certain transformations on host executables. The main services implemented in HostModel are: |  | ||||||
|  |  | ||||||
| * AppHost rewriter:  Embeds the App Name into the AppHost executable. On Windows, also copies resources from App.dll to the AppHost. |  | ||||||
| * ComHost rewriter: Creates a ComHost with an embedded CLSIDMap file to map CLSIDs to .NET Classes. |  | ||||||
|  |  | ||||||
| * Single-file bundler: Embeds an application and its dependencies into the AppHost, to publish a single executable, as described [here](https://github.com/dotnet/designs/blob/master/accepted/2020/single-file/design.md). |  | ||||||
|  |  | ||||||
| The HostModel library is in the Runtime repo because: |  | ||||||
|  |  | ||||||
| * The implementations of the host and HostModel are closely related, which facilitates easy development, update, and testing. |  | ||||||
| * Separating the HostModel implementation from SDK repo repo aligns with code ownership, and facilitates maintenance. |  | ||||||
|  |  | ||||||
| The build targets/tasks that use the HostModel library are in the SDK repo because: |  | ||||||
|  |  | ||||||
| * This facilitates the MSBuild tasks to be multi-targeted. |  | ||||||
| * It helps generate localized error messages, since SDK repo has the localization infrastructure. |  | ||||||
| @@ -1,49 +0,0 @@ | |||||||
| // If updating HostModel, mark the ResourceUpdater.cs class as partial so these functions can get mixed in |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel |  | ||||||
| { |  | ||||||
|     public partial class ResourceUpdater |  | ||||||
|     { |  | ||||||
|         public ResourceUpdater(string peFile, bool bDeleteExistingResources) |  | ||||||
|         { |  | ||||||
|             hUpdate = Kernel32.BeginUpdateResource(peFile, bDeleteExistingResources); |  | ||||||
|             if (hUpdate.IsInvalid) { |  | ||||||
|                 ThrowExceptionForLastWin32Error(); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public ResourceUpdater AddResource(byte[] data, string lpType, IntPtr lpName, ushort langId) |  | ||||||
|         { |  | ||||||
|             if (hUpdate.IsInvalid) { |  | ||||||
|                 ThrowExceptionForInvalidUpdate(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!IsIntResource(lpName)) { |  | ||||||
|                 throw new ArgumentException("AddResource can only be used with integer resource names"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!Kernel32.UpdateResource(hUpdate, lpType, lpName, langId, data, (uint) data.Length)) { |  | ||||||
|                 ThrowExceptionForLastWin32Error(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return this; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         //public ResourceUpdater ClearResource(string lpType, IntPtr lpName, ushort langId) |  | ||||||
|         //{ |  | ||||||
|         //    if (hUpdate.IsInvalid) { |  | ||||||
|         //        ThrowExceptionForInvalidUpdate(); |  | ||||||
|         //    } |  | ||||||
| 
 |  | ||||||
|         //    if (!IsIntResource(lpName)) { |  | ||||||
|         //        throw new ArgumentException("AddResource can only be used with integer resource names"); |  | ||||||
|         //    } |  | ||||||
| 
 |  | ||||||
|         //    if (!Kernel32.UpdateResource(hUpdate, lpType, lpName, langId, null, 0)) { |  | ||||||
|         //        ThrowExceptionForLastWin32Error(); |  | ||||||
|         //    } |  | ||||||
| 
 |  | ||||||
|         //    return this; |  | ||||||
|         //} |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,458 +0,0 @@ | |||||||
| // Licensed to the .NET Foundation under one or more agreements. |  | ||||||
| // The .NET Foundation licenses this file to you under the MIT license. |  | ||||||
| 
 |  | ||||||
| using System.ComponentModel; |  | ||||||
| using System.Diagnostics; |  | ||||||
| using System.Runtime.InteropServices; |  | ||||||
| 
 |  | ||||||
| namespace Microsoft.NET.HostModel |  | ||||||
| { |  | ||||||
|     /// <summary> |  | ||||||
|     /// Provides methods for modifying the embedded native resources |  | ||||||
|     /// in a PE image. It currently only works on Windows, because it |  | ||||||
|     /// requires various kernel32 APIs. |  | ||||||
|     /// </summary> |  | ||||||
|     public partial class ResourceUpdater : IDisposable |  | ||||||
|     { |  | ||||||
|         private sealed class Kernel32 |  | ||||||
|         { |  | ||||||
|             // |  | ||||||
|             // Native methods for updating resources |  | ||||||
|             // |  | ||||||
| 
 |  | ||||||
|             [DllImport(nameof(Kernel32), CharSet = CharSet.Unicode, SetLastError = true)] |  | ||||||
|             public static extern SafeUpdateHandle BeginUpdateResource(string pFileName, |  | ||||||
|                                                                       [MarshalAs(UnmanagedType.Bool)] bool bDeleteExistingResources); |  | ||||||
| 
 |  | ||||||
|             // Update a resource with data from an IntPtr |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             [return: MarshalAs(UnmanagedType.Bool)] |  | ||||||
|             public static extern bool UpdateResource(SafeUpdateHandle hUpdate, |  | ||||||
|                                                      IntPtr lpType, |  | ||||||
|                                                      IntPtr lpName, |  | ||||||
|                                                      ushort wLanguage, |  | ||||||
|                                                      IntPtr lpData, |  | ||||||
|                                                      uint cbData); |  | ||||||
| 
 |  | ||||||
|             // Update a resource with data from a managed byte[] |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             [return: MarshalAs(UnmanagedType.Bool)] |  | ||||||
|             public static extern bool UpdateResource(SafeUpdateHandle hUpdate, |  | ||||||
|                                                      IntPtr lpType, |  | ||||||
|                                                      IntPtr lpName, |  | ||||||
|                                                      ushort wLanguage, |  | ||||||
|                                                      [MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 5)] byte[] lpData, |  | ||||||
|                                                      uint cbData); |  | ||||||
| 
 |  | ||||||
|             // Update a resource with data from a managed byte[] |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             [return: MarshalAs(UnmanagedType.Bool)] |  | ||||||
|             public static extern bool UpdateResource(SafeUpdateHandle hUpdate, |  | ||||||
|                                                      string lpType, |  | ||||||
|                                                      IntPtr lpName, |  | ||||||
|                                                      ushort wLanguage, |  | ||||||
|                                                      [MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 5)] byte[] lpData, |  | ||||||
|                                                      uint cbData); |  | ||||||
| 
 |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             [return: MarshalAs(UnmanagedType.Bool)] |  | ||||||
|             public static extern bool EndUpdateResource(SafeUpdateHandle hUpdate, |  | ||||||
|                                                         bool fDiscard); |  | ||||||
| 
 |  | ||||||
|             // The IntPtr version of this dllimport is used in the |  | ||||||
|             // SafeHandle implementation |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             [return: MarshalAs(UnmanagedType.Bool)] |  | ||||||
|             public static extern bool EndUpdateResource(IntPtr hUpdate, |  | ||||||
|                                                         bool fDiscard); |  | ||||||
| 
 |  | ||||||
|             public const ushort LangID_LangNeutral_SublangNeutral = 0; |  | ||||||
| 
 |  | ||||||
|             // |  | ||||||
|             // Native methods used to read resources from a PE file |  | ||||||
|             // |  | ||||||
| 
 |  | ||||||
|             // Loading and freeing PE files |  | ||||||
| 
 |  | ||||||
|             public enum LoadLibraryFlags : uint |  | ||||||
|             { |  | ||||||
|                 LOAD_LIBRARY_AS_DATAFILE_EXCLUSIVE = 0x00000040, |  | ||||||
|                 LOAD_LIBRARY_AS_IMAGE_RESOURCE = 0x00000020 |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             [DllImport(nameof(Kernel32), CharSet = CharSet.Unicode, SetLastError = true)] |  | ||||||
|             public static extern IntPtr LoadLibraryEx(string lpFileName, |  | ||||||
|                                                       IntPtr hReservedNull, |  | ||||||
|                                                       LoadLibraryFlags dwFlags); |  | ||||||
| 
 |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             [return: MarshalAs(UnmanagedType.Bool)] |  | ||||||
|             public static extern bool FreeLibrary(IntPtr hModule); |  | ||||||
| 
 |  | ||||||
|             // Enumerating resources |  | ||||||
| 
 |  | ||||||
|             public delegate bool EnumResTypeProc(IntPtr hModule, |  | ||||||
|                                                  IntPtr lpType, |  | ||||||
|                                                  IntPtr lParam); |  | ||||||
| 
 |  | ||||||
|             public delegate bool EnumResNameProc(IntPtr hModule, |  | ||||||
|                                                  IntPtr lpType, |  | ||||||
|                                                  IntPtr lpName, |  | ||||||
|                                                  IntPtr lParam); |  | ||||||
| 
 |  | ||||||
|             public delegate bool EnumResLangProc(IntPtr hModule, |  | ||||||
|                                                  IntPtr lpType, |  | ||||||
|                                                  IntPtr lpName, |  | ||||||
|                                                  ushort wLang, |  | ||||||
|                                                  IntPtr lParam); |  | ||||||
| 
 |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             [return: MarshalAs(UnmanagedType.Bool)] |  | ||||||
|             public static extern bool EnumResourceTypes(IntPtr hModule, |  | ||||||
|                                                          EnumResTypeProc lpEnumFunc, |  | ||||||
|                                                          IntPtr lParam); |  | ||||||
| 
 |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             [return: MarshalAs(UnmanagedType.Bool)] |  | ||||||
|             public static extern bool EnumResourceNames(IntPtr hModule, |  | ||||||
|                                                          IntPtr lpType, |  | ||||||
|                                                          EnumResNameProc lpEnumFunc, |  | ||||||
|                                                          IntPtr lParam); |  | ||||||
| 
 |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             [return: MarshalAs(UnmanagedType.Bool)] |  | ||||||
|             public static extern bool EnumResourceLanguages(IntPtr hModule, |  | ||||||
|                                                             IntPtr lpType, |  | ||||||
|                                                             IntPtr lpName, |  | ||||||
|                                                             EnumResLangProc lpEnumFunc, |  | ||||||
|                                                             IntPtr lParam); |  | ||||||
| 
 |  | ||||||
|             public const int UserStoppedResourceEnumerationHRESULT = unchecked((int) 0x80073B02); |  | ||||||
|             public const int ResourceDataNotFoundHRESULT = unchecked((int) 0x80070714); |  | ||||||
| 
 |  | ||||||
|             // Querying and loading resources |  | ||||||
| 
 |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             public static extern IntPtr FindResourceEx(IntPtr hModule, |  | ||||||
|                                                        IntPtr lpType, |  | ||||||
|                                                        IntPtr lpName, |  | ||||||
|                                                        ushort wLanguage); |  | ||||||
| 
 |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             public static extern IntPtr LoadResource(IntPtr hModule, |  | ||||||
|                                                      IntPtr hResInfo); |  | ||||||
| 
 |  | ||||||
|             [DllImport(nameof(Kernel32))] // does not call SetLastError |  | ||||||
|             public static extern IntPtr LockResource(IntPtr hResData); |  | ||||||
| 
 |  | ||||||
|             [DllImport(nameof(Kernel32), SetLastError = true)] |  | ||||||
|             public static extern uint SizeofResource(IntPtr hModule, |  | ||||||
|                                                      IntPtr hResInfo); |  | ||||||
| 
 |  | ||||||
|             public const int ERROR_CALL_NOT_IMPLEMENTED = 0x78; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Holds the update handle returned by BeginUpdateResource. |  | ||||||
|         /// Normally, native resources for the update handle are |  | ||||||
|         /// released by a call to ResourceUpdater.Update(). In case |  | ||||||
|         /// this doesn't happen, the SafeUpdateHandle will release the |  | ||||||
|         /// native resources for the update handle without updating |  | ||||||
|         /// the target file. |  | ||||||
|         /// </summary> |  | ||||||
|         private sealed class SafeUpdateHandle : SafeHandle |  | ||||||
|         { |  | ||||||
|             public SafeUpdateHandle() : base(IntPtr.Zero, true) |  | ||||||
|             { |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             public override bool IsInvalid => handle == IntPtr.Zero; |  | ||||||
| 
 |  | ||||||
|             protected override bool ReleaseHandle() |  | ||||||
|             { |  | ||||||
|                 // discard pending updates without writing them |  | ||||||
|                 return Kernel32.EndUpdateResource(handle, true); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Holds the native handle for the resource update. |  | ||||||
|         /// </summary> |  | ||||||
|         private readonly SafeUpdateHandle hUpdate; |  | ||||||
| 
 |  | ||||||
|         ///<summary> |  | ||||||
|         /// Determines if the ResourceUpdater is supported by the current operating system. |  | ||||||
|         /// Some versions of Windows, such as Nano Server, do not support the needed APIs. |  | ||||||
|         /// </summary> |  | ||||||
|         public static bool IsSupportedOS() |  | ||||||
|         { |  | ||||||
| #if NETSTANDARD2_0_OR_GREATER || NET5_0_OR_GREATER |  | ||||||
|             if (!RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) { |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
| #endif |  | ||||||
| 
 |  | ||||||
|             try { |  | ||||||
|                 // On Nano Server 1709+, `BeginUpdateResource` is exported but returns a null handle with a zero error |  | ||||||
|                 // Try to call `BeginUpdateResource` with an invalid parameter; the error should be non-zero if supported |  | ||||||
|                 // On Nano Server 20213, `BeginUpdateResource` fails with ERROR_CALL_NOT_IMPLEMENTED |  | ||||||
|                 using (var handle = Kernel32.BeginUpdateResource("", false)) { |  | ||||||
|                     int lastWin32Error = Marshal.GetLastWin32Error(); |  | ||||||
| 
 |  | ||||||
|                     if (handle.IsInvalid && (lastWin32Error == 0 || lastWin32Error == Kernel32.ERROR_CALL_NOT_IMPLEMENTED)) { |  | ||||||
|                         return false; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } catch (EntryPointNotFoundException) { |  | ||||||
|                 // BeginUpdateResource isn't exported from Kernel32 |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Create a resource updater for the given PE file. This will |  | ||||||
|         /// acquire a native resource update handle for the file, |  | ||||||
|         /// preparing it for updates. Resources can be added to this |  | ||||||
|         /// updater, which will queue them for update. The target PE |  | ||||||
|         /// file will not be modified until Update() is called, after |  | ||||||
|         /// which the ResourceUpdater can not be used for further |  | ||||||
|         /// updates. |  | ||||||
|         /// </summary> |  | ||||||
|         public ResourceUpdater(string peFile) |  | ||||||
|         { |  | ||||||
|             hUpdate = Kernel32.BeginUpdateResource(peFile, false); |  | ||||||
|             if (hUpdate.IsInvalid) { |  | ||||||
|                 ThrowExceptionForLastWin32Error(); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Add all resources from a source PE file. It is assumed |  | ||||||
|         /// that the input is a valid PE file. If it is not, an |  | ||||||
|         /// exception will be thrown. This will not modify the target |  | ||||||
|         /// until Update() is called. |  | ||||||
|         /// Throws an InvalidOperationException if Update() was already called. |  | ||||||
|         /// </summary> |  | ||||||
|         public ResourceUpdater AddResourcesFromPEImage(string peFile) |  | ||||||
|         { |  | ||||||
|             if (hUpdate.IsInvalid) { |  | ||||||
|                 ThrowExceptionForInvalidUpdate(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             // Using both flags lets the OS loader decide how to load |  | ||||||
|             // it most efficiently. Either mode will prevent other |  | ||||||
|             // processes from modifying the module while it is loaded. |  | ||||||
|             IntPtr hModule = Kernel32.LoadLibraryEx(peFile, IntPtr.Zero, |  | ||||||
|                                                     Kernel32.LoadLibraryFlags.LOAD_LIBRARY_AS_DATAFILE_EXCLUSIVE | |  | ||||||
|                                                     Kernel32.LoadLibraryFlags.LOAD_LIBRARY_AS_IMAGE_RESOURCE); |  | ||||||
|             if (hModule == IntPtr.Zero) { |  | ||||||
|                 ThrowExceptionForLastWin32Error(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             var enumTypesCallback = new Kernel32.EnumResTypeProc(EnumAndUpdateTypesCallback); |  | ||||||
|             var errorInfo = new EnumResourcesErrorInfo(); |  | ||||||
|             GCHandle errorInfoHandle = GCHandle.Alloc(errorInfo); |  | ||||||
|             var errorInfoPtr = GCHandle.ToIntPtr(errorInfoHandle); |  | ||||||
| 
 |  | ||||||
|             try { |  | ||||||
|                 if (!Kernel32.EnumResourceTypes(hModule, enumTypesCallback, errorInfoPtr)) { |  | ||||||
|                     if (Marshal.GetHRForLastWin32Error() != Kernel32.ResourceDataNotFoundHRESULT) { |  | ||||||
|                         CaptureEnumResourcesErrorInfo(errorInfoPtr); |  | ||||||
|                         errorInfo.ThrowException(); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } finally { |  | ||||||
|                 errorInfoHandle.Free(); |  | ||||||
| 
 |  | ||||||
|                 if (!Kernel32.FreeLibrary(hModule)) { |  | ||||||
|                     ThrowExceptionForLastWin32Error(); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return this; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         internal static bool IsIntResource(IntPtr lpType) |  | ||||||
|         { |  | ||||||
|             return ((uint) lpType >> 16) == 0; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Add a language-neutral integer resource from a byte[] with |  | ||||||
|         /// a particular type and name. This will not modify the |  | ||||||
|         /// target until Update() is called. |  | ||||||
|         /// Throws an InvalidOperationException if Update() was already called. |  | ||||||
|         /// </summary> |  | ||||||
|         public ResourceUpdater AddResource(byte[] data, IntPtr lpType, IntPtr lpName) |  | ||||||
|         { |  | ||||||
|             if (hUpdate.IsInvalid) { |  | ||||||
|                 ThrowExceptionForInvalidUpdate(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!IsIntResource(lpType) || !IsIntResource(lpName)) { |  | ||||||
|                 throw new ArgumentException("AddResource can only be used with integer resource types"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!Kernel32.UpdateResource(hUpdate, lpType, lpName, Kernel32.LangID_LangNeutral_SublangNeutral, data, (uint) data.Length)) { |  | ||||||
|                 ThrowExceptionForLastWin32Error(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return this; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Add a language-neutral integer resource from a byte[] with |  | ||||||
|         /// a particular type and name. This will not modify the |  | ||||||
|         /// target until Update() is called. |  | ||||||
|         /// Throws an InvalidOperationException if Update() was already called. |  | ||||||
|         /// </summary> |  | ||||||
|         public ResourceUpdater AddResource(byte[] data, string lpType, IntPtr lpName) |  | ||||||
|         { |  | ||||||
|             if (hUpdate.IsInvalid) { |  | ||||||
|                 ThrowExceptionForInvalidUpdate(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!IsIntResource(lpName)) { |  | ||||||
|                 throw new ArgumentException("AddResource can only be used with integer resource names"); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!Kernel32.UpdateResource(hUpdate, lpType, lpName, Kernel32.LangID_LangNeutral_SublangNeutral, data, (uint) data.Length)) { |  | ||||||
|                 ThrowExceptionForLastWin32Error(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return this; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         /// <summary> |  | ||||||
|         /// Write the pending resource updates to the target PE |  | ||||||
|         /// file. After this, the ResourceUpdater no longer maintains |  | ||||||
|         /// an update handle, and can not be used for further updates. |  | ||||||
|         /// Throws an InvalidOperationException if Update() was already called. |  | ||||||
|         /// </summary> |  | ||||||
|         public void Update() |  | ||||||
|         { |  | ||||||
|             if (hUpdate.IsInvalid) { |  | ||||||
|                 ThrowExceptionForInvalidUpdate(); |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             try { |  | ||||||
|                 if (!Kernel32.EndUpdateResource(hUpdate, false)) { |  | ||||||
|                     ThrowExceptionForLastWin32Error(); |  | ||||||
|                 } |  | ||||||
|             } finally { |  | ||||||
|                 hUpdate.SetHandleAsInvalid(); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private bool EnumAndUpdateTypesCallback(IntPtr hModule, IntPtr lpType, IntPtr lParam) |  | ||||||
|         { |  | ||||||
|             var enumNamesCallback = new Kernel32.EnumResNameProc(EnumAndUpdateNamesCallback); |  | ||||||
|             if (!Kernel32.EnumResourceNames(hModule, lpType, enumNamesCallback, lParam)) { |  | ||||||
|                 CaptureEnumResourcesErrorInfo(lParam); |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private bool EnumAndUpdateNamesCallback(IntPtr hModule, IntPtr lpType, IntPtr lpName, IntPtr lParam) |  | ||||||
|         { |  | ||||||
|             var enumLanguagesCallback = new Kernel32.EnumResLangProc(EnumAndUpdateLanguagesCallback); |  | ||||||
|             if (!Kernel32.EnumResourceLanguages(hModule, lpType, lpName, enumLanguagesCallback, lParam)) { |  | ||||||
|                 CaptureEnumResourcesErrorInfo(lParam); |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private bool EnumAndUpdateLanguagesCallback(IntPtr hModule, IntPtr lpType, IntPtr lpName, ushort wLang, IntPtr lParam) |  | ||||||
|         { |  | ||||||
|             IntPtr hResource = Kernel32.FindResourceEx(hModule, lpType, lpName, wLang); |  | ||||||
|             if (hResource == IntPtr.Zero) { |  | ||||||
|                 CaptureEnumResourcesErrorInfo(lParam); |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             // hResourceLoaded is just a handle to the resource, which |  | ||||||
|             // can be used to get the resource data |  | ||||||
|             IntPtr hResourceLoaded = Kernel32.LoadResource(hModule, hResource); |  | ||||||
|             if (hResourceLoaded == IntPtr.Zero) { |  | ||||||
|                 CaptureEnumResourcesErrorInfo(lParam); |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             // This doesn't actually lock memory. It just retrieves a |  | ||||||
|             // pointer to the resource data. The pointer is valid |  | ||||||
|             // until the module is unloaded. |  | ||||||
|             IntPtr lpResourceData = Kernel32.LockResource(hResourceLoaded); |  | ||||||
|             if (lpResourceData == IntPtr.Zero) { |  | ||||||
|                 ((EnumResourcesErrorInfo) GCHandle.FromIntPtr(lParam).Target).failedToLockResource = true; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (!Kernel32.UpdateResource(hUpdate, lpType, lpName, wLang, lpResourceData, Kernel32.SizeofResource(hModule, hResource))) { |  | ||||||
|                 CaptureEnumResourcesErrorInfo(lParam); |  | ||||||
|                 return false; |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private class EnumResourcesErrorInfo |  | ||||||
|         { |  | ||||||
|             public int hResult; |  | ||||||
|             public bool failedToLockResource; |  | ||||||
| 
 |  | ||||||
|             public void ThrowException() |  | ||||||
|             { |  | ||||||
|                 if (failedToLockResource) { |  | ||||||
|                     Debug.Assert(hResult == 0); |  | ||||||
|                     throw new ResourceNotAvailableException("Failed to lock resource"); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 Debug.Assert(hResult != 0); |  | ||||||
|                 throw new Win32Exception(hResult); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private static void CaptureEnumResourcesErrorInfo(IntPtr errorInfoPtr) |  | ||||||
|         { |  | ||||||
|             int hResult = Marshal.GetHRForLastWin32Error(); |  | ||||||
|             if (hResult != Kernel32.UserStoppedResourceEnumerationHRESULT) { |  | ||||||
|                 GCHandle errorInfoHandle = GCHandle.FromIntPtr(errorInfoPtr); |  | ||||||
|                 var errorInfo = (EnumResourcesErrorInfo) errorInfoHandle.Target; |  | ||||||
|                 errorInfo.hResult = hResult; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private class ResourceNotAvailableException : Exception |  | ||||||
|         { |  | ||||||
|             public ResourceNotAvailableException(string message) : base(message) |  | ||||||
|             { |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private static void ThrowExceptionForLastWin32Error() |  | ||||||
|         { |  | ||||||
|             throw new Win32Exception(Marshal.GetHRForLastWin32Error()); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         private static void ThrowExceptionForInvalidUpdate() |  | ||||||
|         { |  | ||||||
|             throw new InvalidOperationException("Update handle is invalid. This instance may not be used for further updates"); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public void Dispose() |  | ||||||
|         { |  | ||||||
|             Dispose(true); |  | ||||||
|             GC.SuppressFinalize(this); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         public void Dispose(bool disposing) |  | ||||||
|         { |  | ||||||
|             if (disposing) { |  | ||||||
|                 hUpdate.Dispose(); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,15 +0,0 @@ | |||||||
| <Project Sdk="Microsoft.NET.Sdk"> |  | ||||||
|  |  | ||||||
|   <PropertyGroup> |  | ||||||
|     <TargetFrameworks>net472;net6.0</TargetFrameworks> |  | ||||||
|     <ImplicitUsings>enable</ImplicitUsings> |  | ||||||
|     <NoWarn>$(NoWarn);CA2007;CS8002;IDE0161</NoWarn> |  | ||||||
|     <AllowUnsafeBlocks>true</AllowUnsafeBlocks> |  | ||||||
|     <RootNamespace>Microsoft.NET.HostModel</RootNamespace> |  | ||||||
|   </PropertyGroup> |  | ||||||
|  |  | ||||||
|   <ItemGroup> |  | ||||||
|     <PackageReference Include="System.Reflection.Metadata" Version="8.0.0" Condition="'$(TargetFramework)' == 'net472'" /> |  | ||||||
|   </ItemGroup> |  | ||||||
|  |  | ||||||
| </Project> |  | ||||||
| @@ -1,105 +0,0 @@ | |||||||
| { |  | ||||||
|   "version": 1, |  | ||||||
|   "dependencies": { |  | ||||||
|     ".NETFramework,Version=v4.7.2": { |  | ||||||
|       "Microsoft.SourceLink.GitHub": { |  | ||||||
|         "type": "Direct", |  | ||||||
|         "requested": "[8.0.0, )", |  | ||||||
|         "resolved": "8.0.0", |  | ||||||
|         "contentHash": "G5q7OqtwIyGTkeIOAc3u2ZuV/kicQaec5EaRnc0pIeSnh9LUjj+PYQrJYBURvDt7twGl2PKA7nSN0kz1Zw5bnQ==", |  | ||||||
|         "dependencies": { |  | ||||||
|           "Microsoft.Build.Tasks.Git": "8.0.0", |  | ||||||
|           "Microsoft.SourceLink.Common": "8.0.0" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "Nerdbank.GitVersioning": { |  | ||||||
|         "type": "Direct", |  | ||||||
|         "requested": "[3.6.*, )", |  | ||||||
|         "resolved": "3.6.133", |  | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |  | ||||||
|       }, |  | ||||||
|       "System.Reflection.Metadata": { |  | ||||||
|         "type": "Direct", |  | ||||||
|         "requested": "[8.0.0, )", |  | ||||||
|         "resolved": "8.0.0", |  | ||||||
|         "contentHash": "ptvgrFh7PvWI8bcVqG5rsA/weWM09EnthFHR5SCnS6IN+P4mj6rE1lBDC4U8HL9/57htKAqy4KQ3bBj84cfYyQ==", |  | ||||||
|         "dependencies": { |  | ||||||
|           "System.Collections.Immutable": "8.0.0", |  | ||||||
|           "System.Memory": "4.5.5" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "Microsoft.Build.Tasks.Git": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "8.0.0", |  | ||||||
|         "contentHash": "bZKfSIKJRXLTuSzLudMFte/8CempWjVamNUR5eHJizsy+iuOuO/k2gnh7W0dHJmYY0tBf+gUErfluCv5mySAOQ==" |  | ||||||
|       }, |  | ||||||
|       "Microsoft.SourceLink.Common": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "8.0.0", |  | ||||||
|         "contentHash": "dk9JPxTCIevS75HyEQ0E4OVAFhB2N+V9ShCXf8Q6FkUQZDkgLI12y679Nym1YqsiSysuQskT7Z+6nUf3yab6Vw==" |  | ||||||
|       }, |  | ||||||
|       "System.Buffers": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "4.5.1", |  | ||||||
|         "contentHash": "Rw7ijyl1qqRS0YQD/WycNst8hUUMgrMH4FCn1nNm27M4VxchZ1js3fVjQaANHO5f3sN4isvP4a+Met9Y4YomAg==" |  | ||||||
|       }, |  | ||||||
|       "System.Collections.Immutable": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "8.0.0", |  | ||||||
|         "contentHash": "AurL6Y5BA1WotzlEvVaIDpqzpIPvYnnldxru8oXJU2yFxFUy3+pNXjXd1ymO+RA0rq0+590Q8gaz2l3Sr7fmqg==", |  | ||||||
|         "dependencies": { |  | ||||||
|           "System.Memory": "4.5.5", |  | ||||||
|           "System.Runtime.CompilerServices.Unsafe": "6.0.0" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "System.Memory": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "4.5.5", |  | ||||||
|         "contentHash": "XIWiDvKPXaTveaB7HVganDlOCRoj03l+jrwNvcge/t8vhGYKvqV+dMv6G4SAX2NoNmN0wZfVPTAlFwZcZvVOUw==", |  | ||||||
|         "dependencies": { |  | ||||||
|           "System.Buffers": "4.5.1", |  | ||||||
|           "System.Numerics.Vectors": "4.5.0", |  | ||||||
|           "System.Runtime.CompilerServices.Unsafe": "4.5.3" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "System.Numerics.Vectors": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "4.5.0", |  | ||||||
|         "contentHash": "QQTlPTl06J/iiDbJCiepZ4H//BVraReU4O4EoRw1U02H5TLUIT7xn3GnDp9AXPSlJUDyFs4uWjWafNX6WrAojQ==" |  | ||||||
|       }, |  | ||||||
|       "System.Runtime.CompilerServices.Unsafe": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "6.0.0", |  | ||||||
|         "contentHash": "/iUeP3tq1S0XdNNoMz5C9twLSrM/TH+qElHkXWaPvuNOt+99G75NrV0OS2EqHx5wMN7popYjpc8oTjC1y16DLg==" |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|     "net6.0": { |  | ||||||
|       "Microsoft.SourceLink.GitHub": { |  | ||||||
|         "type": "Direct", |  | ||||||
|         "requested": "[8.0.0, )", |  | ||||||
|         "resolved": "8.0.0", |  | ||||||
|         "contentHash": "G5q7OqtwIyGTkeIOAc3u2ZuV/kicQaec5EaRnc0pIeSnh9LUjj+PYQrJYBURvDt7twGl2PKA7nSN0kz1Zw5bnQ==", |  | ||||||
|         "dependencies": { |  | ||||||
|           "Microsoft.Build.Tasks.Git": "8.0.0", |  | ||||||
|           "Microsoft.SourceLink.Common": "8.0.0" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "Nerdbank.GitVersioning": { |  | ||||||
|         "type": "Direct", |  | ||||||
|         "requested": "[3.6.*, )", |  | ||||||
|         "resolved": "3.6.133", |  | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |  | ||||||
|       }, |  | ||||||
|       "Microsoft.Build.Tasks.Git": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "8.0.0", |  | ||||||
|         "contentHash": "bZKfSIKJRXLTuSzLudMFte/8CempWjVamNUR5eHJizsy+iuOuO/k2gnh7W0dHJmYY0tBf+gUErfluCv5mySAOQ==" |  | ||||||
|       }, |  | ||||||
|       "Microsoft.SourceLink.Common": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "8.0.0", |  | ||||||
|         "contentHash": "dk9JPxTCIevS75HyEQ0E4OVAFhB2N+V9ShCXf8Q6FkUQZDkgLI12y679Nym1YqsiSysuQskT7Z+6nUf3yab6Vw==" |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,30 +1,138 @@ | |||||||
| using System.Runtime.Versioning; | using ICSharpCode.SharpZipLib.Tar; | ||||||
| using Microsoft.Extensions.Logging; | using Microsoft.Extensions.Logging; | ||||||
|  | using Velopack.Compression; | ||||||
| 
 | 
 | ||||||
| namespace Velopack.Packaging.Unix; | namespace Velopack.Packaging.Unix; | ||||||
| 
 | 
 | ||||||
| public class AppImageTool | public class AppImageTool | ||||||
| { | { | ||||||
|     [SupportedOSPlatform("linux")] |  | ||||||
|     public static void CreateLinuxAppImage(string appDir, string outputFile, RuntimeCpu machine, ILogger logger) |     public static void CreateLinuxAppImage(string appDir, string outputFile, RuntimeCpu machine, ILogger logger) | ||||||
|     { |     { | ||||||
|         var tool = HelperFile.AppImageToolX64; |         string runtime = machine switch { | ||||||
| 
 |             RuntimeCpu.x86 => HelperFile.AppImageRuntimeX86, | ||||||
|         string arch = machine switch { |             RuntimeCpu.x64 => HelperFile.AppImageRuntimeX64, | ||||||
|             RuntimeCpu.x86 => "i386", |             RuntimeCpu.arm64 => HelperFile.AppImageRuntimeArm64, | ||||||
|             RuntimeCpu.x64 => "x86_64", |  | ||||||
|             RuntimeCpu.arm64 => "arm_aarch64", |  | ||||||
|             _ => throw new ArgumentOutOfRangeException(nameof(machine), machine, null) |             _ => throw new ArgumentOutOfRangeException(nameof(machine), machine, null) | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         var envVar = new Dictionary<string, string>() { |         string tmpSquashFile = outputFile + ".tmpfs"; | ||||||
|             { "ARCH", arch } |         string tmpTarFile = outputFile + ".tmptar"; | ||||||
|  | 
 | ||||||
|  |         try { | ||||||
|  |             if (VelopackRuntimeInfo.IsWindows) { | ||||||
|  |                 // to workaround a permissions limitation of gensquashfs.exe | ||||||
|  |                 // we need to create a tar archive of the AppDir, setting Linux permissions in the tar header | ||||||
|  |                 // and then use tar2sqfs.exe to create the squashfs filesystem | ||||||
|  |                 logger.Info("Compressing AppDir into tar and setting file permissions"); | ||||||
|  |                 using (var outStream = File.Create(tmpTarFile)) | ||||||
|  |                 using (var tarArchive = TarArchive.CreateOutputTarArchive(outStream)) { | ||||||
|  |                     tarArchive.RootPath = Path.GetFullPath(appDir); | ||||||
|  |                     void AddDirectoryToTar(TarArchive tarArchive, DirectoryInfo dir) | ||||||
|  |                     { | ||||||
|  |                         var directories = dir.GetDirectories(); | ||||||
|  |                         foreach (var directory in directories) { | ||||||
|  |                             AddDirectoryToTar(tarArchive, directory); | ||||||
|  |                         } | ||||||
|  | 
 | ||||||
|  |                         var filenames = dir.GetFiles(); | ||||||
|  |                         foreach (var filename in filenames) { | ||||||
|  |                             var tarEntry = TarEntry.CreateEntryFromFile(filename.FullName); | ||||||
|  |                             tarEntry.TarHeader.Magic = "ustar"; | ||||||
|  |                             tarEntry.TarHeader.Version = "00"; | ||||||
|  |                             tarEntry.TarHeader.ModTime = EasyZip.ZipFormatMinDate; | ||||||
|  |                             tarEntry.TarHeader.Mode = Convert.ToInt32("755", 8); | ||||||
|  |                             tarArchive.WriteEntry(tarEntry, true); | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                     AddDirectoryToTar(tarArchive, new DirectoryInfo(appDir)); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 logger.Info("Converting tar into squashfs filesystem"); | ||||||
|  |                 var tool = HelperFile.FindHelperFile("squashfs-tools\\tar2sqfs.exe"); | ||||||
|  |                 logger.Debug(Exe.RunHostedCommand($"\"{tool}\" \"{tmpSquashFile}\" < \"{tmpTarFile}\"")); | ||||||
|  |             } else { | ||||||
|  |                 Exe.AssertSystemBinaryExists("mksquashfs", "sudo apt install squashfs-tools", "brew install squashfs"); | ||||||
|  |                 var tool = "mksquashfs"; | ||||||
|  |                 List<string> args = | ||||||
|  |                 [ | ||||||
|  |                     appDir, | ||||||
|  |                     tmpSquashFile, | ||||||
|  |                     "-comp", | ||||||
|  |                     "xz", | ||||||
|  |                     "-root-owned", | ||||||
|  |                     "-noappend", | ||||||
|  |                     "-Xdict-size", | ||||||
|  |                     "100%", | ||||||
|  |                     "-b", | ||||||
|  |                     "16384", | ||||||
|  |                     "-mkfs-time", | ||||||
|  |                     "0", | ||||||
|  |                 ]; | ||||||
|  |                 logger.Info("Compressing AppDir into squashfs filesystem"); | ||||||
|  |                 logger.Debug(Exe.InvokeAndThrowIfNonZero(tool, args, null)); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             logger.Info($"Creating AppImage with {Path.GetFileName(runtime)} runtime"); | ||||||
|  |             File.Copy(runtime, outputFile, true); | ||||||
|  | 
 | ||||||
|  |             using var outputfs = File.Open(outputFile, FileMode.Append); | ||||||
|  |             using var squashfs = File.OpenRead(tmpSquashFile); | ||||||
|  |             squashfs.CopyTo(outputfs); | ||||||
|  | 
 | ||||||
|  |             Chmod.ChmodFileAsExecutable(outputFile); | ||||||
|  |         } finally { | ||||||
|  |             Utility.DeleteFileOrDirectoryHard(tmpSquashFile); | ||||||
|  |             Utility.DeleteFileOrDirectoryHard(tmpTarFile); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static void CreateLinuxAppImageOld(string appDir, string outputFile, RuntimeCpu machine, ILogger logger) | ||||||
|  |     { | ||||||
|  |         string runtime = machine switch { | ||||||
|  |             RuntimeCpu.x86 => HelperFile.AppImageRuntimeX86, | ||||||
|  |             RuntimeCpu.x64 => HelperFile.AppImageRuntimeX64, | ||||||
|  |             RuntimeCpu.arm64 => HelperFile.AppImageRuntimeArm64, | ||||||
|  |             _ => throw new ArgumentOutOfRangeException(nameof(machine), machine, null) | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         logger.Info("About to create .AppImage for architecture: " + arch); |         string tool = HelperFile.GetMkSquashFsPath(); | ||||||
|  |         List<string> args = new(); | ||||||
| 
 | 
 | ||||||
|         Chmod.ChmodFileAsExecutable(tool); |         string tmpPath = outputFile + ".tmpfs"; | ||||||
|         Exe.InvokeAndThrowIfNonZero(tool, new[] { appDir, outputFile }, null, envVar); |         if (VelopackRuntimeInfo.IsWindows) { | ||||||
|         Chmod.ChmodFileAsExecutable(outputFile); |             args.Add("--all-root"); | ||||||
|  |             args.Add("--pack-dir"); | ||||||
|  |             args.Add(appDir); | ||||||
|  |             args.Add(tmpPath); | ||||||
|  |         } else { | ||||||
|  |             args.Add(appDir); | ||||||
|  |             args.Add(tmpPath); | ||||||
|  |             args.Add("-comp"); | ||||||
|  |             args.Add("xz"); | ||||||
|  |             args.Add("-root-owned"); | ||||||
|  |             args.Add("-noappend"); | ||||||
|  |             args.Add("-Xdict-size"); | ||||||
|  |             args.Add("100%"); | ||||||
|  |             args.Add("-b"); | ||||||
|  |             args.Add("16384"); | ||||||
|  |             args.Add("-mkfs-time"); | ||||||
|  |             args.Add("0"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         try { | ||||||
|  |             logger.Info("Compressing AppDir into squashfs filesystem"); | ||||||
|  |             Exe.InvokeAndThrowIfNonZero(tool, args, null); | ||||||
|  | 
 | ||||||
|  |             logger.Info($"Creating AppImage with {Path.GetFileName(runtime)} runtime"); | ||||||
|  |             File.Copy(runtime, outputFile, true); | ||||||
|  | 
 | ||||||
|  |             using var outputfs = File.Open(outputFile, FileMode.Append); | ||||||
|  |             using var squashfs = File.OpenRead(tmpPath); | ||||||
|  |             squashfs.CopyTo(outputfs); | ||||||
|  | 
 | ||||||
|  |             Chmod.ChmodFileAsExecutable(outputFile); | ||||||
|  |         } finally { | ||||||
|  |             Utility.DeleteFileOrDirectoryHard(tmpPath); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										46
									
								
								src/Velopack.Packaging.Unix/BinDetect.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								src/Velopack.Packaging.Unix/BinDetect.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | namespace Velopack.Packaging; | ||||||
|  | 
 | ||||||
|  | public class BinDetect | ||||||
|  | { | ||||||
|  |     private enum MagicMachO : uint | ||||||
|  |     { | ||||||
|  |         MH_MAGIC = 0xfeedface, | ||||||
|  |         MH_CIGAM = 0xcefaedfe, | ||||||
|  |         MH_MAGIC_64 = 0xfeedfacf, | ||||||
|  |         MH_CIGAM_64 = 0xcffaedfe | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static bool IsMachOImage(string filePath) | ||||||
|  |     { | ||||||
|  |         using (BinaryReader reader = new BinaryReader(File.OpenRead(filePath))) { | ||||||
|  |             if (reader.BaseStream.Length < 256) // Header size | ||||||
|  |                 return false; | ||||||
|  | 
 | ||||||
|  |             uint magic = reader.ReadUInt32(); | ||||||
|  |             return Enum.IsDefined(typeof(MagicMachO), magic); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // First four bytes of valid ELF, as defined in https://github.com/torvalds/linux/blob/aae703b/include/uapi/linux/elf.h | ||||||
|  |     //    0x7f (DEL), 'E', 'L', 'F' | ||||||
|  |     private static ReadOnlySpan<byte> ElfMagic => "\u007f"u8 + "ELF"u8; | ||||||
|  | 
 | ||||||
|  |     public static bool IsElfImage(string filePath) | ||||||
|  |     { | ||||||
|  |         using FileStream fileStream = File.OpenRead(filePath); | ||||||
|  |         using BinaryReader reader = new(fileStream); | ||||||
|  | 
 | ||||||
|  |         if (reader.BaseStream.Length < 16) // EI_NIDENT = 16 | ||||||
|  |         { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         byte[] eIdent = reader.ReadBytes(4); | ||||||
|  | 
 | ||||||
|  |         return | ||||||
|  |             eIdent[0] == ElfMagic[0] && | ||||||
|  |             eIdent[1] == ElfMagic[1] && | ||||||
|  |             eIdent[2] == ElfMagic[2] && | ||||||
|  |             eIdent[3] == ElfMagic[3]; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -4,8 +4,6 @@ using System.Runtime.Versioning; | |||||||
| 
 | 
 | ||||||
| namespace Velopack.Packaging.Unix; | namespace Velopack.Packaging.Unix; | ||||||
| 
 | 
 | ||||||
| [SupportedOSPlatform("linux")] |  | ||||||
| [SupportedOSPlatform("macos")] |  | ||||||
| public class Chmod | public class Chmod | ||||||
| { | { | ||||||
|     private const string OSX_CSTD_LIB = "libSystem.dylib"; |     private const string OSX_CSTD_LIB = "libSystem.dylib"; | ||||||
|   | |||||||
| @@ -1,11 +1,9 @@ | |||||||
| using System.Runtime.Versioning; | using ELFSharp.ELF; | ||||||
| using ELFSharp.ELF; |  | ||||||
| using Microsoft.Extensions.Logging; | using Microsoft.Extensions.Logging; | ||||||
| using Velopack.Packaging.Abstractions; | using Velopack.Packaging.Abstractions; | ||||||
| 
 | 
 | ||||||
| namespace Velopack.Packaging.Unix.Commands; | namespace Velopack.Packaging.Unix.Commands; | ||||||
| 
 | 
 | ||||||
| [SupportedOSPlatform("linux")] |  | ||||||
| public class LinuxPackCommandRunner : PackageBuilder<LinuxPackOptions> | public class LinuxPackCommandRunner : PackageBuilder<LinuxPackOptions> | ||||||
| { | { | ||||||
|     protected string PortablePackagePath { get; set; } |     protected string PortablePackagePath { get; set; } | ||||||
| @@ -21,14 +19,14 @@ public class LinuxPackCommandRunner : PackageBuilder<LinuxPackOptions> | |||||||
|         var bin = dir.CreateSubdirectory("usr").CreateSubdirectory("bin"); |         var bin = dir.CreateSubdirectory("usr").CreateSubdirectory("bin"); | ||||||
| 
 | 
 | ||||||
|         if (Options.PackIsAppDir) { |         if (Options.PackIsAppDir) { | ||||||
|             Log.Info("Using provided .AppDir, will skip building new one."); |             Log.Info("Using provided AppDir, will skip building new one."); | ||||||
|             CopyFiles(new DirectoryInfo(Options.PackDirectory), dir, progress, true); |             CopyFiles(new DirectoryInfo(Options.PackDirectory), dir, progress, true); | ||||||
|         } else { |         } else { | ||||||
|             Log.Info("Building new .AppDir"); |             Log.Info("Building automatic AppDir from pack directory"); | ||||||
|             var appRunPath = Path.Combine(dir.FullName, "AppRun"); |             var appRunPath = Path.Combine(dir.FullName, "AppRun"); | ||||||
| 
 | 
 | ||||||
|             // app icon |             // app icon | ||||||
|             var icon = Options.Icon ?? HelperFile.GetDefaultAppIcon(); |             var icon = Options.Icon ?? HelperFile.GetDefaultAppIcon(RuntimeOs.Linux); | ||||||
|             var iconFilename = Options.PackId + Path.GetExtension(icon); |             var iconFilename = Options.PackId + Path.GetExtension(icon); | ||||||
|             File.Copy(icon, Path.Combine(dir.FullName, iconFilename), true); |             File.Copy(icon, Path.Combine(dir.FullName, iconFilename), true); | ||||||
| 
 | 
 | ||||||
| @@ -38,19 +36,17 @@ public class LinuxPackCommandRunner : PackageBuilder<LinuxPackOptions> | |||||||
| 
 | 
 | ||||||
|             File.WriteAllText(appRunPath, $$"""
 |             File.WriteAllText(appRunPath, $$"""
 | ||||||
| #!/bin/sh | #!/bin/sh | ||||||
| 
 |  | ||||||
| if [ ! -z "$APPIMAGE" ] && [ ! -z "$APPDIR" ]; then | if [ ! -z "$APPIMAGE" ] && [ ! -z "$APPDIR" ]; then | ||||||
|     MD5=$(echo -n "file://$APPIMAGE" | md5sum | cut -d' ' -f1) |     MD5=$(echo -n "file://$APPIMAGE" | md5sum | cut -d' ' -f1) | ||||||
|     cp "$APPDIR/{{iconFilename}}" "$HOME/.cache/thumbnails/normal/$MD5.png" |     cp "$APPDIR/{{iconFilename}}" "$HOME/.cache/thumbnails/normal/$MD5.png" >/dev/null 2>&1 | ||||||
|     cp "$APPDIR/{{iconFilename}}" "$HOME/.cache/thumbnails/large/$MD5.png" |     cp "$APPDIR/{{iconFilename}}" "$HOME/.cache/thumbnails/large/$MD5.png" >/dev/null 2>&1 | ||||||
|     xdg-icon-resource forceupdate |     xdg-icon-resource forceupdate >/dev/null 2>&1 | ||||||
| fi | fi | ||||||
| 
 |  | ||||||
| HERE="$(dirname "$(readlink -f "${0}")")" | HERE="$(dirname "$(readlink -f "${0}")")" | ||||||
| export PATH="${HERE}"/usr/bin/:"${PATH}" | export PATH="${HERE}"/usr/bin/:"${PATH}" | ||||||
| EXEC=$(grep -e '^Exec=.*' "${HERE}"/*.desktop | head -n 1 | cut -d "=" -f 2 | cut -d " " -f 1 | sed 's/\\s/ /g') | EXEC=$(grep -e '^Exec=.*' "${HERE}"/*.desktop | head -n 1 | cut -d "=" -f 2 | cut -d " " -f 1 | sed 's/\\s/ /g') | ||||||
| exec "${EXEC}" "$@" | exec "${EXEC}" "$@" | ||||||
| """);
 | """.Replace("\r", ""));
 | ||||||
|             Chmod.ChmodFileAsExecutable(appRunPath); |             Chmod.ChmodFileAsExecutable(appRunPath); | ||||||
| 
 | 
 | ||||||
|             var mainExeName = Options.EntryExecutableName ?? Options.PackId; |             var mainExeName = Options.EntryExecutableName ?? Options.PackId; | ||||||
| @@ -71,7 +67,7 @@ Icon={Options.PackId} | |||||||
| Exec={mainExeName} | Exec={mainExeName} | ||||||
| StartupWMClass={Options.PackId} | StartupWMClass={Options.PackId} | ||||||
| Categories={categories}; | Categories={categories}; | ||||||
| """);
 | """.Replace("\r", ""));
 | ||||||
| 
 | 
 | ||||||
|             // copy existing app files  |             // copy existing app files  | ||||||
|             CopyFiles(new DirectoryInfo(packDir), bin, progress, true); |             CopyFiles(new DirectoryInfo(packDir), bin, progress, true); | ||||||
| @@ -79,7 +75,7 @@ Categories={categories}; | |||||||
| 
 | 
 | ||||||
|         // velopack required files |         // velopack required files | ||||||
|         File.WriteAllText(Path.Combine(bin.FullName, "sq.version"), GenerateNuspecContent()); |         File.WriteAllText(Path.Combine(bin.FullName, "sq.version"), GenerateNuspecContent()); | ||||||
|         File.Copy(HelperFile.GetUpdatePath(), Path.Combine(bin.FullName, "UpdateNix"), true); |         File.Copy(HelperFile.GetUpdatePath(RuntimeOs.Linux), Path.Combine(bin.FullName, "UpdateNix"), true); | ||||||
|         progress(100); |         progress(100); | ||||||
|         return Task.FromResult(dir.FullName); |         return Task.FromResult(dir.FullName); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ public class OsxBundleCommandRunner : ICommand<OsxBundleOptions> | |||||||
| 
 | 
 | ||||||
|     public string Bundle(OsxBundleOptions options) |     public string Bundle(OsxBundleOptions options) | ||||||
|     { |     { | ||||||
|         var icon = options.Icon ?? HelperFile.GetDefaultAppIcon(); |         var icon = options.Icon ?? HelperFile.GetDefaultAppIcon(RuntimeOs.OSX); | ||||||
|         var packId = options.PackId; |         var packId = options.PackId; | ||||||
|         var packDirectory = options.PackDirectory; |         var packDirectory = options.PackDirectory; | ||||||
|         var packVersion = options.PackVersion; |         var packVersion = options.PackVersion; | ||||||
| @@ -41,7 +41,7 @@ public class OsxBundleCommandRunner : ICommand<OsxBundleOptions> | |||||||
|             throw new UserInfoException($"--exeName '{mainExePath}' does not exist."); |             throw new UserInfoException($"--exeName '{mainExePath}' does not exist."); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if (!MachO.IsMachOImage(mainExePath)) { |         if (!BinDetect.IsMachOImage(mainExePath)) { | ||||||
|             throw new UserInfoException($"--exeName '{mainExePath}' is not a mach-o executable."); |             throw new UserInfoException($"--exeName '{mainExePath}' is not a mach-o executable."); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|   | |||||||
| @@ -32,10 +32,10 @@ public class OsxPackCommandRunner : PackageBuilder<OsxPackOptions> | |||||||
|         var structure = new OsxStructureBuilder(dir.FullName); |         var structure = new OsxStructureBuilder(dir.FullName); | ||||||
|         var macosdir = structure.MacosDirectory; |         var macosdir = structure.MacosDirectory; | ||||||
|         File.WriteAllText(Path.Combine(macosdir, "sq.version"), GenerateNuspecContent()); |         File.WriteAllText(Path.Combine(macosdir, "sq.version"), GenerateNuspecContent()); | ||||||
|         File.Copy(HelperFile.GetUpdatePath(), Path.Combine(macosdir, "UpdateMac"), true); |         File.Copy(HelperFile.GetUpdatePath(RuntimeOs.OSX), Path.Combine(macosdir, "UpdateMac"), true); | ||||||
| 
 | 
 | ||||||
|         foreach (var f in Directory.GetFiles(macosdir, "*", SearchOption.AllDirectories)) { |         foreach (var f in Directory.GetFiles(macosdir, "*", SearchOption.AllDirectories)) { | ||||||
|             if (MachO.IsMachOImage(f)) { |             if (BinDetect.IsMachOImage(f)) { | ||||||
|                 Log.Debug(f + " is a mach-o binary, chmod as executable."); |                 Log.Debug(f + " is a mach-o binary, chmod as executable."); | ||||||
|                 Chmod.ChmodFileAsExecutable(f); |                 Chmod.ChmodFileAsExecutable(f); | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -8,6 +8,7 @@ | |||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageReference Include="ELFSharp" Version="2.17.3" /> |     <PackageReference Include="ELFSharp" Version="2.17.3" /> | ||||||
|  |     <PackageReference Include="SharpZipLib" Version="1.4.2" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|   | |||||||
| @@ -21,8 +21,18 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|  |       }, | ||||||
|  |       "SharpZipLib": { | ||||||
|  |         "type": "Direct", | ||||||
|  |         "requested": "[1.4.2, )", | ||||||
|  |         "resolved": "1.4.2", | ||||||
|  |         "contentHash": "yjj+3zgz8zgXpiiC3ZdF/iyTBbz2fFvMxZFEBPUcwZjIvXOf37Ylm+K58hqMfIBt5JgU/Z2uoUS67JmTLe973A==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Memory": "4.5.4", | ||||||
|  |           "System.Threading.Tasks.Extensions": "4.5.2" | ||||||
|  |         } | ||||||
|       }, |       }, | ||||||
|       "Markdig": { |       "Markdig": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -52,8 +62,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client": { |       "Microsoft.Identity.Client": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "3TDPEie+9t/NgBhoNifLFwM6nkypYUa8GUHfMnkYtovDDTQM8bsS2KEIP5tAh28BgfJZgof/KlCEKvGxz0H3Eg==", |         "contentHash": "zXPDXjTju1yOZQzi5PeNmQfBzlxv+VlCfgfZ70VqyFuhaPtFXwvbLSlwCOMoVALR/MzOR6xkiadoyVDmWDmqKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.IdentityModel.Abstractions": "6.35.0", |           "Microsoft.IdentityModel.Abstractions": "6.35.0", | ||||||
|           "System.Diagnostics.DiagnosticSource": "6.0.1" |           "System.Diagnostics.DiagnosticSource": "6.0.1" | ||||||
| @@ -61,27 +71,27 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Broker": { |       "Microsoft.Identity.Client.Broker": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "D5M7plyrXrQwhj/PILHLS7G3TqDg8rUO9lR+5awUSXueGxsdA2qHpeKiSR0sVfX6wxziWszOLDUR2vjFFKHPSg==", |         "contentHash": "+13pImNYfPGl04qZh3Rtq8wPfo55FScAUEio4ymPzeWJ/PE04jCsKH6ZIlrYbPCqe6t9u0lru1eZ+ho9Dsnvpw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "Microsoft.Identity.Client.NativeInterop": "0.16.0" |           "Microsoft.Identity.Client.NativeInterop": "0.16.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Extensions.Msal": { |       "Microsoft.Identity.Client.Extensions.Msal": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "tCX3c1bSkwH89AKjtB/oAOLlv661tOsEA8ULOCavOc6te/CKlqmMmFs8zMI58Qlv3XkbLOrswYr50IhDntiiPA==", |         "contentHash": "GCSBhzkr7/7DZDk2kYS9dfihB5ieM2CqwGAhtjuE6T1GtAqzuh62UJhMgO0dSx/PLgIYMzNHnh+pfHK8aWJQJQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "System.IO.FileSystem.AccessControl": "5.0.0", |           "System.IO.FileSystem.AccessControl": "5.0.0", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.5.0" |           "System.Security.Cryptography.ProtectedData": "4.5.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.NativeInterop": { |       "Microsoft.Identity.Client.NativeInterop": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "0.16.0", |         "resolved": "0.16.1", | ||||||
|         "contentHash": "riaIH9tT4kRG4xKFxx+wpuo+VAZVOr1ElbNubcvpFXb82gjEmZtJRzpdiS4tdRYTVh6CUKNz9AINAptActTtBQ==" |         "contentHash": "OvY/+/AHESi24f5zOCf9kL4HXwPxXVQ3A+tMQsJvFk2DmP+sc88FYWL49zlku5q0bvx5yFvBLNpHQeRT9a6A5g==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.IdentityModel.Abstractions": { |       "Microsoft.IdentityModel.Abstractions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -100,8 +110,8 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "System.Buffers": { |       "System.Buffers": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -233,16 +243,16 @@ | |||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Extensions.Logging.Abstractions": "[2.2.0, )", |           "Microsoft.Extensions.Logging.Abstractions": "[2.2.0, )", | ||||||
|           "Newtonsoft.Json": "[13.0.1, )", |           "Newtonsoft.Json": "[13.0.1, )", | ||||||
|           "NuGet.Versioning": "[6.9.1, )" |           "NuGet.Versioning": "[6.10.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.packaging": { |       "velopack.packaging": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Markdig": "[0.37.0, )", |           "Markdig": "[0.37.0, )", | ||||||
|           "Microsoft.Identity.Client": "[4.61.0, )", |           "Microsoft.Identity.Client": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Broker": "[4.61.0, )", |           "Microsoft.Identity.Client.Broker": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.0, )", |           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.2, )", | ||||||
|           "System.Linq.Async": "[6.0.1, )", |           "System.Linq.Async": "[6.0.1, )", | ||||||
|           "System.Net.Http": "[4.3.4, )", |           "System.Net.Http": "[4.3.4, )", | ||||||
|           "Velopack": "[1.0.0, )" |           "Velopack": "[1.0.0, )" | ||||||
| @@ -269,8 +279,14 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|  |       }, | ||||||
|  |       "SharpZipLib": { | ||||||
|  |         "type": "Direct", | ||||||
|  |         "requested": "[1.4.2, )", | ||||||
|  |         "resolved": "1.4.2", | ||||||
|  |         "contentHash": "yjj+3zgz8zgXpiiC3ZdF/iyTBbz2fFvMxZFEBPUcwZjIvXOf37Ylm+K58hqMfIBt5JgU/Z2uoUS67JmTLe973A==" | ||||||
|       }, |       }, | ||||||
|       "Markdig": { |       "Markdig": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -294,8 +310,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client": { |       "Microsoft.Identity.Client": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "3TDPEie+9t/NgBhoNifLFwM6nkypYUa8GUHfMnkYtovDDTQM8bsS2KEIP5tAh28BgfJZgof/KlCEKvGxz0H3Eg==", |         "contentHash": "zXPDXjTju1yOZQzi5PeNmQfBzlxv+VlCfgfZ70VqyFuhaPtFXwvbLSlwCOMoVALR/MzOR6xkiadoyVDmWDmqKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.IdentityModel.Abstractions": "6.35.0", |           "Microsoft.IdentityModel.Abstractions": "6.35.0", | ||||||
|           "System.Diagnostics.DiagnosticSource": "6.0.1" |           "System.Diagnostics.DiagnosticSource": "6.0.1" | ||||||
| @@ -303,26 +319,26 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Broker": { |       "Microsoft.Identity.Client.Broker": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "D5M7plyrXrQwhj/PILHLS7G3TqDg8rUO9lR+5awUSXueGxsdA2qHpeKiSR0sVfX6wxziWszOLDUR2vjFFKHPSg==", |         "contentHash": "+13pImNYfPGl04qZh3Rtq8wPfo55FScAUEio4ymPzeWJ/PE04jCsKH6ZIlrYbPCqe6t9u0lru1eZ+ho9Dsnvpw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "Microsoft.Identity.Client.NativeInterop": "0.16.0" |           "Microsoft.Identity.Client.NativeInterop": "0.16.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Extensions.Msal": { |       "Microsoft.Identity.Client.Extensions.Msal": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "tCX3c1bSkwH89AKjtB/oAOLlv661tOsEA8ULOCavOc6te/CKlqmMmFs8zMI58Qlv3XkbLOrswYr50IhDntiiPA==", |         "contentHash": "GCSBhzkr7/7DZDk2kYS9dfihB5ieM2CqwGAhtjuE6T1GtAqzuh62UJhMgO0dSx/PLgIYMzNHnh+pfHK8aWJQJQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.5.0" |           "System.Security.Cryptography.ProtectedData": "4.5.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.NativeInterop": { |       "Microsoft.Identity.Client.NativeInterop": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "0.16.0", |         "resolved": "0.16.1", | ||||||
|         "contentHash": "riaIH9tT4kRG4xKFxx+wpuo+VAZVOr1ElbNubcvpFXb82gjEmZtJRzpdiS4tdRYTVh6CUKNz9AINAptActTtBQ==" |         "contentHash": "OvY/+/AHESi24f5zOCf9kL4HXwPxXVQ3A+tMQsJvFk2DmP+sc88FYWL49zlku5q0bvx5yFvBLNpHQeRT9a6A5g==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.IdentityModel.Abstractions": { |       "Microsoft.IdentityModel.Abstractions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -346,8 +362,8 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "runtime.debian.8-x64.runtime.native.System.Security.Cryptography.OpenSsl": { |       "runtime.debian.8-x64.runtime.native.System.Security.Cryptography.OpenSsl": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -909,16 +925,16 @@ | |||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", |           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", | ||||||
|           "NuGet.Versioning": "[6.9.1, )" |           "NuGet.Versioning": "[6.10.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.packaging": { |       "velopack.packaging": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Markdig": "[0.37.0, )", |           "Markdig": "[0.37.0, )", | ||||||
|           "Microsoft.Identity.Client": "[4.61.0, )", |           "Microsoft.Identity.Client": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Broker": "[4.61.0, )", |           "Microsoft.Identity.Client.Broker": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.0, )", |           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.2, )", | ||||||
|           "System.Linq.Async": "[6.0.1, )", |           "System.Linq.Async": "[6.0.1, )", | ||||||
|           "System.Net.Http": "[4.3.4, )", |           "System.Net.Http": "[4.3.4, )", | ||||||
|           "Velopack": "[1.0.0, )" |           "Velopack": "[1.0.0, )" | ||||||
|   | |||||||
| @@ -1,12 +1,10 @@ | |||||||
| using System.Diagnostics; | using System.Diagnostics; | ||||||
| using System.Runtime.Versioning; |  | ||||||
| using System.Text.RegularExpressions; | using System.Text.RegularExpressions; | ||||||
| using Microsoft.Extensions.Logging; | using Microsoft.Extensions.Logging; | ||||||
| using Velopack.Packaging.Exceptions; | using Velopack.Packaging.Exceptions; | ||||||
| 
 | 
 | ||||||
| namespace Velopack.Packaging.Windows; | namespace Velopack.Packaging.Windows; | ||||||
| 
 | 
 | ||||||
| [SupportedOSPlatform("windows")] |  | ||||||
| public class CodeSign | public class CodeSign | ||||||
| { | { | ||||||
|     public ILogger Log { get; } |     public ILogger Log { get; } | ||||||
| @@ -88,7 +86,11 @@ public class CodeSign | |||||||
|             if (signAsTemplate) { |             if (signAsTemplate) { | ||||||
|                 command = signArguments.Replace("{{file}}", filesToSignStr); |                 command = signArguments.Replace("{{file}}", filesToSignStr); | ||||||
|             } else { |             } else { | ||||||
|                 command = $"\"{HelperFile.SignToolPath}\" sign {signArguments} {filesToSignStr}"; |                 if (VelopackRuntimeInfo.IsWindows) { | ||||||
|  |                     command = $"\"{HelperFile.SignToolPath}\" sign {signArguments} {filesToSignStr}"; | ||||||
|  |                 } else { | ||||||
|  |                     throw new PlatformNotSupportedException("signtool.exe does not work on non-Windows platforms."); | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             RunSigningCommand(command, rootDir, signLogFile); |             RunSigningCommand(command, rootDir, signLogFile); | ||||||
| @@ -107,22 +109,29 @@ public class CodeSign | |||||||
|         // about how the dotnet tool host works prevents signtool from being able to open a token password |         // about how the dotnet tool host works prevents signtool from being able to open a token password | ||||||
|         // prompt, meaning signing fails for those with an HSM. |         // prompt, meaning signing fails for those with an HSM. | ||||||
| 
 | 
 | ||||||
|         string args = $"/S /C \"{command} >> \"{signLogFile}\" 2>&1\""; |         var fileName = "cmd.exe"; | ||||||
|  |         var args = $"/S /C \"{command} >> \"{signLogFile}\" 2>&1\""; | ||||||
|  | 
 | ||||||
|  |         if (!VelopackRuntimeInfo.IsWindows) { | ||||||
|  |             fileName = "/bin/bash"; | ||||||
|  |             string escapedCommand = command.Replace("'", "'\\''"); | ||||||
|  |             args = $"-c '{escapedCommand} >> \"{signLogFile}\" 2>&1'"; | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         var psi = new ProcessStartInfo { |         var psi = new ProcessStartInfo { | ||||||
|             FileName = "cmd.exe", |             FileName = fileName, | ||||||
|             Arguments = args, |             Arguments = args, | ||||||
|             UseShellExecute = false, |             UseShellExecute = false, | ||||||
|             WorkingDirectory = workDir, |             WorkingDirectory = workDir, | ||||||
|             CreateNoWindow = true, |             CreateNoWindow = true, | ||||||
|         }; |         }; | ||||||
| 
 | 
 | ||||||
|         var process = Process.Start(psi); |         using var process = Process.Start(psi); | ||||||
|         process.WaitForExit(); |         process.WaitForExit(); | ||||||
| 
 | 
 | ||||||
|         if (process.ExitCode != 0) { |         if (process.ExitCode != 0) { | ||||||
|             var cmdWithPasswordHidden = "cmd.exe " + new Regex(@"\/p\s+?[^\s]+").Replace(command, "/p ********"); |             var cmdWithPasswordHidden = fileName + " " + new Regex(@"\/p\s+?[^\s]+").Replace(args, "/p ********"); | ||||||
|             Log.Debug($"Signing command failed: {cmdWithPasswordHidden}"); |             Log.Debug($"Signing command failed - {Environment.NewLine}    {cmdWithPasswordHidden}"); | ||||||
|             var output = File.Exists(signLogFile) ? File.ReadAllText(signLogFile).Trim() : "No output file was created."; |             var output = File.Exists(signLogFile) ? File.ReadAllText(signLogFile).Trim() : "No output file was created."; | ||||||
|             throw new UserInfoException( |             throw new UserInfoException( | ||||||
|                 $"Signing command failed. Specify --verbose argument to print signing command." + Environment.NewLine + |                 $"Signing command failed. Specify --verbose argument to print signing command." + Environment.NewLine + | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| using System.Runtime.Versioning; | using Microsoft.Extensions.Logging; | ||||||
| using Microsoft.Extensions.Logging; |  | ||||||
| using Velopack.Compression; | using Velopack.Compression; | ||||||
| using Velopack.NuGet; | using Velopack.NuGet; | ||||||
| using Velopack.Packaging.Abstractions; | using Velopack.Packaging.Abstractions; | ||||||
| @@ -8,7 +7,6 @@ using Velopack.Windows; | |||||||
| 
 | 
 | ||||||
| namespace Velopack.Packaging.Windows.Commands; | namespace Velopack.Packaging.Windows.Commands; | ||||||
| 
 | 
 | ||||||
| [SupportedOSPlatform("windows")] |  | ||||||
| public class WindowsPackCommandRunner : PackageBuilder<WindowsPackOptions> | public class WindowsPackCommandRunner : PackageBuilder<WindowsPackOptions> | ||||||
| { | { | ||||||
|     public WindowsPackCommandRunner(ILogger logger, IFancyConsole console) |     public WindowsPackCommandRunner(ILogger logger, IFancyConsole console) | ||||||
| @@ -48,7 +46,7 @@ public class WindowsPackCommandRunner : PackageBuilder<WindowsPackOptions> | |||||||
|         packDir = dir.FullName; |         packDir = dir.FullName; | ||||||
| 
 | 
 | ||||||
|         var updatePath = Path.Combine(TempDir.FullName, "Update.exe"); |         var updatePath = Path.Combine(TempDir.FullName, "Update.exe"); | ||||||
|         File.Copy(HelperFile.GetUpdatePath(), updatePath, true); |         File.Copy(HelperFile.GetUpdatePath(RuntimeOs.Windows), updatePath, true); | ||||||
| 
 | 
 | ||||||
|         // check for and delete clickonce manifest |         // check for and delete clickonce manifest | ||||||
|         var clickonceManifests = Directory.EnumerateFiles(packDir, "*.application") |         var clickonceManifests = Directory.EnumerateFiles(packDir, "*.application") | ||||||
| @@ -65,10 +63,10 @@ public class WindowsPackCommandRunner : PackageBuilder<WindowsPackOptions> | |||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // update icon for Update.exe if requested |         // update icon for Update.exe if requested | ||||||
|         if (Options.Icon != null && VelopackRuntimeInfo.IsWindows) { |         if (Options.Icon != null) { | ||||||
|             Rcedit.SetExeIcon(updatePath, Options.Icon); |             var editor = new ResourceEdit(updatePath, Log); | ||||||
|         } else if (Options.Icon != null) { |             editor.SetExeIcon(Options.Icon); | ||||||
|             Log.Warn("Unable to set icon for Update.exe (only supported on windows)."); |             editor.Commit(); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         File.Copy(updatePath, Path.Combine(packDir, "Squirrel.exe"), true); |         File.Copy(updatePath, Path.Combine(packDir, "Squirrel.exe"), true); | ||||||
| @@ -176,11 +174,14 @@ public class WindowsPackCommandRunner : PackageBuilder<WindowsPackOptions> | |||||||
|         var bundledZip = new ZipPackage(releasePkg); |         var bundledZip = new ZipPackage(releasePkg); | ||||||
|         Utility.Retry(() => File.Copy(HelperFile.SetupPath, targetSetupExe, true)); |         Utility.Retry(() => File.Copy(HelperFile.SetupPath, targetSetupExe, true)); | ||||||
|         progress(10); |         progress(10); | ||||||
|         if (VelopackRuntimeInfo.IsWindows) { | 
 | ||||||
|             Rcedit.SetPEVersionBlockFromPackageInfo(targetSetupExe, bundledZip, Options.Icon); |         var editor = new ResourceEdit(targetSetupExe, Log); | ||||||
|         } else { |         editor.SetVersionInfo(bundledZip); | ||||||
|             Log.Warn("Unable to set PE Version on Setup.exe (only supported on windows)"); |         if (Options.Icon != null) { | ||||||
|  |             editor.SetExeIcon(Options.Icon); | ||||||
|         } |         } | ||||||
|  |         editor.Commit(); | ||||||
|  | 
 | ||||||
|         progress(25); |         progress(25); | ||||||
|         Log.Debug($"Creating Setup bundle"); |         Log.Debug($"Creating Setup bundle"); | ||||||
|         SetupBundle.CreatePackageBundle(targetSetupExe, releasePkg); |         SetupBundle.CreatePackageBundle(targetSetupExe, releasePkg); | ||||||
| @@ -231,15 +232,9 @@ public class WindowsPackCommandRunner : PackageBuilder<WindowsPackOptions> | |||||||
| 
 | 
 | ||||||
|         try { |         try { | ||||||
|             Utility.Retry(() => File.Copy(HelperFile.StubExecutablePath, targetStubPath, true)); |             Utility.Retry(() => File.Copy(HelperFile.StubExecutablePath, targetStubPath, true)); | ||||||
|             Utility.Retry(() => { |             var edit = new ResourceEdit(targetStubPath, Log); | ||||||
|                 if (VelopackRuntimeInfo.IsWindows) { |             edit.CopyResourcesFrom(exeToCopy); | ||||||
|                     using var writer = new Microsoft.NET.HostModel.ResourceUpdater(targetStubPath, true); |             edit.Commit(); | ||||||
|                     writer.AddResourcesFromPEImage(exeToCopy); |  | ||||||
|                     writer.Update(); |  | ||||||
|                 } else { |  | ||||||
|                     Log.Warn($"Cannot set resources/icon for {targetStubPath} (only supported on windows)."); |  | ||||||
|                 } |  | ||||||
|             }); |  | ||||||
|         } catch (Exception ex) { |         } catch (Exception ex) { | ||||||
|             Log.Error(ex, $"Error creating StubExecutable and copying resources for '{exeToCopy}'. This stub may or may not work properly."); |             Log.Error(ex, $"Error creating StubExecutable and copying resources for '{exeToCopy}'. This stub may or may not work properly."); | ||||||
|         } |         } | ||||||
|   | |||||||
							
								
								
									
										111
									
								
								src/Velopack.Packaging.Windows/IcoExtract.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										111
									
								
								src/Velopack.Packaging.Windows/IcoExtract.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,111 @@ | |||||||
|  | using System.Diagnostics.CodeAnalysis; | ||||||
|  | using Ico.Codecs; | ||||||
|  | using Ico.Host; | ||||||
|  | using Ico.Model; | ||||||
|  | using Ico.Validation; | ||||||
|  | using Microsoft.Extensions.Logging; | ||||||
|  | 
 | ||||||
|  | namespace Velopack.Packaging.Windows; | ||||||
|  | 
 | ||||||
|  | public class IcoExtract | ||||||
|  | { | ||||||
|  |     private readonly ILogger _logger; | ||||||
|  |     private readonly List<IcoFrame> _frames = new(); | ||||||
|  | 
 | ||||||
|  |     public IcoExtract(ILogger logger) | ||||||
|  |     { | ||||||
|  |         _logger = logger; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public List<IcoFrame> ExtractFrames(FileInfo file) | ||||||
|  |     { | ||||||
|  |         var reporter = new IcoILoggerReporter(_logger); | ||||||
|  |         var context = new ParseContext { | ||||||
|  |             DisplayedPath = file.Name, | ||||||
|  |             FullPath = file.FullName, | ||||||
|  |             GeneratedFrames = new List<IcoFrame>(), | ||||||
|  |             Reporter = reporter, | ||||||
|  |             PngEncoder = new SixLabors.ImageSharp.Formats.Png.PngEncoder { | ||||||
|  |                 CompressionLevel = SixLabors.ImageSharp.Formats.Png.PngCompressionLevel.Level9, | ||||||
|  |                 ColorType = SixLabors.ImageSharp.Formats.Png.PngColorType.RgbWithAlpha, | ||||||
|  |             }, | ||||||
|  |             AllowPaletteTruncation = StrictnessPolicy.Loose, | ||||||
|  |             MaskedImagePixelEmitOptions = StrictnessPolicy.Loose, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         _frames.Clear(); | ||||||
|  |         ExceptionWrapper.Try(() => DoExtractFile(context), context, reporter); | ||||||
|  |         return _frames; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void DoExtractFile(ParseContext context) | ||||||
|  |     { | ||||||
|  |         var length = new FileInfo(context.FullPath).Length; | ||||||
|  |         var data = File.ReadAllBytes(context.FullPath); | ||||||
|  |         IcoDecoder.DoFile(data, context, _frames.Add); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #region IErrorReporter | ||||||
|  | 
 | ||||||
|  |     [ExcludeFromCodeCoverage] | ||||||
|  |     private class IcoILoggerReporter : IErrorReporter | ||||||
|  |     { | ||||||
|  |         private readonly ILogger _logger; | ||||||
|  | 
 | ||||||
|  |         public IcoILoggerReporter(ILogger logger) | ||||||
|  |         { | ||||||
|  |             _logger = logger; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         void IErrorReporter.ErrorLine(IcoErrorCode errorCode, string message) | ||||||
|  |         { | ||||||
|  |             _logger.LogDebug($"Error{GenerateCode(errorCode)}: {message}"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         void IErrorReporter.ErrorLine(IcoErrorCode errorCode, string message, string fileName) | ||||||
|  |         { | ||||||
|  |             _logger.LogDebug($"{fileName}: Error{GenerateCode(errorCode)}: {message}"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         void IErrorReporter.ErrorLine(IcoErrorCode errorCode, string message, string fileName, uint frameNumber) | ||||||
|  |         { | ||||||
|  |             _logger.LogDebug($"{fileName}({frameNumber + 1}): Error{GenerateCode(errorCode)}: {message}"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         void IErrorReporter.InfoLine(string message) | ||||||
|  |         { | ||||||
|  |             _logger.LogDebug(message); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         void IErrorReporter.VerboseLine(string message) | ||||||
|  |         { | ||||||
|  |             _logger.LogDebug(message); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         void IErrorReporter.WarnLine(IcoErrorCode errorCode, string message) | ||||||
|  |         { | ||||||
|  |             _logger.LogDebug($"Warning{GenerateCode(errorCode)}: {message}"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         void IErrorReporter.WarnLine(IcoErrorCode errorCode, string message, string fileName) | ||||||
|  |         { | ||||||
|  |             _logger.LogDebug($"{fileName}: Warning{GenerateCode(errorCode)}: {message}"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         void IErrorReporter.WarnLine(IcoErrorCode errorCode, string message, string fileName, uint frameNumber) | ||||||
|  |         { | ||||||
|  |             _logger.LogDebug($"{fileName}({frameNumber + 1}): Warning{GenerateCode(errorCode)}: {message}"); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         private string GenerateCode(IcoErrorCode code) | ||||||
|  |         { | ||||||
|  |             if (code == IcoErrorCode.NoError) { | ||||||
|  |                 return ""; | ||||||
|  |             } else { | ||||||
|  |                 return $" ICO{(uint) code}"; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #endregion | ||||||
|  | } | ||||||
| @@ -1,37 +0,0 @@ | |||||||
| using System.Runtime.Versioning; |  | ||||||
| using Velopack.NuGet; |  | ||||||
| 
 |  | ||||||
| namespace Velopack.Packaging.Windows; |  | ||||||
| 
 |  | ||||||
| [SupportedOSPlatform("windows")] |  | ||||||
| public class Rcedit |  | ||||||
| { |  | ||||||
|     public static void SetExeIcon(string exePath, string iconPath) |  | ||||||
|     { |  | ||||||
|         var args = new[] { Path.GetFullPath(exePath), "--set-icon", iconPath }; |  | ||||||
|         Utility.Retry(() => Exe.InvokeAndThrowIfNonZero(HelperFile.RceditPath, args, null)); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     [SupportedOSPlatform("windows")] |  | ||||||
|     public static void SetPEVersionBlockFromPackageInfo(string exePath, PackageManifest package, string iconPath = null) |  | ||||||
|     { |  | ||||||
|         var realExePath = Path.GetFullPath(exePath); |  | ||||||
| 
 |  | ||||||
|         List<string> args = new List<string>() { |  | ||||||
|             realExePath, |  | ||||||
|             "--set-version-string", "CompanyName", package.ProductCompany, |  | ||||||
|             "--set-version-string", "LegalCopyright", package.ProductCopyright, |  | ||||||
|             "--set-version-string", "FileDescription", package.ProductDescription, |  | ||||||
|             "--set-version-string", "ProductName", package.ProductName, |  | ||||||
|             "--set-file-version", package.Version.ToString(), |  | ||||||
|             "--set-product-version", package.Version.ToString(), |  | ||||||
|         }; |  | ||||||
| 
 |  | ||||||
|         if (iconPath != null) { |  | ||||||
|             args.Add("--set-icon"); |  | ||||||
|             args.Add(Path.GetFullPath(iconPath)); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         Utility.Retry(() => Exe.InvokeAndThrowIfNonZero(HelperFile.RceditPath, args, null)); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										194
									
								
								src/Velopack.Packaging.Windows/ResourceEdit.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										194
									
								
								src/Velopack.Packaging.Windows/ResourceEdit.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,194 @@ | |||||||
|  | using AsmResolver.PE; | ||||||
|  | using AsmResolver.PE.File; | ||||||
|  | using AsmResolver.PE.File.Headers; | ||||||
|  | using AsmResolver.PE.Win32Resources; | ||||||
|  | using AsmResolver.PE.Win32Resources.Builder; | ||||||
|  | using AsmResolver.PE.Win32Resources.Icon; | ||||||
|  | using AsmResolver.PE.Win32Resources.Version; | ||||||
|  | using Microsoft.Extensions.Logging; | ||||||
|  | using Velopack.NuGet; | ||||||
|  | 
 | ||||||
|  | namespace Velopack.Packaging.Windows; | ||||||
|  | 
 | ||||||
|  | public class ResourceEdit | ||||||
|  | { | ||||||
|  |     const ushort kLangNeutral = 0; | ||||||
|  |     const ushort kCodePageUtf16 = 1200; | ||||||
|  | 
 | ||||||
|  |     private readonly string _exePath; | ||||||
|  |     private readonly ILogger _logger; | ||||||
|  |     private readonly PEFile _file; | ||||||
|  |     private readonly ushort _langId = kLangNeutral; | ||||||
|  | 
 | ||||||
|  |     IResourceDirectory _resources; | ||||||
|  |     private bool _disposed; | ||||||
|  | 
 | ||||||
|  |     public ResourceEdit(string exeFile, ILogger logger) | ||||||
|  |     { | ||||||
|  |         _exePath = exeFile; | ||||||
|  |         _logger = logger; | ||||||
|  |         _file = PEFile.FromBytes(File.ReadAllBytes(exeFile)); | ||||||
|  |         var image = PEImage.FromFile(_file); | ||||||
|  |         _resources = image.Resources ?? new ResourceDirectory((uint) 0); | ||||||
|  | 
 | ||||||
|  |         // if there is already a manifest, we want to keep the existing language ID | ||||||
|  |         try { | ||||||
|  |             var existingInfo = VersionInfoResource.FromDirectory(_resources); | ||||||
|  |             if (existingInfo != null) { | ||||||
|  |                 _langId = (ushort) existingInfo.Lcid; | ||||||
|  |             } | ||||||
|  |         } catch { } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public void SetExeIcon(string iconPath) | ||||||
|  |     { | ||||||
|  |         ThrowIfDisposed(); | ||||||
|  | 
 | ||||||
|  |         // should use this commented code once these issues are fixed in AsmResolver  | ||||||
|  |         // https://github.com/Washi1337/AsmResolver/issues/532 | ||||||
|  |         // https://github.com/Washi1337/AsmResolver/issues/533 | ||||||
|  |         // var iconResource = new IconResource(); | ||||||
|  |         // var group = new IconGroupDirectory(); | ||||||
|  |         // iconResource.AddEntry(1, group); | ||||||
|  |         // iconResource.WriteToDirectory(_resources); | ||||||
|  | 
 | ||||||
|  |         var group = new IconGroupDirectory() { | ||||||
|  |             Type = 1, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         var extractor = new IcoExtract(_logger); | ||||||
|  |         var frames = extractor.ExtractFrames(new FileInfo(iconPath)); | ||||||
|  | 
 | ||||||
|  |         for (var p = 0; p < frames.Count; p++) { | ||||||
|  |             var f = frames[p]; | ||||||
|  | 
 | ||||||
|  |             var dictEntry = new IconGroupDirectoryEntry() { | ||||||
|  |                 BytesInRes = (uint) f.RawData.Length, | ||||||
|  |                 Height = (byte) f.CookedData.Height, | ||||||
|  |                 Width = (byte) f.CookedData.Width, | ||||||
|  |                 Id = (ushort) (p + 1), | ||||||
|  |                 Reserved = 0, | ||||||
|  |                 PixelBitCount = (ushort) f.CookedData.PixelType.BitsPerPixel, | ||||||
|  |                 ColorCount = (byte) f.Encoding.PaletteSize, | ||||||
|  |                 ColorPlanes = 1, | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             var iconEntry = new IconEntry() { | ||||||
|  |                 RawIcon = f.RawData, | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             group.AddEntry(dictEntry, iconEntry); | ||||||
|  |             group.Count++; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         WriteToDirectory(_resources, new Dictionary<uint, IconGroupDirectory> { { 1, group } }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void WriteToDirectory(IResourceDirectory rootDirectory, Dictionary<uint, IconGroupDirectory> _entries) | ||||||
|  |     { | ||||||
|  |         ThrowIfDisposed(); | ||||||
|  | 
 | ||||||
|  |         // this function can be removed once these issues are fixed in AsmResolver  | ||||||
|  |         // https://github.com/Washi1337/AsmResolver/issues/532 | ||||||
|  |         // https://github.com/Washi1337/AsmResolver/issues/533 | ||||||
|  | 
 | ||||||
|  |         var newIconDirectory = new ResourceDirectory(ResourceType.Icon); | ||||||
|  |         foreach (var entry in _entries) { | ||||||
|  |             foreach (var (groupEntry, iconEntry) in entry.Value.GetIconEntries()) { | ||||||
|  |                 newIconDirectory.Entries.Add(new ResourceDirectory(groupEntry.Id) { Entries = { new ResourceData(_langId, iconEntry) } }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var newGroupIconDirectory = new ResourceDirectory(ResourceType.GroupIcon); | ||||||
|  |         foreach (var entry in _entries) { | ||||||
|  |             newGroupIconDirectory.Entries.Add(new ResourceDirectory(entry.Key) { Entries = { new ResourceData(_langId, entry.Value) } }); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         rootDirectory.AddOrReplaceEntry(newIconDirectory); | ||||||
|  |         rootDirectory.AddOrReplaceEntry(newGroupIconDirectory); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public void SetVersionInfo(PackageManifest package) | ||||||
|  |     { | ||||||
|  |         ThrowIfDisposed(); | ||||||
|  | 
 | ||||||
|  |         // We just replace the entire VersionInfo section, so we know that the | ||||||
|  |         // VarFileInfo languages will be correct. | ||||||
|  |         var fileVersion = new Version(package.Version.Major, package.Version.Minor, package.Version.Patch, 0); | ||||||
|  | 
 | ||||||
|  |         var versionInfo = new VersionInfoResource(_langId); | ||||||
|  |         versionInfo.FixedVersionInfo.FileOS = FileOS.NT; | ||||||
|  |         versionInfo.FixedVersionInfo.FileType = FileType.App; | ||||||
|  |         versionInfo.FixedVersionInfo.FileVersion = fileVersion; | ||||||
|  |         versionInfo.FixedVersionInfo.ProductVersion = fileVersion; | ||||||
|  | 
 | ||||||
|  |         StringFileInfo stringInfo = new StringFileInfo(); | ||||||
|  |         versionInfo.AddEntry(stringInfo); | ||||||
|  | 
 | ||||||
|  |         VarFileInfo varInfo = new VarFileInfo(); | ||||||
|  |         versionInfo.AddEntry(varInfo); | ||||||
|  | 
 | ||||||
|  |         var stringTable = new StringTable(_langId, kCodePageUtf16); | ||||||
|  |         stringTable[StringTable.CompanyNameKey] = package.ProductCompany; | ||||||
|  |         stringTable[StringTable.FileDescriptionKey] = package.ProductDescription; | ||||||
|  |         stringTable[StringTable.FileVersionKey] = package.Version.ToFullString(); | ||||||
|  |         stringTable[StringTable.LegalCopyrightKey] = package.ProductCopyright; | ||||||
|  |         stringTable[StringTable.ProductNameKey] = package.ProductName; | ||||||
|  |         stringTable[StringTable.ProductVersionKey] = package.Version.ToFullString(); | ||||||
|  |         stringTable[StringTable.CommentsKey] = $"Generated by Velopack {VelopackRuntimeInfo.VelopackNugetVersion}"; | ||||||
|  |         stringInfo.Tables.Add(stringTable); | ||||||
|  | 
 | ||||||
|  |         var varTable = new VarTable(); | ||||||
|  |         varTable.Values.Add(((uint) kCodePageUtf16 << 16) | _langId); | ||||||
|  |         varInfo.Tables.Add(varTable); | ||||||
|  | 
 | ||||||
|  |         versionInfo.WriteToDirectory(_resources); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public void CopyResourcesFrom(string otherExeFile) | ||||||
|  |     { | ||||||
|  |         ThrowIfDisposed(); | ||||||
|  | 
 | ||||||
|  |         var file = PEFile.FromBytes(File.ReadAllBytes(otherExeFile)); | ||||||
|  |         var image = PEImage.FromFile(file); | ||||||
|  |         _resources = image.Resources; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public void Commit() | ||||||
|  |     { | ||||||
|  |         ThrowIfDisposed(); | ||||||
|  |         _disposed = true; | ||||||
|  | 
 | ||||||
|  |         var sortedResources = new ResourceDirectory((uint) 0); | ||||||
|  |         foreach (var entry in _resources.Entries.OrderBy(e => e.Id).ToArray()) { | ||||||
|  |             _resources.RemoveEntry(entry.Id); | ||||||
|  |             sortedResources.Entries.Add(entry); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var resourceBuffer = new ResourceDirectoryBuffer(); | ||||||
|  |         resourceBuffer.AddDirectory(sortedResources); | ||||||
|  | 
 | ||||||
|  |         var resourceDirectory = _file.OptionalHeader.GetDataDirectory(DataDirectoryIndex.ResourceDirectory); | ||||||
|  | 
 | ||||||
|  |         if (resourceDirectory.IsPresentInPE) { | ||||||
|  |             var section = _file.GetSectionContainingRva(resourceDirectory.VirtualAddress); | ||||||
|  |             section.Contents = resourceBuffer; | ||||||
|  |         } else { | ||||||
|  |             _file.Sections.Add(new PESection(".rsrc", SectionFlags.MemoryRead | SectionFlags.ContentInitializedData, resourceBuffer)); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         _file.UpdateHeaders(); | ||||||
|  |         _file.OptionalHeader.SetDataDirectory(DataDirectoryIndex.ResourceDirectory, | ||||||
|  |             new DataDirectory(resourceBuffer.Rva, resourceBuffer.GetPhysicalSize())); | ||||||
|  | 
 | ||||||
|  |         Utility.Retry(() => { | ||||||
|  |             using var fs = File.Create(_exePath); | ||||||
|  |             _file.Write(fs); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void ThrowIfDisposed() | ||||||
|  |     { | ||||||
|  |         if (_disposed) throw new ObjectDisposedException(nameof(ResourceEdit)); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,6 +1,4 @@ | |||||||
| using System.IO.MemoryMappedFiles; | using System.IO.MemoryMappedFiles; | ||||||
| using Microsoft.NET.HostModel; |  | ||||||
| using Microsoft.NET.HostModel.AppHost; |  | ||||||
| 
 | 
 | ||||||
| namespace Velopack.Packaging.Windows; | namespace Velopack.Packaging.Windows; | ||||||
| 
 | 
 | ||||||
| @@ -24,9 +22,9 @@ public static class SetupBundle | |||||||
|             using var memoryMappedFile = MemoryMappedFile.CreateFromFile(setupPath, FileMode.Open, null, 0, MemoryMappedFileAccess.Read); |             using var memoryMappedFile = MemoryMappedFile.CreateFromFile(setupPath, FileMode.Open, null, 0, MemoryMappedFileAccess.Read); | ||||||
|             using MemoryMappedViewAccessor accessor = memoryMappedFile.CreateViewAccessor(0, 0, MemoryMappedFileAccess.Read); |             using MemoryMappedViewAccessor accessor = memoryMappedFile.CreateViewAccessor(0, 0, MemoryMappedFileAccess.Read); | ||||||
| 
 | 
 | ||||||
|             int position = BinaryUtils.SearchInFile(accessor, bundleSignature); |             int position = SearchInFile(accessor, bundleSignature); | ||||||
|             if (position == -1) { |             if (position == -1) { | ||||||
|                 throw new PlaceHolderNotFoundInAppHostException(bundleSignature); |                 throw new Exception("PlaceHolderNotFoundInAppHostException"); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             offset = accessor.ReadInt64(position - 16); |             offset = accessor.ReadInt64(position - 16); | ||||||
| @@ -74,11 +72,11 @@ public static class SetupBundle | |||||||
|         Array.Copy(BitConverter.GetBytes(bundleLength), 0, data, 8, 8); |         Array.Copy(BitConverter.GetBytes(bundleLength), 0, data, 8, 8); | ||||||
| 
 | 
 | ||||||
|         // replace the beginning of the placeholder with the bytes from 'data' |         // replace the beginning of the placeholder with the bytes from 'data' | ||||||
|         RetryUtil.RetryOnIOError(() => |         RetryOnIOError(() => | ||||||
|             BinaryUtils.SearchAndReplace(setupPath, placeholder, data, pad0s: false)); |             SearchAndReplace(setupPath, placeholder, data, pad0s: false)); | ||||||
| 
 | 
 | ||||||
|         // memory-mapped write does not updating last write time |         // memory-mapped write does not updating last write time | ||||||
|         RetryUtil.RetryOnIOError(() => |         RetryOnIOError(() => | ||||||
|             File.SetLastWriteTimeUtc(setupPath, DateTime.UtcNow)); |             File.SetLastWriteTimeUtc(setupPath, DateTime.UtcNow)); | ||||||
| 
 | 
 | ||||||
|         if (!IsBundle(setupPath, out var offset, out var length)) |         if (!IsBundle(setupPath, out var offset, out var length)) | ||||||
| @@ -86,4 +84,237 @@ public static class SetupBundle | |||||||
| 
 | 
 | ||||||
|         return bundleOffset; |         return bundleOffset; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     internal static unsafe void SearchAndReplace( | ||||||
|  |        MemoryMappedViewAccessor accessor, | ||||||
|  |        byte[] searchPattern, | ||||||
|  |        byte[] patternToReplace, | ||||||
|  |        bool pad0s = true) | ||||||
|  |     { | ||||||
|  |         byte* pointer = null; | ||||||
|  | 
 | ||||||
|  |         try { | ||||||
|  |             accessor.SafeMemoryMappedViewHandle.AcquirePointer(ref pointer); | ||||||
|  |             byte* bytes = pointer + accessor.PointerOffset; | ||||||
|  | 
 | ||||||
|  |             int position = KMPSearch(searchPattern, bytes, accessor.Capacity); | ||||||
|  |             if (position < 0) { | ||||||
|  |                 throw new Exception("PlaceHolderNotFoundInAppHostException"); | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             accessor.WriteArray( | ||||||
|  |                 position: position, | ||||||
|  |                 array: patternToReplace, | ||||||
|  |                 offset: 0, | ||||||
|  |                 count: patternToReplace.Length); | ||||||
|  | 
 | ||||||
|  |             if (pad0s) { | ||||||
|  |                 Pad0(searchPattern, patternToReplace, bytes, position); | ||||||
|  |             } | ||||||
|  |         } finally { | ||||||
|  |             if (pointer != null) { | ||||||
|  |                 accessor.SafeMemoryMappedViewHandle.ReleasePointer(); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private static unsafe void Pad0(byte[] searchPattern, byte[] patternToReplace, byte* bytes, int offset) | ||||||
|  |     { | ||||||
|  |         if (patternToReplace.Length < searchPattern.Length) { | ||||||
|  |             for (int i = patternToReplace.Length; i < searchPattern.Length; i++) { | ||||||
|  |                 bytes[i + offset] = 0x0; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static unsafe void SearchAndReplace( | ||||||
|  |         string filePath, | ||||||
|  |         byte[] searchPattern, | ||||||
|  |         byte[] patternToReplace, | ||||||
|  |         bool pad0s = true) | ||||||
|  |     { | ||||||
|  |         using (var mappedFile = MemoryMappedFile.CreateFromFile(filePath)) { | ||||||
|  |             using (var accessor = mappedFile.CreateViewAccessor()) { | ||||||
|  |                 SearchAndReplace(accessor, searchPattern, patternToReplace, pad0s); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static unsafe int SearchInFile(MemoryMappedViewAccessor accessor, byte[] searchPattern) | ||||||
|  |     { | ||||||
|  |         var safeBuffer = accessor.SafeMemoryMappedViewHandle; | ||||||
|  |         return KMPSearch(searchPattern, (byte*) safeBuffer.DangerousGetHandle(), (int) safeBuffer.ByteLength); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static unsafe int SearchInFile(string filePath, byte[] searchPattern) | ||||||
|  |     { | ||||||
|  |         using (var mappedFile = MemoryMappedFile.CreateFromFile(filePath, FileMode.Open, null, 0, MemoryMappedFileAccess.Read)) { | ||||||
|  |             using (var accessor = mappedFile.CreateViewAccessor(0, 0, MemoryMappedFileAccess.Read)) { | ||||||
|  |                 return SearchInFile(accessor, searchPattern); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // See: https://en.wikipedia.org/wiki/Knuth%E2%80%93Morris%E2%80%93Pratt_algorithm | ||||||
|  |     private static int[] ComputeKMPFailureFunction(byte[] pattern) | ||||||
|  |     { | ||||||
|  |         int[] table = new int[pattern.Length]; | ||||||
|  |         if (pattern.Length >= 1) { | ||||||
|  |             table[0] = -1; | ||||||
|  |         } | ||||||
|  |         if (pattern.Length >= 2) { | ||||||
|  |             table[1] = 0; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         int pos = 2; | ||||||
|  |         int cnd = 0; | ||||||
|  |         while (pos < pattern.Length) { | ||||||
|  |             if (pattern[pos - 1] == pattern[cnd]) { | ||||||
|  |                 table[pos] = cnd + 1; | ||||||
|  |                 cnd++; | ||||||
|  |                 pos++; | ||||||
|  |             } else if (cnd > 0) { | ||||||
|  |                 cnd = table[cnd]; | ||||||
|  |             } else { | ||||||
|  |                 table[pos] = 0; | ||||||
|  |                 pos++; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         return table; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // See: https://en.wikipedia.org/wiki/Knuth%E2%80%93Morris%E2%80%93Pratt_algorithm | ||||||
|  |     private static unsafe int KMPSearch(byte[] pattern, byte* bytes, long bytesLength) | ||||||
|  |     { | ||||||
|  |         int m = 0; | ||||||
|  |         int i = 0; | ||||||
|  |         int[] table = ComputeKMPFailureFunction(pattern); | ||||||
|  | 
 | ||||||
|  |         while (m + i < bytesLength) { | ||||||
|  |             if (pattern[i] == bytes[m + i]) { | ||||||
|  |                 if (i == pattern.Length - 1) { | ||||||
|  |                     return m; | ||||||
|  |                 } | ||||||
|  |                 i++; | ||||||
|  |             } else { | ||||||
|  |                 if (table[i] > -1) { | ||||||
|  |                     m = m + i - table[i]; | ||||||
|  |                     i = table[i]; | ||||||
|  |                 } else { | ||||||
|  |                     m++; | ||||||
|  |                     i = 0; | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return -1; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static void CopyFile(string sourcePath, string destinationPath) | ||||||
|  |     { | ||||||
|  |         var destinationDirectory = new FileInfo(destinationPath).Directory.FullName; | ||||||
|  |         if (!Directory.Exists(destinationDirectory)) { | ||||||
|  |             Directory.CreateDirectory(destinationDirectory); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Copy file to destination path so it inherits the same attributes/permissions. | ||||||
|  |         File.Copy(sourcePath, destinationPath, overwrite: true); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     internal static void WriteToStream(MemoryMappedViewAccessor sourceViewAccessor, FileStream fileStream, long length) | ||||||
|  |     { | ||||||
|  |         int pos = 0; | ||||||
|  |         int bufSize = 16384; //16K | ||||||
|  | 
 | ||||||
|  |         byte[] buf = new byte[bufSize]; | ||||||
|  |         length = Math.Min(length, sourceViewAccessor.Capacity); | ||||||
|  |         do { | ||||||
|  |             int bytesRequested = Math.Min((int) length - pos, bufSize); | ||||||
|  |             if (bytesRequested <= 0) { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             int bytesRead = sourceViewAccessor.ReadArray(pos, buf, 0, bytesRequested); | ||||||
|  |             if (bytesRead > 0) { | ||||||
|  |                 fileStream.Write(buf, 0, bytesRead); | ||||||
|  |                 pos += bytesRead; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         while (true); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public const int NumberOfRetries = 500; | ||||||
|  |     public const int NumMilliSecondsToWait = 100; | ||||||
|  | 
 | ||||||
|  |     public static void RetryOnIOError(Action func) | ||||||
|  |     { | ||||||
|  |         for (int i = 1; i <= NumberOfRetries; i++) { | ||||||
|  |             try { | ||||||
|  |                 func(); | ||||||
|  |                 break; | ||||||
|  |             } catch (IOException) when (i < NumberOfRetries) { | ||||||
|  |                 Thread.Sleep(NumMilliSecondsToWait); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public static void RetryOnWin32Error(Action func) | ||||||
|  |     { | ||||||
|  |         static bool IsKnownIrrecoverableError(int hresult) | ||||||
|  |         { | ||||||
|  |             // Error codes are defined in winerror.h | ||||||
|  |             // The error code is stored in the lowest 16 bits of the HResult | ||||||
|  | 
 | ||||||
|  |             switch (hresult & 0xffff) { | ||||||
|  |             case 0x00000001: // ERROR_INVALID_FUNCTION | ||||||
|  |             case 0x00000002: // ERROR_FILE_NOT_FOUND | ||||||
|  |             case 0x00000003: // ERROR_PATH_NOT_FOUND | ||||||
|  |             case 0x00000006: // ERROR_INVALID_HANDLE | ||||||
|  |             case 0x00000008: // ERROR_NOT_ENOUGH_MEMORY | ||||||
|  |             case 0x0000000B: // ERROR_BAD_FORMAT | ||||||
|  |             case 0x0000000E: // ERROR_OUTOFMEMORY | ||||||
|  |             case 0x0000000F: // ERROR_INVALID_DRIVE | ||||||
|  |             case 0x00000012: // ERROR_NO_MORE_FILES | ||||||
|  |             case 0x00000035: // ERROR_BAD_NETPATH | ||||||
|  |             case 0x00000057: // ERROR_INVALID_PARAMETER | ||||||
|  |             case 0x00000071: // ERROR_NO_MORE_SEARCH_HANDLES | ||||||
|  |             case 0x00000072: // ERROR_INVALID_TARGET_HANDLE | ||||||
|  |             case 0x00000078: // ERROR_CALL_NOT_IMPLEMENTED | ||||||
|  |             case 0x0000007B: // ERROR_INVALID_NAME | ||||||
|  |             case 0x0000007C: // ERROR_INVALID_LEVEL | ||||||
|  |             case 0x0000007D: // ERROR_NO_VOLUME_LABEL | ||||||
|  |             case 0x0000009A: // ERROR_LABEL_TOO_LONG | ||||||
|  |             case 0x000000A0: // ERROR_BAD_ARGUMENTS | ||||||
|  |             case 0x000000A1: // ERROR_BAD_PATHNAME | ||||||
|  |             case 0x000000CE: // ERROR_FILENAME_EXCED_RANGE | ||||||
|  |             case 0x000000DF: // ERROR_FILE_TOO_LARGE | ||||||
|  |             case 0x000003ED: // ERROR_UNRECOGNIZED_VOLUME | ||||||
|  |             case 0x000003EE: // ERROR_FILE_INVALID | ||||||
|  |             case 0x00000651: // ERROR_DEVICE_REMOVED | ||||||
|  |                 return true; | ||||||
|  | 
 | ||||||
|  |             default: | ||||||
|  |                 return false; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         for (int i = 1; i <= NumberOfRetries; i++) { | ||||||
|  |             try { | ||||||
|  |                 func(); | ||||||
|  |                 break; | ||||||
|  |             } catch (HResultException hrex) | ||||||
|  |                   when (i < NumberOfRetries && !IsKnownIrrecoverableError(hrex.Win32HResult)) { | ||||||
|  |                 Thread.Sleep(NumMilliSecondsToWait); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public class HResultException : Exception | ||||||
|  |     { | ||||||
|  |         public readonly int Win32HResult; | ||||||
|  |         public HResultException(int hResult) : base(hResult.ToString("X4")) | ||||||
|  |         { | ||||||
|  |             Win32HResult = hResult; | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
| @@ -4,10 +4,11 @@ | |||||||
|     <TargetFrameworks>net472;net6.0</TargetFrameworks> |     <TargetFrameworks>net472;net6.0</TargetFrameworks> | ||||||
|     <ImplicitUsings>enable</ImplicitUsings> |     <ImplicitUsings>enable</ImplicitUsings> | ||||||
|     <NoWarn>$(NoWarn);CA2007;CS8002</NoWarn> |     <NoWarn>$(NoWarn);CA2007;CS8002</NoWarn> | ||||||
|  |     <AllowUnsafeBlocks>true</AllowUnsafeBlocks> | ||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <ProjectReference Include="..\Velopack.Packaging.HostModel\Velopack.Packaging.HostModel.csproj" /> |     <ProjectReference Include="..\Velopack.IcoLib\Velopack.IcoLib.csproj" /> | ||||||
|     <ProjectReference Include="..\Velopack.Packaging\Velopack.Packaging.csproj" /> |     <ProjectReference Include="..\Velopack.Packaging\Velopack.Packaging.csproj" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -35,8 +35,8 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "AsmResolver": { |       "AsmResolver": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -87,8 +87,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client": { |       "Microsoft.Identity.Client": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "3TDPEie+9t/NgBhoNifLFwM6nkypYUa8GUHfMnkYtovDDTQM8bsS2KEIP5tAh28BgfJZgof/KlCEKvGxz0H3Eg==", |         "contentHash": "zXPDXjTju1yOZQzi5PeNmQfBzlxv+VlCfgfZ70VqyFuhaPtFXwvbLSlwCOMoVALR/MzOR6xkiadoyVDmWDmqKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.IdentityModel.Abstractions": "6.35.0", |           "Microsoft.IdentityModel.Abstractions": "6.35.0", | ||||||
|           "System.Diagnostics.DiagnosticSource": "6.0.1" |           "System.Diagnostics.DiagnosticSource": "6.0.1" | ||||||
| @@ -96,27 +96,27 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Broker": { |       "Microsoft.Identity.Client.Broker": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "D5M7plyrXrQwhj/PILHLS7G3TqDg8rUO9lR+5awUSXueGxsdA2qHpeKiSR0sVfX6wxziWszOLDUR2vjFFKHPSg==", |         "contentHash": "+13pImNYfPGl04qZh3Rtq8wPfo55FScAUEio4ymPzeWJ/PE04jCsKH6ZIlrYbPCqe6t9u0lru1eZ+ho9Dsnvpw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "Microsoft.Identity.Client.NativeInterop": "0.16.0" |           "Microsoft.Identity.Client.NativeInterop": "0.16.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Extensions.Msal": { |       "Microsoft.Identity.Client.Extensions.Msal": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "tCX3c1bSkwH89AKjtB/oAOLlv661tOsEA8ULOCavOc6te/CKlqmMmFs8zMI58Qlv3XkbLOrswYr50IhDntiiPA==", |         "contentHash": "GCSBhzkr7/7DZDk2kYS9dfihB5ieM2CqwGAhtjuE6T1GtAqzuh62UJhMgO0dSx/PLgIYMzNHnh+pfHK8aWJQJQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "System.IO.FileSystem.AccessControl": "5.0.0", |           "System.IO.FileSystem.AccessControl": "5.0.0", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.5.0" |           "System.Security.Cryptography.ProtectedData": "4.5.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.NativeInterop": { |       "Microsoft.Identity.Client.NativeInterop": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "0.16.0", |         "resolved": "0.16.1", | ||||||
|         "contentHash": "riaIH9tT4kRG4xKFxx+wpuo+VAZVOr1ElbNubcvpFXb82gjEmZtJRzpdiS4tdRYTVh6CUKNz9AINAptActTtBQ==" |         "contentHash": "OvY/+/AHESi24f5zOCf9kL4HXwPxXVQ3A+tMQsJvFk2DmP+sc88FYWL49zlku5q0bvx5yFvBLNpHQeRT9a6A5g==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.IdentityModel.Abstractions": { |       "Microsoft.IdentityModel.Abstractions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -135,23 +135,26 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|  |       }, | ||||||
|  |       "SixLabors.ImageSharp": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "2.1.8", | ||||||
|  |         "contentHash": "ML1++vactR5xMW36wHA2nAX3vc6VpR62qYnpopQdLNGbP8BAJ/ckv1IrGIIFRUEQ0JS9EgWr1y1gEa/81f+HaA==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Buffers": "4.5.1", | ||||||
|  |           "System.Memory": "4.5.4", | ||||||
|  |           "System.Numerics.Vectors": "4.5.0", | ||||||
|  |           "System.Runtime.CompilerServices.Unsafe": "5.0.0", | ||||||
|  |           "System.Text.Encoding.CodePages": "5.0.0" | ||||||
|  |         } | ||||||
|       }, |       }, | ||||||
|       "System.Buffers": { |       "System.Buffers": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.5.1", |         "resolved": "4.5.1", | ||||||
|         "contentHash": "Rw7ijyl1qqRS0YQD/WycNst8hUUMgrMH4FCn1nNm27M4VxchZ1js3fVjQaANHO5f3sN4isvP4a+Met9Y4YomAg==" |         "contentHash": "Rw7ijyl1qqRS0YQD/WycNst8hUUMgrMH4FCn1nNm27M4VxchZ1js3fVjQaANHO5f3sN4isvP4a+Met9Y4YomAg==" | ||||||
|       }, |       }, | ||||||
|       "System.Collections.Immutable": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "8.0.0", |  | ||||||
|         "contentHash": "AurL6Y5BA1WotzlEvVaIDpqzpIPvYnnldxru8oXJU2yFxFUy3+pNXjXd1ymO+RA0rq0+590Q8gaz2l3Sr7fmqg==", |  | ||||||
|         "dependencies": { |  | ||||||
|           "System.Memory": "4.5.5", |  | ||||||
|           "System.Runtime.CompilerServices.Unsafe": "6.0.0" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "System.Diagnostics.DiagnosticSource": { |       "System.Diagnostics.DiagnosticSource": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.0.1", |         "resolved": "6.0.1", | ||||||
| @@ -206,15 +209,6 @@ | |||||||
|         "resolved": "4.5.0", |         "resolved": "4.5.0", | ||||||
|         "contentHash": "QQTlPTl06J/iiDbJCiepZ4H//BVraReU4O4EoRw1U02H5TLUIT7xn3GnDp9AXPSlJUDyFs4uWjWafNX6WrAojQ==" |         "contentHash": "QQTlPTl06J/iiDbJCiepZ4H//BVraReU4O4EoRw1U02H5TLUIT7xn3GnDp9AXPSlJUDyFs4uWjWafNX6WrAojQ==" | ||||||
|       }, |       }, | ||||||
|       "System.Reflection.Metadata": { |  | ||||||
|         "type": "Transitive", |  | ||||||
|         "resolved": "8.0.0", |  | ||||||
|         "contentHash": "ptvgrFh7PvWI8bcVqG5rsA/weWM09EnthFHR5SCnS6IN+P4mj6rE1lBDC4U8HL9/57htKAqy4KQ3bBj84cfYyQ==", |  | ||||||
|         "dependencies": { |  | ||||||
|           "System.Collections.Immutable": "8.0.0", |  | ||||||
|           "System.Memory": "4.5.5" |  | ||||||
|         } |  | ||||||
|       }, |  | ||||||
|       "System.Runtime": { |       "System.Runtime": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.3.0", |         "resolved": "4.3.0", | ||||||
| @@ -273,6 +267,14 @@ | |||||||
|         "resolved": "5.0.0", |         "resolved": "5.0.0", | ||||||
|         "contentHash": "t0MGLukB5WAVU9bO3MGzvlGnyJPgUlcwerXn1kzBRjwLKixT96XV0Uza41W49gVd8zEMFu9vQEFlv0IOrytICA==" |         "contentHash": "t0MGLukB5WAVU9bO3MGzvlGnyJPgUlcwerXn1kzBRjwLKixT96XV0Uza41W49gVd8zEMFu9vQEFlv0IOrytICA==" | ||||||
|       }, |       }, | ||||||
|  |       "System.Text.Encoding.CodePages": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "5.0.0", | ||||||
|  |         "contentHash": "NyscU59xX6Uo91qvhOs2Ccho3AR2TnZPomo1Z0K6YpyztBPM/A5VbkzOO19sy3A3i1TtEnTxA7bCe3Us+r5MWg==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Runtime.CompilerServices.Unsafe": "5.0.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "System.Text.Encodings.Web": { |       "System.Text.Encodings.Web": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.0.0", |         "resolved": "6.0.0", | ||||||
| @@ -316,26 +318,26 @@ | |||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Extensions.Logging.Abstractions": "[2.2.0, )", |           "Microsoft.Extensions.Logging.Abstractions": "[2.2.0, )", | ||||||
|           "Newtonsoft.Json": "[13.0.1, )", |           "Newtonsoft.Json": "[13.0.1, )", | ||||||
|           "NuGet.Versioning": "[6.9.1, )" |           "NuGet.Versioning": "[6.10.0, )" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "velopack.icolib": { | ||||||
|  |         "type": "Project", | ||||||
|  |         "dependencies": { | ||||||
|  |           "SixLabors.ImageSharp": "[2.1.8, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.packaging": { |       "velopack.packaging": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Markdig": "[0.37.0, )", |           "Markdig": "[0.37.0, )", | ||||||
|           "Microsoft.Identity.Client": "[4.61.0, )", |           "Microsoft.Identity.Client": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Broker": "[4.61.0, )", |           "Microsoft.Identity.Client.Broker": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.0, )", |           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.2, )", | ||||||
|           "System.Linq.Async": "[6.0.1, )", |           "System.Linq.Async": "[6.0.1, )", | ||||||
|           "System.Net.Http": "[4.3.4, )", |           "System.Net.Http": "[4.3.4, )", | ||||||
|           "Velopack": "[1.0.0, )" |           "Velopack": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |  | ||||||
|       "velopack.packaging.hostmodel": { |  | ||||||
|         "type": "Project", |  | ||||||
|         "dependencies": { |  | ||||||
|           "System.Reflection.Metadata": "[8.0.0, )" |  | ||||||
|         } |  | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     "net6.0": { |     "net6.0": { | ||||||
| @@ -372,8 +374,8 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "AsmResolver": { |       "AsmResolver": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -418,8 +420,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client": { |       "Microsoft.Identity.Client": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "3TDPEie+9t/NgBhoNifLFwM6nkypYUa8GUHfMnkYtovDDTQM8bsS2KEIP5tAh28BgfJZgof/KlCEKvGxz0H3Eg==", |         "contentHash": "zXPDXjTju1yOZQzi5PeNmQfBzlxv+VlCfgfZ70VqyFuhaPtFXwvbLSlwCOMoVALR/MzOR6xkiadoyVDmWDmqKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.IdentityModel.Abstractions": "6.35.0", |           "Microsoft.IdentityModel.Abstractions": "6.35.0", | ||||||
|           "System.Diagnostics.DiagnosticSource": "6.0.1" |           "System.Diagnostics.DiagnosticSource": "6.0.1" | ||||||
| @@ -427,26 +429,26 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Broker": { |       "Microsoft.Identity.Client.Broker": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "D5M7plyrXrQwhj/PILHLS7G3TqDg8rUO9lR+5awUSXueGxsdA2qHpeKiSR0sVfX6wxziWszOLDUR2vjFFKHPSg==", |         "contentHash": "+13pImNYfPGl04qZh3Rtq8wPfo55FScAUEio4ymPzeWJ/PE04jCsKH6ZIlrYbPCqe6t9u0lru1eZ+ho9Dsnvpw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "Microsoft.Identity.Client.NativeInterop": "0.16.0" |           "Microsoft.Identity.Client.NativeInterop": "0.16.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Extensions.Msal": { |       "Microsoft.Identity.Client.Extensions.Msal": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "tCX3c1bSkwH89AKjtB/oAOLlv661tOsEA8ULOCavOc6te/CKlqmMmFs8zMI58Qlv3XkbLOrswYr50IhDntiiPA==", |         "contentHash": "GCSBhzkr7/7DZDk2kYS9dfihB5ieM2CqwGAhtjuE6T1GtAqzuh62UJhMgO0dSx/PLgIYMzNHnh+pfHK8aWJQJQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.5.0" |           "System.Security.Cryptography.ProtectedData": "4.5.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.NativeInterop": { |       "Microsoft.Identity.Client.NativeInterop": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "0.16.0", |         "resolved": "0.16.1", | ||||||
|         "contentHash": "riaIH9tT4kRG4xKFxx+wpuo+VAZVOr1ElbNubcvpFXb82gjEmZtJRzpdiS4tdRYTVh6CUKNz9AINAptActTtBQ==" |         "contentHash": "OvY/+/AHESi24f5zOCf9kL4HXwPxXVQ3A+tMQsJvFk2DmP+sc88FYWL49zlku5q0bvx5yFvBLNpHQeRT9a6A5g==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.IdentityModel.Abstractions": { |       "Microsoft.IdentityModel.Abstractions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -455,8 +457,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.NETCore.Platforms": { |       "Microsoft.NETCore.Platforms": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "1.1.1", |         "resolved": "5.0.0", | ||||||
|         "contentHash": "TMBuzAHpTenGbGgk0SMTwyEkyijY/Eae4ZGsFNYJvAr/LDn1ku3Etp3FPxChmDp5HHF3kzJuoaa08N0xjqAJfQ==" |         "contentHash": "VyPlqzH2wavqquTcYpkIIAQ6WdenuKoFN0BdYBbCWsclXacSOHNQn66Gt4z5NBqEYW0FAPm5rlvki9ZiCij5xQ==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.NETCore.Targets": { |       "Microsoft.NETCore.Targets": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -470,8 +472,8 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "runtime.debian.8-x64.runtime.native.System.Security.Cryptography.OpenSsl": { |       "runtime.debian.8-x64.runtime.native.System.Security.Cryptography.OpenSsl": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -571,6 +573,15 @@ | |||||||
|         "resolved": "4.3.2", |         "resolved": "4.3.2", | ||||||
|         "contentHash": "leXiwfiIkW7Gmn7cgnNcdtNAU70SjmKW3jxGj1iKHOvdn0zRWsgv/l2OJUO5zdGdiv2VRFnAsxxhDgMzofPdWg==" |         "contentHash": "leXiwfiIkW7Gmn7cgnNcdtNAU70SjmKW3jxGj1iKHOvdn0zRWsgv/l2OJUO5zdGdiv2VRFnAsxxhDgMzofPdWg==" | ||||||
|       }, |       }, | ||||||
|  |       "SixLabors.ImageSharp": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "2.1.8", | ||||||
|  |         "contentHash": "ML1++vactR5xMW36wHA2nAX3vc6VpR62qYnpopQdLNGbP8BAJ/ckv1IrGIIFRUEQ0JS9EgWr1y1gEa/81f+HaA==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Runtime.CompilerServices.Unsafe": "5.0.0", | ||||||
|  |           "System.Text.Encoding.CodePages": "5.0.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "System.Collections": { |       "System.Collections": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.3.0", |         "resolved": "4.3.0", | ||||||
| @@ -1010,6 +1021,14 @@ | |||||||
|           "System.Runtime": "4.3.0" |           "System.Runtime": "4.3.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|  |       "System.Text.Encoding.CodePages": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "5.0.0", | ||||||
|  |         "contentHash": "NyscU59xX6Uo91qvhOs2Ccho3AR2TnZPomo1Z0K6YpyztBPM/A5VbkzOO19sy3A3i1TtEnTxA7bCe3Us+r5MWg==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "Microsoft.NETCore.Platforms": "5.0.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "System.Text.Encodings.Web": { |       "System.Text.Encodings.Web": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.0.0", |         "resolved": "6.0.0", | ||||||
| @@ -1050,23 +1069,26 @@ | |||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", |           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", | ||||||
|           "NuGet.Versioning": "[6.9.1, )" |           "NuGet.Versioning": "[6.10.0, )" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "velopack.icolib": { | ||||||
|  |         "type": "Project", | ||||||
|  |         "dependencies": { | ||||||
|  |           "SixLabors.ImageSharp": "[2.1.8, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.packaging": { |       "velopack.packaging": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Markdig": "[0.37.0, )", |           "Markdig": "[0.37.0, )", | ||||||
|           "Microsoft.Identity.Client": "[4.61.0, )", |           "Microsoft.Identity.Client": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Broker": "[4.61.0, )", |           "Microsoft.Identity.Client.Broker": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.0, )", |           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.2, )", | ||||||
|           "System.Linq.Async": "[6.0.1, )", |           "System.Linq.Async": "[6.0.1, )", | ||||||
|           "System.Net.Http": "[4.3.4, )", |           "System.Net.Http": "[4.3.4, )", | ||||||
|           "Velopack": "[1.0.0, )" |           "Velopack": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |  | ||||||
|       "velopack.packaging.hostmodel": { |  | ||||||
|         "type": "Project" |  | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ namespace Velopack.Packaging; | |||||||
| 
 | 
 | ||||||
| public static class Exe | public static class Exe | ||||||
| { | { | ||||||
|     public static void AssertSystemBinaryExists(string binaryName) |     public static void AssertSystemBinaryExists(string binaryName, string linuxInstallCmd, string osxInstallCmd) | ||||||
|     { |     { | ||||||
|         try { |         try { | ||||||
|             if (VelopackRuntimeInfo.IsWindows) { |             if (VelopackRuntimeInfo.IsWindows) { | ||||||
| @@ -20,10 +20,52 @@ public static class Exe | |||||||
|                 throw new PlatformNotSupportedException(); |                 throw new PlatformNotSupportedException(); | ||||||
|             } |             } | ||||||
|         } catch (ProcessFailedException) { |         } catch (ProcessFailedException) { | ||||||
|             throw new Exception($"Could not find '{binaryName}' on the system, ensure it is installed and on the PATH."); |             string recommendedCmd = null; | ||||||
|  |             if (VelopackRuntimeInfo.IsLinux && !String.IsNullOrEmpty(linuxInstallCmd)) | ||||||
|  |                 recommendedCmd = linuxInstallCmd; | ||||||
|  |             else if (VelopackRuntimeInfo.IsOSX && !String.IsNullOrEmpty(osxInstallCmd)) | ||||||
|  |                 recommendedCmd = osxInstallCmd; | ||||||
|  | 
 | ||||||
|  |             string message = $"Could not find '{binaryName}' binary on the system, ensure it is installed and on the PATH."; | ||||||
|  |             if (!String.IsNullOrEmpty(recommendedCmd)) { | ||||||
|  |                 message += $" You might be able to install it by running: '{recommendedCmd}'"; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |             throw new UserInfoException(message); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public static string RunHostedCommand(string command, string workDir = null) | ||||||
|  |     { | ||||||
|  |         using var _1 = Utility.GetTempFileName(out var outputFile); | ||||||
|  |         File.Create(outputFile).Close(); | ||||||
|  | 
 | ||||||
|  |         var fileName = "cmd.exe"; | ||||||
|  |         var args = $"/S /C \"{command} >> \"{outputFile}\" 2>&1\""; | ||||||
|  | 
 | ||||||
|  |         if (!VelopackRuntimeInfo.IsWindows) { | ||||||
|  |             fileName = "/bin/bash"; | ||||||
|  |             string escapedCommand = command.Replace("'", "'\\''"); | ||||||
|  |             args = $"-c '{escapedCommand} >> \"{outputFile}\" 2>&1'"; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         var psi = new ProcessStartInfo { | ||||||
|  |             FileName = fileName, | ||||||
|  |             Arguments = args, | ||||||
|  |             UseShellExecute = false, | ||||||
|  |             WorkingDirectory = workDir, | ||||||
|  |             CreateNoWindow = true, | ||||||
|  |         }; | ||||||
|  | 
 | ||||||
|  |         using var process = Process.Start(psi); | ||||||
|  |         process.WaitForExit(); | ||||||
|  | 
 | ||||||
|  |         var stdout = Utility.Retry(() => File.ReadAllText(outputFile).Trim(), 10, 1000); | ||||||
|  |         var result = (process.ExitCode, stdout, command); | ||||||
|  |         ProcessFailedException.ThrowIfNonZero(result); | ||||||
|  |         return result.Item2; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     public static string InvokeAndThrowIfNonZero(string exePath, IEnumerable<string> args, string workingDir, IDictionary<string, string> envVar = null) |     public static string InvokeAndThrowIfNonZero(string exePath, IEnumerable<string> args, string workingDir, IDictionary<string, string> envVar = null) | ||||||
|     { |     { | ||||||
|         var result = InvokeProcess(exePath, args, workingDir, CancellationToken.None, envVar); |         var result = InvokeProcess(exePath, args, workingDir, CancellationToken.None, envVar); | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ public interface IVelopackFlowServiceClient | |||||||
| 
 | 
 | ||||||
|     Task<Profile?> GetProfileAsync(VelopackServiceOptions? options, CancellationToken cancellationToken); |     Task<Profile?> GetProfileAsync(VelopackServiceOptions? options, CancellationToken cancellationToken); | ||||||
| 
 | 
 | ||||||
|     Task UploadLatestReleaseAssetsAsync(string? channel, string releaseDirectory, string? serviceUrl, CancellationToken cancellationToken); |     Task UploadLatestReleaseAssetsAsync(string? channel, string releaseDirectory, string? serviceUrl, RuntimeOs os, CancellationToken cancellationToken); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| public class VelopackFlowServiceClient(HttpClient HttpClient, ILogger Logger) : IVelopackFlowServiceClient | public class VelopackFlowServiceClient(HttpClient HttpClient, ILogger Logger) : IVelopackFlowServiceClient | ||||||
| @@ -91,10 +91,11 @@ public class VelopackFlowServiceClient(HttpClient HttpClient, ILogger Logger) : | |||||||
|         return await HttpClient.GetFromJsonAsync<Profile>(endpoint, cancellationToken); |         return await HttpClient.GetFromJsonAsync<Profile>(endpoint, cancellationToken); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task UploadLatestReleaseAssetsAsync(string? channel, string releaseDirectory, string? serviceUrl, CancellationToken cancellationToken) |     public async Task UploadLatestReleaseAssetsAsync(string? channel, string releaseDirectory, string? serviceUrl, | ||||||
|  |         RuntimeOs os, CancellationToken cancellationToken) | ||||||
|     { |     { | ||||||
|         channel ??= ReleaseEntryHelper.GetDefaultChannel(VelopackRuntimeInfo.SystemOs); |         channel ??= ReleaseEntryHelper.GetDefaultChannel(os); | ||||||
|         ReleaseEntryHelper helper = new(releaseDirectory, channel, Logger); |         ReleaseEntryHelper helper = new(releaseDirectory, channel, Logger, os); | ||||||
|         var latestAssets = helper.GetLatestAssets().ToList(); |         var latestAssets = helper.GetLatestAssets().ToList(); | ||||||
| 
 | 
 | ||||||
|         List<string> installers = []; |         List<string> installers = []; | ||||||
| @@ -107,13 +108,13 @@ public class VelopackFlowServiceClient(HttpClient HttpClient, ILogger Logger) : | |||||||
|             version = latestAssets[0].Version; |             version = latestAssets[0].Version; | ||||||
| 
 | 
 | ||||||
|             if (VelopackRuntimeInfo.IsWindows || VelopackRuntimeInfo.IsOSX) { |             if (VelopackRuntimeInfo.IsWindows || VelopackRuntimeInfo.IsOSX) { | ||||||
|                 var setupName = ReleaseEntryHelper.GetSuggestedSetupName(packageId, channel); |                 var setupName = ReleaseEntryHelper.GetSuggestedSetupName(packageId, channel, os); | ||||||
|                 if (File.Exists(Path.Combine(releaseDirectory, setupName))) { |                 if (File.Exists(Path.Combine(releaseDirectory, setupName))) { | ||||||
|                     installers.Add(setupName); |                     installers.Add(setupName); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             var portableName = ReleaseEntryHelper.GetSuggestedPortableName(packageId, channel); |             var portableName = ReleaseEntryHelper.GetSuggestedPortableName(packageId, channel, os); | ||||||
|             if (File.Exists(Path.Combine(releaseDirectory, portableName))) { |             if (File.Exists(Path.Combine(releaseDirectory, portableName))) { | ||||||
|                 installers.Add(portableName); |                 installers.Add(portableName); | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -5,5 +5,6 @@ public class VelopackServiceOptions | |||||||
|     public const string DefaultBaseUrl = "https://api.velopack.io/"; |     public const string DefaultBaseUrl = "https://api.velopack.io/"; | ||||||
| 
 | 
 | ||||||
|     public string VelopackBaseUrl { get; set; } = DefaultBaseUrl; |     public string VelopackBaseUrl { get; set; } = DefaultBaseUrl; | ||||||
|  | 
 | ||||||
|     public string ApiKey { get; set; } = string.Empty; |     public string ApiKey { get; set; } = string.Empty; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -4,12 +4,11 @@ namespace Velopack.Packaging; | |||||||
| 
 | 
 | ||||||
| public static class HelperFile | public static class HelperFile | ||||||
| { | { | ||||||
|     public static string GetUpdateExeName(RuntimeOs? os = null) |     public static string GetUpdateExeName(RuntimeOs os) | ||||||
|     { |     { | ||||||
|         var _os = os ?? VelopackRuntimeInfo.SystemOs; |         switch (os) { | ||||||
|         switch (_os) { |  | ||||||
|         case RuntimeOs.Windows: |         case RuntimeOs.Windows: | ||||||
|             return FindHelperFile("Update.exe"); |             return FindHelperFile("update.exe"); | ||||||
| #if DEBUG | #if DEBUG | ||||||
|         case RuntimeOs.Linux: |         case RuntimeOs.Linux: | ||||||
|             return FindHelperFile("update"); |             return FindHelperFile("update"); | ||||||
| @@ -26,38 +25,43 @@ public static class HelperFile | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static string GetUpdatePath(RuntimeOs? os = null) => FindHelperFile(GetUpdateExeName(os)); |     public static string GetUpdatePath(RuntimeOs os) => FindHelperFile(GetUpdateExeName(os)); | ||||||
| 
 | 
 | ||||||
|     public static string GetZstdPath() |     public static string GetZstdPath() | ||||||
|     { |     { | ||||||
|         if (VelopackRuntimeInfo.IsWindows) |         if (VelopackRuntimeInfo.IsWindows) | ||||||
|             return FindHelperFile("zstd.exe"); |             return FindHelperFile("zstd.exe"); | ||||||
|         Exe.AssertSystemBinaryExists("zstd"); |         Exe.AssertSystemBinaryExists("zstd", "sudo apt install zstd", "brew install zstd"); | ||||||
|         return "zstd"; |         return "zstd"; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public static string GetMkSquashFsPath() | ||||||
|  |     { | ||||||
|  |         if (VelopackRuntimeInfo.IsWindows) | ||||||
|  |             return FindHelperFile("squashfs-tools\\gensquashfs.exe"); | ||||||
|  |         Exe.AssertSystemBinaryExists("mksquashfs", "sudo apt install squashfs-tools", "brew install squashfs"); | ||||||
|  |         return "mksquashfs"; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     [SupportedOSPlatform("macos")] |     [SupportedOSPlatform("macos")] | ||||||
|     public static string VelopackEntitlements => FindHelperFile("Velopack.entitlements"); |     public static string VelopackEntitlements => FindHelperFile("Velopack.entitlements"); | ||||||
| 
 | 
 | ||||||
|     [SupportedOSPlatform("linux")] |     public static string AppImageRuntimeArm64 => FindHelperFile("appimagekit-runtime-aarch64"); | ||||||
|     public static string AppImageToolX64 => FindHelperFile("appimagetool-x86_64.AppImage"); |  | ||||||
| 
 | 
 | ||||||
|     [SupportedOSPlatform("windows")] |     public static string AppImageRuntimeX64 => FindHelperFile("appimagekit-runtime-x86_64"); | ||||||
|     public static string SetupPath => FindHelperFile("Setup.exe"); | 
 | ||||||
|  |     public static string AppImageRuntimeX86 => FindHelperFile("appimagekit-runtime-i686"); | ||||||
|  | 
 | ||||||
|  |     public static string SetupPath => FindHelperFile("setup.exe"); | ||||||
| 
 | 
 | ||||||
|     [SupportedOSPlatform("windows")] |  | ||||||
|     public static string StubExecutablePath => FindHelperFile("stub.exe"); |     public static string StubExecutablePath => FindHelperFile("stub.exe"); | ||||||
| 
 | 
 | ||||||
|     [SupportedOSPlatform("windows")] |     [SupportedOSPlatform("windows")] | ||||||
|     public static string SignToolPath => FindHelperFile("signtool.exe"); |     public static string SignToolPath => FindHelperFile("signtool.exe"); | ||||||
| 
 | 
 | ||||||
|     [SupportedOSPlatform("windows")] |     public static string GetDefaultAppIcon(RuntimeOs os) | ||||||
|     public static string RceditPath => FindHelperFile("rcedit.exe"); |  | ||||||
| 
 |  | ||||||
|     public static string GetDefaultAppIcon(RuntimeOs? os = null) |  | ||||||
|     { |     { | ||||||
|         var _os = os ?? VelopackRuntimeInfo.SystemOs; |         switch (os) { | ||||||
|         switch (_os) { |  | ||||||
|         case RuntimeOs.Windows: |         case RuntimeOs.Windows: | ||||||
|             return null; |             return null; | ||||||
|         case RuntimeOs.Linux: |         case RuntimeOs.Linux: | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ namespace Velopack.Packaging; | |||||||
| public abstract class PackageBuilder<T> : ICommand<T> | public abstract class PackageBuilder<T> : ICommand<T> | ||||||
|     where T : class, IPackOptions |     where T : class, IPackOptions | ||||||
| { | { | ||||||
|     protected RuntimeOs SupportedTargetOs { get; } |     protected RuntimeOs TargetOs { get; } | ||||||
| 
 | 
 | ||||||
|     protected ILogger Log { get; } |     protected ILogger Log { get; } | ||||||
| 
 | 
 | ||||||
| @@ -36,25 +36,28 @@ public abstract class PackageBuilder<T> : ICommand<T> | |||||||
| 
 | 
 | ||||||
|     public PackageBuilder(RuntimeOs supportedOs, ILogger logger, IFancyConsole console) |     public PackageBuilder(RuntimeOs supportedOs, ILogger logger, IFancyConsole console) | ||||||
|     { |     { | ||||||
|         SupportedTargetOs = supportedOs; |         TargetOs = supportedOs; | ||||||
|         Log = logger; |         Log = logger; | ||||||
|         Console = console; |         Console = console; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public async Task Run(T options) |     public async Task Run(T options) | ||||||
|     { |     { | ||||||
|         if (options.TargetRuntime?.BaseRID != SupportedTargetOs) |         if (options.TargetRuntime?.BaseRID != TargetOs) { | ||||||
|             throw new UserInfoException($"To build packages for {SupportedTargetOs.GetOsLongName()}, " + |             throw new UserInfoException($"To build packages for {TargetOs.GetOsLongName()}, " + | ||||||
|                 $"the target rid must be {SupportedTargetOs} (actually was {options.TargetRuntime?.BaseRID})."); |                 $"the target rid must be {TargetOs} (actually was {options.TargetRuntime?.BaseRID}). " + | ||||||
|  |                 $"If your real intention was to cross-compile a release for {options.TargetRuntime?.BaseRID} then you " + | ||||||
|  |                 $"should provide an OS directive: eg. 'vpk [{options.TargetRuntime?.BaseRID.GetOsShortName()}] pack ...'"); | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         Log.Info($"Beginning to package Velopack release {options.PackVersion}."); |         Log.Info($"Beginning to package Velopack release {options.PackVersion}."); | ||||||
|         Log.Info("Releases Directory: " + options.ReleaseDir.FullName); |         Log.Info("Releases Directory: " + options.ReleaseDir.FullName); | ||||||
| 
 | 
 | ||||||
|         var releaseDir = options.ReleaseDir; |         var releaseDir = options.ReleaseDir; | ||||||
|         var channel = options.Channel?.ToLower() ?? ReleaseEntryHelper.GetDefaultChannel(SupportedTargetOs); |         var channel = options.Channel?.ToLower() ?? ReleaseEntryHelper.GetDefaultChannel(TargetOs); | ||||||
|         Channel = channel; |         Channel = channel; | ||||||
| 
 | 
 | ||||||
|         var entryHelper = new ReleaseEntryHelper(releaseDir.FullName, channel, Log); |         var entryHelper = new ReleaseEntryHelper(releaseDir.FullName, channel, Log, TargetOs); | ||||||
|         if (entryHelper.DoesSimilarVersionExist(SemanticVersion.Parse(options.PackVersion))) { |         if (entryHelper.DoesSimilarVersionExist(SemanticVersion.Parse(options.PackVersion))) { | ||||||
|             if (await Console.PromptYesNo("A release in this channel with the same or greater version already exists. Do you want to continue and potentially overwrite files?") != true) { |             if (await Console.PromptYesNo("A release in this channel with the same or greater version already exists. Do you want to continue and potentially overwrite files?") != true) { | ||||||
|                 throw new UserInfoException($"There is a release in channel {channel} which is equal or greater to the current version {options.PackVersion}. Please increase the current package version or remove that release."); |                 throw new UserInfoException($"There is a release in channel {channel} which is equal or greater to the current version {options.PackVersion}. Please increase the current package version or remove that release."); | ||||||
| @@ -110,35 +113,35 @@ public abstract class PackageBuilder<T> : ICommand<T> | |||||||
|                 packDirectory = await PreprocessPackDir(progress, packDirectory); |                 packDirectory = await PreprocessPackDir(progress, packDirectory); | ||||||
|             }); |             }); | ||||||
| 
 | 
 | ||||||
|             if (VelopackRuntimeInfo.IsWindows || VelopackRuntimeInfo.IsOSX) { |             if (TargetOs != RuntimeOs.Linux) { | ||||||
|                 await ctx.RunTask("Code-sign application", async (progress) => { |                 await ctx.RunTask("Code-sign application", async (progress) => { | ||||||
|                     await CodeSign(progress, packDirectory); |                     await CodeSign(progress, packDirectory); | ||||||
|                 }); |                 }); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             Task portableTask = null; |             Task portableTask = null; | ||||||
|             if (VelopackRuntimeInfo.IsLinux || !Options.NoPortable) { |             if (TargetOs == RuntimeOs.Linux || !Options.NoPortable) { | ||||||
|                 portableTask = ctx.RunTask("Building portable package", async (progress) => { |                 portableTask = ctx.RunTask("Building portable package", async (progress) => { | ||||||
|                     var suggestedName = ReleaseEntryHelper.GetSuggestedPortableName(packId, channel); |                     var suggestedName = ReleaseEntryHelper.GetSuggestedPortableName(packId, channel, TargetOs); | ||||||
|                     var path = getIncompletePath(suggestedName); |                     var path = getIncompletePath(suggestedName); | ||||||
|                     await CreatePortablePackage(progress, packDirectory, path); |                     await CreatePortablePackage(progress, packDirectory, path); | ||||||
|                 }); |                 }); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             // TODO: hack, this is a prerequisite for building full package but only on linux |             // TODO: hack, this is a prerequisite for building full package but only on linux | ||||||
|             if (VelopackRuntimeInfo.IsLinux) await portableTask; |             if (TargetOs == RuntimeOs.Linux) await portableTask; | ||||||
| 
 | 
 | ||||||
|             string releasePath = null; |             string releasePath = null; | ||||||
|             await ctx.RunTask($"Building release {packVersion}", async (progress) => { |             await ctx.RunTask($"Building release {packVersion}", async (progress) => { | ||||||
|                 var suggestedName = ReleaseEntryHelper.GetSuggestedReleaseName(packId, packVersion, channel, false); |                 var suggestedName = ReleaseEntryHelper.GetSuggestedReleaseName(packId, packVersion, channel, false, TargetOs); | ||||||
|                 releasePath = getIncompletePath(suggestedName); |                 releasePath = getIncompletePath(suggestedName); | ||||||
|                 await CreateReleasePackage(progress, packDirectory, releasePath); |                 await CreateReleasePackage(progress, packDirectory, releasePath); | ||||||
|             }); |             }); | ||||||
| 
 | 
 | ||||||
|             Task setupTask = null; |             Task setupTask = null; | ||||||
|             if (!Options.NoInst && (VelopackRuntimeInfo.IsWindows || VelopackRuntimeInfo.IsOSX)) { |             if (!Options.NoInst && TargetOs != RuntimeOs.Linux) { | ||||||
|                 setupTask = ctx.RunTask("Building setup package", async (progress) => { |                 setupTask = ctx.RunTask("Building setup package", async (progress) => { | ||||||
|                     var suggestedName = ReleaseEntryHelper.GetSuggestedSetupName(packId, channel); |                     var suggestedName = ReleaseEntryHelper.GetSuggestedSetupName(packId, channel, TargetOs); | ||||||
|                     var path = getIncompletePath(suggestedName); |                     var path = getIncompletePath(suggestedName); | ||||||
|                     await CreateSetupPackage(progress, releasePath, packDirectory, path); |                     await CreateSetupPackage(progress, releasePath, packDirectory, path); | ||||||
|                 }); |                 }); | ||||||
| @@ -146,12 +149,12 @@ public abstract class PackageBuilder<T> : ICommand<T> | |||||||
| 
 | 
 | ||||||
|             if (prev != null && options.DeltaMode != DeltaMode.None) { |             if (prev != null && options.DeltaMode != DeltaMode.None) { | ||||||
|                 await ctx.RunTask($"Building delta {prev.Version} -> {packVersion}", async (progress) => { |                 await ctx.RunTask($"Building delta {prev.Version} -> {packVersion}", async (progress) => { | ||||||
|                     var suggestedName = ReleaseEntryHelper.GetSuggestedReleaseName(packId, packVersion, channel, true); |                     var suggestedName = ReleaseEntryHelper.GetSuggestedReleaseName(packId, packVersion, channel, true, TargetOs); | ||||||
|                     var deltaPkg = await CreateDeltaPackage(progress, releasePath, prev.PackageFile, getIncompletePath(suggestedName), options.DeltaMode); |                     var deltaPkg = await CreateDeltaPackage(progress, releasePath, prev.PackageFile, getIncompletePath(suggestedName), options.DeltaMode); | ||||||
|                 }); |                 }); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             if (!VelopackRuntimeInfo.IsLinux && portableTask != null) await portableTask; |             if (TargetOs != RuntimeOs.Linux && portableTask != null) await portableTask; | ||||||
|             if (setupTask != null) await setupTask; |             if (setupTask != null) await setupTask; | ||||||
| 
 | 
 | ||||||
|             await ctx.RunTask("Post-process steps", (progress) => { |             await ctx.RunTask("Post-process steps", (progress) => { | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| using System.Text; | using System.Text; | ||||||
| using Microsoft.Extensions.Logging; | using Microsoft.Extensions.Logging; | ||||||
| using NuGet.Versioning; | using NuGet.Versioning; | ||||||
| using Velopack.Json; | using Velopack.Json; | ||||||
| @@ -13,11 +13,11 @@ public class ReleaseEntryHelper | |||||||
|     private readonly string _channel; |     private readonly string _channel; | ||||||
|     private Dictionary<string, List<VelopackAsset>> _releases; |     private Dictionary<string, List<VelopackAsset>> _releases; | ||||||
| 
 | 
 | ||||||
|     public ReleaseEntryHelper(string outputDir, string channel, ILogger logger) |     public ReleaseEntryHelper(string outputDir, string channel, ILogger logger, RuntimeOs os) | ||||||
|     { |     { | ||||||
|         _outputDir = outputDir; |         _outputDir = outputDir; | ||||||
|         _logger = logger; |         _logger = logger; | ||||||
|         _channel = channel ?? GetDefaultChannel(); |         _channel = channel ?? GetDefaultChannel(os); | ||||||
|         _releases = GetReleasesFromDir(outputDir); |         _releases = GetReleasesFromDir(outputDir); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @@ -153,20 +153,20 @@ public class ReleaseEntryHelper | |||||||
|         return Encoding.UTF8.GetString(ms.ToArray()); |         return Encoding.UTF8.GetString(ms.ToArray()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static string GetSuggestedReleaseName(string id, string version, string channel, bool delta) |     public static string GetSuggestedReleaseName(string id, string version, string channel, bool delta, RuntimeOs os) | ||||||
|     { |     { | ||||||
|         var suffix = GetUniqueAssetSuffix(channel); |         var suffix = GetUniqueAssetSuffix(channel); | ||||||
|         version = SemanticVersion.Parse(version).ToNormalizedString(); |         version = SemanticVersion.Parse(version).ToNormalizedString(); | ||||||
|         if (VelopackRuntimeInfo.IsWindows && channel == GetDefaultChannel(RuntimeOs.Windows)) { |         if (os == RuntimeOs.Windows && channel == GetDefaultChannel(RuntimeOs.Windows)) { | ||||||
|             return $"{id}-{version}{(delta ? "-delta" : "-full")}.nupkg"; |             return $"{id}-{version}{(delta ? "-delta" : "-full")}.nupkg"; | ||||||
|         } |         } | ||||||
|         return $"{id}-{version}{suffix}{(delta ? "-delta" : "-full")}.nupkg"; |         return $"{id}-{version}{suffix}{(delta ? "-delta" : "-full")}.nupkg"; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static string GetSuggestedPortableName(string id, string channel) |     public static string GetSuggestedPortableName(string id, string channel, RuntimeOs os) | ||||||
|     { |     { | ||||||
|         var suffix = GetUniqueAssetSuffix(channel); |         var suffix = GetUniqueAssetSuffix(channel); | ||||||
|         if (VelopackRuntimeInfo.IsLinux) { |         if (os == RuntimeOs.Linux) { | ||||||
|             if (channel == GetDefaultChannel(RuntimeOs.Linux)) { |             if (channel == GetDefaultChannel(RuntimeOs.Linux)) { | ||||||
|                 return $"{id}.AppImage"; |                 return $"{id}.AppImage"; | ||||||
|             } else { |             } else { | ||||||
| @@ -177,12 +177,12 @@ public class ReleaseEntryHelper | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static string GetSuggestedSetupName(string id, string channel) |     public static string GetSuggestedSetupName(string id, string channel, RuntimeOs os) | ||||||
|     { |     { | ||||||
|         var suffix = GetUniqueAssetSuffix(channel); |         var suffix = GetUniqueAssetSuffix(channel); | ||||||
|         if (VelopackRuntimeInfo.IsWindows) |         if (os == RuntimeOs.Windows) | ||||||
|             return $"{id}{suffix}-Setup.exe"; |             return $"{id}{suffix}-Setup.exe"; | ||||||
|         else if (VelopackRuntimeInfo.IsOSX) |         else if (os == RuntimeOs.OSX) | ||||||
|             return $"{id}{suffix}-Setup.pkg"; |             return $"{id}{suffix}-Setup.pkg"; | ||||||
|         else |         else | ||||||
|             throw new PlatformNotSupportedException("Platform not supported."); |             throw new PlatformNotSupportedException("Platform not supported."); | ||||||
| @@ -193,9 +193,8 @@ public class ReleaseEntryHelper | |||||||
|         return "-" + channel; |         return "-" + channel; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public static string GetDefaultChannel(RuntimeOs? os = null) |     public static string GetDefaultChannel(RuntimeOs os) | ||||||
|     { |     { | ||||||
|         os ??= VelopackRuntimeInfo.SystemOs; |  | ||||||
|         if (os == RuntimeOs.Windows) return "win"; |         if (os == RuntimeOs.Windows) return "win"; | ||||||
|         if (os == RuntimeOs.OSX) return "osx"; |         if (os == RuntimeOs.OSX) return "osx"; | ||||||
|         if (os == RuntimeOs.Linux) return "linux"; |         if (os == RuntimeOs.Linux) return "linux"; | ||||||
|   | |||||||
| @@ -15,9 +15,9 @@ | |||||||
|     <PackageReference Include="System.Linq.Async" Version="6.0.1" /> |     <PackageReference Include="System.Linq.Async" Version="6.0.1" /> | ||||||
|     <PackageReference Include="System.Net.Http" Version="4.3.4" /> |     <PackageReference Include="System.Net.Http" Version="4.3.4" /> | ||||||
|     <PackageReference Include="Markdig" Version="0.37.0" /> |     <PackageReference Include="Markdig" Version="0.37.0" /> | ||||||
|     <PackageReference Include="Microsoft.Identity.Client" Version="4.61.0" /> |     <PackageReference Include="Microsoft.Identity.Client" Version="4.61.2" /> | ||||||
|     <PackageReference Include="Microsoft.Identity.Client.Broker" Version="4.61.0" /> |     <PackageReference Include="Microsoft.Identity.Client.Broker" Version="4.61.2" /> | ||||||
|     <PackageReference Include="Microsoft.Identity.Client.Extensions.Msal" Version="4.61.0" /> |     <PackageReference Include="Microsoft.Identity.Client.Extensions.Msal" Version="4.61.2" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
| </Project> | </Project> | ||||||
|   | |||||||
| @@ -13,9 +13,9 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client": { |       "Microsoft.Identity.Client": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[4.61.0, )", |         "requested": "[4.61.2, )", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "3TDPEie+9t/NgBhoNifLFwM6nkypYUa8GUHfMnkYtovDDTQM8bsS2KEIP5tAh28BgfJZgof/KlCEKvGxz0H3Eg==", |         "contentHash": "zXPDXjTju1yOZQzi5PeNmQfBzlxv+VlCfgfZ70VqyFuhaPtFXwvbLSlwCOMoVALR/MzOR6xkiadoyVDmWDmqKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.IdentityModel.Abstractions": "6.35.0", |           "Microsoft.IdentityModel.Abstractions": "6.35.0", | ||||||
|           "System.Diagnostics.DiagnosticSource": "6.0.1" |           "System.Diagnostics.DiagnosticSource": "6.0.1" | ||||||
| @@ -23,21 +23,21 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Broker": { |       "Microsoft.Identity.Client.Broker": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[4.61.0, )", |         "requested": "[4.61.2, )", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "D5M7plyrXrQwhj/PILHLS7G3TqDg8rUO9lR+5awUSXueGxsdA2qHpeKiSR0sVfX6wxziWszOLDUR2vjFFKHPSg==", |         "contentHash": "+13pImNYfPGl04qZh3Rtq8wPfo55FScAUEio4ymPzeWJ/PE04jCsKH6ZIlrYbPCqe6t9u0lru1eZ+ho9Dsnvpw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "Microsoft.Identity.Client.NativeInterop": "0.16.0" |           "Microsoft.Identity.Client.NativeInterop": "0.16.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Extensions.Msal": { |       "Microsoft.Identity.Client.Extensions.Msal": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[4.61.0, )", |         "requested": "[4.61.2, )", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "tCX3c1bSkwH89AKjtB/oAOLlv661tOsEA8ULOCavOc6te/CKlqmMmFs8zMI58Qlv3XkbLOrswYr50IhDntiiPA==", |         "contentHash": "GCSBhzkr7/7DZDk2kYS9dfihB5ieM2CqwGAhtjuE6T1GtAqzuh62UJhMgO0dSx/PLgIYMzNHnh+pfHK8aWJQJQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "System.IO.FileSystem.AccessControl": "5.0.0", |           "System.IO.FileSystem.AccessControl": "5.0.0", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.5.0" |           "System.Security.Cryptography.ProtectedData": "4.5.0" | ||||||
|         } |         } | ||||||
| @@ -55,8 +55,8 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "System.Linq.Async": { |       "System.Linq.Async": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
| @@ -96,8 +96,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.NativeInterop": { |       "Microsoft.Identity.Client.NativeInterop": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "0.16.0", |         "resolved": "0.16.1", | ||||||
|         "contentHash": "riaIH9tT4kRG4xKFxx+wpuo+VAZVOr1ElbNubcvpFXb82gjEmZtJRzpdiS4tdRYTVh6CUKNz9AINAptActTtBQ==" |         "contentHash": "OvY/+/AHESi24f5zOCf9kL4HXwPxXVQ3A+tMQsJvFk2DmP+sc88FYWL49zlku5q0bvx5yFvBLNpHQeRT9a6A5g==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.IdentityModel.Abstractions": { |       "Microsoft.IdentityModel.Abstractions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -116,8 +116,8 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "System.Buffers": { |       "System.Buffers": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -233,7 +233,7 @@ | |||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Extensions.Logging.Abstractions": "[2.2.0, )", |           "Microsoft.Extensions.Logging.Abstractions": "[2.2.0, )", | ||||||
|           "Newtonsoft.Json": "[13.0.1, )", |           "Newtonsoft.Json": "[13.0.1, )", | ||||||
|           "NuGet.Versioning": "[6.9.1, )" |           "NuGet.Versioning": "[6.10.0, )" | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
| @@ -246,9 +246,9 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client": { |       "Microsoft.Identity.Client": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[4.61.0, )", |         "requested": "[4.61.2, )", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "3TDPEie+9t/NgBhoNifLFwM6nkypYUa8GUHfMnkYtovDDTQM8bsS2KEIP5tAh28BgfJZgof/KlCEKvGxz0H3Eg==", |         "contentHash": "zXPDXjTju1yOZQzi5PeNmQfBzlxv+VlCfgfZ70VqyFuhaPtFXwvbLSlwCOMoVALR/MzOR6xkiadoyVDmWDmqKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.IdentityModel.Abstractions": "6.35.0", |           "Microsoft.IdentityModel.Abstractions": "6.35.0", | ||||||
|           "System.Diagnostics.DiagnosticSource": "6.0.1" |           "System.Diagnostics.DiagnosticSource": "6.0.1" | ||||||
| @@ -256,21 +256,21 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Broker": { |       "Microsoft.Identity.Client.Broker": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[4.61.0, )", |         "requested": "[4.61.2, )", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "D5M7plyrXrQwhj/PILHLS7G3TqDg8rUO9lR+5awUSXueGxsdA2qHpeKiSR0sVfX6wxziWszOLDUR2vjFFKHPSg==", |         "contentHash": "+13pImNYfPGl04qZh3Rtq8wPfo55FScAUEio4ymPzeWJ/PE04jCsKH6ZIlrYbPCqe6t9u0lru1eZ+ho9Dsnvpw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "Microsoft.Identity.Client.NativeInterop": "0.16.0" |           "Microsoft.Identity.Client.NativeInterop": "0.16.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Extensions.Msal": { |       "Microsoft.Identity.Client.Extensions.Msal": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[4.61.0, )", |         "requested": "[4.61.2, )", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "tCX3c1bSkwH89AKjtB/oAOLlv661tOsEA8ULOCavOc6te/CKlqmMmFs8zMI58Qlv3XkbLOrswYr50IhDntiiPA==", |         "contentHash": "GCSBhzkr7/7DZDk2kYS9dfihB5ieM2CqwGAhtjuE6T1GtAqzuh62UJhMgO0dSx/PLgIYMzNHnh+pfHK8aWJQJQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.5.0" |           "System.Security.Cryptography.ProtectedData": "4.5.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
| @@ -287,8 +287,8 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "System.Linq.Async": { |       "System.Linq.Async": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
| @@ -350,8 +350,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.NativeInterop": { |       "Microsoft.Identity.Client.NativeInterop": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "0.16.0", |         "resolved": "0.16.1", | ||||||
|         "contentHash": "riaIH9tT4kRG4xKFxx+wpuo+VAZVOr1ElbNubcvpFXb82gjEmZtJRzpdiS4tdRYTVh6CUKNz9AINAptActTtBQ==" |         "contentHash": "OvY/+/AHESi24f5zOCf9kL4HXwPxXVQ3A+tMQsJvFk2DmP+sc88FYWL49zlku5q0bvx5yFvBLNpHQeRT9a6A5g==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.IdentityModel.Abstractions": { |       "Microsoft.IdentityModel.Abstractions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -375,8 +375,8 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "runtime.debian.8-x64.runtime.native.System.Security.Cryptography.OpenSsl": { |       "runtime.debian.8-x64.runtime.native.System.Security.Cryptography.OpenSsl": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -897,7 +897,7 @@ | |||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", |           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", | ||||||
|           "NuGet.Versioning": "[6.9.1, )" |           "NuGet.Versioning": "[6.10.0, )" | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -19,6 +19,7 @@ public class PublishCommandRunner(IVelopackFlowServiceClient Client) : ICommand< | |||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         await Client.UploadLatestReleaseAssetsAsync(options.Channel, options.ReleaseDirectory, options.VelopackBaseUrl, token); |         await Client.UploadLatestReleaseAssetsAsync(options.Channel, options.ReleaseDirectory, | ||||||
|  |             options.VelopackBaseUrl, options.TargetOs, token); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -5,6 +5,8 @@ namespace Velopack.Vpk.Commands.Flow; | |||||||
| #nullable enable | #nullable enable | ||||||
| public sealed class PublishOptions : VelopackServiceOptions | public sealed class PublishOptions : VelopackServiceOptions | ||||||
| { | { | ||||||
|  |     public RuntimeOs TargetOs { get; set; } | ||||||
|  | 
 | ||||||
|     public string ReleaseDirectory { get; set; } = ""; |     public string ReleaseDirectory { get; set; } = ""; | ||||||
| 
 | 
 | ||||||
|     public string? Channel { get; set; } |     public string? Channel { get; set; } | ||||||
|   | |||||||
| @@ -5,6 +5,8 @@ namespace Velopack.Vpk.Commands; | |||||||
| 
 | 
 | ||||||
| public class BaseCommand : CliCommand | public class BaseCommand : CliCommand | ||||||
| { | { | ||||||
|  |     public RuntimeOs TargetOs { get; private set; } | ||||||
|  | 
 | ||||||
|     private readonly Dictionary<CliOption, Action<ParseResult, IConfiguration>> _setters = new(); |     private readonly Dictionary<CliOption, Action<ParseResult, IConfiguration>> _setters = new(); | ||||||
| 
 | 
 | ||||||
|     private readonly Dictionary<CliOption, string> _envHelp = new(); |     private readonly Dictionary<CliOption, string> _envHelp = new(); | ||||||
| @@ -60,17 +62,18 @@ public class BaseCommand : CliCommand | |||||||
| 
 | 
 | ||||||
|     public string GetEnvVariableName(CliOption option) => _envHelp.ContainsKey(option) ? _envHelp[option] : null; |     public string GetEnvVariableName(CliOption option) => _envHelp.ContainsKey(option) ? _envHelp[option] : null; | ||||||
| 
 | 
 | ||||||
|     public virtual void SetProperties(ParseResult context, IConfiguration config) |     public virtual void SetProperties(ParseResult context, IConfiguration config, RuntimeOs targetOs) | ||||||
|     { |     { | ||||||
|  |         TargetOs = targetOs; | ||||||
|         foreach (var kvp in _setters) { |         foreach (var kvp in _setters) { | ||||||
|             kvp.Value(context, config); |             kvp.Value(context, config); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public virtual ParseResult ParseAndApply(string command, IConfiguration config = null) |     public virtual ParseResult ParseAndApply(string command, IConfiguration config = null, RuntimeOs? targetOs = null) | ||||||
|     { |     { | ||||||
|         var x = Parse(command); |         var x = Parse(command); | ||||||
|         SetProperties(x, config ?? new ConfigurationManager()); |         SetProperties(x, config ?? new ConfigurationManager(), targetOs ?? VelopackRuntimeInfo.SystemOs); | ||||||
|         return x; |         return x; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -11,11 +11,6 @@ public abstract class PlatformCommand : OutputCommand | |||||||
|         TargetRuntimeOption = AddOption<string>((v) => TargetRuntime = v, "-r", "--runtime") |         TargetRuntimeOption = AddOption<string>((v) => TargetRuntime = v, "-r", "--runtime") | ||||||
|             .SetDescription("The target runtime to build packages for.") |             .SetDescription("The target runtime to build packages for.") | ||||||
|             .SetArgumentHelpName("RID") |             .SetArgumentHelpName("RID") | ||||||
|             .SetDefault(VelopackRuntimeInfo.SystemOs.GetOsShortName()) |  | ||||||
|             .MustBeSupportedRid(); |             .MustBeSupportedRid(); | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     public RID GetRid() => RID.Parse(TargetRuntime ?? VelopackRuntimeInfo.SystemOs.GetOsShortName()); |  | ||||||
| 
 |  | ||||||
|     public RuntimeOs GetRuntimeOs() => GetRid().BaseRID; |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -5,9 +5,9 @@ namespace Velopack.Vpk.Logging; | |||||||
| public class BasicConsole : IFancyConsole | public class BasicConsole : IFancyConsole | ||||||
| { | { | ||||||
|     private readonly ILogger logger; |     private readonly ILogger logger; | ||||||
|     private readonly DefaultPromptValueFactory defaultFactory; |     private readonly VelopackDefaults defaultFactory; | ||||||
| 
 | 
 | ||||||
|     public BasicConsole(ILogger logger, DefaultPromptValueFactory defaultFactory) |     public BasicConsole(ILogger logger, VelopackDefaults defaultFactory) | ||||||
|     { |     { | ||||||
|         this.logger = logger; |         this.logger = logger; | ||||||
|         this.defaultFactory = defaultFactory; |         this.defaultFactory = defaultFactory; | ||||||
|   | |||||||
| @@ -1,5 +0,0 @@ | |||||||
| namespace Velopack.Vpk.Logging; |  | ||||||
| 
 |  | ||||||
| public record DefaultPromptValueFactory(bool DefaultPromptValue) |  | ||||||
| { |  | ||||||
| } |  | ||||||
| @@ -7,9 +7,9 @@ namespace Velopack.Vpk.Logging; | |||||||
| public class SpectreConsole : IFancyConsole | public class SpectreConsole : IFancyConsole | ||||||
| { | { | ||||||
|     private readonly ILogger logger; |     private readonly ILogger logger; | ||||||
|     private readonly DefaultPromptValueFactory defaultFactory; |     private readonly VelopackDefaults defaultFactory; | ||||||
| 
 | 
 | ||||||
|     public SpectreConsole(ILogger logger, DefaultPromptValueFactory defaultFactory) |     public SpectreConsole(ILogger logger, VelopackDefaults defaultFactory) | ||||||
|     { |     { | ||||||
|         this.logger = logger; |         this.logger = logger; | ||||||
|         this.defaultFactory = defaultFactory; |         this.defaultFactory = defaultFactory; | ||||||
|   | |||||||
| @@ -37,17 +37,32 @@ public class Program | |||||||
|         .SetRecursive(true) |         .SetRecursive(true) | ||||||
|         .SetDescription("'yes' by instead of 'no' in non-interactive prompts."); |         .SetDescription("'yes' by instead of 'no' in non-interactive prompts."); | ||||||
| 
 | 
 | ||||||
|  |     public static CliDirective WindowsDirective { get; } = new CliDirective("win") { | ||||||
|  |         Description = "Show and run Windows specific commands." | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     public static CliDirective LinuxDirective { get; } = new CliDirective("linux") { | ||||||
|  |         Description = "Show and run Linux specific commands." | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     public static CliDirective OsxDirective { get; } = new CliDirective("osx") { | ||||||
|  |         Description = "Show and run MacOS specific commands." | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|     public static readonly string INTRO |     public static readonly string INTRO | ||||||
|         = $"Velopack CLI {VelopackRuntimeInfo.VelopackDisplayVersion}, for distributing applications."; |         = $"Velopack CLI {VelopackRuntimeInfo.VelopackDisplayVersion}, for distributing applications."; | ||||||
| 
 | 
 | ||||||
|     public static async Task<int> Main(string[] args) |     public static async Task<int> Main(string[] args) | ||||||
|     { |     { | ||||||
|         CliCommand rootCommand = new CliCommand("vpk", INTRO) { |         CliRootCommand rootCommand = new CliRootCommand(INTRO); | ||||||
|             new LongHelpCommand(), |         rootCommand.Options.Clear(); // remove the default help option | ||||||
|             LegacyConsoleOption, |         rootCommand.Options.Add(new LongHelpCommand()); | ||||||
|             YesOption, |         rootCommand.Options.Add(LegacyConsoleOption); | ||||||
|             VerboseOption, |         rootCommand.Options.Add(YesOption); | ||||||
|         }; |         rootCommand.Options.Add(VerboseOption); | ||||||
|  |         rootCommand.Directives.Add(WindowsDirective); | ||||||
|  |         rootCommand.Directives.Add(LinuxDirective); | ||||||
|  |         rootCommand.Directives.Add(OsxDirective); | ||||||
| 
 | 
 | ||||||
|         rootCommand.TreatUnmatchedTokensAsErrors = false; |         rootCommand.TreatUnmatchedTokensAsErrors = false; | ||||||
|         ParseResult parseResult = rootCommand.Parse(args); |         ParseResult parseResult = rootCommand.Parse(args); | ||||||
| @@ -56,6 +71,9 @@ public class Program | |||||||
|             || Console.IsOutputRedirected |             || Console.IsOutputRedirected | ||||||
|             || Console.IsErrorRedirected; |             || Console.IsErrorRedirected; | ||||||
|         bool defaultYes = parseResult.GetValue(YesOption); |         bool defaultYes = parseResult.GetValue(YesOption); | ||||||
|  |         bool directiveWin = parseResult.GetResult(WindowsDirective) != null; | ||||||
|  |         bool directiveLinux = parseResult.GetResult(LinuxDirective) != null; | ||||||
|  |         bool directiveOsx = parseResult.GetResult(OsxDirective) != null; | ||||||
|         rootCommand.TreatUnmatchedTokensAsErrors = true; |         rootCommand.TreatUnmatchedTokensAsErrors = true; | ||||||
| 
 | 
 | ||||||
|         var builder = Host.CreateEmptyApplicationBuilder(new HostApplicationBuilderSettings { |         var builder = Host.CreateEmptyApplicationBuilder(new HostApplicationBuilderSettings { | ||||||
| @@ -66,20 +84,49 @@ public class Program | |||||||
|         }); |         }); | ||||||
| 
 | 
 | ||||||
|         SetupConfig(builder); |         SetupConfig(builder); | ||||||
|         SetupLogging(builder, verbose, legacyConsole, defaultYes); |         SetupLogging(builder, verbose, legacyConsole); | ||||||
|         SetupVelopackService(builder.Services); |         SetupVelopackService(builder.Services); | ||||||
| 
 | 
 | ||||||
|  |         RuntimeOs targetOs = VelopackRuntimeInfo.SystemOs; | ||||||
|  |         if (new bool[] { directiveWin, directiveLinux, directiveOsx }.Count(x => x) > 1) { | ||||||
|  |             throw new UserInfoException( | ||||||
|  |                 "Invalid arguments: Only one OS directive can be specified at a time: either [win], [linux], or [osx]."); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (directiveWin) { | ||||||
|  |             targetOs = RuntimeOs.Windows; | ||||||
|  |         } else if (directiveLinux) { | ||||||
|  |             targetOs = RuntimeOs.Linux; | ||||||
|  |         } else if (directiveOsx) { | ||||||
|  |             targetOs = RuntimeOs.OSX; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         builder.Services.AddSingleton(new VelopackDefaults(defaultYes, targetOs)); | ||||||
|  | 
 | ||||||
|         var host = builder.Build(); |         var host = builder.Build(); | ||||||
|         var provider = host.Services; |         var provider = host.Services; | ||||||
|  |         var logger = provider.GetRequiredService<Microsoft.Extensions.Logging.ILogger>(); | ||||||
| 
 | 
 | ||||||
|         if (VelopackRuntimeInfo.IsWindows) { |         if (targetOs != VelopackRuntimeInfo.SystemOs) { | ||||||
|  |             logger.LogInformation($"Directive enabled for cross-compiling from {VelopackRuntimeInfo.SystemOs} (current os) to {targetOs}."); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         switch (targetOs) { | ||||||
|  |         case RuntimeOs.Windows: | ||||||
|             rootCommand.AddCommand<WindowsPackCommand, WindowsPackCommandRunner, WindowsPackOptions>(provider); |             rootCommand.AddCommand<WindowsPackCommand, WindowsPackCommandRunner, WindowsPackOptions>(provider); | ||||||
|         } else if (VelopackRuntimeInfo.IsOSX) { |             break; | ||||||
|             rootCommand.AddCommand<OsxBundleCommand, OsxBundleCommandRunner, OsxBundleOptions>(provider); |         case RuntimeOs.Linux: | ||||||
|             rootCommand.AddCommand<OsxPackCommand, OsxPackCommandRunner, OsxPackOptions>(provider); |  | ||||||
|         } else if (VelopackRuntimeInfo.IsLinux) { |  | ||||||
|             rootCommand.AddCommand<LinuxPackCommand, LinuxPackCommandRunner, LinuxPackOptions>(provider); |             rootCommand.AddCommand<LinuxPackCommand, LinuxPackCommandRunner, LinuxPackOptions>(provider); | ||||||
|         } else { |             break; | ||||||
|  |         case RuntimeOs.OSX: | ||||||
|  |             if (VelopackRuntimeInfo.IsOSX) { | ||||||
|  |                 rootCommand.AddCommand<OsxBundleCommand, OsxBundleCommandRunner, OsxBundleOptions>(provider); | ||||||
|  |                 rootCommand.AddCommand<OsxPackCommand, OsxPackCommandRunner, OsxPackOptions>(provider); | ||||||
|  |             } else { | ||||||
|  |                 throw new NotSupportedException($"Cross-compiling from {VelopackRuntimeInfo.SystemOs} to MacOS is not supported."); | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  |         default: | ||||||
|             throw new NotSupportedException("Unsupported OS platform: " + VelopackRuntimeInfo.SystemOs.GetOsLongName()); |             throw new NotSupportedException("Unsupported OS platform: " + VelopackRuntimeInfo.SystemOs.GetOsLongName()); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
| @@ -122,15 +169,13 @@ public class Program | |||||||
|         builder.Services.AddTransient(s => s.GetService<ILoggerFactory>().CreateLogger("vpk")); |         builder.Services.AddTransient(s => s.GetService<ILoggerFactory>().CreateLogger("vpk")); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private static void SetupLogging(IHostApplicationBuilder builder, bool verbose, bool legacyConsole, bool defaultPromptValue) |     private static void SetupLogging(IHostApplicationBuilder builder, bool verbose, bool legacyConsole) | ||||||
|     { |     { | ||||||
|         var conf = new LoggerConfiguration() |         var conf = new LoggerConfiguration() | ||||||
|             .MinimumLevel.Is(verbose ? LogEventLevel.Debug : LogEventLevel.Information) |             .MinimumLevel.Is(verbose ? LogEventLevel.Debug : LogEventLevel.Information) | ||||||
|             .MinimumLevel.Override("Microsoft", LogEventLevel.Warning) |             .MinimumLevel.Override("Microsoft", LogEventLevel.Warning) | ||||||
|             .MinimumLevel.Override("System", LogEventLevel.Warning); |             .MinimumLevel.Override("System", LogEventLevel.Warning); | ||||||
| 
 | 
 | ||||||
|         builder.Services.AddSingleton(new DefaultPromptValueFactory(defaultPromptValue)); |  | ||||||
| 
 |  | ||||||
|         if (legacyConsole) { |         if (legacyConsole) { | ||||||
|             // spectre can have issues with redirected output, so we disable it. |             // spectre can have issues with redirected output, so we disable it. | ||||||
|             builder.Services.AddSingleton<IFancyConsole, BasicConsole>(); |             builder.Services.AddSingleton<IFancyConsole, BasicConsole>(); | ||||||
| @@ -195,12 +240,15 @@ public static class ProgramCommandExtensions | |||||||
|         var command = new TCli(); |         var command = new TCli(); | ||||||
|         command.SetAction(async (ctx, token) => { |         command.SetAction(async (ctx, token) => { | ||||||
|             var logger = provider.GetRequiredService<Microsoft.Extensions.Logging.ILogger>(); |             var logger = provider.GetRequiredService<Microsoft.Extensions.Logging.ILogger>(); | ||||||
|  |             var console = provider.GetRequiredService<IFancyConsole>(); | ||||||
|             var config = provider.GetRequiredService<IConfiguration>(); |             var config = provider.GetRequiredService<IConfiguration>(); | ||||||
|  |             var defaults = provider.GetRequiredService<VelopackDefaults>(); | ||||||
|  | 
 | ||||||
|             logger.LogInformation($"[bold]{Program.INTRO}[/]"); |             logger.LogInformation($"[bold]{Program.INTRO}[/]"); | ||||||
|             var updateCheck = new UpdateChecker(logger); |             var updateCheck = new UpdateChecker(logger); | ||||||
|             await updateCheck.CheckForUpdates(); |             await updateCheck.CheckForUpdates(); | ||||||
| 
 | 
 | ||||||
|             command.SetProperties(ctx, config); |             command.SetProperties(ctx, config, defaults.TargetOs); | ||||||
|             var options = OptionMapper.Map<TOpt>(command); |             var options = OptionMapper.Map<TOpt>(command); | ||||||
| 
 | 
 | ||||||
|             try { |             try { | ||||||
| @@ -210,10 +258,10 @@ public static class ProgramCommandExtensions | |||||||
|                 return 0; |                 return 0; | ||||||
|             } catch (Exception ex) when (ex is ProcessFailedException or UserInfoException) { |             } catch (Exception ex) when (ex is ProcessFailedException or UserInfoException) { | ||||||
|                 // some exceptions are just user info / user error, so don't need a stack trace. |                 // some exceptions are just user info / user error, so don't need a stack trace. | ||||||
|                 logger.Fatal($"[bold orange3]{ex.Message}[/]"); |                 logger.Fatal($"[bold orange3]{console.EscapeMarkup(ex.Message)}[/]"); | ||||||
|                 return -1; |                 return -1; | ||||||
|             } catch (Exception ex) { |             } catch (Exception ex) { | ||||||
|                 logger.Fatal(ex, $"Command {typeof(TCli).Name} had an exception."); |                 logger.Fatal(ex); | ||||||
|                 return -1; |                 return -1; | ||||||
|             } |             } | ||||||
|         }); |         }); | ||||||
|   | |||||||
| @@ -23,7 +23,7 @@ | |||||||
|     <PackageReference Include="Serilog.Extensions.Hosting" Version="8.0.0" /> |     <PackageReference Include="Serilog.Extensions.Hosting" Version="8.0.0" /> | ||||||
|     <PackageReference Include="Serilog.Expressions" Version="4.0.0" /> |     <PackageReference Include="Serilog.Expressions" Version="4.0.0" /> | ||||||
|     <PackageReference Include="System.CommandLine" Version="2.0.0-beta4.23407.1" /> |     <PackageReference Include="System.CommandLine" Version="2.0.0-beta4.23407.1" /> | ||||||
|     <PackageReference Include="NuGet.Protocol" Version="6.9.1" /> |     <PackageReference Include="NuGet.Protocol" Version="6.10.0" /> | ||||||
|     <PackageReference Include="Spectre.Console" Version="0.49.1" /> |     <PackageReference Include="Spectre.Console" Version="0.49.1" /> | ||||||
|     <PackageReference Include="Riok.Mapperly" Version="3.5.1" /> |     <PackageReference Include="Riok.Mapperly" Version="3.5.1" /> | ||||||
|     <PackageReference Include="Humanizer.Core" Version="2.14.1" /> |     <PackageReference Include="Humanizer.Core" Version="2.14.1" /> | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								src/Velopack.Vpk/VelopackDefaults.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/Velopack.Vpk/VelopackDefaults.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | namespace Velopack.Vpk; | ||||||
|  | 
 | ||||||
|  | public record VelopackDefaults | ||||||
|  | { | ||||||
|  |     public bool DefaultPromptValue { get; } | ||||||
|  |     public RuntimeOs TargetOs { get; } | ||||||
|  | 
 | ||||||
|  |     public VelopackDefaults(bool defaultPromptValue) | ||||||
|  |         : this(defaultPromptValue, VelopackRuntimeInfo.SystemOs) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public VelopackDefaults(bool defaultPromptValue, RuntimeOs targetOs) | ||||||
|  |     { | ||||||
|  |         DefaultPromptValue = defaultPromptValue; | ||||||
|  |         TargetOs = targetOs; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -64,16 +64,16 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "NuGet.Protocol": { |       "NuGet.Protocol": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[6.9.1, )", |         "requested": "[6.10.0, )", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "h3bdjqUY4jvwM02D/7QM4FR8x/bbf4Pyjrc1Etw7an2OrWKPUSx0vJPdapKzioxIw7GXl/aVUM/DCeIc3S9brA==", |         "contentHash": "/3r1avHk5IlqoGlXlb1ezNgtTIQyMTR5DAgh1WBcllivpbADpM9rvsFeemvcnndaFuQkEgc7a2egQZEnOK15ew==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "NuGet.Packaging": "6.9.1" |           "NuGet.Packaging": "6.10.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Riok.Mapperly": { |       "Riok.Mapperly": { | ||||||
| @@ -169,15 +169,15 @@ | |||||||
|       }, |       }, | ||||||
|       "AWSSDK.Core": { |       "AWSSDK.Core": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "3.7.304.1", |         "resolved": "3.7.304.10", | ||||||
|         "contentHash": "O/gyE5ptF5zEc3QDk3JI3+AOgBfRfmg6eLH5z3x7hUPmV1Wxu0V4Fm86FSyT6czRviMmQGY0q0SVMbWRbP4vDA==" |         "contentHash": "0CXnPzoM+KYXODm2bvRW8eYs7ic2VLop45sphL8FBfvxHfBK/3OJgpxEE2InSdWS1Iby+6KXpz3NplJJY5+Y1A==" | ||||||
|       }, |       }, | ||||||
|       "AWSSDK.S3": { |       "AWSSDK.S3": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "3.7.308", |         "resolved": "3.7.308.8", | ||||||
|         "contentHash": "U4LWi1yTKVK6IFHWQc4anKPJKXj9dPN+dT1oXto4ndXXlr+DVzm9dULkquyXBjAHlr6jf339ojuYlE6t9Mk6pQ==", |         "contentHash": "lUQgJsj9n/sJo1CRQFsiB2Gqh3rC21I4e5CWR1b+oYBnc9m8LIsGewctX6YQRzGzuv7Im7iIZo7XoirmYhhqEA==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "AWSSDK.Core": "[3.7.304.1, 4.0.0)" |           "AWSSDK.Core": "[3.7.304.10, 4.0.0)" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Azure.Core": { |       "Azure.Core": { | ||||||
| @@ -493,8 +493,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client": { |       "Microsoft.Identity.Client": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "3TDPEie+9t/NgBhoNifLFwM6nkypYUa8GUHfMnkYtovDDTQM8bsS2KEIP5tAh28BgfJZgof/KlCEKvGxz0H3Eg==", |         "contentHash": "zXPDXjTju1yOZQzi5PeNmQfBzlxv+VlCfgfZ70VqyFuhaPtFXwvbLSlwCOMoVALR/MzOR6xkiadoyVDmWDmqKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.IdentityModel.Abstractions": "6.35.0", |           "Microsoft.IdentityModel.Abstractions": "6.35.0", | ||||||
|           "System.Diagnostics.DiagnosticSource": "6.0.1" |           "System.Diagnostics.DiagnosticSource": "6.0.1" | ||||||
| @@ -502,26 +502,26 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Broker": { |       "Microsoft.Identity.Client.Broker": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "D5M7plyrXrQwhj/PILHLS7G3TqDg8rUO9lR+5awUSXueGxsdA2qHpeKiSR0sVfX6wxziWszOLDUR2vjFFKHPSg==", |         "contentHash": "+13pImNYfPGl04qZh3Rtq8wPfo55FScAUEio4ymPzeWJ/PE04jCsKH6ZIlrYbPCqe6t9u0lru1eZ+ho9Dsnvpw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "Microsoft.Identity.Client.NativeInterop": "0.16.0" |           "Microsoft.Identity.Client.NativeInterop": "0.16.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Extensions.Msal": { |       "Microsoft.Identity.Client.Extensions.Msal": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "tCX3c1bSkwH89AKjtB/oAOLlv661tOsEA8ULOCavOc6te/CKlqmMmFs8zMI58Qlv3XkbLOrswYr50IhDntiiPA==", |         "contentHash": "GCSBhzkr7/7DZDk2kYS9dfihB5ieM2CqwGAhtjuE6T1GtAqzuh62UJhMgO0dSx/PLgIYMzNHnh+pfHK8aWJQJQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.5.0" |           "System.Security.Cryptography.ProtectedData": "4.5.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.NativeInterop": { |       "Microsoft.Identity.Client.NativeInterop": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "0.16.0", |         "resolved": "0.16.1", | ||||||
|         "contentHash": "riaIH9tT4kRG4xKFxx+wpuo+VAZVOr1ElbNubcvpFXb82gjEmZtJRzpdiS4tdRYTVh6CUKNz9AINAptActTtBQ==" |         "contentHash": "OvY/+/AHESi24f5zOCf9kL4HXwPxXVQ3A+tMQsJvFk2DmP+sc88FYWL49zlku5q0bvx5yFvBLNpHQeRT9a6A5g==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.IdentityModel.Abstractions": { |       "Microsoft.IdentityModel.Abstractions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -530,8 +530,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.NETCore.Platforms": { |       "Microsoft.NETCore.Platforms": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "1.1.1", |         "resolved": "5.0.0", | ||||||
|         "contentHash": "TMBuzAHpTenGbGgk0SMTwyEkyijY/Eae4ZGsFNYJvAr/LDn1ku3Etp3FPxChmDp5HHF3kzJuoaa08N0xjqAJfQ==" |         "contentHash": "VyPlqzH2wavqquTcYpkIIAQ6WdenuKoFN0BdYBbCWsclXacSOHNQn66Gt4z5NBqEYW0FAPm5rlvki9ZiCij5xQ==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.NETCore.Targets": { |       "Microsoft.NETCore.Targets": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -550,41 +550,41 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Common": { |       "NuGet.Common": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "FbuWZBjQ1NJXBDqCwSddN2yvw3Plq3sTCIh0nc66Hu8jrNr+BOaxlKZv78jvJ+pSy8BvurYOdF9sl9KoORjrtg==", |         "contentHash": "ujuzfDwUylDALwZmqRi7I5Jx4E9/8vR2c0Hq8zRj8zCkR8KarSp84WPJ2uE/qwAOjpBYGek06wWgGJ3ABQ/WxA==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "NuGet.Frameworks": "6.9.1" |           "NuGet.Frameworks": "6.10.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "NuGet.Configuration": { |       "NuGet.Configuration": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "GM06pcUzWdNsizeGciqCjAhryfI1F/rQPETLDF+8pDRgzVpA+wKAR01/4aFU+IXzugnQ9LqOb5YyCRuR1OVZiQ==", |         "contentHash": "PbtCCdFC/K3GG3fdstIIMVraiXcxC1IgnzqayowckSlIg2xSxvcqAWd5Z7G1V6st5JzJi9THSnvmy/kchwh80A==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "NuGet.Common": "6.9.1", |           "NuGet.Common": "6.10.0", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.4.0" |           "System.Security.Cryptography.ProtectedData": "4.4.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "NuGet.Frameworks": { |       "NuGet.Frameworks": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "DaKh3lenPUvzGccPkbI97BIvA27z+/UsL3ankfoZlX/4vBVDK5N1sheFTQ+GuJf+IgSzsJz/A21SPUpQLHwUtA==" |         "contentHash": "bKaC87Q8rxK7ozFN9Eyo0YVUmd4r2s8pbNxHI7sHRqL16OAP0yEdCU5wpDkG9cKLHpZCahgoQUfAVC0UadVU+A==" | ||||||
|       }, |       }, | ||||||
|       "NuGet.Packaging": { |       "NuGet.Packaging": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "6FyasOxKInCELJ+pGy8f17ub9st6ofFeNcBNTo7CRiPJlxyhJfKGKNpfe3HHYwvnZhc3Vdfr0kSR+f1BVGDuTA==", |         "contentHash": "QSznYvf+HejToKt9zOB2z9F1C4R01lnrUUcJ+WxL1ihazTBfrXWPCCAQDtQFSlJlibiJBn/GEoq1gMeaeSo2qQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Newtonsoft.Json": "13.0.3", |           "Newtonsoft.Json": "13.0.3", | ||||||
|           "NuGet.Configuration": "6.9.1", |           "NuGet.Configuration": "6.10.0", | ||||||
|           "NuGet.Versioning": "6.9.1", |           "NuGet.Versioning": "6.10.0", | ||||||
|           "System.Security.Cryptography.Pkcs": "6.0.4" |           "System.Security.Cryptography.Pkcs": "6.0.4" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "Octokit": { |       "Octokit": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -703,6 +703,20 @@ | |||||||
|           "Serilog": "3.1.1" |           "Serilog": "3.1.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|  |       "SharpZipLib": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "1.4.2", | ||||||
|  |         "contentHash": "yjj+3zgz8zgXpiiC3ZdF/iyTBbz2fFvMxZFEBPUcwZjIvXOf37Ylm+K58hqMfIBt5JgU/Z2uoUS67JmTLe973A==" | ||||||
|  |       }, | ||||||
|  |       "SixLabors.ImageSharp": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "2.1.8", | ||||||
|  |         "contentHash": "ML1++vactR5xMW36wHA2nAX3vc6VpR62qYnpopQdLNGbP8BAJ/ckv1IrGIIFRUEQ0JS9EgWr1y1gEa/81f+HaA==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Runtime.CompilerServices.Unsafe": "5.0.0", | ||||||
|  |           "System.Text.Encoding.CodePages": "5.0.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "System.ClientModel": { |       "System.ClientModel": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "1.0.0", |         "resolved": "1.0.0", | ||||||
| @@ -1193,6 +1207,14 @@ | |||||||
|           "System.Runtime": "4.3.0" |           "System.Runtime": "4.3.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|  |       "System.Text.Encoding.CodePages": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "5.0.0", | ||||||
|  |         "contentHash": "NyscU59xX6Uo91qvhOs2Ccho3AR2TnZPomo1Z0K6YpyztBPM/A5VbkzOO19sy3A3i1TtEnTxA7bCe3Us+r5MWg==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "Microsoft.NETCore.Platforms": "5.0.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "System.Text.Encodings.Web": { |       "System.Text.Encodings.Web": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "8.0.0", |         "resolved": "8.0.0", | ||||||
| @@ -1238,37 +1260,41 @@ | |||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", |           "Microsoft.Extensions.Logging.Abstractions": "[6.0.0, )", | ||||||
|           "NuGet.Versioning": "[6.9.1, )" |           "NuGet.Versioning": "[6.10.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.deployment": { |       "velopack.deployment": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "AWSSDK.S3": "[3.7.308, )", |           "AWSSDK.S3": "[3.7.308.8, )", | ||||||
|           "Azure.Storage.Blobs": "[12.20.0, )", |           "Azure.Storage.Blobs": "[12.20.0, )", | ||||||
|           "Octokit": "[11.0.1, )", |           "Octokit": "[11.0.1, )", | ||||||
|           "Velopack.Packaging": "[1.0.0, )" |           "Velopack.Packaging": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|  |       "velopack.icolib": { | ||||||
|  |         "type": "Project", | ||||||
|  |         "dependencies": { | ||||||
|  |           "SixLabors.ImageSharp": "[2.1.8, )" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "velopack.packaging": { |       "velopack.packaging": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Markdig": "[0.37.0, )", |           "Markdig": "[0.37.0, )", | ||||||
|           "Microsoft.Identity.Client": "[4.61.0, )", |           "Microsoft.Identity.Client": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Broker": "[4.61.0, )", |           "Microsoft.Identity.Client.Broker": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.0, )", |           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.2, )", | ||||||
|           "System.Linq.Async": "[6.0.1, )", |           "System.Linq.Async": "[6.0.1, )", | ||||||
|           "System.Net.Http": "[4.3.4, )", |           "System.Net.Http": "[4.3.4, )", | ||||||
|           "Velopack": "[1.0.0, )" |           "Velopack": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.packaging.hostmodel": { |  | ||||||
|         "type": "Project" |  | ||||||
|       }, |  | ||||||
|       "velopack.packaging.unix": { |       "velopack.packaging.unix": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "ELFSharp": "[2.17.3, )", |           "ELFSharp": "[2.17.3, )", | ||||||
|  |           "SharpZipLib": "[1.4.2, )", | ||||||
|           "Velopack.Packaging": "[1.0.0, )" |           "Velopack.Packaging": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
| @@ -1277,8 +1303,8 @@ | |||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "AsmResolver.DotNet": "[5.5.1, )", |           "AsmResolver.DotNet": "[5.5.1, )", | ||||||
|           "AsmResolver.PE.Win32Resources": "[5.5.1, )", |           "AsmResolver.PE.Win32Resources": "[5.5.1, )", | ||||||
|           "Velopack.Packaging": "[1.0.0, )", |           "Velopack.IcoLib": "[1.1.1, )", | ||||||
|           "Velopack.Packaging.HostModel": "[1.0.0, )" |           "Velopack.Packaging": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
| @@ -1346,16 +1372,16 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "NuGet.Protocol": { |       "NuGet.Protocol": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[6.9.1, )", |         "requested": "[6.10.0, )", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "h3bdjqUY4jvwM02D/7QM4FR8x/bbf4Pyjrc1Etw7an2OrWKPUSx0vJPdapKzioxIw7GXl/aVUM/DCeIc3S9brA==", |         "contentHash": "/3r1avHk5IlqoGlXlb1ezNgtTIQyMTR5DAgh1WBcllivpbADpM9rvsFeemvcnndaFuQkEgc7a2egQZEnOK15ew==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "NuGet.Packaging": "6.9.1" |           "NuGet.Packaging": "6.10.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Riok.Mapperly": { |       "Riok.Mapperly": { | ||||||
| @@ -1448,15 +1474,15 @@ | |||||||
|       }, |       }, | ||||||
|       "AWSSDK.Core": { |       "AWSSDK.Core": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "3.7.304.1", |         "resolved": "3.7.304.10", | ||||||
|         "contentHash": "O/gyE5ptF5zEc3QDk3JI3+AOgBfRfmg6eLH5z3x7hUPmV1Wxu0V4Fm86FSyT6czRviMmQGY0q0SVMbWRbP4vDA==" |         "contentHash": "0CXnPzoM+KYXODm2bvRW8eYs7ic2VLop45sphL8FBfvxHfBK/3OJgpxEE2InSdWS1Iby+6KXpz3NplJJY5+Y1A==" | ||||||
|       }, |       }, | ||||||
|       "AWSSDK.S3": { |       "AWSSDK.S3": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "3.7.308", |         "resolved": "3.7.308.8", | ||||||
|         "contentHash": "U4LWi1yTKVK6IFHWQc4anKPJKXj9dPN+dT1oXto4ndXXlr+DVzm9dULkquyXBjAHlr6jf339ojuYlE6t9Mk6pQ==", |         "contentHash": "lUQgJsj9n/sJo1CRQFsiB2Gqh3rC21I4e5CWR1b+oYBnc9m8LIsGewctX6YQRzGzuv7Im7iIZo7XoirmYhhqEA==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "AWSSDK.Core": "[3.7.304.1, 4.0.0)" |           "AWSSDK.Core": "[3.7.304.10, 4.0.0)" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Azure.Core": { |       "Azure.Core": { | ||||||
| @@ -1767,8 +1793,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client": { |       "Microsoft.Identity.Client": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "3TDPEie+9t/NgBhoNifLFwM6nkypYUa8GUHfMnkYtovDDTQM8bsS2KEIP5tAh28BgfJZgof/KlCEKvGxz0H3Eg==", |         "contentHash": "zXPDXjTju1yOZQzi5PeNmQfBzlxv+VlCfgfZ70VqyFuhaPtFXwvbLSlwCOMoVALR/MzOR6xkiadoyVDmWDmqKw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.IdentityModel.Abstractions": "6.35.0", |           "Microsoft.IdentityModel.Abstractions": "6.35.0", | ||||||
|           "System.Diagnostics.DiagnosticSource": "6.0.1" |           "System.Diagnostics.DiagnosticSource": "6.0.1" | ||||||
| @@ -1776,26 +1802,26 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Broker": { |       "Microsoft.Identity.Client.Broker": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "D5M7plyrXrQwhj/PILHLS7G3TqDg8rUO9lR+5awUSXueGxsdA2qHpeKiSR0sVfX6wxziWszOLDUR2vjFFKHPSg==", |         "contentHash": "+13pImNYfPGl04qZh3Rtq8wPfo55FScAUEio4ymPzeWJ/PE04jCsKH6ZIlrYbPCqe6t9u0lru1eZ+ho9Dsnvpw==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "Microsoft.Identity.Client.NativeInterop": "0.16.0" |           "Microsoft.Identity.Client.NativeInterop": "0.16.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.Extensions.Msal": { |       "Microsoft.Identity.Client.Extensions.Msal": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.61.0", |         "resolved": "4.61.2", | ||||||
|         "contentHash": "tCX3c1bSkwH89AKjtB/oAOLlv661tOsEA8ULOCavOc6te/CKlqmMmFs8zMI58Qlv3XkbLOrswYr50IhDntiiPA==", |         "contentHash": "GCSBhzkr7/7DZDk2kYS9dfihB5ieM2CqwGAhtjuE6T1GtAqzuh62UJhMgO0dSx/PLgIYMzNHnh+pfHK8aWJQJQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Identity.Client": "4.61.0", |           "Microsoft.Identity.Client": "4.61.2", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.5.0" |           "System.Security.Cryptography.ProtectedData": "4.5.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Identity.Client.NativeInterop": { |       "Microsoft.Identity.Client.NativeInterop": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "0.16.0", |         "resolved": "0.16.1", | ||||||
|         "contentHash": "riaIH9tT4kRG4xKFxx+wpuo+VAZVOr1ElbNubcvpFXb82gjEmZtJRzpdiS4tdRYTVh6CUKNz9AINAptActTtBQ==" |         "contentHash": "OvY/+/AHESi24f5zOCf9kL4HXwPxXVQ3A+tMQsJvFk2DmP+sc88FYWL49zlku5q0bvx5yFvBLNpHQeRT9a6A5g==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.IdentityModel.Abstractions": { |       "Microsoft.IdentityModel.Abstractions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -1804,8 +1830,8 @@ | |||||||
|       }, |       }, | ||||||
|       "Microsoft.NETCore.Platforms": { |       "Microsoft.NETCore.Platforms": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "1.1.1", |         "resolved": "5.0.0", | ||||||
|         "contentHash": "TMBuzAHpTenGbGgk0SMTwyEkyijY/Eae4ZGsFNYJvAr/LDn1ku3Etp3FPxChmDp5HHF3kzJuoaa08N0xjqAJfQ==" |         "contentHash": "VyPlqzH2wavqquTcYpkIIAQ6WdenuKoFN0BdYBbCWsclXacSOHNQn66Gt4z5NBqEYW0FAPm5rlvki9ZiCij5xQ==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.NETCore.Targets": { |       "Microsoft.NETCore.Targets": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -1824,41 +1850,41 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Common": { |       "NuGet.Common": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "FbuWZBjQ1NJXBDqCwSddN2yvw3Plq3sTCIh0nc66Hu8jrNr+BOaxlKZv78jvJ+pSy8BvurYOdF9sl9KoORjrtg==", |         "contentHash": "ujuzfDwUylDALwZmqRi7I5Jx4E9/8vR2c0Hq8zRj8zCkR8KarSp84WPJ2uE/qwAOjpBYGek06wWgGJ3ABQ/WxA==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "NuGet.Frameworks": "6.9.1" |           "NuGet.Frameworks": "6.10.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "NuGet.Configuration": { |       "NuGet.Configuration": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "GM06pcUzWdNsizeGciqCjAhryfI1F/rQPETLDF+8pDRgzVpA+wKAR01/4aFU+IXzugnQ9LqOb5YyCRuR1OVZiQ==", |         "contentHash": "PbtCCdFC/K3GG3fdstIIMVraiXcxC1IgnzqayowckSlIg2xSxvcqAWd5Z7G1V6st5JzJi9THSnvmy/kchwh80A==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "NuGet.Common": "6.9.1", |           "NuGet.Common": "6.10.0", | ||||||
|           "System.Security.Cryptography.ProtectedData": "4.4.0" |           "System.Security.Cryptography.ProtectedData": "4.4.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "NuGet.Frameworks": { |       "NuGet.Frameworks": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "DaKh3lenPUvzGccPkbI97BIvA27z+/UsL3ankfoZlX/4vBVDK5N1sheFTQ+GuJf+IgSzsJz/A21SPUpQLHwUtA==" |         "contentHash": "bKaC87Q8rxK7ozFN9Eyo0YVUmd4r2s8pbNxHI7sHRqL16OAP0yEdCU5wpDkG9cKLHpZCahgoQUfAVC0UadVU+A==" | ||||||
|       }, |       }, | ||||||
|       "NuGet.Packaging": { |       "NuGet.Packaging": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "6FyasOxKInCELJ+pGy8f17ub9st6ofFeNcBNTo7CRiPJlxyhJfKGKNpfe3HHYwvnZhc3Vdfr0kSR+f1BVGDuTA==", |         "contentHash": "QSznYvf+HejToKt9zOB2z9F1C4R01lnrUUcJ+WxL1ihazTBfrXWPCCAQDtQFSlJlibiJBn/GEoq1gMeaeSo2qQ==", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Newtonsoft.Json": "13.0.3", |           "Newtonsoft.Json": "13.0.3", | ||||||
|           "NuGet.Configuration": "6.9.1", |           "NuGet.Configuration": "6.10.0", | ||||||
|           "NuGet.Versioning": "6.9.1", |           "NuGet.Versioning": "6.10.0", | ||||||
|           "System.Security.Cryptography.Pkcs": "6.0.4" |           "System.Security.Cryptography.Pkcs": "6.0.4" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "Octokit": { |       "Octokit": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -1977,6 +2003,20 @@ | |||||||
|           "Serilog": "3.1.1" |           "Serilog": "3.1.1" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|  |       "SharpZipLib": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "1.4.2", | ||||||
|  |         "contentHash": "yjj+3zgz8zgXpiiC3ZdF/iyTBbz2fFvMxZFEBPUcwZjIvXOf37Ylm+K58hqMfIBt5JgU/Z2uoUS67JmTLe973A==" | ||||||
|  |       }, | ||||||
|  |       "SixLabors.ImageSharp": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "2.1.8", | ||||||
|  |         "contentHash": "ML1++vactR5xMW36wHA2nAX3vc6VpR62qYnpopQdLNGbP8BAJ/ckv1IrGIIFRUEQ0JS9EgWr1y1gEa/81f+HaA==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "System.Runtime.CompilerServices.Unsafe": "5.0.0", | ||||||
|  |           "System.Text.Encoding.CodePages": "5.0.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "System.ClientModel": { |       "System.ClientModel": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "1.0.0", |         "resolved": "1.0.0", | ||||||
| @@ -2243,6 +2283,11 @@ | |||||||
|           "Microsoft.NETCore.Targets": "1.1.0" |           "Microsoft.NETCore.Targets": "1.1.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|  |       "System.Runtime.CompilerServices.Unsafe": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "5.0.0", | ||||||
|  |         "contentHash": "ZD9TMpsmYJLrxbbmdvhwt9YEgG5WntEnZ/d1eH8JBX9LBp+Ju8BSBhUGbZMNVHHomWo2KVImJhTDl2hIgw/6MA==" | ||||||
|  |       }, | ||||||
|       "System.Runtime.Extensions": { |       "System.Runtime.Extensions": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "4.3.0", |         "resolved": "4.3.0", | ||||||
| @@ -2454,6 +2499,14 @@ | |||||||
|           "System.Runtime": "4.3.0" |           "System.Runtime": "4.3.0" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|  |       "System.Text.Encoding.CodePages": { | ||||||
|  |         "type": "Transitive", | ||||||
|  |         "resolved": "5.0.0", | ||||||
|  |         "contentHash": "NyscU59xX6Uo91qvhOs2Ccho3AR2TnZPomo1Z0K6YpyztBPM/A5VbkzOO19sy3A3i1TtEnTxA7bCe3Us+r5MWg==", | ||||||
|  |         "dependencies": { | ||||||
|  |           "Microsoft.NETCore.Platforms": "5.0.0" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "System.Text.Encodings.Web": { |       "System.Text.Encodings.Web": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|         "resolved": "8.0.0", |         "resolved": "8.0.0", | ||||||
| @@ -2495,37 +2548,41 @@ | |||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Microsoft.Extensions.Logging.Abstractions": "[8.0.0, )", |           "Microsoft.Extensions.Logging.Abstractions": "[8.0.0, )", | ||||||
|           "NuGet.Versioning": "[6.9.1, )" |           "NuGet.Versioning": "[6.10.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.deployment": { |       "velopack.deployment": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "AWSSDK.S3": "[3.7.308, )", |           "AWSSDK.S3": "[3.7.308.8, )", | ||||||
|           "Azure.Storage.Blobs": "[12.20.0, )", |           "Azure.Storage.Blobs": "[12.20.0, )", | ||||||
|           "Octokit": "[11.0.1, )", |           "Octokit": "[11.0.1, )", | ||||||
|           "Velopack.Packaging": "[1.0.0, )" |           "Velopack.Packaging": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|  |       "velopack.icolib": { | ||||||
|  |         "type": "Project", | ||||||
|  |         "dependencies": { | ||||||
|  |           "SixLabors.ImageSharp": "[2.1.8, )" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|       "velopack.packaging": { |       "velopack.packaging": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "Markdig": "[0.37.0, )", |           "Markdig": "[0.37.0, )", | ||||||
|           "Microsoft.Identity.Client": "[4.61.0, )", |           "Microsoft.Identity.Client": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Broker": "[4.61.0, )", |           "Microsoft.Identity.Client.Broker": "[4.61.2, )", | ||||||
|           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.0, )", |           "Microsoft.Identity.Client.Extensions.Msal": "[4.61.2, )", | ||||||
|           "System.Linq.Async": "[6.0.1, )", |           "System.Linq.Async": "[6.0.1, )", | ||||||
|           "System.Net.Http": "[4.3.4, )", |           "System.Net.Http": "[4.3.4, )", | ||||||
|           "Velopack": "[1.0.0, )" |           "Velopack": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|       "velopack.packaging.hostmodel": { |  | ||||||
|         "type": "Project" |  | ||||||
|       }, |  | ||||||
|       "velopack.packaging.unix": { |       "velopack.packaging.unix": { | ||||||
|         "type": "Project", |         "type": "Project", | ||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "ELFSharp": "[2.17.3, )", |           "ELFSharp": "[2.17.3, )", | ||||||
|  |           "SharpZipLib": "[1.4.2, )", | ||||||
|           "Velopack.Packaging": "[1.0.0, )" |           "Velopack.Packaging": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
| @@ -2534,8 +2591,8 @@ | |||||||
|         "dependencies": { |         "dependencies": { | ||||||
|           "AsmResolver.DotNet": "[5.5.1, )", |           "AsmResolver.DotNet": "[5.5.1, )", | ||||||
|           "AsmResolver.PE.Win32Resources": "[5.5.1, )", |           "AsmResolver.PE.Win32Resources": "[5.5.1, )", | ||||||
|           "Velopack.Packaging": "[1.0.0, )", |           "Velopack.IcoLib": "[1.1.1, )", | ||||||
|           "Velopack.Packaging.HostModel": "[1.0.0, )" |           "Velopack.Packaging": "[1.0.0, )" | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -99,7 +99,7 @@ namespace Velopack.Compression | |||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         private static char s_pathSeperator = '/'; |         private static char s_pathSeperator = '/'; | ||||||
|         private static readonly DateTime ZipFormatMinDate = new DateTime(1980, 1, 1, 0, 0, 0, DateTimeKind.Utc); |         public static readonly DateTime ZipFormatMinDate = new DateTime(1980, 1, 1, 0, 0, 0, DateTimeKind.Utc); | ||||||
| 
 | 
 | ||||||
|         private static async Task DeterministicCreateFromDirectoryAsync(string sourceDirectoryName, string destinationArchiveFileName, CompressionLevel compressionLevel, |         private static async Task DeterministicCreateFromDirectoryAsync(string sourceDirectoryName, string destinationArchiveFileName, CompressionLevel compressionLevel, | ||||||
|             Action<int> progress, CancellationToken cancelToken) |             Action<int> progress, CancellationToken cancelToken) | ||||||
|   | |||||||
| @@ -16,7 +16,7 @@ | |||||||
|   </PropertyGroup> |   </PropertyGroup> | ||||||
|  |  | ||||||
|   <ItemGroup> |   <ItemGroup> | ||||||
|     <PackageReference Include="NuGet.Versioning" Version="6.9.1" /> |     <PackageReference Include="NuGet.Versioning" Version="6.10.0" /> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
|   <ItemGroup Condition=" $(TargetFramework.StartsWith('net4')) "> |   <ItemGroup Condition=" $(TargetFramework.StartsWith('net4')) "> | ||||||
|   | |||||||
| @@ -30,8 +30,8 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "Newtonsoft.Json": { |       "Newtonsoft.Json": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
| @@ -41,9 +41,9 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[6.9.1, )", |         "requested": "[6.10.0, )", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Build.Tasks.Git": { |       "Microsoft.Build.Tasks.Git": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -81,8 +81,8 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "Newtonsoft.Json": { |       "Newtonsoft.Json": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
| @@ -92,9 +92,9 @@ | |||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[6.9.1, )", |         "requested": "[6.10.0, )", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Build.Tasks.Git": { |       "Microsoft.Build.Tasks.Git": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -127,14 +127,14 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[6.9.1, )", |         "requested": "[6.10.0, )", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Build.Tasks.Git": { |       "Microsoft.Build.Tasks.Git": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
| @@ -176,14 +176,14 @@ | |||||||
|       "Nerdbank.GitVersioning": { |       "Nerdbank.GitVersioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[3.6.*, )", |         "requested": "[3.6.*, )", | ||||||
|         "resolved": "3.6.133", |         "resolved": "3.6.139", | ||||||
|         "contentHash": "VZWMd5YAeDxpjWjAP/X6bAxnRMiEf6tES/ITN0X5CHJgkWLLeHGmEALivmTAfYM6P+P/3Szy6VCITUAkqjcHVw==" |         "contentHash": "rq0Ub/Jik7PtMtZtLn0tHuJ01Yt36RQ+eeBe+S7qnJ/EFOX6D4T9zuYD3vQPYKGI6Ro4t2iWgFm3fGDgjBrMfg==" | ||||||
|       }, |       }, | ||||||
|       "NuGet.Versioning": { |       "NuGet.Versioning": { | ||||||
|         "type": "Direct", |         "type": "Direct", | ||||||
|         "requested": "[6.9.1, )", |         "requested": "[6.10.0, )", | ||||||
|         "resolved": "6.9.1", |         "resolved": "6.10.0", | ||||||
|         "contentHash": "ypnSvEtpNGo48bAWn95J1oHChycCXcevFSbn53fqzLxlXFSZP7dawu8p/7mHAfGufZQSV2sBpW80XQGIfXO8kQ==" |         "contentHash": "ytTA08tgZWo/Pbk333hnzQfiMyyynkkwMt4GyC5T0bzExYYSRNrkwv4jT8jmWr5LUOYsumEZvXdoven2SA7YZw==" | ||||||
|       }, |       }, | ||||||
|       "Microsoft.Build.Tasks.Git": { |       "Microsoft.Build.Tasks.Git": { | ||||||
|         "type": "Transitive", |         "type": "Transitive", | ||||||
|   | |||||||
| @@ -38,10 +38,10 @@ | |||||||
|  |  | ||||||
|   <ItemGroup Condition=" $(MSBuildProjectName.EndsWith('Tests')) "> |   <ItemGroup Condition=" $(MSBuildProjectName.EndsWith('Tests')) "> | ||||||
|     <ProjectReference Include="..\Divergic.Logging.Xunit\Divergic.Logging.Xunit.csproj" /> |     <ProjectReference Include="..\Divergic.Logging.Xunit\Divergic.Logging.Xunit.csproj" /> | ||||||
|     <PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.9.0" /> |     <PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.10.0" /> | ||||||
|     <PackageReference Include="Microsoft.CSharp" Version="4.7.0" /> |     <PackageReference Include="Microsoft.CSharp" Version="4.7.0" /> | ||||||
|     <PackageReference Include="xunit" Version="2.8.0" /> |     <PackageReference Include="xunit" Version="2.8.1" /> | ||||||
|     <PackageReference Include="xunit.runner.visualstudio" Version="2.8.0" /> |     <PackageReference Include="xunit.runner.visualstudio" Version="2.8.1" /> | ||||||
|     <PackageReference Include="Xunit.SkippableFact" Version="1.4.13" /> |     <PackageReference Include="Xunit.SkippableFact" Version="1.4.13" /> | ||||||
|     <PackageReference Include="coverlet.msbuild" Version="6.0.1"> |     <PackageReference Include="coverlet.msbuild" Version="6.0.1"> | ||||||
|       <PrivateAssets>all</PrivateAssets> |       <PrivateAssets>all</PrivateAssets> | ||||||
|   | |||||||
| @@ -15,6 +15,10 @@ | |||||||
|       <PrivateAssets>all</PrivateAssets> |       <PrivateAssets>all</PrivateAssets> | ||||||
|       <IncludeAssets>runtime; build; native; contentfiles; analyzers</IncludeAssets> |       <IncludeAssets>runtime; build; native; contentfiles; analyzers</IncludeAssets> | ||||||
|     </PackageReference> |     </PackageReference> | ||||||
|  |     <PackageReference Update="GitHubActionsTestLogger" Version="2.4.1"> | ||||||
|  |       <PrivateAssets>all</PrivateAssets> | ||||||
|  |       <IncludeAssets>runtime; build; native; contentfiles; analyzers</IncludeAssets> | ||||||
|  |     </PackageReference> | ||||||
|   </ItemGroup> |   </ItemGroup> | ||||||
|  |  | ||||||
| </Project> | </Project> | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user