mirror of
https://jihulab.com/DGP-Studio/Snap.Hutao.Deployment.git
synced 2025-11-19 21:08:45 +08:00
Compare commits
3 Commits
1.15.1
...
feat/webvi
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
98ae857c60 | ||
|
|
738fe060bc | ||
|
|
d3f8500ebe |
5
.github/workflows/Publish.yml
vendored
5
.github/workflows/Publish.yml
vendored
@@ -7,7 +7,7 @@ on:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
publish:
|
publish:
|
||||||
runs-on: ubuntu-latest
|
runs-on: windows-latest
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout Repository
|
- name: Checkout Repository
|
||||||
@@ -18,6 +18,9 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
dotnet-version: '8.x'
|
dotnet-version: '8.x'
|
||||||
|
|
||||||
|
- name: Build Tool
|
||||||
|
run: dotnet publish src/Snap.Hutao.Deployment/Snap.Hutao.Deployment.csproj
|
||||||
|
|
||||||
- name: Pack
|
- name: Pack
|
||||||
run: dotnet pack src/Snap.Hutao.Deployment.Runtime/Snap.Hutao.Deployment.Runtime.csproj
|
run: dotnet pack src/Snap.Hutao.Deployment.Runtime/Snap.Hutao.Deployment.Runtime.csproj
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
<package xmlns="http://schemas.microsoft.com/packaging/2012/06/nuspec.xsd">
|
<package xmlns="http://schemas.microsoft.com/packaging/2012/06/nuspec.xsd">
|
||||||
<metadata>
|
<metadata>
|
||||||
<id>Snap.Hutao.Deployment.Runtime</id>
|
<id>Snap.Hutao.Deployment.Runtime</id>
|
||||||
<version>1.15.1</version>
|
<version>1.9.0</version>
|
||||||
<authors>DGP Studio</authors>
|
<authors>DGP Studio</authors>
|
||||||
<developmentDependency>true</developmentDependency>
|
<developmentDependency>true</developmentDependency>
|
||||||
<requireLicenseAcceptance>false</requireLicenseAcceptance>
|
<requireLicenseAcceptance>false</requireLicenseAcceptance>
|
||||||
|
|||||||
Binary file not shown.
95
src/Snap.Hutao.Deployment/EdgeWebView2Dependency.cs
Normal file
95
src/Snap.Hutao.Deployment/EdgeWebView2Dependency.cs
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
using Microsoft.Win32;
|
||||||
|
using System;
|
||||||
|
using System.Diagnostics;
|
||||||
|
using System.IO;
|
||||||
|
using System.Net.Http;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace Snap.Hutao.Deployment;
|
||||||
|
|
||||||
|
internal static partial class EdgeWebView2Dependency
|
||||||
|
{
|
||||||
|
private const string EdgeWebView2DownloadUrl = "https://go.microsoft.com/fwlink/p/?LinkId=2124703";
|
||||||
|
|
||||||
|
private const string EdgeWebView2PerUserPath = @"HKEY_CURRENT_USER\Software\Microsoft\EdgeUpdate\Clients";
|
||||||
|
private const string EdgeWebView2PerMachinePath = @"HKEY_LOCAL_MACHINE\SOFTWARE\WOW6432Node\Microsoft\EdgeUpdate\Clients";
|
||||||
|
private const string EdgeWebView2GuidKey = "{F3017226-FE2A-4295-8BDF-00C3A9A7E4C5}";
|
||||||
|
|
||||||
|
public static async Task EnsureAsync(bool installWebView2, bool isUpdateMode)
|
||||||
|
{
|
||||||
|
if (!installWebView2 || isUpdateMode)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Registry.GetValue(EdgeWebView2PerUserPath, EdgeWebView2GuidKey, null) is not null || Registry.GetValue(EdgeWebView2PerMachinePath, EdgeWebView2GuidKey, null) is not null)
|
||||||
|
{
|
||||||
|
Console.WriteLine("WebView2 already installed.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Console.WriteLine("WebView2 not found, start downloading and installing WebView2...");
|
||||||
|
await DownloadWebView2InstallerAndInstallAsync().ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task DownloadWebView2InstallerAndInstallAsync()
|
||||||
|
{
|
||||||
|
string webView2InstallerPath = Path.Combine(Path.GetTempPath(), "MicrosoftEdgeWebview2Setup.exe");
|
||||||
|
try
|
||||||
|
{
|
||||||
|
using (HttpClient httpClient = new())
|
||||||
|
{
|
||||||
|
HttpShardCopyWorkerOptions<DownloadStatus> options = new()
|
||||||
|
{
|
||||||
|
HttpClient = httpClient,
|
||||||
|
SourceUrl = EdgeWebView2DownloadUrl,
|
||||||
|
DestinationFilePath = webView2InstallerPath,
|
||||||
|
StatusFactory = (bytesRead, totalBytes) => new DownloadStatus(bytesRead, totalBytes),
|
||||||
|
};
|
||||||
|
|
||||||
|
using (HttpShardCopyWorker<DownloadStatus> worker = await HttpShardCopyWorker<DownloadStatus>.CreateAsync(options).ConfigureAwait(false))
|
||||||
|
{
|
||||||
|
Progress<DownloadStatus> progress = new(ConsoleWriteProgress);
|
||||||
|
await worker.CopyAsync(progress).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Console.WriteLine("Start installing WebView2...");
|
||||||
|
Process installerProcess = new()
|
||||||
|
{
|
||||||
|
StartInfo = new()
|
||||||
|
{
|
||||||
|
Arguments = "/silent /install",
|
||||||
|
FileName = webView2InstallerPath,
|
||||||
|
RedirectStandardOutput = true,
|
||||||
|
RedirectStandardError = true,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
using (installerProcess)
|
||||||
|
{
|
||||||
|
installerProcess.OutputDataReceived += (sender, e) => Console.WriteLine(e.Data);
|
||||||
|
installerProcess.ErrorDataReceived += (sender, e) => Console.WriteLine(e.Data);
|
||||||
|
installerProcess.Start();
|
||||||
|
Console.WriteLine("-----> WebView2 Output begin -----");
|
||||||
|
installerProcess.BeginOutputReadLine();
|
||||||
|
installerProcess.BeginErrorReadLine();
|
||||||
|
|
||||||
|
await installerProcess.WaitForExitAsync().ConfigureAwait(false);
|
||||||
|
Console.WriteLine("<----- WebView2 Output end -------");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
if (File.Exists(webView2InstallerPath))
|
||||||
|
{
|
||||||
|
File.Delete(webView2InstallerPath);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void ConsoleWriteProgress(DownloadStatus status)
|
||||||
|
{
|
||||||
|
Console.Write($"\r{status.ProgressDescription}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,7 +5,6 @@ using System.IO;
|
|||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
using System.Threading;
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Windows.Security.Isolation;
|
|
||||||
|
|
||||||
namespace Snap.Hutao.Deployment;
|
namespace Snap.Hutao.Deployment;
|
||||||
|
|
||||||
@@ -56,12 +55,7 @@ internal sealed class HttpShardCopyWorker<TStatus> : IDisposable
|
|||||||
public Task CopyAsync(IProgress<TStatus> progress, CancellationToken token = default)
|
public Task CopyAsync(IProgress<TStatus> progress, CancellationToken token = default)
|
||||||
{
|
{
|
||||||
ShardProgress shardProgress = new(progress, statusFactory, contentLength);
|
ShardProgress shardProgress = new(progress, statusFactory, contentLength);
|
||||||
ParallelOptions options = new()
|
return Parallel.ForEachAsync(shards, token, (shard, token) => CopyShardAsync(shard, shardProgress, token));
|
||||||
{
|
|
||||||
MaxDegreeOfParallelism = Math.Clamp(2, Environment.ProcessorCount, 6),
|
|
||||||
CancellationToken = token,
|
|
||||||
};
|
|
||||||
return Parallel.ForEachAsync(shards, options, (shard, token) => CopyShardAsync(shard, shardProgress, token));
|
|
||||||
|
|
||||||
async ValueTask CopyShardAsync(Shard shard, IProgress<ShardStatus> progress, CancellationToken token)
|
async ValueTask CopyShardAsync(Shard shard, IProgress<ShardStatus> progress, CancellationToken token)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ internal static partial class Invocation
|
|||||||
string? path = context.ParseResult.GetValueForOption(InvocationOptions.PackagePath);
|
string? path = context.ParseResult.GetValueForOption(InvocationOptions.PackagePath);
|
||||||
string? name = context.ParseResult.GetValueForOption(InvocationOptions.FamilyName);
|
string? name = context.ParseResult.GetValueForOption(InvocationOptions.FamilyName);
|
||||||
bool isUpdateMode = context.ParseResult.GetValueForOption(InvocationOptions.UpdateBehavior);
|
bool isUpdateMode = context.ParseResult.GetValueForOption(InvocationOptions.UpdateBehavior);
|
||||||
|
bool installWebView2 = context.ParseResult.GetValueForOption(InvocationOptions.InstallWebView2);
|
||||||
|
|
||||||
if (!isUpdateMode)
|
if (!isUpdateMode)
|
||||||
{
|
{
|
||||||
@@ -24,7 +25,6 @@ internal static partial class Invocation
|
|||||||
ArgumentException.ThrowIfNullOrEmpty(path);
|
ArgumentException.ThrowIfNullOrEmpty(path);
|
||||||
|
|
||||||
Console.WriteLine($"""
|
Console.WriteLine($"""
|
||||||
Snap Hutao Deployment Tool [1.15.1]
|
|
||||||
PackagePath: {path}
|
PackagePath: {path}
|
||||||
FamilyName: {name}
|
FamilyName: {name}
|
||||||
------------------------------------------------------------
|
------------------------------------------------------------
|
||||||
@@ -46,11 +46,78 @@ internal static partial class Invocation
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await Certificate.EnsureGlobalSignCodeSigningRootR45Async().ConfigureAwait(false);
|
||||||
|
await WindowsAppSDKDependency.EnsureAsync(path).ConfigureAwait(false);
|
||||||
|
await EdgeWebView2Dependency.EnsureAsync(installWebView2, isUpdateMode).ConfigureAwait(false);
|
||||||
|
await RunDeploymentCoreAsync(path, name, isUpdateMode).ConfigureAwait(false);
|
||||||
|
await ExitAsync(isUpdateMode).ConfigureAwait(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task RunDeploymentCoreAsync(string path, string? name, bool isUpdateMode)
|
||||||
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await Certificate.EnsureGlobalSignCodeSigningRootR45Async().ConfigureAwait(false);
|
Console.WriteLine("Initializing PackageManager...");
|
||||||
await WindowsAppSDKDependency.EnsureAsync(path).ConfigureAwait(false);
|
PackageManager packageManager = new();
|
||||||
await RunDeploymentCoreAsync(path, name, isUpdateMode).ConfigureAwait(false);
|
AddPackageOptions addPackageOptions = new()
|
||||||
|
{
|
||||||
|
ForceAppShutdown = true,
|
||||||
|
RetainFilesOnFailure = true,
|
||||||
|
};
|
||||||
|
|
||||||
|
Console.WriteLine("Start deploying...");
|
||||||
|
IProgress<DeploymentProgress> progress = new Progress<DeploymentProgress>(p =>
|
||||||
|
{
|
||||||
|
Console.WriteLine($"[Deploying]: State: {p.state} Progress: {p.percentage}%");
|
||||||
|
});
|
||||||
|
DeploymentResult result = await packageManager
|
||||||
|
.AddPackageByUriAsync(new Uri(path), addPackageOptions)
|
||||||
|
.AsTask(progress)
|
||||||
|
.ConfigureAwait(false);
|
||||||
|
|
||||||
|
if (result.IsRegistered)
|
||||||
|
{
|
||||||
|
Console.WriteLine("Package deployed.");
|
||||||
|
if (string.IsNullOrEmpty(name))
|
||||||
|
{
|
||||||
|
Console.WriteLine("FamilyName not provided, enumerating packages.");
|
||||||
|
|
||||||
|
foreach (Windows.ApplicationModel.Package package in packageManager.FindPackages())
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
if (package is { DisplayName: "Snap Hutao", PublisherDisplayName: "DGP Studio" })
|
||||||
|
{
|
||||||
|
name = package.Id.FamilyName;
|
||||||
|
Console.WriteLine($"Package found: {name}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (COMException ex)
|
||||||
|
{
|
||||||
|
// ERROR_MRM_MAP_NOT_FOUND or ERROR_NOT_FOUND
|
||||||
|
if (ex.HResult is not unchecked((int)0x80073B1F) or unchecked((int)0x80070490))
|
||||||
|
{
|
||||||
|
throw;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Console.WriteLine("Starting app...");
|
||||||
|
Process.Start(new ProcessStartInfo()
|
||||||
|
{
|
||||||
|
UseShellExecute = true,
|
||||||
|
FileName = $@"shell:AppsFolder\{name}!App",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Console.WriteLine($"""
|
||||||
|
ActivityId: {result.ActivityId}
|
||||||
|
ExtendedErrorCode: {result.ExtendedErrorCode}
|
||||||
|
ErrorText: {result.ErrorText}
|
||||||
|
""");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@@ -59,86 +126,14 @@ internal static partial class Invocation
|
|||||||
{ex}
|
{ex}
|
||||||
""");
|
""");
|
||||||
}
|
}
|
||||||
finally
|
|
||||||
{
|
|
||||||
await ExitAsync(isUpdateMode).ConfigureAwait(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static async Task RunDeploymentCoreAsync(string path, string? name, bool isUpdateMode)
|
|
||||||
{
|
|
||||||
Console.WriteLine("Initializing PackageManager...");
|
|
||||||
PackageManager packageManager = new();
|
|
||||||
AddPackageOptions addPackageOptions = new()
|
|
||||||
{
|
|
||||||
ForceAppShutdown = true,
|
|
||||||
RetainFilesOnFailure = true,
|
|
||||||
};
|
|
||||||
|
|
||||||
Console.WriteLine("Start deploying...");
|
|
||||||
IProgress<DeploymentProgress> progress = new Progress<DeploymentProgress>(p =>
|
|
||||||
{
|
|
||||||
Console.WriteLine($"[Deploying]: State: {p.state} Progress: {p.percentage}%");
|
|
||||||
});
|
|
||||||
DeploymentResult result = await packageManager
|
|
||||||
.AddPackageByUriAsync(new Uri(path), addPackageOptions)
|
|
||||||
.AsTask(progress)
|
|
||||||
.ConfigureAwait(false);
|
|
||||||
|
|
||||||
if (result.IsRegistered)
|
|
||||||
{
|
|
||||||
Console.WriteLine("Package deployed.");
|
|
||||||
if (string.IsNullOrEmpty(name))
|
|
||||||
{
|
|
||||||
Console.WriteLine("FamilyName not provided, enumerating packages.");
|
|
||||||
|
|
||||||
foreach (Windows.ApplicationModel.Package package in packageManager.FindPackages())
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (package is { DisplayName: "Snap Hutao", PublisherDisplayName: "DGP Studio" })
|
|
||||||
{
|
|
||||||
name = package.Id.FamilyName;
|
|
||||||
Console.WriteLine($"Package found: {name}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (COMException ex)
|
|
||||||
{
|
|
||||||
// ERROR_MRM_MAP_NOT_FOUND or ERROR_NOT_FOUND
|
|
||||||
if (ex.HResult is not (unchecked((int)0x80073B1F) or unchecked((int)0x80070490)))
|
|
||||||
{
|
|
||||||
throw;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Console.WriteLine("Starting app...");
|
|
||||||
Process.Start(new ProcessStartInfo()
|
|
||||||
{
|
|
||||||
UseShellExecute = true,
|
|
||||||
FileName = $@"shell:AppsFolder\{name}!App",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Console.WriteLine($"""
|
|
||||||
ActivityId: {result.ActivityId}
|
|
||||||
ExtendedErrorCode: {result.ExtendedErrorCode}
|
|
||||||
ErrorText: {result.ErrorText}
|
|
||||||
""");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static async ValueTask ExitAsync(bool isUpdateMode)
|
private static async ValueTask ExitAsync(bool isUpdateMode)
|
||||||
{
|
{
|
||||||
if (!isUpdateMode)
|
if (!isUpdateMode)
|
||||||
{
|
{
|
||||||
Console.WriteLine("Press enter to exit...");
|
Console.WriteLine("Press any key to exit...");
|
||||||
while (Console.ReadKey(true).Key != ConsoleKey.Enter)
|
Console.ReadKey();
|
||||||
{
|
|
||||||
//Pending enter key
|
|
||||||
}
|
|
||||||
FreeConsole();
|
FreeConsole();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|||||||
@@ -19,4 +19,9 @@ internal static class InvocationOptions
|
|||||||
"--update-behavior",
|
"--update-behavior",
|
||||||
() => false,
|
() => false,
|
||||||
"Change behavior of the tool into update mode");
|
"Change behavior of the tool into update mode");
|
||||||
|
|
||||||
|
public static readonly Option<bool> InstallWebView2 = new(
|
||||||
|
"--install-webview2",
|
||||||
|
() => false,
|
||||||
|
"Install WebView2 if not found.");
|
||||||
}
|
}
|
||||||
@@ -16,6 +16,7 @@ internal static partial class Program
|
|||||||
root.AddOption(InvocationOptions.PackagePath);
|
root.AddOption(InvocationOptions.PackagePath);
|
||||||
root.AddOption(InvocationOptions.FamilyName);
|
root.AddOption(InvocationOptions.FamilyName);
|
||||||
root.AddOption(InvocationOptions.UpdateBehavior);
|
root.AddOption(InvocationOptions.UpdateBehavior);
|
||||||
|
root.AddOption(InvocationOptions.InstallWebView2);
|
||||||
|
|
||||||
root.SetHandler(Invocation.RunDeploymentAsync);
|
root.SetHandler(Invocation.RunDeploymentAsync);
|
||||||
|
|
||||||
|
|||||||
@@ -21,7 +21,6 @@
|
|||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="System.CommandLine" Version="2.0.0-beta4.22272.1" />
|
<PackageReference Include="System.CommandLine" Version="2.0.0-beta4.22272.1" />
|
||||||
<PackageReference Include="System.ServiceProcess.ServiceController" Version="8.0.0" />
|
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
|
|||||||
@@ -1,12 +1,8 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.IO.Compression;
|
using System.IO.Compression;
|
||||||
using System.Linq;
|
|
||||||
using System.Net.Http;
|
using System.Net.Http;
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
using System.ServiceProcess;
|
|
||||||
using System.Text.RegularExpressions;
|
using System.Text.RegularExpressions;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
using Windows.Management.Deployment;
|
using Windows.Management.Deployment;
|
||||||
@@ -20,50 +16,30 @@ internal static partial class WindowsAppSDKDependency
|
|||||||
public static async Task EnsureAsync(string packagePath)
|
public static async Task EnsureAsync(string packagePath)
|
||||||
{
|
{
|
||||||
using FileStream packageStream = File.OpenRead(packagePath);
|
using FileStream packageStream = File.OpenRead(packagePath);
|
||||||
ZipArchive package = default!;
|
using ZipArchive package = new(packageStream, ZipArchiveMode.Read);
|
||||||
try
|
|
||||||
{
|
|
||||||
package = new(packageStream, ZipArchiveMode.Read);
|
|
||||||
}
|
|
||||||
catch (InvalidDataException)
|
|
||||||
{
|
|
||||||
Console.WriteLine("Msix Package corrupted, please re-launch Deployment and try again");
|
|
||||||
try
|
|
||||||
{
|
|
||||||
File.Delete(packagePath);
|
|
||||||
}
|
|
||||||
catch
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
throw;
|
(string packageName, string msixVersion) = await ExtractRuntimePackageNameAndMsixMinVersionFromAppManifestAsync(package).ConfigureAwait(false);
|
||||||
|
if (string.IsNullOrEmpty(packageName) || string.IsNullOrEmpty(msixVersion))
|
||||||
|
{
|
||||||
|
Console.WriteLine("No Windows App Runtime version found in Msix/AppxManifest.xml");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
using (package)
|
if (CheckRuntimeInstalled(packageName, msixVersion))
|
||||||
{
|
{
|
||||||
(string packageName, string msixVersion) = await ExtractRuntimePackageNameAndMsixMinVersionFromAppManifestAsync(package).ConfigureAwait(false);
|
return;
|
||||||
if (string.IsNullOrEmpty(packageName) || string.IsNullOrEmpty(msixVersion))
|
}
|
||||||
{
|
|
||||||
Console.WriteLine("No Windows App Runtime version found in Msix/AppxManifest.xml");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (await CheckRuntimeInstalledAndVerifyAsync(packageName, msixVersion).ConfigureAwait(false))
|
string sdkVersion = await ExtractSDKVersionFromDepsJsonAsync(package).ConfigureAwait(false);
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
string sdkVersion = await ExtractSDKVersionFromDepsJsonAsync(package).ConfigureAwait(false);
|
if (string.IsNullOrEmpty(sdkVersion))
|
||||||
|
{
|
||||||
|
Console.WriteLine("No Windows App SDK version found in Msix/Snap.Hutao.deps.json");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(sdkVersion))
|
Console.WriteLine("Start downloading SDK installer...");
|
||||||
{
|
await DownloadWindowsAppRuntimeInstallAndInstallAsync(sdkVersion).ConfigureAwait(false);
|
||||||
Console.WriteLine("No Windows App SDK version found in Msix/Snap.Hutao.deps.json");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Console.WriteLine("Start downloading SDK installer...");
|
|
||||||
await DownloadWindowsAppRuntimeInstallAndInstallAsync(sdkVersion).ConfigureAwait(false);
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static async Task<string> ExtractSDKVersionFromDepsJsonAsync(ZipArchive package)
|
private static async Task<string> ExtractSDKVersionFromDepsJsonAsync(ZipArchive package)
|
||||||
@@ -87,21 +63,19 @@ internal static partial class WindowsAppSDKDependency
|
|||||||
return string.Empty;
|
return string.Empty;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static async Task<bool> CheckRuntimeInstalledAndVerifyAsync(string packageName, string msixVersion)
|
private static bool CheckRuntimeInstalled(string packageName, string msixVersion)
|
||||||
{
|
{
|
||||||
Version msixMinVersion = new(msixVersion);
|
Version msixMinVersion = new(msixVersion);
|
||||||
|
|
||||||
List<bool> results = [];
|
|
||||||
|
|
||||||
foreach (Windows.ApplicationModel.Package installed in new PackageManager().FindPackages())
|
foreach (Windows.ApplicationModel.Package installed in new PackageManager().FindPackages())
|
||||||
{
|
{
|
||||||
if (installed.Id.Name == packageName && installed.Id.Version.ToVersion() >= msixMinVersion)
|
if (installed.Id.Name == packageName && installed.Id.Version.ToVersion() >= msixMinVersion)
|
||||||
{
|
{
|
||||||
results.Add(await installed.VerifyContentIntegrityAsync());
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return results.Count > 0 && results.Aggregate((result, element) => result || element);
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static async Task DownloadWindowsAppRuntimeInstallAndInstallAsync(string version)
|
private static async Task DownloadWindowsAppRuntimeInstallAndInstallAsync(string version)
|
||||||
@@ -126,23 +100,6 @@ internal static partial class WindowsAppSDKDependency
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ServiceController serviceController = new("appxsvc");
|
|
||||||
if (serviceController.CanStop)
|
|
||||||
{
|
|
||||||
Console.WriteLine("Stopping AppxSvc...");
|
|
||||||
serviceController.Stop();
|
|
||||||
serviceController.WaitForStatus(ServiceControllerStatus.Stopped, TimeSpan.FromSeconds(5));
|
|
||||||
|
|
||||||
if (serviceController.Status is not ServiceControllerStatus.Stopped)
|
|
||||||
{
|
|
||||||
Console.WriteLine("Can not stop AppxSvc, timeout...");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
Console.WriteLine("Can not stop AppxSvc, disallowed...");
|
|
||||||
}
|
|
||||||
|
|
||||||
Console.WriteLine("Start installing SDK...");
|
Console.WriteLine("Start installing SDK...");
|
||||||
Process installerProcess = new()
|
Process installerProcess = new()
|
||||||
{
|
{
|
||||||
@@ -159,13 +116,12 @@ internal static partial class WindowsAppSDKDependency
|
|||||||
installerProcess.OutputDataReceived += (sender, e) => Console.WriteLine(e.Data);
|
installerProcess.OutputDataReceived += (sender, e) => Console.WriteLine(e.Data);
|
||||||
installerProcess.ErrorDataReceived += (sender, e) => Console.WriteLine(e.Data);
|
installerProcess.ErrorDataReceived += (sender, e) => Console.WriteLine(e.Data);
|
||||||
installerProcess.Start();
|
installerProcess.Start();
|
||||||
Console.WriteLine("-----> WindowsAppRuntimeInstall Output begin");
|
Console.WriteLine("-----> WindowsAppRuntimeInstall Output begin -----");
|
||||||
installerProcess.BeginOutputReadLine();
|
installerProcess.BeginOutputReadLine();
|
||||||
installerProcess.BeginErrorReadLine();
|
installerProcess.BeginErrorReadLine();
|
||||||
|
|
||||||
await installerProcess.WaitForExitAsync().ConfigureAwait(false);
|
await installerProcess.WaitForExitAsync().ConfigureAwait(false);
|
||||||
Marshal.ThrowExceptionForHR(installerProcess.ExitCode);
|
Console.WriteLine("<----- WindowsAppRuntimeInstall Output end -------");
|
||||||
Console.WriteLine("<----- WindowsAppRuntimeInstall Output end");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
finally
|
finally
|
||||||
|
|||||||
Reference in New Issue
Block a user