From 6f9125cf0a7d817fda04124aa3c14da5e290a4fd Mon Sep 17 00:00:00 2001 From: Dion Date: Mon, 25 Mar 2024 15:19:58 +0100 Subject: [PATCH 01/11] add deps cli --- starsky/starsky.sln | 7 +++ starsky/starskydependenciescli/Program.cs | 48 +++++++++++++++++++ .../starskydependenciescli.csproj | 18 +++++++ 3 files changed, 73 insertions(+) create mode 100644 starsky/starskydependenciescli/Program.cs create mode 100644 starsky/starskydependenciescli/starskydependenciescli.csproj diff --git a/starsky/starsky.sln b/starsky/starsky.sln index ef8b5bbfa1..71d36f2bb1 100644 --- a/starsky/starsky.sln +++ b/starsky/starsky.sln @@ -166,6 +166,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "starsky.feature.settings", EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "starsky.feature.desktop", "starsky.feature.desktop\starsky.feature.desktop.csproj", "{B88C2815-D154-4C6D-AE37-2E150AEBF73D}" EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "starskydependenciescli", "starskydependenciescli\starskydependenciescli.csproj", "{CDFE2199-CB93-49E4-AF8A-98A3239F1D3F}" +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Any CPU = Debug|Any CPU @@ -368,6 +370,10 @@ Global {B88C2815-D154-4C6D-AE37-2E150AEBF73D}.Debug|Any CPU.Build.0 = Debug|Any CPU {B88C2815-D154-4C6D-AE37-2E150AEBF73D}.Release|Any CPU.ActiveCfg = Release|Any CPU {B88C2815-D154-4C6D-AE37-2E150AEBF73D}.Release|Any CPU.Build.0 = Release|Any CPU + {CDFE2199-CB93-49E4-AF8A-98A3239F1D3F}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {CDFE2199-CB93-49E4-AF8A-98A3239F1D3F}.Debug|Any CPU.Build.0 = Debug|Any CPU + {CDFE2199-CB93-49E4-AF8A-98A3239F1D3F}.Release|Any CPU.ActiveCfg = Release|Any CPU + {CDFE2199-CB93-49E4-AF8A-98A3239F1D3F}.Release|Any CPU.Build.0 = Release|Any CPU EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE @@ -428,5 +434,6 @@ Global {A62C129C-5D0C-4A0A-B5AA-261E041FF55D} = {4B9276C3-651E-48D3-B3A7-3F4D74F3D01A} {F2C4C9DE-22A1-4B34-AC1D-0F08353E0742} = {4B9276C3-651E-48D3-B3A7-3F4D74F3D01A} {B88C2815-D154-4C6D-AE37-2E150AEBF73D} = {4B9276C3-651E-48D3-B3A7-3F4D74F3D01A} + {CDFE2199-CB93-49E4-AF8A-98A3239F1D3F} = {B974FC20-C3EE-4EB0-AF25-F0D8DA2C28D7} EndGlobalSection EndGlobal diff --git a/starsky/starskydependenciescli/Program.cs b/starsky/starskydependenciescli/Program.cs new file mode 100644 index 0000000000..e748ad45cb --- /dev/null +++ b/starsky/starskydependenciescli/Program.cs @@ -0,0 +1,48 @@ +using System.Threading.Tasks; + +namespace starskyDependenciesCli +{ + public static class Program + { + public static async Task Main(string[] args) + { + // Use args in application + new ArgsHelper().SetEnvironmentByArgs(args); + + var services = new ServiceCollection(); + + // Setup AppSettings + services = await SetupAppSettings.FirstStepToAddSingleton(services); + + // Inject services + RegisterDependencies.Configure(services); + var serviceProvider = services.BuildServiceProvider(); + var appSettings = serviceProvider.GetRequiredService(); + + services.AddOpenTelemetryMonitoring(appSettings); + services.AddTelemetryLogging(appSettings); + + new SetupDatabaseTypes(appSettings, services).BuilderDb(); + serviceProvider = services.BuildServiceProvider(); + + var geoReverseLookup = serviceProvider.GetRequiredService(); + var geoLocationWrite = serviceProvider.GetRequiredService(); + var geoFileDownload = serviceProvider.GetRequiredService(); + + var selectorStorage = serviceProvider.GetRequiredService(); + + var console = serviceProvider.GetRequiredService(); + var exifToolDownload = serviceProvider.GetRequiredService(); + var logger = serviceProvider.GetRequiredService(); + + // Migrations before geo-tools (not needed for this specific app, but helps the process) + await RunMigrations.Run(serviceProvider.GetRequiredService(), + logger, appSettings); + + // Help and other Command Line Tools args are included in the Geo tools + await new GeoCli(geoReverseLookup, geoLocationWrite, selectorStorage, + appSettings, console, geoFileDownload, exifToolDownload, logger) + .CommandLineAsync(args); + } + } +} diff --git a/starsky/starskydependenciescli/starskydependenciescli.csproj b/starsky/starskydependenciescli/starskydependenciescli.csproj new file mode 100644 index 0000000000..039e874339 --- /dev/null +++ b/starsky/starskydependenciescli/starskydependenciescli.csproj @@ -0,0 +1,18 @@ + + + + Exe + net8.0 + + {04d4b161-7dad-4258-8edc-f0a77e39dc3f} + 8.0.3 + starskydependenciescli + 0.6.0 + enable + disable + + + + true + + From 210caff019d33c62e1e4088b72debc55bb99eae0 Mon Sep 17 00:00:00 2001 From: Dion Date: Wed, 27 Mar 2024 20:17:10 +0100 Subject: [PATCH 02/11] WIP --- .../DependenciesCliService.cs | 36 ++++++++ .../Interfaces/IDependenciesCliService.cs | 5 ++ .../starsky.feature.dependenciescli.csproj | 13 +++ .../Services/PackageTelemetry.cs | 90 ++++++++++--------- .../Services/HttpClientHelper.cs | 4 +- .../Helpers/PlatformParser.cs | 30 +++++++ starsky/starsky.sln | 7 ++ starsky/starskydependenciescli/Program.cs | 26 +++--- .../default-init-launchSettings.json | 11 +++ .../starskydependenciescli.csproj | 12 +++ .../Helpers/PackageTelemetryTest.cs | 17 +--- .../Helpers/PlatformParserTest.cs | 23 +++++ 12 files changed, 205 insertions(+), 69 deletions(-) create mode 100644 starsky/starsky.feature.dependenciescli/DependenciesCliService.cs create mode 100644 starsky/starsky.feature.dependenciescli/Interfaces/IDependenciesCliService.cs create mode 100644 starsky/starsky.feature.dependenciescli/starsky.feature.dependenciescli.csproj create mode 100644 starsky/starsky.foundation.platform/Helpers/PlatformParser.cs create mode 100644 starsky/starskydependenciescli/default-init-launchSettings.json create mode 100644 starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs diff --git a/starsky/starsky.feature.dependenciescli/DependenciesCliService.cs b/starsky/starsky.feature.dependenciescli/DependenciesCliService.cs new file mode 100644 index 0000000000..57cf97735f --- /dev/null +++ b/starsky/starsky.feature.dependenciescli/DependenciesCliService.cs @@ -0,0 +1,36 @@ +using System.Runtime.InteropServices; +using starsky.foundation.database.Data; +using starsky.foundation.database.Helpers; +using starsky.foundation.platform.Helpers; +using starsky.foundation.platform.Interfaces; +using starsky.foundation.platform.Models; +using starsky.foundation.writemeta.Interfaces; + +namespace starsky.feature.dependenciescli; + +public class DependenciesCliService +{ + private IExifToolDownload _exifToolDownload; + private readonly ApplicationDbContext _dbContext; + private IWebLogger _logger; + private AppSettings _appSettings; + + public DependenciesCliService(IExifToolDownload exifToolDownload, + ApplicationDbContext dbContext, IWebLogger logger, AppSettings appSettings) + { + _exifToolDownload = exifToolDownload; + _dbContext = dbContext; + _logger = logger; + _appSettings = appSettings; + } + + public async Task Run(OSPlatform? currentPlatform = null, Architecture? architecture = null) + { + currentPlatform ??= PlatformParser.GetCurrentOsPlatform(); + await RunMigrations.Run(_dbContext, _logger, _appSettings); + await _exifToolDownload.DownloadExifTool(currentPlatform == OSPlatform.Windows); + + } + + +} diff --git a/starsky/starsky.feature.dependenciescli/Interfaces/IDependenciesCliService.cs b/starsky/starsky.feature.dependenciescli/Interfaces/IDependenciesCliService.cs new file mode 100644 index 0000000000..9b3898d81a --- /dev/null +++ b/starsky/starsky.feature.dependenciescli/Interfaces/IDependenciesCliService.cs @@ -0,0 +1,5 @@ +namespace starsky.feature.dependenciescli.Interfaces; + +public interface IDependenciesCliService +{ +} diff --git a/starsky/starsky.feature.dependenciescli/starsky.feature.dependenciescli.csproj b/starsky/starsky.feature.dependenciescli/starsky.feature.dependenciescli.csproj new file mode 100644 index 0000000000..a7c1900e1b --- /dev/null +++ b/starsky/starsky.feature.dependenciescli/starsky.feature.dependenciescli.csproj @@ -0,0 +1,13 @@ + + + + net8.0 + enable + enable + + + + + + + diff --git a/starsky/starsky.feature.packagetelemetry/Services/PackageTelemetry.cs b/starsky/starsky.feature.packagetelemetry/Services/PackageTelemetry.cs index db76e87d42..95e896a70e 100644 --- a/starsky/starsky.feature.packagetelemetry/Services/PackageTelemetry.cs +++ b/starsky/starsky.feature.packagetelemetry/Services/PackageTelemetry.cs @@ -1,7 +1,6 @@ using System; using System.Collections.Generic; using System.Globalization; -using System.Linq; using System.Net.Http; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; @@ -17,9 +16,9 @@ using starsky.foundation.platform.Models; [assembly: InternalsVisibleTo("starskytest")] + namespace starsky.feature.packagetelemetry.Services { - [Service(typeof(IPackageTelemetry), InjectionLifetime = InjectionLifetime.Scoped)] public class PackageTelemetry : IPackageTelemetry { @@ -29,7 +28,8 @@ public class PackageTelemetry : IPackageTelemetry private readonly IQuery _query; private readonly IDeviceIdService _deviceIdService; - public PackageTelemetry(IHttpClientHelper httpClientHelper, AppSettings appSettings, IWebLogger logger, IQuery query, IDeviceIdService deviceIdService) + public PackageTelemetry(IHttpClientHelper httpClientHelper, AppSettings appSettings, + IWebLogger logger, IQuery query, IDeviceIdService deviceIdService) { _httpClientHelper = httpClientHelper; _appSettings = appSettings; @@ -38,49 +38,46 @@ public PackageTelemetry(IHttpClientHelper httpClientHelper, AppSettings appSetti _deviceIdService = deviceIdService; } - internal const string PackageTelemetryUrl = "qdraw.nl/special/starsky/telemetry/index.php"; + internal const string PackageTelemetryUrl = "qdraw.nl/special/starsky/telemetry/v2.php"; internal static object? GetPropValue(object? src, string propName) { return src?.GetType().GetProperty(propName)?.GetValue(src, null); } - internal static OSPlatform? GetCurrentOsPlatform() - { - OSPlatform? currentPlatform = null; - foreach ( var platform in new List{OSPlatform.Linux, - OSPlatform.Windows, OSPlatform.OSX, - OSPlatform.FreeBSD}.Where(RuntimeInformation.IsOSPlatform) ) - { - currentPlatform = platform; - } - - return currentPlatform; - } - - internal List> GetSystemData(OSPlatform? currentPlatform = null, string? deviceId = null) + internal List> GetSystemData( + OSPlatform? currentPlatform = null, string? deviceId = null) { - currentPlatform ??= GetCurrentOsPlatform(); + currentPlatform ??= PlatformParser.GetCurrentOsPlatform(); var dockerContainer = currentPlatform == OSPlatform.Linux && - Environment.GetEnvironmentVariable( - "DOTNET_RUNNING_IN_CONTAINER") == "true"; + Environment.GetEnvironmentVariable( + "DOTNET_RUNNING_IN_CONTAINER") == "true"; var data = new List> { - new KeyValuePair("UTCTime", DateTime.UtcNow.ToString(CultureInfo.InvariantCulture)), + new KeyValuePair("UTCTime", + DateTime.UtcNow.ToString(CultureInfo.InvariantCulture)), new KeyValuePair("AppVersion", _appSettings.AppVersion), - new KeyValuePair("NetVersion", RuntimeInformation.FrameworkDescription), - new KeyValuePair("OSArchitecture", RuntimeInformation.OSArchitecture.ToString()), - new KeyValuePair("ProcessArchitecture", RuntimeInformation.ProcessArchitecture.ToString()), - new KeyValuePair("OSVersion", Environment.OSVersion.Version.ToString()), - new KeyValuePair("OSDescriptionLong", RuntimeInformation.OSDescription.Replace(";", " ")), + new KeyValuePair("NetVersion", + RuntimeInformation.FrameworkDescription), + new KeyValuePair("OSArchitecture", + RuntimeInformation.OSArchitecture.ToString()), + new KeyValuePair("ProcessArchitecture", + RuntimeInformation.ProcessArchitecture.ToString()), + new KeyValuePair("OSVersion", + Environment.OSVersion.Version.ToString()), + new KeyValuePair("OSDescriptionLong", + RuntimeInformation.OSDescription.Replace(";", " ")), new KeyValuePair("OSPlatform", currentPlatform.ToString()!), new KeyValuePair("DockerContainer", dockerContainer.ToString()), - new KeyValuePair("CurrentCulture", CultureInfo.CurrentCulture.ThreeLetterISOLanguageName), - new KeyValuePair("AspNetCoreEnvironment", Environment.GetEnvironmentVariable("ASPNETCORE_ENVIRONMENT") ?? "Not set"), + new KeyValuePair("CurrentCulture", + CultureInfo.CurrentCulture.ThreeLetterISOLanguageName), + new KeyValuePair("AspNetCoreEnvironment", + Environment.GetEnvironmentVariable("ASPNETCORE_ENVIRONMENT") ?? "Not set"), new KeyValuePair("WebsiteName", GetEncryptedSiteName()), new KeyValuePair("DeviceId", deviceId ?? "Not set"), + new KeyValuePair("RuntimeIdentifier", RuntimeInformation.RuntimeIdentifier), }; return data; } @@ -89,11 +86,14 @@ private static string GetEncryptedSiteName() { var siteName = Environment.GetEnvironmentVariable("WEBSITE_SITE_NAME"); - return string.IsNullOrEmpty(siteName) ? "Not set" : Sha256.ComputeSha256(siteName); // used in Azure web apps + return string.IsNullOrEmpty(siteName) + ? "Not set" + : Sha256.ComputeSha256(siteName); // used in Azure web apps } - internal async Task>> AddDatabaseData(List> data) + internal async Task>> AddDatabaseData( + List> data) { var fileIndexItemTotalCount = -1; var fileIndexItemDirectoryCount = -1; @@ -112,22 +112,27 @@ internal async Task>> AddDatabaseData(List> { - new KeyValuePair("FileIndexItemTotalCount",fileIndexItemTotalCount.ToString()), - new KeyValuePair("FileIndexItemDirectoryCount",fileIndexItemDirectoryCount.ToString()), - new KeyValuePair("FileIndexItemCount",fileIndexItemCount.ToString()) + new KeyValuePair("FileIndexItemTotalCount", + fileIndexItemTotalCount.ToString()), + new KeyValuePair("FileIndexItemDirectoryCount", + fileIndexItemDirectoryCount.ToString()), + new KeyValuePair("FileIndexItemCount", + fileIndexItemCount.ToString()) }); return data; } - internal List> AddAppSettingsData(List> data) + internal List> AddAppSettingsData( + List> data) { var type = typeof(AppSettings); var properties = type.GetProperties(); // ReSharper disable once LoopCanBeConvertedToQuery foreach ( var property in properties ) { - var someAttribute = Array.Find(Attribute.GetCustomAttributes(property), x => x is PackageTelemetryAttribute); + var someAttribute = Array.Find(Attribute.GetCustomAttributes(property), + x => x is PackageTelemetryAttribute); if ( someAttribute == null ) { continue; @@ -143,13 +148,16 @@ internal List> AddAppSettingsData(List) || - propValue?.GetType() == typeof(Dictionary>) ) + propValue?.GetType() == + typeof(Dictionary>) ) { value = ParseContent(propValue); } - data.Add(new KeyValuePair("AppSettings" + property.Name, value ?? "null")); + data.Add(new KeyValuePair("AppSettings" + property.Name, + value ?? "null")); } + return data; } @@ -160,7 +168,8 @@ internal static string ParseContent(object propValue) private async Task PostData(HttpContent formUrlEncodedContent) { - return ( await _httpClientHelper.PostString("https://" + PackageTelemetryUrl, formUrlEncodedContent, + return ( await _httpClientHelper.PostString("https://" + PackageTelemetryUrl, + formUrlEncodedContent, _appSettings.EnablePackageTelemetryDebug == true) ).Key; } @@ -171,7 +180,7 @@ private async Task PostData(HttpContent formUrlEncodedContent) return null; } - var currentOsPlatform = GetCurrentOsPlatform(); + var currentOsPlatform = PlatformParser.GetCurrentOsPlatform(); var deviceId = await _deviceIdService.DeviceId(currentOsPlatform); var telemetryDataItems = GetSystemData(currentOsPlatform, deviceId); @@ -194,4 +203,3 @@ private async Task PostData(HttpContent formUrlEncodedContent) } } } - diff --git a/starsky/starsky.foundation.http/Services/HttpClientHelper.cs b/starsky/starsky.foundation.http/Services/HttpClientHelper.cs index 8292f10cf0..1c80632de5 100644 --- a/starsky/starsky.foundation.http/Services/HttpClientHelper.cs +++ b/starsky/starsky.foundation.http/Services/HttpClientHelper.cs @@ -10,6 +10,7 @@ using starsky.foundation.injection; using starsky.foundation.platform.Helpers; using starsky.foundation.platform.Interfaces; +using starsky.foundation.storage.Helpers; using starsky.foundation.storage.Interfaces; using starsky.foundation.storage.Storage; @@ -100,12 +101,13 @@ public async Task> PostString(string sourceHttpUrl, // // allow whitelist and https only if ( !_allowedDomains.Contains(sourceUri.Host) || sourceUri.Scheme != "https" ) return new KeyValuePair(false, string.Empty); - + try { using ( HttpResponseMessage response = await _httpProvider.PostAsync(sourceHttpUrl, httpContent) ) using ( Stream streamToReadFrom = await response.Content.ReadAsStreamAsync() ) { + var reader = new StreamReader(streamToReadFrom, Encoding.UTF8); var result = await reader.ReadToEndAsync(); return new KeyValuePair(response.StatusCode == HttpStatusCode.OK, result); diff --git a/starsky/starsky.foundation.platform/Helpers/PlatformParser.cs b/starsky/starsky.foundation.platform/Helpers/PlatformParser.cs new file mode 100644 index 0000000000..d45de65925 --- /dev/null +++ b/starsky/starsky.foundation.platform/Helpers/PlatformParser.cs @@ -0,0 +1,30 @@ +using System.Collections.Generic; +using System.Linq; +using System.Runtime.InteropServices; + +namespace starsky.foundation.platform.Helpers; + +public static class PlatformParser +{ + public static OSPlatform? GetCurrentOsPlatform() + { + OSPlatform? currentPlatform = null; + foreach ( var platform in new List + { + OSPlatform.Linux, + OSPlatform.Windows, + OSPlatform.OSX, + OSPlatform.FreeBSD + }.Where(RuntimeInformation.IsOSPlatform) ) + { + currentPlatform = platform; + } + + return currentPlatform; + } + + public static string GetCurrentArchitecture() + { + return RuntimeInformation.RuntimeIdentifier; + } +} diff --git a/starsky/starsky.sln b/starsky/starsky.sln index 71d36f2bb1..1d3ed6df59 100644 --- a/starsky/starsky.sln +++ b/starsky/starsky.sln @@ -168,6 +168,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "starsky.feature.desktop", " EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "starskydependenciescli", "starskydependenciescli\starskydependenciescli.csproj", "{CDFE2199-CB93-49E4-AF8A-98A3239F1D3F}" EndProject +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "starsky.feature.dependenciescli", "starsky.feature.dependenciescli\starsky.feature.dependenciescli.csproj", "{C34BB48B-0C65-4897-80A9-87CB1BDD5F48}" +EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution Debug|Any CPU = Debug|Any CPU @@ -374,6 +376,10 @@ Global {CDFE2199-CB93-49E4-AF8A-98A3239F1D3F}.Debug|Any CPU.Build.0 = Debug|Any CPU {CDFE2199-CB93-49E4-AF8A-98A3239F1D3F}.Release|Any CPU.ActiveCfg = Release|Any CPU {CDFE2199-CB93-49E4-AF8A-98A3239F1D3F}.Release|Any CPU.Build.0 = Release|Any CPU + {C34BB48B-0C65-4897-80A9-87CB1BDD5F48}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {C34BB48B-0C65-4897-80A9-87CB1BDD5F48}.Debug|Any CPU.Build.0 = Debug|Any CPU + {C34BB48B-0C65-4897-80A9-87CB1BDD5F48}.Release|Any CPU.ActiveCfg = Release|Any CPU + {C34BB48B-0C65-4897-80A9-87CB1BDD5F48}.Release|Any CPU.Build.0 = Release|Any CPU EndGlobalSection GlobalSection(SolutionProperties) = preSolution HideSolutionNode = FALSE @@ -435,5 +441,6 @@ Global {F2C4C9DE-22A1-4B34-AC1D-0F08353E0742} = {4B9276C3-651E-48D3-B3A7-3F4D74F3D01A} {B88C2815-D154-4C6D-AE37-2E150AEBF73D} = {4B9276C3-651E-48D3-B3A7-3F4D74F3D01A} {CDFE2199-CB93-49E4-AF8A-98A3239F1D3F} = {B974FC20-C3EE-4EB0-AF25-F0D8DA2C28D7} + {C34BB48B-0C65-4897-80A9-87CB1BDD5F48} = {4B9276C3-651E-48D3-B3A7-3F4D74F3D01A} EndGlobalSection EndGlobal diff --git a/starsky/starskydependenciescli/Program.cs b/starsky/starskydependenciescli/Program.cs index e748ad45cb..9f69260df7 100644 --- a/starsky/starskydependenciescli/Program.cs +++ b/starsky/starskydependenciescli/Program.cs @@ -1,4 +1,15 @@ using System.Threading.Tasks; +using Microsoft.Extensions.DependencyInjection; +using starsky.foundation.database.Data; +using starsky.foundation.database.Helpers; +using starsky.foundation.injection; +using starsky.foundation.platform.Helpers; +using starsky.foundation.platform.Interfaces; +using starsky.foundation.platform.Models; +using starsky.foundation.storage.Interfaces; +using starsky.foundation.webtelemetry.Extensions; +using starsky.foundation.webtelemetry.Helpers; +using starsky.foundation.writemeta.Interfaces; namespace starskyDependenciesCli { @@ -25,24 +36,11 @@ public static async Task Main(string[] args) new SetupDatabaseTypes(appSettings, services).BuilderDb(); serviceProvider = services.BuildServiceProvider(); - var geoReverseLookup = serviceProvider.GetRequiredService(); - var geoLocationWrite = serviceProvider.GetRequiredService(); - var geoFileDownload = serviceProvider.GetRequiredService(); - - var selectorStorage = serviceProvider.GetRequiredService(); - - var console = serviceProvider.GetRequiredService(); var exifToolDownload = serviceProvider.GetRequiredService(); var logger = serviceProvider.GetRequiredService(); - // Migrations before geo-tools (not needed for this specific app, but helps the process) - await RunMigrations.Run(serviceProvider.GetRequiredService(), - logger, appSettings); - // Help and other Command Line Tools args are included in the Geo tools - await new GeoCli(geoReverseLookup, geoLocationWrite, selectorStorage, - appSettings, console, geoFileDownload, exifToolDownload, logger) - .CommandLineAsync(args); + exifToolDownload.DownloadExifTool() } } } diff --git a/starsky/starskydependenciescli/default-init-launchSettings.json b/starsky/starskydependenciescli/default-init-launchSettings.json new file mode 100644 index 0000000000..bec23311cc --- /dev/null +++ b/starsky/starskydependenciescli/default-init-launchSettings.json @@ -0,0 +1,11 @@ +{ + "$schema": "http://json.schemastore.org/launchsettings.json", + "profiles": { + "starskydependenciescli": { + "commandName": "Project", + "commandLineArgs": "-h", + "environmentVariables": { + } + } + } +} diff --git a/starsky/starskydependenciescli/starskydependenciescli.csproj b/starsky/starskydependenciescli/starskydependenciescli.csproj index 039e874339..4812f874fe 100644 --- a/starsky/starskydependenciescli/starskydependenciescli.csproj +++ b/starsky/starskydependenciescli/starskydependenciescli.csproj @@ -15,4 +15,16 @@ true + + + + + + + + + + + + diff --git a/starsky/starskytest/starsky.feature.packagetelemetry/Helpers/PackageTelemetryTest.cs b/starsky/starskytest/starsky.feature.packagetelemetry/Helpers/PackageTelemetryTest.cs index 53525e69e0..8123add981 100644 --- a/starsky/starskytest/starsky.feature.packagetelemetry/Helpers/PackageTelemetryTest.cs +++ b/starsky/starskytest/starsky.feature.packagetelemetry/Helpers/PackageTelemetryTest.cs @@ -17,19 +17,6 @@ namespace starskytest.starsky.feature.packagetelemetry.Helpers [TestClass] public sealed class PackageTelemetryTest { - [TestMethod] - public void GetCurrentOsPlatformTest() - { - var content = PackageTelemetry.GetCurrentOsPlatform(); - Assert.IsNotNull(content); - - var allOsPlatforms = new List - { - OSPlatform.Linux, OSPlatform.Windows, OSPlatform.OSX, OSPlatform.FreeBSD - }; - Assert.IsTrue(allOsPlatforms.Contains(content.Value)); - } - [TestMethod] public void GetSystemDataTest() { @@ -55,6 +42,10 @@ public void GetSystemDataTest() Assert.IsTrue(systemData.Exists(p => p.Key == "DockerContainer")); Assert.IsTrue(systemData.Exists(p => p.Key == "CurrentCulture")); Assert.IsTrue(systemData.Exists(p => p.Key == "AspNetCoreEnvironment")); + Assert.IsTrue(systemData.Exists(p => p.Key == "RuntimeIdentifier")); + Assert.AreEqual(systemData.Find(p => p.Key == "RuntimeIdentifier").Value, + RuntimeInformation.RuntimeIdentifier); + Assert.IsTrue(systemData.Exists(p => p.Key == "DeviceId")); } [TestMethod] diff --git a/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs b/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs new file mode 100644 index 0000000000..c013dea38c --- /dev/null +++ b/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs @@ -0,0 +1,23 @@ +using System.Collections.Generic; +using System.Runtime.InteropServices; +using Microsoft.VisualStudio.TestTools.UnitTesting; +using starsky.foundation.platform.Helpers; + +namespace starskytest.starsky.foundation.platform.Helpers; + +[TestClass] +public class PlatformParserTest +{ + [TestMethod] + public void GetCurrentOsPlatformTest() + { + var content = PlatformParser.GetCurrentOsPlatform(); + Assert.IsNotNull(content); + + var allOsPlatforms = new List + { + OSPlatform.Linux, OSPlatform.Windows, OSPlatform.OSX, OSPlatform.FreeBSD + }; + Assert.IsTrue(allOsPlatforms.Contains(content.Value)); + } +} From 10fcd6e7d239dd4012c2684d5bc91996396bb62f Mon Sep 17 00:00:00 2001 From: Dion Date: Thu, 28 Mar 2024 19:06:47 +0100 Subject: [PATCH 03/11] rename and add setup --- .../DependenciesCliService.cs | 36 -------------- .../Interfaces/IDependenciesCliService.cs | 5 -- .../ExternalDependenciesService.cs | 48 +++++++++++++++++++ .../IExternalDependenciesService.cs | 8 ++++ ...arsky.feature.externaldependencies.csproj} | 2 + .../Services/GeoFileDownload.cs | 10 ++-- .../Helpers/ArgsHelper.cs | 27 ++++++++++- .../Helpers/PlatformParser.cs | 26 +++++++--- starsky/starsky.sln | 2 +- starsky/starskydependenciescli/Program.cs | 14 +++--- .../starskydependenciescli.csproj | 3 +- .../FakeMocks/FakeExifToolDownload.cs | 3 +- .../Services/ImportCliTest.cs | 2 +- .../Helpers/ArgsHelperTest.cs | 17 +++++++ .../Helpers/PlatformParserTest.cs | 16 +++++++ 15 files changed, 153 insertions(+), 66 deletions(-) delete mode 100644 starsky/starsky.feature.dependenciescli/DependenciesCliService.cs delete mode 100644 starsky/starsky.feature.dependenciescli/Interfaces/IDependenciesCliService.cs create mode 100644 starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs create mode 100644 starsky/starsky.feature.externaldependencies/Interfaces/IExternalDependenciesService.cs rename starsky/{starsky.feature.dependenciescli/starsky.feature.dependenciescli.csproj => starsky.feature.externaldependencies/starsky.feature.externaldependencies.csproj} (67%) diff --git a/starsky/starsky.feature.dependenciescli/DependenciesCliService.cs b/starsky/starsky.feature.dependenciescli/DependenciesCliService.cs deleted file mode 100644 index 57cf97735f..0000000000 --- a/starsky/starsky.feature.dependenciescli/DependenciesCliService.cs +++ /dev/null @@ -1,36 +0,0 @@ -using System.Runtime.InteropServices; -using starsky.foundation.database.Data; -using starsky.foundation.database.Helpers; -using starsky.foundation.platform.Helpers; -using starsky.foundation.platform.Interfaces; -using starsky.foundation.platform.Models; -using starsky.foundation.writemeta.Interfaces; - -namespace starsky.feature.dependenciescli; - -public class DependenciesCliService -{ - private IExifToolDownload _exifToolDownload; - private readonly ApplicationDbContext _dbContext; - private IWebLogger _logger; - private AppSettings _appSettings; - - public DependenciesCliService(IExifToolDownload exifToolDownload, - ApplicationDbContext dbContext, IWebLogger logger, AppSettings appSettings) - { - _exifToolDownload = exifToolDownload; - _dbContext = dbContext; - _logger = logger; - _appSettings = appSettings; - } - - public async Task Run(OSPlatform? currentPlatform = null, Architecture? architecture = null) - { - currentPlatform ??= PlatformParser.GetCurrentOsPlatform(); - await RunMigrations.Run(_dbContext, _logger, _appSettings); - await _exifToolDownload.DownloadExifTool(currentPlatform == OSPlatform.Windows); - - } - - -} diff --git a/starsky/starsky.feature.dependenciescli/Interfaces/IDependenciesCliService.cs b/starsky/starsky.feature.dependenciescli/Interfaces/IDependenciesCliService.cs deleted file mode 100644 index 9b3898d81a..0000000000 --- a/starsky/starsky.feature.dependenciescli/Interfaces/IDependenciesCliService.cs +++ /dev/null @@ -1,5 +0,0 @@ -namespace starsky.feature.dependenciescli.Interfaces; - -public interface IDependenciesCliService -{ -} diff --git a/starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs b/starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs new file mode 100644 index 0000000000..5b39ba8387 --- /dev/null +++ b/starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs @@ -0,0 +1,48 @@ +using System.Runtime.InteropServices; +using starsky.feature.externaldependencies.Interfaces; +using starsky.feature.geolookup.Interfaces; +using starsky.foundation.database.Data; +using starsky.foundation.database.Helpers; +using starsky.foundation.injection; +using starsky.foundation.platform.Helpers; +using starsky.foundation.platform.Interfaces; +using starsky.foundation.platform.Models; +using starsky.foundation.writemeta.Interfaces; + +namespace starsky.feature.externaldependencies; + +[Service(typeof(IExternalDependenciesService), InjectionLifetime = InjectionLifetime.Scoped)] +public class ExternalDependenciesService : IExternalDependenciesService +{ + private readonly IExifToolDownload _exifToolDownload; + private readonly ApplicationDbContext _dbContext; + private readonly IWebLogger _logger; + private readonly AppSettings _appSettings; + private readonly IGeoFileDownload _geoFileDownload; + + public ExternalDependenciesService(IExifToolDownload exifToolDownload, + ApplicationDbContext dbContext, IWebLogger logger, AppSettings appSettings, + IGeoFileDownload geoFileDownload) + { + _exifToolDownload = exifToolDownload; + _dbContext = dbContext; + _logger = logger; + _appSettings = appSettings; + _geoFileDownload = geoFileDownload; + } + + public async Task SetupAsync(string runtimeIdentifer) + { + } + + public async Task SetupAsync(OSPlatform? currentPlatform = null, + Architecture? architecture = null) + { + currentPlatform ??= PlatformParser.GetCurrentOsPlatform(); + + await RunMigrations.Run(_dbContext, _logger, _appSettings); + await _exifToolDownload.DownloadExifTool(currentPlatform == OSPlatform.Windows); + + await _geoFileDownload.DownloadAsync(); + } +} diff --git a/starsky/starsky.feature.externaldependencies/Interfaces/IExternalDependenciesService.cs b/starsky/starsky.feature.externaldependencies/Interfaces/IExternalDependenciesService.cs new file mode 100644 index 0000000000..2e44d64eac --- /dev/null +++ b/starsky/starsky.feature.externaldependencies/Interfaces/IExternalDependenciesService.cs @@ -0,0 +1,8 @@ +using System.Runtime.InteropServices; + +namespace starsky.feature.externaldependencies.Interfaces; + +public interface IExternalDependenciesService +{ + Task SetupAsync(OSPlatform? currentPlatform = null, Architecture? architecture = null); +} diff --git a/starsky/starsky.feature.dependenciescli/starsky.feature.dependenciescli.csproj b/starsky/starsky.feature.externaldependencies/starsky.feature.externaldependencies.csproj similarity index 67% rename from starsky/starsky.feature.dependenciescli/starsky.feature.dependenciescli.csproj rename to starsky/starsky.feature.externaldependencies/starsky.feature.externaldependencies.csproj index a7c1900e1b..0b33052ab8 100644 --- a/starsky/starsky.feature.dependenciescli/starsky.feature.dependenciescli.csproj +++ b/starsky/starsky.feature.externaldependencies/starsky.feature.externaldependencies.csproj @@ -4,9 +4,11 @@ net8.0 enable enable + starsky.feature.externaldependencies + diff --git a/starsky/starsky.feature.geolookup/Services/GeoFileDownload.cs b/starsky/starsky.feature.geolookup/Services/GeoFileDownload.cs index b1622a255a..9a7b0f2600 100644 --- a/starsky/starsky.feature.geolookup/Services/GeoFileDownload.cs +++ b/starsky/starsky.feature.geolookup/Services/GeoFileDownload.cs @@ -39,7 +39,7 @@ public async Task DownloadAsync() RemoveFailedDownload(); CreateDependenciesFolder(); const string https = "https://"; - const string admin1codesasciiTxt = "admin1CodesASCII.txt"; + const string admin1CodesasciiTxt = "admin1CodesASCII.txt"; if ( !_hostStorage.ExistFile( Path.Combine(_appSettings.DependenciesFolder, CountryName + ".txt")) ) @@ -59,18 +59,18 @@ await _httpClientHelper.Download(https + MirrorUrl + CountryName + ".zip", } if ( !_hostStorage.ExistFile( - Path.Combine(_appSettings.DependenciesFolder, admin1codesasciiTxt)) ) + Path.Combine(_appSettings.DependenciesFolder, admin1CodesasciiTxt)) ) { // code for the second administrative division, // a county in the US, see file admin2Codes.txt; varchar(80) var outputFile = Path.Combine(_appSettings.DependenciesFolder, - admin1codesasciiTxt); + admin1CodesasciiTxt); var baseResult = await _httpClientHelper.Download(https + - BaseUrl + admin1codesasciiTxt, outputFile); + BaseUrl + admin1CodesasciiTxt, outputFile); if ( !baseResult ) { await _httpClientHelper.Download(https + - MirrorUrl + admin1codesasciiTxt, outputFile); + MirrorUrl + admin1CodesasciiTxt, outputFile); } } } diff --git a/starsky/starsky.foundation.platform/Helpers/ArgsHelper.cs b/starsky/starsky.foundation.platform/Helpers/ArgsHelper.cs index 030ff415fb..36e7c51f69 100644 --- a/starsky/starsky.foundation.platform/Helpers/ArgsHelper.cs +++ b/starsky/starsky.foundation.platform/Helpers/ArgsHelper.cs @@ -5,6 +5,7 @@ using System.IO; using System.Linq; using System.Reflection; +using System.Runtime.InteropServices; using System.Text.RegularExpressions; using starsky.foundation.platform.Interfaces; using starsky.foundation.platform.Models; @@ -504,7 +505,7 @@ private void AppSpecificHelp() /// private void ShowVersions() { - var version = System.Runtime.InteropServices.RuntimeInformation.FrameworkDescription; + var version = RuntimeInformation.FrameworkDescription; _console.WriteLine($".NET Version - {version}"); _console.WriteLine($"Starsky Version - {_appSettings.AppVersion} " + "- build at: " + @@ -636,6 +637,30 @@ public static ConsoleOutputMode GetConsoleOutputMode(IReadOnlyList args) return outputMode; } + /// + /// Get input runtime + /// + /// arg list + /// path + public static OSPlatform? GetRuntime(IReadOnlyList args) + { + OSPlatform? outputMode = null; + for ( var arg = 0; arg < args.Count; arg++ ) + { + if ( args[arg]? + .Equals("--runtime", StringComparison.CurrentCultureIgnoreCase) != true || + ( arg + 1 ) == args.Count ) + { + continue; + } + + var runtimeItem = args[arg + 1]; + outputMode = PlatformParser.RuntimeIdentifier(runtimeItem); + } + + return outputMode; + } + /// /// Get the user input from -n or --name /// diff --git a/starsky/starsky.foundation.platform/Helpers/PlatformParser.cs b/starsky/starsky.foundation.platform/Helpers/PlatformParser.cs index d45de65925..a6c31c58eb 100644 --- a/starsky/starsky.foundation.platform/Helpers/PlatformParser.cs +++ b/starsky/starsky.foundation.platform/Helpers/PlatformParser.cs @@ -11,10 +11,7 @@ public static class PlatformParser OSPlatform? currentPlatform = null; foreach ( var platform in new List { - OSPlatform.Linux, - OSPlatform.Windows, - OSPlatform.OSX, - OSPlatform.FreeBSD + OSPlatform.Linux, OSPlatform.Windows, OSPlatform.OSX, OSPlatform.FreeBSD }.Where(RuntimeInformation.IsOSPlatform) ) { currentPlatform = platform; @@ -22,9 +19,24 @@ public static class PlatformParser return currentPlatform; } - - public static string GetCurrentArchitecture() + + public static OSPlatform? RuntimeIdentifier(string? runtimeIdentifier) { - return RuntimeInformation.RuntimeIdentifier; + if ( runtimeIdentifier == null ) + { + return null; + } + + if ( runtimeIdentifier.StartsWith("win-") ) + { + return OSPlatform.Windows; + } + + if ( runtimeIdentifier.StartsWith("linux-") ) + { + return OSPlatform.Linux; + } + + return runtimeIdentifier.StartsWith("osx-") ? OSPlatform.OSX : null; } } diff --git a/starsky/starsky.sln b/starsky/starsky.sln index 1d3ed6df59..221edd1aa0 100644 --- a/starsky/starsky.sln +++ b/starsky/starsky.sln @@ -168,7 +168,7 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "starsky.feature.desktop", " EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "starskydependenciescli", "starskydependenciescli\starskydependenciescli.csproj", "{CDFE2199-CB93-49E4-AF8A-98A3239F1D3F}" EndProject -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "starsky.feature.dependenciescli", "starsky.feature.dependenciescli\starsky.feature.dependenciescli.csproj", "{C34BB48B-0C65-4897-80A9-87CB1BDD5F48}" +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "starsky.feature.externaldependencies", "starsky.feature.externaldependencies\starsky.feature.externaldependencies.csproj", "{C34BB48B-0C65-4897-80A9-87CB1BDD5F48}" EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution diff --git a/starsky/starskydependenciescli/Program.cs b/starsky/starskydependenciescli/Program.cs index 9f69260df7..56f3abfbac 100644 --- a/starsky/starskydependenciescli/Program.cs +++ b/starsky/starskydependenciescli/Program.cs @@ -1,15 +1,13 @@ using System.Threading.Tasks; using Microsoft.Extensions.DependencyInjection; -using starsky.foundation.database.Data; +using starsky.feature.externaldependencies; +using starsky.feature.externaldependencies.Interfaces; using starsky.foundation.database.Helpers; using starsky.foundation.injection; using starsky.foundation.platform.Helpers; -using starsky.foundation.platform.Interfaces; using starsky.foundation.platform.Models; -using starsky.foundation.storage.Interfaces; using starsky.foundation.webtelemetry.Extensions; using starsky.foundation.webtelemetry.Helpers; -using starsky.foundation.writemeta.Interfaces; namespace starskyDependenciesCli { @@ -32,15 +30,15 @@ public static async Task Main(string[] args) services.AddOpenTelemetryMonitoring(appSettings); services.AddTelemetryLogging(appSettings); + services.AddScoped(); new SetupDatabaseTypes(appSettings, services).BuilderDb(); serviceProvider = services.BuildServiceProvider(); - var exifToolDownload = serviceProvider.GetRequiredService(); - var logger = serviceProvider.GetRequiredService(); + var dependenciesService = + serviceProvider.GetRequiredService(); - - exifToolDownload.DownloadExifTool() + await dependenciesService.SetupAsync(); } } } diff --git a/starsky/starskydependenciescli/starskydependenciescli.csproj b/starsky/starskydependenciescli/starskydependenciescli.csproj index 4812f874fe..4c05ca666b 100644 --- a/starsky/starskydependenciescli/starskydependenciescli.csproj +++ b/starsky/starskydependenciescli/starskydependenciescli.csproj @@ -17,6 +17,7 @@ + @@ -26,5 +27,5 @@ - + diff --git a/starsky/starskytest/FakeMocks/FakeExifToolDownload.cs b/starsky/starskytest/FakeMocks/FakeExifToolDownload.cs index ea5360ec1f..df52df19a1 100644 --- a/starsky/starskytest/FakeMocks/FakeExifToolDownload.cs +++ b/starsky/starskytest/FakeMocks/FakeExifToolDownload.cs @@ -7,9 +7,10 @@ namespace starskytest.FakeMocks public class FakeExifToolDownload : IExifToolDownload { public List Called { get; set; } = new List(); + public Task DownloadExifTool(bool isWindows, int minimumSize = 30) { - Called.Add(true); + Called.Add(isWindows); return Task.FromResult(true); } } diff --git a/starsky/starskytest/starsky.feature.import/Services/ImportCliTest.cs b/starsky/starskytest/starsky.feature.import/Services/ImportCliTest.cs index d4bd569c1a..ae3c2e53a6 100644 --- a/starsky/starskytest/starsky.feature.import/Services/ImportCliTest.cs +++ b/starsky/starskytest/starsky.feature.import/Services/ImportCliTest.cs @@ -22,7 +22,7 @@ public async Task ImporterCli_CheckIfExifToolIsCalled() new FakeIImport(new FakeSelectorStorage()), new AppSettings { TempFolder = "/___not___found_" }, fakeConsole, new FakeIWebLogger(), fakeExifToolDownload) - .Importer(new List().ToArray()); + .Importer([]); Assert.IsTrue(fakeExifToolDownload.Called.Count != 0); } diff --git a/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs b/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs index f65ccaddb7..cd0d5fe542 100644 --- a/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs +++ b/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs @@ -629,5 +629,22 @@ public void Name() var args = new List { "-n", "test" }.ToArray(); Assert.AreEqual("test", ArgsHelper.GetName(args)); } + + [DataTestMethod] + [DataRow("osx-arm64", "OSX")] + [DataRow("osx-x64", "OSX")] + [DataRow("linux-x64", "LINUX")] + [DataRow("linux-arm", "LINUX")] + [DataRow("linux-arm64", "LINUX")] + [DataRow("win-x64", "WINDOWS")] + [DataRow("win-x86", "WINDOWS")] + [DataRow("win-arm64", "WINDOWS")] + [DataRow("test", "")] + [DataRow(null, "")] + public void GetRuntimeTest(string input, string expected) + { + var args = new List { "--runtime", input }.ToArray(); + Assert.AreEqual(expected, ArgsHelper.GetRuntime(args).ToString()); + } } } diff --git a/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs b/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs index c013dea38c..e836675c6a 100644 --- a/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs +++ b/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs @@ -20,4 +20,20 @@ public void GetCurrentOsPlatformTest() }; Assert.IsTrue(allOsPlatforms.Contains(content.Value)); } + + [DataTestMethod] + [DataRow("osx-arm64", "OSX")] + [DataRow("osx-x64", "OSX")] + [DataRow("linux-x64", "LINUX")] + [DataRow("linux-arm", "LINUX")] + [DataRow("linux-arm64", "LINUX")] + [DataRow("win-x64", "WINDOWS")] + [DataRow("win-x86", "WINDOWS")] + [DataRow("win-arm64", "WINDOWS")] + [DataRow("test", "")] + [DataRow(null, "")] + public void RuntimeIdentifierTest(string input, string expected) + { + Assert.AreEqual(expected, PlatformParser.RuntimeIdentifier(input).ToString()); + } } From af83e2a808422363e8cc028d082fa3b45464226a Mon Sep 17 00:00:00 2001 From: Dion Date: Thu, 28 Mar 2024 19:19:05 +0100 Subject: [PATCH 04/11] add code --- .../ExternalDependenciesService.cs | 3 +- .../ExternalDependenciesServiceTest.cs | 58 +++++++++++++++++++ .../DataProtection/SqlXmlRepositoryTest.cs | 2 +- starsky/starskytest/starskytest.csproj | 1 + 4 files changed, 62 insertions(+), 2 deletions(-) create mode 100644 starsky/starskytest/starsky.feature.externaldependencies/ExternalDependenciesServiceTest.cs diff --git a/starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs b/starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs index 5b39ba8387..67eee7f799 100644 --- a/starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs +++ b/starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs @@ -31,8 +31,9 @@ public ExternalDependenciesService(IExifToolDownload exifToolDownload, _geoFileDownload = geoFileDownload; } - public async Task SetupAsync(string runtimeIdentifer) + public async Task SetupAsync(List args) { + await SetupAsync(ArgsHelper.GetRuntime(args)); } public async Task SetupAsync(OSPlatform? currentPlatform = null, diff --git a/starsky/starskytest/starsky.feature.externaldependencies/ExternalDependenciesServiceTest.cs b/starsky/starskytest/starsky.feature.externaldependencies/ExternalDependenciesServiceTest.cs new file mode 100644 index 0000000000..b96ae8c272 --- /dev/null +++ b/starsky/starskytest/starsky.feature.externaldependencies/ExternalDependenciesServiceTest.cs @@ -0,0 +1,58 @@ +using System.Threading.Tasks; +using Microsoft.EntityFrameworkCore; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.VisualStudio.TestTools.UnitTesting; +using starsky.feature.externaldependencies; +using starsky.foundation.database.Data; +using starsky.foundation.platform.Models; +using starskytest.FakeMocks; + +namespace starskytest.starsky.feature.externaldependencies; + +[TestClass] +public class ExternalDependenciesServiceTest +{ + private readonly ApplicationDbContext _dbContext; + + private static IServiceScopeFactory CreateNewScope() + { + var services = new ServiceCollection(); + services.AddDbContext(options => + options.UseInMemoryDatabase(nameof(ExternalDependenciesServiceTest))); + var serviceProvider = services.BuildServiceProvider(); + return serviceProvider.GetRequiredService(); + } + + public ExternalDependenciesServiceTest() + { + var serviceScope = CreateNewScope(); + var scope = serviceScope.CreateScope(); + _dbContext = scope.ServiceProvider.GetRequiredService(); + } + + [TestMethod] + public async Task ExternalDependenciesServiceTest_ExifTool_Default() + { + var fakeExifToolDownload = new FakeExifToolDownload(); + var externalDependenciesService = new ExternalDependenciesService(fakeExifToolDownload, + _dbContext, new FakeIWebLogger(), new AppSettings(), new FakeIGeoFileDownload()); + await externalDependenciesService.SetupAsync([]); + + Assert.AreEqual(new AppSettings().IsWindows, fakeExifToolDownload.Called[0]); + } + + [DataTestMethod] + [DataRow("osx-arm64", false)] + [DataRow("linux-x64", false)] + [DataRow("win-x64", true)] + public async Task ExternalDependenciesServiceTest_ExifTool_DataTestMethod(string input, bool expected) + { + var fakeExifToolDownload = new FakeExifToolDownload(); + var externalDependenciesService = new ExternalDependenciesService(fakeExifToolDownload, + _dbContext, new FakeIWebLogger(), new AppSettings(), new FakeIGeoFileDownload()); + + await externalDependenciesService.SetupAsync(["--runtime", input]); + + Assert.AreEqual(expected, fakeExifToolDownload.Called[0]); + } +} diff --git a/starsky/starskytest/starsky.foundation.database/DataProtection/SqlXmlRepositoryTest.cs b/starsky/starskytest/starsky.foundation.database/DataProtection/SqlXmlRepositoryTest.cs index 9ea09b7bde..248a15872d 100644 --- a/starsky/starskytest/starsky.foundation.database/DataProtection/SqlXmlRepositoryTest.cs +++ b/starsky/starskytest/starsky.foundation.database/DataProtection/SqlXmlRepositoryTest.cs @@ -22,7 +22,7 @@ public class SqlXmlRepositoryTest private readonly SqlXmlRepository _repository; private readonly ApplicationDbContext _dbContext; - private IServiceScopeFactory CreateNewScope() + private static IServiceScopeFactory CreateNewScope() { var services = new ServiceCollection(); services.AddDbContext(options => options.UseInMemoryDatabase(nameof(SqlXmlRepositoryTest))); diff --git a/starsky/starskytest/starskytest.csproj b/starsky/starskytest/starskytest.csproj index 0a684b8cdd..dc0e6727a6 100644 --- a/starsky/starskytest/starskytest.csproj +++ b/starsky/starskytest/starskytest.csproj @@ -44,6 +44,7 @@ + From 59fd5ed00d010677b5ce2701aece8ad1c706c6df Mon Sep 17 00:00:00 2001 From: Dion Date: Fri, 29 Mar 2024 15:06:53 +0100 Subject: [PATCH 05/11] WIP --- .../ExternalDependenciesService.cs | 23 ++++++--- .../IExternalDependenciesService.cs | 4 +- .../Helpers/ArgsHelper.cs | 14 +++--- .../Helpers/PlatformParser.cs | 50 ++++++++++++++++--- starsky/starskydependenciescli/Program.cs | 2 +- .../ExternalDependenciesServiceTest.cs | 41 +++++++++++++-- .../Helpers/ArgsHelperTest.cs | 2 +- .../Helpers/PlatformParserTest.cs | 26 +++++----- 8 files changed, 120 insertions(+), 42 deletions(-) diff --git a/starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs b/starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs index 67eee7f799..b4edcd19cd 100644 --- a/starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs +++ b/starsky/starsky.feature.externaldependencies/ExternalDependenciesService.cs @@ -31,18 +31,29 @@ public ExternalDependenciesService(IExifToolDownload exifToolDownload, _geoFileDownload = geoFileDownload; } - public async Task SetupAsync(List args) + public async Task SetupAsync(string[] args) { await SetupAsync(ArgsHelper.GetRuntime(args)); } - public async Task SetupAsync(OSPlatform? currentPlatform = null, - Architecture? architecture = null) + public async Task SetupAsync(List<(OSPlatform?, Architecture?)> currentPlatforms) { - currentPlatform ??= PlatformParser.GetCurrentOsPlatform(); - await RunMigrations.Run(_dbContext, _logger, _appSettings); - await _exifToolDownload.DownloadExifTool(currentPlatform == OSPlatform.Windows); + + + if ( currentPlatforms.Count == 0 ) + { + currentPlatforms = + [ + ( PlatformParser.GetCurrentOsPlatform(), + PlatformParser.GetCurrentArchitecture() ) + ]; + } + + foreach ( var (osPlatform, _) in currentPlatforms ) + { + await _exifToolDownload.DownloadExifTool(osPlatform == OSPlatform.Windows); + } await _geoFileDownload.DownloadAsync(); } diff --git a/starsky/starsky.feature.externaldependencies/Interfaces/IExternalDependenciesService.cs b/starsky/starsky.feature.externaldependencies/Interfaces/IExternalDependenciesService.cs index 2e44d64eac..da6308a8c4 100644 --- a/starsky/starsky.feature.externaldependencies/Interfaces/IExternalDependenciesService.cs +++ b/starsky/starsky.feature.externaldependencies/Interfaces/IExternalDependenciesService.cs @@ -1,8 +1,6 @@ -using System.Runtime.InteropServices; - namespace starsky.feature.externaldependencies.Interfaces; public interface IExternalDependenciesService { - Task SetupAsync(OSPlatform? currentPlatform = null, Architecture? architecture = null); + Task SetupAsync(string[] args); } diff --git a/starsky/starsky.foundation.platform/Helpers/ArgsHelper.cs b/starsky/starsky.foundation.platform/Helpers/ArgsHelper.cs index 36e7c51f69..5d974568dc 100644 --- a/starsky/starsky.foundation.platform/Helpers/ArgsHelper.cs +++ b/starsky/starsky.foundation.platform/Helpers/ArgsHelper.cs @@ -642,23 +642,25 @@ public static ConsoleOutputMode GetConsoleOutputMode(IReadOnlyList args) /// /// arg list /// path - public static OSPlatform? GetRuntime(IReadOnlyList args) + public static List<(OSPlatform?, Architecture?)> GetRuntime(IReadOnlyList args) { - OSPlatform? outputMode = null; + var outputModeList = new List<(OSPlatform?, Architecture?)>(); for ( var arg = 0; arg < args.Count; arg++ ) { if ( args[arg]? - .Equals("--runtime", StringComparison.CurrentCultureIgnoreCase) != true || - ( arg + 1 ) == args.Count ) + .Equals("--runtime", + StringComparison.CurrentCultureIgnoreCase) != true || + arg + 1 == args.Count ) { continue; } var runtimeItem = args[arg + 1]; - outputMode = PlatformParser.RuntimeIdentifier(runtimeItem); + var parsedRuntimeList = PlatformParser.RuntimeIdentifier(runtimeItem); + outputModeList.AddRange(parsedRuntimeList); } - return outputMode; + return outputModeList; } /// diff --git a/starsky/starsky.foundation.platform/Helpers/PlatformParser.cs b/starsky/starsky.foundation.platform/Helpers/PlatformParser.cs index a6c31c58eb..be979cd8ee 100644 --- a/starsky/starsky.foundation.platform/Helpers/PlatformParser.cs +++ b/starsky/starsky.foundation.platform/Helpers/PlatformParser.cs @@ -20,23 +20,57 @@ public static class PlatformParser return currentPlatform; } - public static OSPlatform? RuntimeIdentifier(string? runtimeIdentifier) + public static List<(OSPlatform?, Architecture?)> RuntimeIdentifier(string? runtimeIdentifiers) { - if ( runtimeIdentifier == null ) + var result = new List<(OSPlatform?, Architecture?)>(); + if ( runtimeIdentifiers == null ) { - return null; + return []; } - if ( runtimeIdentifier.StartsWith("win-") ) + var runtimes = runtimeIdentifiers.Split(",").Where(x => !string.IsNullOrEmpty(x)).ToList(); + foreach ( var runtime in runtimes ) + { + var singleRuntimeIdentifier = SingleRuntimeIdentifier(runtime); + if ( singleRuntimeIdentifier is { Item1: not null, Item2: not null } ) + { + result.Add(singleRuntimeIdentifier); + } + } + + return result; + } + + private static (OSPlatform?, Architecture?) SingleRuntimeIdentifier(string? runtimeIdentifier) + { + if ( runtimeIdentifier == null ) { - return OSPlatform.Windows; + return ( null, null ); } - if ( runtimeIdentifier.StartsWith("linux-") ) + switch ( runtimeIdentifier ) { - return OSPlatform.Linux; + case "win-x64": + return ( OSPlatform.Windows, Architecture.X64 ); + case "win-arm64": + return ( OSPlatform.Windows, Architecture.Arm64 ); + case "linux-arm": + return ( OSPlatform.Linux, Architecture.Arm ); + case "linux-arm64": + return ( OSPlatform.Linux, Architecture.Arm64 ); + case "linux-x64": + return ( OSPlatform.Linux, Architecture.X64 ); + case "osx-x64": + return ( OSPlatform.OSX, Architecture.X64 ); + case "osx-arm64": + return ( OSPlatform.OSX, Architecture.Arm64 ); + default: + return ( null, null ); } + } - return runtimeIdentifier.StartsWith("osx-") ? OSPlatform.OSX : null; + public static Architecture GetCurrentArchitecture() + { + return Architecture.Wasm; } } diff --git a/starsky/starskydependenciescli/Program.cs b/starsky/starskydependenciescli/Program.cs index 56f3abfbac..fa012c4703 100644 --- a/starsky/starskydependenciescli/Program.cs +++ b/starsky/starskydependenciescli/Program.cs @@ -38,7 +38,7 @@ public static async Task Main(string[] args) var dependenciesService = serviceProvider.GetRequiredService(); - await dependenciesService.SetupAsync(); + await dependenciesService.SetupAsync(args); } } } diff --git a/starsky/starskytest/starsky.feature.externaldependencies/ExternalDependenciesServiceTest.cs b/starsky/starskytest/starsky.feature.externaldependencies/ExternalDependenciesServiceTest.cs index b96ae8c272..b24955f6a8 100644 --- a/starsky/starskytest/starsky.feature.externaldependencies/ExternalDependenciesServiceTest.cs +++ b/starsky/starskytest/starsky.feature.externaldependencies/ExternalDependenciesServiceTest.cs @@ -1,3 +1,4 @@ +using System.Collections.Generic; using System.Threading.Tasks; using Microsoft.EntityFrameworkCore; using Microsoft.Extensions.DependencyInjection; @@ -34,9 +35,13 @@ public ExternalDependenciesServiceTest() public async Task ExternalDependenciesServiceTest_ExifTool_Default() { var fakeExifToolDownload = new FakeExifToolDownload(); + var appSettings = new AppSettings + { + DatabaseType = AppSettings.DatabaseTypeList.InMemoryDatabase + }; var externalDependenciesService = new ExternalDependenciesService(fakeExifToolDownload, - _dbContext, new FakeIWebLogger(), new AppSettings(), new FakeIGeoFileDownload()); - await externalDependenciesService.SetupAsync([]); + _dbContext, new FakeIWebLogger(), appSettings, new FakeIGeoFileDownload()); + await externalDependenciesService.SetupAsync(new List().ToArray()); Assert.AreEqual(new AppSettings().IsWindows, fakeExifToolDownload.Called[0]); } @@ -45,14 +50,40 @@ public async Task ExternalDependenciesServiceTest_ExifTool_Default() [DataRow("osx-arm64", false)] [DataRow("linux-x64", false)] [DataRow("win-x64", true)] - public async Task ExternalDependenciesServiceTest_ExifTool_DataTestMethod(string input, bool expected) + public async Task ExternalDependenciesServiceTest_ExifTool_Single_DataTestMethod(string input, + bool expected) { var fakeExifToolDownload = new FakeExifToolDownload(); + var appSettings = new AppSettings + { + DatabaseType = AppSettings.DatabaseTypeList.InMemoryDatabase + }; var externalDependenciesService = new ExternalDependenciesService(fakeExifToolDownload, - _dbContext, new FakeIWebLogger(), new AppSettings(), new FakeIGeoFileDownload()); - + _dbContext, new FakeIWebLogger(), appSettings, new FakeIGeoFileDownload()); + await externalDependenciesService.SetupAsync(["--runtime", input]); Assert.AreEqual(expected, fakeExifToolDownload.Called[0]); } + + [DataTestMethod] + [DataRow("osx-arm64,osx-x64", false, false)] + [DataRow("linux-x64,osx-x64", false, false)] + [DataRow("win-x64,osx-x64", true, false)] + public async Task ExternalDependenciesServiceTest_ExifTool_Multiple_DataTestMethod(string input, + bool expected1, bool expected2) + { + var appSettings = new AppSettings + { + DatabaseType = AppSettings.DatabaseTypeList.InMemoryDatabase + }; + var fakeExifToolDownload = new FakeExifToolDownload(); + var externalDependenciesService = new ExternalDependenciesService(fakeExifToolDownload, + _dbContext, new FakeIWebLogger(), appSettings, new FakeIGeoFileDownload()); + + await externalDependenciesService.SetupAsync(["--runtime", input]); + + Assert.AreEqual(expected1, fakeExifToolDownload.Called[0]); + Assert.AreEqual(expected2, fakeExifToolDownload.Called[1]); + } } diff --git a/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs b/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs index cd0d5fe542..42656a606d 100644 --- a/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs +++ b/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs @@ -644,7 +644,7 @@ public void Name() public void GetRuntimeTest(string input, string expected) { var args = new List { "--runtime", input }.ToArray(); - Assert.AreEqual(expected, ArgsHelper.GetRuntime(args).ToString()); + Assert.AreEqual(expected, ArgsHelper.GetRuntime(args)[0].Item1.ToString()); } } } diff --git a/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs b/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs index e836675c6a..69a68b57a2 100644 --- a/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs +++ b/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs @@ -22,18 +22,20 @@ public void GetCurrentOsPlatformTest() } [DataTestMethod] - [DataRow("osx-arm64", "OSX")] - [DataRow("osx-x64", "OSX")] - [DataRow("linux-x64", "LINUX")] - [DataRow("linux-arm", "LINUX")] - [DataRow("linux-arm64", "LINUX")] - [DataRow("win-x64", "WINDOWS")] - [DataRow("win-x86", "WINDOWS")] - [DataRow("win-arm64", "WINDOWS")] - [DataRow("test", "")] - [DataRow(null, "")] - public void RuntimeIdentifierTest(string input, string expected) + [DataRow("osx-arm64", "OSX", "Arm64")] + [DataRow("osx-x64", "OSX", "X64")] + [DataRow("linux-x64", "LINUX", "X64")] + [DataRow("linux-arm", "LINUX", "Arm")] + [DataRow("linux-arm64", "LINUX", "Arm64")] + [DataRow("win-x64", "WINDOWS", "X64")] + [DataRow("win-x86", "WINDOWS", "X86")] + [DataRow("win-arm64", "WINDOWS", "Arm64")] + [DataRow("test", "", "")] + [DataRow(null, "", "")] + public void RuntimeIdentifierTest(string input, string expectedOs, string expectedArch) { - Assert.AreEqual(expected, PlatformParser.RuntimeIdentifier(input).ToString()); + var result = PlatformParser.RuntimeIdentifier(input); + Assert.AreEqual(expectedOs, result[0].Item1.ToString()); + Assert.AreEqual(expectedArch, result[0].Item2.ToString()); } } From 70eb824fd3113f6501bcf247dee9b6e417b80147 Mon Sep 17 00:00:00 2001 From: Dion Date: Thu, 4 Apr 2024 08:55:10 +0200 Subject: [PATCH 06/11] WIP --- starsky/build/Build.cs | 8 +- starsky/build/helpers/DotnetGenericHelper.cs | 18 ++- .../Services/ExifToolDownload.cs | 150 +++++++++++------- starsky/starskydependenciescli/Program.cs | 7 +- .../default-init-launchSettings.json | 0 .../starskydependenciescli.csproj | 1 + .../Helpers/ArgsHelperTest.cs | 13 +- .../Helpers/PlatformParserTest.cs | 13 +- 8 files changed, 142 insertions(+), 68 deletions(-) rename starsky/starskydependenciescli/{ => Properties}/default-init-launchSettings.json (100%) diff --git a/starsky/build/Build.cs b/starsky/build/Build.cs index 3a4c74ce55..9132613248 100644 --- a/starsky/build/Build.cs +++ b/starsky/build/Build.cs @@ -181,7 +181,7 @@ List GetRuntimesWithoutGeneric() /// /// Link to GeoCli.csproj /// - const string GeoCliCsproj = "starskygeocli/starskygeocli.csproj"; + const string starskyDependenciesCliCsproj = "starskydependenciescli/starskydependenciescli.csproj"; /// /// Npm and node are required for preflight checks and building frontend code @@ -315,7 +315,8 @@ void ShowSettingsInfo() Configuration); DotnetTestHelper.TestNetCoreGenericCommand(Configuration, IsUnitTestDisabled()); - DotnetGenericHelper.DownloadDependencies(Configuration, GeoCliCsproj, + DotnetGenericHelper.DownloadDependencies(Configuration, starskyDependenciesCliCsproj, + GetRuntimesWithoutGeneric(), NoDependencies, GenericRuntimeName); MergeCoverageFiles.Merge(IsUnitTestDisabled()); SonarQube.SonarEnd(IsUnitTestDisabled(), NoSonar); @@ -330,7 +331,8 @@ void ShowSettingsInfo() .Executes(() => { DotnetGenericHelper.DownloadDependencies(Configuration, - "starskygeocli/starskygeocli.csproj", NoDependencies, + "starskydependenciescli/starskydependenciescli.csproj", + GetRuntimesWithoutGeneric(), NoDependencies, GenericRuntimeName); DotnetRuntimeSpecificHelper.CopyDependenciesFiles(NoDependencies, GenericRuntimeName, GetRuntimesWithoutGeneric()); diff --git a/starsky/build/helpers/DotnetGenericHelper.cs b/starsky/build/helpers/DotnetGenericHelper.cs index 9cc4a5a85f..489dbe60d6 100644 --- a/starsky/build/helpers/DotnetGenericHelper.cs +++ b/starsky/build/helpers/DotnetGenericHelper.cs @@ -1,4 +1,5 @@ using System; +using System.Collections.Generic; using System.IO; using build; using Nuke.Common.ProjectModel; @@ -43,11 +44,12 @@ public static void BuildNetCoreGenericCommand(Solution solution, /// Download Exiftool and geo deps /// /// is Release - /// geo.csproj file + /// dependenciesCli.csproj file + /// which version are downloaded /// skip this step if true (external deps) /// genericNetcoreFolder public static void DownloadDependencies(Configuration configuration, - string geoCliCsproj, bool noDependencies, + string dependenciesCliCsproj, List getRuntimesWithoutGeneric, bool noDependencies, string genericNetcoreFolder) { if ( noDependencies ) @@ -56,6 +58,12 @@ public static void DownloadDependencies(Configuration configuration, return; } + if ( getRuntimesWithoutGeneric.Count == 0 ) + { + Log.Information("skip deps build due generic build"); + return; + } + var genericDepsFullPath = Path.Combine(BasePath(), genericNetcoreFolder, "dependencies"); Log.Information("genericDepsFullPath: {GenericDepsFullPath}", genericDepsFullPath); @@ -65,16 +73,16 @@ public static void DownloadDependencies(Configuration configuration, Environment.SetEnvironmentVariable("app__DependenciesFolder", genericDepsFullPath); Log.Information("Next: DownloadDependencies"); Log.Information("Run: {Path}", Path.Combine( - WorkingDirectory.GetSolutionParentFolder(), geoCliCsproj) + WorkingDirectory.GetSolutionParentFolder(), dependenciesCliCsproj) ); DotNetRun(p => p .SetConfiguration(configuration) .EnableNoRestore() .EnableNoBuild() - .SetApplicationArguments("--runtime linux-x64,win-x64") + .SetApplicationArguments($"--runtime {string.Join(',', getRuntimesWithoutGeneric)}") .SetProjectFile(Path.Combine(WorkingDirectory.GetSolutionParentFolder(), - geoCliCsproj))); + dependenciesCliCsproj))); } catch ( Exception exception ) { diff --git a/starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs b/starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs index dc268ebbb3..45f40cc2d3 100644 --- a/starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs +++ b/starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs @@ -20,11 +20,16 @@ using starsky.foundation.writemeta.Interfaces; [assembly: InternalsVisibleTo("starskytest")] + namespace starsky.foundation.writemeta.Services { [Service(typeof(IExifToolDownload), InjectionLifetime = InjectionLifetime.Singleton)] - [SuppressMessage("Usage", "S1075:Refactor your code not to use hardcoded absolute paths or URIs", Justification = "Source of files")] - [SuppressMessage("Usage", "S4790:Make sure this weak hash algorithm is not used in a sensitive context here.", Justification = "Safe")] + [SuppressMessage("Usage", + "S1075:Refactor your code not to use hardcoded absolute paths or URIs", + Justification = "Source of files")] + [SuppressMessage("Usage", + "S4790:Make sure this weak hash algorithm is not used in a sensitive context here.", + Justification = "Safe")] public sealed class ExifToolDownload : IExifToolDownload { private readonly IHttpClientHelper _httpClientHelper; @@ -33,11 +38,18 @@ public sealed class ExifToolDownload : IExifToolDownload private readonly IWebLogger _logger; private const string CheckSumLocation = "https://exiftool.org/checksums.txt"; - private const string CheckSumLocationMirror = "https://qdraw.nl/special/mirror/exiftool/checksums.txt"; - private const string ExiftoolDownloadBasePath = "https://exiftool.org/"; // with slash at the end - private const string ExiftoolDownloadBasePathMirror = "https://qdraw.nl/special/mirror/exiftool/"; // with slash at the end - public ExifToolDownload(IHttpClientHelper httpClientHelper, AppSettings appSettings, IWebLogger logger) + private const string CheckSumLocationMirror = + "https://qdraw.nl/special/mirror/exiftool/checksums.txt"; + + private const string + ExiftoolDownloadBasePath = "https://exiftool.org/"; // with slash at the end + + private const string ExiftoolDownloadBasePathMirror = + "https://qdraw.nl/special/mirror/exiftool/"; // with slash at the end + + public ExifToolDownload(IHttpClientHelper httpClientHelper, AppSettings appSettings, + IWebLogger logger) { _httpClientHelper = httpClientHelper; _appSettings = appSettings; @@ -45,7 +57,8 @@ public ExifToolDownload(IHttpClientHelper httpClientHelper, AppSettings appSetti _logger = logger; } - internal ExifToolDownload(IHttpClientHelper httpClientHelper, AppSettings appSettings, IWebLogger logger, IStorage storage) + internal ExifToolDownload(IHttpClientHelper httpClientHelper, AppSettings appSettings, + IWebLogger logger, IStorage storage) { _httpClientHelper = httpClientHelper; _appSettings = appSettings; @@ -61,8 +74,8 @@ internal ExifToolDownload(IHttpClientHelper httpClientHelper, AppSettings appSet /// public async Task DownloadExifTool(bool isWindows, int minimumSize = 30) { - if ( _appSettings.ExiftoolSkipDownloadOnStartup == true || ( - _appSettings.AddSwaggerExport == true && _appSettings.AddSwaggerExportExitAfter == true ) ) + if ( _appSettings.ExiftoolSkipDownloadOnStartup == true || _appSettings is + { AddSwaggerExport: true, AddSwaggerExportExitAfter: true } ) { var name = _appSettings.ExiftoolSkipDownloadOnStartup == true ? "ExiftoolSkipDownloadOnStartup" @@ -74,22 +87,25 @@ public async Task DownloadExifTool(bool isWindows, int minimumSize = 30) CreateDirectoryDependenciesFolderIfNotExists(); if ( isWindows && - ( !_hostFileSystemStorage.ExistFile(ExeExifToolWindowsFullFilePath()) || - _hostFileSystemStorage.Info(ExeExifToolWindowsFullFilePath()).Size <= minimumSize ) ) + ( !_hostFileSystemStorage.ExistFile(ExeExifToolWindowsFullFilePath()) || + _hostFileSystemStorage.Info(ExeExifToolWindowsFullFilePath()).Size <= + minimumSize ) ) { return await StartDownloadForWindows(); } if ( !isWindows && - ( !_hostFileSystemStorage.ExistFile(ExeExifToolUnixFullFilePath()) || - _hostFileSystemStorage.Info(ExeExifToolUnixFullFilePath()).Size <= minimumSize ) ) + ( !_hostFileSystemStorage.ExistFile(ExeExifToolUnixFullFilePath()) || + _hostFileSystemStorage.Info(ExeExifToolUnixFullFilePath()).Size <= + minimumSize ) ) { return await StartDownloadForUnix(); } if ( _appSettings.IsVerbose() ) { - var debugPath = isWindows ? ExeExifToolWindowsFullFilePath() + var debugPath = isWindows + ? ExeExifToolWindowsFullFilePath() : ExeExifToolUnixFullFilePath(); _logger.LogInformation($"[DownloadExifTool] {debugPath}"); } @@ -103,8 +119,9 @@ public async Task DownloadExifTool(bool isWindows, int minimumSize = 30) private void CreateDirectoryDependenciesFolderIfNotExists() { if ( _hostFileSystemStorage.ExistFolder(_appSettings - .DependenciesFolder) ) return; - _logger.LogInformation("[DownloadExifTool] Create Directory: " + _appSettings.DependenciesFolder); + .DependenciesFolder) ) return; + _logger.LogInformation("[DownloadExifTool] Create Directory: " + + _appSettings.DependenciesFolder); _hostFileSystemStorage.CreateDirectory(_appSettings.DependenciesFolder); } @@ -115,13 +132,17 @@ private void CreateDirectoryDependenciesFolderIfNotExists() { return checksums; } - _logger.LogError($"Checksum loading failed {CheckSumLocation}, next retry from mirror ~ error > " + checksums.Value); + + _logger.LogError( + $"Checksum loading failed {CheckSumLocation}, next retry from mirror ~ error > " + + checksums.Value); checksums = await _httpClientHelper.ReadString(CheckSumLocationMirror); if ( checksums.Key ) return new KeyValuePair(false, checksums.Value); _logger.LogError($"Checksum loading failed {CheckSumLocationMirror}" + - $", next stop; please connect to internet and restart app ~ error > " + checksums.Value); + $", next stop; please connect to internet and restart app ~ error > " + + checksums.Value); return null; } @@ -137,7 +158,8 @@ internal async Task StartDownloadForUnix() internal static string GetUnixTarGzFromChecksum(string checksumsValue) { // (?<=SHA1\()Image-ExifTool-[\d\.]+\.zip - var regexExifToolForWindowsName = new Regex(@"(?<=SHA1\()Image-ExifTool-[0-9\.]+\.tar.gz", + var regexExifToolForWindowsName = new Regex( + @"(?<=SHA1\()Image-ExifTool-[0-9\.]+\.tar.gz", RegexOptions.None, TimeSpan.FromMilliseconds(100)); return regexExifToolForWindowsName.Match(checksumsValue).Value; } @@ -145,47 +167,56 @@ internal static string GetUnixTarGzFromChecksum(string checksumsValue) private string ExeExifToolUnixFullFilePath() { var path = Path.Combine(_appSettings.DependenciesFolder, - "exiftool-unix", - "exiftool"); + "exiftool-unix", + "exiftool"); return path; } internal async Task DownloadForUnix(string matchExifToolForUnixName, - string[] getChecksumsFromTextFile, bool downloadFromMirror = false) + IEnumerable getChecksumsFromTextFile, bool downloadFromMirror = false) { + if ( _hostFileSystemStorage.ExistFile(ExeExifToolUnixFullFilePath()) ) return true; - if ( _hostFileSystemStorage.ExistFile( - ExeExifToolUnixFullFilePath()) ) return true; - - var tarGzArchiveFullFilePath = Path.Combine(_appSettings.DependenciesFolder, "exiftool.tar.gz"); + var tarGzArchiveFullFilePath = + Path.Combine(_appSettings.TempFolder, "exiftool.tar.gz"); var url = $"{ExiftoolDownloadBasePath}{matchExifToolForUnixName}"; - if ( downloadFromMirror ) url = $"{ExiftoolDownloadBasePathMirror}{matchExifToolForUnixName}"; + if ( downloadFromMirror ) + url = $"{ExiftoolDownloadBasePathMirror}{matchExifToolForUnixName}"; var unixDownloaded = await _httpClientHelper.Download(url, tarGzArchiveFullFilePath); if ( !unixDownloaded ) { - throw new HttpRequestException($"file is not downloaded {matchExifToolForUnixName}"); + throw new HttpRequestException( + $"file is not downloaded {matchExifToolForUnixName}"); } + if ( !CheckSha1(tarGzArchiveFullFilePath, getChecksumsFromTextFile) ) { - throw new HttpRequestException($"checksum for {tarGzArchiveFullFilePath} is not valid"); + throw new HttpRequestException( + $"checksum for {tarGzArchiveFullFilePath} is not valid"); } await new TarBal(_hostFileSystemStorage).ExtractTarGz( - _hostFileSystemStorage.ReadStream(tarGzArchiveFullFilePath), _appSettings.DependenciesFolder, CancellationToken.None); + _hostFileSystemStorage.ReadStream(tarGzArchiveFullFilePath), + _appSettings.DependenciesFolder, CancellationToken.None); - var imageExifToolVersionFolder = _hostFileSystemStorage.GetDirectories(_appSettings.DependenciesFolder) - .FirstOrDefault(p => p.StartsWith(Path.Combine(_appSettings.DependenciesFolder, "Image-ExifTool-"))); + var imageExifToolVersionFolder = _hostFileSystemStorage + .GetDirectories(_appSettings.DependenciesFolder) + .FirstOrDefault(p => + p.StartsWith(Path.Combine(_appSettings.DependenciesFolder, "Image-ExifTool-"))); if ( imageExifToolVersionFolder != null ) { - var exifToolUnixFolderFullFilePath = Path.Combine(_appSettings.DependenciesFolder, "exiftool-unix"); + var exifToolUnixFolderFullFilePath = + Path.Combine(_appSettings.DependenciesFolder, "exiftool-unix"); if ( _hostFileSystemStorage.ExistFolder(exifToolUnixFolderFullFilePath) ) { _hostFileSystemStorage.FolderDelete( exifToolUnixFolderFullFilePath); } - _hostFileSystemStorage.FolderMove(imageExifToolVersionFolder, exifToolUnixFolderFullFilePath); + + _hostFileSystemStorage.FolderMove(imageExifToolVersionFolder, + exifToolUnixFolderFullFilePath); } else { @@ -196,8 +227,10 @@ internal async Task DownloadForUnix(string matchExifToolForUnixName, // remove tar.gz file afterwards _hostFileSystemStorage.FileDelete(tarGzArchiveFullFilePath); - var exifToolExePath = Path.Combine(_appSettings.DependenciesFolder, "exiftool-unix", "exiftool"); - _logger.LogInformation($"[DownloadForUnix] ExifTool is just downloaded: {exifToolExePath} for {_appSettings.ApplicationType}"); + var exifToolExePath = + Path.Combine(_appSettings.DependenciesFolder, "exiftool-unix", "exiftool"); + _logger.LogInformation( + $"[DownloadForUnix] ExifTool is just downloaded: {exifToolExePath} for {_appSettings.ApplicationType}"); return await RunChmodOnExifToolUnixExe(); } @@ -215,10 +248,12 @@ internal async Task RunChmodOnExifToolUnixExe() } // command.run does not care about the $PATH - var result = await Command.Run("/bin/chmod", "0755", ExeExifToolUnixFullFilePath()).Task; + var result = await Command.Run("/bin/chmod", "0755", ExeExifToolUnixFullFilePath()) + .Task; if ( result.Success ) return true; - _logger.LogError($"command failed with exit code {result.ExitCode}: {result.StandardError}"); + _logger.LogError( + $"command failed with exit code {result.ExitCode}: {result.StandardError}"); return false; } @@ -250,13 +285,13 @@ internal string[] GetChecksumsFromTextFile(string checksumsValue, int max = 8) { var regexExifToolForWindowsName = new Regex("[a-z0-9]{40}", RegexOptions.None, TimeSpan.FromMilliseconds(100)); - var results = regexExifToolForWindowsName.Matches(checksumsValue). - Select(m => m.Value). - ToArray(); + var results = regexExifToolForWindowsName.Matches(checksumsValue).Select(m => m.Value) + .ToArray(); if ( results.Length < max ) return results; - _logger.LogError($"More than {max} checksums found, this is not expected, code stops now"); - return Array.Empty(); + _logger.LogError( + $"More than {max} checksums found, this is not expected, code stops now"); + return []; } /// @@ -272,36 +307,44 @@ internal bool CheckSha1(string fullFilePath, IEnumerable checkSumOptions using var hashAlgorithm = SHA1.Create(); var byteHash = hashAlgorithm.ComputeHash(buffer); - var hash = BitConverter.ToString(byteHash).Replace("-", string.Empty).ToLowerInvariant(); - return checkSumOptions.AsEnumerable().Any(p => p.Equals(hash, StringComparison.InvariantCultureIgnoreCase)); + var hash = BitConverter.ToString(byteHash).Replace("-", string.Empty) + .ToLowerInvariant(); + return checkSumOptions.AsEnumerable().Any(p => + p.Equals(hash, StringComparison.InvariantCultureIgnoreCase)); } private string ExeExifToolWindowsFullFilePath() { - return Path.Combine(Path.Combine(_appSettings.DependenciesFolder, "exiftool-windows"), "exiftool.exe"); + return Path.Combine(Path.Combine(_appSettings.DependenciesFolder, "exiftool-windows"), + "exiftool.exe"); } internal async Task DownloadForWindows(string matchExifToolForWindowsName, string[] getChecksumsFromTextFile, bool downloadFromMirror = false) { if ( _hostFileSystemStorage.ExistFile( - ExeExifToolWindowsFullFilePath()) ) return true; + ExeExifToolWindowsFullFilePath()) ) return true; - var zipArchiveFullFilePath = Path.Combine(_appSettings.DependenciesFolder, "exiftool.zip"); - var windowsExifToolFolder = Path.Combine(_appSettings.DependenciesFolder, "exiftool-windows"); + var zipArchiveFullFilePath = + Path.Combine(_appSettings.DependenciesFolder, "exiftool.zip"); + var windowsExifToolFolder = + Path.Combine(_appSettings.DependenciesFolder, "exiftool-windows"); var url = $"{ExiftoolDownloadBasePath}{matchExifToolForWindowsName}"; - if ( downloadFromMirror ) url = $"{ExiftoolDownloadBasePathMirror}{matchExifToolForWindowsName}"; + if ( downloadFromMirror ) + url = $"{ExiftoolDownloadBasePathMirror}{matchExifToolForWindowsName}"; var windowsDownloaded = await _httpClientHelper.Download(url, zipArchiveFullFilePath); if ( !windowsDownloaded ) { - throw new HttpRequestException($"file is not downloaded {matchExifToolForWindowsName}"); + throw new HttpRequestException( + $"file is not downloaded {matchExifToolForWindowsName}"); } if ( !CheckSha1(zipArchiveFullFilePath, getChecksumsFromTextFile) ) { - throw new HttpRequestException($"checksum for {zipArchiveFullFilePath} is not valid"); + throw new HttpRequestException( + $"checksum for {zipArchiveFullFilePath} is not valid"); } _hostFileSystemStorage.CreateDirectory(windowsExifToolFolder); @@ -310,7 +353,8 @@ internal async Task DownloadForWindows(string matchExifToolForWindowsName, MoveFileIfExist(Path.Combine(windowsExifToolFolder, "exiftool(-k).exe"), Path.Combine(windowsExifToolFolder, "exiftool.exe")); - _logger.LogInformation($"[DownloadForWindows] ExifTool downloaded: {ExeExifToolWindowsFullFilePath()}"); + _logger.LogInformation( + $"[DownloadForWindows] ExifTool downloaded: {ExeExifToolWindowsFullFilePath()}"); return _hostFileSystemStorage.ExistFile(Path.Combine(windowsExifToolFolder, "exiftool.exe")); } diff --git a/starsky/starskydependenciescli/Program.cs b/starsky/starskydependenciescli/Program.cs index fa012c4703..c8ef0dc340 100644 --- a/starsky/starskydependenciescli/Program.cs +++ b/starsky/starskydependenciescli/Program.cs @@ -3,6 +3,8 @@ using starsky.feature.externaldependencies; using starsky.feature.externaldependencies.Interfaces; using starsky.foundation.database.Helpers; +using starsky.foundation.http.Interfaces; +using starsky.foundation.http.Services; using starsky.foundation.injection; using starsky.foundation.platform.Helpers; using starsky.foundation.platform.Models; @@ -30,9 +32,12 @@ public static async Task Main(string[] args) services.AddOpenTelemetryMonitoring(appSettings); services.AddTelemetryLogging(appSettings); - services.AddScoped(); new SetupDatabaseTypes(appSettings, services).BuilderDb(); + + services.AddScoped(); + services.AddScoped(); + serviceProvider = services.BuildServiceProvider(); var dependenciesService = diff --git a/starsky/starskydependenciescli/default-init-launchSettings.json b/starsky/starskydependenciescli/Properties/default-init-launchSettings.json similarity index 100% rename from starsky/starskydependenciescli/default-init-launchSettings.json rename to starsky/starskydependenciescli/Properties/default-init-launchSettings.json diff --git a/starsky/starskydependenciescli/starskydependenciescli.csproj b/starsky/starskydependenciescli/starskydependenciescli.csproj index 4c05ca666b..d8d90559e0 100644 --- a/starsky/starskydependenciescli/starskydependenciescli.csproj +++ b/starsky/starskydependenciescli/starskydependenciescli.csproj @@ -22,6 +22,7 @@ + diff --git a/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs b/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs index 42656a606d..ffee2b88a5 100644 --- a/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs +++ b/starsky/starskytest/starsky.foundation.platform/Helpers/ArgsHelperTest.cs @@ -637,14 +637,21 @@ public void Name() [DataRow("linux-arm", "LINUX")] [DataRow("linux-arm64", "LINUX")] [DataRow("win-x64", "WINDOWS")] - [DataRow("win-x86", "WINDOWS")] [DataRow("win-arm64", "WINDOWS")] - [DataRow("test", "")] - [DataRow(null, "")] public void GetRuntimeTest(string input, string expected) { var args = new List { "--runtime", input }.ToArray(); Assert.AreEqual(expected, ArgsHelper.GetRuntime(args)[0].Item1.ToString()); } + + [DataTestMethod] + [DataRow("test")] + [DataRow(null)] + [DataRow("win-x86")] + public void GetRuntimeTestNoContent(string input) + { + var args = new List { "--runtime", input }.ToArray(); + Assert.AreEqual(0, ArgsHelper.GetRuntime(args).Count); + } } } diff --git a/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs b/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs index 69a68b57a2..b0696f06f2 100644 --- a/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs +++ b/starsky/starskytest/starsky.foundation.platform/Helpers/PlatformParserTest.cs @@ -28,14 +28,21 @@ public void GetCurrentOsPlatformTest() [DataRow("linux-arm", "LINUX", "Arm")] [DataRow("linux-arm64", "LINUX", "Arm64")] [DataRow("win-x64", "WINDOWS", "X64")] - [DataRow("win-x86", "WINDOWS", "X86")] [DataRow("win-arm64", "WINDOWS", "Arm64")] - [DataRow("test", "", "")] - [DataRow(null, "", "")] public void RuntimeIdentifierTest(string input, string expectedOs, string expectedArch) { var result = PlatformParser.RuntimeIdentifier(input); Assert.AreEqual(expectedOs, result[0].Item1.ToString()); Assert.AreEqual(expectedArch, result[0].Item2.ToString()); } + + [DataTestMethod] + [DataRow("test")] + [DataRow(null)] + [DataRow("win-x86")] + public void RuntimeIdentifierTestNoContent(string input) + { + var result = PlatformParser.RuntimeIdentifier(input); + Assert.AreEqual(0, result.Count); + } } From fe6b200dd44190ae6a36cc15e76ed6096e5fac1c Mon Sep 17 00:00:00 2001 From: Dion Date: Thu, 4 Apr 2024 14:49:56 +0200 Subject: [PATCH 07/11] add folder copy --- .../Interfaces/IStorage.cs | 1 + .../Storage/StorageHostFullPathFilesystem.cs | 33 ++++++++++ .../Storage/StorageSubPathFilesystem.cs | 8 +++ .../Storage/StorageThumbnailFilesystem.cs | 5 ++ .../Helpers/CreateFolderIfNotExists.cs | 51 +++++++++++++++ .../Services/ExifToolDownload.cs | 40 ++++++------ starsky/starskytest/FakeMocks/FakeIStorage.cs | 5 ++ .../StorageHostFullPathFilesystemTest.cs | 62 +++++++++++++++++++ 8 files changed, 187 insertions(+), 18 deletions(-) create mode 100644 starsky/starsky.foundation.writemeta/Helpers/CreateFolderIfNotExists.cs diff --git a/starsky/starsky.foundation.storage/Interfaces/IStorage.cs b/starsky/starsky.foundation.storage/Interfaces/IStorage.cs index cfca155725..f4c4c85066 100644 --- a/starsky/starsky.foundation.storage/Interfaces/IStorage.cs +++ b/starsky/starsky.foundation.storage/Interfaces/IStorage.cs @@ -12,6 +12,7 @@ public interface IStorage bool ExistFolder(string path); FolderOrFileModel.FolderOrFileTypeList IsFolderOrFile(string path); void FolderMove(string fromPath, string toPath); + void FolderCopy(string fromPath, string toPath); bool FileMove(string fromPath, string toPath); void FileCopy(string fromPath, string toPath); bool FileDelete(string path); diff --git a/starsky/starsky.foundation.storage/Storage/StorageHostFullPathFilesystem.cs b/starsky/starsky.foundation.storage/Storage/StorageHostFullPathFilesystem.cs index ad92bf4f00..ad5d79fa4a 100644 --- a/starsky/starsky.foundation.storage/Storage/StorageHostFullPathFilesystem.cs +++ b/starsky/starsky.foundation.storage/Storage/StorageHostFullPathFilesystem.cs @@ -320,6 +320,39 @@ public void FolderMove(string fromPath, string toPath) Directory.Move(fromPath, toPath); } + public void FolderCopy(string fromPath, string toPath) + { + var dirs = new Stack(); + dirs.Push(fromPath); + + while (dirs.Count > 0) + { + var currentDir = dirs.Pop(); + var destinationSubDirectory = currentDir.Replace(fromPath, toPath); + + if (!Directory.Exists(destinationSubDirectory)) + { + Directory.CreateDirectory(destinationSubDirectory); + } + + var files = Directory.GetFiles(currentDir); + + foreach (var file in files) + { + var fileName = Path.GetFileName(file); + var destinationFile = Path.Combine(destinationSubDirectory, fileName); + File.Copy(file, destinationFile); + } + + var subdirectories = Directory.GetDirectories(currentDir); + + foreach (var subDirectory in subdirectories) + { + dirs.Push(subDirectory); + } + } + } + /// /// Move file on real filesystem /// diff --git a/starsky/starsky.foundation.storage/Storage/StorageSubPathFilesystem.cs b/starsky/starsky.foundation.storage/Storage/StorageSubPathFilesystem.cs index 695bc37f91..c282542464 100644 --- a/starsky/starsky.foundation.storage/Storage/StorageSubPathFilesystem.cs +++ b/starsky/starsky.foundation.storage/Storage/StorageSubPathFilesystem.cs @@ -101,6 +101,14 @@ public void FolderMove(string fromPath, string toPath) toFileFullPath); } + public void FolderCopy(string fromPath, string toPath) + { + var inputFileFullPath = _appSettings.DatabasePathToFilePath(fromPath); + var toFileFullPath = _appSettings.DatabasePathToFilePath(toPath); + new StorageHostFullPathFilesystem(_logger).FolderCopy(inputFileFullPath, + toFileFullPath); + } + /// /// Move a file /// diff --git a/starsky/starsky.foundation.storage/Storage/StorageThumbnailFilesystem.cs b/starsky/starsky.foundation.storage/Storage/StorageThumbnailFilesystem.cs index 27fac215a1..1e3bff458d 100644 --- a/starsky/starsky.foundation.storage/Storage/StorageThumbnailFilesystem.cs +++ b/starsky/starsky.foundation.storage/Storage/StorageThumbnailFilesystem.cs @@ -69,6 +69,11 @@ public void FolderMove(string fromPath, string toPath) throw new System.NotImplementedException(); } + public void FolderCopy(string fromPath, string toPath) + { + throw new NotImplementedException(); + } + public bool FileMove(string fromPath, string toPath) { var oldThumbPath = CombinePath(fromPath); diff --git a/starsky/starsky.foundation.writemeta/Helpers/CreateFolderIfNotExists.cs b/starsky/starsky.foundation.writemeta/Helpers/CreateFolderIfNotExists.cs new file mode 100644 index 0000000000..b8f767e316 --- /dev/null +++ b/starsky/starsky.foundation.writemeta/Helpers/CreateFolderIfNotExists.cs @@ -0,0 +1,51 @@ +using starsky.foundation.platform.Interfaces; +using starsky.foundation.platform.Models; +using starsky.foundation.storage.Storage; + +namespace starsky.foundation.writemeta.Helpers; + +public class CreateFolderIfNotExists +{ + private readonly StorageHostFullPathFilesystem _hostFileSystemStorage; + private readonly IWebLogger _logger; + private readonly AppSettings _appSettings; + + public CreateFolderIfNotExists(IWebLogger logger, AppSettings appSettings) + { + _hostFileSystemStorage = new StorageHostFullPathFilesystem(logger); + _logger = logger; + _appSettings = appSettings; + } + + internal void CreateDirectoryDependenciesTempFolderIfNotExists() + { + CreateDirectoryDependenciesFolderIfNotExists(); + CreateDirectoryTempFolderIfNotExists(); + } + + private void CreateDirectoryDependenciesFolderIfNotExists() + { + if ( _hostFileSystemStorage.ExistFolder(_appSettings + .DependenciesFolder) ) + { + return; + } + + _logger.LogInformation("[DownloadExifTool] Create Directory: " + + _appSettings.DependenciesFolder); + _hostFileSystemStorage.CreateDirectory(_appSettings.DependenciesFolder); + } + + private void CreateDirectoryTempFolderIfNotExists() + { + if ( _hostFileSystemStorage.ExistFolder(_appSettings + .TempFolder) ) + { + return; + } + + _logger.LogInformation("[DownloadExifTool] Create Directory: " + + _appSettings.TempFolder); + _hostFileSystemStorage.CreateDirectory(_appSettings.TempFolder); + } +} diff --git a/starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs b/starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs index 45f40cc2d3..8f19c71420 100644 --- a/starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs +++ b/starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs @@ -17,6 +17,7 @@ using starsky.foundation.storage.ArchiveFormats; using starsky.foundation.storage.Interfaces; using starsky.foundation.storage.Storage; +using starsky.foundation.writemeta.Helpers; using starsky.foundation.writemeta.Interfaces; [assembly: InternalsVisibleTo("starskytest")] @@ -84,7 +85,8 @@ public async Task DownloadExifTool(bool isWindows, int minimumSize = 30) return false; } - CreateDirectoryDependenciesFolderIfNotExists(); + new CreateFolderIfNotExists(_logger, _appSettings) + .CreateDirectoryDependenciesTempFolderIfNotExists(); if ( isWindows && ( !_hostFileSystemStorage.ExistFile(ExeExifToolWindowsFullFilePath()) || @@ -116,14 +118,6 @@ public async Task DownloadExifTool(bool isWindows, int minimumSize = 30) return await RunChmodOnExifToolUnixExe(); } - private void CreateDirectoryDependenciesFolderIfNotExists() - { - if ( _hostFileSystemStorage.ExistFolder(_appSettings - .DependenciesFolder) ) return; - _logger.LogInformation("[DownloadExifTool] Create Directory: " + - _appSettings.DependenciesFolder); - _hostFileSystemStorage.CreateDirectory(_appSettings.DependenciesFolder); - } internal async Task?> DownloadCheckSums() { @@ -175,7 +169,10 @@ private string ExeExifToolUnixFullFilePath() internal async Task DownloadForUnix(string matchExifToolForUnixName, IEnumerable getChecksumsFromTextFile, bool downloadFromMirror = false) { - if ( _hostFileSystemStorage.ExistFile(ExeExifToolUnixFullFilePath()) ) return true; + if ( _hostFileSystemStorage.ExistFile(ExeExifToolUnixFullFilePath()) ) + { + return true; + } var tarGzArchiveFullFilePath = Path.Combine(_appSettings.TempFolder, "exiftool.tar.gz"); @@ -199,24 +196,31 @@ internal async Task DownloadForUnix(string matchExifToolForUnixName, await new TarBal(_hostFileSystemStorage).ExtractTarGz( _hostFileSystemStorage.ReadStream(tarGzArchiveFullFilePath), - _appSettings.DependenciesFolder, CancellationToken.None); + _appSettings.TempFolder, CancellationToken.None); var imageExifToolVersionFolder = _hostFileSystemStorage - .GetDirectories(_appSettings.DependenciesFolder) + .GetDirectories(_appSettings.TempFolder) .FirstOrDefault(p => - p.StartsWith(Path.Combine(_appSettings.DependenciesFolder, "Image-ExifTool-"))); + p.StartsWith(Path.Combine(_appSettings.TempFolder, "Image-ExifTool-"))); if ( imageExifToolVersionFolder != null ) { - var exifToolUnixFolderFullFilePath = - Path.Combine(_appSettings.DependenciesFolder, "exiftool-unix"); - if ( _hostFileSystemStorage.ExistFolder(exifToolUnixFolderFullFilePath) ) + var exifToolUnixFolderFullFilePathTempFolder = + Path.Combine(_appSettings.TempFolder, "exiftool-unix"); + + if ( _hostFileSystemStorage.ExistFolder(exifToolUnixFolderFullFilePathTempFolder) ) { _hostFileSystemStorage.FolderDelete( - exifToolUnixFolderFullFilePath); + exifToolUnixFolderFullFilePathTempFolder); } _hostFileSystemStorage.FolderMove(imageExifToolVersionFolder, - exifToolUnixFolderFullFilePath); + exifToolUnixFolderFullFilePathTempFolder); + + var exifToolUnixFolderFullFilePath = + Path.Combine(_appSettings.DependenciesFolder, "exiftool-unix"); + + _hostFileSystemStorage.FileCopy(imageExifToolVersionFolder, + exifToolUnixFolderFullFilePathTempFolder); } else { diff --git a/starsky/starskytest/FakeMocks/FakeIStorage.cs b/starsky/starskytest/FakeMocks/FakeIStorage.cs index 398d5f77ad..55bec3f6f5 100644 --- a/starsky/starskytest/FakeMocks/FakeIStorage.cs +++ b/starsky/starskytest/FakeMocks/FakeIStorage.cs @@ -130,6 +130,11 @@ public void FolderMove(string fromPath, string toPath) _outputSubPathFolders[indexOfFolders] = toPath; } + public void FolderCopy(string fromPath, string toPath) + { + throw new NotImplementedException(); + } + public bool FileMove(string fromPath, string toPath) { var existOldFile = ExistFile(fromPath); diff --git a/starsky/starskytest/starsky.foundation.storage/Storage/StorageHostFullPathFilesystemTest.cs b/starsky/starskytest/starsky.foundation.storage/Storage/StorageHostFullPathFilesystemTest.cs index d5f7c31867..c4164519ac 100644 --- a/starsky/starskytest/starsky.foundation.storage/Storage/StorageHostFullPathFilesystemTest.cs +++ b/starsky/starskytest/starsky.foundation.storage/Storage/StorageHostFullPathFilesystemTest.cs @@ -273,6 +273,68 @@ public void IsFileReady_HostService() Assert.IsFalse(hostStorage.ExistFile(filePathNewFile)); } + + [TestMethod] + public void MoveFolder() + { + var hostStorage = new StorageHostFullPathFilesystem(new FakeIWebLogger()); + var beforeDirectoryFullPath = Path.Combine(new CreateAnImage().BasePath, "28934283492349_before"); + var afterDirectoryFullPath = Path.Combine(new CreateAnImage().BasePath, "28934283492349_after"); + hostStorage.FolderDelete(afterDirectoryFullPath); + + hostStorage.CreateDirectory(beforeDirectoryFullPath); + hostStorage.FolderMove(beforeDirectoryFullPath, afterDirectoryFullPath); + + Assert.IsFalse(hostStorage.ExistFolder(beforeDirectoryFullPath)); + Assert.IsTrue(hostStorage.ExistFolder(afterDirectoryFullPath)); + + hostStorage.FolderDelete(beforeDirectoryFullPath); + hostStorage.FolderDelete(afterDirectoryFullPath); + } + + [TestMethod] + public void CopyFolder_FlatFolder() + { + var hostStorage = new StorageHostFullPathFilesystem(new FakeIWebLogger()); + var beforeDirectoryFullPath = Path.Combine(new CreateAnImage().BasePath, "453984583495_before"); + var afterDirectoryFullPath = Path.Combine(new CreateAnImage().BasePath, "453984583495_after"); + hostStorage.FolderDelete(afterDirectoryFullPath); + + hostStorage.CreateDirectory(beforeDirectoryFullPath); + hostStorage.FolderCopy(beforeDirectoryFullPath, afterDirectoryFullPath); + + Assert.IsTrue(hostStorage.ExistFolder(beforeDirectoryFullPath)); + Assert.IsTrue(hostStorage.ExistFolder(afterDirectoryFullPath)); + + hostStorage.FolderDelete(beforeDirectoryFullPath); + hostStorage.FolderDelete(afterDirectoryFullPath); + } + + [TestMethod] + public async Task CopyFolder_ChildItems() + { + var hostStorage = new StorageHostFullPathFilesystem(new FakeIWebLogger()); + var beforeDirectoryFullPath = Path.Combine(new CreateAnImage().BasePath, "9057678234_before"); + var childFullPath = Path.Combine(beforeDirectoryFullPath, "child"); + var childFullPathFile = Path.Combine(childFullPath, "child.test"); + + var afterDirectoryFullPath = Path.Combine(new CreateAnImage().BasePath, "9057678234_after"); + hostStorage.FolderDelete(afterDirectoryFullPath); + + hostStorage.CreateDirectory(beforeDirectoryFullPath); + hostStorage.CreateDirectory(childFullPath); + await hostStorage.WriteStreamAsync(new MemoryStream(new byte[1]), childFullPathFile); + + hostStorage.FolderCopy(beforeDirectoryFullPath, afterDirectoryFullPath); + + Assert.IsTrue(hostStorage.ExistFolder(beforeDirectoryFullPath)); + Assert.IsTrue(hostStorage.ExistFolder(afterDirectoryFullPath)); + Assert.IsTrue(hostStorage.ExistFile(childFullPathFile.Replace("_before", "_after"))); + Assert.IsTrue(hostStorage.ExistFile(childFullPathFile)); + + hostStorage.FolderDelete(beforeDirectoryFullPath); + hostStorage.FolderDelete(afterDirectoryFullPath); + } [TestMethod] public void GetDirectoryRecursive_NotFound() From 15a7c7a99a11f2edd63ee9f4a2c8f48af2d5e42f Mon Sep 17 00:00:00 2001 From: Dion Date: Thu, 4 Apr 2024 14:58:20 +0200 Subject: [PATCH 08/11] tmp --- .../Helpers/ExifToolDownloadTest.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs b/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs index f741ccfd64..44eef08bf7 100644 --- a/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs +++ b/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs @@ -138,7 +138,7 @@ public async Task DownloadCheckSums_BothServicesAreDown() Assert.AreEqual(null, result); } - [TestMethod] + // [TestMethod] public async Task GetExifToolByOs() { var fakeIHttpProvider = new FakeIHttpProvider(new Dictionary @@ -474,7 +474,7 @@ public async Task StartDownloadForUnix_Fail() .StartDownloadForUnix(); } - [TestMethod] + // [TestMethod] public async Task StartDownloadForUnix_2Times() { var fakeIHttpProvider = new FakeIHttpProvider(new Dictionary From 1559e90e241c88252a5d9b61426f1a9b64b05985 Mon Sep 17 00:00:00 2001 From: Dion Date: Thu, 4 Apr 2024 15:02:50 +0200 Subject: [PATCH 09/11] add guid --- .../starsky.feature.externaldependencies.csproj | 1 + 1 file changed, 1 insertion(+) diff --git a/starsky/starsky.feature.externaldependencies/starsky.feature.externaldependencies.csproj b/starsky/starsky.feature.externaldependencies/starsky.feature.externaldependencies.csproj index 0b33052ab8..3ede59eb51 100644 --- a/starsky/starsky.feature.externaldependencies/starsky.feature.externaldependencies.csproj +++ b/starsky/starsky.feature.externaldependencies/starsky.feature.externaldependencies.csproj @@ -2,6 +2,7 @@ net8.0 + {5783f81c-618c-4971-984e-0e405ab1b609} enable enable starsky.feature.externaldependencies From 9f8ad0ff61281904395d80314c0b84e3c61a650f Mon Sep 17 00:00:00 2001 From: Dion Date: Thu, 4 Apr 2024 16:29:05 +0200 Subject: [PATCH 10/11] undo --- .../Helpers/ExifToolDownloadTest.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs b/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs index 44eef08bf7..f741ccfd64 100644 --- a/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs +++ b/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs @@ -138,7 +138,7 @@ public async Task DownloadCheckSums_BothServicesAreDown() Assert.AreEqual(null, result); } - // [TestMethod] + [TestMethod] public async Task GetExifToolByOs() { var fakeIHttpProvider = new FakeIHttpProvider(new Dictionary @@ -474,7 +474,7 @@ public async Task StartDownloadForUnix_Fail() .StartDownloadForUnix(); } - // [TestMethod] + [TestMethod] public async Task StartDownloadForUnix_2Times() { var fakeIHttpProvider = new FakeIHttpProvider(new Dictionary From 0a477b2df61d6ddf7fe255704f67c515e7ee8113 Mon Sep 17 00:00:00 2001 From: Dion Date: Mon, 8 Apr 2024 10:46:56 +0200 Subject: [PATCH 11/11] WIP --- .../Services/ExifToolDownload.cs | 372 ------------------ .../ExifToolDownloadBackgroundService.cs | 1 + .../Services/ExifToolDownloader/CheckSha1.cs | 86 ++++ .../ExifToolDownloader/ExifToolDownload.cs | 115 ++++++ .../ExifToolDownloadUnix.cs | 141 +++++++ .../ExifToolDownloadWindows.cs | 72 ++++ .../ExifToolDownloader/ExifToolLocations.cs | 39 ++ .../Helpers/ExifToolDownloadTest.cs | 1 + 8 files changed, 455 insertions(+), 372 deletions(-) delete mode 100644 starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs create mode 100644 starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/CheckSha1.cs create mode 100644 starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownload.cs create mode 100644 starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownloadUnix.cs create mode 100644 starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownloadWindows.cs create mode 100644 starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolLocations.cs diff --git a/starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs b/starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs deleted file mode 100644 index 8f19c71420..0000000000 --- a/starsky/starsky.foundation.writemeta/Services/ExifToolDownload.cs +++ /dev/null @@ -1,372 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Diagnostics.CodeAnalysis; -using System.IO; -using System.Linq; -using System.Net.Http; -using System.Runtime.CompilerServices; -using System.Security.Cryptography; -using System.Text.RegularExpressions; -using System.Threading; -using System.Threading.Tasks; -using Medallion.Shell; -using starsky.foundation.http.Interfaces; -using starsky.foundation.injection; -using starsky.foundation.platform.Interfaces; -using starsky.foundation.platform.Models; -using starsky.foundation.storage.ArchiveFormats; -using starsky.foundation.storage.Interfaces; -using starsky.foundation.storage.Storage; -using starsky.foundation.writemeta.Helpers; -using starsky.foundation.writemeta.Interfaces; - -[assembly: InternalsVisibleTo("starskytest")] - -namespace starsky.foundation.writemeta.Services -{ - [Service(typeof(IExifToolDownload), InjectionLifetime = InjectionLifetime.Singleton)] - [SuppressMessage("Usage", - "S1075:Refactor your code not to use hardcoded absolute paths or URIs", - Justification = "Source of files")] - [SuppressMessage("Usage", - "S4790:Make sure this weak hash algorithm is not used in a sensitive context here.", - Justification = "Safe")] - public sealed class ExifToolDownload : IExifToolDownload - { - private readonly IHttpClientHelper _httpClientHelper; - private readonly AppSettings _appSettings; - private readonly IStorage _hostFileSystemStorage; - private readonly IWebLogger _logger; - - private const string CheckSumLocation = "https://exiftool.org/checksums.txt"; - - private const string CheckSumLocationMirror = - "https://qdraw.nl/special/mirror/exiftool/checksums.txt"; - - private const string - ExiftoolDownloadBasePath = "https://exiftool.org/"; // with slash at the end - - private const string ExiftoolDownloadBasePathMirror = - "https://qdraw.nl/special/mirror/exiftool/"; // with slash at the end - - public ExifToolDownload(IHttpClientHelper httpClientHelper, AppSettings appSettings, - IWebLogger logger) - { - _httpClientHelper = httpClientHelper; - _appSettings = appSettings; - _hostFileSystemStorage = new StorageHostFullPathFilesystem(logger); - _logger = logger; - } - - internal ExifToolDownload(IHttpClientHelper httpClientHelper, AppSettings appSettings, - IWebLogger logger, IStorage storage) - { - _httpClientHelper = httpClientHelper; - _appSettings = appSettings; - _hostFileSystemStorage = storage; - _logger = logger; - } - - /// - /// Auto Download Exiftool - /// - /// download windows version if true - /// check for min file size in bytes (Default = 30 bytes) - /// - public async Task DownloadExifTool(bool isWindows, int minimumSize = 30) - { - if ( _appSettings.ExiftoolSkipDownloadOnStartup == true || _appSettings is - { AddSwaggerExport: true, AddSwaggerExportExitAfter: true } ) - { - var name = _appSettings.ExiftoolSkipDownloadOnStartup == true - ? "ExiftoolSkipDownloadOnStartup" - : "AddSwaggerExport and AddSwaggerExportExitAfter"; - _logger.LogInformation($"[DownloadExifTool] Skipped due true of {name} setting"); - return false; - } - - new CreateFolderIfNotExists(_logger, _appSettings) - .CreateDirectoryDependenciesTempFolderIfNotExists(); - - if ( isWindows && - ( !_hostFileSystemStorage.ExistFile(ExeExifToolWindowsFullFilePath()) || - _hostFileSystemStorage.Info(ExeExifToolWindowsFullFilePath()).Size <= - minimumSize ) ) - { - return await StartDownloadForWindows(); - } - - if ( !isWindows && - ( !_hostFileSystemStorage.ExistFile(ExeExifToolUnixFullFilePath()) || - _hostFileSystemStorage.Info(ExeExifToolUnixFullFilePath()).Size <= - minimumSize ) ) - { - return await StartDownloadForUnix(); - } - - if ( _appSettings.IsVerbose() ) - { - var debugPath = isWindows - ? ExeExifToolWindowsFullFilePath() - : ExeExifToolUnixFullFilePath(); - _logger.LogInformation($"[DownloadExifTool] {debugPath}"); - } - - // When running deploy scripts rights might reset (only for unix) - if ( isWindows ) return true; - - return await RunChmodOnExifToolUnixExe(); - } - - - internal async Task?> DownloadCheckSums() - { - var checksums = await _httpClientHelper.ReadString(CheckSumLocation); - if ( checksums.Key ) - { - return checksums; - } - - _logger.LogError( - $"Checksum loading failed {CheckSumLocation}, next retry from mirror ~ error > " + - checksums.Value); - - checksums = await _httpClientHelper.ReadString(CheckSumLocationMirror); - if ( checksums.Key ) return new KeyValuePair(false, checksums.Value); - - _logger.LogError($"Checksum loading failed {CheckSumLocationMirror}" + - $", next stop; please connect to internet and restart app ~ error > " + - checksums.Value); - return null; - } - - internal async Task StartDownloadForUnix() - { - var checksums = await DownloadCheckSums(); - if ( checksums == null ) return false; - var matchExifToolForUnixName = GetUnixTarGzFromChecksum(checksums.Value.Value); - return await DownloadForUnix(matchExifToolForUnixName, - GetChecksumsFromTextFile(checksums.Value.Value), !checksums.Value.Key); - } - - internal static string GetUnixTarGzFromChecksum(string checksumsValue) - { - // (?<=SHA1\()Image-ExifTool-[\d\.]+\.zip - var regexExifToolForWindowsName = new Regex( - @"(?<=SHA1\()Image-ExifTool-[0-9\.]+\.tar.gz", - RegexOptions.None, TimeSpan.FromMilliseconds(100)); - return regexExifToolForWindowsName.Match(checksumsValue).Value; - } - - private string ExeExifToolUnixFullFilePath() - { - var path = Path.Combine(_appSettings.DependenciesFolder, - "exiftool-unix", - "exiftool"); - return path; - } - - internal async Task DownloadForUnix(string matchExifToolForUnixName, - IEnumerable getChecksumsFromTextFile, bool downloadFromMirror = false) - { - if ( _hostFileSystemStorage.ExistFile(ExeExifToolUnixFullFilePath()) ) - { - return true; - } - - var tarGzArchiveFullFilePath = - Path.Combine(_appSettings.TempFolder, "exiftool.tar.gz"); - - var url = $"{ExiftoolDownloadBasePath}{matchExifToolForUnixName}"; - if ( downloadFromMirror ) - url = $"{ExiftoolDownloadBasePathMirror}{matchExifToolForUnixName}"; - - var unixDownloaded = await _httpClientHelper.Download(url, tarGzArchiveFullFilePath); - if ( !unixDownloaded ) - { - throw new HttpRequestException( - $"file is not downloaded {matchExifToolForUnixName}"); - } - - if ( !CheckSha1(tarGzArchiveFullFilePath, getChecksumsFromTextFile) ) - { - throw new HttpRequestException( - $"checksum for {tarGzArchiveFullFilePath} is not valid"); - } - - await new TarBal(_hostFileSystemStorage).ExtractTarGz( - _hostFileSystemStorage.ReadStream(tarGzArchiveFullFilePath), - _appSettings.TempFolder, CancellationToken.None); - - var imageExifToolVersionFolder = _hostFileSystemStorage - .GetDirectories(_appSettings.TempFolder) - .FirstOrDefault(p => - p.StartsWith(Path.Combine(_appSettings.TempFolder, "Image-ExifTool-"))); - if ( imageExifToolVersionFolder != null ) - { - var exifToolUnixFolderFullFilePathTempFolder = - Path.Combine(_appSettings.TempFolder, "exiftool-unix"); - - if ( _hostFileSystemStorage.ExistFolder(exifToolUnixFolderFullFilePathTempFolder) ) - { - _hostFileSystemStorage.FolderDelete( - exifToolUnixFolderFullFilePathTempFolder); - } - - _hostFileSystemStorage.FolderMove(imageExifToolVersionFolder, - exifToolUnixFolderFullFilePathTempFolder); - - var exifToolUnixFolderFullFilePath = - Path.Combine(_appSettings.DependenciesFolder, "exiftool-unix"); - - _hostFileSystemStorage.FileCopy(imageExifToolVersionFolder, - exifToolUnixFolderFullFilePathTempFolder); - } - else - { - _logger.LogError($"[DownloadForUnix] ExifTool folder does not exists"); - return false; - } - - // remove tar.gz file afterwards - _hostFileSystemStorage.FileDelete(tarGzArchiveFullFilePath); - - var exifToolExePath = - Path.Combine(_appSettings.DependenciesFolder, "exiftool-unix", "exiftool"); - _logger.LogInformation( - $"[DownloadForUnix] ExifTool is just downloaded: {exifToolExePath} for {_appSettings.ApplicationType}"); - return await RunChmodOnExifToolUnixExe(); - } - - internal async Task RunChmodOnExifToolUnixExe() - { - // need to check again - // when not exist - if ( !_hostFileSystemStorage.ExistFile(ExeExifToolUnixFullFilePath()) ) return false; - if ( _appSettings.IsWindows ) return true; - - if ( !_hostFileSystemStorage.ExistFile("/bin/chmod") ) - { - _logger.LogError("[RunChmodOnExifToolUnixExe] WARNING: /bin/chmod does not exist"); - return true; - } - - // command.run does not care about the $PATH - var result = await Command.Run("/bin/chmod", "0755", ExeExifToolUnixFullFilePath()) - .Task; - if ( result.Success ) return true; - - _logger.LogError( - $"command failed with exit code {result.ExitCode}: {result.StandardError}"); - return false; - } - - internal async Task StartDownloadForWindows() - { - var checksums = await DownloadCheckSums(); - if ( checksums == null ) return false; - - var matchExifToolForWindowsName = GetWindowsZipFromChecksum(checksums.Value.Value); - return await DownloadForWindows(matchExifToolForWindowsName, - GetChecksumsFromTextFile(checksums.Value.Value), !checksums.Value.Key); - } - - internal static string GetWindowsZipFromChecksum(string checksumsValue) - { - // (?<=SHA1\()exiftool-[\d\.]+\.zip - var regexExifToolForWindowsName = new Regex(@"(?<=SHA1\()exiftool-[0-9\.]+\.zip", - RegexOptions.None, TimeSpan.FromMilliseconds(100)); - return regexExifToolForWindowsName.Match(checksumsValue).Value; - } - - /// - /// Parse the content of checksum file - /// - /// input file: see test for example - /// max number of SHA1 results - /// - internal string[] GetChecksumsFromTextFile(string checksumsValue, int max = 8) - { - var regexExifToolForWindowsName = new Regex("[a-z0-9]{40}", - RegexOptions.None, TimeSpan.FromMilliseconds(100)); - var results = regexExifToolForWindowsName.Matches(checksumsValue).Select(m => m.Value) - .ToArray(); - if ( results.Length < max ) return results; - - _logger.LogError( - $"More than {max} checksums found, this is not expected, code stops now"); - return []; - } - - /// - /// Check if SHA1 hash is valid - /// Instead of SHA1CryptoServiceProvider, we use SHA1.Create - /// - /// path of exiftool.exe - /// list of sha1 hashes - /// - internal bool CheckSha1(string fullFilePath, IEnumerable checkSumOptions) - { - using var buffer = _hostFileSystemStorage.ReadStream(fullFilePath); - using var hashAlgorithm = SHA1.Create(); - - var byteHash = hashAlgorithm.ComputeHash(buffer); - var hash = BitConverter.ToString(byteHash).Replace("-", string.Empty) - .ToLowerInvariant(); - return checkSumOptions.AsEnumerable().Any(p => - p.Equals(hash, StringComparison.InvariantCultureIgnoreCase)); - } - - private string ExeExifToolWindowsFullFilePath() - { - return Path.Combine(Path.Combine(_appSettings.DependenciesFolder, "exiftool-windows"), - "exiftool.exe"); - } - - internal async Task DownloadForWindows(string matchExifToolForWindowsName, - string[] getChecksumsFromTextFile, bool downloadFromMirror = false) - { - if ( _hostFileSystemStorage.ExistFile( - ExeExifToolWindowsFullFilePath()) ) return true; - - var zipArchiveFullFilePath = - Path.Combine(_appSettings.DependenciesFolder, "exiftool.zip"); - var windowsExifToolFolder = - Path.Combine(_appSettings.DependenciesFolder, "exiftool-windows"); - - var url = $"{ExiftoolDownloadBasePath}{matchExifToolForWindowsName}"; - if ( downloadFromMirror ) - url = $"{ExiftoolDownloadBasePathMirror}{matchExifToolForWindowsName}"; - - var windowsDownloaded = await _httpClientHelper.Download(url, zipArchiveFullFilePath); - if ( !windowsDownloaded ) - { - throw new HttpRequestException( - $"file is not downloaded {matchExifToolForWindowsName}"); - } - - if ( !CheckSha1(zipArchiveFullFilePath, getChecksumsFromTextFile) ) - { - throw new HttpRequestException( - $"checksum for {zipArchiveFullFilePath} is not valid"); - } - - _hostFileSystemStorage.CreateDirectory(windowsExifToolFolder); - - new Zipper().ExtractZip(zipArchiveFullFilePath, windowsExifToolFolder); - MoveFileIfExist(Path.Combine(windowsExifToolFolder, "exiftool(-k).exe"), - Path.Combine(windowsExifToolFolder, "exiftool.exe")); - - _logger.LogInformation( - $"[DownloadForWindows] ExifTool downloaded: {ExeExifToolWindowsFullFilePath()}"); - return _hostFileSystemStorage.ExistFile(Path.Combine(windowsExifToolFolder, - "exiftool.exe")); - } - - private void MoveFileIfExist(string srcFullPath, string toFullPath) - { - if ( !_hostFileSystemStorage.ExistFile(srcFullPath) ) return; - _hostFileSystemStorage.FileMove(srcFullPath, toFullPath); - } - } -} diff --git a/starsky/starsky.foundation.writemeta/Services/ExifToolDownloadBackgroundService.cs b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloadBackgroundService.cs index a50862ed73..dea77dee14 100644 --- a/starsky/starsky.foundation.writemeta/Services/ExifToolDownloadBackgroundService.cs +++ b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloadBackgroundService.cs @@ -6,6 +6,7 @@ using starsky.foundation.injection; using starsky.foundation.platform.Interfaces; using starsky.foundation.platform.Models; +using starsky.foundation.writemeta.Services.ExifToolDownloader; namespace starsky.foundation.writemeta.Services { diff --git a/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/CheckSha1.cs b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/CheckSha1.cs new file mode 100644 index 0000000000..05590b35cf --- /dev/null +++ b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/CheckSha1.cs @@ -0,0 +1,86 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Security.Cryptography; +using System.Text.RegularExpressions; +using System.Threading.Tasks; +using starsky.foundation.http.Interfaces; +using starsky.foundation.platform.Interfaces; +using starsky.foundation.storage.Interfaces; + +namespace starsky.foundation.writemeta.Services.ExifToolDownloader; + +public class CheckSums(IStorage storage, IHttpClientHelper httpClientHelper, IWebLogger logger) +{ + internal async Task?> DownloadCheckSums(bool isMirror = false) + { + var url = !isMirror + ? ExifToolLocations.CheckSumLocation + : ExifToolLocations.CheckSumLocationMirror; + + var checksums = await httpClientHelper.ReadString(url); + if ( checksums.Key ) + { + return checksums; + } + + return null; + } + + /// + /// Check if SHA1 hash is valid + /// Instead of SHA1CryptoServiceProvider, we use SHA1.Create + /// + /// path of exiftool.exe + /// list of sha1 hashes + /// + internal bool CheckSha1(string fullFilePath, IEnumerable checkSumOptions) + { + using var buffer = storage.ReadStream(fullFilePath); + using var hashAlgorithm = SHA1.Create(); + + var byteHash = hashAlgorithm.ComputeHash(buffer); + var hash = BitConverter.ToString(byteHash).Replace("-", string.Empty) + .ToLowerInvariant(); + return checkSumOptions.AsEnumerable().Any(p => + p.Equals(hash, StringComparison.InvariantCultureIgnoreCase)); + } + + /// + /// Parse the content of checksum file + /// + /// input file: see test for example + /// max number of SHA1 results + /// + internal string[] GetChecksumsFromTextFile(string checksumsValue, int max = 8) + { + var regexExifToolForWindowsName = new Regex("[a-z0-9]{40}", + RegexOptions.None, TimeSpan.FromMilliseconds(100)); + var results = regexExifToolForWindowsName.Matches(checksumsValue).Select(m => m.Value) + .ToArray(); + if ( results.Length < max ) return results; + + logger.LogError( + $"More than {max} checksums found, this is not expected, code stops now"); + return []; + } + + + internal static string GetUnixTarGzFromChecksum(string checksumsValue) + { + // (?<=SHA1\()Image-ExifTool-[\d\.]+\.zip + var regexExifToolForWindowsName = new Regex( + @"(?<=SHA1\()Image-ExifTool-[0-9\.]+\.tar.gz", + RegexOptions.None, TimeSpan.FromMilliseconds(100)); + return regexExifToolForWindowsName.Match(checksumsValue).Value; + } + + internal static string GetWindowsZipFromChecksum(string checksumsValue) + { + // (?<=SHA1\()exiftool-[\d\.]+\.zip + var regexExifToolForWindowsName = new Regex(@"(?<=SHA1\()exiftool-[0-9\.]+\.zip", + RegexOptions.None, TimeSpan.FromMilliseconds(100)); + return regexExifToolForWindowsName.Match(checksumsValue).Value; + } + +} diff --git a/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownload.cs b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownload.cs new file mode 100644 index 0000000000..0e52158ba3 --- /dev/null +++ b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownload.cs @@ -0,0 +1,115 @@ +using System; +using System.Collections.Generic; +using System.Diagnostics.CodeAnalysis; +using System.Runtime.CompilerServices; +using System.Text.RegularExpressions; +using System.Threading.Tasks; +using starsky.foundation.http.Interfaces; +using starsky.foundation.injection; +using starsky.foundation.platform.Interfaces; +using starsky.foundation.platform.Models; +using starsky.foundation.storage.Interfaces; +using starsky.foundation.storage.Storage; +using starsky.foundation.writemeta.Helpers; +using starsky.foundation.writemeta.Interfaces; + +[assembly: InternalsVisibleTo("starskytest")] + +namespace starsky.foundation.writemeta.Services.ExifToolDownloader; + +[Service(typeof(IExifToolDownload), InjectionLifetime = InjectionLifetime.Singleton)] +[SuppressMessage("Usage", + "S1075:Refactor your code not to use hardcoded absolute paths or URIs", + Justification = "Source of files")] +[SuppressMessage("Usage", + "S4790:Make sure this weak hash algorithm is not used in a sensitive context here.", + Justification = "Safe")] +public sealed class ExifToolDownload : IExifToolDownload +{ + private readonly IHttpClientHelper _httpClientHelper; + private readonly AppSettings _appSettings; + private readonly IStorage _hostFileSystemStorage; + private readonly IWebLogger _logger; + private readonly ExifToolLocations _exifToolLocations; + private readonly ExifToolDownloadUnix _exifToolDownloadUnix; + + public ExifToolDownload(IHttpClientHelper httpClientHelper, AppSettings appSettings, + IWebLogger logger) + { + _httpClientHelper = httpClientHelper; + _appSettings = appSettings; + _hostFileSystemStorage = new StorageHostFullPathFilesystem(logger); + _logger = logger; + _exifToolLocations = new ExifToolLocations(_appSettings); + _exifToolDownloadUnix = new ExifToolDownloadUnix(_hostFileSystemStorage, appSettings, httpClientHelper, logger); + _exifToolDownloadWindows = new ExifToolDownloadWindows(_hostFileSystemStorage, appSettings, httpClientHelper, logger); + + } + + internal ExifToolDownload(IHttpClientHelper httpClientHelper, AppSettings appSettings, + IWebLogger logger, IStorage storage) + { + _httpClientHelper = httpClientHelper; + _appSettings = appSettings; + _hostFileSystemStorage = storage; + _logger = logger; + _exifToolLocations = new ExifToolLocations(_appSettings); + _exifToolDownloadUnix = new ExifToolDownloadUnix(_hostFileSystemStorage, appSettings, httpClientHelper, logger); + } + + /// + /// Auto Download Exiftool + /// + /// download windows version if true + /// check for min file size in bytes (Default = 30 bytes) + /// + public async Task DownloadExifTool(bool isWindows, int minimumSize = 30) + { + if ( _appSettings.ExiftoolSkipDownloadOnStartup == true || _appSettings is + { AddSwaggerExport: true, AddSwaggerExportExitAfter: true } ) + { + var name = _appSettings.ExiftoolSkipDownloadOnStartup == true + ? "ExiftoolSkipDownloadOnStartup" + : "AddSwaggerExport and AddSwaggerExportExitAfter"; + _logger.LogInformation($"[DownloadExifTool] Skipped due true of {name} setting"); + return false; + } + + new CreateFolderIfNotExists(_logger, _appSettings) + .CreateDirectoryDependenciesTempFolderIfNotExists(); + + if ( isWindows && + ( !_hostFileSystemStorage.ExistFile( + _exifToolLocations.ExeExifToolWindowsFullFilePath()) || + _hostFileSystemStorage.Info(_exifToolLocations.ExeExifToolWindowsFullFilePath()) + .Size <= + minimumSize ) ) + { + return await StartDownloadForWindows(); + } + + if ( !isWindows && + ( !_hostFileSystemStorage.ExistFile(_exifToolLocations + .ExeExifToolUnixFullFilePath()) || + _hostFileSystemStorage.Info(_exifToolLocations.ExeExifToolUnixFullFilePath()).Size <= + minimumSize ) ) + { + return await _exifToolDownloadUnix.StartDownloadForUnix(); + } + + if ( _appSettings.IsVerbose() ) + { + _logger.LogInformation( + $"[DownloadExifTool] {_exifToolLocations.ExeExifToolFullFilePath(isWindows)}"); + } + + // When running deploy scripts rights might reset (only for unix) + if ( isWindows ) return true; + + return await _exifToolDownloadUnix.RunChmodOnExifToolUnixExe(); + } + + + + +} diff --git a/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownloadUnix.cs b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownloadUnix.cs new file mode 100644 index 0000000000..1e9564cf4b --- /dev/null +++ b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownloadUnix.cs @@ -0,0 +1,141 @@ +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Net.Http; +using System.Threading; +using System.Threading.Tasks; +using Medallion.Shell; +using starsky.foundation.http.Interfaces; +using starsky.foundation.http.Services; +using starsky.foundation.platform.Interfaces; +using starsky.foundation.platform.Models; +using starsky.foundation.storage.ArchiveFormats; +using starsky.foundation.storage.Interfaces; + +namespace starsky.foundation.writemeta.Services.ExifToolDownloader; + +public class ExifToolDownloadUnix +{ + private readonly IStorage _hostFileSystemStorage; + private readonly AppSettings _appSettings; + private readonly IHttpClientHelper _httpClientHelper; + private readonly IWebLogger _logger; + private readonly ExifToolLocations _exifToolLocations; + private readonly CheckSums _checkSha1; + + public ExifToolDownloadUnix(IStorage hostFileSystemStorage, AppSettings appSettings, + IHttpClientHelper httpClientHelper, IWebLogger logger) + { + _hostFileSystemStorage = hostFileSystemStorage; + _appSettings = appSettings; + _httpClientHelper = httpClientHelper; + _logger = logger; + _exifToolLocations = new ExifToolLocations(_appSettings); + _checkSha1 = new CheckSums(_hostFileSystemStorage, httpClientHelper, logger); + } + + internal async Task StartDownloadForUnix() + { + var checksums = await DownloadCheckSums(); + if ( checksums == null ) return false; + var matchExifToolForUnixName = GetUnixTarGzFromChecksum(checksums.Value.Value); + return await DownloadForUnix(matchExifToolForUnixName, + GetChecksumsFromTextFile(checksums.Value.Value), !checksums.Value.Key); + } + + + internal async Task DownloadForUnix(string matchExifToolForUnixName, + IEnumerable getChecksumsFromTextFile, bool downloadFromMirror = false) + { + if ( _hostFileSystemStorage.ExistFile(_exifToolLocations.ExeExifToolUnixFullFilePath()) ) + { + return true; + } + + var tarGzArchiveFullFilePath = + Path.Combine(_appSettings.TempFolder, "exiftool.tar.gz"); + + var url = $"{ExifToolLocations.ExiftoolDownloadBasePath}{matchExifToolForUnixName}"; + if ( downloadFromMirror ) + url = $"{ExifToolLocations.ExiftoolDownloadBasePathMirror}{matchExifToolForUnixName}"; + + var unixDownloaded = await _httpClientHelper.Download(url, tarGzArchiveFullFilePath); + if ( !unixDownloaded ) + { + throw new HttpRequestException( + $"file is not downloaded {matchExifToolForUnixName}"); + } + + if ( !_checkSha1.CheckSha1(tarGzArchiveFullFilePath, getChecksumsFromTextFile) ) + { + throw new HttpRequestException( + $"checksum for {tarGzArchiveFullFilePath} is not valid"); + } + + await new TarBal(_hostFileSystemStorage).ExtractTarGz( + _hostFileSystemStorage.ReadStream(tarGzArchiveFullFilePath), + _appSettings.TempFolder, CancellationToken.None); + + var imageExifToolVersionFolder = _hostFileSystemStorage + .GetDirectories(_appSettings.TempFolder) + .FirstOrDefault(p => + p.StartsWith(Path.Combine(_appSettings.TempFolder, "Image-ExifTool-"))); + if ( imageExifToolVersionFolder != null ) + { + var exifToolUnixFolderFullFilePathTempFolder = + Path.Combine(_appSettings.TempFolder, "exiftool-unix"); + + if ( _hostFileSystemStorage.ExistFolder(exifToolUnixFolderFullFilePathTempFolder) ) + { + _hostFileSystemStorage.FolderDelete( + exifToolUnixFolderFullFilePathTempFolder); + } + + _hostFileSystemStorage.FolderMove(imageExifToolVersionFolder, + exifToolUnixFolderFullFilePathTempFolder); + + var exifToolUnixFolderFullFilePath = + Path.Combine(_appSettings.DependenciesFolder, "exiftool-unix"); + + _hostFileSystemStorage.FileCopy(imageExifToolVersionFolder, + exifToolUnixFolderFullFilePathTempFolder); + } + else + { + _logger.LogError($"[DownloadForUnix] ExifTool folder does not exists"); + return false; + } + + // remove tar.gz file afterwards + _hostFileSystemStorage.FileDelete(tarGzArchiveFullFilePath); + + var exifToolExePath = + Path.Combine(_appSettings.DependenciesFolder, "exiftool-unix", "exiftool"); + _logger.LogInformation( + $"[DownloadForUnix] ExifTool is just downloaded: {exifToolExePath} for {_appSettings.ApplicationType}"); + return await RunChmodOnExifToolUnixExe(); + } + + internal async Task RunChmodOnExifToolUnixExe() + { + // need to check again + // when not exist + if ( !_hostFileSystemStorage.ExistFile(ExeExifToolUnixFullFilePath()) ) return false; + if ( _appSettings.IsWindows ) return true; + + if ( !_hostFileSystemStorage.ExistFile("/bin/chmod") ) + { + _logger.LogError("[RunChmodOnExifToolUnixExe] WARNING: /bin/chmod does not exist"); + return true; + } + + // command.run does not care about the $PATH + var result = await Command.Run("/bin/chmod", "0755", ExeExifToolUnixFullFilePath()) + .Task; + if ( result.Success ) return true; + + _logger.LogError( + $"command failed with exit code {result.ExitCode}: {result.StandardError}"); + return false; + } +} diff --git a/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownloadWindows.cs b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownloadWindows.cs new file mode 100644 index 0000000000..87747d7fac --- /dev/null +++ b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolDownloadWindows.cs @@ -0,0 +1,72 @@ +using System.Collections.Generic; +using System.IO; +using System.Net.Http; +using System.Threading.Tasks; +using starsky.foundation.http.Interfaces; +using starsky.foundation.platform.Interfaces; +using starsky.foundation.platform.Models; +using starsky.foundation.storage.ArchiveFormats; +using starsky.foundation.storage.Interfaces; + +namespace starsky.foundation.writemeta.Services.ExifToolDownloader; + +public class ExifToolDownloadWindows(IStorage hostFileSystemStorage, AppSettings appSettings, + IHttpClientHelper httpClientHelper, IWebLogger logger) +{ + internal async Task StartDownloadForWindows() + { + var checksums = await DownloadCheckSums(); + if ( checksums == null ) return false; + + var matchExifToolForWindowsName = GetWindowsZipFromChecksum(checksums.Value.Value); + return await DownloadForWindows(matchExifToolForWindowsName, + GetChecksumsFromTextFile(checksums.Value.Value), !checksums.Value.Key); + } + + + private void MoveFileIfExist(string srcFullPath, string toFullPath) + { + if ( !_hostFileSystemStorage.ExistFile(srcFullPath) ) return; + _hostFileSystemStorage.FileMove(srcFullPath, toFullPath); + } + + internal async Task DownloadForWindows(string matchExifToolForWindowsName, + IEnumerable getChecksumsFromTextFile, bool downloadFromMirror = false) + { + if ( _hostFileSystemStorage.ExistFile( + ExeExifToolWindowsFullFilePath()) ) return true; + + var zipArchiveFullFilePath = + Path.Combine(_appSettings.DependenciesFolder, "exiftool.zip"); + var windowsExifToolFolder = + Path.Combine(_appSettings.DependenciesFolder, "exiftool-windows"); + + var url = $"{ExiftoolDownloadBasePath}{matchExifToolForWindowsName}"; + if ( downloadFromMirror ) + url = $"{ExiftoolDownloadBasePathMirror}{matchExifToolForWindowsName}"; + + var windowsDownloaded = await _httpClientHelper.Download(url, zipArchiveFullFilePath); + if ( !windowsDownloaded ) + { + throw new HttpRequestException( + $"file is not downloaded {matchExifToolForWindowsName}"); + } + + if ( !CheckSha1(zipArchiveFullFilePath, getChecksumsFromTextFile) ) + { + throw new HttpRequestException( + $"checksum for {zipArchiveFullFilePath} is not valid"); + } + + _hostFileSystemStorage.CreateDirectory(windowsExifToolFolder); + + new Zipper().ExtractZip(zipArchiveFullFilePath, windowsExifToolFolder); + MoveFileIfExist(Path.Combine(windowsExifToolFolder, "exiftool(-k).exe"), + Path.Combine(windowsExifToolFolder, "exiftool.exe")); + + _logger.LogInformation( + $"[DownloadForWindows] ExifTool downloaded: {ExeExifToolWindowsFullFilePath()}"); + return _hostFileSystemStorage.ExistFile(Path.Combine(windowsExifToolFolder, + "exiftool.exe")); + } +} diff --git a/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolLocations.cs b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolLocations.cs new file mode 100644 index 0000000000..1c5b6b2c30 --- /dev/null +++ b/starsky/starsky.foundation.writemeta/Services/ExifToolDownloader/ExifToolLocations.cs @@ -0,0 +1,39 @@ +using System.IO; +using starsky.foundation.platform.Models; + +namespace starsky.foundation.writemeta.Services.ExifToolDownloader; + +public class ExifToolLocations(AppSettings appSettings) +{ + public string ExeExifToolFullFilePath(bool isWindows) + { + return isWindows ? ExeExifToolWindowsFullFilePath() : ExeExifToolUnixFullFilePath(); + } + + internal string ExeExifToolWindowsFullFilePath() + { + return Path.Combine(Path.Combine(appSettings.DependenciesFolder, "exiftool-windows"), + "exiftool.exe"); + } + + internal string ExeExifToolUnixFullFilePath() + { + var path = Path.Combine(appSettings.DependenciesFolder, + "exiftool-unix", + "exiftool"); + return path; + } + + internal const string Https = "https://"; + + internal const string CheckSumLocation = "exiftool.org/checksums.txt"; + + internal const string CheckSumLocationMirror = + "qdraw.nl/special/mirror/exiftool/checksums.txt"; + + internal const string + ExiftoolDownloadBasePath = "exiftool.org/"; // with slash at the end + + internal const string ExiftoolDownloadBasePathMirror = + "qdraw.nl/special/mirror/exiftool/"; // with slash at the end +} diff --git a/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs b/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs index f741ccfd64..7b946c502a 100644 --- a/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs +++ b/starsky/starskytest/starsky.foundation.writemeta/Helpers/ExifToolDownloadTest.cs @@ -14,6 +14,7 @@ using starsky.foundation.storage.Interfaces; using starsky.foundation.storage.Storage; using starsky.foundation.writemeta.Services; +using starsky.foundation.writemeta.Services.ExifToolDownloader; using starskytest.FakeCreateAn; using starskytest.FakeMocks;