diff --git a/assets/mocks/recipes-detail-01.jpeg b/assets/mocks/recipes-detail-01.jpeg new file mode 100644 index 00000000..0bad9f0f Binary files /dev/null and b/assets/mocks/recipes-detail-01.jpeg differ diff --git a/assets/mocks/recipes-detail-step-01.jpeg b/assets/mocks/recipes-detail-step-01.jpeg new file mode 100644 index 00000000..73088325 Binary files /dev/null and b/assets/mocks/recipes-detail-step-01.jpeg differ diff --git a/assets/mocks/recipes-detail-step-02.jpeg b/assets/mocks/recipes-detail-step-02.jpeg new file mode 100644 index 00000000..cfdfeb2e Binary files /dev/null and b/assets/mocks/recipes-detail-step-02.jpeg differ diff --git a/assets/mocks/recipes-detail-step-03.jpeg b/assets/mocks/recipes-detail-step-03.jpeg new file mode 100644 index 00000000..0f05c194 Binary files /dev/null and b/assets/mocks/recipes-detail-step-03.jpeg differ diff --git a/assets/mocks/recipes-overview-01.jpeg b/assets/mocks/recipes-overview-01.jpeg new file mode 100644 index 00000000..591ef03a Binary files /dev/null and b/assets/mocks/recipes-overview-01.jpeg differ diff --git a/generate-client.sh b/generate-client.sh index 616a6ab0..0ee765ee 100644 --- a/generate-client.sh +++ b/generate-client.sh @@ -7,11 +7,14 @@ CLIENT_CLASS="BackendClient" CLIENT_NAMESPACE="HomeBook.Client" OPENAPI_FILE="./source/HomeBook.Backend/HomeBook.Backend.json" CLIENT_OUTPUT_DIR="./source/HomeBook.Client" +CLIENT_CSPROJ="HomeBook.Client.csproj" -# build backend +# Clean output dir except the client csproj +find "${CLIENT_OUTPUT_DIR}" -mindepth 1 ! -name "${CLIENT_CSPROJ}" -exec rm -rf {} + -dotnet restore "${BACKEND_CSPROJ}" -dotnet build "${BACKEND_CSPROJ}" --no-restore -c Release +# build backend +dotnet build "${BACKEND_CSPROJ}" -c Debug +dotnet build "${BACKEND_CSPROJ}" -c Release # install/update kiota dotnet tool install --global Microsoft.OpenApi.Kiota @@ -26,3 +29,6 @@ kiota generate \ echo "Client generation completed successfully!" echo "Output directory: ${CLIENT_OUTPUT_DIR}" + +# nuget restore client +dotnet restore "${CLIENT_OUTPUT_DIR}/${CLIENT_CSPROJ}" diff --git a/homebook.slnx b/homebook.slnx index 4d35a74e..957d0989 100644 --- a/homebook.slnx +++ b/homebook.slnx @@ -1,37 +1,42 @@ - - - + + + + - - - - - - - + + + + + + + - - - - + + + + + + + + - - - - - - - + + + + + + + - - - + + + @@ -63,6 +68,6 @@ - + \ No newline at end of file diff --git a/source/HomeBook.Backend.Abstractions/Contracts/IEndpointDataAccessor.cs b/source/HomeBook.Backend.Abstractions/Contracts/IEndpointDataAccessor.cs new file mode 100644 index 00000000..11d27f84 --- /dev/null +++ b/source/HomeBook.Backend.Abstractions/Contracts/IEndpointDataAccessor.cs @@ -0,0 +1,5 @@ +namespace HomeBook.Backend.Abstractions.Contracts; + +public interface IEndpointDataAccessor +{ +} diff --git a/source/HomeBook.Backend.Abstractions/Contracts/ISearchAggregationResult.cs b/source/HomeBook.Backend.Abstractions/Contracts/ISearchAggregationResult.cs new file mode 100644 index 00000000..69228683 --- /dev/null +++ b/source/HomeBook.Backend.Abstractions/Contracts/ISearchAggregationResult.cs @@ -0,0 +1,8 @@ +namespace HomeBook.Backend.Abstractions.Contracts; + +public interface ISearchAggregationResult +{ + public string ModuleKey { get; } + public int TotalCount { get; } + public IEnumerable Items { get; } +} diff --git a/source/HomeBook.Backend.Abstractions/Contracts/ISearchProvider.cs b/source/HomeBook.Backend.Abstractions/Contracts/ISearchProvider.cs new file mode 100644 index 00000000..64b60bec --- /dev/null +++ b/source/HomeBook.Backend.Abstractions/Contracts/ISearchProvider.cs @@ -0,0 +1,19 @@ +namespace HomeBook.Backend.Abstractions.Contracts; + +/// +/// +/// +public interface ISearchProvider +{ + /// + /// + /// + /// + /// + /// + /// + Task> + SearchAsync(string query, + Guid userId, + CancellationToken cancellationToken = default); +} diff --git a/source/HomeBook.Backend.Abstractions/Contracts/ISearchRegistrationFactory.cs b/source/HomeBook.Backend.Abstractions/Contracts/ISearchRegistrationFactory.cs new file mode 100644 index 00000000..ce8b1818 --- /dev/null +++ b/source/HomeBook.Backend.Abstractions/Contracts/ISearchRegistrationFactory.cs @@ -0,0 +1,6 @@ +namespace HomeBook.Backend.Abstractions.Contracts; + +public interface ISearchRegistrationFactory +{ + ISearchProvider CreateSearchProvider(); +} diff --git a/source/HomeBook.Backend.Abstractions/Contracts/ISearchRegistrationInitiator.cs b/source/HomeBook.Backend.Abstractions/Contracts/ISearchRegistrationInitiator.cs new file mode 100644 index 00000000..d7b3f9fb --- /dev/null +++ b/source/HomeBook.Backend.Abstractions/Contracts/ISearchRegistrationInitiator.cs @@ -0,0 +1,7 @@ +namespace HomeBook.Backend.Abstractions.Contracts; + +public interface ISearchRegistrationInitiator +{ + void AddModule(string moduleId); + void AddServiceProvider(IServiceProvider serviceProvider); +} diff --git a/source/HomeBook.Backend.Abstractions/Contracts/ISearchResultItem.cs b/source/HomeBook.Backend.Abstractions/Contracts/ISearchResultItem.cs new file mode 100644 index 00000000..e956308c --- /dev/null +++ b/source/HomeBook.Backend.Abstractions/Contracts/ISearchResultItem.cs @@ -0,0 +1,10 @@ +namespace HomeBook.Backend.Abstractions.Contracts; + +public interface ISearchResultItem +{ + string Title { get; } + string? Description { get; } + string Url { get; } + string Icon { get; } + string Color { get; } +} diff --git a/source/HomeBook.Backend.Abstractions/Contracts/IUserProvider.cs b/source/HomeBook.Backend.Abstractions/Contracts/IUserProvider.cs index 80146b7b..a8c319a6 100644 --- a/source/HomeBook.Backend.Abstractions/Contracts/IUserProvider.cs +++ b/source/HomeBook.Backend.Abstractions/Contracts/IUserProvider.cs @@ -52,4 +52,13 @@ Task ContainsUserAsync(string username, Task UpdateAdminFlag(Guid userId, bool isAdmin, CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + Task GetUserByIdAsync(Guid userId, + CancellationToken cancellationToken); } diff --git a/source/HomeBook.Backend.Abstractions/HomeBook.Backend.Abstractions.csproj b/source/HomeBook.Backend.Abstractions/HomeBook.Backend.Abstractions.csproj index c9dd3514..cb6aaa84 100644 --- a/source/HomeBook.Backend.Abstractions/HomeBook.Backend.Abstractions.csproj +++ b/source/HomeBook.Backend.Abstractions/HomeBook.Backend.Abstractions.csproj @@ -7,7 +7,7 @@ - + diff --git a/source/HomeBook.Backend.Core.Account/HomeBook.Backend.Core.Account.csproj b/source/HomeBook.Backend.Core.Account/HomeBook.Backend.Core.Account.csproj index 598fd476..2f271a90 100644 --- a/source/HomeBook.Backend.Core.Account/HomeBook.Backend.Core.Account.csproj +++ b/source/HomeBook.Backend.Core.Account/HomeBook.Backend.Core.Account.csproj @@ -14,8 +14,8 @@ - - + + diff --git a/source/HomeBook.Backend.Core.Account/JwtService.cs b/source/HomeBook.Backend.Core.Account/JwtService.cs index 0d8a63a4..dbcf85e7 100644 --- a/source/HomeBook.Backend.Core.Account/JwtService.cs +++ b/source/HomeBook.Backend.Core.Account/JwtService.cs @@ -14,16 +14,20 @@ namespace HomeBook.Backend.Core.Account; /// public class JwtService(IConfiguration configuration) : IJwtService { - private readonly string _secretKey = configuration["Jwt:SecretKey"] ?? throw new InvalidOperationException("JWT SecretKey is required"); - private readonly string _issuer = configuration["Jwt:Issuer"] ?? "HomeBook"; - private readonly string _audience = configuration["Jwt:Audience"] ?? "HomeBook"; - private readonly int _expirationMinutes = int.Parse(configuration["Jwt:ExpirationMinutes"] ?? "60"); + private readonly string _secretKey = configuration["Jwt:SecretKey"] + ?? throw new InvalidOperationException("JWT SecretKey is required"); + + private readonly string _issuer = configuration["Jwt:Issuer"] + ?? "HomeBook"; + + private readonly string _audience = configuration["Jwt:Audience"] + ?? "HomeBook"; + + private readonly int _expirationMinutes = int.Parse(configuration["Jwt:ExpirationMinutes"] + ?? "60"); /// - public JwtTokenResult GenerateToken(Guid userId, string username) - { - return GenerateToken(userId, username, false); - } + public JwtTokenResult GenerateToken(Guid userId, string username) => GenerateToken(userId, username, false); /// public JwtTokenResult GenerateToken(Guid userId, string username, bool isAdmin) @@ -36,14 +40,18 @@ public JwtTokenResult GenerateToken(Guid userId, string username, bool isAdmin) [ new(ClaimTypes.NameIdentifier, userId.ToString()), new(ClaimTypes.Name, username), - new(ClaimTypes.Role, isAdmin ? "Admin" : "User"), - new("IsAdmin", isAdmin.ToString(), ClaimValueTypes.Boolean), new(JwtRegisteredClaimNames.Jti, Guid.NewGuid().ToString()), new(JwtRegisteredClaimNames.Iat, new DateTimeOffset(DateTime.UtcNow).ToUnixTimeSeconds().ToString(), ClaimValueTypes.Integer64) ]; + if (isAdmin) + { + claims = claims.Append(new Claim(ClaimTypes.Role, isAdmin ? "Admin" : "User")).ToArray(); + claims = claims.Append(new Claim("IsAdmin", isAdmin.ToString(), ClaimValueTypes.Boolean)).ToArray(); + } + JwtSecurityToken token = new( issuer: _issuer, audience: _audience, @@ -71,8 +79,8 @@ public bool ValidateToken(string token) { try { - var tokenHandler = new JwtSecurityTokenHandler(); - var key = Encoding.UTF8.GetBytes(_secretKey); + JwtSecurityTokenHandler tokenHandler = new(); + byte[] key = Encoding.UTF8.GetBytes(_secretKey); tokenHandler.ValidateToken(token, new TokenValidationParameters @@ -101,10 +109,10 @@ public bool ValidateToken(string token) { try { - var tokenHandler = new JwtSecurityTokenHandler(); - var key = Encoding.UTF8.GetBytes(_secretKey); + JwtSecurityTokenHandler tokenHandler = new(); + byte[] key = Encoding.UTF8.GetBytes(_secretKey); - var principal = tokenHandler.ValidateToken(token, + ClaimsPrincipal? principal = tokenHandler.ValidateToken(token, new TokenValidationParameters { ValidateIssuerSigningKey = true, @@ -118,7 +126,7 @@ public bool ValidateToken(string token) }, out SecurityToken validatedToken); - var userIdClaim = principal.FindFirst(ClaimTypes.NameIdentifier); + Claim? userIdClaim = principal.FindFirst(ClaimTypes.NameIdentifier); if (userIdClaim != null && Guid.TryParse(userIdClaim.Value, out Guid userId)) { return userId; diff --git a/source/HomeBook.Backend.Core.DataProvider/HomeBook.Backend.Core.DataProvider.csproj b/source/HomeBook.Backend.Core.DataProvider/HomeBook.Backend.Core.DataProvider.csproj index 869637ab..79de7b7b 100644 --- a/source/HomeBook.Backend.Core.DataProvider/HomeBook.Backend.Core.DataProvider.csproj +++ b/source/HomeBook.Backend.Core.DataProvider/HomeBook.Backend.Core.DataProvider.csproj @@ -8,11 +8,10 @@ - - + diff --git a/source/HomeBook.Backend.Core.DataProvider/UserProvider.cs b/source/HomeBook.Backend.Core.DataProvider/UserProvider.cs index 65643f0a..69c29690 100644 --- a/source/HomeBook.Backend.Core.DataProvider/UserProvider.cs +++ b/source/HomeBook.Backend.Core.DataProvider/UserProvider.cs @@ -82,4 +82,10 @@ public Task UpdateAdminFlag(Guid userId, }, cancellationToken); } + + /// + public async Task GetUserByIdAsync(Guid userId, + CancellationToken cancellationToken) => + (await userRepository.GetUserByIdAsync(userId, + cancellationToken))?.ToUserInfo(); } diff --git a/source/HomeBook.Backend.Core.Finances/Extensions/ServiceCollectionExtensions.cs b/source/HomeBook.Backend.Core.Finances/Extensions/ServiceCollectionExtensions.cs deleted file mode 100644 index 837ed346..00000000 --- a/source/HomeBook.Backend.Core.Finances/Extensions/ServiceCollectionExtensions.cs +++ /dev/null @@ -1,24 +0,0 @@ -using FluentValidation; -using HomeBook.Backend.Abstractions; -using HomeBook.Backend.Core.Finances.Contracts; -using HomeBook.Backend.Core.Finances.Validators; -using HomeBook.Backend.Data.Entities; -using Microsoft.Extensions.Configuration; -using Microsoft.Extensions.DependencyInjection; - -namespace HomeBook.Backend.Core.Finances.Extensions; - -public static class ServiceCollectionExtensions -{ - public static IServiceCollection AddBackendCoreFinances(this IServiceCollection services, - IConfiguration configuration, - InstanceStatus instanceStatus) - { - services.AddScoped(); - services.AddScoped(); - - services.AddSingleton, SavingGoalValidator>(); - - return services; - } -} diff --git a/source/HomeBook.Backend.Core.Finances/Mappings/SavingGoalMappings.cs b/source/HomeBook.Backend.Core.Finances/Mappings/SavingGoalMappings.cs deleted file mode 100644 index df644c70..00000000 --- a/source/HomeBook.Backend.Core.Finances/Mappings/SavingGoalMappings.cs +++ /dev/null @@ -1,21 +0,0 @@ -using HomeBook.Backend.Core.Finances.Models; - -namespace HomeBook.Backend.Core.Finances.Mappings; - -public static class SavingGoalMappings -{ - public static SavingGoalDto ToDto(this Data.Entities.SavingGoal savingGoal) - { - return new SavingGoalDto( - savingGoal.Id, - savingGoal.Name, - savingGoal.Color, - savingGoal.Icon ?? string.Empty, - savingGoal.TargetAmount, - savingGoal.CurrentAmount, - savingGoal.MonthlyPayment, - (DTOs.Enums.InterestRateOptions)savingGoal.InterestRateOption, - savingGoal.InterestRate, - savingGoal.TargetDate); - } -} diff --git a/source/HomeBook.Backend.Core.Kitchen/Contracts/IRecipesProvider.cs b/source/HomeBook.Backend.Core.Kitchen/Contracts/IRecipesProvider.cs deleted file mode 100644 index 6251ffd1..00000000 --- a/source/HomeBook.Backend.Core.Kitchen/Contracts/IRecipesProvider.cs +++ /dev/null @@ -1,53 +0,0 @@ -using HomeBook.Backend.Core.Kitchen.Models; - -namespace HomeBook.Backend.Core.Kitchen.Contracts; - -public interface IRecipesProvider -{ - /// - /// - /// - /// - /// - /// - Task GetRecipesAsync(string searchFilter, - CancellationToken cancellationToken); - - /// - /// - /// - /// - /// - /// - Task GetRecipeByIdAsync(Guid id, - CancellationToken cancellationToken); - - /// - /// - /// - /// - /// - /// - Task CreateAsync(string name, - CancellationToken cancellationToken); - - /// - /// - /// - /// - /// - /// - Task DeleteAsync(Guid id, - CancellationToken cancellationToken); - - /// - /// - /// - /// - /// - /// - /// - Task UpdateNameAsync(Guid id, - string name, - CancellationToken cancellationToken); -} diff --git a/source/HomeBook.Backend.Core.Kitchen/Extensions/ServiceCollectionExtensions.cs b/source/HomeBook.Backend.Core.Kitchen/Extensions/ServiceCollectionExtensions.cs deleted file mode 100644 index 8044ea1a..00000000 --- a/source/HomeBook.Backend.Core.Kitchen/Extensions/ServiceCollectionExtensions.cs +++ /dev/null @@ -1,18 +0,0 @@ -using HomeBook.Backend.Abstractions; -using HomeBook.Backend.Core.Kitchen.Contracts; -using Microsoft.Extensions.Configuration; -using Microsoft.Extensions.DependencyInjection; - -namespace HomeBook.Backend.Core.Kitchen.Extensions; - -public static class ServiceCollectionExtensions -{ - public static IServiceCollection AddBackendCoreKitchen(this IServiceCollection services, - IConfiguration configuration, - InstanceStatus instanceStatus) - { - services.AddScoped(); - - return services; - } -} diff --git a/source/HomeBook.Backend.Core.Kitchen/Mappings/RecipeMappings.cs b/source/HomeBook.Backend.Core.Kitchen/Mappings/RecipeMappings.cs deleted file mode 100644 index 5cb72c2a..00000000 --- a/source/HomeBook.Backend.Core.Kitchen/Mappings/RecipeMappings.cs +++ /dev/null @@ -1,14 +0,0 @@ -using HomeBook.Backend.Core.Kitchen.Models; - -namespace HomeBook.Backend.Core.Kitchen.Mappings; - -public static class RecipeMappings -{ - public static RecipeDto ToDto(this Data.Entities.Recipe recipe) - { - return new RecipeDto( - recipe.Id, - recipe.Name, - recipe.NormalizedName); - } -} diff --git a/source/HomeBook.Backend.Core.Kitchen/Models/RecipeDto.cs b/source/HomeBook.Backend.Core.Kitchen/Models/RecipeDto.cs deleted file mode 100644 index 2d682b16..00000000 --- a/source/HomeBook.Backend.Core.Kitchen/Models/RecipeDto.cs +++ /dev/null @@ -1,5 +0,0 @@ -namespace HomeBook.Backend.Core.Kitchen.Models; - -public record RecipeDto(Guid Id, - string Name, - string NormalizedName); diff --git a/source/HomeBook.Backend.Core.Kitchen/RecipesProvider.cs b/source/HomeBook.Backend.Core.Kitchen/RecipesProvider.cs deleted file mode 100644 index 2a2bbcb0..00000000 --- a/source/HomeBook.Backend.Core.Kitchen/RecipesProvider.cs +++ /dev/null @@ -1,78 +0,0 @@ -using HomeBook.Backend.Core.Kitchen.Contracts; -using HomeBook.Backend.Core.Kitchen.Mappings; -using HomeBook.Backend.Core.Kitchen.Models; -using HomeBook.Backend.Data.Contracts; -using HomeBook.Backend.Data.Entities; -using Microsoft.Extensions.Logging; - -namespace HomeBook.Backend.Core.Kitchen; - -/// -public class RecipesProvider( - ILogger logger, - IRecipesRepository recipesRepository) : IRecipesProvider -{ - /// - public async Task GetRecipesAsync(string searchFilter, - CancellationToken cancellationToken) - { - logger.LogInformation("Retrieving meals with search filter: {SearchFilter}", - searchFilter); - - IEnumerable recipeEntities = await recipesRepository.GetAsync(searchFilter, - cancellationToken); - RecipeDto[] recipes = recipeEntities - .Select(m => m.ToDto()) - .ToArray(); - - return recipes; - } - - /// - public async Task GetRecipeByIdAsync(Guid id, - CancellationToken cancellationToken) => - (await recipesRepository.GetByIdAsync(id, - cancellationToken))?.ToDto(); - - /// - public async Task CreateAsync(string name, - CancellationToken cancellationToken) - { - Recipe entity = new() - { - Name = name - }; - - // TODO: validator - - Guid entityId = await recipesRepository - .CreateOrUpdateAsync(entity, - cancellationToken); - return entityId; - } - - /// - public async Task DeleteAsync(Guid id, CancellationToken cancellationToken) => - await recipesRepository.DeleteAsync(id, - cancellationToken); - - /// - public async Task UpdateNameAsync(Guid id, - string name, - CancellationToken cancellationToken) - { - Recipe entity = await recipesRepository.GetByIdAsync(id, - cancellationToken) - ?? throw new KeyNotFoundException( - $"Recipe with id {id} not found"); - - entity.Name = name; - - // TODO: validator - - await recipesRepository - .CreateOrUpdateAsync(entity, - cancellationToken); - return; - } -} diff --git a/source/HomeBook.Backend.Core.Modules/HomeBook.Backend.Core.Modules.csproj b/source/HomeBook.Backend.Core.Modules/HomeBook.Backend.Core.Modules.csproj new file mode 100644 index 00000000..17b910f6 --- /dev/null +++ b/source/HomeBook.Backend.Core.Modules/HomeBook.Backend.Core.Modules.csproj @@ -0,0 +1,9 @@ + + + + net9.0 + enable + enable + + + diff --git a/source/HomeBook.Backend/OpenApi/Description.cs b/source/HomeBook.Backend.Core.Modules/OpenApi/Description.cs similarity index 88% rename from source/HomeBook.Backend/OpenApi/Description.cs rename to source/HomeBook.Backend.Core.Modules/OpenApi/Description.cs index 36b19f1b..b664c96d 100644 --- a/source/HomeBook.Backend/OpenApi/Description.cs +++ b/source/HomeBook.Backend.Core.Modules/OpenApi/Description.cs @@ -1,4 +1,4 @@ -namespace HomeBook.Backend.OpenApi; +namespace HomeBook.Backend.Core.Modules.OpenApi; public class Description(params string[] descriptionLines) { diff --git a/source/HomeBook.Backend/Utilities/ClaimsPrincipalUtilities.cs b/source/HomeBook.Backend.Core.Modules/Utilities/ClaimsPrincipalUtilities.cs similarity index 90% rename from source/HomeBook.Backend/Utilities/ClaimsPrincipalUtilities.cs rename to source/HomeBook.Backend.Core.Modules/Utilities/ClaimsPrincipalUtilities.cs index 9fbc2c17..54640c9d 100644 --- a/source/HomeBook.Backend/Utilities/ClaimsPrincipalUtilities.cs +++ b/source/HomeBook.Backend.Core.Modules/Utilities/ClaimsPrincipalUtilities.cs @@ -1,6 +1,6 @@ using System.Security.Claims; -namespace HomeBook.Backend.Utilities; +namespace HomeBook.Backend.Core.Modules.Utilities; public static class ClaimsPrincipalUtilities { diff --git a/source/HomeBook.Backend.Core.Search/Extensions/ServiceCollectionExtensions.cs b/source/HomeBook.Backend.Core.Search/Extensions/ServiceCollectionExtensions.cs new file mode 100644 index 00000000..c221e425 --- /dev/null +++ b/source/HomeBook.Backend.Core.Search/Extensions/ServiceCollectionExtensions.cs @@ -0,0 +1,20 @@ +using HomeBook.Backend.Abstractions; +using HomeBook.Backend.Abstractions.Contracts; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.DependencyInjection; + +namespace HomeBook.Backend.Core.Search.Extensions; + +public static class ServiceCollectionExtensions +{ + public static IServiceCollection AddBackendCoreSearch(this IServiceCollection services, + IConfiguration configuration, + InstanceStatus instanceStatus) + { + services.AddSingleton(); + services.AddSingleton(x => x.GetRequiredService()); + services.AddSingleton(x => x.GetRequiredService()); + + return services; + } +} diff --git a/source/HomeBook.Backend.Core.Search/HomeBook.Backend.Core.Search.csproj b/source/HomeBook.Backend.Core.Search/HomeBook.Backend.Core.Search.csproj new file mode 100644 index 00000000..bad3431d --- /dev/null +++ b/source/HomeBook.Backend.Core.Search/HomeBook.Backend.Core.Search.csproj @@ -0,0 +1,14 @@ + + + + net9.0 + enable + enable + + + + + + + + diff --git a/source/HomeBook.Backend.Core.Search/Models/SearchAggregationResult.cs b/source/HomeBook.Backend.Core.Search/Models/SearchAggregationResult.cs new file mode 100644 index 00000000..4dbcc9ac --- /dev/null +++ b/source/HomeBook.Backend.Core.Search/Models/SearchAggregationResult.cs @@ -0,0 +1,8 @@ +using HomeBook.Backend.Abstractions.Contracts; + +namespace HomeBook.Backend.Core.Search.Models; + +public record SearchAggregationResult( + string ModuleKey, + int TotalCount, + IEnumerable Items) : ISearchAggregationResult; diff --git a/source/HomeBook.Backend.Core.Search/SearchProvider.cs b/source/HomeBook.Backend.Core.Search/SearchProvider.cs new file mode 100644 index 00000000..8e09759e --- /dev/null +++ b/source/HomeBook.Backend.Core.Search/SearchProvider.cs @@ -0,0 +1,61 @@ +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.Core.Search.Models; +using HomeBook.Backend.Modules.Abstractions; +using Microsoft.Extensions.Logging; + +namespace HomeBook.Backend.Core.Search; + +/// +public class SearchProvider( + ILogger logger, + IEnumerable modules) : ISearchProvider +{ + /// + public async Task> SearchAsync(string query, + Guid userId, + CancellationToken cancellationToken = default) + { + IEnumerable> moduleSearchTasks = modules + .Select(async module => + { + try + { + logger.LogDebug("Requesting module {Module} for search query '{Query}'", + module.Name, + query); + + SearchResult result = await module.SearchAsync(query, + userId, + cancellationToken); + + logger.LogDebug("Module {Module} returned search result with {Count} items for query '{Query}'", + module.Name, + result.Items.Count(), + query); + + SearchAggregationResult moduleSearchResult = new(module.Key, + result.TotalCount, + result.Items); + return moduleSearchResult; + } + catch (OperationCanceledException) + { + // Task was cancelled, return null + return null; + } + catch (Exception err) + { + logger.LogError(err, + "Error while requesting module {Module} for search query '{Query}'", + module.Name, + query); + + return null; + } + }) + .Where(result => result is not null)!; + + IReadOnlyList searchResults = await Task.WhenAll(moduleSearchTasks.ToArray()); + return searchResults; + } +} diff --git a/source/HomeBook.Backend.Core.Search/SearchRegistrationFactory.cs b/source/HomeBook.Backend.Core.Search/SearchRegistrationFactory.cs new file mode 100644 index 00000000..cf19d337 --- /dev/null +++ b/source/HomeBook.Backend.Core.Search/SearchRegistrationFactory.cs @@ -0,0 +1,35 @@ +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.Modules.Abstractions; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Logging; + +namespace HomeBook.Backend.Core.Search; + +public class SearchRegistrationFactory() + : ISearchRegistrationFactory, + ISearchRegistrationInitiator +{ + private IServiceProvider _serviceProvider = null!; + private readonly List _registeredModules = []; + + public void AddModule(string moduleId) => _registeredModules.Add(moduleId); + + public void AddServiceProvider(IServiceProvider serviceProvider) => _serviceProvider = serviceProvider; + + public ISearchProvider CreateSearchProvider() + { + List modules = []; + foreach (string moduleId in _registeredModules) + { + IModule module = _serviceProvider.GetRequiredKeyedService(moduleId); + IBackendModuleSearchRegistrar registrar = (IBackendModuleSearchRegistrar)module; + modules.Add(registrar); + } + + ILoggerFactory loggerFactory = _serviceProvider.GetRequiredService(); + SearchProvider searchProvider = new(loggerFactory.CreateLogger(), + modules); + + return searchProvider; + } +} diff --git a/source/HomeBook.Backend.Core/HomeBook.Backend.Core.csproj b/source/HomeBook.Backend.Core/HomeBook.Backend.Core.csproj index 3f0fffac..6a1367cf 100644 --- a/source/HomeBook.Backend.Core/HomeBook.Backend.Core.csproj +++ b/source/HomeBook.Backend.Core/HomeBook.Backend.Core.csproj @@ -7,7 +7,7 @@ - + diff --git a/source/HomeBook.Backend.Core/StringNormalizer.cs b/source/HomeBook.Backend.Core/StringNormalizer.cs index 9d497c3b..49ba5930 100644 --- a/source/HomeBook.Backend.Core/StringNormalizer.cs +++ b/source/HomeBook.Backend.Core/StringNormalizer.cs @@ -9,7 +9,6 @@ public string Normalize(string input) if (string.IsNullOrWhiteSpace(input)) return string.Empty; - string normalized = input.Trim().ToLowerInvariant(); // Replace common diacritics with their base characters diff --git a/source/HomeBook.Backend.DTOs/HomeBook.Backend.DTOs.csproj b/source/HomeBook.Backend.DTOs/HomeBook.Backend.DTOs.csproj index 17b910f6..f61d0e78 100644 --- a/source/HomeBook.Backend.DTOs/HomeBook.Backend.DTOs.csproj +++ b/source/HomeBook.Backend.DTOs/HomeBook.Backend.DTOs.csproj @@ -6,4 +6,8 @@ enable + + + + diff --git a/source/HomeBook.Backend.DTOs/Requests/Kitchen/CreateRecipeRequest.cs b/source/HomeBook.Backend.DTOs/Requests/Kitchen/CreateRecipeRequest.cs deleted file mode 100644 index 10dfbe4c..00000000 --- a/source/HomeBook.Backend.DTOs/Requests/Kitchen/CreateRecipeRequest.cs +++ /dev/null @@ -1,3 +0,0 @@ -namespace HomeBook.Backend.DTOs.Requests.Kitchen; - -public record CreateRecipeRequest(string Name); diff --git a/source/HomeBook.Backend.DTOs/Responses/Kitchen/RecipeResponse.cs b/source/HomeBook.Backend.DTOs/Responses/Kitchen/RecipeResponse.cs deleted file mode 100644 index 679296b6..00000000 --- a/source/HomeBook.Backend.DTOs/Responses/Kitchen/RecipeResponse.cs +++ /dev/null @@ -1,9 +0,0 @@ -using System.Diagnostics; - -namespace HomeBook.Backend.DTOs.Responses.Kitchen; - -[DebuggerDisplay("{Name}")] -public record RecipeResponse( - Guid Id, - string Name, - string NormalizedName); diff --git a/source/HomeBook.Backend.DTOs/Responses/Search/SearchResponse.cs b/source/HomeBook.Backend.DTOs/Responses/Search/SearchResponse.cs new file mode 100644 index 00000000..dba2ec65 --- /dev/null +++ b/source/HomeBook.Backend.DTOs/Responses/Search/SearchResponse.cs @@ -0,0 +1,15 @@ +namespace HomeBook.Backend.DTOs.Responses.Search; + +public record SearchResponse(SearchModuleResponse[] SearchModuleResponses); + +public record SearchModuleResponse( + string ModuleKey, + int TotalCount, + IEnumerable Items); + +public record SearchItemResponse( + string Title, + string? Description, + string Url, + string Icon, + string Color); diff --git a/source/HomeBook.Backend.Data.Mysql/HomeBook.Backend.Data.Mysql.csproj b/source/HomeBook.Backend.Data.Mysql/HomeBook.Backend.Data.Mysql.csproj index 37fc593a..d70e322a 100644 --- a/source/HomeBook.Backend.Data.Mysql/HomeBook.Backend.Data.Mysql.csproj +++ b/source/HomeBook.Backend.Data.Mysql/HomeBook.Backend.Data.Mysql.csproj @@ -11,7 +11,7 @@ - + all runtime; build; native; contentfiles; analyzers; buildtransitive diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251121154310_AddBasicPropertiesToRecipe.Designer.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251121154310_AddBasicPropertiesToRecipe.Designer.cs new file mode 100644 index 00000000..e3707ccc --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251121154310_AddBasicPropertiesToRecipe.Designer.cs @@ -0,0 +1,275 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251121154310_AddBasicPropertiesToRecipe")] + partial class AddBasicPropertiesToRecipe + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.10") + .HasAnnotation("Relational:MaxIdentifierLength", 64); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("Ingredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("CaloriesKcal") + .HasColumnType("int"); + + b.Property("Description") + .HasColumnType("longtext"); + + b.Property("DurationMinutes") + .HasColumnType("int"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Servings") + .HasColumnType("int"); + + b.HasKey("Id"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("char(36)"); + + b.Property("IngredientId") + .HasColumnType("char(36)"); + + b.Property("Quantity") + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("varchar(7)"); + + b.Property("CurrentAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("Icon") + .HasColumnType("longtext"); + + b.Property("InterestRate") + .HasColumnType("decimal(18,2)"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("decimal(18,2)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("TargetAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("TargetDate") + .HasColumnType("datetime(6)"); + + b.Property("UserId") + .HasColumnType("char(36)"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("datetime(6)"); + + b.Property("Disabled") + .HasColumnType("datetime(6)"); + + b.Property("IsAdmin") + .HasColumnType("tinyint(1)"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("varchar(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("char(36)"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") + .WithMany("RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany() + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Ingredient"); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Navigation("RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251121154310_AddBasicPropertiesToRecipe.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251121154310_AddBasicPropertiesToRecipe.cs new file mode 100644 index 00000000..55c376b7 --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251121154310_AddBasicPropertiesToRecipe.cs @@ -0,0 +1,38 @@ +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + /// + public partial class AddBasicPropertiesToRecipe : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.AddColumn( + name: "Description", + table: "Recipes", + type: "longtext", + nullable: true); + + migrationBuilder.AddColumn( + name: "Servings", + table: "Recipes", + type: "int", + nullable: true); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropColumn( + name: "Description", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "Servings", + table: "Recipes"); + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251124002034_AddUserToRecipe.Designer.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251124002034_AddUserToRecipe.Designer.cs new file mode 100644 index 00000000..1f880caa --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251124002034_AddUserToRecipe.Designer.cs @@ -0,0 +1,289 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251124002034_AddUserToRecipe")] + partial class AddUserToRecipe + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.11") + .HasAnnotation("Relational:MaxIdentifierLength", 64); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("Ingredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("CaloriesKcal") + .HasColumnType("int"); + + b.Property("Description") + .HasColumnType("longtext"); + + b.Property("DurationMinutes") + .HasColumnType("int"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Servings") + .HasColumnType("int"); + + b.Property("UserId") + .HasColumnType("char(36)"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("char(36)"); + + b.Property("IngredientId") + .HasColumnType("char(36)"); + + b.Property("Quantity") + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("varchar(7)"); + + b.Property("CurrentAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("Icon") + .HasColumnType("longtext"); + + b.Property("InterestRate") + .HasColumnType("decimal(18,2)"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("decimal(18,2)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("TargetAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("TargetDate") + .HasColumnType("datetime(6)"); + + b.Property("UserId") + .HasColumnType("char(36)"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("datetime(6)"); + + b.Property("Disabled") + .HasColumnType("datetime(6)"); + + b.Property("IsAdmin") + .HasColumnType("tinyint(1)"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("varchar(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("char(36)"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") + .WithMany("RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany() + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Ingredient"); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Navigation("RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251124002034_AddUserToRecipe.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251124002034_AddUserToRecipe.cs new file mode 100644 index 00000000..1671c620 --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251124002034_AddUserToRecipe.cs @@ -0,0 +1,49 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + /// + public partial class AddUserToRecipe : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.AddColumn( + name: "UserId", + table: "Recipes", + type: "char(36)", + nullable: true); + + migrationBuilder.CreateIndex( + name: "IX_Recipes_UserId", + table: "Recipes", + column: "UserId"); + + migrationBuilder.AddForeignKey( + name: "FK_Recipes_Users_UserId", + table: "Recipes", + column: "UserId", + principalTable: "Users", + principalColumn: "Id"); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_Recipes_Users_UserId", + table: "Recipes"); + + migrationBuilder.DropIndex( + name: "IX_Recipes_UserId", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "UserId", + table: "Recipes"); + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251208213633_AddAdditionalDataToRecipe.Designer.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251208213633_AddAdditionalDataToRecipe.Designer.cs new file mode 100644 index 00000000..f9a67458 --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251208213633_AddAdditionalDataToRecipe.Designer.cs @@ -0,0 +1,301 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251208213633_AddAdditionalDataToRecipe")] + partial class AddAdditionalDataToRecipe + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.11") + .HasAnnotation("Relational:MaxIdentifierLength", 64); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("Ingredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("CaloriesKcal") + .HasColumnType("int"); + + b.Property("Comments") + .HasColumnType("longtext"); + + b.Property("Description") + .HasColumnType("longtext"); + + b.Property("DurationCookingMinutes") + .HasColumnType("int"); + + b.Property("DurationRestingMinutes") + .HasColumnType("int"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("int"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Servings") + .HasColumnType("int"); + + b.Property("Source") + .HasColumnType("longtext"); + + b.Property("UserId") + .HasColumnType("char(36)"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("char(36)"); + + b.Property("IngredientId") + .HasColumnType("char(36)"); + + b.Property("Quantity") + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("varchar(7)"); + + b.Property("CurrentAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("Icon") + .HasColumnType("longtext"); + + b.Property("InterestRate") + .HasColumnType("decimal(18,2)"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("decimal(18,2)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("TargetAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("TargetDate") + .HasColumnType("datetime(6)"); + + b.Property("UserId") + .HasColumnType("char(36)"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("datetime(6)"); + + b.Property("Disabled") + .HasColumnType("datetime(6)"); + + b.Property("IsAdmin") + .HasColumnType("tinyint(1)"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("varchar(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("char(36)"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") + .WithMany("RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany() + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Ingredient"); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Navigation("RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251208213633_AddAdditionalDataToRecipe.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251208213633_AddAdditionalDataToRecipe.cs new file mode 100644 index 00000000..5963b03c --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251208213633_AddAdditionalDataToRecipe.cs @@ -0,0 +1,68 @@ +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + /// + public partial class AddAdditionalDataToRecipe : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.RenameColumn( + name: "DurationMinutes", + table: "Recipes", + newName: "DurationWorkingMinutes"); + + migrationBuilder.AddColumn( + name: "Comments", + table: "Recipes", + type: "longtext", + nullable: true); + + migrationBuilder.AddColumn( + name: "DurationCookingMinutes", + table: "Recipes", + type: "int", + nullable: true); + + migrationBuilder.AddColumn( + name: "DurationRestingMinutes", + table: "Recipes", + type: "int", + nullable: true); + + migrationBuilder.AddColumn( + name: "Source", + table: "Recipes", + type: "longtext", + nullable: true); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropColumn( + name: "Comments", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "DurationCookingMinutes", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "DurationRestingMinutes", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "Source", + table: "Recipes"); + + migrationBuilder.RenameColumn( + name: "DurationWorkingMinutes", + table: "Recipes", + newName: "DurationMinutes"); + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251208232927_AddRecipeOptimizations.Designer.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251208232927_AddRecipeOptimizations.Designer.cs new file mode 100644 index 00000000..69590db4 --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251208232927_AddRecipeOptimizations.Designer.cs @@ -0,0 +1,345 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251208232927_AddRecipeOptimizations")] + partial class AddRecipeOptimizations + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.11") + .HasAnnotation("Relational:MaxIdentifierLength", 64); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("CaloriesKcal") + .HasColumnType("int"); + + b.Property("Comments") + .HasColumnType("longtext"); + + b.Property("Description") + .HasColumnType("longtext"); + + b.Property("DurationCookingMinutes") + .HasColumnType("int"); + + b.Property("DurationRestingMinutes") + .HasColumnType("int"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("int"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Servings") + .HasColumnType("int"); + + b.Property("Source") + .HasColumnType("longtext"); + + b.Property("UserId") + .HasColumnType("char(36)"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("char(36)"); + + b.Property("IngredientId") + .HasColumnType("char(36)"); + + b.Property("Quantity") + .HasColumnType("double"); + + b.Property("RecipeIngredientId") + .HasColumnType("char(36)"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("RecipeIngredientId"); + + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("varchar(500)"); + + b.Property("RecipeId") + .HasColumnType("char(36)"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("int"); + + b.HasKey("Id"); + + b.HasIndex("RecipeId"); + + b.ToTable("RecipeSteps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("varchar(7)"); + + b.Property("CurrentAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("Icon") + .HasColumnType("longtext"); + + b.Property("InterestRate") + .HasColumnType("decimal(18,2)"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("decimal(18,2)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("TargetAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("TargetDate") + .HasColumnType("datetime(6)"); + + b.Property("UserId") + .HasColumnType("char(36)"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("datetime(6)"); + + b.Property("Disabled") + .HasColumnType("datetime(6)"); + + b.Property("IsAdmin") + .HasColumnType("tinyint(1)"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("varchar(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("char(36)"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Recipe2RecipeIngredient") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") + .HasForeignKey("RecipeIngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Navigation("Recipe2RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251208232927_AddRecipeOptimizations.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251208232927_AddRecipeOptimizations.cs new file mode 100644 index 00000000..386d2310 --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251208232927_AddRecipeOptimizations.cs @@ -0,0 +1,235 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + /// + public partial class AddRecipeOptimizations : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_RecipeIngredients_Ingredients_IngredientId", + table: "RecipeIngredients"); + + migrationBuilder.DropForeignKey( + name: "FK_RecipeIngredients_Recipes_RecipeId", + table: "RecipeIngredients"); + + migrationBuilder.DropTable( + name: "Ingredients"); + + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients"); + + migrationBuilder.DropIndex( + name: "IX_RecipeIngredients_IngredientId", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "RecipeId", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "Quantity", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "Unit", + table: "RecipeIngredients"); + + migrationBuilder.RenameColumn( + name: "IngredientId", + table: "RecipeIngredients", + newName: "Id"); + + migrationBuilder.AddColumn( + name: "Name", + table: "RecipeIngredients", + type: "varchar(100)", + maxLength: 100, + nullable: false, + defaultValue: ""); + + migrationBuilder.AddColumn( + name: "NormalizedName", + table: "RecipeIngredients", + type: "varchar(100)", + maxLength: 100, + nullable: false, + defaultValue: ""); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients", + column: "Id"); + + migrationBuilder.CreateTable( + name: "Recipe2RecipeIngredient", + columns: table => new + { + RecipeId = table.Column(type: "char(36)", nullable: false), + IngredientId = table.Column(type: "char(36)", nullable: false), + RecipeIngredientId = table.Column(type: "char(36)", nullable: false), + Quantity = table.Column(type: "double", nullable: true), + Unit = table.Column(type: "varchar(20)", maxLength: 20, nullable: true) + }, + constraints: table => + { + table.PrimaryKey("PK_Recipe2RecipeIngredient", x => new { x.RecipeId, x.IngredientId }); + table.ForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_RecipeIngredientId", + column: x => x.RecipeIngredientId, + principalTable: "RecipeIngredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + table.ForeignKey( + name: "FK_Recipe2RecipeIngredient_Recipes_RecipeId", + column: x => x.RecipeId, + principalTable: "Recipes", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + }) + .Annotation("MySQL:Charset", "utf8mb4"); + + migrationBuilder.CreateTable( + name: "RecipeSteps", + columns: table => new + { + Id = table.Column(type: "char(36)", nullable: false), + RecipeId = table.Column(type: "char(36)", nullable: false), + Description = table.Column(type: "varchar(500)", maxLength: 500, nullable: false), + TimerDurationInSeconds = table.Column(type: "int", nullable: true) + }, + constraints: table => + { + table.PrimaryKey("PK_RecipeSteps", x => x.Id); + table.ForeignKey( + name: "FK_RecipeSteps_Recipes_RecipeId", + column: x => x.RecipeId, + principalTable: "Recipes", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + }) + .Annotation("MySQL:Charset", "utf8mb4"); + + migrationBuilder.CreateIndex( + name: "IX_RecipeIngredients_NormalizedName", + table: "RecipeIngredients", + column: "NormalizedName", + unique: true); + + migrationBuilder.CreateIndex( + name: "IX_Recipe2RecipeIngredient_RecipeIngredientId", + table: "Recipe2RecipeIngredient", + column: "RecipeIngredientId"); + + migrationBuilder.CreateIndex( + name: "IX_RecipeSteps_RecipeId", + table: "RecipeSteps", + column: "RecipeId"); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropTable( + name: "Recipe2RecipeIngredient"); + + migrationBuilder.DropTable( + name: "RecipeSteps"); + + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients"); + + migrationBuilder.DropIndex( + name: "IX_RecipeIngredients_NormalizedName", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "Name", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "NormalizedName", + table: "RecipeIngredients"); + + migrationBuilder.RenameColumn( + name: "Id", + table: "RecipeIngredients", + newName: "IngredientId"); + + migrationBuilder.AddColumn( + name: "RecipeId", + table: "RecipeIngredients", + type: "char(36)", + nullable: false, + defaultValue: new Guid("00000000-0000-0000-0000-000000000000")); + + migrationBuilder.AddColumn( + name: "Quantity", + table: "RecipeIngredients", + type: "varchar(50)", + maxLength: 50, + nullable: true); + + migrationBuilder.AddColumn( + name: "Unit", + table: "RecipeIngredients", + type: "varchar(20)", + maxLength: 20, + nullable: true); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients", + columns: new[] { "RecipeId", "IngredientId" }); + + migrationBuilder.CreateTable( + name: "Ingredients", + columns: table => new + { + Id = table.Column(type: "char(36)", nullable: false), + Name = table.Column(type: "varchar(100)", maxLength: 100, nullable: false), + NormalizedName = table.Column(type: "varchar(100)", maxLength: 100, nullable: false) + }, + constraints: table => + { + table.PrimaryKey("PK_Ingredients", x => x.Id); + }) + .Annotation("MySQL:Charset", "utf8mb4"); + + migrationBuilder.CreateIndex( + name: "IX_RecipeIngredients_IngredientId", + table: "RecipeIngredients", + column: "IngredientId"); + + migrationBuilder.CreateIndex( + name: "IX_Ingredients_NormalizedName", + table: "Ingredients", + column: "NormalizedName", + unique: true); + + migrationBuilder.AddForeignKey( + name: "FK_RecipeIngredients_Ingredients_IngredientId", + table: "RecipeIngredients", + column: "IngredientId", + principalTable: "Ingredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + + migrationBuilder.AddForeignKey( + name: "FK_RecipeIngredients_Recipes_RecipeId", + table: "RecipeIngredients", + column: "RecipeId", + principalTable: "Recipes", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251209132340_AddRecipeStepPosition.Designer.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251209132340_AddRecipeStepPosition.Designer.cs new file mode 100644 index 00000000..472de9cc --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251209132340_AddRecipeStepPosition.Designer.cs @@ -0,0 +1,342 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251209132340_AddRecipeStepPosition")] + partial class AddRecipeStepPosition + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.11") + .HasAnnotation("Relational:MaxIdentifierLength", 64); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("CaloriesKcal") + .HasColumnType("int"); + + b.Property("Comments") + .HasColumnType("longtext"); + + b.Property("Description") + .HasColumnType("longtext"); + + b.Property("DurationCookingMinutes") + .HasColumnType("int"); + + b.Property("DurationRestingMinutes") + .HasColumnType("int"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("int"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Servings") + .HasColumnType("int"); + + b.Property("Source") + .HasColumnType("longtext"); + + b.Property("UserId") + .HasColumnType("char(36)"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("char(36)"); + + b.Property("IngredientId") + .HasColumnType("char(36)"); + + b.Property("Quantity") + .HasColumnType("double"); + + b.Property("RecipeIngredientId") + .HasColumnType("char(36)"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("RecipeIngredientId"); + + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("RecipeId") + .HasColumnType("char(36)"); + + b.Property("Position") + .HasColumnType("int"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("varchar(500)"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("int"); + + b.HasKey("RecipeId", "Position"); + + b.ToTable("RecipeSteps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("varchar(7)"); + + b.Property("CurrentAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("Icon") + .HasColumnType("longtext"); + + b.Property("InterestRate") + .HasColumnType("decimal(18,2)"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("decimal(18,2)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("TargetAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("TargetDate") + .HasColumnType("datetime(6)"); + + b.Property("UserId") + .HasColumnType("char(36)"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("datetime(6)"); + + b.Property("Disabled") + .HasColumnType("datetime(6)"); + + b.Property("IsAdmin") + .HasColumnType("tinyint(1)"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("varchar(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("char(36)"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Recipe2RecipeIngredient") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") + .HasForeignKey("RecipeIngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Navigation("Recipe2RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251209132340_AddRecipeStepPosition.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251209132340_AddRecipeStepPosition.cs new file mode 100644 index 00000000..ea8deb61 --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251209132340_AddRecipeStepPosition.cs @@ -0,0 +1,68 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + /// + public partial class AddRecipeStepPosition : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps"); + + migrationBuilder.DropIndex( + name: "IX_RecipeSteps_RecipeId", + table: "RecipeSteps"); + + migrationBuilder.DropColumn( + name: "Id", + table: "RecipeSteps"); + + migrationBuilder.AddColumn( + name: "Position", + table: "RecipeSteps", + type: "int", + nullable: false, + defaultValue: 0); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps", + columns: new[] { "RecipeId", "Position" }); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps"); + + migrationBuilder.DropColumn( + name: "Position", + table: "RecipeSteps"); + + migrationBuilder.AddColumn( + name: "Id", + table: "RecipeSteps", + type: "char(36)", + nullable: false, + defaultValue: new Guid("00000000-0000-0000-0000-000000000000")); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps", + column: "Id"); + + migrationBuilder.CreateIndex( + name: "IX_RecipeSteps_RecipeId", + table: "RecipeSteps", + column: "RecipeId"); + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251213230426_FixRecipe2RecipeIngredient.Designer.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251213230426_FixRecipe2RecipeIngredient.Designer.cs new file mode 100644 index 00000000..3c56e183 --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251213230426_FixRecipe2RecipeIngredient.Designer.cs @@ -0,0 +1,339 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251213230426_FixRecipe2RecipeIngredient")] + partial class FixRecipe2RecipeIngredient + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.11") + .HasAnnotation("Relational:MaxIdentifierLength", 64); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("CaloriesKcal") + .HasColumnType("int"); + + b.Property("Comments") + .HasColumnType("longtext"); + + b.Property("Description") + .HasColumnType("longtext"); + + b.Property("DurationCookingMinutes") + .HasColumnType("int"); + + b.Property("DurationRestingMinutes") + .HasColumnType("int"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("int"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Servings") + .HasColumnType("int"); + + b.Property("Source") + .HasColumnType("longtext"); + + b.Property("UserId") + .HasColumnType("char(36)"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("char(36)"); + + b.Property("IngredientId") + .HasColumnType("char(36)"); + + b.Property("Quantity") + .HasColumnType("double"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("RecipeId") + .HasColumnType("char(36)"); + + b.Property("Position") + .HasColumnType("int"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("varchar(500)"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("int"); + + b.HasKey("RecipeId", "Position"); + + b.ToTable("RecipeSteps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("varchar(7)"); + + b.Property("CurrentAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("Icon") + .HasColumnType("longtext"); + + b.Property("InterestRate") + .HasColumnType("decimal(18,2)"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("decimal(18,2)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("TargetAmount") + .HasColumnType("decimal(18,2)"); + + b.Property("TargetDate") + .HasColumnType("datetime(6)"); + + b.Property("UserId") + .HasColumnType("char(36)"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("datetime(6)"); + + b.Property("Disabled") + .HasColumnType("datetime(6)"); + + b.Property("IsAdmin") + .HasColumnType("tinyint(1)"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("varchar(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("varchar(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("char(36)"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("varchar(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("longtext"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Recipe2RecipeIngredient") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Navigation("Recipe2RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/20251213230426_FixRecipe2RecipeIngredient.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/20251213230426_FixRecipe2RecipeIngredient.cs new file mode 100644 index 00000000..c41b0b65 --- /dev/null +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/20251213230426_FixRecipe2RecipeIngredient.cs @@ -0,0 +1,72 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Mysql.Migrations +{ + /// + public partial class FixRecipe2RecipeIngredient : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_RecipeIngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.DropIndex( + name: "IX_Recipe2RecipeIngredient_RecipeIngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.DropColumn( + name: "RecipeIngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.CreateIndex( + name: "IX_Recipe2RecipeIngredient_IngredientId", + table: "Recipe2RecipeIngredient", + column: "IngredientId"); + + migrationBuilder.AddForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_IngredientId", + table: "Recipe2RecipeIngredient", + column: "IngredientId", + principalTable: "RecipeIngredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_IngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.DropIndex( + name: "IX_Recipe2RecipeIngredient_IngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.AddColumn( + name: "RecipeIngredientId", + table: "Recipe2RecipeIngredient", + type: "char(36)", + nullable: false, + defaultValue: new Guid("00000000-0000-0000-0000-000000000000")); + + migrationBuilder.CreateIndex( + name: "IX_Recipe2RecipeIngredient_RecipeIngredientId", + table: "Recipe2RecipeIngredient", + column: "RecipeIngredientId"); + + migrationBuilder.AddForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_RecipeIngredientId", + table: "Recipe2RecipeIngredient", + column: "RecipeIngredientId", + principalTable: "RecipeIngredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + } + } +} diff --git a/source/HomeBook.Backend.Data.Mysql/Migrations/AppDbContextModelSnapshot.cs b/source/HomeBook.Backend.Data.Mysql/Migrations/AppDbContextModelSnapshot.cs index 06dc84e1..4908e6c1 100644 --- a/source/HomeBook.Backend.Data.Mysql/Migrations/AppDbContextModelSnapshot.cs +++ b/source/HomeBook.Backend.Data.Mysql/Migrations/AppDbContextModelSnapshot.cs @@ -16,7 +16,7 @@ protected override void BuildModel(ModelBuilder modelBuilder) { #pragma warning disable 612, 618 modelBuilder - .HasAnnotation("ProductVersion", "9.0.10") + .HasAnnotation("ProductVersion", "9.0.11") .HasAnnotation("Relational:MaxIdentifierLength", 64); modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => @@ -39,40 +39,28 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.ToTable("Configurations"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => { b.Property("Id") .ValueGeneratedOnAdd() .HasColumnType("char(36)"); - b.Property("Name") - .IsRequired() - .HasMaxLength(100) - .HasColumnType("varchar(100)"); - - b.Property("NormalizedName") - .IsRequired() - .HasMaxLength(100) - .HasColumnType("varchar(100)"); - - b.HasKey("Id"); + b.Property("CaloriesKcal") + .HasColumnType("int"); - b.HasIndex("NormalizedName") - .IsUnique(); + b.Property("Comments") + .HasColumnType("longtext"); - b.ToTable("Ingredients"); - }); + b.Property("Description") + .HasColumnType("longtext"); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => - { - b.Property("Id") - .ValueGeneratedOnAdd() - .HasColumnType("char(36)"); + b.Property("DurationCookingMinutes") + .HasColumnType("int"); - b.Property("CaloriesKcal") + b.Property("DurationRestingMinutes") .HasColumnType("int"); - b.Property("DurationMinutes") + b.Property("DurationWorkingMinutes") .HasColumnType("int"); b.Property("Name") @@ -85,12 +73,23 @@ protected override void BuildModel(ModelBuilder modelBuilder) .HasMaxLength(50) .HasColumnType("varchar(50)"); + b.Property("Servings") + .HasColumnType("int"); + + b.Property("Source") + .HasColumnType("longtext"); + + b.Property("UserId") + .HasColumnType("char(36)"); + b.HasKey("Id"); + b.HasIndex("UserId"); + b.ToTable("Recipes"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => { b.Property("RecipeId") .HasColumnType("char(36)"); @@ -98,9 +97,8 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.Property("IngredientId") .HasColumnType("char(36)"); - b.Property("Quantity") - .HasMaxLength(50) - .HasColumnType("varchar(50)"); + b.Property("Quantity") + .HasColumnType("double"); b.Property("Unit") .HasMaxLength(20) @@ -110,9 +108,54 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.HasIndex("IngredientId"); + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("char(36)"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("varchar(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + b.ToTable("RecipeIngredients"); }); + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("RecipeId") + .HasColumnType("char(36)"); + + b.Property("Position") + .HasColumnType("int"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("varchar(500)"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("int"); + + b.HasKey("RecipeId", "Position"); + + b.ToTable("RecipeSteps"); + }); + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => { b.Property("Id") @@ -215,21 +258,41 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.ToTable("UserPreferences"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => { - b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") - .WithMany("RecipeIngredients") + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") .HasForeignKey("IngredientId") .OnDelete(DeleteBehavior.Cascade) .IsRequired(); b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") - .WithMany() + .WithMany("Recipe2RecipeIngredient") .HasForeignKey("RecipeId") .OnDelete(DeleteBehavior.Cascade) .IsRequired(); - b.Navigation("Ingredient"); + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); b.Navigation("Recipe"); }); @@ -256,9 +319,16 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.Navigation("User"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => { - b.Navigation("RecipeIngredients"); + b.Navigation("Recipe2RecipeIngredients"); }); #pragma warning restore 612, 618 } diff --git a/source/HomeBook.Backend.Data.PostgreSql/HomeBook.Backend.Data.PostgreSql.csproj b/source/HomeBook.Backend.Data.PostgreSql/HomeBook.Backend.Data.PostgreSql.csproj index d130bf14..81d0bfdf 100644 --- a/source/HomeBook.Backend.Data.PostgreSql/HomeBook.Backend.Data.PostgreSql.csproj +++ b/source/HomeBook.Backend.Data.PostgreSql/HomeBook.Backend.Data.PostgreSql.csproj @@ -11,11 +11,11 @@ - + all runtime; build; native; contentfiles; analyzers; buildtransitive - + diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251121154304_AddBasicPropertiesToRecipe.Designer.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251121154304_AddBasicPropertiesToRecipe.Designer.cs new file mode 100644 index 00000000..389baf89 --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251121154304_AddBasicPropertiesToRecipe.Designer.cs @@ -0,0 +1,278 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; +using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251121154304_AddBasicPropertiesToRecipe")] + partial class AddBasicPropertiesToRecipe + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.10") + .HasAnnotation("Relational:MaxIdentifierLength", 63); + + NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("Ingredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("CaloriesKcal") + .HasColumnType("integer"); + + b.Property("Description") + .HasColumnType("text"); + + b.Property("DurationMinutes") + .HasColumnType("integer"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Servings") + .HasColumnType("integer"); + + b.HasKey("Id"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("uuid"); + + b.Property("IngredientId") + .HasColumnType("uuid"); + + b.Property("Quantity") + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("character varying(7)"); + + b.Property("CurrentAmount") + .HasColumnType("numeric"); + + b.Property("Icon") + .HasColumnType("text"); + + b.Property("InterestRate") + .HasColumnType("numeric"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("numeric"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("TargetAmount") + .HasColumnType("numeric"); + + b.Property("TargetDate") + .HasColumnType("timestamp with time zone"); + + b.Property("UserId") + .HasColumnType("uuid"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("timestamp with time zone"); + + b.Property("Disabled") + .HasColumnType("timestamp with time zone"); + + b.Property("IsAdmin") + .HasColumnType("boolean"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("character varying(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("uuid"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") + .WithMany("RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany() + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Ingredient"); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Navigation("RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251121154304_AddBasicPropertiesToRecipe.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251121154304_AddBasicPropertiesToRecipe.cs new file mode 100644 index 00000000..f3e376de --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251121154304_AddBasicPropertiesToRecipe.cs @@ -0,0 +1,38 @@ +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + /// + public partial class AddBasicPropertiesToRecipe : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.AddColumn( + name: "Description", + table: "Recipes", + type: "text", + nullable: true); + + migrationBuilder.AddColumn( + name: "Servings", + table: "Recipes", + type: "integer", + nullable: true); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropColumn( + name: "Description", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "Servings", + table: "Recipes"); + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251124002028_AddUserToRecipe.Designer.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251124002028_AddUserToRecipe.Designer.cs new file mode 100644 index 00000000..cbbc2821 --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251124002028_AddUserToRecipe.Designer.cs @@ -0,0 +1,292 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; +using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251124002028_AddUserToRecipe")] + partial class AddUserToRecipe + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.11") + .HasAnnotation("Relational:MaxIdentifierLength", 63); + + NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("Ingredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("CaloriesKcal") + .HasColumnType("integer"); + + b.Property("Description") + .HasColumnType("text"); + + b.Property("DurationMinutes") + .HasColumnType("integer"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Servings") + .HasColumnType("integer"); + + b.Property("UserId") + .HasColumnType("uuid"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("uuid"); + + b.Property("IngredientId") + .HasColumnType("uuid"); + + b.Property("Quantity") + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("character varying(7)"); + + b.Property("CurrentAmount") + .HasColumnType("numeric"); + + b.Property("Icon") + .HasColumnType("text"); + + b.Property("InterestRate") + .HasColumnType("numeric"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("numeric"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("TargetAmount") + .HasColumnType("numeric"); + + b.Property("TargetDate") + .HasColumnType("timestamp with time zone"); + + b.Property("UserId") + .HasColumnType("uuid"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("timestamp with time zone"); + + b.Property("Disabled") + .HasColumnType("timestamp with time zone"); + + b.Property("IsAdmin") + .HasColumnType("boolean"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("character varying(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("uuid"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") + .WithMany("RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany() + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Ingredient"); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Navigation("RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251124002028_AddUserToRecipe.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251124002028_AddUserToRecipe.cs new file mode 100644 index 00000000..2b1b6b1c --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251124002028_AddUserToRecipe.cs @@ -0,0 +1,49 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + /// + public partial class AddUserToRecipe : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.AddColumn( + name: "UserId", + table: "Recipes", + type: "uuid", + nullable: true); + + migrationBuilder.CreateIndex( + name: "IX_Recipes_UserId", + table: "Recipes", + column: "UserId"); + + migrationBuilder.AddForeignKey( + name: "FK_Recipes_Users_UserId", + table: "Recipes", + column: "UserId", + principalTable: "Users", + principalColumn: "Id"); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_Recipes_Users_UserId", + table: "Recipes"); + + migrationBuilder.DropIndex( + name: "IX_Recipes_UserId", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "UserId", + table: "Recipes"); + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208213656_AddAdditionalDataToRecipe.Designer.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208213656_AddAdditionalDataToRecipe.Designer.cs new file mode 100644 index 00000000..7618f24d --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208213656_AddAdditionalDataToRecipe.Designer.cs @@ -0,0 +1,304 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; +using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251208213656_AddAdditionalDataToRecipe")] + partial class AddAdditionalDataToRecipe + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.11") + .HasAnnotation("Relational:MaxIdentifierLength", 63); + + NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("Ingredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("CaloriesKcal") + .HasColumnType("integer"); + + b.Property("Comments") + .HasColumnType("text"); + + b.Property("Description") + .HasColumnType("text"); + + b.Property("DurationCookingMinutes") + .HasColumnType("integer"); + + b.Property("DurationRestingMinutes") + .HasColumnType("integer"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("integer"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Servings") + .HasColumnType("integer"); + + b.Property("Source") + .HasColumnType("text"); + + b.Property("UserId") + .HasColumnType("uuid"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("uuid"); + + b.Property("IngredientId") + .HasColumnType("uuid"); + + b.Property("Quantity") + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("character varying(7)"); + + b.Property("CurrentAmount") + .HasColumnType("numeric"); + + b.Property("Icon") + .HasColumnType("text"); + + b.Property("InterestRate") + .HasColumnType("numeric"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("numeric"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("TargetAmount") + .HasColumnType("numeric"); + + b.Property("TargetDate") + .HasColumnType("timestamp with time zone"); + + b.Property("UserId") + .HasColumnType("uuid"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("timestamp with time zone"); + + b.Property("Disabled") + .HasColumnType("timestamp with time zone"); + + b.Property("IsAdmin") + .HasColumnType("boolean"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("character varying(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("uuid"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") + .WithMany("RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany() + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Ingredient"); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Navigation("RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208213656_AddAdditionalDataToRecipe.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208213656_AddAdditionalDataToRecipe.cs new file mode 100644 index 00000000..a81d7e1f --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208213656_AddAdditionalDataToRecipe.cs @@ -0,0 +1,68 @@ +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + /// + public partial class AddAdditionalDataToRecipe : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.RenameColumn( + name: "DurationMinutes", + table: "Recipes", + newName: "DurationWorkingMinutes"); + + migrationBuilder.AddColumn( + name: "Comments", + table: "Recipes", + type: "text", + nullable: true); + + migrationBuilder.AddColumn( + name: "DurationCookingMinutes", + table: "Recipes", + type: "integer", + nullable: true); + + migrationBuilder.AddColumn( + name: "DurationRestingMinutes", + table: "Recipes", + type: "integer", + nullable: true); + + migrationBuilder.AddColumn( + name: "Source", + table: "Recipes", + type: "text", + nullable: true); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropColumn( + name: "Comments", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "DurationCookingMinutes", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "DurationRestingMinutes", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "Source", + table: "Recipes"); + + migrationBuilder.RenameColumn( + name: "DurationWorkingMinutes", + table: "Recipes", + newName: "DurationMinutes"); + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208232934_AddRecipeOptimizations.Designer.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208232934_AddRecipeOptimizations.Designer.cs new file mode 100644 index 00000000..f72b9960 --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208232934_AddRecipeOptimizations.Designer.cs @@ -0,0 +1,348 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; +using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251208232934_AddRecipeOptimizations")] + partial class AddRecipeOptimizations + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.11") + .HasAnnotation("Relational:MaxIdentifierLength", 63); + + NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("CaloriesKcal") + .HasColumnType("integer"); + + b.Property("Comments") + .HasColumnType("text"); + + b.Property("Description") + .HasColumnType("text"); + + b.Property("DurationCookingMinutes") + .HasColumnType("integer"); + + b.Property("DurationRestingMinutes") + .HasColumnType("integer"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("integer"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Servings") + .HasColumnType("integer"); + + b.Property("Source") + .HasColumnType("text"); + + b.Property("UserId") + .HasColumnType("uuid"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("uuid"); + + b.Property("IngredientId") + .HasColumnType("uuid"); + + b.Property("Quantity") + .HasColumnType("double precision"); + + b.Property("RecipeIngredientId") + .HasColumnType("uuid"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("RecipeIngredientId"); + + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("character varying(500)"); + + b.Property("RecipeId") + .HasColumnType("uuid"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("integer"); + + b.HasKey("Id"); + + b.HasIndex("RecipeId"); + + b.ToTable("RecipeSteps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("character varying(7)"); + + b.Property("CurrentAmount") + .HasColumnType("numeric"); + + b.Property("Icon") + .HasColumnType("text"); + + b.Property("InterestRate") + .HasColumnType("numeric"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("numeric"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("TargetAmount") + .HasColumnType("numeric"); + + b.Property("TargetDate") + .HasColumnType("timestamp with time zone"); + + b.Property("UserId") + .HasColumnType("uuid"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("timestamp with time zone"); + + b.Property("Disabled") + .HasColumnType("timestamp with time zone"); + + b.Property("IsAdmin") + .HasColumnType("boolean"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("character varying(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("uuid"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Recipe2RecipeIngredient") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") + .HasForeignKey("RecipeIngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Navigation("Recipe2RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208232934_AddRecipeOptimizations.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208232934_AddRecipeOptimizations.cs new file mode 100644 index 00000000..655127e2 --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251208232934_AddRecipeOptimizations.cs @@ -0,0 +1,232 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + /// + public partial class AddRecipeOptimizations : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_RecipeIngredients_Ingredients_IngredientId", + table: "RecipeIngredients"); + + migrationBuilder.DropForeignKey( + name: "FK_RecipeIngredients_Recipes_RecipeId", + table: "RecipeIngredients"); + + migrationBuilder.DropTable( + name: "Ingredients"); + + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients"); + + migrationBuilder.DropIndex( + name: "IX_RecipeIngredients_IngredientId", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "RecipeId", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "Quantity", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "Unit", + table: "RecipeIngredients"); + + migrationBuilder.RenameColumn( + name: "IngredientId", + table: "RecipeIngredients", + newName: "Id"); + + migrationBuilder.AddColumn( + name: "Name", + table: "RecipeIngredients", + type: "character varying(100)", + maxLength: 100, + nullable: false, + defaultValue: ""); + + migrationBuilder.AddColumn( + name: "NormalizedName", + table: "RecipeIngredients", + type: "character varying(100)", + maxLength: 100, + nullable: false, + defaultValue: ""); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients", + column: "Id"); + + migrationBuilder.CreateTable( + name: "Recipe2RecipeIngredient", + columns: table => new + { + RecipeId = table.Column(type: "uuid", nullable: false), + IngredientId = table.Column(type: "uuid", nullable: false), + RecipeIngredientId = table.Column(type: "uuid", nullable: false), + Quantity = table.Column(type: "double precision", nullable: true), + Unit = table.Column(type: "character varying(20)", maxLength: 20, nullable: true) + }, + constraints: table => + { + table.PrimaryKey("PK_Recipe2RecipeIngredient", x => new { x.RecipeId, x.IngredientId }); + table.ForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_RecipeIngredientId", + column: x => x.RecipeIngredientId, + principalTable: "RecipeIngredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + table.ForeignKey( + name: "FK_Recipe2RecipeIngredient_Recipes_RecipeId", + column: x => x.RecipeId, + principalTable: "Recipes", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + }); + + migrationBuilder.CreateTable( + name: "RecipeSteps", + columns: table => new + { + Id = table.Column(type: "uuid", nullable: false), + RecipeId = table.Column(type: "uuid", nullable: false), + Description = table.Column(type: "character varying(500)", maxLength: 500, nullable: false), + TimerDurationInSeconds = table.Column(type: "integer", nullable: true) + }, + constraints: table => + { + table.PrimaryKey("PK_RecipeSteps", x => x.Id); + table.ForeignKey( + name: "FK_RecipeSteps_Recipes_RecipeId", + column: x => x.RecipeId, + principalTable: "Recipes", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + }); + + migrationBuilder.CreateIndex( + name: "IX_RecipeIngredients_NormalizedName", + table: "RecipeIngredients", + column: "NormalizedName", + unique: true); + + migrationBuilder.CreateIndex( + name: "IX_Recipe2RecipeIngredient_RecipeIngredientId", + table: "Recipe2RecipeIngredient", + column: "RecipeIngredientId"); + + migrationBuilder.CreateIndex( + name: "IX_RecipeSteps_RecipeId", + table: "RecipeSteps", + column: "RecipeId"); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropTable( + name: "Recipe2RecipeIngredient"); + + migrationBuilder.DropTable( + name: "RecipeSteps"); + + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients"); + + migrationBuilder.DropIndex( + name: "IX_RecipeIngredients_NormalizedName", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "Name", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "NormalizedName", + table: "RecipeIngredients"); + + migrationBuilder.RenameColumn( + name: "Id", + table: "RecipeIngredients", + newName: "IngredientId"); + + migrationBuilder.AddColumn( + name: "RecipeId", + table: "RecipeIngredients", + type: "uuid", + nullable: false, + defaultValue: new Guid("00000000-0000-0000-0000-000000000000")); + + migrationBuilder.AddColumn( + name: "Quantity", + table: "RecipeIngredients", + type: "character varying(50)", + maxLength: 50, + nullable: true); + + migrationBuilder.AddColumn( + name: "Unit", + table: "RecipeIngredients", + type: "character varying(20)", + maxLength: 20, + nullable: true); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients", + columns: new[] { "RecipeId", "IngredientId" }); + + migrationBuilder.CreateTable( + name: "Ingredients", + columns: table => new + { + Id = table.Column(type: "uuid", nullable: false), + Name = table.Column(type: "character varying(100)", maxLength: 100, nullable: false), + NormalizedName = table.Column(type: "character varying(100)", maxLength: 100, nullable: false) + }, + constraints: table => + { + table.PrimaryKey("PK_Ingredients", x => x.Id); + }); + + migrationBuilder.CreateIndex( + name: "IX_RecipeIngredients_IngredientId", + table: "RecipeIngredients", + column: "IngredientId"); + + migrationBuilder.CreateIndex( + name: "IX_Ingredients_NormalizedName", + table: "Ingredients", + column: "NormalizedName", + unique: true); + + migrationBuilder.AddForeignKey( + name: "FK_RecipeIngredients_Ingredients_IngredientId", + table: "RecipeIngredients", + column: "IngredientId", + principalTable: "Ingredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + + migrationBuilder.AddForeignKey( + name: "FK_RecipeIngredients_Recipes_RecipeId", + table: "RecipeIngredients", + column: "RecipeId", + principalTable: "Recipes", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251209132345_AddRecipeStepPosition.Designer.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251209132345_AddRecipeStepPosition.Designer.cs new file mode 100644 index 00000000..d5b9519a --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251209132345_AddRecipeStepPosition.Designer.cs @@ -0,0 +1,345 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; +using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251209132345_AddRecipeStepPosition")] + partial class AddRecipeStepPosition + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.11") + .HasAnnotation("Relational:MaxIdentifierLength", 63); + + NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("CaloriesKcal") + .HasColumnType("integer"); + + b.Property("Comments") + .HasColumnType("text"); + + b.Property("Description") + .HasColumnType("text"); + + b.Property("DurationCookingMinutes") + .HasColumnType("integer"); + + b.Property("DurationRestingMinutes") + .HasColumnType("integer"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("integer"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Servings") + .HasColumnType("integer"); + + b.Property("Source") + .HasColumnType("text"); + + b.Property("UserId") + .HasColumnType("uuid"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("uuid"); + + b.Property("IngredientId") + .HasColumnType("uuid"); + + b.Property("Quantity") + .HasColumnType("double precision"); + + b.Property("RecipeIngredientId") + .HasColumnType("uuid"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("RecipeIngredientId"); + + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("RecipeId") + .HasColumnType("uuid"); + + b.Property("Position") + .HasColumnType("integer"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("character varying(500)"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("integer"); + + b.HasKey("RecipeId", "Position"); + + b.ToTable("RecipeSteps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("character varying(7)"); + + b.Property("CurrentAmount") + .HasColumnType("numeric"); + + b.Property("Icon") + .HasColumnType("text"); + + b.Property("InterestRate") + .HasColumnType("numeric"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("numeric"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("TargetAmount") + .HasColumnType("numeric"); + + b.Property("TargetDate") + .HasColumnType("timestamp with time zone"); + + b.Property("UserId") + .HasColumnType("uuid"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("timestamp with time zone"); + + b.Property("Disabled") + .HasColumnType("timestamp with time zone"); + + b.Property("IsAdmin") + .HasColumnType("boolean"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("character varying(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("uuid"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Recipe2RecipeIngredient") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") + .HasForeignKey("RecipeIngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Navigation("Recipe2RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251209132345_AddRecipeStepPosition.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251209132345_AddRecipeStepPosition.cs new file mode 100644 index 00000000..1e765020 --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251209132345_AddRecipeStepPosition.cs @@ -0,0 +1,68 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + /// + public partial class AddRecipeStepPosition : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps"); + + migrationBuilder.DropIndex( + name: "IX_RecipeSteps_RecipeId", + table: "RecipeSteps"); + + migrationBuilder.DropColumn( + name: "Id", + table: "RecipeSteps"); + + migrationBuilder.AddColumn( + name: "Position", + table: "RecipeSteps", + type: "integer", + nullable: false, + defaultValue: 0); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps", + columns: new[] { "RecipeId", "Position" }); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps"); + + migrationBuilder.DropColumn( + name: "Position", + table: "RecipeSteps"); + + migrationBuilder.AddColumn( + name: "Id", + table: "RecipeSteps", + type: "uuid", + nullable: false, + defaultValue: new Guid("00000000-0000-0000-0000-000000000000")); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps", + column: "Id"); + + migrationBuilder.CreateIndex( + name: "IX_RecipeSteps_RecipeId", + table: "RecipeSteps", + column: "RecipeId"); + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251213230432_FixRecipe2RecipeIngredient.Designer.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251213230432_FixRecipe2RecipeIngredient.Designer.cs new file mode 100644 index 00000000..46478965 --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251213230432_FixRecipe2RecipeIngredient.Designer.cs @@ -0,0 +1,342 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; +using Npgsql.EntityFrameworkCore.PostgreSQL.Metadata; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251213230432_FixRecipe2RecipeIngredient")] + partial class FixRecipe2RecipeIngredient + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder + .HasAnnotation("ProductVersion", "9.0.11") + .HasAnnotation("Relational:MaxIdentifierLength", 63); + + NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("CaloriesKcal") + .HasColumnType("integer"); + + b.Property("Comments") + .HasColumnType("text"); + + b.Property("Description") + .HasColumnType("text"); + + b.Property("DurationCookingMinutes") + .HasColumnType("integer"); + + b.Property("DurationRestingMinutes") + .HasColumnType("integer"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("integer"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Servings") + .HasColumnType("integer"); + + b.Property("Source") + .HasColumnType("text"); + + b.Property("UserId") + .HasColumnType("uuid"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("uuid"); + + b.Property("IngredientId") + .HasColumnType("uuid"); + + b.Property("Quantity") + .HasColumnType("double precision"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("RecipeId") + .HasColumnType("uuid"); + + b.Property("Position") + .HasColumnType("integer"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("character varying(500)"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("integer"); + + b.HasKey("RecipeId", "Position"); + + b.ToTable("RecipeSteps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("character varying(7)"); + + b.Property("CurrentAmount") + .HasColumnType("numeric"); + + b.Property("Icon") + .HasColumnType("text"); + + b.Property("InterestRate") + .HasColumnType("numeric"); + + b.Property("InterestRateOption") + .IsRequired() + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("numeric"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("TargetAmount") + .HasColumnType("numeric"); + + b.Property("TargetDate") + .HasColumnType("timestamp with time zone"); + + b.Property("UserId") + .HasColumnType("uuid"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("timestamp with time zone"); + + b.Property("Disabled") + .HasColumnType("timestamp with time zone"); + + b.Property("IsAdmin") + .HasColumnType("boolean"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("character varying(512)"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("character varying(20)"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("uuid"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("character varying(50)"); + + b.Property("Value") + .IsRequired() + .HasColumnType("text"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Recipe2RecipeIngredient") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Navigation("Recipe2RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251213230432_FixRecipe2RecipeIngredient.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251213230432_FixRecipe2RecipeIngredient.cs new file mode 100644 index 00000000..0d0e78cb --- /dev/null +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/20251213230432_FixRecipe2RecipeIngredient.cs @@ -0,0 +1,72 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.PostgreSql.Migrations +{ + /// + public partial class FixRecipe2RecipeIngredient : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_RecipeIngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.DropIndex( + name: "IX_Recipe2RecipeIngredient_RecipeIngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.DropColumn( + name: "RecipeIngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.CreateIndex( + name: "IX_Recipe2RecipeIngredient_IngredientId", + table: "Recipe2RecipeIngredient", + column: "IngredientId"); + + migrationBuilder.AddForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_IngredientId", + table: "Recipe2RecipeIngredient", + column: "IngredientId", + principalTable: "RecipeIngredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_IngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.DropIndex( + name: "IX_Recipe2RecipeIngredient_IngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.AddColumn( + name: "RecipeIngredientId", + table: "Recipe2RecipeIngredient", + type: "uuid", + nullable: false, + defaultValue: new Guid("00000000-0000-0000-0000-000000000000")); + + migrationBuilder.CreateIndex( + name: "IX_Recipe2RecipeIngredient_RecipeIngredientId", + table: "Recipe2RecipeIngredient", + column: "RecipeIngredientId"); + + migrationBuilder.AddForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_RecipeIngredientId", + table: "Recipe2RecipeIngredient", + column: "RecipeIngredientId", + principalTable: "RecipeIngredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + } + } +} diff --git a/source/HomeBook.Backend.Data.PostgreSql/Migrations/AppDbContextModelSnapshot.cs b/source/HomeBook.Backend.Data.PostgreSql/Migrations/AppDbContextModelSnapshot.cs index 894f2ecd..d639153f 100644 --- a/source/HomeBook.Backend.Data.PostgreSql/Migrations/AppDbContextModelSnapshot.cs +++ b/source/HomeBook.Backend.Data.PostgreSql/Migrations/AppDbContextModelSnapshot.cs @@ -17,7 +17,7 @@ protected override void BuildModel(ModelBuilder modelBuilder) { #pragma warning disable 612, 618 modelBuilder - .HasAnnotation("ProductVersion", "9.0.10") + .HasAnnotation("ProductVersion", "9.0.11") .HasAnnotation("Relational:MaxIdentifierLength", 63); NpgsqlModelBuilderExtensions.UseIdentityByDefaultColumns(modelBuilder); @@ -42,40 +42,28 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.ToTable("Configurations"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => { b.Property("Id") .ValueGeneratedOnAdd() .HasColumnType("uuid"); - b.Property("Name") - .IsRequired() - .HasMaxLength(100) - .HasColumnType("character varying(100)"); - - b.Property("NormalizedName") - .IsRequired() - .HasMaxLength(100) - .HasColumnType("character varying(100)"); - - b.HasKey("Id"); + b.Property("CaloriesKcal") + .HasColumnType("integer"); - b.HasIndex("NormalizedName") - .IsUnique(); + b.Property("Comments") + .HasColumnType("text"); - b.ToTable("Ingredients"); - }); + b.Property("Description") + .HasColumnType("text"); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => - { - b.Property("Id") - .ValueGeneratedOnAdd() - .HasColumnType("uuid"); + b.Property("DurationCookingMinutes") + .HasColumnType("integer"); - b.Property("CaloriesKcal") + b.Property("DurationRestingMinutes") .HasColumnType("integer"); - b.Property("DurationMinutes") + b.Property("DurationWorkingMinutes") .HasColumnType("integer"); b.Property("Name") @@ -88,12 +76,23 @@ protected override void BuildModel(ModelBuilder modelBuilder) .HasMaxLength(50) .HasColumnType("character varying(50)"); + b.Property("Servings") + .HasColumnType("integer"); + + b.Property("Source") + .HasColumnType("text"); + + b.Property("UserId") + .HasColumnType("uuid"); + b.HasKey("Id"); + b.HasIndex("UserId"); + b.ToTable("Recipes"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => { b.Property("RecipeId") .HasColumnType("uuid"); @@ -101,9 +100,8 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.Property("IngredientId") .HasColumnType("uuid"); - b.Property("Quantity") - .HasMaxLength(50) - .HasColumnType("character varying(50)"); + b.Property("Quantity") + .HasColumnType("double precision"); b.Property("Unit") .HasMaxLength(20) @@ -113,9 +111,54 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.HasIndex("IngredientId"); + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("uuid"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("character varying(100)"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + b.ToTable("RecipeIngredients"); }); + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("RecipeId") + .HasColumnType("uuid"); + + b.Property("Position") + .HasColumnType("integer"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("character varying(500)"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("integer"); + + b.HasKey("RecipeId", "Position"); + + b.ToTable("RecipeSteps"); + }); + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => { b.Property("Id") @@ -218,21 +261,41 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.ToTable("UserPreferences"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => { - b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") - .WithMany("RecipeIngredients") + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") .HasForeignKey("IngredientId") .OnDelete(DeleteBehavior.Cascade) .IsRequired(); b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") - .WithMany() + .WithMany("Recipe2RecipeIngredient") .HasForeignKey("RecipeId") .OnDelete(DeleteBehavior.Cascade) .IsRequired(); - b.Navigation("Ingredient"); + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); b.Navigation("Recipe"); }); @@ -259,9 +322,16 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.Navigation("User"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => { - b.Navigation("RecipeIngredients"); + b.Navigation("Recipe2RecipeIngredients"); }); #pragma warning restore 612, 618 } diff --git a/source/HomeBook.Backend.Data.Sqlite/HomeBook.Backend.Data.Sqlite.csproj b/source/HomeBook.Backend.Data.Sqlite/HomeBook.Backend.Data.Sqlite.csproj index 2c084795..dbc28e03 100644 --- a/source/HomeBook.Backend.Data.Sqlite/HomeBook.Backend.Data.Sqlite.csproj +++ b/source/HomeBook.Backend.Data.Sqlite/HomeBook.Backend.Data.Sqlite.csproj @@ -11,12 +11,12 @@ - + all runtime; build; native; contentfiles; analyzers; buildtransitive - - + + diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251121154255_AddBasicPropertiesToRecipe.Designer.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251121154255_AddBasicPropertiesToRecipe.Designer.cs new file mode 100644 index 00000000..dc32dddb --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251121154255_AddBasicPropertiesToRecipe.Designer.cs @@ -0,0 +1,272 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251121154255_AddBasicPropertiesToRecipe")] + partial class AddBasicPropertiesToRecipe + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder.HasAnnotation("ProductVersion", "9.0.10"); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("Ingredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("CaloriesKcal") + .HasColumnType("INTEGER"); + + b.Property("Description") + .HasColumnType("TEXT"); + + b.Property("DurationMinutes") + .HasColumnType("INTEGER"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Servings") + .HasColumnType("INTEGER"); + + b.HasKey("Id"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("TEXT"); + + b.Property("IngredientId") + .HasColumnType("TEXT"); + + b.Property("Quantity") + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("TEXT"); + + b.Property("CurrentAmount") + .HasColumnType("TEXT"); + + b.Property("Icon") + .HasColumnType("TEXT"); + + b.Property("InterestRate") + .HasColumnType("TEXT"); + + b.Property("InterestRateOption") + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("TargetAmount") + .HasColumnType("TEXT"); + + b.Property("TargetDate") + .HasColumnType("TEXT"); + + b.Property("UserId") + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Disabled") + .HasColumnType("TEXT"); + + b.Property("IsAdmin") + .HasColumnType("INTEGER"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("TEXT"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("TEXT"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") + .WithMany("RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany() + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Ingredient"); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Navigation("RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251121154255_AddBasicPropertiesToRecipe.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251121154255_AddBasicPropertiesToRecipe.cs new file mode 100644 index 00000000..5d5f513b --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251121154255_AddBasicPropertiesToRecipe.cs @@ -0,0 +1,38 @@ +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + /// + public partial class AddBasicPropertiesToRecipe : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.AddColumn( + name: "Description", + table: "Recipes", + type: "TEXT", + nullable: true); + + migrationBuilder.AddColumn( + name: "Servings", + table: "Recipes", + type: "INTEGER", + nullable: true); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropColumn( + name: "Description", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "Servings", + table: "Recipes"); + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251124002006_AddUserToRecipe.Designer.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251124002006_AddUserToRecipe.Designer.cs new file mode 100644 index 00000000..839f8fe1 --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251124002006_AddUserToRecipe.Designer.cs @@ -0,0 +1,286 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251124002006_AddUserToRecipe")] + partial class AddUserToRecipe + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder.HasAnnotation("ProductVersion", "9.0.11"); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("Ingredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("CaloriesKcal") + .HasColumnType("INTEGER"); + + b.Property("Description") + .HasColumnType("TEXT"); + + b.Property("DurationMinutes") + .HasColumnType("INTEGER"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Servings") + .HasColumnType("INTEGER"); + + b.Property("UserId") + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("TEXT"); + + b.Property("IngredientId") + .HasColumnType("TEXT"); + + b.Property("Quantity") + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("TEXT"); + + b.Property("CurrentAmount") + .HasColumnType("TEXT"); + + b.Property("Icon") + .HasColumnType("TEXT"); + + b.Property("InterestRate") + .HasColumnType("TEXT"); + + b.Property("InterestRateOption") + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("TargetAmount") + .HasColumnType("TEXT"); + + b.Property("TargetDate") + .HasColumnType("TEXT"); + + b.Property("UserId") + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Disabled") + .HasColumnType("TEXT"); + + b.Property("IsAdmin") + .HasColumnType("INTEGER"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("TEXT"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("TEXT"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") + .WithMany("RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany() + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Ingredient"); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Navigation("RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251124002006_AddUserToRecipe.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251124002006_AddUserToRecipe.cs new file mode 100644 index 00000000..0f6885a1 --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251124002006_AddUserToRecipe.cs @@ -0,0 +1,49 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + /// + public partial class AddUserToRecipe : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.AddColumn( + name: "UserId", + table: "Recipes", + type: "TEXT", + nullable: true); + + migrationBuilder.CreateIndex( + name: "IX_Recipes_UserId", + table: "Recipes", + column: "UserId"); + + migrationBuilder.AddForeignKey( + name: "FK_Recipes_Users_UserId", + table: "Recipes", + column: "UserId", + principalTable: "Users", + principalColumn: "Id"); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_Recipes_Users_UserId", + table: "Recipes"); + + migrationBuilder.DropIndex( + name: "IX_Recipes_UserId", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "UserId", + table: "Recipes"); + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208213701_AddAdditionalDataToRecipe.Designer.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208213701_AddAdditionalDataToRecipe.Designer.cs new file mode 100644 index 00000000..66486eaf --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208213701_AddAdditionalDataToRecipe.Designer.cs @@ -0,0 +1,298 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251208213701_AddAdditionalDataToRecipe")] + partial class AddAdditionalDataToRecipe + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder.HasAnnotation("ProductVersion", "9.0.11"); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("Ingredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("CaloriesKcal") + .HasColumnType("INTEGER"); + + b.Property("Comments") + .HasColumnType("TEXT"); + + b.Property("Description") + .HasColumnType("TEXT"); + + b.Property("DurationCookingMinutes") + .HasColumnType("INTEGER"); + + b.Property("DurationRestingMinutes") + .HasColumnType("INTEGER"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("INTEGER"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Servings") + .HasColumnType("INTEGER"); + + b.Property("Source") + .HasColumnType("TEXT"); + + b.Property("UserId") + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("TEXT"); + + b.Property("IngredientId") + .HasColumnType("TEXT"); + + b.Property("Quantity") + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("TEXT"); + + b.Property("CurrentAmount") + .HasColumnType("TEXT"); + + b.Property("Icon") + .HasColumnType("TEXT"); + + b.Property("InterestRate") + .HasColumnType("TEXT"); + + b.Property("InterestRateOption") + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("TargetAmount") + .HasColumnType("TEXT"); + + b.Property("TargetDate") + .HasColumnType("TEXT"); + + b.Property("UserId") + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Disabled") + .HasColumnType("TEXT"); + + b.Property("IsAdmin") + .HasColumnType("INTEGER"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("TEXT"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("TEXT"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") + .WithMany("RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany() + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Ingredient"); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + { + b.Navigation("RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208213701_AddAdditionalDataToRecipe.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208213701_AddAdditionalDataToRecipe.cs new file mode 100644 index 00000000..97eb3175 --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208213701_AddAdditionalDataToRecipe.cs @@ -0,0 +1,68 @@ +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + /// + public partial class AddAdditionalDataToRecipe : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.RenameColumn( + name: "DurationMinutes", + table: "Recipes", + newName: "DurationWorkingMinutes"); + + migrationBuilder.AddColumn( + name: "Comments", + table: "Recipes", + type: "TEXT", + nullable: true); + + migrationBuilder.AddColumn( + name: "DurationCookingMinutes", + table: "Recipes", + type: "INTEGER", + nullable: true); + + migrationBuilder.AddColumn( + name: "DurationRestingMinutes", + table: "Recipes", + type: "INTEGER", + nullable: true); + + migrationBuilder.AddColumn( + name: "Source", + table: "Recipes", + type: "TEXT", + nullable: true); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropColumn( + name: "Comments", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "DurationCookingMinutes", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "DurationRestingMinutes", + table: "Recipes"); + + migrationBuilder.DropColumn( + name: "Source", + table: "Recipes"); + + migrationBuilder.RenameColumn( + name: "DurationWorkingMinutes", + table: "Recipes", + newName: "DurationMinutes"); + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208232940_AddRecipeOptimizations.Designer.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208232940_AddRecipeOptimizations.Designer.cs new file mode 100644 index 00000000..5156d865 --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208232940_AddRecipeOptimizations.Designer.cs @@ -0,0 +1,342 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251208232940_AddRecipeOptimizations")] + partial class AddRecipeOptimizations + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder.HasAnnotation("ProductVersion", "9.0.11"); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("CaloriesKcal") + .HasColumnType("INTEGER"); + + b.Property("Comments") + .HasColumnType("TEXT"); + + b.Property("Description") + .HasColumnType("TEXT"); + + b.Property("DurationCookingMinutes") + .HasColumnType("INTEGER"); + + b.Property("DurationRestingMinutes") + .HasColumnType("INTEGER"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("INTEGER"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Servings") + .HasColumnType("INTEGER"); + + b.Property("Source") + .HasColumnType("TEXT"); + + b.Property("UserId") + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("TEXT"); + + b.Property("IngredientId") + .HasColumnType("TEXT"); + + b.Property("Quantity") + .HasColumnType("REAL"); + + b.Property("RecipeIngredientId") + .HasColumnType("TEXT"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("RecipeIngredientId"); + + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("TEXT"); + + b.Property("RecipeId") + .HasColumnType("TEXT"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("INTEGER"); + + b.HasKey("Id"); + + b.HasIndex("RecipeId"); + + b.ToTable("RecipeSteps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("TEXT"); + + b.Property("CurrentAmount") + .HasColumnType("TEXT"); + + b.Property("Icon") + .HasColumnType("TEXT"); + + b.Property("InterestRate") + .HasColumnType("TEXT"); + + b.Property("InterestRateOption") + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("TargetAmount") + .HasColumnType("TEXT"); + + b.Property("TargetDate") + .HasColumnType("TEXT"); + + b.Property("UserId") + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Disabled") + .HasColumnType("TEXT"); + + b.Property("IsAdmin") + .HasColumnType("INTEGER"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("TEXT"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("TEXT"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Recipe2RecipeIngredient") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") + .HasForeignKey("RecipeIngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Navigation("Recipe2RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208232940_AddRecipeOptimizations.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208232940_AddRecipeOptimizations.cs new file mode 100644 index 00000000..36a40ec0 --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251208232940_AddRecipeOptimizations.cs @@ -0,0 +1,232 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + /// + public partial class AddRecipeOptimizations : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_RecipeIngredients_Ingredients_IngredientId", + table: "RecipeIngredients"); + + migrationBuilder.DropForeignKey( + name: "FK_RecipeIngredients_Recipes_RecipeId", + table: "RecipeIngredients"); + + migrationBuilder.DropTable( + name: "Ingredients"); + + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients"); + + migrationBuilder.DropIndex( + name: "IX_RecipeIngredients_IngredientId", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "RecipeId", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "Quantity", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "Unit", + table: "RecipeIngredients"); + + migrationBuilder.RenameColumn( + name: "IngredientId", + table: "RecipeIngredients", + newName: "Id"); + + migrationBuilder.AddColumn( + name: "Name", + table: "RecipeIngredients", + type: "TEXT", + maxLength: 100, + nullable: false, + defaultValue: ""); + + migrationBuilder.AddColumn( + name: "NormalizedName", + table: "RecipeIngredients", + type: "TEXT", + maxLength: 100, + nullable: false, + defaultValue: ""); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients", + column: "Id"); + + migrationBuilder.CreateTable( + name: "Recipe2RecipeIngredient", + columns: table => new + { + RecipeId = table.Column(type: "TEXT", nullable: false), + IngredientId = table.Column(type: "TEXT", nullable: false), + RecipeIngredientId = table.Column(type: "TEXT", nullable: false), + Quantity = table.Column(type: "REAL", nullable: true), + Unit = table.Column(type: "TEXT", maxLength: 20, nullable: true) + }, + constraints: table => + { + table.PrimaryKey("PK_Recipe2RecipeIngredient", x => new { x.RecipeId, x.IngredientId }); + table.ForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_RecipeIngredientId", + column: x => x.RecipeIngredientId, + principalTable: "RecipeIngredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + table.ForeignKey( + name: "FK_Recipe2RecipeIngredient_Recipes_RecipeId", + column: x => x.RecipeId, + principalTable: "Recipes", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + }); + + migrationBuilder.CreateTable( + name: "RecipeSteps", + columns: table => new + { + Id = table.Column(type: "TEXT", nullable: false), + RecipeId = table.Column(type: "TEXT", nullable: false), + Description = table.Column(type: "TEXT", maxLength: 500, nullable: false), + TimerDurationInSeconds = table.Column(type: "INTEGER", nullable: true) + }, + constraints: table => + { + table.PrimaryKey("PK_RecipeSteps", x => x.Id); + table.ForeignKey( + name: "FK_RecipeSteps_Recipes_RecipeId", + column: x => x.RecipeId, + principalTable: "Recipes", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + }); + + migrationBuilder.CreateIndex( + name: "IX_RecipeIngredients_NormalizedName", + table: "RecipeIngredients", + column: "NormalizedName", + unique: true); + + migrationBuilder.CreateIndex( + name: "IX_Recipe2RecipeIngredient_RecipeIngredientId", + table: "Recipe2RecipeIngredient", + column: "RecipeIngredientId"); + + migrationBuilder.CreateIndex( + name: "IX_RecipeSteps_RecipeId", + table: "RecipeSteps", + column: "RecipeId"); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropTable( + name: "Recipe2RecipeIngredient"); + + migrationBuilder.DropTable( + name: "RecipeSteps"); + + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients"); + + migrationBuilder.DropIndex( + name: "IX_RecipeIngredients_NormalizedName", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "Name", + table: "RecipeIngredients"); + + migrationBuilder.DropColumn( + name: "NormalizedName", + table: "RecipeIngredients"); + + migrationBuilder.RenameColumn( + name: "Id", + table: "RecipeIngredients", + newName: "IngredientId"); + + migrationBuilder.AddColumn( + name: "RecipeId", + table: "RecipeIngredients", + type: "TEXT", + nullable: false, + defaultValue: new Guid("00000000-0000-0000-0000-000000000000")); + + migrationBuilder.AddColumn( + name: "Quantity", + table: "RecipeIngredients", + type: "TEXT", + maxLength: 50, + nullable: true); + + migrationBuilder.AddColumn( + name: "Unit", + table: "RecipeIngredients", + type: "TEXT", + maxLength: 20, + nullable: true); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeIngredients", + table: "RecipeIngredients", + columns: new[] { "RecipeId", "IngredientId" }); + + migrationBuilder.CreateTable( + name: "Ingredients", + columns: table => new + { + Id = table.Column(type: "TEXT", nullable: false), + Name = table.Column(type: "TEXT", maxLength: 100, nullable: false), + NormalizedName = table.Column(type: "TEXT", maxLength: 100, nullable: false) + }, + constraints: table => + { + table.PrimaryKey("PK_Ingredients", x => x.Id); + }); + + migrationBuilder.CreateIndex( + name: "IX_RecipeIngredients_IngredientId", + table: "RecipeIngredients", + column: "IngredientId"); + + migrationBuilder.CreateIndex( + name: "IX_Ingredients_NormalizedName", + table: "Ingredients", + column: "NormalizedName", + unique: true); + + migrationBuilder.AddForeignKey( + name: "FK_RecipeIngredients_Ingredients_IngredientId", + table: "RecipeIngredients", + column: "IngredientId", + principalTable: "Ingredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + + migrationBuilder.AddForeignKey( + name: "FK_RecipeIngredients_Recipes_RecipeId", + table: "RecipeIngredients", + column: "RecipeId", + principalTable: "Recipes", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251209132351_AddRecipeStepPosition.Designer.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251209132351_AddRecipeStepPosition.Designer.cs new file mode 100644 index 00000000..beac973c --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251209132351_AddRecipeStepPosition.Designer.cs @@ -0,0 +1,339 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251209132351_AddRecipeStepPosition")] + partial class AddRecipeStepPosition + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder.HasAnnotation("ProductVersion", "9.0.11"); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("CaloriesKcal") + .HasColumnType("INTEGER"); + + b.Property("Comments") + .HasColumnType("TEXT"); + + b.Property("Description") + .HasColumnType("TEXT"); + + b.Property("DurationCookingMinutes") + .HasColumnType("INTEGER"); + + b.Property("DurationRestingMinutes") + .HasColumnType("INTEGER"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("INTEGER"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Servings") + .HasColumnType("INTEGER"); + + b.Property("Source") + .HasColumnType("TEXT"); + + b.Property("UserId") + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("TEXT"); + + b.Property("IngredientId") + .HasColumnType("TEXT"); + + b.Property("Quantity") + .HasColumnType("REAL"); + + b.Property("RecipeIngredientId") + .HasColumnType("TEXT"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("RecipeIngredientId"); + + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("RecipeId") + .HasColumnType("TEXT"); + + b.Property("Position") + .HasColumnType("INTEGER"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("TEXT"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("INTEGER"); + + b.HasKey("RecipeId", "Position"); + + b.ToTable("RecipeSteps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("TEXT"); + + b.Property("CurrentAmount") + .HasColumnType("TEXT"); + + b.Property("Icon") + .HasColumnType("TEXT"); + + b.Property("InterestRate") + .HasColumnType("TEXT"); + + b.Property("InterestRateOption") + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("TargetAmount") + .HasColumnType("TEXT"); + + b.Property("TargetDate") + .HasColumnType("TEXT"); + + b.Property("UserId") + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Disabled") + .HasColumnType("TEXT"); + + b.Property("IsAdmin") + .HasColumnType("INTEGER"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("TEXT"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("TEXT"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Recipe2RecipeIngredient") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") + .HasForeignKey("RecipeIngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Navigation("Recipe2RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251209132351_AddRecipeStepPosition.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251209132351_AddRecipeStepPosition.cs new file mode 100644 index 00000000..b976444d --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251209132351_AddRecipeStepPosition.cs @@ -0,0 +1,68 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + /// + public partial class AddRecipeStepPosition : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps"); + + migrationBuilder.DropIndex( + name: "IX_RecipeSteps_RecipeId", + table: "RecipeSteps"); + + migrationBuilder.DropColumn( + name: "Id", + table: "RecipeSteps"); + + migrationBuilder.AddColumn( + name: "Position", + table: "RecipeSteps", + type: "INTEGER", + nullable: false, + defaultValue: 0); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps", + columns: new[] { "RecipeId", "Position" }); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps"); + + migrationBuilder.DropColumn( + name: "Position", + table: "RecipeSteps"); + + migrationBuilder.AddColumn( + name: "Id", + table: "RecipeSteps", + type: "TEXT", + nullable: false, + defaultValue: new Guid("00000000-0000-0000-0000-000000000000")); + + migrationBuilder.AddPrimaryKey( + name: "PK_RecipeSteps", + table: "RecipeSteps", + column: "Id"); + + migrationBuilder.CreateIndex( + name: "IX_RecipeSteps_RecipeId", + table: "RecipeSteps", + column: "RecipeId"); + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251213230408_FixRecipe2RecipeIngredient.Designer.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251213230408_FixRecipe2RecipeIngredient.Designer.cs new file mode 100644 index 00000000..4070f735 --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251213230408_FixRecipe2RecipeIngredient.Designer.cs @@ -0,0 +1,336 @@ +// +using System; +using HomeBook.Backend.Data; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Infrastructure; +using Microsoft.EntityFrameworkCore.Migrations; +using Microsoft.EntityFrameworkCore.Storage.ValueConversion; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + [DbContext(typeof(AppDbContext))] + [Migration("20251213230408_FixRecipe2RecipeIngredient")] + partial class FixRecipe2RecipeIngredient + { + /// + protected override void BuildTargetModel(ModelBuilder modelBuilder) + { +#pragma warning disable 612, 618 + modelBuilder.HasAnnotation("ProductVersion", "9.0.11"); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Key") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Configurations"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("CaloriesKcal") + .HasColumnType("INTEGER"); + + b.Property("Comments") + .HasColumnType("TEXT"); + + b.Property("Description") + .HasColumnType("TEXT"); + + b.Property("DurationCookingMinutes") + .HasColumnType("INTEGER"); + + b.Property("DurationRestingMinutes") + .HasColumnType("INTEGER"); + + b.Property("DurationWorkingMinutes") + .HasColumnType("INTEGER"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Servings") + .HasColumnType("INTEGER"); + + b.Property("Source") + .HasColumnType("TEXT"); + + b.Property("UserId") + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("Recipes"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.Property("RecipeId") + .HasColumnType("TEXT"); + + b.Property("IngredientId") + .HasColumnType("TEXT"); + + b.Property("Quantity") + .HasColumnType("REAL"); + + b.Property("Unit") + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("RecipeId", "IngredientId"); + + b.HasIndex("IngredientId"); + + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + + b.ToTable("RecipeIngredients"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("RecipeId") + .HasColumnType("TEXT"); + + b.Property("Position") + .HasColumnType("INTEGER"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("TEXT"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("INTEGER"); + + b.HasKey("RecipeId", "Position"); + + b.ToTable("RecipeSteps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Color") + .IsRequired() + .HasMaxLength(7) + .HasColumnType("TEXT"); + + b.Property("CurrentAmount") + .HasColumnType("TEXT"); + + b.Property("Icon") + .HasColumnType("TEXT"); + + b.Property("InterestRate") + .HasColumnType("TEXT"); + + b.Property("InterestRateOption") + .HasColumnType("text"); + + b.Property("MonthlyPayment") + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("TargetAmount") + .HasColumnType("TEXT"); + + b.Property("TargetDate") + .HasColumnType("TEXT"); + + b.Property("UserId") + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("UserId"); + + b.ToTable("SavingGoals"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.User", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Created") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Disabled") + .HasColumnType("TEXT"); + + b.Property("IsAdmin") + .HasColumnType("INTEGER"); + + b.Property("PasswordHash") + .IsRequired() + .HasMaxLength(512) + .HasColumnType("TEXT"); + + b.Property("PasswordHashType") + .IsRequired() + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Username") + .IsRequired() + .HasMaxLength(20) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.ToTable("Users"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.Property("UserId") + .HasColumnType("TEXT"); + + b.Property("Key") + .HasMaxLength(50) + .HasColumnType("TEXT"); + + b.Property("Value") + .IsRequired() + .HasColumnType("TEXT"); + + b.HasKey("UserId", "Key"); + + b.ToTable("UserPreferences"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") + .HasForeignKey("IngredientId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Recipe2RecipeIngredient") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("Recipe"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.UserPreference", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Navigation("Recipe2RecipeIngredients"); + }); +#pragma warning restore 612, 618 + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/20251213230408_FixRecipe2RecipeIngredient.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251213230408_FixRecipe2RecipeIngredient.cs new file mode 100644 index 00000000..68019fe1 --- /dev/null +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/20251213230408_FixRecipe2RecipeIngredient.cs @@ -0,0 +1,72 @@ +using System; +using Microsoft.EntityFrameworkCore.Migrations; + +#nullable disable + +namespace HomeBook.Backend.Data.Sqlite.Migrations +{ + /// + public partial class FixRecipe2RecipeIngredient : Migration + { + /// + protected override void Up(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_RecipeIngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.DropIndex( + name: "IX_Recipe2RecipeIngredient_RecipeIngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.DropColumn( + name: "RecipeIngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.CreateIndex( + name: "IX_Recipe2RecipeIngredient_IngredientId", + table: "Recipe2RecipeIngredient", + column: "IngredientId"); + + migrationBuilder.AddForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_IngredientId", + table: "Recipe2RecipeIngredient", + column: "IngredientId", + principalTable: "RecipeIngredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + } + + /// + protected override void Down(MigrationBuilder migrationBuilder) + { + migrationBuilder.DropForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_IngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.DropIndex( + name: "IX_Recipe2RecipeIngredient_IngredientId", + table: "Recipe2RecipeIngredient"); + + migrationBuilder.AddColumn( + name: "RecipeIngredientId", + table: "Recipe2RecipeIngredient", + type: "TEXT", + nullable: false, + defaultValue: new Guid("00000000-0000-0000-0000-000000000000")); + + migrationBuilder.CreateIndex( + name: "IX_Recipe2RecipeIngredient_RecipeIngredientId", + table: "Recipe2RecipeIngredient", + column: "RecipeIngredientId"); + + migrationBuilder.AddForeignKey( + name: "FK_Recipe2RecipeIngredient_RecipeIngredients_RecipeIngredientId", + table: "Recipe2RecipeIngredient", + column: "RecipeIngredientId", + principalTable: "RecipeIngredients", + principalColumn: "Id", + onDelete: ReferentialAction.Cascade); + } + } +} diff --git a/source/HomeBook.Backend.Data.Sqlite/Migrations/AppDbContextModelSnapshot.cs b/source/HomeBook.Backend.Data.Sqlite/Migrations/AppDbContextModelSnapshot.cs index 727bd028..a1ff3eae 100644 --- a/source/HomeBook.Backend.Data.Sqlite/Migrations/AppDbContextModelSnapshot.cs +++ b/source/HomeBook.Backend.Data.Sqlite/Migrations/AppDbContextModelSnapshot.cs @@ -15,7 +15,7 @@ partial class AppDbContextModelSnapshot : ModelSnapshot protected override void BuildModel(ModelBuilder modelBuilder) { #pragma warning disable 612, 618 - modelBuilder.HasAnnotation("ProductVersion", "9.0.10"); + modelBuilder.HasAnnotation("ProductVersion", "9.0.11"); modelBuilder.Entity("HomeBook.Backend.Data.Entities.Configuration", b => { @@ -37,40 +37,28 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.ToTable("Configurations"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => { b.Property("Id") .ValueGeneratedOnAdd() .HasColumnType("TEXT"); - b.Property("Name") - .IsRequired() - .HasMaxLength(100) - .HasColumnType("TEXT"); + b.Property("CaloriesKcal") + .HasColumnType("INTEGER"); - b.Property("NormalizedName") - .IsRequired() - .HasMaxLength(100) + b.Property("Comments") .HasColumnType("TEXT"); - b.HasKey("Id"); - - b.HasIndex("NormalizedName") - .IsUnique(); - - b.ToTable("Ingredients"); - }); - - modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => - { - b.Property("Id") - .ValueGeneratedOnAdd() + b.Property("Description") .HasColumnType("TEXT"); - b.Property("CaloriesKcal") + b.Property("DurationCookingMinutes") .HasColumnType("INTEGER"); - b.Property("DurationMinutes") + b.Property("DurationRestingMinutes") + .HasColumnType("INTEGER"); + + b.Property("DurationWorkingMinutes") .HasColumnType("INTEGER"); b.Property("Name") @@ -83,12 +71,23 @@ protected override void BuildModel(ModelBuilder modelBuilder) .HasMaxLength(50) .HasColumnType("TEXT"); + b.Property("Servings") + .HasColumnType("INTEGER"); + + b.Property("Source") + .HasColumnType("TEXT"); + + b.Property("UserId") + .HasColumnType("TEXT"); + b.HasKey("Id"); + b.HasIndex("UserId"); + b.ToTable("Recipes"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => { b.Property("RecipeId") .HasColumnType("TEXT"); @@ -96,9 +95,8 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.Property("IngredientId") .HasColumnType("TEXT"); - b.Property("Quantity") - .HasMaxLength(50) - .HasColumnType("TEXT"); + b.Property("Quantity") + .HasColumnType("REAL"); b.Property("Unit") .HasMaxLength(20) @@ -108,9 +106,54 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.HasIndex("IngredientId"); + b.ToTable("Recipe2RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + { + b.Property("Id") + .ValueGeneratedOnAdd() + .HasColumnType("TEXT"); + + b.Property("Name") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.Property("NormalizedName") + .IsRequired() + .HasMaxLength(100) + .HasColumnType("TEXT"); + + b.HasKey("Id"); + + b.HasIndex("NormalizedName") + .IsUnique(); + b.ToTable("RecipeIngredients"); }); + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.Property("RecipeId") + .HasColumnType("TEXT"); + + b.Property("Position") + .HasColumnType("INTEGER"); + + b.Property("Description") + .IsRequired() + .HasMaxLength(500) + .HasColumnType("TEXT"); + + b.Property("TimerDurationInSeconds") + .HasColumnType("INTEGER"); + + b.HasKey("RecipeId", "Position"); + + b.ToTable("RecipeSteps"); + }); + modelBuilder.Entity("HomeBook.Backend.Data.Entities.SavingGoal", b => { b.Property("Id") @@ -212,21 +255,41 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.ToTable("UserPreferences"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.User", "User") + .WithMany() + .HasForeignKey("UserId"); + + b.Navigation("User"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe2RecipeIngredient", b => { - b.HasOne("HomeBook.Backend.Data.Entities.Ingredient", "Ingredient") - .WithMany("RecipeIngredients") + b.HasOne("HomeBook.Backend.Data.Entities.RecipeIngredient", "RecipeIngredient") + .WithMany("Recipe2RecipeIngredients") .HasForeignKey("IngredientId") .OnDelete(DeleteBehavior.Cascade) .IsRequired(); b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") - .WithMany() + .WithMany("Recipe2RecipeIngredient") .HasForeignKey("RecipeId") .OnDelete(DeleteBehavior.Cascade) .IsRequired(); - b.Navigation("Ingredient"); + b.Navigation("Recipe"); + + b.Navigation("RecipeIngredient"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeStep", b => + { + b.HasOne("HomeBook.Backend.Data.Entities.Recipe", "Recipe") + .WithMany("Steps") + .HasForeignKey("RecipeId") + .OnDelete(DeleteBehavior.Cascade) + .IsRequired(); b.Navigation("Recipe"); }); @@ -253,9 +316,16 @@ protected override void BuildModel(ModelBuilder modelBuilder) b.Navigation("User"); }); - modelBuilder.Entity("HomeBook.Backend.Data.Entities.Ingredient", b => + modelBuilder.Entity("HomeBook.Backend.Data.Entities.Recipe", b => + { + b.Navigation("Recipe2RecipeIngredient"); + + b.Navigation("Steps"); + }); + + modelBuilder.Entity("HomeBook.Backend.Data.Entities.RecipeIngredient", b => { - b.Navigation("RecipeIngredients"); + b.Navigation("Recipe2RecipeIngredients"); }); #pragma warning restore 612, 618 } diff --git a/source/HomeBook.Backend.Data/AppDbContext.cs b/source/HomeBook.Backend.Data/AppDbContext.cs index 5d8c28fd..884924c7 100644 --- a/source/HomeBook.Backend.Data/AppDbContext.cs +++ b/source/HomeBook.Backend.Data/AppDbContext.cs @@ -14,8 +14,9 @@ public class AppDbContext( public DbSet UserPreferences { get; set; } = null!; public DbSet SavingGoals { get; set; } = null!; public DbSet Recipes { get; set; } = null!; - public DbSet Ingredients { get; set; } = null!; + public DbSet Recipe2RecipeIngredients { get; set; } = null!; public DbSet RecipeIngredients { get; set; } = null!; + public DbSet RecipeSteps { get; set; } = null!; private readonly IEnumerable? _saveChangesInterceptors = saveChangesInterceptors; diff --git a/source/HomeBook.Backend.Data/Contracts/IIngredientRepository.cs b/source/HomeBook.Backend.Data/Contracts/IIngredientRepository.cs new file mode 100644 index 00000000..88027832 --- /dev/null +++ b/source/HomeBook.Backend.Data/Contracts/IIngredientRepository.cs @@ -0,0 +1,37 @@ +using HomeBook.Backend.Data.Entities; + +namespace HomeBook.Backend.Data.Contracts; + +public interface IIngredientRepository +{ + /// + /// + /// + /// + /// + /// + Task CreateOrUpdateAsync(RecipeIngredient entity, + CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + /// + Task GetByIdAsync(Guid entityId, + CancellationToken cancellationToken, + AppDbContext? appDbContext = null); + + /// + /// + /// + /// + /// + /// + /// + Task GetByName(string name, + CancellationToken cancellationToken, + AppDbContext? appDbContext = null); +} diff --git a/source/HomeBook.Backend.Data/Contracts/IRecipesRepository.cs b/source/HomeBook.Backend.Data/Contracts/IRecipesRepository.cs index 01e32127..f849b41e 100644 --- a/source/HomeBook.Backend.Data/Contracts/IRecipesRepository.cs +++ b/source/HomeBook.Backend.Data/Contracts/IRecipesRepository.cs @@ -41,4 +41,46 @@ Task CreateOrUpdateAsync(Recipe entity, /// Task DeleteAsync(Guid entityId, CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + Task CreateOrUpdateAsync(Recipe2RecipeIngredient entity, + CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + /// + /// + Task GetAsync(Guid recipeId, + Guid ingredientId, + CancellationToken cancellationToken, + AppDbContext? appDbContext = null); + + /// + /// + /// + /// + /// + /// + Task CreateRecipeStepAsync(RecipeStep entity, + CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + /// + Task UpdateRecipeNameAsync(Guid id, + string name, + CancellationToken cancellationToken); } diff --git a/source/HomeBook.Backend.Data/Contracts/IUserPreferenceRepository.cs b/source/HomeBook.Backend.Data/Contracts/IUserPreferenceRepository.cs index 774e36a4..f339618c 100644 --- a/source/HomeBook.Backend.Data/Contracts/IUserPreferenceRepository.cs +++ b/source/HomeBook.Backend.Data/Contracts/IUserPreferenceRepository.cs @@ -21,9 +21,9 @@ public interface IUserPreferenceRepository /// /// sets the preference value for a specific user by key /// - /// + /// /// /// - Task SetPreferenceForUserByKeyAsync(UserPreference preference, + Task SetPreferenceForUserByKeyAsync(UserPreference entity, CancellationToken cancellationToken); } diff --git a/source/HomeBook.Backend.Data/Entities/Ingredient.cs b/source/HomeBook.Backend.Data/Entities/Ingredient.cs deleted file mode 100644 index a6fd892b..00000000 --- a/source/HomeBook.Backend.Data/Entities/Ingredient.cs +++ /dev/null @@ -1,26 +0,0 @@ -using System.ComponentModel.DataAnnotations; -using HomeBook.Backend.Abstractions.Contracts; -using Microsoft.EntityFrameworkCore; - -namespace HomeBook.Backend.Data.Entities; - -[Index(nameof(NormalizedName), IsUnique = true)] -public class Ingredient : INormalizable -{ - public Guid Id { get; set; } - - [Required] - [MaxLength(100)] - public string Name { get; set; } = null!; - - [Required] - [MaxLength(100)] - public string NormalizedName { get; set; } = null!; - - public ICollection RecipeIngredients { get; set; } = []; - - public void Normalize(IStringNormalizer normalizer) - { - NormalizedName = normalizer.Normalize(Name); - } -} diff --git a/source/HomeBook.Backend.Data/Entities/Recipe.cs b/source/HomeBook.Backend.Data/Entities/Recipe.cs index f31beb86..54f1f47c 100644 --- a/source/HomeBook.Backend.Data/Entities/Recipe.cs +++ b/source/HomeBook.Backend.Data/Entities/Recipe.cs @@ -6,6 +6,7 @@ namespace HomeBook.Backend.Data.Entities; [DebuggerDisplay("[{nameof(Recipe)}] {Name}")] +[Table("Recipes")] public class Recipe : INormalizable { [Key] @@ -21,24 +22,31 @@ public class Recipe : INormalizable [MaxLength(50)] public string NormalizedName { get; set; } = null!; - public int? DurationMinutes { get; set; } + public string? Description { get; set; } - [NotMapped] - public TimeSpan? Duration - { - get => - DurationMinutes.HasValue - ? TimeSpan.FromMinutes(DurationMinutes.Value) - : null; - - set => - DurationMinutes = value.HasValue - ? (int)value.Value.TotalMinutes - : null; - } + public int? DurationWorkingMinutes { get; set; } + + public int? DurationCookingMinutes { get; set; } + + public int? DurationRestingMinutes { get; set; } public int? CaloriesKcal { get; set; } + public int? Servings { get; set; } + + public string? Comments { get; set; } + + public string? Source { get; set; } + + [ForeignKey(nameof(User))] + public Guid? UserId { get; set; } + + public virtual User? User { get; set; } + + public virtual ICollection Recipe2RecipeIngredient { get; set; } = new List(); + + public virtual ICollection Steps { get; set; } = new List(); + public void Normalize(IStringNormalizer normalizer) { NormalizedName = normalizer.Normalize(Name); diff --git a/source/HomeBook.Backend.Data/Entities/Recipe2RecipeIngredient.cs b/source/HomeBook.Backend.Data/Entities/Recipe2RecipeIngredient.cs new file mode 100644 index 00000000..1dbeead5 --- /dev/null +++ b/source/HomeBook.Backend.Data/Entities/Recipe2RecipeIngredient.cs @@ -0,0 +1,27 @@ +using System.ComponentModel.DataAnnotations; +using System.ComponentModel.DataAnnotations.Schema; +using Microsoft.EntityFrameworkCore; + +namespace HomeBook.Backend.Data.Entities; + +[PrimaryKey(nameof(RecipeId), nameof(IngredientId))] +[Table("Recipe2RecipeIngredient")] +public class Recipe2RecipeIngredient +{ + [Required] + public Guid RecipeId { get; set; } + + [ForeignKey(nameof(RecipeId))] + public Recipe Recipe { get; set; } = null!; + + [Required] + public Guid IngredientId { get; set; } + + [ForeignKey(nameof(IngredientId))] + public RecipeIngredient RecipeIngredient { get; set; } = null!; + + public double? Quantity { get; set; } + + [MaxLength(20)] + public string? Unit { get; set; } +} diff --git a/source/HomeBook.Backend.Data/Entities/RecipeIngredient.cs b/source/HomeBook.Backend.Data/Entities/RecipeIngredient.cs index 2d8265e0..de71c383 100644 --- a/source/HomeBook.Backend.Data/Entities/RecipeIngredient.cs +++ b/source/HomeBook.Backend.Data/Entities/RecipeIngredient.cs @@ -1,22 +1,33 @@ using System.ComponentModel.DataAnnotations; +using System.ComponentModel.DataAnnotations.Schema; +using HomeBook.Backend.Abstractions.Contracts; using Microsoft.EntityFrameworkCore; namespace HomeBook.Backend.Data.Entities; -[PrimaryKey(nameof(RecipeId), nameof(IngredientId))] -public class RecipeIngredient +[Index(nameof(NormalizedName), IsUnique = true)] +[Table("RecipeIngredients")] +public class RecipeIngredient : INormalizable { + [Key] + [DatabaseGenerated(DatabaseGeneratedOption.Identity)] [Required] - public Guid RecipeId { get; set; } - public virtual Recipe Recipe { get; set; } = null!; + public Guid Id { get; set; } [Required] - public Guid IngredientId { get; set; } - public virtual Ingredient Ingredient { get; set; } = null!; + [MaxLength(100)] + public string Name { get; set; } = null!; + + [Required] + [MaxLength(100)] + public string NormalizedName { get; set; } = null!; + + public virtual ICollection Recipe2RecipeIngredients { get; set; } = + new List(); - [MaxLength(50)] - public string? Quantity { get; set; } - [MaxLength(20)] - public string? Unit { get; set; } + public void Normalize(IStringNormalizer normalizer) + { + NormalizedName = normalizer.Normalize(Name); + } } diff --git a/source/HomeBook.Backend.Data/Entities/RecipeStep.cs b/source/HomeBook.Backend.Data/Entities/RecipeStep.cs new file mode 100644 index 00000000..88476cc6 --- /dev/null +++ b/source/HomeBook.Backend.Data/Entities/RecipeStep.cs @@ -0,0 +1,23 @@ +using System.ComponentModel.DataAnnotations; +using System.ComponentModel.DataAnnotations.Schema; +using Microsoft.EntityFrameworkCore; + +namespace HomeBook.Backend.Data.Entities; + +[PrimaryKey(nameof(RecipeId), nameof(Position))] +[Table("RecipeSteps")] +public class RecipeStep +{ + [Required] + public Guid RecipeId { get; set; } + public virtual Recipe Recipe { get; set; } = null!; + + [Required] + public int Position { get; set; } + + [Required] + [StringLength(500)] + public string Description { get; set; } + + public int? TimerDurationInSeconds { get; set; } +} diff --git a/source/HomeBook.Backend.Data/Extensions/ServiceCollectionExtensions.cs b/source/HomeBook.Backend.Data/Extensions/ServiceCollectionExtensions.cs index e735de0d..c0aece2c 100644 --- a/source/HomeBook.Backend.Data/Extensions/ServiceCollectionExtensions.cs +++ b/source/HomeBook.Backend.Data/Extensions/ServiceCollectionExtensions.cs @@ -22,6 +22,7 @@ public static IServiceCollection AddBackendData(this IServiceCollection services services.AddScoped(); services.AddScoped(); services.AddScoped(); + services.AddScoped(); services.AddSingleton(); diff --git a/source/HomeBook.Backend.Data/HomeBook.Backend.Data.csproj b/source/HomeBook.Backend.Data/HomeBook.Backend.Data.csproj index b67bd95b..a52f8df8 100644 --- a/source/HomeBook.Backend.Data/HomeBook.Backend.Data.csproj +++ b/source/HomeBook.Backend.Data/HomeBook.Backend.Data.csproj @@ -11,9 +11,9 @@ - - - + + + runtime; build; native; contentfiles; analyzers; buildtransitive all diff --git a/source/HomeBook.Backend.Data/Interceptors/NormalizationInterceptor.cs b/source/HomeBook.Backend.Data/Interceptors/NormalizationInterceptor.cs index 2ae8ad4a..616249d7 100644 --- a/source/HomeBook.Backend.Data/Interceptors/NormalizationInterceptor.cs +++ b/source/HomeBook.Backend.Data/Interceptors/NormalizationInterceptor.cs @@ -14,7 +14,8 @@ public override InterceptionResult SavingChanges(DbContextEventData eventDa { DbContext dbContext = eventData.Context!; - IEnumerable entries = dbContext.ChangeTracker.Entries() + var allEntries = dbContext.ChangeTracker.Entries(); + IEnumerable entries = allEntries .Where(e => e.State is EntityState.Added or EntityState.Modified) .Select(e => e.Entity) .OfType(); @@ -31,7 +32,8 @@ public override ValueTask> SavingChangesAsync(DbContextE { DbContext dbContext = eventData.Context!; - IEnumerable entries = dbContext.ChangeTracker.Entries() + var allEntries = dbContext.ChangeTracker.Entries(); + IEnumerable entries = allEntries .Where(e => e.State is EntityState.Added or EntityState.Modified) .Select(e => e.Entity) .OfType(); diff --git a/source/HomeBook.Backend.Data/Repositories/IngredientRepository.cs b/source/HomeBook.Backend.Data/Repositories/IngredientRepository.cs new file mode 100644 index 00000000..201dfa5b --- /dev/null +++ b/source/HomeBook.Backend.Data/Repositories/IngredientRepository.cs @@ -0,0 +1,86 @@ +using System.Linq.Expressions; +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.Data.Contracts; +using HomeBook.Backend.Data.Entities; +using Microsoft.EntityFrameworkCore; +using Microsoft.EntityFrameworkCore.Query; + +namespace HomeBook.Backend.Data.Repositories; + +/// +public class IngredientRepository( + IDbContextFactory factory, + IStringNormalizer stringNormalizer) : IIngredientRepository +{ + /// + public async Task CreateOrUpdateAsync(RecipeIngredient entity, + CancellationToken cancellationToken) + { + await using AppDbContext dbContext = await factory.CreateDbContextAsync(cancellationToken); + + entity.Normalize(stringNormalizer); + + RecipeIngredient? existing = await GetByName(entity.Name, + cancellationToken, + dbContext); + + if (existing is null) + { + dbContext.Add(entity); + } + else + { + await dbContext.RecipeIngredients + .Where(u => u.Id == entity.Id) + .ExecuteUpdateAsync(UpdateEntityProperties(entity), + cancellationToken: cancellationToken); + } + + await dbContext.SaveChangesAsync(cancellationToken); + return entity.Id; + } + + private static Expression, SetPropertyCalls>> + UpdateEntityProperties(RecipeIngredient entity) + { + return s => s + .SetProperty(u => u.Name, entity.Name) + .SetProperty(u => u.NormalizedName, entity.NormalizedName); + } + + /// + public async Task GetByIdAsync(Guid entityId, + CancellationToken cancellationToken, + AppDbContext? appDbContext = null) + { + if (appDbContext is null) + { + await using AppDbContext newDbContext = await factory.CreateDbContextAsync(cancellationToken); + return await GetByIdAsync(entityId, cancellationToken, newDbContext); + } + + RecipeIngredient? entity = await appDbContext.Set() + .FirstOrDefaultAsync(ri => ri.Id == entityId, cancellationToken); + + return entity; + } + + /// + public async Task GetByName(string name, + CancellationToken cancellationToken, + AppDbContext? appDbContext = null) + { + if (appDbContext is null) + { + await using AppDbContext newDbContext = await factory.CreateDbContextAsync(cancellationToken); + return await GetByName(name, cancellationToken, newDbContext); + } + + string normalizedName = stringNormalizer.Normalize(name); + + RecipeIngredient? entity = await appDbContext.Set() + .FirstOrDefaultAsync(ri => ri.NormalizedName == normalizedName, cancellationToken); + + return entity; + } +} diff --git a/source/HomeBook.Backend.Data/Repositories/RecipesRepository.cs b/source/HomeBook.Backend.Data/Repositories/RecipesRepository.cs index 51cf727c..7fdb85e6 100644 --- a/source/HomeBook.Backend.Data/Repositories/RecipesRepository.cs +++ b/source/HomeBook.Backend.Data/Repositories/RecipesRepository.cs @@ -40,8 +40,10 @@ public async Task> GetAsync(string? searchFilter, } Recipe? entity = await appDbContext.Set() - .Where(e => e.Id == entityId) - .FirstOrDefaultAsync(cancellationToken); + .Include(r => r.Recipe2RecipeIngredient) + .ThenInclude(ri => ri.RecipeIngredient) + .Include(r => r.Steps) + .FirstOrDefaultAsync(r => r.Id == entityId, cancellationToken); return entity; } @@ -58,14 +60,28 @@ public async Task CreateOrUpdateAsync(Recipe entity, if (existing is null) { - dbContext.Recipes.Add(entity); + dbContext.Add(entity); + await dbContext.SaveChangesAsync(cancellationToken); } else { - dbContext.Entry(existing).CurrentValues.SetValues(entity); + await dbContext.Recipes + .Where(u => u.Id == entity.Id) + .ExecuteUpdateAsync(x => x + .SetProperty(u => u.Name, entity.Name) + .SetProperty(u => u.NormalizedName, stringNormalizer.Normalize(entity.Name)) + .SetProperty(u => u.Description, entity.Description) + .SetProperty(u => u.DurationWorkingMinutes, entity.DurationWorkingMinutes) + .SetProperty(u => u.DurationCookingMinutes, entity.DurationCookingMinutes) + .SetProperty(u => u.DurationRestingMinutes, entity.DurationRestingMinutes) + .SetProperty(u => u.CaloriesKcal, entity.CaloriesKcal) + .SetProperty(u => u.Servings, entity.Servings) + .SetProperty(u => u.Comments, entity.Comments) + .SetProperty(u => u.Source, entity.Source) + .SetProperty(u => u.NormalizedName, entity.NormalizedName), + cancellationToken: cancellationToken); } - await dbContext.SaveChangesAsync(cancellationToken); return entity.Id; } @@ -79,4 +95,83 @@ await dbContext.Set() .Where(e => e.Id == entityId) .ExecuteDeleteAsync(cancellationToken); } + + /// + public async Task CreateOrUpdateAsync(Recipe2RecipeIngredient entity, + CancellationToken cancellationToken) + { + await using AppDbContext dbContext = await factory.CreateDbContextAsync(cancellationToken); + + Recipe2RecipeIngredient? existing = await GetAsync(entity.RecipeId, + entity.IngredientId, + cancellationToken, + dbContext); + + if (existing is null) + { + dbContext.Add(entity); + } + else + { + await dbContext.Recipe2RecipeIngredients + .Where(u => u.RecipeId == entity.RecipeId + && u.IngredientId == entity.IngredientId) + .ExecuteUpdateAsync(x => x + .SetProperty(u => u.Quantity, entity.Quantity) + .SetProperty(u => u.Unit, entity.Unit), + cancellationToken: cancellationToken); + } + + await dbContext.SaveChangesAsync(cancellationToken); + return entity; + } + + /// + public async Task GetAsync(Guid recipeId, + Guid ingredientId, + CancellationToken cancellationToken, + AppDbContext? appDbContext = null) + { + if (appDbContext is null) + { + await using AppDbContext newDbContext = await factory.CreateDbContextAsync(cancellationToken); + return await GetAsync(recipeId, ingredientId, cancellationToken, newDbContext); + } + + Recipe2RecipeIngredient? entity = await appDbContext.Set() + .Include(r2ri => r2ri.RecipeIngredient) + .Include(r2ri => r2ri.Recipe) + .FirstOrDefaultAsync(r2ri => r2ri.RecipeId == recipeId + && r2ri.IngredientId == ingredientId, + cancellationToken); + + return entity; + } + + /// + public async Task CreateRecipeStepAsync(RecipeStep entity, + CancellationToken cancellationToken) + { + await using AppDbContext dbContext = await factory.CreateDbContextAsync(cancellationToken); + + dbContext.RecipeSteps.Add(entity); + + await dbContext.SaveChangesAsync(cancellationToken); + return entity; + } + + /// + public async Task UpdateRecipeNameAsync(Guid id, + string name, + CancellationToken cancellationToken) + { + await using AppDbContext dbContext = await factory.CreateDbContextAsync(cancellationToken); + + await dbContext.Recipes + .Where(r => r.Id == id) + .ExecuteUpdateAsync(x => x + .SetProperty(r => r.Name, name) + .SetProperty(r => r.NormalizedName, stringNormalizer.Normalize(name)), + cancellationToken: cancellationToken); + } } diff --git a/source/HomeBook.Backend.Data/Repositories/UserPreferenceRepository.cs b/source/HomeBook.Backend.Data/Repositories/UserPreferenceRepository.cs index df029c0e..fc6826f5 100644 --- a/source/HomeBook.Backend.Data/Repositories/UserPreferenceRepository.cs +++ b/source/HomeBook.Backend.Data/Repositories/UserPreferenceRepository.cs @@ -23,24 +23,24 @@ public class UserPreferenceRepository( } /// - public async Task SetPreferenceForUserByKeyAsync(UserPreference preference, + public async Task SetPreferenceForUserByKeyAsync(UserPreference entity, CancellationToken cancellationToken) { await using AppDbContext dbContext = await factory.CreateDbContextAsync(cancellationToken); - UserPreference? existingPreference = await dbContext.Set() - .FirstOrDefaultAsync(c => c.UserId == preference.UserId - && c.Key == preference.Key, + UserPreference? existingEntity = await dbContext.Set() + .FirstOrDefaultAsync(c => c.UserId == entity.UserId + && c.Key == entity.Key, cancellationToken); - if (existingPreference is null) + if (existingEntity is null) { - dbContext.Add(preference); + dbContext.Add(entity); } else { - existingPreference.Value = preference.Value; - dbContext.Update(existingPreference); + existingEntity.Value = entity.Value; + dbContext.Update(existingEntity); } await dbContext.SaveChangesAsync(cancellationToken); diff --git a/source/HomeBook.Backend.Core.Finances/Contracts/IFinanceCalculationsService.cs b/source/HomeBook.Backend.Module.Finances/Contracts/IFinanceCalculationsService.cs similarity index 92% rename from source/HomeBook.Backend.Core.Finances/Contracts/IFinanceCalculationsService.cs rename to source/HomeBook.Backend.Module.Finances/Contracts/IFinanceCalculationsService.cs index 71b9e98f..b71bbd9f 100644 --- a/source/HomeBook.Backend.Core.Finances/Contracts/IFinanceCalculationsService.cs +++ b/source/HomeBook.Backend.Module.Finances/Contracts/IFinanceCalculationsService.cs @@ -1,6 +1,6 @@ -using HomeBook.Backend.Core.Finances.Models; +using HomeBook.Backend.Module.Finances.Models; -namespace HomeBook.Backend.Core.Finances.Contracts; +namespace HomeBook.Backend.Module.Finances.Contracts; /// /// diff --git a/source/HomeBook.Backend.Core.Finances/Contracts/ISavingGoalsProvider.cs b/source/HomeBook.Backend.Module.Finances/Contracts/ISavingGoalsProvider.cs similarity index 96% rename from source/HomeBook.Backend.Core.Finances/Contracts/ISavingGoalsProvider.cs rename to source/HomeBook.Backend.Module.Finances/Contracts/ISavingGoalsProvider.cs index cbb67f7a..cb846bfa 100644 --- a/source/HomeBook.Backend.Core.Finances/Contracts/ISavingGoalsProvider.cs +++ b/source/HomeBook.Backend.Module.Finances/Contracts/ISavingGoalsProvider.cs @@ -1,7 +1,7 @@ -using HomeBook.Backend.Core.Finances.Models; -using HomeBook.Backend.DTOs.Enums; +using HomeBook.Backend.Module.Finances.Enums; +using HomeBook.Backend.Module.Finances.Models; -namespace HomeBook.Backend.Core.Finances.Contracts; +namespace HomeBook.Backend.Module.Finances.Contracts; /// /// diff --git a/source/HomeBook.Backend.Module.Finances/Endpoints/CalculationsEndpoints.cs b/source/HomeBook.Backend.Module.Finances/Endpoints/CalculationsEndpoints.cs new file mode 100644 index 00000000..0cc0cce7 --- /dev/null +++ b/source/HomeBook.Backend.Module.Finances/Endpoints/CalculationsEndpoints.cs @@ -0,0 +1,37 @@ +using HomeBook.Backend.Core.Modules.OpenApi; +using HomeBook.Backend.Module.Finances.Handler; +using HomeBook.Backend.Module.Finances.Responses; +using HomeBook.Backend.Modules.Abstractions; +using Microsoft.AspNetCore.Builder; +using Microsoft.AspNetCore.Http; +using Microsoft.AspNetCore.Routing; + +namespace HomeBook.Backend.Module.Finances.Endpoints; + +public static class FinanceCalculationsEndpoints +{ + public static IEndpointBuilder MapCalculationEndpoints(this IEndpointBuilder builder) + { + builder.AddEndpoint(routeBuilder => + { + RouteGroupBuilder group = routeBuilder + .MapGroup("/calculations") + .WithDescription("Endpoints to manage finances calculations") + .RequireAuthorization(); + + group.MapPost("/savings", CalculationsHandler.HandleCalculateSavings) + .WithName("CalculateSavings") + .WithDescription(new Description( + "returns the calculated savings based on the provided parameters", + "HTTP 200: successfully calculated savings", + "HTTP 401: User is not authorized", + "HTTP 500: Unknown error while getting saving goals")) + .RequireAuthorization() + .Produces(StatusCodes.Status200OK) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status500InternalServerError); + }); + + return builder; + } +} diff --git a/source/HomeBook.Backend.Module.Finances/Endpoints/SavingGoalEndpoints.cs b/source/HomeBook.Backend.Module.Finances/Endpoints/SavingGoalEndpoints.cs new file mode 100644 index 00000000..e305cf24 --- /dev/null +++ b/source/HomeBook.Backend.Module.Finances/Endpoints/SavingGoalEndpoints.cs @@ -0,0 +1,129 @@ +using HomeBook.Backend.Core.Modules.OpenApi; +using HomeBook.Backend.Module.Finances.Handler; +using HomeBook.Backend.Module.Finances.Responses; +using HomeBook.Backend.Modules.Abstractions; +using Microsoft.AspNetCore.Builder; +using Microsoft.AspNetCore.Http; +using Microsoft.AspNetCore.Routing; + +namespace HomeBook.Backend.Module.Finances.Endpoints; + +public static class SavingGoalEndpoints +{ + public static IEndpointBuilder MapSavingGoalEndpoints(this IEndpointBuilder builder) + { + builder.AddEndpoint(routeBuilder => + { + RouteGroupBuilder group = routeBuilder + .MapGroup("/saving-goals") + .WithDescription("Endpoints to manage finances saving goals") + .RequireAuthorization(); + + group.MapGet("/", SavingGoalHandler.HandleGetSavingGoals) + .WithName("GetSavingGoals") + .WithDescription(new Description( + "returns the users finances saving goals", + "HTTP 200: Finances saving goals were found", + "HTTP 401: User is not authorized", + "HTTP 500: Unknown error while getting saving goals")) + .RequireAuthorization() + .Produces(StatusCodes.Status200OK) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status500InternalServerError); + + group.MapPost("/", SavingGoalHandler.HandleCreateSavingGoal) + .WithName("CreateSavingGoal") + .WithDescription(new Description( + "creates a new finances saving goal for the user", + "HTTP 201: Finances saving goal was created", + "HTTP 400: Invalid request data", + "HTTP 401: User is not authorized", + "HTTP 500: Unknown error while creating saving goal")) + .RequireAuthorization() + .Produces(StatusCodes.Status201Created) + .Produces(StatusCodes.Status400BadRequest) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status500InternalServerError); + + group.MapPatch("/{savingGoalId:guid}/name", SavingGoalHandler.HandleUpdateSavingGoalName) + .WithName("UpdateSavingGoalName") + .WithDescription(new Description( + "updates the name of an existing finances saving goal for the user", + "HTTP 200: Finances saving goal was updated", + "HTTP 400: Invalid request data", + "HTTP 401: User is not authorized", + "HTTP 404: Saving goal not found", + "HTTP 500: Unknown error while updating saving goal")) + .RequireAuthorization() + .Produces(StatusCodes.Status200OK) + .Produces(StatusCodes.Status400BadRequest) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status404NotFound) + .Produces(StatusCodes.Status500InternalServerError); + + group.MapPatch("/{savingGoalId:guid}/appearance", SavingGoalHandler.HandleUpdateSavingGoalAppearance) + .WithName("UpdateSavingGoalAppearance") + .WithDescription(new Description( + "updates the appearance of an existing finances saving goal for the user", + "HTTP 200: Finances saving goal was updated", + "HTTP 400: Invalid request data", + "HTTP 401: User is not authorized", + "HTTP 404: Saving goal not found", + "HTTP 500: Unknown error while updating saving goal")) + .RequireAuthorization() + .Produces(StatusCodes.Status200OK) + .Produces(StatusCodes.Status400BadRequest) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status404NotFound) + .Produces(StatusCodes.Status500InternalServerError); + + group.MapPatch("/{savingGoalId:guid}/amounts", SavingGoalHandler.HandleUpdateSavingGoalAmounts) + .WithName("UpdateSavingGoalAmounts") + .WithDescription(new Description( + "updates the amounts of an existing finances saving goal for the user", + "HTTP 200: Finances saving goal was updated", + "HTTP 400: Invalid request data", + "HTTP 401: User is not authorized", + "HTTP 404: Saving goal not found", + "HTTP 500: Unknown error while updating saving goal")) + .RequireAuthorization() + .Produces(StatusCodes.Status200OK) + .Produces(StatusCodes.Status400BadRequest) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status404NotFound) + .Produces(StatusCodes.Status500InternalServerError); + + group.MapPatch("/{savingGoalId:guid}/info", SavingGoalHandler.HandleUpdateSavingGoalInfo) + .WithName("UpdateSavingGoalInfo") + .WithDescription(new Description( + "updates the info (target date, etc.) of an existing finances saving goal for the user", + "HTTP 200: Finances saving goal was updated", + "HTTP 400: Invalid request data", + "HTTP 401: User is not authorized", + "HTTP 404: Saving goal not found", + "HTTP 500: Unknown error while updating saving goal")) + .RequireAuthorization() + .Produces(StatusCodes.Status200OK) + .Produces(StatusCodes.Status400BadRequest) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status404NotFound) + .Produces(StatusCodes.Status500InternalServerError); + + group.MapDelete("/{id:guid}", SavingGoalHandler.HandleDeleteSavingGoal) + .WithName("DeleteSavingGoal") + .WithDescription(new Description( + "deletes an existing finances saving goal for the user", + "HTTP 200: Finances saving goal was deleted", + "HTTP 401: User is not authorized", + "HTTP 404: Saving goal not found", + "HTTP 500: Unknown error while deleting saving goal")) + .RequireAuthorization() + .Produces(StatusCodes.Status200OK) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status404NotFound) + .Produces(StatusCodes.Status500InternalServerError); + }); + + return builder; + } +} diff --git a/source/HomeBook.Backend.DTOs/Enums/InterestRateOptions.cs b/source/HomeBook.Backend.Module.Finances/Enums/InterestRateOptions.cs similarity index 58% rename from source/HomeBook.Backend.DTOs/Enums/InterestRateOptions.cs rename to source/HomeBook.Backend.Module.Finances/Enums/InterestRateOptions.cs index bbe36fbf..41a0b575 100644 --- a/source/HomeBook.Backend.DTOs/Enums/InterestRateOptions.cs +++ b/source/HomeBook.Backend.Module.Finances/Enums/InterestRateOptions.cs @@ -1,4 +1,4 @@ -namespace HomeBook.Backend.DTOs.Enums; +namespace HomeBook.Backend.Module.Finances.Enums; public enum InterestRateOptions { diff --git a/source/HomeBook.Backend/Handler/FinanceCalculationsHandler.cs b/source/HomeBook.Backend.Module.Finances/Handler/CalculationsHandler.cs similarity index 67% rename from source/HomeBook.Backend/Handler/FinanceCalculationsHandler.cs rename to source/HomeBook.Backend.Module.Finances/Handler/CalculationsHandler.cs index 623ec3e6..46b03f3f 100644 --- a/source/HomeBook.Backend/Handler/FinanceCalculationsHandler.cs +++ b/source/HomeBook.Backend.Module.Finances/Handler/CalculationsHandler.cs @@ -1,12 +1,15 @@ -using HomeBook.Backend.Core.Finances.Contracts; -using HomeBook.Backend.Core.Finances.Models; -using HomeBook.Backend.DTOs.Requests.Finances; -using HomeBook.Backend.Mappings; +using HomeBook.Backend.Module.Finances.Contracts; +using HomeBook.Backend.Module.Finances.Enums; +using HomeBook.Backend.Module.Finances.Mappings; +using HomeBook.Backend.Module.Finances.Models; +using HomeBook.Backend.Module.Finances.Requests; +using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Mvc; +using Microsoft.Extensions.Logging; -namespace HomeBook.Backend.Handler; +namespace HomeBook.Backend.Module.Finances.Handler; -public class FinanceCalculationsHandler +public class CalculationsHandler { /// /// calculates the savings based on the provided parameters @@ -17,7 +20,7 @@ public class FinanceCalculationsHandler /// /// public static async Task HandleCalculateSavings([FromBody] CalculateSavingRequest request, - [FromServices] ILogger logger, + [FromServices] ILogger logger, [FromServices] IFinanceCalculationsService financeCalculationsService, CancellationToken cancellationToken) { @@ -25,19 +28,19 @@ public static async Task HandleCalculateSavings([FromBody] CalculateSav { SavingCalculationResult result = request.InterestRateOption switch { - DTOs.Enums.InterestRateOptions.MONTHLY => financeCalculationsService.CalculateMonthlySavings( + InterestRateOptions.MONTHLY => financeCalculationsService.CalculateMonthlySavings( request.TargetAmount, request.TargetDate, request.InterestRate, request.TargetSimpleRate), - DTOs.Enums.InterestRateOptions.YEARLY => financeCalculationsService.CalculateYearlySavings( + InterestRateOptions.YEARLY => financeCalculationsService.CalculateYearlySavings( request.TargetAmount, request.TargetDate, request.InterestRate, request.TargetSimpleRate), - DTOs.Enums.InterestRateOptions.NONE => financeCalculationsService.CalculateSavings( + InterestRateOptions.NONE => financeCalculationsService.CalculateSavings( request.TargetAmount, request.TargetDate), diff --git a/source/HomeBook.Backend/Handler/FinanceSavingGoalHandler.cs b/source/HomeBook.Backend.Module.Finances/Handler/SavingGoalHandler.cs similarity index 90% rename from source/HomeBook.Backend/Handler/FinanceSavingGoalHandler.cs rename to source/HomeBook.Backend.Module.Finances/Handler/SavingGoalHandler.cs index b9ff8605..9275daa0 100644 --- a/source/HomeBook.Backend/Handler/FinanceSavingGoalHandler.cs +++ b/source/HomeBook.Backend.Module.Finances/Handler/SavingGoalHandler.cs @@ -1,15 +1,17 @@ using System.Security.Claims; -using HomeBook.Backend.Core.Finances.Contracts; -using HomeBook.Backend.Core.Finances.Models; -using HomeBook.Backend.DTOs.Requests.Finances; -using HomeBook.Backend.DTOs.Responses.Finances; -using HomeBook.Backend.Mappings; -using HomeBook.Backend.Utilities; +using HomeBook.Backend.Core.Modules.Utilities; +using HomeBook.Backend.Module.Finances.Contracts; +using HomeBook.Backend.Module.Finances.Mappings; +using HomeBook.Backend.Module.Finances.Models; +using HomeBook.Backend.Module.Finances.Requests; +using HomeBook.Backend.Module.Finances.Responses; +using Microsoft.AspNetCore.Http; using Microsoft.AspNetCore.Mvc; +using Microsoft.Extensions.Logging; -namespace HomeBook.Backend.Handler; +namespace HomeBook.Backend.Module.Finances.Handler; -public class FinanceSavingGoalHandler +public class SavingGoalHandler { /// /// gets the user finance saving goals @@ -20,7 +22,7 @@ public class FinanceSavingGoalHandler /// /// public static async Task HandleGetSavingGoals(ClaimsPrincipal user, - [FromServices] ILogger logger, + [FromServices] ILogger logger, [FromServices] ISavingGoalsProvider savingGoalsProvider, CancellationToken cancellationToken) { @@ -54,7 +56,7 @@ public static async Task HandleGetSavingGoals(ClaimsPrincipal user, /// public static async Task HandleCreateSavingGoal(ClaimsPrincipal user, [FromBody] CreateSavingGoalRequest request, - [FromServices] ILogger logger, + [FromServices] ILogger logger, [FromServices] ISavingGoalsProvider savingGoalsProvider, CancellationToken cancellationToken) { @@ -96,7 +98,7 @@ public static async Task HandleCreateSavingGoal(ClaimsPrincipal user, public static async Task HandleUpdateSavingGoalName(Guid savingGoalId, ClaimsPrincipal user, [FromBody] UpdateSavingGoalNameRequest request, - [FromServices] ILogger logger, + [FromServices] ILogger logger, [FromServices] ISavingGoalsProvider savingGoalsProvider, CancellationToken cancellationToken) { @@ -133,7 +135,7 @@ await savingGoalsProvider.UpdateSavingGoalNameAsync(userId, public static async Task HandleUpdateSavingGoalAppearance(Guid savingGoalId, ClaimsPrincipal user, [FromBody] UpdateSavingGoalAppearanceRequest request, - [FromServices] ILogger logger, + [FromServices] ILogger logger, [FromServices] ISavingGoalsProvider savingGoalsProvider, CancellationToken cancellationToken) { @@ -171,7 +173,7 @@ await savingGoalsProvider.UpdateSavingGoalAppearanceAsync(userId, public static async Task HandleUpdateSavingGoalAmounts(Guid savingGoalId, ClaimsPrincipal user, [FromBody] UpdateSavingGoalAmountsRequest request, - [FromServices] ILogger logger, + [FromServices] ILogger logger, [FromServices] ISavingGoalsProvider savingGoalsProvider, CancellationToken cancellationToken) { @@ -212,7 +214,7 @@ await savingGoalsProvider.UpdateSavingGoalAmountsAsync(userId, public static async Task HandleUpdateSavingGoalInfo(Guid savingGoalId, ClaimsPrincipal user, [FromBody] UpdateSavingGoalInfoRequest request, - [FromServices] ILogger logger, + [FromServices] ILogger logger, [FromServices] ISavingGoalsProvider savingGoalsProvider, CancellationToken cancellationToken) { @@ -247,7 +249,7 @@ await savingGoalsProvider.UpdateSavingGoalInfoAsync(userId, /// public static async Task HandleDeleteSavingGoal(Guid id, ClaimsPrincipal user, - [FromServices] ILogger logger, + [FromServices] ILogger logger, [FromServices] ISavingGoalsProvider savingGoalsProvider, CancellationToken cancellationToken) { diff --git a/source/HomeBook.Backend.Core.Finances/HomeBook.Backend.Core.Finances.csproj b/source/HomeBook.Backend.Module.Finances/HomeBook.Backend.Module.Finances.csproj similarity index 56% rename from source/HomeBook.Backend.Core.Finances/HomeBook.Backend.Core.Finances.csproj rename to source/HomeBook.Backend.Module.Finances/HomeBook.Backend.Module.Finances.csproj index 1314ef4c..79343c25 100644 --- a/source/HomeBook.Backend.Core.Finances/HomeBook.Backend.Core.Finances.csproj +++ b/source/HomeBook.Backend.Module.Finances/HomeBook.Backend.Module.Finances.csproj @@ -7,12 +7,13 @@ - - + - + + + diff --git a/source/HomeBook.Backend/Mappings/CalculationMappings.cs b/source/HomeBook.Backend.Module.Finances/Mappings/CalculationMappings.cs similarity index 61% rename from source/HomeBook.Backend/Mappings/CalculationMappings.cs rename to source/HomeBook.Backend.Module.Finances/Mappings/CalculationMappings.cs index a9a0ccfe..78bc14e9 100644 --- a/source/HomeBook.Backend/Mappings/CalculationMappings.cs +++ b/source/HomeBook.Backend.Module.Finances/Mappings/CalculationMappings.cs @@ -1,7 +1,7 @@ -using HomeBook.Backend.Core.Finances.Models; -using HomeBook.Backend.DTOs.Responses.Finances; +using HomeBook.Backend.Module.Finances.Models; +using HomeBook.Backend.Module.Finances.Responses; -namespace HomeBook.Backend.Mappings; +namespace HomeBook.Backend.Module.Finances.Mappings; public static class CalculationMappings { diff --git a/source/HomeBook.Backend.Module.Finances/Mappings/SavingGoalMappings.cs b/source/HomeBook.Backend.Module.Finances/Mappings/SavingGoalMappings.cs new file mode 100644 index 00000000..f88a3be3 --- /dev/null +++ b/source/HomeBook.Backend.Module.Finances/Mappings/SavingGoalMappings.cs @@ -0,0 +1,38 @@ +using HomeBook.Backend.Module.Finances.Enums; +using HomeBook.Backend.Module.Finances.Models; +using HomeBook.Backend.Module.Finances.Responses; + +namespace HomeBook.Backend.Module.Finances.Mappings; + +public static class SavingGoalMappings +{ + public static SavingGoalDto ToDto(this Data.Entities.SavingGoal savingGoal) + { + return new SavingGoalDto( + savingGoal.Id, + savingGoal.Name, + savingGoal.Color, + savingGoal.Icon ?? string.Empty, + savingGoal.TargetAmount, + savingGoal.CurrentAmount, + savingGoal.MonthlyPayment, + (InterestRateOptions)savingGoal.InterestRateOption, + savingGoal.InterestRate, + savingGoal.TargetDate); + } + + public static SavingGoalResponse ToResponse(this SavingGoalDto savingGoal) + { + return new SavingGoalResponse( + savingGoal.Id, + savingGoal.Name, + savingGoal.Color, + savingGoal.Icon ?? string.Empty, + savingGoal.TargetAmount, + savingGoal.CurrentAmount, + savingGoal.MonthlyPayment, + (InterestRateOptions)savingGoal.InterestRateOption, + savingGoal.InterestRate, + savingGoal.TargetDate); + } +} diff --git a/source/HomeBook.Backend.Core.Finances/Models/SavingCalculationResult.cs b/source/HomeBook.Backend.Module.Finances/Models/SavingCalculationResult.cs similarity index 72% rename from source/HomeBook.Backend.Core.Finances/Models/SavingCalculationResult.cs rename to source/HomeBook.Backend.Module.Finances/Models/SavingCalculationResult.cs index bed99052..6be10d20 100644 --- a/source/HomeBook.Backend.Core.Finances/Models/SavingCalculationResult.cs +++ b/source/HomeBook.Backend.Module.Finances/Models/SavingCalculationResult.cs @@ -1,4 +1,4 @@ -namespace HomeBook.Backend.Core.Finances.Models; +namespace HomeBook.Backend.Module.Finances.Models; public record SavingCalculationResult(short MonthsNeeded, decimal MonthlyPayment, diff --git a/source/HomeBook.Backend.Core.Finances/Models/SavingGoalDto.cs b/source/HomeBook.Backend.Module.Finances/Models/SavingGoalDto.cs similarity index 73% rename from source/HomeBook.Backend.Core.Finances/Models/SavingGoalDto.cs rename to source/HomeBook.Backend.Module.Finances/Models/SavingGoalDto.cs index 213222f1..455e8f63 100644 --- a/source/HomeBook.Backend.Core.Finances/Models/SavingGoalDto.cs +++ b/source/HomeBook.Backend.Module.Finances/Models/SavingGoalDto.cs @@ -1,6 +1,6 @@ -using HomeBook.Backend.DTOs.Enums; +using HomeBook.Backend.Module.Finances.Enums; -namespace HomeBook.Backend.Core.Finances.Models; +namespace HomeBook.Backend.Module.Finances.Models; public record SavingGoalDto( Guid Id, diff --git a/source/HomeBook.Backend.Module.Finances/Module.cs b/source/HomeBook.Backend.Module.Finances/Module.cs new file mode 100644 index 00000000..3a334f32 --- /dev/null +++ b/source/HomeBook.Backend.Module.Finances/Module.cs @@ -0,0 +1,78 @@ +using FluentValidation; +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.Core.Finances.Validators; +using HomeBook.Backend.Data.Entities; +using HomeBook.Backend.Module.Finances.Contracts; +using HomeBook.Backend.Module.Finances.Endpoints; +using HomeBook.Backend.Module.Finances.Provider; +using HomeBook.Backend.Module.Finances.Services; +using HomeBook.Backend.Modules.Abstractions; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.DependencyInjection; + +namespace HomeBook.Backend.Module.Finances; + +public class Module : IModule, + IBackendModuleEndpointRegistrar, + IBackendModuleServiceRegistrar, + IBackendModuleSearchRegistrar +{ + public string Name { get; } = "Finances Module"; + public string Description { get; } = "Provides financial management features"; + public string Key { get; } = "finances"; + public string Author { get; } = "HomeBook"; + public Version Version { get; } = new("1.0.0"); + + public async Task InitializeAsync() + { + await Task.CompletedTask; + } + + public void RegisterEndpoints(IEndpointBuilder builder, IConfiguration configuration) + { + builder.MapCalculationEndpoints() + .MapSavingGoalEndpoints(); + } + + public static void RegisterServices(IServiceCollection services, IConfiguration configuration) + { + services.AddScoped(); + services.AddScoped(); + + services.AddSingleton, SavingGoalValidator>(); + } + + public async Task SearchAsync(string query, + Guid userId, + CancellationToken cancellationToken = default) + { + await Task.Delay(5000, cancellationToken); // Simulate some search delay + + var items = new List + { + new SearchResultItem( + Title: "Budget Overview", + Description: "View your monthly budget summary", + Url: "/finances/budget-overview", + Icon: "budget_icon", + Color: "green" + ), + new SearchResultItem( + Title: "Expense Tracker", + Description: "Track your daily expenses", + Url: "/finances/expense-tracker", + Icon: "expense_icon", + Color: "red" + ), + new SearchResultItem( + Title: "Saving Goals", + Description: "Manage your saving goals", + Url: "/finances/saving-goals", + Icon: "saving_icon", + Color: "blue" + ) + }; + return new SearchResult(items.Count, + items); + } +} diff --git a/source/HomeBook.Backend.Core.Finances/SavingGoalsProvider.cs b/source/HomeBook.Backend.Module.Finances/Provider/SavingGoalsProvider.cs similarity index 96% rename from source/HomeBook.Backend.Core.Finances/SavingGoalsProvider.cs rename to source/HomeBook.Backend.Module.Finances/Provider/SavingGoalsProvider.cs index 868700d1..69e7aa81 100644 --- a/source/HomeBook.Backend.Core.Finances/SavingGoalsProvider.cs +++ b/source/HomeBook.Backend.Module.Finances/Provider/SavingGoalsProvider.cs @@ -1,12 +1,12 @@ -using HomeBook.Backend.Core.Finances.Contracts; -using HomeBook.Backend.Core.Finances.Mappings; -using HomeBook.Backend.Core.Finances.Models; using HomeBook.Backend.Data.Contracts; using HomeBook.Backend.Data.Entities; -using HomeBook.Backend.DTOs.Enums; +using HomeBook.Backend.Module.Finances.Contracts; +using HomeBook.Backend.Module.Finances.Enums; +using HomeBook.Backend.Module.Finances.Mappings; +using HomeBook.Backend.Module.Finances.Models; using Microsoft.Extensions.Logging; -namespace HomeBook.Backend.Core.Finances; +namespace HomeBook.Backend.Module.Finances.Provider; public class SavingGoalsProvider( ILogger logger, diff --git a/source/HomeBook.Backend.DTOs/Requests/Finances/CalculateSavingRequest.cs b/source/HomeBook.Backend.Module.Finances/Requests/CalculateSavingRequest.cs similarity index 81% rename from source/HomeBook.Backend.DTOs/Requests/Finances/CalculateSavingRequest.cs rename to source/HomeBook.Backend.Module.Finances/Requests/CalculateSavingRequest.cs index 30b01e45..ec394452 100644 --- a/source/HomeBook.Backend.DTOs/Requests/Finances/CalculateSavingRequest.cs +++ b/source/HomeBook.Backend.Module.Finances/Requests/CalculateSavingRequest.cs @@ -1,6 +1,6 @@ -using HomeBook.Backend.DTOs.Enums; +using HomeBook.Backend.Module.Finances.Enums; -namespace HomeBook.Backend.DTOs.Requests.Finances; +namespace HomeBook.Backend.Module.Finances.Requests; /// /// diff --git a/source/HomeBook.Backend.DTOs/Requests/Finances/CreateSavingGoalRequest.cs b/source/HomeBook.Backend.Module.Finances/Requests/CreateSavingGoalRequest.cs similarity index 73% rename from source/HomeBook.Backend.DTOs/Requests/Finances/CreateSavingGoalRequest.cs rename to source/HomeBook.Backend.Module.Finances/Requests/CreateSavingGoalRequest.cs index fce53260..54a93603 100644 --- a/source/HomeBook.Backend.DTOs/Requests/Finances/CreateSavingGoalRequest.cs +++ b/source/HomeBook.Backend.Module.Finances/Requests/CreateSavingGoalRequest.cs @@ -1,6 +1,6 @@ -using HomeBook.Backend.DTOs.Enums; +using HomeBook.Backend.Module.Finances.Enums; -namespace HomeBook.Backend.DTOs.Requests.Finances; +namespace HomeBook.Backend.Module.Finances.Requests; public record CreateSavingGoalRequest( string Name, diff --git a/source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalAmountsRequest.cs b/source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalAmountsRequest.cs similarity index 67% rename from source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalAmountsRequest.cs rename to source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalAmountsRequest.cs index ab51b76e..3a35ba42 100644 --- a/source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalAmountsRequest.cs +++ b/source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalAmountsRequest.cs @@ -1,6 +1,6 @@ -using HomeBook.Backend.DTOs.Enums; +using HomeBook.Backend.Module.Finances.Enums; -namespace HomeBook.Backend.DTOs.Requests.Finances; +namespace HomeBook.Backend.Module.Finances.Requests; public record UpdateSavingGoalAmountsRequest( decimal? TargetAmount, diff --git a/source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalAppearanceRequest.cs b/source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalAppearanceRequest.cs similarity index 59% rename from source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalAppearanceRequest.cs rename to source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalAppearanceRequest.cs index bd1862c9..542f1848 100644 --- a/source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalAppearanceRequest.cs +++ b/source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalAppearanceRequest.cs @@ -1,3 +1,3 @@ -namespace HomeBook.Backend.DTOs.Requests.Finances; +namespace HomeBook.Backend.Module.Finances.Requests; public record UpdateSavingGoalAppearanceRequest(string Color, string Icon); diff --git a/source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalInfoRequest.cs b/source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalInfoRequest.cs similarity index 55% rename from source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalInfoRequest.cs rename to source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalInfoRequest.cs index d4e47dde..85e916e7 100644 --- a/source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalInfoRequest.cs +++ b/source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalInfoRequest.cs @@ -1,3 +1,3 @@ -namespace HomeBook.Backend.DTOs.Requests.Finances; +namespace HomeBook.Backend.Module.Finances.Requests; public record UpdateSavingGoalInfoRequest(DateTime? TargetDate); diff --git a/source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalNameRequest.cs b/source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalNameRequest.cs similarity index 51% rename from source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalNameRequest.cs rename to source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalNameRequest.cs index d718969d..58aa0707 100644 --- a/source/HomeBook.Backend.DTOs/Requests/Finances/UpdateSavingGoalNameRequest.cs +++ b/source/HomeBook.Backend.Module.Finances/Requests/UpdateSavingGoalNameRequest.cs @@ -1,3 +1,3 @@ -namespace HomeBook.Backend.DTOs.Requests.Finances; +namespace HomeBook.Backend.Module.Finances.Requests; public record UpdateSavingGoalNameRequest(string Name); diff --git a/source/HomeBook.Backend.DTOs/Responses/Finances/CalculatedSavingResponse.cs b/source/HomeBook.Backend.Module.Finances/Responses/CalculatedSavingResponse.cs similarity index 71% rename from source/HomeBook.Backend.DTOs/Responses/Finances/CalculatedSavingResponse.cs rename to source/HomeBook.Backend.Module.Finances/Responses/CalculatedSavingResponse.cs index 5e2b3404..5a3937d7 100644 --- a/source/HomeBook.Backend.DTOs/Responses/Finances/CalculatedSavingResponse.cs +++ b/source/HomeBook.Backend.Module.Finances/Responses/CalculatedSavingResponse.cs @@ -1,4 +1,4 @@ -namespace HomeBook.Backend.DTOs.Responses.Finances; +namespace HomeBook.Backend.Module.Finances.Responses; public record CalculatedSavingResponse(short MonthsNeeded, decimal MonthlyPayment, diff --git a/source/HomeBook.Backend.DTOs/Responses/Finances/SavingGoalListResponse.cs b/source/HomeBook.Backend.Module.Finances/Responses/SavingGoalListResponse.cs similarity index 57% rename from source/HomeBook.Backend.DTOs/Responses/Finances/SavingGoalListResponse.cs rename to source/HomeBook.Backend.Module.Finances/Responses/SavingGoalListResponse.cs index 21aabdb7..24abdf45 100644 --- a/source/HomeBook.Backend.DTOs/Responses/Finances/SavingGoalListResponse.cs +++ b/source/HomeBook.Backend.Module.Finances/Responses/SavingGoalListResponse.cs @@ -1,3 +1,3 @@ -namespace HomeBook.Backend.DTOs.Responses.Finances; +namespace HomeBook.Backend.Module.Finances.Responses; public record SavingGoalListResponse(SavingGoalResponse[] SavingGoals); diff --git a/source/HomeBook.Backend.DTOs/Responses/Finances/SavingGoalResponse.cs b/source/HomeBook.Backend.Module.Finances/Responses/SavingGoalResponse.cs similarity index 78% rename from source/HomeBook.Backend.DTOs/Responses/Finances/SavingGoalResponse.cs rename to source/HomeBook.Backend.Module.Finances/Responses/SavingGoalResponse.cs index 46328b41..40b4fc8e 100644 --- a/source/HomeBook.Backend.DTOs/Responses/Finances/SavingGoalResponse.cs +++ b/source/HomeBook.Backend.Module.Finances/Responses/SavingGoalResponse.cs @@ -1,7 +1,7 @@ using System.Diagnostics; -using HomeBook.Backend.DTOs.Enums; +using HomeBook.Backend.Module.Finances.Enums; -namespace HomeBook.Backend.DTOs.Responses.Finances; +namespace HomeBook.Backend.Module.Finances.Responses; [DebuggerDisplay("{Name} ({CurrentAmount} / {TargetAmount})")] public record SavingGoalResponse( diff --git a/source/HomeBook.Backend.Core.Finances/FinanceCalculationsService.cs b/source/HomeBook.Backend.Module.Finances/Services/FinanceCalculationsService.cs similarity index 96% rename from source/HomeBook.Backend.Core.Finances/FinanceCalculationsService.cs rename to source/HomeBook.Backend.Module.Finances/Services/FinanceCalculationsService.cs index 5576e479..12e3e947 100644 --- a/source/HomeBook.Backend.Core.Finances/FinanceCalculationsService.cs +++ b/source/HomeBook.Backend.Module.Finances/Services/FinanceCalculationsService.cs @@ -1,8 +1,8 @@ using HomeBook.Backend.Abstractions.Contracts; -using HomeBook.Backend.Core.Finances.Contracts; -using HomeBook.Backend.Core.Finances.Models; +using HomeBook.Backend.Module.Finances.Contracts; +using HomeBook.Backend.Module.Finances.Models; -namespace HomeBook.Backend.Core.Finances; +namespace HomeBook.Backend.Module.Finances.Services; /// public class FinanceCalculationsService(IDateTimeProvider dateTimeProvider) diff --git a/source/HomeBook.Backend.Core.Finances/Validators/SavingGoalValidator.cs b/source/HomeBook.Backend.Module.Finances/Validators/SavingGoalValidator.cs similarity index 100% rename from source/HomeBook.Backend.Core.Finances/Validators/SavingGoalValidator.cs rename to source/HomeBook.Backend.Module.Finances/Validators/SavingGoalValidator.cs diff --git a/source/HomeBook.Backend.Module.Kitchen/Contracts/IRecipesProvider.cs b/source/HomeBook.Backend.Module.Kitchen/Contracts/IRecipesProvider.cs new file mode 100644 index 00000000..7e8b7e29 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Contracts/IRecipesProvider.cs @@ -0,0 +1,120 @@ +using HomeBook.Backend.Data.Entities; +using HomeBook.Backend.Module.Kitchen.Models; + +namespace HomeBook.Backend.Module.Kitchen.Contracts; + +public interface IRecipesProvider +{ + /// + /// + /// + /// + /// + /// + Task GetRecipesAsync(string searchFilter, + CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + Task GetRecipeByIdAsync(Guid id, + CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + Task CreateAsync(string name, + Guid userId, + string? description, + int? servings, + int? durationWorkingMinutes, + int? durationCookingMinutes, + int? durationRestingMinutes, + int? caloriesKcal, + string? comments, + string? source, + CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + /// + /// + /// + Task AddIngredientToRecipeAsync(Guid recipeId, + string name, + double? quantity, + string? unit, + CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + Task GetIngredientByNameAsync(string name, + CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + Task CreateIngredientAsync(string name, + CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + /// + /// + /// + Task AddStepToRecipeAsync(Guid recipeId, + int position, + string description, + int? timerDurationInSeconds, + CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + Task DeleteAsync(Guid id, + CancellationToken cancellationToken); + + /// + /// + /// + /// + /// + /// + /// + Task UpdateNameAsync(Guid id, + string name, + CancellationToken cancellationToken); +} diff --git a/source/HomeBook.Backend.Module.Kitchen/Endpoints/RecipeEndpoints.cs b/source/HomeBook.Backend.Module.Kitchen/Endpoints/RecipeEndpoints.cs new file mode 100644 index 00000000..4248e9c2 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Endpoints/RecipeEndpoints.cs @@ -0,0 +1,79 @@ +using HomeBook.Backend.Core.Modules.OpenApi; +using HomeBook.Backend.Module.Kitchen.Handler; +using HomeBook.Backend.Module.Kitchen.Responses; +using HomeBook.Backend.Modules.Abstractions; +using Microsoft.AspNetCore.Builder; +using Microsoft.AspNetCore.Http; +using Microsoft.AspNetCore.Routing; + +namespace HomeBook.Backend.Module.Kitchen.Endpoints; + +public static class RecipeEndpoints +{ + public static IEndpointBuilder MapRecipeEndpoints(this IEndpointBuilder builder) + { + builder.AddEndpoint(routeBuilder => + { + RouteGroupBuilder group = routeBuilder + .MapGroup("/recipes") + .WithDescription("Endpoints to manage recipes informations") + .RequireAuthorization(); + + group.MapGet("/", RecipeHandler.HandleGetRecipes) + .WithName("GetRecipes") + .WithDescription(new Description( + "returns recipes matching the search filter", + "HTTP 200: Recipes were found", + "HTTP 401: User is not authorized", + "HTTP 500: Unknown error while getting recipes")) + .RequireAuthorization() + .Produces(StatusCodes.Status200OK) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status500InternalServerError); + + group.MapGet("/{id:guid}", RecipeHandler.HandleGetRecipeById) + .WithName("GetRecipeById") + .WithDescription(new Description( + "returns recipe by id", + "HTTP 200: Recipes were found", + "HTTP 404: Recipe not found", + "HTTP 401: User is not authorized", + "HTTP 500: Unknown error while getting recipes")) + .RequireAuthorization() + .Produces(StatusCodes.Status200OK) + .Produces(StatusCodes.Status404NotFound) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status500InternalServerError); + + group.MapPost("/", RecipeHandler.HandleCreateRecipe) + .WithName("CreateRecipe") + .WithDescription(new Description( + "creates a new recipe", + "HTTP 201: Recipe was created", + "HTTP 400: Invalid request data", + "HTTP 401: User is not authorized", + "HTTP 500: Unknown error while creating recipe")) + .RequireAuthorization() + .Produces(StatusCodes.Status201Created) + .Produces(StatusCodes.Status400BadRequest) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status500InternalServerError); + + group.MapDelete("/{id:guid}", RecipeHandler.HandleDeleteRecipe) + .WithName("DeleteRecipe") + .WithDescription(new Description( + "deletes an existing recipe", + "HTTP 200: Recipe was deleted", + "HTTP 401: User is not authorized", + "HTTP 404: Saving goal not found", + "HTTP 500: Unknown error while deleting recipe")) + .RequireAuthorization() + .Produces(StatusCodes.Status200OK) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status404NotFound) + .Produces(StatusCodes.Status500InternalServerError); + }); + + return builder; + } +} diff --git a/source/HomeBook.Backend.Module.Kitchen/Handler/RecipeHandler.cs b/source/HomeBook.Backend.Module.Kitchen/Handler/RecipeHandler.cs new file mode 100644 index 00000000..0de60460 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Handler/RecipeHandler.cs @@ -0,0 +1,221 @@ +using System.Security.Claims; +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.Core.Modules.Utilities; +using HomeBook.Backend.Data.Entities; +using HomeBook.Backend.Module.Kitchen.Contracts; +using HomeBook.Backend.Module.Kitchen.Mappings; +using HomeBook.Backend.Module.Kitchen.Models; +using HomeBook.Backend.Module.Kitchen.Requests; +using HomeBook.Backend.Module.Kitchen.Responses; +using Microsoft.AspNetCore.Http; +using Microsoft.AspNetCore.Mvc; +using Microsoft.Extensions.Logging; + +namespace HomeBook.Backend.Module.Kitchen.Handler; + +public class RecipeHandler +{ + /// + /// returns recipes matching the search filter + /// + /// + /// + /// + /// + /// + /// + public static async Task HandleGetRecipes(string searchFilter, + [FromServices] ILogger logger, + [FromServices] IRecipesProvider recipesProvider, + [FromServices] IUserProvider userProvider, + CancellationToken cancellationToken) + { + try + { + RecipeDto[] recipeDtos = await recipesProvider.GetRecipesAsync(searchFilter, + cancellationToken); + + List recipes = []; + foreach (RecipeDto recipeDto in recipeDtos) + { + RecipeResponse recipeResponse = await recipeDto.ToResponseAsync(async userId => + await userProvider.GetUserByIdAsync(userId, cancellationToken) + ); + recipes.Add(recipeResponse); + } + + return TypedResults.Ok(new RecipesListResponse(recipes.ToArray())); + } + catch (Exception err) + { + logger.LogError(err, "Error while getting recipes"); + return TypedResults.InternalServerError(err.Message); + } + } + + /// + /// returns recipe by id + /// + /// + /// + /// + /// + /// + /// + public static async Task HandleGetRecipeById(Guid id, + [FromServices] ILogger logger, + [FromServices] IRecipesProvider recipesProvider, + [FromServices] IUserProvider userProvider, + CancellationToken cancellationToken) + { + try + { + RecipeDto? recipeDto = await recipesProvider.GetRecipeByIdAsync(id, + cancellationToken); + + if (recipeDto is null) + return TypedResults.NotFound(); + + RecipeDetailResponse response = await recipeDto.ToDetailResponseAsync(async userId => + await userProvider.GetUserByIdAsync(userId, cancellationToken) + ); + return TypedResults.Ok(response); + } + catch (Exception err) + { + logger.LogError(err, "Error while getting recipes"); + return TypedResults.InternalServerError(err.Message); + } + } + + /// + /// + /// + /// + /// + /// + /// + /// + /// + public static async Task HandleCreateRecipe(ClaimsPrincipal user, + [FromBody] CreateRecipeRequest request, + [FromServices] ILogger logger, + [FromServices] IRecipesProvider recipesProvider, + CancellationToken cancellationToken) + { + try + { + Guid userId = user.GetUserId(); + + Guid createdRecipeId = await recipesProvider.CreateAsync(request.Name, + userId, + request.Description, + request.Servings, + request.DurationWorkingMinutes, + request.DurationCookingMinutes, + request.DurationRestingMinutes, + request.CaloriesKcal, + request.Comments, + request.Source, + cancellationToken); + + if (request.Ingredients is not null + && request.Ingredients.Length != 0) + foreach (CreateRecipeIngredientRequest ingredient in request.Ingredients) + { + await recipesProvider.AddIngredientToRecipeAsync(createdRecipeId, + ingredient.Name, + ingredient.Quantity, + ingredient.Unit, + cancellationToken); + } + + if (request.Steps is not null + && request.Steps.Length != 0) + foreach (CreateRecipeStepRequest si in request.Steps) + { + await recipesProvider.AddStepToRecipeAsync(createdRecipeId, + si.Position, + si.Description, + si.TimerDurationInSeconds, + cancellationToken); + } + + return TypedResults.Ok(); + } + catch (Exception err) + { + logger.LogError(err, "Error while creating recipe"); + return TypedResults.InternalServerError(err.Message); + } + } + + /// + /// + /// + /// + /// + /// + /// + /// + /// + public static async Task HandleDeleteRecipe(Guid id, + ClaimsPrincipal user, + [FromServices] ILogger logger, + [FromServices] IRecipesProvider recipesProvider, + CancellationToken cancellationToken) + { + try + { + Guid userId = user.GetUserId(); + + await recipesProvider.DeleteAsync(id, + cancellationToken); + + return TypedResults.Ok(); + } + catch (Exception err) + { + logger.LogError(err, + "Error while deleting recipe for {Id}", + id); + return TypedResults.InternalServerError(err.Message); + } + } + + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + public static async Task HandleUpdateRecipeName(Guid id, + ClaimsPrincipal user, + [FromBody] UpdateRecipeNameRequest request, + [FromServices] ILogger logger, + [FromServices] IRecipesProvider recipesProvider, + CancellationToken cancellationToken) + { + try + { + Guid userId = user.GetUserId(); + + await recipesProvider.UpdateNameAsync(id, + request.Name, + cancellationToken); + + return TypedResults.Ok(); + } + catch (Exception err) + { + logger.LogError(err, + "Error while updating recipe name for {Id}", + id); + return TypedResults.InternalServerError(err.Message); + } + } +} diff --git a/source/HomeBook.Backend.Core.Kitchen/HomeBook.Backend.Core.Kitchen.csproj b/source/HomeBook.Backend.Module.Kitchen/HomeBook.Backend.Module.Kitchen.csproj similarity index 56% rename from source/HomeBook.Backend.Core.Kitchen/HomeBook.Backend.Core.Kitchen.csproj rename to source/HomeBook.Backend.Module.Kitchen/HomeBook.Backend.Module.Kitchen.csproj index 1314ef4c..623d052a 100644 --- a/source/HomeBook.Backend.Core.Kitchen/HomeBook.Backend.Core.Kitchen.csproj +++ b/source/HomeBook.Backend.Module.Kitchen/HomeBook.Backend.Module.Kitchen.csproj @@ -7,12 +7,13 @@ - - + - + + + diff --git a/source/HomeBook.Backend.Module.Kitchen/Mappings/RecipeMappings.cs b/source/HomeBook.Backend.Module.Kitchen/Mappings/RecipeMappings.cs new file mode 100644 index 00000000..ba26b538 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Mappings/RecipeMappings.cs @@ -0,0 +1,142 @@ +using HomeBook.Backend.Abstractions.Models.UserManagement; +using HomeBook.Backend.Module.Kitchen.Models; +using HomeBook.Backend.Module.Kitchen.Responses; + +namespace HomeBook.Backend.Module.Kitchen.Mappings; + +public static class RecipeMappings +{ + public static RecipeDto ToDto(this Data.Entities.Recipe recipe) + { + return new RecipeDto( + recipe.Id, + recipe.UserId, + recipe.Name, + recipe.NormalizedName, + recipe.Description, + recipe.Servings, + recipe.DurationWorkingMinutes, + recipe.DurationCookingMinutes, + recipe.DurationRestingMinutes, + recipe.CaloriesKcal, + recipe.Comments, + recipe.Source, + recipe.Recipe2RecipeIngredient.Select(i => i.ToDto()).ToArray(), + recipe.Steps.Select(s => s.ToDto()).ToArray()); + } + + public static RecipeIngredientDto ToDto(this Data.Entities.Recipe2RecipeIngredient r2ri) + { + return new RecipeIngredientDto(r2ri.IngredientId, + r2ri.RecipeIngredient.Name, + r2ri.RecipeIngredient.NormalizedName, + r2ri.Quantity, + r2ri.Unit); + } + + public static RecipeStepDto ToDto(this Data.Entities.RecipeStep rs) + { + return new RecipeStepDto( + rs.RecipeId, + rs.Position, + rs.Description, + rs.TimerDurationInSeconds); + } + + public static async Task ToResponseAsync(this RecipeDto r, + Func> getUserInfoAsync) + { + string? username = null; + if (r.UserId.HasValue) + { + UserInfo? userInfo = await getUserInfoAsync(r.UserId.Value); + username = userInfo?.Username; + } + + return new RecipeResponse(r.Id, + username, + r.Name, + r.NormalizedName, + r.Description, + r.Servings, + + r.DurationWorkingMinutes, + r.DurationCookingMinutes, + r.DurationRestingMinutes, + r.CaloriesKcal, + r.Comments, + r.Source); + } + + public static async Task ToDetailResponseAsync(this RecipeDto recipe, + Func> getUserInfoAsync) + { + string? username = null; + if (recipe.UserId.HasValue) + { + UserInfo? userInfo = await getUserInfoAsync(recipe.UserId.Value); + username = userInfo?.Username; + } + + // var ingredients = new List + // { + // new(Guid.NewGuid(), + // 1, + // "Scheibe", + // "Schinken", + // "gekochter, oder anderer Belag, z.B. Putenbrust oder Salami"), + // new(Guid.NewGuid(), 1, null, "Ei", "gekocht"), + // new(Guid.NewGuid(), 1, "Scheibe", "Käse", "am besten Emmentaler , Ihr könnt aber auch anderen nehmen"), + // new(Guid.NewGuid(), 1, null, "Salatblatt", null), + // new(Guid.NewGuid(), 0.5, null, "Tomate", null), + // new(Guid.NewGuid(), null, null, "Salz und Pfeffer, Grillgewürz", null), + // new(Guid.NewGuid(), 3, "EL", "Mayonaise", null), + // new(Guid.NewGuid(), 2, "Scheiben", "Sandwich Toast", null), + // new(Guid.NewGuid(), 4, "x", "Zahnstocher", "o.ä. zum Fixieren") + // }; + // + // var steps = new List + // { + // new(Guid.NewGuid(), + // "Tomaten waschen und schneiden. Salat waschen und in einzelne Blätter teilen."), + // new(Guid.NewGuid(), + // "Eier hart kochen und anschließend in Scheiben schneiden.", + // 600), + // new(Guid.NewGuid(), + // "Beide Scheiben Toast einseitig mit Mayonaise ca. 2mm beschmieren. Den Rest Mayonaise benötigen wir noch später. Über den Toast ein bischen Salz,Pfeffer und bei Bedarf auch Grillgewürz streuen. Das Grillgewürz verleiht dem Ganzem einen \"neuen\" Geschmack."), + // }; + + return new RecipeDetailResponse(recipe.Id, + username, + recipe.Name, + recipe.NormalizedName, + recipe.Description, + recipe.Servings, + recipe.Ingredients.Select(x => x.ToResponse()).ToArray(), + recipe.Steps.Select(x => x.ToResponse()).ToArray(), + recipe.DurationWorkingMinutes, + recipe.DurationCookingMinutes, + recipe.DurationRestingMinutes, + recipe.CaloriesKcal, + recipe.Comments, + recipe.Source); + } + + public static RecipeIngredientResponse ToResponse(this RecipeIngredientDto ri) + { + return new RecipeIngredientResponse(ri.Id, + ri.Name, + ri.NormalizedName, + ri.Quantity, + ri.Unit); + } + + public static RecipeStepResponse ToResponse(this RecipeStepDto rs) + { + return new RecipeStepResponse( + rs.RecipeId, + rs.Position, + rs.Description, + rs.TimerDurationInSeconds); + } +} diff --git a/source/HomeBook.Backend.Module.Kitchen/Models/RecipeDto.cs b/source/HomeBook.Backend.Module.Kitchen/Models/RecipeDto.cs new file mode 100644 index 00000000..9ddb323a --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Models/RecipeDto.cs @@ -0,0 +1,17 @@ +namespace HomeBook.Backend.Module.Kitchen.Models; + +public record RecipeDto( + Guid Id, + Guid? UserId, + string Name, + string NormalizedName, + string? Description, + int? Servings, + int? DurationWorkingMinutes, + int? DurationCookingMinutes, + int? DurationRestingMinutes, + int? CaloriesKcal, + string? Comments, + string? Source, + RecipeIngredientDto[] Ingredients, + RecipeStepDto[] Steps); diff --git a/source/HomeBook.Backend.Module.Kitchen/Models/RecipeIngredientDto.cs b/source/HomeBook.Backend.Module.Kitchen/Models/RecipeIngredientDto.cs new file mode 100644 index 00000000..414174e0 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Models/RecipeIngredientDto.cs @@ -0,0 +1,7 @@ +namespace HomeBook.Backend.Module.Kitchen.Models; + +public record RecipeIngredientDto(Guid Id, + string Name, + string NormalizedName, + double? Quantity, + string? Unit); diff --git a/source/HomeBook.Backend.Module.Kitchen/Models/RecipeStepDto.cs b/source/HomeBook.Backend.Module.Kitchen/Models/RecipeStepDto.cs new file mode 100644 index 00000000..7538b198 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Models/RecipeStepDto.cs @@ -0,0 +1,7 @@ +namespace HomeBook.Backend.Module.Kitchen.Models; + +public record RecipeStepDto( + Guid RecipeId, + int Position, + string Description, + int? TimerDurationInSeconds); diff --git a/source/HomeBook.Backend.Module.Kitchen/Module.cs b/source/HomeBook.Backend.Module.Kitchen/Module.cs new file mode 100644 index 00000000..2eb57629 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Module.cs @@ -0,0 +1,56 @@ +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.Module.Kitchen.Contracts; +using HomeBook.Backend.Module.Kitchen.Endpoints; +using HomeBook.Backend.Module.Kitchen.Provider; +using HomeBook.Backend.Modules.Abstractions; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.DependencyInjection; + +namespace HomeBook.Backend.Module.Kitchen; + +public class Module : IModule, + IBackendModuleEndpointRegistrar, + IBackendModuleServiceRegistrar, + IBackendModuleSearchRegistrar +{ + public string Name { get; } = "Kitchen Module"; + public string Description { get; } = "Provides kitchen and recipe management features"; + public string Key { get; } = "kitchen"; + public string Author { get; } = "HomeBook"; + public Version Version { get; } = new("1.0.0"); + + public async Task InitializeAsync() + { + await Task.CompletedTask; + } + + public void RegisterEndpoints(IEndpointBuilder builder, IConfiguration configuration) + { + builder.MapRecipeEndpoints(); + } + + public static void RegisterServices(IServiceCollection services, IConfiguration configuration) + { + services.AddScoped(); + } + + public async Task SearchAsync(string query, + Guid userId, + CancellationToken cancellationToken = default) + { + await Task.Delay(3500, cancellationToken); // Simulate some search delay + + var items = new List + { + new SearchResultItem( + Title: "Expense Tracker", + Description: "Track your daily expenses", + Url: "/finances/expense-tracker", + Icon: "expense_icon", + Color: "red" + ) + }; + return new SearchResult(items.Count, + items); + } +} diff --git a/source/HomeBook.Backend.Module.Kitchen/Provider/RecipesProvider.cs b/source/HomeBook.Backend.Module.Kitchen/Provider/RecipesProvider.cs new file mode 100644 index 00000000..c9c23690 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Provider/RecipesProvider.cs @@ -0,0 +1,174 @@ +using HomeBook.Backend.Data.Contracts; +using HomeBook.Backend.Data.Entities; +using HomeBook.Backend.Module.Kitchen.Contracts; +using HomeBook.Backend.Module.Kitchen.Mappings; +using HomeBook.Backend.Module.Kitchen.Models; +using Microsoft.Extensions.Logging; + +namespace HomeBook.Backend.Module.Kitchen.Provider; + +/// +public class RecipesProvider( + ILogger logger, + IRecipesRepository recipesRepository, + IIngredientRepository ingredientRepository) : IRecipesProvider +{ + /// + public async Task GetRecipesAsync(string searchFilter, + CancellationToken cancellationToken) + { + logger.LogInformation("Retrieving meals with search filter: {SearchFilter}", + searchFilter); + + IEnumerable recipeEntities = await recipesRepository.GetAsync(searchFilter, + cancellationToken); + RecipeDto[] recipes = recipeEntities + .Select(m => m.ToDto()) + .ToArray(); + + return recipes; + } + + /// + public async Task GetRecipeByIdAsync(Guid id, + CancellationToken cancellationToken) => + (await recipesRepository.GetByIdAsync(id, + cancellationToken))?.ToDto(); + + /// + public async Task CreateAsync(string name, + Guid userId, + string? description, + int? servings, + int? durationWorkingMinutes, + int? durationCookingMinutes, + int? durationRestingMinutes, + int? caloriesKcal, + string? comments, + string? source, + CancellationToken cancellationToken) + { + Recipe entity = new() + { + UserId = userId, + Name = name, + NormalizedName = name, + Description = description, + DurationWorkingMinutes = durationWorkingMinutes, + DurationCookingMinutes = durationCookingMinutes, + DurationRestingMinutes = durationRestingMinutes, + CaloriesKcal = caloriesKcal, + Servings = servings, + Comments = comments, + Source = source + }; + + // TODO: validator + + Guid entityId = await recipesRepository + .CreateOrUpdateAsync(entity, + cancellationToken); + return entityId; + } + + /// + public async Task AddIngredientToRecipeAsync(Guid recipeId, + string name, + double? quantity, + string? unit, + CancellationToken cancellationToken) + { + string normalizedName = name; // TODO: normalize + RecipeIngredient? ingredient = await GetIngredientByNameAsync(name, + cancellationToken); + + Guid? ingredientId = ingredient?.Id ?? await CreateIngredientAsync(name, + cancellationToken); + + Recipe2RecipeIngredient entity = new() + { + RecipeId = recipeId, + IngredientId = ingredientId!.Value, + Quantity = quantity, + Unit = unit + }; + + // TODO: validator + + logger.LogInformation("Adding ingredient {IngredientName} to recipe {RecipeId}", + name, + recipeId); + + Recipe2RecipeIngredient updatedEntity = await recipesRepository + .CreateOrUpdateAsync(entity, + cancellationToken); + return updatedEntity; + } + + /// + public async Task GetIngredientByNameAsync(string name, + CancellationToken cancellationToken) => + await ingredientRepository.GetByName(name, + cancellationToken); + + /// + public async Task CreateIngredientAsync(string name, + CancellationToken cancellationToken) + { + RecipeIngredient entity = new() + { + Name = name + }; + + Guid entityId = await ingredientRepository.CreateOrUpdateAsync(entity, + cancellationToken); + + return entityId; + } + + /// + public async Task AddStepToRecipeAsync(Guid recipeId, + int position, + string description, + int? timerDurationInSeconds, + CancellationToken cancellationToken) + { + RecipeStep entity = new() + { + RecipeId = recipeId, + Description = description, + Position = position, + TimerDurationInSeconds = timerDurationInSeconds + }; + + // TODO: validator + + RecipeStep recipeStep = await recipesRepository.CreateRecipeStepAsync(entity, + cancellationToken); + return recipeStep; + } + + /// + public async Task DeleteAsync(Guid id, CancellationToken cancellationToken) => + await recipesRepository.DeleteAsync(id, + cancellationToken); + + /// + public async Task UpdateNameAsync(Guid id, + string name, + CancellationToken cancellationToken) + { + // Recipe entity = await recipesRepository.GetByIdAsync(id, + // cancellationToken) + // ?? throw new KeyNotFoundException( + // $"Recipe with id {id} not found"); + // + // entity.Name = name; + + // TODO: validator + + await recipesRepository.UpdateRecipeNameAsync(id, + name, + cancellationToken); + } +} diff --git a/source/HomeBook.Backend.Module.Kitchen/Requests/CreateRecipeIngredientRequest.cs b/source/HomeBook.Backend.Module.Kitchen/Requests/CreateRecipeIngredientRequest.cs new file mode 100644 index 00000000..5321ec48 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Requests/CreateRecipeIngredientRequest.cs @@ -0,0 +1,6 @@ +namespace HomeBook.Backend.Module.Kitchen.Requests; + +public record CreateRecipeIngredientRequest( + string Name, + double? Quantity, + string? Unit); diff --git a/source/HomeBook.Backend.Module.Kitchen/Requests/CreateRecipeRequest.cs b/source/HomeBook.Backend.Module.Kitchen/Requests/CreateRecipeRequest.cs new file mode 100644 index 00000000..25be5a6e --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Requests/CreateRecipeRequest.cs @@ -0,0 +1,14 @@ +namespace HomeBook.Backend.Module.Kitchen.Requests; + +public record CreateRecipeRequest( + string Name, + string? Description, + int? Servings, + CreateRecipeIngredientRequest[]? Ingredients, + CreateRecipeStepRequest[]? Steps, + int? DurationWorkingMinutes, + int? DurationCookingMinutes, + int? DurationRestingMinutes, + int? CaloriesKcal, + string? Comments, + string? Source); diff --git a/source/HomeBook.Backend.Module.Kitchen/Requests/CreateRecipeStepRequest.cs b/source/HomeBook.Backend.Module.Kitchen/Requests/CreateRecipeStepRequest.cs new file mode 100644 index 00000000..3a46cfde --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Requests/CreateRecipeStepRequest.cs @@ -0,0 +1,6 @@ +namespace HomeBook.Backend.Module.Kitchen.Requests; + +public record CreateRecipeStepRequest( + string Description, + int Position, + int? TimerDurationInSeconds); diff --git a/source/HomeBook.Backend.DTOs/Requests/Kitchen/UpdateRecipeNameRequest.cs b/source/HomeBook.Backend.Module.Kitchen/Requests/UpdateRecipeNameRequest.cs similarity index 50% rename from source/HomeBook.Backend.DTOs/Requests/Kitchen/UpdateRecipeNameRequest.cs rename to source/HomeBook.Backend.Module.Kitchen/Requests/UpdateRecipeNameRequest.cs index 1bb18d66..ead7aee9 100644 --- a/source/HomeBook.Backend.DTOs/Requests/Kitchen/UpdateRecipeNameRequest.cs +++ b/source/HomeBook.Backend.Module.Kitchen/Requests/UpdateRecipeNameRequest.cs @@ -1,3 +1,3 @@ -namespace HomeBook.Backend.DTOs.Requests.Kitchen; +namespace HomeBook.Backend.Module.Kitchen.Requests; public record UpdateRecipeNameRequest(string Name); diff --git a/source/HomeBook.Backend.Module.Kitchen/Responses/IngredientsListResponse.cs b/source/HomeBook.Backend.Module.Kitchen/Responses/IngredientsListResponse.cs new file mode 100644 index 00000000..56684ae1 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Responses/IngredientsListResponse.cs @@ -0,0 +1,3 @@ +namespace HomeBook.Backend.Module.Kitchen.Responses; + +public record IngredientsListResponse(RecipeIngredientResponse[] Ingredients); diff --git a/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeDetailResponse.cs b/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeDetailResponse.cs new file mode 100644 index 00000000..77277323 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeDetailResponse.cs @@ -0,0 +1,20 @@ +using System.Diagnostics; + +namespace HomeBook.Backend.Module.Kitchen.Responses; + +[DebuggerDisplay("{Name}")] +public record RecipeDetailResponse( + Guid Id, + string? Username, + string Name, + string NormalizedName, + string? Description, + int? Servings, + RecipeIngredientResponse[] Ingredients, + RecipeStepResponse[] Steps, + int? DurationWorkingMinutes, + int? DurationCookingMinutes, + int? DurationRestingMinutes, + int? CaloriesKcal, + string? Comments, + string? Source); diff --git a/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeIngredientResponse.cs b/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeIngredientResponse.cs new file mode 100644 index 00000000..3c380b84 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeIngredientResponse.cs @@ -0,0 +1,11 @@ +using System.Diagnostics; + +namespace HomeBook.Backend.Module.Kitchen.Responses; + +[DebuggerDisplay("{Quantity} {Unit} {Name}")] +public record RecipeIngredientResponse( + Guid Id, + string Name, + string? NormalizedName, + double? Quantity, + string? Unit); diff --git a/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeResponse.cs b/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeResponse.cs new file mode 100644 index 00000000..67568eea --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeResponse.cs @@ -0,0 +1,19 @@ +using System.Diagnostics; + +namespace HomeBook.Backend.Module.Kitchen.Responses; + +[DebuggerDisplay("{Name}")] +public record RecipeResponse( + Guid Id, + string? Username, + string Name, + string NormalizedName, + string? Description, + int? Servings, + + int? DurationWorkingMinutes, + int? DurationCookingMinutes, + int? DurationRestingMinutes, + int? CaloriesKcal, + string? Comments, + string? Source); diff --git a/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeStepResponse.cs b/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeStepResponse.cs new file mode 100644 index 00000000..e9a59404 --- /dev/null +++ b/source/HomeBook.Backend.Module.Kitchen/Responses/RecipeStepResponse.cs @@ -0,0 +1,10 @@ +using System.Diagnostics; + +namespace HomeBook.Backend.Module.Kitchen.Responses; + +[DebuggerDisplay("{Description}")] +public record RecipeStepResponse( + Guid RecipeId, + int Position, + string Description, + int? TimerDurationInSeconds = null); diff --git a/source/HomeBook.Backend.DTOs/Responses/Kitchen/RecipesListResponse.cs b/source/HomeBook.Backend.Module.Kitchen/Responses/RecipesListResponse.cs similarity index 53% rename from source/HomeBook.Backend.DTOs/Responses/Kitchen/RecipesListResponse.cs rename to source/HomeBook.Backend.Module.Kitchen/Responses/RecipesListResponse.cs index c9b71ad2..bc7cb083 100644 --- a/source/HomeBook.Backend.DTOs/Responses/Kitchen/RecipesListResponse.cs +++ b/source/HomeBook.Backend.Module.Kitchen/Responses/RecipesListResponse.cs @@ -1,3 +1,3 @@ -namespace HomeBook.Backend.DTOs.Responses.Kitchen; +namespace HomeBook.Backend.Module.Kitchen.Responses; public record RecipesListResponse(RecipeResponse[] Recipes); diff --git a/source/HomeBook.Backend.Modules.Abstractions/HomeBook.Backend.Modules.Abstractions.csproj b/source/HomeBook.Backend.Modules.Abstractions/HomeBook.Backend.Modules.Abstractions.csproj new file mode 100644 index 00000000..8344a5e2 --- /dev/null +++ b/source/HomeBook.Backend.Modules.Abstractions/HomeBook.Backend.Modules.Abstractions.csproj @@ -0,0 +1,23 @@ + + + + net9.0 + enable + enable + + + + + + + + + + + + + + + + + diff --git a/source/HomeBook.Backend.Modules.Abstractions/IBackendModuleEndpointRegistrar.cs b/source/HomeBook.Backend.Modules.Abstractions/IBackendModuleEndpointRegistrar.cs new file mode 100644 index 00000000..f2e74754 --- /dev/null +++ b/source/HomeBook.Backend.Modules.Abstractions/IBackendModuleEndpointRegistrar.cs @@ -0,0 +1,9 @@ +using Microsoft.Extensions.Configuration; + +namespace HomeBook.Backend.Modules.Abstractions; + +public interface IBackendModuleEndpointRegistrar +{ + void RegisterEndpoints(IEndpointBuilder builder, + IConfiguration configuration); +} diff --git a/source/HomeBook.Backend.Modules.Abstractions/IBackendModuleSearchRegistrar.cs b/source/HomeBook.Backend.Modules.Abstractions/IBackendModuleSearchRegistrar.cs new file mode 100644 index 00000000..5ef5648a --- /dev/null +++ b/source/HomeBook.Backend.Modules.Abstractions/IBackendModuleSearchRegistrar.cs @@ -0,0 +1,30 @@ +namespace HomeBook.Backend.Modules.Abstractions; + +public interface IBackendModuleSearchRegistrar +{ + /// + /// the display name of this module + /// + string Name { get; } + + /// + /// the key of this module (used for endpoint grouping, etc.) + /// + string Key { get; } + + /// + /// the author of this module + /// + string Author { get; } + + /// + /// + /// + /// + /// + /// + /// + Task SearchAsync(string query, + Guid userId, + CancellationToken cancellationToken = default); +} diff --git a/source/HomeBook.Backend.Modules.Abstractions/IBackendModuleServiceRegistrar.cs b/source/HomeBook.Backend.Modules.Abstractions/IBackendModuleServiceRegistrar.cs new file mode 100644 index 00000000..ff4c94e7 --- /dev/null +++ b/source/HomeBook.Backend.Modules.Abstractions/IBackendModuleServiceRegistrar.cs @@ -0,0 +1,15 @@ +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.DependencyInjection; + +namespace HomeBook.Backend.Modules.Abstractions; + +public interface IBackendModuleServiceRegistrar +{ + /// + /// register required services for this module. + /// + /// + /// + static abstract void RegisterServices(IServiceCollection services, + IConfiguration configuration); +} diff --git a/source/HomeBook.Backend.Modules.Abstractions/IEndpointBuilder.cs b/source/HomeBook.Backend.Modules.Abstractions/IEndpointBuilder.cs new file mode 100644 index 00000000..e04c35e1 --- /dev/null +++ b/source/HomeBook.Backend.Modules.Abstractions/IEndpointBuilder.cs @@ -0,0 +1,8 @@ +using Microsoft.AspNetCore.Routing; + +namespace HomeBook.Backend.Modules.Abstractions; + +public interface IEndpointBuilder +{ + IEndpointBuilder AddEndpoint(Action groupBuilderAction); +} diff --git a/source/HomeBook.Backend.Modules.Abstractions/IModule.cs b/source/HomeBook.Backend.Modules.Abstractions/IModule.cs new file mode 100644 index 00000000..edca8316 --- /dev/null +++ b/source/HomeBook.Backend.Modules.Abstractions/IModule.cs @@ -0,0 +1,34 @@ +namespace HomeBook.Backend.Modules.Abstractions; + +public interface IModule +{ + /// + /// the display name of this module + /// + string Name { get; } + + /// + /// the description of this module (for open api for example) + /// + string Description { get; } + + /// + /// the key of this module (used for endpoint grouping, etc.) + /// + string Key { get; } + + /// + /// the author of this module + /// + string Author { get; } + + /// + /// the version of this module + /// + Version Version { get; } + + /// + /// contains the initialization logic for the module. + /// + Task InitializeAsync(); +} diff --git a/source/HomeBook.Backend.Modules.Abstractions/SearchResultItem.cs b/source/HomeBook.Backend.Modules.Abstractions/SearchResultItem.cs new file mode 100644 index 00000000..b569ee43 --- /dev/null +++ b/source/HomeBook.Backend.Modules.Abstractions/SearchResultItem.cs @@ -0,0 +1,14 @@ +using HomeBook.Backend.Abstractions.Contracts; + +namespace HomeBook.Backend.Modules.Abstractions; + +public record SearchResult( + int TotalCount, + IEnumerable Items); + +public record SearchResultItem( + string Title, + string? Description, + string Url, + string Icon, + string Color) : ISearchResultItem; diff --git a/source/HomeBook.Backend/Endpoints/FinanceCalculationsEndpoints.cs b/source/HomeBook.Backend/Endpoints/FinanceCalculationsEndpoints.cs deleted file mode 100644 index 85cc48e6..00000000 --- a/source/HomeBook.Backend/Endpoints/FinanceCalculationsEndpoints.cs +++ /dev/null @@ -1,34 +0,0 @@ -using HomeBook.Backend.DTOs.Responses.Finances; -using HomeBook.Backend.Handler; -using HomeBook.Backend.OpenApi; - -namespace HomeBook.Backend.Endpoints; - -public static class FinanceCalculationsEndpoints -{ - public static IEndpointRouteBuilder MapFinancesCalculationEndpoints(this IEndpointRouteBuilder routeBuilder) - { - RouteGroupBuilder group = routeBuilder - .MapGroup("/finances/calculations") - .WithDescription("Endpoints for finances calculations") - .RequireAuthorization(); - - group.MapPost("/savings", FinanceCalculationsHandler.HandleCalculateSavings) - .WithName("CalculateSavings") - .WithTags("Finances", "Savings", "Calculate") - .WithDescription(new Description( - "returns the calculated savings based on the provided parameters", - "HTTP 200: successfully calculated savings", - "HTTP 401: User is not authorized", - "HTTP 500: Unknown error while getting saving goals")) - .RequireAuthorization() - .WithOpenApi(operation => new(operation) - { - }) - .Produces(StatusCodes.Status200OK) - .Produces(StatusCodes.Status401Unauthorized) - .Produces(StatusCodes.Status500InternalServerError); - - return routeBuilder; - } -} diff --git a/source/HomeBook.Backend/Endpoints/FinanceSavingGoalEndpoints.cs b/source/HomeBook.Backend/Endpoints/FinanceSavingGoalEndpoints.cs deleted file mode 100644 index 47bdd07b..00000000 --- a/source/HomeBook.Backend/Endpoints/FinanceSavingGoalEndpoints.cs +++ /dev/null @@ -1,151 +0,0 @@ -using HomeBook.Backend.DTOs.Responses.Finances; -using HomeBook.Backend.Handler; -using HomeBook.Backend.OpenApi; -using HomeBook.Backend.Responses; - -namespace HomeBook.Backend.Endpoints; - -public static class FinanceSavingGoalEndpoints -{ - public static IEndpointRouteBuilder MapFinancesSavingGoalEndpoints(this IEndpointRouteBuilder routeBuilder) - { - RouteGroupBuilder group = routeBuilder - .MapGroup("/finances/saving-goals") - .WithDescription("Endpoints for finances saving goals") - .RequireAuthorization(); - - group.MapGet("/", FinanceSavingGoalHandler.HandleGetSavingGoals) - .WithName("GetSavingGoals") - .WithTags("Finances", "SavingGoals") - .WithDescription(new Description( - "returns the users finances saving goals", - "HTTP 200: Finances saving goals were found", - "HTTP 401: User is not authorized", - "HTTP 500: Unknown error while getting saving goals")) - .RequireAuthorization() - .WithOpenApi(operation => new(operation) - { - }) - .Produces(StatusCodes.Status200OK) - .Produces(StatusCodes.Status401Unauthorized) - .Produces(StatusCodes.Status500InternalServerError); - - group.MapPost("/", FinanceSavingGoalHandler.HandleCreateSavingGoal) - .WithName("CreateSavingGoal") - .WithTags("Finances", "SavingGoals") - .WithDescription(new Description( - "creates a new finances saving goal for the user", - "HTTP 201: Finances saving goal was created", - "HTTP 400: Invalid request data", - "HTTP 401: User is not authorized", - "HTTP 500: Unknown error while creating saving goal")) - .RequireAuthorization() - .WithOpenApi(operation => new(operation) - { - }) - .Produces(StatusCodes.Status201Created) - .Produces(StatusCodes.Status400BadRequest) - .Produces(StatusCodes.Status401Unauthorized) - .Produces(StatusCodes.Status500InternalServerError); - - group.MapPatch("/{savingGoalId:guid}/name", FinanceSavingGoalHandler.HandleUpdateSavingGoalName) - .WithName("UpdateSavingGoalName") - .WithTags("Finances", "SavingGoals") - .WithDescription(new Description( - "updates the name of an existing finances saving goal for the user", - "HTTP 200: Finances saving goal was updated", - "HTTP 400: Invalid request data", - "HTTP 401: User is not authorized", - "HTTP 404: Saving goal not found", - "HTTP 500: Unknown error while updating saving goal")) - .RequireAuthorization() - .WithOpenApi(operation => new(operation) - { - }) - .Produces(StatusCodes.Status200OK) - .Produces(StatusCodes.Status400BadRequest) - .Produces(StatusCodes.Status401Unauthorized) - .Produces(StatusCodes.Status404NotFound) - .Produces(StatusCodes.Status500InternalServerError); - - group.MapPatch("/{savingGoalId:guid}/appearance", FinanceSavingGoalHandler.HandleUpdateSavingGoalAppearance) - .WithName("UpdateSavingGoalAppearance") - .WithTags("Finances", "SavingGoals") - .WithDescription(new Description( - "updates the appearance of an existing finances saving goal for the user", - "HTTP 200: Finances saving goal was updated", - "HTTP 400: Invalid request data", - "HTTP 401: User is not authorized", - "HTTP 404: Saving goal not found", - "HTTP 500: Unknown error while updating saving goal")) - .RequireAuthorization() - .WithOpenApi(operation => new(operation) - { - }) - .Produces(StatusCodes.Status200OK) - .Produces(StatusCodes.Status400BadRequest) - .Produces(StatusCodes.Status401Unauthorized) - .Produces(StatusCodes.Status404NotFound) - .Produces(StatusCodes.Status500InternalServerError); - - group.MapPatch("/{savingGoalId:guid}/amounts", FinanceSavingGoalHandler.HandleUpdateSavingGoalAmounts) - .WithName("UpdateSavingGoalAmounts") - .WithTags("Finances", "SavingGoals") - .WithDescription(new Description( - "updates the amounts of an existing finances saving goal for the user", - "HTTP 200: Finances saving goal was updated", - "HTTP 400: Invalid request data", - "HTTP 401: User is not authorized", - "HTTP 404: Saving goal not found", - "HTTP 500: Unknown error while updating saving goal")) - .RequireAuthorization() - .WithOpenApi(operation => new(operation) - { - }) - .Produces(StatusCodes.Status200OK) - .Produces(StatusCodes.Status400BadRequest) - .Produces(StatusCodes.Status401Unauthorized) - .Produces(StatusCodes.Status404NotFound) - .Produces(StatusCodes.Status500InternalServerError); - - group.MapPatch("/{savingGoalId:guid}/info", FinanceSavingGoalHandler.HandleUpdateSavingGoalInfo) - .WithName("UpdateSavingGoalInfo") - .WithTags("Finances", "SavingGoals") - .WithDescription(new Description( - "updates the info (target date, etc.) of an existing finances saving goal for the user", - "HTTP 200: Finances saving goal was updated", - "HTTP 400: Invalid request data", - "HTTP 401: User is not authorized", - "HTTP 404: Saving goal not found", - "HTTP 500: Unknown error while updating saving goal")) - .RequireAuthorization() - .WithOpenApi(operation => new(operation) - { - }) - .Produces(StatusCodes.Status200OK) - .Produces(StatusCodes.Status400BadRequest) - .Produces(StatusCodes.Status401Unauthorized) - .Produces(StatusCodes.Status404NotFound) - .Produces(StatusCodes.Status500InternalServerError); - - group.MapDelete("/{id:guid}", FinanceSavingGoalHandler.HandleDeleteSavingGoal) - .WithName("DeleteSavingGoal") - .WithTags("Finances", "SavingGoals") - .WithDescription(new Description( - "deletes an existing finances saving goal for the user", - "HTTP 200: Finances saving goal was deleted", - "HTTP 401: User is not authorized", - "HTTP 404: Saving goal not found", - "HTTP 500: Unknown error while deleting saving goal")) - .RequireAuthorization() - .WithOpenApi(operation => new(operation) - { - }) - .Produces(StatusCodes.Status200OK) - .Produces(StatusCodes.Status401Unauthorized) - .Produces(StatusCodes.Status404NotFound) - .Produces(StatusCodes.Status500InternalServerError); - - return routeBuilder; - } -} diff --git a/source/HomeBook.Backend/Endpoints/KitchenRecipeEndpoints.cs b/source/HomeBook.Backend/Endpoints/KitchenRecipeEndpoints.cs deleted file mode 100644 index 4df73e7d..00000000 --- a/source/HomeBook.Backend/Endpoints/KitchenRecipeEndpoints.cs +++ /dev/null @@ -1,70 +0,0 @@ -using HomeBook.Backend.DTOs.Responses.Kitchen; -using HomeBook.Backend.Handler; -using HomeBook.Backend.OpenApi; - -namespace HomeBook.Backend.Endpoints; - -public static class KitchenRecipeEndpoints -{ - public static IEndpointRouteBuilder MapKitchenRecipeEndpoints(this IEndpointRouteBuilder routeBuilder) - { - RouteGroupBuilder group = routeBuilder - .MapGroup("/kitchen/recipes") - .WithDescription("Endpoints to manage recipes informations") - .RequireAuthorization(); - - group.MapGet("/", KitchenRecipeHandler.HandleGetRecipes) - .WithName("GetRecipes") - .WithTags("Kitchen", "Recipes") - .WithDescription(new Description( - "returns recipes matching the search filter", - "HTTP 200: Recipes were found", - "HTTP 401: User is not authorized", - "HTTP 500: Unknown error while getting recipes")) - .RequireAuthorization() - .WithOpenApi(operation => new(operation) - { - }) - .Produces(StatusCodes.Status200OK) - .Produces(StatusCodes.Status401Unauthorized) - .Produces(StatusCodes.Status500InternalServerError); - - group.MapPost("/", KitchenRecipeHandler.HandleCreateRecipe) - .WithName("CreateRecipe") - .WithTags("Kitchen", "Recipes") - .WithDescription(new Description( - "creates a new recipe", - "HTTP 201: Recipe was created", - "HTTP 400: Invalid request data", - "HTTP 401: User is not authorized", - "HTTP 500: Unknown error while creating recipe")) - .RequireAuthorization() - .WithOpenApi(operation => new(operation) - { - }) - .Produces(StatusCodes.Status201Created) - .Produces(StatusCodes.Status400BadRequest) - .Produces(StatusCodes.Status401Unauthorized) - .Produces(StatusCodes.Status500InternalServerError); - - group.MapDelete("/{id:guid}", KitchenRecipeHandler.HandleDeleteRecipe) - .WithName("DeleteRecipe") - .WithTags("Kitchen", "Recipes") - .WithDescription(new Description( - "deletes an existing recipe", - "HTTP 200: Recipe was deleted", - "HTTP 401: User is not authorized", - "HTTP 404: Saving goal not found", - "HTTP 500: Unknown error while deleting recipe")) - .RequireAuthorization() - .WithOpenApi(operation => new(operation) - { - }) - .Produces(StatusCodes.Status200OK) - .Produces(StatusCodes.Status401Unauthorized) - .Produces(StatusCodes.Status404NotFound) - .Produces(StatusCodes.Status500InternalServerError); - - return routeBuilder; - } -} diff --git a/source/HomeBook.Backend/Endpoints/SearchEndpoints.cs b/source/HomeBook.Backend/Endpoints/SearchEndpoints.cs new file mode 100644 index 00000000..1e47f5ce --- /dev/null +++ b/source/HomeBook.Backend/Endpoints/SearchEndpoints.cs @@ -0,0 +1,34 @@ +using HomeBook.Backend.Core.Modules.OpenApi; +using HomeBook.Backend.DTOs.Responses.Search; +using HomeBook.Backend.Handler; + +namespace HomeBook.Backend.Endpoints; + +public static class SearchEndpoints +{ + public static IEndpointRouteBuilder MapSearchEndpoints(this IEndpointRouteBuilder routeBuilder) + { + RouteGroupBuilder group = routeBuilder + .MapGroup("/search") + .WithDescription("Endpoints for search functionality") + .RequireAuthorization(); + + group.MapGet("/", SearchHandler.HandleSearch) + .WithName("Search") + .WithTags("Search") + .WithDescription(new Description( + "returns search results based on the query", + "HTTP 200: Search results were found", + "HTTP 401: User is not authorized", + "HTTP 500: Unknown error while getting preference")) + .RequireAuthorization() + .WithOpenApi(operation => new(operation) + { + }) + .Produces(StatusCodes.Status200OK) + .Produces(StatusCodes.Status401Unauthorized) + .Produces(StatusCodes.Status500InternalServerError); + + return routeBuilder; + } +} diff --git a/source/HomeBook.Backend/Endpoints/SetupEndpoints.cs b/source/HomeBook.Backend/Endpoints/SetupEndpoints.cs index 5b3b4e0c..53875de9 100644 --- a/source/HomeBook.Backend/Endpoints/SetupEndpoints.cs +++ b/source/HomeBook.Backend/Endpoints/SetupEndpoints.cs @@ -1,5 +1,5 @@ +using HomeBook.Backend.Core.Modules.OpenApi; using HomeBook.Backend.Handler; -using HomeBook.Backend.OpenApi; using HomeBook.Backend.Requests; using HomeBook.Backend.Responses; diff --git a/source/HomeBook.Backend/Endpoints/UpdateEndpoints.cs b/source/HomeBook.Backend/Endpoints/UpdateEndpoints.cs index 571bfc51..3f1301bc 100644 --- a/source/HomeBook.Backend/Endpoints/UpdateEndpoints.cs +++ b/source/HomeBook.Backend/Endpoints/UpdateEndpoints.cs @@ -1,5 +1,5 @@ +using HomeBook.Backend.Core.Modules.OpenApi; using HomeBook.Backend.Handler; -using HomeBook.Backend.OpenApi; namespace HomeBook.Backend.Endpoints; diff --git a/source/HomeBook.Backend/Endpoints/UserEndpoints.cs b/source/HomeBook.Backend/Endpoints/UserEndpoints.cs index 36629a5e..d28f9a62 100644 --- a/source/HomeBook.Backend/Endpoints/UserEndpoints.cs +++ b/source/HomeBook.Backend/Endpoints/UserEndpoints.cs @@ -1,5 +1,5 @@ +using HomeBook.Backend.Core.Modules.OpenApi; using HomeBook.Backend.Handler; -using HomeBook.Backend.OpenApi; using HomeBook.Backend.Responses; namespace HomeBook.Backend.Endpoints; diff --git a/source/HomeBook.Backend/Endpoints/VersionEndpoints.cs b/source/HomeBook.Backend/Endpoints/VersionEndpoints.cs index 285aa799..de91b9a7 100644 --- a/source/HomeBook.Backend/Endpoints/VersionEndpoints.cs +++ b/source/HomeBook.Backend/Endpoints/VersionEndpoints.cs @@ -15,9 +15,6 @@ public static IEndpointRouteBuilder MapVersionEndpoints(this IEndpointRouteBuild .WithTags("Version") .WithDescription("returns the version of the backend service") // .RequireAuthorization(policy => policy.RequireRole("read")) - .WithOpenApi(operation => new(operation) - { - }) .Produces(StatusCodes.Status200OK) .Produces(StatusCodes.Status500InternalServerError); diff --git a/source/HomeBook.Backend/Extensions/ServiceCollectionExtensions.cs b/source/HomeBook.Backend/Extensions/ServiceCollectionExtensions.cs index d6eb7237..a84d3a6a 100644 --- a/source/HomeBook.Backend/Extensions/ServiceCollectionExtensions.cs +++ b/source/HomeBook.Backend/Extensions/ServiceCollectionExtensions.cs @@ -6,13 +6,11 @@ using HomeBook.Backend.Core; using HomeBook.Backend.Core.Account.Extensions; using HomeBook.Backend.Core.DataProvider; -using HomeBook.Backend.Core.DataProvider.Extensions; using HomeBook.Backend.Core.Extensions; -using HomeBook.Backend.Core.Finances.Extensions; using HomeBook.Backend.Core.HashProvider; -using HomeBook.Backend.Core.Kitchen.Extensions; using HomeBook.Backend.Core.Licenses; using HomeBook.Backend.Core.Licenses.Extensions; +using HomeBook.Backend.Core.Search.Extensions; using Homebook.Backend.Core.Setup; using Homebook.Backend.Core.Setup.Extensions; using Homebook.Backend.Core.Setup.Factories; @@ -20,12 +18,10 @@ using Homebook.Backend.Core.Setup.Provider; using Homebook.Backend.Core.Setup.Validators; using HomeBook.Backend.Data; -using HomeBook.Backend.Data.Entities; using HomeBook.Backend.Data.Extensions; using HomeBook.Backend.Data.Mysql.Extensions; using HomeBook.Backend.Data.PostgreSql.Extensions; using HomeBook.Backend.Data.Sqlite.Extensions; -using HomeBook.Backend.Data.Validators; using HomeBook.Backend.Factories; using HomeBook.Backend.Provider; using HomeBook.Backend.Services; @@ -47,15 +43,13 @@ public static IServiceCollection AddDependenciesForRuntime(this IServiceCollecti IConfiguration configuration, InstanceStatus instanceStatus) { - services.AddBackendServices(configuration, instanceStatus); - services.AddBackendCore(configuration, instanceStatus); - services.AddBackendCoreSetup(configuration, instanceStatus); - services.AddBackendCoreLicenses(configuration, instanceStatus); - services.AddBackendDatabaseProvider(configuration, instanceStatus); - services.AddAccountServices(configuration, instanceStatus); - - services.AddBackendCoreFinances(configuration, instanceStatus); - services.AddBackendCoreKitchen(configuration, instanceStatus); + services.AddBackendServices(configuration, instanceStatus) + .AddBackendCore(configuration, instanceStatus) + .AddBackendCoreSetup(configuration, instanceStatus) + .AddBackendCoreLicenses(configuration, instanceStatus) + .AddBackendCoreSearch(configuration, instanceStatus) + .AddBackendDatabaseProvider(configuration, instanceStatus) + .AddAccountServices(configuration, instanceStatus); return services; } diff --git a/source/HomeBook.Backend/Factories/ModuleBuilderExtensions.cs b/source/HomeBook.Backend/Factories/ModuleBuilderExtensions.cs new file mode 100644 index 00000000..3442fafd --- /dev/null +++ b/source/HomeBook.Backend/Factories/ModuleBuilderExtensions.cs @@ -0,0 +1,16 @@ +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.ModuleCore; + +namespace HomeBook.Backend.Factories; + +public static class ModuleBuilderExtensions +{ + public static void RegisterModulesInSearchFactory(this ModuleBuilder moduleBuilder, + ISearchRegistrationInitiator searchRegistrationInitiator) + { + foreach (string moduleId in moduleBuilder.SearchEnabledModules) + { + searchRegistrationInitiator.AddModule(moduleId); + } + } +} diff --git a/source/HomeBook.Backend/Handler/InfoHandler.cs b/source/HomeBook.Backend/Handler/InfoHandler.cs index 04f6e0ce..309c638e 100644 --- a/source/HomeBook.Backend/Handler/InfoHandler.cs +++ b/source/HomeBook.Backend/Handler/InfoHandler.cs @@ -21,7 +21,7 @@ public static async Task HandleGetInstanceInfo( (instanceDefaultName ?? string.Empty)); return TypedResults.Ok(response); } - catch (Exception) + catch (Exception err) { return TypedResults.Problem("An error occurred while retrieving instance information.", statusCode: 500); } diff --git a/source/HomeBook.Backend/Handler/KitchenRecipeHandler.cs b/source/HomeBook.Backend/Handler/KitchenRecipeHandler.cs deleted file mode 100644 index b38ed89b..00000000 --- a/source/HomeBook.Backend/Handler/KitchenRecipeHandler.cs +++ /dev/null @@ -1,143 +0,0 @@ -using System.Security.Claims; -using HomeBook.Backend.Core.Kitchen.Contracts; -using HomeBook.Backend.Core.Kitchen.Models; -using HomeBook.Backend.DTOs.Requests.Kitchen; -using HomeBook.Backend.DTOs.Responses.Kitchen; -using HomeBook.Backend.Mappings; -using HomeBook.Backend.Utilities; -using Microsoft.AspNetCore.Mvc; - -namespace HomeBook.Backend.Handler; - -public class KitchenRecipeHandler -{ - /// - /// returns recipes matching the search filter - /// - /// - /// - /// - /// - /// - public static async Task HandleGetRecipes(string searchFilter, - [FromServices] ILogger logger, - [FromServices] IRecipesProvider recipesProvider, - CancellationToken cancellationToken) - { - try - { - RecipeDto[] recipeDtos = await recipesProvider.GetRecipesAsync(searchFilter, - cancellationToken); - RecipeResponse[] recipes = recipeDtos - .Select(sg => sg.ToResponse()) - .ToArray(); - - return TypedResults.Ok(new RecipesListResponse(recipes)); - } - catch (Exception err) - { - logger.LogError(err, "Error while getting recipes"); - return TypedResults.InternalServerError(err.Message); - } - } - - /// - /// - /// - /// - /// - /// - /// - /// - /// - public static async Task HandleCreateRecipe(ClaimsPrincipal user, - [FromBody] CreateRecipeRequest request, - [FromServices] ILogger logger, - [FromServices] IRecipesProvider recipesProvider, - CancellationToken cancellationToken) - { - try - { - Guid userId = user.GetUserId(); - - Guid createdId = await recipesProvider.CreateAsync(request.Name, - cancellationToken); - - return TypedResults.Ok(); - } - catch (Exception err) - { - logger.LogError(err, "Error while creating recipe"); - return TypedResults.InternalServerError(err.Message); - } - } - - /// - /// - /// - /// - /// - /// - /// - /// - /// - public static async Task HandleDeleteRecipe(Guid id, - ClaimsPrincipal user, - [FromServices] ILogger logger, - [FromServices] IRecipesProvider recipesProvider, - CancellationToken cancellationToken) - { - try - { - Guid userId = user.GetUserId(); - - await recipesProvider.DeleteAsync(id, - cancellationToken); - - return TypedResults.Ok(); - } - catch (Exception err) - { - logger.LogError(err, - "Error while deleting recipe for {Id}", - id); - return TypedResults.InternalServerError(err.Message); - } - } - - /// - /// - /// - /// - /// - /// - /// - /// - /// - /// - public static async Task HandleUpdateRecipeName(Guid id, - ClaimsPrincipal user, - [FromBody] UpdateRecipeNameRequest request, - [FromServices] ILogger logger, - [FromServices] IRecipesProvider recipesProvider, - CancellationToken cancellationToken) - { - try - { - Guid userId = user.GetUserId(); - - await recipesProvider.UpdateNameAsync(id, - request.Name, - cancellationToken); - - return TypedResults.Ok(); - } - catch (Exception err) - { - logger.LogError(err, - "Error while updating recipe name for {Id}", - id); - return TypedResults.InternalServerError(err.Message); - } - } -} diff --git a/source/HomeBook.Backend/Handler/SearchHandler.cs b/source/HomeBook.Backend/Handler/SearchHandler.cs new file mode 100644 index 00000000..91250168 --- /dev/null +++ b/source/HomeBook.Backend/Handler/SearchHandler.cs @@ -0,0 +1,47 @@ +using System.Security.Claims; +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.Core.Modules.Utilities; +using HomeBook.Backend.DTOs.Responses.Search; +using HomeBook.Backend.Mappings; +using Microsoft.AspNetCore.Mvc; + +namespace HomeBook.Backend.Handler; + +public class SearchHandler +{ + /// + /// + /// + /// + /// + /// + /// + /// + /// + public static async Task HandleSearch(ClaimsPrincipal user, + [FromQuery(Name = "s")] string query, + [FromServices] ILogger logger, + [FromServices] ISearchRegistrationFactory searchRegistrationFactory, + CancellationToken cancellationToken) + { + try + { + ISearchProvider searchProvider = searchRegistrationFactory + .CreateSearchProvider(); + IEnumerable searchAggregationResults = await searchProvider + .SearchAsync(query, + user.GetUserId(), + cancellationToken); + + SearchResponse response = searchAggregationResults.ToResponse(); + return TypedResults.Ok(response); + } + catch (Exception err) + { + logger.LogError(err, + "Error while handling search request for query '{Query}'", + query); + return TypedResults.InternalServerError(err.Message); + } + } +} diff --git a/source/HomeBook.Backend/Handler/UserHandler.cs b/source/HomeBook.Backend/Handler/UserHandler.cs index dffcee8a..8ead6537 100644 --- a/source/HomeBook.Backend/Handler/UserHandler.cs +++ b/source/HomeBook.Backend/Handler/UserHandler.cs @@ -1,9 +1,9 @@ using Microsoft.AspNetCore.Mvc; using System.Security.Claims; using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.Core.Modules.Utilities; using HomeBook.Backend.Requests; using HomeBook.Backend.Responses; -using HomeBook.Backend.Utilities; namespace HomeBook.Backend.Handler; diff --git a/source/HomeBook.Backend/HomeBook.Backend.csproj b/source/HomeBook.Backend/HomeBook.Backend.csproj index f9f4aea8..36fe1958 100644 --- a/source/HomeBook.Backend/HomeBook.Backend.csproj +++ b/source/HomeBook.Backend/HomeBook.Backend.csproj @@ -4,7 +4,6 @@ net9.0 enable enable - . @@ -19,12 +18,12 @@ - - + + all runtime; build; native; contentfiles; analyzers; buildtransitive - + @@ -44,8 +43,7 @@ - - + @@ -54,10 +52,11 @@ - - + + + diff --git a/source/HomeBook.Backend/HomeBookBuilder.cs b/source/HomeBook.Backend/HomeBookBuilder.cs new file mode 100644 index 00000000..54831aef --- /dev/null +++ b/source/HomeBook.Backend/HomeBookBuilder.cs @@ -0,0 +1,29 @@ +using System.Runtime.CompilerServices; +using HomeBook.Backend.Options; + +namespace HomeBook.Backend; + +public static class HomeBookBuilder +{ + private static readonly ConditionalWeakTable _options = new(); + + /// + /// default HomeBook configuration + /// + /// + /// + public static HomeBookOptions HomeBook(this WebApplicationBuilder builder) => + _options.GetValue(builder, _ => new HomeBookOptions()); + + public static void BuildHomeBook(this WebApplicationBuilder builder) + { + HomeBookOptions options = builder.HomeBook(); + + // if (builder.HomeBook().StartMenuBuilder is IStartMenuRegistrator smr) + // { + // smr.RegisterStartMenuItems( + // builder.Services, + // builder.Configuration); + // } + } +} diff --git a/source/HomeBook.Backend/Mappings/MealMappings.cs b/source/HomeBook.Backend/Mappings/MealMappings.cs deleted file mode 100644 index d2f8e732..00000000 --- a/source/HomeBook.Backend/Mappings/MealMappings.cs +++ /dev/null @@ -1,14 +0,0 @@ -using HomeBook.Backend.Core.Kitchen.Models; -using HomeBook.Backend.DTOs.Responses.Kitchen; - -namespace HomeBook.Backend.Mappings; - -public static class MealMappings -{ - public static RecipeResponse ToResponse(this RecipeDto recipe) - { - return new RecipeResponse(recipe.Id, - recipe.Name, - recipe.NormalizedName); - } -} diff --git a/source/HomeBook.Backend/Mappings/SavingGoalMappings.cs b/source/HomeBook.Backend/Mappings/SavingGoalMappings.cs deleted file mode 100644 index 9759a28b..00000000 --- a/source/HomeBook.Backend/Mappings/SavingGoalMappings.cs +++ /dev/null @@ -1,23 +0,0 @@ -using HomeBook.Backend.Core.Finances.Models; -using HomeBook.Backend.DTOs.Responses.Finances; -using HomeBook.Backend.Responses; - -namespace HomeBook.Backend.Mappings; - -public static class SavingGoalMappings -{ - public static SavingGoalResponse ToResponse(this SavingGoalDto savingGoal) - { - return new SavingGoalResponse( - savingGoal.Id, - savingGoal.Name, - savingGoal.Color, - savingGoal.Icon, - savingGoal.TargetAmount, - savingGoal.CurrentAmount, - savingGoal.MonthlyPayment, - savingGoal.InterestRateOption, - savingGoal.InterestRate, - savingGoal.TargetDate); - } -} diff --git a/source/HomeBook.Backend/Mappings/SearchMappings.cs b/source/HomeBook.Backend/Mappings/SearchMappings.cs new file mode 100644 index 00000000..45072df3 --- /dev/null +++ b/source/HomeBook.Backend/Mappings/SearchMappings.cs @@ -0,0 +1,28 @@ +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.DTOs.Responses.Search; + +namespace HomeBook.Backend.Mappings; + +public static class SearchMappings +{ + public static SearchResponse ToResponse(this IEnumerable results) + { + List moduleResponses = []; + moduleResponses.AddRange(results + .Select(result => new SearchModuleResponse( + result.ModuleKey, + result.TotalCount, + result.Items.Select(x => x.ToResponse())))); + + SearchResponse response = new(moduleResponses.ToArray()); + return response; + } + + public static SearchItemResponse ToResponse(this ISearchResultItem item) => + new( + item.Title, + item.Description, + item.Url, + item.Icon, + item.Color); +} diff --git a/source/HomeBook.Backend/ModuleCore/EndpointBuilder.cs b/source/HomeBook.Backend/ModuleCore/EndpointBuilder.cs new file mode 100644 index 00000000..b0a256ac --- /dev/null +++ b/source/HomeBook.Backend/ModuleCore/EndpointBuilder.cs @@ -0,0 +1,16 @@ +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.Modules.Abstractions; + +namespace HomeBook.Backend.ModuleCore; + +public class EndpointBuilder(IEndpointRouteBuilder groupBuilder) + : IEndpointBuilder, + IEndpointDataAccessor +{ + public IEndpointBuilder AddEndpoint(Action groupBuilderAction) + { + groupBuilderAction(groupBuilder); + + return this; + } +} diff --git a/source/HomeBook.Backend/ModuleCore/ModuleBuilder.cs b/source/HomeBook.Backend/ModuleCore/ModuleBuilder.cs new file mode 100644 index 00000000..d1da18bc --- /dev/null +++ b/source/HomeBook.Backend/ModuleCore/ModuleBuilder.cs @@ -0,0 +1,52 @@ +using System.Reflection; +using HomeBook.Backend.Modules.Abstractions; +using HomeBook.Backend.Options; + +namespace HomeBook.Backend.ModuleCore; + +public class ModuleBuilder( + HomeBookOptions homeBookOptions, + IServiceCollection serviceCollection, + IConfiguration configuration) +{ + private readonly List _searchEnabledModules = []; + + public IReadOnlyList SearchEnabledModules => _searchEnabledModules.AsReadOnly(); + + /// + /// adds a module to the service collection if the module is enabled. + /// + /// + /// + /// + public ModuleBuilder AddModule() where T : class, IModule + { + string moduleId = typeof(T).FullName + ?? throw new InvalidOperationException("Module type must have a full name."); + + // register the module + RegisterModule(moduleId); + + return this; + } + + private void RegisterModule(string moduleId) where T : class, IModule + { + // register the IModule itself + serviceCollection.AddSingleton(); + serviceCollection.AddKeyedSingleton(moduleId); + + // implements the Module the IBackendModuleServiceRegistrar interface? + if (typeof(IBackendModuleServiceRegistrar).IsAssignableFrom(typeof(T))) + { + MethodInfo? method = typeof(T).GetMethod( + "RegisterServices", + BindingFlags.Public | BindingFlags.Static + ); + method?.Invoke(null, [serviceCollection, configuration]); + } + + if (typeof(IBackendModuleSearchRegistrar).IsAssignableFrom(typeof(T))) + _searchEnabledModules.Add(moduleId); + } +} diff --git a/source/HomeBook.Backend/ModuleCore/ModuleExtensions.cs b/source/HomeBook.Backend/ModuleCore/ModuleExtensions.cs new file mode 100644 index 00000000..bb060346 --- /dev/null +++ b/source/HomeBook.Backend/ModuleCore/ModuleExtensions.cs @@ -0,0 +1,138 @@ +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.Core.Search; +using HomeBook.Backend.Factories; +using HomeBook.Backend.Modules.Abstractions; +using HomeBook.Backend.Options; + +namespace HomeBook.Backend.ModuleCore; + +public static class ModuleExtensions +{ + private static ModuleBuilder? _moduleBuilder = null; + private static SearchRegistrationFactory? _searchRegistrationFactory = null; + + /// + /// use in Blazor Server + /// + /// + /// + /// + public static void AddModules(this WebApplicationBuilder builder, + HomeBookOptions homeBookOptions, + Action builderAction) + { + builder.Services.AddModules( + homeBookOptions, + builder.Configuration, + builderAction); + } + + /// + /// + /// + /// + /// + /// + /// + public static void AddModules(this IServiceCollection sc, + HomeBookOptions hb, + IConfiguration c, + Action builderAction) + { + _searchRegistrationFactory = new(); + _moduleBuilder = new ModuleBuilder(hb, sc, c); + builderAction(_moduleBuilder); + } + + /// + /// use in Blazor Server + /// + /// + public static async Task RunModulesPostBuild(this WebApplication host) + { + CancellationToken cancellationToken = CancellationToken.None; + + ISearchRegistrationInitiator searchRegistrationInitiator = host.Services + .GetRequiredService(); + searchRegistrationInitiator.AddServiceProvider(host.Services); + + // register the search provider with modules + // sc.AddSingleton(x => + // { + // _searchRegistrationFactory.AddServiceProvider(x); + // return _searchRegistrationFactory!; + // }); + + await host.RunModulesPostBuild(host.Services, + host.Configuration); + + // call startup service if needed + } + + /// + /// general post build logic + /// + /// + /// + /// + public static async Task RunModulesPostBuild(this WebApplication host, + IServiceProvider sp, + IConfiguration c) + { + if (_moduleBuilder is null) + return; + + // register search enabled modules in search registration factory + ISearchRegistrationInitiator searchRegistrationInitiator = sp + .GetRequiredService(); + _moduleBuilder.RegisterModulesInSearchFactory(searchRegistrationInitiator); + + IEnumerable modules = sp.GetServices(); + + // initialize all modules + foreach (IModule module in modules) + { + // register endpoints + try + { + await host.RegisterEndpointsForModuleAsync(module); + } + catch (NotImplementedException) + { + // do nothing + } + + // call the initialization logic + try + { + await module.InitializeAsync(); + } + catch (NotImplementedException) + { + // do nothing + } + } + } + + public static async Task RegisterEndpointsForModuleAsync(this WebApplication host, + IModule module) + { + if (module is not IBackendModuleEndpointRegistrar registrar) + return; + IBackendModuleEndpointRegistrar endpointRegistrar = registrar; + + IConfiguration configuration = host.Configuration; + + // register endpoint group for module + RouteGroupBuilder moduleEndpointGroup = host.MapGroup($"/modules/{module.Key}") + .WithDescription(module.Description) + .WithTags([ + module.Name + ]); + + IEndpointBuilder builder = new EndpointBuilder(moduleEndpointGroup); + endpointRegistrar.RegisterEndpoints(builder, configuration); + + IEndpointDataAccessor endpointDataAccessor = (IEndpointDataAccessor)builder; + } +} diff --git a/source/HomeBook.Backend/Options/HomeBookOptions.cs b/source/HomeBook.Backend/Options/HomeBookOptions.cs new file mode 100644 index 00000000..3ca6d9ad --- /dev/null +++ b/source/HomeBook.Backend/Options/HomeBookOptions.cs @@ -0,0 +1,6 @@ +namespace HomeBook.Backend.Options; + +public class HomeBookOptions +{ + +} \ No newline at end of file diff --git a/source/HomeBook.Backend/Program.cs b/source/HomeBook.Backend/Program.cs index cdac1f26..650f5850 100644 --- a/source/HomeBook.Backend/Program.cs +++ b/source/HomeBook.Backend/Program.cs @@ -1,9 +1,11 @@ +using HomeBook.Backend; using HomeBook.Backend.Abstractions; using HomeBook.Backend.Endpoints; using HomeBook.Backend.EnvironmentHandler; using HomeBook.Backend.Extensions; using HomeBook.Backend.Core.Account.Extensions; using HomeBook.Backend.Middleware; +using HomeBook.Backend.ModuleCore; using Scalar.AspNetCore; using Serilog; @@ -12,7 +14,7 @@ EnvironmentLoader.LoadEnvFile(developmentEnvFile); #endif -var builder = WebApplication.CreateBuilder(args); +WebApplicationBuilder builder = WebApplication.CreateBuilder(args); builder.Configuration.Sources.Clear(); builder.Configuration .AddJsonFile("appsettings.json", optional: false, reloadOnChange: true) @@ -50,7 +52,6 @@ builder.Services.AddJwtAuthentication(builder.Configuration, instanceStatus); if (builder.Environment.IsDevelopment()) -{ builder.Services.AddCors(options => { options.AddDefaultPolicy(policy => @@ -60,9 +61,19 @@ .AllowAnyMethod(); }); }); -} -var app = builder.Build(); +if (instanceStatus == InstanceStatus.RUNNING) + builder.AddModules( + builder.HomeBook(), + (moduleBuilder) => + { + // app modules + moduleBuilder + .AddModule() + .AddModule(); + }); + +WebApplication app = builder.Build(); Log.Information("HomeBook Backend application starting up - Version: {Version}", app.Configuration["Version"] ?? "Unknown"); @@ -105,12 +116,13 @@ .MapAccountEndpoints() .MapInfoEndpoints() .MapUserEndpoints() - .MapFinancesCalculationEndpoints() - .MapFinancesSavingGoalEndpoints() - .MapKitchenRecipeEndpoints(); + .MapSearchEndpoints(); break; } #endregion +if (instanceStatus == InstanceStatus.RUNNING) + await app.RunModulesPostBuild(); + app.Run(); diff --git a/source/HomeBook.Backend/appsettings.json b/source/HomeBook.Backend/appsettings.json index 8562cd02..a44ae904 100644 --- a/source/HomeBook.Backend/appsettings.json +++ b/source/HomeBook.Backend/appsettings.json @@ -1,4 +1,5 @@ { + "Version": "1.0.0", "Logging": { "LogLevel": { "Default": "Information", @@ -27,6 +28,5 @@ "Issuer": "HomeBook", "Audience": "HomeBook", "ExpirationMinutes": 60 - }, - "Version": "1.0.110" + } } diff --git a/source/HomeBook.Client/BackendClient.cs b/source/HomeBook.Client/BackendClient.cs index f5d987ee..4240941d 100644 --- a/source/HomeBook.Client/BackendClient.cs +++ b/source/HomeBook.Client/BackendClient.cs @@ -1,10 +1,10 @@ // #pragma warning disable CS0618 using HomeBook.Client.Account; -using HomeBook.Client.Finances; using HomeBook.Client.Info; -using HomeBook.Client.Kitchen; +using HomeBook.Client.Modules; using HomeBook.Client.Platform; +using HomeBook.Client.Search; using HomeBook.Client.Setup; using HomeBook.Client.System; using HomeBook.Client.Update; @@ -33,26 +33,26 @@ public partial class BackendClient : BaseRequestBuilder { get => new global::HomeBook.Client.Account.AccountRequestBuilder(PathParameters, RequestAdapter); } - /// The finances property - public global::HomeBook.Client.Finances.FinancesRequestBuilder Finances - { - get => new global::HomeBook.Client.Finances.FinancesRequestBuilder(PathParameters, RequestAdapter); - } /// The info property public global::HomeBook.Client.Info.InfoRequestBuilder Info { get => new global::HomeBook.Client.Info.InfoRequestBuilder(PathParameters, RequestAdapter); } - /// The kitchen property - public global::HomeBook.Client.Kitchen.KitchenRequestBuilder Kitchen + /// The modules property + public global::HomeBook.Client.Modules.ModulesRequestBuilder Modules { - get => new global::HomeBook.Client.Kitchen.KitchenRequestBuilder(PathParameters, RequestAdapter); + get => new global::HomeBook.Client.Modules.ModulesRequestBuilder(PathParameters, RequestAdapter); } /// The platform property public global::HomeBook.Client.Platform.PlatformRequestBuilder Platform { get => new global::HomeBook.Client.Platform.PlatformRequestBuilder(PathParameters, RequestAdapter); } + /// The search property + public global::HomeBook.Client.Search.SearchRequestBuilder Search + { + get => new global::HomeBook.Client.Search.SearchRequestBuilder(PathParameters, RequestAdapter); + } /// The setup property public global::HomeBook.Client.Setup.SetupRequestBuilder Setup { diff --git a/source/HomeBook.Client/Finances/SavingGoals/Item/WithSavingGoalItemRequestBuilder.cs b/source/HomeBook.Client/Finances/SavingGoals/Item/WithSavingGoalItemRequestBuilder.cs deleted file mode 100644 index 583a48ce..00000000 --- a/source/HomeBook.Client/Finances/SavingGoals/Item/WithSavingGoalItemRequestBuilder.cs +++ /dev/null @@ -1,124 +0,0 @@ -// -#pragma warning disable CS0618 -using HomeBook.Client.Finances.SavingGoals.Item.Amounts; -using HomeBook.Client.Finances.SavingGoals.Item.Appearance; -using HomeBook.Client.Finances.SavingGoals.Item.Info; -using HomeBook.Client.Finances.SavingGoals.Item.Name; -using Microsoft.Kiota.Abstractions.Extensions; -using Microsoft.Kiota.Abstractions.Serialization; -using Microsoft.Kiota.Abstractions; -using System.Collections.Generic; -using System.IO; -using System.Threading.Tasks; -using System.Threading; -using System; -namespace HomeBook.Client.Finances.SavingGoals.Item -{ - /// - /// Builds and executes requests for operations under \finances\saving-goals\{savingGoalId} - /// - [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] - public partial class WithSavingGoalItemRequestBuilder : BaseRequestBuilder - { - /// The amounts property - public global::HomeBook.Client.Finances.SavingGoals.Item.Amounts.AmountsRequestBuilder Amounts - { - get => new global::HomeBook.Client.Finances.SavingGoals.Item.Amounts.AmountsRequestBuilder(PathParameters, RequestAdapter); - } - /// The appearance property - public global::HomeBook.Client.Finances.SavingGoals.Item.Appearance.AppearanceRequestBuilder Appearance - { - get => new global::HomeBook.Client.Finances.SavingGoals.Item.Appearance.AppearanceRequestBuilder(PathParameters, RequestAdapter); - } - /// The info property - public global::HomeBook.Client.Finances.SavingGoals.Item.Info.InfoRequestBuilder Info - { - get => new global::HomeBook.Client.Finances.SavingGoals.Item.Info.InfoRequestBuilder(PathParameters, RequestAdapter); - } - /// The name property - public global::HomeBook.Client.Finances.SavingGoals.Item.Name.NameRequestBuilder Name - { - get => new global::HomeBook.Client.Finances.SavingGoals.Item.Name.NameRequestBuilder(PathParameters, RequestAdapter); - } - /// - /// Instantiates a new and sets the default values. - /// - /// Path parameters for the request - /// The request adapter to use to execute the requests. - public WithSavingGoalItemRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{savingGoalId}?id={id}", pathParameters) - { - } - /// - /// Instantiates a new and sets the default values. - /// - /// The raw URL to use for the request builder. - /// The request adapter to use to execute the requests. - public WithSavingGoalItemRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{savingGoalId}?id={id}", rawUrl) - { - } - /// - /// deletes an existing finances saving goal for the userHTTP 200: Finances saving goal was deletedHTTP 401: User is not authorizedHTTP 404: Saving goal not foundHTTP 500: Unknown error while deleting saving goal - /// - /// A - /// Cancellation token to use when cancelling requests - /// Configuration for the request such as headers, query parameters, and middleware options. -#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER -#nullable enable - public async Task DeleteAsync(Action>? requestConfiguration = default, CancellationToken cancellationToken = default) - { -#nullable restore -#else - public async Task DeleteAsync(Action> requestConfiguration = default, CancellationToken cancellationToken = default) - { -#endif - var requestInfo = ToDeleteRequestInformation(requestConfiguration); - return await RequestAdapter.SendPrimitiveAsync(requestInfo, default, cancellationToken).ConfigureAwait(false); - } - /// - /// deletes an existing finances saving goal for the userHTTP 200: Finances saving goal was deletedHTTP 401: User is not authorizedHTTP 404: Saving goal not foundHTTP 500: Unknown error while deleting saving goal - /// - /// A - /// Configuration for the request such as headers, query parameters, and middleware options. -#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER -#nullable enable - public RequestInformation ToDeleteRequestInformation(Action>? requestConfiguration = default) - { -#nullable restore -#else - public RequestInformation ToDeleteRequestInformation(Action> requestConfiguration = default) - { -#endif - var requestInfo = new RequestInformation(Method.DELETE, UrlTemplate, PathParameters); - requestInfo.Configure(requestConfiguration); - requestInfo.Headers.TryAdd("Accept", "application/json"); - return requestInfo; - } - /// - /// Returns a request builder with the provided arbitrary URL. Using this method means any other path or query parameters are ignored. - /// - /// A - /// The raw URL to use for the request builder. - public global::HomeBook.Client.Finances.SavingGoals.Item.WithSavingGoalItemRequestBuilder WithUrl(string rawUrl) - { - return new global::HomeBook.Client.Finances.SavingGoals.Item.WithSavingGoalItemRequestBuilder(rawUrl, RequestAdapter); - } - /// - /// deletes an existing finances saving goal for the userHTTP 200: Finances saving goal was deletedHTTP 401: User is not authorizedHTTP 404: Saving goal not foundHTTP 500: Unknown error while deleting saving goal - /// - [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] - public partial class WithSavingGoalItemRequestBuilderDeleteQueryParameters - { - [QueryParameter("id")] - public Guid? Id { get; set; } - } - /// - /// Configuration for the request such as headers, query parameters, and middleware options. - /// - [Obsolete("This class is deprecated. Please use the generic RequestConfiguration class generated by the generator.")] - [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] - public partial class WithSavingGoalItemRequestBuilderDeleteRequestConfiguration : RequestConfiguration - { - } - } -} -#pragma warning restore CS0618 diff --git a/source/HomeBook.Client/Models/CreateRecipeIngredientRequest.cs b/source/HomeBook.Client/Models/CreateRecipeIngredientRequest.cs new file mode 100644 index 00000000..47a4f15e --- /dev/null +++ b/source/HomeBook.Client/Models/CreateRecipeIngredientRequest.cs @@ -0,0 +1,79 @@ +// +#pragma warning disable CS0618 +using Microsoft.Kiota.Abstractions.Extensions; +using Microsoft.Kiota.Abstractions.Serialization; +using System.Collections.Generic; +using System.IO; +using System; +namespace HomeBook.Client.Models +{ + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + #pragma warning disable CS1591 + public partial class CreateRecipeIngredientRequest : IAdditionalDataHolder, IParsable + #pragma warning restore CS1591 + { + /// Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well. + public IDictionary AdditionalData { get; set; } + /// The name property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Name { get; set; } +#nullable restore +#else + public string Name { get; set; } +#endif + /// The quantity property + public double? Quantity { get; set; } + /// The unit property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Unit { get; set; } +#nullable restore +#else + public string Unit { get; set; } +#endif + /// + /// Instantiates a new and sets the default values. + /// + public CreateRecipeIngredientRequest() + { + AdditionalData = new Dictionary(); + } + /// + /// Creates a new instance of the appropriate class based on discriminator value + /// + /// A + /// The parse node to use to read the discriminator value and create the object + public static global::HomeBook.Client.Models.CreateRecipeIngredientRequest CreateFromDiscriminatorValue(IParseNode parseNode) + { + if(ReferenceEquals(parseNode, null)) throw new ArgumentNullException(nameof(parseNode)); + return new global::HomeBook.Client.Models.CreateRecipeIngredientRequest(); + } + /// + /// The deserialization information for the current model + /// + /// A IDictionary<string, Action<IParseNode>> + public virtual IDictionary> GetFieldDeserializers() + { + return new Dictionary> + { + { "name", n => { Name = n.GetStringValue(); } }, + { "quantity", n => { Quantity = n.GetDoubleValue(); } }, + { "unit", n => { Unit = n.GetStringValue(); } }, + }; + } + /// + /// Serializes information the current object + /// + /// Serialization writer to use to serialize this model + public virtual void Serialize(ISerializationWriter writer) + { + if(ReferenceEquals(writer, null)) throw new ArgumentNullException(nameof(writer)); + writer.WriteStringValue("name", Name); + writer.WriteDoubleValue("quantity", Quantity); + writer.WriteStringValue("unit", Unit); + writer.WriteAdditionalData(AdditionalData); + } + } +} +#pragma warning restore CS0618 diff --git a/source/HomeBook.Client/Models/CreateRecipeRequest.cs b/source/HomeBook.Client/Models/CreateRecipeRequest.cs index 2fe7990e..42fd621d 100644 --- a/source/HomeBook.Client/Models/CreateRecipeRequest.cs +++ b/source/HomeBook.Client/Models/CreateRecipeRequest.cs @@ -14,6 +14,38 @@ public partial class CreateRecipeRequest : IAdditionalDataHolder, IParsable { /// Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well. public IDictionary AdditionalData { get; set; } + /// The caloriesKcal property + public int? CaloriesKcal { get; set; } + /// The comments property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Comments { get; set; } +#nullable restore +#else + public string Comments { get; set; } +#endif + /// The description property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Description { get; set; } +#nullable restore +#else + public string Description { get; set; } +#endif + /// The durationCookingMinutes property + public int? DurationCookingMinutes { get; set; } + /// The durationRestingMinutes property + public int? DurationRestingMinutes { get; set; } + /// The durationWorkingMinutes property + public int? DurationWorkingMinutes { get; set; } + /// The ingredients property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public List? Ingredients { get; set; } +#nullable restore +#else + public List Ingredients { get; set; } +#endif /// The name property #if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER #nullable enable @@ -21,6 +53,24 @@ public partial class CreateRecipeRequest : IAdditionalDataHolder, IParsable #nullable restore #else public string Name { get; set; } +#endif + /// The servings property + public int? Servings { get; set; } + /// The source property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Source { get; set; } +#nullable restore +#else + public string Source { get; set; } +#endif + /// The steps property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public List? Steps { get; set; } +#nullable restore +#else + public List Steps { get; set; } #endif /// /// Instantiates a new and sets the default values. @@ -47,7 +97,17 @@ public virtual IDictionary> GetFieldDeserializers() { return new Dictionary> { + { "caloriesKcal", n => { CaloriesKcal = n.GetIntValue(); } }, + { "comments", n => { Comments = n.GetStringValue(); } }, + { "description", n => { Description = n.GetStringValue(); } }, + { "durationCookingMinutes", n => { DurationCookingMinutes = n.GetIntValue(); } }, + { "durationRestingMinutes", n => { DurationRestingMinutes = n.GetIntValue(); } }, + { "durationWorkingMinutes", n => { DurationWorkingMinutes = n.GetIntValue(); } }, + { "ingredients", n => { Ingredients = n.GetCollectionOfObjectValues(global::HomeBook.Client.Models.CreateRecipeIngredientRequest.CreateFromDiscriminatorValue)?.AsList(); } }, { "name", n => { Name = n.GetStringValue(); } }, + { "servings", n => { Servings = n.GetIntValue(); } }, + { "source", n => { Source = n.GetStringValue(); } }, + { "steps", n => { Steps = n.GetCollectionOfObjectValues(global::HomeBook.Client.Models.CreateRecipeStepRequest.CreateFromDiscriminatorValue)?.AsList(); } }, }; } /// @@ -57,7 +117,17 @@ public virtual IDictionary> GetFieldDeserializers() public virtual void Serialize(ISerializationWriter writer) { if(ReferenceEquals(writer, null)) throw new ArgumentNullException(nameof(writer)); + writer.WriteIntValue("caloriesKcal", CaloriesKcal); + writer.WriteStringValue("comments", Comments); + writer.WriteStringValue("description", Description); + writer.WriteIntValue("durationCookingMinutes", DurationCookingMinutes); + writer.WriteIntValue("durationRestingMinutes", DurationRestingMinutes); + writer.WriteIntValue("durationWorkingMinutes", DurationWorkingMinutes); + writer.WriteCollectionOfObjectValues("ingredients", Ingredients); writer.WriteStringValue("name", Name); + writer.WriteIntValue("servings", Servings); + writer.WriteStringValue("source", Source); + writer.WriteCollectionOfObjectValues("steps", Steps); writer.WriteAdditionalData(AdditionalData); } } diff --git a/source/HomeBook.Client/Models/CreateRecipeStepRequest.cs b/source/HomeBook.Client/Models/CreateRecipeStepRequest.cs new file mode 100644 index 00000000..c33520d2 --- /dev/null +++ b/source/HomeBook.Client/Models/CreateRecipeStepRequest.cs @@ -0,0 +1,73 @@ +// +#pragma warning disable CS0618 +using Microsoft.Kiota.Abstractions.Extensions; +using Microsoft.Kiota.Abstractions.Serialization; +using System.Collections.Generic; +using System.IO; +using System; +namespace HomeBook.Client.Models +{ + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + #pragma warning disable CS1591 + public partial class CreateRecipeStepRequest : IAdditionalDataHolder, IParsable + #pragma warning restore CS1591 + { + /// Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well. + public IDictionary AdditionalData { get; set; } + /// The description property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Description { get; set; } +#nullable restore +#else + public string Description { get; set; } +#endif + /// The position property + public int? Position { get; set; } + /// The timerDurationInSeconds property + public int? TimerDurationInSeconds { get; set; } + /// + /// Instantiates a new and sets the default values. + /// + public CreateRecipeStepRequest() + { + AdditionalData = new Dictionary(); + } + /// + /// Creates a new instance of the appropriate class based on discriminator value + /// + /// A + /// The parse node to use to read the discriminator value and create the object + public static global::HomeBook.Client.Models.CreateRecipeStepRequest CreateFromDiscriminatorValue(IParseNode parseNode) + { + if(ReferenceEquals(parseNode, null)) throw new ArgumentNullException(nameof(parseNode)); + return new global::HomeBook.Client.Models.CreateRecipeStepRequest(); + } + /// + /// The deserialization information for the current model + /// + /// A IDictionary<string, Action<IParseNode>> + public virtual IDictionary> GetFieldDeserializers() + { + return new Dictionary> + { + { "description", n => { Description = n.GetStringValue(); } }, + { "position", n => { Position = n.GetIntValue(); } }, + { "timerDurationInSeconds", n => { TimerDurationInSeconds = n.GetIntValue(); } }, + }; + } + /// + /// Serializes information the current object + /// + /// Serialization writer to use to serialize this model + public virtual void Serialize(ISerializationWriter writer) + { + if(ReferenceEquals(writer, null)) throw new ArgumentNullException(nameof(writer)); + writer.WriteStringValue("description", Description); + writer.WriteIntValue("position", Position); + writer.WriteIntValue("timerDurationInSeconds", TimerDurationInSeconds); + writer.WriteAdditionalData(AdditionalData); + } + } +} +#pragma warning restore CS0618 diff --git a/source/HomeBook.Client/Models/RecipeDetailResponse.cs b/source/HomeBook.Client/Models/RecipeDetailResponse.cs new file mode 100644 index 00000000..6205954d --- /dev/null +++ b/source/HomeBook.Client/Models/RecipeDetailResponse.cs @@ -0,0 +1,159 @@ +// +#pragma warning disable CS0618 +using Microsoft.Kiota.Abstractions.Extensions; +using Microsoft.Kiota.Abstractions.Serialization; +using System.Collections.Generic; +using System.IO; +using System; +namespace HomeBook.Client.Models +{ + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + #pragma warning disable CS1591 + public partial class RecipeDetailResponse : IAdditionalDataHolder, IParsable + #pragma warning restore CS1591 + { + /// Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well. + public IDictionary AdditionalData { get; set; } + /// The caloriesKcal property + public int? CaloriesKcal { get; set; } + /// The comments property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Comments { get; set; } +#nullable restore +#else + public string Comments { get; set; } +#endif + /// The description property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Description { get; set; } +#nullable restore +#else + public string Description { get; set; } +#endif + /// The durationCookingMinutes property + public int? DurationCookingMinutes { get; set; } + /// The durationRestingMinutes property + public int? DurationRestingMinutes { get; set; } + /// The durationWorkingMinutes property + public int? DurationWorkingMinutes { get; set; } + /// The id property + public Guid? Id { get; set; } + /// The ingredients property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public List? Ingredients { get; set; } +#nullable restore +#else + public List Ingredients { get; set; } +#endif + /// The name property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Name { get; set; } +#nullable restore +#else + public string Name { get; set; } +#endif + /// The normalizedName property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? NormalizedName { get; set; } +#nullable restore +#else + public string NormalizedName { get; set; } +#endif + /// The servings property + public int? Servings { get; set; } + /// The source property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Source { get; set; } +#nullable restore +#else + public string Source { get; set; } +#endif + /// The steps property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public List? Steps { get; set; } +#nullable restore +#else + public List Steps { get; set; } +#endif + /// The username property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Username { get; set; } +#nullable restore +#else + public string Username { get; set; } +#endif + /// + /// Instantiates a new and sets the default values. + /// + public RecipeDetailResponse() + { + AdditionalData = new Dictionary(); + } + /// + /// Creates a new instance of the appropriate class based on discriminator value + /// + /// A + /// The parse node to use to read the discriminator value and create the object + public static global::HomeBook.Client.Models.RecipeDetailResponse CreateFromDiscriminatorValue(IParseNode parseNode) + { + if(ReferenceEquals(parseNode, null)) throw new ArgumentNullException(nameof(parseNode)); + return new global::HomeBook.Client.Models.RecipeDetailResponse(); + } + /// + /// The deserialization information for the current model + /// + /// A IDictionary<string, Action<IParseNode>> + public virtual IDictionary> GetFieldDeserializers() + { + return new Dictionary> + { + { "caloriesKcal", n => { CaloriesKcal = n.GetIntValue(); } }, + { "comments", n => { Comments = n.GetStringValue(); } }, + { "description", n => { Description = n.GetStringValue(); } }, + { "durationCookingMinutes", n => { DurationCookingMinutes = n.GetIntValue(); } }, + { "durationRestingMinutes", n => { DurationRestingMinutes = n.GetIntValue(); } }, + { "durationWorkingMinutes", n => { DurationWorkingMinutes = n.GetIntValue(); } }, + { "id", n => { Id = n.GetGuidValue(); } }, + { "ingredients", n => { Ingredients = n.GetCollectionOfObjectValues(global::HomeBook.Client.Models.RecipeIngredientResponse.CreateFromDiscriminatorValue)?.AsList(); } }, + { "name", n => { Name = n.GetStringValue(); } }, + { "normalizedName", n => { NormalizedName = n.GetStringValue(); } }, + { "servings", n => { Servings = n.GetIntValue(); } }, + { "source", n => { Source = n.GetStringValue(); } }, + { "steps", n => { Steps = n.GetCollectionOfObjectValues(global::HomeBook.Client.Models.RecipeStepResponse.CreateFromDiscriminatorValue)?.AsList(); } }, + { "username", n => { Username = n.GetStringValue(); } }, + }; + } + /// + /// Serializes information the current object + /// + /// Serialization writer to use to serialize this model + public virtual void Serialize(ISerializationWriter writer) + { + if(ReferenceEquals(writer, null)) throw new ArgumentNullException(nameof(writer)); + writer.WriteIntValue("caloriesKcal", CaloriesKcal); + writer.WriteStringValue("comments", Comments); + writer.WriteStringValue("description", Description); + writer.WriteIntValue("durationCookingMinutes", DurationCookingMinutes); + writer.WriteIntValue("durationRestingMinutes", DurationRestingMinutes); + writer.WriteIntValue("durationWorkingMinutes", DurationWorkingMinutes); + writer.WriteGuidValue("id", Id); + writer.WriteCollectionOfObjectValues("ingredients", Ingredients); + writer.WriteStringValue("name", Name); + writer.WriteStringValue("normalizedName", NormalizedName); + writer.WriteIntValue("servings", Servings); + writer.WriteStringValue("source", Source); + writer.WriteCollectionOfObjectValues("steps", Steps); + writer.WriteStringValue("username", Username); + writer.WriteAdditionalData(AdditionalData); + } + } +} +#pragma warning restore CS0618 diff --git a/source/HomeBook.Client/Models/RecipeIngredientResponse.cs b/source/HomeBook.Client/Models/RecipeIngredientResponse.cs new file mode 100644 index 00000000..7a41ebdc --- /dev/null +++ b/source/HomeBook.Client/Models/RecipeIngredientResponse.cs @@ -0,0 +1,93 @@ +// +#pragma warning disable CS0618 +using Microsoft.Kiota.Abstractions.Extensions; +using Microsoft.Kiota.Abstractions.Serialization; +using System.Collections.Generic; +using System.IO; +using System; +namespace HomeBook.Client.Models +{ + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + #pragma warning disable CS1591 + public partial class RecipeIngredientResponse : IAdditionalDataHolder, IParsable + #pragma warning restore CS1591 + { + /// Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well. + public IDictionary AdditionalData { get; set; } + /// The id property + public Guid? Id { get; set; } + /// The name property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Name { get; set; } +#nullable restore +#else + public string Name { get; set; } +#endif + /// The normalizedName property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? NormalizedName { get; set; } +#nullable restore +#else + public string NormalizedName { get; set; } +#endif + /// The quantity property + public double? Quantity { get; set; } + /// The unit property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Unit { get; set; } +#nullable restore +#else + public string Unit { get; set; } +#endif + /// + /// Instantiates a new and sets the default values. + /// + public RecipeIngredientResponse() + { + AdditionalData = new Dictionary(); + } + /// + /// Creates a new instance of the appropriate class based on discriminator value + /// + /// A + /// The parse node to use to read the discriminator value and create the object + public static global::HomeBook.Client.Models.RecipeIngredientResponse CreateFromDiscriminatorValue(IParseNode parseNode) + { + if(ReferenceEquals(parseNode, null)) throw new ArgumentNullException(nameof(parseNode)); + return new global::HomeBook.Client.Models.RecipeIngredientResponse(); + } + /// + /// The deserialization information for the current model + /// + /// A IDictionary<string, Action<IParseNode>> + public virtual IDictionary> GetFieldDeserializers() + { + return new Dictionary> + { + { "id", n => { Id = n.GetGuidValue(); } }, + { "name", n => { Name = n.GetStringValue(); } }, + { "normalizedName", n => { NormalizedName = n.GetStringValue(); } }, + { "quantity", n => { Quantity = n.GetDoubleValue(); } }, + { "unit", n => { Unit = n.GetStringValue(); } }, + }; + } + /// + /// Serializes information the current object + /// + /// Serialization writer to use to serialize this model + public virtual void Serialize(ISerializationWriter writer) + { + if(ReferenceEquals(writer, null)) throw new ArgumentNullException(nameof(writer)); + writer.WriteGuidValue("id", Id); + writer.WriteStringValue("name", Name); + writer.WriteStringValue("normalizedName", NormalizedName); + writer.WriteDoubleValue("quantity", Quantity); + writer.WriteStringValue("unit", Unit); + writer.WriteAdditionalData(AdditionalData); + } + } +} +#pragma warning restore CS0618 diff --git a/source/HomeBook.Client/Models/RecipeResponse.cs b/source/HomeBook.Client/Models/RecipeResponse.cs index 4a62c9cf..c4bebd1c 100644 --- a/source/HomeBook.Client/Models/RecipeResponse.cs +++ b/source/HomeBook.Client/Models/RecipeResponse.cs @@ -14,6 +14,30 @@ public partial class RecipeResponse : IAdditionalDataHolder, IParsable { /// Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well. public IDictionary AdditionalData { get; set; } + /// The caloriesKcal property + public int? CaloriesKcal { get; set; } + /// The comments property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Comments { get; set; } +#nullable restore +#else + public string Comments { get; set; } +#endif + /// The description property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Description { get; set; } +#nullable restore +#else + public string Description { get; set; } +#endif + /// The durationCookingMinutes property + public int? DurationCookingMinutes { get; set; } + /// The durationRestingMinutes property + public int? DurationRestingMinutes { get; set; } + /// The durationWorkingMinutes property + public int? DurationWorkingMinutes { get; set; } /// The id property public Guid? Id { get; set; } /// The name property @@ -31,6 +55,24 @@ public partial class RecipeResponse : IAdditionalDataHolder, IParsable #nullable restore #else public string NormalizedName { get; set; } +#endif + /// The servings property + public int? Servings { get; set; } + /// The source property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Source { get; set; } +#nullable restore +#else + public string Source { get; set; } +#endif + /// The username property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Username { get; set; } +#nullable restore +#else + public string Username { get; set; } #endif /// /// Instantiates a new and sets the default values. @@ -57,9 +99,18 @@ public virtual IDictionary> GetFieldDeserializers() { return new Dictionary> { + { "caloriesKcal", n => { CaloriesKcal = n.GetIntValue(); } }, + { "comments", n => { Comments = n.GetStringValue(); } }, + { "description", n => { Description = n.GetStringValue(); } }, + { "durationCookingMinutes", n => { DurationCookingMinutes = n.GetIntValue(); } }, + { "durationRestingMinutes", n => { DurationRestingMinutes = n.GetIntValue(); } }, + { "durationWorkingMinutes", n => { DurationWorkingMinutes = n.GetIntValue(); } }, { "id", n => { Id = n.GetGuidValue(); } }, { "name", n => { Name = n.GetStringValue(); } }, { "normalizedName", n => { NormalizedName = n.GetStringValue(); } }, + { "servings", n => { Servings = n.GetIntValue(); } }, + { "source", n => { Source = n.GetStringValue(); } }, + { "username", n => { Username = n.GetStringValue(); } }, }; } /// @@ -69,9 +120,18 @@ public virtual IDictionary> GetFieldDeserializers() public virtual void Serialize(ISerializationWriter writer) { if(ReferenceEquals(writer, null)) throw new ArgumentNullException(nameof(writer)); + writer.WriteIntValue("caloriesKcal", CaloriesKcal); + writer.WriteStringValue("comments", Comments); + writer.WriteStringValue("description", Description); + writer.WriteIntValue("durationCookingMinutes", DurationCookingMinutes); + writer.WriteIntValue("durationRestingMinutes", DurationRestingMinutes); + writer.WriteIntValue("durationWorkingMinutes", DurationWorkingMinutes); writer.WriteGuidValue("id", Id); writer.WriteStringValue("name", Name); writer.WriteStringValue("normalizedName", NormalizedName); + writer.WriteIntValue("servings", Servings); + writer.WriteStringValue("source", Source); + writer.WriteStringValue("username", Username); writer.WriteAdditionalData(AdditionalData); } } diff --git a/source/HomeBook.Client/Models/RecipeStepResponse.cs b/source/HomeBook.Client/Models/RecipeStepResponse.cs new file mode 100644 index 00000000..f3df6ba0 --- /dev/null +++ b/source/HomeBook.Client/Models/RecipeStepResponse.cs @@ -0,0 +1,77 @@ +// +#pragma warning disable CS0618 +using Microsoft.Kiota.Abstractions.Extensions; +using Microsoft.Kiota.Abstractions.Serialization; +using System.Collections.Generic; +using System.IO; +using System; +namespace HomeBook.Client.Models +{ + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + #pragma warning disable CS1591 + public partial class RecipeStepResponse : IAdditionalDataHolder, IParsable + #pragma warning restore CS1591 + { + /// Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well. + public IDictionary AdditionalData { get; set; } + /// The description property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Description { get; set; } +#nullable restore +#else + public string Description { get; set; } +#endif + /// The position property + public int? Position { get; set; } + /// The recipeId property + public Guid? RecipeId { get; set; } + /// The timerDurationInSeconds property + public int? TimerDurationInSeconds { get; set; } + /// + /// Instantiates a new and sets the default values. + /// + public RecipeStepResponse() + { + AdditionalData = new Dictionary(); + } + /// + /// Creates a new instance of the appropriate class based on discriminator value + /// + /// A + /// The parse node to use to read the discriminator value and create the object + public static global::HomeBook.Client.Models.RecipeStepResponse CreateFromDiscriminatorValue(IParseNode parseNode) + { + if(ReferenceEquals(parseNode, null)) throw new ArgumentNullException(nameof(parseNode)); + return new global::HomeBook.Client.Models.RecipeStepResponse(); + } + /// + /// The deserialization information for the current model + /// + /// A IDictionary<string, Action<IParseNode>> + public virtual IDictionary> GetFieldDeserializers() + { + return new Dictionary> + { + { "description", n => { Description = n.GetStringValue(); } }, + { "position", n => { Position = n.GetIntValue(); } }, + { "recipeId", n => { RecipeId = n.GetGuidValue(); } }, + { "timerDurationInSeconds", n => { TimerDurationInSeconds = n.GetIntValue(); } }, + }; + } + /// + /// Serializes information the current object + /// + /// Serialization writer to use to serialize this model + public virtual void Serialize(ISerializationWriter writer) + { + if(ReferenceEquals(writer, null)) throw new ArgumentNullException(nameof(writer)); + writer.WriteStringValue("description", Description); + writer.WriteIntValue("position", Position); + writer.WriteGuidValue("recipeId", RecipeId); + writer.WriteIntValue("timerDurationInSeconds", TimerDurationInSeconds); + writer.WriteAdditionalData(AdditionalData); + } + } +} +#pragma warning restore CS0618 diff --git a/source/HomeBook.Client/Models/SearchItemResponse.cs b/source/HomeBook.Client/Models/SearchItemResponse.cs new file mode 100644 index 00000000..caf8546d --- /dev/null +++ b/source/HomeBook.Client/Models/SearchItemResponse.cs @@ -0,0 +1,105 @@ +// +#pragma warning disable CS0618 +using Microsoft.Kiota.Abstractions.Extensions; +using Microsoft.Kiota.Abstractions.Serialization; +using System.Collections.Generic; +using System.IO; +using System; +namespace HomeBook.Client.Models +{ + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + #pragma warning disable CS1591 + public partial class SearchItemResponse : IAdditionalDataHolder, IParsable + #pragma warning restore CS1591 + { + /// Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well. + public IDictionary AdditionalData { get; set; } + /// The color property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Color { get; set; } +#nullable restore +#else + public string Color { get; set; } +#endif + /// The description property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Description { get; set; } +#nullable restore +#else + public string Description { get; set; } +#endif + /// The icon property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Icon { get; set; } +#nullable restore +#else + public string Icon { get; set; } +#endif + /// The title property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Title { get; set; } +#nullable restore +#else + public string Title { get; set; } +#endif + /// The url property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? Url { get; set; } +#nullable restore +#else + public string Url { get; set; } +#endif + /// + /// Instantiates a new and sets the default values. + /// + public SearchItemResponse() + { + AdditionalData = new Dictionary(); + } + /// + /// Creates a new instance of the appropriate class based on discriminator value + /// + /// A + /// The parse node to use to read the discriminator value and create the object + public static global::HomeBook.Client.Models.SearchItemResponse CreateFromDiscriminatorValue(IParseNode parseNode) + { + if(ReferenceEquals(parseNode, null)) throw new ArgumentNullException(nameof(parseNode)); + return new global::HomeBook.Client.Models.SearchItemResponse(); + } + /// + /// The deserialization information for the current model + /// + /// A IDictionary<string, Action<IParseNode>> + public virtual IDictionary> GetFieldDeserializers() + { + return new Dictionary> + { + { "color", n => { Color = n.GetStringValue(); } }, + { "description", n => { Description = n.GetStringValue(); } }, + { "icon", n => { Icon = n.GetStringValue(); } }, + { "title", n => { Title = n.GetStringValue(); } }, + { "url", n => { Url = n.GetStringValue(); } }, + }; + } + /// + /// Serializes information the current object + /// + /// Serialization writer to use to serialize this model + public virtual void Serialize(ISerializationWriter writer) + { + if(ReferenceEquals(writer, null)) throw new ArgumentNullException(nameof(writer)); + writer.WriteStringValue("color", Color); + writer.WriteStringValue("description", Description); + writer.WriteStringValue("icon", Icon); + writer.WriteStringValue("title", Title); + writer.WriteStringValue("url", Url); + writer.WriteAdditionalData(AdditionalData); + } + } +} +#pragma warning restore CS0618 diff --git a/source/HomeBook.Client/Models/SearchModuleResponse.cs b/source/HomeBook.Client/Models/SearchModuleResponse.cs new file mode 100644 index 00000000..5e1a161c --- /dev/null +++ b/source/HomeBook.Client/Models/SearchModuleResponse.cs @@ -0,0 +1,79 @@ +// +#pragma warning disable CS0618 +using Microsoft.Kiota.Abstractions.Extensions; +using Microsoft.Kiota.Abstractions.Serialization; +using System.Collections.Generic; +using System.IO; +using System; +namespace HomeBook.Client.Models +{ + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + #pragma warning disable CS1591 + public partial class SearchModuleResponse : IAdditionalDataHolder, IParsable + #pragma warning restore CS1591 + { + /// Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well. + public IDictionary AdditionalData { get; set; } + /// The items property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public List? Items { get; set; } +#nullable restore +#else + public List Items { get; set; } +#endif + /// The moduleKey property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public string? ModuleKey { get; set; } +#nullable restore +#else + public string ModuleKey { get; set; } +#endif + /// The totalCount property + public int? TotalCount { get; set; } + /// + /// Instantiates a new and sets the default values. + /// + public SearchModuleResponse() + { + AdditionalData = new Dictionary(); + } + /// + /// Creates a new instance of the appropriate class based on discriminator value + /// + /// A + /// The parse node to use to read the discriminator value and create the object + public static global::HomeBook.Client.Models.SearchModuleResponse CreateFromDiscriminatorValue(IParseNode parseNode) + { + if(ReferenceEquals(parseNode, null)) throw new ArgumentNullException(nameof(parseNode)); + return new global::HomeBook.Client.Models.SearchModuleResponse(); + } + /// + /// The deserialization information for the current model + /// + /// A IDictionary<string, Action<IParseNode>> + public virtual IDictionary> GetFieldDeserializers() + { + return new Dictionary> + { + { "items", n => { Items = n.GetCollectionOfObjectValues(global::HomeBook.Client.Models.SearchItemResponse.CreateFromDiscriminatorValue)?.AsList(); } }, + { "moduleKey", n => { ModuleKey = n.GetStringValue(); } }, + { "totalCount", n => { TotalCount = n.GetIntValue(); } }, + }; + } + /// + /// Serializes information the current object + /// + /// Serialization writer to use to serialize this model + public virtual void Serialize(ISerializationWriter writer) + { + if(ReferenceEquals(writer, null)) throw new ArgumentNullException(nameof(writer)); + writer.WriteCollectionOfObjectValues("items", Items); + writer.WriteStringValue("moduleKey", ModuleKey); + writer.WriteIntValue("totalCount", TotalCount); + writer.WriteAdditionalData(AdditionalData); + } + } +} +#pragma warning restore CS0618 diff --git a/source/HomeBook.Client/Models/SearchResponse.cs b/source/HomeBook.Client/Models/SearchResponse.cs new file mode 100644 index 00000000..b4f266bc --- /dev/null +++ b/source/HomeBook.Client/Models/SearchResponse.cs @@ -0,0 +1,65 @@ +// +#pragma warning disable CS0618 +using Microsoft.Kiota.Abstractions.Extensions; +using Microsoft.Kiota.Abstractions.Serialization; +using System.Collections.Generic; +using System.IO; +using System; +namespace HomeBook.Client.Models +{ + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + #pragma warning disable CS1591 + public partial class SearchResponse : IAdditionalDataHolder, IParsable + #pragma warning restore CS1591 + { + /// Stores additional data not described in the OpenAPI description found when deserializing. Can be used for serialization as well. + public IDictionary AdditionalData { get; set; } + /// The searchModuleResponses property +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public List? SearchModuleResponses { get; set; } +#nullable restore +#else + public List SearchModuleResponses { get; set; } +#endif + /// + /// Instantiates a new and sets the default values. + /// + public SearchResponse() + { + AdditionalData = new Dictionary(); + } + /// + /// Creates a new instance of the appropriate class based on discriminator value + /// + /// A + /// The parse node to use to read the discriminator value and create the object + public static global::HomeBook.Client.Models.SearchResponse CreateFromDiscriminatorValue(IParseNode parseNode) + { + if(ReferenceEquals(parseNode, null)) throw new ArgumentNullException(nameof(parseNode)); + return new global::HomeBook.Client.Models.SearchResponse(); + } + /// + /// The deserialization information for the current model + /// + /// A IDictionary<string, Action<IParseNode>> + public virtual IDictionary> GetFieldDeserializers() + { + return new Dictionary> + { + { "searchModuleResponses", n => { SearchModuleResponses = n.GetCollectionOfObjectValues(global::HomeBook.Client.Models.SearchModuleResponse.CreateFromDiscriminatorValue)?.AsList(); } }, + }; + } + /// + /// Serializes information the current object + /// + /// Serialization writer to use to serialize this model + public virtual void Serialize(ISerializationWriter writer) + { + if(ReferenceEquals(writer, null)) throw new ArgumentNullException(nameof(writer)); + writer.WriteCollectionOfObjectValues("searchModuleResponses", SearchModuleResponses); + writer.WriteAdditionalData(AdditionalData); + } + } +} +#pragma warning restore CS0618 diff --git a/source/HomeBook.Client/Finances/Calculations/CalculationsRequestBuilder.cs b/source/HomeBook.Client/Modules/Finances/Calculations/CalculationsRequestBuilder.cs similarity index 62% rename from source/HomeBook.Client/Finances/Calculations/CalculationsRequestBuilder.cs rename to source/HomeBook.Client/Modules/Finances/Calculations/CalculationsRequestBuilder.cs index 57d24265..d5ac4798 100644 --- a/source/HomeBook.Client/Finances/Calculations/CalculationsRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Finances/Calculations/CalculationsRequestBuilder.cs @@ -1,39 +1,39 @@ // #pragma warning disable CS0618 -using HomeBook.Client.Finances.Calculations.Savings; +using HomeBook.Client.Modules.Finances.Calculations.Savings; using Microsoft.Kiota.Abstractions.Extensions; using Microsoft.Kiota.Abstractions; using System.Collections.Generic; using System.IO; using System.Threading.Tasks; using System; -namespace HomeBook.Client.Finances.Calculations +namespace HomeBook.Client.Modules.Finances.Calculations { /// - /// Builds and executes requests for operations under \finances\calculations + /// Builds and executes requests for operations under \modules\finances\calculations /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class CalculationsRequestBuilder : BaseRequestBuilder { /// The savings property - public global::HomeBook.Client.Finances.Calculations.Savings.SavingsRequestBuilder Savings + public global::HomeBook.Client.Modules.Finances.Calculations.Savings.SavingsRequestBuilder Savings { - get => new global::HomeBook.Client.Finances.Calculations.Savings.SavingsRequestBuilder(PathParameters, RequestAdapter); + get => new global::HomeBook.Client.Modules.Finances.Calculations.Savings.SavingsRequestBuilder(PathParameters, RequestAdapter); } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public CalculationsRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/calculations", pathParameters) + public CalculationsRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/calculations", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public CalculationsRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/calculations", rawUrl) + public CalculationsRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/calculations", rawUrl) { } } diff --git a/source/HomeBook.Client/Finances/Calculations/Savings/SavingsRequestBuilder.cs b/source/HomeBook.Client/Modules/Finances/Calculations/Savings/SavingsRequestBuilder.cs similarity index 85% rename from source/HomeBook.Client/Finances/Calculations/Savings/SavingsRequestBuilder.cs rename to source/HomeBook.Client/Modules/Finances/Calculations/Savings/SavingsRequestBuilder.cs index f03377d9..e8774221 100644 --- a/source/HomeBook.Client/Finances/Calculations/Savings/SavingsRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Finances/Calculations/Savings/SavingsRequestBuilder.cs @@ -9,28 +9,28 @@ using System.Threading.Tasks; using System.Threading; using System; -namespace HomeBook.Client.Finances.Calculations.Savings +namespace HomeBook.Client.Modules.Finances.Calculations.Savings { /// - /// Builds and executes requests for operations under \finances\calculations\savings + /// Builds and executes requests for operations under \modules\finances\calculations\savings /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class SavingsRequestBuilder : BaseRequestBuilder { /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public SavingsRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/calculations/savings", pathParameters) + public SavingsRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/calculations/savings", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public SavingsRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/calculations/savings", rawUrl) + public SavingsRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/calculations/savings", rawUrl) { } /// @@ -78,11 +78,11 @@ public RequestInformation ToPostRequestInformation(global::HomeBook.Client.Model /// /// Returns a request builder with the provided arbitrary URL. Using this method means any other path or query parameters are ignored. /// - /// A + /// A /// The raw URL to use for the request builder. - public global::HomeBook.Client.Finances.Calculations.Savings.SavingsRequestBuilder WithUrl(string rawUrl) + public global::HomeBook.Client.Modules.Finances.Calculations.Savings.SavingsRequestBuilder WithUrl(string rawUrl) { - return new global::HomeBook.Client.Finances.Calculations.Savings.SavingsRequestBuilder(rawUrl, RequestAdapter); + return new global::HomeBook.Client.Modules.Finances.Calculations.Savings.SavingsRequestBuilder(rawUrl, RequestAdapter); } /// /// Configuration for the request such as headers, query parameters, and middleware options. diff --git a/source/HomeBook.Client/Finances/FinancesRequestBuilder.cs b/source/HomeBook.Client/Modules/Finances/FinancesRequestBuilder.cs similarity index 58% rename from source/HomeBook.Client/Finances/FinancesRequestBuilder.cs rename to source/HomeBook.Client/Modules/Finances/FinancesRequestBuilder.cs index 4c8506ed..a840ce8f 100644 --- a/source/HomeBook.Client/Finances/FinancesRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Finances/FinancesRequestBuilder.cs @@ -1,45 +1,45 @@ // #pragma warning disable CS0618 -using HomeBook.Client.Finances.Calculations; -using HomeBook.Client.Finances.SavingGoals; +using HomeBook.Client.Modules.Finances.Calculations; +using HomeBook.Client.Modules.Finances.SavingGoals; using Microsoft.Kiota.Abstractions.Extensions; using Microsoft.Kiota.Abstractions; using System.Collections.Generic; using System.IO; using System.Threading.Tasks; using System; -namespace HomeBook.Client.Finances +namespace HomeBook.Client.Modules.Finances { /// - /// Builds and executes requests for operations under \finances + /// Builds and executes requests for operations under \modules\finances /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class FinancesRequestBuilder : BaseRequestBuilder { /// The calculations property - public global::HomeBook.Client.Finances.Calculations.CalculationsRequestBuilder Calculations + public global::HomeBook.Client.Modules.Finances.Calculations.CalculationsRequestBuilder Calculations { - get => new global::HomeBook.Client.Finances.Calculations.CalculationsRequestBuilder(PathParameters, RequestAdapter); + get => new global::HomeBook.Client.Modules.Finances.Calculations.CalculationsRequestBuilder(PathParameters, RequestAdapter); } /// The savingGoals property - public global::HomeBook.Client.Finances.SavingGoals.SavingGoalsRequestBuilder SavingGoals + public global::HomeBook.Client.Modules.Finances.SavingGoals.SavingGoalsRequestBuilder SavingGoals { - get => new global::HomeBook.Client.Finances.SavingGoals.SavingGoalsRequestBuilder(PathParameters, RequestAdapter); + get => new global::HomeBook.Client.Modules.Finances.SavingGoals.SavingGoalsRequestBuilder(PathParameters, RequestAdapter); } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public FinancesRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances", pathParameters) + public FinancesRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public FinancesRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances", rawUrl) + public FinancesRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances", rawUrl) { } } diff --git a/source/HomeBook.Client/Finances/SavingGoals/Item/Amounts/AmountsRequestBuilder.cs b/source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Amounts/AmountsRequestBuilder.cs similarity index 84% rename from source/HomeBook.Client/Finances/SavingGoals/Item/Amounts/AmountsRequestBuilder.cs rename to source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Amounts/AmountsRequestBuilder.cs index 93a8aa1a..fa7306a1 100644 --- a/source/HomeBook.Client/Finances/SavingGoals/Item/Amounts/AmountsRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Amounts/AmountsRequestBuilder.cs @@ -9,28 +9,28 @@ using System.Threading.Tasks; using System.Threading; using System; -namespace HomeBook.Client.Finances.SavingGoals.Item.Amounts +namespace HomeBook.Client.Modules.Finances.SavingGoals.Item.Amounts { /// - /// Builds and executes requests for operations under \finances\saving-goals\{-id}\amounts + /// Builds and executes requests for operations under \modules\finances\saving-goals\{-id}\amounts /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class AmountsRequestBuilder : BaseRequestBuilder { /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public AmountsRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{%2Did}/amounts", pathParameters) + public AmountsRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals/{%2Did}/amounts", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public AmountsRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{%2Did}/amounts", rawUrl) + public AmountsRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals/{%2Did}/amounts", rawUrl) { } /// @@ -78,11 +78,11 @@ public RequestInformation ToPatchRequestInformation(global::HomeBook.Client.Mode /// /// Returns a request builder with the provided arbitrary URL. Using this method means any other path or query parameters are ignored. /// - /// A + /// A /// The raw URL to use for the request builder. - public global::HomeBook.Client.Finances.SavingGoals.Item.Amounts.AmountsRequestBuilder WithUrl(string rawUrl) + public global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Amounts.AmountsRequestBuilder WithUrl(string rawUrl) { - return new global::HomeBook.Client.Finances.SavingGoals.Item.Amounts.AmountsRequestBuilder(rawUrl, RequestAdapter); + return new global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Amounts.AmountsRequestBuilder(rawUrl, RequestAdapter); } /// /// Configuration for the request such as headers, query parameters, and middleware options. diff --git a/source/HomeBook.Client/Finances/SavingGoals/Item/Appearance/AppearanceRequestBuilder.cs b/source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Appearance/AppearanceRequestBuilder.cs similarity index 84% rename from source/HomeBook.Client/Finances/SavingGoals/Item/Appearance/AppearanceRequestBuilder.cs rename to source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Appearance/AppearanceRequestBuilder.cs index 089839de..7fcb5ab0 100644 --- a/source/HomeBook.Client/Finances/SavingGoals/Item/Appearance/AppearanceRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Appearance/AppearanceRequestBuilder.cs @@ -9,28 +9,28 @@ using System.Threading.Tasks; using System.Threading; using System; -namespace HomeBook.Client.Finances.SavingGoals.Item.Appearance +namespace HomeBook.Client.Modules.Finances.SavingGoals.Item.Appearance { /// - /// Builds and executes requests for operations under \finances\saving-goals\{-id}\appearance + /// Builds and executes requests for operations under \modules\finances\saving-goals\{-id}\appearance /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class AppearanceRequestBuilder : BaseRequestBuilder { /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public AppearanceRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{%2Did}/appearance", pathParameters) + public AppearanceRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals/{%2Did}/appearance", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public AppearanceRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{%2Did}/appearance", rawUrl) + public AppearanceRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals/{%2Did}/appearance", rawUrl) { } /// @@ -78,11 +78,11 @@ public RequestInformation ToPatchRequestInformation(global::HomeBook.Client.Mode /// /// Returns a request builder with the provided arbitrary URL. Using this method means any other path or query parameters are ignored. /// - /// A + /// A /// The raw URL to use for the request builder. - public global::HomeBook.Client.Finances.SavingGoals.Item.Appearance.AppearanceRequestBuilder WithUrl(string rawUrl) + public global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Appearance.AppearanceRequestBuilder WithUrl(string rawUrl) { - return new global::HomeBook.Client.Finances.SavingGoals.Item.Appearance.AppearanceRequestBuilder(rawUrl, RequestAdapter); + return new global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Appearance.AppearanceRequestBuilder(rawUrl, RequestAdapter); } /// /// Configuration for the request such as headers, query parameters, and middleware options. diff --git a/source/HomeBook.Client/Finances/SavingGoals/Item/Info/InfoRequestBuilder.cs b/source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Info/InfoRequestBuilder.cs similarity index 85% rename from source/HomeBook.Client/Finances/SavingGoals/Item/Info/InfoRequestBuilder.cs rename to source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Info/InfoRequestBuilder.cs index dd75f6dd..125fba7e 100644 --- a/source/HomeBook.Client/Finances/SavingGoals/Item/Info/InfoRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Info/InfoRequestBuilder.cs @@ -9,28 +9,28 @@ using System.Threading.Tasks; using System.Threading; using System; -namespace HomeBook.Client.Finances.SavingGoals.Item.Info +namespace HomeBook.Client.Modules.Finances.SavingGoals.Item.Info { /// - /// Builds and executes requests for operations under \finances\saving-goals\{-id}\info + /// Builds and executes requests for operations under \modules\finances\saving-goals\{-id}\info /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class InfoRequestBuilder : BaseRequestBuilder { /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public InfoRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{%2Did}/info", pathParameters) + public InfoRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals/{%2Did}/info", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public InfoRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{%2Did}/info", rawUrl) + public InfoRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals/{%2Did}/info", rawUrl) { } /// @@ -78,11 +78,11 @@ public RequestInformation ToPatchRequestInformation(global::HomeBook.Client.Mode /// /// Returns a request builder with the provided arbitrary URL. Using this method means any other path or query parameters are ignored. /// - /// A + /// A /// The raw URL to use for the request builder. - public global::HomeBook.Client.Finances.SavingGoals.Item.Info.InfoRequestBuilder WithUrl(string rawUrl) + public global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Info.InfoRequestBuilder WithUrl(string rawUrl) { - return new global::HomeBook.Client.Finances.SavingGoals.Item.Info.InfoRequestBuilder(rawUrl, RequestAdapter); + return new global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Info.InfoRequestBuilder(rawUrl, RequestAdapter); } /// /// Configuration for the request such as headers, query parameters, and middleware options. diff --git a/source/HomeBook.Client/Finances/SavingGoals/Item/ItemRequestBuilder.cs b/source/HomeBook.Client/Modules/Finances/SavingGoals/Item/ItemRequestBuilder.cs similarity index 69% rename from source/HomeBook.Client/Finances/SavingGoals/Item/ItemRequestBuilder.cs rename to source/HomeBook.Client/Modules/Finances/SavingGoals/Item/ItemRequestBuilder.cs index 2ebeba64..36abc775 100644 --- a/source/HomeBook.Client/Finances/SavingGoals/Item/ItemRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Finances/SavingGoals/Item/ItemRequestBuilder.cs @@ -1,9 +1,9 @@ // #pragma warning disable CS0618 -using HomeBook.Client.Finances.SavingGoals.Item.Amounts; -using HomeBook.Client.Finances.SavingGoals.Item.Appearance; -using HomeBook.Client.Finances.SavingGoals.Item.Info; -using HomeBook.Client.Finances.SavingGoals.Item.Name; +using HomeBook.Client.Modules.Finances.SavingGoals.Item.Amounts; +using HomeBook.Client.Modules.Finances.SavingGoals.Item.Appearance; +using HomeBook.Client.Modules.Finances.SavingGoals.Item.Info; +using HomeBook.Client.Modules.Finances.SavingGoals.Item.Name; using Microsoft.Kiota.Abstractions.Extensions; using Microsoft.Kiota.Abstractions.Serialization; using Microsoft.Kiota.Abstractions; @@ -12,48 +12,48 @@ using System.Threading.Tasks; using System.Threading; using System; -namespace HomeBook.Client.Finances.SavingGoals.Item +namespace HomeBook.Client.Modules.Finances.SavingGoals.Item { /// - /// Builds and executes requests for operations under \finances\saving-goals\{-id} + /// Builds and executes requests for operations under \modules\finances\saving-goals\{-id} /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class ItemRequestBuilder : BaseRequestBuilder { /// The amounts property - public global::HomeBook.Client.Finances.SavingGoals.Item.Amounts.AmountsRequestBuilder Amounts + public global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Amounts.AmountsRequestBuilder Amounts { - get => new global::HomeBook.Client.Finances.SavingGoals.Item.Amounts.AmountsRequestBuilder(PathParameters, RequestAdapter); + get => new global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Amounts.AmountsRequestBuilder(PathParameters, RequestAdapter); } /// The appearance property - public global::HomeBook.Client.Finances.SavingGoals.Item.Appearance.AppearanceRequestBuilder Appearance + public global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Appearance.AppearanceRequestBuilder Appearance { - get => new global::HomeBook.Client.Finances.SavingGoals.Item.Appearance.AppearanceRequestBuilder(PathParameters, RequestAdapter); + get => new global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Appearance.AppearanceRequestBuilder(PathParameters, RequestAdapter); } /// The info property - public global::HomeBook.Client.Finances.SavingGoals.Item.Info.InfoRequestBuilder Info + public global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Info.InfoRequestBuilder Info { - get => new global::HomeBook.Client.Finances.SavingGoals.Item.Info.InfoRequestBuilder(PathParameters, RequestAdapter); + get => new global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Info.InfoRequestBuilder(PathParameters, RequestAdapter); } /// The name property - public global::HomeBook.Client.Finances.SavingGoals.Item.Name.NameRequestBuilder Name + public global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Name.NameRequestBuilder Name { - get => new global::HomeBook.Client.Finances.SavingGoals.Item.Name.NameRequestBuilder(PathParameters, RequestAdapter); + get => new global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Name.NameRequestBuilder(PathParameters, RequestAdapter); } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public ItemRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{%2Did}", pathParameters) + public ItemRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals/{%2Did}", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public ItemRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{%2Did}", rawUrl) + public ItemRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals/{%2Did}", rawUrl) { } /// @@ -96,11 +96,11 @@ public RequestInformation ToDeleteRequestInformation(Action /// Returns a request builder with the provided arbitrary URL. Using this method means any other path or query parameters are ignored. /// - /// A + /// A /// The raw URL to use for the request builder. - public global::HomeBook.Client.Finances.SavingGoals.Item.ItemRequestBuilder WithUrl(string rawUrl) + public global::HomeBook.Client.Modules.Finances.SavingGoals.Item.ItemRequestBuilder WithUrl(string rawUrl) { - return new global::HomeBook.Client.Finances.SavingGoals.Item.ItemRequestBuilder(rawUrl, RequestAdapter); + return new global::HomeBook.Client.Modules.Finances.SavingGoals.Item.ItemRequestBuilder(rawUrl, RequestAdapter); } /// /// Configuration for the request such as headers, query parameters, and middleware options. diff --git a/source/HomeBook.Client/Finances/SavingGoals/Item/Name/NameRequestBuilder.cs b/source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Name/NameRequestBuilder.cs similarity index 85% rename from source/HomeBook.Client/Finances/SavingGoals/Item/Name/NameRequestBuilder.cs rename to source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Name/NameRequestBuilder.cs index 0420ae5b..7cb53598 100644 --- a/source/HomeBook.Client/Finances/SavingGoals/Item/Name/NameRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Finances/SavingGoals/Item/Name/NameRequestBuilder.cs @@ -9,28 +9,28 @@ using System.Threading.Tasks; using System.Threading; using System; -namespace HomeBook.Client.Finances.SavingGoals.Item.Name +namespace HomeBook.Client.Modules.Finances.SavingGoals.Item.Name { /// - /// Builds and executes requests for operations under \finances\saving-goals\{-id}\name + /// Builds and executes requests for operations under \modules\finances\saving-goals\{-id}\name /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class NameRequestBuilder : BaseRequestBuilder { /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public NameRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{%2Did}/name", pathParameters) + public NameRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals/{%2Did}/name", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public NameRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals/{%2Did}/name", rawUrl) + public NameRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals/{%2Did}/name", rawUrl) { } /// @@ -78,11 +78,11 @@ public RequestInformation ToPatchRequestInformation(global::HomeBook.Client.Mode /// /// Returns a request builder with the provided arbitrary URL. Using this method means any other path or query parameters are ignored. /// - /// A + /// A /// The raw URL to use for the request builder. - public global::HomeBook.Client.Finances.SavingGoals.Item.Name.NameRequestBuilder WithUrl(string rawUrl) + public global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Name.NameRequestBuilder WithUrl(string rawUrl) { - return new global::HomeBook.Client.Finances.SavingGoals.Item.Name.NameRequestBuilder(rawUrl, RequestAdapter); + return new global::HomeBook.Client.Modules.Finances.SavingGoals.Item.Name.NameRequestBuilder(rawUrl, RequestAdapter); } /// /// Configuration for the request such as headers, query parameters, and middleware options. diff --git a/source/HomeBook.Client/Finances/SavingGoals/SavingGoalsRequestBuilder.cs b/source/HomeBook.Client/Modules/Finances/SavingGoals/SavingGoalsRequestBuilder.cs similarity index 82% rename from source/HomeBook.Client/Finances/SavingGoals/SavingGoalsRequestBuilder.cs rename to source/HomeBook.Client/Modules/Finances/SavingGoals/SavingGoalsRequestBuilder.cs index fa9eb0aa..f5c72fff 100644 --- a/source/HomeBook.Client/Finances/SavingGoals/SavingGoalsRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Finances/SavingGoals/SavingGoalsRequestBuilder.cs @@ -1,7 +1,7 @@ // #pragma warning disable CS0618 -using HomeBook.Client.Finances.SavingGoals.Item; using HomeBook.Client.Models; +using HomeBook.Client.Modules.Finances.SavingGoals.Item; using Microsoft.Kiota.Abstractions.Extensions; using Microsoft.Kiota.Abstractions.Serialization; using Microsoft.Kiota.Abstractions; @@ -10,53 +10,53 @@ using System.Threading.Tasks; using System.Threading; using System; -namespace HomeBook.Client.Finances.SavingGoals +namespace HomeBook.Client.Modules.Finances.SavingGoals { /// - /// Builds and executes requests for operations under \finances\saving-goals + /// Builds and executes requests for operations under \modules\finances\saving-goals /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class SavingGoalsRequestBuilder : BaseRequestBuilder { - /// Gets an item from the HomeBook.Client.finances.savingGoals.item collection + /// Gets an item from the HomeBook.Client.modules.finances.savingGoals.item collection /// Unique identifier of the item - /// A - public global::HomeBook.Client.Finances.SavingGoals.Item.ItemRequestBuilder this[Guid position] + /// A + public global::HomeBook.Client.Modules.Finances.SavingGoals.Item.ItemRequestBuilder this[Guid position] { get { var urlTplParams = new Dictionary(PathParameters); urlTplParams.Add("%2Did", position); - return new global::HomeBook.Client.Finances.SavingGoals.Item.ItemRequestBuilder(urlTplParams, RequestAdapter); + return new global::HomeBook.Client.Modules.Finances.SavingGoals.Item.ItemRequestBuilder(urlTplParams, RequestAdapter); } } - /// Gets an item from the HomeBook.Client.finances.savingGoals.item collection + /// Gets an item from the HomeBook.Client.modules.finances.savingGoals.item collection /// Unique identifier of the item - /// A + /// A [Obsolete("This indexer is deprecated and will be removed in the next major version. Use the one with the typed parameter instead.")] - public global::HomeBook.Client.Finances.SavingGoals.Item.ItemRequestBuilder this[string position] + public global::HomeBook.Client.Modules.Finances.SavingGoals.Item.ItemRequestBuilder this[string position] { get { var urlTplParams = new Dictionary(PathParameters); if (!string.IsNullOrWhiteSpace(position)) urlTplParams.Add("%2Did", position); - return new global::HomeBook.Client.Finances.SavingGoals.Item.ItemRequestBuilder(urlTplParams, RequestAdapter); + return new global::HomeBook.Client.Modules.Finances.SavingGoals.Item.ItemRequestBuilder(urlTplParams, RequestAdapter); } } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public SavingGoalsRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals", pathParameters) + public SavingGoalsRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public SavingGoalsRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/finances/saving-goals", rawUrl) + public SavingGoalsRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/finances/saving-goals", rawUrl) { } /// @@ -140,11 +140,11 @@ public RequestInformation ToPostRequestInformation(global::HomeBook.Client.Model /// /// Returns a request builder with the provided arbitrary URL. Using this method means any other path or query parameters are ignored. /// - /// A + /// A /// The raw URL to use for the request builder. - public global::HomeBook.Client.Finances.SavingGoals.SavingGoalsRequestBuilder WithUrl(string rawUrl) + public global::HomeBook.Client.Modules.Finances.SavingGoals.SavingGoalsRequestBuilder WithUrl(string rawUrl) { - return new global::HomeBook.Client.Finances.SavingGoals.SavingGoalsRequestBuilder(rawUrl, RequestAdapter); + return new global::HomeBook.Client.Modules.Finances.SavingGoals.SavingGoalsRequestBuilder(rawUrl, RequestAdapter); } /// /// Configuration for the request such as headers, query parameters, and middleware options. diff --git a/source/HomeBook.Client/Kitchen/KitchenRequestBuilder.cs b/source/HomeBook.Client/Modules/Kitchen/KitchenRequestBuilder.cs similarity index 66% rename from source/HomeBook.Client/Kitchen/KitchenRequestBuilder.cs rename to source/HomeBook.Client/Modules/Kitchen/KitchenRequestBuilder.cs index ef7b70fe..fcf88bc0 100644 --- a/source/HomeBook.Client/Kitchen/KitchenRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Kitchen/KitchenRequestBuilder.cs @@ -1,39 +1,39 @@ // #pragma warning disable CS0618 -using HomeBook.Client.Kitchen.Recipes; +using HomeBook.Client.Modules.Kitchen.Recipes; using Microsoft.Kiota.Abstractions.Extensions; using Microsoft.Kiota.Abstractions; using System.Collections.Generic; using System.IO; using System.Threading.Tasks; using System; -namespace HomeBook.Client.Kitchen +namespace HomeBook.Client.Modules.Kitchen { /// - /// Builds and executes requests for operations under \kitchen + /// Builds and executes requests for operations under \modules\kitchen /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class KitchenRequestBuilder : BaseRequestBuilder { /// The recipes property - public global::HomeBook.Client.Kitchen.Recipes.RecipesRequestBuilder Recipes + public global::HomeBook.Client.Modules.Kitchen.Recipes.RecipesRequestBuilder Recipes { - get => new global::HomeBook.Client.Kitchen.Recipes.RecipesRequestBuilder(PathParameters, RequestAdapter); + get => new global::HomeBook.Client.Modules.Kitchen.Recipes.RecipesRequestBuilder(PathParameters, RequestAdapter); } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public KitchenRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/kitchen", pathParameters) + public KitchenRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/kitchen", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public KitchenRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/kitchen", rawUrl) + public KitchenRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/kitchen", rawUrl) { } } diff --git a/source/HomeBook.Client/Kitchen/Recipes/Item/RecipesItemRequestBuilder.cs b/source/HomeBook.Client/Modules/Kitchen/Recipes/Item/RecipesItemRequestBuilder.cs similarity index 53% rename from source/HomeBook.Client/Kitchen/Recipes/Item/RecipesItemRequestBuilder.cs rename to source/HomeBook.Client/Modules/Kitchen/Recipes/Item/RecipesItemRequestBuilder.cs index 3db94d63..62c92e15 100644 --- a/source/HomeBook.Client/Kitchen/Recipes/Item/RecipesItemRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Kitchen/Recipes/Item/RecipesItemRequestBuilder.cs @@ -1,5 +1,6 @@ // #pragma warning disable CS0618 +using HomeBook.Client.Models; using Microsoft.Kiota.Abstractions.Extensions; using Microsoft.Kiota.Abstractions.Serialization; using Microsoft.Kiota.Abstractions; @@ -8,28 +9,28 @@ using System.Threading.Tasks; using System.Threading; using System; -namespace HomeBook.Client.Kitchen.Recipes.Item +namespace HomeBook.Client.Modules.Kitchen.Recipes.Item { /// - /// Builds and executes requests for operations under \kitchen\recipes\{id} + /// Builds and executes requests for operations under \modules\kitchen\recipes\{id} /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class RecipesItemRequestBuilder : BaseRequestBuilder { /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public RecipesItemRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/kitchen/recipes/{id}", pathParameters) + public RecipesItemRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/kitchen/recipes/{id}", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public RecipesItemRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/kitchen/recipes/{id}", rawUrl) + public RecipesItemRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/kitchen/recipes/{id}", rawUrl) { } /// @@ -51,6 +52,24 @@ public async Task DeleteAsync(Action(requestInfo, default, cancellationToken).ConfigureAwait(false); } /// + /// returns recipe by idHTTP 200: Recipes were foundHTTP 404: Recipe not foundHTTP 401: User is not authorizedHTTP 500: Unknown error while getting recipes + /// + /// A + /// Cancellation token to use when cancelling requests + /// Configuration for the request such as headers, query parameters, and middleware options. +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public async Task GetAsync(Action>? requestConfiguration = default, CancellationToken cancellationToken = default) + { +#nullable restore +#else + public async Task GetAsync(Action> requestConfiguration = default, CancellationToken cancellationToken = default) + { +#endif + var requestInfo = ToGetRequestInformation(requestConfiguration); + return await RequestAdapter.SendAsync(requestInfo, global::HomeBook.Client.Models.RecipeDetailResponse.CreateFromDiscriminatorValue, default, cancellationToken).ConfigureAwait(false); + } + /// /// deletes an existing recipeHTTP 200: Recipe was deletedHTTP 401: User is not authorizedHTTP 404: Saving goal not foundHTTP 500: Unknown error while deleting recipe /// /// A @@ -70,13 +89,32 @@ public RequestInformation ToDeleteRequestInformation(Action + /// returns recipe by idHTTP 200: Recipes were foundHTTP 404: Recipe not foundHTTP 401: User is not authorizedHTTP 500: Unknown error while getting recipes + /// + /// A + /// Configuration for the request such as headers, query parameters, and middleware options. +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public RequestInformation ToGetRequestInformation(Action>? requestConfiguration = default) + { +#nullable restore +#else + public RequestInformation ToGetRequestInformation(Action> requestConfiguration = default) + { +#endif + var requestInfo = new RequestInformation(Method.GET, UrlTemplate, PathParameters); + requestInfo.Configure(requestConfiguration); + requestInfo.Headers.TryAdd("Accept", "application/json"); + return requestInfo; + } + /// /// Returns a request builder with the provided arbitrary URL. Using this method means any other path or query parameters are ignored. /// - /// A + /// A /// The raw URL to use for the request builder. - public global::HomeBook.Client.Kitchen.Recipes.Item.RecipesItemRequestBuilder WithUrl(string rawUrl) + public global::HomeBook.Client.Modules.Kitchen.Recipes.Item.RecipesItemRequestBuilder WithUrl(string rawUrl) { - return new global::HomeBook.Client.Kitchen.Recipes.Item.RecipesItemRequestBuilder(rawUrl, RequestAdapter); + return new global::HomeBook.Client.Modules.Kitchen.Recipes.Item.RecipesItemRequestBuilder(rawUrl, RequestAdapter); } /// /// Configuration for the request such as headers, query parameters, and middleware options. @@ -86,6 +124,14 @@ public RequestInformation ToDeleteRequestInformation(Action { } + /// + /// Configuration for the request such as headers, query parameters, and middleware options. + /// + [Obsolete("This class is deprecated. Please use the generic RequestConfiguration class generated by the generator.")] + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + public partial class RecipesItemRequestBuilderGetRequestConfiguration : RequestConfiguration + { + } } } #pragma warning restore CS0618 diff --git a/source/HomeBook.Client/Kitchen/Recipes/RecipesRequestBuilder.cs b/source/HomeBook.Client/Modules/Kitchen/Recipes/RecipesRequestBuilder.cs similarity index 76% rename from source/HomeBook.Client/Kitchen/Recipes/RecipesRequestBuilder.cs rename to source/HomeBook.Client/Modules/Kitchen/Recipes/RecipesRequestBuilder.cs index 2468efa2..ca6769b1 100644 --- a/source/HomeBook.Client/Kitchen/Recipes/RecipesRequestBuilder.cs +++ b/source/HomeBook.Client/Modules/Kitchen/Recipes/RecipesRequestBuilder.cs @@ -1,7 +1,7 @@ // #pragma warning disable CS0618 -using HomeBook.Client.Kitchen.Recipes.Item; using HomeBook.Client.Models; +using HomeBook.Client.Modules.Kitchen.Recipes.Item; using Microsoft.Kiota.Abstractions.Extensions; using Microsoft.Kiota.Abstractions.Serialization; using Microsoft.Kiota.Abstractions; @@ -10,53 +10,53 @@ using System.Threading.Tasks; using System.Threading; using System; -namespace HomeBook.Client.Kitchen.Recipes +namespace HomeBook.Client.Modules.Kitchen.Recipes { /// - /// Builds and executes requests for operations under \kitchen\recipes + /// Builds and executes requests for operations under \modules\kitchen\recipes /// [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] public partial class RecipesRequestBuilder : BaseRequestBuilder { - /// Gets an item from the HomeBook.Client.kitchen.recipes.item collection + /// Gets an item from the HomeBook.Client.modules.kitchen.recipes.item collection /// Unique identifier of the item - /// A - public global::HomeBook.Client.Kitchen.Recipes.Item.RecipesItemRequestBuilder this[Guid position] + /// A + public global::HomeBook.Client.Modules.Kitchen.Recipes.Item.RecipesItemRequestBuilder this[Guid position] { get { var urlTplParams = new Dictionary(PathParameters); urlTplParams.Add("id", position); - return new global::HomeBook.Client.Kitchen.Recipes.Item.RecipesItemRequestBuilder(urlTplParams, RequestAdapter); + return new global::HomeBook.Client.Modules.Kitchen.Recipes.Item.RecipesItemRequestBuilder(urlTplParams, RequestAdapter); } } - /// Gets an item from the HomeBook.Client.kitchen.recipes.item collection + /// Gets an item from the HomeBook.Client.modules.kitchen.recipes.item collection /// Unique identifier of the item - /// A + /// A [Obsolete("This indexer is deprecated and will be removed in the next major version. Use the one with the typed parameter instead.")] - public global::HomeBook.Client.Kitchen.Recipes.Item.RecipesItemRequestBuilder this[string position] + public global::HomeBook.Client.Modules.Kitchen.Recipes.Item.RecipesItemRequestBuilder this[string position] { get { var urlTplParams = new Dictionary(PathParameters); if (!string.IsNullOrWhiteSpace(position)) urlTplParams.Add("id", position); - return new global::HomeBook.Client.Kitchen.Recipes.Item.RecipesItemRequestBuilder(urlTplParams, RequestAdapter); + return new global::HomeBook.Client.Modules.Kitchen.Recipes.Item.RecipesItemRequestBuilder(urlTplParams, RequestAdapter); } } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// Path parameters for the request /// The request adapter to use to execute the requests. - public RecipesRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/kitchen/recipes?searchFilter={searchFilter}", pathParameters) + public RecipesRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/kitchen/recipes?searchFilter={searchFilter}", pathParameters) { } /// - /// Instantiates a new and sets the default values. + /// Instantiates a new and sets the default values. /// /// The raw URL to use for the request builder. /// The request adapter to use to execute the requests. - public RecipesRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/kitchen/recipes?searchFilter={searchFilter}", rawUrl) + public RecipesRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules/kitchen/recipes?searchFilter={searchFilter}", rawUrl) { } /// @@ -67,11 +67,11 @@ public RecipesRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : ba /// Configuration for the request such as headers, query parameters, and middleware options. #if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER #nullable enable - public async Task GetAsync(Action>? requestConfiguration = default, CancellationToken cancellationToken = default) + public async Task GetAsync(Action>? requestConfiguration = default, CancellationToken cancellationToken = default) { #nullable restore #else - public async Task GetAsync(Action> requestConfiguration = default, CancellationToken cancellationToken = default) + public async Task GetAsync(Action> requestConfiguration = default, CancellationToken cancellationToken = default) { #endif var requestInfo = ToGetRequestInformation(requestConfiguration); @@ -103,11 +103,11 @@ public async Task PostAsync(global::HomeBook.Client.Models.CreateRecipeRequest b /// Configuration for the request such as headers, query parameters, and middleware options. #if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER #nullable enable - public RequestInformation ToGetRequestInformation(Action>? requestConfiguration = default) + public RequestInformation ToGetRequestInformation(Action>? requestConfiguration = default) { #nullable restore #else - public RequestInformation ToGetRequestInformation(Action> requestConfiguration = default) + public RequestInformation ToGetRequestInformation(Action> requestConfiguration = default) { #endif var requestInfo = new RequestInformation(Method.GET, UrlTemplate, PathParameters); @@ -131,7 +131,7 @@ public RequestInformation ToPostRequestInformation(global::HomeBook.Client.Model { #endif if(ReferenceEquals(body, null)) throw new ArgumentNullException(nameof(body)); - var requestInfo = new RequestInformation(Method.POST, "{+baseurl}/kitchen/recipes", PathParameters); + var requestInfo = new RequestInformation(Method.POST, "{+baseurl}/modules/kitchen/recipes", PathParameters); requestInfo.Configure(requestConfiguration); requestInfo.Headers.TryAdd("Accept", "application/json"); requestInfo.SetContentFromParsable(RequestAdapter, "application/json", body); @@ -140,11 +140,11 @@ public RequestInformation ToPostRequestInformation(global::HomeBook.Client.Model /// /// Returns a request builder with the provided arbitrary URL. Using this method means any other path or query parameters are ignored. /// - /// A + /// A /// The raw URL to use for the request builder. - public global::HomeBook.Client.Kitchen.Recipes.RecipesRequestBuilder WithUrl(string rawUrl) + public global::HomeBook.Client.Modules.Kitchen.Recipes.RecipesRequestBuilder WithUrl(string rawUrl) { - return new global::HomeBook.Client.Kitchen.Recipes.RecipesRequestBuilder(rawUrl, RequestAdapter); + return new global::HomeBook.Client.Modules.Kitchen.Recipes.RecipesRequestBuilder(rawUrl, RequestAdapter); } /// /// returns recipes matching the search filterHTTP 200: Recipes were foundHTTP 401: User is not authorizedHTTP 500: Unknown error while getting recipes @@ -167,7 +167,7 @@ public partial class RecipesRequestBuilderGetQueryParameters /// [Obsolete("This class is deprecated. Please use the generic RequestConfiguration class generated by the generator.")] [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] - public partial class RecipesRequestBuilderGetRequestConfiguration : RequestConfiguration + public partial class RecipesRequestBuilderGetRequestConfiguration : RequestConfiguration { } /// diff --git a/source/HomeBook.Client/Modules/ModulesRequestBuilder.cs b/source/HomeBook.Client/Modules/ModulesRequestBuilder.cs new file mode 100644 index 00000000..0556eecf --- /dev/null +++ b/source/HomeBook.Client/Modules/ModulesRequestBuilder.cs @@ -0,0 +1,47 @@ +// +#pragma warning disable CS0618 +using HomeBook.Client.Modules.Finances; +using HomeBook.Client.Modules.Kitchen; +using Microsoft.Kiota.Abstractions.Extensions; +using Microsoft.Kiota.Abstractions; +using System.Collections.Generic; +using System.IO; +using System.Threading.Tasks; +using System; +namespace HomeBook.Client.Modules +{ + /// + /// Builds and executes requests for operations under \modules + /// + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + public partial class ModulesRequestBuilder : BaseRequestBuilder + { + /// The finances property + public global::HomeBook.Client.Modules.Finances.FinancesRequestBuilder Finances + { + get => new global::HomeBook.Client.Modules.Finances.FinancesRequestBuilder(PathParameters, RequestAdapter); + } + /// The kitchen property + public global::HomeBook.Client.Modules.Kitchen.KitchenRequestBuilder Kitchen + { + get => new global::HomeBook.Client.Modules.Kitchen.KitchenRequestBuilder(PathParameters, RequestAdapter); + } + /// + /// Instantiates a new and sets the default values. + /// + /// Path parameters for the request + /// The request adapter to use to execute the requests. + public ModulesRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules", pathParameters) + { + } + /// + /// Instantiates a new and sets the default values. + /// + /// The raw URL to use for the request builder. + /// The request adapter to use to execute the requests. + public ModulesRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/modules", rawUrl) + { + } + } +} +#pragma warning restore CS0618 diff --git a/source/HomeBook.Client/Search/SearchRequestBuilder.cs b/source/HomeBook.Client/Search/SearchRequestBuilder.cs new file mode 100644 index 00000000..7fe39f18 --- /dev/null +++ b/source/HomeBook.Client/Search/SearchRequestBuilder.cs @@ -0,0 +1,108 @@ +// +#pragma warning disable CS0618 +using HomeBook.Client.Models; +using Microsoft.Kiota.Abstractions.Extensions; +using Microsoft.Kiota.Abstractions.Serialization; +using Microsoft.Kiota.Abstractions; +using System.Collections.Generic; +using System.IO; +using System.Threading.Tasks; +using System.Threading; +using System; +namespace HomeBook.Client.Search +{ + /// + /// Builds and executes requests for operations under \search + /// + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + public partial class SearchRequestBuilder : BaseRequestBuilder + { + /// + /// Instantiates a new and sets the default values. + /// + /// Path parameters for the request + /// The request adapter to use to execute the requests. + public SearchRequestBuilder(Dictionary pathParameters, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/search?s={s}", pathParameters) + { + } + /// + /// Instantiates a new and sets the default values. + /// + /// The raw URL to use for the request builder. + /// The request adapter to use to execute the requests. + public SearchRequestBuilder(string rawUrl, IRequestAdapter requestAdapter) : base(requestAdapter, "{+baseurl}/search?s={s}", rawUrl) + { + } + /// + /// returns search results based on the queryHTTP 200: Search results were foundHTTP 401: User is not authorizedHTTP 500: Unknown error while getting preference + /// + /// A + /// Cancellation token to use when cancelling requests + /// Configuration for the request such as headers, query parameters, and middleware options. +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public async Task GetAsync(Action>? requestConfiguration = default, CancellationToken cancellationToken = default) + { +#nullable restore +#else + public async Task GetAsync(Action> requestConfiguration = default, CancellationToken cancellationToken = default) + { +#endif + var requestInfo = ToGetRequestInformation(requestConfiguration); + return await RequestAdapter.SendAsync(requestInfo, global::HomeBook.Client.Models.SearchResponse.CreateFromDiscriminatorValue, default, cancellationToken).ConfigureAwait(false); + } + /// + /// returns search results based on the queryHTTP 200: Search results were foundHTTP 401: User is not authorizedHTTP 500: Unknown error while getting preference + /// + /// A + /// Configuration for the request such as headers, query parameters, and middleware options. +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + public RequestInformation ToGetRequestInformation(Action>? requestConfiguration = default) + { +#nullable restore +#else + public RequestInformation ToGetRequestInformation(Action> requestConfiguration = default) + { +#endif + var requestInfo = new RequestInformation(Method.GET, UrlTemplate, PathParameters); + requestInfo.Configure(requestConfiguration); + requestInfo.Headers.TryAdd("Accept", "application/json"); + return requestInfo; + } + /// + /// Returns a request builder with the provided arbitrary URL. Using this method means any other path or query parameters are ignored. + /// + /// A + /// The raw URL to use for the request builder. + public global::HomeBook.Client.Search.SearchRequestBuilder WithUrl(string rawUrl) + { + return new global::HomeBook.Client.Search.SearchRequestBuilder(rawUrl, RequestAdapter); + } + /// + /// returns search results based on the queryHTTP 200: Search results were foundHTTP 401: User is not authorizedHTTP 500: Unknown error while getting preference + /// + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + public partial class SearchRequestBuilderGetQueryParameters + { +#if NETSTANDARD2_1_OR_GREATER || NETCOREAPP3_1_OR_GREATER +#nullable enable + [QueryParameter("s")] + public string? S { get; set; } +#nullable restore +#else + [QueryParameter("s")] + public string S { get; set; } +#endif + } + /// + /// Configuration for the request such as headers, query parameters, and middleware options. + /// + [Obsolete("This class is deprecated. Please use the generic RequestConfiguration class generated by the generator.")] + [global::System.CodeDom.Compiler.GeneratedCode("Kiota", "1.0.0")] + public partial class SearchRequestBuilderGetRequestConfiguration : RequestConfiguration + { + } + } +} +#pragma warning restore CS0618 diff --git a/source/HomeBook.Client/kiota-lock.json b/source/HomeBook.Client/kiota-lock.json index b73ab2da..bcb7e856 100644 --- a/source/HomeBook.Client/kiota-lock.json +++ b/source/HomeBook.Client/kiota-lock.json @@ -1,5 +1,5 @@ { - "descriptionHash": "E8E4759BA46CD41CFFA957E9C23B989A77BDDBFCAA215E0A3D1A47036E3869BBCA9E07A579CB47DE3144AC8E5EA51E20883E9EB2D3BF75650416D6B5CC42D2CA", + "descriptionHash": "73735C6E167C9BA1B701A913F45406572D1DA5D5AD1F1D0C992D8408EEB6E1296AC1E036C8AFF6BD23B505FD288D0AE7FF7FC8DEAA1333B19FE9F8A822C7D956", "descriptionLocation": "../HomeBook.Backend/HomeBook.Backend.json", "lockFileVersion": "1.0.0", "kiotaVersion": "1.29.0", diff --git a/source/HomeBook.Frontend.Core/Icons/GlassMorphism.cs b/source/HomeBook.Frontend.Core/Icons/GlassMorphism.cs index a1dc63c7..14d393af 100644 --- a/source/HomeBook.Frontend.Core/Icons/GlassMorphism.cs +++ b/source/HomeBook.Frontend.Core/Icons/GlassMorphism.cs @@ -38,6 +38,7 @@ public partial class GlassMorphism public const string Tableware = "\n\n"; public const string FillColor = "\n\n"; public const string Web = "\n\n"; + public const string Home = "\n\n"; } } } diff --git a/source/HomeBook.Frontend.Core/Icons/LiquidGlassColor.cs b/source/HomeBook.Frontend.Core/Icons/LiquidGlassColor.cs index a443082a..1436e9d8 100644 --- a/source/HomeBook.Frontend.Core/Icons/LiquidGlassColor.cs +++ b/source/HomeBook.Frontend.Core/Icons/LiquidGlassColor.cs @@ -27,6 +27,7 @@ public partial class LiquidGlassColor public const string Globe = "\n\n"; public const string RestaurantBuilding = "\n\n"; public const string ColorPalette = "\n\n"; + public const string Home = "\n\n"; } } } diff --git a/source/HomeBook.Frontend.Core/Icons/Windows11Colored.cs b/source/HomeBook.Frontend.Core/Icons/Windows11Colored.cs new file mode 100644 index 00000000..3cff0079 --- /dev/null +++ b/source/HomeBook.Frontend.Core/Icons/Windows11Colored.cs @@ -0,0 +1,15 @@ +namespace HomeBook.Frontend.Core.Icons; + +public partial class HomeBookIcons +{ + public partial class Icons8 + { + public partial class Windows11 + { + public partial class Colored + { + public const string Home = "\n\n"; + } + } + } +} diff --git a/source/HomeBook.Frontend.Core/Icons/Windows11Filled.cs b/source/HomeBook.Frontend.Core/Icons/Windows11Filled.cs index f4c4bf72..785c6aab 100644 --- a/source/HomeBook.Frontend.Core/Icons/Windows11Filled.cs +++ b/source/HomeBook.Frontend.Core/Icons/Windows11Filled.cs @@ -63,6 +63,27 @@ public partial class Filled public const string Search = "\n\n"; public const string SearchBar = "\n\n"; public const string SearchAdvanced = "\n\n"; + public const string Clock = "\n\n"; + public const string Stopwatch1 = "\n\n"; + public const string Stopwatch = "\n\n"; + public const string Timer = "\n\n"; + public const string Future = "\n\n"; + public const string UserGroups = "\n\n"; + public const string Group = "\n\n"; + public const string People = "\n\n"; + public const string Unavailable = "\n\n"; + public const string Multiply = "\n\n"; + public const string Cancel = "\n\n"; + public const string DeleteShield = "\n\n"; + public const string TrashEmpty = "\n\n"; + public const string TrashFull = "\n\n"; + public const string TrashCan = "\n\n"; + public const string Trash = "\n\n"; + public const string Edit = "\n\n"; + public const string EditPencil = "\n\n"; + public const string PenSquared = "\n\n"; + public const string Pencil = "\n\n"; + public const string EditDelivery = "\n\n"; public const string Tag = "\n\n"; public const string Tags = "\n\n"; @@ -81,8 +102,6 @@ public partial class Filled public const string ShippingContainer = "\n\n"; public const string GitHub = "\n\n"; public const string Page = "\n\n"; - public const string Cancel = "\n\n"; - public const string Trash = "\n\n"; public const string EditUser = "\n\n"; public const string DeleteUser = "\n\n"; public const string AddUser = "\n\n"; diff --git a/source/HomeBook.Frontend.Core/Icons/Windows11Outline.cs b/source/HomeBook.Frontend.Core/Icons/Windows11Outline.cs index b17276e0..7fa2308c 100644 --- a/source/HomeBook.Frontend.Core/Icons/Windows11Outline.cs +++ b/source/HomeBook.Frontend.Core/Icons/Windows11Outline.cs @@ -63,6 +63,27 @@ public partial class Outline public const string Search = "\n\n"; public const string SearchBar = "\n\n"; public const string SearchAdvanced = "\n\n"; + public const string Clock = "\n\n"; + public const string Stopwatch1 = "\n\n"; + public const string Stopwatch = "\n\n"; + public const string Timer = "\n\n"; + public const string Future = "\n\n"; + public const string UserGroups = "\n\n"; + public const string Group = "\n\n"; + public const string People = "\n\n"; + public const string Unavailable = "\n\n"; + public const string Multiply = "\n\n"; + public const string Cancel = "\n\n"; + public const string DeleteShield = "\n\n"; + public const string TrashEmpty = "\n\n"; + public const string TrashFull = "\n\n"; + public const string TrashCan = "\n\n"; + public const string Trash = "\n\n"; + public const string Edit = "\n\n"; + public const string EditPencil = "\n\n"; + public const string PenSquared = "\n\n"; + public const string Pencil = "\n\n"; + public const string EditDelivery = "\n\n"; } } } diff --git a/source/HomeBook.Frontend.Module.Finances/HomeBook.Frontend.Module.Finances.csproj b/source/HomeBook.Frontend.Module.Finances/HomeBook.Frontend.Module.Finances.csproj index a11069e1..1d6be1f1 100644 --- a/source/HomeBook.Frontend.Module.Finances/HomeBook.Frontend.Module.Finances.csproj +++ b/source/HomeBook.Frontend.Module.Finances/HomeBook.Frontend.Module.Finances.csproj @@ -10,7 +10,7 @@ - + diff --git a/source/HomeBook.Frontend.Module.Finances/Pages/Saving/Add.razor.cs b/source/HomeBook.Frontend.Module.Finances/Pages/Saving/Add.razor.cs index 47353d4a..b846ed7e 100644 --- a/source/HomeBook.Frontend.Module.Finances/Pages/Saving/Add.razor.cs +++ b/source/HomeBook.Frontend.Module.Finances/Pages/Saving/Add.razor.cs @@ -75,7 +75,7 @@ protected async Task NextStep() // calculate with target date string? token = await AuthenticationService.GetTokenAsync(cancellationToken); - CalculatedSavingResponse? response = await BackendClient.Finances.Calculations.Savings + CalculatedSavingResponse? response = await BackendClient.Modules.Finances.Calculations.Savings .PostAsync(new CalculateSavingRequest { TargetAmount = Convert.ToDouble(_model.TargetAmount), @@ -125,7 +125,7 @@ protected async Task SaveAsync() CancellationToken cancellationToken = CancellationToken.None; string? token = await AuthenticationService.GetTokenAsync(cancellationToken); - await BackendClient.Finances.SavingGoals + await BackendClient.Modules.Finances.SavingGoals .PostAsync( _summaryVM.ToRequest(), x => diff --git a/source/HomeBook.Frontend.Module.Finances/Services/SavingGoalService.cs b/source/HomeBook.Frontend.Module.Finances/Services/SavingGoalService.cs index 8927d250..2b3fe8ac 100644 --- a/source/HomeBook.Frontend.Module.Finances/Services/SavingGoalService.cs +++ b/source/HomeBook.Frontend.Module.Finances/Services/SavingGoalService.cs @@ -16,7 +16,7 @@ public class SavingGoalService( public async Task> GetAllSavingGoalsAsync(CancellationToken cancellationToken = default) { string? token = await authenticationService.GetTokenAsync(cancellationToken); - SavingGoalListResponse? response = await backendClient.Finances.SavingGoals.GetAsync(x => + SavingGoalListResponse? response = await backendClient.Modules.Finances.SavingGoals.GetAsync(x => { x.Headers.Add("Authorization", $"Bearer {token}"); }, diff --git a/source/HomeBook.Frontend.Module.Kitchen/Components/UiMealCard.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Components/UiMealCard.razor.cs index 44b27ab1..5ac05b60 100644 --- a/source/HomeBook.Frontend.Module.Kitchen/Components/UiMealCard.razor.cs +++ b/source/HomeBook.Frontend.Module.Kitchen/Components/UiMealCard.razor.cs @@ -1,5 +1,4 @@ using HomeBook.Frontend.Module.Kitchen.ViewModels; -using HomeBook.Frontend.UI.Components; using HomeBook.Frontend.UI.Utilities; using Microsoft.AspNetCore.Components; @@ -9,7 +8,7 @@ public partial class UiMealCard : ComponentBase { protected string Style => new HtmlArgumentBuilder("") - .AddClass($"background-color: color-mix(in srgb, {MealTypeColor}, transparent 90%);") + .AddClass($"background-color: color-mix(in srgb, {MealTypeColor}, transparent 75%)") .Build(); protected string CssClass => @@ -31,7 +30,7 @@ public partial class UiMealCard : ComponentBase public string Class { get; set; } = string.Empty; [Parameter] - public MealItemViewModel? Meal { get; set; } = null; + public RecipeViewModel? Meal { get; set; } = null; [Parameter] public EventCallback OnAdd { get; set; } @@ -39,13 +38,13 @@ public partial class UiMealCard : ComponentBase [Parameter] public EventCallback OnDelete { get; set; } - private async Task HandleAddClick() + public async Task HandleAddClick() { if (OnAdd.HasDelegate) await OnAdd.InvokeAsync(null); } - private async Task HandleDeleteClick() + protected async Task HandleDeleteClick() { if (OnDelete.HasDelegate) await OnDelete.InvokeAsync(null); diff --git a/source/HomeBook.Frontend.Module.Kitchen/Components/UiMealDay.razor b/source/HomeBook.Frontend.Module.Kitchen/Components/UiMealDay.razor new file mode 100644 index 00000000..782240cd --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Components/UiMealDay.razor @@ -0,0 +1,46 @@ +@using HomeBook.Frontend.Core.Icons +@using HomeBook.Frontend.Module.Kitchen.Enums +@using HomeBook.Frontend.Module.Kitchen.Resources +@using Microsoft.Extensions.Localization + +@inject IStringLocalizer Loc +@inject IDialogService DialogService + + + + + + @MealPlanItem.Date.ToString("D") + + + +
+ + + + + + + +
+ +
diff --git a/source/HomeBook.Frontend.Module.Kitchen/Components/UiMealDay.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Components/UiMealDay.razor.cs new file mode 100644 index 00000000..566d45d0 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Components/UiMealDay.razor.cs @@ -0,0 +1,102 @@ +using HomeBook.Frontend.Module.Kitchen.Dialogs; +using HomeBook.Frontend.Module.Kitchen.Enums; +using HomeBook.Frontend.Module.Kitchen.Models; +using HomeBook.Frontend.Module.Kitchen.ViewModels; +using HomeBook.Frontend.UI.Utilities; +using Microsoft.AspNetCore.Components; +using MudBlazor; + +namespace HomeBook.Frontend.Module.Kitchen.Components; + +public partial class UiMealDay : ComponentBase +{ + [Parameter] + public MealPlanItemViewModel MealPlanItem { get; set; } = null!; + + [Parameter] + public EventCallback OnMealPlanItemChanged { get; set; } + + protected override async Task OnParametersSetAsync() + { + await base.OnParametersSetAsync(); + + ArgumentNullException.ThrowIfNull(MealPlanItem, nameof(MealPlanItem)); + } + + protected string HeaderCssClass => + new HtmlArgumentBuilder("ui-meal-card") + .AddClass("border-0") + .AddClass($"ui-color-bg-gradient-{MealPlanItem!.ColorName}", + !string.IsNullOrEmpty(MealPlanItem.ColorName)) + .Build(); + + private async Task OnMealAdd(MealType mealType, DateOnly date) + { + IDialogReference dialogReference = await DialogService.ShowAsync( + "+Gericht auswählen", + new DialogOptions() + { + MaxWidth = MaxWidth.Small, + FullWidth = true, + CloseOnEscapeKey = true, + CloseButton = true + }); + + DialogResult? dialogResult = await dialogReference.Result; + if (dialogResult is null) + return; + + RecipeViewModel? meal = (dialogResult.Data as RecipeViewModel); + if (meal is null) + return; + + // MealPlanItemViewModel? mealPlanItem = _mealPlanItems.FirstOrDefault(item => item.Date == date); + switch (mealType) + { + case MealType.Breakfast: + MealPlanItem!.Breakfast = meal; + break; + case MealType.Lunch: + MealPlanItem!.Lunch = meal; + break; + case MealType.Dinner: + MealPlanItem!.Dinner = meal; + break; + default: + throw new ArgumentOutOfRangeException(nameof(mealType), mealType, null); + } + + StateHasChanged(); + + MealPlanChangedDto dto = new(MealPlanChangedAction.Added, + meal, + mealType, + date); + await OnMealPlanItemChanged.InvokeAsync(dto); + } + + private async Task OnMealDelete(MealType mealType, DateOnly date) + { + // MealPlanItemViewModel? mealPlanItem = _mealPlanItems.FirstOrDefault(item => item.Date == date); + switch (mealType) + { + case MealType.Breakfast: + MealPlanItem!.Breakfast = null; + break; + case MealType.Lunch: + MealPlanItem!.Lunch = null; + break; + case MealType.Dinner: + MealPlanItem!.Dinner = null; + break; + } + + StateHasChanged(); + + MealPlanChangedDto dto = new(MealPlanChangedAction.Removed, + null, + mealType, + date); + await OnMealPlanItemChanged.InvokeAsync(dto); + } +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Contracts/IMealService.cs b/source/HomeBook.Frontend.Module.Kitchen/Contracts/IMealService.cs deleted file mode 100644 index 5b97fd4e..00000000 --- a/source/HomeBook.Frontend.Module.Kitchen/Contracts/IMealService.cs +++ /dev/null @@ -1,15 +0,0 @@ -using HomeBook.Frontend.Module.Kitchen.Models; - -namespace HomeBook.Frontend.Module.Kitchen.Contracts; - -public interface IMealService -{ - /// - /// - /// - /// - /// - /// - Task> GetMealsAsync(string filter, - CancellationToken cancellationToken = default); -} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Contracts/IRecipeService.cs b/source/HomeBook.Frontend.Module.Kitchen/Contracts/IRecipeService.cs new file mode 100644 index 00000000..8fb9e4a8 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Contracts/IRecipeService.cs @@ -0,0 +1,71 @@ +using HomeBook.Frontend.Module.Kitchen.Models; + +namespace HomeBook.Frontend.Module.Kitchen.Contracts; + +public interface IRecipeService +{ + /// + /// + /// + /// + /// + /// + Task> GetRecipesAsync(string? filter, + CancellationToken cancellationToken = default); + + /// + /// + /// + /// + /// + /// + Task GetRecipeByIdAsync(Guid id, + CancellationToken cancellationToken = default); + + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + /// + Task CreateRecipeAsync(string name, + string? description = null, + int? servings = null, + RecipeStepDto[]? steps = null, + RecipeIngredientDto[]? ingredients = null, + int? durationWorkingMinutes = null, + int? durationCookingMinutes = null, + int? durationRestingMinutes = null, + int? caloriesKcal = null, + string? comments = null, + string? source = null, + CancellationToken cancellationToken = default); + + /// + /// + /// + /// + /// + /// + Task CreateRecipeAsync(string name, + CancellationToken cancellationToken = default); + + /// + /// + /// + /// + /// + /// + Task DeleteRecipeAsync(Guid recipeId, + CancellationToken cancellationToken = default); +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Dialogs/MealSelectDialog.razor b/source/HomeBook.Frontend.Module.Kitchen/Dialogs/MealSelectDialog.razor index d4a88f17..88a12ac8 100644 --- a/source/HomeBook.Frontend.Module.Kitchen/Dialogs/MealSelectDialog.razor +++ b/source/HomeBook.Frontend.Module.Kitchen/Dialogs/MealSelectDialog.razor @@ -1,21 +1,33 @@ +@using HomeBook.Frontend.Core.Icons @using HomeBook.Frontend.Module.Kitchen.Contracts @using HomeBook.Frontend.Module.Kitchen.ViewModels -@inject IMealService MealService +@inject IRecipeService RecipeService - +
+ + + + + +
@if (_isLoading) { @@ -23,11 +35,11 @@ Indeterminate="true"/> } - + - @foreach (MealItemViewModel meal in _mealItems) + @foreach (RecipeViewModel meal in _mealItems) { - @@ -47,14 +59,4 @@
- - - - +Neues Gericht anlegen - - - -
diff --git a/source/HomeBook.Frontend.Module.Kitchen/Dialogs/MealSelectDialog.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Dialogs/MealSelectDialog.razor.cs index 26a9b0f9..1ec5b6f0 100644 --- a/source/HomeBook.Frontend.Module.Kitchen/Dialogs/MealSelectDialog.razor.cs +++ b/source/HomeBook.Frontend.Module.Kitchen/Dialogs/MealSelectDialog.razor.cs @@ -11,10 +11,12 @@ public partial class MealSelectDialog : ComponentBase [CascadingParameter] private IMudDialogInstance MudDialog { get; set; } - private List _mealItems = []; + private List _mealItems = []; private bool _isLoading { get; set; } private CancellationTokenSource _cancellationTokenSource = new(); - private Timer? _searchDebounceTimer; + private Timer? _searchInputDebounceTimer; + + private MudTextField? _searchTextField; protected override async Task OnAfterRenderAsync(bool firstRender) { @@ -23,38 +25,37 @@ protected override async Task OnAfterRenderAsync(bool firstRender) if (!firstRender) return; - // TODO: load available meals from service - await SearchAsync(string.Empty); + await SearchAsync(); } - private string? _text; + private string? _searchText; private void OnSearchTextFieldValueChanged(string? value) { - _text = value; + _searchText = value; // Reset existing timer if any - _searchDebounceTimer?.Dispose(); + _searchInputDebounceTimer?.Dispose(); // Create new timer with 3 seconds delay long delayMilliseconds = 1000; - _searchDebounceTimer = new Timer(TimerCallback, + _searchInputDebounceTimer = new Timer(SearchInputDebounceTimerCallback, null, TimeSpan.FromMilliseconds(delayMilliseconds), Timeout.InfiniteTimeSpan); } - private async void TimerCallback(object? state) + private async void SearchInputDebounceTimerCallback(object? state) { await InvokeAsync(async () => { - _searchDebounceTimer = null; + _searchInputDebounceTimer = null; - await SearchAsync(_text ?? string.Empty); + await SearchAsync(_searchText ?? string.Empty); }); } - private async Task SearchAsync(string searchText) + private async Task SearchAsync(string? searchText = null) { CancellationToken cancellationToken = _cancellationTokenSource.Token; @@ -66,7 +67,7 @@ private async Task SearchAsync(string searchText) // Check if cancelled before starting cancellationToken.ThrowIfCancellationRequested(); - IEnumerable meals = await MealService.GetMealsAsync(searchText, + IEnumerable meals = await RecipeService.GetRecipesAsync(searchText, cancellationToken); _mealItems.Clear(); @@ -92,10 +93,14 @@ private async Task SearchAsync(string searchText) _isLoading = false; StateHasChanged(); } + + if (_searchTextField is not null) + await _searchTextField.FocusAsync(); + StateHasChanged(); } } - private async Task SelectMealAsync(MealItemViewModel meal) + private async Task SelectMealAsync(RecipeViewModel meal) { await ShutdownAsync(); @@ -115,7 +120,22 @@ private async Task ShutdownAsync() _cancellationTokenSource.Dispose(); // Cleanup timer - if (_searchDebounceTimer is not null) - await _searchDebounceTimer.DisposeAsync(); + if (_searchInputDebounceTimer is not null) + await _searchInputDebounceTimer.DisposeAsync(); + } + + private async Task CreateAsNewMeal() + { + string? mealName = _searchText?.Trim(); + + if (string.IsNullOrWhiteSpace(mealName)) + return; + + CancellationToken cancellationToken = _cancellationTokenSource.Token; + await RecipeService.CreateRecipeAsync(mealName, + cancellationToken); + + // reload + await SearchAsync(mealName); } } diff --git a/source/HomeBook.Frontend.Module.Kitchen/Enums/MealPlanChangedAction.cs b/source/HomeBook.Frontend.Module.Kitchen/Enums/MealPlanChangedAction.cs new file mode 100644 index 00000000..2d293b13 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Enums/MealPlanChangedAction.cs @@ -0,0 +1,7 @@ +namespace HomeBook.Frontend.Module.Kitchen.Enums; + +public enum MealPlanChangedAction +{ + Added, + Removed +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/HomeBook.Frontend.Module.Kitchen.csproj b/source/HomeBook.Frontend.Module.Kitchen/HomeBook.Frontend.Module.Kitchen.csproj index a11069e1..98ee1a22 100644 --- a/source/HomeBook.Frontend.Module.Kitchen/HomeBook.Frontend.Module.Kitchen.csproj +++ b/source/HomeBook.Frontend.Module.Kitchen/HomeBook.Frontend.Module.Kitchen.csproj @@ -10,7 +10,8 @@ - + + diff --git a/source/HomeBook.Frontend.Module.Kitchen/Mappings/MealMappings.cs b/source/HomeBook.Frontend.Module.Kitchen/Mappings/MealMappings.cs deleted file mode 100644 index 45b6b927..00000000 --- a/source/HomeBook.Frontend.Module.Kitchen/Mappings/MealMappings.cs +++ /dev/null @@ -1,26 +0,0 @@ -using HomeBook.Frontend.Module.Kitchen.Models; -using HomeBook.Frontend.Module.Kitchen.ViewModels; - -namespace HomeBook.Frontend.Module.Kitchen.Mappings; - -public static class MealMappings -{ - public static MealItemViewModel ToViewModel(this RecipeDto recipe) - { - return new MealItemViewModel - { - Name = recipe.Name, - Ingredients = recipe.Ingredients, - Duration = recipe.Duration, - CaloriesKcal = recipe.CaloriesKcal - }; - } - - public static RecipeDto ToDto(this HomeBook.Client.Models.RecipeResponse r) => - new( - r.Id!.Value, - r.Name!, - "", - null, - null); -} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Mappings/RecipeMappings.cs b/source/HomeBook.Frontend.Module.Kitchen/Mappings/RecipeMappings.cs new file mode 100644 index 00000000..ba318908 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Mappings/RecipeMappings.cs @@ -0,0 +1,87 @@ +using HomeBook.Client.Models; +using HomeBook.Frontend.Module.Kitchen.ViewModels; +using HomeBook.Frontend.Module.Kitchen.Models; +using HomeBook.Frontend.Module.Kitchen.ViewModels; + +namespace HomeBook.Frontend.Module.Kitchen.Mappings; + +public static class RecipeMappings +{ + public static RecipeDetailViewModel ToViewModel(this RecipeDetailDto recipe) + { + TimeSpan? duration = recipe.DurationInMinutes.HasValue + ? TimeSpan.FromMinutes(recipe.DurationInMinutes.Value) + : null; + + return new RecipeDetailViewModel + { + Id = recipe.Id, + Username = recipe.Username, + Name = recipe.Name, + Description = recipe.Description, + Servings = recipe.Servings, + CaloriesKcal = recipe.CaloriesKcal, + Duration = duration, + // Ingredients = recipe.Ingredients, + Image = TestImageMappings.PlaceholderImage + }; + } + + public static RecipeViewModel ToViewModel(this RecipeDto recipe) + { + TimeSpan? duration = recipe.DurationInMinutes.HasValue + ? TimeSpan.FromMinutes(recipe.DurationInMinutes.Value) + : null; + + return new RecipeViewModel + { + Id = recipe.Id, + Username = recipe.Username, + Name = recipe.Name, + Description = recipe.Description, + Servings = recipe.Servings, + CaloriesKcal = recipe.CaloriesKcal, + Duration = duration, + Ingredients = recipe.Ingredients, + Image = TestImageMappings.PlaceholderImage + }; + } + + public static RecipeDto ToDto(this HomeBook.Client.Models.RecipeResponse r) => + new( + r.Id!.Value, + r.Username!, + r.Name!, + r.Description!, + r.Servings, + r.CaloriesKcal, + r.DurationCookingMinutes, + ""); + + public static RecipeDetailDto ToDto(this HomeBook.Client.Models.RecipeDetailResponse r) => + new( + r.Id!.Value, + r.Username!, + r.Name!, + r.Description!, + r.Servings, + r.CaloriesKcal, + r.DurationCookingMinutes, + ""); + + public static CreateRecipeStepRequest ToRequest(this RecipeStepDto dto) => + new() + { + Description = dto.Description, + Position = dto.Position, + TimerDurationInSeconds = dto.TimerDurationInSeconds + }; + + public static CreateRecipeIngredientRequest ToRequest(this RecipeIngredientDto dto) => + new() + { + Name = dto.Name, + Quantity = dto.Quantity, + Unit = dto.Unit, + }; +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Mappings/TestImageMappings.cs b/source/HomeBook.Frontend.Module.Kitchen/Mappings/TestImageMappings.cs new file mode 100644 index 00000000..86fcf142 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Mappings/TestImageMappings.cs @@ -0,0 +1,7 @@ +namespace HomeBook.Frontend.Module.Kitchen.Mappings; + +public static class TestImageMappings +{ + public static string PlaceholderImage = + ""; +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Models/MealPlanChangedDto.cs b/source/HomeBook.Frontend.Module.Kitchen/Models/MealPlanChangedDto.cs new file mode 100644 index 00000000..b7d25ceb --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Models/MealPlanChangedDto.cs @@ -0,0 +1,10 @@ +using HomeBook.Frontend.Module.Kitchen.Enums; +using HomeBook.Frontend.Module.Kitchen.ViewModels; + +namespace HomeBook.Frontend.Module.Kitchen.Models; + +public record MealPlanChangedDto( + MealPlanChangedAction Action, + RecipeViewModel? Recipe, + MealType MealType, + DateOnly Date); diff --git a/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeDetailDto.cs b/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeDetailDto.cs new file mode 100644 index 00000000..9b4e1b84 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeDetailDto.cs @@ -0,0 +1,11 @@ +namespace HomeBook.Frontend.Module.Kitchen.Models; + +public record RecipeDetailDto( + Guid Id, + string Username, + string Name, + string Description, + int? Servings, + int? CaloriesKcal, + int? DurationInMinutes, + string Ingredients); diff --git a/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeDto.cs b/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeDto.cs index 4eddbd5b..47161c6f 100644 --- a/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeDto.cs +++ b/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeDto.cs @@ -2,7 +2,10 @@ namespace HomeBook.Frontend.Module.Kitchen.Models; public record RecipeDto( Guid Id, + string Username, string Name, - string Ingredients, - TimeSpan? Duration, - int? CaloriesKcal); + string Description, + int? Servings, + int? CaloriesKcal, + int? DurationInMinutes, + string Ingredients); diff --git a/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeIngredientDto.cs b/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeIngredientDto.cs new file mode 100644 index 00000000..b2e5087c --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeIngredientDto.cs @@ -0,0 +1,6 @@ +namespace HomeBook.Frontend.Module.Kitchen.Models; + +public record RecipeIngredientDto( + string Name, + double? Quantity, + string? Unit); diff --git a/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeStepDto.cs b/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeStepDto.cs new file mode 100644 index 00000000..17998dc0 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Models/RecipeStepDto.cs @@ -0,0 +1,6 @@ +namespace HomeBook.Frontend.Module.Kitchen.Models; + +public record RecipeStepDto( + string Description, + int Position, + int? TimerDurationInSeconds); diff --git a/source/HomeBook.Frontend.Module.Kitchen/Module.cs b/source/HomeBook.Frontend.Module.Kitchen/Module.cs index 53ad4d76..1e3ff9fa 100644 --- a/source/HomeBook.Frontend.Module.Kitchen/Module.cs +++ b/source/HomeBook.Frontend.Module.Kitchen/Module.cs @@ -52,7 +52,7 @@ public static void RegisterWidgets(IWidgetBuilder builder, public static void RegisterServices(IServiceCollection services, IConfiguration configuration) { - services.AddSingleton(); + services.AddSingleton(); } /// diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/MealPlan/PlanOverview.razor b/source/HomeBook.Frontend.Module.Kitchen/Pages/MealPlan/PlanOverview.razor index 303edb4c..8ccd5cba 100644 --- a/source/HomeBook.Frontend.Module.Kitchen/Pages/MealPlan/PlanOverview.razor +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/MealPlan/PlanOverview.razor @@ -3,6 +3,7 @@ @using HomeBook.Frontend.Core.Icons @using HomeBook.Frontend.Module.Kitchen.Components @using HomeBook.Frontend.Module.Kitchen.Enums +@using HomeBook.Frontend.Module.Kitchen.Models @using HomeBook.Frontend.Module.Kitchen.Resources @using HomeBook.Frontend.Module.Kitchen.ViewModels @using Microsoft.Extensions.Localization @@ -12,65 +13,34 @@ - + - - @_startDate.ToString(Loc[nameof(Strings.DateTimeFormat_ShortDateOnlyFormat)]) - @_endDate.ToString(Loc[nameof(Strings.DateTimeFormat_ShortDateOnlyFormat)]) - + - + + @_startDate.ToString(Loc[nameof(Strings.DateTimeFormat_ShortDateOnlyFormat)]) - @_endDate.ToString(Loc[nameof(Strings.DateTimeFormat_ShortDateOnlyFormat)]) + - + - - @string.Format(Loc[nameof(Strings.PlanOverview_Header_CalendarWeek_TextTemplate)], @_calendarWeek) - + - + + @string.Format(Loc[nameof(Strings.PlanOverview_Header_CalendarWeek_TextTemplate)], @_calendarWeek) + - - - @foreach (MealPlanItemViewModel meal in _mealPlanItems) - { - - - - - @meal.Date.ToString("D") - - - -
+ - + - - - - -
+
- + @foreach (MealPlanItemViewModel mealPlanItem in _mealPlanItems) + { + }
diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/MealPlan/PlanOverview.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Pages/MealPlan/PlanOverview.razor.cs index fb349c43..8c7d8d21 100644 --- a/source/HomeBook.Frontend.Module.Kitchen/Pages/MealPlan/PlanOverview.razor.cs +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/MealPlan/PlanOverview.razor.cs @@ -1,8 +1,7 @@ -using HomeBook.Frontend.Module.Kitchen.Dialogs; using HomeBook.Frontend.Module.Kitchen.Enums; +using HomeBook.Frontend.Module.Kitchen.Models; using HomeBook.Frontend.Module.Kitchen.ViewModels; using Microsoft.AspNetCore.Components; -using MudBlazor; namespace HomeBook.Frontend.Module.Kitchen.Pages.MealPlan; @@ -32,21 +31,21 @@ protected override async Task OnAfterRenderAsync(bool firstRender) Id = Guid.NewGuid(), Date = DateOnly.FromDateTime(DateTime.Today), ColorName = "cerulean", - Breakfast = new MealItemViewModel() + Breakfast = new RecipeViewModel() { Name = "Omelette mit Speck", Ingredients = "Eier, Speck, Milch, Gewürze", Duration = TimeSpan.FromMinutes(15), CaloriesKcal = 350 }, - Lunch = new MealItemViewModel() + Lunch = new RecipeViewModel() { Name = "Würstchen mit Kartoffelsalat", Ingredients = "Würstchen, Kartoffeln, Mayonnaise, Zwiebeln, Gurken", Duration = TimeSpan.FromMinutes(135), CaloriesKcal = 800 }, - Dinner = new MealItemViewModel() + Dinner = new RecipeViewModel() { Name = "Bratkartoffeln mit Spiegelei", Ingredients = "Kartoffeln, Eier, Zwiebeln, Gewürze", @@ -58,7 +57,7 @@ protected override async Task OnAfterRenderAsync(bool firstRender) Id = Guid.NewGuid(), Date = DateOnly.FromDateTime(DateTime.Today.AddDays(1)), ColorName = "fern", - Lunch = new MealItemViewModel() + Lunch = new RecipeViewModel() { Name = "Würstchen mit Kartoffelsalat", Ingredients = "Würstchen, Kartoffeln, Mayonnaise, Zwiebeln, Gurken", @@ -70,7 +69,7 @@ protected override async Task OnAfterRenderAsync(bool firstRender) Id = Guid.NewGuid(), Date = DateOnly.FromDateTime(DateTime.Today.AddDays(2)), ColorName = "amber", - Dinner = new MealItemViewModel() + Dinner = new RecipeViewModel() { Name = "Bratkartoffeln mit Spiegelei", Ingredients = "Kartoffeln, Eier, Zwiebeln, Gewürze", @@ -82,7 +81,7 @@ protected override async Task OnAfterRenderAsync(bool firstRender) Id = Guid.NewGuid(), Date = DateOnly.FromDateTime(DateTime.Today.AddDays(3)), ColorName = "azure", - Breakfast = new MealItemViewModel() + Breakfast = new RecipeViewModel() { Name = "Fischstäbchen mit Kartoffelpüree", Ingredients = "Fischstäbchen, Kartoffeln, Butter, Milch", @@ -94,7 +93,7 @@ protected override async Task OnAfterRenderAsync(bool firstRender) Id = Guid.NewGuid(), Date = DateOnly.FromDateTime(DateTime.Today.AddDays(4)), ColorName = "chartreuse", - Breakfast = new MealItemViewModel() + Breakfast = new RecipeViewModel() { Name = "Eintopf mit Würstchen", Ingredients = "Würstchen, Kartoffeln, Gemüse, Gewürze", @@ -106,7 +105,7 @@ protected override async Task OnAfterRenderAsync(bool firstRender) Id = Guid.NewGuid(), Date = DateOnly.FromDateTime(DateTime.Today.AddDays(5)), ColorName = "jade", - Breakfast = new MealItemViewModel() + Breakfast = new RecipeViewModel() { Name = "Hähnchenschenkel mit Reis", Ingredients = "Hähnchenschenkel, Reis, Gewürze", @@ -118,7 +117,7 @@ protected override async Task OnAfterRenderAsync(bool firstRender) Id = Guid.NewGuid(), Date = DateOnly.FromDateTime(DateTime.Today.AddDays(6)), ColorName = "plum", - Breakfast = new MealItemViewModel() + Breakfast = new RecipeViewModel() { Name = "Roastbeef mit Gemüse", Ingredients = "Kartoffeln, Gemüse, Gewürze", @@ -130,57 +129,34 @@ protected override async Task OnAfterRenderAsync(bool firstRender) StateHasChanged(); } - private async Task OnMealAdd(MealType mealType, DateOnly date) + private async Task OnMealAdd(MealType mealType, DateOnly date, RecipeViewModel meal) { - IDialogReference dialogReference = await DialogService.ShowAsync( - "+Gericht auswählen", - new DialogOptions() - { - MaxWidth = MaxWidth.Small, - FullWidth = true, - CloseOnEscapeKey = true, - CloseButton = true - }); - - DialogResult? dialogResult = await dialogReference.Result; - if (dialogResult is null) - return; + // TODO: call rest - MealItemViewModel meal = (dialogResult.Data as MealItemViewModel)!; + int i = 0; + } - MealPlanItemViewModel? mealPlanItem = _mealPlanItems.FirstOrDefault(item => item.Date == date); - switch (mealType) - { - case MealType.Breakfast: - mealPlanItem!.Breakfast = meal; - break; - case MealType.Lunch: - mealPlanItem!.Lunch = meal; - break; - case MealType.Dinner: - mealPlanItem!.Dinner = meal; - break; - } + private async Task OnMealDelete(MealType mealType, DateOnly date) + { + // TODO: call rest - StateHasChanged(); + int i = 0; } - private async Task OnMealDelete(MealType mealType, DateOnly date) + private async Task OnMealPlanItemChanged(MealPlanChangedDto eventArgs) { - MealPlanItemViewModel? mealPlanItem = _mealPlanItems.FirstOrDefault(item => item.Date == date); - switch (mealType) + switch (eventArgs.Action) { - case MealType.Breakfast: - mealPlanItem!.Breakfast = null; - break; - case MealType.Lunch: - mealPlanItem!.Lunch = null; + case MealPlanChangedAction.Removed: + // remove meal from plan + await OnMealDelete(eventArgs.MealType, eventArgs.Date); break; - case MealType.Dinner: - mealPlanItem!.Dinner = null; + case MealPlanChangedAction.Added: + // add or update meal in plan + await OnMealAdd(eventArgs.MealType, eventArgs.Date, eventArgs.Recipe!); break; + default: + throw new ArgumentOutOfRangeException(); } - - StateHasChanged(); } } diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeDescriptionServings.razor b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeDescriptionServings.razor new file mode 100644 index 00000000..c21ab4d9 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeDescriptionServings.razor @@ -0,0 +1,35 @@ + + + + + + + @switch (NumberOfServings) + { + case 1: + + +Für 1 Portion + + break; + default: + + +Für @NumberOfServings Portionen + + break; + } + + + + + + + + diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeDescriptionServings.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeDescriptionServings.razor.cs new file mode 100644 index 00000000..7faa55aa --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeDescriptionServings.razor.cs @@ -0,0 +1,27 @@ +using Microsoft.AspNetCore.Components; + +namespace HomeBook.Frontend.Module.Kitchen.Pages.Recipes.Components; + +public partial class UiRecipeDescriptionServings : ComponentBase +{ + [Parameter] + public string Description { get; set; } = string.Empty; + + protected int _numberOfServings; + + [Parameter] + public int NumberOfServings + { + get => _numberOfServings; + set => _numberOfServings = value; + } + + [Parameter] + public EventCallback NumberOfServingsChanged { get; set; } + + private async Task OnNumberOfServingsChanged(int value) + { + await NumberOfServingsChanged.InvokeAsync(value); + StateHasChanged(); + } +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeImageUpload.razor b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeImageUpload.razor new file mode 100644 index 00000000..4a239905 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeImageUpload.razor @@ -0,0 +1,121 @@ +@using Microsoft.AspNetCore.Components.Forms + +@inject ISnackbar Snackbar + + + +Rezeptbild + + + + +Füge jetzt oder später bis zu zehn Bilder von deinem Rezept hinzu (Empfohlene Breite mind. 1024px, nur + JPG- oder PNG-Dateien). Das erste Bild ist dein Vorschaubild. + + + + + + + + + + +Open file picker + + + + ---Upload--- + + + + +Clear + + + + + + +@code { +#nullable enable + private const string DefaultDragClass = "relative rounded-xl border-2 border-dashed pa-3 mt-3 mx-3 frosted-bg-b1 mud-height-full"; + private string _dragClass = DefaultDragClass; + private readonly List _fileNames = new(); + private MudFileUpload>? _fileUpload; + + private async Task ClearAsync() + { + await (_fileUpload?.ClearAsync() ?? Task.CompletedTask); + _fileNames.Clear(); + ClearDragClass(); + } + + private Task OpenFilePickerAsync() + => _fileUpload?.OpenFilePickerAsync() ?? Task.CompletedTask; + + private void OnInputFileChanged(InputFileChangeEventArgs e) + { + ClearDragClass(); + var files = e.GetMultipleFiles(); + foreach (var file in files) + { + _fileNames.Add(file.Name); + } + } + + private void Upload() + { + // Upload the files here + Snackbar.Configuration.PositionClass = Defaults.Classes.Position.TopCenter; + Snackbar.Add("TODO: Upload your files!"); + } + + private void SetDragClass() + => _dragClass = $"{DefaultDragClass} mud-border-primary"; + + private void ClearDragClass() + => _dragClass = DefaultDragClass; + +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeImageUpload.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeImageUpload.razor.cs new file mode 100644 index 00000000..488cf2aa --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeImageUpload.razor.cs @@ -0,0 +1,7 @@ +using Microsoft.AspNetCore.Components; + +namespace HomeBook.Frontend.Module.Kitchen.Pages.Recipes.Components; + +public partial class UiRecipeImageUpload : ComponentBase +{ +} \ No newline at end of file diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeIngredientsList.razor b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeIngredientsList.razor new file mode 100644 index 00000000..0280a446 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeIngredientsList.razor @@ -0,0 +1,64 @@ +@using HomeBook.Frontend.Core.Icons +@using HomeBook.Frontend.Module.Kitchen.ViewModels + + + +Zutaten + + + + + @foreach (IngredientViewModel ingredient in Ingredients) + { + + +
+ + + +
+ + + @ingredient.DisplayText + + + @if (!string.IsNullOrWhiteSpace(ingredient.AdditionalText)) + { + @ingredient.AdditionalText + } + +
+ + + + +
+ +
+ } + + + +
+ + + + + +
+ +
+ +
diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeIngredientsList.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeIngredientsList.razor.cs new file mode 100644 index 00000000..d525b2c4 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeIngredientsList.razor.cs @@ -0,0 +1,10 @@ +using HomeBook.Frontend.Module.Kitchen.ViewModels; +using Microsoft.AspNetCore.Components; + +namespace HomeBook.Frontend.Module.Kitchen.Pages.Recipes.Components; + +public partial class UiRecipeIngredientsList : ComponentBase +{ + [Parameter] + public IEnumerable Ingredients { get; set; } = new List(); +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeMetaEdit.razor b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeMetaEdit.razor new file mode 100644 index 00000000..40b31d52 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeMetaEdit.razor @@ -0,0 +1,141 @@ + + +Dauer & weitere Angaben + + + + + + + + + + +Arbeitszeit + + +
+ + +
+ +
+ + + + + +Koch/Backzeit + + +
+ + +
+
+ + + + + +Ruhezeit + + +
+ + +
+
+ + + + + +Kalorien/Portion + + +
+ +
+
+ +
+ +
+ + + + + +Kommentar + + + + + + +Quelle + + + + + diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeMetaEdit.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeMetaEdit.razor.cs new file mode 100644 index 00000000..7ed46be2 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeMetaEdit.razor.cs @@ -0,0 +1,7 @@ +using Microsoft.AspNetCore.Components; + +namespace HomeBook.Frontend.Module.Kitchen.Pages.Recipes.Components; + +public partial class UiRecipeMetaEdit : ComponentBase +{ +} \ No newline at end of file diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeStepsList.razor b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeStepsList.razor new file mode 100644 index 00000000..1e63bf4e --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeStepsList.razor @@ -0,0 +1,86 @@ +@using HomeBook.Frontend.Core.Icons + + +Zubereitung + + + + + @foreach (var (step, index) in Steps.Select((item, index) => (item, index))) + { + + +
+ +Schritt @(index + 1) + +
+ + + +
+ @step +
+ + + + +
+ +
+ +
+ } + + + +
+ +Schritt @(Steps.Count() + 1) + +
+ + + + +
+ + + +Timer für diesen Schritt + + +
+ + +
+ +
+ +
+ +
diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeStepsList.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeStepsList.razor.cs new file mode 100644 index 00000000..cb0a6823 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Components/UiRecipeStepsList.razor.cs @@ -0,0 +1,10 @@ +using HomeBook.Frontend.Module.Kitchen.ViewModels; +using Microsoft.AspNetCore.Components; + +namespace HomeBook.Frontend.Module.Kitchen.Pages.Recipes.Components; + +public partial class UiRecipeStepsList : ComponentBase +{ + [Parameter] + public IEnumerable Steps { get; set; } = new List(); +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Edit.razor b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Edit.razor new file mode 100644 index 00000000..08bc69f5 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Edit.razor @@ -0,0 +1,123 @@ +@page "/Kitchen/Recipes/{RecipeId:guid}/Edit" + +@using HomeBook.Frontend.Core.Icons +@using HomeBook.Frontend.Module.Kitchen.Contracts +@using HomeBook.Frontend.Module.Kitchen.Pages.Recipes.Components +@using HomeBook.Frontend.Module.Kitchen.ViewModels +@using Microsoft.AspNetCore.Components.Forms + +@inject IRecipeService RecipeService +@inject NavigationManager NavigationManager +@inject ISnackbar Snackbar + +@if (!_isLoading + && _recipe != null) +{ + + + + + + + + + + + + + + @_recipe.Name + + + + + + + + + + + + + + + + + + + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + @* *@ + + + +
+ + + +Abort + + + + + + +Save Recipe + + +
+
+ +
+} + + + + @if (_isLoading) + { +
+ +
+ } + + @if (!_isLoading) + { + } + +
diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Edit.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Edit.razor.cs new file mode 100644 index 00000000..63c94461 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Edit.razor.cs @@ -0,0 +1,83 @@ +using HomeBook.Frontend.Module.Kitchen.Mappings; +using HomeBook.Frontend.Module.Kitchen.Models; +using HomeBook.Frontend.Module.Kitchen.ViewModels; +using Microsoft.AspNetCore.Components; +using MudBlazor; + +namespace HomeBook.Frontend.Module.Kitchen.Pages.Recipes; + +public partial class Edit : ComponentBase +{ + [Parameter] + public Guid RecipeId { get; set; } + + private bool _isLoading = false; + private RecipeDetailViewModel? _recipe = null; + + protected override async Task OnAfterRenderAsync(bool firstRender) + { + await base.OnAfterRenderAsync(firstRender); + + if (!firstRender) + return; + + await LoadRecipeAsync(); + } + + private async Task LoadRecipeAsync() + { + CancellationToken cancellationToken = CancellationToken.None; + _isLoading = true; + StateHasChanged(); + + try + { + RecipeDetailDto? recipeDto = await RecipeService.GetRecipeByIdAsync(RecipeId, + cancellationToken); + if (recipeDto is null) + { + // recipe not found + Snackbar.Add("+Recipe could not be found.", Severity.Error); + NavigationManager.NavigateTo("/Kitchen/Recipes"); + } + + _recipe = recipeDto.ToViewModel(); + } + catch (Exception err) + { + int i = 0; + } + finally + { + _isLoading = false; + StateHasChanged(); + } + } + + private async Task DeleteRecipe() + { + try + { + await RecipeService.DeleteRecipeAsync(RecipeId); + Snackbar.Add("+Recipe deleted successfully.", Severity.Success); + + NavigationManager.NavigateTo("/Kitchen/Recipes"); + } + catch (Exception err) + { + Snackbar.Add("+Recipe could not be deleted. " + err.Message, Severity.Error); + } + } + + private void SaveRecipe() + { + var servings = _recipe.Servings; + + int i = 0; + } + + private void AbortEditingRecipe() + { + NavigationManager.NavigateTo("/Kitchen/Recipes"); + } +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Overview.razor b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Overview.razor index 2717ded7..6e9bf586 100644 --- a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Overview.razor +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Overview.razor @@ -1,5 +1,101 @@ @page "/Kitchen/Recipes" +@using HomeBook.Frontend.Core.Icons +@using HomeBook.Frontend.Module.Kitchen.Contracts +@using HomeBook.Frontend.Module.Kitchen.ViewModels +@using Humanizer + +@inject IRecipeService RecipeService + + + + + + + +Create Test Recipe + + + + + + + @if (_isLoading) + { +
+ +
+ } + + @if (!_isLoading) + { + + + @foreach (RecipeViewModel recipe in _recipes) + { + + + + + + + + + + @recipe.Name + @if (!string.IsNullOrEmpty(recipe.Description)) + { + @recipe.Description + } + + + @if (recipe.HasAnnotations) + { + + + @if (recipe.Duration is not null) + { + + + @recipe.Duration.Value.Humanize() + + } + + @if (recipe.Servings is not null) + { + + + @recipe.Servings +servingins + + } + + + } + + + + + + + } + + + } +
diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Overview.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Overview.razor.cs index c09cdceb..9633319a 100644 --- a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Overview.razor.cs +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/Overview.razor.cs @@ -1,7 +1,79 @@ +using HomeBook.Frontend.Module.Kitchen.Mappings; +using HomeBook.Frontend.Module.Kitchen.Models; +using HomeBook.Frontend.Module.Kitchen.ViewModels; using Microsoft.AspNetCore.Components; namespace HomeBook.Frontend.Module.Kitchen.Pages.Recipes; public partial class Overview : ComponentBase { -} \ No newline at end of file + private List _recipes = []; + private bool _isLoading = false; + + protected override async Task OnAfterRenderAsync(bool firstRender) + { + await base.OnAfterRenderAsync(firstRender); + + if (!firstRender) + return; + + await LoadRecipesAsync(); + } + + private async Task LoadRecipesAsync() + { + CancellationToken cancellationToken = CancellationToken.None; + _isLoading = true; + StateHasChanged(); + + try + { + IEnumerable recipes = await RecipeService.GetRecipesAsync(string.Empty, + cancellationToken); + _recipes.Clear(); + foreach (RecipeDto recipe in recipes) + { + _recipes.Add(recipe.ToViewModel()); + } + } + catch (Exception err) + { + int i = 0; + } + finally + { + _isLoading = false; + StateHasChanged(); + } + } + + private async Task CreateRecipeAsync() + { + string recipeName = "Cheeseburger"; + string description = + "Ein leckerer Cheeseburger mit saftigem Rindfleisch, geschmolzenem Käse, frischem Salat, Tomaten und Zwiebeln, serviert in einem weichen Brötchen."; + int servings = 2; + var steps = new List(); + var ingredients = new List(); + int durationWorkingMinutes = 45; + int durationCookingMinutes = 45; + int durationRestingMinutes = 45; + int caloriesKcal = 3250; + string comments = "3250"; + string source = "3250"; + CancellationToken cancellationToken = CancellationToken.None; + + await RecipeService.CreateRecipeAsync(recipeName, + description, + servings, + steps.ToArray(), + ingredients.ToArray(), + durationWorkingMinutes, + durationCookingMinutes, + durationRestingMinutes, + caloriesKcal, + comments, + source, + cancellationToken); + } +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/View.razor b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/View.razor new file mode 100644 index 00000000..71ff7cc3 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/View.razor @@ -0,0 +1,63 @@ +@page "/Kitchen/Recipes/{RecipeId:guid}/View" + +@using HomeBook.Frontend.Core.Icons +@using HomeBook.Frontend.Module.Kitchen.Contracts + +@inject IRecipeService RecipeService +@inject NavigationManager NavigationManager +@inject ISnackbar Snackbar + +@if (!_isLoading + && _recipe != null) +{ + + + + + + + + + + + + + + @_recipe.Name + + + + + + + + + + + + + +} + + + + @if (_isLoading) + { +
+ +
+ } + + @if (!_isLoading) + { + } + +
diff --git a/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/View.razor.cs b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/View.razor.cs new file mode 100644 index 00000000..04c9cb78 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Pages/Recipes/View.razor.cs @@ -0,0 +1,71 @@ +using HomeBook.Frontend.Module.Kitchen.Mappings; +using HomeBook.Frontend.Module.Kitchen.Models; +using HomeBook.Frontend.Module.Kitchen.ViewModels; +using Microsoft.AspNetCore.Components; +using MudBlazor; + +namespace HomeBook.Frontend.Module.Kitchen.Pages.Recipes; + +public partial class View : ComponentBase +{ + [Parameter] + public Guid RecipeId { get; set; } + + private bool _isLoading = false; + private RecipeDetailViewModel? _recipe = null; + + protected override async Task OnAfterRenderAsync(bool firstRender) + { + await base.OnAfterRenderAsync(firstRender); + + if (!firstRender) + return; + + await LoadRecipeAsync(); + } + + private async Task LoadRecipeAsync() + { + CancellationToken cancellationToken = CancellationToken.None; + _isLoading = true; + StateHasChanged(); + + try + { + RecipeDetailDto? recipeDto = await RecipeService.GetRecipeByIdAsync(RecipeId, + cancellationToken); + if (recipeDto is null) + { + // recipe not found + Snackbar.Add("+Recipe could not be found.", Severity.Error); + NavigationManager.NavigateTo("/Kitchen/Recipes"); + } + + _recipe = recipeDto.ToViewModel(); + } + catch (Exception err) + { + int i = 0; + } + finally + { + _isLoading = false; + StateHasChanged(); + } + } + + private async Task DeleteRecipe() + { + try + { + await RecipeService.DeleteRecipeAsync(RecipeId); + Snackbar.Add("+Recipe deleted successfully.", Severity.Success); + + NavigationManager.NavigateTo("/Kitchen/Recipes"); + } + catch (Exception err) + { + Snackbar.Add("+Recipe could not be deleted. " + err.Message, Severity.Error); + } + } +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Services/MealService.cs b/source/HomeBook.Frontend.Module.Kitchen/Services/MealService.cs deleted file mode 100644 index d6e53d8e..00000000 --- a/source/HomeBook.Frontend.Module.Kitchen/Services/MealService.cs +++ /dev/null @@ -1,39 +0,0 @@ -using HomeBook.Client; -using HomeBook.Client.Models; -using HomeBook.Frontend.Abstractions.Contracts; -using HomeBook.Frontend.Module.Kitchen.Contracts; -using HomeBook.Frontend.Module.Kitchen.Mappings; -using HomeBook.Frontend.Module.Kitchen.Models; - -namespace HomeBook.Frontend.Module.Kitchen.Services; - -/// -public class MealService( - IAuthenticationService authenticationService, - BackendClient backendClient) : IMealService -{ - /// - public async Task> GetMealsAsync(string filter, - CancellationToken cancellationToken = default) - { - cancellationToken.ThrowIfCancellationRequested(); - - string? token = await authenticationService.GetTokenAsync(cancellationToken); - RecipesListResponse? response = await backendClient.Kitchen.Recipes.GetAsync(x => - { - x.Headers.Add("Authorization", $"Bearer {token}"); - }, - cancellationToken); - - cancellationToken.ThrowIfCancellationRequested(); - - if (response is null) - return []; - - List result = (response.Recipes ?? []) - .Select(x => x.ToDto()) - .ToList(); - - return result; - } -} diff --git a/source/HomeBook.Frontend.Module.Kitchen/Services/RecipeService.cs b/source/HomeBook.Frontend.Module.Kitchen/Services/RecipeService.cs new file mode 100644 index 00000000..851d9174 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/Services/RecipeService.cs @@ -0,0 +1,128 @@ +using HomeBook.Client; +using HomeBook.Client.Models; +using HomeBook.Frontend.Abstractions.Contracts; +using HomeBook.Frontend.Module.Kitchen.Contracts; +using HomeBook.Frontend.Module.Kitchen.Mappings; +using HomeBook.Frontend.Module.Kitchen.Models; + +namespace HomeBook.Frontend.Module.Kitchen.Services; + +/// +public class RecipeService( + IAuthenticationService authenticationService, + BackendClient backendClient) : IRecipeService +{ + /// + public async Task> GetRecipesAsync(string? filter, + CancellationToken cancellationToken = default) + { + cancellationToken.ThrowIfCancellationRequested(); + + string? token = await authenticationService.GetTokenAsync(cancellationToken); + RecipesListResponse? response = await backendClient.Modules.Kitchen.Recipes.GetAsync(x => + { + x.Headers.Add("Authorization", $"Bearer {token}"); + + if (!string.IsNullOrWhiteSpace(filter)) + x.QueryParameters.SearchFilter = filter; + }, + cancellationToken); + + cancellationToken.ThrowIfCancellationRequested(); + + if (response is null) + return []; + + List result = (response.Recipes ?? []) + .Select(x => x.ToDto()) + .ToList(); + + return result; + } + + /// + public async Task GetRecipeByIdAsync(Guid id, + CancellationToken cancellationToken = default) + { + string? token = await authenticationService.GetTokenAsync(cancellationToken); + RecipeDetailResponse? response = await backendClient.Modules.Kitchen.Recipes[id] + .GetAsync(x => + { + x.Headers.Add("Authorization", $"Bearer {token}"); + }, + cancellationToken); + + return response?.ToDto(); + } + + /// + public async Task CreateRecipeAsync(string name, + string? description = null, + int? servings = null, + RecipeStepDto[]? steps = null, + RecipeIngredientDto[]? ingredients = null, + int? durationWorkingMinutes = null, + int? durationCookingMinutes = null, + int? durationRestingMinutes = null, + int? caloriesKcal = null, + string? comments = null, + string? source = null, + CancellationToken cancellationToken = default) + { + cancellationToken.ThrowIfCancellationRequested(); + + string? token = await authenticationService.GetTokenAsync(cancellationToken); + CreateRecipeRequest request = new() + { + Name = name, + Description = description, + Servings = servings, + Ingredients = (ingredients ?? []).Select(x => x.ToRequest()).ToList(), + Steps = (steps ?? []).Select(x => x.ToRequest()).ToList(), + DurationWorkingMinutes = durationWorkingMinutes, + DurationCookingMinutes = durationCookingMinutes, + DurationRestingMinutes = durationRestingMinutes, + CaloriesKcal = caloriesKcal, + Comments = comments, + Source = source + }; + + await backendClient.Modules.Kitchen.Recipes.PostAsync(request, + x => + { + x.Headers.Add("Authorization", $"Bearer {token}"); + }, + cancellationToken); + } + + /// + public async Task CreateRecipeAsync(string name, + CancellationToken cancellationToken = default) => + await CreateRecipeAsync(name, + null, + null, + null, + null, + null, + null, + null, + null, + null, + null, + cancellationToken); + + /// + public async Task DeleteRecipeAsync(Guid recipeId, + CancellationToken cancellationToken = default) + { + cancellationToken.ThrowIfCancellationRequested(); + + string? token = await authenticationService.GetTokenAsync(cancellationToken); + await backendClient.Modules.Kitchen.Recipes[recipeId] + .DeleteAsync(x => + { + x.Headers.Add("Authorization", $"Bearer {token}"); + }, + cancellationToken); + } +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/ViewModels/IngredientViewModel.cs b/source/HomeBook.Frontend.Module.Kitchen/ViewModels/IngredientViewModel.cs new file mode 100644 index 00000000..312c6026 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/ViewModels/IngredientViewModel.cs @@ -0,0 +1,19 @@ +using System.Globalization; +using System.Text.RegularExpressions; + +namespace HomeBook.Frontend.Module.Kitchen.ViewModels; + +public class IngredientViewModel +{ + public Guid Id { get; set; } + public decimal? Quantity { get; set; } + public string? Unit { get; set; } + public string Name { get; set; } + public string? AdditionalText { get; set; } + + public string DisplayText => + (Quantity.HasValue ? Quantity.Value.ToString(CultureInfo.InvariantCulture) + " " : "") + + (Unit != null ? Unit + " " : "") + + Name + + (AdditionalText != null ? ", " + AdditionalText : ""); +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/ViewModels/MealItemViewModel.cs b/source/HomeBook.Frontend.Module.Kitchen/ViewModels/MealItemViewModel.cs deleted file mode 100644 index 28e50714..00000000 --- a/source/HomeBook.Frontend.Module.Kitchen/ViewModels/MealItemViewModel.cs +++ /dev/null @@ -1,9 +0,0 @@ -namespace HomeBook.Frontend.Module.Kitchen.ViewModels; - -public class MealItemViewModel -{ - public string Name { get; set; } - public string Ingredients { get; set; } - public TimeSpan? Duration { get; set; } - public int? CaloriesKcal { get; set; } -} diff --git a/source/HomeBook.Frontend.Module.Kitchen/ViewModels/MealPlanItemViewModel.cs b/source/HomeBook.Frontend.Module.Kitchen/ViewModels/MealPlanItemViewModel.cs index 95c95e5b..87a18e85 100644 --- a/source/HomeBook.Frontend.Module.Kitchen/ViewModels/MealPlanItemViewModel.cs +++ b/source/HomeBook.Frontend.Module.Kitchen/ViewModels/MealPlanItemViewModel.cs @@ -5,7 +5,7 @@ public class MealPlanItemViewModel public Guid Id { get; set; } public string ColorName { get; set; } public DateOnly Date { get; set; } - public MealItemViewModel? Breakfast { get; set; } - public MealItemViewModel? Lunch { get; set; } - public MealItemViewModel? Dinner { get; set; } + public RecipeViewModel? Breakfast { get; set; } + public RecipeViewModel? Lunch { get; set; } + public RecipeViewModel? Dinner { get; set; } } diff --git a/source/HomeBook.Frontend.Module.Kitchen/ViewModels/RecipeDetailViewModel.cs b/source/HomeBook.Frontend.Module.Kitchen/ViewModels/RecipeDetailViewModel.cs new file mode 100644 index 00000000..da94a63c --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/ViewModels/RecipeDetailViewModel.cs @@ -0,0 +1,29 @@ +namespace HomeBook.Frontend.Module.Kitchen.ViewModels; + +public class RecipeDetailViewModel +{ + public Guid Id { get; set; } + public string Username { get; set; } + public string Name { get; set; } + public string Description { get; set; } + public int? Servings { get; set; } + public int? CaloriesKcal { get; set; } + public TimeSpan? Duration { get; set; } + public TimeSpan? DurationWorkingMinutes { get; set; } + public TimeSpan? DurationCookingMinutes { get; set; } + public TimeSpan? DurationRestingMinutes { get; set; } + public IEnumerable Ingredients { get; set; } + public IEnumerable Steps { get; set; } + public string Image { get; set; } + public string Source { get; set; } + public string Comment { get; set; } + + public int NumberOfServings { get; set; } + + public bool HasAnnotations => Duration.HasValue || Servings.HasValue; + + public RecipeDetailViewModel() + { + NumberOfServings = Servings ?? 1; + } +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/ViewModels/RecipeViewModel.cs b/source/HomeBook.Frontend.Module.Kitchen/ViewModels/RecipeViewModel.cs new file mode 100644 index 00000000..de3caa14 --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/ViewModels/RecipeViewModel.cs @@ -0,0 +1,16 @@ +namespace HomeBook.Frontend.Module.Kitchen.ViewModels; + +public class RecipeViewModel +{ + public Guid Id { get; set; } + public string Username { get; set; } + public string Name { get; set; } + public string Description { get; set; } + public int? Servings { get; set; } + public int? CaloriesKcal { get; set; } + public TimeSpan? Duration { get; set; } + public string Ingredients { get; set; } + public string Image { get; set; } + + public bool HasAnnotations => Duration.HasValue || Servings.HasValue; +} diff --git a/source/HomeBook.Frontend.Module.Kitchen/ViewModels/StepViewModel.cs b/source/HomeBook.Frontend.Module.Kitchen/ViewModels/StepViewModel.cs new file mode 100644 index 00000000..7affe24d --- /dev/null +++ b/source/HomeBook.Frontend.Module.Kitchen/ViewModels/StepViewModel.cs @@ -0,0 +1,8 @@ +namespace HomeBook.Frontend.Module.Kitchen.ViewModels; + +public class StepViewModel +{ + public Guid Id { get; set; } + public string Description { get; set; } + public int? TimerDurationInSeconds { get; set; } +} diff --git a/source/HomeBook.Frontend.Module.PlatformInfo/HomeBook.Frontend.Module.PlatformInfo.csproj b/source/HomeBook.Frontend.Module.PlatformInfo/HomeBook.Frontend.Module.PlatformInfo.csproj index 629255bf..8ee5a72c 100644 --- a/source/HomeBook.Frontend.Module.PlatformInfo/HomeBook.Frontend.Module.PlatformInfo.csproj +++ b/source/HomeBook.Frontend.Module.PlatformInfo/HomeBook.Frontend.Module.PlatformInfo.csproj @@ -10,7 +10,7 @@ - + diff --git a/source/HomeBook.Frontend.Modules.Abstractions/HomeBook.Frontend.Modules.Abstractions.csproj b/source/HomeBook.Frontend.Modules.Abstractions/HomeBook.Frontend.Modules.Abstractions.csproj index 28a3b0b3..5fc2d1c3 100644 --- a/source/HomeBook.Frontend.Modules.Abstractions/HomeBook.Frontend.Modules.Abstractions.csproj +++ b/source/HomeBook.Frontend.Modules.Abstractions/HomeBook.Frontend.Modules.Abstractions.csproj @@ -10,10 +10,10 @@ - + - + diff --git a/source/HomeBook.Frontend.Services/HomeBook.Frontend.Services.csproj b/source/HomeBook.Frontend.Services/HomeBook.Frontend.Services.csproj index 134ec293..08efab3c 100644 --- a/source/HomeBook.Frontend.Services/HomeBook.Frontend.Services.csproj +++ b/source/HomeBook.Frontend.Services/HomeBook.Frontend.Services.csproj @@ -7,11 +7,11 @@ - + - + diff --git a/source/HomeBook.Frontend.UI/Components/UiNumericGroup.razor b/source/HomeBook.Frontend.UI/Components/UiNumericGroup.razor new file mode 100644 index 00000000..c95ac0e9 --- /dev/null +++ b/source/HomeBook.Frontend.UI/Components/UiNumericGroup.razor @@ -0,0 +1,28 @@ +@using HomeBook.Frontend.Core.Icons + +@typeparam T + +
+ + + + + + + +
diff --git a/source/HomeBook.Frontend.UI/Components/UiNumericGroup.razor.cs b/source/HomeBook.Frontend.UI/Components/UiNumericGroup.razor.cs new file mode 100644 index 00000000..b832b8c0 --- /dev/null +++ b/source/HomeBook.Frontend.UI/Components/UiNumericGroup.razor.cs @@ -0,0 +1,48 @@ +using Microsoft.AspNetCore.Components; +using MudBlazor; + +namespace HomeBook.Frontend.UI.Components; + +public partial class UiNumericGroup : ComponentBase +{ + private MudNumericField _numericInput = null!; + + protected T _value; + + [Parameter] + public T Value + { + get => _value; + set => _value = value; + } + + [Parameter] + public T Min { get; set; } + + [Parameter] + public T Max { get; set; } + + [Parameter] + public T Step { get; set; } + + [Parameter] + public EventCallback ValueChanged { get; set; } + + private async Task OnValueChanged(T? value) + { + await ValueChanged.InvokeAsync(value); + StateHasChanged(); + } + + public async Task Decrement() + { + await _numericInput.Decrement(); + StateHasChanged(); + } + + public async Task Increment() + { + await _numericInput.Increment(); + StateHasChanged(); + } +} diff --git a/source/HomeBook.Frontend.UI/HomeBook.Frontend.UI.csproj b/source/HomeBook.Frontend.UI/HomeBook.Frontend.UI.csproj index 15caf5f9..723ad710 100644 --- a/source/HomeBook.Frontend.UI/HomeBook.Frontend.UI.csproj +++ b/source/HomeBook.Frontend.UI/HomeBook.Frontend.UI.csproj @@ -11,7 +11,7 @@ - + diff --git a/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.Designer.cs b/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.Designer.cs index 1b8c3b43..bb4f69fc 100644 --- a/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.Designer.cs +++ b/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.Designer.cs @@ -1442,5 +1442,11 @@ public static string Settings_NavMenu_DeveloperColors_Text { return ResourceManager.GetString("Settings_NavMenu_DeveloperColors_Text", resourceCulture); } } + + public static string MainLayout_SearchTextField_Placeholder { + get { + return ResourceManager.GetString("MainLayout_SearchTextField_Placeholder", resourceCulture); + } + } } } diff --git a/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.de-DE.resx b/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.de-DE.resx index 27b8cec2..16b1b307 100644 --- a/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.de-DE.resx +++ b/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.de-DE.resx @@ -111,4 +111,7 @@ Dev Colors + + Suche nach... + diff --git a/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.resx b/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.resx index 7d563752..5cf4dd8f 100644 --- a/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.resx +++ b/source/HomeBook.Frontend.UI/Resources/LocalizationStrings.resx @@ -722,4 +722,7 @@ Dev Colors + + Search for... + diff --git a/source/HomeBook.Frontend/HomeBook.Frontend.csproj b/source/HomeBook.Frontend/HomeBook.Frontend.csproj index 90a6d81d..cb2274a4 100644 --- a/source/HomeBook.Frontend/HomeBook.Frontend.csproj +++ b/source/HomeBook.Frontend/HomeBook.Frontend.csproj @@ -9,13 +9,13 @@ - - - - + + + + - + diff --git a/source/HomeBook.Frontend/Layout/MainLayout.razor b/source/HomeBook.Frontend/Layout/MainLayout.razor index 8256abbe..9f058f66 100644 --- a/source/HomeBook.Frontend/Layout/MainLayout.razor +++ b/source/HomeBook.Frontend/Layout/MainLayout.razor @@ -49,11 +49,16 @@ + + - + - + @Loc[nameof(LocalizationStrings.AppTitle)] - @Loc[nameof(LocalizationStrings.AppSlogan)] @@ -58,7 +60,8 @@ - + + Color="Color.Primary"> @Loc[nameof(LocalizationStrings.Settings_About_ApplicationArea_Title)] @@ -83,7 +86,7 @@ + Color="Color.Primary"/> @Loc[nameof(LocalizationStrings.Settings_About_ApplicationArea_Backend_Text)] @_backendDotnetVersion @@ -93,7 +96,7 @@ + Color="Color.Primary"/> @Loc[nameof(LocalizationStrings.Settings_About_ApplicationArea_Database_Text)] @_databaseProvider @@ -111,7 +114,7 @@ + Color="Color.Primary"/> @Loc[nameof(LocalizationStrings.Settings_About_ApplicationArea_UI_Text)] @_uiDotnetVersion @@ -121,7 +124,7 @@ + Color="Color.Primary"/> @Loc[nameof(LocalizationStrings.Settings_About_ApplicationArea_Deployment_Text)] @_deploymentType @@ -139,7 +142,7 @@ - + Color="Color.Primary"> @Loc[nameof(LocalizationStrings.Settings_About_LicensesArea_Title)] @@ -169,8 +172,8 @@ - + @((MarkupString)(string.Format(Loc[nameof(LocalizationStrings.Settings_About_LicensesArea_Copyright_TextTemplate)], DateTime.Now.Year))) diff --git a/source/HomeBook.Frontend/Pages/Settings/Developer/Icons.razor b/source/HomeBook.Frontend/Pages/Settings/Developer/Icons.razor index bcff2d6c..b1bf8773 100644 --- a/source/HomeBook.Frontend/Pages/Settings/Developer/Icons.razor +++ b/source/HomeBook.Frontend/Pages/Settings/Developer/Icons.razor @@ -25,6 +25,9 @@ + + diff --git a/source/HomeBook.Frontend/Pages/Settings/SettingsNavMenu.razor b/source/HomeBook.Frontend/Pages/Settings/SettingsNavMenu.razor index e3397483..4a8c4aea 100644 --- a/source/HomeBook.Frontend/Pages/Settings/SettingsNavMenu.razor +++ b/source/HomeBook.Frontend/Pages/Settings/SettingsNavMenu.razor @@ -6,7 +6,7 @@ @inject IStringLocalizer Loc @inject IConfiguration Configuration - + + string? response = await backendClient.Account.Logout.PostAsync( + x => { }, cancellationToken); @@ -134,7 +135,7 @@ public async Task IsAuthenticatedAsync(CancellationToken cancellationToken if (string.IsNullOrEmpty(expiresAtString) || !DateTime.TryParse(expiresAtString, out DateTime expiresAt)) - return true; + return false; if (DateTime.UtcNow < expiresAt) return true; @@ -173,7 +174,7 @@ public async Task IsAuthenticatedAsync(CancellationToken cancellationToken // Parse JWT token to extract claims string[] tokenParts = token.Split('.'); if (tokenParts.Length != 3) - return null; + throw new FormatException("Invalid JWT token format"); // Decode payload string payload = tokenParts[1]; @@ -219,13 +220,12 @@ public async Task IsAuthenticatedAsync(CancellationToken cancellationToken /// public async Task IsCurrentUserAdminAsync(CancellationToken cancellationToken = default) { - ClaimsPrincipal user = await GetCurrentUserAsync(cancellationToken); - Claim? isAdminClaim = user.FindFirst("IsAdmin"); + ClaimsPrincipal? user = await GetCurrentUserAsync(cancellationToken); + Claim? isAdminClaim = user?.FindFirst("IsAdmin"); - if (isAdminClaim != null && bool.TryParse(isAdminClaim.Value, out bool isAdmin)) - { + if (isAdminClaim is not null + && bool.TryParse(isAdminClaim.Value, out bool isAdmin)) return isAdmin; - } return false; } diff --git a/source/HomeBook.Frontend/Setup/SetupSteps/BackendConnectionSetupStep.razor.cs b/source/HomeBook.Frontend/Setup/SetupSteps/BackendConnectionSetupStep.razor.cs index 84983152..a8e7b257 100644 --- a/source/HomeBook.Frontend/Setup/SetupSteps/BackendConnectionSetupStep.razor.cs +++ b/source/HomeBook.Frontend/Setup/SetupSteps/BackendConnectionSetupStep.razor.cs @@ -85,11 +85,6 @@ private async Task ConnectToServerAsync(CancellationToken cancellationToken) throw new SetupCheckException( Loc[nameof(LocalizationStrings.Setup_BackendConnection_Check_VersionError_Message)]); - string appVersion = Configuration.GetSection("Version").Value ?? ""; - if (appVersion != versionResponse) - throw new SetupCheckException( - Loc[nameof(LocalizationStrings.Setup_BackendConnection_Check_VersionMatchError_Message)]); - try { await BackendClient.Setup.Availability.GetAsync(x => diff --git a/source/HomeBook.Frontend/Styles/_components.scss b/source/HomeBook.Frontend/Styles/_components.scss index f3ddfe6e..c1e2c4cb 100644 --- a/source/HomeBook.Frontend/Styles/_components.scss +++ b/source/HomeBook.Frontend/Styles/_components.scss @@ -3,6 +3,7 @@ @import "components/mud-drawer"; @import "components/mud-card"; @import "components/mud-stepper"; +@import "components/mud-file-upload"; @import "components/ui-wave-background"; @import "components/ui-stripe-background"; diff --git a/source/HomeBook.Frontend/Styles/components/_mud-file-upload.scss b/source/HomeBook.Frontend/Styles/components/_mud-file-upload.scss new file mode 100644 index 00000000..1e8d7abd --- /dev/null +++ b/source/HomeBook.Frontend/Styles/components/_mud-file-upload.scss @@ -0,0 +1,11 @@ +.mud-file-upload { + + .html-fileupload-hidden { + position: absolute; + width: 100%; + height: 100%; + overflow: hidden; + z-index: 10; + opacity: 0; + } +} diff --git a/source/HomeBook.Frontend/wwwroot/appsettings.json b/source/HomeBook.Frontend/wwwroot/appsettings.json index adc605b5..696804f1 100644 --- a/source/HomeBook.Frontend/wwwroot/appsettings.json +++ b/source/HomeBook.Frontend/wwwroot/appsettings.json @@ -1,5 +1,5 @@ { - "Version": "1.0.110", + "Version": "1.0.0", "Backend": { "Host": "/api" }, diff --git a/source/HomeBook.Frontend/wwwroot/css/app.css b/source/HomeBook.Frontend/wwwroot/css/app.css index d2653dd4..176bc9dc 100644 --- a/source/HomeBook.Frontend/wwwroot/css/app.css +++ b/source/HomeBook.Frontend/wwwroot/css/app.css @@ -1647,6 +1647,15 @@ display: none; } +.mud-file-upload .html-fileupload-hidden { + position: absolute; + width: 100%; + height: 100%; + overflow: hidden; + z-index: 10; + opacity: 0; +} + .ui-wavebackground-container { width: 100vw; height: 100vh; diff --git a/source/HomeBook.Frontend/wwwroot/css/app.css.map b/source/HomeBook.Frontend/wwwroot/css/app.css.map index 21c6cdf8..01142e40 100644 --- a/source/HomeBook.Frontend/wwwroot/css/app.css.map +++ b/source/HomeBook.Frontend/wwwroot/css/app.css.map @@ -1 +1 @@ -{"version":3,"sourceRoot":"","sources":["../../Styles/_blazorbase.scss","../../Styles/variables/_breakpoints.scss","../../Styles/variables/_frosted-ui.scss","../../Styles/variables/_border-radius.scss","../../Styles/variables/_ui-colored-icon.scss","../../Styles/variables/_ui-startmenu-item.scss","../../Styles/utilities/_width.scss","../../Styles/utilities/_float.scss","../../Styles/styles/_frosted.scss","../../Styles/styles/_colors.scss","../../Styles/styles/_color-palette.scss","../../Styles/styles/_settings-colors.scss","../../Styles/styles/_typography.scss","../../Styles/components/_mud-nav-menu.scss","../../Styles/components/_mud-paper.scss","../../Styles/components/_mud-drawer.scss","../../Styles/components/_mud-card.scss","../../Styles/components/_mud-stepper.scss","../../Styles/components/_ui-wave-background.scss","../../Styles/components/_ui-stripe-background.scss","../../Styles/components/_ui-countdown-alert.scss","../../Styles/components/_ui-dev-icons.scss","../../Styles/components/_ui-widgets.scss","../../Styles/components/_ui-icon.scss","../../Styles/components/_ui-settings-item.scss","../../Styles/components/_ui-startmenu.scss","../../Styles/components/_ui-progress-item.scss","../../Styles/components/_ui-colored-icon.scss","../../Styles/components/_ui-detail-card.scss","../../Styles/components/_ui-value-card.scss","../../Styles/components/_ui-layout.scss","../../Styles/components/_ui-wallpaper.scss","../../Styles/views/_setup.scss","../../Styles/views/_login.scss"],"names":[],"mappings":"AAAA;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;;;AAGJ;EACI;;;AAGJ;EACI;EACA;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;;;AAGJ;EACI;;;AC3DJ;AAOA;AAIA;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;;ACXJ;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;;ACjBJ;EACI;;;ACCJ;EACI;EACA;EACA;;;ACHJ;EACI;EACA;EACA;;;ACHJ;EAAO;;;AACP;EAAQ;;;AACR;EAAQ;;;AACR;EAAQ;;;AACR;EAAS;;;AACT;EAAU;;;AAeV;EATE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AAQ5B;EAbE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AAY5B;EAjBE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AAgB5B;EArBE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AAoB5B;EAzBE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AAwB5B;EA7BE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AA4B5B;EAjCE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;ACpB5B;EACI;;;AAGJ;EACI;;;AAGJ;EACI;;;AC0DA;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;ACxDR;EACI;EACA;EACA;;;AAGJ;EACI;EACA;;;AAGJ;EACI;;;AAGJ;EACI;EACA;;;AAGJ;EACI;;;AAGJ;EACI;EACA;;;AAGJ;EACI;;;ACkBJ;EAEQ;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;;;AAeJ;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AChEpB;EACI;EACA;EACA;EACA;;;ACTJ;EACI;;;ACKQ;EACI;;AAOR;EACI;EACA;EACA;;AAGJ;EACI;EACA;;AAGJ;EACI;EACA;;;ACrBA;EACI;EACA;;AAGJ;EACI;EACA;;AAWA;EACI;EACA;;AAGJ;EACI;EACA;;;AC1BZ;EACI;EACA;;AAEA;EACI;;AAIR;EACI;EACA;;AAEA;EACI;;AAIR;EACI;EACA;;AAEA;EACI;;;AC1BZ;EACI;EACA;EACA;EACA;;AAEA;EACI;EACA;;AAGJ;EACI;;AAIR;EACI;EACA;EACA;;AAGJ;EACI;EACA;EACA;;AAGJ;EACI;;;AC3BA;EACI;;;AC0BZ;EACI;EACA;;AAEA;EACI,YAlBc;EAmBd;EACA;EACA;;AAGJ;EACI;EACA;;AAGJ;EACI;EACA;;AAGJ;EACI;EACA;;AAGJ;EACI;EACA;EACA;EACA,OAnCK;EAoCL,QArCM;EAsCN;EACA;EACA;EACA;EACA;EACA;;AAGJ;EACI;EACA;EACA;;AAGJ;EACI;EACA;EACA;;;AAIR;EACI;IACI;;EAEJ;IACI;;EAEJ;IACI;;;AAIR;EACI;IACI;;EAGJ;IACI;;EAGJ;IACI;;EAGJ;IACI;;EAGJ;IACI;;;AClHR;EACI;EACA;;;AAGJ;EACI;EACA;;AAEA;AACI;AACA;AACA;EACA;EACA;EACA;EACA;EACA;;AAGJ;AACI;AACA;AACA;EACA;EACA;EACA;EACA;EACA;;AAGJ;AACI;AACA;AACA;EACA;EACA;EACA;EACA;EACA;;AAlCR;AAoCI;AAOA;AAOA;AACA;AAOA;AACA;AAOA;AACA;;;ACxEJ;EACI;EACA;EACA;;AAEA;EACI;EACA;EACA;EACA;;;ACPJ;EACI;;;ACHR;EACI;EACA;EACA;EACA;EACA;EAEA;;;AAGJ;EACI;EACA;;;AAGJ;EACI;EACA;EAGA;EACA;EAGA;EAGA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;;AAEA;EACI;;AAGJ;EACI;;AAGJ;EACI;;AAGJ;EACI;;AAGJ;EACI;;AAGJ;EACI;;AAGJ;EACI;;;AAIR;EAEI;;;AC7DA;EACI;;AAEA;EACI;EACA;;AAGJ;EACI;EACA;EACA;;AAEA;EACI;;ACbZ;EACI;;AAEA;EAHJ;IAIQ;;;;ACVZ;EACI;EACA;;AAEA;EACI;;;AAIR;EACI;EACA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;EACA;EACA;EACA;;;AAGJ;EACI;;AAUA;EACI;;;AC7CR;EACI;EACA;EACA;;AAEA;AAAA;AAAA;EAGI;EACA;;AAGJ;AAAA;EAEI;;AAEA;AAAA;AAAA;AAAA;EAEI;EACA;;AAGJ;AAAA;AAAA;AAAA;EAEI;;AAIR;EACI;;AAEA;EACI;;;AC9BR;EACI;EACA;;AAGJ;EACI;EACA;EACA;;;ACVR;EACI;;;AAGJ;EACI;;AAEA;EACI;;AAGJ;EACI;;AAGJ;EACI;;;AChBR;EACI;;AAEA;EACI;;AAGJ;EACI;;;AAIR;EACI;;AAEA;EACI;;AAGJ;EACI;;;ACpBR;EACI;EACA;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;;;AAKA;EAFJ;IAGQ;IACA;;;;ACzBR;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;;ACRJ;EACI;EACA;EACA;EACA;;AAEA;EACI;;;AAIR;EACI;EACA;;;AAOJ;EACI;;;AAGJ;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAEA;EACI;;;AAIR;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAEA;EACI;;AAGJ;EACI;EACA;EACA;EACA;EACA;EAEA;EACA;EACA;EACA;;AAEA;EACI;EACA;EACA;;AA4BA;EACI;;AAGJ;EACI;;AAGJ;EACI;;AAGJ;EACI;;;AC9GhB;EACI;EACA;EACA;EACA;;;AAGJ;EACI;EACA;;;AAGJ;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAEA;EACI","file":"app.css"} \ No newline at end of file +{"version":3,"sourceRoot":"","sources":["../../Styles/_blazorbase.scss","../../Styles/variables/_breakpoints.scss","../../Styles/variables/_frosted-ui.scss","../../Styles/variables/_border-radius.scss","../../Styles/variables/_ui-colored-icon.scss","../../Styles/variables/_ui-startmenu-item.scss","../../Styles/utilities/_width.scss","../../Styles/utilities/_float.scss","../../Styles/styles/_frosted.scss","../../Styles/styles/_colors.scss","../../Styles/styles/_color-palette.scss","../../Styles/styles/_settings-colors.scss","../../Styles/styles/_typography.scss","../../Styles/components/_mud-nav-menu.scss","../../Styles/components/_mud-paper.scss","../../Styles/components/_mud-drawer.scss","../../Styles/components/_mud-card.scss","../../Styles/components/_mud-stepper.scss","../../Styles/components/_mud-file-upload.scss","../../Styles/components/_ui-wave-background.scss","../../Styles/components/_ui-stripe-background.scss","../../Styles/components/_ui-countdown-alert.scss","../../Styles/components/_ui-dev-icons.scss","../../Styles/components/_ui-widgets.scss","../../Styles/components/_ui-icon.scss","../../Styles/components/_ui-settings-item.scss","../../Styles/components/_ui-startmenu.scss","../../Styles/components/_ui-progress-item.scss","../../Styles/components/_ui-colored-icon.scss","../../Styles/components/_ui-detail-card.scss","../../Styles/components/_ui-value-card.scss","../../Styles/components/_ui-layout.scss","../../Styles/components/_ui-wallpaper.scss","../../Styles/views/_setup.scss","../../Styles/views/_login.scss"],"names":[],"mappings":"AAAA;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;;;AAGJ;EACI;;;AAGJ;EACI;EACA;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;;;AAGJ;EACI;;;AC3DJ;AAOA;AAIA;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;;ACXJ;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;;ACjBJ;EACI;;;ACCJ;EACI;EACA;EACA;;;ACHJ;EACI;EACA;EACA;;;ACHJ;EAAO;;;AACP;EAAQ;;;AACR;EAAQ;;;AACR;EAAQ;;;AACR;EAAS;;;AACT;EAAU;;;AAeV;EATE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AAQ5B;EAbE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AAY5B;EAjBE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AAgB5B;EArBE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AAoB5B;EAzBE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AAwB5B;EA7BE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;AA4B5B;EAjCE;IAAuB;;EACvB;IAAwB;;EACxB;IAAwB;;EACxB;IAAwB;;EACxB;IAAyB;;EACzB;IAA0B;;;ACpB5B;EACI;;;AAGJ;EACI;;;AAGJ;EACI;;;AC0DA;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;AAOJ;EA7DA;EAEA,YACI;EAaJ;EACA;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;AAuCA;EACI;;;AAIR;EArCA,YACI;EAaJ;;AAEA;EACI;EACA;EACA;EACA;EACA;EACA;;;ACxDR;EACI;EACA;EACA;;;AAGJ;EACI;EACA;;;AAGJ;EACI;;;AAGJ;EACI;EACA;;;AAGJ;EACI;;;AAGJ;EACI;EACA;;;AAGJ;EACI;;;ACkBJ;EAEQ;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;EANA;EACA;EAIA;EACA;AAAA;AAAA;AAAA;;;AAeJ;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OATgB;;;AAYpB;EACI;EAKA,OAlBgB;;;AAGpB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AAMhB;EACI;;;AAGJ;EACI;EACA,OAZY;;;AAehB;EACI;EAKA,OArBY;;;AChEpB;EACI;EACA;EACA;EACA;;;ACTJ;EACI;;;ACKQ;EACI;;AAOR;EACI;EACA;EACA;;AAGJ;EACI;EACA;;AAGJ;EACI;EACA;;;ACrBA;EACI;EACA;;AAGJ;EACI;EACA;;AAWA;EACI;EACA;;AAGJ;EACI;EACA;;;AC1BZ;EACI;EACA;;AAEA;EACI;;AAIR;EACI;EACA;;AAEA;EACI;;AAIR;EACI;EACA;;AAEA;EACI;;;AC1BZ;EACI;EACA;EACA;EACA;;AAEA;EACI;EACA;;AAGJ;EACI;;AAIR;EACI;EACA;EACA;;AAGJ;EACI;EACA;EACA;;AAGJ;EACI;;;AC3BA;EACI;;;ACHR;EACI;EACA;EACA;EACA;EACA;EACA;;;ACuBR;EACI;EACA;;AAEA;EACI,YAlBc;EAmBd;EACA;EACA;;AAGJ;EACI;EACA;;AAGJ;EACI;EACA;;AAGJ;EACI;EACA;;AAGJ;EACI;EACA;EACA;EACA,OAnCK;EAoCL,QArCM;EAsCN;EACA;EACA;EACA;EACA;EACA;;AAGJ;EACI;EACA;EACA;;AAGJ;EACI;EACA;EACA;;;AAIR;EACI;IACI;;EAEJ;IACI;;EAEJ;IACI;;;AAIR;EACI;IACI;;EAGJ;IACI;;EAGJ;IACI;;EAGJ;IACI;;EAGJ;IACI;;;AClHR;EACI;EACA;;;AAGJ;EACI;EACA;;AAEA;AACI;AACA;AACA;EACA;EACA;EACA;EACA;EACA;;AAGJ;AACI;AACA;AACA;EACA;EACA;EACA;EACA;EACA;;AAGJ;AACI;AACA;AACA;EACA;EACA;EACA;EACA;EACA;;AAlCR;AAoCI;AAOA;AAOA;AACA;AAOA;AACA;AAOA;AACA;;;ACxEJ;EACI;EACA;EACA;;AAEA;EACI;EACA;EACA;EACA;;;ACPJ;EACI;;;ACHR;EACI;EACA;EACA;EACA;EACA;EAEA;;;AAGJ;EACI;EACA;;;AAGJ;EACI;EACA;EAGA;EACA;EAGA;EAGA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;;AAEA;EACI;;AAGJ;EACI;;AAGJ;EACI;;AAGJ;EACI;;AAGJ;EACI;;AAGJ;EACI;;AAGJ;EACI;;;AAIR;EAEI;;;AC7DA;EACI;;AAEA;EACI;EACA;;AAGJ;EACI;EACA;EACA;;AAEA;EACI;;ACbZ;EACI;;AAEA;EAHJ;IAIQ;;;;ACVZ;EACI;EACA;;AAEA;EACI;;;AAIR;EACI;EACA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;EACA;EACA;EACA;;;AAGJ;EACI;;AAUA;EACI;;;AC7CR;EACI;EACA;EACA;;AAEA;AAAA;AAAA;EAGI;EACA;;AAGJ;AAAA;EAEI;;AAEA;AAAA;AAAA;AAAA;EAEI;EACA;;AAGJ;AAAA;AAAA;AAAA;EAEI;;AAIR;EACI;;AAEA;EACI;;;AC9BR;EACI;EACA;;AAGJ;EACI;EACA;EACA;;;ACVR;EACI;;;AAGJ;EACI;;AAEA;EACI;;AAGJ;EACI;;AAGJ;EACI;;;AChBR;EACI;;AAEA;EACI;;AAGJ;EACI;;;AAIR;EACI;;AAEA;EACI;;AAGJ;EACI;;;ACpBR;EACI;EACA;EACA;EACA;EACA;;;AAGJ;EACI;EACA;EACA;;;AAGJ;EACI;EACA;EACA;EACA;;;AAKA;EAFJ;IAGQ;IACA;;;;ACzBR;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;;ACRJ;EACI;EACA;EACA;EACA;;AAEA;EACI;;;AAIR;EACI;EACA;;;AAOJ;EACI;;;AAGJ;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAEA;EACI;;;AAIR;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAEA;EACI;;AAGJ;EACI;EACA;EACA;EACA;EACA;EAEA;EACA;EACA;EACA;;AAEA;EACI;EACA;EACA;;AA4BA;EACI;;AAGJ;EACI;;AAGJ;EACI;;AAGJ;EACI;;;AC9GhB;EACI;EACA;EACA;EACA;;;AAGJ;EACI;EACA;;;AAGJ;EACI;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAEA;EACI","file":"app.css"} \ No newline at end of file diff --git a/source/HomeBook.Frontend/wwwroot/css/app.min.css b/source/HomeBook.Frontend/wwwroot/css/app.min.css index e1330056..591c22af 100644 --- a/source/HomeBook.Frontend/wwwroot/css/app.min.css +++ b/source/HomeBook.Frontend/wwwroot/css/app.min.css @@ -1 +1 @@ -#blazor-error-ui{color-scheme:light only;background:#ffffe0;bottom:0;box-shadow:0 -1px 2px rgba(0,0,0,.2);box-sizing:border-box;display:none;left:0;padding:.6rem 1.25rem .7rem 1.25rem;position:fixed;width:100%;z-index:1000}#blazor-error-ui .dismiss{cursor:pointer;position:absolute;right:.75rem;top:.5rem}.blazor-error-boundary{background:url() no-repeat 1rem/1.8rem,#b32121;padding:1rem 1rem 1rem 3.7rem;color:#fff}.blazor-error-boundary::after{content:"An error has occurred."}.loading-progress{position:relative;display:block;width:8rem;height:8rem;margin:20vh auto 1rem auto}.loading-progress circle{fill:none;stroke:#e0e0e0;stroke-width:.6rem;transform-origin:50% 50%;transform:rotate(-90deg)}.loading-progress circle:last-child{stroke:#1b6ec2;stroke-dasharray:calc(3.141*var(--blazor-load-percentage, 0%)*.8),500%;transition:stroke-dasharray .05s ease-in-out}.loading-progress-text{position:absolute;text-align:center;font-weight:bold;inset:calc(20vh + 3.25rem) 0 auto .2rem}.loading-progress-text:after{content:var(--blazor-load-percentage-text, "Loading")}:root{--hb-breakpoint-xs: 0px;--hb-breakpoint-sm: 600px;--hb-breakpoint-md: 960px;--hb-breakpoint-lg: 1280px;--hb-breakpoint-xl: 1920px;--hb-breakpoint-xxl: 2560px;--hb-breakpoint-xxxl: 3840px;--hb-breakpoint-xxxxl: 5120px}:root{--hb-frosted-color: rgb(255, 255, 255);--hb-frosted-alpha-min: 0.05;--hb-frosted-alpha-step: 0.05;--hb-frosted-blur: 6px;--hb-frosted-steps: 10;--hb-frosted-border-alpha: 0.05;--hb-frosted-gradient-deg: 160deg;--hb-frosted-gradient-add: 0.25;--hb-frosted-inner-shadow-alpha: 0.3}:root{--hb-ui-icon-border-radius: 6px}:root{--hb-ui-colored-icon-background-opacity: 75%;--hb-ui-colored-icon-borderradius: 8px;--hb-ui-colored-icon-size: 1.5em}:root{--hb-startmenu-background-icon-size: 140px;--hb-startmenu-background-icon-opacity: 0.075;--hb-startmenu-container-background: transparent}.w-0{width:0% !important}.w-25{width:25% !important}.w-50{width:50% !important}.w-75{width:75% !important}.w-100{width:100% !important}.w-auto{width:auto !important}@media(min-width: 600px){.w-sm-0{width:0% !important}.w-sm-25{width:25% !important}.w-sm-50{width:50% !important}.w-sm-75{width:75% !important}.w-sm-100{width:100% !important}.w-sm-auto{width:auto !important}}@media(min-width: 960px){.w-md-0{width:0% !important}.w-md-25{width:25% !important}.w-md-50{width:50% !important}.w-md-75{width:75% !important}.w-md-100{width:100% !important}.w-md-auto{width:auto !important}}@media(min-width: 1280px){.w-lg-0{width:0% !important}.w-lg-25{width:25% !important}.w-lg-50{width:50% !important}.w-lg-75{width:75% !important}.w-lg-100{width:100% !important}.w-lg-auto{width:auto !important}}@media(min-width: 1920px){.w-xl-0{width:0% !important}.w-xl-25{width:25% !important}.w-xl-50{width:50% !important}.w-xl-75{width:75% !important}.w-xl-100{width:100% !important}.w-xl-auto{width:auto !important}}@media(min-width: 2560px){.w-xxl-0{width:0% !important}.w-xxl-25{width:25% !important}.w-xxl-50{width:50% !important}.w-xxl-75{width:75% !important}.w-xxl-100{width:100% !important}.w-xxl-auto{width:auto !important}}@media(min-width: 3840px){.w-xxxl-0{width:0% !important}.w-xxxl-25{width:25% !important}.w-xxxl-50{width:50% !important}.w-xxxl-75{width:75% !important}.w-xxxl-100{width:100% !important}.w-xxxl-auto{width:auto !important}}@media(min-width: 5120px){.w-xxxxl-0{width:0% !important}.w-xxxxl-25{width:25% !important}.w-xxxxl-50{width:50% !important}.w-xxxxl-75{width:75% !important}.w-xxxxl-100{width:100% !important}.w-xxxxl-auto{width:auto !important}}.float-right{float:right !important}.float-left{float:left !important}.float-none{float:none !important}.frosted-b1{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.05) 0%, rgba(255, 255, 255, 0.3) 100%),hsla(0,0%,100%,.05) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.1)}.frosted-b1::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.35);border-radius:inherit}.frosted-b1.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b1{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.05) 0%, rgba(255, 255, 255, 0.3) 100%),hsla(0,0%,100%,.05) !important;backdrop-filter:blur(6px)}.frosted-bg-b1::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.35);border-radius:inherit}.frosted-b2{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.1) 0%, rgba(255, 255, 255, 0.35) 100%),hsla(0,0%,100%,.1) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.15)}.frosted-b2::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.4);border-radius:inherit}.frosted-b2.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b2{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.1) 0%, rgba(255, 255, 255, 0.35) 100%),hsla(0,0%,100%,.1) !important;backdrop-filter:blur(6px)}.frosted-bg-b2::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.4);border-radius:inherit}.frosted-b3{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.15) 0%, rgba(255, 255, 255, 0.4) 100%),hsla(0,0%,100%,.15) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.2)}.frosted-b3::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.45);border-radius:inherit}.frosted-b3.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b3{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.15) 0%, rgba(255, 255, 255, 0.4) 100%),hsla(0,0%,100%,.15) !important;backdrop-filter:blur(6px)}.frosted-bg-b3::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.45);border-radius:inherit}.frosted-b4{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.2) 0%, rgba(255, 255, 255, 0.45) 100%),hsla(0,0%,100%,.2) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.25)}.frosted-b4::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.5);border-radius:inherit}.frosted-b4.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b4{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.2) 0%, rgba(255, 255, 255, 0.45) 100%),hsla(0,0%,100%,.2) !important;backdrop-filter:blur(6px)}.frosted-bg-b4::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.5);border-radius:inherit}.frosted-b5{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.25) 0%, rgba(255, 255, 255, 0.5) 100%),hsla(0,0%,100%,.25) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.3)}.frosted-b5::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.55);border-radius:inherit}.frosted-b5.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b5,.mud-drawer.frosted{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.25) 0%, rgba(255, 255, 255, 0.5) 100%),hsla(0,0%,100%,.25) !important;backdrop-filter:blur(6px)}.frosted-bg-b5::before,.mud-drawer.frosted::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.55);border-radius:inherit}.frosted-b6{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.3) 0%, rgba(255, 255, 255, 0.55) 100%),hsla(0,0%,100%,.3) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.35)}.frosted-b6::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.6);border-radius:inherit}.frosted-b6.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b6{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.3) 0%, rgba(255, 255, 255, 0.55) 100%),hsla(0,0%,100%,.3) !important;backdrop-filter:blur(6px)}.frosted-bg-b6::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.6);border-radius:inherit}.frosted-b7{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.35) 0%, rgba(255, 255, 255, 0.6) 100%),hsla(0,0%,100%,.35) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.4)}.frosted-b7::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.65);border-radius:inherit}.frosted-b7.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b7{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.35) 0%, rgba(255, 255, 255, 0.6) 100%),hsla(0,0%,100%,.35) !important;backdrop-filter:blur(6px)}.frosted-bg-b7::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.65);border-radius:inherit}.frosted-b8{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.4) 0%, rgba(255, 255, 255, 0.65) 100%),hsla(0,0%,100%,.4) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.45)}.frosted-b8::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.7);border-radius:inherit}.frosted-b8.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b8{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.4) 0%, rgba(255, 255, 255, 0.65) 100%),hsla(0,0%,100%,.4) !important;backdrop-filter:blur(6px)}.frosted-bg-b8::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.7);border-radius:inherit}.frosted-b9{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.45) 0%, rgba(255, 255, 255, 0.7) 100%),hsla(0,0%,100%,.45) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.5)}.frosted-b9::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.75);border-radius:inherit}.frosted-b9.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b9{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.45) 0%, rgba(255, 255, 255, 0.7) 100%),hsla(0,0%,100%,.45) !important;backdrop-filter:blur(6px)}.frosted-bg-b9::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.75);border-radius:inherit}.frosted-b10{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.5) 0%, rgba(255, 255, 255, 0.75) 100%),hsla(0,0%,100%,.5) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.55)}.frosted-b10::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.8);border-radius:inherit}.frosted-b10.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b10{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.5) 0%, rgba(255, 255, 255, 0.75) 100%),hsla(0,0%,100%,.5) !important;backdrop-filter:blur(6px)}.frosted-bg-b10::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.8);border-radius:inherit}:root{--hb-github-color: #24292e;--hb-docker-color: #2496ed;--hb-ubuntu-color: #e95420}.github-bg{background-color:#24292e !important;color:#fff !important}.github-text{color:#24292e !important}.docker-bg{background-color:#2496ed !important;color:#fff !important}.docker-text{color:#2496ed !important}.ubuntu-bg{background-color:#e95420 !important;color:#fff !important}.ubuntu-text{color:#e95420 !important}:root{--hb-color-brand-github: rgb(36, 41, 46);--hb-color-brand-github-rgb: 36, 41, 46;--hb-color-brand-github-dark: rgb(28.8, 32.8, 36.8);--hb-color-brand-github-dark-rgb: 28.8, 32.8, 36.8;--hb-color-brand-docker: rgb(36, 150, 237);--hb-color-brand-docker-rgb: 36, 150, 237;--hb-color-brand-docker-dark: rgb(16.5873417722, 121.6405063291, 201.8126582278);--hb-color-brand-docker-dark-rgb: 16.5873417722, 121.6405063291, 201.8126582278;--hb-color-brand-ubuntu: rgb(233, 84, 32);--hb-color-brand-ubuntu-rgb: 233, 84, 32;--hb-color-brand-ubuntu-dark: rgb(192.9632653061, 64.0326530612, 19.0367346939);--hb-color-brand-ubuntu-dark-rgb: 192.9632653061, 64.0326530612, 19.0367346939;--hb-color-aqua: rgb(0, 180, 216);--hb-color-aqua-rgb: 0, 180, 216;--hb-color-aqua-dark: rgb(0, 144, 172.8);--hb-color-aqua-dark-rgb: 0, 144, 172.8;--hb-color-amethyst: rgb(147, 51, 234);--hb-color-amethyst-rgb: 147, 51, 234;--hb-color-amethyst-dark: rgb(118.56, 21.28, 206.72);--hb-color-amethyst-dark-rgb: 118.56, 21.28, 206.72;--hb-color-indigo: rgb(63, 81, 181);--hb-color-indigo-rgb: 63, 81, 181;--hb-color-indigo-dark: rgb(50.4, 64.8, 144.8);--hb-color-indigo-dark-rgb: 50.4, 64.8, 144.8;--hb-color-denim: rgb(47, 102, 144);--hb-color-denim-rgb: 47, 102, 144;--hb-color-denim-dark: rgb(37.6, 81.6, 115.2);--hb-color-denim-dark-rgb: 37.6, 81.6, 115.2;--hb-color-cerulean: rgb(81, 73, 211);--hb-color-cerulean-rgb: 81, 73, 211;--hb-color-cerulean-dark: rgb(52.2761061947, 44.2336283186, 182.9663716814);--hb-color-cerulean-dark-rgb: 52.2761061947, 44.2336283186, 182.9663716814;--hb-color-ocean: rgb(30, 96, 145);--hb-color-ocean-rgb: 30, 96, 145;--hb-color-ocean-dark: rgb(24, 76.8, 116);--hb-color-ocean-dark-rgb: 24, 76.8, 116;--hb-color-purple: rgb(147, 52, 234);--hb-color-purple-rgb: 147, 52, 234;--hb-color-purple-dark: rgb(118.4857142857, 21.45, 207.35);--hb-color-purple-dark-rgb: 118.4857142857, 21.45, 207.35;--hb-color-azure: rgb(79, 158, 248);--hb-color-azure-rgb: 79, 158, 248;--hb-color-azure-dark: rgb(16.1016393443, 123.3344262295, 245.4983606557);--hb-color-azure-dark-rgb: 16.1016393443, 123.3344262295, 245.4983606557;--hb-color-lavender: rgb(192, 132, 252);--hb-color-lavender-rgb: 192, 132, 252;--hb-color-lavender-dark: rgb(153.6, 57.0285714286, 250.1714285714);--hb-color-lavender-dark-rgb: 153.6, 57.0285714286, 250.1714285714;--hb-color-plum: rgb(142, 69, 133);--hb-color-plum-rgb: 142, 69, 133;--hb-color-plum-dark: rgb(113.6, 55.2, 106.4);--hb-color-plum-dark-rgb: 113.6, 55.2, 106.4;--hb-color-rose: rgb(229, 115, 115);--hb-color-rose-rgb: 229, 115, 115;--hb-color-rose-dark: rgb(218.2240963855, 56.9759036145, 56.9759036145);--hb-color-rose-dark-rgb: 218.2240963855, 56.9759036145, 56.9759036145;--hb-color-pink: rgb(233, 30, 99);--hb-color-pink-rgb: 233, 30, 99;--hb-color-pink-dark: rgb(191.6599190283, 18.7400809717, 77.5157894737);--hb-color-pink-dark-rgb: 191.6599190283, 18.7400809717, 77.5157894737;--hb-color-crimson: rgb(220, 38, 38);--hb-color-crimson-rgb: 220, 38, 38;--hb-color-crimson-dark: rgb(177.7333333333, 28.6666666667, 28.6666666667);--hb-color-crimson-dark-rgb: 177.7333333333, 28.6666666667, 28.6666666667;--hb-color-ruby: rgb(22, 163, 74);--hb-color-ruby-rgb: 22, 163, 74;--hb-color-ruby-dark: rgb(17.6, 130.4, 59.2);--hb-color-ruby-dark-rgb: 17.6, 130.4, 59.2;--hb-color-spring: rgb(0, 230, 118);--hb-color-spring-rgb: 0, 230, 118;--hb-color-spring-dark: rgb(0, 184, 94.4);--hb-color-spring-dark-rgb: 0, 184, 94.4;--hb-color-apple: rgb(46, 204, 113);--hb-color-apple-rgb: 46, 204, 113;--hb-color-apple-dark: rgb(36.8, 163.2, 90.4);--hb-color-apple-dark-rgb: 36.8, 163.2, 90.4;--hb-color-lime: rgb(203, 220, 56);--hb-color-lime-rgb: 203, 220, 56;--hb-color-lime-dark: rgb(171.7333333333, 187.7743589744, 33.0256410256);--hb-color-lime-dark-rgb: 171.7333333333, 187.7743589744, 33.0256410256;--hb-color-emerald: rgb(15, 184, 129);--hb-color-emerald-rgb: 15, 184, 129;--hb-color-emerald-dark: rgb(12, 147.2, 103.2);--hb-color-emerald-dark-rgb: 12, 147.2, 103.2;--hb-color-teal: rgb(0, 150, 136);--hb-color-teal-rgb: 0, 150, 136;--hb-color-teal-dark: rgb(0, 120, 108.8);--hb-color-teal-dark-rgb: 0, 120, 108.8;--hb-color-petrol: rgb(0, 128, 128);--hb-color-petrol-rgb: 0, 128, 128;--hb-color-petrol-dark: rgb(0, 102.4, 102.4);--hb-color-petrol-dark-rgb: 0, 102.4, 102.4;--hb-color-wine: rgb(114, 47, 55);--hb-color-wine-rgb: 114, 47, 55;--hb-color-wine-dark: rgb(91.2, 37.6, 44);--hb-color-wine-dark-rgb: 91.2, 37.6, 44;--hb-color-mulberry: rgb(155, 34, 66);--hb-color-mulberry-rgb: 155, 34, 66;--hb-color-mulberry-dark: rgb(124, 27.2, 52.8);--hb-color-mulberry-dark-rgb: 124, 27.2, 52.8;--hb-color-fern: rgb(98, 139, 72);--hb-color-fern-rgb: 98, 139, 72;--hb-color-fern-dark: rgb(78.4, 111.2, 57.6);--hb-color-fern-dark-rgb: 78.4, 111.2, 57.6;--hb-color-honey: rgb(246, 190, 79);--hb-color-honey-rgb: 246, 190, 79;--hb-color-honey-dark: rgb(242.8378378378, 167.1621621622, 17.1621621622);--hb-color-honey-dark-rgb: 242.8378378378, 167.1621621622, 17.1621621622;--hb-color-lemon: rgb(255, 241, 118);--hb-color-lemon-rgb: 255, 241, 118;--hb-color-lemon-dark: rgb(255, 233.3766423358, 43.4);--hb-color-lemon-dark-rgb: 255, 233.3766423358, 43.4;--hb-color-gold: rgb(255, 181, 71);--hb-color-gold-rgb: 255, 181, 71;--hb-color-gold-dark: rgb(255, 154.7782608696, 5.8);--hb-color-gold-dark-rgb: 255, 154.7782608696, 5.8;--hb-color-amber: rgb(255, 193, 7);--hb-color-amber-rgb: 255, 193, 7;--hb-color-amber-dark: rgb(209.6, 157.2, 0);--hb-color-amber-dark-rgb: 209.6, 157.2, 0;--hb-color-coral: rgb(255, 127, 80);--hb-color-coral-rgb: 255, 127, 80;--hb-color-coral-dark: rgb(255, 77.9942857143, 13);--hb-color-coral-dark-rgb: 255, 77.9942857143, 13;--hb-color-orange: rgb(244, 158, 11);--hb-color-orange-rgb: 244, 158, 11;--hb-color-orange-dark: rgb(195.2, 126.4, 8.8);--hb-color-orange-dark-rgb: 195.2, 126.4, 8.8;--hb-color-caramel: rgb(199, 122, 53);--hb-color-caramel-rgb: 199, 122, 53;--hb-color-caramel-dark: rgb(159.2, 97.6, 42.4);--hb-color-caramel-dark-rgb: 159.2, 97.6, 42.4;--hb-color-stone: rgb(120, 113, 108);--hb-color-stone-rgb: 120, 113, 108;--hb-color-stone-dark: rgb(96, 90.4, 86.4);--hb-color-stone-dark-rgb: 96, 90.4, 86.4;--hb-color-graphite: rgb(61, 61, 61);--hb-color-graphite-rgb: 61, 61, 61;--hb-color-graphite-dark: rgb(48.8, 48.8, 48.8);--hb-color-graphite-dark-rgb: 48.8, 48.8, 48.8;--hb-color-charcoal: rgb(75, 85, 99);--hb-color-charcoal-rgb: 75, 85, 99;--hb-color-charcoal-dark: rgb(60, 68, 79.2);--hb-color-charcoal-dark-rgb: 60, 68, 79.2;--hb-color-storm: rgb(71, 85, 105);--hb-color-storm-rgb: 71, 85, 105;--hb-color-storm-dark: rgb(56.8, 68, 84);--hb-color-storm-dark-rgb: 56.8, 68, 84;--hb-color-smoke: rgb(115, 115, 115);--hb-color-smoke-rgb: 115, 115, 115;--hb-color-smoke-dark: #5c5c5c;--hb-color-smoke-dark-rgb: 92, 92, 92;--hb-color-shadow: rgb(31, 41, 55);--hb-color-shadow-rgb: 31, 41, 55;--hb-color-shadow-dark: rgb(24.8, 32.8, 44);--hb-color-shadow-dark-rgb: 24.8, 32.8, 44;--hb-color-steel: rgb(107, 114, 128);--hb-color-steel-rgb: 107, 114, 128;--hb-color-steel-dark: rgb(85.6, 91.2, 102.4);--hb-color-steel-dark-rgb: 85.6, 91.2, 102.4;--hb-color-peach: rgb(255, 188, 154);--hb-color-peach-rgb: 255, 188, 154;--hb-color-peach-dark: rgb(255, 133.7366336634, 72.2);--hb-color-peach-dark-rgb: 255, 133.7366336634, 72.2;--hb-color-sunset: rgb(251, 113, 133);--hb-color-sunset-rgb: 251, 113, 133;--hb-color-sunset-dark: rgb(249.0054794521, 42.1945205479, 72.1671232877);--hb-color-sunset-dark-rgb: 249.0054794521, 42.1945205479, 72.1671232877;--hb-color-chartreuse: rgb(167, 244, 50);--hb-color-chartreuse-rgb: 167, 244, 50;--hb-color-chartreuse-dark: rgb(139.3777777778, 223.2222222222, 11.9777777778);--hb-color-chartreuse-dark-rgb: 139.3777777778, 223.2222222222, 11.9777777778;--hb-color-jade: rgb(0, 191, 165);--hb-color-jade-rgb: 0, 191, 165;--hb-color-jade-dark: rgb(0, 152.8, 132);--hb-color-jade-dark-rgb: 0, 152.8, 132}.ui-color-brand-github{color:var(--hb-color-brand-github)}.ui-color-bg-brand-github{background-color:var(--hb-color-brand-github);color:#fff}.ui-color-bg-gradient-brand-github{background:linear-gradient(to right, var(--hb-color-brand-github), var(--hb-color-brand-github-dark));color:#fff}.ui-color-brand-docker{color:var(--hb-color-brand-docker)}.ui-color-bg-brand-docker{background-color:var(--hb-color-brand-docker);color:#fff}.ui-color-bg-gradient-brand-docker{background:linear-gradient(to right, var(--hb-color-brand-docker), var(--hb-color-brand-docker-dark));color:#fff}.ui-color-brand-ubuntu{color:var(--hb-color-brand-ubuntu)}.ui-color-bg-brand-ubuntu{background-color:var(--hb-color-brand-ubuntu);color:#fff}.ui-color-bg-gradient-brand-ubuntu{background:linear-gradient(to right, var(--hb-color-brand-ubuntu), var(--hb-color-brand-ubuntu-dark));color:#fff}.ui-color-aqua{color:var(--hb-color-aqua)}.ui-color-bg-aqua{background-color:var(--hb-color-aqua);color:#fff}.ui-color-bg-gradient-aqua{background:linear-gradient(to right, var(--hb-color-aqua), var(--hb-color-aqua-dark));color:#fff}.ui-color-amethyst{color:var(--hb-color-amethyst)}.ui-color-bg-amethyst{background-color:var(--hb-color-amethyst);color:#fff}.ui-color-bg-gradient-amethyst{background:linear-gradient(to right, var(--hb-color-amethyst), var(--hb-color-amethyst-dark));color:#fff}.ui-color-indigo{color:var(--hb-color-indigo)}.ui-color-bg-indigo{background-color:var(--hb-color-indigo);color:#fff}.ui-color-bg-gradient-indigo{background:linear-gradient(to right, var(--hb-color-indigo), var(--hb-color-indigo-dark));color:#fff}.ui-color-denim{color:var(--hb-color-denim)}.ui-color-bg-denim{background-color:var(--hb-color-denim);color:#fff}.ui-color-bg-gradient-denim{background:linear-gradient(to right, var(--hb-color-denim), var(--hb-color-denim-dark));color:#fff}.ui-color-cerulean{color:var(--hb-color-cerulean)}.ui-color-bg-cerulean{background-color:var(--hb-color-cerulean);color:#fff}.ui-color-bg-gradient-cerulean{background:linear-gradient(to right, var(--hb-color-cerulean), var(--hb-color-cerulean-dark));color:#fff}.ui-color-ocean{color:var(--hb-color-ocean)}.ui-color-bg-ocean{background-color:var(--hb-color-ocean);color:#fff}.ui-color-bg-gradient-ocean{background:linear-gradient(to right, var(--hb-color-ocean), var(--hb-color-ocean-dark));color:#fff}.ui-color-purple{color:var(--hb-color-purple)}.ui-color-bg-purple{background-color:var(--hb-color-purple);color:#fff}.ui-color-bg-gradient-purple{background:linear-gradient(to right, var(--hb-color-purple), var(--hb-color-purple-dark));color:#fff}.ui-color-azure{color:var(--hb-color-azure)}.ui-color-bg-azure{background-color:var(--hb-color-azure);color:#000}.ui-color-bg-gradient-azure{background:linear-gradient(to right, var(--hb-color-azure), var(--hb-color-azure-dark));color:#000}.ui-color-lavender{color:var(--hb-color-lavender)}.ui-color-bg-lavender{background-color:var(--hb-color-lavender);color:#000}.ui-color-bg-gradient-lavender{background:linear-gradient(to right, var(--hb-color-lavender), var(--hb-color-lavender-dark));color:#000}.ui-color-plum{color:var(--hb-color-plum)}.ui-color-bg-plum{background-color:var(--hb-color-plum);color:#fff}.ui-color-bg-gradient-plum{background:linear-gradient(to right, var(--hb-color-plum), var(--hb-color-plum-dark));color:#fff}.ui-color-rose{color:var(--hb-color-rose)}.ui-color-bg-rose{background-color:var(--hb-color-rose);color:#000}.ui-color-bg-gradient-rose{background:linear-gradient(to right, var(--hb-color-rose), var(--hb-color-rose-dark));color:#000}.ui-color-pink{color:var(--hb-color-pink)}.ui-color-bg-pink{background-color:var(--hb-color-pink);color:#fff}.ui-color-bg-gradient-pink{background:linear-gradient(to right, var(--hb-color-pink), var(--hb-color-pink-dark));color:#fff}.ui-color-crimson{color:var(--hb-color-crimson)}.ui-color-bg-crimson{background-color:var(--hb-color-crimson);color:#fff}.ui-color-bg-gradient-crimson{background:linear-gradient(to right, var(--hb-color-crimson), var(--hb-color-crimson-dark));color:#fff}.ui-color-ruby{color:var(--hb-color-ruby)}.ui-color-bg-ruby{background-color:var(--hb-color-ruby);color:#fff}.ui-color-bg-gradient-ruby{background:linear-gradient(to right, var(--hb-color-ruby), var(--hb-color-ruby-dark));color:#fff}.ui-color-spring{color:var(--hb-color-spring)}.ui-color-bg-spring{background-color:var(--hb-color-spring);color:#fff}.ui-color-bg-gradient-spring{background:linear-gradient(to right, var(--hb-color-spring), var(--hb-color-spring-dark));color:#fff}.ui-color-apple{color:var(--hb-color-apple)}.ui-color-bg-apple{background-color:var(--hb-color-apple);color:#fff}.ui-color-bg-gradient-apple{background:linear-gradient(to right, var(--hb-color-apple), var(--hb-color-apple-dark));color:#fff}.ui-color-lime{color:var(--hb-color-lime)}.ui-color-bg-lime{background-color:var(--hb-color-lime);color:#fff}.ui-color-bg-gradient-lime{background:linear-gradient(to right, var(--hb-color-lime), var(--hb-color-lime-dark));color:#fff}.ui-color-emerald{color:var(--hb-color-emerald)}.ui-color-bg-emerald{background-color:var(--hb-color-emerald);color:#fff}.ui-color-bg-gradient-emerald{background:linear-gradient(to right, var(--hb-color-emerald), var(--hb-color-emerald-dark));color:#fff}.ui-color-teal{color:var(--hb-color-teal)}.ui-color-bg-teal{background-color:var(--hb-color-teal);color:#fff}.ui-color-bg-gradient-teal{background:linear-gradient(to right, var(--hb-color-teal), var(--hb-color-teal-dark));color:#fff}.ui-color-petrol{color:var(--hb-color-petrol)}.ui-color-bg-petrol{background-color:var(--hb-color-petrol);color:#fff}.ui-color-bg-gradient-petrol{background:linear-gradient(to right, var(--hb-color-petrol), var(--hb-color-petrol-dark));color:#fff}.ui-color-wine{color:var(--hb-color-wine)}.ui-color-bg-wine{background-color:var(--hb-color-wine);color:#fff}.ui-color-bg-gradient-wine{background:linear-gradient(to right, var(--hb-color-wine), var(--hb-color-wine-dark));color:#fff}.ui-color-mulberry{color:var(--hb-color-mulberry)}.ui-color-bg-mulberry{background-color:var(--hb-color-mulberry);color:#fff}.ui-color-bg-gradient-mulberry{background:linear-gradient(to right, var(--hb-color-mulberry), var(--hb-color-mulberry-dark));color:#fff}.ui-color-fern{color:var(--hb-color-fern)}.ui-color-bg-fern{background-color:var(--hb-color-fern);color:#fff}.ui-color-bg-gradient-fern{background:linear-gradient(to right, var(--hb-color-fern), var(--hb-color-fern-dark));color:#fff}.ui-color-honey{color:var(--hb-color-honey)}.ui-color-bg-honey{background-color:var(--hb-color-honey);color:#000}.ui-color-bg-gradient-honey{background:linear-gradient(to right, var(--hb-color-honey), var(--hb-color-honey-dark));color:#000}.ui-color-lemon{color:var(--hb-color-lemon)}.ui-color-bg-lemon{background-color:var(--hb-color-lemon);color:#000}.ui-color-bg-gradient-lemon{background:linear-gradient(to right, var(--hb-color-lemon), var(--hb-color-lemon-dark));color:#000}.ui-color-gold{color:var(--hb-color-gold)}.ui-color-bg-gold{background-color:var(--hb-color-gold);color:#000}.ui-color-bg-gradient-gold{background:linear-gradient(to right, var(--hb-color-gold), var(--hb-color-gold-dark));color:#000}.ui-color-amber{color:var(--hb-color-amber)}.ui-color-bg-amber{background-color:var(--hb-color-amber);color:#fff}.ui-color-bg-gradient-amber{background:linear-gradient(to right, var(--hb-color-amber), var(--hb-color-amber-dark));color:#fff}.ui-color-coral{color:var(--hb-color-coral)}.ui-color-bg-coral{background-color:var(--hb-color-coral);color:#000}.ui-color-bg-gradient-coral{background:linear-gradient(to right, var(--hb-color-coral), var(--hb-color-coral-dark));color:#000}.ui-color-orange{color:var(--hb-color-orange)}.ui-color-bg-orange{background-color:var(--hb-color-orange);color:#fff}.ui-color-bg-gradient-orange{background:linear-gradient(to right, var(--hb-color-orange), var(--hb-color-orange-dark));color:#fff}.ui-color-caramel{color:var(--hb-color-caramel)}.ui-color-bg-caramel{background-color:var(--hb-color-caramel);color:#fff}.ui-color-bg-gradient-caramel{background:linear-gradient(to right, var(--hb-color-caramel), var(--hb-color-caramel-dark));color:#fff}.ui-color-stone{color:var(--hb-color-stone)}.ui-color-bg-stone{background-color:var(--hb-color-stone);color:#fff}.ui-color-bg-gradient-stone{background:linear-gradient(to right, var(--hb-color-stone), var(--hb-color-stone-dark));color:#fff}.ui-color-graphite{color:var(--hb-color-graphite)}.ui-color-bg-graphite{background-color:var(--hb-color-graphite);color:#fff}.ui-color-bg-gradient-graphite{background:linear-gradient(to right, var(--hb-color-graphite), var(--hb-color-graphite-dark));color:#fff}.ui-color-charcoal{color:var(--hb-color-charcoal)}.ui-color-bg-charcoal{background-color:var(--hb-color-charcoal);color:#fff}.ui-color-bg-gradient-charcoal{background:linear-gradient(to right, var(--hb-color-charcoal), var(--hb-color-charcoal-dark));color:#fff}.ui-color-storm{color:var(--hb-color-storm)}.ui-color-bg-storm{background-color:var(--hb-color-storm);color:#fff}.ui-color-bg-gradient-storm{background:linear-gradient(to right, var(--hb-color-storm), var(--hb-color-storm-dark));color:#fff}.ui-color-smoke{color:var(--hb-color-smoke)}.ui-color-bg-smoke{background-color:var(--hb-color-smoke);color:#fff}.ui-color-bg-gradient-smoke{background:linear-gradient(to right, var(--hb-color-smoke), var(--hb-color-smoke-dark));color:#fff}.ui-color-shadow{color:var(--hb-color-shadow)}.ui-color-bg-shadow{background-color:var(--hb-color-shadow);color:#fff}.ui-color-bg-gradient-shadow{background:linear-gradient(to right, var(--hb-color-shadow), var(--hb-color-shadow-dark));color:#fff}.ui-color-steel{color:var(--hb-color-steel)}.ui-color-bg-steel{background-color:var(--hb-color-steel);color:#fff}.ui-color-bg-gradient-steel{background:linear-gradient(to right, var(--hb-color-steel), var(--hb-color-steel-dark));color:#fff}.ui-color-peach{color:var(--hb-color-peach)}.ui-color-bg-peach{background-color:var(--hb-color-peach);color:#000}.ui-color-bg-gradient-peach{background:linear-gradient(to right, var(--hb-color-peach), var(--hb-color-peach-dark));color:#000}.ui-color-sunset{color:var(--hb-color-sunset)}.ui-color-bg-sunset{background-color:var(--hb-color-sunset);color:#000}.ui-color-bg-gradient-sunset{background:linear-gradient(to right, var(--hb-color-sunset), var(--hb-color-sunset-dark));color:#000}.ui-color-chartreuse{color:var(--hb-color-chartreuse)}.ui-color-bg-chartreuse{background-color:var(--hb-color-chartreuse);color:#fff}.ui-color-bg-gradient-chartreuse{background:linear-gradient(to right, var(--hb-color-chartreuse), var(--hb-color-chartreuse-dark));color:#fff}.ui-color-jade{color:var(--hb-color-jade)}.ui-color-bg-jade{background-color:var(--hb-color-jade);color:#fff}.ui-color-bg-gradient-jade{background:linear-gradient(to right, var(--hb-color-jade), var(--hb-color-jade-dark));color:#fff}:root{--hb-settings-color-instance-name: #2563EB;--hb-settings-color-default-language: #16A34A;--hb-settings-color-user-actions-disable: #FFA726;--hb-settings-color-user-actions-delete: #E53935}.text-dark{color:var(--mud-palette-text-primary)}.mud-navmenu.mud-navmenu-first-level>.mud-nav-item>.mud-nav-link{border-radius:inherit}.mud-navmenu.mud-navmenu-xxl .mud-nav-link{display:flex !important;align-items:center !important;padding:8px 8px !important}.mud-navmenu.mud-navmenu-xxl .mud-icon-root{width:2rem;height:2rem}.mud-navmenu.mud-navmenu-xxl .mud-nav-link-text{font-size:1rem;color:var(--mud-palette-primary)}.mud-paper>.mud-list>.mud-list-item:first-child{border-top-left-radius:var(--mud-default-borderradius);border-top-right-radius:var(--mud-default-borderradius)}.mud-paper>.mud-list>.mud-list-item:last-child{border-bottom-left-radius:var(--mud-default-borderradius);border-bottom-right-radius:var(--mud-default-borderradius)}.mud-paper>.mud-navmenu.mud-navmenu-first-level>.mud-nav-item:first-child{border-top-left-radius:var(--mud-default-borderradius);border-top-right-radius:var(--mud-default-borderradius)}.mud-paper>.mud-navmenu.mud-navmenu-first-level>.mud-nav-item:last-child{border-bottom-left-radius:var(--mud-default-borderradius);border-bottom-right-radius:var(--mud-default-borderradius)}.mud-drawer.frosted.mud-theme-primary{color:var(--mud-palette-primary-text) !important;background:rgb(from var(--mud-palette-primary) r g b/0.7) !important}.mud-drawer.frosted.mud-theme-primary::before{content:none !important}.mud-drawer.frosted.mud-theme-secondary{color:var(--mud-palette-secondary-text) !important;background:rgb(from var(--mud-palette-secondary) r g b/0.7) !important}.mud-drawer.frosted.mud-theme-secondary::before{content:none !important}.mud-drawer.frosted.mud-theme-tertiary{color:var(--mud-palette-tertiary-text) !important;background:rgb(from var(--mud-palette-tertiary) r g b/0.7) !important}.mud-drawer.frosted.mud-theme-tertiary::before{content:none !important}.mud-card>.mud-card-header{border-color:var(--mud-palette-divider);border-width:1px;border-style:solid none none none;padding:8px 12px}.mud-card>.mud-card-header .mud-card-header-actions{margin:0 !important;align-self:center}.mud-card>.mud-card-header .mud-typography{padding:4px !important}.mud-card>.mud-card-content{border-color:var(--mud-palette-divider);border-width:1px;border-style:solid none none none}.mud-card>.mud-card-actions{border-bottom-left-radius:var(--mud-default-borderradius);border-bottom-right-radius:var(--mud-default-borderradius);background-color:var(--mud-palette-background-gray)}.mud-card .no-divider{border:none}.mud-stepper.stepper-no-actions .mud-stepper-actions{display:none}.ui-wavebackground-container{width:100vw;height:100vh}.ui-wavebackground-container.ui-bg-01{background:linear-gradient(315deg, rgb(24, 19, 89) 2%, rgb(1, 55, 125) 25%, rgb(50, 150, 200) 45%, rgb(164, 236, 248) 70%, rgb(66, 111, 160) 98%);animation:gradient 150s ease infinite;background-size:400% 400%;background-attachment:fixed}.ui-wavebackground-container.ui-bg-02{background:url("/img/bg/ws25.jpg") no-repeat center center fixed;background-size:cover}.ui-wavebackground-container.ui-bg-03{background:url("/img/bg/wp12118355.webp") no-repeat center center fixed;background-size:cover}.ui-wavebackground-container.ui-bg-04{background:url("/img/bg/wp12118355-light.webp") no-repeat center center fixed;background-size:cover}.ui-wavebackground-container .wave{background:hsla(0,0%,100%,.25);border-radius:1000% 1000% 0 0;position:fixed;width:400%;height:30em;animation:wave 30s -3s linear infinite;transform:translate3d(0, 0, 0);opacity:.8;bottom:0;left:0;z-index:-100}.ui-wavebackground-container .wave:nth-of-type(2){bottom:-1.25em;animation:wave 54s linear reverse infinite;opacity:.8}.ui-wavebackground-container .wave:nth-of-type(3){bottom:-2.5em;animation:wave 60s -1s reverse infinite;opacity:.9}@keyframes gradient{0%{background-position:0% 0%}50%{background-position:100% 100%}100%{background-position:0% 0%}}@keyframes wave{2%{transform:translateX(1)}25%{transform:translateX(-25%)}50%{transform:translateX(-50%)}75%{transform:translateX(-25%)}100%{transform:translateX(1)}}.ui-stripe-background-container{width:100vw;height:100vh}#ui-stripe-background-canvas{width:inherit;height:inherit}#ui-stripe-background-canvas.build-mode-alpha{--gradient-color-1: #003a7a;--gradient-color-2: #003f90;--gradient-color-3: #00366e;--gradient-color-4: #002790;--gradient-color-5: #005fe0}#ui-stripe-background-canvas.build-mode-beta{--gradient-color-1: #1b5d8a;--gradient-color-2: #1c5d89;--gradient-color-3: #009ac7;--gradient-color-4: #114d7b;--gradient-color-5: #26a4d2}#ui-stripe-background-canvas.build-mode-release{--gradient-color-1: #7ebbfc;--gradient-color-2: #3366ff;--gradient-color-3: #1340c8;--gradient-color-4: #0029a3;--gradient-color-5: #1c1d7c}.ui-countdown-alert{overflow:hidden;position:relative;border-radius:var(--mud-default-borderradius)}.ui-countdown-alert .mud-progress-linear{position:absolute;bottom:0;left:0;width:100%}.ui-dev-icons-container .mud-icon-root{font-size:60px}:root{--cell-size: 170px;--cell-size: 73px;--cell-gap: 24px;--widget-border-radius: 12px;--ui-widget-drawer-width: calc((var(--cell-size) * 8) + (var(--cell-gap) * 7) + var(--cell-gap));--hb-widget-padding: calc(var(--widget-border-radius) / 3)}.hb-widget-container{height:inherit;width:inherit}.hb-widget-grid{display:grid;gap:var(--cell-gap);grid-template-columns:repeat(auto-fill, var(--cell-size));grid-template-rows:repeat(auto-fill, var(--cell-size));grid-auto-rows:var(--cell-size);width:100%;height:100%}.ui-widget-list{display:flex;flex-direction:column;row-gap:var(--cell-gap);padding:calc(var(--cell-gap)/2)}.ui-widget{overflow:hidden;display:flex;flex-direction:column}.ui-widget>*{flex:1 1 auto}.ui-widget.w-2{width:calc(var(--cell-size)*2 + var(--cell-gap)*1)}.ui-widget.w-4{width:calc(var(--cell-size)*4 + var(--cell-gap)*3)}.ui-widget.w-8{width:calc(var(--cell-size)*8 + var(--cell-gap)*7)}.ui-widget.h-1{height:calc(var(--cell-size)*1 + var(--cell-gap)*0)}.ui-widget.h-2{height:calc(var(--cell-size)*2 + var(--cell-gap)*1)}.ui-widget.h-4{height:calc(var(--cell-size)*4 + var(--cell-gap)*3)}.hb-widget-selection-drawer{width:var(--ui-widget-drawer-width)}.ui-icon>.ui-icon-el{border-radius:var(--hb-ui-icon-border-radius, 0)}.ui-icon>.ui-icon-el.ui-icon-filled{color:#fff;padding:2px}.ui-icon>.ui-icon-el.ui-icon-outlined{border-width:1px;border-style:solid;padding:1px}.ui-icon>.ui-icon-el.ui-icon-outlined.mud-icon-size-large{border-width:2px}.ui-settings-item .ui-settings-item-content{width:100%}@media(min-width: 960px){.ui-settings-item .ui-settings-item-content{width:240px}}.ui-startmenu-item-link{background:var(--hb-startmenu-container-background);width:100%}.ui-startmenu-item-link>span{height:100%}.ui-startmenu-container{width:100%;height:inherit;overflow:hidden}.ui-startmenu-item-background-icon{height:var(--hb-startmenu-background-icon-size);width:var(--hb-startmenu-background-icon-size);opacity:var(--hb-startmenu-background-icon-opacity);position:absolute;z-index:-1;top:-25px;left:-25px}.ui-startmenu-item-content{height:inherit}.ui-startmenu-item-content .ui-startmenu-item-nav{margin-top:auto !important}.ui-progress-item{display:flex;flex-direction:column;width:100%}.ui-progress-item .ui-progress-item-header,.ui-progress-item .ui-progress-item-progress,.ui-progress-item .ui-progress-item-footer{display:flex;width:100%}.ui-progress-item .ui-progress-item-header,.ui-progress-item .ui-progress-item-footer{align-items:center}.ui-progress-item .ui-progress-item-header .ui-progress-item-header-start,.ui-progress-item .ui-progress-item-header .ui-progress-item-footer-start,.ui-progress-item .ui-progress-item-footer .ui-progress-item-header-start,.ui-progress-item .ui-progress-item-footer .ui-progress-item-footer-start{flex:1 1 auto;min-width:0}.ui-progress-item .ui-progress-item-header .ui-progress-item-header-end,.ui-progress-item .ui-progress-item-header .ui-progress-item-footer-end,.ui-progress-item .ui-progress-item-footer .ui-progress-item-header-end,.ui-progress-item .ui-progress-item-footer .ui-progress-item-footer-end{flex:0 0 auto}.ui-progress-item .ui-progress-item-progress{width:100%}.ui-progress-item .ui-progress-item-progress .mud-progress-linear-bar{background-color:var(--ui-element-accent-color) !important}.ui-colored-icon-container .ui-colored-icon-frame{background-color:color-mix(in srgb, var(--ui-colored-icon-color), transparent var(--hb-ui-colored-icon-background-opacity));border-radius:var(--hb-ui-colored-icon-borderradius)}.ui-colored-icon-container .ui-colored-icon{color:var(--ui-colored-icon-color);height:var(--hb-ui-colored-icon-size);width:var(--hb-ui-colored-icon-size)}.ui-detail-card{color:inherit}.ui-detail-card-container{color:inherit}.ui-detail-card-container .ui-detail-card-content{color:inherit}.ui-detail-card-container .ui-detail-card-header{color:inherit}.ui-detail-card-container .ui-detail-card-footer{color:inherit}.ui-value-header{color:inherit}.ui-value-header .ui-value-card-title-text{color:var(--ui-value-card-caption-color)}.ui-value-header .ui-value-display-text{font-weight:500}.ui-value-footer-text{color:inherit}.ui-value-footer-text .ui-value-footer-highlighted-text{color:var(--ui-value-card-accent-color)}.ui-value-footer-text .ui-value-footer-notice-text{color:var(--ui-value-card-caption-color)}.hb-drawer{border-radius:12px;margin:8px;height:calc(100vh - 16px) !important;width:calc(var(--mud-drawer-width, var(--mud-drawer-width-left)) - 16px) !important;top:0px}.hb-header-appbar{border-radius:12px;margin-top:8px !important;width:calc(100% - 8px - var(--mud-drawer-width-left)) !important}.hb-footer-appbar{border-radius:12px;margin-bottom:8px;margin-right:8px !important;width:calc(100% - 8px - var(--mud-drawer-width-left)) !important}@media(min-width: 600px){.hb-main-content{padding-top:calc(var(--mud-appbar-height) + 16px);padding-bottom:calc(var(--mud-appbar-height) + 16px)}}.ui-wallpaper{position:fixed;top:0;left:0;width:100vw;height:100vh;z-index:-1;background-size:cover;background-position:center}.hb-setup-container{height:100vh;padding-top:120px;padding-bottom:120px;transition:opacity 3s ease}.hb-setup-container.is-finished{opacity:0}.hb-setup-card{height:100%;overflow-y:hidden}.hb-license-agreement-content{overflow-y:auto}.setup-background{transition:all 2s ease;position:absolute;top:0;left:0;width:100vw;height:100vh;z-index:-100;margin:auto;overflow:hidden}.setup-background.is-finished{opacity:0}.setup-tiled-container{position:absolute;top:0;left:0;width:100vw;height:100vh;z-index:-90;overflow:hidden;transition:opacity 4s ease}.setup-tiled-container.is-hidden{opacity:0}.setup-tiled-container .setup-tiled-background{position:relative;width:150vw;height:150vh;left:-25vw;top:-25vh;transform:rotate(10deg);display:grid;grid-template-columns:1fr 1fr;grid-template-rows:1fr 1fr}.setup-tiled-container .setup-tiled-background .bg-tile{transition:all 6s ease;position:relative;background:radial-gradient(ellipse at center, rgba(255, 255, 255, 0.5) 0%, rgba(255, 255, 255, 0) 75%)}.setup-tiled-container.is-finished .bg-tile:nth-child(1){transform:translate(-200vw, -200px)}.setup-tiled-container.is-finished .bg-tile:nth-child(2){transform:translate(200px, -200vh)}.setup-tiled-container.is-finished .bg-tile:nth-child(3){transform:translate(-200px, 200vh)}.setup-tiled-container.is-finished .bg-tile:nth-child(4){transform:translate(200vw, 200px)}.hb-account-login-container{height:100vh;padding-top:120px;padding-bottom:120px;transition:opacity 3s ease}.hb-account-login-card{height:100%;overflow-y:hidden}.account-login-background{transition:all 2s ease;position:absolute;top:0;left:0;width:100vw;height:100vh;z-index:-100;margin:auto;overflow:hidden}.account-login-background.is-finished{opacity:0}/*# sourceMappingURL=app.min.css.map */ +#blazor-error-ui{color-scheme:light only;background:#ffffe0;bottom:0;box-shadow:0 -1px 2px rgba(0,0,0,.2);box-sizing:border-box;display:none;left:0;padding:.6rem 1.25rem .7rem 1.25rem;position:fixed;width:100%;z-index:1000}#blazor-error-ui .dismiss{cursor:pointer;position:absolute;right:.75rem;top:.5rem}.blazor-error-boundary{background:url() no-repeat 1rem/1.8rem,#b32121;padding:1rem 1rem 1rem 3.7rem;color:#fff}.blazor-error-boundary::after{content:"An error has occurred."}.loading-progress{position:relative;display:block;width:8rem;height:8rem;margin:20vh auto 1rem auto}.loading-progress circle{fill:none;stroke:#e0e0e0;stroke-width:.6rem;transform-origin:50% 50%;transform:rotate(-90deg)}.loading-progress circle:last-child{stroke:#1b6ec2;stroke-dasharray:calc(3.141*var(--blazor-load-percentage, 0%)*.8),500%;transition:stroke-dasharray .05s ease-in-out}.loading-progress-text{position:absolute;text-align:center;font-weight:bold;inset:calc(20vh + 3.25rem) 0 auto .2rem}.loading-progress-text:after{content:var(--blazor-load-percentage-text, "Loading")}:root{--hb-breakpoint-xs: 0px;--hb-breakpoint-sm: 600px;--hb-breakpoint-md: 960px;--hb-breakpoint-lg: 1280px;--hb-breakpoint-xl: 1920px;--hb-breakpoint-xxl: 2560px;--hb-breakpoint-xxxl: 3840px;--hb-breakpoint-xxxxl: 5120px}:root{--hb-frosted-color: rgb(255, 255, 255);--hb-frosted-alpha-min: 0.05;--hb-frosted-alpha-step: 0.05;--hb-frosted-blur: 6px;--hb-frosted-steps: 10;--hb-frosted-border-alpha: 0.05;--hb-frosted-gradient-deg: 160deg;--hb-frosted-gradient-add: 0.25;--hb-frosted-inner-shadow-alpha: 0.3}:root{--hb-ui-icon-border-radius: 6px}:root{--hb-ui-colored-icon-background-opacity: 75%;--hb-ui-colored-icon-borderradius: 8px;--hb-ui-colored-icon-size: 1.5em}:root{--hb-startmenu-background-icon-size: 140px;--hb-startmenu-background-icon-opacity: 0.075;--hb-startmenu-container-background: transparent}.w-0{width:0% !important}.w-25{width:25% !important}.w-50{width:50% !important}.w-75{width:75% !important}.w-100{width:100% !important}.w-auto{width:auto !important}@media(min-width: 600px){.w-sm-0{width:0% !important}.w-sm-25{width:25% !important}.w-sm-50{width:50% !important}.w-sm-75{width:75% !important}.w-sm-100{width:100% !important}.w-sm-auto{width:auto !important}}@media(min-width: 960px){.w-md-0{width:0% !important}.w-md-25{width:25% !important}.w-md-50{width:50% !important}.w-md-75{width:75% !important}.w-md-100{width:100% !important}.w-md-auto{width:auto !important}}@media(min-width: 1280px){.w-lg-0{width:0% !important}.w-lg-25{width:25% !important}.w-lg-50{width:50% !important}.w-lg-75{width:75% !important}.w-lg-100{width:100% !important}.w-lg-auto{width:auto !important}}@media(min-width: 1920px){.w-xl-0{width:0% !important}.w-xl-25{width:25% !important}.w-xl-50{width:50% !important}.w-xl-75{width:75% !important}.w-xl-100{width:100% !important}.w-xl-auto{width:auto !important}}@media(min-width: 2560px){.w-xxl-0{width:0% !important}.w-xxl-25{width:25% !important}.w-xxl-50{width:50% !important}.w-xxl-75{width:75% !important}.w-xxl-100{width:100% !important}.w-xxl-auto{width:auto !important}}@media(min-width: 3840px){.w-xxxl-0{width:0% !important}.w-xxxl-25{width:25% !important}.w-xxxl-50{width:50% !important}.w-xxxl-75{width:75% !important}.w-xxxl-100{width:100% !important}.w-xxxl-auto{width:auto !important}}@media(min-width: 5120px){.w-xxxxl-0{width:0% !important}.w-xxxxl-25{width:25% !important}.w-xxxxl-50{width:50% !important}.w-xxxxl-75{width:75% !important}.w-xxxxl-100{width:100% !important}.w-xxxxl-auto{width:auto !important}}.float-right{float:right !important}.float-left{float:left !important}.float-none{float:none !important}.frosted-b1{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.05) 0%, rgba(255, 255, 255, 0.3) 100%),hsla(0,0%,100%,.05) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.1)}.frosted-b1::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.35);border-radius:inherit}.frosted-b1.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b1{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.05) 0%, rgba(255, 255, 255, 0.3) 100%),hsla(0,0%,100%,.05) !important;backdrop-filter:blur(6px)}.frosted-bg-b1::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.35);border-radius:inherit}.frosted-b2{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.1) 0%, rgba(255, 255, 255, 0.35) 100%),hsla(0,0%,100%,.1) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.15)}.frosted-b2::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.4);border-radius:inherit}.frosted-b2.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b2{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.1) 0%, rgba(255, 255, 255, 0.35) 100%),hsla(0,0%,100%,.1) !important;backdrop-filter:blur(6px)}.frosted-bg-b2::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.4);border-radius:inherit}.frosted-b3{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.15) 0%, rgba(255, 255, 255, 0.4) 100%),hsla(0,0%,100%,.15) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.2)}.frosted-b3::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.45);border-radius:inherit}.frosted-b3.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b3{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.15) 0%, rgba(255, 255, 255, 0.4) 100%),hsla(0,0%,100%,.15) !important;backdrop-filter:blur(6px)}.frosted-bg-b3::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.45);border-radius:inherit}.frosted-b4{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.2) 0%, rgba(255, 255, 255, 0.45) 100%),hsla(0,0%,100%,.2) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.25)}.frosted-b4::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.5);border-radius:inherit}.frosted-b4.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b4{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.2) 0%, rgba(255, 255, 255, 0.45) 100%),hsla(0,0%,100%,.2) !important;backdrop-filter:blur(6px)}.frosted-bg-b4::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.5);border-radius:inherit}.frosted-b5{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.25) 0%, rgba(255, 255, 255, 0.5) 100%),hsla(0,0%,100%,.25) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.3)}.frosted-b5::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.55);border-radius:inherit}.frosted-b5.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b5,.mud-drawer.frosted{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.25) 0%, rgba(255, 255, 255, 0.5) 100%),hsla(0,0%,100%,.25) !important;backdrop-filter:blur(6px)}.frosted-bg-b5::before,.mud-drawer.frosted::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.55);border-radius:inherit}.frosted-b6{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.3) 0%, rgba(255, 255, 255, 0.55) 100%),hsla(0,0%,100%,.3) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.35)}.frosted-b6::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.6);border-radius:inherit}.frosted-b6.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b6{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.3) 0%, rgba(255, 255, 255, 0.55) 100%),hsla(0,0%,100%,.3) !important;backdrop-filter:blur(6px)}.frosted-bg-b6::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.6);border-radius:inherit}.frosted-b7{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.35) 0%, rgba(255, 255, 255, 0.6) 100%),hsla(0,0%,100%,.35) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.4)}.frosted-b7::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.65);border-radius:inherit}.frosted-b7.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b7{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.35) 0%, rgba(255, 255, 255, 0.6) 100%),hsla(0,0%,100%,.35) !important;backdrop-filter:blur(6px)}.frosted-bg-b7::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.65);border-radius:inherit}.frosted-b8{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.4) 0%, rgba(255, 255, 255, 0.65) 100%),hsla(0,0%,100%,.4) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.45)}.frosted-b8::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.7);border-radius:inherit}.frosted-b8.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b8{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.4) 0%, rgba(255, 255, 255, 0.65) 100%),hsla(0,0%,100%,.4) !important;backdrop-filter:blur(6px)}.frosted-bg-b8::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.7);border-radius:inherit}.frosted-b9{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.45) 0%, rgba(255, 255, 255, 0.7) 100%),hsla(0,0%,100%,.45) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.5)}.frosted-b9::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.75);border-radius:inherit}.frosted-b9.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b9{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.45) 0%, rgba(255, 255, 255, 0.7) 100%),hsla(0,0%,100%,.45) !important;backdrop-filter:blur(6px)}.frosted-bg-b9::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.75);border-radius:inherit}.frosted-b10{position:relative;background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.5) 0%, rgba(255, 255, 255, 0.75) 100%),hsla(0,0%,100%,.5) !important;backdrop-filter:blur(6px);border:1px solid hsla(0,0%,100%,.55)}.frosted-b10::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.8);border-radius:inherit}.frosted-b10.no-backdrop-blur{backdrop-filter:none}.frosted-bg-b10{background:radial-gradient(ellipse 80% 30% at top center, rgba(255, 255, 255, 0.15) 0%, transparent 80%),linear-gradient(160deg, rgba(255, 255, 255, 0.5) 0%, rgba(255, 255, 255, 0.75) 100%),hsla(0,0%,100%,.5) !important;backdrop-filter:blur(6px)}.frosted-bg-b10::before{content:"";position:absolute;inset:0;pointer-events:none;box-shadow:inset 0px 4px 48px hsla(0,0%,100%,.8);border-radius:inherit}:root{--hb-github-color: #24292e;--hb-docker-color: #2496ed;--hb-ubuntu-color: #e95420}.github-bg{background-color:#24292e !important;color:#fff !important}.github-text{color:#24292e !important}.docker-bg{background-color:#2496ed !important;color:#fff !important}.docker-text{color:#2496ed !important}.ubuntu-bg{background-color:#e95420 !important;color:#fff !important}.ubuntu-text{color:#e95420 !important}:root{--hb-color-brand-github: rgb(36, 41, 46);--hb-color-brand-github-rgb: 36, 41, 46;--hb-color-brand-github-dark: rgb(28.8, 32.8, 36.8);--hb-color-brand-github-dark-rgb: 28.8, 32.8, 36.8;--hb-color-brand-docker: rgb(36, 150, 237);--hb-color-brand-docker-rgb: 36, 150, 237;--hb-color-brand-docker-dark: rgb(16.5873417722, 121.6405063291, 201.8126582278);--hb-color-brand-docker-dark-rgb: 16.5873417722, 121.6405063291, 201.8126582278;--hb-color-brand-ubuntu: rgb(233, 84, 32);--hb-color-brand-ubuntu-rgb: 233, 84, 32;--hb-color-brand-ubuntu-dark: rgb(192.9632653061, 64.0326530612, 19.0367346939);--hb-color-brand-ubuntu-dark-rgb: 192.9632653061, 64.0326530612, 19.0367346939;--hb-color-aqua: rgb(0, 180, 216);--hb-color-aqua-rgb: 0, 180, 216;--hb-color-aqua-dark: rgb(0, 144, 172.8);--hb-color-aqua-dark-rgb: 0, 144, 172.8;--hb-color-amethyst: rgb(147, 51, 234);--hb-color-amethyst-rgb: 147, 51, 234;--hb-color-amethyst-dark: rgb(118.56, 21.28, 206.72);--hb-color-amethyst-dark-rgb: 118.56, 21.28, 206.72;--hb-color-indigo: rgb(63, 81, 181);--hb-color-indigo-rgb: 63, 81, 181;--hb-color-indigo-dark: rgb(50.4, 64.8, 144.8);--hb-color-indigo-dark-rgb: 50.4, 64.8, 144.8;--hb-color-denim: rgb(47, 102, 144);--hb-color-denim-rgb: 47, 102, 144;--hb-color-denim-dark: rgb(37.6, 81.6, 115.2);--hb-color-denim-dark-rgb: 37.6, 81.6, 115.2;--hb-color-cerulean: rgb(81, 73, 211);--hb-color-cerulean-rgb: 81, 73, 211;--hb-color-cerulean-dark: rgb(52.2761061947, 44.2336283186, 182.9663716814);--hb-color-cerulean-dark-rgb: 52.2761061947, 44.2336283186, 182.9663716814;--hb-color-ocean: rgb(30, 96, 145);--hb-color-ocean-rgb: 30, 96, 145;--hb-color-ocean-dark: rgb(24, 76.8, 116);--hb-color-ocean-dark-rgb: 24, 76.8, 116;--hb-color-purple: rgb(147, 52, 234);--hb-color-purple-rgb: 147, 52, 234;--hb-color-purple-dark: rgb(118.4857142857, 21.45, 207.35);--hb-color-purple-dark-rgb: 118.4857142857, 21.45, 207.35;--hb-color-azure: rgb(79, 158, 248);--hb-color-azure-rgb: 79, 158, 248;--hb-color-azure-dark: rgb(16.1016393443, 123.3344262295, 245.4983606557);--hb-color-azure-dark-rgb: 16.1016393443, 123.3344262295, 245.4983606557;--hb-color-lavender: rgb(192, 132, 252);--hb-color-lavender-rgb: 192, 132, 252;--hb-color-lavender-dark: rgb(153.6, 57.0285714286, 250.1714285714);--hb-color-lavender-dark-rgb: 153.6, 57.0285714286, 250.1714285714;--hb-color-plum: rgb(142, 69, 133);--hb-color-plum-rgb: 142, 69, 133;--hb-color-plum-dark: rgb(113.6, 55.2, 106.4);--hb-color-plum-dark-rgb: 113.6, 55.2, 106.4;--hb-color-rose: rgb(229, 115, 115);--hb-color-rose-rgb: 229, 115, 115;--hb-color-rose-dark: rgb(218.2240963855, 56.9759036145, 56.9759036145);--hb-color-rose-dark-rgb: 218.2240963855, 56.9759036145, 56.9759036145;--hb-color-pink: rgb(233, 30, 99);--hb-color-pink-rgb: 233, 30, 99;--hb-color-pink-dark: rgb(191.6599190283, 18.7400809717, 77.5157894737);--hb-color-pink-dark-rgb: 191.6599190283, 18.7400809717, 77.5157894737;--hb-color-crimson: rgb(220, 38, 38);--hb-color-crimson-rgb: 220, 38, 38;--hb-color-crimson-dark: rgb(177.7333333333, 28.6666666667, 28.6666666667);--hb-color-crimson-dark-rgb: 177.7333333333, 28.6666666667, 28.6666666667;--hb-color-ruby: rgb(22, 163, 74);--hb-color-ruby-rgb: 22, 163, 74;--hb-color-ruby-dark: rgb(17.6, 130.4, 59.2);--hb-color-ruby-dark-rgb: 17.6, 130.4, 59.2;--hb-color-spring: rgb(0, 230, 118);--hb-color-spring-rgb: 0, 230, 118;--hb-color-spring-dark: rgb(0, 184, 94.4);--hb-color-spring-dark-rgb: 0, 184, 94.4;--hb-color-apple: rgb(46, 204, 113);--hb-color-apple-rgb: 46, 204, 113;--hb-color-apple-dark: rgb(36.8, 163.2, 90.4);--hb-color-apple-dark-rgb: 36.8, 163.2, 90.4;--hb-color-lime: rgb(203, 220, 56);--hb-color-lime-rgb: 203, 220, 56;--hb-color-lime-dark: rgb(171.7333333333, 187.7743589744, 33.0256410256);--hb-color-lime-dark-rgb: 171.7333333333, 187.7743589744, 33.0256410256;--hb-color-emerald: rgb(15, 184, 129);--hb-color-emerald-rgb: 15, 184, 129;--hb-color-emerald-dark: rgb(12, 147.2, 103.2);--hb-color-emerald-dark-rgb: 12, 147.2, 103.2;--hb-color-teal: rgb(0, 150, 136);--hb-color-teal-rgb: 0, 150, 136;--hb-color-teal-dark: rgb(0, 120, 108.8);--hb-color-teal-dark-rgb: 0, 120, 108.8;--hb-color-petrol: rgb(0, 128, 128);--hb-color-petrol-rgb: 0, 128, 128;--hb-color-petrol-dark: rgb(0, 102.4, 102.4);--hb-color-petrol-dark-rgb: 0, 102.4, 102.4;--hb-color-wine: rgb(114, 47, 55);--hb-color-wine-rgb: 114, 47, 55;--hb-color-wine-dark: rgb(91.2, 37.6, 44);--hb-color-wine-dark-rgb: 91.2, 37.6, 44;--hb-color-mulberry: rgb(155, 34, 66);--hb-color-mulberry-rgb: 155, 34, 66;--hb-color-mulberry-dark: rgb(124, 27.2, 52.8);--hb-color-mulberry-dark-rgb: 124, 27.2, 52.8;--hb-color-fern: rgb(98, 139, 72);--hb-color-fern-rgb: 98, 139, 72;--hb-color-fern-dark: rgb(78.4, 111.2, 57.6);--hb-color-fern-dark-rgb: 78.4, 111.2, 57.6;--hb-color-honey: rgb(246, 190, 79);--hb-color-honey-rgb: 246, 190, 79;--hb-color-honey-dark: rgb(242.8378378378, 167.1621621622, 17.1621621622);--hb-color-honey-dark-rgb: 242.8378378378, 167.1621621622, 17.1621621622;--hb-color-lemon: rgb(255, 241, 118);--hb-color-lemon-rgb: 255, 241, 118;--hb-color-lemon-dark: rgb(255, 233.3766423358, 43.4);--hb-color-lemon-dark-rgb: 255, 233.3766423358, 43.4;--hb-color-gold: rgb(255, 181, 71);--hb-color-gold-rgb: 255, 181, 71;--hb-color-gold-dark: rgb(255, 154.7782608696, 5.8);--hb-color-gold-dark-rgb: 255, 154.7782608696, 5.8;--hb-color-amber: rgb(255, 193, 7);--hb-color-amber-rgb: 255, 193, 7;--hb-color-amber-dark: rgb(209.6, 157.2, 0);--hb-color-amber-dark-rgb: 209.6, 157.2, 0;--hb-color-coral: rgb(255, 127, 80);--hb-color-coral-rgb: 255, 127, 80;--hb-color-coral-dark: rgb(255, 77.9942857143, 13);--hb-color-coral-dark-rgb: 255, 77.9942857143, 13;--hb-color-orange: rgb(244, 158, 11);--hb-color-orange-rgb: 244, 158, 11;--hb-color-orange-dark: rgb(195.2, 126.4, 8.8);--hb-color-orange-dark-rgb: 195.2, 126.4, 8.8;--hb-color-caramel: rgb(199, 122, 53);--hb-color-caramel-rgb: 199, 122, 53;--hb-color-caramel-dark: rgb(159.2, 97.6, 42.4);--hb-color-caramel-dark-rgb: 159.2, 97.6, 42.4;--hb-color-stone: rgb(120, 113, 108);--hb-color-stone-rgb: 120, 113, 108;--hb-color-stone-dark: rgb(96, 90.4, 86.4);--hb-color-stone-dark-rgb: 96, 90.4, 86.4;--hb-color-graphite: rgb(61, 61, 61);--hb-color-graphite-rgb: 61, 61, 61;--hb-color-graphite-dark: rgb(48.8, 48.8, 48.8);--hb-color-graphite-dark-rgb: 48.8, 48.8, 48.8;--hb-color-charcoal: rgb(75, 85, 99);--hb-color-charcoal-rgb: 75, 85, 99;--hb-color-charcoal-dark: rgb(60, 68, 79.2);--hb-color-charcoal-dark-rgb: 60, 68, 79.2;--hb-color-storm: rgb(71, 85, 105);--hb-color-storm-rgb: 71, 85, 105;--hb-color-storm-dark: rgb(56.8, 68, 84);--hb-color-storm-dark-rgb: 56.8, 68, 84;--hb-color-smoke: rgb(115, 115, 115);--hb-color-smoke-rgb: 115, 115, 115;--hb-color-smoke-dark: #5c5c5c;--hb-color-smoke-dark-rgb: 92, 92, 92;--hb-color-shadow: rgb(31, 41, 55);--hb-color-shadow-rgb: 31, 41, 55;--hb-color-shadow-dark: rgb(24.8, 32.8, 44);--hb-color-shadow-dark-rgb: 24.8, 32.8, 44;--hb-color-steel: rgb(107, 114, 128);--hb-color-steel-rgb: 107, 114, 128;--hb-color-steel-dark: rgb(85.6, 91.2, 102.4);--hb-color-steel-dark-rgb: 85.6, 91.2, 102.4;--hb-color-peach: rgb(255, 188, 154);--hb-color-peach-rgb: 255, 188, 154;--hb-color-peach-dark: rgb(255, 133.7366336634, 72.2);--hb-color-peach-dark-rgb: 255, 133.7366336634, 72.2;--hb-color-sunset: rgb(251, 113, 133);--hb-color-sunset-rgb: 251, 113, 133;--hb-color-sunset-dark: rgb(249.0054794521, 42.1945205479, 72.1671232877);--hb-color-sunset-dark-rgb: 249.0054794521, 42.1945205479, 72.1671232877;--hb-color-chartreuse: rgb(167, 244, 50);--hb-color-chartreuse-rgb: 167, 244, 50;--hb-color-chartreuse-dark: rgb(139.3777777778, 223.2222222222, 11.9777777778);--hb-color-chartreuse-dark-rgb: 139.3777777778, 223.2222222222, 11.9777777778;--hb-color-jade: rgb(0, 191, 165);--hb-color-jade-rgb: 0, 191, 165;--hb-color-jade-dark: rgb(0, 152.8, 132);--hb-color-jade-dark-rgb: 0, 152.8, 132}.ui-color-brand-github{color:var(--hb-color-brand-github)}.ui-color-bg-brand-github{background-color:var(--hb-color-brand-github);color:#fff}.ui-color-bg-gradient-brand-github{background:linear-gradient(to right, var(--hb-color-brand-github), var(--hb-color-brand-github-dark));color:#fff}.ui-color-brand-docker{color:var(--hb-color-brand-docker)}.ui-color-bg-brand-docker{background-color:var(--hb-color-brand-docker);color:#fff}.ui-color-bg-gradient-brand-docker{background:linear-gradient(to right, var(--hb-color-brand-docker), var(--hb-color-brand-docker-dark));color:#fff}.ui-color-brand-ubuntu{color:var(--hb-color-brand-ubuntu)}.ui-color-bg-brand-ubuntu{background-color:var(--hb-color-brand-ubuntu);color:#fff}.ui-color-bg-gradient-brand-ubuntu{background:linear-gradient(to right, var(--hb-color-brand-ubuntu), var(--hb-color-brand-ubuntu-dark));color:#fff}.ui-color-aqua{color:var(--hb-color-aqua)}.ui-color-bg-aqua{background-color:var(--hb-color-aqua);color:#fff}.ui-color-bg-gradient-aqua{background:linear-gradient(to right, var(--hb-color-aqua), var(--hb-color-aqua-dark));color:#fff}.ui-color-amethyst{color:var(--hb-color-amethyst)}.ui-color-bg-amethyst{background-color:var(--hb-color-amethyst);color:#fff}.ui-color-bg-gradient-amethyst{background:linear-gradient(to right, var(--hb-color-amethyst), var(--hb-color-amethyst-dark));color:#fff}.ui-color-indigo{color:var(--hb-color-indigo)}.ui-color-bg-indigo{background-color:var(--hb-color-indigo);color:#fff}.ui-color-bg-gradient-indigo{background:linear-gradient(to right, var(--hb-color-indigo), var(--hb-color-indigo-dark));color:#fff}.ui-color-denim{color:var(--hb-color-denim)}.ui-color-bg-denim{background-color:var(--hb-color-denim);color:#fff}.ui-color-bg-gradient-denim{background:linear-gradient(to right, var(--hb-color-denim), var(--hb-color-denim-dark));color:#fff}.ui-color-cerulean{color:var(--hb-color-cerulean)}.ui-color-bg-cerulean{background-color:var(--hb-color-cerulean);color:#fff}.ui-color-bg-gradient-cerulean{background:linear-gradient(to right, var(--hb-color-cerulean), var(--hb-color-cerulean-dark));color:#fff}.ui-color-ocean{color:var(--hb-color-ocean)}.ui-color-bg-ocean{background-color:var(--hb-color-ocean);color:#fff}.ui-color-bg-gradient-ocean{background:linear-gradient(to right, var(--hb-color-ocean), var(--hb-color-ocean-dark));color:#fff}.ui-color-purple{color:var(--hb-color-purple)}.ui-color-bg-purple{background-color:var(--hb-color-purple);color:#fff}.ui-color-bg-gradient-purple{background:linear-gradient(to right, var(--hb-color-purple), var(--hb-color-purple-dark));color:#fff}.ui-color-azure{color:var(--hb-color-azure)}.ui-color-bg-azure{background-color:var(--hb-color-azure);color:#000}.ui-color-bg-gradient-azure{background:linear-gradient(to right, var(--hb-color-azure), var(--hb-color-azure-dark));color:#000}.ui-color-lavender{color:var(--hb-color-lavender)}.ui-color-bg-lavender{background-color:var(--hb-color-lavender);color:#000}.ui-color-bg-gradient-lavender{background:linear-gradient(to right, var(--hb-color-lavender), var(--hb-color-lavender-dark));color:#000}.ui-color-plum{color:var(--hb-color-plum)}.ui-color-bg-plum{background-color:var(--hb-color-plum);color:#fff}.ui-color-bg-gradient-plum{background:linear-gradient(to right, var(--hb-color-plum), var(--hb-color-plum-dark));color:#fff}.ui-color-rose{color:var(--hb-color-rose)}.ui-color-bg-rose{background-color:var(--hb-color-rose);color:#000}.ui-color-bg-gradient-rose{background:linear-gradient(to right, var(--hb-color-rose), var(--hb-color-rose-dark));color:#000}.ui-color-pink{color:var(--hb-color-pink)}.ui-color-bg-pink{background-color:var(--hb-color-pink);color:#fff}.ui-color-bg-gradient-pink{background:linear-gradient(to right, var(--hb-color-pink), var(--hb-color-pink-dark));color:#fff}.ui-color-crimson{color:var(--hb-color-crimson)}.ui-color-bg-crimson{background-color:var(--hb-color-crimson);color:#fff}.ui-color-bg-gradient-crimson{background:linear-gradient(to right, var(--hb-color-crimson), var(--hb-color-crimson-dark));color:#fff}.ui-color-ruby{color:var(--hb-color-ruby)}.ui-color-bg-ruby{background-color:var(--hb-color-ruby);color:#fff}.ui-color-bg-gradient-ruby{background:linear-gradient(to right, var(--hb-color-ruby), var(--hb-color-ruby-dark));color:#fff}.ui-color-spring{color:var(--hb-color-spring)}.ui-color-bg-spring{background-color:var(--hb-color-spring);color:#fff}.ui-color-bg-gradient-spring{background:linear-gradient(to right, var(--hb-color-spring), var(--hb-color-spring-dark));color:#fff}.ui-color-apple{color:var(--hb-color-apple)}.ui-color-bg-apple{background-color:var(--hb-color-apple);color:#fff}.ui-color-bg-gradient-apple{background:linear-gradient(to right, var(--hb-color-apple), var(--hb-color-apple-dark));color:#fff}.ui-color-lime{color:var(--hb-color-lime)}.ui-color-bg-lime{background-color:var(--hb-color-lime);color:#fff}.ui-color-bg-gradient-lime{background:linear-gradient(to right, var(--hb-color-lime), var(--hb-color-lime-dark));color:#fff}.ui-color-emerald{color:var(--hb-color-emerald)}.ui-color-bg-emerald{background-color:var(--hb-color-emerald);color:#fff}.ui-color-bg-gradient-emerald{background:linear-gradient(to right, var(--hb-color-emerald), var(--hb-color-emerald-dark));color:#fff}.ui-color-teal{color:var(--hb-color-teal)}.ui-color-bg-teal{background-color:var(--hb-color-teal);color:#fff}.ui-color-bg-gradient-teal{background:linear-gradient(to right, var(--hb-color-teal), var(--hb-color-teal-dark));color:#fff}.ui-color-petrol{color:var(--hb-color-petrol)}.ui-color-bg-petrol{background-color:var(--hb-color-petrol);color:#fff}.ui-color-bg-gradient-petrol{background:linear-gradient(to right, var(--hb-color-petrol), var(--hb-color-petrol-dark));color:#fff}.ui-color-wine{color:var(--hb-color-wine)}.ui-color-bg-wine{background-color:var(--hb-color-wine);color:#fff}.ui-color-bg-gradient-wine{background:linear-gradient(to right, var(--hb-color-wine), var(--hb-color-wine-dark));color:#fff}.ui-color-mulberry{color:var(--hb-color-mulberry)}.ui-color-bg-mulberry{background-color:var(--hb-color-mulberry);color:#fff}.ui-color-bg-gradient-mulberry{background:linear-gradient(to right, var(--hb-color-mulberry), var(--hb-color-mulberry-dark));color:#fff}.ui-color-fern{color:var(--hb-color-fern)}.ui-color-bg-fern{background-color:var(--hb-color-fern);color:#fff}.ui-color-bg-gradient-fern{background:linear-gradient(to right, var(--hb-color-fern), var(--hb-color-fern-dark));color:#fff}.ui-color-honey{color:var(--hb-color-honey)}.ui-color-bg-honey{background-color:var(--hb-color-honey);color:#000}.ui-color-bg-gradient-honey{background:linear-gradient(to right, var(--hb-color-honey), var(--hb-color-honey-dark));color:#000}.ui-color-lemon{color:var(--hb-color-lemon)}.ui-color-bg-lemon{background-color:var(--hb-color-lemon);color:#000}.ui-color-bg-gradient-lemon{background:linear-gradient(to right, var(--hb-color-lemon), var(--hb-color-lemon-dark));color:#000}.ui-color-gold{color:var(--hb-color-gold)}.ui-color-bg-gold{background-color:var(--hb-color-gold);color:#000}.ui-color-bg-gradient-gold{background:linear-gradient(to right, var(--hb-color-gold), var(--hb-color-gold-dark));color:#000}.ui-color-amber{color:var(--hb-color-amber)}.ui-color-bg-amber{background-color:var(--hb-color-amber);color:#fff}.ui-color-bg-gradient-amber{background:linear-gradient(to right, var(--hb-color-amber), var(--hb-color-amber-dark));color:#fff}.ui-color-coral{color:var(--hb-color-coral)}.ui-color-bg-coral{background-color:var(--hb-color-coral);color:#000}.ui-color-bg-gradient-coral{background:linear-gradient(to right, var(--hb-color-coral), var(--hb-color-coral-dark));color:#000}.ui-color-orange{color:var(--hb-color-orange)}.ui-color-bg-orange{background-color:var(--hb-color-orange);color:#fff}.ui-color-bg-gradient-orange{background:linear-gradient(to right, var(--hb-color-orange), var(--hb-color-orange-dark));color:#fff}.ui-color-caramel{color:var(--hb-color-caramel)}.ui-color-bg-caramel{background-color:var(--hb-color-caramel);color:#fff}.ui-color-bg-gradient-caramel{background:linear-gradient(to right, var(--hb-color-caramel), var(--hb-color-caramel-dark));color:#fff}.ui-color-stone{color:var(--hb-color-stone)}.ui-color-bg-stone{background-color:var(--hb-color-stone);color:#fff}.ui-color-bg-gradient-stone{background:linear-gradient(to right, var(--hb-color-stone), var(--hb-color-stone-dark));color:#fff}.ui-color-graphite{color:var(--hb-color-graphite)}.ui-color-bg-graphite{background-color:var(--hb-color-graphite);color:#fff}.ui-color-bg-gradient-graphite{background:linear-gradient(to right, var(--hb-color-graphite), var(--hb-color-graphite-dark));color:#fff}.ui-color-charcoal{color:var(--hb-color-charcoal)}.ui-color-bg-charcoal{background-color:var(--hb-color-charcoal);color:#fff}.ui-color-bg-gradient-charcoal{background:linear-gradient(to right, var(--hb-color-charcoal), var(--hb-color-charcoal-dark));color:#fff}.ui-color-storm{color:var(--hb-color-storm)}.ui-color-bg-storm{background-color:var(--hb-color-storm);color:#fff}.ui-color-bg-gradient-storm{background:linear-gradient(to right, var(--hb-color-storm), var(--hb-color-storm-dark));color:#fff}.ui-color-smoke{color:var(--hb-color-smoke)}.ui-color-bg-smoke{background-color:var(--hb-color-smoke);color:#fff}.ui-color-bg-gradient-smoke{background:linear-gradient(to right, var(--hb-color-smoke), var(--hb-color-smoke-dark));color:#fff}.ui-color-shadow{color:var(--hb-color-shadow)}.ui-color-bg-shadow{background-color:var(--hb-color-shadow);color:#fff}.ui-color-bg-gradient-shadow{background:linear-gradient(to right, var(--hb-color-shadow), var(--hb-color-shadow-dark));color:#fff}.ui-color-steel{color:var(--hb-color-steel)}.ui-color-bg-steel{background-color:var(--hb-color-steel);color:#fff}.ui-color-bg-gradient-steel{background:linear-gradient(to right, var(--hb-color-steel), var(--hb-color-steel-dark));color:#fff}.ui-color-peach{color:var(--hb-color-peach)}.ui-color-bg-peach{background-color:var(--hb-color-peach);color:#000}.ui-color-bg-gradient-peach{background:linear-gradient(to right, var(--hb-color-peach), var(--hb-color-peach-dark));color:#000}.ui-color-sunset{color:var(--hb-color-sunset)}.ui-color-bg-sunset{background-color:var(--hb-color-sunset);color:#000}.ui-color-bg-gradient-sunset{background:linear-gradient(to right, var(--hb-color-sunset), var(--hb-color-sunset-dark));color:#000}.ui-color-chartreuse{color:var(--hb-color-chartreuse)}.ui-color-bg-chartreuse{background-color:var(--hb-color-chartreuse);color:#fff}.ui-color-bg-gradient-chartreuse{background:linear-gradient(to right, var(--hb-color-chartreuse), var(--hb-color-chartreuse-dark));color:#fff}.ui-color-jade{color:var(--hb-color-jade)}.ui-color-bg-jade{background-color:var(--hb-color-jade);color:#fff}.ui-color-bg-gradient-jade{background:linear-gradient(to right, var(--hb-color-jade), var(--hb-color-jade-dark));color:#fff}:root{--hb-settings-color-instance-name: #2563EB;--hb-settings-color-default-language: #16A34A;--hb-settings-color-user-actions-disable: #FFA726;--hb-settings-color-user-actions-delete: #E53935}.text-dark{color:var(--mud-palette-text-primary)}.mud-navmenu.mud-navmenu-first-level>.mud-nav-item>.mud-nav-link{border-radius:inherit}.mud-navmenu.mud-navmenu-xxl .mud-nav-link{display:flex !important;align-items:center !important;padding:8px 8px !important}.mud-navmenu.mud-navmenu-xxl .mud-icon-root{width:2rem;height:2rem}.mud-navmenu.mud-navmenu-xxl .mud-nav-link-text{font-size:1rem;color:var(--mud-palette-primary)}.mud-paper>.mud-list>.mud-list-item:first-child{border-top-left-radius:var(--mud-default-borderradius);border-top-right-radius:var(--mud-default-borderradius)}.mud-paper>.mud-list>.mud-list-item:last-child{border-bottom-left-radius:var(--mud-default-borderradius);border-bottom-right-radius:var(--mud-default-borderradius)}.mud-paper>.mud-navmenu.mud-navmenu-first-level>.mud-nav-item:first-child{border-top-left-radius:var(--mud-default-borderradius);border-top-right-radius:var(--mud-default-borderradius)}.mud-paper>.mud-navmenu.mud-navmenu-first-level>.mud-nav-item:last-child{border-bottom-left-radius:var(--mud-default-borderradius);border-bottom-right-radius:var(--mud-default-borderradius)}.mud-drawer.frosted.mud-theme-primary{color:var(--mud-palette-primary-text) !important;background:rgb(from var(--mud-palette-primary) r g b/0.7) !important}.mud-drawer.frosted.mud-theme-primary::before{content:none !important}.mud-drawer.frosted.mud-theme-secondary{color:var(--mud-palette-secondary-text) !important;background:rgb(from var(--mud-palette-secondary) r g b/0.7) !important}.mud-drawer.frosted.mud-theme-secondary::before{content:none !important}.mud-drawer.frosted.mud-theme-tertiary{color:var(--mud-palette-tertiary-text) !important;background:rgb(from var(--mud-palette-tertiary) r g b/0.7) !important}.mud-drawer.frosted.mud-theme-tertiary::before{content:none !important}.mud-card>.mud-card-header{border-color:var(--mud-palette-divider);border-width:1px;border-style:solid none none none;padding:8px 12px}.mud-card>.mud-card-header .mud-card-header-actions{margin:0 !important;align-self:center}.mud-card>.mud-card-header .mud-typography{padding:4px !important}.mud-card>.mud-card-content{border-color:var(--mud-palette-divider);border-width:1px;border-style:solid none none none}.mud-card>.mud-card-actions{border-bottom-left-radius:var(--mud-default-borderradius);border-bottom-right-radius:var(--mud-default-borderradius);background-color:var(--mud-palette-background-gray)}.mud-card .no-divider{border:none}.mud-stepper.stepper-no-actions .mud-stepper-actions{display:none}.mud-file-upload .html-fileupload-hidden{position:absolute;width:100%;height:100%;overflow:hidden;z-index:10;opacity:0}.ui-wavebackground-container{width:100vw;height:100vh}.ui-wavebackground-container.ui-bg-01{background:linear-gradient(315deg, rgb(24, 19, 89) 2%, rgb(1, 55, 125) 25%, rgb(50, 150, 200) 45%, rgb(164, 236, 248) 70%, rgb(66, 111, 160) 98%);animation:gradient 150s ease infinite;background-size:400% 400%;background-attachment:fixed}.ui-wavebackground-container.ui-bg-02{background:url("/img/bg/ws25.jpg") no-repeat center center fixed;background-size:cover}.ui-wavebackground-container.ui-bg-03{background:url("/img/bg/wp12118355.webp") no-repeat center center fixed;background-size:cover}.ui-wavebackground-container.ui-bg-04{background:url("/img/bg/wp12118355-light.webp") no-repeat center center fixed;background-size:cover}.ui-wavebackground-container .wave{background:hsla(0,0%,100%,.25);border-radius:1000% 1000% 0 0;position:fixed;width:400%;height:30em;animation:wave 30s -3s linear infinite;transform:translate3d(0, 0, 0);opacity:.8;bottom:0;left:0;z-index:-100}.ui-wavebackground-container .wave:nth-of-type(2){bottom:-1.25em;animation:wave 54s linear reverse infinite;opacity:.8}.ui-wavebackground-container .wave:nth-of-type(3){bottom:-2.5em;animation:wave 60s -1s reverse infinite;opacity:.9}@keyframes gradient{0%{background-position:0% 0%}50%{background-position:100% 100%}100%{background-position:0% 0%}}@keyframes wave{2%{transform:translateX(1)}25%{transform:translateX(-25%)}50%{transform:translateX(-50%)}75%{transform:translateX(-25%)}100%{transform:translateX(1)}}.ui-stripe-background-container{width:100vw;height:100vh}#ui-stripe-background-canvas{width:inherit;height:inherit}#ui-stripe-background-canvas.build-mode-alpha{--gradient-color-1: #003a7a;--gradient-color-2: #003f90;--gradient-color-3: #00366e;--gradient-color-4: #002790;--gradient-color-5: #005fe0}#ui-stripe-background-canvas.build-mode-beta{--gradient-color-1: #1b5d8a;--gradient-color-2: #1c5d89;--gradient-color-3: #009ac7;--gradient-color-4: #114d7b;--gradient-color-5: #26a4d2}#ui-stripe-background-canvas.build-mode-release{--gradient-color-1: #7ebbfc;--gradient-color-2: #3366ff;--gradient-color-3: #1340c8;--gradient-color-4: #0029a3;--gradient-color-5: #1c1d7c}.ui-countdown-alert{overflow:hidden;position:relative;border-radius:var(--mud-default-borderradius)}.ui-countdown-alert .mud-progress-linear{position:absolute;bottom:0;left:0;width:100%}.ui-dev-icons-container .mud-icon-root{font-size:60px}:root{--cell-size: 170px;--cell-size: 73px;--cell-gap: 24px;--widget-border-radius: 12px;--ui-widget-drawer-width: calc((var(--cell-size) * 8) + (var(--cell-gap) * 7) + var(--cell-gap));--hb-widget-padding: calc(var(--widget-border-radius) / 3)}.hb-widget-container{height:inherit;width:inherit}.hb-widget-grid{display:grid;gap:var(--cell-gap);grid-template-columns:repeat(auto-fill, var(--cell-size));grid-template-rows:repeat(auto-fill, var(--cell-size));grid-auto-rows:var(--cell-size);width:100%;height:100%}.ui-widget-list{display:flex;flex-direction:column;row-gap:var(--cell-gap);padding:calc(var(--cell-gap)/2)}.ui-widget{overflow:hidden;display:flex;flex-direction:column}.ui-widget>*{flex:1 1 auto}.ui-widget.w-2{width:calc(var(--cell-size)*2 + var(--cell-gap)*1)}.ui-widget.w-4{width:calc(var(--cell-size)*4 + var(--cell-gap)*3)}.ui-widget.w-8{width:calc(var(--cell-size)*8 + var(--cell-gap)*7)}.ui-widget.h-1{height:calc(var(--cell-size)*1 + var(--cell-gap)*0)}.ui-widget.h-2{height:calc(var(--cell-size)*2 + var(--cell-gap)*1)}.ui-widget.h-4{height:calc(var(--cell-size)*4 + var(--cell-gap)*3)}.hb-widget-selection-drawer{width:var(--ui-widget-drawer-width)}.ui-icon>.ui-icon-el{border-radius:var(--hb-ui-icon-border-radius, 0)}.ui-icon>.ui-icon-el.ui-icon-filled{color:#fff;padding:2px}.ui-icon>.ui-icon-el.ui-icon-outlined{border-width:1px;border-style:solid;padding:1px}.ui-icon>.ui-icon-el.ui-icon-outlined.mud-icon-size-large{border-width:2px}.ui-settings-item .ui-settings-item-content{width:100%}@media(min-width: 960px){.ui-settings-item .ui-settings-item-content{width:240px}}.ui-startmenu-item-link{background:var(--hb-startmenu-container-background);width:100%}.ui-startmenu-item-link>span{height:100%}.ui-startmenu-container{width:100%;height:inherit;overflow:hidden}.ui-startmenu-item-background-icon{height:var(--hb-startmenu-background-icon-size);width:var(--hb-startmenu-background-icon-size);opacity:var(--hb-startmenu-background-icon-opacity);position:absolute;z-index:-1;top:-25px;left:-25px}.ui-startmenu-item-content{height:inherit}.ui-startmenu-item-content .ui-startmenu-item-nav{margin-top:auto !important}.ui-progress-item{display:flex;flex-direction:column;width:100%}.ui-progress-item .ui-progress-item-header,.ui-progress-item .ui-progress-item-progress,.ui-progress-item .ui-progress-item-footer{display:flex;width:100%}.ui-progress-item .ui-progress-item-header,.ui-progress-item .ui-progress-item-footer{align-items:center}.ui-progress-item .ui-progress-item-header .ui-progress-item-header-start,.ui-progress-item .ui-progress-item-header .ui-progress-item-footer-start,.ui-progress-item .ui-progress-item-footer .ui-progress-item-header-start,.ui-progress-item .ui-progress-item-footer .ui-progress-item-footer-start{flex:1 1 auto;min-width:0}.ui-progress-item .ui-progress-item-header .ui-progress-item-header-end,.ui-progress-item .ui-progress-item-header .ui-progress-item-footer-end,.ui-progress-item .ui-progress-item-footer .ui-progress-item-header-end,.ui-progress-item .ui-progress-item-footer .ui-progress-item-footer-end{flex:0 0 auto}.ui-progress-item .ui-progress-item-progress{width:100%}.ui-progress-item .ui-progress-item-progress .mud-progress-linear-bar{background-color:var(--ui-element-accent-color) !important}.ui-colored-icon-container .ui-colored-icon-frame{background-color:color-mix(in srgb, var(--ui-colored-icon-color), transparent var(--hb-ui-colored-icon-background-opacity));border-radius:var(--hb-ui-colored-icon-borderradius)}.ui-colored-icon-container .ui-colored-icon{color:var(--ui-colored-icon-color);height:var(--hb-ui-colored-icon-size);width:var(--hb-ui-colored-icon-size)}.ui-detail-card{color:inherit}.ui-detail-card-container{color:inherit}.ui-detail-card-container .ui-detail-card-content{color:inherit}.ui-detail-card-container .ui-detail-card-header{color:inherit}.ui-detail-card-container .ui-detail-card-footer{color:inherit}.ui-value-header{color:inherit}.ui-value-header .ui-value-card-title-text{color:var(--ui-value-card-caption-color)}.ui-value-header .ui-value-display-text{font-weight:500}.ui-value-footer-text{color:inherit}.ui-value-footer-text .ui-value-footer-highlighted-text{color:var(--ui-value-card-accent-color)}.ui-value-footer-text .ui-value-footer-notice-text{color:var(--ui-value-card-caption-color)}.hb-drawer{border-radius:12px;margin:8px;height:calc(100vh - 16px) !important;width:calc(var(--mud-drawer-width, var(--mud-drawer-width-left)) - 16px) !important;top:0px}.hb-header-appbar{border-radius:12px;margin-top:8px !important;width:calc(100% - 8px - var(--mud-drawer-width-left)) !important}.hb-footer-appbar{border-radius:12px;margin-bottom:8px;margin-right:8px !important;width:calc(100% - 8px - var(--mud-drawer-width-left)) !important}@media(min-width: 600px){.hb-main-content{padding-top:calc(var(--mud-appbar-height) + 16px);padding-bottom:calc(var(--mud-appbar-height) + 16px)}}.ui-wallpaper{position:fixed;top:0;left:0;width:100vw;height:100vh;z-index:-1;background-size:cover;background-position:center}.hb-setup-container{height:100vh;padding-top:120px;padding-bottom:120px;transition:opacity 3s ease}.hb-setup-container.is-finished{opacity:0}.hb-setup-card{height:100%;overflow-y:hidden}.hb-license-agreement-content{overflow-y:auto}.setup-background{transition:all 2s ease;position:absolute;top:0;left:0;width:100vw;height:100vh;z-index:-100;margin:auto;overflow:hidden}.setup-background.is-finished{opacity:0}.setup-tiled-container{position:absolute;top:0;left:0;width:100vw;height:100vh;z-index:-90;overflow:hidden;transition:opacity 4s ease}.setup-tiled-container.is-hidden{opacity:0}.setup-tiled-container .setup-tiled-background{position:relative;width:150vw;height:150vh;left:-25vw;top:-25vh;transform:rotate(10deg);display:grid;grid-template-columns:1fr 1fr;grid-template-rows:1fr 1fr}.setup-tiled-container .setup-tiled-background .bg-tile{transition:all 6s ease;position:relative;background:radial-gradient(ellipse at center, rgba(255, 255, 255, 0.5) 0%, rgba(255, 255, 255, 0) 75%)}.setup-tiled-container.is-finished .bg-tile:nth-child(1){transform:translate(-200vw, -200px)}.setup-tiled-container.is-finished .bg-tile:nth-child(2){transform:translate(200px, -200vh)}.setup-tiled-container.is-finished .bg-tile:nth-child(3){transform:translate(-200px, 200vh)}.setup-tiled-container.is-finished .bg-tile:nth-child(4){transform:translate(200vw, 200px)}.hb-account-login-container{height:100vh;padding-top:120px;padding-bottom:120px;transition:opacity 3s ease}.hb-account-login-card{height:100%;overflow-y:hidden}.account-login-background{transition:all 2s ease;position:absolute;top:0;left:0;width:100vw;height:100vh;z-index:-100;margin:auto;overflow:hidden}.account-login-background.is-finished{opacity:0}/*# sourceMappingURL=app.min.css.map */ diff --git a/source/HomeBook.Frontend/wwwroot/css/app.min.css.map b/source/HomeBook.Frontend/wwwroot/css/app.min.css.map index aced992c..69dcce38 100644 --- a/source/HomeBook.Frontend/wwwroot/css/app.min.css.map +++ b/source/HomeBook.Frontend/wwwroot/css/app.min.css.map @@ -1 +1 @@ -{"version":3,"sourceRoot":"","sources":["../../Styles/_blazorbase.scss","../../Styles/variables/_breakpoints.scss","../../Styles/variables/_frosted-ui.scss","../../Styles/variables/_border-radius.scss","../../Styles/variables/_ui-colored-icon.scss","../../Styles/variables/_ui-startmenu-item.scss","../../Styles/utilities/_width.scss","../../Styles/utilities/_float.scss","../../Styles/styles/_frosted.scss","../../Styles/styles/_colors.scss","../../Styles/styles/_color-palette.scss","../../Styles/styles/_settings-colors.scss","../../Styles/styles/_typography.scss","../../Styles/components/_mud-nav-menu.scss","../../Styles/components/_mud-paper.scss","../../Styles/components/_mud-drawer.scss","../../Styles/components/_mud-card.scss","../../Styles/components/_mud-stepper.scss","../../Styles/components/_ui-wave-background.scss","../../Styles/components/_ui-stripe-background.scss","../../Styles/components/_ui-countdown-alert.scss","../../Styles/components/_ui-dev-icons.scss","../../Styles/components/_ui-widgets.scss","../../Styles/components/_ui-icon.scss","../../Styles/components/_ui-settings-item.scss","../../Styles/components/_ui-startmenu.scss","../../Styles/components/_ui-progress-item.scss","../../Styles/components/_ui-colored-icon.scss","../../Styles/components/_ui-detail-card.scss","../../Styles/components/_ui-value-card.scss","../../Styles/components/_ui-layout.scss","../../Styles/components/_ui-wallpaper.scss","../../Styles/views/_setup.scss","../../Styles/views/_login.scss"],"names":[],"mappings":"AAAA,iBACI,wBACA,mBACA,SACA,qCACA,sBACA,aACA,OACA,oCACA,eACA,WACA,aAGJ,0BACI,eACA,kBACA,aACA,UAGJ,uBACI,ipDACA,8BACA,WAGJ,8BACI,iCAGJ,kBACI,kBACA,cACA,WACA,YACA,2BAGJ,yBACI,UACA,eACA,mBACA,yBACA,yBAGJ,oCACI,eACA,uEACA,6CAGJ,uBACI,kBACA,kBACA,iBACA,wCAGJ,6BACI,sDChDJ,MACI,wBACA,0BACA,0BACA,2BACA,2BACA,4BACA,6BACA,8BCXJ,MACI,uCACA,6BACA,8BACA,uBACA,uBACA,gCACA,kCACA,gCACA,qCCjBJ,MACI,gCCCJ,MACI,6CACA,uCACA,iCCHJ,MACI,2CACA,8CACA,iDCHJ,yBACA,2BACA,2BACA,2BACA,6BACA,8BAeA,yBATE,4BACA,8BACA,8BACA,8BACA,gCACA,kCAQF,yBAbE,4BACA,8BACA,8BACA,8BACA,gCACA,kCAYF,0BAjBE,4BACA,8BACA,8BACA,8BACA,gCACA,kCAgBF,0BArBE,4BACA,8BACA,8BACA,8BACA,gCACA,kCAoBF,0BAzBE,6BACA,+BACA,+BACA,+BACA,iCACA,mCAwBF,0BA7BE,8BACA,gCACA,gCACA,gCACA,kCACA,oCA4BF,0BAjCE,+BACA,iCACA,iCACA,iCACA,mCACA,qCCpBF,aACI,uBAGJ,YACI,sBAGJ,YACI,sBC0DA,YA7DA,kBAEA,WACI,kNAaJ,0BACA,oCAEA,oBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,kNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAOJ,YA7DA,kBAEA,WACI,iNAaJ,0BACA,qCAEA,oBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,iNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAOJ,YA7DA,kBAEA,WACI,kNAaJ,0BACA,oCAEA,oBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,kNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAOJ,YA7DA,kBAEA,WACI,iNAaJ,0BACA,qCAEA,oBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,iNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAOJ,YA7DA,kBAEA,WACI,kNAaJ,0BACA,oCAEA,oBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAuCA,6BACI,qBAIR,mCArCA,WACI,kNAaJ,0BAEA,mDACI,WACA,kBACA,QACA,oBACA,kDACA,sBAOJ,YA7DA,kBAEA,WACI,iNAaJ,0BACA,qCAEA,oBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,iNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAOJ,YA7DA,kBAEA,WACI,kNAaJ,0BACA,oCAEA,oBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,kNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAOJ,YA7DA,kBAEA,WACI,iNAaJ,0BACA,qCAEA,oBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,iNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAOJ,YA7DA,kBAEA,WACI,kNAaJ,0BACA,oCAEA,oBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,kNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAOJ,aA7DA,kBAEA,WACI,iNAaJ,0BACA,qCAEA,qBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAuCA,8BACI,qBAIR,gBArCA,WACI,iNAaJ,0BAEA,wBACI,WACA,kBACA,QACA,oBACA,iDACA,sBCxDR,MACI,2BACA,2BACA,2BAGJ,WACI,oCACA,sBAGJ,aACI,yBAGJ,WACI,oCACA,sBAGJ,aACI,yBAGJ,WACI,oCACA,sBAGJ,aACI,yBCkBJ,MAEQ,yCACA,wCAIA,oDACA,mDANA,2CACA,0CAIA,iFACA,gFANA,0CACA,yCAIA,gFACA,+EANA,kCACA,iCAIA,yCACA,wCANA,uCACA,sCAIA,qDACA,oDANA,oCACA,mCAIA,+CACA,8CANA,oCACA,mCAIA,8CACA,6CANA,sCACA,qCAIA,4EACA,2EANA,mCACA,kCAIA,0CACA,yCANA,qCACA,oCAIA,2DACA,0DANA,oCACA,mCAIA,0EACA,yEANA,wCACA,uCAIA,oEACA,mEANA,mCACA,kCAIA,8CACA,6CANA,oCACA,mCAIA,wEACA,uEANA,kCACA,iCAIA,wEACA,uEANA,qCACA,oCAIA,2EACA,0EANA,kCACA,iCAIA,6CACA,4CANA,oCACA,mCAIA,0CACA,yCANA,oCACA,mCAIA,8CACA,6CANA,mCACA,kCAIA,yEACA,wEANA,sCACA,qCAIA,+CACA,8CANA,kCACA,iCAIA,yCACA,wCANA,oCACA,mCAIA,6CACA,4CANA,kCACA,iCAIA,0CACA,yCANA,sCACA,qCAIA,+CACA,8CANA,kCACA,iCAIA,6CACA,4CANA,oCACA,mCAIA,0EACA,yEANA,qCACA,oCAIA,sDACA,qDANA,mCACA,kCAIA,oDACA,mDANA,mCACA,kCAIA,4CACA,2CANA,oCACA,mCAIA,mDACA,kDANA,qCACA,oCAIA,+CACA,8CANA,sCACA,qCAIA,gDACA,+CANA,qCACA,oCAIA,2CACA,0CANA,qCACA,oCAIA,gDACA,+CANA,qCACA,oCAIA,4CACA,2CANA,mCACA,kCAIA,yCACA,wCANA,qCACA,oCAIA,+BACA,sCANA,mCACA,kCAIA,4CACA,2CANA,qCACA,oCAIA,8CACA,6CANA,qCACA,oCAIA,sDACA,qDANA,sCACA,qCAIA,0EACA,yEANA,yCACA,wCAIA,+EACA,8EANA,kCACA,iCAIA,yCACA,wCAeJ,uBACI,mCAGJ,0BACI,8CACA,MAZY,KAehB,mCACI,sGAKA,MArBY,KAMhB,uBACI,mCAGJ,0BACI,8CACA,MAZY,KAehB,mCACI,sGAKA,MArBY,KAMhB,uBACI,mCAGJ,0BACI,8CACA,MAZY,KAehB,mCACI,sGAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,mBACI,+BAGJ,sBACI,0CACA,MAZY,KAehB,+BACI,8FAKA,MArBY,KAMhB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,mBACI,+BAGJ,sBACI,0CACA,MAZY,KAehB,+BACI,8FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MATgB,KAYpB,4BACI,wFAKA,MAlBgB,KAGpB,mBACI,+BAGJ,sBACI,0CACA,MATgB,KAYpB,+BACI,8FAKA,MAlBgB,KAGpB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MATgB,KAYpB,2BACI,sFAKA,MAlBgB,KAGpB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,kBACI,8BAGJ,qBACI,yCACA,MAZY,KAehB,8BACI,4FAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,kBACI,8BAGJ,qBACI,yCACA,MAZY,KAehB,8BACI,4FAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,mBACI,+BAGJ,sBACI,0CACA,MAZY,KAehB,+BACI,8FAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MATgB,KAYpB,4BACI,wFAKA,MAlBgB,KAGpB,gBACI,4BAGJ,mBACI,uCACA,MATgB,KAYpB,4BACI,wFAKA,MAlBgB,KAGpB,eACI,2BAGJ,kBACI,sCACA,MATgB,KAYpB,2BACI,sFAKA,MAlBgB,KAGpB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MATgB,KAYpB,4BACI,wFAKA,MAlBgB,KAGpB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,kBACI,8BAGJ,qBACI,yCACA,MAZY,KAehB,8BACI,4FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,mBACI,+BAGJ,sBACI,0CACA,MAZY,KAehB,+BACI,8FAKA,MArBY,KAMhB,mBACI,+BAGJ,sBACI,0CACA,MAZY,KAehB,+BACI,8FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MATgB,KAYpB,4BACI,wFAKA,MAlBgB,KAGpB,iBACI,6BAGJ,oBACI,wCACA,MATgB,KAYpB,6BACI,0FAKA,MAlBgB,KAGpB,qBACI,iCAGJ,wBACI,4CACA,MAZY,KAehB,iCACI,kGAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KChEpB,MACI,2CACA,8CACA,kDACA,iDCTJ,WACI,sCCKQ,iEACI,sBAOR,2CACI,wBACA,8BACA,2BAGJ,4CACI,WACA,YAGJ,gDACI,eACA,iCCrBA,gDACI,uDACA,wDAGJ,+CACI,0DACA,2DAWA,0EACI,uDACA,wDAGJ,yEACI,0DACA,2DC1BZ,sCACI,iDACA,qEAEA,8CACI,wBAIR,wCACI,mDACA,uEAEA,gDACI,wBAIR,uCACI,kDACA,sEAEA,+CACI,wBC1BZ,2BACI,wCACA,iBACA,kCACA,iBAEA,oDACI,oBACA,kBAGJ,2CACI,uBAIR,4BACI,wCACA,iBACA,kCAGJ,4BACI,0DACA,2DACA,oDAGJ,sBACI,YC3BA,qDACI,aC0BZ,6BACI,YACA,aAEA,sCACI,WAlBc,uIAmBd,sCACA,0BACA,4BAGJ,sCACI,iEACA,sBAGJ,sCACI,wEACA,sBAGJ,sCACI,8EACA,sBAGJ,mCACI,+BACA,8BACA,eACA,MAnCK,KAoCL,OArCM,KAsCN,uCACA,+BACA,WACA,SACA,OACA,aAGJ,kDACI,eACA,2CACA,WAGJ,kDACI,cACA,wCACA,WAIR,oBACI,GACI,0BAEJ,IACI,8BAEJ,KACI,2BAIR,gBACI,GACI,wBAGJ,IACI,2BAGJ,IACI,2BAGJ,IACI,2BAGJ,KACI,yBClHR,gCACI,YACA,aAGJ,6BACI,cACA,eAEA,8CAII,4BACA,4BACA,4BACA,4BACA,4BAGJ,6CAII,4BACA,4BACA,4BACA,4BACA,4BAGJ,gDAII,4BACA,4BACA,4BACA,4BACA,4BCvCR,oBACI,gBACA,kBACA,8CAEA,yCACI,kBACA,SACA,OACA,WCPJ,uCACI,eCHR,MACI,mBACA,kBACA,iBACA,6BACA,iGAEA,2DAGJ,qBACI,eACA,cAGJ,gBACI,aACA,oBAGA,0DACA,uDAGA,gCAGA,WACA,YAGJ,gBACI,aACA,sBACA,wBACA,gCAGJ,WACI,gBACA,aACA,sBAEA,aACI,cAGJ,eACI,mDAGJ,eACI,mDAGJ,eACI,mDAGJ,eACI,oDAGJ,eACI,oDAGJ,eACI,oDAIR,4BAEI,oCC7DA,qBACI,iDAEA,oCACI,WACA,YAGJ,sCACI,iBACA,mBACA,YAEA,0DACI,iBCbZ,4CACI,WAEA,yBAHJ,4CAIQ,aCVZ,wBACI,oDACA,WAEA,6BACI,YAIR,wBACI,WACA,eACA,gBAGJ,mCACI,gDACA,+CACA,oDACA,kBACA,WACA,UACA,WAGJ,2BACI,eAUA,kDACI,2BC7CR,kBACI,aACA,sBACA,WAEA,mIAGI,aACA,WAGJ,sFAEI,mBAEA,wSAEI,cACA,YAGJ,gSAEI,cAIR,6CACI,WAEA,sEACI,2DC9BR,kDACI,4HACA,qDAGJ,4CACI,mCACA,sCACA,qCCVR,gBACI,cAGJ,0BACI,cAEA,kDACI,cAGJ,iDACI,cAGJ,iDACI,cChBR,iBACI,cAEA,2CACI,yCAGJ,wCACI,gBAIR,sBACI,cAEA,wDACI,wCAGJ,mDACI,yCCpBR,WACI,mBACA,WACA,qCACA,oFACA,QAGJ,kBACI,mBACA,0BACA,iEAGJ,kBACI,mBACA,kBACA,4BACA,iEAKA,yBAFJ,iBAGQ,kDACA,sDCzBR,cACI,eACA,MACA,OACA,YACA,aACA,WACA,sBACA,2BCRJ,oBACI,aACA,kBACA,qBACA,2BAEA,gCACI,UAIR,eACI,YACA,kBAOJ,8BACI,gBAGJ,kBACI,uBACA,kBACA,MACA,OACA,YACA,aACA,aACA,YACA,gBAEA,8BACI,UAIR,uBACI,kBACA,MACA,OACA,YACA,aACA,YACA,gBACA,2BAEA,iCACI,UAGJ,+CACI,kBACA,YACA,aACA,WACA,UAEA,wBACA,aACA,8BACA,2BAEA,wDACI,uBACA,kBACA,uGA4BA,yDACI,oCAGJ,yDACI,mCAGJ,yDACI,mCAGJ,yDACI,kCC9GhB,4BACI,aACA,kBACA,qBACA,2BAGJ,uBACI,YACA,kBAGJ,0BACI,uBACA,kBACA,MACA,OACA,YACA,aACA,aACA,YACA,gBAEA,sCACI","file":"app.min.css"} \ No newline at end of file +{"version":3,"sourceRoot":"","sources":["../../Styles/_blazorbase.scss","../../Styles/variables/_breakpoints.scss","../../Styles/variables/_frosted-ui.scss","../../Styles/variables/_border-radius.scss","../../Styles/variables/_ui-colored-icon.scss","../../Styles/variables/_ui-startmenu-item.scss","../../Styles/utilities/_width.scss","../../Styles/utilities/_float.scss","../../Styles/styles/_frosted.scss","../../Styles/styles/_colors.scss","../../Styles/styles/_color-palette.scss","../../Styles/styles/_settings-colors.scss","../../Styles/styles/_typography.scss","../../Styles/components/_mud-nav-menu.scss","../../Styles/components/_mud-paper.scss","../../Styles/components/_mud-drawer.scss","../../Styles/components/_mud-card.scss","../../Styles/components/_mud-stepper.scss","../../Styles/components/_mud-file-upload.scss","../../Styles/components/_ui-wave-background.scss","../../Styles/components/_ui-stripe-background.scss","../../Styles/components/_ui-countdown-alert.scss","../../Styles/components/_ui-dev-icons.scss","../../Styles/components/_ui-widgets.scss","../../Styles/components/_ui-icon.scss","../../Styles/components/_ui-settings-item.scss","../../Styles/components/_ui-startmenu.scss","../../Styles/components/_ui-progress-item.scss","../../Styles/components/_ui-colored-icon.scss","../../Styles/components/_ui-detail-card.scss","../../Styles/components/_ui-value-card.scss","../../Styles/components/_ui-layout.scss","../../Styles/components/_ui-wallpaper.scss","../../Styles/views/_setup.scss","../../Styles/views/_login.scss"],"names":[],"mappings":"AAAA,iBACI,wBACA,mBACA,SACA,qCACA,sBACA,aACA,OACA,oCACA,eACA,WACA,aAGJ,0BACI,eACA,kBACA,aACA,UAGJ,uBACI,ipDACA,8BACA,WAGJ,8BACI,iCAGJ,kBACI,kBACA,cACA,WACA,YACA,2BAGJ,yBACI,UACA,eACA,mBACA,yBACA,yBAGJ,oCACI,eACA,uEACA,6CAGJ,uBACI,kBACA,kBACA,iBACA,wCAGJ,6BACI,sDChDJ,MACI,wBACA,0BACA,0BACA,2BACA,2BACA,4BACA,6BACA,8BCXJ,MACI,uCACA,6BACA,8BACA,uBACA,uBACA,gCACA,kCACA,gCACA,qCCjBJ,MACI,gCCCJ,MACI,6CACA,uCACA,iCCHJ,MACI,2CACA,8CACA,iDCHJ,yBACA,2BACA,2BACA,2BACA,6BACA,8BAeA,yBATE,4BACA,8BACA,8BACA,8BACA,gCACA,kCAQF,yBAbE,4BACA,8BACA,8BACA,8BACA,gCACA,kCAYF,0BAjBE,4BACA,8BACA,8BACA,8BACA,gCACA,kCAgBF,0BArBE,4BACA,8BACA,8BACA,8BACA,gCACA,kCAoBF,0BAzBE,6BACA,+BACA,+BACA,+BACA,iCACA,mCAwBF,0BA7BE,8BACA,gCACA,gCACA,gCACA,kCACA,oCA4BF,0BAjCE,+BACA,iCACA,iCACA,iCACA,mCACA,qCCpBF,aACI,uBAGJ,YACI,sBAGJ,YACI,sBC0DA,YA7DA,kBAEA,WACI,kNAaJ,0BACA,oCAEA,oBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,kNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAOJ,YA7DA,kBAEA,WACI,iNAaJ,0BACA,qCAEA,oBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,iNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAOJ,YA7DA,kBAEA,WACI,kNAaJ,0BACA,oCAEA,oBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,kNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAOJ,YA7DA,kBAEA,WACI,iNAaJ,0BACA,qCAEA,oBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,iNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAOJ,YA7DA,kBAEA,WACI,kNAaJ,0BACA,oCAEA,oBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAuCA,6BACI,qBAIR,mCArCA,WACI,kNAaJ,0BAEA,mDACI,WACA,kBACA,QACA,oBACA,kDACA,sBAOJ,YA7DA,kBAEA,WACI,iNAaJ,0BACA,qCAEA,oBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,iNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAOJ,YA7DA,kBAEA,WACI,kNAaJ,0BACA,oCAEA,oBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,kNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAOJ,YA7DA,kBAEA,WACI,iNAaJ,0BACA,qCAEA,oBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,iNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAOJ,YA7DA,kBAEA,WACI,kNAaJ,0BACA,oCAEA,oBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAuCA,6BACI,qBAIR,eArCA,WACI,kNAaJ,0BAEA,uBACI,WACA,kBACA,QACA,oBACA,kDACA,sBAOJ,aA7DA,kBAEA,WACI,iNAaJ,0BACA,qCAEA,qBACI,WACA,kBACA,QACA,oBACA,iDACA,sBAuCA,8BACI,qBAIR,gBArCA,WACI,iNAaJ,0BAEA,wBACI,WACA,kBACA,QACA,oBACA,iDACA,sBCxDR,MACI,2BACA,2BACA,2BAGJ,WACI,oCACA,sBAGJ,aACI,yBAGJ,WACI,oCACA,sBAGJ,aACI,yBAGJ,WACI,oCACA,sBAGJ,aACI,yBCkBJ,MAEQ,yCACA,wCAIA,oDACA,mDANA,2CACA,0CAIA,iFACA,gFANA,0CACA,yCAIA,gFACA,+EANA,kCACA,iCAIA,yCACA,wCANA,uCACA,sCAIA,qDACA,oDANA,oCACA,mCAIA,+CACA,8CANA,oCACA,mCAIA,8CACA,6CANA,sCACA,qCAIA,4EACA,2EANA,mCACA,kCAIA,0CACA,yCANA,qCACA,oCAIA,2DACA,0DANA,oCACA,mCAIA,0EACA,yEANA,wCACA,uCAIA,oEACA,mEANA,mCACA,kCAIA,8CACA,6CANA,oCACA,mCAIA,wEACA,uEANA,kCACA,iCAIA,wEACA,uEANA,qCACA,oCAIA,2EACA,0EANA,kCACA,iCAIA,6CACA,4CANA,oCACA,mCAIA,0CACA,yCANA,oCACA,mCAIA,8CACA,6CANA,mCACA,kCAIA,yEACA,wEANA,sCACA,qCAIA,+CACA,8CANA,kCACA,iCAIA,yCACA,wCANA,oCACA,mCAIA,6CACA,4CANA,kCACA,iCAIA,0CACA,yCANA,sCACA,qCAIA,+CACA,8CANA,kCACA,iCAIA,6CACA,4CANA,oCACA,mCAIA,0EACA,yEANA,qCACA,oCAIA,sDACA,qDANA,mCACA,kCAIA,oDACA,mDANA,mCACA,kCAIA,4CACA,2CANA,oCACA,mCAIA,mDACA,kDANA,qCACA,oCAIA,+CACA,8CANA,sCACA,qCAIA,gDACA,+CANA,qCACA,oCAIA,2CACA,0CANA,qCACA,oCAIA,gDACA,+CANA,qCACA,oCAIA,4CACA,2CANA,mCACA,kCAIA,yCACA,wCANA,qCACA,oCAIA,+BACA,sCANA,mCACA,kCAIA,4CACA,2CANA,qCACA,oCAIA,8CACA,6CANA,qCACA,oCAIA,sDACA,qDANA,sCACA,qCAIA,0EACA,yEANA,yCACA,wCAIA,+EACA,8EANA,kCACA,iCAIA,yCACA,wCAeJ,uBACI,mCAGJ,0BACI,8CACA,MAZY,KAehB,mCACI,sGAKA,MArBY,KAMhB,uBACI,mCAGJ,0BACI,8CACA,MAZY,KAehB,mCACI,sGAKA,MArBY,KAMhB,uBACI,mCAGJ,0BACI,8CACA,MAZY,KAehB,mCACI,sGAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,mBACI,+BAGJ,sBACI,0CACA,MAZY,KAehB,+BACI,8FAKA,MArBY,KAMhB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,mBACI,+BAGJ,sBACI,0CACA,MAZY,KAehB,+BACI,8FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MATgB,KAYpB,4BACI,wFAKA,MAlBgB,KAGpB,mBACI,+BAGJ,sBACI,0CACA,MATgB,KAYpB,+BACI,8FAKA,MAlBgB,KAGpB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MATgB,KAYpB,2BACI,sFAKA,MAlBgB,KAGpB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,kBACI,8BAGJ,qBACI,yCACA,MAZY,KAehB,8BACI,4FAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,kBACI,8BAGJ,qBACI,yCACA,MAZY,KAehB,8BACI,4FAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,mBACI,+BAGJ,sBACI,0CACA,MAZY,KAehB,+BACI,8FAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MATgB,KAYpB,4BACI,wFAKA,MAlBgB,KAGpB,gBACI,4BAGJ,mBACI,uCACA,MATgB,KAYpB,4BACI,wFAKA,MAlBgB,KAGpB,eACI,2BAGJ,kBACI,sCACA,MATgB,KAYpB,2BACI,sFAKA,MAlBgB,KAGpB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MATgB,KAYpB,4BACI,wFAKA,MAlBgB,KAGpB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,kBACI,8BAGJ,qBACI,yCACA,MAZY,KAehB,8BACI,4FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,mBACI,+BAGJ,sBACI,0CACA,MAZY,KAehB,+BACI,8FAKA,MArBY,KAMhB,mBACI,+BAGJ,sBACI,0CACA,MAZY,KAehB,+BACI,8FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,iBACI,6BAGJ,oBACI,wCACA,MAZY,KAehB,6BACI,0FAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MAZY,KAehB,4BACI,wFAKA,MArBY,KAMhB,gBACI,4BAGJ,mBACI,uCACA,MATgB,KAYpB,4BACI,wFAKA,MAlBgB,KAGpB,iBACI,6BAGJ,oBACI,wCACA,MATgB,KAYpB,6BACI,0FAKA,MAlBgB,KAGpB,qBACI,iCAGJ,wBACI,4CACA,MAZY,KAehB,iCACI,kGAKA,MArBY,KAMhB,eACI,2BAGJ,kBACI,sCACA,MAZY,KAehB,2BACI,sFAKA,MArBY,KChEpB,MACI,2CACA,8CACA,kDACA,iDCTJ,WACI,sCCKQ,iEACI,sBAOR,2CACI,wBACA,8BACA,2BAGJ,4CACI,WACA,YAGJ,gDACI,eACA,iCCrBA,gDACI,uDACA,wDAGJ,+CACI,0DACA,2DAWA,0EACI,uDACA,wDAGJ,yEACI,0DACA,2DC1BZ,sCACI,iDACA,qEAEA,8CACI,wBAIR,wCACI,mDACA,uEAEA,gDACI,wBAIR,uCACI,kDACA,sEAEA,+CACI,wBC1BZ,2BACI,wCACA,iBACA,kCACA,iBAEA,oDACI,oBACA,kBAGJ,2CACI,uBAIR,4BACI,wCACA,iBACA,kCAGJ,4BACI,0DACA,2DACA,oDAGJ,sBACI,YC3BA,qDACI,aCHR,yCACI,kBACA,WACA,YACA,gBACA,WACA,UCuBR,6BACI,YACA,aAEA,sCACI,WAlBc,uIAmBd,sCACA,0BACA,4BAGJ,sCACI,iEACA,sBAGJ,sCACI,wEACA,sBAGJ,sCACI,8EACA,sBAGJ,mCACI,+BACA,8BACA,eACA,MAnCK,KAoCL,OArCM,KAsCN,uCACA,+BACA,WACA,SACA,OACA,aAGJ,kDACI,eACA,2CACA,WAGJ,kDACI,cACA,wCACA,WAIR,oBACI,GACI,0BAEJ,IACI,8BAEJ,KACI,2BAIR,gBACI,GACI,wBAGJ,IACI,2BAGJ,IACI,2BAGJ,IACI,2BAGJ,KACI,yBClHR,gCACI,YACA,aAGJ,6BACI,cACA,eAEA,8CAII,4BACA,4BACA,4BACA,4BACA,4BAGJ,6CAII,4BACA,4BACA,4BACA,4BACA,4BAGJ,gDAII,4BACA,4BACA,4BACA,4BACA,4BCvCR,oBACI,gBACA,kBACA,8CAEA,yCACI,kBACA,SACA,OACA,WCPJ,uCACI,eCHR,MACI,mBACA,kBACA,iBACA,6BACA,iGAEA,2DAGJ,qBACI,eACA,cAGJ,gBACI,aACA,oBAGA,0DACA,uDAGA,gCAGA,WACA,YAGJ,gBACI,aACA,sBACA,wBACA,gCAGJ,WACI,gBACA,aACA,sBAEA,aACI,cAGJ,eACI,mDAGJ,eACI,mDAGJ,eACI,mDAGJ,eACI,oDAGJ,eACI,oDAGJ,eACI,oDAIR,4BAEI,oCC7DA,qBACI,iDAEA,oCACI,WACA,YAGJ,sCACI,iBACA,mBACA,YAEA,0DACI,iBCbZ,4CACI,WAEA,yBAHJ,4CAIQ,aCVZ,wBACI,oDACA,WAEA,6BACI,YAIR,wBACI,WACA,eACA,gBAGJ,mCACI,gDACA,+CACA,oDACA,kBACA,WACA,UACA,WAGJ,2BACI,eAUA,kDACI,2BC7CR,kBACI,aACA,sBACA,WAEA,mIAGI,aACA,WAGJ,sFAEI,mBAEA,wSAEI,cACA,YAGJ,gSAEI,cAIR,6CACI,WAEA,sEACI,2DC9BR,kDACI,4HACA,qDAGJ,4CACI,mCACA,sCACA,qCCVR,gBACI,cAGJ,0BACI,cAEA,kDACI,cAGJ,iDACI,cAGJ,iDACI,cChBR,iBACI,cAEA,2CACI,yCAGJ,wCACI,gBAIR,sBACI,cAEA,wDACI,wCAGJ,mDACI,yCCpBR,WACI,mBACA,WACA,qCACA,oFACA,QAGJ,kBACI,mBACA,0BACA,iEAGJ,kBACI,mBACA,kBACA,4BACA,iEAKA,yBAFJ,iBAGQ,kDACA,sDCzBR,cACI,eACA,MACA,OACA,YACA,aACA,WACA,sBACA,2BCRJ,oBACI,aACA,kBACA,qBACA,2BAEA,gCACI,UAIR,eACI,YACA,kBAOJ,8BACI,gBAGJ,kBACI,uBACA,kBACA,MACA,OACA,YACA,aACA,aACA,YACA,gBAEA,8BACI,UAIR,uBACI,kBACA,MACA,OACA,YACA,aACA,YACA,gBACA,2BAEA,iCACI,UAGJ,+CACI,kBACA,YACA,aACA,WACA,UAEA,wBACA,aACA,8BACA,2BAEA,wDACI,uBACA,kBACA,uGA4BA,yDACI,oCAGJ,yDACI,mCAGJ,yDACI,mCAGJ,yDACI,kCC9GhB,4BACI,aACA,kBACA,qBACA,2BAGJ,uBACI,YACA,kBAGJ,0BACI,uBACA,kBACA,MACA,OACA,YACA,aACA,aACA,YACA,gBAEA,sCACI","file":"app.min.css"} \ No newline at end of file diff --git a/source/HomeBook.Frontend/wwwroot/img/bg/ws25.jpg b/source/HomeBook.Frontend/wwwroot/img/bg/ws25.jpg deleted file mode 100644 index df940f84..00000000 Binary files a/source/HomeBook.Frontend/wwwroot/img/bg/ws25.jpg and /dev/null differ diff --git a/source/HomeBook.Frontend/wwwroot/index.html b/source/HomeBook.Frontend/wwwroot/index.html index d7c8c288..66895c1e 100644 --- a/source/HomeBook.Frontend/wwwroot/index.html +++ b/source/HomeBook.Frontend/wwwroot/index.html @@ -2,49 +2,51 @@ - - + + HomeBook - + - + - + - + - + - + - + - - + + -
- - - - -
-
+
+ + + + +
+
-
- An unhandled error has occurred. - Reload - 🗙 -
+
+ An unhandled error has occurred. + Reload + 🗙 +
- + - - + + + + diff --git a/source/HomeBook.UnitTests/Backend/Core/Account/JwtServiceTests.cs b/source/HomeBook.UnitTests/Backend/Core/Account/JwtServiceTests.cs new file mode 100644 index 00000000..870fdb04 --- /dev/null +++ b/source/HomeBook.UnitTests/Backend/Core/Account/JwtServiceTests.cs @@ -0,0 +1,103 @@ +using HomeBook.Backend.Core.Account; +using Microsoft.Extensions.Configuration; +using NSubstitute.ExceptionExtensions; + +namespace HomeBook.UnitTests.Backend.Core.Account; + +[TestFixture] +public class JwtServiceTests +{ + private IConfiguration _configuration = null!; + private JwtService _instance = null!; + + [SetUp] + public void SetUp() + { + _configuration = new ConfigurationBuilder() + .AddInMemoryCollection(new Dictionary + { + { + "Jwt:SecretKey", "this-is-a-big-s3cr3t-key-with-a-lot-of-data-and-value" + }, + { + "Jwt:Issuer", "HomeBookTest" + }, + { + "Jwt:Audience", "HomeBookTest" + }, + { + "Jwt:ExpirationMinutes", "120" + } + }) + .Build(); + _instance = new JwtService(_configuration); + } + + [TestCase(true)] + [TestCase(false)] + public void GenerateToken_AndVerify_Return(bool withAdmin) + { + // Arrange + var userId = Guid.NewGuid(); + var username = "testuser"; + + // Act & Assert + var jwt = _instance.GenerateToken(userId, username, withAdmin); + jwt.ShouldNotBeNull(); + + var token = jwt.Token; + token.ShouldNotBeNullOrEmpty(); + + var isValidToken = _instance.ValidateToken(token); + isValidToken.ShouldBeTrue(); + + var hasAdminFlag = _instance.IsAdminFromToken(token); + hasAdminFlag.ShouldBe(withAdmin); + + var userIdFromToken = _instance.GetUserIdFromToken(token); + userIdFromToken.ShouldBe(userId); + } + + [Test] + public void GenerateTokenWithOutAdminFlag_Return() + { + // Arrange + var userId = Guid.NewGuid(); + var username = "testuser"; + + // Act & Assert + var jwt = _instance.GenerateToken(userId, username); + jwt.ShouldNotBeNull(); + + var token = jwt.Token; + token.ShouldNotBeNullOrEmpty(); + + var isValidToken = _instance.ValidateToken(token); + isValidToken.ShouldBeTrue(); + + var hasAdminFlag = _instance.IsAdminFromToken(token); + hasAdminFlag.ShouldBeFalse(); + + var userIdFromToken = _instance.GetUserIdFromToken(token); + userIdFromToken.ShouldBe(userId); + } + + [Test] + public void GenerateTokenWithInvalidToken_ThrowsAndReturns() + { + // Arrange + + // Act & Assert + var token = "this-is-an-invalid-token"; + token.ShouldNotBeNullOrEmpty(); + + var isValidToken = _instance.ValidateToken(token); + isValidToken.ShouldBeFalse(); + + var hasAdminFlag = _instance.IsAdminFromToken(token); + hasAdminFlag.ShouldBeFalse(); + + var userIdFromToken = _instance.GetUserIdFromToken(token); + userIdFromToken.ShouldBeNull(); + } +} diff --git a/source/HomeBook.UnitTests/Backend/Core/Finances/FinanceCalculationsServiceTests.cs b/source/HomeBook.UnitTests/Backend/Core/Finances/FinanceCalculationsServiceTests.cs index 9b49d069..d05637ba 100644 --- a/source/HomeBook.UnitTests/Backend/Core/Finances/FinanceCalculationsServiceTests.cs +++ b/source/HomeBook.UnitTests/Backend/Core/Finances/FinanceCalculationsServiceTests.cs @@ -1,6 +1,7 @@ using System.Globalization; using HomeBook.Backend.Abstractions.Contracts; using HomeBook.Backend.Core.Finances; +using HomeBook.Backend.Module.Finances.Services; using HomeBook.UnitTests.TestCore.Helper; using NSubstitute; diff --git a/source/HomeBook.UnitTests/Backend/Handler/FinanceSavingGoalHandlerE2ETests.cs b/source/HomeBook.UnitTests/Backend/Handler/FinanceSavingGoalHandlerE2ETests.cs index 73cdb412..e3cb75d7 100644 --- a/source/HomeBook.UnitTests/Backend/Handler/FinanceSavingGoalHandlerE2ETests.cs +++ b/source/HomeBook.UnitTests/Backend/Handler/FinanceSavingGoalHandlerE2ETests.cs @@ -1,14 +1,17 @@ using System.Security.Claims; using HomeBook.Backend.Abstractions; using HomeBook.Backend.Abstractions.Contracts; -using HomeBook.Backend.Core.Finances.Contracts; +using HomeBook.Backend.Core.Search; using HomeBook.Backend.Data.Sqlite; using HomeBook.Backend.Data.Sqlite.Extensions; -using HomeBook.Backend.DTOs.Enums; -using HomeBook.Backend.DTOs.Requests.Finances; -using HomeBook.Backend.DTOs.Responses.Finances; using HomeBook.Backend.Extensions; -using HomeBook.Backend.Handler; +using HomeBook.Backend.Factories; +using HomeBook.Backend.Module.Finances.Contracts; +using HomeBook.Backend.Module.Finances.Enums; +using HomeBook.Backend.Module.Finances.Handler; +using HomeBook.Backend.Module.Finances.Requests; +using HomeBook.Backend.Module.Finances.Responses; +using HomeBook.UnitTests.TestCore.Backend; using HomeBook.UnitTests.TestCore.Helper; using Microsoft.AspNetCore.Http.HttpResults; using Microsoft.Data.Sqlite; @@ -71,6 +74,7 @@ public async Task RunFullLifecycle_Returns() ["Database:UseInMemory"] = "true", ["Database:Provider"] = "SQLITE" }); + SearchRegistrationFactory srf = new(); IServiceProvider serviceProvider = new ServiceCollection() .AddLogging() .AddSingleton(configuration) @@ -78,6 +82,7 @@ public async Task RunFullLifecycle_Returns() .AddBackendDataSqlite(configuration) .AddKeyedSingleton("SQLITE") .AddDependenciesForRuntime(configuration, InstanceStatus.SETUP) + .AddBackendModulesForTestEnvironment(configuration, srf) .BuildServiceProvider(); // apply migrations @@ -118,16 +123,16 @@ public async Task RunFullLifecycle_Returns() InterestRateOptions.MONTHLY, 0, null); - var createSavingGoalResult = await FinanceSavingGoalHandler.HandleCreateSavingGoal(testuser, + var createSavingGoalResult = await SavingGoalHandler.HandleCreateSavingGoal(testuser, createRequest, - _loggerFactory.CreateLogger(), + _loggerFactory.CreateLogger(), savingGoalsProvider, cancellationToken); createSavingGoalResult.ShouldBeOfType(); // get all after creation - var savingGoalsResult1 = await FinanceSavingGoalHandler.HandleGetSavingGoals(testuser, - _loggerFactory.CreateLogger(), + var savingGoalsResult1 = await SavingGoalHandler.HandleGetSavingGoals(testuser, + _loggerFactory.CreateLogger(), savingGoalsProvider, cancellationToken); var savingGoalsResponse1 = savingGoalsResult1.ShouldBeOfType>(); @@ -141,16 +146,16 @@ public async Task RunFullLifecycle_Returns() savingGoalsResponse1.Value.SavingGoals[0].MonthlyPayment.ShouldBe(500); // update saving goal - var updateSavingGoalNameResult = await FinanceSavingGoalHandler.HandleUpdateSavingGoalName(createdSavingGoalId, + var updateSavingGoalNameResult = await SavingGoalHandler.HandleUpdateSavingGoalName(createdSavingGoalId, testuser, new UpdateSavingGoalNameRequest("Updated Saving Goal"), - _loggerFactory.CreateLogger(), + _loggerFactory.CreateLogger(), savingGoalsProvider, cancellationToken); updateSavingGoalNameResult.ShouldBeOfType(); // update saving goal current amount - var updateSavingGoalAmountsResult = await FinanceSavingGoalHandler.HandleUpdateSavingGoalAmounts( + var updateSavingGoalAmountsResult = await SavingGoalHandler.HandleUpdateSavingGoalAmounts( createdSavingGoalId, testuser, new UpdateSavingGoalAmountsRequest(null, @@ -158,35 +163,35 @@ public async Task RunFullLifecycle_Returns() null, null, null), - _loggerFactory.CreateLogger(), + _loggerFactory.CreateLogger(), savingGoalsProvider, cancellationToken); updateSavingGoalAmountsResult.ShouldBeOfType(); // update saving goal current amount var dt = DateTime.UtcNow.AddYears(3); - var updateSavingGoalInfoResult = await FinanceSavingGoalHandler.HandleUpdateSavingGoalInfo( + var updateSavingGoalInfoResult = await SavingGoalHandler.HandleUpdateSavingGoalInfo( createdSavingGoalId, testuser, new UpdateSavingGoalInfoRequest(new DateTime(dt.Year, dt.Month, dt.Day)), - _loggerFactory.CreateLogger(), + _loggerFactory.CreateLogger(), savingGoalsProvider, cancellationToken); updateSavingGoalInfoResult.ShouldBeOfType(); // update saving goal current amount - var updateSavingGoalAppearanceResult = await FinanceSavingGoalHandler.HandleUpdateSavingGoalAppearance( + var updateSavingGoalAppearanceResult = await SavingGoalHandler.HandleUpdateSavingGoalAppearance( createdSavingGoalId, testuser, new UpdateSavingGoalAppearanceRequest("#0000ff", ""), - _loggerFactory.CreateLogger(), + _loggerFactory.CreateLogger(), savingGoalsProvider, cancellationToken); updateSavingGoalAppearanceResult.ShouldBeOfType(); // get all after update - var savingGoalsResult2 = await FinanceSavingGoalHandler.HandleGetSavingGoals(testuser, - _loggerFactory.CreateLogger(), + var savingGoalsResult2 = await SavingGoalHandler.HandleGetSavingGoals(testuser, + _loggerFactory.CreateLogger(), savingGoalsProvider, cancellationToken); var savingGoalsResponse2 = savingGoalsResult2.ShouldBeOfType>(); @@ -200,16 +205,16 @@ public async Task RunFullLifecycle_Returns() savingGoalsResponse2.Value.SavingGoals[0].TargetDate.ShouldBe(new DateTime(dt.Year, dt.Month, dt.Day)); // delete saving goal - var deleteSavingGoalResult = await FinanceSavingGoalHandler.HandleDeleteSavingGoal(createdSavingGoalId, + var deleteSavingGoalResult = await SavingGoalHandler.HandleDeleteSavingGoal(createdSavingGoalId, testuser, - _loggerFactory.CreateLogger(), + _loggerFactory.CreateLogger(), savingGoalsProvider, cancellationToken); deleteSavingGoalResult.ShouldBeOfType(); // get all after delete - var savingGoalsResult3 = await FinanceSavingGoalHandler.HandleGetSavingGoals(testuser, - _loggerFactory.CreateLogger(), + var savingGoalsResult3 = await SavingGoalHandler.HandleGetSavingGoals(testuser, + _loggerFactory.CreateLogger(), savingGoalsProvider, cancellationToken); var savingGoalsResponse3 = savingGoalsResult3.ShouldBeOfType>(); diff --git a/source/HomeBook.UnitTests/Backend/Handler/InfoHandlerE2ETests.cs b/source/HomeBook.UnitTests/Backend/Handler/InfoHandlerE2ETests.cs new file mode 100644 index 00000000..65d55b9a --- /dev/null +++ b/source/HomeBook.UnitTests/Backend/Handler/InfoHandlerE2ETests.cs @@ -0,0 +1,107 @@ +using HomeBook.Backend.Abstractions; +using HomeBook.Backend.Abstractions.Contracts; +using HomeBook.Backend.Data.Sqlite; +using HomeBook.Backend.Data.Sqlite.Extensions; +using HomeBook.Backend.Extensions; +using HomeBook.Backend.Handler; +using HomeBook.Backend.Responses; +using HomeBook.UnitTests.TestCore.Helper; +using Microsoft.AspNetCore.Http.HttpResults; +using Microsoft.Data.Sqlite; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Logging; + +namespace HomeBook.UnitTests.Backend.Handler; + +[TestFixture] +public class InfoHandlerE2ETests +{ + private ILoggerFactory _loggerFactory; + private SqliteConnection _keepAlive = null!; + + [SetUp] + public void SetUpSubstitutes() + { + // create logger + _loggerFactory = LoggerFactory.Create(builder => + { + builder.AddSimpleConsole(options => + { + options.IncludeScopes = true; + options.SingleLine = true; + options.TimestampFormat = "HH:mm:ss "; + }) + .SetMinimumLevel(LogLevel.Debug); + }); + + // create sqlite in memory + var connectionString = ConnectionStringBuilder.BuildInMemory(); + + _keepAlive = new SqliteConnection(connectionString); + _keepAlive.Open(); + } + + [TearDown] + public void TearDown() + { + // delete sqlite in memory + _keepAlive.Close(); + + // dispose logger + _loggerFactory.Dispose(); + } + + [Test] + public async Task RunFullLifecycle_Returns() + { + // Arrange + CancellationToken cancellationToken = CancellationToken.None; + IConfigurationRoot configuration = ConfigurationHelper.CreateConfigurationRoot(new Dictionary + { + ["Environment"] = "UnitTests", + ["Database:UseInMemory"] = "true", + ["Database:Provider"] = "SQLITE" + }); + IServiceProvider serviceProvider = new ServiceCollection() + .AddLogging() + .AddSingleton(configuration) + .AddSingleton(configuration) + .AddBackendDataSqlite(configuration) + .AddKeyedSingleton("SQLITE") + .AddDependenciesForRuntime(configuration, InstanceStatus.RUNNING) + .BuildServiceProvider(); + // apply migrations + var databaseMigrator = serviceProvider.GetKeyedService("SQLITE"); + await databaseMigrator.MigrateAsync(cancellationToken); + + // write test data + IInstanceConfigurationProvider instanceConfigurationProvider = serviceProvider.GetRequiredService(); + await instanceConfigurationProvider.SetHomeBookInstanceDefaultLocaleAsync("te-ST", cancellationToken); + await instanceConfigurationProvider.SetHomeBookInstanceNameAsync("Test Instance", cancellationToken); + + + // Act & Assert + var instanceInfoResult = await InfoHandler.HandleGetInstanceInfo( + instanceConfigurationProvider, + cancellationToken); + var instanceInfoResponse = instanceInfoResult.ShouldBeOfType>(); + instanceInfoResponse.Value.ShouldNotBeNull(); + instanceInfoResponse.Value.Name.ShouldBe("Test Instance"); + instanceInfoResponse.Value.DefaultLocale.ShouldBe("te-ST"); + + var instanceNameResult = await InfoHandler.HandleGetInstanceName( + instanceConfigurationProvider, + cancellationToken); + var instanceNameResponse = instanceNameResult.ShouldBeOfType>(); + instanceNameResponse.Value.ShouldNotBeNull(); + instanceNameResponse.Value.ShouldBe("Test Instance"); + + var instanceDefaultLocaleResult = await InfoHandler.HandleGetInstanceDefaultLocale( + instanceConfigurationProvider, + cancellationToken); + var instanceDefaultLocaleResponse = instanceDefaultLocaleResult.ShouldBeOfType>(); + instanceDefaultLocaleResponse.Value.ShouldNotBeNull(); + instanceDefaultLocaleResponse.Value.ShouldBe("te-ST"); + } +} diff --git a/source/HomeBook.UnitTests/Backend/Handler/KitchenRecipeHandlerE2ETests.cs b/source/HomeBook.UnitTests/Backend/Handler/KitchenRecipeHandlerE2ETests.cs index 371402f2..baec72b0 100644 --- a/source/HomeBook.UnitTests/Backend/Handler/KitchenRecipeHandlerE2ETests.cs +++ b/source/HomeBook.UnitTests/Backend/Handler/KitchenRecipeHandlerE2ETests.cs @@ -1,13 +1,15 @@ using System.Security.Claims; using HomeBook.Backend.Abstractions; using HomeBook.Backend.Abstractions.Contracts; -using HomeBook.Backend.Core.Kitchen.Contracts; +using HomeBook.Backend.Core.Search; using HomeBook.Backend.Data.Sqlite; using HomeBook.Backend.Data.Sqlite.Extensions; -using HomeBook.Backend.DTOs.Requests.Kitchen; -using HomeBook.Backend.DTOs.Responses.Kitchen; using HomeBook.Backend.Extensions; -using HomeBook.Backend.Handler; +using HomeBook.Backend.Module.Kitchen.Contracts; +using HomeBook.Backend.Module.Kitchen.Handler; +using HomeBook.Backend.Module.Kitchen.Requests; +using HomeBook.Backend.Module.Kitchen.Responses; +using HomeBook.UnitTests.TestCore.Backend; using HomeBook.UnitTests.TestCore.Helper; using Microsoft.AspNetCore.Http.HttpResults; using Microsoft.Data.Sqlite; @@ -18,7 +20,7 @@ namespace HomeBook.UnitTests.Backend.Handler; [TestFixture] -public class KitchenRecipeHandlerE2ETests +public class KitchenRecipeHandlerE2ETests : TestBase { private ILoggerFactory _loggerFactory; private SqliteConnection _keepAlive = null!; @@ -56,27 +58,21 @@ public void TearDown() } [Test] - public async Task RunFullLifecycle_Returns() + public async Task RunRecipesFullLifecycle_Returns() { // Arrange CancellationToken cancellationToken = CancellationToken.None; string testUserName = "testuser"; string testUserPassword = "s3cr3tP@ssw0rd!"; - // create configuration and service provider - IConfigurationRoot configuration = ConfigurationHelper.CreateConfigurationRoot(new Dictionary - { - ["Environment"] = "UnitTests", - ["Database:UseInMemory"] = "true", - ["Database:Provider"] = "SQLITE" - }); - IServiceProvider serviceProvider = new ServiceCollection() - .AddLogging() - .AddSingleton(configuration) - .AddSingleton(configuration) + SearchRegistrationFactory srf = new(); + IConfigurationRoot configuration = CreateTestConfiguration(); + IServiceCollection serviceCollection = CreateTestServiceProvider(configuration); + IServiceProvider serviceProvider = serviceCollection .AddBackendDataSqlite(configuration) .AddKeyedSingleton("SQLITE") .AddDependenciesForRuntime(configuration, InstanceStatus.RUNNING) + .AddBackendModulesForTestEnvironment(configuration, srf) .BuildServiceProvider(); // apply migrations @@ -89,12 +85,184 @@ public async Task RunFullLifecycle_Returns() .Where(line => !string.IsNullOrWhiteSpace(line)) .ToList(); tables.ShouldContain("__EFMigrationsHistory"); - tables.ShouldContain("Users"); - tables.ShouldContain("Configurations"); - tables.ShouldContain("UserPreferences"); + tables.ShouldContain("RecipeSteps"); + tables.ShouldContain("RecipeIngredients"); + tables.ShouldContain("Recipe2RecipeIngredient"); tables.ShouldContain("Recipes"); + + // create user + IUserProvider userProvider = serviceProvider.GetRequiredService(); + Guid testUserId = await userProvider.CreateUserAsync(testUserName, + testUserPassword, + cancellationToken); + ClaimsPrincipal testuser = UserHelper.CreateTestUser(testUserId, + testUserName); + + // create instances for tests + IRecipesProvider recipesProvider = serviceProvider.GetRequiredService(); + + // Act & Assert + + // get all recipes + var recipesResult1 = await RecipeHandler.HandleGetRecipes("", + _loggerFactory.CreateLogger(), + recipesProvider, + userProvider, + cancellationToken); + var recipesResponse1 = recipesResult1.ShouldBeOfType>(); + recipesResponse1.Value.ShouldNotBeNull(); + recipesResponse1.Value.Recipes.Length.ShouldBe(0); + + // create recipes + var createRecipeResult1 = await RecipeHandler.HandleCreateRecipe(testuser, + new CreateRecipeRequest("Gyros-Pita", null, null, null, null, null, null, null, null, null, null), + _loggerFactory.CreateLogger(), + recipesProvider, + cancellationToken); + createRecipeResult1.ShouldBeOfType(); + var createRecipeResult2 = await RecipeHandler.HandleCreateRecipe(testuser, + new CreateRecipeRequest("Nana's Italian Roulade", + null, + null, + null, + null, + null, + null, + null, + null, + null, + null), + _loggerFactory.CreateLogger(), + recipesProvider, + cancellationToken); + createRecipeResult2.ShouldBeOfType(); + var createRecipeResult3 = await RecipeHandler.HandleCreateRecipe(testuser, + new CreateRecipeRequest("Pancakes", null, null, null, null, null, null, null, null, null, null), + _loggerFactory.CreateLogger(), + recipesProvider, + cancellationToken); + createRecipeResult3.ShouldBeOfType(); + var createRecipeResult4 = await RecipeHandler.HandleCreateRecipe(testuser, + new CreateRecipeRequest("Pasta à la Roma", null, null, null, null, null, null, null, null, null, null), + _loggerFactory.CreateLogger(), + recipesProvider, + cancellationToken); + createRecipeResult4.ShouldBeOfType(); + var createRecipeResult5 = await RecipeHandler.HandleCreateRecipe(testuser, + new CreateRecipeRequest("Rührei mit Kräutern", null, null, null, null, null, null, null, null, null, null), + _loggerFactory.CreateLogger(), + recipesProvider, + cancellationToken); + createRecipeResult5.ShouldBeOfType(); + + // get all recipes + var recipesResult2 = await RecipeHandler.HandleGetRecipes("", + _loggerFactory.CreateLogger(), + recipesProvider, + userProvider, + cancellationToken); + var recipesResponse2 = recipesResult2.ShouldBeOfType>(); + recipesResponse2.Value.ShouldNotBeNull(); + recipesResponse2.Value.Recipes.Length.ShouldBe(5); + recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Gyros-Pita" + && r.NormalizedName == "gyros-pita"); + recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Nana's Italian Roulade" + && r.NormalizedName == "nanas-italian-roulade"); + recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Pancakes" + && r.NormalizedName == "pancakes"); + recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Pasta à la Roma" + && r.NormalizedName == "pasta-a-la-roma"); + recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Rührei mit Kräutern" + && r.NormalizedName == "ruhrei-mit-krautern"); + + var recipeToDelete = recipesResponse2.Value.Recipes.First(r => r.Name == "Pancakes"); + var deleteRecipeResult = await RecipeHandler.HandleDeleteRecipe(recipeToDelete.Id, + testuser, + _loggerFactory.CreateLogger(), + recipesProvider, + cancellationToken); + deleteRecipeResult.ShouldBeOfType(); + + // get all recipes + var recipesResult3 = await RecipeHandler.HandleGetRecipes("", + _loggerFactory.CreateLogger(), + recipesProvider, + userProvider, + cancellationToken); + var recipesResponse3 = recipesResult3.ShouldBeOfType>(); + recipesResponse3.Value.ShouldNotBeNull(); + recipesResponse3.Value.Recipes.Length.ShouldBe(4); + recipesResponse3.Value.Recipes.ShouldContain(r => r.Name == "Gyros-Pita" + && r.NormalizedName == "gyros-pita"); + recipesResponse3.Value.Recipes.ShouldContain(r => r.Name == "Nana's Italian Roulade" + && r.NormalizedName == "nanas-italian-roulade"); + recipesResponse3.Value.Recipes.ShouldContain(r => r.Name == "Pasta à la Roma" + && r.NormalizedName == "pasta-a-la-roma"); + recipesResponse3.Value.Recipes.ShouldContain(r => r.Name == "Rührei mit Kräutern" + && r.NormalizedName == "ruhrei-mit-krautern"); + + var recipeToUpdate = recipesResponse3.Value.Recipes.First(r => r.Name == "Gyros-Pita"); + var updateRecipeResult = await RecipeHandler.HandleUpdateRecipeName(recipeToUpdate.Id, + testuser, + new UpdateRecipeNameRequest("Gyros Wrap"), + _loggerFactory.CreateLogger(), + recipesProvider, + cancellationToken); + updateRecipeResult.ShouldBeOfType(); + + // get all recipes + var recipesResult4 = await RecipeHandler.HandleGetRecipes("", + _loggerFactory.CreateLogger(), + recipesProvider, + userProvider, + cancellationToken); + var recipesResponse4 = recipesResult4.ShouldBeOfType>(); + recipesResponse4.Value.ShouldNotBeNull(); + recipesResponse4.Value.Recipes.Length.ShouldBe(4); + recipesResponse4.Value.Recipes.ShouldContain(r => r.Name == "Gyros Wrap" + && r.NormalizedName == "gyros-wrap"); + recipesResponse4.Value.Recipes.ShouldContain(r => r.Name == "Nana's Italian Roulade" + && r.NormalizedName == "nanas-italian-roulade"); + recipesResponse4.Value.Recipes.ShouldContain(r => r.Name == "Pasta à la Roma" + && r.NormalizedName == "pasta-a-la-roma"); + recipesResponse4.Value.Recipes.ShouldContain(r => r.Name == "Rührei mit Kräutern" + && r.NormalizedName == "ruhrei-mit-krautern"); + } + + [Test] + public async Task RunRecipesWithStepsAndIngredientsFullLifecycle_Returns() + { + // Arrange + CancellationToken cancellationToken = CancellationToken.None; + string testUserName = "testuser"; + string testUserPassword = "s3cr3tP@ssw0rd!"; + + // create configuration and service provider + SearchRegistrationFactory srf = new(); + IConfigurationRoot configuration = CreateTestConfiguration(); + IServiceCollection serviceCollection = CreateTestServiceProvider(configuration); + + IServiceProvider serviceProvider = serviceCollection + .AddBackendDataSqlite(configuration) + .AddKeyedSingleton("SQLITE") + .AddDependenciesForRuntime(configuration, InstanceStatus.RUNNING) + .AddBackendModulesForTestEnvironment(configuration, srf) + .BuildServiceProvider(); + + // apply migrations + var databaseMigrator = serviceProvider.GetKeyedService("SQLITE"); + await databaseMigrator.MigrateAsync(cancellationToken); + + // verify that migrations were applied + var tables = SqliteHelper.GetAllTableNames(_keepAlive) + .Split(Environment.NewLine) + .Where(line => !string.IsNullOrWhiteSpace(line)) + .ToList(); + tables.ShouldContain("__EFMigrationsHistory"); + tables.ShouldContain("RecipeSteps"); + tables.ShouldContain("RecipeIngredients"); + tables.ShouldContain("Recipe2RecipeIngredient"); tables.ShouldContain("Recipes"); - tables.ShouldContain("Ingredients"); // create user IUserProvider userProvider = serviceProvider.GetRequiredService(); @@ -110,77 +278,122 @@ public async Task RunFullLifecycle_Returns() // Act & Assert // get all recipes - var recipesResult1 = await KitchenRecipeHandler.HandleGetRecipes("", - _loggerFactory.CreateLogger(), + var recipesResult1 = await RecipeHandler.HandleGetRecipes("", + _loggerFactory.CreateLogger(), recipesProvider, + userProvider, cancellationToken); var recipesResponse1 = recipesResult1.ShouldBeOfType>(); recipesResponse1.Value.ShouldNotBeNull(); recipesResponse1.Value.Recipes.Length.ShouldBe(0); // create recipes - var createRecipeResult1 = await KitchenRecipeHandler.HandleCreateRecipe(testuser, - new CreateRecipeRequest("Gyros-Pita"), - _loggerFactory.CreateLogger(), + var createRecipeResult1 = await RecipeHandler.HandleCreateRecipe(testuser, + new CreateRecipeRequest("Gyros-Pita", null, null, null, null, null, null, null, null, null, null), + _loggerFactory.CreateLogger(), recipesProvider, cancellationToken); createRecipeResult1.ShouldBeOfType(); - var createRecipeResult2 = await KitchenRecipeHandler.HandleCreateRecipe(testuser, - new CreateRecipeRequest("Nana's Italian Roulade"), - _loggerFactory.CreateLogger(), + var createRecipeResult2 = await RecipeHandler.HandleCreateRecipe(testuser, + new CreateRecipeRequest("Nana's Italian Roulade", + null, + null, + null, + null, + null, + null, + null, + null, + null, + null), + _loggerFactory.CreateLogger(), recipesProvider, cancellationToken); createRecipeResult2.ShouldBeOfType(); - var createRecipeResult3 = await KitchenRecipeHandler.HandleCreateRecipe(testuser, - new CreateRecipeRequest("Pancakes"), - _loggerFactory.CreateLogger(), + var createRecipeResult3 = await RecipeHandler.HandleCreateRecipe(testuser, + new CreateRecipeRequest("Pancakes", + "Leckere Pfannkuchen mit Schoko-Creme", + 4, + [ + new CreateRecipeIngredientRequest("Mehl", 500, "Gramm"), + new CreateRecipeIngredientRequest("Salz", 1, "Priese"), + new CreateRecipeIngredientRequest("Milch", 1, "Liter"), + new CreateRecipeIngredientRequest("Schokolade", 4, "EL"), + ], + [ + new CreateRecipeStepRequest("Mehl mit Salz und Milch mischen.", 0, null), + new CreateRecipeStepRequest("Den Teig in eine Pfanne geben und knapp eine Minute anbacken lassen.", + 1, + 60), + new CreateRecipeStepRequest("Die Schokolade schmelzen lassen", 2, (5 * 60)), + new CreateRecipeStepRequest("Die Schokolade über den fertigen Pfannkuchen verteilen", 3, null) + ], + 8, + 4, + null, + 2750, + "Dies ist ein Test-Kommentar", + "Das habe ich mir selbst ausgedacht"), + _loggerFactory.CreateLogger(), recipesProvider, cancellationToken); createRecipeResult3.ShouldBeOfType(); - var createRecipeResult4 = await KitchenRecipeHandler.HandleCreateRecipe(testuser, - new CreateRecipeRequest("Pasta à la Roma"), - _loggerFactory.CreateLogger(), + var createRecipeResult4 = await RecipeHandler.HandleCreateRecipe(testuser, + new CreateRecipeRequest("Pasta à la Roma", null, null, null, null, null, null, null, null, null, null), + _loggerFactory.CreateLogger(), recipesProvider, cancellationToken); createRecipeResult4.ShouldBeOfType(); - var createRecipeResult5 = await KitchenRecipeHandler.HandleCreateRecipe(testuser, - new CreateRecipeRequest("Rührei mit Kräutern"), - _loggerFactory.CreateLogger(), + var createRecipeResult5 = await RecipeHandler.HandleCreateRecipe(testuser, + new CreateRecipeRequest("Rührei mit Kräutern", null, null, null, null, null, null, null, null, null, null), + _loggerFactory.CreateLogger(), recipesProvider, cancellationToken); createRecipeResult5.ShouldBeOfType(); // get all recipes - var recipesResult2 = await KitchenRecipeHandler.HandleGetRecipes("", - _loggerFactory.CreateLogger(), + var recipesResult2 = await RecipeHandler.HandleGetRecipes("", + _loggerFactory.CreateLogger(), recipesProvider, + userProvider, cancellationToken); var recipesResponse2 = recipesResult2.ShouldBeOfType>(); recipesResponse2.Value.ShouldNotBeNull(); recipesResponse2.Value.Recipes.Length.ShouldBe(5); - recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Gyros-Pita"); - recipesResponse2.Value.Recipes.ShouldContain(r => r.NormalizedName == "gyros-pita"); - recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Nana's Italian Roulade"); - recipesResponse2.Value.Recipes.ShouldContain(r => r.NormalizedName == "nanas-italian-roulade"); - recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Pancakes"); - recipesResponse2.Value.Recipes.ShouldContain(r => r.NormalizedName == "pancakes"); - recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Pasta à la Roma"); - recipesResponse2.Value.Recipes.ShouldContain(r => r.NormalizedName == "pasta-a-la-roma"); - recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Rührei mit Kräutern"); - recipesResponse2.Value.Recipes.ShouldContain(r => r.NormalizedName == "ruhrei-mit-krautern"); + recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Gyros-Pita" + && r.NormalizedName == "gyros-pita"); + recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Nana's Italian Roulade" + && r.NormalizedName == "nanas-italian-roulade"); + + recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Pancakes" + && r.NormalizedName == "pancakes" + && r.Description == "Leckere Pfannkuchen mit Schoko-Creme" + && r.DurationWorkingMinutes == 8 + && r.DurationCookingMinutes == 4 + && r.DurationRestingMinutes == null + && r.CaloriesKcal == 2750 + && r.Servings == 4 + && r.Comments == "Dies ist ein Test-Kommentar" + && r.Source == "Das habe ich mir selbst ausgedacht"); + + recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Pasta à la Roma" + && r.NormalizedName == "pasta-a-la-roma"); + recipesResponse2.Value.Recipes.ShouldContain(r => r.Name == "Rührei mit Kräutern" + && r.NormalizedName == "ruhrei-mit-krautern"); var recipeToDelete = recipesResponse2.Value.Recipes.First(r => r.Name == "Pancakes"); - var deleteRecipeResult = await KitchenRecipeHandler.HandleDeleteRecipe(recipeToDelete.Id, + var deleteRecipeResult = await RecipeHandler.HandleDeleteRecipe(recipeToDelete.Id, testuser, - _loggerFactory.CreateLogger(), + _loggerFactory.CreateLogger(), recipesProvider, cancellationToken); deleteRecipeResult.ShouldBeOfType(); // get all recipes - var recipesResult3 = await KitchenRecipeHandler.HandleGetRecipes("", - _loggerFactory.CreateLogger(), + var recipesResult3 = await RecipeHandler.HandleGetRecipes("", + _loggerFactory.CreateLogger(), recipesProvider, + userProvider, cancellationToken); var recipesResponse3 = recipesResult3.ShouldBeOfType>(); recipesResponse3.Value.ShouldNotBeNull(); @@ -195,18 +408,19 @@ public async Task RunFullLifecycle_Returns() recipesResponse3.Value.Recipes.ShouldContain(r => r.NormalizedName == "ruhrei-mit-krautern"); var recipeToUpdate = recipesResponse3.Value.Recipes.First(r => r.Name == "Gyros-Pita"); - var updateRecipeResult = await KitchenRecipeHandler.HandleUpdateRecipeName(recipeToUpdate.Id, + var updateRecipeResult = await RecipeHandler.HandleUpdateRecipeName(recipeToUpdate.Id, testuser, new UpdateRecipeNameRequest("Gyros Wrap"), - _loggerFactory.CreateLogger(), + _loggerFactory.CreateLogger(), recipesProvider, cancellationToken); updateRecipeResult.ShouldBeOfType(); // get all recipes - var recipesResult4 = await KitchenRecipeHandler.HandleGetRecipes("", - _loggerFactory.CreateLogger(), + var recipesResult4 = await RecipeHandler.HandleGetRecipes("", + _loggerFactory.CreateLogger(), recipesProvider, + userProvider, cancellationToken); var recipesResponse4 = recipesResult4.ShouldBeOfType>(); recipesResponse4.Value.ShouldNotBeNull(); diff --git a/source/HomeBook.UnitTests/Backend/OpenApi/DescriptionTests.cs b/source/HomeBook.UnitTests/Backend/OpenApi/DescriptionTests.cs index 52091083..8f319b5e 100644 --- a/source/HomeBook.UnitTests/Backend/OpenApi/DescriptionTests.cs +++ b/source/HomeBook.UnitTests/Backend/OpenApi/DescriptionTests.cs @@ -1,4 +1,4 @@ -using HomeBook.Backend.OpenApi; +using HomeBook.Backend.Core.Modules.OpenApi; namespace HomeBook.UnitTests.Backend.OpenApi; diff --git a/source/HomeBook.UnitTests/Backend/TestBase.cs b/source/HomeBook.UnitTests/Backend/TestBase.cs new file mode 100644 index 00000000..1a16d59f --- /dev/null +++ b/source/HomeBook.UnitTests/Backend/TestBase.cs @@ -0,0 +1,36 @@ +using HomeBook.UnitTests.TestCore.Helper; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Logging; + +namespace HomeBook.UnitTests.Backend; + +public class TestBase +{ + public IConfigurationRoot CreateTestConfiguration() + { + IConfigurationRoot configuration = ConfigurationHelper.CreateConfigurationRoot(new Dictionary + { + ["Environment"] = "UnitTests", + ["Database:UseInMemory"] = "true", + ["Database:Provider"] = "SQLITE" + }); + + return configuration; + } + + public IServiceCollection CreateTestServiceProvider(IConfigurationRoot configuration) + { + IServiceCollection serviceCollection = new ServiceCollection() + .AddLogging(builder => + { + builder.ClearProviders(); + builder.AddConsole(); + builder.SetMinimumLevel(LogLevel.Information); + }) + .AddSingleton(configuration) + .AddSingleton(configuration); + + return serviceCollection; + } +} diff --git a/source/HomeBook.UnitTests/Frontend/Services/AuthenticationServiceTests.cs b/source/HomeBook.UnitTests/Frontend/Services/AuthenticationServiceTests.cs new file mode 100644 index 00000000..cab4a12e --- /dev/null +++ b/source/HomeBook.UnitTests/Frontend/Services/AuthenticationServiceTests.cs @@ -0,0 +1,426 @@ +using HomeBook.Client; +using HomeBook.Client.Models; +using HomeBook.Frontend.Services; +using HomeBook.UnitTests.TestCore.Frontend; +using HomeBook.UnitTests.TestCore.Helper; +using Microsoft.Extensions.Logging; +using Microsoft.Kiota.Abstractions; +using Microsoft.Kiota.Abstractions.Serialization; +using NSubstitute; +using NSubstitute.ExceptionExtensions; + +namespace HomeBook.UnitTests.Frontend.Services; + +[TestFixture] +public class AuthenticationServiceTests +{ + private ILoggerFactory _loggerFactory; + private IRequestAdapter _backendClientAdapter; + private BackendClient _backendClient; + private TestJSRuntime _jsRuntime; + private AuthenticationService _instance; + + [SetUp] + public void SetUp() + { + _loggerFactory = LoggerFactory.Create(builder => + { + builder.AddSimpleConsole(options => + { + options.IncludeScopes = true; + options.SingleLine = true; + options.TimestampFormat = "HH:mm:ss "; + }) + .SetMinimumLevel(LogLevel.Debug); + }); + _backendClientAdapter = Substitute.For(); + _backendClient = new BackendClient(_backendClientAdapter); + _jsRuntime = new TestJSRuntime(); + _instance = new AuthenticationService( + _loggerFactory.CreateLogger(), + _backendClient, + _jsRuntime); + } + + [TearDown] + public void TearDown() + { + _loggerFactory.Dispose(); + } + + [Test] + public async Task LoginAsync_WithData_Return() + { + // arrange + var userId = Guid.NewGuid(); + var username = "testuser"; + var password = "s3cr3t"; + var cancellationToken = CancellationToken.None; + var tokenResponse = "a-s-e-c-r-e-t-t-o-k-e-n"; + var refreshTokenResponse = "a-s-e-c-r-e-t-r-e-f-r-e-s-h-t-o-k-e-n"; + var expiresAt = DateTime.UtcNow.AddHours(1); + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + + _jsRuntime.SetupResult("localStorage.getItem", tokenResponse, "authToken"); + _jsRuntime.SetupResult("localStorage.getItem", expectedExpiresAt, "expiresAt"); + + RequestInformation? postRequest = null; + _backendClientAdapter.SendAsync( + Arg.Do(req => postRequest = req), + Arg.Any>(), + Arg.Any>?>(), + Arg.Any()) + .Returns(new LoginResponse() + { + Token = tokenResponse, + RefreshToken = refreshTokenResponse, + ExpiresAt = expiresAt, + UserId = userId, + Username = username + }); + var eventTriggered = false; + _instance.AuthenticationStateChanged += isAuthenticated => + { + isAuthenticated.ShouldBeTrue(); + eventTriggered = true; + }; + + // act + var result = await _instance.LoginAsync(username, + password, + cancellationToken); + + // assert + _jsRuntime.Called("localStorage.setItem", "authToken", tokenResponse).ShouldBeTrue(); + _jsRuntime.Called("localStorage.setItem", "refreshToken", refreshTokenResponse).ShouldBeTrue(); + _jsRuntime.Called("localStorage.setItem", "expiresAt", expectedExpiresAt).ShouldBeTrue(); + eventTriggered.ShouldBeTrue(); + + result.ShouldBeTrue(); + + var isAuthenticated = await _instance.IsAuthenticatedAsync(cancellationToken); + isAuthenticated.ShouldBeTrue(); + } + + [Test] + public async Task LoginAsync_WithNoResponseFromServer_Return() + { + var userId = Guid.NewGuid(); + var username = "testuser"; + var password = "s3cr3t"; + var cancellationToken = CancellationToken.None; + var tokenResponse = "a-s-e-c-r-e-t-t-o-k-e-n"; + var refreshTokenResponse = "a-s-e-c-r-e-t-r-e-f-r-e-s-h-t-o-k-e-n"; + var expiresAt = DateTime.UtcNow.AddHours(1); + + RequestInformation? postRequest = null; + _backendClientAdapter.SendAsync( + Arg.Do(req => postRequest = req), + Arg.Any>(), + Arg.Any>?>(), + Arg.Any()) + .Returns((LoginResponse)null!); + var eventTriggered = false; + _instance.AuthenticationStateChanged += isAuthenticated => + { + isAuthenticated.ShouldBeTrue(); + eventTriggered = true; + }; + var result = await _instance.LoginAsync(username, + password, + cancellationToken); + + _jsRuntime.Called("localStorage.setItem", "authToken", tokenResponse).ShouldBeFalse(); + _jsRuntime.Called("localStorage.setItem", "refreshToken", refreshTokenResponse).ShouldBeFalse(); + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + _jsRuntime.Called("localStorage.setItem", "expiresAt", expectedExpiresAt).ShouldBeFalse(); + eventTriggered.ShouldBeFalse(); + + result.ShouldBeFalse(); + } + + [Test] + public async Task LoginAsync_WithHttp400AsInvalidAuthFromServer_Return() + { + var userId = Guid.NewGuid(); + var username = "testuser"; + var password = "s3cr3t"; + var cancellationToken = CancellationToken.None; + var tokenResponse = "a-s-e-c-r-e-t-t-o-k-e-n"; + var refreshTokenResponse = "a-s-e-c-r-e-t-r-e-f-r-e-s-h-t-o-k-e-n"; + var expiresAt = DateTime.UtcNow.AddHours(1); + + RequestInformation? postRequest = null; + _backendClientAdapter.SendAsync( + Arg.Do(req => postRequest = req), + Arg.Any>(), + Arg.Any>?>(), + Arg.Any()) + .ThrowsAsync(new ApiException("Bad Request").WithStatusCode(400)); + var eventTriggered = false; + _instance.AuthenticationStateChanged += isAuthenticated => + { + isAuthenticated.ShouldBeTrue(); + eventTriggered = true; + }; + var result = await _instance.LoginAsync(username, + password, + cancellationToken); + + _jsRuntime.Called("localStorage.setItem", "authToken", tokenResponse).ShouldBeFalse(); + _jsRuntime.Called("localStorage.setItem", "refreshToken", refreshTokenResponse).ShouldBeFalse(); + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + _jsRuntime.Called("localStorage.setItem", "expiresAt", expectedExpiresAt).ShouldBeFalse(); + eventTriggered.ShouldBeFalse(); + + result.ShouldBeFalse(); + } + + [Test] + public async Task LoginAsync_WithHttp401AsInvalidAuthFromServer_Return() + { + var userId = Guid.NewGuid(); + var username = "testuser"; + var password = "s3cr3t"; + var cancellationToken = CancellationToken.None; + var tokenResponse = "a-s-e-c-r-e-t-t-o-k-e-n"; + var refreshTokenResponse = "a-s-e-c-r-e-t-r-e-f-r-e-s-h-t-o-k-e-n"; + var expiresAt = DateTime.UtcNow.AddHours(1); + + RequestInformation? postRequest = null; + _backendClientAdapter.SendAsync( + Arg.Do(req => postRequest = req), + Arg.Any>(), + Arg.Any>?>(), + Arg.Any()) + .ThrowsAsync(new ApiException("Unauthorized").WithStatusCode(401)); + var eventTriggered = false; + _instance.AuthenticationStateChanged += isAuthenticated => + { + isAuthenticated.ShouldBeTrue(); + eventTriggered = true; + }; + var result = await _instance.LoginAsync(username, + password, + cancellationToken); + + _jsRuntime.Called("localStorage.setItem", "authToken", tokenResponse).ShouldBeFalse(); + _jsRuntime.Called("localStorage.setItem", "refreshToken", refreshTokenResponse).ShouldBeFalse(); + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + _jsRuntime.Called("localStorage.setItem", "expiresAt", expectedExpiresAt).ShouldBeFalse(); + eventTriggered.ShouldBeFalse(); + + result.ShouldBeFalse(); + } + + [Test] + public async Task LoginAsync_WithHttp500AsInvalidAuthFromServer_Return() + { + var userId = Guid.NewGuid(); + var username = "testuser"; + var password = "s3cr3t"; + var cancellationToken = CancellationToken.None; + var tokenResponse = "a-s-e-c-r-e-t-t-o-k-e-n"; + var refreshTokenResponse = "a-s-e-c-r-e-t-r-e-f-r-e-s-h-t-o-k-e-n"; + var expiresAt = DateTime.UtcNow.AddHours(1); + + RequestInformation? postRequest = null; + _backendClientAdapter.SendAsync( + Arg.Do(req => postRequest = req), + Arg.Any>(), + Arg.Any>?>(), + Arg.Any()) + .ThrowsAsync(new ApiException("Internal Server Error").WithStatusCode(500)); + var eventTriggered = false; + _instance.AuthenticationStateChanged += isAuthenticated => + { + isAuthenticated.ShouldBeTrue(); + eventTriggered = true; + }; + var result = await _instance.LoginAsync(username, + password, + cancellationToken); + + _jsRuntime.Called("localStorage.setItem", "authToken", tokenResponse).ShouldBeFalse(); + _jsRuntime.Called("localStorage.setItem", "refreshToken", refreshTokenResponse).ShouldBeFalse(); + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + _jsRuntime.Called("localStorage.setItem", "expiresAt", expectedExpiresAt).ShouldBeFalse(); + eventTriggered.ShouldBeFalse(); + + result.ShouldBeFalse(); + } + + [Test] + public async Task IsAuthenticatedAsync_WithoutTokenStored_Return() + { + // arrange + var cancellationToken = CancellationToken.None; + var tokenResponse = "a-s-e-c-r-e-t-t-o-k-e-n"; + var refreshTokenResponse = "a-s-e-c-r-e-t-r-e-f-r-e-s-h-t-o-k-e-n"; + var expiresAt = DateTime.UtcNow.AddHours(1); + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + + _jsRuntime.SetupResult("localStorage.getItem", (string)null!, "authToken"); + _jsRuntime.SetupResult("localStorage.getItem", expectedExpiresAt, "expiresAt"); + + // act + var isAuthenticated = await _instance.IsAuthenticatedAsync(cancellationToken); + + // assert + isAuthenticated.ShouldBeFalse(); + } + + [Test] + public async Task IsAuthenticatedAsync_WithoutExpiredStored_Return() + { + // arrange + var cancellationToken = CancellationToken.None; + var tokenResponse = "a-s-e-c-r-e-t-t-o-k-e-n"; + var refreshTokenResponse = "a-s-e-c-r-e-t-r-e-f-r-e-s-h-t-o-k-e-n"; + var expiresAt = DateTime.UtcNow.AddHours(1); + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + + _jsRuntime.SetupResult("localStorage.getItem", tokenResponse, "authToken"); + _jsRuntime.SetupResult("localStorage.getItem", (string)null!, "expiresAt"); + + // act + var isAuthenticated = await _instance.IsAuthenticatedAsync(cancellationToken); + + // assert + isAuthenticated.ShouldBeFalse(); + } + + [Test] + public async Task IsAuthenticatedAsync_WithExpiredToken_Return() + { + // arrange + var cancellationToken = CancellationToken.None; + var tokenResponse = "a-s-e-c-r-e-t-t-o-k-e-n"; + var refreshTokenResponse = "a-s-e-c-r-e-t-r-e-f-r-e-s-h-t-o-k-e-n"; + var expiresAt = DateTime.UtcNow.AddHours(-1); + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + + _jsRuntime.SetupResult("localStorage.getItem", tokenResponse, "authToken"); + _jsRuntime.SetupResult("localStorage.getItem", expectedExpiresAt, "expiresAt"); + + // act + var eventTriggered = false; + _instance.AuthenticationStateChanged += isAuthenticated => + { + isAuthenticated.ShouldBeFalse(); + eventTriggered = true; + }; + var isAuthenticated = await _instance.IsAuthenticatedAsync(cancellationToken); + + // assert + isAuthenticated.ShouldBeFalse(); + _jsRuntime.Called("localStorage.removeItem", "authToken").ShouldBeTrue(); + _jsRuntime.Called("localStorage.removeItem", "refreshToken").ShouldBeTrue(); + _jsRuntime.Called("localStorage.removeItem", "expiresAt").ShouldBeTrue(); + eventTriggered.ShouldBeTrue(); + } + + [Test] + public async Task IsAuthenticatedAsync_WithInvalidExpired_Return() + { + // arrange + var cancellationToken = CancellationToken.None; + var tokenResponse = "a-s-e-c-r-e-t-t-o-k-e-n"; + var refreshTokenResponse = "a-s-e-c-r-e-t-r-e-f-r-e-s-h-t-o-k-e-n"; + var expiresAt = DateTime.UtcNow.AddHours(-1); + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + + _jsRuntime.SetupResult("localStorage.getItem", tokenResponse, "authToken"); + _jsRuntime.SetupResult("localStorage.getItem", "invalid-expired-datetime", "expiresAt"); + + // act + var isAuthenticated = await _instance.IsAuthenticatedAsync(cancellationToken); + + // assert + isAuthenticated.ShouldBeFalse(); + } + + [Test] + public async Task IsAdminOrThrowAsync_WithValidUserAsAdmin_Return() + { + // arrange + var cancellationToken = CancellationToken.None; + var userId = Guid.NewGuid(); + var username = "testuser"; + var expiresAt = DateTime.UtcNow.AddHours(1); + var testToken = JwtTokenHelper.GenerateToken(userId, username, expiresAt, true); + var tokenResponse = testToken.Token; + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + + _jsRuntime.SetupResult("localStorage.getItem", tokenResponse, "authToken"); + _jsRuntime.SetupResult("localStorage.getItem", expectedExpiresAt, "expiresAt"); + + // act & assert + await Should.NotThrowAsync(async () => await _instance.IsAdminOrThrowAsync(CancellationToken.None)); + } + + [Test] + public async Task IsAdminOrThrowAsync_WithValidUserAsNonAdmin_Throws() + { + // arrange + var cancellationToken = CancellationToken.None; + var userId = Guid.NewGuid(); + var username = "testuser"; + var expiresAt = DateTime.UtcNow.AddHours(1); + var testToken = JwtTokenHelper.GenerateToken(userId, username, expiresAt, false); + var tokenResponse = testToken.Token; + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + + _jsRuntime.SetupResult("localStorage.getItem", tokenResponse, "authToken"); + _jsRuntime.SetupResult("localStorage.getItem", expectedExpiresAt, "expiresAt"); + + // act & assert + var exception = await Should.ThrowAsync(async () => + await _instance.IsAdminOrThrowAsync(CancellationToken.None)); + + exception.Message.ShouldBe("User is not authorized to access system information."); + } + + [Test] + public async Task IsAdminOrThrowAsync_WithoutAdminFlagInToken_Throws() + { + // arrange + var cancellationToken = CancellationToken.None; + var userId = Guid.NewGuid(); + var username = "testuser"; + var expiresAt = DateTime.UtcNow.AddHours(1); + var testToken = JwtTokenHelper.GenerateToken(userId, username, expiresAt, false, true); + var tokenResponse = testToken.Token; + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + + _jsRuntime.SetupResult("localStorage.getItem", tokenResponse, "authToken"); + _jsRuntime.SetupResult("localStorage.getItem", expectedExpiresAt, "expiresAt"); + + // act & assert + var exception = await Should.ThrowAsync(async () => + await _instance.IsAdminOrThrowAsync(CancellationToken.None)); + + exception.Message.ShouldBe("User is not authorized to access system information."); + } + + [Test] + public async Task IsAdminOrThrowAsync_WithInvalidToken_Throws() + { + // arrange + var cancellationToken = CancellationToken.None; + var userId = Guid.NewGuid(); + var username = "testuser"; + var expiresAt = DateTime.UtcNow.AddHours(1); + var tokenResponse = "invalid-token"; + var expectedExpiresAt = new DateTimeOffset(expiresAt, TimeSpan.FromHours(0)).DateTime.ToString("O"); + + _jsRuntime.SetupResult("localStorage.getItem", tokenResponse, "authToken"); + _jsRuntime.SetupResult("localStorage.getItem", expectedExpiresAt, "expiresAt"); + + // act & assert + var exception = await Should.ThrowAsync(async () => + await _instance.IsAdminOrThrowAsync(CancellationToken.None)); + + exception.Message.ShouldBe("User is not authorized to access system information."); + } +} diff --git a/source/HomeBook.UnitTests/HomeBook.UnitTests.csproj b/source/HomeBook.UnitTests/HomeBook.UnitTests.csproj index 8af75c08..da34a608 100644 --- a/source/HomeBook.UnitTests/HomeBook.UnitTests.csproj +++ b/source/HomeBook.UnitTests/HomeBook.UnitTests.csproj @@ -22,7 +22,7 @@ runtime; build; native; contentfiles; analyzers; buildtransitive - +
@@ -39,7 +39,6 @@ - diff --git a/source/HomeBook.UnitTests/TestCore/Backend/TestModulesExtensions.cs b/source/HomeBook.UnitTests/TestCore/Backend/TestModulesExtensions.cs new file mode 100644 index 00000000..3c8b1ff9 --- /dev/null +++ b/source/HomeBook.UnitTests/TestCore/Backend/TestModulesExtensions.cs @@ -0,0 +1,27 @@ +using HomeBook.Backend.Core.Search; +using HomeBook.Backend.Factories; +using HomeBook.Backend.ModuleCore; +using HomeBook.Backend.Options; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.DependencyInjection; + +namespace HomeBook.UnitTests.TestCore.Backend; + +public static class TestModulesExtensions +{ + public static IServiceCollection AddBackendModulesForTestEnvironment(this IServiceCollection sc, + IConfiguration c, + SearchRegistrationFactory srf) + { + HomeBookOptions hb = new(); + ModuleBuilder moduleBuilder = new(hb, sc, c); + + moduleBuilder + .AddModule() + .AddModule(); + + moduleBuilder.RegisterModulesInSearchFactory(srf); + + return sc; + } +} diff --git a/source/HomeBook.UnitTests/TestCore/Frontend/TestJSRuntime.cs b/source/HomeBook.UnitTests/TestCore/Frontend/TestJSRuntime.cs new file mode 100644 index 00000000..076999d4 --- /dev/null +++ b/source/HomeBook.UnitTests/TestCore/Frontend/TestJSRuntime.cs @@ -0,0 +1,147 @@ +using Microsoft.JSInterop; + +namespace HomeBook.UnitTests.TestCore.Frontend; + +public class TestJSRuntime : IJSRuntime +{ + private readonly Dictionary _setupResults = new(); + private readonly List<(string Identifier, object?[]? Args)> _calledMethods = new(); + + + /// + /// Sets up a result for a specific JS function call with specific arguments + /// + /// The JS function identifier + /// The result to return when this function is called + /// The arguments that should match for this setup + public void SetupResult(string identifier, object? result, params object?[]? args) + { + string key = CreateKey(identifier, args); + _setupResults[key] = result; + } + + /// + /// Checks if a specific JS function was called with the given arguments + /// + /// The JS function identifier + /// The arguments that were passed + /// True if the function was called with these arguments, false otherwise + public bool Called(string identifier, params object?[]? args) + { + return _calledMethods.Any(call => call.Identifier == identifier && ArgumentsMatch(call.Args, args)); + } + + /// + /// Checks if a specific JS function was called (ignoring arguments) + /// + /// The JS function identifier + /// True if the function was called at least once, false otherwise + public bool HasCall(string identifier) + { + return _calledMethods.Any(call => call.Identifier == identifier); + } + + /// + /// Checks if a specific JS function was called with the given arguments + /// + /// The JS function identifier + /// The arguments that were passed + /// True if the function was called with these arguments, false otherwise + public bool HasCall(string identifier, params object?[]? args) + { + return Called(identifier, args); + } + + /// + /// Creates a unique key from identifier and arguments + /// + /// The JS function identifier + /// The arguments + /// A unique key representing the combination + private static string CreateKey(string identifier, object?[]? args) + { + if (args == null || args.Length == 0) + { + return identifier; + } + + string argsString = string.Join("-", args.Select(arg => arg?.ToString() ?? "null")); + return $"{identifier}-{argsString}"; + } + + /// + /// Gets all recorded method calls + /// + /// List of all method calls with their identifiers and arguments + public IReadOnlyList<(string Identifier, object?[]? Args)> GetAllCalls() + { + return _calledMethods.AsReadOnly(); + } + + /// + /// Clears all recorded method calls and setup results + /// + public void Clear() + { + _calledMethods.Clear(); + _setupResults.Clear(); + } + + public ValueTask InvokeAsync(string identifier, object?[]? args) + { + return InvokeAsync(identifier, CancellationToken.None, args); + } + + public ValueTask InvokeAsync(string identifier, + CancellationToken cancellationToken, + object?[]? args) + { + // Record the method call + _calledMethods.Add((identifier, args)); + + // Create key for lookup + string key = CreateKey(identifier, args); + + // Return configured result if available + if (_setupResults.TryGetValue(key, out object? result)) + { + if (result is TValue typedResult) + { + return ValueTask.FromResult(typedResult); + } + + if (result == null && !typeof(TValue).IsValueType) + { + return ValueTask.FromResult(default(TValue)!); + } + + // Try to convert the result to the expected type + try + { + return ValueTask.FromResult((TValue)Convert.ChangeType(result, typeof(TValue))!); + } + catch + { + // If conversion fails, return default value + return ValueTask.FromResult(default(TValue)!); + } + } + + // Return default value if no result is configured + return ValueTask.FromResult(default(TValue)!); + } + + private static bool ArgumentsMatch(object?[]? args1, object?[]? args2) + { + if (args1 == null && args2 == null) return true; + if (args1 == null || args2 == null) return false; + if (args1.Length != args2.Length) return false; + + for (int i = 0; i < args1.Length; i++) + { + if (!Equals(args1[i], args2[i])) return false; + } + + return true; + } +} diff --git a/source/HomeBook.UnitTests/TestCore/Helper/ApiExceptionExtensions.cs b/source/HomeBook.UnitTests/TestCore/Helper/ApiExceptionExtensions.cs new file mode 100644 index 00000000..554b5a21 --- /dev/null +++ b/source/HomeBook.UnitTests/TestCore/Helper/ApiExceptionExtensions.cs @@ -0,0 +1,12 @@ +using Microsoft.Kiota.Abstractions; + +namespace HomeBook.UnitTests.TestCore.Helper; + +public static class ApiExceptionExtensions +{ + public static ApiException WithStatusCode(this ApiException exception, int statusCode) + { + exception.ResponseStatusCode = statusCode; + return exception; + } +} diff --git a/source/HomeBook.UnitTests/TestCore/Helper/JwtTokenHelper.cs b/source/HomeBook.UnitTests/TestCore/Helper/JwtTokenHelper.cs new file mode 100644 index 00000000..dffd70e7 --- /dev/null +++ b/source/HomeBook.UnitTests/TestCore/Helper/JwtTokenHelper.cs @@ -0,0 +1,99 @@ +using System.IdentityModel.Tokens.Jwt; +using System.Security.Claims; +using System.Security.Cryptography; +using System.Text; +using Microsoft.IdentityModel.Tokens; + +namespace HomeBook.UnitTests.TestCore.Helper; + +public static class JwtTokenHelper +{ + private const string SecretKey = "this-is-a-very-long-secret-key-for-jwt-token-generation"; + private const string Issuer = "HomeBook"; + private const string Audience = "HomeBook"; + + public static JwtTokenResult GenerateToken(Guid userId, + string username, + DateTime expiresAt, + bool isAdmin, + bool removeAdminFlag = false) + { + SymmetricSecurityKey key = new(Encoding.UTF8.GetBytes(SecretKey)); + SigningCredentials credentials = new(key, SecurityAlgorithms.HmacSha256); + + List claims = + [ + new(ClaimTypes.NameIdentifier, userId.ToString()), + new(ClaimTypes.Name, username), + new(JwtRegisteredClaimNames.Jti, Guid.NewGuid().ToString()), + new(JwtRegisteredClaimNames.Iat, + new DateTimeOffset(DateTime.UtcNow).ToUnixTimeSeconds().ToString(), + ClaimValueTypes.Integer64) + ]; + + if (!removeAdminFlag) + claims.Add(new(ClaimTypes.Role, isAdmin ? "Admin" : "User")); + if (!removeAdminFlag) + claims.Add(new Claim("IsAdmin", isAdmin.ToString(), ClaimValueTypes.Boolean)); + + JwtSecurityToken token = new( + issuer: Issuer, + audience: Audience, + claims: claims.ToArray(), + expires: expiresAt, + signingCredentials: credentials + ); + + JwtSecurityTokenHandler tokenHandler = new(); + string? tokenString = tokenHandler.WriteToken(token); + string refreshToken = GenerateRefreshToken(); + + return new JwtTokenResult + { + Token = tokenString, + RefreshToken = refreshToken, + ExpiresAt = expiresAt, + UserId = userId, + Username = username + }; + } + + private static string GenerateRefreshToken() + { + var randomNumber = new byte[64]; + using var rng = RandomNumberGenerator.Create(); + rng.GetBytes(randomNumber); + return Convert.ToBase64String(randomNumber); + } +} + +/// +/// Result model for JWT token generation +/// +public record JwtTokenResult +{ + /// + /// The JWT access token + /// + public required string Token { get; init; } + + /// + /// The refresh token + /// + public required string RefreshToken { get; init; } + + /// + /// Token expiration date and time + /// + public required DateTime ExpiresAt { get; init; } + + /// + /// User ID associated with the token + /// + public required Guid UserId { get; init; } + + /// + /// Username associated with the token + /// + public required string Username { get; init; } +} diff --git a/source/Homebook.Backend.Core.Setup/Homebook.Backend.Core.Setup.csproj b/source/Homebook.Backend.Core.Setup/Homebook.Backend.Core.Setup.csproj index 5d9488b3..4e45dbf1 100644 --- a/source/Homebook.Backend.Core.Setup/Homebook.Backend.Core.Setup.csproj +++ b/source/Homebook.Backend.Core.Setup/Homebook.Backend.Core.Setup.csproj @@ -7,7 +7,7 @@ - +