diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 75915fe8..932b9c21 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -27,9 +27,9 @@ jobs: uses: actions/checkout@v4 - name: Install .NET Core - uses: actions/setup-dotnet@v3 + uses: actions/setup-dotnet@v4 with: - dotnet-version: '7.0.x' + dotnet-version: '8.0.x' - name: Build & test (Release) run: dotnet test src -c Release diff --git a/Dockerfile b/Dockerfile index 0c8c702f..a6722338 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,20 +1,20 @@ -FROM --platform=$BUILDPLATFORM mcr.microsoft.com/dotnet/sdk:7.0 AS build -ARG TARGETARCH -WORKDIR /app - -RUN mkdir -p src/UnityNuGet && mkdir -p src/UnityNuGet.Server && mkdir -p src/UnityNuGet.Tests - -COPY src/*.sln src -COPY src/UnityNuGet/*.csproj src/UnityNuGet -COPY src/UnityNuGet.Server/*.csproj src/UnityNuGet.Server -COPY src/UnityNuGet.Tests/*.csproj src/UnityNuGet.Tests -RUN dotnet restore src -a $TARGETARCH - -COPY . ./ -RUN dotnet publish src -a $TARGETARCH -c Release -o /app/src/out - -# Build runtime image -FROM mcr.microsoft.com/dotnet/aspnet:7.0 -WORKDIR /app -COPY --from=build /app/src/out . -ENTRYPOINT ["dotnet", "UnityNuGet.Server.dll"] +FROM --platform=$BUILDPLATFORM mcr.microsoft.com/dotnet/sdk:8.0 AS build +ARG TARGETARCH +WORKDIR /app + +RUN mkdir -p src/UnityNuGet && mkdir -p src/UnityNuGet.Server && mkdir -p src/UnityNuGet.Tests + +COPY src/*.sln src +COPY src/UnityNuGet/*.csproj src/UnityNuGet +COPY src/UnityNuGet.Server/*.csproj src/UnityNuGet.Server +COPY src/UnityNuGet.Tests/*.csproj src/UnityNuGet.Tests +RUN dotnet restore src -a $TARGETARCH + +COPY . ./ +RUN dotnet publish src -a $TARGETARCH -c Release -o /app/src/out + +# Build runtime image +FROM mcr.microsoft.com/dotnet/aspnet:8.0 +WORKDIR /app +COPY --from=build /app/src/out . +ENTRYPOINT ["dotnet", "UnityNuGet.Server.dll"] diff --git a/src/Directory.Build.props b/src/Directory.Build.props new file mode 100644 index 00000000..b91d59b2 --- /dev/null +++ b/src/Directory.Build.props @@ -0,0 +1,8 @@ + + + + enable + true + + + diff --git a/src/Directory.Packages.props b/src/Directory.Packages.props new file mode 100644 index 00000000..a423acc8 --- /dev/null +++ b/src/Directory.Packages.props @@ -0,0 +1,24 @@ + + + true + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/UnityNuGet.Server/Controllers/ApiController.cs b/src/UnityNuGet.Server/Controllers/ApiController.cs index 95aade7a..43d3cc15 100644 --- a/src/UnityNuGet.Server/Controllers/ApiController.cs +++ b/src/UnityNuGet.Server/Controllers/ApiController.cs @@ -1,130 +1,124 @@ -using System.IO; -using System.Linq; -using System.Text; -using Microsoft.AspNetCore.Mvc; -using UnityNuGet.Npm; - -namespace UnityNuGet.Server.Controllers -{ - /// - /// Main entry to emulate the following NPM endpoints: - /// - /// - "-/all": query all packages (return json) - /// - "{packageId}": query a specific package (return json) - /// - "{package_id}/-/{package_file}": download a specific package - /// - [Route("")] - [ApiController] - public class ApiController : ControllerBase - { - private readonly RegistryCacheSingleton _cacheSingleton; - private readonly RegistryCacheReport _registryCacheReport; - - public ApiController(RegistryCacheSingleton cacheSingleton, RegistryCacheReport registryCacheReport) - { - _cacheSingleton = cacheSingleton; - _registryCacheReport = registryCacheReport; - } +using System.IO; +using System.Linq; +using System.Text; +using Microsoft.AspNetCore.Mvc; +using UnityNuGet.Npm; + +namespace UnityNuGet.Server.Controllers +{ + /// + /// Main entry to emulate the following NPM endpoints: + /// + /// - "-/all": query all packages (return json) + /// - "{packageId}": query a specific package (return json) + /// - "{package_id}/-/{package_file}": download a specific package + /// + [Route("")] + [ApiController] + public class ApiController(RegistryCacheSingleton cacheSingleton, RegistryCacheReport registryCacheReport) : ControllerBase + { + private readonly RegistryCacheSingleton _cacheSingleton = cacheSingleton; + private readonly RegistryCacheReport _registryCacheReport = registryCacheReport; // GET / - [HttpGet("")] - public IActionResult Home() - { - return Ok(); - } - - // GET -/all - [HttpGet("-/all")] - public JsonResult GetAll() - { - if (!TryGetInstance(out var instance, out var error)) return new JsonResult(error); - - var result = instance?.All(); - return new JsonResult(result); - } - - // GET {packageId} - [HttpGet("{id}")] - public JsonResult GetPackage(string id) - { - if (!TryGetInstance(out var instance, out var error)) return new JsonResult(error); - - var package = instance?.GetPackage(id); - if (package == null) - { - return new JsonResult(NpmError.NotFound); - } - - return new JsonResult(package); - } - - // GET {package_id}/-/{package_file} - [HttpGet("{id}/-/{file}")] - [HttpHead("{id}/-/{file}")] - public IActionResult DownloadPackage(string id, string file) - { - if (!TryGetInstance(out var instance, out var error)) return new JsonResult(error); - - var package = instance?.GetPackage(id); - if (package == null) - { - return new JsonResult(NpmError.NotFound); - } - - if (!file.StartsWith(id + "-") || !file.EndsWith(".tgz")) - { - return new JsonResult(NpmError.NotFound); - } - - var filePath = instance?.GetPackageFilePath(file); - if (string.IsNullOrEmpty(filePath) || !System.IO.File.Exists(filePath)) - { - return new JsonResult(NpmError.NotFound); - } - - // This method can be called with HEAD request, so in that case we just calculate the content length - if (Request.Method.Equals("HEAD")) - { - Response.ContentType = "application/octet-stream"; - Response.ContentLength = new FileInfo(filePath).Length; - return Ok(); - } - else - { - return new PhysicalFileResult(filePath, "application/octet-stream") { FileDownloadName = file }; - } - } - - private bool TryGetInstance(out RegistryCache? cacheInstance, out NpmError? npmError) - { - var instance = _cacheSingleton.Instance; - cacheInstance = instance; - - if (instance == null) - { - if (_registryCacheReport.ErrorMessages.Any()) - { - var stringBuilder = new StringBuilder(); - stringBuilder.AppendLine("Error initializing the server:"); - - foreach (var error in _registryCacheReport.ErrorMessages) - { - stringBuilder.AppendLine(error); - } - - npmError = new NpmError("not_initialized", stringBuilder.ToString()); - } - else - { - npmError = new NpmError("not_initialized", $"The server is initializing ({_registryCacheReport.Progress:F1}% completed). Please retry later..."); - } - } - else - { - npmError = null; - } - - return instance != null; - } - } -} + [HttpGet("")] + public IActionResult Home() + { + return Ok(); + } + + // GET -/all + [HttpGet("-/all")] + public JsonResult GetAll() + { + if (!TryGetInstance(out var instance, out var error)) return new JsonResult(error); + + var result = instance?.All(); + return new JsonResult(result); + } + + // GET {packageId} + [HttpGet("{id}")] + public JsonResult GetPackage(string id) + { + if (!TryGetInstance(out var instance, out var error)) return new JsonResult(error); + + var package = instance?.GetPackage(id); + if (package == null) + { + return new JsonResult(NpmError.NotFound); + } + + return new JsonResult(package); + } + + // GET {package_id}/-/{package_file} + [HttpGet("{id}/-/{file}")] + [HttpHead("{id}/-/{file}")] + public IActionResult DownloadPackage(string id, string file) + { + if (!TryGetInstance(out var instance, out var error)) return new JsonResult(error); + + var package = instance?.GetPackage(id); + if (package == null) + { + return new JsonResult(NpmError.NotFound); + } + + if (!file.StartsWith(id + "-") || !file.EndsWith(".tgz")) + { + return new JsonResult(NpmError.NotFound); + } + + var filePath = instance?.GetPackageFilePath(file); + if (string.IsNullOrEmpty(filePath) || !System.IO.File.Exists(filePath)) + { + return new JsonResult(NpmError.NotFound); + } + + // This method can be called with HEAD request, so in that case we just calculate the content length + if (Request.Method.Equals("HEAD")) + { + Response.ContentType = "application/octet-stream"; + Response.ContentLength = new FileInfo(filePath).Length; + return Ok(); + } + else + { + return new PhysicalFileResult(filePath, "application/octet-stream") { FileDownloadName = file }; + } + } + + private bool TryGetInstance(out RegistryCache? cacheInstance, out NpmError? npmError) + { + var instance = _cacheSingleton.Instance; + cacheInstance = instance; + + if (instance == null) + { + if (_registryCacheReport.ErrorMessages.Any()) + { + var stringBuilder = new StringBuilder(); + stringBuilder.AppendLine("Error initializing the server:"); + + foreach (var error in _registryCacheReport.ErrorMessages) + { + stringBuilder.AppendLine(error); + } + + npmError = new NpmError("not_initialized", stringBuilder.ToString()); + } + else + { + npmError = new NpmError("not_initialized", $"The server is initializing ({_registryCacheReport.Progress:F1}% completed). Please retry later..."); + } + } + else + { + npmError = null; + } + + return instance != null; + } + } +} diff --git a/src/UnityNuGet.Server/NuGetRedirectLogger.cs b/src/UnityNuGet.Server/NuGetRedirectLogger.cs index 1dd600ff..3e03d271 100644 --- a/src/UnityNuGet.Server/NuGetRedirectLogger.cs +++ b/src/UnityNuGet.Server/NuGetRedirectLogger.cs @@ -1,47 +1,42 @@ -using System; -using System.Threading.Tasks; -using Microsoft.Extensions.Logging; -using NuGet.Common; -using ILogger = Microsoft.Extensions.Logging.ILogger; -using LogLevel = NuGet.Common.LogLevel; - -namespace UnityNuGet.Server -{ - /// - /// A NuGet logger redirecting to a - /// - public class NuGetRedirectLogger : LoggerBase - { - private readonly ILogger _logger; - - public NuGetRedirectLogger(ILogger logger) - { - _logger = logger ?? throw new ArgumentNullException(nameof(logger)); - } - - public override void Log(ILogMessage message) - { - LoggerExtensions.Log(_logger, GetLogLevel(message.Level), "{Message}", message.Message); - } - - public override Task LogAsync(ILogMessage message) - { - LoggerExtensions.Log(_logger, GetLogLevel(message.Level), "{Message}", message.Message); - return Task.CompletedTask; - } - - private static Microsoft.Extensions.Logging.LogLevel GetLogLevel(LogLevel logLevel) - { - return logLevel switch - { - LogLevel.Debug => Microsoft.Extensions.Logging.LogLevel.Debug, - LogLevel.Verbose => Microsoft.Extensions.Logging.LogLevel.Trace, - LogLevel.Information => Microsoft.Extensions.Logging.LogLevel.Information, - LogLevel.Minimal => Microsoft.Extensions.Logging.LogLevel.Information, - LogLevel.Warning => Microsoft.Extensions.Logging.LogLevel.Warning, - LogLevel.Error => Microsoft.Extensions.Logging.LogLevel.Error, - _ => throw new ArgumentOutOfRangeException(nameof(logLevel), logLevel, null), - }; - } - } -} +using System; +using System.Threading.Tasks; +using Microsoft.Extensions.Logging; +using NuGet.Common; +using ILogger = Microsoft.Extensions.Logging.ILogger; +using LogLevel = NuGet.Common.LogLevel; + +namespace UnityNuGet.Server +{ + /// + /// A NuGet logger redirecting to a + /// + public class NuGetRedirectLogger(ILogger logger) : LoggerBase + { + private readonly ILogger _logger = logger ?? throw new ArgumentNullException(nameof(logger)); + + public override void Log(ILogMessage message) + { + LoggerExtensions.Log(_logger, GetLogLevel(message.Level), "{Message}", message.Message); + } + + public override Task LogAsync(ILogMessage message) + { + LoggerExtensions.Log(_logger, GetLogLevel(message.Level), "{Message}", message.Message); + return Task.CompletedTask; + } + + private static Microsoft.Extensions.Logging.LogLevel GetLogLevel(LogLevel logLevel) + { + return logLevel switch + { + LogLevel.Debug => Microsoft.Extensions.Logging.LogLevel.Debug, + LogLevel.Verbose => Microsoft.Extensions.Logging.LogLevel.Trace, + LogLevel.Information => Microsoft.Extensions.Logging.LogLevel.Information, + LogLevel.Minimal => Microsoft.Extensions.Logging.LogLevel.Information, + LogLevel.Warning => Microsoft.Extensions.Logging.LogLevel.Warning, + LogLevel.Error => Microsoft.Extensions.Logging.LogLevel.Error, + _ => throw new ArgumentOutOfRangeException(nameof(logLevel), logLevel, null), + }; + } + } +} diff --git a/src/UnityNuGet.Server/RegistryCacheInitializer.cs b/src/UnityNuGet.Server/RegistryCacheInitializer.cs index 9d6f8892..4fdc8f65 100644 --- a/src/UnityNuGet.Server/RegistryCacheInitializer.cs +++ b/src/UnityNuGet.Server/RegistryCacheInitializer.cs @@ -1,82 +1,73 @@ -using System; -using System.IO; -using System.Linq; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.AspNetCore.Hosting; -using Microsoft.Extensions.Configuration; -using Microsoft.Extensions.Hosting; -using Microsoft.Extensions.Logging; -using Microsoft.Extensions.Options; - -namespace UnityNuGet.Server -{ - public class RegistryCacheInitializer : IHostedService - { - private readonly IConfiguration configuration; - private readonly IHostEnvironment hostEnvironment; - private readonly ILoggerFactory loggerFactory; - private readonly RegistryOptions registryOptions; - private readonly RegistryCacheSingleton registryCacheSingleton; - - public RegistryCacheInitializer(IConfiguration configuration, IHostEnvironment hostEnvironment, ILoggerFactory loggerFactory, IOptions registryOptionsAccessor, RegistryCacheSingleton registryCacheSingleton) - { - this.configuration = configuration; - this.hostEnvironment = hostEnvironment; - this.loggerFactory = loggerFactory; - registryOptions = registryOptionsAccessor.Value; - this.registryCacheSingleton = registryCacheSingleton; - } - - public Task StartAsync(CancellationToken cancellationToken) - { - var loggerRedirect = new NuGetRedirectLogger(loggerFactory.CreateLogger("NuGet")); - - Uri uri = registryOptions.RootHttpUrl!; - - bool isDevelopment = hostEnvironment.IsDevelopment(); - if (isDevelopment) - { - var urls = configuration[WebHostDefaults.ServerUrlsKey]; - - // Select HTTPS in production, HTTP in development - var url = (urls?.Split(';').FirstOrDefault(x => !x.StartsWith("https"))) ?? throw new InvalidOperationException($"Unable to find a proper server URL from `{urls}`. Expecting a `http://...` URL in development"); - - // https://docs.microsoft.com/en-us/dotnet/core/tools/dotnet-environment-variables#dotnet_running_in_container-and-dotnet_running_in_containers - bool runningInContainer = configuration.GetValue("DOTNET_RUNNING_IN_CONTAINER"); - - uri = new Uri(runningInContainer ? url.Replace("+", "localhost") : url); - } - - // Get the current directory from registry options (prepend binary folder in dev) - string unityPackageFolder; - - if (isDevelopment) - { - var currentDirectory = Path.GetDirectoryName(typeof(Startup).Assembly.Location)!; - unityPackageFolder = Path.Combine(currentDirectory, new DirectoryInfo(registryOptions.RootPersistentFolder!).Name); - } - else - { - if (Path.IsPathRooted(registryOptions.RootPersistentFolder)) - { - unityPackageFolder = registryOptions.RootPersistentFolder; - } - else - { - unityPackageFolder = Path.Combine(Directory.GetCurrentDirectory(), registryOptions.RootPersistentFolder!); - } - } - loggerRedirect.LogInformation($"Using Unity Package folder `{unityPackageFolder}`"); - - // Add the cache accessible from the services - registryCacheSingleton.UnityPackageFolder = unityPackageFolder; - registryCacheSingleton.ServerUri = uri; - registryCacheSingleton.NuGetRedirectLogger = loggerRedirect; - - return Task.CompletedTask; - } - - public Task StopAsync(CancellationToken cancellationToken) => Task.CompletedTask; - } -} +using System; +using System.IO; +using System.Linq; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.AspNetCore.Hosting; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.Hosting; +using Microsoft.Extensions.Logging; +using Microsoft.Extensions.Options; + +namespace UnityNuGet.Server +{ + public class RegistryCacheInitializer(IConfiguration configuration, IHostEnvironment hostEnvironment, ILoggerFactory loggerFactory, IOptions registryOptionsAccessor, RegistryCacheSingleton registryCacheSingleton) : IHostedService + { + private readonly IConfiguration configuration = configuration; + private readonly IHostEnvironment hostEnvironment = hostEnvironment; + private readonly ILoggerFactory loggerFactory = loggerFactory; + private readonly RegistryOptions registryOptions = registryOptionsAccessor.Value; + private readonly RegistryCacheSingleton registryCacheSingleton = registryCacheSingleton; + + public Task StartAsync(CancellationToken cancellationToken) + { + var loggerRedirect = new NuGetRedirectLogger(loggerFactory.CreateLogger("NuGet")); + + Uri uri = registryOptions.RootHttpUrl!; + + bool isDevelopment = hostEnvironment.IsDevelopment(); + if (isDevelopment) + { + var urls = configuration[WebHostDefaults.ServerUrlsKey]; + + // Select HTTPS in production, HTTP in development + var url = (urls?.Split(';').FirstOrDefault(x => !x.StartsWith("https"))) ?? throw new InvalidOperationException($"Unable to find a proper server URL from `{urls}`. Expecting a `http://...` URL in development"); + + // https://docs.microsoft.com/en-us/dotnet/core/tools/dotnet-environment-variables#dotnet_running_in_container-and-dotnet_running_in_containers + bool runningInContainer = configuration.GetValue("DOTNET_RUNNING_IN_CONTAINER"); + + uri = new Uri(runningInContainer ? url.Replace("+", "localhost") : url); + } + + // Get the current directory from registry options (prepend binary folder in dev) + string unityPackageFolder; + + if (isDevelopment) + { + var currentDirectory = Path.GetDirectoryName(typeof(Startup).Assembly.Location)!; + unityPackageFolder = Path.Combine(currentDirectory, new DirectoryInfo(registryOptions.RootPersistentFolder!).Name); + } + else + { + if (Path.IsPathRooted(registryOptions.RootPersistentFolder)) + { + unityPackageFolder = registryOptions.RootPersistentFolder; + } + else + { + unityPackageFolder = Path.Combine(Directory.GetCurrentDirectory(), registryOptions.RootPersistentFolder!); + } + } + loggerRedirect.LogInformation($"Using Unity Package folder `{unityPackageFolder}`"); + + // Add the cache accessible from the services + registryCacheSingleton.UnityPackageFolder = unityPackageFolder; + registryCacheSingleton.ServerUri = uri; + registryCacheSingleton.NuGetRedirectLogger = loggerRedirect; + + return Task.CompletedTask; + } + + public Task StopAsync(CancellationToken cancellationToken) => Task.CompletedTask; + } +} diff --git a/src/UnityNuGet.Server/RegistryCacheReport.cs b/src/UnityNuGet.Server/RegistryCacheReport.cs index 4861e521..f50b4ad3 100644 --- a/src/UnityNuGet.Server/RegistryCacheReport.cs +++ b/src/UnityNuGet.Server/RegistryCacheReport.cs @@ -1,84 +1,78 @@ -using System; -using System.Collections.Generic; -using Microsoft.Extensions.Options; - -namespace UnityNuGet.Server -{ - /// - /// Main class that stores relevant messages that may appear during the build of the Unity packages with . - /// - public class RegistryCacheReport - { - private readonly IList _informationMessages = new List(); - private readonly IList _warningMessages = new List(); - private readonly IList _errorMessages = new List(); - - private DateTime? _lastUpdate; - - private readonly RegistryCacheSingleton _registryCacheSingleton; - private readonly RegistryOptions _registryOptions; - - public RegistryCacheReport(RegistryCacheSingleton registryCacheSingleton, IOptions registryOptionsAccessor) - { - _registryCacheSingleton = registryCacheSingleton; - _registryOptions = registryOptionsAccessor.Value; - } - - public IEnumerable InformationMeessages => _informationMessages; - - public IEnumerable WarningMessages => _warningMessages; - - public IEnumerable ErrorMessages => _errorMessages; - - public bool Running { get; private set; } - - public double Progress - { - get - { - var currentIndex = _registryCacheSingleton.ProgressPackageIndex; - var totalCount = _registryCacheSingleton.ProgressTotalPackageCount; - var percent = totalCount != 0 ? (double)currentIndex * 100 / totalCount : 0; - - return percent; - } - } - - public TimeSpan? TimeRemainingForNextUpdate - { - get - { - if (_errorMessages.Count == 0) - { - return _lastUpdate != null ? _lastUpdate.Value.Add(_registryOptions.UpdateInterval) - DateTime.UtcNow : null; - } - else - { - return TimeSpan.FromSeconds(0); - } - } - } - - public void AddInformation(string message) => _informationMessages.Add(message); - - public void AddWarning(string message) => _warningMessages.Add(message); - - public void AddError(string message) => _errorMessages.Add(message); - - public void Start() - { - Running = true; - - _informationMessages.Clear(); - _warningMessages.Clear(); - _errorMessages.Clear(); - } - - public void Complete() - { - Running = false; - - _lastUpdate = DateTime.UtcNow; - } - } -} +using System; +using System.Collections.Generic; +using Microsoft.Extensions.Options; + +namespace UnityNuGet.Server +{ + /// + /// Main class that stores relevant messages that may appear during the build of the Unity packages with . + /// + public class RegistryCacheReport(RegistryCacheSingleton registryCacheSingleton, IOptions registryOptionsAccessor) + { + private readonly RegistryCacheSingleton _registryCacheSingleton = registryCacheSingleton; + private readonly RegistryOptions _registryOptions = registryOptionsAccessor.Value; + + private readonly List _informationMessages = []; + private readonly List _warningMessages = []; + private readonly List _errorMessages = []; + + private DateTime? _lastUpdate; + + public IEnumerable InformationMeessages => _informationMessages; + + public IEnumerable WarningMessages => _warningMessages; + + public IEnumerable ErrorMessages => _errorMessages; + + public bool Running { get; private set; } + + public double Progress + { + get + { + var currentIndex = _registryCacheSingleton.ProgressPackageIndex; + var totalCount = _registryCacheSingleton.ProgressTotalPackageCount; + var percent = totalCount != 0 ? (double)currentIndex * 100 / totalCount : 0; + + return percent; + } + } + + public TimeSpan? TimeRemainingForNextUpdate + { + get + { + if (_errorMessages.Count == 0) + { + return _lastUpdate != null ? _lastUpdate.Value.Add(_registryOptions.UpdateInterval) - DateTime.UtcNow : null; + } + else + { + return TimeSpan.FromSeconds(0); + } + } + } + + public void AddInformation(string message) => _informationMessages.Add(message); + + public void AddWarning(string message) => _warningMessages.Add(message); + + public void AddError(string message) => _errorMessages.Add(message); + + public void Start() + { + Running = true; + + _informationMessages.Clear(); + _warningMessages.Clear(); + _errorMessages.Clear(); + } + + public void Complete() + { + Running = false; + + _lastUpdate = DateTime.UtcNow; + } + } +} diff --git a/src/UnityNuGet.Server/RegistryCacheUpdater.cs b/src/UnityNuGet.Server/RegistryCacheUpdater.cs index d3dbdd95..aed12d3f 100644 --- a/src/UnityNuGet.Server/RegistryCacheUpdater.cs +++ b/src/UnityNuGet.Server/RegistryCacheUpdater.cs @@ -1,82 +1,74 @@ -using System; -using System.Linq; -using System.Threading; -using System.Threading.Tasks; -using Microsoft.Extensions.Hosting; -using Microsoft.Extensions.Logging; -using Microsoft.Extensions.Options; - -namespace UnityNuGet.Server -{ - /// - /// Update the RegistryCache at a regular interval - /// - internal sealed class RegistryCacheUpdater : BackgroundService - { - private readonly RegistryCacheReport _registryCacheReport; - private readonly RegistryCacheSingleton _currentRegistryCache; - private readonly ILogger _logger; - private readonly RegistryOptions _registryOptions; - - public RegistryCacheUpdater(RegistryCacheReport registryCacheReport, RegistryCacheSingleton currentRegistryCache, ILogger logger, IOptions registryOptionsAccessor) - { - _registryCacheReport = registryCacheReport; - _currentRegistryCache = currentRegistryCache; - _logger = logger; - _registryOptions = registryOptionsAccessor.Value; - } - - protected override async Task ExecuteAsync(CancellationToken stoppingToken) - { - try - { - while (!stoppingToken.IsCancellationRequested) - { - _logger.LogInformation("Starting to update RegistryCache"); - - _registryCacheReport.Start(); - - var newRegistryCache = new RegistryCache(_currentRegistryCache.UnityPackageFolder!, _currentRegistryCache.ServerUri!, _registryOptions.UnityScope!, _registryOptions.MinimumUnityVersion!, _registryOptions.PackageNameNuGetPostFix!, _registryOptions.TargetFrameworks!, _currentRegistryCache.NuGetRedirectLogger!) - { - // Update progress - OnProgress = (current, total) => - { - _currentRegistryCache.ProgressTotalPackageCount = total; - _currentRegistryCache.ProgressPackageIndex = current; - }, - OnInformation = message => _registryCacheReport.AddInformation(message), - OnWarning = message => _registryCacheReport.AddWarning(message), - OnError = message => _registryCacheReport.AddError(message) - }; - - await newRegistryCache.Build(); - - if (_registryCacheReport.ErrorMessages.Any()) - { - _logger.LogInformation("RegistryCache not updated due to errors. See previous logs"); - } - else - { - // Update the registry cache in the services - _currentRegistryCache.Instance = newRegistryCache; - - _logger.LogInformation("RegistryCache successfully updated"); - } - - _registryCacheReport.Complete(); - - await Task.Delay((int)_registryOptions.UpdateInterval.TotalMilliseconds, stoppingToken); - } - } - catch (Exception ex) - { - string message = "Error while building a new registry cache."; - - _logger.LogError(ex, "{Message}", message); - - _registryCacheReport.AddError($"{message}. Reason: {ex}"); - _registryCacheReport.Complete(); - } - } - } -} +using System; +using System.Linq; +using System.Threading; +using System.Threading.Tasks; +using Microsoft.Extensions.Hosting; +using Microsoft.Extensions.Logging; +using Microsoft.Extensions.Options; + +namespace UnityNuGet.Server +{ + /// + /// Update the RegistryCache at a regular interval + /// + internal sealed class RegistryCacheUpdater(RegistryCacheReport registryCacheReport, RegistryCacheSingleton currentRegistryCache, ILogger logger, IOptions registryOptionsAccessor) : BackgroundService + { + private readonly RegistryCacheReport _registryCacheReport = registryCacheReport; + private readonly RegistryCacheSingleton _currentRegistryCache = currentRegistryCache; + private readonly ILogger _logger = logger; + private readonly RegistryOptions _registryOptions = registryOptionsAccessor.Value; + + protected override async Task ExecuteAsync(CancellationToken stoppingToken) + { + try + { + while (!stoppingToken.IsCancellationRequested) + { + _logger.LogInformation("Starting to update RegistryCache"); + + _registryCacheReport.Start(); + + var newRegistryCache = new RegistryCache(_currentRegistryCache.UnityPackageFolder!, _currentRegistryCache.ServerUri!, _registryOptions.UnityScope!, _registryOptions.MinimumUnityVersion!, _registryOptions.PackageNameNuGetPostFix!, _registryOptions.TargetFrameworks!, _currentRegistryCache.NuGetRedirectLogger!) + { + // Update progress + OnProgress = (current, total) => + { + _currentRegistryCache.ProgressTotalPackageCount = total; + _currentRegistryCache.ProgressPackageIndex = current; + }, + OnInformation = message => _registryCacheReport.AddInformation(message), + OnWarning = message => _registryCacheReport.AddWarning(message), + OnError = message => _registryCacheReport.AddError(message) + }; + + await newRegistryCache.Build(); + + if (_registryCacheReport.ErrorMessages.Any()) + { + _logger.LogInformation("RegistryCache not updated due to errors. See previous logs"); + } + else + { + // Update the registry cache in the services + _currentRegistryCache.Instance = newRegistryCache; + + _logger.LogInformation("RegistryCache successfully updated"); + } + + _registryCacheReport.Complete(); + + await Task.Delay((int)_registryOptions.UpdateInterval.TotalMilliseconds, stoppingToken); + } + } + catch (Exception ex) + { + string message = "Error while building a new registry cache."; + + _logger.LogError(ex, "{Message}", message); + + _registryCacheReport.AddError($"{message}. Reason: {ex}"); + _registryCacheReport.Complete(); + } + } + } +} diff --git a/src/UnityNuGet.Server/UnityNuGet.Server.csproj b/src/UnityNuGet.Server/UnityNuGet.Server.csproj index 6a2933d7..56d92248 100644 --- a/src/UnityNuGet.Server/UnityNuGet.Server.csproj +++ b/src/UnityNuGet.Server/UnityNuGet.Server.csproj @@ -1,12 +1,10 @@ - + - net7.0 - enable + net8.0 /subscriptions/b6745039-70e7-4641-994b-5457cb220e2a/resourcegroups/Default-ApplicationInsights-EastUS/providers/microsoft.insights/components/unitynuget-registry /subscriptions/b6745039-70e7-4641-994b-5457cb220e2a/resourcegroups/Default-ApplicationInsights-EastUS/providers/microsoft.insights/components/unitynuget-registry 1be0a769-8d75-4a27-99e0-128afcc0ffee - true @@ -16,11 +14,11 @@ - - - - - + + + + + diff --git a/src/UnityNuGet.Tests/NativeTests.cs b/src/UnityNuGet.Tests/NativeTests.cs index 1dd28f4e..60419540 100644 --- a/src/UnityNuGet.Tests/NativeTests.cs +++ b/src/UnityNuGet.Tests/NativeTests.cs @@ -1,53 +1,53 @@ -using System; -using System.IO; -using System.Threading.Tasks; -using NUnit.Framework; - -namespace UnityNuGet.Tests -{ - [Ignore("Ignore native libs tests")] - public class NativeTests - { - [Test] - public async Task TestBuild() - { - var unityPackages = Path.Combine(Path.GetDirectoryName(typeof(RegistryCacheTests).Assembly.Location), "unity_packages"); - Directory.Delete(unityPackages, true); - - var errorsTriggered = false; - - var registryCache = new RegistryCache( - unityPackages, - new Uri("http://localhost/"), - "org.nuget", - "2019.1", - " (NuGet)", - new RegistryTargetFramework[] { - new RegistryTargetFramework { Name = "netstandard2.1", DefineConstraints = new string[] { "UNITY_2021_2_OR_NEWER"} }, - new RegistryTargetFramework { Name = "netstandard2.0", DefineConstraints = new string[] { "!UNITY_2021_2_OR_NEWER" } }, - }, - new NuGetConsoleLogger()) - { - Filter = "rhino3dm", - OnError = message => - { - errorsTriggered = true; - } - }; - - await registryCache.Build(); - - Assert.False(errorsTriggered, "The registry failed to build, check the logs"); - var allResult = registryCache.All(); - var allResultJson = allResult.ToJson(); - - StringAssert.Contains("org.nuget.rhino3dm", allResultJson); - - var rhinoPackage = registryCache.GetPackage("org.nuget.rhino3dm"); - Assert.NotNull(rhinoPackage); - var rhinopackageJson = rhinoPackage.ToJson(); - StringAssert.Contains("org.nuget.rhino3dm", rhinopackageJson); - StringAssert.Contains("7.11.0", rhinopackageJson); - } - } -} +using System; +using System.IO; +using System.Threading.Tasks; +using NUnit.Framework; + +namespace UnityNuGet.Tests +{ + [Ignore("Ignore native libs tests")] + public class NativeTests + { + [Test] + public async Task TestBuild() + { + var unityPackages = Path.Combine(Path.GetDirectoryName(typeof(RegistryCacheTests).Assembly.Location)!, "unity_packages"); + Directory.Delete(unityPackages, true); + + var errorsTriggered = false; + + var registryCache = new RegistryCache( + unityPackages, + new Uri("http://localhost/"), + "org.nuget", + "2019.1", + " (NuGet)", + [ + new() { Name = "netstandard2.1", DefineConstraints = ["UNITY_2021_2_OR_NEWER"] }, + new() { Name = "netstandard2.0", DefineConstraints = ["!UNITY_2021_2_OR_NEWER"] }, + ], + new NuGetConsoleLogger()) + { + Filter = "rhino3dm", + OnError = message => + { + errorsTriggered = true; + } + }; + + await registryCache.Build(); + + Assert.That(errorsTriggered, Is.False, "The registry failed to build, check the logs"); + var allResult = registryCache.All(); + var allResultJson = allResult.ToJson(); + + Assert.That(allResultJson, Does.Contain("org.nuget.rhino3dm")); + + var rhinoPackage = registryCache.GetPackage("org.nuget.rhino3dm"); + Assert.That(rhinoPackage, Is.Not.Null); + var rhinopackageJson = rhinoPackage!.ToJson(); + Assert.That(rhinopackageJson, Does.Contain("org.nuget.rhino3dm")); + Assert.That(rhinopackageJson, Does.Contain("7.11.0")); + } + } +} diff --git a/src/UnityNuGet.Tests/NuGetHelperTests.cs b/src/UnityNuGet.Tests/NuGetHelperTests.cs index 44329295..17c63a47 100644 --- a/src/UnityNuGet.Tests/NuGetHelperTests.cs +++ b/src/UnityNuGet.Tests/NuGetHelperTests.cs @@ -18,7 +18,7 @@ public class NuGetHelperTests [TestCase("analyzers/Test.resources.dll")] public void IsApplicableAnalyzerResource_Valid(string input) { - Assert.True(NuGetHelper.IsApplicableAnalyzerResource(input)); + Assert.That(NuGetHelper.IsApplicableAnalyzerResource(input), Is.True); } [Test] @@ -31,7 +31,7 @@ public void IsApplicableAnalyzerResource_Valid(string input) [TestCase("analyzers/Test.dll")] public void IsApplicableAnalyzerResource_Invalid(string input) { - Assert.False(NuGetHelper.IsApplicableAnalyzerResource(input)); + Assert.That(NuGetHelper.IsApplicableAnalyzerResource(input), Is.False); } // Examples: @@ -48,7 +48,7 @@ public void IsApplicableAnalyzerResource_Invalid(string input) [TestCase("analyzers/Test.dll")] public void IsApplicableUnitySupportedRoslynVersionFolder_Valid(string input) { - Assert.True(NuGetHelper.IsApplicableUnitySupportedRoslynVersionFolder(input)); + Assert.That(NuGetHelper.IsApplicableUnitySupportedRoslynVersionFolder(input), Is.True); } [Test] @@ -56,7 +56,7 @@ public void IsApplicableUnitySupportedRoslynVersionFolder_Valid(string input) [TestCase("analyzers/dotnet/roslyn4.0/Test.dll")] public void IsApplicableUnitySupportedRoslynVersionFolder_Invalid(string input) { - Assert.False(NuGetHelper.IsApplicableUnitySupportedRoslynVersionFolder(input)); + Assert.That(NuGetHelper.IsApplicableUnitySupportedRoslynVersionFolder(input), Is.False); } [Test] @@ -64,17 +64,17 @@ public void GetCompatiblePackageDependencyGroups_SpecificSingleFramework() { IList packageDependencyGroups = new PackageDependencyGroup[] { - new PackageDependencyGroup(CommonFrameworks.NetStandard13, Array.Empty()), - new PackageDependencyGroup(CommonFrameworks.NetStandard16, Array.Empty()), - new PackageDependencyGroup(CommonFrameworks.NetStandard20, Array.Empty()), - new PackageDependencyGroup(CommonFrameworks.NetStandard21, Array.Empty()) + new(CommonFrameworks.NetStandard13, Array.Empty()), + new(CommonFrameworks.NetStandard16, Array.Empty()), + new(CommonFrameworks.NetStandard20, Array.Empty()), + new(CommonFrameworks.NetStandard21, Array.Empty()) }; - IEnumerable targetFrameworks = new RegistryTargetFramework[] { new RegistryTargetFramework { Framework = CommonFrameworks.NetStandard20 } }; + IEnumerable targetFrameworks = new RegistryTargetFramework[] { new() { Framework = CommonFrameworks.NetStandard20 } }; IEnumerable compatibleDependencyGroups = NuGetHelper.GetCompatiblePackageDependencyGroups(packageDependencyGroups, targetFrameworks); - CollectionAssert.AreEqual(new PackageDependencyGroup[] { packageDependencyGroups[2] }, compatibleDependencyGroups); + Assert.That(compatibleDependencyGroups, Is.EqualTo(new PackageDependencyGroup[] { packageDependencyGroups[2] }).AsCollection); } [Test] @@ -82,17 +82,17 @@ public void GetCompatiblePackageDependencyGroups_SpecificMultipleFrameworks() { IList packageDependencyGroups = new PackageDependencyGroup[] { - new PackageDependencyGroup(CommonFrameworks.NetStandard13, Array.Empty()), - new PackageDependencyGroup(CommonFrameworks.NetStandard16, Array.Empty()), - new PackageDependencyGroup(CommonFrameworks.NetStandard20, Array.Empty()), - new PackageDependencyGroup(CommonFrameworks.NetStandard21, Array.Empty()) + new(CommonFrameworks.NetStandard13, Array.Empty()), + new(CommonFrameworks.NetStandard16, Array.Empty()), + new(CommonFrameworks.NetStandard20, Array.Empty()), + new(CommonFrameworks.NetStandard21, Array.Empty()) }; - IEnumerable targetFrameworks = new RegistryTargetFramework[] { new RegistryTargetFramework { Framework = CommonFrameworks.NetStandard20 }, new RegistryTargetFramework { Framework = CommonFrameworks.NetStandard21 } }; + IEnumerable targetFrameworks = new RegistryTargetFramework[] { new() { Framework = CommonFrameworks.NetStandard20 }, new() { Framework = CommonFrameworks.NetStandard21 } }; IEnumerable compatibleDependencyGroups = NuGetHelper.GetCompatiblePackageDependencyGroups(packageDependencyGroups, targetFrameworks); - CollectionAssert.AreEqual(new PackageDependencyGroup[] { packageDependencyGroups[2], packageDependencyGroups[3] }, compatibleDependencyGroups); + Assert.That(compatibleDependencyGroups, Is.EqualTo(new PackageDependencyGroup[] { packageDependencyGroups[2], packageDependencyGroups[3] }).AsCollection); } [Test] @@ -100,14 +100,14 @@ public void GetCompatiblePackageDependencyGroups_AnyFramework() { IList packageDependencyGroups = new PackageDependencyGroup[] { - new PackageDependencyGroup(new NuGetFramework(SpecialIdentifiers.Any), Array.Empty()) + new(new NuGetFramework(SpecialIdentifiers.Any), Array.Empty()) }; - IEnumerable targetFrameworks = new RegistryTargetFramework[] { new RegistryTargetFramework { Framework = CommonFrameworks.NetStandard20 } }; + IEnumerable targetFrameworks = new RegistryTargetFramework[] { new() { Framework = CommonFrameworks.NetStandard20 } }; IEnumerable compatibleDependencyGroups = NuGetHelper.GetCompatiblePackageDependencyGroups(packageDependencyGroups, targetFrameworks); - CollectionAssert.AreEqual(packageDependencyGroups, compatibleDependencyGroups); + Assert.That(compatibleDependencyGroups, Is.EqualTo(packageDependencyGroups).AsCollection); } } } diff --git a/src/UnityNuGet.Tests/PlatformDefinitionTests.cs b/src/UnityNuGet.Tests/PlatformDefinitionTests.cs index 6c4377b0..7ac0b412 100644 --- a/src/UnityNuGet.Tests/PlatformDefinitionTests.cs +++ b/src/UnityNuGet.Tests/PlatformDefinitionTests.cs @@ -1,132 +1,144 @@ -using System; -using System.Collections.Generic; -using System.IO; -using System.Linq; -using NUnit.Framework; - -namespace UnityNuGet.Tests -{ - public class PlatformDefinitionTests - { - [Test] - public void CanFindDefinitions() - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - - // Look-up by OS should return the most general configuration - var win = platformDefs.Find(UnityOs.Windows); - Assert.IsNotNull(win); - Assert.AreEqual(win.Cpu, UnityCpu.AnyCpu); - - // Look-up explicit configuration +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using NUnit.Framework; + +namespace UnityNuGet.Tests +{ + public class PlatformDefinitionTests + { + [Test] + public void CanFindDefinitions() + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + + // Look-up by OS should return the most general configuration + var win = platformDefs.Find(UnityOs.Windows); + Assert.That(win, Is.Not.Null); + Assert.That(win!.Cpu, Is.EqualTo(UnityCpu.AnyCpu)); + + // Look-up explicit configuration var win64 = platformDefs.Find(UnityOs.Windows, UnityCpu.X64); - Assert.IsNotNull(win64); - Assert.AreEqual(win64.Os, win.Os); - Assert.AreEqual(win64.Cpu, UnityCpu.X64); - Assert.True(win.Children.Contains(win64)); - - // Look-up invalid configuration - var and = platformDefs.Find(UnityOs.Android, UnityCpu.None); - Assert.IsNull(and); - } - - [Test] - public void RemainingPlatforms_NoneVisited() - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var visited = new HashSet(); - - // If no platform was visited, the remaining platforms should be the (AnyOS, AnyCPU) config. - var remaining = platformDefs.GetRemainingPlatforms(visited); - Assert.IsNotNull(remaining); - Assert.AreEqual(1, remaining.Count); - Assert.AreEqual(remaining.First(), platformDefs); - } - - [Test] - public void RemainingPlatforms_OneVisited() - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - - foreach (var child in platformDefs.Children) + Assert.Multiple(() => { - var visited = new HashSet() { child }; - var remaining = platformDefs.GetRemainingPlatforms(visited); - - // We should get all other children, except the one already visited - Assert.AreEqual(platformDefs.Children.Count, remaining.Count + 1); - foreach (var r in remaining) + Assert.That(win64, Is.Not.Null); + Assert.That(win.Os, Is.EqualTo(win64!.Os)); + }); + Assert.Multiple(() => + { + Assert.That(win64.Cpu, Is.EqualTo(UnityCpu.X64)); + Assert.That(win.Children, Does.Contain(win64)); + }); + + // Look-up invalid configuration + var and = platformDefs.Find(UnityOs.Android, UnityCpu.None); + Assert.That(and, Is.Null); + } + + [Test] + public void RemainingPlatforms_NoneVisited() + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var visited = new HashSet(); + + // If no platform was visited, the remaining platforms should be the (AnyOS, AnyCPU) config. + var remaining = platformDefs.GetRemainingPlatforms(visited); + Assert.That(remaining, Is.Not.Null); + Assert.Multiple(() => + { + Assert.That(remaining, Has.Count.EqualTo(1)); + Assert.That(platformDefs, Is.EqualTo(remaining.First())); + }); + } + + [Test] + public void RemainingPlatforms_OneVisited() + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + + foreach (var child in platformDefs.Children) + { + var visited = new HashSet() { child }; + var remaining = platformDefs.GetRemainingPlatforms(visited); + + // We should get all other children, except the one already visited + Assert.That(remaining.Count + 1, Is.EqualTo(platformDefs.Children.Count)); + foreach (var r in remaining) { - Assert.AreNotEqual(r, child); - Assert.IsTrue(platformDefs.Children.Contains(r)); - } - } - } - - [Test] - public void RemainingPlatforms_LeafVisited() - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var win64 = platformDefs.Find(UnityOs.Windows, UnityCpu.X64); - var visited = new HashSet() { win64 }; - - // The remaining platforms should be all non-windows, as well as all !x64 windows - var expected = platformDefs.Children - .Except(new[] { win64.Parent }) - .Concat( - win64.Parent.Children - .Except(new[] { win64 })) - .ToHashSet(); - var actual = platformDefs.GetRemainingPlatforms(visited); - Assert.IsTrue(expected.SetEquals(actual)); - } - - [TestCase("")] - [TestCase("base")] - public void TestConfigPath_Root(string basePath) - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var file = new PlatformFile("a/b/c.dll", platformDefs); - - // We don't use extra paths for the (AnyOS, AnyCPU) configuration - var actual = file.GetDestinationPath(basePath); - var expected = Path.Combine( - basePath, - Path.GetFileName(file.SourcePath)); - Assert.AreEqual(actual, expected); - } - - [TestCase("")] - [TestCase("base")] - public void TestConfigPath_OsOnly(string basePath) - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var win = platformDefs.Find(UnityOs.Windows); - var file = new PlatformFile("a/b/c.dll", win); - - var actual = file.GetDestinationPath(basePath); - var expected = Path.Combine( - basePath, - "Windows", - Path.GetFileName(file.SourcePath)); - Assert.AreEqual(actual, expected); - } - - [TestCase("")] - [TestCase("base")] - public void TestConfigPath_Full(string basePath) - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var win64 = platformDefs.Find(UnityOs.Windows, UnityCpu.X64); - var file = new PlatformFile("a/b/c.dll", win64); - - var actual = file.GetDestinationPath(basePath); - var expected = Path.Combine( - basePath, - "Windows", - "x86_64", - Path.GetFileName(file.SourcePath)); - Assert.AreEqual(actual, expected); - } - } -} + Assert.Multiple(() => + { + Assert.That(child, Is.Not.EqualTo(r)); + Assert.That(platformDefs.Children, Does.Contain(r)); + }); + } + } + } + + [Test] + public void RemainingPlatforms_LeafVisited() + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var win64 = platformDefs.Find(UnityOs.Windows, UnityCpu.X64); + var visited = new HashSet() { win64! }; + + // The remaining platforms should be all non-windows, as well as all !x64 windows + var expected = platformDefs.Children + .Except(new[] { win64!.Parent }) + .Concat( + win64.Parent!.Children + .Except(new[] { win64 })) + .ToHashSet(); + var actual = platformDefs.GetRemainingPlatforms(visited); + Assert.That(expected.SetEquals(actual), Is.True); + } + + [TestCase("")] + [TestCase("base")] + public void TestConfigPath_Root(string basePath) + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var file = new PlatformFile("a/b/c.dll", platformDefs); + + // We don't use extra paths for the (AnyOS, AnyCPU) configuration + var actual = file.GetDestinationPath(basePath); + var expected = Path.Combine( + basePath, + Path.GetFileName(file.SourcePath)); + Assert.That(expected, Is.EqualTo(actual)); + } + + [TestCase("")] + [TestCase("base")] + public void TestConfigPath_OsOnly(string basePath) + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var win = platformDefs.Find(UnityOs.Windows); + var file = new PlatformFile("a/b/c.dll", win!); + + var actual = file.GetDestinationPath(basePath); + var expected = Path.Combine( + basePath, + "Windows", + Path.GetFileName(file.SourcePath)); + Assert.That(expected, Is.EqualTo(actual)); + } + + [TestCase("")] + [TestCase("base")] + public void TestConfigPath_Full(string basePath) + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var win64 = platformDefs.Find(UnityOs.Windows, UnityCpu.X64); + var file = new PlatformFile("a/b/c.dll", win64!); + + var actual = file.GetDestinationPath(basePath); + var expected = Path.Combine( + basePath, + "Windows", + "x86_64", + Path.GetFileName(file.SourcePath)); + Assert.That(expected, Is.EqualTo(actual)); + } + } +} diff --git a/src/UnityNuGet.Tests/RegistryCacheTests.cs b/src/UnityNuGet.Tests/RegistryCacheTests.cs index 6e7372ba..e13c464f 100644 --- a/src/UnityNuGet.Tests/RegistryCacheTests.cs +++ b/src/UnityNuGet.Tests/RegistryCacheTests.cs @@ -12,17 +12,17 @@ public async Task TestBuild() { var errorsTriggered = false; - var unityPackages = Path.Combine(Path.GetDirectoryName(typeof(RegistryCacheTests).Assembly.Location), "unity_packages"); + var unityPackages = Path.Combine(Path.GetDirectoryName(typeof(RegistryCacheTests).Assembly.Location)!, "unity_packages"); var registryCache = new RegistryCache( unityPackages, new Uri("http://localhost/"), "org.nuget", "2019.1", " (NuGet)", - new RegistryTargetFramework[] { - new RegistryTargetFramework { Name = "netstandard2.1", DefineConstraints = new string[] { "UNITY_2021_2_OR_NEWER"} }, - new RegistryTargetFramework { Name = "netstandard2.0", DefineConstraints = new string[] { "!UNITY_2021_2_OR_NEWER" } }, - }, + [ + new() { Name = "netstandard2.1", DefineConstraints = ["UNITY_2021_2_OR_NEWER"] }, + new() { Name = "netstandard2.0", DefineConstraints = ["!UNITY_2021_2_OR_NEWER"] }, + ], new NuGetConsoleLogger()) { OnError = message => @@ -36,20 +36,20 @@ public async Task TestBuild() await registryCache.Build(); - Assert.False(errorsTriggered, "The registry failed to build, check the logs"); + Assert.That(errorsTriggered, Is.False, "The registry failed to build, check the logs"); var allResult = registryCache.All(); - Assert.True(allResult.Packages.Count >= 3); + Assert.That(allResult.Packages, Has.Count.GreaterThanOrEqualTo(3)); var allResultJson = allResult.ToJson(); - StringAssert.Contains("org.nuget.scriban", allResultJson); - StringAssert.Contains("org.nuget.system.runtime.compilerservices.unsafe", allResultJson); + Assert.That(allResultJson, Does.Contain("org.nuget.scriban")); + Assert.That(allResultJson, Does.Contain("org.nuget.system.runtime.compilerservices.unsafe")); var scribanPackage = registryCache.GetPackage("org.nuget.scriban"); - Assert.NotNull(scribanPackage); - var scribanPackageJson = scribanPackage.ToJson(); - StringAssert.Contains("org.nuget.scriban", scribanPackageJson); - StringAssert.Contains("2.1.0", scribanPackageJson); + Assert.That(scribanPackage, Is.Not.Null); + var scribanPackageJson = scribanPackage!.ToJson(); + Assert.That(scribanPackageJson, Does.Contain("org.nuget.scriban")); + Assert.That(scribanPackageJson, Does.Contain("2.1.0")); } } } diff --git a/src/UnityNuGet.Tests/RegistryTests.cs b/src/UnityNuGet.Tests/RegistryTests.cs index 21dee362..5d7681e2 100644 --- a/src/UnityNuGet.Tests/RegistryTests.cs +++ b/src/UnityNuGet.Tests/RegistryTests.cs @@ -1,4 +1,7 @@ -using System.Linq; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Text; using System.Text.RegularExpressions; using System.Threading; using System.Threading.Tasks; @@ -10,7 +13,6 @@ using NuGet.Protocol.Core.Types; using NUnit.Framework; using static NuGet.Frameworks.FrameworkConstants; -using System.IO; namespace UnityNuGet.Tests { @@ -23,7 +25,7 @@ public void Make_Sure_That_The_Order_In_The_Registry_Is_Respected() var originalPackageNames = registry.Select(r => r.Key).ToArray(); var sortedPackageNames = originalPackageNames.OrderBy(p => p).ToArray(); - Assert.AreEqual(sortedPackageNames, originalPackageNames); + Assert.That(originalPackageNames, Is.EqualTo(sortedPackageNames)); } [Test] @@ -32,7 +34,7 @@ public void Ensure_That_Packages_Already_Included_In_Net_Standard_Are_not_Includ var registry = Registry.GetInstance(); var packageNames = registry.Select(r => r.Key).Where(DotNetHelper.IsNetStandard20Assembly).ToArray(); - Assert.IsEmpty(packageNames); + Assert.That(packageNames, Is.Empty); } [Test] @@ -57,14 +59,14 @@ public async Task CanParse_PackageWithRuntimes() var runtimeLibs = await RuntimeLibraries .GetSupportedRuntimeLibsAsync(downloadResult.PackageReader, CommonFrameworks.NetStandard20, logger) .ToListAsync(); - Assert.IsTrue(runtimeLibs.Any()); + Assert.That(runtimeLibs, Is.Not.Empty); // Make sure these runtime libraries are only for Windows var platformDefs = PlatformDefinition.CreateAllPlatforms(); var win = platformDefs.Find(UnityOs.Windows); foreach (var (file, os, cpu) in runtimeLibs) { - Assert.AreEqual(platformDefs.Find(os, cpu), win); + Assert.That(platformDefs.Find(os, cpu), Is.EqualTo(win)); } // Get the lib files @@ -73,7 +75,7 @@ public async Task CanParse_PackageWithRuntimes() versions, new RegistryTargetFramework[] { - new RegistryTargetFramework + new() { Framework = CommonFrameworks.NetStandard20, }, @@ -88,7 +90,7 @@ public async Task CanParse_PackageWithRuntimes() var runtimeFiles = runtimeLibs .Select(l => Path.GetFileName(l.file)) .ToHashSet(); - Assert.IsTrue(libFiles.SetEquals(runtimeFiles)); + Assert.That(libFiles.SetEquals(runtimeFiles), Is.True); } [Test] @@ -103,7 +105,7 @@ public async Task Ensure_Min_Version_Is_Correct_Ignoring_Analyzers_And_Native_Li var repository = Repository.Factory.GetCoreV3("https://api.nuget.org/v3/index.json"); var resource = await repository.GetResourceAsync(); - var nuGetFrameworks = new RegistryTargetFramework[] { new RegistryTargetFramework { Framework = CommonFrameworks.NetStandard20 } }; + var nuGetFrameworks = new RegistryTargetFramework[] { new() { Framework = CommonFrameworks.NetStandard20 } }; var excludedPackages = new string[] { // All versions target "Any" and not .netstandard2.0 / 2.1 @@ -151,7 +153,7 @@ public async Task Ensure_Min_Version_Is_Correct_Ignoring_Analyzers_And_Native_Li if (packageIdentity != null) { - Assert.AreEqual(packageIdentity.Version, versionRange.MinVersion, $"Package {packageId}"); + Assert.That(versionRange!.MinVersion, Is.EqualTo(packageIdentity.Version), $"Package {packageId}"); } else { @@ -159,7 +161,7 @@ public async Task Ensure_Min_Version_Is_Correct_Ignoring_Analyzers_And_Native_Li var downloadResult = await PackageDownloader.GetDownloadResourceResultAsync( new SourceRepository[] { repository }, - new PackageIdentity(registryKvp.Key, registryKvp.Value.Version.MinVersion), + new PackageIdentity(registryKvp.Key, registryKvp.Value.Version!.MinVersion), new PackageDownloadContext(cache), SettingsUtility.GetGlobalPackagesFolder(settings), logger, cancellationToken); @@ -177,5 +179,62 @@ public async Task Ensure_Min_Version_Is_Correct_Ignoring_Analyzers_And_Native_Li } } } + + [Test] + public async Task Ensure_Do_Not_Exceed_The_Maximum_Number_Of_Allowed_Versions() + { + const int maxAllowedVersions = 100; + + var registry = Registry.GetInstance(); + + var logger = NullLogger.Instance; + var cancellationToken = CancellationToken.None; + + var cache = new SourceCacheContext(); + var repository = Repository.Factory.GetCoreV3("https://api.nuget.org/v3/index.json"); + var resource = await repository.GetResourceAsync(); + + List<(string packageId, int versionCount)> packages = []; + + foreach (var registryKvp in registry.Where(r => !r.Value.Analyzer && !r.Value.Ignored)) + { + var packageId = registryKvp.Key; + + var versionRange = registryKvp.Value.Version; + + var dependencyPackageMetas = await resource.GetMetadataAsync( + packageId, + includePrerelease: false, + includeUnlisted: false, + cache, + logger, + cancellationToken); + + var versions = dependencyPackageMetas.Where(v => versionRange!.Satisfies(v.Identity.Version)).ToArray(); + + if (versions.Length > maxAllowedVersions) + { + packages.Add((registryKvp.Key, versions.Length)); + } + } + + StringBuilder stringBuilder = new(); + + foreach (var (packageId, versionCount) in packages.OrderByDescending(p => p.versionCount)) + { + stringBuilder.AppendLine($"{packageId} -> {versionCount}"); + } + + if (stringBuilder.Length == 0) + { + const bool trueConstant = true; + + Assert.That(trueConstant, Is.True); + } + else + { + Assert.Inconclusive(stringBuilder.ToString()); + } + } } } diff --git a/src/UnityNuGet.Tests/UnityMetaTests.cs b/src/UnityNuGet.Tests/UnityMetaTests.cs index 6a1299c1..4ecc8350 100644 --- a/src/UnityNuGet.Tests/UnityMetaTests.cs +++ b/src/UnityNuGet.Tests/UnityMetaTests.cs @@ -1,162 +1,173 @@ -using System; -using System.Linq; -using System.Text.RegularExpressions; -using NUnit.Framework; - -namespace UnityNuGet.Tests +using System; +using System.Linq; +using System.Text.RegularExpressions; +using NUnit.Framework; + +namespace UnityNuGet.Tests { - public class UnityMetaTests - { - [Test] - public void GetMetaForDll_FormatsDefineConstraintsProperly_WithoutConstraints() - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var anyOs = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); - var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), anyOs, Array.Empty(), Array.Empty()); - StringAssert.DoesNotContain("defineConstraints", output); - - // This is on the same line in the template, so ensure it's intact - StringAssert.Contains("\n isPreloaded: 0\n", output); - } - - [Test] - public void GetMetaForDll_FormatsLabelsProperly_WithoutLabels() - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var anyOs = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); - var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), anyOs, Array.Empty(), Array.Empty()); - StringAssert.DoesNotContain("labels", output); - - // This is on the same line in the template, so ensure it's intact - StringAssert.Contains("\nPluginImporter:\n", output); +#pragma warning disable CA1861 // Avoid constant arrays as arguments + public class UnityMetaTests + { + [Test] + public void GetMetaForDll_FormatsDefineConstraintsProperly_WithoutConstraints() + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var anyOs = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); + var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), anyOs!, Array.Empty(), Array.Empty()); + Assert.That(output, Does.Not.Contain("defineConstraints")); + + // This is on the same line in the template, so ensure it's intact + Assert.That(output, Does.Contain("\n isPreloaded: 0\n")); + } + + [Test] + public void GetMetaForDll_FormatsLabelsProperly_WithoutLabels() + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var anyOs = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); + var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), anyOs!, Array.Empty(), Array.Empty()); + Assert.That(output, Does.Not.Contain("labels")); + + // This is on the same line in the template, so ensure it's intact + Assert.That(output, Does.Contain("\nPluginImporter:\n")); } [TestCase(new[] { "FIRST" }, "\n defineConstraints:\n - FIRST\n")] - [TestCase(new[] { "FIRST", "SECOND" }, "\n defineConstraints:\n - FIRST\n - SECOND\n")] - public void GetMetaForDll_FormatsDefineConstraintsProperly_WithConstraints( - string[] constraints, string expected) - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var anyOs = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); - var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), anyOs, Array.Empty(), constraints); - - StringAssert.Contains(expected, output); - - // This is on the same line in the template, so ensure it's intact - StringAssert.Contains("\n isPreloaded: 0\n", output); - } - - [TestCase(new[] { "FIRST" }, "\nlabels:\n - FIRST\n")] - [TestCase(new[] { "FIRST", "SECOND" }, "\nlabels:\n - FIRST\n - SECOND\n")] - public void GetMetaForDll_FormatsLabelsProperly_WithLabels( - string[] labels, string expected) - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var anyOs = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); - var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), anyOs, labels, Array.Empty()); - - StringAssert.Contains(expected, output); - - // This is on the same line in the template, so ensure it's intact - StringAssert.Contains("\nPluginImporter:\n", output); - } - - [TestCase(true, "1")] - [TestCase(false, "0")] - public void GetMetaForDll_FormatsAnyPlatformEnabledProperly(bool value, string expected) - { - PlatformDefinition platformDef; - - if (value) + [TestCase(new[] { "FIRST", "SECOND" }, "\n defineConstraints:\n - FIRST\n - SECOND\n")] + public void GetMetaForDll_FormatsDefineConstraintsProperly_WithConstraints( + string[] constraints, string expected) + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var anyOs = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); + var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), anyOs!, Array.Empty(), constraints); + + Assert.That(output, Does.Contain(expected)); + + // This is on the same line in the template, so ensure it's intact + Assert.That(output, Does.Contain("\n isPreloaded: 0\n")); + } + + [TestCase(new[] { "FIRST" }, "\nlabels:\n - FIRST\n")] + [TestCase(new[] { "FIRST", "SECOND" }, "\nlabels:\n - FIRST\n - SECOND\n")] + public void GetMetaForDll_FormatsLabelsProperly_WithLabels( + string[] labels, string expected) + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var anyOs = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); + var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), anyOs!, labels, Array.Empty()); + + Assert.That(output, Does.Contain(expected)); + + // This is on the same line in the template, so ensure it's intact + Assert.That(output, Does.Contain("\nPluginImporter:\n")); + } + + [TestCase(true, "1")] + [TestCase(false, "0")] + public void GetMetaForDll_FormatsAnyPlatformEnabledProperly(bool value, string expected) + { + PlatformDefinition? platformDef; + + if (value) + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + platformDef = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); + } + else + { + platformDef = new PlatformDefinition(UnityOs.AnyOs, UnityCpu.None, isEditorConfig: false); + } + + var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), platformDef!, Array.Empty(), Array.Empty()); + + Assert.That(output, Does.Contain($"\n platformData:\n - first:\n Any:\n second:\n enabled: {expected}\n")); + } + + [Test] + public void GetMetaForDll_ContainsNoWindowsNewlines() + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var anyOs = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); + var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), anyOs!, Array.Empty(), new[] { "TEST" }); + Assert.That(output, Does.Not.Contain("\r")); + } + + [TestCase(UnityOs.Android, "Android", "Android")] + [TestCase(UnityOs.WebGL, "WebGL", "WebGL")] + [TestCase(UnityOs.iOS, "iPhone", "iOS")] + public void GetMetaForDll_NonEditor(UnityOs os, string platformName, string osName) + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var output = UnityMeta.GetMetaForDll( + Guid.NewGuid(), + platformDefs.Find(os)!, + Array.Empty(), + Array.Empty()); + + // There should be a single 'Exclude Android: 0' match + var excludeRegex = new Regex("Exclude (.*): 0"); + var excludeMatches = excludeRegex.Matches(output); + Assert.That(excludeMatches, Is.Not.Null); + Assert.That(excludeMatches, Has.Count.EqualTo(1)); + Assert.Multiple(() => { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - platformDef = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); - } - else + Assert.That(excludeMatches.Single().Groups, Has.Count.EqualTo(2)); + Assert.That(osName, Is.EqualTo(excludeMatches.Single().Groups[1].Value)); + }); + + // There should be a single 'enabled: 1' match + var enableRegex = new Regex("enabled: 1"); + var enableMatches = enableRegex.Matches(output); + Assert.That(enableMatches, Is.Not.Null); + Assert.Multiple(() => { - platformDef = new PlatformDefinition(UnityOs.AnyOs, UnityCpu.None, isEditorConfig: false); - } - - var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), platformDef, Array.Empty(), Array.Empty()); - - StringAssert.Contains($"\n platformData:\n - first:\n Any:\n second:\n enabled: {expected}\n", output); - } - - [Test] - public void GetMetaForDll_ContainsNoWindowsNewlines() - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var anyOs = platformDefs.Find(UnityOs.AnyOs, UnityCpu.AnyCpu); - var output = UnityMeta.GetMetaForDll(Guid.NewGuid(), anyOs, Array.Empty(), new[] { "TEST" }); - StringAssert.DoesNotContain("\r", output); - } - - [TestCase(UnityOs.Android, "Android", "Android")] - [TestCase(UnityOs.WebGL, "WebGL", "WebGL")] - [TestCase(UnityOs.iOS, "iPhone", "iOS")] - public void GetMetaForDll_NonEditor(UnityOs os, string platformName, string osName) - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var output = UnityMeta.GetMetaForDll( - Guid.NewGuid(), - platformDefs.Find(os), - Array.Empty(), - Array.Empty()); - - // There should be a single 'Exclude Android: 0' match - var excludeRegex = new Regex("Exclude (.*): 0"); - var excludeMatches = excludeRegex.Matches(output); - Assert.IsNotNull(excludeMatches); - Assert.AreEqual(excludeMatches.Count, 1); - Assert.AreEqual(excludeMatches.Single().Groups.Count, 2); - Assert.AreEqual(excludeMatches.Single().Groups[1].Value, osName); - - // There should be a single 'enabled: 1' match - var enableRegex = new Regex("enabled: 1"); + Assert.That(enableMatches, Has.Count.EqualTo(1)); + + Assert.That(output, Does.Contain($"- first:\n {platformName}: {osName}\n second:\n enabled: 1\n")); + }); + } + + [TestCase(UnityOs.Windows, new[] { "Win", "Win64" })] + [TestCase(UnityOs.Linux, new[] { "Linux64" })] + [TestCase(UnityOs.OSX, new[] { "OSXUniversal" })] + public void GetMetaForDll_Editor(UnityOs os, string[] osNames) + { + var platformDefs = PlatformDefinition.CreateAllPlatforms(); + var pDef = platformDefs.Find(os); + var output = UnityMeta.GetMetaForDll( + Guid.NewGuid(), + pDef!, + Array.Empty(), + Array.Empty()); + + // There should be only 'Exclude Editor: 0' and 'Exclude {{ osName }}: 0' matches + var excludeRegex = new Regex("Exclude (.*): 0"); + var excludeMatches = excludeRegex.Matches(output); + Assert.That(excludeMatches, Is.Not.Null); + var actualExcludes = excludeMatches + .Select(match => match.Groups[1].Value) + .ToHashSet(); + + var expectedExcludes = osNames + .Append("Editor") + .ToHashSet(); + Assert.That(actualExcludes.SetEquals(expectedExcludes), Is.True); + + // There should be as many 'enabled: 1' matches as exclude matches + var enableRegex = new Regex("enabled: 1"); var enableMatches = enableRegex.Matches(output); - Assert.IsNotNull(enableMatches); - Assert.AreEqual(enableMatches.Count, 1); - - StringAssert.Contains($"- first:\n {platformName}: {osName}\n second:\n enabled: 1\n", output); - } - - [TestCase(UnityOs.Windows, new[] { "Win", "Win64" })] - [TestCase(UnityOs.Linux, new[] { "Linux64" })] - [TestCase(UnityOs.OSX, new[] { "OSXUniversal" })] - public void GetMetaForDll_Editor(UnityOs os, string[] osNames) - { - var platformDefs = PlatformDefinition.CreateAllPlatforms(); - var pDef = platformDefs.Find(os); - var output = UnityMeta.GetMetaForDll( - Guid.NewGuid(), - pDef, - Array.Empty(), - Array.Empty()); - - // There should be only 'Exclude Editor: 0' and 'Exclude {{ osName }}: 0' matches - var excludeRegex = new Regex("Exclude (.*): 0"); - var excludeMatches = excludeRegex.Matches(output); - Assert.IsNotNull(excludeMatches); - var actualExcludes = excludeMatches - .Select(match => match.Groups[1].Value) - .ToHashSet(); - - var expectedExcludes = osNames - .Append("Editor") - .ToHashSet(); - Assert.IsTrue(actualExcludes.SetEquals(expectedExcludes)); - - // There should be as many 'enabled: 1' matches as exclude matches - var enableRegex = new Regex("enabled: 1"); - var enableMatches = enableRegex.Matches(output); - Assert.IsNotNull(enableMatches); - Assert.AreEqual(enableMatches.Count, excludeMatches.Count); - - foreach (var osName in actualExcludes) + Assert.Multiple(() => { - var platformName = (osName == "Editor") ? osName : "Standalone"; - StringAssert.Contains($"- first:\n {platformName}: {osName}\n second:\n enabled: 1\n", output); - } - } + Assert.That(enableMatches, Is.Not.Null); + Assert.That(excludeMatches, Has.Count.EqualTo(enableMatches.Count)); + }); + + foreach (var osName in actualExcludes) + { + var platformName = (osName == "Editor") ? osName : "Standalone"; + Assert.That(output, Does.Contain($"- first:\n {platformName}: {osName}\n second:\n enabled: 1\n")); + } + } } -} +#pragma warning restore CA1861 // Avoid constant arrays as arguments +} diff --git a/src/UnityNuGet.Tests/UnityNuGet.Tests.csproj b/src/UnityNuGet.Tests/UnityNuGet.Tests.csproj index f278a3b9..9b67fe0d 100644 --- a/src/UnityNuGet.Tests/UnityNuGet.Tests.csproj +++ b/src/UnityNuGet.Tests/UnityNuGet.Tests.csproj @@ -1,16 +1,19 @@ - + - net7.0 + net8.0 false false - true - - - + + + + all + runtime; build; native; contentfiles; analyzers; buildtransitive + + diff --git a/src/UnityNuGet/DotNetHelper.cs b/src/UnityNuGet/DotNetHelper.cs index fb8be2ed..baf4df53 100644 --- a/src/UnityNuGet/DotNetHelper.cs +++ b/src/UnityNuGet/DotNetHelper.cs @@ -1,238 +1,236 @@ -using System.Collections.Generic; - -namespace UnityNuGet -{ - /// - /// Helper class to identify which NuGet packages are replaced by .NET Standard 2.0 / 2.1 - /// - static class DotNetHelper - { - public static bool IsNetStandard20Assembly(string packageId) => NetStandard20Assemblies.Contains(packageId); - - public static bool IsNetStandard21Assembly(string packageId) => NetStandard21Assemblies.Contains(packageId); - - // Retrieved from NuGet package (/build/netstandard2.0/ref): https://www.nuget.org/packages/NETStandard.Library/2.0.3 - private static readonly HashSet NetStandard20Assemblies = new() - { - "Microsoft.Win32.Primitives", - "System.AppContext", - "System.Collections.Concurrent", - "System.Collections", - "System.Collections.NonGeneric", - "System.Collections.Specialized", - "System.ComponentModel.Composition", - "System.ComponentModel", - "System.ComponentModel.EventBasedAsync", - "System.ComponentModel.Primitives", - "System.ComponentModel.TypeConverter", - "System.Console", - "System.Core", - "System.Data.Common", - "System.Data", - "System.Diagnostics.Contracts", - "System.Diagnostics.Debug", - "System.Diagnostics.FileVersionInfo", - "System.Diagnostics.Process", - "System.Diagnostics.StackTrace", - "System.Diagnostics.TextWriterTraceListener", - "System.Diagnostics.Tools", - "System.Diagnostics.TraceSource", - "System.Diagnostics.Tracing", - "System", - "System.Drawing", - "System.Drawing.Primitives", - "System.Dynamic.Runtime", - "System.Globalization.Calendars", - "System.Globalization", - "System.Globalization.Extensions", - "System.IO.Compression", - "System.IO.Compression.FileSystem", - "System.IO.Compression.ZipFile", - "System.IO", - "System.IO.FileSystem", - "System.IO.FileSystem.DriveInfo", - "System.IO.FileSystem.Primitives", - "System.IO.FileSystem.Watcher", - "System.IO.IsolatedStorage", - "System.IO.MemoryMappedFiles", - "System.IO.Pipes", - "System.IO.UnmanagedMemoryStream", - "System.Linq", - "System.Linq.Expressions", - "System.Linq.Parallel", - "System.Linq.Queryable", - "System.Net", - "System.Net.Http", - "System.Net.NameResolution", - "System.Net.NetworkInformation", - "System.Net.Ping", - "System.Net.Primitives", - "System.Net.Requests", - "System.Net.Security", - "System.Net.Sockets", - "System.Net.WebHeaderCollection", - "System.Net.WebSockets.Client", - "System.Net.WebSockets", - "System.Numerics", - "System.ObjectModel", - "System.Reflection", - "System.Reflection.Extensions", - "System.Reflection.Primitives", - "System.Resources.Reader", - "System.Resources.ResourceManager", - "System.Resources.Writer", - "System.Runtime.CompilerServices.VisualC", - "System.Runtime", - "System.Runtime.Extensions", - "System.Runtime.Handles", - "System.Runtime.InteropServices", - "System.Runtime.InteropServices.RuntimeInformation", - "System.Runtime.Numerics", - "System.Runtime.Serialization", - "System.Runtime.Serialization.Formatters", - "System.Runtime.Serialization.Json", - "System.Runtime.Serialization.Primitives", - "System.Runtime.Serialization.Xml", - "System.Security.Claims", - "System.Security.Cryptography.Algorithms", - "System.Security.Cryptography.Csp", - "System.Security.Cryptography.Encoding", - "System.Security.Cryptography.Primitives", - "System.Security.Cryptography.X509Certificates", - "System.Security.Principal", - "System.Security.SecureString", - "System.ServiceModel.Web", - "System.Text.Encoding", - "System.Text.Encoding.Extensions", - "System.Text.RegularExpressions", - "System.Threading", - "System.Threading.Overlapped", - "System.Threading.Tasks", - "System.Threading.Tasks.Parallel", - "System.Threading.Thread", - "System.Threading.ThreadPool", - "System.Threading.Timer", - "System.Transactions", - "System.ValueTuple", - "System.Web", - "System.Windows", - "System.Xml", - "System.Xml.Linq", - "System.Xml.ReaderWriter", - "System.Xml.Serialization", - "System.Xml.XDocument", - "System.Xml.XmlDocument", - "System.Xml.XmlSerializer", - "System.Xml.XPath", - "System.Xml.XPath.XDocument" - }; - - // Retrieved from NuGet package (/ref/netstandard2.1): https://www.nuget.org/packages/NETStandard.Library.Ref - private static readonly HashSet NetStandard21Assemblies = new() - { - "Microsoft.Win32.Primitives", - "System.AppContext", - "System.Buffers", - "System.Collections.Concurrent", - "System.Collections", - "System.Collections.NonGeneric", - "System.Collections.Specialized", - "System.ComponentModel", - "System.ComponentModel.EventBasedAsync", - "System.ComponentModel.Primitives", - "System.ComponentModel.TypeConverter", - "System.Console", - "System.Data.Common", - "System.Diagnostics.Contracts", - "System.Diagnostics.Debug", - "System.Diagnostics.FileVersionInfo", - "System.Diagnostics.Process", - "System.Diagnostics.StackTrace", - "System.Diagnostics.TextWriterTraceListener", - "System.Diagnostics.Tools", - "System.Diagnostics.TraceSource", - "System.Diagnostics.Tracing", - "System.Drawing.Primitives", - "System.Dynamic.Runtime", - "System.Globalization.Calendars", - "System.Globalization", - "System.Globalization.Extensions", - "System.IO.Compression", - "System.IO.Compression.ZipFile", - "System.IO", - "System.IO.FileSystem", - "System.IO.FileSystem.DriveInfo", - "System.IO.FileSystem.Primitives", - "System.IO.FileSystem.Watcher", - "System.IO.IsolatedStorage", - "System.IO.MemoryMappedFiles", - "System.IO.Pipes", - "System.IO.UnmanagedMemoryStream", - "System.Linq", - "System.Linq.Expressions", - "System.Linq.Parallel", - "System.Linq.Queryable", - "System.Memory", - "System.Net.Http", - "System.Net.NameResolution", - "System.Net.NetworkInformation", - "System.Net.Ping", - "System.Net.Primitives", - "System.Net.Requests", - "System.Net.Security", - "System.Net.Sockets", - "System.Net.WebHeaderCollection", - "System.Net.WebSockets.Client", - "System.Net.WebSockets", - "System.Numerics.Vectors", - "System.ObjectModel", - "System.Reflection.DispatchProxy", - "System.Reflection", - "System.Reflection.Emit", - "System.Reflection.Emit.ILGeneration", - "System.Reflection.Emit.Lightweight", - "System.Reflection.Extensions", - "System.Reflection.Primitives", - "System.Resources.Reader", - "System.Resources.ResourceManager", - "System.Resources.Writer", - "System.Runtime.CompilerServices.VisualC", - "System.Runtime", - "System.Runtime.Extensions", - "System.Runtime.Handles", - "System.Runtime.InteropServices", - "System.Runtime.InteropServices.RuntimeInformation", - "System.Runtime.Numerics", - "System.Runtime.Serialization.Formatters", - "System.Runtime.Serialization.Json", - "System.Runtime.Serialization.Primitives", - "System.Runtime.Serialization.Xml", - "System.Security.Claims", - "System.Security.Cryptography.Algorithms", - "System.Security.Cryptography.Csp", - "System.Security.Cryptography.Encoding", - "System.Security.Cryptography.Primitives", - "System.Security.Cryptography.X509Certificates", - "System.Security.Principal", - "System.Security.SecureString", - "System.Text.Encoding", - "System.Text.Encoding.Extensions", - "System.Text.RegularExpressions", - "System.Threading", - "System.Threading.Overlapped", - "System.Threading.Tasks", - "System.Threading.Tasks.Extensions", - "System.Threading.Tasks.Parallel", - "System.Threading.Thread", - "System.Threading.ThreadPool", - "System.Threading.Timer", - "System.ValueTuple", - "System.Xml.ReaderWriter", - "System.Xml.XDocument", - "System.Xml.XmlDocument", - "System.Xml.XmlSerializer", - "System.Xml.XPath", - "System.Xml.XPath.XDocument", - }; - } -} +using System.Collections.Generic; + +namespace UnityNuGet +{ + /// + /// Helper class to identify which NuGet packages are replaced by .NET Standard 2.0 / 2.1 + /// + static class DotNetHelper + { + public static bool IsNetStandard20Assembly(string packageId) => NetStandard20Assemblies.Contains(packageId); + + public static bool IsNetStandard21Assembly(string packageId) => NetStandard21Assemblies.Contains(packageId); + + // Retrieved from NuGet package (/build/netstandard2.0/ref): https://www.nuget.org/packages/NETStandard.Library/2.0.3 + private static readonly HashSet NetStandard20Assemblies = [ + "Microsoft.Win32.Primitives", + "System.AppContext", + "System.Collections.Concurrent", + "System.Collections", + "System.Collections.NonGeneric", + "System.Collections.Specialized", + "System.ComponentModel.Composition", + "System.ComponentModel", + "System.ComponentModel.EventBasedAsync", + "System.ComponentModel.Primitives", + "System.ComponentModel.TypeConverter", + "System.Console", + "System.Core", + "System.Data.Common", + "System.Data", + "System.Diagnostics.Contracts", + "System.Diagnostics.Debug", + "System.Diagnostics.FileVersionInfo", + "System.Diagnostics.Process", + "System.Diagnostics.StackTrace", + "System.Diagnostics.TextWriterTraceListener", + "System.Diagnostics.Tools", + "System.Diagnostics.TraceSource", + "System.Diagnostics.Tracing", + "System", + "System.Drawing", + "System.Drawing.Primitives", + "System.Dynamic.Runtime", + "System.Globalization.Calendars", + "System.Globalization", + "System.Globalization.Extensions", + "System.IO.Compression", + "System.IO.Compression.FileSystem", + "System.IO.Compression.ZipFile", + "System.IO", + "System.IO.FileSystem", + "System.IO.FileSystem.DriveInfo", + "System.IO.FileSystem.Primitives", + "System.IO.FileSystem.Watcher", + "System.IO.IsolatedStorage", + "System.IO.MemoryMappedFiles", + "System.IO.Pipes", + "System.IO.UnmanagedMemoryStream", + "System.Linq", + "System.Linq.Expressions", + "System.Linq.Parallel", + "System.Linq.Queryable", + "System.Net", + "System.Net.Http", + "System.Net.NameResolution", + "System.Net.NetworkInformation", + "System.Net.Ping", + "System.Net.Primitives", + "System.Net.Requests", + "System.Net.Security", + "System.Net.Sockets", + "System.Net.WebHeaderCollection", + "System.Net.WebSockets.Client", + "System.Net.WebSockets", + "System.Numerics", + "System.ObjectModel", + "System.Reflection", + "System.Reflection.Extensions", + "System.Reflection.Primitives", + "System.Resources.Reader", + "System.Resources.ResourceManager", + "System.Resources.Writer", + "System.Runtime.CompilerServices.VisualC", + "System.Runtime", + "System.Runtime.Extensions", + "System.Runtime.Handles", + "System.Runtime.InteropServices", + "System.Runtime.InteropServices.RuntimeInformation", + "System.Runtime.Numerics", + "System.Runtime.Serialization", + "System.Runtime.Serialization.Formatters", + "System.Runtime.Serialization.Json", + "System.Runtime.Serialization.Primitives", + "System.Runtime.Serialization.Xml", + "System.Security.Claims", + "System.Security.Cryptography.Algorithms", + "System.Security.Cryptography.Csp", + "System.Security.Cryptography.Encoding", + "System.Security.Cryptography.Primitives", + "System.Security.Cryptography.X509Certificates", + "System.Security.Principal", + "System.Security.SecureString", + "System.ServiceModel.Web", + "System.Text.Encoding", + "System.Text.Encoding.Extensions", + "System.Text.RegularExpressions", + "System.Threading", + "System.Threading.Overlapped", + "System.Threading.Tasks", + "System.Threading.Tasks.Parallel", + "System.Threading.Thread", + "System.Threading.ThreadPool", + "System.Threading.Timer", + "System.Transactions", + "System.ValueTuple", + "System.Web", + "System.Windows", + "System.Xml", + "System.Xml.Linq", + "System.Xml.ReaderWriter", + "System.Xml.Serialization", + "System.Xml.XDocument", + "System.Xml.XmlDocument", + "System.Xml.XmlSerializer", + "System.Xml.XPath", + "System.Xml.XPath.XDocument" + ]; + + // Retrieved from NuGet package (/ref/netstandard2.1): https://www.nuget.org/packages/NETStandard.Library.Ref + private static readonly HashSet NetStandard21Assemblies = [ + "Microsoft.Win32.Primitives", + "System.AppContext", + "System.Buffers", + "System.Collections.Concurrent", + "System.Collections", + "System.Collections.NonGeneric", + "System.Collections.Specialized", + "System.ComponentModel", + "System.ComponentModel.EventBasedAsync", + "System.ComponentModel.Primitives", + "System.ComponentModel.TypeConverter", + "System.Console", + "System.Data.Common", + "System.Diagnostics.Contracts", + "System.Diagnostics.Debug", + "System.Diagnostics.FileVersionInfo", + "System.Diagnostics.Process", + "System.Diagnostics.StackTrace", + "System.Diagnostics.TextWriterTraceListener", + "System.Diagnostics.Tools", + "System.Diagnostics.TraceSource", + "System.Diagnostics.Tracing", + "System.Drawing.Primitives", + "System.Dynamic.Runtime", + "System.Globalization.Calendars", + "System.Globalization", + "System.Globalization.Extensions", + "System.IO.Compression", + "System.IO.Compression.ZipFile", + "System.IO", + "System.IO.FileSystem", + "System.IO.FileSystem.DriveInfo", + "System.IO.FileSystem.Primitives", + "System.IO.FileSystem.Watcher", + "System.IO.IsolatedStorage", + "System.IO.MemoryMappedFiles", + "System.IO.Pipes", + "System.IO.UnmanagedMemoryStream", + "System.Linq", + "System.Linq.Expressions", + "System.Linq.Parallel", + "System.Linq.Queryable", + "System.Memory", + "System.Net.Http", + "System.Net.NameResolution", + "System.Net.NetworkInformation", + "System.Net.Ping", + "System.Net.Primitives", + "System.Net.Requests", + "System.Net.Security", + "System.Net.Sockets", + "System.Net.WebHeaderCollection", + "System.Net.WebSockets.Client", + "System.Net.WebSockets", + "System.Numerics.Vectors", + "System.ObjectModel", + "System.Reflection.DispatchProxy", + "System.Reflection", + "System.Reflection.Emit", + "System.Reflection.Emit.ILGeneration", + "System.Reflection.Emit.Lightweight", + "System.Reflection.Extensions", + "System.Reflection.Primitives", + "System.Resources.Reader", + "System.Resources.ResourceManager", + "System.Resources.Writer", + "System.Runtime.CompilerServices.VisualC", + "System.Runtime", + "System.Runtime.Extensions", + "System.Runtime.Handles", + "System.Runtime.InteropServices", + "System.Runtime.InteropServices.RuntimeInformation", + "System.Runtime.Numerics", + "System.Runtime.Serialization.Formatters", + "System.Runtime.Serialization.Json", + "System.Runtime.Serialization.Primitives", + "System.Runtime.Serialization.Xml", + "System.Security.Claims", + "System.Security.Cryptography.Algorithms", + "System.Security.Cryptography.Csp", + "System.Security.Cryptography.Encoding", + "System.Security.Cryptography.Primitives", + "System.Security.Cryptography.X509Certificates", + "System.Security.Principal", + "System.Security.SecureString", + "System.Text.Encoding", + "System.Text.Encoding.Extensions", + "System.Text.RegularExpressions", + "System.Threading", + "System.Threading.Overlapped", + "System.Threading.Tasks", + "System.Threading.Tasks.Extensions", + "System.Threading.Tasks.Parallel", + "System.Threading.Thread", + "System.Threading.ThreadPool", + "System.Threading.Timer", + "System.ValueTuple", + "System.Xml.ReaderWriter", + "System.Xml.XDocument", + "System.Xml.XmlDocument", + "System.Xml.XmlSerializer", + "System.Xml.XPath", + "System.Xml.XPath.XDocument", + ]; + } +} diff --git a/src/UnityNuGet/Npm/NpmError.cs b/src/UnityNuGet/Npm/NpmError.cs index 4c77053a..17a8a992 100644 --- a/src/UnityNuGet/Npm/NpmError.cs +++ b/src/UnityNuGet/Npm/NpmError.cs @@ -1,24 +1,18 @@ -using Newtonsoft.Json; - -namespace UnityNuGet.Npm -{ - /// - /// A simple object to return npm errors. Used mainly for returning - /// - public class NpmError : NpmObject - { - public static readonly NpmError NotFound = new("not_found", "document not found"); - - public NpmError(string error, string reason) - { - Error = error; - Reason = reason; - } - - [JsonProperty("error")] - public string Error { get; } - - [JsonProperty("reason")] - public string Reason { get; } - } -} +using Newtonsoft.Json; + +namespace UnityNuGet.Npm +{ + /// + /// A simple object to return npm errors. Used mainly for returning + /// + public class NpmError(string error, string reason) : NpmObject + { + public static readonly NpmError NotFound = new("not_found", "document not found"); + + [JsonProperty("error")] + public string Error { get; } = error; + + [JsonProperty("reason")] + public string Reason { get; } = reason; + } +} diff --git a/src/UnityNuGet/Npm/NpmPackage.cs b/src/UnityNuGet/Npm/NpmPackage.cs index dc5420ec..5275f683 100644 --- a/src/UnityNuGet/Npm/NpmPackage.cs +++ b/src/UnityNuGet/Npm/NpmPackage.cs @@ -1,51 +1,51 @@ -using System; -using System.Collections.Generic; -using Newtonsoft.Json; - -namespace UnityNuGet.Npm -{ - /// - /// Describes a full NPM package (used as a response to `{packageId}`) - /// - public class NpmPackage : NpmObject - { - public NpmPackage() - { - Revision = "1-0"; - DistTags = new(); - Versions = new(); - Time = new(); - Users = new(); - } - - [JsonProperty("_id")] - public string? Id { get; set; } - - [JsonProperty("_rev")] - public string Revision { get; set; } - - [JsonProperty("name")] - public string? Name { get; set; } - - [JsonProperty("license")] - public string? License { get; set; } - - [JsonProperty("description")] - public string? Description { get; set; } - - [JsonProperty("dist-tags")] - public Dictionary DistTags { get; } - - [JsonProperty("versions")] - public Dictionary Versions { get; } - - [JsonProperty("time")] - public Dictionary Time { get; } - - [JsonProperty("repository", NullValueHandling = NullValueHandling.Ignore)] - public NpmSourceRepository? Repository { get; set; } - - [JsonProperty("users")] - public Dictionary Users { get; } - } -} +using System; +using System.Collections.Generic; +using Newtonsoft.Json; + +namespace UnityNuGet.Npm +{ + /// + /// Describes a full NPM package (used as a response to `{packageId}`) + /// + public class NpmPackage : NpmObject + { + public NpmPackage() + { + Revision = "1-0"; + DistTags = []; + Versions = []; + Time = []; + Users = []; + } + + [JsonProperty("_id")] + public string? Id { get; set; } + + [JsonProperty("_rev")] + public string Revision { get; set; } + + [JsonProperty("name")] + public string? Name { get; set; } + + [JsonProperty("license")] + public string? License { get; set; } + + [JsonProperty("description")] + public string? Description { get; set; } + + [JsonProperty("dist-tags")] + public Dictionary DistTags { get; } + + [JsonProperty("versions")] + public Dictionary Versions { get; } + + [JsonProperty("time")] + public Dictionary Time { get; } + + [JsonProperty("repository", NullValueHandling = NullValueHandling.Ignore)] + public NpmSourceRepository? Repository { get; set; } + + [JsonProperty("users")] + public Dictionary Users { get; } + } +} diff --git a/src/UnityNuGet/Npm/NpmPackageInfo.cs b/src/UnityNuGet/Npm/NpmPackageInfo.cs index 31834ecb..b2401a70 100644 --- a/src/UnityNuGet/Npm/NpmPackageInfo.cs +++ b/src/UnityNuGet/Npm/NpmPackageInfo.cs @@ -1,40 +1,40 @@ -using System; -using System.Collections.Generic; -using Newtonsoft.Json; - -namespace UnityNuGet.Npm -{ - /// - /// Describes a package for "all" listing, used by - /// - public class NpmPackageInfo : NpmObject - { - public NpmPackageInfo() - { - Maintainers = new(); - Versions = new(); - Keywords = new(); - } - - [JsonProperty("name")] - public string? Name { get; set; } - - [JsonProperty("description")] - public string? Description { get; set; } - - [JsonProperty("maintainers")] - public List Maintainers { get; } - - [JsonProperty("versions")] - public Dictionary Versions { get; } - - [JsonProperty("time")] - public DateTimeOffset? Time { get; set; } - - [JsonProperty("keywords")] - public List Keywords { get; } - - [JsonProperty("author")] - public string? Author { get; set; } - } -} +using System; +using System.Collections.Generic; +using Newtonsoft.Json; + +namespace UnityNuGet.Npm +{ + /// + /// Describes a package for "all" listing, used by + /// + public class NpmPackageInfo : NpmObject + { + public NpmPackageInfo() + { + Maintainers = []; + Versions = []; + Keywords = []; + } + + [JsonProperty("name")] + public string? Name { get; set; } + + [JsonProperty("description")] + public string? Description { get; set; } + + [JsonProperty("maintainers")] + public List Maintainers { get; } + + [JsonProperty("versions")] + public Dictionary Versions { get; } + + [JsonProperty("time")] + public DateTimeOffset? Time { get; set; } + + [JsonProperty("keywords")] + public List Keywords { get; } + + [JsonProperty("author")] + public string? Author { get; set; } + } +} diff --git a/src/UnityNuGet/Npm/NpmPackageListAllResponse.cs b/src/UnityNuGet/Npm/NpmPackageListAllResponse.cs index 7293a718..34ac33f0 100644 --- a/src/UnityNuGet/Npm/NpmPackageListAllResponse.cs +++ b/src/UnityNuGet/Npm/NpmPackageListAllResponse.cs @@ -1,50 +1,48 @@ -using System.Collections.Generic; -using Newtonsoft.Json; -using Newtonsoft.Json.Linq; - -namespace UnityNuGet.Npm -{ - /// - /// Represents a `-/all` listing package NPM response - /// - /// - /// NOTE: only used to serialize from C# to JSON (JSON To C# is not implemented) - /// - public class NpmPackageListAllResponse : NpmObject - { - public NpmPackageListAllResponse() - { - Unused = 99999; - Packages = new(); - } - - [JsonProperty("_updated")] - public int Unused { get; set; } - - [JsonIgnore] - public Dictionary Packages { get; } - - // everything else gets stored here - [JsonExtensionData] -#pragma warning disable IDE0051 // Remove unused private members - private IDictionary AdditionalData -#pragma warning restore IDE0051 // Remove unused private members - { - get - { - var marshalPackages = new Dictionary(); - foreach (var packagePair in Packages) - { - marshalPackages.Add(packagePair.Key, JObject.FromObject(packagePair.Value)); - } - - return marshalPackages; - } - } - - public void Reset() - { - Packages.Clear(); - } - } -} +using System.Collections.Generic; +using Newtonsoft.Json; +using Newtonsoft.Json.Linq; + +namespace UnityNuGet.Npm +{ + /// + /// Represents a `-/all` listing package NPM response + /// + /// + /// NOTE: only used to serialize from C# to JSON (JSON To C# is not implemented) + /// + public class NpmPackageListAllResponse : NpmObject + { + public NpmPackageListAllResponse() + { + Unused = 99999; + Packages = []; + } + + [JsonProperty("_updated")] + public int Unused { get; set; } + + [JsonIgnore] + public Dictionary Packages { get; } + + // Everything else gets stored here + [JsonExtensionData] + private IDictionary AdditionalData + { + get + { + var marshalPackages = new Dictionary(); + foreach (var packagePair in Packages) + { + marshalPackages.Add(packagePair.Key, JObject.FromObject(packagePair.Value)); + } + + return marshalPackages; + } + } + + public void Reset() + { + Packages.Clear(); + } + } +} diff --git a/src/UnityNuGet/Npm/NpmPackageRegistry.cs b/src/UnityNuGet/Npm/NpmPackageRegistry.cs index 3db6003b..4eb69f90 100644 --- a/src/UnityNuGet/Npm/NpmPackageRegistry.cs +++ b/src/UnityNuGet/Npm/NpmPackageRegistry.cs @@ -1,38 +1,38 @@ -using System.Collections.Generic; - -namespace UnityNuGet.Npm -{ - /// - /// Used to store all type of responses (all packages, or single packages but also unlisted packages) - /// - public class NpmPackageRegistry : NpmObject - { - public NpmPackageRegistry() - { - Packages = new(); - ListedPackageInfos = new(); - UnlistedPackageInfos = new(); - } - - public Dictionary Packages { get; } - - public NpmPackageListAllResponse ListedPackageInfos { get; } - - public NpmPackageListAllResponse UnlistedPackageInfos { get; } - - public void AddPackage(NpmPackageCacheEntry entry, bool isListed) - { - var package = entry.Package!; - Packages.Add(package.Id!, package); - var packageInfos = isListed ? ListedPackageInfos : UnlistedPackageInfos; - packageInfos.Packages.Add(package.Id!, entry.Info!); - } - - public void Reset() - { - Packages.Clear(); - ListedPackageInfos.Reset(); - UnlistedPackageInfos.Reset(); - } - } -} +using System.Collections.Generic; + +namespace UnityNuGet.Npm +{ + /// + /// Used to store all type of responses (all packages, or single packages but also unlisted packages) + /// + public class NpmPackageRegistry : NpmObject + { + public NpmPackageRegistry() + { + Packages = []; + ListedPackageInfos = new(); + UnlistedPackageInfos = new(); + } + + public Dictionary Packages { get; } + + public NpmPackageListAllResponse ListedPackageInfos { get; } + + public NpmPackageListAllResponse UnlistedPackageInfos { get; } + + public void AddPackage(NpmPackageCacheEntry entry, bool isListed) + { + var package = entry.Package!; + Packages.Add(package.Id!, package); + var packageInfos = isListed ? ListedPackageInfos : UnlistedPackageInfos; + packageInfos.Packages.Add(package.Id!, entry.Info!); + } + + public void Reset() + { + Packages.Clear(); + ListedPackageInfos.Reset(); + UnlistedPackageInfos.Reset(); + } + } +} diff --git a/src/UnityNuGet/Npm/NpmPackageVersion.cs b/src/UnityNuGet/Npm/NpmPackageVersion.cs index de90dee1..932f7f92 100644 --- a/src/UnityNuGet/Npm/NpmPackageVersion.cs +++ b/src/UnityNuGet/Npm/NpmPackageVersion.cs @@ -1,51 +1,51 @@ -using System.Collections.Generic; -using Newtonsoft.Json; - -namespace UnityNuGet.Npm -{ - /// - /// Describes a version of a - /// - public class NpmPackageVersion : NpmObject - { - public NpmPackageVersion() - { - Dependencies = new(); - Distribution = new(); - Scripts = new(); - } - - [JsonProperty("name")] - public string? Name { get; set; } - - [JsonProperty("version")] - public string? Version { get; set; } - - [JsonProperty("dist")] - public NpmDistribution Distribution { get; } - - [JsonProperty("dependencies")] - public Dictionary Dependencies { get; } - - [JsonProperty("_id")] - public string? Id { get; set; } - - [JsonProperty("unity", NullValueHandling = NullValueHandling.Ignore)] - public string? Unity { get; set; } - - [JsonProperty("description", NullValueHandling = NullValueHandling.Ignore)] - public string? Description { get; set; } - - [JsonProperty("displayName", NullValueHandling = NullValueHandling.Ignore)] - public string? DisplayName { get; set; } - - [JsonProperty("scripts", NullValueHandling = NullValueHandling.Ignore)] - public Dictionary Scripts { get; } - - [JsonProperty("repository", NullValueHandling = NullValueHandling.Ignore)] - public NpmSourceRepository? Repository { get; set; } - - [JsonProperty("author", NullValueHandling = NullValueHandling.Ignore)] - public string? Author { get; set; } - } -} +using System.Collections.Generic; +using Newtonsoft.Json; + +namespace UnityNuGet.Npm +{ + /// + /// Describes a version of a + /// + public class NpmPackageVersion : NpmObject + { + public NpmPackageVersion() + { + Dependencies = []; + Distribution = new(); + Scripts = []; + } + + [JsonProperty("name")] + public string? Name { get; set; } + + [JsonProperty("version")] + public string? Version { get; set; } + + [JsonProperty("dist")] + public NpmDistribution Distribution { get; } + + [JsonProperty("dependencies")] + public Dictionary Dependencies { get; } + + [JsonProperty("_id")] + public string? Id { get; set; } + + [JsonProperty("unity", NullValueHandling = NullValueHandling.Ignore)] + public string? Unity { get; set; } + + [JsonProperty("description", NullValueHandling = NullValueHandling.Ignore)] + public string? Description { get; set; } + + [JsonProperty("displayName", NullValueHandling = NullValueHandling.Ignore)] + public string? DisplayName { get; set; } + + [JsonProperty("scripts", NullValueHandling = NullValueHandling.Ignore)] + public Dictionary Scripts { get; } + + [JsonProperty("repository", NullValueHandling = NullValueHandling.Ignore)] + public NpmSourceRepository? Repository { get; set; } + + [JsonProperty("author", NullValueHandling = NullValueHandling.Ignore)] + public string? Author { get; set; } + } +} diff --git a/src/UnityNuGet/NuGetHelper.cs b/src/UnityNuGet/NuGetHelper.cs index 7266c3d0..ff4f6111 100644 --- a/src/UnityNuGet/NuGetHelper.cs +++ b/src/UnityNuGet/NuGetHelper.cs @@ -8,10 +8,11 @@ namespace UnityNuGet { - static class NuGetHelper + static partial class NuGetHelper { // https://learn.microsoft.com/en-us/visualstudio/extensibility/roslyn-version-support - private static readonly Regex roslynVersionRegex = new(@"/roslyn(\d+)\.(\d+)\.?(\d*)/"); + [GeneratedRegex(@"/roslyn(\d+)\.(\d+)\.?(\d*)/")] + private static partial Regex RoslynVersion(); // https://docs.unity3d.com/Manual/roslyn-analyzers.html private static readonly Version unityRoslynSupportedVersion = new(3, 8, 0); @@ -63,7 +64,7 @@ bool IsResource() public static bool IsApplicableUnitySupportedRoslynVersionFolder(string file) { - var roslynVersionMatch = roslynVersionRegex.Match(file); + var roslynVersionMatch = RoslynVersion().Match(file); bool hasRoslynVersionFolder = roslynVersionMatch.Success; bool hasUnitySupportedRoslynVersionFolder = hasRoslynVersionFolder && diff --git a/src/UnityNuGet/PlatformDefinition.cs b/src/UnityNuGet/PlatformDefinition.cs index 08d8e277..04e247ae 100644 --- a/src/UnityNuGet/PlatformDefinition.cs +++ b/src/UnityNuGet/PlatformDefinition.cs @@ -1,407 +1,392 @@ -using System; -using System.Collections.Generic; -using System.IO; -using System.Linq; - -namespace UnityNuGet -{ - /// - /// All operating systems supported by Unity - /// - public enum UnityOs - { - AnyOs, - Windows, - Linux, - OSX, - Android, - WebGL, - iOS - } - - /// - /// All CPUs supported by Unity - /// - public enum UnityCpu - { - AnyCpu, - X64, - X86, - ARM64, - ARMv7, - None, +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; + +namespace UnityNuGet +{ + /// + /// All operating systems supported by Unity + /// + public enum UnityOs + { + AnyOs, + Windows, + Linux, + OSX, + Android, + WebGL, + iOS + } + + /// + /// All CPUs supported by Unity + /// + public enum UnityCpu + { + AnyCpu, + X64, + X86, + ARM64, + ARMv7, + None, + } + + /// + /// Extensions for and + /// + internal static class UnityEnumExtensions + { + /// + /// Converts a to string. + /// + /// The value to be converted. + /// This method should be used for constructing package paths for OS-dependent files. + /// A string representation of the provided value. + public static string GetPathName(this UnityOs os) + { + return os switch + { + UnityOs.AnyOs => string.Empty, + _ => GetName(os), + }; + } + + /// + /// Converts a to a string representation accepted by the Unity .meta file format. + /// + /// The value to be converted. + /// A string representation of the provided value. + public static string GetName(this UnityOs os) + { + return os switch + { + UnityOs.AnyOs => "AnyOS", + UnityOs.Windows => "Windows", + UnityOs.Linux => "Linux", + UnityOs.OSX => "OSX", + UnityOs.Android => "Android", + UnityOs.WebGL => "WebGL", + UnityOs.iOS => "iOS", + _ => throw new ArgumentException($"Unknown OS {os}"), + }; + } + + /// + /// Converts a to string. + /// + /// The value to be converted. + /// This method should be used for constructing package paths for CPU-dependent files. + /// A string representation of the provided value. + public static string GetPathName(this UnityCpu cpu) + { + return cpu switch + { + UnityCpu.AnyCpu => string.Empty, + _ => GetName(cpu), + }; + } + + /// + /// Converts a to a string representation accepted by the Unity .meta file format. + /// + /// The value to be converted. + /// A string representation of the provided value. + public static string GetName(this UnityCpu cpu) + { + return cpu switch + { + UnityCpu.AnyCpu => "AnyCPU", + UnityCpu.X64 => "x86_64", + UnityCpu.X86 => "x86", + UnityCpu.ARM64 => "ARM64", + UnityCpu.ARMv7 => "ARMv7", + UnityCpu.None => "None", + _ => throw new ArgumentException($"Unknown CPU {cpu}"), + }; + } } - /// - /// Extensions for and - /// - internal static class UnityEnumExtensions - { - /// - /// Converts a to string. - /// - /// The value to be converted. - /// This method should be used for constructing package paths for OS-dependent files. - /// A string representation of the provided value. - public static string GetPathName(this UnityOs os) - { - return os switch - { - UnityOs.AnyOs => string.Empty, - _ => GetName(os), - }; - } - - /// - /// Converts a to a string representation accepted by the Unity .meta file format. - /// - /// The value to be converted. - /// A string representation of the provided value. - public static string GetName(this UnityOs os) - { - return os switch - { - UnityOs.AnyOs => "AnyOS", - UnityOs.Windows => "Windows", - UnityOs.Linux => "Linux", - UnityOs.OSX => "OSX", - UnityOs.Android => "Android", - UnityOs.WebGL => "WebGL", - UnityOs.iOS => "iOS", - _ => throw new ArgumentException($"Unknown OS {os}"), - }; - } - - /// - /// Converts a to string. - /// - /// The value to be converted. - /// This method should be used for constructing package paths for CPU-dependent files. - /// A string representation of the provided value. - public static string GetPathName(this UnityCpu cpu) - { - return cpu switch - { - UnityCpu.AnyCpu => string.Empty, - _ => GetName(cpu), - }; - } - - /// - /// Converts a to a string representation accepted by the Unity .meta file format. - /// - /// The value to be converted. - /// A string representation of the provided value. - public static string GetName(this UnityCpu cpu) - { - return cpu switch - { - UnityCpu.AnyCpu => "AnyCPU", - UnityCpu.X64 => "x86_64", - UnityCpu.X86 => "x86", - UnityCpu.ARM64 => "ARM64", - UnityCpu.ARMv7 => "ARMv7", - UnityCpu.None => "None", - _ => throw new ArgumentException($"Unknown CPU {cpu}"), - }; - } + /// + /// A subtree of hierarchical (os, cpu) configuration tuples that are supported by Unity. + /// + /// + /// The root node is typically the most general configuration, i.e. + /// supporting and . Leaf nodes are typically specialized, + /// targeting a specific OS and CPU flavor. + /// + /// + /// Creates a new instance. + /// + /// The OS. + /// The CPU flavor. + /// True if the Unity editor is available in this (os, cpu) tuple. + internal class PlatformDefinition(UnityOs os, UnityCpu cpu, bool isEditorConfig) + { + private readonly UnityOs _os = os; + private readonly UnityCpu _cpu = cpu; + private readonly bool _isEditor = isEditorConfig; + private readonly List _children = []; + + private PlatformDefinition? _parent = null; + + /// + /// The parent that is a superset of this. + /// + public PlatformDefinition? Parent + { + get => _parent; + + private set + { + _parent = value; + } + } + + /// + /// The child configurations this is a superset of. + /// + public IReadOnlyList Children + { + get => _children; + + private set + { + _children.AddRange(value); + + foreach (var child in _children) + { + child.Parent = this; + } + } + } + + /// + /// The distance from the root in the configuration tree. + /// + public int Depth + => (_parent == null) ? 0 : (1 + _parent.Depth); + + /// + /// The operating system. + /// + public UnityOs Os + => _os; + + /// + /// The CPU flavor. + /// + public UnityCpu Cpu + => _cpu; + + /// + public override string ToString() + => $"{_os}.{_cpu}"; + + /// + /// Attempts to find a that matches the given + /// and among the descendants of this configuration. + /// + /// The operating system to match. + /// The CPU flavor to match. + /// A matching . + public PlatformDefinition? Find(UnityOs os, UnityCpu? cpu = default) + { + // Test self + if ((_os == os) && ((cpu == null) || (_cpu == cpu))) + { + return this; + } + + // Recurse to children + return _children + .Select(c => c.Find(os, cpu)) + .Where(c => c != null) + .FirstOrDefault(); + } + + /// + /// Attempts to find a for which the Unity editor is available + /// among the descendants of this configuration. + /// + /// A matching . + public PlatformDefinition? FindEditor() + { + // Test self + if (_isEditor) + { + return this; + } + + // Recurse to children + return _children + .Select(c => c.FindEditor()) + .Where(c => c != null) + .FirstOrDefault(); + } + + /// + /// Returns the difference set of configurations this is a superset of, + /// that are not already part of the given visited set. + /// + /// The set of already visited configurations, that should not be part of the returned set. + /// + /// A set of configurations this is a superset of, + /// that are not already part of the given visited set. + /// + public HashSet GetRemainingPlatforms(IReadOnlySet visitedPlatforms) + { + var remainingPlatforms = new HashSet + { + // Push the root + this + }; + + for (bool found = true; found;) + { + found = false; + + foreach (var p in remainingPlatforms) + { + // Remove p if already visited + if (visitedPlatforms.Contains(p)) + { + remainingPlatforms.Remove(p); + found = true; + break; + } + + // If p has descendants that were visited, we can't use it and need to expand it + if (p.HasVisitedDescendants(visitedPlatforms)) + { + remainingPlatforms.Remove(p); + + foreach (var c in p.Children) + { + remainingPlatforms.Add(c); + } + + found = true; + break; + } + } + } + + return remainingPlatforms; + } + + /// + /// Creates the tree of all known (os, cpu) configurations supported by Unity + /// + /// + public static PlatformDefinition CreateAllPlatforms() + { + var root = new PlatformDefinition(UnityOs.AnyOs, UnityCpu.AnyCpu, isEditorConfig: true) + { + Children = new List() + { + new(UnityOs.Windows, UnityCpu.AnyCpu, isEditorConfig: true) + { + Children = new List() + { + new(UnityOs.Windows, UnityCpu.X64, isEditorConfig: true), + new(UnityOs.Windows, UnityCpu.X86, isEditorConfig: false), + }, + }, + new(UnityOs.Linux, UnityCpu.X64, isEditorConfig: true), + new(UnityOs.Android, UnityCpu.ARMv7, isEditorConfig: false), + new(UnityOs.WebGL, UnityCpu.AnyCpu, isEditorConfig: false), + new(UnityOs.iOS, UnityCpu.AnyCpu, isEditorConfig: false), + new(UnityOs.OSX, UnityCpu.AnyCpu, isEditorConfig: true) + { + Children = new List() + { + new(UnityOs.OSX, UnityCpu.X64, isEditorConfig: true), + new(UnityOs.OSX, UnityCpu.ARM64, isEditorConfig: true), + }, + } + } + }; + + return root; + } + + /// + /// Returns true if either this or any of its descendants + /// is also part of the given set of visited configurations. + /// The set of visited configurations to test against. + /// + private bool HasVisitedDescendants(IReadOnlySet visitedPlatforms) + { + if (visitedPlatforms.Contains(this)) + { + return true; + } + + foreach (var c in _children) + { + if (c.HasVisitedDescendants(visitedPlatforms)) + { + return true; + } + } + + return false; + } } - /// - /// A subtree of hierarchical (os, cpu) configuration tuples that are supported by Unity. - /// - /// - /// The root node is typically the most general configuration, i.e. - /// supporting and . Leaf nodes are typically specialized, - /// targeting a specific OS and CPU flavor. - /// - internal class PlatformDefinition - { - private readonly UnityOs _os; - private readonly UnityCpu _cpu; - private readonly bool _isEditor; - private readonly List _children; - - private PlatformDefinition? _parent; - - /// - /// Creates a new instance. - /// - /// The OS. - /// The CPU flavor. - /// True if the Unity editor is available in this (os, cpu) tuple. - public PlatformDefinition(UnityOs os, UnityCpu cpu, bool isEditorConfig) - { - _os = os; - _cpu = cpu; - _parent = null; - _children = new(); - _isEditor = isEditorConfig; - } - - /// - /// The parent that is a superset of this. - /// - public PlatformDefinition? Parent - { - get => _parent; - - private set - { - _parent = value; - } - } - - /// - /// The child configurations this is a superset of. - /// - public IReadOnlyList Children - { - get => _children; - - private set - { - _children.AddRange(value); - - foreach (var child in _children) - { - child.Parent = this; - } - } - } - - /// - /// The distance from the root in the configuration tree. - /// - public int Depth - => (_parent == null) ? 0 : (1 + _parent.Depth); - - /// - /// The operating system. - /// - public UnityOs Os - => _os; - - /// - /// The CPU flavor. - /// - public UnityCpu Cpu - => _cpu; - - /// - public override string ToString() - => $"{_os}.{_cpu}"; - - /// - /// Attempts to find a that matches the given - /// and among the descendants of this configuration. - /// - /// The operating system to match. - /// The CPU flavor to match. - /// A matching . - public PlatformDefinition? Find(UnityOs os, UnityCpu? cpu = default) - { - // Test self - if ((_os == os) && ((cpu == null) || (_cpu == cpu))) - { - return this; - } - - // Recurse to children - return _children - .Select(c => c.Find(os, cpu)) - .Where(c => c != null) - .FirstOrDefault(); - } - - /// - /// Attempts to find a for which the Unity editor is available - /// among the descendants of this configuration. - /// - /// A matching . - public PlatformDefinition? FindEditor() - { - // Test self - if (_isEditor) - { - return this; - } - - // Recurse to children - return _children - .Select(c => c.FindEditor()) - .Where(c => c != null) - .FirstOrDefault(); - } - - /// - /// Returns the difference set of configurations this is a superset of, - /// that are not already part of the given visited set. - /// - /// The set of already visited configurations, that should not be part of the returned set. - /// - /// A set of configurations this is a superset of, - /// that are not already part of the given visited set. - /// - public HashSet GetRemainingPlatforms(IReadOnlySet visitedPlatforms) - { - var remainingPlatforms = new HashSet - { - // Push the root - this - }; - - for (bool found = true; found;) - { - found = false; - - foreach (var p in remainingPlatforms) - { - // Remove p if already visited - if (visitedPlatforms.Contains(p)) - { - remainingPlatforms.Remove(p); - found = true; - break; - } - - // If p has descendants that were visited, we can't use it and need to expand it - if (p.HasVisitedDescendants(visitedPlatforms)) - { - remainingPlatforms.Remove(p); - - foreach (var c in p.Children) - { - remainingPlatforms.Add(c); - } - - found = true; - break; - } - } - } - - return remainingPlatforms; - } - - /// - /// Creates the tree of all known (os, cpu) configurations supported by Unity - /// - /// - public static PlatformDefinition CreateAllPlatforms() - { - var root = new PlatformDefinition(UnityOs.AnyOs, UnityCpu.AnyCpu, isEditorConfig: true) - { - Children = new List() - { - new PlatformDefinition(UnityOs.Windows, UnityCpu.AnyCpu, isEditorConfig: true) - { - Children = new List() - { - new PlatformDefinition(UnityOs.Windows, UnityCpu.X64, isEditorConfig: true), - new PlatformDefinition(UnityOs.Windows, UnityCpu.X86, isEditorConfig: false), - }, - }, - new PlatformDefinition(UnityOs.Linux, UnityCpu.X64, isEditorConfig: true), - new PlatformDefinition(UnityOs.Android, UnityCpu.ARMv7, isEditorConfig: false), - new PlatformDefinition(UnityOs.WebGL, UnityCpu.AnyCpu, isEditorConfig: false), - new PlatformDefinition(UnityOs.iOS, UnityCpu.AnyCpu, isEditorConfig: false), - new PlatformDefinition(UnityOs.OSX, UnityCpu.AnyCpu, isEditorConfig: true) - { - Children = new List() - { - new PlatformDefinition(UnityOs.OSX, UnityCpu.X64, isEditorConfig: true), - new PlatformDefinition(UnityOs.OSX, UnityCpu.ARM64, isEditorConfig: true), - }, - } - } - }; - - return root; - } - - /// - /// Returns true if either this or any of its descendants - /// is also part of the given set of visited configurations. - /// The set of visited configurations to test against. - /// - private bool HasVisitedDescendants(IReadOnlySet visitedPlatforms) - { - if (visitedPlatforms.Contains(this)) - { - return true; - } - - foreach (var c in _children) - { - if (c.HasVisitedDescendants(visitedPlatforms)) - { - return true; - } - } - - return false; - } - } - - /// - /// Associates a file path with the the file is compatible with. - /// - internal class PlatformFile - { - private readonly PlatformDefinition _platform; - private readonly string _sourcePath; - - /// - /// Creates a new instance. - /// - /// The full path of the file in the original NuGet package. - /// The platform the file is compatible with. - public PlatformFile(string sourcePath, PlatformDefinition platform) - { - _sourcePath = sourcePath; - _platform = platform; - } - - /// - /// The full path of the file in the original NuGet package. - /// - public string SourcePath - => _sourcePath; - - /// - /// The the file is compatible with. - /// - public PlatformDefinition Platform - => _platform; - - /// - /// Returns the full path of the file in the UPM package. - /// - /// The file path to start from. - /// The full path of the file in the UPM package. - public string GetDestinationPath(string basePath) - { - // We start with just the base path - var fullPath = basePath; - var depth = _platform.Depth; - - if (depth > 0) - { - // Our configuration is not AnyOS, add a / to our full path - fullPath = Path.Combine(fullPath, _platform.Os.GetPathName()); - } - - if (depth > 1) - { - // Our CPU os not AnyCPU, add a / to our full path - fullPath = Path.Combine(fullPath, _platform.Cpu.GetPathName()); - } - - // Finally, append the file name and return - var fileName = Path.GetFileName(_sourcePath); - fullPath = Path.Combine(fullPath, fileName); - return fullPath; - } - } -} + /// + /// Associates a file path with the the file is compatible with. + /// + /// + /// Creates a new instance. + /// + /// The full path of the file in the original NuGet package. + /// The platform the file is compatible with. + internal class PlatformFile(string sourcePath, PlatformDefinition platform) + { + private readonly PlatformDefinition _platform = platform; + private readonly string _sourcePath = sourcePath; + + /// + /// The full path of the file in the original NuGet package. + /// + public string SourcePath + => _sourcePath; + + /// + /// The the file is compatible with. + /// + public PlatformDefinition Platform + => _platform; + + /// + /// Returns the full path of the file in the UPM package. + /// + /// The file path to start from. + /// The full path of the file in the UPM package. + public string GetDestinationPath(string basePath) + { + // We start with just the base path + var fullPath = basePath; + var depth = _platform.Depth; + + if (depth > 0) + { + // Our configuration is not AnyOS, add a / to our full path + fullPath = Path.Combine(fullPath, _platform.Os.GetPathName()); + } + + if (depth > 1) + { + // Our CPU os not AnyCPU, add a / to our full path + fullPath = Path.Combine(fullPath, _platform.Cpu.GetPathName()); + } + + // Finally, append the file name and return + var fileName = Path.GetFileName(_sourcePath); + fullPath = Path.Combine(fullPath, fileName); + return fullPath; + } + } +} diff --git a/src/UnityNuGet/Registry.cs b/src/UnityNuGet/Registry.cs index de0d0f04..a9c19297 100644 --- a/src/UnityNuGet/Registry.cs +++ b/src/UnityNuGet/Registry.cs @@ -1,44 +1,39 @@ -using System; -using System.Collections.Generic; -using System.IO; -using System.Runtime.Serialization; -using Newtonsoft.Json; - -namespace UnityNuGet -{ - /// - /// Loads the `registry.json` file at startup - /// - [Serializable] - public sealed class Registry : Dictionary - { - private const string RegistryFileName = "registry.json"; - private static readonly object LockRead = new(); - private static Registry? _registry = null; - - // A comparer is established for cases where the dependency name is not set to the correct case. - // Example: https://www.nuget.org/packages/NeoSmart.Caching.Sqlite/0.1.0#dependencies-body-tab - public Registry() : base(StringComparer.OrdinalIgnoreCase) - { - } - - private Registry(SerializationInfo info, StreamingContext context) : base(info, context) - { - } - - public static Registry Parse(string json) - { - ArgumentNullException.ThrowIfNull(json); - return JsonConvert.DeserializeObject(json, JsonCommonExtensions.Settings)!; - } - - public static Registry GetInstance() - { - lock (LockRead) - { - _registry ??= Parse(File.ReadAllText(Path.Combine(Path.GetDirectoryName(typeof(Registry).Assembly.Location)!, RegistryFileName))); - } - return _registry; - } - } -} +using System; +using System.Collections.Generic; +using System.IO; +using Newtonsoft.Json; + +namespace UnityNuGet +{ + /// + /// Loads the `registry.json` file at startup + /// + [Serializable] + public sealed class Registry : Dictionary + { + private const string RegistryFileName = "registry.json"; + private static readonly object LockRead = new(); + private static Registry? _registry = null; + + // A comparer is established for cases where the dependency name is not set to the correct case. + // Example: https://www.nuget.org/packages/NeoSmart.Caching.Sqlite/0.1.0#dependencies-body-tab + public Registry() : base(StringComparer.OrdinalIgnoreCase) + { + } + + public static Registry Parse(string json) + { + ArgumentNullException.ThrowIfNull(json); + return JsonConvert.DeserializeObject(json, JsonCommonExtensions.Settings)!; + } + + public static Registry GetInstance() + { + lock (LockRead) + { + _registry ??= Parse(File.ReadAllText(Path.Combine(Path.GetDirectoryName(typeof(Registry).Assembly.Location)!, RegistryFileName))); + } + return _registry; + } + } +} diff --git a/src/UnityNuGet/RegistryCache.cs b/src/UnityNuGet/RegistryCache.cs index c63f4b3a..323a2f0e 100644 --- a/src/UnityNuGet/RegistryCache.cs +++ b/src/UnityNuGet/RegistryCache.cs @@ -280,7 +280,7 @@ private async Task BuildInternal() var resolvedDependencyGroups = NuGetHelper.GetCompatiblePackageDependencyGroups(packageMeta.DependencySets, _targetFrameworks).ToList(); - if (!packageEntry.Analyzer && !resolvedDependencyGroups.Any()) + if (!packageEntry.Analyzer && resolvedDependencyGroups.Count == 0) { using var downloadResult = await GetPackageDownloadResourceResult(packageIdentity); @@ -570,7 +570,7 @@ RegistryEntry packageEntry { if (!collectedItems.TryGetValue(item, out var frameworksPerGroup)) { - frameworksPerGroup = new HashSet(); + frameworksPerGroup = []; collectedItems.Add(item, frameworksPerGroup); } frameworksPerGroup.Add(targetFramework); @@ -643,16 +643,16 @@ RegistryEntry packageEntry meta = UnityMeta.GetMetaForDll( GetStableGuid(identity, fileInUnityPackage), new PlatformDefinition(UnityOs.AnyOs, UnityCpu.None, isEditorConfig: false), - new string[] { "RoslynAnalyzer" }, - Array.Empty()); + ["RoslynAnalyzer"], + []); } else { meta = UnityMeta.GetMetaForDll( GetStableGuid(identity, fileInUnityPackage), new PlatformDefinition(UnityOs.AnyOs, UnityCpu.None, isEditorConfig: false), - Array.Empty(), - Array.Empty()); + [], + []); } } else @@ -765,7 +765,7 @@ RegistryEntry packageEntry } else { - folders = Array.Empty(); + folders = []; } string folder = string.Empty; @@ -789,7 +789,7 @@ RegistryEntry packageEntry // use NET_STANDARD var defineConstraints = hasMultiNetStandard || hasOnlyNetStandard21 - || isPackageNetStandard21Assembly ? frameworks.First(x => x.Framework == item.TargetFramework).DefineConstraints : Array.Empty(); + || isPackageNetStandard21Assembly ? frameworks.First(x => x.Framework == item.TargetFramework).DefineConstraints : []; meta = UnityMeta.GetMetaForDll( GetStableGuid(identity, fileInUnityPackage), @@ -902,7 +902,7 @@ RegistryEntry packageEntry if (licenseUrlText != null) { licenseUrlText = licenseUrlText.Trim(); - if (licenseUrlText.StartsWith("<")) + if (licenseUrlText.StartsWith('<')) { try { @@ -1112,7 +1112,7 @@ private static UnityAsmdef CreateAnalyzerAmsdef(PackageIdentity packageIdentity) return new() { Name = $"{packageIdentity.Id}_Unity", // Add _Unity suffix because Unity has a validation so that assemblies names do not collide with asmdefs assembly names - IncludePlatforms = new string[] { "Editor" } + IncludePlatforms = ["Editor"] }; } @@ -1164,8 +1164,15 @@ private void LogError(string message) private static List SplitCommaSeparatedString(string input) { var list = new List(); - if (input == null) return list; - foreach (var entry in input.Split(new[] { ',', ';' }, StringSplitOptions.RemoveEmptyEntries)) + + if (input == null) + { + return list; + } + + char[] separators = [',', ';']; + + foreach (var entry in input.Split(separators, StringSplitOptions.RemoveEmptyEntries)) { list.Add(entry.Trim()); } diff --git a/src/UnityNuGet/RegistryEntry.cs b/src/UnityNuGet/RegistryEntry.cs index 2c2c67d8..8c00a404 100644 --- a/src/UnityNuGet/RegistryEntry.cs +++ b/src/UnityNuGet/RegistryEntry.cs @@ -1,27 +1,27 @@ -using System.Collections.Generic; -using Newtonsoft.Json; -using NuGet.Versioning; - -namespace UnityNuGet -{ - /// - /// An entry in the - /// - public class RegistryEntry - { - [JsonProperty("ignore")] - public bool Ignored { get; set; } - - [JsonProperty("listed")] - public bool Listed { get; set; } - - [JsonProperty("version")] - public VersionRange? Version { get; set; } - - [JsonProperty("defineConstraints")] - public List DefineConstraints { get; set; } = new(); - - [JsonProperty("analyzer")] - public bool Analyzer { get; set; } - } -} +using System.Collections.Generic; +using Newtonsoft.Json; +using NuGet.Versioning; + +namespace UnityNuGet +{ + /// + /// An entry in the + /// + public class RegistryEntry + { + [JsonProperty("ignore")] + public bool Ignored { get; set; } + + [JsonProperty("listed")] + public bool Listed { get; set; } + + [JsonProperty("version")] + public VersionRange? Version { get; set; } + + [JsonProperty("defineConstraints")] + public List DefineConstraints { get; set; } = []; + + [JsonProperty("analyzer")] + public bool Analyzer { get; set; } + } +} diff --git a/src/UnityNuGet/UnityNuGet.csproj b/src/UnityNuGet/UnityNuGet.csproj index 1e3306b1..6943335e 100644 --- a/src/UnityNuGet/UnityNuGet.csproj +++ b/src/UnityNuGet/UnityNuGet.csproj @@ -1,11 +1,8 @@ - + - net7.0 - enable + net8.0 0.14.0 - 9.0 - true @@ -23,14 +20,14 @@ - - - - - - - - + + + + + + + + diff --git a/src/UnityNuGet/UnityPackage.cs b/src/UnityNuGet/UnityPackage.cs index ae6be33d..b07b41d8 100644 --- a/src/UnityNuGet/UnityPackage.cs +++ b/src/UnityNuGet/UnityPackage.cs @@ -1,41 +1,41 @@ -using System.Collections.Generic; -using Newtonsoft.Json; - -namespace UnityNuGet -{ - /// - /// - /// - public class UnityPackage : JsonObjectBase - { - public UnityPackage() - { - Keywords = new List(); - Dependencies = new Dictionary(); - } - - [JsonProperty("name")] - public string? Name { get; set; } - - [JsonProperty("displayName")] - public string? DisplayName { get; set; } - - [JsonProperty("version")] - public string? Version { get; set; } - - [JsonProperty("unity")] - public string? Unity { get; set; } - - [JsonProperty("description")] - public string? Description { get; set; } - - [JsonProperty("keywords")] - public List Keywords { get; } - - [JsonProperty("category")] - public string? Category { get; set; } - - [JsonProperty("dependencies")] - public Dictionary Dependencies { get; } - } -} +using System.Collections.Generic; +using Newtonsoft.Json; + +namespace UnityNuGet +{ + /// + /// + /// + public class UnityPackage : JsonObjectBase + { + public UnityPackage() + { + Keywords = []; + Dependencies = []; + } + + [JsonProperty("name")] + public string? Name { get; set; } + + [JsonProperty("displayName")] + public string? DisplayName { get; set; } + + [JsonProperty("version")] + public string? Version { get; set; } + + [JsonProperty("unity")] + public string? Unity { get; set; } + + [JsonProperty("description")] + public string? Description { get; set; } + + [JsonProperty("keywords")] + public List Keywords { get; } + + [JsonProperty("category")] + public string? Category { get; set; } + + [JsonProperty("dependencies")] + public Dictionary Dependencies { get; } + } +} diff --git a/src/global.json b/src/global.json index 3a95bf88..f7fb55b4 100644 --- a/src/global.json +++ b/src/global.json @@ -1,6 +1,6 @@ { "sdk": { - "version": "7.0.100", + "version": "8.0.100", "rollForward": "latestMinor", "allowPrerelease": false }