Catalog/imagecatalog/Program.cs

239 lines
8.5 KiB
C#
Raw Normal View History

using System.Runtime.InteropServices;
using Catalog.Communication.DependencyInjection;
using ImageCatalog;
2024-10-14 22:18:03 +02:00
using ImageCatalog_2.Services;
2025-07-23 17:16:06 +02:00
using MaddoShared;
2024-10-14 22:18:03 +02:00
using Microsoft.Extensions.DependencyInjection;
2026-02-04 23:16:06 +01:00
using AutoMapper;
using Microsoft.Extensions.Logging;
2025-07-28 14:45:03 +02:00
using Microsoft.Extensions.Logging.Abstractions;
using Microsoft.Extensions.Logging.Console;
using System.IO;
2025-07-28 14:45:03 +02:00
using Microsoft.Extensions.Options;
using Avalonia;
2024-10-14 22:18:03 +02:00
2025-07-28 14:45:03 +02:00
namespace ImageCatalog_2;
static class Program
2024-10-14 22:18:03 +02:00
{
#if WINDOWS
2025-07-28 14:45:03 +02:00
[DllImport("kernel32.dll", SetLastError = true)]
private static extern bool AllocConsole();
[DllImport("kernel32.dll", SetLastError = true)]
private static extern bool FreeConsole();
2025-07-28 14:45:03 +02:00
[DllImport("kernel32.dll", SetLastError = true)]
static extern IntPtr GetStdHandle(int nStdHandle);
2025-07-28 14:45:03 +02:00
private const int STD_OUTPUT_HANDLE = -11;
private const int STD_ERROR_HANDLE = -12;
2025-07-28 14:45:03 +02:00
[DllImport("kernel32.dll", SetLastError = true)]
static extern bool SetStdHandle(int nStdHandle, IntPtr handle);
2025-07-28 14:45:03 +02:00
[DllImport("kernel32.dll", SetLastError = true)]
static extern IntPtr GetConsoleWindow();
2025-07-28 14:45:03 +02:00
[DllImport("kernel32.dll", SetLastError = true)]
static extern bool AttachConsole(int dwProcessId);
2026-05-09 17:27:05 +02:00
private const int ATTACH_PARENT_PROCESS = -1;
[DllImport("kernel32.dll", SetLastError = true)]
private static extern bool GenerateConsoleCtrlEvent(uint dwCtrlEvent, uint dwProcessGroupId);
[DllImport("kernel32.dll", SetLastError = true)]
private static extern bool SetConsoleCtrlHandler(ConsoleCtrlDelegate? handlerRoutine, bool add);
2025-07-28 14:45:03 +02:00
[DllImport("kernel32.dll", SetLastError = true)]
static extern IntPtr CreateFile(
string lpFileName,
uint dwDesiredAccess,
uint dwShareMode,
IntPtr lpSecurityAttributes,
uint dwCreationDisposition,
uint dwFlagsAndAttributes,
IntPtr hTemplateFile);
2025-07-28 14:45:03 +02:00
private const uint GENERIC_WRITE = 0x40000000;
private const uint OPEN_EXISTING = 3;
private const uint CTRL_C_EVENT = 0;
private delegate bool ConsoleCtrlDelegate(uint ctrlType);
2025-07-28 14:45:03 +02:00
private static void RedirectConsoleOutput()
{
var stdOutHandle = CreateFile("CONOUT$", GENERIC_WRITE, 0, IntPtr.Zero, OPEN_EXISTING, 0, IntPtr.Zero);
var safeFileHandle = new Microsoft.Win32.SafeHandles.SafeFileHandle(stdOutHandle, true);
var fileStream = new FileStream(safeFileHandle, FileAccess.Write);
var standardOutput = new StreamWriter(fileStream) { AutoFlush = true };
Console.SetOut(standardOutput);
Console.SetError(standardOutput);
}
internal static bool TrySendConsoleInterrupt(int processId)
{
_ = processId;
return false;
}
#endif
public static IServiceProvider ServiceProvider { get; private set; } = default!;
public static Avalonia.AppBuilder BuildAvaloniaApp()
=> Avalonia.AppBuilder.Configure<AvaloniaApp>()
.UsePlatformDetect()
.LogToTrace();
2025-07-28 14:45:03 +02:00
[STAThread]
2026-05-09 17:27:05 +02:00
static int Main(string[] args)
2025-07-28 14:45:03 +02:00
{
#if WINDOWS
2026-05-09 17:27:05 +02:00
if (CommandLineOperationRunner.IsHeadlessRequest(args))
{
AttachConsole(ATTACH_PARENT_PROCESS);
}
else
{
AllocConsole();
}
2025-07-28 14:45:03 +02:00
RedirectConsoleOutput();
#endif
2025-07-28 14:45:03 +02:00
var serviceCollection = new ServiceCollection();
ConfigureServices(serviceCollection);
2024-10-14 22:18:03 +02:00
2025-07-28 14:45:03 +02:00
ServiceProvider = serviceCollection.BuildServiceProvider();
2024-10-14 22:18:03 +02:00
2026-05-09 17:27:05 +02:00
if (CommandLineOperationRunner.IsHeadlessRequest(args))
{
return CommandLineOperationRunner.RunAsync(ServiceProvider, args ?? Array.Empty<string>())
.GetAwaiter()
.GetResult();
}
BuildAvaloniaApp().StartWithClassicDesktopLifetime(args ?? Array.Empty<string>());
2026-05-09 17:27:05 +02:00
return 0;
2025-07-28 14:45:03 +02:00
}
2024-10-14 22:18:03 +02:00
2025-07-28 14:45:03 +02:00
private static void ConfigureServices(ServiceCollection services)
{
2026-02-04 23:16:06 +01:00
services.AddAutoMapper(cfg => { }, typeof(Program));
2025-07-28 14:45:03 +02:00
services.AddTransient<ITestService, TestService>();
2026-02-04 19:48:03 +01:00
services.AddTransient<ISettingsService, SettingsService>();
2024-10-14 22:18:03 +02:00
services.AddTransient<DataModel>(sp =>
{
var testService = sp.GetRequiredService<ITestService>();
var settingsService = sp.GetRequiredService<ISettingsService>();
2026-02-21 15:53:52 +01:00
var imageCreation = sp.GetRequiredService<ImageCreationService>();
var aiExtractionService = sp.GetRequiredService<IAiExtractionService>();
var imageProcessingCoordinator = sp.GetRequiredService<IImageProcessingCoordinator>();
var picSettings = sp.GetRequiredService<PicSettings>();
var mapper = sp.GetRequiredService<IMapper>();
var logger = sp.GetRequiredService<ILogger<DataModel>>();
var versionProvider = sp.GetService<MaddoShared.IVersionProvider>();
return new DataModel(testService, settingsService, imageCreation, aiExtractionService, imageProcessingCoordinator, picSettings, mapper, logger, versionProvider);
});
2024-10-14 23:05:18 +02:00
services.AddTransient<IAiExtractionService, AiExtractionService>();
services.AddTransient<IImageProcessingCoordinator, ImageProcessingCoordinator>();
2026-02-21 15:53:52 +01:00
services.AddTransient<ImageCreationService>();
#if WINDOWS
2026-02-21 15:53:52 +01:00
services.AddTransient<ImageCreatorGDI>();
#endif
2026-02-21 15:53:52 +01:00
services.AddTransient<ImageCreatorImageSharp>();
services.AddTransient<ImageCreatorMapper>();
services.AddTransient<IImageCreator>(sp => sp.GetRequiredService<ImageCreatorMapper>());
2025-07-28 09:15:45 +02:00
var userPrefsPath = System.IO.Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData),
"ImageCatalog", "userprefs.xml");
services.AddSingleton(new ParametriSetup(userPrefsPath));
services.AddSingleton<PickerPreferenceService>();
2025-07-28 14:45:03 +02:00
services.AddSingleton<PicSettings>();
services.AddCatalogCommunication(options =>
{
options.BaseUri = new Uri("https://www.regalamiunsorriso.it/");
options.AdminPageBasePath = "admin/pg_RUS";
options.ReceiveFilePath = "ReceiveFile.abl";
options.RequestTimeout = TimeSpan.FromSeconds(30);
options.RetryCount = 2;
options.RetryBaseDelay = TimeSpan.FromMilliseconds(250);
});
services.AddTransient<AvaloniaMainWindow>();
services.AddSingleton<MaddoShared.IVersionProvider, MaddoShared.VersionProvider>();
2025-07-28 14:45:03 +02:00
services.AddLogging(configure =>
{
configure.AddCustomFormatter();
configure.AddConsole();
configure.SetMinimumLevel(LogLevel.Debug);
});
2024-10-14 22:18:03 +02:00
}
}
2025-07-28 14:45:03 +02:00
public static class ConsoleLoggerExtensions
{
public static ILoggingBuilder AddCustomFormatter(
this ILoggingBuilder builder) =>
builder
.AddConsole(options => options.FormatterName = nameof(CustomLoggingFormatter))
.AddConsoleFormatter<CustomLoggingFormatter, ConsoleFormatterOptions>()
.AddFilter("LuckyPennySoftware.AutoMapper.License", LogLevel.None);
}
public sealed class CustomLoggingFormatter : ConsoleFormatter, IDisposable
{
private readonly IDisposable? _optionsReloadToken;
private ConsoleFormatterOptions _formatterOptions;
public CustomLoggingFormatter(IOptionsMonitor<ConsoleFormatterOptions> options)
2025-07-28 14:45:03 +02:00
// Case insensitive
: base(nameof(CustomLoggingFormatter)) =>
(_optionsReloadToken, _formatterOptions) =
(options.OnChange(ReloadLoggerOptions), options.CurrentValue);
private void ReloadLoggerOptions(ConsoleFormatterOptions options) =>
_formatterOptions = options;
public override void Write<TState>(
in LogEntry<TState> logEntry,
IExternalScopeProvider? scopeProvider,
TextWriter? textWriter)
{
if (textWriter is null)
{
return;
}
string? message =
logEntry.Formatter?.Invoke(
logEntry.State, logEntry.Exception);
if (message is null)
{
return;
}
var timestamp = DateTimeOffset.Now.ToString("yyyy-MM-dd HH:mm:ss.fff");
var level = logEntry.LogLevel.ToString().ToUpperInvariant();
var category = logEntry.Category ?? "App";
var line = $"{timestamp} [{level}] {category}: {message}";
textWriter.WriteLine(line);
System.Diagnostics.Debug.WriteLine(line);
if (logEntry.Exception is not null)
{
var exceptionText = logEntry.Exception.ToString();
textWriter.WriteLine(exceptionText);
System.Diagnostics.Debug.WriteLine(exceptionText);
}
}
public void Dispose() => _optionsReloadToken?.Dispose();
}