12 Commits

Author SHA1 Message Date
Lightczx
de074e40e0 code style [skip ci] 2024-01-31 15:36:29 +08:00
Lightczx
4cf4c6caca fix 2024-01-31 15:24:22 +08:00
Lightczx
02dae0e199 Update Snap.Hutao.Deployment.exe 2024-01-31 13:49:18 +08:00
Lightczx
291c754deb update version 2024-01-31 13:48:46 +08:00
Lightczx
a4fd59df08 add msix corruted message 2024-01-31 13:32:55 +08:00
DismissedLight
39ccb73e2f 1.14 2024-01-18 23:12:32 +08:00
DismissedLight
ae96cdf793 fix IOE on WAS version check 2024-01-18 23:10:03 +08:00
DismissedLight
0571c386e9 1.13 2024-01-16 22:03:51 +08:00
qhy040404
95d7a85494 Improve WAS install pipeline (#5)
Co-authored-by: DismissedLight <1686188646@qq.com>
2024-01-16 21:23:25 +08:00
qhy040404
d8b3f8ae34 minor fix 2024-01-12 09:24:52 +08:00
DismissedLight
da33fc0ba3 Merge pull request #4 from DGP-Studio/fix/crash 2024-01-12 09:19:54 +08:00
qhy040404
d6f5b7374d global try and enter to exit 2024-01-12 09:13:13 +08:00
8 changed files with 153 additions and 100 deletions

View File

@@ -7,7 +7,7 @@ on:
jobs: jobs:
publish: publish:
runs-on: windows-latest runs-on: ubuntu-latest
steps: steps:
- name: Checkout Repository - name: Checkout Repository
@@ -18,9 +18,6 @@ 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

View File

@@ -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.9.0</version> <version>1.15.2</version>
<authors>DGP Studio</authors> <authors>DGP Studio</authors>
<developmentDependency>true</developmentDependency> <developmentDependency>true</developmentDependency>
<requireLicenseAcceptance>false</requireLicenseAcceptance> <requireLicenseAcceptance>false</requireLicenseAcceptance>

View File

@@ -55,7 +55,12 @@ 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);
return Parallel.ForEachAsync(shards, token, (shard, token) => CopyShardAsync(shard, shardProgress, token)); ParallelOptions options = new()
{
MaxDegreeOfParallelism = Math.Clamp(Environment.ProcessorCount, 2, 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)
{ {

View File

@@ -24,6 +24,7 @@ internal static partial class Invocation
ArgumentException.ThrowIfNullOrEmpty(path); ArgumentException.ThrowIfNullOrEmpty(path);
Console.WriteLine($""" Console.WriteLine($"""
Snap Hutao Deployment Tool [1.15.2]
PackagePath: {path} PackagePath: {path}
FamilyName: {name} FamilyName: {name}
------------------------------------------------------------ ------------------------------------------------------------
@@ -45,77 +46,11 @@ internal static partial class Invocation
} }
} }
await Certificate.EnsureGlobalSignCodeSigningRootR45Async().ConfigureAwait(false);
await WindowsAppSDKDependency.EnsureAsync(path).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
{ {
Console.WriteLine("Initializing PackageManager..."); await Certificate.EnsureGlobalSignCodeSigningRootR45Async().ConfigureAwait(false);
PackageManager packageManager = new(); await WindowsAppSDKDependency.EnsureAsync(path).ConfigureAwait(false);
AddPackageOptions addPackageOptions = new() await RunDeploymentCoreAsync(path, name, isUpdateMode).ConfigureAwait(false);
{
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)
{ {
@@ -124,14 +59,86 @@ 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 any key to exit..."); Console.WriteLine("Press enter to exit...");
Console.ReadKey(); while (Console.ReadKey(true).Key != ConsoleKey.Enter)
{
//Pending enter key
}
FreeConsole(); FreeConsole();
} }
else else

View File

@@ -1,5 +1,4 @@
using System.CommandLine; using System.CommandLine;
using System.Runtime.InteropServices;
using System.Threading.Tasks; using System.Threading.Tasks;
namespace Snap.Hutao.Deployment; namespace Snap.Hutao.Deployment;

View File

@@ -21,6 +21,7 @@
<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>

View File

@@ -1,8 +1,12 @@
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;
@@ -16,30 +20,50 @@ 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);
using ZipArchive package = new(packageStream, ZipArchiveMode.Read); ZipArchive package = default!;
try
(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"); package = new(packageStream, ZipArchiveMode.Read);
return; }
catch (InvalidDataException)
{
Console.WriteLine("Msix Package corrupted, please re-launch Deployment and try again");
try
{
File.Delete(packagePath);
}
catch
{
}
throw;
} }
if (CheckRuntimeInstalled(packageName, msixVersion)) using (package)
{ {
return; (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;
}
string sdkVersion = await ExtractSDKVersionFromDepsJsonAsync(package).ConfigureAwait(false); if (await CheckRuntimeInstalledAndVerifyAsync(packageName, msixVersion).ConfigureAwait(false))
{
return;
}
if (string.IsNullOrEmpty(sdkVersion)) string sdkVersion = await ExtractSDKVersionFromDepsJsonAsync(package).ConfigureAwait(false);
{
Console.WriteLine("No Windows App SDK version found in Msix/Snap.Hutao.deps.json");
return;
}
Console.WriteLine("Start downloading SDK installer..."); if (string.IsNullOrEmpty(sdkVersion))
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)
@@ -63,19 +87,21 @@ internal static partial class WindowsAppSDKDependency
return string.Empty; return string.Empty;
} }
private static bool CheckRuntimeInstalled(string packageName, string msixVersion) private static async Task<bool> CheckRuntimeInstalledAndVerifyAsync(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)
{ {
return true; results.Add(await installed.VerifyContentIntegrityAsync());
} }
} }
return false; return results.Count > 0 && results.Aggregate((result, element) => result || element);
} }
private static async Task DownloadWindowsAppRuntimeInstallAndInstallAsync(string version) private static async Task DownloadWindowsAppRuntimeInstallAndInstallAsync(string version)
@@ -100,6 +126,23 @@ 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()
{ {
@@ -116,12 +159,13 @@ 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);
Console.WriteLine("<----- WindowsAppRuntimeInstall Output end -------"); Marshal.ThrowExceptionForHR(installerProcess.ExitCode);
Console.WriteLine("<----- WindowsAppRuntimeInstall Output end");
} }
} }
finally finally