diff --git a/src/Api/Core/FluentCMS.Api.Core.Events/Block.cs b/src/Api/Core/FluentCMS.Api.Core.Events/Block.cs new file mode 100644 index 000000000..8810b3c47 --- /dev/null +++ b/src/Api/Core/FluentCMS.Api.Core.Events/Block.cs @@ -0,0 +1,21 @@ +namespace FluentCMS.Api.Core.Events; + +public class BlockAddedEvent(Block block) : EventBase +{ + public Block Block { get; } = block; +} + +public class BlockUpdatedEvent(Block block) : EventBase +{ + public Block Block { get; } = block; +} + +public class BlockRemovedEvent(Block block) : EventBase +{ + public Block Block { get; } = block; +} + +public class BlocksRemovedBySiteEvent(IEnumerable blocks) : EventBase +{ + public IEnumerable Block { get; } = blocks; +} diff --git a/src/Api/Core/FluentCMS.Api.Core.Events/Page.cs b/src/Api/Core/FluentCMS.Api.Core.Events/Page.cs new file mode 100644 index 000000000..628f5058e --- /dev/null +++ b/src/Api/Core/FluentCMS.Api.Core.Events/Page.cs @@ -0,0 +1,16 @@ +namespace FluentCMS.Api.Core.Events; + +public class PageAddedEvent(Page page) : EventBase +{ + public Page Page { get; } = page; +} + +public class PageUpdatedEvent(Page page) : EventBase +{ + public Page Page { get; } = page; +} + +public class PageRemovedEvent(Page page) : EventBase +{ + public Page Page { get; } = page; +} diff --git a/src/Api/Core/FluentCMS.Api.Core.Models/Block.cs b/src/Api/Core/FluentCMS.Api.Core.Models/Block.cs new file mode 100644 index 000000000..10b27dbe8 --- /dev/null +++ b/src/Api/Core/FluentCMS.Api.Core.Models/Block.cs @@ -0,0 +1,9 @@ +namespace FluentCMS.Api.Core.Models; + +public class Block : SiteAssociatedEntity +{ + public string Name { get; set; } = string.Empty; + public string Category { get; set; } = string.Empty; + public string? Description { get; set; } + public string Content { get; set; } = string.Empty; +} \ No newline at end of file diff --git a/src/Api/Core/FluentCMS.Api.Core.Models/File.cs b/src/Api/Core/FluentCMS.Api.Core.Models/File.cs index baadf248f..5c49ac752 100644 --- a/src/Api/Core/FluentCMS.Api.Core.Models/File.cs +++ b/src/Api/Core/FluentCMS.Api.Core.Models/File.cs @@ -5,6 +5,7 @@ public class File : SiteAssociatedEntity public string Name { get; set; } = default!; public string NormalizedName { get; set; } = default!; public Guid FolderId { get; set; } + public Guid SiteId { get; set; } public string Extension { get; set; } = default!; public string ContentType { get; set; } = default!; public long Size { get; set; } diff --git a/src/Api/Core/FluentCMS.Api.Core.Models/Folder.cs b/src/Api/Core/FluentCMS.Api.Core.Models/Folder.cs index f89d402ed..ac5e01eeb 100644 --- a/src/Api/Core/FluentCMS.Api.Core.Models/Folder.cs +++ b/src/Api/Core/FluentCMS.Api.Core.Models/Folder.cs @@ -1,8 +1,19 @@ -namespace FluentCMS.Api.Core.Models; +using System.Text.Json.Serialization; + +namespace FluentCMS.Api.Core.Models; public class Folder : SiteAssociatedEntity { public string Name { get; set; } = default!; public string NormalizedName { get; set; } = default!; + public Guid SiteId { get; set; } public Guid? ParentId { get; set; } + public long Size { get; set; } + + public ICollection Files { get; set; } = new List(); + public ICollection Folders { get; set; } = new List(); + + + [JsonIgnore] + public Folder? ParentFolder { get; set; } = default!; } diff --git a/src/Api/Core/FluentCMS.Api.Core.Models/Identity/Role.cs b/src/Api/Core/FluentCMS.Api.Core.Models/Identity/Role.cs index 390708c3f..0f15b324a 100644 --- a/src/Api/Core/FluentCMS.Api.Core.Models/Identity/Role.cs +++ b/src/Api/Core/FluentCMS.Api.Core.Models/Identity/Role.cs @@ -1,6 +1,6 @@ namespace FluentCMS.Api.Core.Models.Identity; -public class Role : IdentityRole, ISiteAssociatedEntity +public class Role : IdentityRole, IEntity { public string Description { get; set; } = string.Empty; @@ -12,7 +12,6 @@ public class Role : IdentityRole, ISiteAssociatedEntity public string? UpdatedBy { get; set; } public DateTime? UpdatedAt { get; set; } public int Version { get; set; } - public Guid SiteId { get; set; } public Role() { diff --git a/src/Api/Core/FluentCMS.Api.Core.Models/Identity/UserRole.cs b/src/Api/Core/FluentCMS.Api.Core.Models/Identity/UserRole.cs index 0f3abea01..5e28b1cdb 100644 --- a/src/Api/Core/FluentCMS.Api.Core.Models/Identity/UserRole.cs +++ b/src/Api/Core/FluentCMS.Api.Core.Models/Identity/UserRole.cs @@ -1,6 +1,6 @@ namespace FluentCMS.Api.Core.Models.Identity; -public class UserRole : IdentityUserRole, ISiteAssociatedEntity +public class UserRole : IdentityUserRole { // IAuditableEntity implementations public Guid Id { get; set; } @@ -9,5 +9,4 @@ public class UserRole : IdentityUserRole, ISiteAssociatedEntity public string? UpdatedBy { get; set; } public DateTime? UpdatedAt { get; set; } public int Version { get; set; } - public Guid SiteId { get; set; } } diff --git a/src/Api/Core/FluentCMS.Api.Core.Models/Layout.cs b/src/Api/Core/FluentCMS.Api.Core.Models/Layout.cs index 955f44a69..49aedc803 100644 --- a/src/Api/Core/FluentCMS.Api.Core.Models/Layout.cs +++ b/src/Api/Core/FluentCMS.Api.Core.Models/Layout.cs @@ -2,6 +2,7 @@ public class Layout : SiteAssociatedEntity { + public Guid SiteId { get; set; } public string Name { get; set; } = default!; public string Body { get; set; } = default!; public string Head { get; set; } = default!; diff --git a/src/Api/Core/FluentCMS.Api.Core.Models/Page.cs b/src/Api/Core/FluentCMS.Api.Core.Models/Page.cs index d42c2fead..576222595 100644 --- a/src/Api/Core/FluentCMS.Api.Core.Models/Page.cs +++ b/src/Api/Core/FluentCMS.Api.Core.Models/Page.cs @@ -3,11 +3,18 @@ public class Page : SiteAssociatedEntity { public string Title { get; set; } = string.Empty; - public Guid? ParentId { get; set; } + public string Slug { get; set; } = string.Empty; public int Order { get; set; } - public string Path { get; set; } = string.Empty; // URL path, only one segment without forward slash (/) + public Guid SiteId { get; set; } + public Guid? ParentId { get; set; } public Guid? LayoutId { get; set; } public Guid? EditLayoutId { get; set; } public Guid? DetailLayoutId { get; set; } - public bool Locked { get; set; } = false; -} + public string? MetaTitle { get; set; } + public string? MetaDescription { get; set; } + public bool? RobotsIndex { get; set; } + public bool? RobotsFollow { get; set; } + public string? OgType { get; set; } + public string? Head { get; set; } + // public bool Locked { get; set; } = false; +} \ No newline at end of file diff --git a/src/Api/Core/FluentCMS.Api.Core.Models/Site.cs b/src/Api/Core/FluentCMS.Api.Core.Models/Site.cs index f10ccfbc8..fdb94b266 100644 --- a/src/Api/Core/FluentCMS.Api.Core.Models/Site.cs +++ b/src/Api/Core/FluentCMS.Api.Core.Models/Site.cs @@ -3,9 +3,17 @@ public class Site : AuditableEntity { public string Name { get; set; } = default!; - public string? Description { get; set; } public List Urls { get; set; } = []; - public Guid LayoutId { get; set; } - public Guid DetailLayoutId { get; set; } - public Guid EditLayoutId { get; set; } + public string? Description { get; set; } + public Guid? LayoutId { get; set; } + public Guid? DetailLayoutId { get; set; } + public Guid? EditLayoutId { get; set; } + public string? MetaTitle { get; set; } + public string? MetaDescription { get; set; } + public bool? RobotsIndex { get; set; } + public bool? RobotsFollow { get; set; } + public string? RobotsTxt { get; set; } + public string? GoogleTagsId { get; set; } + public string? OgType { get; set; } + public string? Head { get; set; } } diff --git a/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/IFileRepository.cs b/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/IFileRepository.cs deleted file mode 100644 index a624a9a2a..000000000 --- a/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/IFileRepository.cs +++ /dev/null @@ -1,8 +0,0 @@ -using File = FluentCMS.Api.Core.Models.File; - -namespace FluentCMS.Api.Core.Repositories.Abstractions; - -public interface IFileRepository : ISiteAssociatedRepository -{ - Task GetByName(Guid siteId, Guid folderId, string normalizedFileName, CancellationToken cancellationToken = default); -} diff --git a/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/IFolderRepository.cs b/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/IFolderRepository.cs deleted file mode 100644 index 0c0e04ac5..000000000 --- a/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/IFolderRepository.cs +++ /dev/null @@ -1,6 +0,0 @@ -namespace FluentCMS.Api.Core.Repositories.Abstractions; - -public interface IFolderRepository : ISiteAssociatedRepository -{ - Task GetByName(Guid siteId, Guid? parentId, string normalizedName, CancellationToken cancellationToken = default); -} diff --git a/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/ILayoutRepository.cs b/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/ILayoutRepository.cs deleted file mode 100644 index d6dc2753a..000000000 --- a/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/ILayoutRepository.cs +++ /dev/null @@ -1,5 +0,0 @@ -namespace FluentCMS.Api.Core.Repositories.Abstractions; - -public interface ILayoutRepository : ISiteAssociatedRepository -{ -} diff --git a/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/IPageRepository.cs b/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/IPageRepository.cs deleted file mode 100644 index c07de4fdb..000000000 --- a/src/Api/Core/FluentCMS.Api.Core.Repositories.Abstractions/IPageRepository.cs +++ /dev/null @@ -1,5 +0,0 @@ -namespace FluentCMS.Api.Core.Repositories.Abstractions; - -public interface IPageRepository : ISiteAssociatedRepository -{ -} diff --git a/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/FileRepository.cs b/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/FileRepository.cs deleted file mode 100644 index eafc5bca6..000000000 --- a/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/FileRepository.cs +++ /dev/null @@ -1,11 +0,0 @@ -using File = FluentCMS.Api.Core.Models.File; - -namespace FluentCMS.Api.Core.Repositories.EntityFramework; - -public class FileRepository(CmsCoreDbContext dbContext) : SiteAssociatedRepository(dbContext), IFileRepository -{ - public async Task GetByName(Guid siteId, Guid folderId, string normalizedFileName, CancellationToken cancellationToken = default) - { - return await Query().FirstOrDefault(x => x.SiteId == siteId && x.FolderId == folderId && x.NormalizedName == normalizedFileName, cancellationToken); - } -} diff --git a/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/FolderRepository.cs b/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/FolderRepository.cs deleted file mode 100644 index 12907f20c..000000000 --- a/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/FolderRepository.cs +++ /dev/null @@ -1,9 +0,0 @@ -namespace FluentCMS.Api.Core.Repositories.EntityFramework; - -public class FolderRepository(CmsCoreDbContext dbContext) : SiteAssociatedRepository(dbContext), IFolderRepository -{ - public async Task GetByName(Guid siteId, Guid? parentId, string normalizedName, CancellationToken cancellationToken = default) - { - return await Query().FirstOrDefault(x => x.SiteId == siteId && x.ParentId == parentId && x.NormalizedName == normalizedName, cancellationToken); - } -} diff --git a/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/LayoutRepository.cs b/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/LayoutRepository.cs deleted file mode 100644 index 67e99f91f..000000000 --- a/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/LayoutRepository.cs +++ /dev/null @@ -1,3 +0,0 @@ -namespace FluentCMS.Api.Core.Repositories.EntityFramework; - -public class LayoutRepository(CmsCoreDbContext dbContext) : SiteAssociatedRepository(dbContext), ILayoutRepository; diff --git a/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/PageRepository.cs b/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/PageRepository.cs deleted file mode 100644 index 72e14a3ae..000000000 --- a/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/PageRepository.cs +++ /dev/null @@ -1,5 +0,0 @@ -namespace FluentCMS.Api.Core.Repositories.EntityFramework; - -public class PageRepository(CmsCoreDbContext dbContext) : SiteAssociatedRepository(dbContext), IPageRepository -{ -} diff --git a/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/ServiceCollectionExtensions.cs b/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/ServiceCollectionExtensions.cs index 6c0c41a79..c8e015f0a 100644 --- a/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/ServiceCollectionExtensions.cs +++ b/src/Api/Core/FluentCMS.Api.Core.Repositories.EntityFramework/ServiceCollectionExtensions.cs @@ -5,10 +5,6 @@ public static class ServiceCollectionExtensions public static IServiceCollection AddEntityFrameworkRpositories(this IServiceCollection services) { services.AddScoped(); - services.AddScoped(); - services.AddScoped(); - services.AddScoped(); - services.AddScoped(); services.AddScoped(); services.AddScoped(); services.AddScoped(); diff --git a/src/Api/FluentCMS.Api/.gitignore b/src/Api/FluentCMS.Api/.gitignore new file mode 100644 index 000000000..45afc3700 --- /dev/null +++ b/src/Api/FluentCMS.Api/.gitignore @@ -0,0 +1 @@ +files \ No newline at end of file diff --git a/src/Api/FluentCMS.Api/FluentCMS.Api.csproj b/src/Api/FluentCMS.Api/FluentCMS.Api.csproj index b03a44eb7..c8ffa1bc5 100644 --- a/src/Api/FluentCMS.Api/FluentCMS.Api.csproj +++ b/src/Api/FluentCMS.Api/FluentCMS.Api.csproj @@ -15,6 +15,7 @@ + diff --git a/src/Api/FluentCMS.Api/Program.cs b/src/Api/FluentCMS.Api/Program.cs index 92e95e9d3..079937ac4 100644 --- a/src/Api/FluentCMS.Api/Program.cs +++ b/src/Api/FluentCMS.Api/Program.cs @@ -1,6 +1,7 @@ using FluentCMS.Api.Core.Filters; using FluentCMS.Api.Core.Repositories.EntityFramework; using FluentCMS.Api.Plugins.TodoManagement.Repositories; +using FluentCMS.Api.Plugins.AIAgentManagement.Repositories; using FluentCMS.Infrastructure.Configuration.EntityFramework; using FluentCMS.Infrastructure.Configuration.EntityFramework.Sqlite; using FluentCMS.Infrastructure.EventBus.InMemory; @@ -54,6 +55,20 @@ validationOptions.IgnoreExceptions = false; // Fail fast on errors validationOptions.Conditions.Add(new EnvironmentCondition(builder.Environment, e => e.IsDevelopment())); }); + + // Specific database for Agent library + options.For() + .UseSqlite("DataSource=agent.db;Cache=Shared") + .EnableDataSeeding(seedingOptions => + { + seedingOptions.IgnoreExceptions = false; // Fail fast on errors + seedingOptions.Conditions.Add(new EnvironmentCondition(builder.Environment, e => e.IsDevelopment())); + }) + .EnableSchemaValidation(validationOptions => + { + validationOptions.IgnoreExceptions = false; // Fail fast on errors + validationOptions.Conditions.Add(new EnvironmentCondition(builder.Environment, e => e.IsDevelopment())); + }); }); services.AddDbConfiguration(); @@ -72,7 +87,7 @@ // Add plugin system services.AddPluginSystem(builder.Configuration, options => { - options.ScanAssemblyPatterns = ["FluentCMS.*"]; + options.ScanAssemblyPatterns = ["FluentCMS.Api.Plugins.*"]; options.LoggerFactory = loggerFactory; }); diff --git a/src/Api/FluentCMS.Api/appsettings.json b/src/Api/FluentCMS.Api/appsettings.json index e0315e0af..0ead022f3 100644 --- a/src/Api/FluentCMS.Api/appsettings.json +++ b/src/Api/FluentCMS.Api/appsettings.json @@ -2,6 +2,13 @@ "ConnectionStrings": { "DefaultConnection": "DataSource=app.db;Cache=Shared" }, + "OpenRouter": { + "ApiKey": "", + "Model": "openai/gpt-4.1-nano", + "AttributionReferer": "https://fluentcms.com", + "AttributionTitle": "FluentCMS", + "IncludeStandardReferrerHeader": true + }, "Providers": { "Caching": [ { diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/AIAgentPlugin.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/AIAgentPlugin.cs new file mode 100644 index 000000000..54c4ccbc0 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/AIAgentPlugin.cs @@ -0,0 +1,20 @@ +namespace FluentCMS.Api.Plugins.AIAgentManagement; + +[Plugin] +public class AIAgentPlugin : IPluginStartup +{ + public void Configure(IApplicationBuilder app) + { + } + + public void ConfigureServices(IServiceCollection services, IConfiguration? configuration) + { + services.AddDatabaseContext(); + services.AddDataSeeder(); + services.AddSchemaValidator(); + services.AddScoped(); + services.AddScoped(); + services.AddScoped(); + services.AddOpenRouterChatClient(configuration); + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Controllers/AgentsController.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Controllers/AgentsController.cs new file mode 100644 index 000000000..7d46bad4d --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Controllers/AgentsController.cs @@ -0,0 +1,146 @@ +using Microsoft.AspNetCore.Mvc; +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +namespace FluentCMS.Api.Plugins.AIAgentManagement.Controllers; + +[ApiController] +public class AgentsController : BaseController +{ + private readonly IAgentRepository _agentRepository; + private readonly ILogger _logger; + + public AgentsController(IAgentRepository agentRepository, ILogger logger) + { + _agentRepository = agentRepository; + _logger = logger; + } + + private static AgentResponseDto MapToResponseDto(Agent agent) + { + return new AgentResponseDto + { + Id = agent.Id, + Name = agent.Name, + Description = agent.Description, + Model = agent.Model, + SystemPrompt = agent.SystemPrompt, + }; + } + + [HttpGet] + public async Task> GetAll(CancellationToken cancellationToken) + { + var agents = await _agentRepository.Query().ToList(cancellationToken); + + var agentResponses = agents.Select(MapToResponseDto).ToList(); + return SuccessList(agentResponses, 1, agentResponses.Count, agentResponses.Count); + + } + + [HttpGet("{id}")] + public async Task>> GetById(Guid id, CancellationToken cancellationToken) + { + var agent = await _agentRepository.Query().OrderBy(agent => agent.Id).FirstOrDefault(a => a.Id == id, cancellationToken); + if (agent == null) + return NotFound("Agent not found"); + + var agentResponse = MapToResponseDto(agent); + return Success(agentResponse); + } + + [HttpPost] + public async Task> Create([FromBody] Agent createAgentRequest) + { + try + { + if (!ModelState.IsValid) + { + return BadRequest(ModelState); + } + + var createdAgent = await _agentRepository.Add(createAgentRequest); + return CreatedAtAction(nameof(GetById), new { id = createdAgent.Id }, createdAgent); + } + catch (Exception ex) + { + _logger.LogError(ex, "Error creating agent"); + return StatusCode(500, "An error occurred while creating the agent"); + } + } + + [HttpPut] + public async Task> Update([FromBody] Agent updateAgentRequest) + { + try + { + if (!ModelState.IsValid) + { + return BadRequest(ModelState); + } + + var updatedAgent = await _agentRepository.Update(updateAgentRequest); + + if (updatedAgent == null) + { + return NotFound($"Agent with ID {updateAgentRequest.Id} not found"); + } + + return Ok(updatedAgent); + } + catch (Exception ex) + { + return StatusCode(500, "An error occurred while updating the agent"); + } + } + + [HttpDelete("{id:guid}")] + public async Task Remove(Guid id, CancellationToken cancellationToken) + { + try + { + var agent = await _agentRepository.Query().OrderBy(agent => agent.Id).FirstOrDefault(a => a.Id == id, cancellationToken); + if (agent is null) + { + return NotFound($"Agent with ID {id} not found"); + } + + var success = await _agentRepository.Remove(agent, cancellationToken); + + + return NoContent(); + } + catch (Exception ex) + { + _logger.LogError(ex, "Error deleting agent with ID {Id}", id); + return StatusCode(500, "An error occurred while deleting the agent"); + } + } +} + + +public class CreateAgentRequest +{ + public string Name { get; set; } + public string Description { get; set; } + public string Model { get; set; } + public string SystemPrompt { get; set; } +} + +public class UpdateAgentRequest +{ + public Guid Id { get; set; } + public string? Name { get; set; } + public string? Description { get; set; } + public string? Model { get; set; } + public string? SystemPrompt { get; set; } +} + +public class AgentResponseDto +{ + public Guid Id { get; set; } + public string Name { get; set; } + public string Description { get; set; } + public string Model { get; set; } + public string SystemPrompt { get; set; } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Controllers/ThreadsController.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Controllers/ThreadsController.cs new file mode 100644 index 000000000..27f9a213a --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Controllers/ThreadsController.cs @@ -0,0 +1,51 @@ +using Microsoft.Agents.AI; +using Microsoft.Extensions.AI; + +namespace FluentCMS.Api.Plugins.AIAgentManagement.Controllers; + +public class ThreadsController(OpenRouterClient openRouterClient, Tools tools) : BaseController +{ + [HttpPost] + public async Task> Start(CancellationToken cancellationToken = default) + { + var chatClient = openRouterClient.GetChatClient(); + AIAgent agent = chatClient.CreateAIAgent(instructions: "You are a senior full-stack software developer", + tools: [ + AIFunctionFactory.Create(tools.ListFiles), + AIFunctionFactory.Create(tools.WriteFile), + AIFunctionFactory.Create(tools.ReadFile)]); + + + var path = "C:\\Projects\\microsoft-agent\\ConsoleApp1"; + var instructions = $""" + You are a senior full-stack software developer. + Your task is to review and understand the source code files in the given path: {path}. + After reviewing the files, you need to create a comprehensive README.md file in markdown format that explains the source code. + The README.md file should include: + 1. An overview of the project. + 2. A brief description of each source code file and its purpose. + 3. Any important classes, methods, or functions and their roles. + 4. Instructions on how to set up and run the project if applicable. + Please write the markdown content in {path}\\README.md file into the root folder + """; + // Non-streaming agent interaction with function tools. + var agentResponse = await agent.RunAsync(instructions, cancellationToken: cancellationToken); + return Success(agentResponse); + } + + +} + +//public class ThreadDto +//{ +// public Guid? Id { get; set; } +// public Guid AgentId { get; set; } +// public Guid TemplateId { get; set; } +// public string Message { get; set; } +//} + +//public class ThreadResponseDto: AgentRunResponse +//{ +// public Guid Id { get; set; } +// public Guid TemplateId { get; set; } +//} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Controllers/ThreadsManagementController.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Controllers/ThreadsManagementController.cs new file mode 100644 index 000000000..8fb9ac131 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Controllers/ThreadsManagementController.cs @@ -0,0 +1,145 @@ +using Microsoft.AspNetCore.Mvc; +// using Microsoft.Threads.AI; +using Microsoft.Extensions.AI; + +namespace FluentCMS.Api.Plugins.AIAgentManagement.Controllers; + +[ApiController] +public class ThreadsManagementController : BaseController +{ + private readonly IThreadRepository _threadRepository; + private readonly ILogger _logger; + + public ThreadsManagementController(IThreadRepository threadRepository, ILogger logger) + { + _threadRepository = threadRepository; + _logger = logger; + } + + private static ThreadResponseDto MapToResponseDto(AIThread thread) + { + return new ThreadResponseDto + { + Id = thread.Id, + Name = thread.Name, + Description = thread.Description, + Model = thread.Model, + SystemPrompt = thread.SystemPrompt, + }; + } + + [HttpGet] + public async Task> GetAll(CancellationToken cancellationToken) + { + var threads = await _threadRepository.Query().ToList(cancellationToken); + + var threadResponses = threads.Select(MapToResponseDto).ToList(); + return SuccessList(threadResponses, 1, threadResponses.Count, threadResponses.Count); + + } + + [HttpGet("{id}")] + public async Task>> GetById(Guid id, CancellationToken cancellationToken) + { + var thread = await _threadRepository.Query().OrderBy(thread => thread.Id).FirstOrDefault(a => a.Id == id, cancellationToken); + if (thread == null) + return NotFound("AIThread not found"); + + var threadResponse = MapToResponseDto(thread); + return Success(threadResponse); + } + + [HttpPost] + public async Task> Create([FromBody] AIThread createThreadRequest) + { + try + { + if (!ModelState.IsValid) + { + return BadRequest(ModelState); + } + + var createdThread = await _threadRepository.Add(createThreadRequest); + return CreatedAtAction(nameof(GetById), new { id = createdThread.Id }, createdThread); + } + catch (Exception ex) + { + _logger.LogError(ex, "Error creating thread"); + return StatusCode(500, "An error occurred while creating the thread"); + } + } + + [HttpPut] + public async Task> Update([FromBody] AIThread updateThreadRequest) + { + try + { + if (!ModelState.IsValid) + { + return BadRequest(ModelState); + } + + var updatedThread = await _threadRepository.Update(updateThreadRequest); + + if (updatedThread == null) + { + return NotFound($"AIThread with ID {updateThreadRequest.Id} not found"); + } + + return Ok(updatedThread); + } + catch (Exception ex) + { + return StatusCode(500, "An error occurred while updating the thread"); + } + } + + [HttpDelete("{id:guid}")] + public async Task Remove(Guid id, CancellationToken cancellationToken) + { + try + { + var thread = await _threadRepository.Query().OrderBy(thread => thread.Id).FirstOrDefault(a => a.Id == id, cancellationToken); + if (thread is null) + { + return NotFound($"AIThread with ID {id} not found"); + } + + var success = await _threadRepository.Remove(thread, cancellationToken); + + + return NoContent(); + } + catch (Exception ex) + { + _logger.LogError(ex, "Error deleting thread with ID {Id}", id); + return StatusCode(500, "An error occurred while deleting the thread"); + } + } +} + +public class CreateThreadRequest +{ + public string Name { get; set; } + public string Description { get; set; } + public string Model { get; set; } + public string SystemPrompt { get; set; } +} + +public class UpdateThreadRequest +{ + public Guid Id { get; set; } + public string? Name { get; set; } + public string? Description { get; set; } + public string? Model { get; set; } + public string? SystemPrompt { get; set; } +} + +public class ThreadResponseDto +{ + public Guid Id { get; set; } + public string Name { get; set; } + public string Description { get; set; } + public string Model { get; set; } + public string SystemPrompt { get; set; } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/FluentCMS.Api.Plugins.AIAgentManagement.csproj b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/FluentCMS.Api.Plugins.AIAgentManagement.csproj new file mode 100644 index 000000000..bfa876cad --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/FluentCMS.Api.Plugins.AIAgentManagement.csproj @@ -0,0 +1,29 @@ + + + + net9.0 + enable + enable + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Models/AIThread.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Models/AIThread.cs new file mode 100644 index 000000000..91dd5a62e --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Models/AIThread.cs @@ -0,0 +1,9 @@ +namespace FluentCMS.Api.Plugins.AIAgentManagement.Models; + +public class AIThread : AuditableEntity +{ + public string Name { get; set; } = string.Empty; + public string Description { get; set; } = string.Empty; + public string Model { get; set; } + public string SystemPrompt { get; set; } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Models/Agent.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Models/Agent.cs new file mode 100644 index 000000000..9b3c615fe --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Models/Agent.cs @@ -0,0 +1,9 @@ +namespace FluentCMS.Api.Plugins.AIAgentManagement.Models; + +public class Agent : AuditableEntity +{ + public string Name { get; set; } = string.Empty; + public string Description { get; set; } = string.Empty; + public string Model { get; set; } + public string SystemPrompt { get; set; } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Models/AgentInfo.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Models/AgentInfo.cs new file mode 100644 index 000000000..9b05125f6 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Models/AgentInfo.cs @@ -0,0 +1,33 @@ +namespace FluentCMS.Api.Plugins.AIAgentManagement.Models; + +public class AgentInfo : AuditableEntity +{ + public string Name { get; set; } = string.Empty; + public string Description { get; set; } = string.Empty; + public List Functions { get; set; } = []; + public string Instructions { get; set; } = string.Empty; +} + +public class FunctionInfo: AuditableEntity +{ + public string Name { get; set; } = string.Empty; + public string? Description { get; set; } + public string Type { get; set; } = string.Empty; + public string Method { get; set; } = string.Empty; +} + +public class LlmInfo : AuditableEntity +{ + +} + +public class ProjectInfo : AuditableEntity +{ + public string Name { get; set; } = string.Empty; + public string Description { get; set; } = string.Empty; +} + +public class ChatTemplate +{ + +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/OpenRouterClient.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/OpenRouterClient.cs new file mode 100644 index 000000000..71e78da55 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/OpenRouterClient.cs @@ -0,0 +1,74 @@ +namespace FluentCMS.Api.Plugins.AIAgentManagement; + +/// +/// Options for configuring an OpenRouter-backed chat client. +/// +public sealed class OpenRouterClientOptions +{ + /// + /// OpenRouter API key (required). Example: "sk-or-v1-..." + /// + public string ApiKey { get; set; } = string.Empty; + + /// + /// Default model slug (required). Example: "openrouter/auto" or "anthropic/claude-3.5-sonnet". + /// + public string Model { get; set; } = "openrouter/auto"; + + /// + /// OpenRouter base endpoint. Defaults to "https://openrouter.ai/api/v1". + /// + public Uri Endpoint { get; set; } = new Uri("https://openrouter.ai/api/v1"); + + /// + /// Optional attribution header. Use your site/app URL (scheme + host). + /// OpenRouter recommends setting "HTTP-Referer" to your public URL for analytics. + /// + public string? AttributionReferer { get; set; } + + /// + /// Optional attribution header. A short, human-friendly app title. + /// OpenRouter recommends "X-Title" for analytics. + /// + public string? AttributionTitle { get; set; } + + /// + /// Optional request timeout (HttpClient). Default: 100 seconds. + /// + public TimeSpan Timeout { get; set; } = TimeSpan.FromSeconds(100); + + /// + /// Set true to also include the standard "Referer" header alongside "HTTP-Referer". + /// (Some proxies strip non-standard headers; this duplicates for safety.) + /// + public bool IncludeStandardReferrerHeader { get; set; } = true; +} + +public static class OpenRouterServiceCollectionExtensions +{ + public static IServiceCollection AddOpenRouterChatClient( + this IServiceCollection services, + IConfiguration config, + string sectionName = "OpenRouter") + { + var options = config.GetSection(sectionName); + services.Configure(options); + services.TryAddSingleton(); + return services; + } +} + +public class OpenRouterClient(IOptions options) +{ + public ChatClient GetChatClient() + { + var opt = options.Value; + var aiClient = new OpenAIClientOptions + { + Endpoint = opt.Endpoint, + }; + var credential = new ApiKeyCredential(opt.ApiKey); + var client = new ChatClient(model: opt.Model, credential: credential, options: aiClient); + return client; + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AIDbContext.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AIDbContext.cs new file mode 100644 index 000000000..4f41da321 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AIDbContext.cs @@ -0,0 +1,57 @@ +namespace FluentCMS.Api.Plugins.AIAgentManagement.Repositories; + +internal class AIDbContext(DbContextOptions options) : DbContext(options), IAIAgentDatabaseMarker +{ + public DbSet Threads => Set(); + public DbSet Agents => Set(); + + protected override void OnModelCreating(ModelBuilder modelBuilder) + { + base.OnModelCreating(modelBuilder); + + // Configure the Agent entity + modelBuilder.Entity() + .HasKey(t => t.Id); + + modelBuilder.Entity() + .Property(t => t.Name) + .IsRequired() + .HasMaxLength(150); + + modelBuilder.Entity() + .Property(t => t.Model) + .IsRequired() + .HasMaxLength(150); + + modelBuilder.Entity() + .Property(t => t.Description) + .HasMaxLength(500); + + modelBuilder.Entity() + .Property(t => t.SystemPrompt) + .HasMaxLength(5000); + + + // Configure the Thread entity + modelBuilder.Entity() + .HasKey(t => t.Id); + + modelBuilder.Entity() + .Property(t => t.Name) + .IsRequired() + .HasMaxLength(150); + + modelBuilder.Entity() + .Property(t => t.Model) + .IsRequired() + .HasMaxLength(150); + + modelBuilder.Entity() + .Property(t => t.Description) + .HasMaxLength(500); + + modelBuilder.Entity() + .Property(t => t.SystemPrompt) + .HasMaxLength(5000); + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AgentDataSeeder.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AgentDataSeeder.cs new file mode 100644 index 000000000..bd32f79ae --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AgentDataSeeder.cs @@ -0,0 +1,50 @@ +namespace FluentCMS.Api.Plugins.AIAgentManagement.Repositories; + +internal class AgentDataSeeder(AIDbContext dbContext, ILogger logger) : BaseDataSeeder(dbContext, logger) +{ + public override int Priority => 10000; + + public override async Task SeedData(CancellationToken cancellationToken = default) + { + logger.LogInformation("Seeding initial agent items into the database..."); + await DbContext.Agents.AddRangeAsync([ + new Agent + { + Name = "Code Companion", + Description = "Assists with writing, refactoring, and optimizing code across multiple languages.", + SystemPrompt = "You are an expert software engineer. Generate clean, efficient, and secure code. Avoid unnecessary comments and keep responses minimal.", + Model = "openai/gpt-4.1" + }, + new Agent + { + Name = "Debug Analyzer", + Description = "Specializes in debugging and error diagnosis for C#, JavaScript, and Python codebases.", + SystemPrompt = "You are a debugging assistant. Identify issues, explain causes briefly, and propose direct fixes.", + Model = "openai/gpt-4.1-mini" + }, + new Agent + { + Name = "Architecture Advisor", + Description = "Provides guidance on software design patterns, system architecture, and scalability decisions.", + SystemPrompt = "You are a senior software architect. Recommend architecture decisions based on scalability, maintainability, and performance.", + Model = "openai/gpt-4.1" + }, + new Agent + { + Name = "Test Writer", + Description = "Generates and improves unit, integration, and end-to-end tests for various frameworks.", + SystemPrompt = "You are a testing expert. Write thorough, maintainable, and minimal test code using best practices.", + Model = "openai/gpt-4.1-mini" + }, + new Agent + { + Name = "Doc Generator", + Description = "Creates concise technical documentation, code summaries, and API references.", + SystemPrompt = "You are a technical documentation assistant. Generate clear, professional, and accurate documentation from code or descriptions.", + Model = "openai/gpt-4.1-nano" + } + ], cancellationToken); + + await DbContext.SaveChangesAsync(cancellationToken); + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AgentRepository.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AgentRepository.cs new file mode 100644 index 000000000..e5680f78c --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AgentRepository.cs @@ -0,0 +1,10 @@ +using FluentCMS.Api.Plugins.AIAgentManagement.Models; + +namespace FluentCMS.Api.Plugins.AIAgentManagement.Repositories; + + +public interface IAgentRepository : IRepository +{ +} + +internal class AgentRepository(AIDbContext dbContext) : Repository(dbContext), IAgentRepository; diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AgentSchemaValidator.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AgentSchemaValidator.cs new file mode 100644 index 000000000..f26f31085 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/AgentSchemaValidator.cs @@ -0,0 +1,6 @@ +namespace FluentCMS.Api.Plugins.AIAgentManagement.Repositories; + +internal class AgentSchemaValidator(AIDbContext dbContext, ILogger logger) : BaseSchemaValidator(dbContext, logger) +{ + public override int Priority => 10000; +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/IAIAgentDatabaseMarker.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/IAIAgentDatabaseMarker.cs new file mode 100644 index 000000000..7ed2c2d95 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/IAIAgentDatabaseMarker.cs @@ -0,0 +1,6 @@ +namespace FluentCMS.Api.Plugins.AIAgentManagement.Repositories; + +public interface IAIAgentDatabaseMarker : IDatabaseArea +{ + // Marker interface for database +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/ThreadRepository.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/ThreadRepository.cs new file mode 100644 index 000000000..a3dd8acdd --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Repositories/ThreadRepository.cs @@ -0,0 +1,10 @@ +using FluentCMS.Api.Plugins.AIAgentManagement.Models; + +namespace FluentCMS.Api.Plugins.AIAgentManagement.Repositories; + + +public interface IThreadRepository : IRepository +{ +} + +internal class ThreadRepository(AIDbContext dbContext) : Repository(dbContext), IThreadRepository; diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Tools.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Tools.cs new file mode 100644 index 000000000..c98893e16 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/Tools.cs @@ -0,0 +1,53 @@ +namespace FluentCMS.Api.Plugins.AIAgentManagement; + +public class Tools(ILogger logger) +{ + [Description("Request to read the contents of a file at the specified path. Use this when you need to examine the contents of an existing file you do not know the contents of, for example to analyze code, review text files, or extract information from configuration files. Automatically extracts raw text from PDF and DOCX files. May not be suitable for other types of binary files, as it returns the raw content as a string. Do NOT use this tool to list the contents of a directory. Only use this tool on files.")] + public string ReadFile([Description("The path of the file to read")] string path) + { + logger.LogInformation("ReadFile called with path: {Path}", path); + var fullPath = path; + if (!File.Exists(fullPath)) + { + logger.LogWarning("File not found: {FullPath}", fullPath); + return $"Error: File at path '{fullPath}' does not exist."; + } + logger.LogInformation("Reading file: {FullPath}", fullPath); + return File.ReadAllText(fullPath); + } + + [Description("Request to write content to a file at the specified path. If the file exists, it will be overwritten with the provided content. If the file doesn't exist, it will be created. This tool will automatically create any directories needed to write the file.")] + public void WriteFile( + [Description("The path of the file to write to")] string path, + [Description("The content to write to the file. ALWAYS provide the COMPLETE intended content of the file, without any truncation or omissions. You MUST include ALL parts of the file, even if they haven't been modified.")] string content) + { + logger.LogInformation("WriteFile called with path: {Path}", path); + var fullPath = path; + var directory = Path.GetDirectoryName(fullPath); + if (!Directory.Exists(directory)) + { + logger.LogInformation("Creating directory: {Directory}", directory); + Directory.CreateDirectory(directory); + } + logger.LogInformation("Writing to file: {FullPath}", fullPath); + File.WriteAllText(fullPath, content); + } + + [Description("Request to list files and directories within the specified directory. If recursive is true, it will list all files and directories recursively. If recursive is false or not provided, it will only list the top-level contents. Do not use this tool to confirm the existence of files you may have created, as the user will let you know if the files were created successfully or not.")] + public List ListFiles( + [Description("The path of the directory to list contents for")] string directoryPath, + [Description("Whether to list files recursively. Use true for recursive listing, false or omit for top-level only.")] bool recursive = false) + { + logger.LogInformation("ListFiles called with directoryPath: {DirectoryPath}, recursive: {Recursive}", directoryPath, recursive); + var fullPath = directoryPath; + if (!Directory.Exists(fullPath)) + { + logger.LogWarning("Directory not found: {FullPath}", fullPath); + return [$"Error: Directory at path '{fullPath}' does not exist."]; + } + logger.LogInformation("Listing files in directory: {FullPath}", fullPath); + var searchOption = recursive ? SearchOption.AllDirectories : SearchOption.TopDirectoryOnly; + var entries = Directory.GetFileSystemEntries(fullPath, "*", searchOption).ToList(); + return entries; + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/_GlobalUsings.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/_GlobalUsings.cs new file mode 100644 index 000000000..4103eba21 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.AIAgentManagement/_GlobalUsings.cs @@ -0,0 +1,22 @@ +global using FluentCMS.Api.Core; +global using FluentCMS.Api.Core.Controllers; +global using FluentCMS.Infrastructure; +global using FluentCMS.Infrastructure.Plugins.Abstractions; +global using FluentCMS.Infrastructure.Repositories.Abstractions; +global using FluentCMS.Infrastructure.Repositories.EntityFramework; +global using FluentCMS.Infrastructure.Repositories.EntityFramework.Configuration; +global using Microsoft.AspNetCore.Builder; +global using Microsoft.AspNetCore.Mvc; +global using Microsoft.EntityFrameworkCore; +global using Microsoft.Extensions.Configuration; +global using Microsoft.Extensions.DependencyInjection; +global using Microsoft.Extensions.Logging; +global using System.ComponentModel.DataAnnotations; +global using Microsoft.Extensions.DependencyInjection.Extensions; +global using Microsoft.Extensions.Options; +global using OpenAI; +global using OpenAI.Chat; +global using System.ClientModel; +global using FluentCMS.Api.Plugins.AIAgentManagement.Repositories; +global using System.ComponentModel; +global using FluentCMS.Api.Plugins.AIAgentManagement.Models; diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/CmsCoreManagementPlugin.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/CmsCoreManagementPlugin.cs index 4a05e2333..855d12679 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/CmsCoreManagementPlugin.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/CmsCoreManagementPlugin.cs @@ -13,11 +13,22 @@ public void ConfigureServices(IServiceCollection services, IConfiguration? confi // Register database context services.AddDatabaseContext(); + services.AddDataSeeder(); services.AddSchemaValidator(); services.AddScoped(); + services.AddScoped(); + services.AddScoped(); + services.AddScoped(); + services.AddScoped(); + services.AddScoped(); services.AddScoped(); + services.AddScoped(); + services.AddScoped(); + services.AddScoped(); + services.AddScoped(); + services.AddScoped(); } public void Configure(IApplicationBuilder app) diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/BlocksController.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/BlocksController.cs new file mode 100644 index 000000000..edeec2ff3 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/BlocksController.cs @@ -0,0 +1,54 @@ +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Controllers; + +public class BlocksController(IBlockService blockService) : BaseController +{ + + [HttpGet] + public async Task> GetBySiteId([FromQuery] Guid siteId, CancellationToken cancellationToken = default) + { + var blocks = await blockService.GetBySiteId(siteId, cancellationToken); + + var blocksDto = Mapper.Map>(blocks); + + return SuccessList(blocksDto); + } + + [HttpGet] + public async Task> GetAll(CancellationToken cancellationToken = default) + { + var blocks = await blockService.GetAll(cancellationToken); + var blocksDto = Mapper.Map>(blocks); + return SuccessList(blocksDto); + } + + [HttpGet("{id:guid}")] + public async Task> GetById(Guid id, CancellationToken cancellationToken = default) + { + var block = await blockService.GetById(id, cancellationToken); + return Success(Mapper.Map(block)); + } + + [HttpPost] + public async Task> Add(BlockAddRequest request, CancellationToken cancellationToken = default) + { + var block = Mapper.Map(request); + await blockService.Add(block, cancellationToken); + return Success(Mapper.Map(block)); + } + + [HttpPut("{id:guid}")] + public async Task> Update(Guid id, BlockUpdateRequest request, CancellationToken cancellationToken = default) + { + var block = await blockService.GetById(id, cancellationToken); + Mapper.Map(request, block); + await blockService.Update(block, cancellationToken); + return Success(Mapper.Map(block)); + } + + [HttpDelete("{id:guid}")] + public async Task Remove(Guid id, CancellationToken cancellationToken = default) + { + await blockService.Remove(id, cancellationToken); + return Success(); + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/FilesController.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/FilesController.cs new file mode 100644 index 000000000..441a0c968 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/FilesController.cs @@ -0,0 +1,108 @@ +using FluentCMS.Api.Plugins.CmsCoreManagement.Dtos; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Controllers; + +public class FilesController(IFileService fileService, IFolderService folderService, ISiteService siteService, IMapper mapper) : BaseController +{ + // [HttpGet("{id}")] + // public async Task> GetById([FromRoute] Guid id, CancellationToken cancellationToken = default) + // { + // var file = await fileService.GetById(id, cancellationToken); + // var fileResponse = mapper.Map(file); + // fileResponse.Path = await fileService.GetFilePath(file); + // return Ok(fileResponse); + // } + + [HttpPost] + public async Task> Upload([FromQuery] Guid folderId, [FromForm] IEnumerable files, CancellationToken cancellationToken = default) + { + var filesResponse = new List(); + + var folder = await folderService.GetById(folderId, cancellationToken); + + foreach (var formFile in files.Where(x => x.Length > 0)) + { + var file = new Core.Models.File + { + FolderId = folderId, + SiteId = folder.SiteId, + Name = formFile.FileName, + Size = formFile.Length, + ContentType = formFile.ContentType, + Extension = System.IO.Path.GetExtension(formFile.FileName), + }; + + await fileService.Create(file, formFile.OpenReadStream(), cancellationToken); + + var fileResponse = mapper.Map(file); + // fileResponse.Path = await fileService.GetFilePath(file); + filesResponse.Add(fileResponse); + } + + return SuccessList(filesResponse); + } + + // [HttpGet("{id}")] + // public async Task Download([FromRoute] Guid id, CancellationToken cancellationToken = default) + // { + // var file = await fileService.GetById(id, cancellationToken); + // var fileStream = await fileService.GetStream(id, cancellationToken); + // return Results.File(fileStream, contentType: file.ContentType, fileDownloadName: file.Name, lastModified: file.ModifiedAt ?? file.CreatedAt); + // } + + // [HttpGet] + // [DecodeQueryParam] + // public async Task Download([FromQuery] string url, CancellationToken cancellationToken = default) + // { + // // example url = https://example.com/folder1/folder2/file.ext + // // extract folder path and file name from the url + // var uri = new Uri(url); + + // // Get site by domain name + // var domain = uri.Authority; + // var site = await siteService.GetByUrl(domain, cancellationToken); + + // // Get the full path from the URL (without the scheme and domain) + // string fullPath = uri.AbsolutePath; + + // // Extract the file name + // var fileName = System.IO.Path.GetFileName(fullPath); + + // // Extract the folder path (excluding the file name) + // var folderPath = fullPath[..fullPath.LastIndexOf('/')]; + + // // find folders by url + // var folder = await folderService.GetByPath(site.Id, folderPath, cancellationToken); + + // // find file by name and folder id + // var file = await fileService.GetByName(folder.Id, fileName, cancellationToken); + + // // get file stream + // var fileStream = await fileService.GetStream(file.Id, cancellationToken); + + // return Results.File(fileStream, contentType: file.ContentType, fileDownloadName: file.Name, lastModified: file.ModifiedAt ?? file.CreatedAt); + // } + + [HttpDelete("{id}")] + public async Task> Remove([FromRoute] Guid id, CancellationToken cancellationToken = default) + { + await fileService.Remove(id, cancellationToken); + return Success(true); + } + + [HttpPut] + public async Task> Rename([FromBody] FileRenameRequest request, CancellationToken cancellationToken = default) + { + var file = await fileService.Rename(request.Id, request.Name, cancellationToken); + var fileResponse = mapper.Map(file); + return Success(fileResponse); + } + + [HttpPut] + public async Task> Move([FromBody] FileMoveRequest request, CancellationToken cancellationToken = default) + { + var file = await fileService.Move(request.Id, request.FolderId, cancellationToken); + var fileResponse = mapper.Map(file); + return Success(fileResponse); + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/FoldersController.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/FoldersController.cs new file mode 100644 index 000000000..0b0fbadd4 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/FoldersController.cs @@ -0,0 +1,93 @@ +using FluentCMS.Api.Plugins.CmsCoreManagement.Dtos; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Controllers; + +public class FoldersController(IFolderService folderService, IFileService fileService, IMapper mapper) : BaseController +{ + [HttpGet()] + public async Task> GetRoot([FromQuery] Guid siteId, CancellationToken cancellationToken = default) + { + var folder = await folderService.GetRoot(siteId, cancellationToken); + + var files = await fileService.GetByFolderId(folder.Id, cancellationToken); + var folders = await folderService.GetByFolderId(folder.Id, cancellationToken); + + var folderDto = mapper.Map(folder); + folderDto.Files = mapper.Map>(files); + folderDto.Folders = mapper.Map>(folders); + + if (folder.ParentId != null) + { + var parentFolder = await folderService.GetById(folder.ParentId.Value, cancellationToken); + folderDto.ParentFolder = mapper.Map(parentFolder); + } + + return Success(folderDto); + } + + [HttpGet("{id}")] + public async Task> GetById([FromRoute] Guid id, CancellationToken cancellationToken = default) + { + var folder = await folderService.GetById(id, cancellationToken); + var files = await fileService.GetByFolderId(folder.Id, cancellationToken); + var folders = await folderService.GetByFolderId(folder.Id, cancellationToken); + + var folderDto = mapper.Map(folder); + folderDto.Files = mapper.Map>(files); + folderDto.Folders = mapper.Map>(folders); + + if (folder.ParentId != null) + { + var parentFolder = await folderService.GetById(folder.ParentId.Value, cancellationToken); + folderDto.ParentFolder = mapper.Map(parentFolder); + } + + return Success(folderDto); + } + + [HttpPost] + public async Task> Create([FromBody] FolderAddRequest request, CancellationToken cancellationToken = default) + { + var folder = mapper.Map(request); + await folderService.Create(folder, cancellationToken); + + var folderDto = mapper.Map(folder); + return Success(folderDto); + } + + [HttpPut] + public async Task> Rename([FromBody] FolderRenameRequest request, CancellationToken cancellationToken = default) + { + var folder = await folderService.Rename(request.Id, request.Name, cancellationToken); + + var folderDto = mapper.Map(folder); + + return Success(folderDto); + } + + [HttpPut] + public async Task> Move([FromBody] FolderMoveRequest request, CancellationToken cancellationToken = default) + { + var folder = await folderService.Move(request.Id, request.ParentId, cancellationToken); + + var folderDto = mapper.Map(folder); + + return Success(folderDto); + } + + [HttpGet] + public async Task> GetParentFolders([FromQuery] Guid folderId, CancellationToken cancellationToken = default) + { + var folders = await folderService.GetParentFolders(folderId, cancellationToken); + + var response = mapper.Map>(folders); + return SuccessList(response); + } + + [HttpDelete("{id:guid}")] + public async Task> Remove([FromRoute] Guid id, CancellationToken cancellationToken = default) + { + await folderService.Remove(id, cancellationToken); + return Success(true); + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/LayoutsController.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/LayoutsController.cs new file mode 100644 index 000000000..6e74dad20 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/LayoutsController.cs @@ -0,0 +1,52 @@ +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Controllers; + +public class LayoutsController(ILayoutService layoutService) : BaseController +{ + + [HttpGet] + public async Task> GetBySiteId([FromQuery] Guid siteId, CancellationToken cancellationToken = default) + { + var layouts = await layoutService.GetBySiteId(siteId, cancellationToken); + var layoutsDto = Mapper.Map>(layouts); + return SuccessList(layoutsDto); + } + + [HttpGet] + public async Task> GetAll(CancellationToken cancellationToken = default) + { + var layouts = await layoutService.GetAll(cancellationToken); + var layoutsDto = Mapper.Map>(layouts); + return SuccessList(layoutsDto); + } + + [HttpGet("{id:guid}")] + public async Task> GetById(Guid id, CancellationToken cancellationToken = default) + { + var layout = await layoutService.GetById(id, cancellationToken); + return Success(Mapper.Map(layout)); + } + + [HttpPost] + public async Task> Add(LayoutAddRequest request, CancellationToken cancellationToken = default) + { + var layout = Mapper.Map(request); + await layoutService.Add(layout, cancellationToken); + return Success(Mapper.Map(layout)); + } + + [HttpPut("{id:guid}")] + public async Task> Update(Guid id, LayoutUpdateRequest request, CancellationToken cancellationToken = default) + { + var layout = await layoutService.GetById(id, cancellationToken); + Mapper.Map(request, layout); + await layoutService.Update(layout, cancellationToken); + return Success(Mapper.Map(layout)); + } + + [HttpDelete("{id:guid}")] + public async Task Remove(Guid id, CancellationToken cancellationToken = default) + { + await layoutService.Remove(id, cancellationToken); + return Success(); + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/PagesController.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/PagesController.cs new file mode 100644 index 000000000..db352f1fd --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/PagesController.cs @@ -0,0 +1,57 @@ +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Controllers; + +public class PagesController(IPageService pageService) : BaseController +{ + + [HttpGet] + public async Task> GetBySiteId([FromQuery] Guid siteId, CancellationToken cancellationToken = default) + { + var pages = await pageService.GetBySiteId(siteId, cancellationToken); + + var pagesDto = Mapper.Map>(pages); + + foreach (var page in pagesDto) + page.FullPath = await pageService.GetPageUrl(page.Id, cancellationToken); + + return SuccessList(pagesDto); + } + + [HttpGet] + public async Task> GetAll(CancellationToken cancellationToken = default) + { + var pages = await pageService.GetAll(cancellationToken); + var pagesDto = Mapper.Map>(pages); + return SuccessList(pagesDto); + } + + [HttpGet("{id:guid}")] + public async Task> GetById(Guid id, CancellationToken cancellationToken = default) + { + var page = await pageService.GetById(id, cancellationToken); + return Success(Mapper.Map(page)); + } + + [HttpPost] + public async Task> Add(PageAddRequest request, CancellationToken cancellationToken = default) + { + var page = Mapper.Map(request); + await pageService.Add(page, cancellationToken); + return Success(Mapper.Map(page)); + } + + [HttpPut("{id:guid}")] + public async Task> Update(Guid id, PageUpdateRequest request, CancellationToken cancellationToken = default) + { + var page = await pageService.GetById(id, cancellationToken); + Mapper.Map(request, page); + await pageService.Update(page, cancellationToken); + return Success(Mapper.Map(page)); + } + + [HttpDelete("{id:guid}")] + public async Task Remove(Guid id, CancellationToken cancellationToken = default) + { + await pageService.Remove(id, cancellationToken); + return Success(); + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/SiteController.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/SitesController.cs similarity index 81% rename from src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/SiteController.cs rename to src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/SitesController.cs index 72ce499c9..213f6a4ab 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/SiteController.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Controllers/SitesController.cs @@ -1,6 +1,6 @@ namespace FluentCMS.Api.Plugins.CmsCoreManagement.Controllers; -public class SiteController(ISiteService siteService) : BaseController +public class SitesController(ISiteService siteService) : BaseController { [HttpGet] public async Task> GetAll(CancellationToken cancellationToken = default) @@ -17,6 +17,14 @@ public async Task> GetById(Guid id, CancellationToken cance return Success(Mapper.Map(site)); } + + [HttpGet] + public async Task> GetByUrl([FromQuery] string url, CancellationToken cancellationToken = default) + { + var site = await siteService.GetByUrl(url, cancellationToken); + return Success(Mapper.Map(site)); + } + [HttpPost] public async Task> Add(SiteAddRequest request, CancellationToken cancellationToken = default) { diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Block.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Block.cs new file mode 100644 index 000000000..caf21bf7d --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Block.cs @@ -0,0 +1,30 @@ +using System.ComponentModel.DataAnnotations; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Dtos; + +public class BlockDto +{ + public Guid Id { get; set; } + public Guid SiteId { get; set; } + public string Name { get; set; } + public string Category { get; set; } + public string? Description { get; set; } + public string Content { get; set; } +} + +public class BlockAddRequest +{ + public required string Name { get; set; } + public required Guid SiteId { get; set; } + public string Category { get; set; } = string.Empty; + public string? Description { get; set; } = string.Empty; + public string Content { get; set; } = string.Empty; +} + +public class BlockUpdateRequest +{ + public string Name { get; set; } = string.Empty; + public string Category { get; set; } = string.Empty; + public string? Description { get; set; } = string.Empty; + public string Content { get; set; } = string.Empty; +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/File.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/File.cs new file mode 100644 index 000000000..b0ed9bf0f --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/File.cs @@ -0,0 +1,27 @@ +using System.ComponentModel.DataAnnotations; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Dtos; + +public class FileDto +{ + public Guid Id { get; set; } + public Guid SiteId { get; set; } + public string Name { get; set; } + public string NormalizedName { get; set; } = default!; + public Guid FolderId { get; set; } + public string Extension { get; set; } = default!; + public string ContentType { get; set; } = default!; + public long Size { get; set; } +} + +public class FileMoveRequest +{ + public Guid Id { get; set; } + public Guid FolderId { get; set; } +} + +public class FileRenameRequest +{ + public Guid Id { get; set; } + public string Name { get; set; } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Folder.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Folder.cs new file mode 100644 index 000000000..becf85709 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Folder.cs @@ -0,0 +1,33 @@ +using System.ComponentModel.DataAnnotations; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Dtos; + +public class FolderDto +{ + public Guid Id { get; set; } + public Guid SiteId { get; set; } + public string Name { get; set; } + public string NormalizedName { get; set; } + public List Files { get; set; } = []; + public List Folders { get; set; } = []; + public FolderDto? ParentFolder { get; set; } +} + +public class FolderAddRequest +{ + public string Name { get; set; } + public Guid SiteId { get; set; } + public Guid ParentId { get; set; } +} + +public class FolderRenameRequest +{ + public Guid Id { get; set; } + public string Name { get; set; } +} + +public class FolderMoveRequest +{ + public Guid Id { get; set; } + public Guid ParentId { get; set; } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Layout.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Layout.cs new file mode 100644 index 000000000..8a4770b90 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Layout.cs @@ -0,0 +1,27 @@ +using System.ComponentModel.DataAnnotations; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Dtos; + +public class LayoutDto +{ + public Guid Id { get; set; } + public Guid SiteId { get; set; } + public string Name { get; set; } + public string Head { get; set; } + public string Body { get; set; } +} + +public class LayoutAddRequest +{ + public required string Name { get; set; } + public required Guid SiteId { get; set; } + public string Head { get; set; } = string.Empty; + public string Body { get; set; } = string.Empty; +} + +public class LayoutUpdateRequest +{ + public string Name { get; set; } = string.Empty; + public string Head { get; set; } = string.Empty; + public string Body { get; set; } = string.Empty; +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/MappingProfile.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/MappingProfile.cs index 183298a9d..d8871783a 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/MappingProfile.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/MappingProfile.cs @@ -7,5 +7,27 @@ public MappingProfile() CreateMap(); CreateMap(); CreateMap(); + + + CreateMap(); + CreateMap(); + CreateMap(); + + CreateMap(); + CreateMap(); + CreateMap(); + + CreateMap(); + CreateMap(); + CreateMap(); + + CreateMap(); + CreateMap(); + CreateMap(); + + CreateMap(); + CreateMap(); + CreateMap(); + CreateMap(); } } diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Page.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Page.cs new file mode 100644 index 000000000..56d33abc7 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Page.cs @@ -0,0 +1,60 @@ +using System.ComponentModel.DataAnnotations; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Dtos; + +public class PageDto +{ + public Guid Id { get; set; } + public Guid SiteId { get; set; } + public string Title { get; set; } = default!; + public string Slug { get; set; } + public string? FullPath { get; set; } + public int Order { get; set; } + public Guid ParentId { get; set; } + public Guid LayoutId { get; set; } + public Guid EditLayoutId { get; set; } + public Guid DetailLayoutId { get; set; } + public string? MetaTitle { get; set; } + public string? MetaDescription { get; set; } + public bool RobotsIndex { get; set; } + public bool RobotsFollow { get; set; } + public string? OgType { get; set; } + public string? Head { get; set; } +} + +public class PageAddRequest +{ + [Required] + public string Title { get; set; } = default!; + public Guid SiteId { get; set; } + public string? Slug { get; set; } + public int Order { get; set; } + public Guid? ParentId { get; set; } + public Guid? LayoutId { get; set; } + public Guid? EditLayoutId { get; set; } + public Guid? DetailLayoutId { get; set; } + public string? MetaTitle { get; set; } + public string? MetaDescription { get; set; } + public bool RobotsIndex { get; set; } + public bool RobotsFollow { get; set; } + public string? OgType { get; set; } + public string? Head { get; set; } +} + +public class PageUpdateRequest +{ + [Required] + public string Title { get; set; } = default!; + public string? Slug { get; set; } + public int Order { get; set; } + public Guid? ParentId { get; set; } + public Guid? LayoutId { get; set; } + public Guid? EditLayoutId { get; set; } + public Guid? DetailLayoutId { get; set; } + public string? MetaTitle { get; set; } + public string? MetaDescription { get; set; } + public bool RobotsIndex { get; set; } + public bool RobotsFollow { get; set; } + public string? OgType { get; set; } + public string? Head { get; set; } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Site.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Site.cs index 692c52a5c..2657ca3ac 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Site.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Dtos/Site.cs @@ -8,6 +8,18 @@ public class SiteDto public string Name { get; set; } = default!; public string? Description { get; set; } public List Urls { get; set; } = []; + + public Guid? LayoutId { get; set; } + public Guid? DetailLayoutId { get; set; } + public Guid? EditLayoutId { get; set; } + public string? MetaTitle { get; set; } + public string? MetaDescription { get; set; } + public bool? RobotsIndex { get; set; } + public bool? RobotsFollow { get; set; } + public string? RobotsTxt { get; set; } + public string? GoogleTagsId { get; set; } + public string? OgType { get; set; } + public string? Head { get; set; } } public class SiteAddRequest @@ -19,6 +31,19 @@ public class SiteAddRequest [Required] public List Urls { get; set; } = []; + + + public Guid? LayoutId { get; set; } + public Guid? DetailLayoutId { get; set; } + public Guid? EditLayoutId { get; set; } + public string? MetaTitle { get; set; } + public string? MetaDescription { get; set; } + public bool? RobotsIndex { get; set; } + public bool? RobotsFollow { get; set; } + public string? RobotsTxt { get; set; } + public string? GoogleTagsId { get; set; } + public string? OgType { get; set; } + public string? Head { get; set; } } public class SiteUpdateRequest @@ -30,4 +55,16 @@ public class SiteUpdateRequest [Required] public List Urls { get; set; } = []; + + public Guid? LayoutId { get; set; } + public Guid? DetailLayoutId { get; set; } + public Guid? EditLayoutId { get; set; } + public string? MetaTitle { get; set; } + public string? MetaDescription { get; set; } + public bool? RobotsIndex { get; set; } + public bool? RobotsFollow { get; set; } + public string? RobotsTxt { get; set; } + public string? GoogleTagsId { get; set; } + public string? OgType { get; set; } + public string? Head { get; set; } } diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/MessageCodes.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/MessageCodes.cs index c0e1cbb76..b8aba97b1 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/MessageCodes.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/MessageCodes.cs @@ -12,4 +12,43 @@ public static class MessageCodes public const string SiteFailedToRemove = "Site.FailedToRemove"; #endregion + + #region Page + + public const string PageUrlCannotBeEmpty = "Page.UrlCannotBeEmpty"; + public const string PageUrlIsInvalid = "Page.UrlIsInvalid"; + public const string PageUrlMustBeUnique = "Page.UrlMustBeUnique"; + public const string PageFailedToAdd = "Page.FailedToAdd"; + public const string PageFailedToUpdate = "Page.FailedToUpdate"; + public const string PageFailedToRemove = "Page.FailedToRemove"; + + #endregion + + #region Folder + public const string FolderInvalidName = "Folder.InvalidName"; + public const string FolderAlreadyExists = "Folder.AlreadyExists"; + public const string FolderUnableToCreate = "Folder.UnableToCreate"; + public const string FolderParentNotFound = "Folder.ParentNotFound"; + public const string FolderNotFound = "Folder.NotFound"; + + public const string FolderCannotRenameRootFolder = "Folder.CannotRenameRootFolder"; + public const string FolderUnableToUpdate = "Folder.UnableToUpdate"; + + public const string FolderCannotMoveToItself = "Folder.CannotMoveToItself"; + public const string FolderCannotMoveRootFolder = "Folder.CannotMoveRootFolder"; + public const string FolderCannotMoveToChild = "Folder.CannotMoveToChild"; + + + #endregion + + #region File + + public const string FileUnableToDelete = "File.UnableToDelete"; + public const string FileNotFound = "File.NotFound"; + public const string FileInvalidName = "File.InvalidName"; + public const string FileAlreadyExists = "File.AlreadyExists"; + public const string FileUnableToUpdate = "File.UnableToUpdate"; + + #endregion + } diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/BlockRepository.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/BlockRepository.cs new file mode 100644 index 000000000..de1df7f46 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/BlockRepository.cs @@ -0,0 +1,11 @@ +using FluentCMS.Api.Core.Repositories.EntityFramework; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Repositories; + +public interface IBlockRepository : ISiteAssociatedRepository +{ +} + +internal class BlockRepository(CmsCoreDbContext dbContext) : SiteAssociatedRepository(dbContext), IBlockRepository +{ +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/CmsCoreDataSeeder.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/CmsCoreDataSeeder.cs new file mode 100644 index 000000000..38cfd8695 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/CmsCoreDataSeeder.cs @@ -0,0 +1,190 @@ +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Repositories; + +internal class CmsCoreDataSeeder(CmsCoreDbContext dbContext, ILogger logger) + : BaseDataSeeder(dbContext, logger) +{ + public override int Priority => 10000; + + public override async Task SeedData(CancellationToken cancellationToken = default) + { + logger.LogInformation("Seeding initial site entities into the database..."); + + var sites = new[] + { + new + { + Name = "Default Site", + Url = "localhost:5022", + Title = "Default CMS Site", + Emoji = "🌐", + Pages = new[] + { + ("Home", ""), + ("About Us", "about"), + ("Contact", "contact"), + ("Blog", "blog") + }, + Folders = new[] + { + "Images", + "Documents", + "Videos", + "Temp", + "Archive" + } + }, + new + { + Name = "Marketing Hub", + Url = "127.0.0.1:5022", + Title = "Marketing Hub", + Emoji = "📣", + Pages = new[] + { + ("Home", ""), + ("Campaigns", "campaigns"), + ("Landing Pages", "landing-pages"), + ("Case Studies", "case-studies"), + ("Team", "team") + }, + Folders = new[] + { + "Images", + "Brand", + "Ads", + "EmailTemplates", + "Analytics" + } + }, + new + { + Name = "Docs Portal", + Url = "0.0.0.0:5022", + Title = "Documentation Portal", + Emoji = "📄", + Pages = new[] + { + ("Home", ""), + ("Getting Started", "getting-started"), + ("API Reference", "api"), + ("Guides", "guides"), + ("Changelog", "changelog"), + ("FAQ", "faq") + }, + Folders = new[] + { + "Guides", + "ApiSamples", + "Releases", + "Images", + "Internal" + } + } + }; + + foreach (var s in sites) + { + var site = new Site + { + Name = s.Name, + Urls = [s.Url], + Description = $"{s.Name} seeded site {s.Emoji}", + MetaTitle = s.Title, + MetaDescription = $"{s.Name} instance", + RobotsIndex = true, + RobotsFollow = true, + RobotsTxt = "User-agent: *\nAllow: /", + OgType = "website", + Head = $"" + }; + + await DbContext.Sites.AddAsync(site, cancellationToken); + await DbContext.SaveChangesAsync(cancellationToken); + + var layout = new Layout + { + SiteId = site.Id, + Name = $"{s.Name} Layout", + Head = $"{s.Title} {s.Emoji}", + Body = $"

{s.Name} Layout Body {s.Emoji}

" + }; + + await DbContext.Layouts.AddAsync(layout, cancellationToken); + await DbContext.SaveChangesAsync(cancellationToken); + + site.LayoutId = layout.Id; + site.EditLayoutId = layout.Id; + site.DetailLayoutId = layout.Id; + + DbContext.Sites.Update(site); + await DbContext.SaveChangesAsync(cancellationToken); + + + var order = 0; + foreach (var p in s.Pages) + { + var page = new Page + { + SiteId = site.Id, + Title = p.Item1 + " " + s.Emoji, + Slug = p.Item2, + Order = order++, + LayoutId = layout.Id, + EditLayoutId = layout.Id, + DetailLayoutId = layout.Id, + MetaTitle = p.Item1, + RobotsIndex = true, + RobotsFollow = true, + OgType = "website" + }; + + await DbContext.Pages.AddAsync(page, cancellationToken); + } + + var rootFolder = new Folder + { + SiteId = site.Id, + Name = "Files", + NormalizedName = "files", + ParentId = null, + Size = 0 + }; + + await DbContext.Folders.AddAsync(rootFolder, cancellationToken); + + var childFolders = new List(); + + foreach (var f in s.Folders) + { + childFolders.Add(new Folder + { + SiteId = site.Id, + Name = f, + NormalizedName = f.ToLower(), + ParentId = rootFolder.Id, + Size = 0 + }); + } + + await DbContext.Folders.AddRangeAsync(childFolders, cancellationToken); + + var nested = new List(); + + foreach (var cf in childFolders.Take(2)) + { + nested.Add(new Folder + { + SiteId = site.Id, + Name = $"{cf.Name} Sub", + NormalizedName = $"{cf.NormalizedName}-sub", + ParentId = cf.Id, + Size = 0 + }); + } + + await DbContext.Folders.AddRangeAsync(nested, cancellationToken); + + await DbContext.SaveChangesAsync(cancellationToken); + } + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/CmsCoreDbContext.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/CmsCoreDbContext.cs index a66592e20..1eadd2f9b 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/CmsCoreDbContext.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/CmsCoreDbContext.cs @@ -4,14 +4,198 @@ namespace FluentCMS.Api.Plugins.CmsCoreManagement.Repositories; internal class CmsCoreDbContext(DbContextOptions options) : DbContext(options), ICmsCoreDatabaseMarker { + public DbSet Folders => Set(); + public DbSet Files => Set(); public DbSet Sites => Set(); + public DbSet Pages => Set(); + public DbSet Layouts => Set(); + public DbSet Blocks => Set(); protected override void OnModelCreating(ModelBuilder modelBuilder) { base.OnModelCreating(modelBuilder); - // Configure the Site entity - modelBuilder.Entity() - .HasKey(s => s.Id); + modelBuilder.Entity(e => + { + e.HasKey(x => x.Id); + e.Property(x => x.Name).IsRequired().HasMaxLength(200); + + e.Property(x => x.Description); + + e.Property(x => x.Urls) + .HasConversion( + v => string.Join(";", v), + v => v.Split(";", StringSplitOptions.RemoveEmptyEntries).ToList()); + + e.Property(x => x.MetaTitle); + e.Property(x => x.MetaDescription); + e.Property(x => x.RobotsIndex); + e.Property(x => x.RobotsFollow); + e.Property(x => x.RobotsTxt); + e.Property(x => x.GoogleTagsId); + e.Property(x => x.OgType); + e.Property(x => x.Head); + + // Layout relationships (nullable, no cascade) + e.HasOne() + .WithMany() + .HasForeignKey(x => x.LayoutId) + .IsRequired(false) + .OnDelete(DeleteBehavior.Restrict); + + e.HasOne() + .WithMany() + .HasForeignKey(x => x.EditLayoutId) + .IsRequired(false) + .OnDelete(DeleteBehavior.Restrict); + + e.HasOne() + .WithMany() + .HasForeignKey(x => x.DetailLayoutId) + .IsRequired(false) + .OnDelete(DeleteBehavior.Restrict); + }); + + modelBuilder.Entity(e => + { + e.HasKey(x => x.Id); + e.Property(x => x.Name).IsRequired().HasMaxLength(200); + e.Property(x => x.Head); + e.Property(x => x.Body); + + e.HasOne() + .WithMany() + .HasForeignKey("SiteId") + .IsRequired() + .OnDelete(DeleteBehavior.Cascade); + }); + + + modelBuilder.Entity(e => + { + e.HasKey(x => x.Id); + e.Property(x => x.Name).IsRequired().HasMaxLength(200); + e.Property(x => x.Category).IsRequired().HasMaxLength(200); + e.Property(x => x.Description); + e.Property(x => x.Content); + + e.HasOne() + .WithMany() + .HasForeignKey("SiteId") + .IsRequired() + .OnDelete(DeleteBehavior.Cascade); + }); + + modelBuilder.Entity(e => + { + e.HasKey(x => x.Id); + e.Property(x => x.Title).IsRequired().HasMaxLength(200); + e.Property(x => x.Slug); // can be empty for / (home page) + e.Property(x => x.Order); + e.Property(x => x.MetaTitle); + e.Property(x => x.MetaDescription); + e.Property(x => x.RobotsIndex); + e.Property(x => x.RobotsFollow); + e.Property(x => x.OgType); + e.Property(x => x.Head); + e.HasOne() + .WithMany() + .HasForeignKey(x => x.LayoutId) + .IsRequired(false) + .OnDelete(DeleteBehavior.Restrict); + + e.HasOne() + .WithMany() + .HasForeignKey(x => x.EditLayoutId) + .IsRequired(false) + .OnDelete(DeleteBehavior.Restrict); + + e.HasOne() + .WithMany() + .HasForeignKey(x => x.DetailLayoutId) + .IsRequired(false) + .OnDelete(DeleteBehavior.Restrict); + + // Make ParentId nullable + e.HasOne() + .WithMany() + .HasForeignKey(x => x.ParentId) + .IsRequired(false) + .OnDelete(DeleteBehavior.Restrict); + + e.HasOne() + .WithMany() + .HasForeignKey("SiteId") + .IsRequired() + .OnDelete(DeleteBehavior.Cascade); + + }); + + modelBuilder.Entity(e => + { + e.HasKey(x => x.Id); + + e.Property(x => x.Name) + .IsRequired() + .HasMaxLength(255); + + e.Property(x => x.NormalizedName) + .IsRequired() + .HasMaxLength(500); + + e.Property(x => x.Size) + .IsRequired(); + + e.Property(x => x.Extension) + .IsRequired() + .HasMaxLength(50); + + e.Property(x => x.ContentType) + .IsRequired() + .HasMaxLength(200); + + e.HasOne() + .WithMany(x => x.Files) + .HasForeignKey(x => x.FolderId) + .OnDelete(DeleteBehavior.Cascade); + + e.Property("SiteId"); + e.HasOne() + .WithMany() + .HasForeignKey("SiteId") + .OnDelete(DeleteBehavior.Cascade); + + }); + + modelBuilder.Entity(e => + { + e.HasKey(x => x.Id); + + e.Property(x => x.Name) + .IsRequired() + .HasMaxLength(255); + + e.Property(x => x.NormalizedName) + .IsRequired() + .HasMaxLength(500); + + e.Property(x => x.Size) + .IsRequired(); + + + e.HasOne(f => f.ParentFolder) + .WithMany(f => f.Folders) + .HasForeignKey(f => f.ParentId) + .IsRequired(false) + .OnDelete(DeleteBehavior.Restrict); + + e.Property("SiteId"); + e.HasOne() + .WithMany() + .HasForeignKey("SiteId") + .OnDelete(DeleteBehavior.Cascade); + + }); + } } diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/FileRepository.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/FileRepository.cs new file mode 100644 index 000000000..779b0e008 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/FileRepository.cs @@ -0,0 +1,24 @@ +using FluentCMS.Api.Core.Repositories.EntityFramework; +using System.Threading.Tasks; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Repositories; + +// Should be SiteAssociatedRepository +public interface IFileRepository : ISiteAssociatedRepository +{ + Task> GetByFolderId(Guid folderId, CancellationToken cancellationToken = default); + Task GetByName(Guid SiteId, Guid? folderId, string normalizedName, CancellationToken cancellationToken); +} + +internal class FileRepository(CmsCoreDbContext dbContext) : SiteAssociatedRepository(dbContext), IFileRepository +{ + public async Task> GetByFolderId(Guid folderId, CancellationToken cancellationToken) + { + return await Query().Where(x => x.FolderId == folderId).ToList(cancellationToken); + } + public async Task GetByName(Guid SiteId, Guid? folderId, string normalizedName, CancellationToken cancellationToken) + { + return await Query().Where(x => x.FolderId == folderId && x.NormalizedName == normalizedName).SingleOrDefault(cancellationToken); + } + +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/FolderRepository.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/FolderRepository.cs new file mode 100644 index 000000000..97c2cd27e --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/FolderRepository.cs @@ -0,0 +1,31 @@ +using FluentCMS.Api.Core.Repositories.EntityFramework; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Repositories; + +// Should be SiteAssociatedRepository +public interface IFolderRepository : ISiteAssociatedRepository +{ + Task GetByName(Guid SiteId, Guid? parentId, string normalizedName, CancellationToken cancellationToken); + Task> GetByFolderId(Guid folderId, CancellationToken cancellationToken); + +} + +internal class FolderRepository(CmsCoreDbContext dbContext) : SiteAssociatedRepository(dbContext), IFolderRepository +{ + + public async Task GetByName(Guid siteId, Guid? parentId, string normalizedName, CancellationToken cancellationToken) + { + var query = Query().Where(x =>x.SiteId == siteId && x.NormalizedName == normalizedName); + + if(parentId != null) + { + query = query.Where(x => x.ParentId == parentId!); + } + return await query.SingleOrDefault(cancellationToken); + } + + public async Task> GetByFolderId(Guid folderId, CancellationToken cancellationToken) + { + return await Query().Where(x => x.ParentId == folderId).ToList(cancellationToken); + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/LayoutRepository.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/LayoutRepository.cs new file mode 100644 index 000000000..df907596f --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/LayoutRepository.cs @@ -0,0 +1,12 @@ +using FluentCMS.Api.Core.Repositories.EntityFramework; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Repositories; + +// Should be SiteAssociatedRepository +public interface ILayoutRepository : ISiteAssociatedRepository +{ +} + +internal class LayoutRepository(CmsCoreDbContext dbContext) : SiteAssociatedRepository(dbContext), ILayoutRepository +{ +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/PageRepository.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/PageRepository.cs new file mode 100644 index 000000000..d7c20adef --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/PageRepository.cs @@ -0,0 +1,11 @@ +using FluentCMS.Api.Core.Repositories.EntityFramework; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Repositories; + +public interface IPageRepository : ISiteAssociatedRepository +{ +} + +internal class PageRepository(CmsCoreDbContext dbContext) : SiteAssociatedRepository(dbContext), IPageRepository +{ +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/SiteRepository.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/SiteRepository.cs index c9c17c991..e1bf95d2f 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/SiteRepository.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Repositories/SiteRepository.cs @@ -1,4 +1,6 @@ -namespace FluentCMS.Api.Plugins.CmsCoreManagement.Repositories; +using Microsoft.EntityFrameworkCore; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Repositories; public interface ISiteRepository : IRepository { @@ -9,6 +11,8 @@ internal class SiteRepository(CmsCoreDbContext dbContext) : Repository GetByUrl(string url, CancellationToken cancellationToken = default) { - return await Query().Where(x => x.Urls.Contains(url)).SingleOrDefault(cancellationToken); + return (await Query().ToList(cancellationToken)) + .FirstOrDefault(s => s.Urls.Any(u => u.Contains(url, StringComparison.OrdinalIgnoreCase))); + } } diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/BlockService.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/BlockService.cs new file mode 100644 index 000000000..2b44b8556 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/BlockService.cs @@ -0,0 +1,57 @@ +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Services; + +public interface IBlockService +{ + Task> GetAll(CancellationToken cancellationToken = default); + Task> GetBySiteId(Guid siteId, CancellationToken cancellationToken = default); + Task GetById(Guid id, CancellationToken cancellationToken = default); + Task Add(Block block, CancellationToken cancellationToken = default); + Task Update(Block block, CancellationToken cancellationToken = default); + Task Remove(Guid id, CancellationToken cancellationToken = default); +} + +internal class BlockService(IBlockRepository blockRepository, IEventPublisher eventPublisher) : IBlockService +{ + public async Task> GetAll(CancellationToken cancellationToken = default) + { + return await blockRepository.GetAll(cancellationToken); + } + + public async Task> GetBySiteId(Guid siteId, CancellationToken cancellationToken = default) + { + return await blockRepository.GetAllForSite(siteId, cancellationToken); + } + + public async Task GetById(Guid id, CancellationToken cancellationToken = default) + { + return await blockRepository.GetById(id, cancellationToken); + } + + public async Task Add(Block block, CancellationToken cancellationToken = default) + { + + await blockRepository.Add(block, cancellationToken); + + await eventPublisher.Publish(new BlockAddedEvent(block), cancellationToken); + + return block; + } + + public async Task Update(Block block, CancellationToken cancellationToken = default) + { + await blockRepository.Update(block, cancellationToken); + + await eventPublisher.Publish(new BlockUpdatedEvent(block), cancellationToken); + + return block; + } + + public async Task Remove(Guid id, CancellationToken cancellationToken = default) + { + var deletedBlock = await blockRepository.Remove(id, cancellationToken); + + await eventPublisher.Publish(new BlockRemovedEvent(deletedBlock), cancellationToken); + + return deletedBlock; + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/FileService.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/FileService.cs new file mode 100644 index 000000000..3c6ec071f --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/FileService.cs @@ -0,0 +1,170 @@ +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Services; + +public interface IFileService +{ + Task> GetByFolderId(Guid folderId, CancellationToken cancellationToken = default); + + Task Create(Core.Models.File file, System.IO.Stream fileContent, CancellationToken cancellationToken = default); +// Task GetById(Guid id, CancellationToken cancellationToken = default); + // Task GetByName(Guid folderId, string fileName, CancellationToken cancellationToken = default); + Task Rename(Guid id, string name, CancellationToken cancellationToken = default); + Task Remove(Guid id, CancellationToken cancellationToken = default); + Task Move(Guid id, Guid folderId, CancellationToken cancellationToken = default); +// Task GetFilePath(File file, CancellationToken cancellationToken = default); +// Task GetStream(Guid id, CancellationToken cancellationToken = default); +// Task> GetAll(Guid siteId, CancellationToken cancellationToken = default); +} + +internal class FileService(IFileRepository fileRepository, IFolderRepository folderRepository, IFolderService folderService) : IFileService +{ +// // TODO: use IFileStorageProvider fileStorageProvider. currently files will not be saved. + public async Task Create(Core.Models.File file, System.IO.Stream fileContent, CancellationToken cancellationToken = default) + { + var folder = await folderRepository.GetById(file.FolderId, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FolderNotFound); + + // if (folder.SiteId != file.SiteId) + // throw new EnhancedException(MessageCodes.FolderNotFound); + + file.NormalizedName = GetNormalizedFileName(file.Name); + + // check if file with the same name already exists + var existingFile = await fileRepository.GetByName(folder.SiteId, folder.Id, file.NormalizedName, cancellationToken); + if (existingFile != null) + { + // add a suffix to the new file's name to avoid conflicts with preiously uploaded files + var fileNameWithoutExtension = System.IO.Path.GetFileNameWithoutExtension(file.Name); + var fileExtension = System.IO.Path.GetExtension(file.Name); + var suffix = 1; + do + { + file.Name = $"{fileNameWithoutExtension} ({suffix}){fileExtension}"; + file.NormalizedName = GetNormalizedFileName(file.Name); + existingFile = await fileRepository.GetByName(folder.SiteId, folder.Id, file.NormalizedName, cancellationToken); + suffix++; + } while (existingFile != null); + } + + await fileRepository.Add(file, cancellationToken); + + var uploadsFolder = "files"; + Directory.CreateDirectory(uploadsFolder); // ensure folder exists + + var extension = Path.GetExtension(file.Name); + var filePath = Path.Combine(uploadsFolder, file.Id + extension); + + using var fileStream = System.IO.File.Create(filePath); + await fileContent.CopyToAsync(fileStream, cancellationToken); + + return file; + } + + public async Task Remove(Guid id, CancellationToken cancellationToken = default) + { + return await fileRepository.Remove(id, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FileUnableToDelete); + } + +// public async Task> GetAll(Guid siteId, CancellationToken cancellationToken = default) +// { +// return await fileRepository.GetAllForSite(siteId, cancellationToken); +// } + + public async Task> GetByFolderId(Guid folderId, CancellationToken cancellationToken) + { + return await fileRepository.GetByFolderId(folderId, cancellationToken); + } + +// public async Task GetById(Guid id, CancellationToken cancellationToken = default) +// { +// var file = await fileRepository.GetById(id, cancellationToken) ?? +// throw new AppException(ExceptionCodes.FileNotFound); + +// return file; +// } + +// public async Task GetByName(Guid folderId, string fileName, CancellationToken cancellationToken = default) +// { +// var folder = await folderRepository.GetById(folderId, cancellationToken) ?? +// throw new AppException(ExceptionCodes.FolderNotFound); + +// var normalizedFileName = GetNormalizedFileName(fileName); + +// return await fileRepository.GetByName(folder.SiteId, folder.Id, normalizedFileName, cancellationToken) ?? +// throw new AppException(ExceptionCodes.FileNotFound); +// } + +// public async Task GetFilePath(File file, CancellationToken cancellationToken = default) +// { +// var folders = await folderService.GetParentFolders(file.FolderId, cancellationToken) ?? +// throw new AppException(ExceptionCodes.FolderNotFound); + +// return string.Join("/", folders.Select(x => x.Name)) + "/" + file.Name; +// } + +// public async Task GetStream(Guid id, CancellationToken cancellationToken = default) +// { +// return await fileStorageProvider.Download(id.ToString(), cancellationToken) ?? +// throw new AppException(ExceptionCodes.FileNotFound); +// } + + public async Task Rename(Guid id, string name, CancellationToken cancellationToken = default) + { + var normalizedFileName = GetNormalizedFileName(name); + + var file = await fileRepository.GetById(id, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FileNotFound); + + if (!IsValidFileName(normalizedFileName)) + throw new EnhancedException(MessageCodes.FileInvalidName); + + // check if file with the same name already exists + var existingFile = await fileRepository.GetByName(file.SiteId, file.FolderId, normalizedFileName, cancellationToken); + if (existingFile != null) + throw new EnhancedException(MessageCodes.FileAlreadyExists); + + file.Name = name; + file.NormalizedName = normalizedFileName; + + return await fileRepository.Update(file, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FileUnableToUpdate); + } + + public async Task Move(Guid id, Guid folderId, CancellationToken cancellationToken = default) + { + var folder = await folderRepository.GetById(folderId, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FolderNotFound); + + var file = await fileRepository.GetById(id, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FileNotFound); + + if (file.SiteId != folder.SiteId) + throw new EnhancedException(MessageCodes.FolderNotFound); + + // check if file with the same name already exists + var exisitingFile = await fileRepository.GetByName(file.SiteId, folder.Id, file.NormalizedName, cancellationToken); + if (exisitingFile != null) + throw new EnhancedException(MessageCodes.FileAlreadyExists); + + file.FolderId = folderId; + + return await fileRepository.Update(file, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FileUnableToUpdate); + } + + private static readonly Regex _fileNameRegex = new(@"[a-zA-Z0-9_\-\.\(\)\s]+(\.[a-zA-Z0-9]{2,6})?"); + + private static bool IsValidFileName(string fileName) + { + if (string.IsNullOrWhiteSpace(fileName)) + return false; // Folder name should not be empty or whitespace + + return _fileNameRegex.IsMatch(fileName); + } + + private static string GetNormalizedFileName(string fileName) + { + var normalized = fileName.Trim().ToLower(); + return normalized; + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/FolderService.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/FolderService.cs new file mode 100644 index 000000000..081461369 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/FolderService.cs @@ -0,0 +1,204 @@ +// using System.Text.RegularExpressions; + +using System.Data; +using System.Globalization; +using System.Runtime.CompilerServices; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Services; + +public interface IFolderService +{ + Task Create(Folder folder, CancellationToken cancellationToken = default); + Task GetRoot(Guid siteId, CancellationToken cancellationToken = default); + Task GetById(Guid id, CancellationToken cancellationToken = default); + // Task GetByPath(Guid siteId, string folderPath, CancellationToken cancellationToken = default); + Task> GetByFolderId(Guid folderId, CancellationToken cancellationToken = default); + Task> GetParentFolders(Guid folderId, CancellationToken cancellationToken = default); + Task Rename(Guid folderId, string name, CancellationToken cancellationToken = default); + Task Move(Guid folderId, Guid parentFolderId, CancellationToken cancellationToken = default); + Task Remove(Guid id, CancellationToken cancellationToken = default); +} + +internal class FolderService(IFolderRepository folderRepository, IFileRepository fileRepository) : IFolderService +{ + public async Task GetById(Guid id, CancellationToken cancellationToken = default) + { + return await folderRepository.GetById(id, cancellationToken); + } + public async Task Create(Folder folder, CancellationToken cancellationToken = default) + { + folder.NormalizedName = GetNormalizedFolderName(folder.Name); + + if (!IsValidFolderName(folder.NormalizedName)) + throw new EnhancedException(MessageCodes.FolderInvalidName); + + // check if parent folder exists + if (folder.ParentId == null) + throw new EnhancedException(MessageCodes.FolderParentNotFound); + + _ = await folderRepository.GetById(folder.ParentId.Value, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FolderParentNotFound); + + // // check if folder with the same name already exists + var existingFolder = await folderRepository.GetByName(folder.SiteId, folder.ParentId, folder.NormalizedName, cancellationToken); + if (existingFolder != null) + throw new EnhancedException(MessageCodes.FolderAlreadyExists); + + return await folderRepository.Add(folder, cancellationToken); + // return await folderRepository.Create(folder, cancellationToken) ?? + // throw new EnhancedException(MessageCodes.FolderUnableToCreate); + } + + public async Task GetRoot(Guid siteId, CancellationToken cancellationToken = default) + { + var result = await folderRepository.GetByName(siteId, default, "files", cancellationToken); + Console.WriteLine("Getroot: " + siteId.ToString()); + Console.WriteLine("Getroot: " + result.Id); + return result; + } + + public async Task> GetParentFolders(Guid folderId, CancellationToken cancellationToken = default) + { + var folders = new List(); + var currentFolder = await folderRepository.GetById(folderId, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FolderNotFound); + + folders.Add(currentFolder); + + while (currentFolder != null && currentFolder.ParentId != null) + { + currentFolder = await folderRepository.GetById(currentFolder.ParentId.Value, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FolderNotFound); + if (currentFolder != null) + folders.Add(currentFolder); + } + + folders.Reverse(); + + return folders; + } + + + public async Task> GetByFolderId(Guid folderId, CancellationToken cancellationToken) + { + return await folderRepository.GetByFolderId(folderId, cancellationToken); + } + + public async Task Remove(Guid id, CancellationToken cancellationToken = default) + { + // TODO: Delete sub folders and files + var existingFolder = await folderRepository.GetById(id, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FolderNotFound); + + return await folderRepository.Remove(id, cancellationToken); + } + + // public Task GetByPath(Guid siteId, string folderPath, CancellationToken cancellationToken = default) + // { + // folderPath = GetNormalizedFolderName(folderPath); + + // // split folder path + // string[] folderNames = folderPath.Split('/', StringSplitOptions.RemoveEmptyEntries); + + // return FindFolder(siteId, null, folderNames, 0, cancellationToken); + // } + + // // find folder recursively by parent folder + // private async Task FindFolder(Guid siteId, Guid? parentId, string[] folderNames, int index, CancellationToken cancellationToken) + // { + // if (index == folderNames.Length - 1) + // { + // return await folderRepository.GetByName(siteId, parentId, folderNames[index], cancellationToken) ?? + // throw new AppException(ExceptionCodes.FolderNotFound); + // } + // var folder = await folderRepository.GetByName(siteId, parentId, folderNames[index], cancellationToken) ?? + // throw new AppException(ExceptionCodes.FolderNotFound); + + // return await FindFolder(siteId, folder.Id, folderNames, index + 1, cancellationToken); + // } + + public async Task Rename(Guid folderId, string name, CancellationToken cancellationToken = default) + { + var normalizedName = GetNormalizedFolderName(name); + + if (!IsValidFolderName(normalizedName)) + throw new EnhancedException(MessageCodes.FolderInvalidName); + + var existingFolder = await folderRepository.GetById(folderId, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FolderNotFound); + + // check if folder with the same name already exists + var existingFolderSameName = await folderRepository.GetByName(existingFolder.SiteId, existingFolder.ParentId, normalizedName, cancellationToken); + + // check if the folder is not root folder + if (existingFolder.ParentId == null) + throw new EnhancedException(MessageCodes.FolderCannotRenameRootFolder); + + if (existingFolderSameName != null && existingFolderSameName.Id != folderId) + throw new EnhancedException(MessageCodes.FolderAlreadyExists); + + existingFolder.Name = name; + existingFolder.NormalizedName = normalizedName; + + return await folderRepository.Update(existingFolder, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FolderUnableToUpdate); + } + + public async Task Move(Guid folderId, Guid parentFolderId, CancellationToken cancellationToken = default) + { + // check if folder is not moving to itself + if (folderId == parentFolderId) + throw new EnhancedException(MessageCodes.FolderCannotMoveToItself); + + // check if folder exists + var existingFolder = await folderRepository.GetById(folderId, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FolderNotFound); + + // check if the folder is not root folder + if (existingFolder.ParentId == null) + throw new EnhancedException(MessageCodes.FolderCannotMoveRootFolder); + + // check if parent folder exists + var parentFolder = await folderRepository.GetById(parentFolderId, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FolderParentNotFound); + + // check if folder is not moving to its child + var parentFolders = await GetParentFolders(parentFolderId, cancellationToken); + if (parentFolders.Any(x => x.Id == folderId)) + throw new EnhancedException(MessageCodes.FolderCannotMoveToChild); + + existingFolder.ParentId = parentFolderId; + + return await folderRepository.Update(existingFolder, cancellationToken) ?? + throw new EnhancedException(MessageCodes.FolderUnableToUpdate); + } + + private static string GetNormalizedFolderName(string folderName) + { + return folderName.Trim().ToLower(); + } + + // Regular expression to allow only alphanumeric, spaces, underscores, and certain special characters + private static readonly Regex _regex = new(@"^[\w\-. ]+$"); + + private static bool IsValidFolderName(string folderName) + { + if (string.IsNullOrWhiteSpace(folderName)) + return false; // Folder name should not be empty or whitespace + + return _regex.IsMatch(folderName); + } + + // private async Task DeleteSubFolders(Guid parentFolderId, List allFolders, List allFiles, CancellationToken cancellationToken = default) + // { + // var fileIds = allFiles.Where(x => x.FolderId == parentFolderId).Select(x => x.Id); + // await fileRepository.DeleteMany(fileIds, cancellationToken); + + // foreach (var childFolder in allFolders.Where(x => x.ParentId == parentFolderId)) + // await DeleteSubFolders(childFolder.Id, allFolders, allFiles, cancellationToken); + + // return await folderRepository.Delete(parentFolderId, cancellationToken) ?? + // throw new AppException(ExceptionCodes.FolderUnableToDelete); + // } + +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/LayoutService.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/LayoutService.cs new file mode 100644 index 000000000..d0889010f --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/LayoutService.cs @@ -0,0 +1,57 @@ +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Services; + +public interface ILayoutService +{ + Task> GetAll(CancellationToken cancellationToken = default); + Task> GetBySiteId(Guid siteId, CancellationToken cancellationToken = default); + Task GetById(Guid id, CancellationToken cancellationToken = default); + Task Add(Layout layout, CancellationToken cancellationToken = default); + Task Update(Layout layout, CancellationToken cancellationToken = default); + Task Remove(Guid id, CancellationToken cancellationToken = default); +} + +internal class LayoutService(ILayoutRepository layoutRepository, IEventPublisher eventPublisher) : ILayoutService +{ + public async Task> GetAll(CancellationToken cancellationToken = default) + { + return await layoutRepository.GetAll(cancellationToken); + } + + public async Task> GetBySiteId(Guid siteId, CancellationToken cancellationToken = default) + { + return await layoutRepository.GetAllForSite(siteId, cancellationToken); + } + + public async Task GetById(Guid id, CancellationToken cancellationToken = default) + { + return await layoutRepository.GetById(id, cancellationToken); + } + + public async Task Add(Layout layout, CancellationToken cancellationToken = default) + { + + await layoutRepository.Add(layout, cancellationToken); + + await eventPublisher.Publish(new LayoutAddedEvent(layout), cancellationToken); + + return layout; + } + + public async Task Update(Layout layout, CancellationToken cancellationToken = default) + { + await layoutRepository.Update(layout, cancellationToken); + + await eventPublisher.Publish(new LayoutUpdatedEvent(layout), cancellationToken); + + return layout; + } + + public async Task Remove(Guid id, CancellationToken cancellationToken = default) + { + var deletedLayout = await layoutRepository.Remove(id, cancellationToken); + + await eventPublisher.Publish(new LayoutRemovedEvent(deletedLayout), cancellationToken); + + return deletedLayout; + } +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/PageService.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/PageService.cs new file mode 100644 index 000000000..3ab416427 --- /dev/null +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/Services/PageService.cs @@ -0,0 +1,133 @@ +using FluentCMS.Api.Core.Models; + +namespace FluentCMS.Api.Plugins.CmsCoreManagement.Services; + +public interface IPageService +{ + Task> GetAll(CancellationToken cancellationToken = default); + Task GetById(Guid id, CancellationToken cancellationToken = default); + Task> GetBySiteId(Guid siteId, CancellationToken cancellationToken = default); + // Task GetByUrl(string url, CancellationToken cancellationToken = default); + Task GetPageUrl(Guid id, CancellationToken cancellationToken = default); + Task Add(Page page, CancellationToken cancellationToken = default); + Task Update(Page page, CancellationToken cancellationToken = default); + Task Remove(Guid id, CancellationToken cancellationToken = default); +} + +internal class PageService(IPageRepository pageRepository, IEventPublisher eventPublisher) : IPageService +{ + public const string VALID_DOMAIN_NAME_REGEX = @"^(([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]*[a-zA-Z0-9])\.)*([A-Za-z0-9]|[A-Za-z0-9][A-Za-z0-9\-]*[A-Za-z0-9])(:\d{1,5})?$"; + + public async Task> GetAll(CancellationToken cancellationToken = default) + { + return await pageRepository.GetAll(cancellationToken); + } + + public async Task GetById(Guid id, CancellationToken cancellationToken = default) + { + return await pageRepository.GetById(id, cancellationToken); + } + + public async Task> GetBySiteId(Guid siteId, CancellationToken cancellationToken = default) + { + return await pageRepository.GetAllForSite(siteId, cancellationToken); + } + + public async Task GetPageUrl(Guid id, CancellationToken cancellationToken = default) + { + var segments = new List(); + var current = await pageRepository.GetById(id, cancellationToken); + + while (current != null) + { + segments.Add(current.Slug); + current = current.ParentId != null + ? await pageRepository.GetById(current.ParentId.Value, cancellationToken) + : null; + } + + segments.Reverse(); + return "/" + string.Join("/", segments); + } + + // public async Task GetByUrl(string url, CancellationToken cancellationToken = default) + // { + // url = ValidateAndFormatUrl(url); + // // no need to check permissions + // return await pageRepository.GetByUrl(url, cancellationToken) ?? + // throw new EntityNotFoundException(); + // } + + public async Task Add(Page page, CancellationToken cancellationToken = default) + { + // ValidateAndFormatUrls(page); + + // check if page url are unique + // var allPages = await pageRepository.GetAll(cancellationToken); + // if (allPages.Any(x => x.Urls.Any(y => page.Urls.Contains(y)))) + // throw new EnhancedException(MessageCodes.PageUrlMustBeUnique); + + await pageRepository.Add(page, cancellationToken); + + await eventPublisher.Publish(new PageAddedEvent(page), cancellationToken); + + return page; + } + + public async Task Update(Page page, CancellationToken cancellationToken = default) + { + // ValidateAndFormatUrls(page); + + // check if page url is unique + // var allPages = await pageRepository.GetAll(cancellationToken); + // if (allPages.Any(x => x.Id != page.Id && x.Urls.Any(y => page.Urls.Contains(y)))) + // throw new EnhancedException(MessageCodes.PageUrlMustBeUnique); + + await pageRepository.Update(page, cancellationToken); + + await eventPublisher.Publish(new PageUpdatedEvent(page), cancellationToken); + + return page; + } + + public async Task Remove(Guid id, CancellationToken cancellationToken = default) + { + var deletedPage = await pageRepository.Remove(id, cancellationToken); + + await eventPublisher.Publish(new PageRemovedEvent(deletedPage), cancellationToken); + + return deletedPage; + } + + + // private static void ValidateAndFormatUrls(Page page) + // { + // page.Urls = [.. page.Urls.Select(ValidateAndFormatUrl)]; + // } + + + // private static string ValidateAndFormatUrl(string url) + // { + // // Trim spaces + // url = url.Trim(); + + // // Check if URL is empty + // if (string.IsNullOrEmpty(url)) + // throw new EnhancedException(MessageCodes.PageUrlCannotBeEmpty); + + // // Convert to lowercase + // url = url.ToLowerInvariant(); + + // // if the url ends with a slash, remove it + // if (url.EndsWith('/')) + // url = url[..^1]; + + // // Validate URL format using regex + // if (!Regex.IsMatch(url, VALID_DOMAIN_NAME_REGEX)) + // throw new EnhancedException(MessageCodes.PageUrlIsInvalid); + + // // Return the formatted URL if valid + // return url; + // } + +} diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/_GlobalUsings.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/_GlobalUsings.cs index 8e88e9b17..cdabd667b 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/_GlobalUsings.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.CmsCoreManagement/_GlobalUsings.cs @@ -11,6 +11,7 @@ global using FluentCMS.Api.Core.Controllers; global using FluentCMS.Api.Plugins.CmsCoreManagement.Services; global using FluentCMS.Api.Core; +global using Microsoft.AspNetCore.Http; global using Microsoft.AspNetCore.Mvc; global using FluentCMS.Api.Plugins.CmsCoreManagement.Dtos; global using FluentCMS.Infrastructure.Plugins.Abstractions; diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Controllers/UsersController.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Controllers/UsersController.cs index 0b60575f8..6756c84ad 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Controllers/UsersController.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Controllers/UsersController.cs @@ -28,10 +28,8 @@ public async Task> Add(UserAddRequest request, Cancellation [HttpPut("{id:guid}")] public async Task> Update(Guid id, UserUpdateRequest request, CancellationToken cancellationToken = default) { - var user = await userService.GetById(id, cancellationToken); - Mapper.Map(request, user); - await userService.Update(user, cancellationToken); - return Success(Mapper.Map(user)); + var updated = await userService.Update(id, request, cancellationToken); + return Success(Mapper.Map(updated)); } [HttpDelete("{id:guid}")] diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Repositories/RoleRepository.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Repositories/RoleRepository.cs index a2853f952..d76f6aad8 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Repositories/RoleRepository.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Repositories/RoleRepository.cs @@ -1,7 +1,7 @@ namespace FluentCMS.Api.Plugins.IdentityManagement.Repositories; -public interface IRoleRepository : ISiteAssociatedRepository +public interface IRoleRepository : IRepository { } -internal class RoleRepository(AppIdentityDbContext dbContext) : SiteAssociatedRepository(dbContext), IRoleRepository; +internal class RoleRepository(AppIdentityDbContext dbContext) : Repository(dbContext), IRoleRepository; diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Repositories/UserRoleRepository.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Repositories/UserRoleRepository.cs index 39373e24f..ab0b71f6e 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Repositories/UserRoleRepository.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Repositories/UserRoleRepository.cs @@ -1,14 +1,14 @@ namespace FluentCMS.Api.Plugins.IdentityManagement.Repositories; -public interface IUserRoleRepository : ISiteAssociatedRepository +public interface IUserRoleRepository : IRepository { - Task> GetUserRoles(Guid userId, Guid siteId, CancellationToken cancellationToken = default); + Task> GetUserRoles(Guid userId, CancellationToken cancellationToken = default); } -internal class UserRoleRepository(AppIdentityDbContext dbContext) : SiteAssociatedRepository(dbContext), IUserRoleRepository +internal class UserRoleRepository(AppIdentityDbContext dbContext) : Repository(dbContext), IUserRoleRepository { - public async Task> GetUserRoles(Guid userId, Guid siteId, CancellationToken cancellationToken = default) + public async Task> GetUserRoles(Guid userId, CancellationToken cancellationToken = default) { - return await Query().Where(x => x.SiteId == siteId && x.UserId == userId).ToList(cancellationToken); + return await Query().Where(x => x.UserId == userId).ToList(cancellationToken); } } diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Services/RoleService.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Services/RoleService.cs index e6eeebe91..a9947aa6f 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Services/RoleService.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Services/RoleService.cs @@ -15,7 +15,7 @@ internal class RoleService(ISecurityContext securityContext, IRoleRepository rol { public async Task> GetAll(CancellationToken cancellationToken = default) { - return await roleRepository.GetAllForSite(securityContext.GetSiteId(), cancellationToken); + return await roleRepository.GetAll(cancellationToken); } public async Task Add(Role role, CancellationToken cancellationToken) @@ -23,8 +23,6 @@ public async Task Add(Role role, CancellationToken cancellationToken) if (role.Type != RoleTypes.UserDefined) throw new EnhancedException(MessageCodes.RoleInvalidTypeForCreation); - ValidateSiteContext(role); - var identityResult = await roleManager.CreateAsync(role); identityResult.ThrowIfInvalid(); @@ -38,14 +36,17 @@ public async Task Update(Role role, CancellationToken cancellationToken) if (role.Type != RoleTypes.UserDefined) throw new EnhancedException(MessageCodes.RoleInvalidTypeForCreation); - ValidateSiteContext(role); - var existing = await roleManager.FindByIdAsync(role.Id.ToString()) ?? throw new EntityNotFoundException(role.Id); - ValidateSiteContext(existing); - var identityResult = await roleManager.UpdateAsync(role); + existing.Name = role.Name; + existing.NormalizedName = role.Name.ToUpperInvariant(); + existing.Description = role.Description; + + var identityResult = await roleManager.UpdateAsync(existing); + + identityResult.ThrowIfInvalid(); await eventPublisher.Publish(new RoleUpdatedEvent(role), cancellationToken); @@ -60,8 +61,6 @@ public async Task Remove(Guid roleId, CancellationToken cancellationToken if (role.Type != RoleTypes.UserDefined) throw new EnhancedException(MessageCodes.RoleDefaultCanNotBeDeleted); - ValidateSiteContext(role); - var identityResult = await roleManager.DeleteAsync(role); identityResult.ThrowIfInvalid(); @@ -73,20 +72,19 @@ public async Task Remove(Guid roleId, CancellationToken cancellationToken public async Task GetById(Guid roleId, CancellationToken cancellationToken = default) { var role = await roleRepository.GetById(roleId, cancellationToken); - ValidateSiteContext(role); return role; } public async Task> GetUserRoles(Guid userId, CancellationToken cancellationToken = default) { - var userRoles = await userRoleRepository.GetUserRoles(userId, securityContext.GetSiteId(), cancellationToken); + var userRoles = await userRoleRepository.GetUserRoles(userId, cancellationToken); var roles = await roleRepository.GetByIds(userRoles.Select(ur => ur.RoleId), cancellationToken); return roles; } public async Task UpdateUserRoles(Guid userId, IEnumerable roleIds, CancellationToken cancellationToken = default) { - var existingUserRoles = await userRoleRepository.GetUserRoles(userId, securityContext.GetSiteId(), cancellationToken); + var existingUserRoles = await userRoleRepository.GetUserRoles(userId, cancellationToken); var roleIdsToRemove = existingUserRoles.Where(r => !roleIds.Contains(r.RoleId)); var roleIdsToAdd = roleIds.Where(r => !existingUserRoles.Any(er => er.RoleId == r)); @@ -95,13 +93,6 @@ public async Task UpdateUserRoles(Guid userId, IEnumerable roleIds, Cancel { UserId = userId, RoleId = rid, - SiteId = securityContext.GetSiteId() }), cancellationToken); } - - private void ValidateSiteContext(Role role) - { - if (role.SiteId != securityContext.GetSiteId()) - throw new EnhancedException(MessageCodes.RoleSiteIdMismatch); - } } diff --git a/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Services/UserService.cs b/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Services/UserService.cs index 445fb9e65..a71574bf9 100644 --- a/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Services/UserService.cs +++ b/src/Api/Plugins/FluentCMS.Api.Plugins.IdentityManagement/Services/UserService.cs @@ -3,7 +3,7 @@ public interface IUserService { Task Add(User user, string password, CancellationToken cancellationToken = default); - Task Update(User user, CancellationToken cancellationToken = default); + Task Update(Guid id, UserUpdateRequest user, CancellationToken cancellationToken = default); Task Remove(Guid id, CancellationToken cancellationToken = default); Task ChangePassword(Guid userId, string newPassword, CancellationToken cancellationToken = default); Task> GetAll(CancellationToken cancellationToken = default); @@ -36,8 +36,22 @@ public async Task ChangePassword(Guid userId, string newPassword, Cancella return user; } - public async Task Update(User user, CancellationToken cancellationToken = default) + public async Task Update(Guid id, UserUpdateRequest request, CancellationToken cancellationToken = default) { + // Get the tracked identity user (with SecurityStamp, ConcurrencyStamp, etc.) + var user = await userManager.FindByIdAsync(id.ToString()); + if (user == null) + throw new Exception("User not found"); + + // Update only the fields you expose in the API + user.Email = request.Email; + user.Description = request.Description; + user.Suspended = request.Suspended; + user.IsSuperAdmin = request.IsSuperAdmin; + user.EmailConfirmed = request.EmailConfirmed; + user.LockoutEnabled = request.Locked; + + // Identity will validate security stamp, concurrency stamp, etc. var result = await userManager.UpdateAsync(user); result.ThrowIfInvalid(); diff --git a/src/Backend/FluentCMS.Services/FolderService.cs b/src/Backend/FluentCMS.Services/FolderService.cs index ad071c3b9..38c3a61fc 100644 --- a/src/Backend/FluentCMS.Services/FolderService.cs +++ b/src/Backend/FluentCMS.Services/FolderService.cs @@ -73,25 +73,24 @@ public async Task GetById(Guid id, CancellationToken cancellationToken = return folder; } - public async Task> GetParentFolders(Guid folderId, CancellationToken cancellationToken = default) { var folders = new List(); - var currentFolder = await folderRepository.GetById(folderId, cancellationToken) ?? - throw new AppException(ExceptionCodes.FolderNotFound); + var current = await folderRepository.GetById(folderId, cancellationToken) + ?? throw new AppException(ExceptionCodes.FolderNotFound); - folders.Add(currentFolder); - - while (currentFolder != null && currentFolder.ParentId != null) + while (current != null) { - currentFolder = await folderRepository.GetById(currentFolder.ParentId.Value, cancellationToken) ?? - throw new AppException(ExceptionCodes.FolderNotFound); - if (currentFolder != null) - folders.Add(currentFolder); + folders.Add(current); + + if (current.ParentId == null) + break; + + current = await folderRepository.GetById(current.ParentId.Value, cancellationToken) + ?? throw new AppException(ExceptionCodes.FolderNotFound); } folders.Reverse(); - return folders; } diff --git a/src/FluentCMS.sln b/src/FluentCMS.sln index be2256d0d..8f50c3494 100644 --- a/src/FluentCMS.sln +++ b/src/FluentCMS.sln @@ -215,9 +215,14 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "FluentCMS.Web.UI.Components EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "TestApp", "UI\TestApp\TestApp.csproj", "{F7295900-404E-470D-94C5-8BB07E85FA85}" EndProject + +Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "FluentCMS.Api.Plugins.AIAgentManagement", "Api\Plugins\FluentCMS.Api.Plugins.AIAgentManagement\FluentCMS.Api.Plugins.AIAgentManagement.csproj", "{3F2740FD-A4CC-153C-4E2C-EF1EF4B4F634}" +EndProject + Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "FluentCMS.Infrastructure.EventBus.Example", "Infrastructure\EventBus\FluentCMS.Infrastructure.EventBus.Example\FluentCMS.Infrastructure.EventBus.Example.csproj", "{2CAD7D0E-0872-750D-3309-F0930D2BBE3E}" EndProject Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "FluentCMS.Infrastructure.EventBus.Tests", "Infrastructure\EventBus\FluentCMS.Infrastructure.EventBus.Tests\FluentCMS.Infrastructure.EventBus.Tests.csproj", "{95243C21-EF32-E6B5-527C-31182D655B4D}" + EndProject Global GlobalSection(SolutionConfigurationPlatforms) = preSolution @@ -537,6 +542,10 @@ Global {F7295900-404E-470D-94C5-8BB07E85FA85}.Debug|Any CPU.Build.0 = Debug|Any CPU {F7295900-404E-470D-94C5-8BB07E85FA85}.Release|Any CPU.ActiveCfg = Release|Any CPU {F7295900-404E-470D-94C5-8BB07E85FA85}.Release|Any CPU.Build.0 = Release|Any CPU + {3F2740FD-A4CC-153C-4E2C-EF1EF4B4F634}.Debug|Any CPU.ActiveCfg = Debug|Any CPU + {3F2740FD-A4CC-153C-4E2C-EF1EF4B4F634}.Debug|Any CPU.Build.0 = Debug|Any CPU + {3F2740FD-A4CC-153C-4E2C-EF1EF4B4F634}.Release|Any CPU.ActiveCfg = Release|Any CPU + {3F2740FD-A4CC-153C-4E2C-EF1EF4B4F634}.Release|Any CPU.Build.0 = Release|Any CPU {2CAD7D0E-0872-750D-3309-F0930D2BBE3E}.Debug|Any CPU.ActiveCfg = Debug|Any CPU {2CAD7D0E-0872-750D-3309-F0930D2BBE3E}.Debug|Any CPU.Build.0 = Debug|Any CPU {2CAD7D0E-0872-750D-3309-F0930D2BBE3E}.Release|Any CPU.ActiveCfg = Release|Any CPU @@ -648,6 +657,7 @@ Global {F1016288-2279-471B-90F0-5F584883FD2A} = {BA962DC1-64AB-4F30-A69C-DABFE345845B} {62BE2AB5-64E3-4F42-B65E-DBAF82C67FAE} = {F40F4AFA-1FF0-484C-A27F-0E00FCC5D638} {F7295900-404E-470D-94C5-8BB07E85FA85} = {F40F4AFA-1FF0-484C-A27F-0E00FCC5D638} + {3F2740FD-A4CC-153C-4E2C-EF1EF4B4F634} = {BA962DC1-64AB-4F30-A69C-DABFE345845B} {2CAD7D0E-0872-750D-3309-F0930D2BBE3E} = {A387EFCC-4699-4B0D-A746-DA1AD9A8436C} {95243C21-EF32-E6B5-527C-31182D655B4D} = {A387EFCC-4699-4B0D-A746-DA1AD9A8436C} EndGlobalSection diff --git a/src/UI/FluentCMS.Web.ApiClients/ApiClientException.cs b/src/UI/FluentCMS.Web.ApiClients/ApiClientException.cs new file mode 100644 index 000000000..bc2109d37 --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/ApiClientException.cs @@ -0,0 +1,32 @@ +using System.Text.Json; + +namespace FluentCMS.Web.ApiClients; + +public class ApiClientException : Exception +{ + public int StatusCode { get; private set; } + public string? Response { get; private set; } + public IReadOnlyDictionary> Headers { get; private set; } + + public ApiExceptionResult? ApiResult { get; private set; } + + public ApiClientException(string message, int statusCode, string? response, IReadOnlyDictionary> headers, Exception? innerException) : base(message, innerException) + { + StatusCode = statusCode; + Response = response; + Headers = headers; + + if (string.IsNullOrEmpty(response)) + return; + + try + { + var options = new JsonSerializerOptions { PropertyNamingPolicy = JsonNamingPolicy.CamelCase }; + ApiResult = JsonSerializer.Deserialize(response, options); + } + catch (Exception) + { + // ignored + } + } +} diff --git a/src/UI/FluentCMS.Web.ApiClients/ApiClientFactory.cs b/src/UI/FluentCMS.Web.ApiClients/ApiClientFactory.cs new file mode 100644 index 000000000..dd34d22f2 --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/ApiClientFactory.cs @@ -0,0 +1,33 @@ +using Microsoft.Extensions.DependencyInjection; + +namespace FluentCMS.Web.ApiClients; + +// factory pattern implementation to return IApiClient +public class ApiClientFactory(IServiceProvider serviceProvider) +{ + private T Get() where T : IApiClient + { + return serviceProvider.GetRequiredService(); + } + + public IPagesClient Pages => Get(); + public ISitesClient Sites => Get(); + // public IApiTokenClient ApiToken => Get(); + public IRolesClient Roles => Get(); + public IUsersClient Users => Get(); + // public ISettingsClient Settings => Get(); + // public IPluginClient Plugin => Get(); + // public IPluginContentClient PluginContent => Get(); + // public IPluginDefinitionClient PluginDefinition => Get(); + public ILayoutsClient Layouts => Get(); + public IAccountsClient Accounts => Get(); + // public IContentsClient Content => Get(); + // public IContentTypeClient ContentType => Get(); + public IFoldersClient Folders => Get(); + public IFilesClient Files => Get(); + // public IGlobalSettingsClient GlobalSettings => Get(); + // public ISetupClient Setup => Get(); + // public IUserRolesClient UserRole => Get(); + public IBlocksClient Blocks => Get(); + +} diff --git a/src/UI/FluentCMS.Web.ApiClients/ApiClientServiceExtensions.cs b/src/UI/FluentCMS.Web.ApiClients/ApiClientServiceExtensions.cs new file mode 100644 index 000000000..8a00a423c --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/ApiClientServiceExtensions.cs @@ -0,0 +1,93 @@ +using FluentCMS.Web.ApiClients; +using Microsoft.AspNetCore.Components.Authorization; +using Microsoft.AspNetCore.Http; +using Microsoft.Extensions.Configuration; +using Microsoft.Extensions.Options; +using System.Net.Http.Headers; +using System.Reflection; +using System.Security.Claims; + +namespace Microsoft.Extensions.DependencyInjection; + +public static class ApiClientServiceExtensions +{ + public const string HTTP_CLIENT_API_NAME = "FluentCMS.Web.Api"; + + public static IServiceCollection AddApiClients(this IServiceCollection services, IConfiguration configuration) + { + services.Configure(configuration.GetSection("ClientSettings")); + + // TODO: these DI registration should be done automatically inside plugins + // services.AddScoped(); + // services.AddScoped(); + + services.AddScoped(); + + services.AddHttpClient(HTTP_CLIENT_API_NAME, (sp, client) => + { + using var scope = sp.CreateScope(); + var clientSettings = scope.ServiceProvider.GetService>()?.Value; + + var apiUrl = clientSettings?.Url; + if (string.IsNullOrWhiteSpace(apiUrl)) + throw new NullReferenceException("AppSettings.Url is null!"); + + client.BaseAddress = new Uri(apiUrl); + client.DefaultRequestHeaders.Accept.Clear(); + client.DefaultRequestHeaders.Accept.Add(new MediaTypeWithQualityHeaderValue("application/json")); + }); + + // find all interfaces that inherit from IApiClient + var apiClientTypes = Assembly.GetAssembly(typeof(IApiClient))?.GetTypes() + .Where(t => t.IsInterface && typeof(IApiClient).IsAssignableFrom(t)); + + // register all IApiClient interfaces with their implementation + foreach (var apiClientType in apiClientTypes ?? []) + { + // find the implementation of the interface + var implementationType = Assembly.GetAssembly(apiClientType)?.GetTypes() + .FirstOrDefault(t => apiClientType.IsAssignableFrom(t) && !t.IsInterface); + + // register the interface with the implementation + if (implementationType != null) + services.AddScoped(apiClientType, sp => + { + var httpClient = sp.GetRequiredService().CreateClient(HTTP_CLIENT_API_NAME); + + var clientSettings = sp.GetService>()?.CurrentValue; + var apiKey = clientSettings?.Key ?? ""; + httpClient.DefaultRequestHeaders.Add("X-API-AUTH", apiKey); + + var userToken = GetUserJwt(sp); + + if (!string.IsNullOrEmpty(userToken)) + httpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", userToken); + + var ctor = implementationType.GetConstructor([typeof(HttpClient)]) ?? + throw new InvalidOperationException($"Could not find constructor for {implementationType.Name}"); + + return ctor.Invoke([httpClient]); + }); + } + return services; + } + + private static string? GetUserJwt(IServiceProvider sp) + { + try + { + var authStateProvider = sp.GetRequiredService(); + var authStateTask = authStateProvider.GetAuthenticationStateAsync(); + var authState = authStateTask.GetAwaiter().GetResult(); + + if (authState?.User?.Identity == null || !authState.User.Identity.IsAuthenticated) + return default; + + return authState.User.FindFirstValue("jwt") ?? default; + } + catch (Exception) + { + return default; + } + } +} diff --git a/src/UI/FluentCMS.Web.ApiClients/ApiExceptionResult.cs b/src/UI/FluentCMS.Web.ApiClients/ApiExceptionResult.cs new file mode 100644 index 000000000..593d09bd5 --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/ApiExceptionResult.cs @@ -0,0 +1,11 @@ +namespace FluentCMS.Web.ApiClients; + +public class ApiExceptionResult +{ + public List? Errors { get; set; } + public string TraceId { get; set; } = string.Empty; + public string SessionId { get; set; } = string.Empty; + public string UniqueId { get; set; } = string.Empty; + public double Duration { get; set; } + public int Status { get; set; } +} diff --git a/src/UI/FluentCMS.Web.ApiClients/ClientSettings.cs b/src/UI/FluentCMS.Web.ApiClients/ClientSettings.cs new file mode 100644 index 000000000..35d71f860 --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/ClientSettings.cs @@ -0,0 +1,7 @@ +namespace FluentCMS.Web.ApiClients; + +public class ClientSettings +{ + public string Url { get; set; } = default!; + public string Key { get; set; } = default!; +} diff --git a/src/UI/FluentCMS.Web.ApiClients/FilesClient.cs b/src/UI/FluentCMS.Web.ApiClients/FilesClient.cs new file mode 100644 index 000000000..9e41c7d7b --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/FilesClient.cs @@ -0,0 +1,74 @@ +namespace FluentCMS.Web.ApiClients; + +public partial interface IFilesClient : IApiClient +{ + Task DownloadGetResponseMessageByIdAsync(Guid? id, CancellationToken cancellationToken = default); + Task DownloadGetResponseMessageAsync(string? url, CancellationToken cancellationToken = default); +} + +public partial class FilesClient +{ + public async Task DownloadGetResponseMessageByIdAsync(Guid? id, CancellationToken cancellationToken = default) + { + var disposeClient_ = false; + try + { + using var request_ = new HttpRequestMessage(); + + request_.Method = new HttpMethod("GET"); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/File/Download/{Id}" + urlBuilder_.Append("api/Files/Download/"); + + urlBuilder_.Append(id); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new Uri(url_, UriKind.RelativeOrAbsolute); + + return await _httpClient.SendAsync(request_, HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + } + finally + { + if (disposeClient_) + _httpClient.Dispose(); + } + } + + public async Task DownloadGetResponseMessageAsync(string? url, CancellationToken cancellationToken = default) + { + var disposeClient_ = false; + try + { + using var request_ = new HttpRequestMessage(); + + request_.Method = new HttpMethod("GET"); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/File/Download" + urlBuilder_.Append("api/Files/Download"); + urlBuilder_.Append('?'); + if (url != null) + { + var htmlDecodedUrl = System.Net.WebUtility.HtmlDecode(url); + urlBuilder_.Append(Uri.EscapeDataString("url")).Append('=').Append(Uri.EscapeDataString(htmlDecodedUrl)); + } + else + { + urlBuilder_.Length--; + } + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new Uri(url_, UriKind.RelativeOrAbsolute); + + return await _httpClient.SendAsync(request_, HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + } + finally + { + if (disposeClient_) + _httpClient.Dispose(); + } + } +} diff --git a/src/UI/FluentCMS.Web.ApiClients/FluentCMS.Web.ApiClients.csproj b/src/UI/FluentCMS.Web.ApiClients/FluentCMS.Web.ApiClients.csproj new file mode 100644 index 000000000..b35f5ae62 --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/FluentCMS.Web.ApiClients.csproj @@ -0,0 +1,38 @@ + + + net9.0 + enable + enable + true + FluentCMS.Web.ApiClients + 0.0.1 + Amir Pournasserian + FluentCMS + Web ApiClients. + fluentcms;cms;apiclient;plugin;core + https://github.com/fluentcms/FluentCMS + https://fluentcms.com + icon.png + MIT + README.md + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.ApiClients/GeneratedApiClients.cs b/src/UI/FluentCMS.Web.ApiClients/GeneratedApiClients.cs new file mode 100755 index 000000000..f8b32e26b --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/GeneratedApiClients.cs @@ -0,0 +1,10982 @@ +//---------------------- +// +// Generated using the NSwag toolchain v14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0)) (http://NSwag.org) +// +//---------------------- + +#nullable enable + +#pragma warning disable 108 // Disable "CS0108 '{derivedDto}.ToJson()' hides inherited member '{dtoBase}.ToJson()'. Use the new keyword if hiding was intended." +#pragma warning disable 114 // Disable "CS0114 '{derivedDto}.RaisePropertyChanged(String)' hides inherited member 'dtoBase.RaisePropertyChanged(String)'. To make the current member override that implementation, add the override keyword. Otherwise add the new keyword." +#pragma warning disable 472 // Disable "CS0472 The result of the expression is always 'false' since a value of type 'Int32' is never equal to 'null' of type 'Int32?' +#pragma warning disable 612 // Disable "CS0612 '...' is obsolete" +#pragma warning disable 649 // Disable "CS0649 Field is never assigned to, and will always have its default value null" +#pragma warning disable 1573 // Disable "CS1573 Parameter '...' has no matching param tag in the XML comment for ... +#pragma warning disable 1591 // Disable "CS1591 Missing XML comment for publicly visible type or member ..." +#pragma warning disable 8073 // Disable "CS8073 The result of the expression is always 'false' since a value of type 'T' is never equal to 'null' of type 'T?'" +#pragma warning disable 3016 // Disable "CS3016 Arrays as attribute arguments is not CLS-compliant" +#pragma warning disable 8600 // Disable "CS8600 Converting null literal or possible null value to non-nullable type" +#pragma warning disable 8602 // Disable "CS8602 Dereference of a possibly null reference" +#pragma warning disable 8603 // Disable "CS8603 Possible null reference return" +#pragma warning disable 8604 // Disable "CS8604 Possible null reference argument for parameter" +#pragma warning disable 8625 // Disable "CS8625 Cannot convert null literal to non-nullable reference type" +#pragma warning disable 8765 // Disable "CS8765 Nullability of type of parameter doesn't match overridden member (possibly because of nullability attributes)." + +namespace FluentCMS.Web.ApiClients +{ + using System = global::System; + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface IAccountsClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RegisterAsync(AccountRegisterRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RegisterAsync(AccountRegisterRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task LoginAsync(AccountLoginRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task LoginAsync(AccountLoginRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task LogoutAsync(); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task LogoutAsync(System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task ConfirmEmailAsync(AccountConfirmEmailRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task ConfirmEmailAsync(AccountConfirmEmailRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task ResendConfirmationAsync(AccountResendConfirmationRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task ResendConfirmationAsync(AccountResendConfirmationRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task ForgotPasswordAsync(AccountForgotPasswordRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task ForgotPasswordAsync(AccountForgotPasswordRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task ResetPasswordAsync(AccountResetPasswordRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task ResetPasswordAsync(AccountResetPasswordRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task ChangePasswordAsync(AccountChangePasswordRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task ChangePasswordAsync(AccountChangePasswordRequest? body, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AccountsClient : IAccountsClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public AccountsClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RegisterAsync(AccountRegisterRequest? body) + { + return RegisterAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RegisterAsync(AccountRegisterRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Accounts/Register" + urlBuilder_.Append("api/Accounts/Register"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task LoginAsync(AccountLoginRequest? body) + { + return LoginAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task LoginAsync(AccountLoginRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Accounts/Login" + urlBuilder_.Append("api/Accounts/Login"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task LogoutAsync() + { + return LogoutAsync(System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task LogoutAsync(System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Accounts/Logout" + urlBuilder_.Append("api/Accounts/Logout"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task ConfirmEmailAsync(AccountConfirmEmailRequest? body) + { + return ConfirmEmailAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task ConfirmEmailAsync(AccountConfirmEmailRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Accounts/ConfirmEmail" + urlBuilder_.Append("api/Accounts/ConfirmEmail"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task ResendConfirmationAsync(AccountResendConfirmationRequest? body) + { + return ResendConfirmationAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task ResendConfirmationAsync(AccountResendConfirmationRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Accounts/ResendConfirmation" + urlBuilder_.Append("api/Accounts/ResendConfirmation"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task ForgotPasswordAsync(AccountForgotPasswordRequest? body) + { + return ForgotPasswordAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task ForgotPasswordAsync(AccountForgotPasswordRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Accounts/ForgotPassword" + urlBuilder_.Append("api/Accounts/ForgotPassword"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task ResetPasswordAsync(AccountResetPasswordRequest? body) + { + return ResetPasswordAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task ResetPasswordAsync(AccountResetPasswordRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Accounts/ResetPassword" + urlBuilder_.Append("api/Accounts/ResetPassword"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task ChangePasswordAsync(AccountChangePasswordRequest? body) + { + return ChangePasswordAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task ChangePasswordAsync(AccountChangePasswordRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Accounts/ChangePassword" + urlBuilder_.Append("api/Accounts/ChangePassword"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface IAgentsClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task CreateAsync(Agent? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task CreateAsync(Agent? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(Agent? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(Agent? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AgentsClient : IAgentsClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public AgentsClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetAllAsync() + { + return GetAllAsync(System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Agents/GetAll" + urlBuilder_.Append("api/Agents/GetAll"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetByIdAsync(System.Guid id) + { + return GetByIdAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Agents/GetById/{id}" + urlBuilder_.Append("api/Agents/GetById/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task CreateAsync(Agent? body) + { + return CreateAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task CreateAsync(Agent? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Agents/Create" + urlBuilder_.Append("api/Agents/Create"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task UpdateAsync(Agent? body) + { + return UpdateAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task UpdateAsync(Agent? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Agents/Update" + urlBuilder_.Append("api/Agents/Update"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RemoveAsync(System.Guid id) + { + return RemoveAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("DELETE"); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Agents/Remove/{id}" + urlBuilder_.Append("api/Agents/Remove/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + return; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface IBlocksClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(BlockAddRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(BlockAddRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, BlockUpdateRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, BlockUpdateRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class BlocksClient : IBlocksClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public BlocksClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId) + { + return GetBySiteIdAsync(siteId, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Blocks/GetBySiteId" + urlBuilder_.Append("api/Blocks/GetBySiteId"); + urlBuilder_.Append('?'); + if (siteId != null) + { + urlBuilder_.Append(System.Uri.EscapeDataString("siteId")).Append('=').Append(System.Uri.EscapeDataString(ConvertToString(siteId, System.Globalization.CultureInfo.InvariantCulture))).Append('&'); + } + urlBuilder_.Length--; + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetAllAsync() + { + return GetAllAsync(System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Blocks/GetAll" + urlBuilder_.Append("api/Blocks/GetAll"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetByIdAsync(System.Guid id) + { + return GetByIdAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Blocks/GetById/{id}" + urlBuilder_.Append("api/Blocks/GetById/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task AddAsync(BlockAddRequest? body) + { + return AddAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task AddAsync(BlockAddRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Blocks/Add" + urlBuilder_.Append("api/Blocks/Add"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task UpdateAsync(System.Guid id, BlockUpdateRequest? body) + { + return UpdateAsync(id, body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task UpdateAsync(System.Guid id, BlockUpdateRequest? body, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Blocks/Update/{id}" + urlBuilder_.Append("api/Blocks/Update/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RemoveAsync(System.Guid id) + { + return RemoveAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("DELETE"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Blocks/Remove/{id}" + urlBuilder_.Append("api/Blocks/Remove/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface IFilesClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UploadAsync(System.Guid? folderId, System.Collections.Generic.IEnumerable files); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UploadAsync(System.Guid? folderId, System.Collections.Generic.IEnumerable files, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RenameAsync(FileRenameRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RenameAsync(FileRenameRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task MoveAsync(FileMoveRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task MoveAsync(FileMoveRequest? body, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FilesClient : IFilesClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public FilesClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task UploadAsync(System.Guid? folderId, System.Collections.Generic.IEnumerable files) + { + return UploadAsync(folderId, files, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task UploadAsync(System.Guid? folderId, System.Collections.Generic.IEnumerable files, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var boundary_ = System.Guid.NewGuid().ToString(); + var content_ = new System.Net.Http.MultipartFormDataContent(boundary_); + content_.Headers.Remove("Content-Type"); + content_.Headers.TryAddWithoutValidation("Content-Type", "multipart/form-data; boundary=" + boundary_); + + if (files == null) + throw new System.ArgumentNullException("files"); + else + { + foreach (var item_ in files) + { + var content_files_ = new System.Net.Http.StreamContent(item_.Data); + if (!string.IsNullOrEmpty(item_.ContentType)) + content_files_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse(item_.ContentType); + content_.Add(content_files_, "files", item_.FileName ?? "files"); + } + } + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Files/Upload" + urlBuilder_.Append("api/Files/Upload"); + urlBuilder_.Append('?'); + if (folderId != null) + { + urlBuilder_.Append(System.Uri.EscapeDataString("folderId")).Append('=').Append(System.Uri.EscapeDataString(ConvertToString(folderId, System.Globalization.CultureInfo.InvariantCulture))).Append('&'); + } + urlBuilder_.Length--; + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RemoveAsync(System.Guid id) + { + return RemoveAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("DELETE"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Files/Remove/{id}" + urlBuilder_.Append("api/Files/Remove/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RenameAsync(FileRenameRequest? body) + { + return RenameAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RenameAsync(FileRenameRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Files/Rename" + urlBuilder_.Append("api/Files/Rename"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task MoveAsync(FileMoveRequest? body) + { + return MoveAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task MoveAsync(FileMoveRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Files/Move" + urlBuilder_.Append("api/Files/Move"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface IFoldersClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetRootAsync(System.Guid? siteId); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetRootAsync(System.Guid? siteId, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task CreateAsync(FolderAddRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task CreateAsync(FolderAddRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RenameAsync(FolderRenameRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RenameAsync(FolderRenameRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task MoveAsync(FolderMoveRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task MoveAsync(FolderMoveRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetParentFoldersAsync(System.Guid? folderId); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetParentFoldersAsync(System.Guid? folderId, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FoldersClient : IFoldersClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public FoldersClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetRootAsync(System.Guid? siteId) + { + return GetRootAsync(siteId, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetRootAsync(System.Guid? siteId, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Folders/GetRoot" + urlBuilder_.Append("api/Folders/GetRoot"); + urlBuilder_.Append('?'); + if (siteId != null) + { + urlBuilder_.Append(System.Uri.EscapeDataString("siteId")).Append('=').Append(System.Uri.EscapeDataString(ConvertToString(siteId, System.Globalization.CultureInfo.InvariantCulture))).Append('&'); + } + urlBuilder_.Length--; + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetByIdAsync(System.Guid id) + { + return GetByIdAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Folders/GetById/{id}" + urlBuilder_.Append("api/Folders/GetById/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task CreateAsync(FolderAddRequest? body) + { + return CreateAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task CreateAsync(FolderAddRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Folders/Create" + urlBuilder_.Append("api/Folders/Create"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RenameAsync(FolderRenameRequest? body) + { + return RenameAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RenameAsync(FolderRenameRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Folders/Rename" + urlBuilder_.Append("api/Folders/Rename"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task MoveAsync(FolderMoveRequest? body) + { + return MoveAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task MoveAsync(FolderMoveRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Folders/Move" + urlBuilder_.Append("api/Folders/Move"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetParentFoldersAsync(System.Guid? folderId) + { + return GetParentFoldersAsync(folderId, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetParentFoldersAsync(System.Guid? folderId, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Folders/GetParentFolders" + urlBuilder_.Append("api/Folders/GetParentFolders"); + urlBuilder_.Append('?'); + if (folderId != null) + { + urlBuilder_.Append(System.Uri.EscapeDataString("folderId")).Append('=').Append(System.Uri.EscapeDataString(ConvertToString(folderId, System.Globalization.CultureInfo.InvariantCulture))).Append('&'); + } + urlBuilder_.Length--; + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RemoveAsync(System.Guid id) + { + return RemoveAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("DELETE"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Folders/Remove/{id}" + urlBuilder_.Append("api/Folders/Remove/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface ILayoutsClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(LayoutAddRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(LayoutAddRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, LayoutUpdateRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, LayoutUpdateRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class LayoutsClient : ILayoutsClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public LayoutsClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId) + { + return GetBySiteIdAsync(siteId, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Layouts/GetBySiteId" + urlBuilder_.Append("api/Layouts/GetBySiteId"); + urlBuilder_.Append('?'); + if (siteId != null) + { + urlBuilder_.Append(System.Uri.EscapeDataString("siteId")).Append('=').Append(System.Uri.EscapeDataString(ConvertToString(siteId, System.Globalization.CultureInfo.InvariantCulture))).Append('&'); + } + urlBuilder_.Length--; + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetAllAsync() + { + return GetAllAsync(System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Layouts/GetAll" + urlBuilder_.Append("api/Layouts/GetAll"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetByIdAsync(System.Guid id) + { + return GetByIdAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Layouts/GetById/{id}" + urlBuilder_.Append("api/Layouts/GetById/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task AddAsync(LayoutAddRequest? body) + { + return AddAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task AddAsync(LayoutAddRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Layouts/Add" + urlBuilder_.Append("api/Layouts/Add"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task UpdateAsync(System.Guid id, LayoutUpdateRequest? body) + { + return UpdateAsync(id, body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task UpdateAsync(System.Guid id, LayoutUpdateRequest? body, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Layouts/Update/{id}" + urlBuilder_.Append("api/Layouts/Update/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RemoveAsync(System.Guid id) + { + return RemoveAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("DELETE"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Layouts/Remove/{id}" + urlBuilder_.Append("api/Layouts/Remove/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface IPagesClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(PageAddRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(PageAddRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, PageUpdateRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, PageUpdateRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class PagesClient : IPagesClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public PagesClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId) + { + return GetBySiteIdAsync(siteId, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetBySiteIdAsync(System.Guid? siteId, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Pages/GetBySiteId" + urlBuilder_.Append("api/Pages/GetBySiteId"); + urlBuilder_.Append('?'); + if (siteId != null) + { + urlBuilder_.Append(System.Uri.EscapeDataString("siteId")).Append('=').Append(System.Uri.EscapeDataString(ConvertToString(siteId, System.Globalization.CultureInfo.InvariantCulture))).Append('&'); + } + urlBuilder_.Length--; + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetAllAsync() + { + return GetAllAsync(System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Pages/GetAll" + urlBuilder_.Append("api/Pages/GetAll"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetByIdAsync(System.Guid id) + { + return GetByIdAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Pages/GetById/{id}" + urlBuilder_.Append("api/Pages/GetById/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task AddAsync(PageAddRequest? body) + { + return AddAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task AddAsync(PageAddRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Pages/Add" + urlBuilder_.Append("api/Pages/Add"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task UpdateAsync(System.Guid id, PageUpdateRequest? body) + { + return UpdateAsync(id, body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task UpdateAsync(System.Guid id, PageUpdateRequest? body, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Pages/Update/{id}" + urlBuilder_.Append("api/Pages/Update/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RemoveAsync(System.Guid id) + { + return RemoveAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("DELETE"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Pages/Remove/{id}" + urlBuilder_.Append("api/Pages/Remove/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface IRolesClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(RoleAddRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(RoleAddRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, RoleUpdateRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, RoleUpdateRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetUserRolesAsync(System.Guid? userId); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetUserRolesAsync(System.Guid? userId, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateUserRolesAsync(UserRolesUpdateRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateUserRolesAsync(UserRolesUpdateRequest? body, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class RolesClient : IRolesClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public RolesClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetByIdAsync(System.Guid id) + { + return GetByIdAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Roles/GetById/{id}" + urlBuilder_.Append("api/Roles/GetById/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetAllAsync() + { + return GetAllAsync(System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Roles/GetAll" + urlBuilder_.Append("api/Roles/GetAll"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task AddAsync(RoleAddRequest? body) + { + return AddAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task AddAsync(RoleAddRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Roles/Add" + urlBuilder_.Append("api/Roles/Add"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task UpdateAsync(System.Guid id, RoleUpdateRequest? body) + { + return UpdateAsync(id, body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task UpdateAsync(System.Guid id, RoleUpdateRequest? body, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Roles/Update/{id}" + urlBuilder_.Append("api/Roles/Update/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RemoveAsync(System.Guid id) + { + return RemoveAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("DELETE"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Roles/Remove/{id}" + urlBuilder_.Append("api/Roles/Remove/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetUserRolesAsync(System.Guid? userId) + { + return GetUserRolesAsync(userId, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetUserRolesAsync(System.Guid? userId, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Roles/GetUserRoles" + urlBuilder_.Append("api/Roles/GetUserRoles"); + urlBuilder_.Append('?'); + if (userId != null) + { + urlBuilder_.Append(System.Uri.EscapeDataString("userId")).Append('=').Append(System.Uri.EscapeDataString(ConvertToString(userId, System.Globalization.CultureInfo.InvariantCulture))).Append('&'); + } + urlBuilder_.Length--; + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task UpdateUserRolesAsync(UserRolesUpdateRequest? body) + { + return UpdateUserRolesAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task UpdateUserRolesAsync(UserRolesUpdateRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Roles/UpdateUserRoles" + urlBuilder_.Append("api/Roles/UpdateUserRoles"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface ISitesClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByUrlAsync(string? url); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByUrlAsync(string? url, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(SiteAddRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(SiteAddRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, SiteUpdateRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, SiteUpdateRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class SitesClient : ISitesClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public SitesClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetAllAsync() + { + return GetAllAsync(System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Sites/GetAll" + urlBuilder_.Append("api/Sites/GetAll"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetByIdAsync(System.Guid id) + { + return GetByIdAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Sites/GetById/{id}" + urlBuilder_.Append("api/Sites/GetById/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetByUrlAsync(string? url) + { + return GetByUrlAsync(url, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetByUrlAsync(string? url, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Sites/GetByUrl" + urlBuilder_.Append("api/Sites/GetByUrl"); + urlBuilder_.Append('?'); + if (url != null) + { + urlBuilder_.Append(System.Uri.EscapeDataString("url")).Append('=').Append(System.Uri.EscapeDataString(ConvertToString(url, System.Globalization.CultureInfo.InvariantCulture))).Append('&'); + } + urlBuilder_.Length--; + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task AddAsync(SiteAddRequest? body) + { + return AddAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task AddAsync(SiteAddRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Sites/Add" + urlBuilder_.Append("api/Sites/Add"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task UpdateAsync(System.Guid id, SiteUpdateRequest? body) + { + return UpdateAsync(id, body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task UpdateAsync(System.Guid id, SiteUpdateRequest? body, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Sites/Update/{id}" + urlBuilder_.Append("api/Sites/Update/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RemoveAsync(System.Guid id) + { + return RemoveAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("DELETE"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Sites/Remove/{id}" + urlBuilder_.Append("api/Sites/Remove/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface IThreadsClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task StartAsync(); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task StartAsync(System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class ThreadsClient : IThreadsClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public ThreadsClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task StartAsync() + { + return StartAsync(System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task StartAsync(System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Content = new System.Net.Http.StringContent(string.Empty, System.Text.Encoding.UTF8, "application/json"); + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Threads/Start" + urlBuilder_.Append("api/Threads/Start"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface IThreadsManagementClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task CreateAsync(AIThread? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task CreateAsync(AIThread? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(AIThread? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(AIThread? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class ThreadsManagementClient : IThreadsManagementClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public ThreadsManagementClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetAllAsync() + { + return GetAllAsync(System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/ThreadsManagement/GetAll" + urlBuilder_.Append("api/ThreadsManagement/GetAll"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetByIdAsync(System.Guid id) + { + return GetByIdAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/ThreadsManagement/GetById/{id}" + urlBuilder_.Append("api/ThreadsManagement/GetById/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task CreateAsync(AIThread? body) + { + return CreateAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task CreateAsync(AIThread? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/ThreadsManagement/Create" + urlBuilder_.Append("api/ThreadsManagement/Create"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task UpdateAsync(AIThread? body) + { + return UpdateAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task UpdateAsync(AIThread? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/ThreadsManagement/Update" + urlBuilder_.Append("api/ThreadsManagement/Update"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RemoveAsync(System.Guid id) + { + return RemoveAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("DELETE"); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/ThreadsManagement/Remove/{id}" + urlBuilder_.Append("api/ThreadsManagement/Remove/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + return; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface ITodosClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task CreateAsync(TodoCreateDto? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task CreateAsync(TodoCreateDto? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, TodoUpdateDto? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, TodoUpdateDto? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task DeleteAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task DeleteAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class TodosClient : ITodosClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public TodosClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetAllAsync() + { + return GetAllAsync(System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Todos/GetAll" + urlBuilder_.Append("api/Todos/GetAll"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetByIdAsync(System.Guid id) + { + return GetByIdAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Todos/GetById/{id}" + urlBuilder_.Append("api/Todos/GetById/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task CreateAsync(TodoCreateDto? body) + { + return CreateAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task CreateAsync(TodoCreateDto? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Todos/Create" + urlBuilder_.Append("api/Todos/Create"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task UpdateAsync(System.Guid id, TodoUpdateDto? body) + { + return UpdateAsync(id, body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task UpdateAsync(System.Guid id, TodoUpdateDto? body, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Todos/Update/{id}" + urlBuilder_.Append("api/Todos/Update/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task DeleteAsync(System.Guid id) + { + return DeleteAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task DeleteAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("DELETE"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Todos/Delete/{id}" + urlBuilder_.Append("api/Todos/Delete/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial interface IUsersClient : IApiClient + { + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(UserAddRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task AddAsync(UserAddRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, UserUpdateRequest? body); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task UpdateAsync(System.Guid id, UserUpdateRequest? body, System.Threading.CancellationToken cancellationToken); + + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id); + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken); + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class UsersClient : IUsersClient + { + private System.Net.Http.HttpClient _httpClient; + private static System.Lazy _settings = new System.Lazy(CreateSerializerSettings, true); + private System.Text.Json.JsonSerializerOptions _instanceSettings; + + #pragma warning disable CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + public UsersClient(System.Net.Http.HttpClient httpClient) + #pragma warning restore CS8618 // Non-nullable field must contain a non-null value when exiting constructor. Consider declaring as nullable. + { + _httpClient = httpClient; + Initialize(); + } + + private static System.Text.Json.JsonSerializerOptions CreateSerializerSettings() + { + var settings = new System.Text.Json.JsonSerializerOptions(); + var converters = new System.Text.Json.Serialization.JsonConverter[] { new DictionaryJsonConverter() }; + foreach(var converter in converters) + settings.Converters.Add(converter); + UpdateJsonSerializerSettings(settings); + return settings; + } + + public System.Text.Json.JsonSerializerOptions JsonSerializerSettings { get { return _instanceSettings ?? _settings.Value; } } + + static partial void UpdateJsonSerializerSettings(System.Text.Json.JsonSerializerOptions settings); + + partial void Initialize(); + + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, string url); + partial void PrepareRequest(System.Net.Http.HttpClient client, System.Net.Http.HttpRequestMessage request, System.Text.StringBuilder urlBuilder); + partial void ProcessResponse(System.Net.Http.HttpClient client, System.Net.Http.HttpResponseMessage response); + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetAllAsync() + { + return GetAllAsync(System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetAllAsync(System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Users/GetAll" + urlBuilder_.Append("api/Users/GetAll"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task GetByIdAsync(System.Guid id) + { + return GetByIdAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task GetByIdAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("GET"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Users/GetById/{id}" + urlBuilder_.Append("api/Users/GetById/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task AddAsync(UserAddRequest? body) + { + return AddAsync(body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task AddAsync(UserAddRequest? body, System.Threading.CancellationToken cancellationToken) + { + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("POST"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Users/Add" + urlBuilder_.Append("api/Users/Add"); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task UpdateAsync(System.Guid id, UserUpdateRequest? body) + { + return UpdateAsync(id, body, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task UpdateAsync(System.Guid id, UserUpdateRequest? body, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + var json_ = System.Text.Json.JsonSerializer.SerializeToUtf8Bytes(body, JsonSerializerSettings); + var content_ = new System.Net.Http.ByteArrayContent(json_); + content_.Headers.ContentType = System.Net.Http.Headers.MediaTypeHeaderValue.Parse("application/json"); + request_.Content = content_; + request_.Method = new System.Net.Http.HttpMethod("PUT"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Users/Update/{id}" + urlBuilder_.Append("api/Users/Update/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + /// OK + /// A server side error occurred. + public virtual System.Threading.Tasks.Task RemoveAsync(System.Guid id) + { + return RemoveAsync(id, System.Threading.CancellationToken.None); + } + + /// A cancellation token that can be used by other objects or threads to receive notice of cancellation. + /// OK + /// A server side error occurred. + public virtual async System.Threading.Tasks.Task RemoveAsync(System.Guid id, System.Threading.CancellationToken cancellationToken) + { + if (id == null) + throw new System.ArgumentNullException("id"); + + var client_ = _httpClient; + var disposeClient_ = false; + try + { + using (var request_ = new System.Net.Http.HttpRequestMessage()) + { + request_.Method = new System.Net.Http.HttpMethod("DELETE"); + request_.Headers.Accept.Add(System.Net.Http.Headers.MediaTypeWithQualityHeaderValue.Parse("application/json")); + + var urlBuilder_ = new System.Text.StringBuilder(); + + // Operation Path: "api/Users/Remove/{id}" + urlBuilder_.Append("api/Users/Remove/"); + urlBuilder_.Append(System.Uri.EscapeDataString(ConvertToString(id, System.Globalization.CultureInfo.InvariantCulture))); + + PrepareRequest(client_, request_, urlBuilder_); + + var url_ = urlBuilder_.ToString(); + request_.RequestUri = new System.Uri(url_, System.UriKind.RelativeOrAbsolute); + + PrepareRequest(client_, request_, url_); + + var response_ = await client_.SendAsync(request_, System.Net.Http.HttpCompletionOption.ResponseHeadersRead, cancellationToken).ConfigureAwait(false); + var disposeResponse_ = true; + try + { + var headers_ = new System.Collections.Generic.Dictionary>(); + foreach (var item_ in response_.Headers) + headers_[item_.Key] = item_.Value; + if (response_.Content != null && response_.Content.Headers != null) + { + foreach (var item_ in response_.Content.Headers) + headers_[item_.Key] = item_.Value; + } + + ProcessResponse(client_, response_); + + var status_ = (int)response_.StatusCode; + if (status_ == 200) + { + var objectResponse_ = await ReadObjectResponseAsync(response_, headers_, cancellationToken).ConfigureAwait(false); + if (objectResponse_.Object == null) + { + throw new ApiClientException("Response was null which was not expected.", status_, objectResponse_.Text, headers_, null); + } + return objectResponse_.Object; + } + else + { + var responseData_ = response_.Content == null ? null : await ReadAsStringAsync(response_.Content, cancellationToken).ConfigureAwait(false); + throw new ApiClientException("The HTTP status code of the response was not expected (" + status_ + ").", status_, responseData_, headers_, null); + } + } + finally + { + if (disposeResponse_) + response_.Dispose(); + } + } + } + finally + { + if (disposeClient_) + client_.Dispose(); + } + } + + protected struct ObjectResponseResult + { + public ObjectResponseResult(T responseObject, string responseText) + { + this.Object = responseObject; + this.Text = responseText; + } + + public T Object { get; } + + public string Text { get; } + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStringAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStringAsync(cancellationToken); + #else + return content.ReadAsStringAsync(); + #endif + } + + [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)] + private static System.Threading.Tasks.Task ReadAsStreamAsync(System.Net.Http.HttpContent content, System.Threading.CancellationToken cancellationToken) + { + #if NET5_0_OR_GREATER + return content.ReadAsStreamAsync(cancellationToken); + #else + return content.ReadAsStreamAsync(); + #endif + } + + public bool ReadResponseAsString { get; set; } + + protected virtual async System.Threading.Tasks.Task> ReadObjectResponseAsync(System.Net.Http.HttpResponseMessage response, System.Collections.Generic.IReadOnlyDictionary> headers, System.Threading.CancellationToken cancellationToken) + { + if (response == null || response.Content == null) + { + return new ObjectResponseResult(default(T)!, string.Empty); + } + + if (ReadResponseAsString) + { + var responseText = await ReadAsStringAsync(response.Content, cancellationToken).ConfigureAwait(false); + try + { + var typedBody = System.Text.Json.JsonSerializer.Deserialize(responseText, JsonSerializerSettings); + return new ObjectResponseResult(typedBody!, responseText); + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body string as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, responseText, headers, exception); + } + } + else + { + try + { + using (var responseStream = await ReadAsStreamAsync(response.Content, cancellationToken).ConfigureAwait(false)) + { + var typedBody = await System.Text.Json.JsonSerializer.DeserializeAsync(responseStream, JsonSerializerSettings, cancellationToken).ConfigureAwait(false); + return new ObjectResponseResult(typedBody!, string.Empty); + } + } + catch (System.Text.Json.JsonException exception) + { + var message = "Could not deserialize the response body stream as " + typeof(T).FullName + "."; + throw new ApiClientException(message, (int)response.StatusCode, string.Empty, headers, exception); + } + } + } + + private string ConvertToString(object? value, System.Globalization.CultureInfo cultureInfo) + { + if (value == null) + { + return ""; + } + + if (value is System.Enum) + { + var name = System.Enum.GetName(value.GetType(), value); + if (name != null) + { + var field_ = System.Reflection.IntrospectionExtensions.GetTypeInfo(value.GetType()).GetDeclaredField(name); + if (field_ != null) + { + var attribute = System.Reflection.CustomAttributeExtensions.GetCustomAttribute(field_, typeof(System.Runtime.Serialization.EnumMemberAttribute)) + as System.Runtime.Serialization.EnumMemberAttribute; + if (attribute != null) + { + return attribute.Value != null ? attribute.Value : name; + } + } + + var converted = System.Convert.ToString(System.Convert.ChangeType(value, System.Enum.GetUnderlyingType(value.GetType()), cultureInfo)); + return converted == null ? string.Empty : converted; + } + } + else if (value is bool) + { + return System.Convert.ToString((bool)value, cultureInfo).ToLowerInvariant(); + } + else if (value is byte[]) + { + return System.Convert.ToBase64String((byte[]) value); + } + else if (value is string[]) + { + return string.Join(",", (string[])value); + } + else if (value.GetType().IsArray) + { + var valueArray = (System.Array)value; + var valueTextArray = new string[valueArray.Length]; + for (var i = 0; i < valueArray.Length; i++) + { + valueTextArray[i] = ConvertToString(valueArray.GetValue(i), cultureInfo); + } + return string.Join(",", valueTextArray); + } + + var result = System.Convert.ToString(value, cultureInfo); + return result == null ? "" : result; + } + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AIAnnotation + { + + [System.Text.Json.Serialization.JsonPropertyName("annotatedRegions")] + public System.Collections.Generic.ICollection? AnnotatedRegions { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("additionalProperties")] + public System.Collections.Generic.Dictionary? AdditionalProperties { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AIContent + { + + [System.Text.Json.Serialization.JsonPropertyName("annotations")] + public System.Collections.Generic.ICollection? Annotations { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("additionalProperties")] + public System.Collections.Generic.Dictionary? AdditionalProperties { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AIThread + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("createdAt")] + public System.DateTime CreatedAt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("updatedAt")] + public System.DateTime? UpdatedAt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("createdBy")] + public string? CreatedBy { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("updatedBy")] + public string? UpdatedBy { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("version")] + public int Version { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("model")] + public string? Model { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("systemPrompt")] + public string? SystemPrompt { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AccountChangePasswordRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("userName")] + [System.ComponentModel.DataAnnotations.Required] + public string UserName { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("oldPassword")] + [System.ComponentModel.DataAnnotations.Required] + public string OldPassword { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("newPassword")] + [System.ComponentModel.DataAnnotations.Required] + [System.ComponentModel.DataAnnotations.StringLength(int.MaxValue, MinimumLength = 3)] + public string NewPassword { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("confirmPassword")] + [System.ComponentModel.DataAnnotations.Required] + public string ConfirmPassword { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AccountConfirmEmailRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("userName")] + [System.ComponentModel.DataAnnotations.Required] + public string UserName { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("email")] + [System.ComponentModel.DataAnnotations.Required] + public string Email { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("token")] + [System.ComponentModel.DataAnnotations.Required] + public string Token { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AccountForgotPasswordRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("email")] + [System.ComponentModel.DataAnnotations.Required] + public string Email { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AccountLoginRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("userName")] + [System.ComponentModel.DataAnnotations.Required] + public string UserName { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("password")] + [System.ComponentModel.DataAnnotations.Required] + public string Password { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AccountRegisterRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("email")] + [System.ComponentModel.DataAnnotations.Required] + public string Email { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("userName")] + [System.ComponentModel.DataAnnotations.Required] + [System.ComponentModel.DataAnnotations.StringLength(int.MaxValue, MinimumLength = 3)] + public string UserName { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("password")] + [System.ComponentModel.DataAnnotations.Required] + [System.ComponentModel.DataAnnotations.StringLength(int.MaxValue, MinimumLength = 3)] + public string Password { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("confirmPassword")] + [System.ComponentModel.DataAnnotations.Required] + public string ConfirmPassword { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AccountResendConfirmationRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("email")] + [System.ComponentModel.DataAnnotations.Required] + public string Email { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AccountResetPasswordRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("email")] + [System.ComponentModel.DataAnnotations.Required] + public string Email { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("token")] + [System.ComponentModel.DataAnnotations.Required] + public string Token { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("newPassword")] + [System.ComponentModel.DataAnnotations.Required] + [System.ComponentModel.DataAnnotations.StringLength(int.MaxValue, MinimumLength = 3)] + public string NewPassword { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("confirmPassword")] + [System.ComponentModel.DataAnnotations.Required] + public string ConfirmPassword { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class Agent + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("createdAt")] + public System.DateTime CreatedAt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("updatedAt")] + public System.DateTime? UpdatedAt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("createdBy")] + public string? CreatedBy { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("updatedBy")] + public string? UpdatedBy { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("version")] + public int Version { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("model")] + public string? Model { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("systemPrompt")] + public string? SystemPrompt { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AgentResponseDto + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("model")] + public string? Model { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("systemPrompt")] + public string? SystemPrompt { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AgentResponseDtoApiListResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public System.Collections.Generic.ICollection? Data { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pagination")] + public PaginationInfo Pagination { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AgentResponseDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public AgentResponseDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AgentRunResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("messages")] + public System.Collections.Generic.ICollection? Messages { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("agentId")] + public string? AgentId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("responseId")] + public string? ResponseId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("continuationToken")] + public object? ContinuationToken { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("createdAt")] + public System.DateTime? CreatedAt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("usage")] + public UsageDetails Usage { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("additionalProperties")] + public System.Collections.Generic.Dictionary? AdditionalProperties { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AgentRunResponseApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public AgentRunResponse Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class AnnotatedRegion + { + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class ApiError + { + + [System.Text.Json.Serialization.JsonPropertyName("code")] + public string? Code { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class ApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class BlockAddRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("siteId")] + [System.ComponentModel.DataAnnotations.Required(AllowEmptyStrings = true)] + public System.Guid SiteId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("category")] + public string? Category { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("content")] + public string? Content { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class BlockDto + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("siteId")] + public System.Guid SiteId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("category")] + public string? Category { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("content")] + public string? Content { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class BlockDtoApiListResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public System.Collections.Generic.ICollection? Data { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pagination")] + public PaginationInfo Pagination { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class BlockDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public BlockDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class BlockUpdateRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("category")] + public string? Category { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("content")] + public string? Content { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class BooleanApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public bool Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class ChatMessage + { + + [System.Text.Json.Serialization.JsonPropertyName("authorName")] + public string? AuthorName { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("createdAt")] + public System.DateTime? CreatedAt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("role")] + public ChatRole Role { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("contents")] + public System.Collections.Generic.ICollection? Contents { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageId")] + public string? MessageId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("additionalProperties")] + public System.Collections.Generic.Dictionary? AdditionalProperties { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class ChatRole + { + + [System.Text.Json.Serialization.JsonPropertyName("value")] + public string? Value { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FileDto + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("siteId")] + public System.Guid SiteId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("normalizedName")] + public string? NormalizedName { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("folderId")] + public System.Guid FolderId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("extension")] + public string? Extension { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("contentType")] + public string? ContentType { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("size")] + public long Size { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FileDtoApiListResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public System.Collections.Generic.ICollection? Data { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pagination")] + public PaginationInfo Pagination { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FileDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public FileDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FileMoveRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("folderId")] + public System.Guid FolderId { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FileRenameRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FolderAddRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("siteId")] + public System.Guid SiteId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("parentId")] + public System.Guid ParentId { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FolderDto + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("siteId")] + public System.Guid SiteId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("normalizedName")] + public string? NormalizedName { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("files")] + public System.Collections.Generic.ICollection? Files { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("folders")] + public System.Collections.Generic.ICollection? Folders { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("parentFolder")] + public FolderDto ParentFolder { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FolderDtoApiListResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public System.Collections.Generic.ICollection? Data { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pagination")] + public PaginationInfo Pagination { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FolderDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public FolderDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FolderMoveRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("parentId")] + public System.Guid ParentId { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FolderRenameRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class LayoutAddRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("siteId")] + [System.ComponentModel.DataAnnotations.Required(AllowEmptyStrings = true)] + public System.Guid SiteId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("head")] + public string? Head { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("body")] + public string? Body { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class LayoutDto + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("siteId")] + public System.Guid SiteId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("head")] + public string? Head { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("body")] + public string? Body { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class LayoutDtoApiListResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public System.Collections.Generic.ICollection? Data { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pagination")] + public PaginationInfo Pagination { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class LayoutDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public LayoutDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class LayoutUpdateRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("head")] + public string? Head { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("body")] + public string? Body { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class LoginDto + { + + [System.Text.Json.Serialization.JsonPropertyName("token")] + public string? Token { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class LoginDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public LoginDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class PageAddRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("title")] + [System.ComponentModel.DataAnnotations.Required] + public string Title { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("siteId")] + public System.Guid SiteId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("slug")] + public string? Slug { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("order")] + public int Order { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("parentId")] + public System.Guid? ParentId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("layoutId")] + public System.Guid? LayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("editLayoutId")] + public System.Guid? EditLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("detailLayoutId")] + public System.Guid? DetailLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaTitle")] + public string? MetaTitle { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaDescription")] + public string? MetaDescription { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsIndex")] + public bool RobotsIndex { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsFollow")] + public bool RobotsFollow { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("ogType")] + public string? OgType { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("head")] + public string? Head { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class PageDto + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("siteId")] + public System.Guid SiteId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("title")] + public string? Title { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("slug")] + public string? Slug { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("fullPath")] + public string? FullPath { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("order")] + public int Order { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("parentId")] + public System.Guid ParentId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("layoutId")] + public System.Guid LayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("editLayoutId")] + public System.Guid EditLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("detailLayoutId")] + public System.Guid DetailLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaTitle")] + public string? MetaTitle { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaDescription")] + public string? MetaDescription { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsIndex")] + public bool RobotsIndex { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsFollow")] + public bool RobotsFollow { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("ogType")] + public string? OgType { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("head")] + public string? Head { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class PageDtoApiListResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public System.Collections.Generic.ICollection? Data { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pagination")] + public PaginationInfo Pagination { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class PageDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public PageDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class PageUpdateRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("title")] + [System.ComponentModel.DataAnnotations.Required] + public string Title { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("slug")] + public string? Slug { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("order")] + public int Order { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("parentId")] + public System.Guid? ParentId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("layoutId")] + public System.Guid? LayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("editLayoutId")] + public System.Guid? EditLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("detailLayoutId")] + public System.Guid? DetailLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaTitle")] + public string? MetaTitle { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaDescription")] + public string? MetaDescription { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsIndex")] + public bool RobotsIndex { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsFollow")] + public bool RobotsFollow { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("ogType")] + public string? OgType { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("head")] + public string? Head { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class PaginationInfo + { + + [System.Text.Json.Serialization.JsonPropertyName("page")] + public int Page { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pageSize")] + public int PageSize { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("totalPages")] + public int TotalPages { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("totalCount")] + public int TotalCount { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("hasNextPage")] + public bool HasNextPage { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("hasPreviousPage")] + public bool HasPreviousPage { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class RoleAddRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("name")] + [System.ComponentModel.DataAnnotations.Required] + [System.ComponentModel.DataAnnotations.StringLength(256, MinimumLength = 1)] + public string Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + [System.ComponentModel.DataAnnotations.StringLength(1000)] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("siteId")] + [System.ComponentModel.DataAnnotations.Required(AllowEmptyStrings = true)] + public System.Guid SiteId { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class RoleDto + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("siteId")] + public System.Guid SiteId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("type")] + [System.Text.Json.Serialization.JsonConverter(typeof(System.Text.Json.Serialization.JsonStringEnumConverter))] + public RoleTypes Type { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("createdAt")] + public System.DateTime CreatedAt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("updatedAt")] + public System.DateTime? UpdatedAt { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class RoleDtoApiListResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public System.Collections.Generic.ICollection? Data { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pagination")] + public PaginationInfo Pagination { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class RoleDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public RoleDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public enum RoleTypes + { + + [System.Runtime.Serialization.EnumMember(Value = @"UserDefined")] + UserDefined = 0, + + [System.Runtime.Serialization.EnumMember(Value = @"Administrators")] + Administrators = 1, + + [System.Runtime.Serialization.EnumMember(Value = @"Authenticated")] + Authenticated = 2, + + [System.Runtime.Serialization.EnumMember(Value = @"Guest")] + Guest = 3, + + [System.Runtime.Serialization.EnumMember(Value = @"AllUsers")] + AllUsers = 4, + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class RoleUpdateRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("name")] + [System.ComponentModel.DataAnnotations.Required] + [System.ComponentModel.DataAnnotations.StringLength(256, MinimumLength = 1)] + public string Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + [System.ComponentModel.DataAnnotations.StringLength(1000)] + public string? Description { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class SiteAddRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("name")] + [System.ComponentModel.DataAnnotations.Required] + public string Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("urls")] + [System.ComponentModel.DataAnnotations.Required] + public System.Collections.Generic.ICollection Urls { get; set; } = new System.Collections.ObjectModel.Collection(); + + [System.Text.Json.Serialization.JsonPropertyName("layoutId")] + public System.Guid? LayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("detailLayoutId")] + public System.Guid? DetailLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("editLayoutId")] + public System.Guid? EditLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaTitle")] + public string? MetaTitle { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaDescription")] + public string? MetaDescription { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsIndex")] + public bool? RobotsIndex { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsFollow")] + public bool? RobotsFollow { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsTxt")] + public string? RobotsTxt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("googleTagsId")] + public string? GoogleTagsId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("ogType")] + public string? OgType { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("head")] + public string? Head { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class SiteDto + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("urls")] + public System.Collections.Generic.ICollection? Urls { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("layoutId")] + public System.Guid? LayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("detailLayoutId")] + public System.Guid? DetailLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("editLayoutId")] + public System.Guid? EditLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaTitle")] + public string? MetaTitle { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaDescription")] + public string? MetaDescription { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsIndex")] + public bool? RobotsIndex { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsFollow")] + public bool? RobotsFollow { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsTxt")] + public string? RobotsTxt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("googleTagsId")] + public string? GoogleTagsId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("ogType")] + public string? OgType { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("head")] + public string? Head { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class SiteDtoApiListResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public System.Collections.Generic.ICollection? Data { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pagination")] + public PaginationInfo Pagination { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class SiteDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public SiteDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class SiteUpdateRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("name")] + [System.ComponentModel.DataAnnotations.Required] + public string Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("urls")] + [System.ComponentModel.DataAnnotations.Required] + public System.Collections.Generic.ICollection Urls { get; set; } = new System.Collections.ObjectModel.Collection(); + + [System.Text.Json.Serialization.JsonPropertyName("layoutId")] + public System.Guid? LayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("detailLayoutId")] + public System.Guid? DetailLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("editLayoutId")] + public System.Guid? EditLayoutId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaTitle")] + public string? MetaTitle { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("metaDescription")] + public string? MetaDescription { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsIndex")] + public bool? RobotsIndex { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsFollow")] + public bool? RobotsFollow { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("robotsTxt")] + public string? RobotsTxt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("googleTagsId")] + public string? GoogleTagsId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("ogType")] + public string? OgType { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("head")] + public string? Head { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class ThreadResponseDto + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("name")] + public string? Name { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("model")] + public string? Model { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("systemPrompt")] + public string? SystemPrompt { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class ThreadResponseDtoApiListResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public System.Collections.Generic.ICollection? Data { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pagination")] + public PaginationInfo Pagination { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class ThreadResponseDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public ThreadResponseDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class TodoCreateDto + { + + [System.Text.Json.Serialization.JsonPropertyName("title")] + [System.ComponentModel.DataAnnotations.Required] + public string Title { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("isCompleted")] + public bool IsCompleted { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("dueDate")] + public System.DateTime? DueDate { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class TodoResponseDto + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("title")] + public string? Title { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("isCompleted")] + public bool IsCompleted { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("dueDate")] + public System.DateTime? DueDate { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("createdAt")] + public System.DateTime CreatedAt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("version")] + public int Version { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class TodoResponseDtoApiListResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public System.Collections.Generic.ICollection? Data { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pagination")] + public PaginationInfo Pagination { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class TodoResponseDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public TodoResponseDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class TodoUpdateDto + { + + [System.Text.Json.Serialization.JsonPropertyName("title")] + [System.ComponentModel.DataAnnotations.Required] + public string Title { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("isCompleted")] + public bool IsCompleted { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("dueDate")] + public System.DateTime? DueDate { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class UsageDetails + { + + [System.Text.Json.Serialization.JsonPropertyName("inputTokenCount")] + public long? InputTokenCount { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("outputTokenCount")] + public long? OutputTokenCount { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("totalTokenCount")] + public long? TotalTokenCount { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("additionalCounts")] + public System.Collections.Generic.Dictionary? AdditionalCounts { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class UserAddRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("username")] + [System.ComponentModel.DataAnnotations.Required] + public string Username { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("password")] + [System.ComponentModel.DataAnnotations.Required] + public string Password { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("email")] + [System.ComponentModel.DataAnnotations.Required] + public string Email { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("suspended")] + public bool Suspended { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("isSuperAdmin")] + public bool IsSuperAdmin { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("emailConfirmed")] + public bool EmailConfirmed { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("locked")] + public bool Locked { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class UserDto + { + + [System.Text.Json.Serialization.JsonPropertyName("id")] + public System.Guid Id { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("username")] + public string? Username { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("email")] + public string? Email { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("isSuperAdmin")] + public bool IsSuperAdmin { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("lastLogin")] + public System.DateTime? LastLogin { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("loginCount")] + public int LoginCount { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("passwordChangedAt")] + public System.DateTime? PasswordChangedAt { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("passwordChangedBy")] + public string? PasswordChangedBy { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("emailConfirmed")] + public bool EmailConfirmed { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("phoneNumberConfirmed")] + public bool PhoneNumberConfirmed { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("suspended")] + public bool Suspended { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("lockoutEnd")] + public System.DateTime? LockoutEnd { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("lockoutEnabled")] + public bool LockoutEnabled { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("accessFailedCount")] + public int AccessFailedCount { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class UserDtoApiListResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public System.Collections.Generic.ICollection? Data { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("pagination")] + public PaginationInfo Pagination { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class UserDtoApiResponse + { + + [System.Text.Json.Serialization.JsonPropertyName("success")] + public bool Success { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("message")] + public string? Message { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("messageCode")] + public string? MessageCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("errors")] + public System.Collections.Generic.ICollection? Errors { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("timestamp")] + public System.DateTime Timestamp { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("traceId")] + public string? TraceId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("statusCode")] + public int StatusCode { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("sessionId")] + public string? SessionId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("uniqueId")] + public string? UniqueId { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("duration")] + public double Duration { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("data")] + public UserDto Data { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class UserRolesUpdateRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("roleIds")] + [System.ComponentModel.DataAnnotations.Required] + [System.ComponentModel.DataAnnotations.MinLength(1)] + public System.Collections.Generic.ICollection RoleIds { get; set; } = new System.Collections.ObjectModel.Collection(); + + [System.Text.Json.Serialization.JsonPropertyName("userId")] + [System.ComponentModel.DataAnnotations.Required(AllowEmptyStrings = true)] + public System.Guid UserId { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class UserUpdateRequest + { + + [System.Text.Json.Serialization.JsonPropertyName("email")] + [System.ComponentModel.DataAnnotations.Required] + public string Email { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("suspended")] + public bool Suspended { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("isSuperAdmin")] + public bool IsSuperAdmin { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("emailConfirmed")] + public bool EmailConfirmed { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("locked")] + public bool Locked { get; set; } = default!; + + [System.Text.Json.Serialization.JsonPropertyName("description")] + public string? Description { get; set; } = default!; + + } + + [System.CodeDom.Compiler.GeneratedCode("NSwag", "14.6.3.0 (NJsonSchema v11.5.2.0 (Newtonsoft.Json v13.0.0.0))")] + public partial class FileParameter + { + public FileParameter(System.IO.Stream data) + : this (data, null, null) + { + } + + public FileParameter(System.IO.Stream data, string? fileName) + : this (data, fileName, null) + { + } + + public FileParameter(System.IO.Stream data, string? fileName, string? contentType) + { + Data = data; + FileName = fileName; + ContentType = contentType; + } + + public System.IO.Stream Data { get; private set; } + + public string? FileName { get; private set; } + + public string? ContentType { get; private set; } + } + + +} + +#pragma warning restore 108 +#pragma warning restore 114 +#pragma warning restore 472 +#pragma warning restore 612 +#pragma warning restore 649 +#pragma warning restore 1573 +#pragma warning restore 1591 +#pragma warning restore 8073 +#pragma warning restore 3016 +#pragma warning restore 8600 +#pragma warning restore 8602 +#pragma warning restore 8603 +#pragma warning restore 8604 +#pragma warning restore 8625 +#pragma warning restore 8765 \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.ApiClients/IApiClient.cs b/src/UI/FluentCMS.Web.ApiClients/IApiClient.cs new file mode 100644 index 000000000..826e653cd --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/IApiClient.cs @@ -0,0 +1,6 @@ +namespace FluentCMS.Web.ApiClients; + +// Marker interface for all API clients +public interface IApiClient +{ +} diff --git a/src/UI/FluentCMS.Web.ApiClients/IContent.cs b/src/UI/FluentCMS.Web.ApiClients/IContent.cs new file mode 100644 index 000000000..194861a6f --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/IContent.cs @@ -0,0 +1,6 @@ +// namespace FluentCMS.Web.ApiClients; + +// public interface IContent +// { +// Guid Id { get; set; } +// } diff --git a/src/UI/FluentCMS.Web.ApiClients/ObjectExtensions.cs b/src/UI/FluentCMS.Web.ApiClients/ObjectExtensions.cs new file mode 100644 index 000000000..eb264251f --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/ObjectExtensions.cs @@ -0,0 +1,57 @@ +// using System.Reflection; + +// namespace FluentCMS.Web.ApiClients; + +// public static class ObjectExtensions +// { +// public static List ToContentList(this ICollection items) where T : IContent, new() +// { +// var result = new List(); +// foreach (var item in items) +// { +// if (item.Data != null) +// { +// var model = item.Data.ToContent(); +// model.Id = item.Id; +// result.Add(model); +// } + +// } +// return result; +// } + +// public static T ToContent(this Dictionary source) where T : IContent, new() +// { +// var someObject = new T(); +// var someObjectType = someObject.GetType(); + +// if (someObjectType != null) +// { +// foreach (var item in source) +// { +// someObjectType.GetProperty(item.Key)?.SetValue(someObject, item.Value, null); +// } +// } +// return someObject; +// } + +// private const BindingFlags _bindingAttr = BindingFlags.Instance | BindingFlags.Public | BindingFlags.GetProperty; + +// public static Dictionary ToDictionary(this IContent source) +// { +// var result = new Dictionary(); +// foreach (var prop in source.GetType().GetProperties(_bindingAttr)) +// { +// var propValue = prop.GetValue(source, null); +// if (propValue != null) +// { +// result.Add(prop.Name, propValue); +// } +// } + +// result.Remove("Id"); +// result.Remove("id"); + +// return result; +// } +// } diff --git a/src/UI/FluentCMS.Web.ApiClients/README.md b/src/UI/FluentCMS.Web.ApiClients/README.md new file mode 100644 index 000000000..22d11d039 --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/README.md @@ -0,0 +1 @@ +# FluentCMS Web ApiClients \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.ApiClients/Services/AuthManager.cs b/src/UI/FluentCMS.Web.ApiClients/Services/AuthManager.cs new file mode 100644 index 000000000..1147aa7c3 --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/Services/AuthManager.cs @@ -0,0 +1,47 @@ +// using Microsoft.AspNetCore.Authentication; +// using Microsoft.AspNetCore.Authentication.Cookies; +// using Microsoft.AspNetCore.Http; +// using System.Security.Claims; + +// namespace FluentCMS.Web.ApiClients.Services; + +// public class AuthManager(ApiClientFactory apiClient) +// { +// public async Task Logout(HttpContext httpContext) +// { +// ArgumentNullException.ThrowIfNull(httpContext); + +// await httpContext.SignOutAsync(CookieAuthenticationDefaults.AuthenticationScheme); +// } + +// public async Task Login(HttpContext httpContext, string username, string password, bool isPersist) +// { +// ArgumentNullException.ThrowIfNull(httpContext); + +// var accountResponse = await apiClient.Account.AuthenticateAsync(new UserLoginRequest +// { +// Username = username, +// Password = password, +// }); + +// var account = accountResponse.Data ?? throw new InvalidOperationException("Something bad happened!"); + +// var identityClaims = new ClaimsIdentity(CookieAuthenticationDefaults.AuthenticationScheme); + +// identityClaims.AddClaim(new Claim(ClaimTypes.Sid, account.UserId.ToString())); +// identityClaims.AddClaim(new Claim(ClaimTypes.NameIdentifier, account.UserName ?? string.Empty)); +// identityClaims.AddClaim(new Claim(ClaimTypes.Email, account.Email ?? string.Empty)); +// identityClaims.AddClaim(new Claim("jwt", account.Token ?? string.Empty)); + +// var cookieClaims = new ClaimsPrincipal(identityClaims); + +// await httpContext.SignInAsync(CookieAuthenticationDefaults.AuthenticationScheme, +// cookieClaims, +// new AuthenticationProperties +// { +// IsPersistent = true, +// IssuedUtc = DateTimeOffset.UtcNow, +// ExpiresUtc = DateTimeOffset.UtcNow.AddDays(10) // 100 days +// }); +// } +// } diff --git a/src/UI/FluentCMS.Web.ApiClients/config.nswag b/src/UI/FluentCMS.Web.ApiClients/config.nswag new file mode 100644 index 000000000..2257e249f --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/config.nswag @@ -0,0 +1,103 @@ +{ + "runtime": "Net100", + "defaultVariables": "", + "documentGenerator": { + "fromDocument": { + "json": "", + "url": "http://localhost:5093/swagger/v1/swagger.json", + "output": "./swagger.json", + "newLineBehavior": "Auto" + } + }, + "codeGenerators": { + "openApiToCSharpClient": { + "clientBaseClass": null, + "configurationClass": null, + "generateClientClasses": true, + "suppressClientClassesOutput": false, + "generateClientInterfaces": true, + "suppressClientInterfacesOutput": false, + "clientBaseInterface": "IApiClient", + "injectHttpClient": true, + "disposeHttpClient": true, + "protectedMethods": [], + "generateExceptionClasses": false, + "exceptionClass": "ApiClientException", + "wrapDtoExceptions": true, + "useHttpClientCreationMethod": false, + "httpClientType": "System.Net.Http.HttpClient", + "useHttpRequestMessageCreationMethod": false, + "useBaseUrl": false, + "generateBaseUrlProperty": true, + "generateSyncMethods": false, + "generatePrepareRequestAndProcessResponseAsAsyncMethods": false, + "exposeJsonSerializerSettings": true, + "clientClassAccessModifier": "public", + "typeAccessModifier": "public", + "propertySetterAccessModifier": "", + "generateNativeRecords": false, + "generateContractsOutput": false, + "contractsNamespace": null, + "contractsOutputFilePath": null, + "parameterDateTimeFormat": "s", + "parameterDateFormat": "yyyy-MM-dd", + "generateUpdateJsonSerializerSettingsMethod": true, + "useRequestAndResponseSerializationSettings": false, + "serializeTypeInformation": false, + "queryNullValue": "", + "className": "{controller}Client", + "operationGenerationMode": "MultipleClientsFromPathSegments", + "additionalNamespaceUsages": [], + "additionalContractNamespaceUsages": [], + "generateOptionalParameters": false, + "generateJsonMethods": false, + "enforceFlagEnums": false, + "parameterArrayType": "System.Collections.Generic.IEnumerable", + "parameterDictionaryType": "System.Collections.Generic.Dictionary", + "responseArrayType": "System.Collections.Generic.ICollection", + "responseDictionaryType": "System.Collections.Generic.Dictionary", + "wrapResponses": false, + "wrapResponseMethods": [], + "generateResponseClasses": true, + "responseClass": "SwaggerResponse", + "namespace": "FluentCMS.Web.ApiClients", + "requiredPropertiesMustBeDefined": true, + "dateType": "System.DateTime", + "jsonConverters": [ + "DictionaryJsonConverter" + ], + "anyType": "object", + "dateTimeType": "System.DateTime", + "timeType": "System.TimeSpan", + "timeSpanType": "System.TimeSpan", + "arrayType": "System.Collections.Generic.ICollection", + "arrayInstanceType": "System.Collections.ObjectModel.Collection", + "dictionaryType": "System.Collections.Generic.Dictionary", + "dictionaryInstanceType": "System.Collections.Generic.Dictionary", + "arrayBaseType": "System.Collections.ObjectModel.Collection", + "dictionaryBaseType": "System.Collections.Generic.Dictionary", + "classStyle": "Poco", + "jsonLibrary": "SystemTextJson", + "generateDefaultValues": true, + "generateDataAnnotations": true, + "excludedTypeNames": [], + "excludedParameterNames": [], + "handleReferences": false, + "generateImmutableArrayProperties": false, + "generateImmutableDictionaryProperties": false, + "jsonSerializerSettingsTransformationMethod": "", + "inlineNamedArrays": false, + "inlineNamedDictionaries": false, + "inlineNamedTuples": true, + "inlineNamedAny": false, + "generateDtoTypes": true, + "generateOptionalPropertiesAsNullable": false, + "generateNullableReferenceTypes": true, + "templateDirectory": null, + "serviceHost": null, + "serviceSchemes": null, + "output": "./GeneratedApiClients.cs", + "newLineBehavior": "Auto" + } + } +} diff --git a/src/UI/FluentCMS.Web.ApiClients/swagger.json b/src/UI/FluentCMS.Web.ApiClients/swagger.json new file mode 100644 index 000000000..08ec2f75a --- /dev/null +++ b/src/UI/FluentCMS.Web.ApiClients/swagger.json @@ -0,0 +1,5382 @@ +{ + "openapi": "3.0.4", + "info": { + "title": "FluentCMS API", + "version": "v1.0.0" + }, + "paths": { + "/api/Accounts/Register": { + "post": { + "tags": [ + "Accounts" + ], + "operationId": "Register", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AccountRegisterRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/AccountRegisterRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/AccountRegisterRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Accounts/Login": { + "post": { + "tags": [ + "Accounts" + ], + "operationId": "Login", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AccountLoginRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/AccountLoginRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/AccountLoginRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LoginDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Accounts/Logout": { + "get": { + "tags": [ + "Accounts" + ], + "operationId": "Logout", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Accounts/ConfirmEmail": { + "post": { + "tags": [ + "Accounts" + ], + "operationId": "ConfirmEmail", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AccountConfirmEmailRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/AccountConfirmEmailRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/AccountConfirmEmailRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Accounts/ResendConfirmation": { + "post": { + "tags": [ + "Accounts" + ], + "operationId": "ResendConfirmation", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AccountResendConfirmationRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/AccountResendConfirmationRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/AccountResendConfirmationRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Accounts/ForgotPassword": { + "post": { + "tags": [ + "Accounts" + ], + "operationId": "ForgotPassword", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AccountForgotPasswordRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/AccountForgotPasswordRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/AccountForgotPasswordRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Accounts/ResetPassword": { + "post": { + "tags": [ + "Accounts" + ], + "operationId": "ResetPassword", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AccountResetPasswordRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/AccountResetPasswordRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/AccountResetPasswordRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Accounts/ChangePassword": { + "post": { + "tags": [ + "Accounts" + ], + "operationId": "ChangePassword", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AccountChangePasswordRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/AccountChangePasswordRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/AccountChangePasswordRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Agents/GetAll": { + "get": { + "tags": [ + "Agents" + ], + "operationId": "GetAll", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AgentResponseDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Agents/GetById/{id}": { + "get": { + "tags": [ + "Agents" + ], + "operationId": "GetById", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AgentResponseDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Agents/Create": { + "post": { + "tags": [ + "Agents" + ], + "operationId": "Create", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Agent" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/Agent" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/Agent" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Agent" + } + } + } + } + } + } + }, + "/api/Agents/Update": { + "put": { + "tags": [ + "Agents" + ], + "operationId": "Update", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Agent" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/Agent" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/Agent" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/Agent" + } + } + } + } + } + } + }, + "/api/Agents/Remove/{id}": { + "delete": { + "tags": [ + "Agents" + ], + "operationId": "Remove", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK" + } + } + } + }, + "/api/Blocks/GetBySiteId": { + "get": { + "tags": [ + "Blocks" + ], + "operationId": "GetBySiteId", + "parameters": [ + { + "name": "siteId", + "in": "query", + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BlockDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Blocks/GetAll": { + "get": { + "tags": [ + "Blocks" + ], + "operationId": "GetAll2", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BlockDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Blocks/GetById/{id}": { + "get": { + "tags": [ + "Blocks" + ], + "operationId": "GetById2", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BlockDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Blocks/Add": { + "post": { + "tags": [ + "Blocks" + ], + "operationId": "Add", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BlockAddRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/BlockAddRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/BlockAddRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BlockDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Blocks/Update/{id}": { + "put": { + "tags": [ + "Blocks" + ], + "operationId": "Update2", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BlockUpdateRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/BlockUpdateRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/BlockUpdateRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BlockDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Blocks/Remove/{id}": { + "delete": { + "tags": [ + "Blocks" + ], + "operationId": "Remove2", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Files/Upload": { + "post": { + "tags": [ + "Files" + ], + "operationId": "Upload", + "parameters": [ + { + "name": "folderId", + "in": "query", + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "requestBody": { + "content": { + "multipart/form-data": { + "schema": { + "type": "object", + "properties": { + "files": { + "type": "array", + "items": { + "type": "string", + "format": "binary" + } + } + } + }, + "encoding": { + "files": { + "style": "form" + } + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FileDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Files/Remove/{id}": { + "delete": { + "tags": [ + "Files" + ], + "operationId": "Remove3", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BooleanApiResponse" + } + } + } + } + } + } + }, + "/api/Files/Rename": { + "put": { + "tags": [ + "Files" + ], + "operationId": "Rename", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FileRenameRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/FileRenameRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/FileRenameRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FileDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Files/Move": { + "put": { + "tags": [ + "Files" + ], + "operationId": "Move", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FileMoveRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/FileMoveRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/FileMoveRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FileDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Folders/GetRoot": { + "get": { + "tags": [ + "Folders" + ], + "operationId": "GetRoot", + "parameters": [ + { + "name": "siteId", + "in": "query", + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FolderDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Folders/GetById/{id}": { + "get": { + "tags": [ + "Folders" + ], + "operationId": "GetById3", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FolderDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Folders/Create": { + "post": { + "tags": [ + "Folders" + ], + "operationId": "Create2", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FolderAddRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/FolderAddRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/FolderAddRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FolderDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Folders/Rename": { + "put": { + "tags": [ + "Folders" + ], + "operationId": "Rename2", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FolderRenameRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/FolderRenameRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/FolderRenameRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FolderDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Folders/Move": { + "put": { + "tags": [ + "Folders" + ], + "operationId": "Move2", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FolderMoveRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/FolderMoveRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/FolderMoveRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FolderDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Folders/GetParentFolders": { + "get": { + "tags": [ + "Folders" + ], + "operationId": "GetParentFolders", + "parameters": [ + { + "name": "folderId", + "in": "query", + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/FolderDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Folders/Remove/{id}": { + "delete": { + "tags": [ + "Folders" + ], + "operationId": "Remove4", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/BooleanApiResponse" + } + } + } + } + } + } + }, + "/api/Layouts/GetBySiteId": { + "get": { + "tags": [ + "Layouts" + ], + "operationId": "GetBySiteId2", + "parameters": [ + { + "name": "siteId", + "in": "query", + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LayoutDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Layouts/GetAll": { + "get": { + "tags": [ + "Layouts" + ], + "operationId": "GetAll3", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LayoutDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Layouts/GetById/{id}": { + "get": { + "tags": [ + "Layouts" + ], + "operationId": "GetById4", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LayoutDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Layouts/Add": { + "post": { + "tags": [ + "Layouts" + ], + "operationId": "Add2", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LayoutAddRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/LayoutAddRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/LayoutAddRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LayoutDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Layouts/Update/{id}": { + "put": { + "tags": [ + "Layouts" + ], + "operationId": "Update3", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LayoutUpdateRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/LayoutUpdateRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/LayoutUpdateRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/LayoutDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Layouts/Remove/{id}": { + "delete": { + "tags": [ + "Layouts" + ], + "operationId": "Remove5", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Pages/GetBySiteId": { + "get": { + "tags": [ + "Pages" + ], + "operationId": "GetBySiteId3", + "parameters": [ + { + "name": "siteId", + "in": "query", + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/PageDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Pages/GetAll": { + "get": { + "tags": [ + "Pages" + ], + "operationId": "GetAll4", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/PageDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Pages/GetById/{id}": { + "get": { + "tags": [ + "Pages" + ], + "operationId": "GetById5", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/PageDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Pages/Add": { + "post": { + "tags": [ + "Pages" + ], + "operationId": "Add3", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/PageAddRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/PageAddRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/PageAddRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/PageDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Pages/Update/{id}": { + "put": { + "tags": [ + "Pages" + ], + "operationId": "Update4", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/PageUpdateRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/PageUpdateRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/PageUpdateRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/PageDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Pages/Remove/{id}": { + "delete": { + "tags": [ + "Pages" + ], + "operationId": "Remove6", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Roles/GetById/{id}": { + "get": { + "tags": [ + "Roles" + ], + "operationId": "GetById6", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/RoleDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Roles/GetAll": { + "get": { + "tags": [ + "Roles" + ], + "operationId": "GetAll5", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/RoleDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Roles/Add": { + "post": { + "tags": [ + "Roles" + ], + "operationId": "Add4", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/RoleAddRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/RoleAddRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/RoleAddRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/RoleDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Roles/Update/{id}": { + "put": { + "tags": [ + "Roles" + ], + "operationId": "Update5", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/RoleUpdateRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/RoleUpdateRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/RoleUpdateRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/RoleDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Roles/Remove/{id}": { + "delete": { + "tags": [ + "Roles" + ], + "operationId": "Remove7", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Roles/GetUserRoles": { + "get": { + "tags": [ + "Roles" + ], + "operationId": "GetUserRoles", + "parameters": [ + { + "name": "userId", + "in": "query", + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/RoleDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Roles/UpdateUserRoles": { + "post": { + "tags": [ + "Roles" + ], + "operationId": "UpdateUserRoles", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/UserRolesUpdateRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/UserRolesUpdateRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/UserRolesUpdateRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Sites/GetAll": { + "get": { + "tags": [ + "Sites" + ], + "operationId": "GetAll6", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/SiteDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Sites/GetById/{id}": { + "get": { + "tags": [ + "Sites" + ], + "operationId": "GetById7", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/SiteDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Sites/GetByUrl": { + "get": { + "tags": [ + "Sites" + ], + "operationId": "GetByUrl", + "parameters": [ + { + "name": "url", + "in": "query", + "schema": { + "type": "string" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/SiteDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Sites/Add": { + "post": { + "tags": [ + "Sites" + ], + "operationId": "Add5", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/SiteAddRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/SiteAddRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/SiteAddRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/SiteDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Sites/Update/{id}": { + "put": { + "tags": [ + "Sites" + ], + "operationId": "Update6", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/SiteUpdateRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/SiteUpdateRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/SiteUpdateRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/SiteDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Sites/Remove/{id}": { + "delete": { + "tags": [ + "Sites" + ], + "operationId": "Remove8", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Threads/Start": { + "post": { + "tags": [ + "Threads" + ], + "operationId": "Start", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AgentRunResponseApiResponse" + } + } + } + } + } + } + }, + "/api/ThreadsManagement/GetAll": { + "get": { + "tags": [ + "ThreadsManagement" + ], + "operationId": "GetAll7", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ThreadResponseDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/ThreadsManagement/GetById/{id}": { + "get": { + "tags": [ + "ThreadsManagement" + ], + "operationId": "GetById8", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ThreadResponseDtoApiResponse" + } + } + } + } + } + } + }, + "/api/ThreadsManagement/Create": { + "post": { + "tags": [ + "ThreadsManagement" + ], + "operationId": "Create3", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AIThread" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/AIThread" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/AIThread" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AIThread" + } + } + } + } + } + } + }, + "/api/ThreadsManagement/Update": { + "put": { + "tags": [ + "ThreadsManagement" + ], + "operationId": "Update7", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AIThread" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/AIThread" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/AIThread" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/AIThread" + } + } + } + } + } + } + }, + "/api/ThreadsManagement/Remove/{id}": { + "delete": { + "tags": [ + "ThreadsManagement" + ], + "operationId": "Remove9", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK" + } + } + } + }, + "/api/Todos/GetAll": { + "get": { + "tags": [ + "Todos" + ], + "operationId": "GetAll8", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/TodoResponseDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Todos/GetById/{id}": { + "get": { + "tags": [ + "Todos" + ], + "operationId": "GetById9", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/TodoResponseDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Todos/Create": { + "post": { + "tags": [ + "Todos" + ], + "operationId": "Create4", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/TodoCreateDto" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/TodoCreateDto" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/TodoCreateDto" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/TodoResponseDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Todos/Update/{id}": { + "put": { + "tags": [ + "Todos" + ], + "operationId": "Update8", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/TodoUpdateDto" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/TodoUpdateDto" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/TodoUpdateDto" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/TodoResponseDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Todos/Delete/{id}": { + "delete": { + "tags": [ + "Todos" + ], + "operationId": "Delete", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + }, + "/api/Users/GetAll": { + "get": { + "tags": [ + "Users" + ], + "operationId": "GetAll9", + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/UserDtoApiListResponse" + } + } + } + } + } + } + }, + "/api/Users/GetById/{id}": { + "get": { + "tags": [ + "Users" + ], + "operationId": "GetById10", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/UserDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Users/Add": { + "post": { + "tags": [ + "Users" + ], + "operationId": "Add6", + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/UserAddRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/UserAddRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/UserAddRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/UserDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Users/Update/{id}": { + "put": { + "tags": [ + "Users" + ], + "operationId": "Update9", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "requestBody": { + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/UserUpdateRequest" + } + }, + "text/json": { + "schema": { + "$ref": "#/components/schemas/UserUpdateRequest" + } + }, + "application/*+json": { + "schema": { + "$ref": "#/components/schemas/UserUpdateRequest" + } + } + } + }, + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/UserDtoApiResponse" + } + } + } + } + } + } + }, + "/api/Users/Remove/{id}": { + "delete": { + "tags": [ + "Users" + ], + "operationId": "Remove10", + "parameters": [ + { + "name": "id", + "in": "path", + "required": true, + "schema": { + "type": "string", + "format": "uuid" + } + } + ], + "responses": { + "200": { + "description": "OK", + "content": { + "application/json": { + "schema": { + "$ref": "#/components/schemas/ApiResponse" + } + } + } + } + } + } + } + }, + "components": { + "schemas": { + "AIAnnotation": { + "type": "object", + "additionalProperties": false, + "properties": { + "annotatedRegions": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/AnnotatedRegion" + } + }, + "additionalProperties": { + "type": "object", + "nullable": true, + "additionalProperties": {} + } + } + }, + "AIContent": { + "type": "object", + "additionalProperties": false, + "properties": { + "annotations": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/AIAnnotation" + } + }, + "additionalProperties": { + "type": "object", + "nullable": true, + "additionalProperties": {} + } + } + }, + "AIThread": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "createdAt": { + "type": "string", + "format": "date-time" + }, + "updatedAt": { + "type": "string", + "format": "date-time", + "nullable": true + }, + "createdBy": { + "type": "string", + "nullable": true + }, + "updatedBy": { + "type": "string", + "nullable": true + }, + "version": { + "type": "integer", + "format": "int32" + }, + "name": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + }, + "model": { + "type": "string", + "nullable": true + }, + "systemPrompt": { + "type": "string", + "nullable": true + } + } + }, + "AccountChangePasswordRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "confirmPassword", + "newPassword", + "oldPassword", + "userName" + ], + "properties": { + "userName": { + "type": "string", + "minLength": 1 + }, + "oldPassword": { + "type": "string", + "minLength": 1 + }, + "newPassword": { + "type": "string", + "minLength": 3 + }, + "confirmPassword": { + "type": "string", + "minLength": 1 + } + } + }, + "AccountConfirmEmailRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "email", + "token", + "userName" + ], + "properties": { + "userName": { + "type": "string", + "minLength": 1 + }, + "email": { + "type": "string", + "format": "email", + "minLength": 1 + }, + "token": { + "type": "string", + "minLength": 1 + } + } + }, + "AccountForgotPasswordRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "email" + ], + "properties": { + "email": { + "type": "string", + "format": "email", + "minLength": 1 + } + } + }, + "AccountLoginRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "password", + "userName" + ], + "properties": { + "userName": { + "type": "string", + "minLength": 1 + }, + "password": { + "type": "string", + "minLength": 1 + } + } + }, + "AccountRegisterRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "confirmPassword", + "email", + "password", + "userName" + ], + "properties": { + "email": { + "type": "string", + "format": "email", + "minLength": 1 + }, + "userName": { + "type": "string", + "minLength": 3 + }, + "password": { + "type": "string", + "minLength": 3 + }, + "confirmPassword": { + "type": "string", + "minLength": 1 + } + } + }, + "AccountResendConfirmationRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "email" + ], + "properties": { + "email": { + "type": "string", + "format": "email", + "minLength": 1 + } + } + }, + "AccountResetPasswordRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "confirmPassword", + "email", + "newPassword", + "token" + ], + "properties": { + "email": { + "type": "string", + "format": "email", + "minLength": 1 + }, + "token": { + "type": "string", + "minLength": 1 + }, + "newPassword": { + "type": "string", + "minLength": 3 + }, + "confirmPassword": { + "type": "string", + "minLength": 1 + } + } + }, + "Agent": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "createdAt": { + "type": "string", + "format": "date-time" + }, + "updatedAt": { + "type": "string", + "format": "date-time", + "nullable": true + }, + "createdBy": { + "type": "string", + "nullable": true + }, + "updatedBy": { + "type": "string", + "nullable": true + }, + "version": { + "type": "integer", + "format": "int32" + }, + "name": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + }, + "model": { + "type": "string", + "nullable": true + }, + "systemPrompt": { + "type": "string", + "nullable": true + } + } + }, + "AgentResponseDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "name": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + }, + "model": { + "type": "string", + "nullable": true + }, + "systemPrompt": { + "type": "string", + "nullable": true + } + } + }, + "AgentResponseDtoApiListResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/AgentResponseDto" + } + }, + "pagination": { + "$ref": "#/components/schemas/PaginationInfo" + } + } + }, + "AgentResponseDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/AgentResponseDto" + } + } + }, + "AgentRunResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "messages": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ChatMessage" + } + }, + "agentId": { + "type": "string", + "nullable": true + }, + "responseId": { + "type": "string", + "nullable": true + }, + "continuationToken": { + "nullable": true + }, + "createdAt": { + "type": "string", + "format": "date-time", + "nullable": true + }, + "usage": { + "$ref": "#/components/schemas/UsageDetails" + }, + "additionalProperties": { + "type": "object", + "nullable": true, + "additionalProperties": {} + } + } + }, + "AgentRunResponseApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/AgentRunResponse" + } + } + }, + "AnnotatedRegion": { + "type": "object", + "additionalProperties": false + }, + "ApiError": { + "type": "object", + "additionalProperties": false, + "properties": { + "code": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + } + } + }, + "ApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + } + } + }, + "BlockAddRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "name", + "siteId" + ], + "properties": { + "name": { + "type": "string", + "nullable": true + }, + "siteId": { + "type": "string", + "format": "uuid" + }, + "category": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + }, + "content": { + "type": "string", + "nullable": true + } + } + }, + "BlockDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "siteId": { + "type": "string", + "format": "uuid" + }, + "name": { + "type": "string", + "nullable": true + }, + "category": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + }, + "content": { + "type": "string", + "nullable": true + } + } + }, + "BlockDtoApiListResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/BlockDto" + } + }, + "pagination": { + "$ref": "#/components/schemas/PaginationInfo" + } + } + }, + "BlockDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/BlockDto" + } + } + }, + "BlockUpdateRequest": { + "type": "object", + "additionalProperties": false, + "properties": { + "name": { + "type": "string", + "nullable": true + }, + "category": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + }, + "content": { + "type": "string", + "nullable": true + } + } + }, + "BooleanApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "boolean" + } + } + }, + "ChatMessage": { + "type": "object", + "additionalProperties": false, + "properties": { + "authorName": { + "type": "string", + "nullable": true + }, + "createdAt": { + "type": "string", + "format": "date-time", + "nullable": true + }, + "role": { + "$ref": "#/components/schemas/ChatRole" + }, + "contents": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/AIContent" + } + }, + "messageId": { + "type": "string", + "nullable": true + }, + "additionalProperties": { + "type": "object", + "nullable": true, + "additionalProperties": {} + } + } + }, + "ChatRole": { + "type": "object", + "additionalProperties": false, + "properties": { + "value": { + "type": "string", + "nullable": true + } + } + }, + "FileDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "siteId": { + "type": "string", + "format": "uuid" + }, + "name": { + "type": "string", + "nullable": true + }, + "normalizedName": { + "type": "string", + "nullable": true + }, + "folderId": { + "type": "string", + "format": "uuid" + }, + "extension": { + "type": "string", + "nullable": true + }, + "contentType": { + "type": "string", + "nullable": true + }, + "size": { + "type": "integer", + "format": "int64" + } + } + }, + "FileDtoApiListResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/FileDto" + } + }, + "pagination": { + "$ref": "#/components/schemas/PaginationInfo" + } + } + }, + "FileDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/FileDto" + } + } + }, + "FileMoveRequest": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "folderId": { + "type": "string", + "format": "uuid" + } + } + }, + "FileRenameRequest": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "name": { + "type": "string", + "nullable": true + } + } + }, + "FolderAddRequest": { + "type": "object", + "additionalProperties": false, + "properties": { + "name": { + "type": "string", + "nullable": true + }, + "siteId": { + "type": "string", + "format": "uuid" + }, + "parentId": { + "type": "string", + "format": "uuid" + } + } + }, + "FolderDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "siteId": { + "type": "string", + "format": "uuid" + }, + "name": { + "type": "string", + "nullable": true + }, + "normalizedName": { + "type": "string", + "nullable": true + }, + "files": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/FileDto" + } + }, + "folders": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/FolderDto" + } + }, + "parentFolder": { + "$ref": "#/components/schemas/FolderDto" + } + } + }, + "FolderDtoApiListResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/FolderDto" + } + }, + "pagination": { + "$ref": "#/components/schemas/PaginationInfo" + } + } + }, + "FolderDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/FolderDto" + } + } + }, + "FolderMoveRequest": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "parentId": { + "type": "string", + "format": "uuid" + } + } + }, + "FolderRenameRequest": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "name": { + "type": "string", + "nullable": true + } + } + }, + "LayoutAddRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "name", + "siteId" + ], + "properties": { + "name": { + "type": "string", + "nullable": true + }, + "siteId": { + "type": "string", + "format": "uuid" + }, + "head": { + "type": "string", + "nullable": true + }, + "body": { + "type": "string", + "nullable": true + } + } + }, + "LayoutDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "siteId": { + "type": "string", + "format": "uuid" + }, + "name": { + "type": "string", + "nullable": true + }, + "head": { + "type": "string", + "nullable": true + }, + "body": { + "type": "string", + "nullable": true + } + } + }, + "LayoutDtoApiListResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/LayoutDto" + } + }, + "pagination": { + "$ref": "#/components/schemas/PaginationInfo" + } + } + }, + "LayoutDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/LayoutDto" + } + } + }, + "LayoutUpdateRequest": { + "type": "object", + "additionalProperties": false, + "properties": { + "name": { + "type": "string", + "nullable": true + }, + "head": { + "type": "string", + "nullable": true + }, + "body": { + "type": "string", + "nullable": true + } + } + }, + "LoginDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "token": { + "type": "string", + "nullable": true + } + } + }, + "LoginDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/LoginDto" + } + } + }, + "PageAddRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "title" + ], + "properties": { + "title": { + "type": "string", + "minLength": 1 + }, + "siteId": { + "type": "string", + "format": "uuid" + }, + "slug": { + "type": "string", + "nullable": true + }, + "order": { + "type": "integer", + "format": "int32" + }, + "parentId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "layoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "editLayoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "detailLayoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "metaTitle": { + "type": "string", + "nullable": true + }, + "metaDescription": { + "type": "string", + "nullable": true + }, + "robotsIndex": { + "type": "boolean" + }, + "robotsFollow": { + "type": "boolean" + }, + "ogType": { + "type": "string", + "nullable": true + }, + "head": { + "type": "string", + "nullable": true + } + } + }, + "PageDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "siteId": { + "type": "string", + "format": "uuid" + }, + "title": { + "type": "string", + "nullable": true + }, + "slug": { + "type": "string", + "nullable": true + }, + "fullPath": { + "type": "string", + "nullable": true + }, + "order": { + "type": "integer", + "format": "int32" + }, + "parentId": { + "type": "string", + "format": "uuid" + }, + "layoutId": { + "type": "string", + "format": "uuid" + }, + "editLayoutId": { + "type": "string", + "format": "uuid" + }, + "detailLayoutId": { + "type": "string", + "format": "uuid" + }, + "metaTitle": { + "type": "string", + "nullable": true + }, + "metaDescription": { + "type": "string", + "nullable": true + }, + "robotsIndex": { + "type": "boolean" + }, + "robotsFollow": { + "type": "boolean" + }, + "ogType": { + "type": "string", + "nullable": true + }, + "head": { + "type": "string", + "nullable": true + } + } + }, + "PageDtoApiListResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/PageDto" + } + }, + "pagination": { + "$ref": "#/components/schemas/PaginationInfo" + } + } + }, + "PageDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/PageDto" + } + } + }, + "PageUpdateRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "title" + ], + "properties": { + "title": { + "type": "string", + "minLength": 1 + }, + "slug": { + "type": "string", + "nullable": true + }, + "order": { + "type": "integer", + "format": "int32" + }, + "parentId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "layoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "editLayoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "detailLayoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "metaTitle": { + "type": "string", + "nullable": true + }, + "metaDescription": { + "type": "string", + "nullable": true + }, + "robotsIndex": { + "type": "boolean" + }, + "robotsFollow": { + "type": "boolean" + }, + "ogType": { + "type": "string", + "nullable": true + }, + "head": { + "type": "string", + "nullable": true + } + } + }, + "PaginationInfo": { + "type": "object", + "additionalProperties": false, + "properties": { + "page": { + "type": "integer", + "format": "int32" + }, + "pageSize": { + "type": "integer", + "format": "int32" + }, + "totalPages": { + "type": "integer", + "format": "int32" + }, + "totalCount": { + "type": "integer", + "format": "int32" + }, + "hasNextPage": { + "type": "boolean" + }, + "hasPreviousPage": { + "type": "boolean" + } + } + }, + "RoleAddRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "name", + "siteId" + ], + "properties": { + "name": { + "type": "string", + "maxLength": 256, + "minLength": 1 + }, + "description": { + "type": "string", + "maxLength": 1000, + "minLength": 0, + "nullable": true + }, + "siteId": { + "type": "string", + "format": "uuid" + } + } + }, + "RoleDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "name": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + }, + "siteId": { + "type": "string", + "format": "uuid" + }, + "type": { + "$ref": "#/components/schemas/RoleTypes" + }, + "createdAt": { + "type": "string", + "format": "date-time" + }, + "updatedAt": { + "type": "string", + "format": "date-time", + "nullable": true + } + } + }, + "RoleDtoApiListResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/RoleDto" + } + }, + "pagination": { + "$ref": "#/components/schemas/PaginationInfo" + } + } + }, + "RoleDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/RoleDto" + } + } + }, + "RoleTypes": { + "type": "string", + "enum": [ + "UserDefined", + "Administrators", + "Authenticated", + "Guest", + "AllUsers" + ] + }, + "RoleUpdateRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "name" + ], + "properties": { + "name": { + "type": "string", + "maxLength": 256, + "minLength": 1 + }, + "description": { + "type": "string", + "maxLength": 1000, + "minLength": 0, + "nullable": true + } + } + }, + "SiteAddRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "name", + "urls" + ], + "properties": { + "name": { + "type": "string", + "minLength": 1 + }, + "description": { + "type": "string", + "nullable": true + }, + "urls": { + "type": "array", + "items": { + "type": "string" + } + }, + "layoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "detailLayoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "editLayoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "metaTitle": { + "type": "string", + "nullable": true + }, + "metaDescription": { + "type": "string", + "nullable": true + }, + "robotsIndex": { + "type": "boolean", + "nullable": true + }, + "robotsFollow": { + "type": "boolean", + "nullable": true + }, + "robotsTxt": { + "type": "string", + "nullable": true + }, + "googleTagsId": { + "type": "string", + "nullable": true + }, + "ogType": { + "type": "string", + "nullable": true + }, + "head": { + "type": "string", + "nullable": true + } + } + }, + "SiteDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "name": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + }, + "urls": { + "type": "array", + "nullable": true, + "items": { + "type": "string" + } + }, + "layoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "detailLayoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "editLayoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "metaTitle": { + "type": "string", + "nullable": true + }, + "metaDescription": { + "type": "string", + "nullable": true + }, + "robotsIndex": { + "type": "boolean", + "nullable": true + }, + "robotsFollow": { + "type": "boolean", + "nullable": true + }, + "robotsTxt": { + "type": "string", + "nullable": true + }, + "googleTagsId": { + "type": "string", + "nullable": true + }, + "ogType": { + "type": "string", + "nullable": true + }, + "head": { + "type": "string", + "nullable": true + } + } + }, + "SiteDtoApiListResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/SiteDto" + } + }, + "pagination": { + "$ref": "#/components/schemas/PaginationInfo" + } + } + }, + "SiteDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/SiteDto" + } + } + }, + "SiteUpdateRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "name", + "urls" + ], + "properties": { + "name": { + "type": "string", + "minLength": 1 + }, + "description": { + "type": "string", + "nullable": true + }, + "urls": { + "type": "array", + "items": { + "type": "string" + } + }, + "layoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "detailLayoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "editLayoutId": { + "type": "string", + "format": "uuid", + "nullable": true + }, + "metaTitle": { + "type": "string", + "nullable": true + }, + "metaDescription": { + "type": "string", + "nullable": true + }, + "robotsIndex": { + "type": "boolean", + "nullable": true + }, + "robotsFollow": { + "type": "boolean", + "nullable": true + }, + "robotsTxt": { + "type": "string", + "nullable": true + }, + "googleTagsId": { + "type": "string", + "nullable": true + }, + "ogType": { + "type": "string", + "nullable": true + }, + "head": { + "type": "string", + "nullable": true + } + } + }, + "ThreadResponseDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "name": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + }, + "model": { + "type": "string", + "nullable": true + }, + "systemPrompt": { + "type": "string", + "nullable": true + } + } + }, + "ThreadResponseDtoApiListResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ThreadResponseDto" + } + }, + "pagination": { + "$ref": "#/components/schemas/PaginationInfo" + } + } + }, + "ThreadResponseDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/ThreadResponseDto" + } + } + }, + "TodoCreateDto": { + "type": "object", + "additionalProperties": false, + "required": [ + "title" + ], + "properties": { + "title": { + "type": "string", + "minLength": 1 + }, + "description": { + "type": "string", + "nullable": true + }, + "isCompleted": { + "type": "boolean" + }, + "dueDate": { + "type": "string", + "format": "date-time", + "nullable": true + } + } + }, + "TodoResponseDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "title": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + }, + "isCompleted": { + "type": "boolean" + }, + "dueDate": { + "type": "string", + "format": "date-time", + "nullable": true + }, + "createdAt": { + "type": "string", + "format": "date-time" + }, + "version": { + "type": "integer", + "format": "int32" + } + } + }, + "TodoResponseDtoApiListResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/TodoResponseDto" + } + }, + "pagination": { + "$ref": "#/components/schemas/PaginationInfo" + } + } + }, + "TodoResponseDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/TodoResponseDto" + } + } + }, + "TodoUpdateDto": { + "type": "object", + "additionalProperties": false, + "required": [ + "title" + ], + "properties": { + "title": { + "type": "string", + "minLength": 1 + }, + "description": { + "type": "string", + "nullable": true + }, + "isCompleted": { + "type": "boolean" + }, + "dueDate": { + "type": "string", + "format": "date-time", + "nullable": true + } + } + }, + "UsageDetails": { + "type": "object", + "additionalProperties": false, + "properties": { + "inputTokenCount": { + "type": "integer", + "format": "int64", + "nullable": true + }, + "outputTokenCount": { + "type": "integer", + "format": "int64", + "nullable": true + }, + "totalTokenCount": { + "type": "integer", + "format": "int64", + "nullable": true + }, + "additionalCounts": { + "type": "object", + "nullable": true, + "additionalProperties": { + "type": "integer", + "format": "int64" + } + } + } + }, + "UserAddRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "email", + "emailConfirmed", + "isSuperAdmin", + "locked", + "password", + "suspended", + "username" + ], + "properties": { + "username": { + "type": "string", + "minLength": 1 + }, + "password": { + "type": "string", + "minLength": 1 + }, + "email": { + "type": "string", + "format": "email", + "minLength": 1 + }, + "suspended": { + "type": "boolean" + }, + "isSuperAdmin": { + "type": "boolean" + }, + "emailConfirmed": { + "type": "boolean" + }, + "locked": { + "type": "boolean" + }, + "description": { + "type": "string", + "nullable": true + } + } + }, + "UserDto": { + "type": "object", + "additionalProperties": false, + "properties": { + "id": { + "type": "string", + "format": "uuid" + }, + "username": { + "type": "string", + "nullable": true + }, + "email": { + "type": "string", + "nullable": true + }, + "description": { + "type": "string", + "nullable": true + }, + "isSuperAdmin": { + "type": "boolean" + }, + "lastLogin": { + "type": "string", + "format": "date-time", + "nullable": true + }, + "loginCount": { + "type": "integer", + "format": "int32" + }, + "passwordChangedAt": { + "type": "string", + "format": "date-time", + "nullable": true + }, + "passwordChangedBy": { + "type": "string", + "nullable": true + }, + "emailConfirmed": { + "type": "boolean" + }, + "phoneNumberConfirmed": { + "type": "boolean" + }, + "suspended": { + "type": "boolean" + }, + "lockoutEnd": { + "type": "string", + "format": "date-time", + "nullable": true + }, + "lockoutEnabled": { + "type": "boolean" + }, + "accessFailedCount": { + "type": "integer", + "format": "int32" + } + } + }, + "UserDtoApiListResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/UserDto" + } + }, + "pagination": { + "$ref": "#/components/schemas/PaginationInfo" + } + } + }, + "UserDtoApiResponse": { + "type": "object", + "additionalProperties": false, + "properties": { + "success": { + "type": "boolean" + }, + "message": { + "type": "string", + "nullable": true + }, + "messageCode": { + "type": "string", + "nullable": true + }, + "errors": { + "type": "array", + "nullable": true, + "items": { + "$ref": "#/components/schemas/ApiError" + } + }, + "timestamp": { + "type": "string", + "format": "date-time" + }, + "traceId": { + "type": "string", + "nullable": true + }, + "statusCode": { + "type": "integer", + "format": "int32" + }, + "sessionId": { + "type": "string", + "nullable": true + }, + "uniqueId": { + "type": "string", + "nullable": true + }, + "duration": { + "type": "number", + "format": "double" + }, + "data": { + "$ref": "#/components/schemas/UserDto" + } + } + }, + "UserRolesUpdateRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "roleIds", + "userId" + ], + "properties": { + "roleIds": { + "type": "array", + "minItems": 1, + "items": { + "type": "string", + "format": "uuid" + } + }, + "userId": { + "type": "string", + "format": "uuid" + } + } + }, + "UserUpdateRequest": { + "type": "object", + "additionalProperties": false, + "required": [ + "email", + "emailConfirmed", + "isSuperAdmin", + "locked", + "suspended" + ], + "properties": { + "email": { + "type": "string", + "format": "email", + "minLength": 1 + }, + "suspended": { + "type": "boolean" + }, + "isSuperAdmin": { + "type": "boolean" + }, + "emailConfirmed": { + "type": "boolean" + }, + "locked": { + "type": "boolean" + }, + "description": { + "type": "string", + "nullable": true + } + } + } + }, + "securitySchemes": { + "Bearer": { + "type": "http", + "description": "JWT Authorization header using the Bearer scheme. Example: \"Bearer {token}\"", + "scheme": "bearer", + "bearerFormat": "JWT" + } + } + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/App.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/App.razor new file mode 100644 index 000000000..c5ef431c9 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/App.razor @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/AuthLayout.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/AuthLayout.razor new file mode 100644 index 000000000..6925790da --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/AuthLayout.razor @@ -0,0 +1,15 @@ +@inherits LayoutComponentBase + +
+
+ + + @Body + +
+
\ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/EmptyLayout.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/EmptyLayout.razor new file mode 100644 index 000000000..5c7e6c12e --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/EmptyLayout.razor @@ -0,0 +1,10 @@ +@inherits LayoutComponentBase + + +
+
+
+ @Body +
+
+
\ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/MainLayout.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/MainLayout.razor new file mode 100644 index 000000000..99c8db626 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/MainLayout.razor @@ -0,0 +1,12 @@ +@inherits LayoutComponentBase + +
+
+ + + +
+ @Body +
+
+
\ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/MainLayout.razor.css b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/MainLayout.razor.css new file mode 100644 index 000000000..8aed7e1e3 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/MainLayout.razor.css @@ -0,0 +1,30 @@ +.page { + height: 100dvh; +} + +.page-wrapper { + overflow: auto; + height: 100%; +} + +#blazor-error-ui { + color-scheme: light only; + background: lightyellow; + bottom: 0; + box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.2); + box-sizing: border-box; + display: none; + left: 0; + padding: 0.6rem 1.25rem 0.7rem 1.25rem; + position: fixed; + width: 100%; + z-index: 1000; +} + +#blazor-error-ui .dismiss { + cursor: pointer; + position: absolute; + right: 0.75rem; + top: 0.5rem; +} + diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/NavMenu.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/NavMenu.razor new file mode 100644 index 000000000..d82228282 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/NavMenu.razor @@ -0,0 +1,220 @@ + \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/NavMenu.razor.css b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/NavMenu.razor.css new file mode 100644 index 000000000..e69de29bb diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PageBackButton.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PageBackButton.razor new file mode 100644 index 000000000..ea4ab0c24 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PageBackButton.razor @@ -0,0 +1,16 @@ +@if (string.IsNullOrEmpty(Url)) +{ + +} +else +{ + + + Back + +} + +@code { + [Parameter] + public string Url { get; set; } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PageBody.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PageBody.razor new file mode 100644 index 000000000..d1bedcf20 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PageBody.razor @@ -0,0 +1,15 @@ + +
+
+
+
+ @ChildContent +
+
+
+
+ +@code { + [Parameter] + public RenderFragment ChildContent { get; set; } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PageHeader.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PageHeader.razor new file mode 100644 index 000000000..680792114 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PageHeader.razor @@ -0,0 +1,34 @@ + + +@code { + [Parameter] + public string Title { get; set; } + + [Parameter] + public string BackUrl { get; set; } + + [Parameter] + public RenderFragment ChildContent { get; set; } + + [Parameter] + public RenderFragment? BackFragment { get; set; } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PluginFormPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PluginFormPage.razor new file mode 100644 index 000000000..8cbeb4252 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Layout/PluginFormPage.razor @@ -0,0 +1,83 @@ +@inject HttpClient Http +@inject NavigationManager Navigation + + + + + + + + @if (Model != null) + { + + + + + @ChildContent + + + + @if (!string.IsNullOrEmpty(BackUrl)) + { + Cancel + } + + + + @if (Errors.Count > 0) + { + + @foreach (var error in Errors) + { +

@error

+ } +
+ } +
+ } +
+
+
+ +@code { + [Parameter] + public string Title { get; set; } + + [Parameter] + public string FormName { get; set; } + + [Parameter] + public string BackUrl { get; set; } + + [Parameter] + public string SubmitLabel { get; set; } = "Subimt"; + + [Parameter] + public RenderFragment ChildContent { get; set; } + + [Parameter] + public EventCallback OnSubmit { get; set; } + + [Parameter] + public virtual object Model { get; set; } + + [Parameter] + public List Errors { get; set; } = []; + + public async Task HandleSubmit() + { + try + { + await OnSubmit.InvokeAsync(); + } + catch (ApiClientException ex) + { + Errors.Clear(); + Errors.Add(ex.Message); + } + catch(Exception ex) + { + Errors = [ex.Message]; + } + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/AuthLoginPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/AuthLoginPage.razor new file mode 100644 index 000000000..235d7ee17 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/AuthLoginPage.razor @@ -0,0 +1,58 @@ +@page "/auth/login" +@layout AuthLayout + +@inject HttpClient Http +@inject NavigationManager Navigation +@inject IHttpContextAccessor HttpContextAccessor + + + + Login to your account + + + + + + + + + + + + + Or + + + + + + + + + +@code { + [SupplyParameterFromForm(FormName = "LoginForm")] + private AccountLoginRequest Model { get; set; } = new(); + private async Task HandleSubmit() + { + Console.WriteLine("Form submitted"); + + var responseMessage = await Http.PostAsJsonAsync("http://localhost:5093/api/Accounts/Login", Model); + var response = await responseMessage.Content.ReadFromJsonAsync(); + + if (response?.Data != null) + { + var token = response.Data.Token; + Console.WriteLine(token); + + HttpContextAccessor.HttpContext.Response.Cookies.Append("auth_token", token, new CookieOptions + { + HttpOnly = true, + Secure = true, + SameSite = SameSiteMode.Strict + }); + + Navigation.NavigateTo("/admin", false); + } + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/AuthLogoutPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/AuthLogoutPage.razor new file mode 100644 index 000000000..a65ebb60d --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/AuthLogoutPage.razor @@ -0,0 +1,11 @@ +@page "/auth/logout" +@inject NavigationManager Navigation +@inject HttpClient Http + +@code { + protected override async Task OnInitializedAsync() + { + var response = await Http.GetFromJsonAsync("http://localhost:5093/api/Accounts/Logout"); + Navigation.NavigateTo("/", false); + } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockCreatePage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockCreatePage.razor new file mode 100644 index 000000000..f3a24cd37 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockCreatePage.razor @@ -0,0 +1,49 @@ +@page "/admin/blocks/create" +@inject HttpClient Http +@inject IHttpContextAccessor HttpContextAccessor +@inject NavigationManager Navigation +@inject ApiClientFactory Api + + + + +@code { + private List Errors { get; set; } = []; + + [SupplyParameterFromForm(FormName = "BlockCreateForm")] + private BlockFormModel Model { get; set; } = new(); + + private Guid SiteId { get; set; } + + protected override async Task OnInitializedAsync() + { + var baseUri = new Uri(Navigation.BaseUri); + var host = baseUri.Host.ToLowerInvariant(); + var siteResponse = await Api.Sites.GetByUrlAsync(host); + + if (siteResponse.Data != null) + { + SiteId = siteResponse.Data.Id; + } + } + private async Task HandleSubmit() + { + Errors.Clear(); + Console.WriteLine("Submitting Block model:"); + Console.WriteLine($"Name: {Model.Name}"); + Console.WriteLine($"Category: {Model.Category}"); + Console.WriteLine($"Description: {Model.Description}"); + Console.WriteLine($"Content: {Model.Content}"); + + var response = await Api.Blocks.AddAsync(new BlockAddRequest + { + SiteId = SiteId, + Name = Model.Name, + Category = Model.Category, + Description = Model.Description, + Content = Model.Content, + }); + Navigation.NavigateTo("/admin/blocks"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockEditPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockEditPage.razor new file mode 100644 index 000000000..556c9e210 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockEditPage.razor @@ -0,0 +1,51 @@ +@page "/admin/blocks/edit/{id}" +@inject HttpClient Http +@inject NavigationManager Navigation +@inject IHttpContextAccessor HttpContextAccessor +@inject ApiClientFactory Api + + + + +@code { + [Parameter] + public Guid Id { get; set; } + + [SupplyParameterFromForm(FormName = "BlockEditForm")] + private BlockFormModel Model { get; set; } + private List Errors { get; set; } = new(); + + protected override async Task OnInitializedAsync() + { + if(Model is null) { + Errors.Clear(); + + var block = await Api.Blocks.GetByIdAsync(Id); + + if (block.Data == null) + { + Errors.Add("Block not found"); + return; + } + + Model = new BlockFormModel + { + Name = block.Data.Name, + Category = block.Data.Category, + Description = block.Data.Description, + Content = block.Data.Content, + }; + } + } + + private async Task HandleSubmit() + { + var response = await Api.Blocks.UpdateAsync(Id, new BlockUpdateRequest { + Name = Model.Name, + Category = Model.Category, + Description = Model.Description, + Content = Model.Content, + }); + Navigation.NavigateTo("/admin/blocks"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockForm.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockForm.razor new file mode 100644 index 000000000..6add608c3 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockForm.razor @@ -0,0 +1,17 @@ + + + + + + + +@code { + [Parameter(CaptureUnmatchedValues = true)] + public IReadOnlyDictionary? AdditionalAttributes { get; set; } = default!; + + [Parameter] + public BlockFormModel Model { get; set; } + + [Parameter] + public EventCallback OnSubmit { get; set; } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockFormModel.cs b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockFormModel.cs new file mode 100644 index 000000000..e688abbca --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockFormModel.cs @@ -0,0 +1,7 @@ +public class BlockFormModel +{ + public string Name { get; set; } + public string Category { get; set; } + public string? Description { get; set; } + public string Content { get; set; } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockListPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockListPage.razor new file mode 100644 index 000000000..0c955314f --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Blocks/BlockListPage.razor @@ -0,0 +1,121 @@ +@page "/admin/blocks" + +@using FluentCMS.Web.UI.Components +@rendermode InteractiveServer + +@inject HttpClient Http +@inject NavigationManager Navigation +@inject ApiClientFactory Api + +Blocks + + + + + Create new block + + + + + + @if (Blocks == null) + { + +

Loading...

+
+ } + else if (!Blocks.Any()) + { + +

No Blocks found.

+
+ } + else + { + + @context.Name + @context.Category + @context.Description + + + + + + + + + + + } +
+
+ + + Are you sure to delete this block? + + +@code { + private List Blocks; + private Guid SiteId { get; set; } + protected override async Task OnInitializedAsync() + { + try + { + var baseUri = new Uri(Navigation.BaseUri); + var host = baseUri.Host.ToLowerInvariant(); + var siteResponse = await Api.Sites.GetByUrlAsync(host); + + if (siteResponse.Data != null) + { + SiteId = siteResponse.Data.Id; + await Load(); + } + } + catch (ApiClientException ex) + { + Console.WriteLine("Error fetching block: " + ex.Message); + } + } + + private async Task Load() + { + try + { + var response = await Api.Blocks.GetBySiteIdAsync(SiteId); + + if (response != null) + { + Blocks = response.Data.ToList(); + } + } + catch (ApiClientException ex) + { + Console.WriteLine("Error fetching block: " + ex.Message); + } + } + + private bool DeleteConfirmOpen { get; set; } = false; + private Guid? CurrentId { get; set; } + + private void OpenDeleteConfirm(Guid value) + { + CurrentId = value; + DeleteConfirmOpen = true; + } + + private void OnDeleteCancel() + { + DeleteConfirmOpen = false; + } + + private async Task OnDeleteConfirm() { + if (CurrentId != null) + { + await Api.Blocks.RemoveAsync(CurrentId.Value); + await Load(); + } + DeleteConfirmOpen = false; + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/DashboardPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/DashboardPage.razor new file mode 100644 index 000000000..d6dfbae8e --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/DashboardPage.razor @@ -0,0 +1,14 @@ +@page "/admin" + +Dashboard + + + + + +
    +
  • Getting started guide
  • +
  • Usage dashboard
  • +
  • Test agents
  • +
+
diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Error.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Error.razor new file mode 100644 index 000000000..576cc2d2f --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Error.razor @@ -0,0 +1,36 @@ +@page "/Error" +@using System.Diagnostics + +Error + +

Error.

+

An error occurred while processing your request.

+ +@if (ShowRequestId) +{ +

+ Request ID: @RequestId +

+} + +

Development Mode

+

+ Swapping to Development environment will display more detailed information about the error that occurred. +

+

+ The Development environment shouldn't be enabled for deployed applications. + It can result in displaying sensitive information from exceptions to end users. + For local debugging, enable the Development environment by setting the ASPNETCORE_ENVIRONMENT environment variable to Development + and restarting the app. +

+ +@code{ + [CascadingParameter] + private HttpContext? HttpContext { get; set; } + + private string? RequestId { get; set; } + private bool ShowRequestId => !string.IsNullOrEmpty(RequestId); + + protected override void OnInitialized() => + RequestId = Activity.Current?.Id ?? HttpContext?.TraceIdentifier; +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileListPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileListPage.razor new file mode 100644 index 000000000..b639af132 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileListPage.razor @@ -0,0 +1,417 @@ +@page "/admin/files/{id?}" + +@inject ApiClientFactory Api; +@inject NavigationManager Navigation; + +File Management + + + + + @* @if (!string.IsNullOrEmpty(BackUrl)) + { + + } *@ + + @if (ParentFolders != null) + { + + } + + + + + + + + + + + + + + @if (context.IsFolder) + { + + } + else + { + + } + @context.Name + + + @(context.IsFolder ? "Folder" : context.ContentType!) + + @if (!context.IsParentFolder) + { + @if (context.IsFolder) + { + + + + } + else + { + + + + + + } + } + + + + + + + Are you sure to delete this folder? + + + + Are you sure to delete this file? + + + + + + + + + + + +@code { + private string? BackUrl { get; set; } + + private List Assets { get; set; } + + private List ParentFolders { get; set; } + + private bool CreateFolderModalOpen { get; set; } = false; + private bool FolderMoveModalOpen { get; set; } = false; + private bool FileMoveModalOpen { get; set; } = false; + private bool FileUploadModalOpen { get; set; } = false; + + private bool FolderRenameModalOpen { get; set; } = false; + private bool FileRenameModalOpen { get; set; } = false; + private bool FolderSelectorModalOpen { get; set; } = false; + private bool IsMovingFolder { get; set; } = false; + + [Parameter] + public string? Id { get; set; } + + private Guid? CurrentFolderId { get; set; } + private Guid SiteId { get; set; } + + private async Task OnFolderSelectorCancel() + { + FolderSelectorModalOpen = false; + } + + private async Task OnFolderSelected(Guid folderId) + { + FolderSelectorModalOpen = false; + + if(IsMovingFolder) + { + await Api.Folders.MoveAsync(new FolderMoveRequest { Id = CurrentId.Value, ParentId = folderId}); + } else { + await Api.Files.MoveAsync(new FileMoveRequest { Id = CurrentId.Value, FolderId = folderId}); + } + await Load(); + } + + private async Task OpenFileUploadModal() + { + FileUploadModalOpen = true; + } + + private async Task OnFileUploadCancel() + { + FileUploadModalOpen = false; + } + + private async Task OnFileUpload(List Files) + { + FileUploadModalOpen = false; + await Api.Files.UploadAsync(CurrentFolderId.Value, Files); + await Load(); + } + + private async Task OnFolderRename(FolderRenameRequest model) + { + FolderRenameModalOpen = false; + await Api.Folders.RenameAsync(model); + await Load(); + } + + private async Task OnFolderRenameCancel() + { + FolderRenameModalOpen = false; + } + + private async Task OpenFolderRename(AssetDto folder) + { + FolderRenameModalOpen = true; + CurrentRenamingFolder = new FolderRenameRequest { + Id = folder.Id, + Name = folder.Name + }; + } + + private async Task OpenFileRename(AssetDto file) + { + FileRenameModalOpen = true; + CurrentRenamingFile = new FileRenameRequest { + Id = file.Id, + Name = file.Name + }; + } + + private async Task OnFileRename(FileRenameRequest model) + { + FileRenameModalOpen = false; + await Api.Files.RenameAsync(model); + await Load(); + } + + private async Task OnFileRenameCancel() + { + FileRenameModalOpen = false; + } + + private async Task HandleFileDelete(Guid fileId) + { + CurrentId = fileId; + DeleteFileConfirmOpen = true; + } + + private async Task HandleFolderDelete(Guid folderId) + { + CurrentId = folderId; + DeleteFolderConfirmOpen = true; + } + + private async Task HandleFileMove(Guid fileId) + { + IsMovingFolder = false; + FolderSelectorModalOpen = true; + CurrentId = fileId; + } + + private async Task HandleFolderMove(Guid folderId) + { + IsMovingFolder = true; + FolderSelectorModalOpen = true; + CurrentId = folderId; + } + + private async Task OpenCreateFolderModal() + { + CreateFolderModalOpen = true; + } + + private async Task OnCreateFolder(FolderAddRequest Model) + { + CreateFolderModalOpen = false; + Model.ParentId = CurrentFolderId.Value; + Model.SiteId = SiteId; + await Api.Folders.CreateAsync(Model); + await Load(); + } + + private async Task OnCreateFolderCancel() + { + CreateFolderModalOpen = false; + } + + private async Task Load() + { + if(CurrentFolderId == null) + { + var rootResponse = await Api.Folders.GetRootAsync(SiteId); + if(rootResponse?.Data != null) + { + CurrentFolderId = rootResponse.Data.Id; + } + } + + var folderResponse = await Api.Folders.GetByIdAsync(CurrentFolderId.Value); + if (folderResponse?.Data != null) + { + Assets = folderResponse.Data.Folders.Select(x => new AssetDto + { + Id = x.Id, + Name = x.Name, + NormalizedName = x.NormalizedName, + IsFolder = true + }).ToList(); + + Assets.AddRange(folderResponse.Data.Files.Select(f => new AssetDto + { + Id = f.Id, + Name = f.Name, + NormalizedName = f.NormalizedName, + IsFolder = false, + Extension = f.Extension, + ContentType = f.ContentType, + Size = f.Size + })); + + if (folderResponse.Data.ParentFolder != null) + { + BackUrl = "/admin/files/" + folderResponse.Data.ParentFolder.Id.ToString(); + Assets.Insert(0, new AssetDto + { + Id = folderResponse.Data.ParentFolder.Id, + Name = "(parent)", + NormalizedName = "..", + IsFolder = true, + IsParentFolder = true, + }); + } + else + { + BackUrl = null; + } + } + + + var parentFoldersResponse = await Api.Folders.GetParentFoldersAsync(CurrentFolderId.Value); + if (parentFoldersResponse?.Data != null) + { + ParentFolders = parentFoldersResponse.Data.ToList(); + } + } + + private async Task NavigateFolder(Guid folderId) + { + Console.WriteLine("NavigateFolder" + folderId); + CurrentFolderId = folderId; + await Load(); + } + + protected override async Task OnInitializedAsync() + { + var baseUri = new Uri(Navigation.BaseUri); + var host = baseUri.Host.ToLowerInvariant(); + var siteResponse = await Api.Sites.GetByUrlAsync(host); + + if (siteResponse?.Data != null) + { + SiteId = siteResponse.Data.Id; + } + + if (Guid.TryParse(Id, out var guid)) + { + CurrentFolderId = guid; + } + } + + protected override async Task OnParametersSetAsync() + { + + await Load(); + } + + private bool DeleteFolderConfirmOpen { get; set; } = false; + private bool DeleteFileConfirmOpen { get; set; } = false; + + private Guid? CurrentId { get; set; } + private FileRenameRequest? CurrentRenamingFile { get; set; } + private FolderRenameRequest? CurrentRenamingFolder { get; set; } + + private void OpenFileDeleteConfirm(Guid value) + { + CurrentId = value; + DeleteFileConfirmOpen = true; + } + + private void OpenFolderDeleteConfirm(Guid value) + { + CurrentId = value; + DeleteFileConfirmOpen = true; + } + + private async Task OnDeleteFileCancel() { + DeleteFileConfirmOpen = false; + } + + private async Task OnDeleteFileConfirm() { + if (CurrentId != null) + { + await Api.Files.RemoveAsync(CurrentId.Value); + await Load(); + } + DeleteFileConfirmOpen = false; + } + + private async Task OnDeleteFolderCancel() { + DeleteFolderConfirmOpen = false; + } + + private async Task OnDeleteFolderConfirm() { + if (CurrentId != null) + { + await Api.Folders.RemoveAsync(CurrentId.Value); + await Load(); + } + DeleteFolderConfirmOpen = false; + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileModels.cs b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileModels.cs new file mode 100644 index 000000000..ad9c37c62 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileModels.cs @@ -0,0 +1,13 @@ +namespace FluentCMS.Web.ApiClients; + +public class AssetDto +{ + public Guid Id { get; set; } + public string Name { get; set; } + public string NormalizedName { get; set; } + public bool IsFolder { get; set; } + public bool IsParentFolder { get; set; } = false; + public string? ContentType { get; set; } + public string? Extension { get; set; } + public long? Size { get; set; } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileRenameModal.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileRenameModal.razor new file mode 100644 index 000000000..11633aa5e --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileRenameModal.razor @@ -0,0 +1,52 @@ + + + @if(Model != null) + { + + + + Rename File + + + + + + + + + + + + + + + } + + +@code { + [Parameter] + public FileRenameRequest Model { get; set; } + + [Parameter] + public bool Visible { get; set; } = false; + + [Parameter] + public EventCallback OnCancel { get; set; } + + [Parameter] + public EventCallback OnSubmit { get; set; } + + private async Task HandleCancel() + { + await OnCancel.InvokeAsync(); + } + + private async Task HandleSubmit() + { + await OnSubmit.InvokeAsync(Model); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileUploadModal.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileUploadModal.razor new file mode 100644 index 000000000..8343fa602 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FileUploadModal.razor @@ -0,0 +1,66 @@ + + + + + + Upload File + + + + + + + + @* *@ + + + + + + + + + + + +@code { + [Parameter] + public bool Visible { get; set; } = false; + + [Parameter] + public EventCallback> OnSubmit { get; set; } + + [Parameter] + public EventCallback OnCancel { get; set; } + + int CONFIG_MAX_FILE_SIZE = 1024 * 1024 * 10; + int CONFIG_MAX_FILES = 5; + + private List Files { get; set; } = []; + + private async Task OnFilesChanged(InputFileChangeEventArgs e) + { + Console.WriteLine("ONFILESCHANGED: "); + Files = []; + foreach (var file in e.GetMultipleFiles(CONFIG_MAX_FILES)) + { + var Data = file.OpenReadStream(CONFIG_MAX_FILE_SIZE); + Files.Add(new FileParameter(Data, file.Name, file.ContentType)); + } + await OnSubmit.InvokeAsync(Files); + } + + private async Task HandleCancel() + { + await OnCancel.InvokeAsync(); + } + + private async Task HandleSubmit() + { + await OnSubmit.InvokeAsync(Files); + } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FolderCreateModal.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FolderCreateModal.razor new file mode 100644 index 000000000..585b03ed4 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FolderCreateModal.razor @@ -0,0 +1,52 @@ +@namespace FluentCMS.Web.UI.Admin.Components + + + @if (Model != null) + { + + + + Create New Folder + + + + + + + + + + + + + + + } + + +@code { + [Parameter] + public bool Visible { get; set; } = false; + + [Parameter] + public EventCallback OnSubmit { get; set; } + + [Parameter] + public EventCallback OnCancel { get; set; } + + private FolderAddRequest Model { get; set; } = new(); + + private async Task HandleSubmit() + { + await OnSubmit.InvokeAsync(Model); + } + + private async Task HandleCancel() + { + await OnCancel.InvokeAsync(); + } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FolderRenameModal.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FolderRenameModal.razor new file mode 100644 index 000000000..7dcf176ed --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FolderRenameModal.razor @@ -0,0 +1,53 @@ + + + @if (Model != null) + { + + + + Rename Folder + + + + + + + + + + + + + + + + } + + +@code { + [Parameter] + public FolderRenameRequest Model { get; set; } + + [Parameter] + public bool Visible { get; set; } = false; + + [Parameter] + public EventCallback OnCancel { get; set; } + + [Parameter] + public EventCallback OnSubmit { get; set; } + + private async Task HandleCancel() + { + await OnCancel.InvokeAsync(); + } + + private async Task HandleSubmit() + { + await OnSubmit.InvokeAsync(Model); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FolderSelectorModal.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FolderSelectorModal.razor new file mode 100644 index 000000000..a5989fb68 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Files/FolderSelectorModal.razor @@ -0,0 +1,220 @@ +@namespace FluentCMS.Web.UI.Admin + +@inject ApiClientFactory Api +@inject NavigationManager Navigation + + + + + Choose Folder + + +
+ @if (ParentFolders != null) + { + + } + + @* + @foreach (var breadcrumbItem in ParentFolders) + { + @if (breadcrumbItem.Id == Model) + { + + @breadcrumbItem.Name + + } + else if(breadcrumbItem.Id == RootFolder?.Id) + { + + @breadcrumbItem.Name + + } + else + { + + @breadcrumbItem.Name + + } + } + *@ + @* + + + + *@ + + + + + @if (context.IsFolder && DisabledFolder != context.Id) + { + + @context.Name + } + else + { + + @context.Name + } + + + @(context.IsFolder ? "Folder" : context.ContentType!) + + @if (!context.IsParentFolder) + { + @if (context.IsFolder && context.Id != DisabledFolder) + { + + } + + } + + +
+ + + + + + + + + + +
+ +@code { + [Parameter] + public Guid? DisabledFolder { get; set; } + + [Parameter] + public Guid? CurrentFolderId { get; set; } + + [Parameter] + public Guid? Model { get; set; } = default!; + + [Parameter] + public bool Visible { get; set; } = false; + + [Parameter] + public EventCallback OnSubmit { get; set; } + + [Parameter] + public EventCallback OnCancel { get; set; } + + private List Assets { get; set; } + + private List ParentFolders { get; set; } = new List(); + + private Guid SiteId { get; set; } + + private FolderDto RootFolder { get; set; } + private FolderDto CurrentFolder { get; set; } + + private async Task NavigateFolder(Guid folderId) + { + CurrentFolderId = folderId; + await Load(); + } + + protected override async Task OnInitializedAsync() + { + var baseUri = new Uri(Navigation.BaseUri); + var host = baseUri.Host.ToLowerInvariant(); + var siteResponse = await Api.Sites.GetByUrlAsync(host); + + if (siteResponse?.Data != null) + { + SiteId = siteResponse.Data.Id; + } + + var rootResponse = await Api.Folders.GetRootAsync(SiteId); + if (rootResponse?.Data != null) + { + RootFolder = rootResponse.Data; + } + await Load(); + } + + private async Task Load() + { + if (CurrentFolderId == null) + { + CurrentFolderId = RootFolder.Id; + } + + var folderResponse = await Api.Folders.GetByIdAsync(CurrentFolderId.Value); + if (folderResponse?.Data != null) + { + CurrentFolder = folderResponse.Data; + + Assets = CurrentFolder.Folders.Select(x => new AssetDto + { + Id = x.Id, + Name = x.Name, + NormalizedName = x.NormalizedName, + IsFolder = true + }).ToList(); + + Assets.AddRange(CurrentFolder.Files.Select(f => new AssetDto + { + Id = f.Id, + Name = f.Name, + NormalizedName = f.NormalizedName, + IsFolder = false, + Extension = f.Extension, + ContentType = f.ContentType, + Size = f.Size + })); + + if (CurrentFolder.ParentFolder != null) + { + Assets.Insert(0, new AssetDto + { + Id = CurrentFolder.ParentFolder.Id, + Name = "(parent)", + NormalizedName = "..", + IsFolder = true, + IsParentFolder = true, + }); + } + } + + var parentFoldersResponse = await Api.Folders.GetParentFoldersAsync(CurrentFolderId.Value); + if (parentFoldersResponse?.Data != null) + { + ParentFolders = parentFoldersResponse.Data.ToList(); + } + } + + private async Task HandleCancel() + { + await OnCancel.InvokeAsync(); + } + + private async Task OnChooseFolder() + { + if (CurrentFolderId is null) return; + await OnSubmit.InvokeAsync(CurrentFolderId.Value); + } + +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/HomePage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/HomePage.razor new file mode 100644 index 000000000..9858c5d57 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/HomePage.razor @@ -0,0 +1,58 @@ +@page "/" +@layout EmptyLayout + +Home + +
+
+

Welcome 👋

+
+
+ +
+

Overview

+

+ This is your default home page. Explore features, track activity, and manage resources from one place. +

+ +
+ Login + Admin Panel +
+ +
+ +
+

Features

+
    +
  • + 📘 Documentation +
  • +
  • + 📊 Analytics +
  • +
  • + ⚙️ System Settings +
  • +
  • + 🧪 Testing Sandbox +
  • +
  • + 🧩 Components Library +
  • +
  • + 🚀 Release Notes +
  • +
  • + 👥 User Management +
  • +
+
+ +
+

Status

+
+ All systems operational. No active alerts. +
+
+
diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutCreatePage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutCreatePage.razor new file mode 100644 index 000000000..e265d6875 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutCreatePage.razor @@ -0,0 +1,42 @@ +@page "/admin/layouts/create" +@inject HttpClient Http +@inject IHttpContextAccessor HttpContextAccessor +@inject NavigationManager Navigation +@inject ApiClientFactory Api + + + + +@code { + private List Errors { get; set; } = []; + + [SupplyParameterFromForm(FormName = "LayoutCreateForm")] + private LayoutFormModel Model { get; set; } = new(); + + private Guid SiteId { get; set; } + + protected override async Task OnInitializedAsync() + { + var baseUri = new Uri(Navigation.BaseUri); + var host = baseUri.Host.ToLowerInvariant(); + var siteResponse = await Api.Sites.GetByUrlAsync(host); + + if (siteResponse.Data != null) + { + SiteId = siteResponse.Data.Id; + } + } + private async Task HandleSubmit() + { + Errors.Clear(); + var response = await Api.Layouts.AddAsync(new LayoutAddRequest + { + SiteId = SiteId, + Name = Model.Name, + Head = Model.Head, + Body = Model.Body, + }); + Navigation.NavigateTo("/admin/layouts"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutEditPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutEditPage.razor new file mode 100644 index 000000000..8e32b4244 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutEditPage.razor @@ -0,0 +1,48 @@ +@page "/admin/layouts/edit/{id}" +@inject HttpClient Http +@inject NavigationManager Navigation +@inject IHttpContextAccessor HttpContextAccessor +@inject ApiClientFactory Api + + + +@code { + [Parameter] + public Guid Id { get; set; } + + [SupplyParameterFromForm(FormName = "LayoutEditForm")] + private LayoutFormModel Model { get; set; } + private List Errors { get; set; } = new(); + + protected override async Task OnInitializedAsync() + { + if(Model is null) { + Errors.Clear(); + + var layout = await Api.Layouts.GetByIdAsync(Id); + + if (layout.Data == null) + { + Errors.Add("Layout not found"); + return; + } + + Model = new LayoutFormModel + { + Name = layout.Data.Name, + Head = layout.Data.Head, + Body = layout.Data.Body, + }; + } + } + + private async Task HandleSubmit() + { + var response = await Api.Layouts.UpdateAsync(Id, new LayoutUpdateRequest { + Name = Model.Name, + Head = Model.Head, + Body = Model.Body, + }); + Navigation.NavigateTo("/admin/layouts"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutForm.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutForm.razor new file mode 100644 index 000000000..26a448afd --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutForm.razor @@ -0,0 +1,17 @@ + + + + + + + +@code { + [Parameter(CaptureUnmatchedValues = true)] + public IReadOnlyDictionary? AdditionalAttributes { get; set; } = default!; + + [Parameter] + public LayoutFormModel Model { get; set; } + + [Parameter] + public EventCallback OnSubmit { get; set; } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutFormModel.cs b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutFormModel.cs new file mode 100644 index 000000000..39be98f08 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutFormModel.cs @@ -0,0 +1,6 @@ +public class LayoutFormModel +{ + public string Name { get; set; } + public string? Head { get; set; } + public string? Body { get; set; } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutListPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutListPage.razor new file mode 100644 index 000000000..d3227c03e --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Layouts/LayoutListPage.razor @@ -0,0 +1,118 @@ +@page "/admin/layouts" + +@using FluentCMS.Web.UI.Components +@rendermode InteractiveServer + +@inject HttpClient Http +@inject NavigationManager Navigation +@inject ApiClientFactory Api + +Layouts + + + Create new layout + + + + + @if (Layouts == null) + { + +

Loading...

+
+ } + else if (!Layouts.Any()) + { + +

No Layouts found.

+
+ } + else + { + + @context.Name + + + + + + + + + + + } +
+
+ + + Are you sure to delete this layout? + + +@code { + private List Layouts; + private Guid SiteId { get; set; } + + protected override async Task OnInitializedAsync() + { + try + { + var baseUri = new Uri(Navigation.BaseUri); + var host = baseUri.Host.ToLowerInvariant(); + var siteResponse = await Api.Sites.GetByUrlAsync(host); + + if (siteResponse.Data != null) + { + SiteId = siteResponse.Data.Id; + await Load(); + } + } + catch (ApiClientException ex) + { + Console.WriteLine("Error fetching site: " + ex.Message); + } + } + + private async Task Load() + { + try + { + var response = await Api.Layouts.GetBySiteIdAsync(SiteId); + + if (response != null) + { + Layouts = response.Data.ToList(); + } + + } + catch (ApiClientException ex) + { + Console.WriteLine("Error fetching layouts: " + ex.Message); + } + } + + private bool DeleteConfirmOpen { get; set; } = false; + private Guid? CurrentId { get; set; } + + private void OpenDeleteConfirm(Guid value) + { + CurrentId = value; + DeleteConfirmOpen = true; + } + + private async Task OnDeleteCancel() + { + DeleteConfirmOpen = false; + } + + private async Task OnDeleteConfirm() { + if (CurrentId != null) + { + await Api.Layouts.RemoveAsync(CurrentId.Value); + await Load(); + } + DeleteConfirmOpen = false; + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageCreatePage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageCreatePage.razor new file mode 100644 index 000000000..51701e048 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageCreatePage.razor @@ -0,0 +1,85 @@ +@page "/admin/pages/create" +@inject HttpClient Http +@inject IHttpContextAccessor HttpContextAccessor +@inject ApiClientFactory Api +@inject NavigationManager Navigation + + + +@code { + private List PageOptions { get; set; } + private List LayoutOptions { get; set; } + + private Guid SiteId { get; set; } + + [SupplyParameterFromForm(FormName = "PageCreateForm")] + private PageFormModel Model { get; set; } = new(); + + protected override async Task OnInitializedAsync() + { + var baseUri = new Uri(Navigation.BaseUri); + var host = baseUri.Host.ToLowerInvariant(); + var siteResponse = await Api.Sites.GetByUrlAsync(host); + + if (siteResponse.Data != null) + { + SiteId = siteResponse.Data.Id; + + var pagesResponse = await Api.Pages.GetBySiteIdAsync(SiteId); + if (pagesResponse.Data != null) + { + PageOptions = new List() { new() { Title = "(root)", Value = Guid.Empty } }; + PageOptions.AddRange(pagesResponse.Data.Select(p => new SelectOption { Title = p.Title, Value = p.Id })); + } + + var layoutsResponse = await Api.Layouts.GetBySiteIdAsync(SiteId); + if (layoutsResponse.Data != null) + { + LayoutOptions = new List() { new() { Title = "(default)", Value = Guid.Empty } }; + LayoutOptions.AddRange(layoutsResponse.Data.Select(l => new SelectOption { Title = l.Name, Value = l.Id })); + } + } + + } + + public async Task HandleSubmit() + { + // Log all model fields + Console.WriteLine("Submitting Page model:"); + Console.WriteLine($"Title: {Model.Title}"); + Console.WriteLine($"Slug: {Model.Slug}"); + Console.WriteLine($"Order: {Model.Order}"); + Console.WriteLine($"ParentId: {Model.ParentId}"); + Console.WriteLine($"LayoutId: {Model.LayoutId}"); + Console.WriteLine($"EditLayoutId: {Model.EditLayoutId}"); + Console.WriteLine($"DetailLayoutId: {Model.DetailLayoutId}"); + Console.WriteLine($"MetaTitle: {Model.MetaTitle}"); + Console.WriteLine($"MetaDescription: {Model.MetaDescription}"); + Console.WriteLine($"RobotsIndex: {Model.RobotsIndex}"); + Console.WriteLine($"RobotsFollow: {Model.RobotsFollow}"); + Console.WriteLine($"OgType: {Model.OgType}"); + Console.WriteLine($"Head: {Model.Head}"); + + var request = new PageAddRequest + { + Title = Model.Title, + Slug = Model.Slug, + SiteId = SiteId, + Order = Model.Order, + ParentId = Model.ParentId == Guid.Empty ? null : Model.ParentId, + LayoutId = Model.LayoutId == Guid.Empty ? null : Model.LayoutId, + EditLayoutId = Model.EditLayoutId == Guid.Empty ? null : Model.EditLayoutId, + DetailLayoutId = Model.DetailLayoutId == Guid.Empty ? null : Model.DetailLayoutId, + MetaTitle = Model.MetaTitle, + MetaDescription = Model.MetaDescription, + RobotsIndex = Model.RobotsIndex, + RobotsFollow = Model.RobotsFollow, + OgType = Model.OgType, + Head = Model.Head + }; + + var response = await Api.Pages.AddAsync(request); + Navigation.NavigateTo("/admin/pages"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageEditPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageEditPage.razor new file mode 100644 index 000000000..ac5829a2a --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageEditPage.razor @@ -0,0 +1,119 @@ +@page "/admin/pages/edit/{id}" +@inject HttpClient Http +@inject IHttpContextAccessor HttpContextAccessor +@inject NavigationManager Navigation +@inject ApiClientFactory Api + + + +@code { + [Parameter] + public Guid Id { get; set; } + + private List PageOptions { get; set; } + private List LayoutOptions { get; set; } + + [SupplyParameterFromForm(FormName = "PageEditForm")] + private PageFormModel Model { get; set; } + private List Errors { get; set; } = new(); + + protected override async Task OnInitializedAsync() + { + var baseUri = new Uri(Navigation.BaseUri); + var host = baseUri.Host.ToLowerInvariant(); + var siteResponse = await Api.Sites.GetByUrlAsync(host); + + if (siteResponse.Data != null) + { + var pagesResponse = await Api.Pages.GetBySiteIdAsync(siteResponse.Data.Id); + if (pagesResponse.Data != null) + { + PageOptions = new List() { new() { Title = "(root)", Value = Guid.Empty } }; + PageOptions.AddRange(pagesResponse.Data.Where(x => x.Id != Id).Select(p => new SelectOption + { + Title = + p.Title, + Value = p.Id + })); + Console.WriteLine("Page count: " + PageOptions.Count); + } + + var layoutsResponse = await Api.Layouts.GetBySiteIdAsync(siteResponse.Data.Id); + if (layoutsResponse.Data != null) + { + LayoutOptions = new List() { new() { Title = "(default)", Value = Guid.Empty } }; + LayoutOptions.AddRange(layoutsResponse.Data.Select(l => new SelectOption { Title = l.Name, Value = l.Id })); + Console.WriteLine("Layout count: " + LayoutOptions.Count); + } + + if (Model is null) + { + Errors.Clear(); + + + var page = await Api.Pages.GetByIdAsync(Id); + + if (page.Data == null) + { + Errors.Add("Page not found"); + return; + } + + Model = new PageFormModel + { + Title = page.Data.Title, + Slug = page.Data.Slug, + Order = page.Data.Order, + ParentId = page.Data.ParentId, + LayoutId = page.Data.LayoutId, + EditLayoutId = page.Data.EditLayoutId, + DetailLayoutId = page.Data.DetailLayoutId, + MetaTitle = page.Data.MetaTitle, + MetaDescription = page.Data.MetaDescription, + RobotsIndex = page.Data.RobotsIndex, + RobotsFollow = page.Data.RobotsFollow, + OgType = page.Data.OgType, + }; + } + } + } + + private async Task HandleSubmit() + { + // Log all model fields + Console.WriteLine("Submitting Page model:"); + Console.WriteLine($"Title: {Model.Title}"); + Console.WriteLine($"Slug: {Model.Slug}"); + Console.WriteLine($"Order: {Model.Order}"); + Console.WriteLine($"ParentId: {Model.ParentId}"); + Console.WriteLine($"LayoutId: {Model.LayoutId}"); + Console.WriteLine($"EditLayoutId: {Model.EditLayoutId}"); + Console.WriteLine($"DetailLayoutId: {Model.DetailLayoutId}"); + Console.WriteLine($"MetaTitle: {Model.MetaTitle}"); + Console.WriteLine($"MetaDescription: {Model.MetaDescription}"); + Console.WriteLine($"RobotsIndex: {Model.RobotsIndex}"); + Console.WriteLine($"RobotsFollow: {Model.RobotsFollow}"); + Console.WriteLine($"OgType: {Model.OgType}"); + Console.WriteLine($"Head: {Model.Head}"); + + var request = new PageUpdateRequest + { + Title = Model.Title, + Slug = Model.Slug, + Order = Model.Order, + ParentId = Model.ParentId == Guid.Empty ? null : Model.ParentId, + LayoutId = Model.LayoutId == Guid.Empty ? null : Model.LayoutId, + EditLayoutId = Model.EditLayoutId == Guid.Empty ? null : Model.EditLayoutId, + DetailLayoutId = Model.DetailLayoutId == Guid.Empty ? null : Model.DetailLayoutId, + MetaTitle = Model.MetaTitle, + MetaDescription = Model.MetaDescription, + RobotsIndex = Model.RobotsIndex, + RobotsFollow = Model.RobotsFollow, + OgType = Model.OgType, + }; + + var response = await Api.Pages.UpdateAsync(Id, request); + Navigation.NavigateTo("/admin/pages"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageForm.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageForm.razor new file mode 100644 index 000000000..637bd7fe7 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageForm.razor @@ -0,0 +1,47 @@ + + + + + + + + + + + + + + + + + + + +@code { + private List OgTypeOptions = + [ + new SelectOptionString { Title = "(default)", Value = string.Empty }, + new SelectOptionString { Title = "Website", Value = "website" }, + new SelectOptionString { Title = "Article", Value = "article" }, + new SelectOptionString { Title = "Product", Value = "product" }, + new SelectOptionString { Title = "Video", Value = "video" } + ]; + + [Parameter(CaptureUnmatchedValues = true)] + public IReadOnlyDictionary? AdditionalAttributes { get; set; } = default!; + + [Parameter] + public string BackUrl { get; set; } + + [Parameter] + public EventCallback OnSubmit { get; set; } + + [Parameter] + public PageFormModel Model { get; set; } + + [Parameter] + public List LayoutOptions { get; set; } + + [Parameter] + public List PageOptions { get; set; } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageFormModel.cs b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageFormModel.cs new file mode 100644 index 000000000..5f74be1fb --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageFormModel.cs @@ -0,0 +1,22 @@ +namespace FluentCMS.Web.UI.Admin; + +public class PageFormModel +{ + public Guid? Id { get; set; } + public Guid? ParentId { get; set; } + public string Title { get; set; } = default!; + public string Slug { get; set; } + public int Order { get; set; } + + public Guid? LayoutId { get; set; } + public Guid? EditLayoutId { get; set; } + public Guid? DetailLayoutId { get; set; } + + public string? MetaTitle { get; set; } + public string? MetaDescription { get; set; } + public bool RobotsIndex { get; set; } + public bool RobotsFollow { get; set; } + public string? OgType { get; set; } + + public string? Head { get; set; } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageListPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageListPage.razor new file mode 100644 index 000000000..2776276e6 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Pages/PageListPage.razor @@ -0,0 +1,136 @@ +@page "/admin/pages" + +@using FluentCMS.Web.UI.Components + +@inject HttpClient Http +@inject NavigationManager Navigation +@inject ApiClientFactory Api + +Pages + + + + + Create new page + + + + + + @if (Pages == null) + { + +

Loading...

+
+ } + else if (!Pages.Any()) + { + +

No Pages found.

+
+ } + else + { + + @context.Title + @context.Slug + @context.FullPath + + @(Pages.FirstOrDefault(x => x.Id == context.ParentId)?.Title ?? "(root)") + + + + @(Layouts.FirstOrDefault(x => x.Id == context.LayoutId)?.Name ?? "(default)") + + + + + + + + + + + + + + + } +
+
+ + + Are you sure to delete this page? + + + +@code { + private List? Pages; + + private List Layouts; + private Guid? SiteId { get; set; } + protected override async Task OnInitializedAsync() + { + try + { + var baseUri = new Uri(Navigation.BaseUri); + var host = baseUri.Host.ToLowerInvariant(); + var siteResponse = await Api.Sites.GetByUrlAsync(host); + + if (siteResponse.Data != null) + { + SiteId = siteResponse.Data.Id; + + var layoutsResponse = await Api.Layouts.GetAllAsync(); + if (layoutsResponse.Data != null) + { + Layouts = layoutsResponse.Data.ToList(); + await Load(); + } + } + + } + catch (ApiClientException ex) + { + Console.WriteLine("Error fetching page: " + ex.Message); + } + } + + private async Task Load() + { + if (SiteId != null) + { + var response = await Api.Pages.GetBySiteIdAsync(SiteId.Value); + + if (response != null) + { + Pages = response.Data.ToList(); + } + } + + } + + private bool DeleteConfirmOpen { get; set; } = false; + private Guid? CurrentId { get; set; } + + private void OpenDeleteConfirm(Guid value) + { + CurrentId = value; + DeleteConfirmOpen = true; + } + + private async Task OnDeleteCancel() { + DeleteConfirmOpen = false; + } + private async Task OnDeleteConfirm() { + if(CurrentId != null) + { + await Api.Pages.RemoveAsync(CurrentId.Value); + await Load(); + } + DeleteConfirmOpen = false; + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/ProfilePage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/ProfilePage.razor new file mode 100644 index 000000000..8260c058a --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/ProfilePage.razor @@ -0,0 +1,46 @@ +@page "/admin/profile" +@inject HttpClient Http + +@* + + + + + + + + + + *@ + + + + + + + + + + + + +@code { + + private AccountChangePasswordRequest ChangePasswordModel { get; set; } = new(); + private LoginDto ProfileModel { get; set; } = new(); + + private async Task OnInitialized() + { + Console.WriteLine("Load profile"); + } + + private async Task OnProfileSubmit() + { + Console.WriteLine("Profile submit"); + } + private async Task OnChangePasswordSubmit() + { + Console.WriteLine("change password submit"); + + } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleCreatePage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleCreatePage.razor new file mode 100644 index 000000000..e1013b6c1 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleCreatePage.razor @@ -0,0 +1,29 @@ +@page "/admin/roles/create" +@inject HttpClient Http + +@inject IHttpContextAccessor HttpContextAccessor +@inject NavigationManager Navigation +@inject ApiClientFactory Api + + + +@code { + private List Errors { get; set; } = []; + + [SupplyParameterFromForm(FormName="RoleCreateForm")] + private RoleFormModel Model { get; set; } = new(); + + protected override async Task OnInitializedAsync() + { + Console.WriteLine("Initialized"); + } + + public async Task HandleSubmit() + { + await Api.Roles.AddAsync(new RoleAddRequest { + Name = Model.Name, + Description = Model.Description, + }); + Navigation.NavigateTo("/admin/roles"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleEditPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleEditPage.razor new file mode 100644 index 000000000..afb0c61c6 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleEditPage.razor @@ -0,0 +1,55 @@ +@page "/admin/roles/edit/{id}" +@inject HttpClient Http +@inject IHttpContextAccessor HttpContextAccessor +@inject NavigationManager Navigation +@inject ApiClientFactory Api + +Edit Role + + + + + + +@code { + [Parameter] + public Guid Id { get; set; } + + [SupplyParameterFromForm(FormName = "RoleEditForm")] + private RoleFormModel Model { get; set; } + private List Errors { get; set; } = new(); + + protected override async Task OnInitializedAsync() + { + if (Model is null) + { + + Errors.Clear(); + + var response = await Api.Roles.GetByIdAsync(Id); + + if (response?.Data == null) + { + Errors.Add("Role not found"); + return; + } + + Model = new RoleFormModel + { + Name = response.Data.Name, + Description = response.Data.Description, + }; + } + + } + + private async Task HandleSubmit() + { + var response = await Api.Roles.UpdateAsync(Id, new RoleUpdateRequest { + Name = Model.Name, + Description = Model.Description, + }); + + Navigation.NavigateTo("/admin/roles"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleForm.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleForm.razor new file mode 100644 index 000000000..d3a34e34e --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleForm.razor @@ -0,0 +1,19 @@ + + + + + + +@code { + [Parameter(CaptureUnmatchedValues = true)] + public IReadOnlyDictionary? AdditionalAttributes { get; set; } = default!; + + [Parameter] + public string BackUrl { get; set; } + + [Parameter] + public EventCallback OnSubmit { get; set; } + + [Parameter] + public RoleFormModel Model { get; set; } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleFormModel.cs b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleFormModel.cs new file mode 100644 index 000000000..f22675fd2 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleFormModel.cs @@ -0,0 +1,8 @@ +namespace FluentCMS.Web.UI.Admin; + +public class RoleFormModel +{ + public Guid? Id { get; set; } + public string Name { get; set; } = string.Empty; + public string Description { get; set; } = string.Empty; +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleListPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleListPage.razor new file mode 100644 index 000000000..b286eac17 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Roles/RoleListPage.razor @@ -0,0 +1,101 @@ +@page "/admin/roles" + +@using FluentCMS.Web.UI.Components + +@inject HttpClient Http +@inject ApiClientFactory Api + +Roles + + + + + Create new role + + + + + + @if (Roles == null) + { + +

Loading...

+
+ } + else if (!Roles.Any()) + { + +

No Roles found.

+
+ } + else + { + + @context.Name + @context.Description + + + + + + + + + + } +
+
+ + + Are you sure to delete this role? + + +@code { + private List Roles = new List(); + + protected override async Task OnInitializedAsync() + { + await Load(); + } + + private async Task Load() + { + try + { + var response = await Api.Roles.GetAllAsync(); + + if (response != null) + { + Roles = response.Data.ToList(); + } + } + catch (ApiClientException ex) + { + Console.WriteLine("Error fetching roles: " + ex.Message); + } + } + + private bool DeleteConfirmOpen { get; set; } = false; + private Guid? CurrentId { get; set; } + + private void OpenDeleteConfirm(Guid value) + { + CurrentId = value; + DeleteConfirmOpen = true; + } + + private async Task OnDeleteCancel() { + DeleteConfirmOpen = false; + } + + private async Task OnDeleteConfirm() { + if(CurrentId != null) + { + await Api.Roles.RemoveAsync(CurrentId.Value); + await Load(); + } + DeleteConfirmOpen = false; + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/SelectOption.cs b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/SelectOption.cs new file mode 100644 index 000000000..215493048 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/SelectOption.cs @@ -0,0 +1,12 @@ +namespace FluentCMS.Web.UI.Admin; + +public class SelectOption +{ + public string Title { get; set; } = string.Empty; + public Guid? Value { get; set; } +} +public class SelectOptionString +{ + public string Title { get; set; } = string.Empty; + public string Value { get; set; } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/SettingsPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/SettingsPage.razor new file mode 100644 index 000000000..5d98d8590 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/SettingsPage.razor @@ -0,0 +1,15 @@ +@page "/admin/settings" + +Settings + + + + + +
    +
  • Settings
  • +
  • Base URL
  • +
  • API Key
  • +
  • Save + Cancel (go to home)
  • +
+
diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteCreatePage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteCreatePage.razor new file mode 100644 index 000000000..2409b4a8d --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteCreatePage.razor @@ -0,0 +1,77 @@ +@page "/admin/sites/create" + +@using System.Text.Json; +@using System.Text.Json.Serialization; + +@inject HttpClient Http +@inject IHttpContextAccessor HttpContextAccessor +@inject NavigationManager Navigation +@inject ApiClientFactory Api + + + +@code { + private List Errors { get; set; } = []; + + [SupplyParameterFromForm(FormName = "SiteCreateForm")] + private SiteFormModel Model { get; set; } = new(); + + private List LayoutOptions { get; set; } + + protected override async Task OnInitializedAsync() + { + Console.WriteLine("Initialized"); + var resp = await Api.Layouts.GetAllAsync(); + + if(resp?.Data != null) + { + LayoutOptions = new List { + new() { + Title = "(none)", + Value = null + } + }; + LayoutOptions.AddRange( + resp.Data.Select(x => new SelectOption + { + Title = x.Name, + Value = x.Id + }) + ); + } + } + + public async Task OnSubmit() + { + Errors.Clear(); + + var request = new SiteAddRequest + { + Name = Model.Name, + Description = Model.Description, + Urls = [.. Model.Urls.Split(",")], + MetaTitle = Model.MetaTitle, + MetaDescription = Model.MetaDescription, + LayoutId = Model.LayoutId, + EditLayoutId = Model.EditLayoutId, + DetailLayoutId = Model.DetailLayoutId, + GoogleTagsId = Model.GoogleTagsId, + Head = Model.Head, + OgType = Model.OgType, + RobotsTxt = Model.RobotsTxt, + RobotsFollow = Model.RobotsFollow, + RobotsIndex = Model.RobotsIndex, + }; + + Console.WriteLine(JsonSerializer.Serialize(request, new JsonSerializerOptions + { + WriteIndented = true, + DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull + })); + + var response = await Api.Sites.AddAsync(request); + + Navigation.NavigateTo("/admin/sites"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteEditPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteEditPage.razor new file mode 100644 index 000000000..da58fab7e --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteEditPage.razor @@ -0,0 +1,95 @@ +@page "/admin/sites/edit/{id}" +@inject HttpClient Http +@inject NavigationManager Navigation +@inject IHttpContextAccessor HttpContextAccessor +@inject ApiClientFactory Api + + + +@code { + [Parameter] + public Guid Id { get; set; } + + [SupplyParameterFromForm(FormName = "SiteEditForm")] + private SiteFormModel Model { get; set; } + private List Errors { get; set; } = new(); + private List LayoutOptions { get; set; } + + protected override async Task OnInitializedAsync() + { + // Load layouts + var resp = await Api.Layouts.GetBySiteIdAsync(Id); + + if(resp?.Data != null) + { + LayoutOptions = new List { + new() { + Title = "(none)", + Value = null + } + }; + LayoutOptions.AddRange( + resp.Data.Select(x => new SelectOption + { + Title = x.Name, + Value = x.Id + }) + ); + } + + if(Model is null) { + Errors.Clear(); + + var site = await Api.Sites.GetByIdAsync(Id); + + if (site.Data == null) + { + Errors.Add("Site not found"); + return; + } + + Model = new SiteFormModel + { + Id = site.Data.Id, + Name = site.Data.Name, + Description = site.Data.Description, + Urls = string.Join(",", site.Data.Urls ?? []), + MetaTitle = site.Data.MetaTitle, + MetaDescription = site.Data.MetaDescription, + GoogleTagsId = site.Data.GoogleTagsId, + LayoutId = site.Data.LayoutId, + DetailLayoutId = site.Data.DetailLayoutId, + EditLayoutId = site.Data.EditLayoutId, + Head = site.Data.Head, + OgType = site.Data.OgType, + RobotsTxt = site.Data.RobotsTxt, + RobotsFollow = site.Data.RobotsFollow.Value, + RobotsIndex = site.Data.RobotsIndex.Value, + }; + } + + + } + + private async Task HandleSubmit() + { + var response = await Api.Sites.UpdateAsync(Id, new SiteUpdateRequest { + Name = Model.Name, + Description = Model.Description, + Urls = [.. Model.Urls.Split(",")], + MetaTitle = Model.MetaTitle, + MetaDescription = Model.MetaDescription, + GoogleTagsId = Model.GoogleTagsId, + LayoutId = Model.LayoutId, + DetailLayoutId = Model.DetailLayoutId, + EditLayoutId = Model.EditLayoutId, + Head = Model.Head, + OgType = Model.OgType, + RobotsTxt = Model.RobotsTxt, + RobotsFollow = Model.RobotsFollow, + RobotsIndex = Model.RobotsIndex, + }); + + Navigation.NavigateTo("/admin/sites"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteForm.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteForm.razor new file mode 100644 index 000000000..4ab0fea4c --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteForm.razor @@ -0,0 +1,61 @@ + +@if(LayoutOptions != null) +{ + + + + + + + @if (Model.Id != null) + { + + + + } + + + + + + + + + + +} + +@code { + private List OgTypeOptions = + [ + new() { Title = "(default)", Value = "" }, + new() { Title = "Website", Value = "website" }, + new() { Title = "Article", Value = "article" }, + new() { Title = "Product", Value = "product" }, + new() { Title = "Video", Value = "video" } + ]; + + [Parameter(CaptureUnmatchedValues = true)] + public IReadOnlyDictionary? AdditionalAttributes { get; set; } = default!; + + [Parameter] + public SiteFormModel Model { get; set; } = default!; + + [Parameter] + public EventCallback OnSubmit { get; set; } = default!; + + [Parameter] + public List LayoutOptions { get; set; } = default!; + + class SelectOptionString + { + public string Title { get; set; } = string.Empty; + public string? Value { get; set; } + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteFormModel.cs b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteFormModel.cs new file mode 100644 index 000000000..024f46f9e --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteFormModel.cs @@ -0,0 +1,20 @@ +namespace FluentCMS.Web.UI.Admin; + +public class SiteFormModel +{ + public Guid? Id { get; set; } + public string? Name { get; set; } + public string? Description { get; set; } + public string? Urls { get; set; } + public string? Head { get; set; } + public Guid? LayoutId { get; set; } + public Guid? EditLayoutId { get; set; } + public Guid? DetailLayoutId { get; set; } + public string? OgType { get; set; } + public string? GoogleTagsId { get; set; } + public string? MetaTitle { get; set; } + public string? MetaDescription { get; set; } + public bool RobotsIndex { get; set; } + public bool RobotsFollow { get; set; } + public string? RobotsTxt { get; set; } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteListPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteListPage.razor new file mode 100644 index 000000000..d96396296 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Sites/SiteListPage.razor @@ -0,0 +1,122 @@ +@page "/admin/sites" + +@inject HttpClient Http +@inject ApiClientFactory Api + +Sites + + + + + Add Site + + + + + + @if (Sites == null) + { + +

Loading...

+
+ } + else if (!Sites.Any()) + { + +

No Sites found.

+
+ } + else + { + + @context.Name + @context.Description + + @if (context.Urls != null) + { + + + @foreach (var url in context!.Urls!) + { + @url + } + + } + + @* + @Layouts.Find(x => x.Id == context.LayoutId)?.Name + + + @Layouts.Find(x => x.Id == context.DetailLayoutId)?.Name + + + @Layouts.Find(x => x.Id == context.EditLayoutId)?.Name + + @context.Model *@ + + + + + + + + } +
+
+ + + Are you sure to delete "" site? + + + + +@code { + private List Layouts { get; set; } + private List? Sites { get; set; } + + + protected override async Task OnInitializedAsync() + { + var resp = await Api.Sites.GetAllAsync(); + + if (resp?.Data != null) + { + Sites = resp.Data.ToList(); + Console.WriteLine(Sites); + + foreach(var site in Sites) + { + Console.WriteLine($"Site: {site.Name} ({site.Description})"); + foreach(var url in site.Urls) + { + Console.WriteLine($" Url: {url}"); + } + } + } + + Layouts = new List { +new LayoutDto { +Id = Guid.NewGuid() +}, +}; + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserCreatePage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserCreatePage.razor new file mode 100644 index 000000000..a1056702e --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserCreatePage.razor @@ -0,0 +1,34 @@ +@page "/admin/users/create" +@inject HttpClient Http +@inject IHttpContextAccessor HttpContextAccessor +@inject NavigationManager Navigation +@inject ApiClientFactory Api + + + +@code { + private List Errors { get; set; } = []; + + [SupplyParameterFromForm(FormName="UserCreateForm")] + private UserFormModel Model { get; set; } = new(); + + public async Task OnInitialized() + { + Console.WriteLine("Initialized"); + } + + public async Task HandleSubmit() + { + Errors.Clear(); + var response = await Api.Users.AddAsync(new UserAddRequest { + Username = Model.Username, + Email = Model.Email, + Description = Model.Description, + Suspended = Model.Suspended, + IsSuperAdmin = Model.IsSuperAdmin, + EmailConfirmed = Model.EmailConfirmed, + Locked = Model.Locked + }); + Navigation.NavigateTo("/admin/users"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserEditPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserEditPage.razor new file mode 100644 index 000000000..962a07950 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserEditPage.razor @@ -0,0 +1,58 @@ +@page "/admin/users/edit/{id}" +@inject HttpClient Http +@inject IHttpContextAccessor HttpContextAccessor +@inject ApiClientFactory Api +@inject NavigationManager Navigation + + + +@code { + [Parameter] + public Guid Id { get; set; } + + [SupplyParameterFromForm(FormName = "UserEditForm")] + private UserFormModel Model { get; set; } + private List Errors { get; set; } = new(); + + protected override async Task OnInitializedAsync() + { + if(Model is null) { + Errors.Clear(); + + var user = await Api.Users.GetByIdAsync(Id); + + if (user.Data == null) + { + Errors.Add("User not found"); + return; + } + + Model = new UserFormModel + { + Id = user.Data.Id, + Username = user.Data.Username, + Email = user.Data.Email, + Description = user.Data.Description, + Suspended = user.Data.Suspended, + IsSuperAdmin = user.Data.IsSuperAdmin, + EmailConfirmed = user.Data.EmailConfirmed, + Locked = user.Data.LockoutEnabled + }; + } + + } + + private async Task HandleSubmit() + { + var response = await Api.Users.UpdateAsync(Id, new UserUpdateRequest { + Email = Model.Email, + Description = Model.Description, + Suspended = Model.Suspended, + IsSuperAdmin = Model.IsSuperAdmin, + EmailConfirmed = Model.EmailConfirmed, + Locked = Model.Locked + }); + Navigation.NavigateTo("/admin/users"); + } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserForm.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserForm.razor new file mode 100644 index 000000000..40a277367 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserForm.razor @@ -0,0 +1,25 @@ + + + + + + + + + + + + +@code { + [Parameter(CaptureUnmatchedValues = true)] + public IReadOnlyDictionary? AdditionalAttributes { get; set; } = default!; + + [Parameter] + public string BackUrl { get; set; } + + [Parameter] + public EventCallback OnSubmit { get; set; } + + [Parameter] + public UserFormModel Model { get; set; } +} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserFormModel.cs b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserFormModel.cs new file mode 100644 index 000000000..d7e69e9af --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserFormModel.cs @@ -0,0 +1,14 @@ +namespace FluentCMS.Web.UI.Admin; + +public class UserFormModel +{ + public Guid? Id { get; set; } + public string? Username { get; set; } + public string? Email { get; set; } + public string? Description { get; set; } + + public bool Suspended { get; set; } + public bool IsSuperAdmin { get; set; } + public bool EmailConfirmed { get; set; } + public bool Locked { get; set; } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserListPage.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserListPage.razor new file mode 100644 index 000000000..36ebde55b --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Pages/Users/UserListPage.razor @@ -0,0 +1,112 @@ +@page "/admin/users" + +@using FluentCMS.Web.UI.Components + +@inject HttpClient Http +@inject ApiClientFactory Api + +Users + + + Create new user + + + + + @if (Users == null) + { + +

Loading...

+
+ } + else if (!Users.Any()) + { + +

No Users found.

+
+ } + else + { + + @context.Username + @context.Email + @context.Description + + @if (context.IsSuperAdmin) + { + Yes + } else { + No + } + + + @if (context.Suspended) + { + Yes + } else { + No + } + + + + + + + + + } +
+
+ + + Are you sure to delete this user? + + +@code { + private List? Users; + + protected override async Task OnInitializedAsync() + { + await Load(); + } + + private async Task Load() + { + try + { + var response = await Api.Users.GetAllAsync(); + if (response != null) + { + Users = response.Data.ToList(); + } + } + catch (ApiClientException ex) + { + Console.WriteLine("Error fetching user: " + ex.Message); + } + } + + + private bool DeleteConfirmOpen { get; set; } = false; + private Guid? CurrentId { get; set; } + + private void OpenDeleteConfirm(Guid value) + { + CurrentId = value; + DeleteConfirmOpen = true; + } + + private async Task OnDeleteCancel() { + DeleteConfirmOpen = false; + } + private async Task OnDeleteConfirm() { + if(CurrentId != null) + { + await Api.Pages.RemoveAsync(CurrentId.Value); + await Load(); + } + DeleteConfirmOpen = false; + } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/Routes.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/Routes.razor new file mode 100644 index 000000000..f756e19df --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/Routes.razor @@ -0,0 +1,6 @@ + + + + + + diff --git a/src/UI/FluentCMS.Web.UI.Admin/Components/_Imports.razor b/src/UI/FluentCMS.Web.UI.Admin/Components/_Imports.razor new file mode 100644 index 000000000..b1c10c018 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Components/_Imports.razor @@ -0,0 +1,13 @@ +@using System.Net.Http +@using System.Net.Http.Json +@using Microsoft.AspNetCore.Components.Forms +@using Microsoft.AspNetCore.Components.Routing +@using Microsoft.AspNetCore.Components.Web +@using static Microsoft.AspNetCore.Components.Web.RenderMode +@using Microsoft.AspNetCore.Components.Web.Virtualization +@using Microsoft.JSInterop +@using FluentCMS.Web.UI.Admin +@using FluentCMS.Web.ApiClients +@using FluentCMS.Web.UI.Admin.Components +@using FluentCMS.Web.UI.Admin.Components.Layout +@using FluentCMS.Web.UI.Components diff --git a/src/UI/FluentCMS.Web.UI.Admin/FluentCMS.Web.UI.Admin.csproj b/src/UI/FluentCMS.Web.UI.Admin/FluentCMS.Web.UI.Admin.csproj new file mode 100644 index 000000000..8c3ccc7cf --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/FluentCMS.Web.UI.Admin.csproj @@ -0,0 +1,18 @@ + + + + net9.0 + enable + enable + + + + + + + + + + + + diff --git a/src/UI/FluentCMS.Web.UI.Admin/Program.cs b/src/UI/FluentCMS.Web.UI.Admin/Program.cs new file mode 100644 index 000000000..ec0539564 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Program.cs @@ -0,0 +1,52 @@ +using FluentCMS.Web.UI.Admin.Components; +using FluentCMS.Web.ApiClients; +using Microsoft.AspNetCore.Builder; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Hosting; +using System; + +var builder = WebApplication.CreateBuilder(args); + +// Add services to the container. +builder.Services.AddRazorComponents() + .AddInteractiveServerComponents(); + +builder.Services.AddEndpointsApiExplorer(); +builder.Services.AddSwaggerGen(); + +builder.Services.AddUIComponents(); +builder.Services.AddControllers(); +builder.Services.AddApiClients(builder.Configuration); +builder.Services.AddHttpContextAccessor(); +builder.Services.AddHttpClient(); + +builder.Services.AddScoped(); + +var app = builder.Build(); + +// Configure the HTTP request pipeline. +if (!app.Environment.IsDevelopment()) +{ + app.UseExceptionHandler("/Error", createScopeForErrors: true); + // The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts. + app.UseHsts(); +} + +if (app.Environment.IsDevelopment()) +{ + app.UseSwagger(); + app.UseSwaggerUI(); +} + +app.UseHttpsRedirection(); + + +app.UseAntiforgery(); + +app.MapStaticAssets(); +app.MapRazorComponents() + .AddInteractiveServerRenderMode(); + +app.MapControllers(); + +app.Run(); diff --git a/src/UI/FluentCMS.Web.UI.Admin/Properties/launchSettings.json b/src/UI/FluentCMS.Web.UI.Admin/Properties/launchSettings.json new file mode 100644 index 000000000..ab050162c --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/Properties/launchSettings.json @@ -0,0 +1,23 @@ +{ + "$schema": "https://json.schemastore.org/launchsettings.json", + "profiles": { + "http": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "http://localhost:5022", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + }, + "https": { + "commandName": "Project", + "dotnetRunMessages": true, + "launchBrowser": true, + "applicationUrl": "https://localhost:7108;http://localhost:5022", + "environmentVariables": { + "ASPNETCORE_ENVIRONMENT": "Development" + } + } + } + } diff --git a/src/UI/FluentCMS.Web.UI.Admin/appsettings.Development.json b/src/UI/FluentCMS.Web.UI.Admin/appsettings.Development.json new file mode 100644 index 000000000..0c208ae91 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/appsettings.Development.json @@ -0,0 +1,8 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + } +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/appsettings.json b/src/UI/FluentCMS.Web.UI.Admin/appsettings.json new file mode 100644 index 000000000..e2eff803d --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/appsettings.json @@ -0,0 +1,13 @@ +{ + "Logging": { + "LogLevel": { + "Default": "Information", + "Microsoft.AspNetCore": "Warning" + } + }, + "ClientSettings": { + "Url": "http://localhost:5093", + "Key": "" + }, + "AllowedHosts": "*" +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/css/app.css b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/css/app.css new file mode 100644 index 000000000..81748151c --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/css/app.css @@ -0,0 +1,17 @@ +.page-title { + margin-top: -4px; +} + +.page-back-button { + display: inline-flex; + align-items: center; + justify-content: center; + gap: 0.25rem; + font-size: 10px; + line-height: 10px; + color: var(--f-secondary); +} + +.page-back-button svg { + margin-top: -2px; +} diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/favicon.png b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/favicon.png new file mode 100644 index 000000000..8422b5969 Binary files /dev/null and b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/favicon.png differ diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/images/logo-dark.svg b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/images/logo-dark.svg new file mode 100644 index 000000000..74bb61e39 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/images/logo-dark.svg @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/images/logo.svg b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/images/logo.svg new file mode 100644 index 000000000..994b7f60e --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/images/logo.svg @@ -0,0 +1,15 @@ + + + + + + + + + + + + + + + diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/app.js b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/app.js new file mode 100644 index 000000000..b53ede26c --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/app.js @@ -0,0 +1 @@ +import './tabler.esm' \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.js b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.js new file mode 100644 index 000000000..b800f7b77 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.js @@ -0,0 +1,38 @@ +/*! + * Tabler v1.4.0 (https://tabler.io) + * Copyright 2018-2025 The Tabler Authors + * Copyright 2018-2025 codecalm.net Paweł Kuna + * Licensed under MIT (https://github.com/tabler/tabler/blob/master/LICENSE) + */ +/** + * demo-theme is specifically loaded right after the body and not deferred + * to ensure we switch to the chosen dark/light theme as fast as possible. + * This will prevent any flashes of the light theme (default) before switching. + */ +const themeConfig = { + "theme": "light", + "theme-base": "gray", + "theme-font": "sans-serif", + "theme-primary": "blue", + "theme-radius": "1" +}; +const params = new Proxy(new URLSearchParams(window.location.search), { + get: (searchParams, prop) => searchParams.get(prop) +}); +for (const key in themeConfig) { + const param = params[key]; + let selectedValue; + if (!!param) { + localStorage.setItem('tabler-' + key, param); + selectedValue = param; + } else { + const storedTheme = localStorage.getItem('tabler-' + key); + selectedValue = storedTheme ? storedTheme : themeConfig[key]; + } + if (selectedValue !== themeConfig[key]) { + document.documentElement.setAttribute('data-bs-' + key, selectedValue); + } else { + document.documentElement.removeAttribute('data-bs-' + key); + } +} +//# sourceMappingURL=tabler-theme.esm.js.map diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.js.map b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.js.map new file mode 100644 index 000000000..a6c3c37ce --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.js.map @@ -0,0 +1 @@ +{"version":3,"file":"tabler-theme.esm.js","sources":["../../js/tabler-theme.js"],"sourcesContent":["/**\n * demo-theme is specifically loaded right after the body and not deferred\n * to ensure we switch to the chosen dark/light theme as fast as possible.\n * This will prevent any flashes of the light theme (default) before switching.\n */\nconst themeConfig = {\n\t\"theme\": \"light\",\n\t\"theme-base\": \"gray\",\n\t\"theme-font\": \"sans-serif\",\n\t\"theme-primary\": \"blue\",\n\t\"theme-radius\": \"1\",\n}\n\nconst params = new Proxy(new URLSearchParams(window.location.search), {\n\tget: (searchParams, prop) => searchParams.get(prop),\n})\n\nfor (const key in themeConfig) {\n\tconst param = params[key]\n\tlet selectedValue\n\n\tif (!!param) {\n\t\tlocalStorage.setItem('tabler-' + key, param)\n\t\tselectedValue = param\n\t} else {\n\t\tconst storedTheme = localStorage.getItem('tabler-' + key)\n\t\tselectedValue = storedTheme ? storedTheme : themeConfig[key]\n\t}\n\n\tif (selectedValue !== themeConfig[key]) {\n\t\tdocument.documentElement.setAttribute('data-bs-' + key, selectedValue)\n\t} else {\n\t\tdocument.documentElement.removeAttribute('data-bs-' + key)\n\t}\n}"],"names":["themeConfig","params","Proxy","URLSearchParams","window","location","search","get","searchParams","prop","key","param","selectedValue","localStorage","setItem","storedTheme","getItem","document","documentElement","setAttribute","removeAttribute"],"mappings":";;;;;;AAAA;AACA;AACA;AACA;AACA;AACA,MAAMA,WAAW,GAAG;AACnB,EAAA,OAAO,EAAE,OAAO;AAChB,EAAA,YAAY,EAAE,MAAM;AACpB,EAAA,YAAY,EAAE,YAAY;AAC1B,EAAA,eAAe,EAAE,MAAM;AACvB,EAAA,cAAc,EAAE;AACjB,CAAC;AAED,MAAMC,MAAM,GAAG,IAAIC,KAAK,CAAC,IAAIC,eAAe,CAACC,MAAM,CAACC,QAAQ,CAACC,MAAM,CAAC,EAAE;EACrEC,GAAG,EAAEA,CAACC,YAAY,EAAEC,IAAI,KAAKD,YAAY,CAACD,GAAG,CAACE,IAAI;AACnD,CAAC,CAAC;AAEF,KAAK,MAAMC,GAAG,IAAIV,WAAW,EAAE;AAC9B,EAAA,MAAMW,KAAK,GAAGV,MAAM,CAACS,GAAG,CAAC;AACzB,EAAA,IAAIE,aAAa;EAEjB,IAAI,CAAC,CAACD,KAAK,EAAE;IACZE,YAAY,CAACC,OAAO,CAAC,SAAS,GAAGJ,GAAG,EAAEC,KAAK,CAAC;AAC5CC,IAAAA,aAAa,GAAGD,KAAK;AACtB,EAAA,CAAC,MAAM;IACN,MAAMI,WAAW,GAAGF,YAAY,CAACG,OAAO,CAAC,SAAS,GAAGN,GAAG,CAAC;IACzDE,aAAa,GAAGG,WAAW,GAAGA,WAAW,GAAGf,WAAW,CAACU,GAAG,CAAC;AAC7D,EAAA;AAEA,EAAA,IAAIE,aAAa,KAAKZ,WAAW,CAACU,GAAG,CAAC,EAAE;IACvCO,QAAQ,CAACC,eAAe,CAACC,YAAY,CAAC,UAAU,GAAGT,GAAG,EAAEE,aAAa,CAAC;AACvE,EAAA,CAAC,MAAM;IACNK,QAAQ,CAACC,eAAe,CAACE,eAAe,CAAC,UAAU,GAAGV,GAAG,CAAC;AAC3D,EAAA;AACD"} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.min.js b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.min.js new file mode 100644 index 000000000..a1130a522 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.min.js @@ -0,0 +1,8 @@ +/*! + * Tabler v1.4.0 (https://tabler.io) + * Copyright 2018-2025 The Tabler Authors + * Copyright 2018-2025 codecalm.net Paweł Kuna + * Licensed under MIT (https://github.com/tabler/tabler/blob/master/LICENSE) + */ +const themeConfig={theme:"light","theme-base":"gray","theme-font":"sans-serif","theme-primary":"blue","theme-radius":"1"},params=new Proxy(new URLSearchParams(window.location.search),{get:(e,t)=>e.get(t)});for(const e in themeConfig){const t=params[e];let a;if(t)localStorage.setItem("tabler-"+e,t),a=t;else{a=localStorage.getItem("tabler-"+e)||themeConfig[e]}a!==themeConfig[e]?document.documentElement.setAttribute("data-bs-"+e,a):document.documentElement.removeAttribute("data-bs-"+e)} +//# sourceMappingURL=tabler-theme.esm.min.js.map \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.min.js.map b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.min.js.map new file mode 100644 index 000000000..abbb38f5d --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.esm.min.js.map @@ -0,0 +1 @@ +{"version":3,"names":["themeConfig","theme","params","Proxy","URLSearchParams","window","location","search","get","searchParams","prop","key","param","selectedValue","localStorage","setItem","getItem","document","documentElement","setAttribute","removeAttribute"],"sources":["../../js/tabler-theme.js"],"sourcesContent":["/**\n * demo-theme is specifically loaded right after the body and not deferred\n * to ensure we switch to the chosen dark/light theme as fast as possible.\n * This will prevent any flashes of the light theme (default) before switching.\n */\nconst themeConfig = {\n\t\"theme\": \"light\",\n\t\"theme-base\": \"gray\",\n\t\"theme-font\": \"sans-serif\",\n\t\"theme-primary\": \"blue\",\n\t\"theme-radius\": \"1\",\n}\n\nconst params = new Proxy(new URLSearchParams(window.location.search), {\n\tget: (searchParams, prop) => searchParams.get(prop),\n})\n\nfor (const key in themeConfig) {\n\tconst param = params[key]\n\tlet selectedValue\n\n\tif (!!param) {\n\t\tlocalStorage.setItem('tabler-' + key, param)\n\t\tselectedValue = param\n\t} else {\n\t\tconst storedTheme = localStorage.getItem('tabler-' + key)\n\t\tselectedValue = storedTheme ? storedTheme : themeConfig[key]\n\t}\n\n\tif (selectedValue !== themeConfig[key]) {\n\t\tdocument.documentElement.setAttribute('data-bs-' + key, selectedValue)\n\t} else {\n\t\tdocument.documentElement.removeAttribute('data-bs-' + key)\n\t}\n}"],"mappings":";;;;;;AAKA,MAAMA,YAAc,CACnBC,MAAS,QACT,aAAc,OACd,aAAc,aACd,gBAAiB,OACjB,eAAgB,KAGXC,OAAS,IAAIC,MAAM,IAAIC,gBAAgBC,OAAOC,SAASC,QAAS,CACrEC,IAAKA,CAACC,EAAcC,IAASD,EAAaD,IAAIE,KAG/C,IAAK,MAAMC,KAAOX,YAAa,CAC9B,MAAMY,EAAQV,OAAOS,GACrB,IAAIE,EAEJ,GAAMD,EACLE,aAAaC,QAAQ,UAAYJ,EAAKC,GACtCC,EAAgBD,MACV,CAENC,EADoBC,aAAaE,QAAQ,UAAYL,IACTX,YAAYW,EACzD,CAEIE,IAAkBb,YAAYW,GACjCM,SAASC,gBAAgBC,aAAa,WAAaR,EAAKE,GAExDI,SAASC,gBAAgBE,gBAAgB,WAAaT,EAExD","ignoreList":[]} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.js b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.js new file mode 100644 index 000000000..3003fdbc2 --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.js @@ -0,0 +1,45 @@ +/*! + * Tabler v1.4.0 (https://tabler.io) + * Copyright 2018-2025 The Tabler Authors + * Copyright 2018-2025 codecalm.net Paweł Kuna + * Licensed under MIT (https://github.com/tabler/tabler/blob/master/LICENSE) + */ +(function (factory) { + typeof define === 'function' && define.amd ? define(factory) : + factory(); +})((function () { 'use strict'; + + /** + * demo-theme is specifically loaded right after the body and not deferred + * to ensure we switch to the chosen dark/light theme as fast as possible. + * This will prevent any flashes of the light theme (default) before switching. + */ + const themeConfig = { + "theme": "light", + "theme-base": "gray", + "theme-font": "sans-serif", + "theme-primary": "blue", + "theme-radius": "1" + }; + const params = new Proxy(new URLSearchParams(window.location.search), { + get: (searchParams, prop) => searchParams.get(prop) + }); + for (const key in themeConfig) { + const param = params[key]; + let selectedValue; + if (!!param) { + localStorage.setItem('tabler-' + key, param); + selectedValue = param; + } else { + const storedTheme = localStorage.getItem('tabler-' + key); + selectedValue = storedTheme ? storedTheme : themeConfig[key]; + } + if (selectedValue !== themeConfig[key]) { + document.documentElement.setAttribute('data-bs-' + key, selectedValue); + } else { + document.documentElement.removeAttribute('data-bs-' + key); + } + } + +})); +//# sourceMappingURL=tabler-theme.js.map diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.js.map b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.js.map new file mode 100644 index 000000000..19ed4828d --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.js.map @@ -0,0 +1 @@ +{"version":3,"file":"tabler-theme.js","sources":["../../js/tabler-theme.js"],"sourcesContent":["/**\n * demo-theme is specifically loaded right after the body and not deferred\n * to ensure we switch to the chosen dark/light theme as fast as possible.\n * This will prevent any flashes of the light theme (default) before switching.\n */\nconst themeConfig = {\n\t\"theme\": \"light\",\n\t\"theme-base\": \"gray\",\n\t\"theme-font\": \"sans-serif\",\n\t\"theme-primary\": \"blue\",\n\t\"theme-radius\": \"1\",\n}\n\nconst params = new Proxy(new URLSearchParams(window.location.search), {\n\tget: (searchParams, prop) => searchParams.get(prop),\n})\n\nfor (const key in themeConfig) {\n\tconst param = params[key]\n\tlet selectedValue\n\n\tif (!!param) {\n\t\tlocalStorage.setItem('tabler-' + key, param)\n\t\tselectedValue = param\n\t} else {\n\t\tconst storedTheme = localStorage.getItem('tabler-' + key)\n\t\tselectedValue = storedTheme ? storedTheme : themeConfig[key]\n\t}\n\n\tif (selectedValue !== themeConfig[key]) {\n\t\tdocument.documentElement.setAttribute('data-bs-' + key, selectedValue)\n\t} else {\n\t\tdocument.documentElement.removeAttribute('data-bs-' + key)\n\t}\n}"],"names":["themeConfig","params","Proxy","URLSearchParams","window","location","search","get","searchParams","prop","key","param","selectedValue","localStorage","setItem","storedTheme","getItem","document","documentElement","setAttribute","removeAttribute"],"mappings":";;;;;;;;;;;CAAA;CACA;CACA;CACA;CACA;CACA,MAAMA,WAAW,GAAG;CACnB,EAAA,OAAO,EAAE,OAAO;CAChB,EAAA,YAAY,EAAE,MAAM;CACpB,EAAA,YAAY,EAAE,YAAY;CAC1B,EAAA,eAAe,EAAE,MAAM;CACvB,EAAA,cAAc,EAAE;CACjB,CAAC;CAED,MAAMC,MAAM,GAAG,IAAIC,KAAK,CAAC,IAAIC,eAAe,CAACC,MAAM,CAACC,QAAQ,CAACC,MAAM,CAAC,EAAE;GACrEC,GAAG,EAAEA,CAACC,YAAY,EAAEC,IAAI,KAAKD,YAAY,CAACD,GAAG,CAACE,IAAI;CACnD,CAAC,CAAC;CAEF,KAAK,MAAMC,GAAG,IAAIV,WAAW,EAAE;CAC9B,EAAA,MAAMW,KAAK,GAAGV,MAAM,CAACS,GAAG,CAAC;CACzB,EAAA,IAAIE,aAAa;GAEjB,IAAI,CAAC,CAACD,KAAK,EAAE;KACZE,YAAY,CAACC,OAAO,CAAC,SAAS,GAAGJ,GAAG,EAAEC,KAAK,CAAC;CAC5CC,IAAAA,aAAa,GAAGD,KAAK;CACtB,EAAA,CAAC,MAAM;KACN,MAAMI,WAAW,GAAGF,YAAY,CAACG,OAAO,CAAC,SAAS,GAAGN,GAAG,CAAC;KACzDE,aAAa,GAAGG,WAAW,GAAGA,WAAW,GAAGf,WAAW,CAACU,GAAG,CAAC;CAC7D,EAAA;CAEA,EAAA,IAAIE,aAAa,KAAKZ,WAAW,CAACU,GAAG,CAAC,EAAE;KACvCO,QAAQ,CAACC,eAAe,CAACC,YAAY,CAAC,UAAU,GAAGT,GAAG,EAAEE,aAAa,CAAC;CACvE,EAAA,CAAC,MAAM;KACNK,QAAQ,CAACC,eAAe,CAACE,eAAe,CAAC,UAAU,GAAGV,GAAG,CAAC;CAC3D,EAAA;CACD;;;;;;"} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.min.js b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.min.js new file mode 100644 index 000000000..d4d9d866c --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.min.js @@ -0,0 +1,8 @@ +/*! + * Tabler v1.4.0 (https://tabler.io) + * Copyright 2018-2025 The Tabler Authors + * Copyright 2018-2025 codecalm.net Paweł Kuna + * Licensed under MIT (https://github.com/tabler/tabler/blob/master/LICENSE) + */ +!function(e){"function"==typeof define&&define.amd?define(e):e()}(function(){"use strict";const e={theme:"light","theme-base":"gray","theme-font":"sans-serif","theme-primary":"blue","theme-radius":"1"},t=new Proxy(new URLSearchParams(window.location.search),{get:(e,t)=>e.get(t)});for(const n in e){const o=t[n];let a;if(o)localStorage.setItem("tabler-"+n,o),a=o;else{a=localStorage.getItem("tabler-"+n)||e[n]}a!==e[n]?document.documentElement.setAttribute("data-bs-"+n,a):document.documentElement.removeAttribute("data-bs-"+n)}}); +//# sourceMappingURL=tabler-theme.min.js.map \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.min.js.map b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.min.js.map new file mode 100644 index 000000000..2181cfb6e --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler-theme.min.js.map @@ -0,0 +1 @@ +{"version":3,"names":["themeConfig","theme","params","Proxy","URLSearchParams","window","location","search","get","searchParams","prop","key","param","selectedValue","localStorage","setItem","getItem","document","documentElement","setAttribute","removeAttribute"],"sources":["../../js/tabler-theme.js"],"sourcesContent":["/**\n * demo-theme is specifically loaded right after the body and not deferred\n * to ensure we switch to the chosen dark/light theme as fast as possible.\n * This will prevent any flashes of the light theme (default) before switching.\n */\nconst themeConfig = {\n\t\"theme\": \"light\",\n\t\"theme-base\": \"gray\",\n\t\"theme-font\": \"sans-serif\",\n\t\"theme-primary\": \"blue\",\n\t\"theme-radius\": \"1\",\n}\n\nconst params = new Proxy(new URLSearchParams(window.location.search), {\n\tget: (searchParams, prop) => searchParams.get(prop),\n})\n\nfor (const key in themeConfig) {\n\tconst param = params[key]\n\tlet selectedValue\n\n\tif (!!param) {\n\t\tlocalStorage.setItem('tabler-' + key, param)\n\t\tselectedValue = param\n\t} else {\n\t\tconst storedTheme = localStorage.getItem('tabler-' + key)\n\t\tselectedValue = storedTheme ? storedTheme : themeConfig[key]\n\t}\n\n\tif (selectedValue !== themeConfig[key]) {\n\t\tdocument.documentElement.setAttribute('data-bs-' + key, selectedValue)\n\t} else {\n\t\tdocument.documentElement.removeAttribute('data-bs-' + key)\n\t}\n}"],"mappings":";;;;;;0FAKA,MAAMA,EAAc,CACnBC,MAAS,QACT,aAAc,OACd,aAAc,aACd,gBAAiB,OACjB,eAAgB,KAGXC,EAAS,IAAIC,MAAM,IAAIC,gBAAgBC,OAAOC,SAASC,QAAS,CACrEC,IAAKA,CAACC,EAAcC,IAASD,EAAaD,IAAIE,KAG/C,IAAK,MAAMC,KAAOX,EAAa,CAC9B,MAAMY,EAAQV,EAAOS,GACrB,IAAIE,EAEJ,GAAMD,EACLE,aAAaC,QAAQ,UAAYJ,EAAKC,GACtCC,EAAgBD,MACV,CAENC,EADoBC,aAAaE,QAAQ,UAAYL,IACTX,EAAYW,EACzD,CAEIE,IAAkBb,EAAYW,GACjCM,SAASC,gBAAgBC,aAAa,WAAaR,EAAKE,GAExDI,SAASC,gBAAgBE,gBAAgB,WAAaT,EAExD,C","ignoreList":[]} \ No newline at end of file diff --git a/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler.esm.js b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler.esm.js new file mode 100644 index 000000000..cb77e5dac --- /dev/null +++ b/src/UI/FluentCMS.Web.UI.Admin/wwwroot/js/tabler.esm.js @@ -0,0 +1,6249 @@ +/*! + * Tabler v1.4.0 (https://tabler.io) + * Copyright 2018-2025 The Tabler Authors + * Copyright 2018-2025 codecalm.net Paweł Kuna + * Licensed under MIT (https://github.com/tabler/tabler/blob/master/LICENSE) + */ +// Autosize plugin +const elements$1 = document.querySelectorAll('[data-bs-toggle="autosize"]'); +if (elements$1.length) { + elements$1.forEach(function (element) { + window.autosize && window.autosize(element); + }); +} + +const elements = document.querySelectorAll('[data-countup]'); +if (elements.length) { + elements.forEach(function (element) { + let options = {}; + try { + const dataOptions = element.getAttribute('data-countup') ? JSON.parse(element.getAttribute('data-countup')) : {}; + options = Object.assign({ + 'enableScrollSpy': true + }, dataOptions); + } catch (error) {} + const value = parseInt(element.innerHTML, 10); + if (window.countUp && window.countUp.CountUp) { + const countUp = new window.countUp.CountUp(element, value, options); + if (!countUp.error) { + countUp.start(); + } + } + }); +} + +// Input mask plugin + +var maskElementList = [].slice.call(document.querySelectorAll('[data-mask]')); +maskElementList.map(function (maskEl) { + window.IMask && new window.IMask(maskEl, { + mask: maskEl.dataset.mask, + lazy: maskEl.dataset['mask-visible'] === 'true' + }); +}); + +var top = 'top'; +var bottom = 'bottom'; +var right = 'right'; +var left = 'left'; +var auto = 'auto'; +var basePlacements = [top, bottom, right, left]; +var start = 'start'; +var end = 'end'; +var clippingParents = 'clippingParents'; +var viewport = 'viewport'; +var popper = 'popper'; +var reference = 'reference'; +var variationPlacements = /*#__PURE__*/basePlacements.reduce(function (acc, placement) { + return acc.concat([placement + "-" + start, placement + "-" + end]); +}, []); +var placements = /*#__PURE__*/[].concat(basePlacements, [auto]).reduce(function (acc, placement) { + return acc.concat([placement, placement + "-" + start, placement + "-" + end]); +}, []); // modifiers that need to read the DOM + +var beforeRead = 'beforeRead'; +var read = 'read'; +var afterRead = 'afterRead'; // pure-logic modifiers + +var beforeMain = 'beforeMain'; +var main = 'main'; +var afterMain = 'afterMain'; // modifier with the purpose to write to the DOM (or write into a framework state) + +var beforeWrite = 'beforeWrite'; +var write = 'write'; +var afterWrite = 'afterWrite'; +var modifierPhases = [beforeRead, read, afterRead, beforeMain, main, afterMain, beforeWrite, write, afterWrite]; + +function getNodeName(element) { + return element ? (element.nodeName || '').toLowerCase() : null; +} + +function getWindow(node) { + if (node == null) { + return window; + } + if (node.toString() !== '[object Window]') { + var ownerDocument = node.ownerDocument; + return ownerDocument ? ownerDocument.defaultView || window : window; + } + return node; +} + +function isElement$1(node) { + var OwnElement = getWindow(node).Element; + return node instanceof OwnElement || node instanceof Element; +} +function isHTMLElement(node) { + var OwnElement = getWindow(node).HTMLElement; + return node instanceof OwnElement || node instanceof HTMLElement; +} +function isShadowRoot(node) { + // IE 11 has no ShadowRoot + if (typeof ShadowRoot === 'undefined') { + return false; + } + var OwnElement = getWindow(node).ShadowRoot; + return node instanceof OwnElement || node instanceof ShadowRoot; +} + +// and applies them to the HTMLElements such as popper and arrow + +function applyStyles(_ref) { + var state = _ref.state; + Object.keys(state.elements).forEach(function (name) { + var style = state.styles[name] || {}; + var attributes = state.attributes[name] || {}; + var element = state.elements[name]; // arrow is optional + virtual elements + + if (!isHTMLElement(element) || !getNodeName(element)) { + return; + } // Flow doesn't support to extend this property, but it's the most + // effective way to apply styles to an HTMLElement + // $FlowFixMe[cannot-write] + + Object.assign(element.style, style); + Object.keys(attributes).forEach(function (name) { + var value = attributes[name]; + if (value === false) { + element.removeAttribute(name); + } else { + element.setAttribute(name, value === true ? '' : value); + } + }); + }); +} +function effect$2(_ref2) { + var state = _ref2.state; + var initialStyles = { + popper: { + position: state.options.strategy, + left: '0', + top: '0', + margin: '0' + }, + arrow: { + position: 'absolute' + }, + reference: {} + }; + Object.assign(state.elements.popper.style, initialStyles.popper); + state.styles = initialStyles; + if (state.elements.arrow) { + Object.assign(state.elements.arrow.style, initialStyles.arrow); + } + return function () { + Object.keys(state.elements).forEach(function (name) { + var element = state.elements[name]; + var attributes = state.attributes[name] || {}; + var styleProperties = Object.keys(state.styles.hasOwnProperty(name) ? state.styles[name] : initialStyles[name]); // Set all values to an empty string to unset them + + var style = styleProperties.reduce(function (style, property) { + style[property] = ''; + return style; + }, {}); // arrow is optional + virtual elements + + if (!isHTMLElement(element) || !getNodeName(element)) { + return; + } + Object.assign(element.style, style); + Object.keys(attributes).forEach(function (attribute) { + element.removeAttribute(attribute); + }); + }); + }; +} // eslint-disable-next-line import/no-unused-modules + +const applyStyles$1 = { + name: 'applyStyles', + enabled: true, + phase: 'write', + fn: applyStyles, + effect: effect$2, + requires: ['computeStyles'] +}; + +function getBasePlacement(placement) { + return placement.split('-')[0]; +} + +var max = Math.max; +var min = Math.min; +var round = Math.round; + +function getUAString() { + var uaData = navigator.userAgentData; + if (uaData != null && uaData.brands && Array.isArray(uaData.brands)) { + return uaData.brands.map(function (item) { + return item.brand + "/" + item.version; + }).join(' '); + } + return navigator.userAgent; +} + +function isLayoutViewport() { + return !/^((?!chrome|android).)*safari/i.test(getUAString()); +} + +function getBoundingClientRect(element, includeScale, isFixedStrategy) { + if (includeScale === void 0) { + includeScale = false; + } + if (isFixedStrategy === void 0) { + isFixedStrategy = false; + } + var clientRect = element.getBoundingClientRect(); + var scaleX = 1; + var scaleY = 1; + if (includeScale && isHTMLElement(element)) { + scaleX = element.offsetWidth > 0 ? round(clientRect.width) / element.offsetWidth || 1 : 1; + scaleY = element.offsetHeight > 0 ? round(clientRect.height) / element.offsetHeight || 1 : 1; + } + var _ref = isElement$1(element) ? getWindow(element) : window, + visualViewport = _ref.visualViewport; + var addVisualOffsets = !isLayoutViewport() && isFixedStrategy; + var x = (clientRect.left + (addVisualOffsets && visualViewport ? visualViewport.offsetLeft : 0)) / scaleX; + var y = (clientRect.top + (addVisualOffsets && visualViewport ? visualViewport.offsetTop : 0)) / scaleY; + var width = clientRect.width / scaleX; + var height = clientRect.height / scaleY; + return { + width: width, + height: height, + top: y, + right: x + width, + bottom: y + height, + left: x, + x: x, + y: y + }; +} + +// means it doesn't take into account transforms. + +function getLayoutRect(element) { + var clientRect = getBoundingClientRect(element); // Use the clientRect sizes if it's not been transformed. + // Fixes https://github.com/popperjs/popper-core/issues/1223 + + var width = element.offsetWidth; + var height = element.offsetHeight; + if (Math.abs(clientRect.width - width) <= 1) { + width = clientRect.width; + } + if (Math.abs(clientRect.height - height) <= 1) { + height = clientRect.height; + } + return { + x: element.offsetLeft, + y: element.offsetTop, + width: width, + height: height + }; +} + +function contains(parent, child) { + var rootNode = child.getRootNode && child.getRootNode(); // First, attempt with faster native method + + if (parent.contains(child)) { + return true; + } // then fallback to custom implementation with Shadow DOM support + else if (rootNode && isShadowRoot(rootNode)) { + var next = child; + do { + if (next && parent.isSameNode(next)) { + return true; + } // $FlowFixMe[prop-missing]: need a better way to handle this... + + next = next.parentNode || next.host; + } while (next); + } // Give up, the result is false + + return false; +} + +function getComputedStyle$1(element) { + return getWindow(element).getComputedStyle(element); +} + +function isTableElement(element) { + return ['table', 'td', 'th'].indexOf(getNodeName(element)) >= 0; +} + +function getDocumentElement(element) { + // $FlowFixMe[incompatible-return]: assume body is always available + return ((isElement$1(element) ? element.ownerDocument : + // $FlowFixMe[prop-missing] + element.document) || window.document).documentElement; +} + +function getParentNode(element) { + if (getNodeName(element) === 'html') { + return element; + } + return ( + // this is a quicker (but less type safe) way to save quite some bytes from the bundle + // $FlowFixMe[incompatible-return] + // $FlowFixMe[prop-missing] + element.assignedSlot || + // step into the shadow DOM of the parent of a slotted node + element.parentNode || ( + // DOM Element detected + isShadowRoot(element) ? element.host : null) || + // ShadowRoot detected + // $FlowFixMe[incompatible-call]: HTMLElement is a Node + getDocumentElement(element) // fallback + ); +} + +function getTrueOffsetParent(element) { + if (!isHTMLElement(element) || + // https://github.com/popperjs/popper-core/issues/837 + getComputedStyle$1(element).position === 'fixed') { + return null; + } + return element.offsetParent; +} // `.offsetParent` reports `null` for fixed elements, while absolute elements +// return the containing block + +function getContainingBlock(element) { + var isFirefox = /firefox/i.test(getUAString()); + var isIE = /Trident/i.test(getUAString()); + if (isIE && isHTMLElement(element)) { + // In IE 9, 10 and 11 fixed elements containing block is always established by the viewport + var elementCss = getComputedStyle$1(element); + if (elementCss.position === 'fixed') { + return null; + } + } + var currentNode = getParentNode(element); + if (isShadowRoot(currentNode)) { + currentNode = currentNode.host; + } + while (isHTMLElement(currentNode) && ['html', 'body'].indexOf(getNodeName(currentNode)) < 0) { + var css = getComputedStyle$1(currentNode); // This is non-exhaustive but covers the most common CSS properties that + // create a containing block. + // https://developer.mozilla.org/en-US/docs/Web/CSS/Containing_block#identifying_the_containing_block + + if (css.transform !== 'none' || css.perspective !== 'none' || css.contain === 'paint' || ['transform', 'perspective'].indexOf(css.willChange) !== -1 || isFirefox && css.willChange === 'filter' || isFirefox && css.filter && css.filter !== 'none') { + return currentNode; + } else { + currentNode = currentNode.parentNode; + } + } + return null; +} // Gets the closest ancestor positioned element. Handles some edge cases, +// such as table ancestors and cross browser bugs. + +function getOffsetParent(element) { + var window = getWindow(element); + var offsetParent = getTrueOffsetParent(element); + while (offsetParent && isTableElement(offsetParent) && getComputedStyle$1(offsetParent).position === 'static') { + offsetParent = getTrueOffsetParent(offsetParent); + } + if (offsetParent && (getNodeName(offsetParent) === 'html' || getNodeName(offsetParent) === 'body' && getComputedStyle$1(offsetParent).position === 'static')) { + return window; + } + return offsetParent || getContainingBlock(element) || window; +} + +function getMainAxisFromPlacement(placement) { + return ['top', 'bottom'].indexOf(placement) >= 0 ? 'x' : 'y'; +} + +function within(min$1, value, max$1) { + return max(min$1, min(value, max$1)); +} +function withinMaxClamp(min, value, max) { + var v = within(min, value, max); + return v > max ? max : v; +} + +function getFreshSideObject() { + return { + top: 0, + right: 0, + bottom: 0, + left: 0 + }; +} + +function mergePaddingObject(paddingObject) { + return Object.assign({}, getFreshSideObject(), paddingObject); +} + +function expandToHashMap(value, keys) { + return keys.reduce(function (hashMap, key) { + hashMap[key] = value; + return hashMap; + }, {}); +} + +var toPaddingObject = function toPaddingObject(padding, state) { + padding = typeof padding === 'function' ? padding(Object.assign({}, state.rects, { + placement: state.placement + })) : padding; + return mergePaddingObject(typeof padding !== 'number' ? padding : expandToHashMap(padding, basePlacements)); +}; +function arrow(_ref) { + var _state$modifiersData$; + var state = _ref.state, + name = _ref.name, + options = _ref.options; + var arrowElement = state.elements.arrow; + var popperOffsets = state.modifiersData.popperOffsets; + var basePlacement = getBasePlacement(state.placement); + var axis = getMainAxisFromPlacement(basePlacement); + var isVertical = [left, right].indexOf(basePlacement) >= 0; + var len = isVertical ? 'height' : 'width'; + if (!arrowElement || !popperOffsets) { + return; + } + var paddingObject = toPaddingObject(options.padding, state); + var arrowRect = getLayoutRect(arrowElement); + var minProp = axis === 'y' ? top : left; + var maxProp = axis === 'y' ? bottom : right; + var endDiff = state.rects.reference[len] + state.rects.reference[axis] - popperOffsets[axis] - state.rects.popper[len]; + var startDiff = popperOffsets[axis] - state.rects.reference[axis]; + var arrowOffsetParent = getOffsetParent(arrowElement); + var clientSize = arrowOffsetParent ? axis === 'y' ? arrowOffsetParent.clientHeight || 0 : arrowOffsetParent.clientWidth || 0 : 0; + var centerToReference = endDiff / 2 - startDiff / 2; // Make sure the arrow doesn't overflow the popper if the center point is + // outside of the popper bounds + + var min = paddingObject[minProp]; + var max = clientSize - arrowRect[len] - paddingObject[maxProp]; + var center = clientSize / 2 - arrowRect[len] / 2 + centerToReference; + var offset = within(min, center, max); // Prevents breaking syntax highlighting... + + var axisProp = axis; + state.modifiersData[name] = (_state$modifiersData$ = {}, _state$modifiersData$[axisProp] = offset, _state$modifiersData$.centerOffset = offset - center, _state$modifiersData$); +} +function effect$1(_ref2) { + var state = _ref2.state, + options = _ref2.options; + var _options$element = options.element, + arrowElement = _options$element === void 0 ? '[data-popper-arrow]' : _options$element; + if (arrowElement == null) { + return; + } // CSS selector + + if (typeof arrowElement === 'string') { + arrowElement = state.elements.popper.querySelector(arrowElement); + if (!arrowElement) { + return; + } + } + if (!contains(state.elements.popper, arrowElement)) { + return; + } + state.elements.arrow = arrowElement; +} // eslint-disable-next-line import/no-unused-modules + +const arrow$1 = { + name: 'arrow', + enabled: true, + phase: 'main', + fn: arrow, + effect: effect$1, + requires: ['popperOffsets'], + requiresIfExists: ['preventOverflow'] +}; + +function getVariation(placement) { + return placement.split('-')[1]; +} + +var unsetSides = { + top: 'auto', + right: 'auto', + bottom: 'auto', + left: 'auto' +}; // Round the offsets to the nearest suitable subpixel based on the DPR. +// Zooming can change the DPR, but it seems to report a value that will +// cleanly divide the values into the appropriate subpixels. + +function roundOffsetsByDPR(_ref, win) { + var x = _ref.x, + y = _ref.y; + var dpr = win.devicePixelRatio || 1; + return { + x: round(x * dpr) / dpr || 0, + y: round(y * dpr) / dpr || 0 + }; +} +function mapToStyles(_ref2) { + var _Object$assign2; + var popper = _ref2.popper, + popperRect = _ref2.popperRect, + placement = _ref2.placement, + variation = _ref2.variation, + offsets = _ref2.offsets, + position = _ref2.position, + gpuAcceleration = _ref2.gpuAcceleration, + adaptive = _ref2.adaptive, + roundOffsets = _ref2.roundOffsets, + isFixed = _ref2.isFixed; + var _offsets$x = offsets.x, + x = _offsets$x === void 0 ? 0 : _offsets$x, + _offsets$y = offsets.y, + y = _offsets$y === void 0 ? 0 : _offsets$y; + var _ref3 = typeof roundOffsets === 'function' ? roundOffsets({ + x: x, + y: y + }) : { + x: x, + y: y + }; + x = _ref3.x; + y = _ref3.y; + var hasX = offsets.hasOwnProperty('x'); + var hasY = offsets.hasOwnProperty('y'); + var sideX = left; + var sideY = top; + var win = window; + if (adaptive) { + var offsetParent = getOffsetParent(popper); + var heightProp = 'clientHeight'; + var widthProp = 'clientWidth'; + if (offsetParent === getWindow(popper)) { + offsetParent = getDocumentElement(popper); + if (getComputedStyle$1(offsetParent).position !== 'static' && position === 'absolute') { + heightProp = 'scrollHeight'; + widthProp = 'scrollWidth'; + } + } // $FlowFixMe[incompatible-cast]: force type refinement, we compare offsetParent with window above, but Flow doesn't detect it + + offsetParent = offsetParent; + if (placement === top || (placement === left || placement === right) && variation === end) { + sideY = bottom; + var offsetY = isFixed && offsetParent === win && win.visualViewport ? win.visualViewport.height : + // $FlowFixMe[prop-missing] + offsetParent[heightProp]; + y -= offsetY - popperRect.height; + y *= gpuAcceleration ? 1 : -1; + } + if (placement === left || (placement === top || placement === bottom) && variation === end) { + sideX = right; + var offsetX = isFixed && offsetParent === win && win.visualViewport ? win.visualViewport.width : + // $FlowFixMe[prop-missing] + offsetParent[widthProp]; + x -= offsetX - popperRect.width; + x *= gpuAcceleration ? 1 : -1; + } + } + var commonStyles = Object.assign({ + position: position + }, adaptive && unsetSides); + var _ref4 = roundOffsets === true ? roundOffsetsByDPR({ + x: x, + y: y + }, getWindow(popper)) : { + x: x, + y: y + }; + x = _ref4.x; + y = _ref4.y; + if (gpuAcceleration) { + var _Object$assign; + return Object.assign({}, commonStyles, (_Object$assign = {}, _Object$assign[sideY] = hasY ? '0' : '', _Object$assign[sideX] = hasX ? '0' : '', _Object$assign.transform = (win.devicePixelRatio || 1) <= 1 ? "translate(" + x + "px, " + y + "px)" : "translate3d(" + x + "px, " + y + "px, 0)", _Object$assign)); + } + return Object.assign({}, commonStyles, (_Object$assign2 = {}, _Object$assign2[sideY] = hasY ? y + "px" : '', _Object$assign2[sideX] = hasX ? x + "px" : '', _Object$assign2.transform = '', _Object$assign2)); +} +function computeStyles(_ref5) { + var state = _ref5.state, + options = _ref5.options; + var _options$gpuAccelerat = options.gpuAcceleration, + gpuAcceleration = _options$gpuAccelerat === void 0 ? true : _options$gpuAccelerat, + _options$adaptive = options.adaptive, + adaptive = _options$adaptive === void 0 ? true : _options$adaptive, + _options$roundOffsets = options.roundOffsets, + roundOffsets = _options$roundOffsets === void 0 ? true : _options$roundOffsets; + var commonStyles = { + placement: getBasePlacement(state.placement), + variation: getVariation(state.placement), + popper: state.elements.popper, + popperRect: state.rects.popper, + gpuAcceleration: gpuAcceleration, + isFixed: state.options.strategy === 'fixed' + }; + if (state.modifiersData.popperOffsets != null) { + state.styles.popper = Object.assign({}, state.styles.popper, mapToStyles(Object.assign({}, commonStyles, { + offsets: state.modifiersData.popperOffsets, + position: state.options.strategy, + adaptive: adaptive, + roundOffsets: roundOffsets + }))); + } + if (state.modifiersData.arrow != null) { + state.styles.arrow = Object.assign({}, state.styles.arrow, mapToStyles(Object.assign({}, commonStyles, { + offsets: state.modifiersData.arrow, + position: 'absolute', + adaptive: false, + roundOffsets: roundOffsets + }))); + } + state.attributes.popper = Object.assign({}, state.attributes.popper, { + 'data-popper-placement': state.placement + }); +} // eslint-disable-next-line import/no-unused-modules + +const computeStyles$1 = { + name: 'computeStyles', + enabled: true, + phase: 'beforeWrite', + fn: computeStyles, + data: {} +}; + +var passive = { + passive: true +}; +function effect(_ref) { + var state = _ref.state, + instance = _ref.instance, + options = _ref.options; + var _options$scroll = options.scroll, + scroll = _options$scroll === void 0 ? true : _options$scroll, + _options$resize = options.resize, + resize = _options$resize === void 0 ? true : _options$resize; + var window = getWindow(state.elements.popper); + var scrollParents = [].concat(state.scrollParents.reference, state.scrollParents.popper); + if (scroll) { + scrollParents.forEach(function (scrollParent) { + scrollParent.addEventListener('scroll', instance.update, passive); + }); + } + if (resize) { + window.addEventListener('resize', instance.update, passive); + } + return function () { + if (scroll) { + scrollParents.forEach(function (scrollParent) { + scrollParent.removeEventListener('scroll', instance.update, passive); + }); + } + if (resize) { + window.removeEventListener('resize', instance.update, passive); + } + }; +} // eslint-disable-next-line import/no-unused-modules + +const eventListeners = { + name: 'eventListeners', + enabled: true, + phase: 'write', + fn: function fn() {}, + effect: effect, + data: {} +}; + +var hash$1 = { + left: 'right', + right: 'left', + bottom: 'top', + top: 'bottom' +}; +function getOppositePlacement(placement) { + return placement.replace(/left|right|bottom|top/g, function (matched) { + return hash$1[matched]; + }); +} + +var hash = { + start: 'end', + end: 'start' +}; +function getOppositeVariationPlacement(placement) { + return placement.replace(/start|end/g, function (matched) { + return hash[matched]; + }); +} + +function getWindowScroll(node) { + var win = getWindow(node); + var scrollLeft = win.pageXOffset; + var scrollTop = win.pageYOffset; + return { + scrollLeft: scrollLeft, + scrollTop: scrollTop + }; +} + +function getWindowScrollBarX(element) { + // If has a CSS width greater than the viewport, then this will be + // incorrect for RTL. + // Popper 1 is broken in this case and never had a bug report so let's assume + // it's not an issue. I don't think anyone ever specifies width on + // anyway. + // Browsers where the left scrollbar doesn't cause an issue report `0` for + // this (e.g. Edge 2019, IE11, Safari) + return getBoundingClientRect(getDocumentElement(element)).left + getWindowScroll(element).scrollLeft; +} + +function getViewportRect(element, strategy) { + var win = getWindow(element); + var html = getDocumentElement(element); + var visualViewport = win.visualViewport; + var width = html.clientWidth; + var height = html.clientHeight; + var x = 0; + var y = 0; + if (visualViewport) { + width = visualViewport.width; + height = visualViewport.height; + var layoutViewport = isLayoutViewport(); + if (layoutViewport || !layoutViewport && strategy === 'fixed') { + x = visualViewport.offsetLeft; + y = visualViewport.offsetTop; + } + } + return { + width: width, + height: height, + x: x + getWindowScrollBarX(element), + y: y + }; +} + +// of the `` and `` rect bounds if horizontally scrollable + +function getDocumentRect(element) { + var _element$ownerDocumen; + var html = getDocumentElement(element); + var winScroll = getWindowScroll(element); + var body = (_element$ownerDocumen = element.ownerDocument) == null ? void 0 : _element$ownerDocumen.body; + var width = max(html.scrollWidth, html.clientWidth, body ? body.scrollWidth : 0, body ? body.clientWidth : 0); + var height = max(html.scrollHeight, html.clientHeight, body ? body.scrollHeight : 0, body ? body.clientHeight : 0); + var x = -winScroll.scrollLeft + getWindowScrollBarX(element); + var y = -winScroll.scrollTop; + if (getComputedStyle$1(body || html).direction === 'rtl') { + x += max(html.clientWidth, body ? body.clientWidth : 0) - width; + } + return { + width: width, + height: height, + x: x, + y: y + }; +} + +function isScrollParent(element) { + // Firefox wants us to check `-x` and `-y` variations as well + var _getComputedStyle = getComputedStyle$1(element), + overflow = _getComputedStyle.overflow, + overflowX = _getComputedStyle.overflowX, + overflowY = _getComputedStyle.overflowY; + return /auto|scroll|overlay|hidden/.test(overflow + overflowY + overflowX); +} + +function getScrollParent(node) { + if (['html', 'body', '#document'].indexOf(getNodeName(node)) >= 0) { + // $FlowFixMe[incompatible-return]: assume body is always available + return node.ownerDocument.body; + } + if (isHTMLElement(node) && isScrollParent(node)) { + return node; + } + return getScrollParent(getParentNode(node)); +} + +/* +given a DOM element, return the list of all scroll parents, up the list of ancesors +until we get to the top window object. This list is what we attach scroll listeners +to, because if any of these parent elements scroll, we'll need to re-calculate the +reference element's position. +*/ + +function listScrollParents(element, list) { + var _element$ownerDocumen; + if (list === void 0) { + list = []; + } + var scrollParent = getScrollParent(element); + var isBody = scrollParent === ((_element$ownerDocumen = element.ownerDocument) == null ? void 0 : _element$ownerDocumen.body); + var win = getWindow(scrollParent); + var target = isBody ? [win].concat(win.visualViewport || [], isScrollParent(scrollParent) ? scrollParent : []) : scrollParent; + var updatedList = list.concat(target); + return isBody ? updatedList : + // $FlowFixMe[incompatible-call]: isBody tells us target will be an HTMLElement here + updatedList.concat(listScrollParents(getParentNode(target))); +} + +function rectToClientRect(rect) { + return Object.assign({}, rect, { + left: rect.x, + top: rect.y, + right: rect.x + rect.width, + bottom: rect.y + rect.height + }); +} + +function getInnerBoundingClientRect(element, strategy) { + var rect = getBoundingClientRect(element, false, strategy === 'fixed'); + rect.top = rect.top + element.clientTop; + rect.left = rect.left + element.clientLeft; + rect.bottom = rect.top + element.clientHeight; + rect.right = rect.left + element.clientWidth; + rect.width = element.clientWidth; + rect.height = element.clientHeight; + rect.x = rect.left; + rect.y = rect.top; + return rect; +} +function getClientRectFromMixedType(element, clippingParent, strategy) { + return clippingParent === viewport ? rectToClientRect(getViewportRect(element, strategy)) : isElement$1(clippingParent) ? getInnerBoundingClientRect(clippingParent, strategy) : rectToClientRect(getDocumentRect(getDocumentElement(element))); +} // A "clipping parent" is an overflowable container with the characteristic of +// clipping (or hiding) overflowing elements with a position different from +// `initial` + +function getClippingParents(element) { + var clippingParents = listScrollParents(getParentNode(element)); + var canEscapeClipping = ['absolute', 'fixed'].indexOf(getComputedStyle$1(element).position) >= 0; + var clipperElement = canEscapeClipping && isHTMLElement(element) ? getOffsetParent(element) : element; + if (!isElement$1(clipperElement)) { + return []; + } // $FlowFixMe[incompatible-return]: https://github.com/facebook/flow/issues/1414 + + return clippingParents.filter(function (clippingParent) { + return isElement$1(clippingParent) && contains(clippingParent, clipperElement) && getNodeName(clippingParent) !== 'body'; + }); +} // Gets the maximum area that the element is visible in due to any number of +// clipping parents + +function getClippingRect(element, boundary, rootBoundary, strategy) { + var mainClippingParents = boundary === 'clippingParents' ? getClippingParents(element) : [].concat(boundary); + var clippingParents = [].concat(mainClippingParents, [rootBoundary]); + var firstClippingParent = clippingParents[0]; + var clippingRect = clippingParents.reduce(function (accRect, clippingParent) { + var rect = getClientRectFromMixedType(element, clippingParent, strategy); + accRect.top = max(rect.top, accRect.top); + accRect.right = min(rect.right, accRect.right); + accRect.bottom = min(rect.bottom, accRect.bottom); + accRect.left = max(rect.left, accRect.left); + return accRect; + }, getClientRectFromMixedType(element, firstClippingParent, strategy)); + clippingRect.width = clippingRect.right - clippingRect.left; + clippingRect.height = clippingRect.bottom - clippingRect.top; + clippingRect.x = clippingRect.left; + clippingRect.y = clippingRect.top; + return clippingRect; +} + +function computeOffsets(_ref) { + var reference = _ref.reference, + element = _ref.element, + placement = _ref.placement; + var basePlacement = placement ? getBasePlacement(placement) : null; + var variation = placement ? getVariation(placement) : null; + var commonX = reference.x + reference.width / 2 - element.width / 2; + var commonY = reference.y + reference.height / 2 - element.height / 2; + var offsets; + switch (basePlacement) { + case top: + offsets = { + x: commonX, + y: reference.y - element.height + }; + break; + case bottom: + offsets = { + x: commonX, + y: reference.y + reference.height + }; + break; + case right: + offsets = { + x: reference.x + reference.width, + y: commonY + }; + break; + case left: + offsets = { + x: reference.x - element.width, + y: commonY + }; + break; + default: + offsets = { + x: reference.x, + y: reference.y + }; + } + var mainAxis = basePlacement ? getMainAxisFromPlacement(basePlacement) : null; + if (mainAxis != null) { + var len = mainAxis === 'y' ? 'height' : 'width'; + switch (variation) { + case start: + offsets[mainAxis] = offsets[mainAxis] - (reference[len] / 2 - element[len] / 2); + break; + case end: + offsets[mainAxis] = offsets[mainAxis] + (reference[len] / 2 - element[len] / 2); + break; + } + } + return offsets; +} + +function detectOverflow(state, options) { + if (options === void 0) { + options = {}; + } + var _options = options, + _options$placement = _options.placement, + placement = _options$placement === void 0 ? state.placement : _options$placement, + _options$strategy = _options.strategy, + strategy = _options$strategy === void 0 ? state.strategy : _options$strategy, + _options$boundary = _options.boundary, + boundary = _options$boundary === void 0 ? clippingParents : _options$boundary, + _options$rootBoundary = _options.rootBoundary, + rootBoundary = _options$rootBoundary === void 0 ? viewport : _options$rootBoundary, + _options$elementConte = _options.elementContext, + elementContext = _options$elementConte === void 0 ? popper : _options$elementConte, + _options$altBoundary = _options.altBoundary, + altBoundary = _options$altBoundary === void 0 ? false : _options$altBoundary, + _options$padding = _options.padding, + padding = _options$padding === void 0 ? 0 : _options$padding; + var paddingObject = mergePaddingObject(typeof padding !== 'number' ? padding : expandToHashMap(padding, basePlacements)); + var altContext = elementContext === popper ? reference : popper; + var popperRect = state.rects.popper; + var element = state.elements[altBoundary ? altContext : elementContext]; + var clippingClientRect = getClippingRect(isElement$1(element) ? element : element.contextElement || getDocumentElement(state.elements.popper), boundary, rootBoundary, strategy); + var referenceClientRect = getBoundingClientRect(state.elements.reference); + var popperOffsets = computeOffsets({ + reference: referenceClientRect, + element: popperRect, + placement: placement + }); + var popperClientRect = rectToClientRect(Object.assign({}, popperRect, popperOffsets)); + var elementClientRect = elementContext === popper ? popperClientRect : referenceClientRect; // positive = overflowing the clipping rect + // 0 or negative = within the clipping rect + + var overflowOffsets = { + top: clippingClientRect.top - elementClientRect.top + paddingObject.top, + bottom: elementClientRect.bottom - clippingClientRect.bottom + paddingObject.bottom, + left: clippingClientRect.left - elementClientRect.left + paddingObject.left, + right: elementClientRect.right - clippingClientRect.right + paddingObject.right + }; + var offsetData = state.modifiersData.offset; // Offsets can be applied only to the popper element + + if (elementContext === popper && offsetData) { + var offset = offsetData[placement]; + Object.keys(overflowOffsets).forEach(function (key) { + var multiply = [right, bottom].indexOf(key) >= 0 ? 1 : -1; + var axis = [top, bottom].indexOf(key) >= 0 ? 'y' : 'x'; + overflowOffsets[key] += offset[axis] * multiply; + }); + } + return overflowOffsets; +} + +function computeAutoPlacement(state, options) { + if (options === void 0) { + options = {}; + } + var _options = options, + placement = _options.placement, + boundary = _options.boundary, + rootBoundary = _options.rootBoundary, + padding = _options.padding, + flipVariations = _options.flipVariations, + _options$allowedAutoP = _options.allowedAutoPlacements, + allowedAutoPlacements = _options$allowedAutoP === void 0 ? placements : _options$allowedAutoP; + var variation = getVariation(placement); + var placements$1 = variation ? flipVariations ? variationPlacements : variationPlacements.filter(function (placement) { + return getVariation(placement) === variation; + }) : basePlacements; + var allowedPlacements = placements$1.filter(function (placement) { + return allowedAutoPlacements.indexOf(placement) >= 0; + }); + if (allowedPlacements.length === 0) { + allowedPlacements = placements$1; + } // $FlowFixMe[incompatible-type]: Flow seems to have problems with two array unions... + + var overflows = allowedPlacements.reduce(function (acc, placement) { + acc[placement] = detectOverflow(state, { + placement: placement, + boundary: boundary, + rootBoundary: rootBoundary, + padding: padding + })[getBasePlacement(placement)]; + return acc; + }, {}); + return Object.keys(overflows).sort(function (a, b) { + return overflows[a] - overflows[b]; + }); +} + +function getExpandedFallbackPlacements(placement) { + if (getBasePlacement(placement) === auto) { + return []; + } + var oppositePlacement = getOppositePlacement(placement); + return [getOppositeVariationPlacement(placement), oppositePlacement, getOppositeVariationPlacement(oppositePlacement)]; +} +function flip(_ref) { + var state = _ref.state, + options = _ref.options, + name = _ref.name; + if (state.modifiersData[name]._skip) { + return; + } + var _options$mainAxis = options.mainAxis, + checkMainAxis = _options$mainAxis === void 0 ? true : _options$mainAxis, + _options$altAxis = options.altAxis, + checkAltAxis = _options$altAxis === void 0 ? true : _options$altAxis, + specifiedFallbackPlacements = options.fallbackPlacements, + padding = options.padding, + boundary = options.boundary, + rootBoundary = options.rootBoundary, + altBoundary = options.altBoundary, + _options$flipVariatio = options.flipVariations, + flipVariations = _options$flipVariatio === void 0 ? true : _options$flipVariatio, + allowedAutoPlacements = options.allowedAutoPlacements; + var preferredPlacement = state.options.placement; + var basePlacement = getBasePlacement(preferredPlacement); + var isBasePlacement = basePlacement === preferredPlacement; + var fallbackPlacements = specifiedFallbackPlacements || (isBasePlacement || !flipVariations ? [getOppositePlacement(preferredPlacement)] : getExpandedFallbackPlacements(preferredPlacement)); + var placements = [preferredPlacement].concat(fallbackPlacements).reduce(function (acc, placement) { + return acc.concat(getBasePlacement(placement) === auto ? computeAutoPlacement(state, { + placement: placement, + boundary: boundary, + rootBoundary: rootBoundary, + padding: padding, + flipVariations: flipVariations, + allowedAutoPlacements: allowedAutoPlacements + }) : placement); + }, []); + var referenceRect = state.rects.reference; + var popperRect = state.rects.popper; + var checksMap = new Map(); + var makeFallbackChecks = true; + var firstFittingPlacement = placements[0]; + for (var i = 0; i < placements.length; i++) { + var placement = placements[i]; + var _basePlacement = getBasePlacement(placement); + var isStartVariation = getVariation(placement) === start; + var isVertical = [top, bottom].indexOf(_basePlacement) >= 0; + var len = isVertical ? 'width' : 'height'; + var overflow = detectOverflow(state, { + placement: placement, + boundary: boundary, + rootBoundary: rootBoundary, + altBoundary: altBoundary, + padding: padding + }); + var mainVariationSide = isVertical ? isStartVariation ? right : left : isStartVariation ? bottom : top; + if (referenceRect[len] > popperRect[len]) { + mainVariationSide = getOppositePlacement(mainVariationSide); + } + var altVariationSide = getOppositePlacement(mainVariationSide); + var checks = []; + if (checkMainAxis) { + checks.push(overflow[_basePlacement] <= 0); + } + if (checkAltAxis) { + checks.push(overflow[mainVariationSide] <= 0, overflow[altVariationSide] <= 0); + } + if (checks.every(function (check) { + return check; + })) { + firstFittingPlacement = placement; + makeFallbackChecks = false; + break; + } + checksMap.set(placement, checks); + } + if (makeFallbackChecks) { + // `2` may be desired in some cases – research later + var numberOfChecks = flipVariations ? 3 : 1; + var _loop = function _loop(_i) { + var fittingPlacement = placements.find(function (placement) { + var checks = checksMap.get(placement); + if (checks) { + return checks.slice(0, _i).every(function (check) { + return check; + }); + } + }); + if (fittingPlacement) { + firstFittingPlacement = fittingPlacement; + return "break"; + } + }; + for (var _i = numberOfChecks; _i > 0; _i--) { + var _ret = _loop(_i); + if (_ret === "break") break; + } + } + if (state.placement !== firstFittingPlacement) { + state.modifiersData[name]._skip = true; + state.placement = firstFittingPlacement; + state.reset = true; + } +} // eslint-disable-next-line import/no-unused-modules + +const flip$1 = { + name: 'flip', + enabled: true, + phase: 'main', + fn: flip, + requiresIfExists: ['offset'], + data: { + _skip: false + } +}; + +function getSideOffsets(overflow, rect, preventedOffsets) { + if (preventedOffsets === void 0) { + preventedOffsets = { + x: 0, + y: 0 + }; + } + return { + top: overflow.top - rect.height - preventedOffsets.y, + right: overflow.right - rect.width + preventedOffsets.x, + bottom: overflow.bottom - rect.height + preventedOffsets.y, + left: overflow.left - rect.width - preventedOffsets.x + }; +} +function isAnySideFullyClipped(overflow) { + return [top, right, bottom, left].some(function (side) { + return overflow[side] >= 0; + }); +} +function hide(_ref) { + var state = _ref.state, + name = _ref.name; + var referenceRect = state.rects.reference; + var popperRect = state.rects.popper; + var preventedOffsets = state.modifiersData.preventOverflow; + var referenceOverflow = detectOverflow(state, { + elementContext: 'reference' + }); + var popperAltOverflow = detectOverflow(state, { + altBoundary: true + }); + var referenceClippingOffsets = getSideOffsets(referenceOverflow, referenceRect); + var popperEscapeOffsets = getSideOffsets(popperAltOverflow, popperRect, preventedOffsets); + var isReferenceHidden = isAnySideFullyClipped(referenceClippingOffsets); + var hasPopperEscaped = isAnySideFullyClipped(popperEscapeOffsets); + state.modifiersData[name] = { + referenceClippingOffsets: referenceClippingOffsets, + popperEscapeOffsets: popperEscapeOffsets, + isReferenceHidden: isReferenceHidden, + hasPopperEscaped: hasPopperEscaped + }; + state.attributes.popper = Object.assign({}, state.attributes.popper, { + 'data-popper-reference-hidden': isReferenceHidden, + 'data-popper-escaped': hasPopperEscaped + }); +} // eslint-disable-next-line import/no-unused-modules + +const hide$1 = { + name: 'hide', + enabled: true, + phase: 'main', + requiresIfExists: ['preventOverflow'], + fn: hide +}; + +function distanceAndSkiddingToXY(placement, rects, offset) { + var basePlacement = getBasePlacement(placement); + var invertDistance = [left, top].indexOf(basePlacement) >= 0 ? -1 : 1; + var _ref = typeof offset === 'function' ? offset(Object.assign({}, rects, { + placement: placement + })) : offset, + skidding = _ref[0], + distance = _ref[1]; + skidding = skidding || 0; + distance = (distance || 0) * invertDistance; + return [left, right].indexOf(basePlacement) >= 0 ? { + x: distance, + y: skidding + } : { + x: skidding, + y: distance + }; +} +function offset(_ref2) { + var state = _ref2.state, + options = _ref2.options, + name = _ref2.name; + var _options$offset = options.offset, + offset = _options$offset === void 0 ? [0, 0] : _options$offset; + var data = placements.reduce(function (acc, placement) { + acc[placement] = distanceAndSkiddingToXY(placement, state.rects, offset); + return acc; + }, {}); + var _data$state$placement = data[state.placement], + x = _data$state$placement.x, + y = _data$state$placement.y; + if (state.modifiersData.popperOffsets != null) { + state.modifiersData.popperOffsets.x += x; + state.modifiersData.popperOffsets.y += y; + } + state.modifiersData[name] = data; +} // eslint-disable-next-line import/no-unused-modules + +const offset$1 = { + name: 'offset', + enabled: true, + phase: 'main', + requires: ['popperOffsets'], + fn: offset +}; + +function popperOffsets(_ref) { + var state = _ref.state, + name = _ref.name; + // Offsets are the actual position the popper needs to have to be + // properly positioned near its reference element + // This is the most basic placement, and will be adjusted by + // the modifiers in the next step + state.modifiersData[name] = computeOffsets({ + reference: state.rects.reference, + element: state.rects.popper, + placement: state.placement + }); +} // eslint-disable-next-line import/no-unused-modules + +const popperOffsets$1 = { + name: 'popperOffsets', + enabled: true, + phase: 'read', + fn: popperOffsets, + data: {} +}; + +function getAltAxis(axis) { + return axis === 'x' ? 'y' : 'x'; +} + +function preventOverflow(_ref) { + var state = _ref.state, + options = _ref.options, + name = _ref.name; + var _options$mainAxis = options.mainAxis, + checkMainAxis = _options$mainAxis === void 0 ? true : _options$mainAxis, + _options$altAxis = options.altAxis, + checkAltAxis = _options$altAxis === void 0 ? false : _options$altAxis, + boundary = options.boundary, + rootBoundary = options.rootBoundary, + altBoundary = options.altBoundary, + padding = options.padding, + _options$tether = options.tether, + tether = _options$tether === void 0 ? true : _options$tether, + _options$tetherOffset = options.tetherOffset, + tetherOffset = _options$tetherOffset === void 0 ? 0 : _options$tetherOffset; + var overflow = detectOverflow(state, { + boundary: boundary, + rootBoundary: rootBoundary, + padding: padding, + altBoundary: altBoundary + }); + var basePlacement = getBasePlacement(state.placement); + var variation = getVariation(state.placement); + var isBasePlacement = !variation; + var mainAxis = getMainAxisFromPlacement(basePlacement); + var altAxis = getAltAxis(mainAxis); + var popperOffsets = state.modifiersData.popperOffsets; + var referenceRect = state.rects.reference; + var popperRect = state.rects.popper; + var tetherOffsetValue = typeof tetherOffset === 'function' ? tetherOffset(Object.assign({}, state.rects, { + placement: state.placement + })) : tetherOffset; + var normalizedTetherOffsetValue = typeof tetherOffsetValue === 'number' ? { + mainAxis: tetherOffsetValue, + altAxis: tetherOffsetValue + } : Object.assign({ + mainAxis: 0, + altAxis: 0 + }, tetherOffsetValue); + var offsetModifierState = state.modifiersData.offset ? state.modifiersData.offset[state.placement] : null; + var data = { + x: 0, + y: 0 + }; + if (!popperOffsets) { + return; + } + if (checkMainAxis) { + var _offsetModifierState$; + var mainSide = mainAxis === 'y' ? top : left; + var altSide = mainAxis === 'y' ? bottom : right; + var len = mainAxis === 'y' ? 'height' : 'width'; + var offset = popperOffsets[mainAxis]; + var min$1 = offset + overflow[mainSide]; + var max$1 = offset - overflow[altSide]; + var additive = tether ? -popperRect[len] / 2 : 0; + var minLen = variation === start ? referenceRect[len] : popperRect[len]; + var maxLen = variation === start ? -popperRect[len] : -referenceRect[len]; // We need to include the arrow in the calculation so the arrow doesn't go + // outside the reference bounds + + var arrowElement = state.elements.arrow; + var arrowRect = tether && arrowElement ? getLayoutRect(arrowElement) : { + width: 0, + height: 0 + }; + var arrowPaddingObject = state.modifiersData['arrow#persistent'] ? state.modifiersData['arrow#persistent'].padding : getFreshSideObject(); + var arrowPaddingMin = arrowPaddingObject[mainSide]; + var arrowPaddingMax = arrowPaddingObject[altSide]; // If the reference length is smaller than the arrow length, we don't want + // to include its full size in the calculation. If the reference is small + // and near the edge of a boundary, the popper can overflow even if the + // reference is not overflowing as well (e.g. virtual elements with no + // width or height) + + var arrowLen = within(0, referenceRect[len], arrowRect[len]); + var minOffset = isBasePlacement ? referenceRect[len] / 2 - additive - arrowLen - arrowPaddingMin - normalizedTetherOffsetValue.mainAxis : minLen - arrowLen - arrowPaddingMin - normalizedTetherOffsetValue.mainAxis; + var maxOffset = isBasePlacement ? -referenceRect[len] / 2 + additive + arrowLen + arrowPaddingMax + normalizedTetherOffsetValue.mainAxis : maxLen + arrowLen + arrowPaddingMax + normalizedTetherOffsetValue.mainAxis; + var arrowOffsetParent = state.elements.arrow && getOffsetParent(state.elements.arrow); + var clientOffset = arrowOffsetParent ? mainAxis === 'y' ? arrowOffsetParent.clientTop || 0 : arrowOffsetParent.clientLeft || 0 : 0; + var offsetModifierValue = (_offsetModifierState$ = offsetModifierState == null ? void 0 : offsetModifierState[mainAxis]) != null ? _offsetModifierState$ : 0; + var tetherMin = offset + minOffset - offsetModifierValue - clientOffset; + var tetherMax = offset + maxOffset - offsetModifierValue; + var preventedOffset = within(tether ? min(min$1, tetherMin) : min$1, offset, tether ? max(max$1, tetherMax) : max$1); + popperOffsets[mainAxis] = preventedOffset; + data[mainAxis] = preventedOffset - offset; + } + if (checkAltAxis) { + var _offsetModifierState$2; + var _mainSide = mainAxis === 'x' ? top : left; + var _altSide = mainAxis === 'x' ? bottom : right; + var _offset = popperOffsets[altAxis]; + var _len = altAxis === 'y' ? 'height' : 'width'; + var _min = _offset + overflow[_mainSide]; + var _max = _offset - overflow[_altSide]; + var isOriginSide = [top, left].indexOf(basePlacement) !== -1; + var _offsetModifierValue = (_offsetModifierState$2 = offsetModifierState == null ? void 0 : offsetModifierState[altAxis]) != null ? _offsetModifierState$2 : 0; + var _tetherMin = isOriginSide ? _min : _offset - referenceRect[_len] - popperRect[_len] - _offsetModifierValue + normalizedTetherOffsetValue.altAxis; + var _tetherMax = isOriginSide ? _offset + referenceRect[_len] + popperRect[_len] - _offsetModifierValue - normalizedTetherOffsetValue.altAxis : _max; + var _preventedOffset = tether && isOriginSide ? withinMaxClamp(_tetherMin, _offset, _tetherMax) : within(tether ? _tetherMin : _min, _offset, tether ? _tetherMax : _max); + popperOffsets[altAxis] = _preventedOffset; + data[altAxis] = _preventedOffset - _offset; + } + state.modifiersData[name] = data; +} // eslint-disable-next-line import/no-unused-modules + +const preventOverflow$1 = { + name: 'preventOverflow', + enabled: true, + phase: 'main', + fn: preventOverflow, + requiresIfExists: ['offset'] +}; + +function getHTMLElementScroll(element) { + return { + scrollLeft: element.scrollLeft, + scrollTop: element.scrollTop + }; +} + +function getNodeScroll(node) { + if (node === getWindow(node) || !isHTMLElement(node)) { + return getWindowScroll(node); + } else { + return getHTMLElementScroll(node); + } +} + +function isElementScaled(element) { + var rect = element.getBoundingClientRect(); + var scaleX = round(rect.width) / element.offsetWidth || 1; + var scaleY = round(rect.height) / element.offsetHeight || 1; + return scaleX !== 1 || scaleY !== 1; +} // Returns the composite rect of an element relative to its offsetParent. +// Composite means it takes into account transforms as well as layout. + +function getCompositeRect(elementOrVirtualElement, offsetParent, isFixed) { + if (isFixed === void 0) { + isFixed = false; + } + var isOffsetParentAnElement = isHTMLElement(offsetParent); + var offsetParentIsScaled = isHTMLElement(offsetParent) && isElementScaled(offsetParent); + var documentElement = getDocumentElement(offsetParent); + var rect = getBoundingClientRect(elementOrVirtualElement, offsetParentIsScaled, isFixed); + var scroll = { + scrollLeft: 0, + scrollTop: 0 + }; + var offsets = { + x: 0, + y: 0 + }; + if (isOffsetParentAnElement || !isOffsetParentAnElement && !isFixed) { + if (getNodeName(offsetParent) !== 'body' || + // https://github.com/popperjs/popper-core/issues/1078 + isScrollParent(documentElement)) { + scroll = getNodeScroll(offsetParent); + } + if (isHTMLElement(offsetParent)) { + offsets = getBoundingClientRect(offsetParent, true); + offsets.x += offsetParent.clientLeft; + offsets.y += offsetParent.clientTop; + } else if (documentElement) { + offsets.x = getWindowScrollBarX(documentElement); + } + } + return { + x: rect.left + scroll.scrollLeft - offsets.x, + y: rect.top + scroll.scrollTop - offsets.y, + width: rect.width, + height: rect.height + }; +} + +function order(modifiers) { + var map = new Map(); + var visited = new Set(); + var result = []; + modifiers.forEach(function (modifier) { + map.set(modifier.name, modifier); + }); // On visiting object, check for its dependencies and visit them recursively + + function sort(modifier) { + visited.add(modifier.name); + var requires = [].concat(modifier.requires || [], modifier.requiresIfExists || []); + requires.forEach(function (dep) { + if (!visited.has(dep)) { + var depModifier = map.get(dep); + if (depModifier) { + sort(depModifier); + } + } + }); + result.push(modifier); + } + modifiers.forEach(function (modifier) { + if (!visited.has(modifier.name)) { + // check for visited object + sort(modifier); + } + }); + return result; +} +function orderModifiers(modifiers) { + // order based on dependencies + var orderedModifiers = order(modifiers); // order based on phase + + return modifierPhases.reduce(function (acc, phase) { + return acc.concat(orderedModifiers.filter(function (modifier) { + return modifier.phase === phase; + })); + }, []); +} + +function debounce(fn) { + var pending; + return function () { + if (!pending) { + pending = new Promise(function (resolve) { + Promise.resolve().then(function () { + pending = undefined; + resolve(fn()); + }); + }); + } + return pending; + }; +} + +function mergeByName(modifiers) { + var merged = modifiers.reduce(function (merged, current) { + var existing = merged[current.name]; + merged[current.name] = existing ? Object.assign({}, existing, current, { + options: Object.assign({}, existing.options, current.options), + data: Object.assign({}, existing.data, current.data) + }) : current; + return merged; + }, {}); // IE11 does not support Object.values + + return Object.keys(merged).map(function (key) { + return merged[key]; + }); +} + +var DEFAULT_OPTIONS = { + placement: 'bottom', + modifiers: [], + strategy: 'absolute' +}; +function areValidElements() { + for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { + args[_key] = arguments[_key]; + } + return !args.some(function (element) { + return !(element && typeof element.getBoundingClientRect === 'function'); + }); +} +function popperGenerator(generatorOptions) { + if (generatorOptions === void 0) { + generatorOptions = {}; + } + var _generatorOptions = generatorOptions, + _generatorOptions$def = _generatorOptions.defaultModifiers, + defaultModifiers = _generatorOptions$def === void 0 ? [] : _generatorOptions$def, + _generatorOptions$def2 = _generatorOptions.defaultOptions, + defaultOptions = _generatorOptions$def2 === void 0 ? DEFAULT_OPTIONS : _generatorOptions$def2; + return function createPopper(reference, popper, options) { + if (options === void 0) { + options = defaultOptions; + } + var state = { + placement: 'bottom', + orderedModifiers: [], + options: Object.assign({}, DEFAULT_OPTIONS, defaultOptions), + modifiersData: {}, + elements: { + reference: reference, + popper: popper + }, + attributes: {}, + styles: {} + }; + var effectCleanupFns = []; + var isDestroyed = false; + var instance = { + state: state, + setOptions: function setOptions(setOptionsAction) { + var options = typeof setOptionsAction === 'function' ? setOptionsAction(state.options) : setOptionsAction; + cleanupModifierEffects(); + state.options = Object.assign({}, defaultOptions, state.options, options); + state.scrollParents = { + reference: isElement$1(reference) ? listScrollParents(reference) : reference.contextElement ? listScrollParents(reference.contextElement) : [], + popper: listScrollParents(popper) + }; // Orders the modifiers based on their dependencies and `phase` + // properties + + var orderedModifiers = orderModifiers(mergeByName([].concat(defaultModifiers, state.options.modifiers))); // Strip out disabled modifiers + + state.orderedModifiers = orderedModifiers.filter(function (m) { + return m.enabled; + }); + runModifierEffects(); + return instance.update(); + }, + // Sync update – it will always be executed, even if not necessary. This + // is useful for low frequency updates where sync behavior simplifies the + // logic. + // For high frequency updates (e.g. `resize` and `scroll` events), always + // prefer the async Popper#update method + forceUpdate: function forceUpdate() { + if (isDestroyed) { + return; + } + var _state$elements = state.elements, + reference = _state$elements.reference, + popper = _state$elements.popper; // Don't proceed if `reference` or `popper` are not valid elements + // anymore + + if (!areValidElements(reference, popper)) { + return; + } // Store the reference and popper rects to be read by modifiers + + state.rects = { + reference: getCompositeRect(reference, getOffsetParent(popper), state.options.strategy === 'fixed'), + popper: getLayoutRect(popper) + }; // Modifiers have the ability to reset the current update cycle. The + // most common use case for this is the `flip` modifier changing the + // placement, which then needs to re-run all the modifiers, because the + // logic was previously ran for the previous placement and is therefore + // stale/incorrect + + state.reset = false; + state.placement = state.options.placement; // On each update cycle, the `modifiersData` property for each modifier + // is filled with the initial data specified by the modifier. This means + // it doesn't persist and is fresh on each update. + // To ensure persistent data, use `${name}#persistent` + + state.orderedModifiers.forEach(function (modifier) { + return state.modifiersData[modifier.name] = Object.assign({}, modifier.data); + }); + for (var index = 0; index < state.orderedModifiers.length; index++) { + if (state.reset === true) { + state.reset = false; + index = -1; + continue; + } + var _state$orderedModifie = state.orderedModifiers[index], + fn = _state$orderedModifie.fn, + _state$orderedModifie2 = _state$orderedModifie.options, + _options = _state$orderedModifie2 === void 0 ? {} : _state$orderedModifie2, + name = _state$orderedModifie.name; + if (typeof fn === 'function') { + state = fn({ + state: state, + options: _options, + name: name, + instance: instance + }) || state; + } + } + }, + // Async and optimistically optimized update – it will not be executed if + // not necessary (debounced to run at most once-per-tick) + update: debounce(function () { + return new Promise(function (resolve) { + instance.forceUpdate(); + resolve(state); + }); + }), + destroy: function destroy() { + cleanupModifierEffects(); + isDestroyed = true; + } + }; + if (!areValidElements(reference, popper)) { + return instance; + } + instance.setOptions(options).then(function (state) { + if (!isDestroyed && options.onFirstUpdate) { + options.onFirstUpdate(state); + } + }); // Modifiers have the ability to execute arbitrary code before the first + // update cycle runs. They will be executed in the same order as the update + // cycle. This is useful when a modifier adds some persistent data that + // other modifiers need to use, but the modifier is run after the dependent + // one. + + function runModifierEffects() { + state.orderedModifiers.forEach(function (_ref) { + var name = _ref.name, + _ref$options = _ref.options, + options = _ref$options === void 0 ? {} : _ref$options, + effect = _ref.effect; + if (typeof effect === 'function') { + var cleanupFn = effect({ + state: state, + name: name, + instance: instance, + options: options + }); + var noopFn = function noopFn() {}; + effectCleanupFns.push(cleanupFn || noopFn); + } + }); + } + function cleanupModifierEffects() { + effectCleanupFns.forEach(function (fn) { + return fn(); + }); + effectCleanupFns = []; + } + return instance; + }; +} +var createPopper$2 = /*#__PURE__*/popperGenerator(); // eslint-disable-next-line import/no-unused-modules + +var defaultModifiers$1 = [eventListeners, popperOffsets$1, computeStyles$1, applyStyles$1]; +var createPopper$1 = /*#__PURE__*/popperGenerator({ + defaultModifiers: defaultModifiers$1 +}); // eslint-disable-next-line import/no-unused-modules + +var defaultModifiers = [eventListeners, popperOffsets$1, computeStyles$1, applyStyles$1, offset$1, flip$1, preventOverflow$1, arrow$1, hide$1]; +var createPopper = /*#__PURE__*/popperGenerator({ + defaultModifiers: defaultModifiers +}); // eslint-disable-next-line import/no-unused-modules + +const Popper = /*#__PURE__*/Object.freeze(/*#__PURE__*/Object.defineProperty({ + __proto__: null, + afterMain, + afterRead, + afterWrite, + applyStyles: applyStyles$1, + arrow: arrow$1, + auto, + basePlacements, + beforeMain, + beforeRead, + beforeWrite, + bottom, + clippingParents, + computeStyles: computeStyles$1, + createPopper, + createPopperBase: createPopper$2, + createPopperLite: createPopper$1, + detectOverflow, + end, + eventListeners, + flip: flip$1, + hide: hide$1, + left, + main, + modifierPhases, + offset: offset$1, + placements, + popper, + popperGenerator, + popperOffsets: popperOffsets$1, + preventOverflow: preventOverflow$1, + read, + reference, + right, + start, + top, + variationPlacements, + viewport, + write +}, Symbol.toStringTag, { value: 'Module' })); + +/*! + * Bootstrap v5.3.7 (https://getbootstrap.com/) + * Copyright 2011-2025 The Bootstrap Authors (https://github.com/twbs/bootstrap/graphs/contributors) + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + */ + +/** + * -------------------------------------------------------------------------- + * Bootstrap dom/data.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const elementMap = new Map(); +const Data = { + set(element, key, instance) { + if (!elementMap.has(element)) { + elementMap.set(element, new Map()); + } + const instanceMap = elementMap.get(element); + + // make it clear we only want one instance per element + // can be removed later when multiple key/instances are fine to be used + if (!instanceMap.has(key) && instanceMap.size !== 0) { + // eslint-disable-next-line no-console + console.error(`Bootstrap doesn't allow more than one instance per element. Bound instance: ${Array.from(instanceMap.keys())[0]}.`); + return; + } + instanceMap.set(key, instance); + }, + get(element, key) { + if (elementMap.has(element)) { + return elementMap.get(element).get(key) || null; + } + return null; + }, + remove(element, key) { + if (!elementMap.has(element)) { + return; + } + const instanceMap = elementMap.get(element); + instanceMap.delete(key); + + // free up element references if there are no instances left for an element + if (instanceMap.size === 0) { + elementMap.delete(element); + } + } +}; + +/** + * -------------------------------------------------------------------------- + * Bootstrap util/index.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +const MAX_UID = 1000000; +const MILLISECONDS_MULTIPLIER = 1000; +const TRANSITION_END = 'transitionend'; + +/** + * Properly escape IDs selectors to handle weird IDs + * @param {string} selector + * @returns {string} + */ +const parseSelector = selector => { + if (selector && window.CSS && window.CSS.escape) { + // document.querySelector needs escaping to handle IDs (html5+) containing for instance / + selector = selector.replace(/#([^\s"#']+)/g, (match, id) => `#${CSS.escape(id)}`); + } + return selector; +}; + +// Shout-out Angus Croll (https://goo.gl/pxwQGp) +const toType = object => { + if (object === null || object === undefined) { + return `${object}`; + } + return Object.prototype.toString.call(object).match(/\s([a-z]+)/i)[1].toLowerCase(); +}; + +/** + * Public Util API + */ + +const getUID = prefix => { + do { + prefix += Math.floor(Math.random() * MAX_UID); + } while (document.getElementById(prefix)); + return prefix; +}; +const getTransitionDurationFromElement = element => { + if (!element) { + return 0; + } + + // Get transition-duration of the element + let { + transitionDuration, + transitionDelay + } = window.getComputedStyle(element); + const floatTransitionDuration = Number.parseFloat(transitionDuration); + const floatTransitionDelay = Number.parseFloat(transitionDelay); + + // Return 0 if element or transition duration is not found + if (!floatTransitionDuration && !floatTransitionDelay) { + return 0; + } + + // If multiple durations are defined, take the first + transitionDuration = transitionDuration.split(',')[0]; + transitionDelay = transitionDelay.split(',')[0]; + return (Number.parseFloat(transitionDuration) + Number.parseFloat(transitionDelay)) * MILLISECONDS_MULTIPLIER; +}; +const triggerTransitionEnd = element => { + element.dispatchEvent(new Event(TRANSITION_END)); +}; +const isElement = object => { + if (!object || typeof object !== 'object') { + return false; + } + if (typeof object.jquery !== 'undefined') { + object = object[0]; + } + return typeof object.nodeType !== 'undefined'; +}; +const getElement = object => { + // it's a jQuery object or a node element + if (isElement(object)) { + return object.jquery ? object[0] : object; + } + if (typeof object === 'string' && object.length > 0) { + return document.querySelector(parseSelector(object)); + } + return null; +}; +const isVisible = element => { + if (!isElement(element) || element.getClientRects().length === 0) { + return false; + } + const elementIsVisible = getComputedStyle(element).getPropertyValue('visibility') === 'visible'; + // Handle `details` element as its content may falsie appear visible when it is closed + const closedDetails = element.closest('details:not([open])'); + if (!closedDetails) { + return elementIsVisible; + } + if (closedDetails !== element) { + const summary = element.closest('summary'); + if (summary && summary.parentNode !== closedDetails) { + return false; + } + if (summary === null) { + return false; + } + } + return elementIsVisible; +}; +const isDisabled = element => { + if (!element || element.nodeType !== Node.ELEMENT_NODE) { + return true; + } + if (element.classList.contains('disabled')) { + return true; + } + if (typeof element.disabled !== 'undefined') { + return element.disabled; + } + return element.hasAttribute('disabled') && element.getAttribute('disabled') !== 'false'; +}; +const findShadowRoot = element => { + if (!document.documentElement.attachShadow) { + return null; + } + + // Can find the shadow root otherwise it'll return the document + if (typeof element.getRootNode === 'function') { + const root = element.getRootNode(); + return root instanceof ShadowRoot ? root : null; + } + if (element instanceof ShadowRoot) { + return element; + } + + // when we don't find a shadow root + if (!element.parentNode) { + return null; + } + return findShadowRoot(element.parentNode); +}; +const noop = () => {}; + +/** + * Trick to restart an element's animation + * + * @param {HTMLElement} element + * @return void + * + * @see https://www.harrytheo.com/blog/2021/02/restart-a-css-animation-with-javascript/#restarting-a-css-animation + */ +const reflow = element => { + element.offsetHeight; // eslint-disable-line no-unused-expressions +}; +const getjQuery = () => { + if (window.jQuery && !document.body.hasAttribute('data-bs-no-jquery')) { + return window.jQuery; + } + return null; +}; +const DOMContentLoadedCallbacks = []; +const onDOMContentLoaded = callback => { + if (document.readyState === 'loading') { + // add listener on the first call when the document is in loading state + if (!DOMContentLoadedCallbacks.length) { + document.addEventListener('DOMContentLoaded', () => { + for (const callback of DOMContentLoadedCallbacks) { + callback(); + } + }); + } + DOMContentLoadedCallbacks.push(callback); + } else { + callback(); + } +}; +const isRTL = () => document.documentElement.dir === 'rtl'; +const defineJQueryPlugin = plugin => { + onDOMContentLoaded(() => { + const $ = getjQuery(); + /* istanbul ignore if */ + if ($) { + const name = plugin.NAME; + const JQUERY_NO_CONFLICT = $.fn[name]; + $.fn[name] = plugin.jQueryInterface; + $.fn[name].Constructor = plugin; + $.fn[name].noConflict = () => { + $.fn[name] = JQUERY_NO_CONFLICT; + return plugin.jQueryInterface; + }; + } + }); +}; +const execute = (possibleCallback, args = [], defaultValue = possibleCallback) => { + return typeof possibleCallback === 'function' ? possibleCallback.call(...args) : defaultValue; +}; +const executeAfterTransition = (callback, transitionElement, waitForTransition = true) => { + if (!waitForTransition) { + execute(callback); + return; + } + const durationPadding = 5; + const emulatedDuration = getTransitionDurationFromElement(transitionElement) + durationPadding; + let called = false; + const handler = ({ + target + }) => { + if (target !== transitionElement) { + return; + } + called = true; + transitionElement.removeEventListener(TRANSITION_END, handler); + execute(callback); + }; + transitionElement.addEventListener(TRANSITION_END, handler); + setTimeout(() => { + if (!called) { + triggerTransitionEnd(transitionElement); + } + }, emulatedDuration); +}; + +/** + * Return the previous/next element of a list. + * + * @param {array} list The list of elements + * @param activeElement The active element + * @param shouldGetNext Choose to get next or previous element + * @param isCycleAllowed + * @return {Element|elem} The proper element + */ +const getNextActiveElement = (list, activeElement, shouldGetNext, isCycleAllowed) => { + const listLength = list.length; + let index = list.indexOf(activeElement); + + // if the element does not exist in the list return an element + // depending on the direction and if cycle is allowed + if (index === -1) { + return !shouldGetNext && isCycleAllowed ? list[listLength - 1] : list[0]; + } + index += shouldGetNext ? 1 : -1; + if (isCycleAllowed) { + index = (index + listLength) % listLength; + } + return list[Math.max(0, Math.min(index, listLength - 1))]; +}; + +/** + * -------------------------------------------------------------------------- + * Bootstrap dom/event-handler.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const namespaceRegex = /[^.]*(?=\..*)\.|.*/; +const stripNameRegex = /\..*/; +const stripUidRegex = /::\d+$/; +const eventRegistry = {}; // Events storage +let uidEvent = 1; +const customEvents = { + mouseenter: 'mouseover', + mouseleave: 'mouseout' +}; +const nativeEvents = new Set(['click', 'dblclick', 'mouseup', 'mousedown', 'contextmenu', 'mousewheel', 'DOMMouseScroll', 'mouseover', 'mouseout', 'mousemove', 'selectstart', 'selectend', 'keydown', 'keypress', 'keyup', 'orientationchange', 'touchstart', 'touchmove', 'touchend', 'touchcancel', 'pointerdown', 'pointermove', 'pointerup', 'pointerleave', 'pointercancel', 'gesturestart', 'gesturechange', 'gestureend', 'focus', 'blur', 'change', 'reset', 'select', 'submit', 'focusin', 'focusout', 'load', 'unload', 'beforeunload', 'resize', 'move', 'DOMContentLoaded', 'readystatechange', 'error', 'abort', 'scroll']); + +/** + * Private methods + */ + +function makeEventUid(element, uid) { + return uid && `${uid}::${uidEvent++}` || element.uidEvent || uidEvent++; +} +function getElementEvents(element) { + const uid = makeEventUid(element); + element.uidEvent = uid; + eventRegistry[uid] = eventRegistry[uid] || {}; + return eventRegistry[uid]; +} +function bootstrapHandler(element, fn) { + return function handler(event) { + hydrateObj(event, { + delegateTarget: element + }); + if (handler.oneOff) { + EventHandler.off(element, event.type, fn); + } + return fn.apply(element, [event]); + }; +} +function bootstrapDelegationHandler(element, selector, fn) { + return function handler(event) { + const domElements = element.querySelectorAll(selector); + for (let { + target + } = event; target && target !== this; target = target.parentNode) { + for (const domElement of domElements) { + if (domElement !== target) { + continue; + } + hydrateObj(event, { + delegateTarget: target + }); + if (handler.oneOff) { + EventHandler.off(element, event.type, selector, fn); + } + return fn.apply(target, [event]); + } + } + }; +} +function findHandler(events, callable, delegationSelector = null) { + return Object.values(events).find(event => event.callable === callable && event.delegationSelector === delegationSelector); +} +function normalizeParameters(originalTypeEvent, handler, delegationFunction) { + const isDelegated = typeof handler === 'string'; + // TODO: tooltip passes `false` instead of selector, so we need to check + const callable = isDelegated ? delegationFunction : handler || delegationFunction; + let typeEvent = getTypeEvent(originalTypeEvent); + if (!nativeEvents.has(typeEvent)) { + typeEvent = originalTypeEvent; + } + return [isDelegated, callable, typeEvent]; +} +function addHandler(element, originalTypeEvent, handler, delegationFunction, oneOff) { + if (typeof originalTypeEvent !== 'string' || !element) { + return; + } + let [isDelegated, callable, typeEvent] = normalizeParameters(originalTypeEvent, handler, delegationFunction); + + // in case of mouseenter or mouseleave wrap the handler within a function that checks for its DOM position + // this prevents the handler from being dispatched the same way as mouseover or mouseout does + if (originalTypeEvent in customEvents) { + const wrapFunction = fn => { + return function (event) { + if (!event.relatedTarget || event.relatedTarget !== event.delegateTarget && !event.delegateTarget.contains(event.relatedTarget)) { + return fn.call(this, event); + } + }; + }; + callable = wrapFunction(callable); + } + const events = getElementEvents(element); + const handlers = events[typeEvent] || (events[typeEvent] = {}); + const previousFunction = findHandler(handlers, callable, isDelegated ? handler : null); + if (previousFunction) { + previousFunction.oneOff = previousFunction.oneOff && oneOff; + return; + } + const uid = makeEventUid(callable, originalTypeEvent.replace(namespaceRegex, '')); + const fn = isDelegated ? bootstrapDelegationHandler(element, handler, callable) : bootstrapHandler(element, callable); + fn.delegationSelector = isDelegated ? handler : null; + fn.callable = callable; + fn.oneOff = oneOff; + fn.uidEvent = uid; + handlers[uid] = fn; + element.addEventListener(typeEvent, fn, isDelegated); +} +function removeHandler(element, events, typeEvent, handler, delegationSelector) { + const fn = findHandler(events[typeEvent], handler, delegationSelector); + if (!fn) { + return; + } + element.removeEventListener(typeEvent, fn, Boolean(delegationSelector)); + delete events[typeEvent][fn.uidEvent]; +} +function removeNamespacedHandlers(element, events, typeEvent, namespace) { + const storeElementEvent = events[typeEvent] || {}; + for (const [handlerKey, event] of Object.entries(storeElementEvent)) { + if (handlerKey.includes(namespace)) { + removeHandler(element, events, typeEvent, event.callable, event.delegationSelector); + } + } +} +function getTypeEvent(event) { + // allow to get the native events from namespaced events ('click.bs.button' --> 'click') + event = event.replace(stripNameRegex, ''); + return customEvents[event] || event; +} +const EventHandler = { + on(element, event, handler, delegationFunction) { + addHandler(element, event, handler, delegationFunction, false); + }, + one(element, event, handler, delegationFunction) { + addHandler(element, event, handler, delegationFunction, true); + }, + off(element, originalTypeEvent, handler, delegationFunction) { + if (typeof originalTypeEvent !== 'string' || !element) { + return; + } + const [isDelegated, callable, typeEvent] = normalizeParameters(originalTypeEvent, handler, delegationFunction); + const inNamespace = typeEvent !== originalTypeEvent; + const events = getElementEvents(element); + const storeElementEvent = events[typeEvent] || {}; + const isNamespace = originalTypeEvent.startsWith('.'); + if (typeof callable !== 'undefined') { + // Simplest case: handler is passed, remove that listener ONLY. + if (!Object.keys(storeElementEvent).length) { + return; + } + removeHandler(element, events, typeEvent, callable, isDelegated ? handler : null); + return; + } + if (isNamespace) { + for (const elementEvent of Object.keys(events)) { + removeNamespacedHandlers(element, events, elementEvent, originalTypeEvent.slice(1)); + } + } + for (const [keyHandlers, event] of Object.entries(storeElementEvent)) { + const handlerKey = keyHandlers.replace(stripUidRegex, ''); + if (!inNamespace || originalTypeEvent.includes(handlerKey)) { + removeHandler(element, events, typeEvent, event.callable, event.delegationSelector); + } + } + }, + trigger(element, event, args) { + if (typeof event !== 'string' || !element) { + return null; + } + const $ = getjQuery(); + const typeEvent = getTypeEvent(event); + const inNamespace = event !== typeEvent; + let jQueryEvent = null; + let bubbles = true; + let nativeDispatch = true; + let defaultPrevented = false; + if (inNamespace && $) { + jQueryEvent = $.Event(event, args); + $(element).trigger(jQueryEvent); + bubbles = !jQueryEvent.isPropagationStopped(); + nativeDispatch = !jQueryEvent.isImmediatePropagationStopped(); + defaultPrevented = jQueryEvent.isDefaultPrevented(); + } + const evt = hydrateObj(new Event(event, { + bubbles, + cancelable: true + }), args); + if (defaultPrevented) { + evt.preventDefault(); + } + if (nativeDispatch) { + element.dispatchEvent(evt); + } + if (evt.defaultPrevented && jQueryEvent) { + jQueryEvent.preventDefault(); + } + return evt; + } +}; +function hydrateObj(obj, meta = {}) { + for (const [key, value] of Object.entries(meta)) { + try { + obj[key] = value; + } catch (_unused) { + Object.defineProperty(obj, key, { + configurable: true, + get() { + return value; + } + }); + } + } + return obj; +} + +/** + * -------------------------------------------------------------------------- + * Bootstrap dom/manipulator.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +function normalizeData(value) { + if (value === 'true') { + return true; + } + if (value === 'false') { + return false; + } + if (value === Number(value).toString()) { + return Number(value); + } + if (value === '' || value === 'null') { + return null; + } + if (typeof value !== 'string') { + return value; + } + try { + return JSON.parse(decodeURIComponent(value)); + } catch (_unused) { + return value; + } +} +function normalizeDataKey(key) { + return key.replace(/[A-Z]/g, chr => `-${chr.toLowerCase()}`); +} +const Manipulator = { + setDataAttribute(element, key, value) { + element.setAttribute(`data-bs-${normalizeDataKey(key)}`, value); + }, + removeDataAttribute(element, key) { + element.removeAttribute(`data-bs-${normalizeDataKey(key)}`); + }, + getDataAttributes(element) { + if (!element) { + return {}; + } + const attributes = {}; + const bsKeys = Object.keys(element.dataset).filter(key => key.startsWith('bs') && !key.startsWith('bsConfig')); + for (const key of bsKeys) { + let pureKey = key.replace(/^bs/, ''); + pureKey = pureKey.charAt(0).toLowerCase() + pureKey.slice(1); + attributes[pureKey] = normalizeData(element.dataset[key]); + } + return attributes; + }, + getDataAttribute(element, key) { + return normalizeData(element.getAttribute(`data-bs-${normalizeDataKey(key)}`)); + } +}; + +/** + * -------------------------------------------------------------------------- + * Bootstrap util/config.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Class definition + */ + +class Config { + // Getters + static get Default() { + return {}; + } + static get DefaultType() { + return {}; + } + static get NAME() { + throw new Error('You have to implement the static method "NAME", for each component!'); + } + _getConfig(config) { + config = this._mergeConfigObj(config); + config = this._configAfterMerge(config); + this._typeCheckConfig(config); + return config; + } + _configAfterMerge(config) { + return config; + } + _mergeConfigObj(config, element) { + const jsonConfig = isElement(element) ? Manipulator.getDataAttribute(element, 'config') : {}; // try to parse + + return { + ...this.constructor.Default, + ...(typeof jsonConfig === 'object' ? jsonConfig : {}), + ...(isElement(element) ? Manipulator.getDataAttributes(element) : {}), + ...(typeof config === 'object' ? config : {}) + }; + } + _typeCheckConfig(config, configTypes = this.constructor.DefaultType) { + for (const [property, expectedTypes] of Object.entries(configTypes)) { + const value = config[property]; + const valueType = isElement(value) ? 'element' : toType(value); + if (!new RegExp(expectedTypes).test(valueType)) { + throw new TypeError(`${this.constructor.NAME.toUpperCase()}: Option "${property}" provided type "${valueType}" but expected type "${expectedTypes}".`); + } + } + } +} + +/** + * -------------------------------------------------------------------------- + * Bootstrap base-component.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const VERSION = '5.3.7'; + +/** + * Class definition + */ + +class BaseComponent extends Config { + constructor(element, config) { + super(); + element = getElement(element); + if (!element) { + return; + } + this._element = element; + this._config = this._getConfig(config); + Data.set(this._element, this.constructor.DATA_KEY, this); + } + + // Public + dispose() { + Data.remove(this._element, this.constructor.DATA_KEY); + EventHandler.off(this._element, this.constructor.EVENT_KEY); + for (const propertyName of Object.getOwnPropertyNames(this)) { + this[propertyName] = null; + } + } + + // Private + _queueCallback(callback, element, isAnimated = true) { + executeAfterTransition(callback, element, isAnimated); + } + _getConfig(config) { + config = this._mergeConfigObj(config, this._element); + config = this._configAfterMerge(config); + this._typeCheckConfig(config); + return config; + } + + // Static + static getInstance(element) { + return Data.get(getElement(element), this.DATA_KEY); + } + static getOrCreateInstance(element, config = {}) { + return this.getInstance(element) || new this(element, typeof config === 'object' ? config : null); + } + static get VERSION() { + return VERSION; + } + static get DATA_KEY() { + return `bs.${this.NAME}`; + } + static get EVENT_KEY() { + return `.${this.DATA_KEY}`; + } + static eventName(name) { + return `${name}${this.EVENT_KEY}`; + } +} + +/** + * -------------------------------------------------------------------------- + * Bootstrap dom/selector-engine.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +const getSelector = element => { + let selector = element.getAttribute('data-bs-target'); + if (!selector || selector === '#') { + let hrefAttribute = element.getAttribute('href'); + + // The only valid content that could double as a selector are IDs or classes, + // so everything starting with `#` or `.`. If a "real" URL is used as the selector, + // `document.querySelector` will rightfully complain it is invalid. + // See https://github.com/twbs/bootstrap/issues/32273 + if (!hrefAttribute || !hrefAttribute.includes('#') && !hrefAttribute.startsWith('.')) { + return null; + } + + // Just in case some CMS puts out a full URL with the anchor appended + if (hrefAttribute.includes('#') && !hrefAttribute.startsWith('#')) { + hrefAttribute = `#${hrefAttribute.split('#')[1]}`; + } + selector = hrefAttribute && hrefAttribute !== '#' ? hrefAttribute.trim() : null; + } + return selector ? selector.split(',').map(sel => parseSelector(sel)).join(',') : null; +}; +const SelectorEngine = { + find(selector, element = document.documentElement) { + return [].concat(...Element.prototype.querySelectorAll.call(element, selector)); + }, + findOne(selector, element = document.documentElement) { + return Element.prototype.querySelector.call(element, selector); + }, + children(element, selector) { + return [].concat(...element.children).filter(child => child.matches(selector)); + }, + parents(element, selector) { + const parents = []; + let ancestor = element.parentNode.closest(selector); + while (ancestor) { + parents.push(ancestor); + ancestor = ancestor.parentNode.closest(selector); + } + return parents; + }, + prev(element, selector) { + let previous = element.previousElementSibling; + while (previous) { + if (previous.matches(selector)) { + return [previous]; + } + previous = previous.previousElementSibling; + } + return []; + }, + // TODO: this is now unused; remove later along with prev() + next(element, selector) { + let next = element.nextElementSibling; + while (next) { + if (next.matches(selector)) { + return [next]; + } + next = next.nextElementSibling; + } + return []; + }, + focusableChildren(element) { + const focusables = ['a', 'button', 'input', 'textarea', 'select', 'details', '[tabindex]', '[contenteditable="true"]'].map(selector => `${selector}:not([tabindex^="-"])`).join(','); + return this.find(focusables, element).filter(el => !isDisabled(el) && isVisible(el)); + }, + getSelectorFromElement(element) { + const selector = getSelector(element); + if (selector) { + return SelectorEngine.findOne(selector) ? selector : null; + } + return null; + }, + getElementFromSelector(element) { + const selector = getSelector(element); + return selector ? SelectorEngine.findOne(selector) : null; + }, + getMultipleElementsFromSelector(element) { + const selector = getSelector(element); + return selector ? SelectorEngine.find(selector) : []; + } +}; + +/** + * -------------------------------------------------------------------------- + * Bootstrap util/component-functions.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +const enableDismissTrigger = (component, method = 'hide') => { + const clickEvent = `click.dismiss${component.EVENT_KEY}`; + const name = component.NAME; + EventHandler.on(document, clickEvent, `[data-bs-dismiss="${name}"]`, function (event) { + if (['A', 'AREA'].includes(this.tagName)) { + event.preventDefault(); + } + if (isDisabled(this)) { + return; + } + const target = SelectorEngine.getElementFromSelector(this) || this.closest(`.${name}`); + const instance = component.getOrCreateInstance(target); + + // Method argument is left, for Alert and only, as it doesn't implement the 'hide' method + instance[method](); + }); +}; + +/** + * -------------------------------------------------------------------------- + * Bootstrap alert.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$f = 'alert'; +const DATA_KEY$a = 'bs.alert'; +const EVENT_KEY$b = `.${DATA_KEY$a}`; +const EVENT_CLOSE = `close${EVENT_KEY$b}`; +const EVENT_CLOSED = `closed${EVENT_KEY$b}`; +const CLASS_NAME_FADE$5 = 'fade'; +const CLASS_NAME_SHOW$8 = 'show'; + +/** + * Class definition + */ + +class Alert extends BaseComponent { + // Getters + static get NAME() { + return NAME$f; + } + + // Public + close() { + const closeEvent = EventHandler.trigger(this._element, EVENT_CLOSE); + if (closeEvent.defaultPrevented) { + return; + } + this._element.classList.remove(CLASS_NAME_SHOW$8); + const isAnimated = this._element.classList.contains(CLASS_NAME_FADE$5); + this._queueCallback(() => this._destroyElement(), this._element, isAnimated); + } + + // Private + _destroyElement() { + this._element.remove(); + EventHandler.trigger(this._element, EVENT_CLOSED); + this.dispose(); + } + + // Static + static jQueryInterface(config) { + return this.each(function () { + const data = Alert.getOrCreateInstance(this); + if (typeof config !== 'string') { + return; + } + if (data[config] === undefined || config.startsWith('_') || config === 'constructor') { + throw new TypeError(`No method named "${config}"`); + } + data[config](this); + }); + } +} + +/** + * Data API implementation + */ + +enableDismissTrigger(Alert, 'close'); + +/** + * jQuery + */ + +defineJQueryPlugin(Alert); + +/** + * -------------------------------------------------------------------------- + * Bootstrap button.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$e = 'button'; +const DATA_KEY$9 = 'bs.button'; +const EVENT_KEY$a = `.${DATA_KEY$9}`; +const DATA_API_KEY$6 = '.data-api'; +const CLASS_NAME_ACTIVE$3 = 'active'; +const SELECTOR_DATA_TOGGLE$5 = '[data-bs-toggle="button"]'; +const EVENT_CLICK_DATA_API$6 = `click${EVENT_KEY$a}${DATA_API_KEY$6}`; + +/** + * Class definition + */ + +class Button extends BaseComponent { + // Getters + static get NAME() { + return NAME$e; + } + + // Public + toggle() { + // Toggle class and sync the `aria-pressed` attribute with the return value of the `.toggle()` method + this._element.setAttribute('aria-pressed', this._element.classList.toggle(CLASS_NAME_ACTIVE$3)); + } + + // Static + static jQueryInterface(config) { + return this.each(function () { + const data = Button.getOrCreateInstance(this); + if (config === 'toggle') { + data[config](); + } + }); + } +} + +/** + * Data API implementation + */ + +EventHandler.on(document, EVENT_CLICK_DATA_API$6, SELECTOR_DATA_TOGGLE$5, event => { + event.preventDefault(); + const button = event.target.closest(SELECTOR_DATA_TOGGLE$5); + const data = Button.getOrCreateInstance(button); + data.toggle(); +}); + +/** + * jQuery + */ + +defineJQueryPlugin(Button); + +/** + * -------------------------------------------------------------------------- + * Bootstrap util/swipe.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$d = 'swipe'; +const EVENT_KEY$9 = '.bs.swipe'; +const EVENT_TOUCHSTART = `touchstart${EVENT_KEY$9}`; +const EVENT_TOUCHMOVE = `touchmove${EVENT_KEY$9}`; +const EVENT_TOUCHEND = `touchend${EVENT_KEY$9}`; +const EVENT_POINTERDOWN = `pointerdown${EVENT_KEY$9}`; +const EVENT_POINTERUP = `pointerup${EVENT_KEY$9}`; +const POINTER_TYPE_TOUCH = 'touch'; +const POINTER_TYPE_PEN = 'pen'; +const CLASS_NAME_POINTER_EVENT = 'pointer-event'; +const SWIPE_THRESHOLD = 40; +const Default$c = { + endCallback: null, + leftCallback: null, + rightCallback: null +}; +const DefaultType$c = { + endCallback: '(function|null)', + leftCallback: '(function|null)', + rightCallback: '(function|null)' +}; + +/** + * Class definition + */ + +class Swipe extends Config { + constructor(element, config) { + super(); + this._element = element; + if (!element || !Swipe.isSupported()) { + return; + } + this._config = this._getConfig(config); + this._deltaX = 0; + this._supportPointerEvents = Boolean(window.PointerEvent); + this._initEvents(); + } + + // Getters + static get Default() { + return Default$c; + } + static get DefaultType() { + return DefaultType$c; + } + static get NAME() { + return NAME$d; + } + + // Public + dispose() { + EventHandler.off(this._element, EVENT_KEY$9); + } + + // Private + _start(event) { + if (!this._supportPointerEvents) { + this._deltaX = event.touches[0].clientX; + return; + } + if (this._eventIsPointerPenTouch(event)) { + this._deltaX = event.clientX; + } + } + _end(event) { + if (this._eventIsPointerPenTouch(event)) { + this._deltaX = event.clientX - this._deltaX; + } + this._handleSwipe(); + execute(this._config.endCallback); + } + _move(event) { + this._deltaX = event.touches && event.touches.length > 1 ? 0 : event.touches[0].clientX - this._deltaX; + } + _handleSwipe() { + const absDeltaX = Math.abs(this._deltaX); + if (absDeltaX <= SWIPE_THRESHOLD) { + return; + } + const direction = absDeltaX / this._deltaX; + this._deltaX = 0; + if (!direction) { + return; + } + execute(direction > 0 ? this._config.rightCallback : this._config.leftCallback); + } + _initEvents() { + if (this._supportPointerEvents) { + EventHandler.on(this._element, EVENT_POINTERDOWN, event => this._start(event)); + EventHandler.on(this._element, EVENT_POINTERUP, event => this._end(event)); + this._element.classList.add(CLASS_NAME_POINTER_EVENT); + } else { + EventHandler.on(this._element, EVENT_TOUCHSTART, event => this._start(event)); + EventHandler.on(this._element, EVENT_TOUCHMOVE, event => this._move(event)); + EventHandler.on(this._element, EVENT_TOUCHEND, event => this._end(event)); + } + } + _eventIsPointerPenTouch(event) { + return this._supportPointerEvents && (event.pointerType === POINTER_TYPE_PEN || event.pointerType === POINTER_TYPE_TOUCH); + } + + // Static + static isSupported() { + return 'ontouchstart' in document.documentElement || navigator.maxTouchPoints > 0; + } +} + +/** + * -------------------------------------------------------------------------- + * Bootstrap carousel.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$c = 'carousel'; +const DATA_KEY$8 = 'bs.carousel'; +const EVENT_KEY$8 = `.${DATA_KEY$8}`; +const DATA_API_KEY$5 = '.data-api'; +const ARROW_LEFT_KEY$1 = 'ArrowLeft'; +const ARROW_RIGHT_KEY$1 = 'ArrowRight'; +const TOUCHEVENT_COMPAT_WAIT = 500; // Time for mouse compat events to fire after touch + +const ORDER_NEXT = 'next'; +const ORDER_PREV = 'prev'; +const DIRECTION_LEFT = 'left'; +const DIRECTION_RIGHT = 'right'; +const EVENT_SLIDE = `slide${EVENT_KEY$8}`; +const EVENT_SLID = `slid${EVENT_KEY$8}`; +const EVENT_KEYDOWN$1 = `keydown${EVENT_KEY$8}`; +const EVENT_MOUSEENTER$1 = `mouseenter${EVENT_KEY$8}`; +const EVENT_MOUSELEAVE$1 = `mouseleave${EVENT_KEY$8}`; +const EVENT_DRAG_START = `dragstart${EVENT_KEY$8}`; +const EVENT_LOAD_DATA_API$3 = `load${EVENT_KEY$8}${DATA_API_KEY$5}`; +const EVENT_CLICK_DATA_API$5 = `click${EVENT_KEY$8}${DATA_API_KEY$5}`; +const CLASS_NAME_CAROUSEL = 'carousel'; +const CLASS_NAME_ACTIVE$2 = 'active'; +const CLASS_NAME_SLIDE = 'slide'; +const CLASS_NAME_END = 'carousel-item-end'; +const CLASS_NAME_START = 'carousel-item-start'; +const CLASS_NAME_NEXT = 'carousel-item-next'; +const CLASS_NAME_PREV = 'carousel-item-prev'; +const SELECTOR_ACTIVE = '.active'; +const SELECTOR_ITEM = '.carousel-item'; +const SELECTOR_ACTIVE_ITEM = SELECTOR_ACTIVE + SELECTOR_ITEM; +const SELECTOR_ITEM_IMG = '.carousel-item img'; +const SELECTOR_INDICATORS = '.carousel-indicators'; +const SELECTOR_DATA_SLIDE = '[data-bs-slide], [data-bs-slide-to]'; +const SELECTOR_DATA_RIDE = '[data-bs-ride="carousel"]'; +const KEY_TO_DIRECTION = { + [ARROW_LEFT_KEY$1]: DIRECTION_RIGHT, + [ARROW_RIGHT_KEY$1]: DIRECTION_LEFT +}; +const Default$b = { + interval: 5000, + keyboard: true, + pause: 'hover', + ride: false, + touch: true, + wrap: true +}; +const DefaultType$b = { + interval: '(number|boolean)', + // TODO:v6 remove boolean support + keyboard: 'boolean', + pause: '(string|boolean)', + ride: '(boolean|string)', + touch: 'boolean', + wrap: 'boolean' +}; + +/** + * Class definition + */ + +class Carousel extends BaseComponent { + constructor(element, config) { + super(element, config); + this._interval = null; + this._activeElement = null; + this._isSliding = false; + this.touchTimeout = null; + this._swipeHelper = null; + this._indicatorsElement = SelectorEngine.findOne(SELECTOR_INDICATORS, this._element); + this._addEventListeners(); + if (this._config.ride === CLASS_NAME_CAROUSEL) { + this.cycle(); + } + } + + // Getters + static get Default() { + return Default$b; + } + static get DefaultType() { + return DefaultType$b; + } + static get NAME() { + return NAME$c; + } + + // Public + next() { + this._slide(ORDER_NEXT); + } + nextWhenVisible() { + // FIXME TODO use `document.visibilityState` + // Don't call next when the page isn't visible + // or the carousel or its parent isn't visible + if (!document.hidden && isVisible(this._element)) { + this.next(); + } + } + prev() { + this._slide(ORDER_PREV); + } + pause() { + if (this._isSliding) { + triggerTransitionEnd(this._element); + } + this._clearInterval(); + } + cycle() { + this._clearInterval(); + this._updateInterval(); + this._interval = setInterval(() => this.nextWhenVisible(), this._config.interval); + } + _maybeEnableCycle() { + if (!this._config.ride) { + return; + } + if (this._isSliding) { + EventHandler.one(this._element, EVENT_SLID, () => this.cycle()); + return; + } + this.cycle(); + } + to(index) { + const items = this._getItems(); + if (index > items.length - 1 || index < 0) { + return; + } + if (this._isSliding) { + EventHandler.one(this._element, EVENT_SLID, () => this.to(index)); + return; + } + const activeIndex = this._getItemIndex(this._getActive()); + if (activeIndex === index) { + return; + } + const order = index > activeIndex ? ORDER_NEXT : ORDER_PREV; + this._slide(order, items[index]); + } + dispose() { + if (this._swipeHelper) { + this._swipeHelper.dispose(); + } + super.dispose(); + } + + // Private + _configAfterMerge(config) { + config.defaultInterval = config.interval; + return config; + } + _addEventListeners() { + if (this._config.keyboard) { + EventHandler.on(this._element, EVENT_KEYDOWN$1, event => this._keydown(event)); + } + if (this._config.pause === 'hover') { + EventHandler.on(this._element, EVENT_MOUSEENTER$1, () => this.pause()); + EventHandler.on(this._element, EVENT_MOUSELEAVE$1, () => this._maybeEnableCycle()); + } + if (this._config.touch && Swipe.isSupported()) { + this._addTouchEventListeners(); + } + } + _addTouchEventListeners() { + for (const img of SelectorEngine.find(SELECTOR_ITEM_IMG, this._element)) { + EventHandler.on(img, EVENT_DRAG_START, event => event.preventDefault()); + } + const endCallBack = () => { + if (this._config.pause !== 'hover') { + return; + } + + // If it's a touch-enabled device, mouseenter/leave are fired as + // part of the mouse compatibility events on first tap - the carousel + // would stop cycling until user tapped out of it; + // here, we listen for touchend, explicitly pause the carousel + // (as if it's the second time we tap on it, mouseenter compat event + // is NOT fired) and after a timeout (to allow for mouse compatibility + // events to fire) we explicitly restart cycling + + this.pause(); + if (this.touchTimeout) { + clearTimeout(this.touchTimeout); + } + this.touchTimeout = setTimeout(() => this._maybeEnableCycle(), TOUCHEVENT_COMPAT_WAIT + this._config.interval); + }; + const swipeConfig = { + leftCallback: () => this._slide(this._directionToOrder(DIRECTION_LEFT)), + rightCallback: () => this._slide(this._directionToOrder(DIRECTION_RIGHT)), + endCallback: endCallBack + }; + this._swipeHelper = new Swipe(this._element, swipeConfig); + } + _keydown(event) { + if (/input|textarea/i.test(event.target.tagName)) { + return; + } + const direction = KEY_TO_DIRECTION[event.key]; + if (direction) { + event.preventDefault(); + this._slide(this._directionToOrder(direction)); + } + } + _getItemIndex(element) { + return this._getItems().indexOf(element); + } + _setActiveIndicatorElement(index) { + if (!this._indicatorsElement) { + return; + } + const activeIndicator = SelectorEngine.findOne(SELECTOR_ACTIVE, this._indicatorsElement); + activeIndicator.classList.remove(CLASS_NAME_ACTIVE$2); + activeIndicator.removeAttribute('aria-current'); + const newActiveIndicator = SelectorEngine.findOne(`[data-bs-slide-to="${index}"]`, this._indicatorsElement); + if (newActiveIndicator) { + newActiveIndicator.classList.add(CLASS_NAME_ACTIVE$2); + newActiveIndicator.setAttribute('aria-current', 'true'); + } + } + _updateInterval() { + const element = this._activeElement || this._getActive(); + if (!element) { + return; + } + const elementInterval = Number.parseInt(element.getAttribute('data-bs-interval'), 10); + this._config.interval = elementInterval || this._config.defaultInterval; + } + _slide(order, element = null) { + if (this._isSliding) { + return; + } + const activeElement = this._getActive(); + const isNext = order === ORDER_NEXT; + const nextElement = element || getNextActiveElement(this._getItems(), activeElement, isNext, this._config.wrap); + if (nextElement === activeElement) { + return; + } + const nextElementIndex = this._getItemIndex(nextElement); + const triggerEvent = eventName => { + return EventHandler.trigger(this._element, eventName, { + relatedTarget: nextElement, + direction: this._orderToDirection(order), + from: this._getItemIndex(activeElement), + to: nextElementIndex + }); + }; + const slideEvent = triggerEvent(EVENT_SLIDE); + if (slideEvent.defaultPrevented) { + return; + } + if (!activeElement || !nextElement) { + // Some weirdness is happening, so we bail + // TODO: change tests that use empty divs to avoid this check + return; + } + const isCycling = Boolean(this._interval); + this.pause(); + this._isSliding = true; + this._setActiveIndicatorElement(nextElementIndex); + this._activeElement = nextElement; + const directionalClassName = isNext ? CLASS_NAME_START : CLASS_NAME_END; + const orderClassName = isNext ? CLASS_NAME_NEXT : CLASS_NAME_PREV; + nextElement.classList.add(orderClassName); + reflow(nextElement); + activeElement.classList.add(directionalClassName); + nextElement.classList.add(directionalClassName); + const completeCallBack = () => { + nextElement.classList.remove(directionalClassName, orderClassName); + nextElement.classList.add(CLASS_NAME_ACTIVE$2); + activeElement.classList.remove(CLASS_NAME_ACTIVE$2, orderClassName, directionalClassName); + this._isSliding = false; + triggerEvent(EVENT_SLID); + }; + this._queueCallback(completeCallBack, activeElement, this._isAnimated()); + if (isCycling) { + this.cycle(); + } + } + _isAnimated() { + return this._element.classList.contains(CLASS_NAME_SLIDE); + } + _getActive() { + return SelectorEngine.findOne(SELECTOR_ACTIVE_ITEM, this._element); + } + _getItems() { + return SelectorEngine.find(SELECTOR_ITEM, this._element); + } + _clearInterval() { + if (this._interval) { + clearInterval(this._interval); + this._interval = null; + } + } + _directionToOrder(direction) { + if (isRTL()) { + return direction === DIRECTION_LEFT ? ORDER_PREV : ORDER_NEXT; + } + return direction === DIRECTION_LEFT ? ORDER_NEXT : ORDER_PREV; + } + _orderToDirection(order) { + if (isRTL()) { + return order === ORDER_PREV ? DIRECTION_LEFT : DIRECTION_RIGHT; + } + return order === ORDER_PREV ? DIRECTION_RIGHT : DIRECTION_LEFT; + } + + // Static + static jQueryInterface(config) { + return this.each(function () { + const data = Carousel.getOrCreateInstance(this, config); + if (typeof config === 'number') { + data.to(config); + return; + } + if (typeof config === 'string') { + if (data[config] === undefined || config.startsWith('_') || config === 'constructor') { + throw new TypeError(`No method named "${config}"`); + } + data[config](); + } + }); + } +} + +/** + * Data API implementation + */ + +EventHandler.on(document, EVENT_CLICK_DATA_API$5, SELECTOR_DATA_SLIDE, function (event) { + const target = SelectorEngine.getElementFromSelector(this); + if (!target || !target.classList.contains(CLASS_NAME_CAROUSEL)) { + return; + } + event.preventDefault(); + const carousel = Carousel.getOrCreateInstance(target); + const slideIndex = this.getAttribute('data-bs-slide-to'); + if (slideIndex) { + carousel.to(slideIndex); + carousel._maybeEnableCycle(); + return; + } + if (Manipulator.getDataAttribute(this, 'slide') === 'next') { + carousel.next(); + carousel._maybeEnableCycle(); + return; + } + carousel.prev(); + carousel._maybeEnableCycle(); +}); +EventHandler.on(window, EVENT_LOAD_DATA_API$3, () => { + const carousels = SelectorEngine.find(SELECTOR_DATA_RIDE); + for (const carousel of carousels) { + Carousel.getOrCreateInstance(carousel); + } +}); + +/** + * jQuery + */ + +defineJQueryPlugin(Carousel); + +/** + * -------------------------------------------------------------------------- + * Bootstrap collapse.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$b = 'collapse'; +const DATA_KEY$7 = 'bs.collapse'; +const EVENT_KEY$7 = `.${DATA_KEY$7}`; +const DATA_API_KEY$4 = '.data-api'; +const EVENT_SHOW$6 = `show${EVENT_KEY$7}`; +const EVENT_SHOWN$6 = `shown${EVENT_KEY$7}`; +const EVENT_HIDE$6 = `hide${EVENT_KEY$7}`; +const EVENT_HIDDEN$6 = `hidden${EVENT_KEY$7}`; +const EVENT_CLICK_DATA_API$4 = `click${EVENT_KEY$7}${DATA_API_KEY$4}`; +const CLASS_NAME_SHOW$7 = 'show'; +const CLASS_NAME_COLLAPSE = 'collapse'; +const CLASS_NAME_COLLAPSING = 'collapsing'; +const CLASS_NAME_COLLAPSED = 'collapsed'; +const CLASS_NAME_DEEPER_CHILDREN = `:scope .${CLASS_NAME_COLLAPSE} .${CLASS_NAME_COLLAPSE}`; +const CLASS_NAME_HORIZONTAL = 'collapse-horizontal'; +const WIDTH = 'width'; +const HEIGHT = 'height'; +const SELECTOR_ACTIVES = '.collapse.show, .collapse.collapsing'; +const SELECTOR_DATA_TOGGLE$4 = '[data-bs-toggle="collapse"]'; +const Default$a = { + parent: null, + toggle: true +}; +const DefaultType$a = { + parent: '(null|element)', + toggle: 'boolean' +}; + +/** + * Class definition + */ + +class Collapse extends BaseComponent { + constructor(element, config) { + super(element, config); + this._isTransitioning = false; + this._triggerArray = []; + const toggleList = SelectorEngine.find(SELECTOR_DATA_TOGGLE$4); + for (const elem of toggleList) { + const selector = SelectorEngine.getSelectorFromElement(elem); + const filterElement = SelectorEngine.find(selector).filter(foundElement => foundElement === this._element); + if (selector !== null && filterElement.length) { + this._triggerArray.push(elem); + } + } + this._initializeChildren(); + if (!this._config.parent) { + this._addAriaAndCollapsedClass(this._triggerArray, this._isShown()); + } + if (this._config.toggle) { + this.toggle(); + } + } + + // Getters + static get Default() { + return Default$a; + } + static get DefaultType() { + return DefaultType$a; + } + static get NAME() { + return NAME$b; + } + + // Public + toggle() { + if (this._isShown()) { + this.hide(); + } else { + this.show(); + } + } + show() { + if (this._isTransitioning || this._isShown()) { + return; + } + let activeChildren = []; + + // find active children + if (this._config.parent) { + activeChildren = this._getFirstLevelChildren(SELECTOR_ACTIVES).filter(element => element !== this._element).map(element => Collapse.getOrCreateInstance(element, { + toggle: false + })); + } + if (activeChildren.length && activeChildren[0]._isTransitioning) { + return; + } + const startEvent = EventHandler.trigger(this._element, EVENT_SHOW$6); + if (startEvent.defaultPrevented) { + return; + } + for (const activeInstance of activeChildren) { + activeInstance.hide(); + } + const dimension = this._getDimension(); + this._element.classList.remove(CLASS_NAME_COLLAPSE); + this._element.classList.add(CLASS_NAME_COLLAPSING); + this._element.style[dimension] = 0; + this._addAriaAndCollapsedClass(this._triggerArray, true); + this._isTransitioning = true; + const complete = () => { + this._isTransitioning = false; + this._element.classList.remove(CLASS_NAME_COLLAPSING); + this._element.classList.add(CLASS_NAME_COLLAPSE, CLASS_NAME_SHOW$7); + this._element.style[dimension] = ''; + EventHandler.trigger(this._element, EVENT_SHOWN$6); + }; + const capitalizedDimension = dimension[0].toUpperCase() + dimension.slice(1); + const scrollSize = `scroll${capitalizedDimension}`; + this._queueCallback(complete, this._element, true); + this._element.style[dimension] = `${this._element[scrollSize]}px`; + } + hide() { + if (this._isTransitioning || !this._isShown()) { + return; + } + const startEvent = EventHandler.trigger(this._element, EVENT_HIDE$6); + if (startEvent.defaultPrevented) { + return; + } + const dimension = this._getDimension(); + this._element.style[dimension] = `${this._element.getBoundingClientRect()[dimension]}px`; + reflow(this._element); + this._element.classList.add(CLASS_NAME_COLLAPSING); + this._element.classList.remove(CLASS_NAME_COLLAPSE, CLASS_NAME_SHOW$7); + for (const trigger of this._triggerArray) { + const element = SelectorEngine.getElementFromSelector(trigger); + if (element && !this._isShown(element)) { + this._addAriaAndCollapsedClass([trigger], false); + } + } + this._isTransitioning = true; + const complete = () => { + this._isTransitioning = false; + this._element.classList.remove(CLASS_NAME_COLLAPSING); + this._element.classList.add(CLASS_NAME_COLLAPSE); + EventHandler.trigger(this._element, EVENT_HIDDEN$6); + }; + this._element.style[dimension] = ''; + this._queueCallback(complete, this._element, true); + } + + // Private + _isShown(element = this._element) { + return element.classList.contains(CLASS_NAME_SHOW$7); + } + _configAfterMerge(config) { + config.toggle = Boolean(config.toggle); // Coerce string values + config.parent = getElement(config.parent); + return config; + } + _getDimension() { + return this._element.classList.contains(CLASS_NAME_HORIZONTAL) ? WIDTH : HEIGHT; + } + _initializeChildren() { + if (!this._config.parent) { + return; + } + const children = this._getFirstLevelChildren(SELECTOR_DATA_TOGGLE$4); + for (const element of children) { + const selected = SelectorEngine.getElementFromSelector(element); + if (selected) { + this._addAriaAndCollapsedClass([element], this._isShown(selected)); + } + } + } + _getFirstLevelChildren(selector) { + const children = SelectorEngine.find(CLASS_NAME_DEEPER_CHILDREN, this._config.parent); + // remove children if greater depth + return SelectorEngine.find(selector, this._config.parent).filter(element => !children.includes(element)); + } + _addAriaAndCollapsedClass(triggerArray, isOpen) { + if (!triggerArray.length) { + return; + } + for (const element of triggerArray) { + element.classList.toggle(CLASS_NAME_COLLAPSED, !isOpen); + element.setAttribute('aria-expanded', isOpen); + } + } + + // Static + static jQueryInterface(config) { + const _config = {}; + if (typeof config === 'string' && /show|hide/.test(config)) { + _config.toggle = false; + } + return this.each(function () { + const data = Collapse.getOrCreateInstance(this, _config); + if (typeof config === 'string') { + if (typeof data[config] === 'undefined') { + throw new TypeError(`No method named "${config}"`); + } + data[config](); + } + }); + } +} + +/** + * Data API implementation + */ + +EventHandler.on(document, EVENT_CLICK_DATA_API$4, SELECTOR_DATA_TOGGLE$4, function (event) { + // preventDefault only for elements (which change the URL) not inside the collapsible element + if (event.target.tagName === 'A' || event.delegateTarget && event.delegateTarget.tagName === 'A') { + event.preventDefault(); + } + for (const element of SelectorEngine.getMultipleElementsFromSelector(this)) { + Collapse.getOrCreateInstance(element, { + toggle: false + }).toggle(); + } +}); + +/** + * jQuery + */ + +defineJQueryPlugin(Collapse); + +/** + * -------------------------------------------------------------------------- + * Bootstrap dropdown.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$a = 'dropdown'; +const DATA_KEY$6 = 'bs.dropdown'; +const EVENT_KEY$6 = `.${DATA_KEY$6}`; +const DATA_API_KEY$3 = '.data-api'; +const ESCAPE_KEY$2 = 'Escape'; +const TAB_KEY$1 = 'Tab'; +const ARROW_UP_KEY$1 = 'ArrowUp'; +const ARROW_DOWN_KEY$1 = 'ArrowDown'; +const RIGHT_MOUSE_BUTTON = 2; // MouseEvent.button value for the secondary button, usually the right button + +const EVENT_HIDE$5 = `hide${EVENT_KEY$6}`; +const EVENT_HIDDEN$5 = `hidden${EVENT_KEY$6}`; +const EVENT_SHOW$5 = `show${EVENT_KEY$6}`; +const EVENT_SHOWN$5 = `shown${EVENT_KEY$6}`; +const EVENT_CLICK_DATA_API$3 = `click${EVENT_KEY$6}${DATA_API_KEY$3}`; +const EVENT_KEYDOWN_DATA_API = `keydown${EVENT_KEY$6}${DATA_API_KEY$3}`; +const EVENT_KEYUP_DATA_API = `keyup${EVENT_KEY$6}${DATA_API_KEY$3}`; +const CLASS_NAME_SHOW$6 = 'show'; +const CLASS_NAME_DROPUP = 'dropup'; +const CLASS_NAME_DROPEND = 'dropend'; +const CLASS_NAME_DROPSTART = 'dropstart'; +const CLASS_NAME_DROPUP_CENTER = 'dropup-center'; +const CLASS_NAME_DROPDOWN_CENTER = 'dropdown-center'; +const SELECTOR_DATA_TOGGLE$3 = '[data-bs-toggle="dropdown"]:not(.disabled):not(:disabled)'; +const SELECTOR_DATA_TOGGLE_SHOWN = `${SELECTOR_DATA_TOGGLE$3}.${CLASS_NAME_SHOW$6}`; +const SELECTOR_MENU = '.dropdown-menu'; +const SELECTOR_NAVBAR = '.navbar'; +const SELECTOR_NAVBAR_NAV = '.navbar-nav'; +const SELECTOR_VISIBLE_ITEMS = '.dropdown-menu .dropdown-item:not(.disabled):not(:disabled)'; +const PLACEMENT_TOP = isRTL() ? 'top-end' : 'top-start'; +const PLACEMENT_TOPEND = isRTL() ? 'top-start' : 'top-end'; +const PLACEMENT_BOTTOM = isRTL() ? 'bottom-end' : 'bottom-start'; +const PLACEMENT_BOTTOMEND = isRTL() ? 'bottom-start' : 'bottom-end'; +const PLACEMENT_RIGHT = isRTL() ? 'left-start' : 'right-start'; +const PLACEMENT_LEFT = isRTL() ? 'right-start' : 'left-start'; +const PLACEMENT_TOPCENTER = 'top'; +const PLACEMENT_BOTTOMCENTER = 'bottom'; +const Default$9 = { + autoClose: true, + boundary: 'clippingParents', + display: 'dynamic', + offset: [0, 2], + popperConfig: null, + reference: 'toggle' +}; +const DefaultType$9 = { + autoClose: '(boolean|string)', + boundary: '(string|element)', + display: 'string', + offset: '(array|string|function)', + popperConfig: '(null|object|function)', + reference: '(string|element|object)' +}; + +/** + * Class definition + */ + +class Dropdown extends BaseComponent { + constructor(element, config) { + super(element, config); + this._popper = null; + this._parent = this._element.parentNode; // dropdown wrapper + // TODO: v6 revert #37011 & change markup https://getbootstrap.com/docs/5.3/forms/input-group/ + this._menu = SelectorEngine.next(this._element, SELECTOR_MENU)[0] || SelectorEngine.prev(this._element, SELECTOR_MENU)[0] || SelectorEngine.findOne(SELECTOR_MENU, this._parent); + this._inNavbar = this._detectNavbar(); + } + + // Getters + static get Default() { + return Default$9; + } + static get DefaultType() { + return DefaultType$9; + } + static get NAME() { + return NAME$a; + } + + // Public + toggle() { + return this._isShown() ? this.hide() : this.show(); + } + show() { + if (isDisabled(this._element) || this._isShown()) { + return; + } + const relatedTarget = { + relatedTarget: this._element + }; + const showEvent = EventHandler.trigger(this._element, EVENT_SHOW$5, relatedTarget); + if (showEvent.defaultPrevented) { + return; + } + this._createPopper(); + + // If this is a touch-enabled device we add extra + // empty mouseover listeners to the body's immediate children; + // only needed because of broken event delegation on iOS + // https://www.quirksmode.org/blog/archives/2014/02/mouse_event_bub.html + if ('ontouchstart' in document.documentElement && !this._parent.closest(SELECTOR_NAVBAR_NAV)) { + for (const element of [].concat(...document.body.children)) { + EventHandler.on(element, 'mouseover', noop); + } + } + this._element.focus(); + this._element.setAttribute('aria-expanded', true); + this._menu.classList.add(CLASS_NAME_SHOW$6); + this._element.classList.add(CLASS_NAME_SHOW$6); + EventHandler.trigger(this._element, EVENT_SHOWN$5, relatedTarget); + } + hide() { + if (isDisabled(this._element) || !this._isShown()) { + return; + } + const relatedTarget = { + relatedTarget: this._element + }; + this._completeHide(relatedTarget); + } + dispose() { + if (this._popper) { + this._popper.destroy(); + } + super.dispose(); + } + update() { + this._inNavbar = this._detectNavbar(); + if (this._popper) { + this._popper.update(); + } + } + + // Private + _completeHide(relatedTarget) { + const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE$5, relatedTarget); + if (hideEvent.defaultPrevented) { + return; + } + + // If this is a touch-enabled device we remove the extra + // empty mouseover listeners we added for iOS support + if ('ontouchstart' in document.documentElement) { + for (const element of [].concat(...document.body.children)) { + EventHandler.off(element, 'mouseover', noop); + } + } + if (this._popper) { + this._popper.destroy(); + } + this._menu.classList.remove(CLASS_NAME_SHOW$6); + this._element.classList.remove(CLASS_NAME_SHOW$6); + this._element.setAttribute('aria-expanded', 'false'); + Manipulator.removeDataAttribute(this._menu, 'popper'); + EventHandler.trigger(this._element, EVENT_HIDDEN$5, relatedTarget); + + // Explicitly return focus to the trigger element + this._element.focus(); + } + _getConfig(config) { + config = super._getConfig(config); + if (typeof config.reference === 'object' && !isElement(config.reference) && typeof config.reference.getBoundingClientRect !== 'function') { + // Popper virtual elements require a getBoundingClientRect method + throw new TypeError(`${NAME$a.toUpperCase()}: Option "reference" provided type "object" without a required "getBoundingClientRect" method.`); + } + return config; + } + _createPopper() { + if (typeof Popper === 'undefined') { + throw new TypeError('Bootstrap\'s dropdowns require Popper (https://popper.js.org/docs/v2/)'); + } + let referenceElement = this._element; + if (this._config.reference === 'parent') { + referenceElement = this._parent; + } else if (isElement(this._config.reference)) { + referenceElement = getElement(this._config.reference); + } else if (typeof this._config.reference === 'object') { + referenceElement = this._config.reference; + } + const popperConfig = this._getPopperConfig(); + this._popper = createPopper(referenceElement, this._menu, popperConfig); + } + _isShown() { + return this._menu.classList.contains(CLASS_NAME_SHOW$6); + } + _getPlacement() { + const parentDropdown = this._parent; + if (parentDropdown.classList.contains(CLASS_NAME_DROPEND)) { + return PLACEMENT_RIGHT; + } + if (parentDropdown.classList.contains(CLASS_NAME_DROPSTART)) { + return PLACEMENT_LEFT; + } + if (parentDropdown.classList.contains(CLASS_NAME_DROPUP_CENTER)) { + return PLACEMENT_TOPCENTER; + } + if (parentDropdown.classList.contains(CLASS_NAME_DROPDOWN_CENTER)) { + return PLACEMENT_BOTTOMCENTER; + } + + // We need to trim the value because custom properties can also include spaces + const isEnd = getComputedStyle(this._menu).getPropertyValue('--bs-position').trim() === 'end'; + if (parentDropdown.classList.contains(CLASS_NAME_DROPUP)) { + return isEnd ? PLACEMENT_TOPEND : PLACEMENT_TOP; + } + return isEnd ? PLACEMENT_BOTTOMEND : PLACEMENT_BOTTOM; + } + _detectNavbar() { + return this._element.closest(SELECTOR_NAVBAR) !== null; + } + _getOffset() { + const { + offset + } = this._config; + if (typeof offset === 'string') { + return offset.split(',').map(value => Number.parseInt(value, 10)); + } + if (typeof offset === 'function') { + return popperData => offset(popperData, this._element); + } + return offset; + } + _getPopperConfig() { + const defaultBsPopperConfig = { + placement: this._getPlacement(), + modifiers: [{ + name: 'preventOverflow', + options: { + boundary: this._config.boundary + } + }, { + name: 'offset', + options: { + offset: this._getOffset() + } + }] + }; + + // Disable Popper if we have a static display or Dropdown is in Navbar + if (this._inNavbar || this._config.display === 'static') { + Manipulator.setDataAttribute(this._menu, 'popper', 'static'); // TODO: v6 remove + defaultBsPopperConfig.modifiers = [{ + name: 'applyStyles', + enabled: false + }]; + } + return { + ...defaultBsPopperConfig, + ...execute(this._config.popperConfig, [undefined, defaultBsPopperConfig]) + }; + } + _selectMenuItem({ + key, + target + }) { + const items = SelectorEngine.find(SELECTOR_VISIBLE_ITEMS, this._menu).filter(element => isVisible(element)); + if (!items.length) { + return; + } + + // if target isn't included in items (e.g. when expanding the dropdown) + // allow cycling to get the last item in case key equals ARROW_UP_KEY + getNextActiveElement(items, target, key === ARROW_DOWN_KEY$1, !items.includes(target)).focus(); + } + + // Static + static jQueryInterface(config) { + return this.each(function () { + const data = Dropdown.getOrCreateInstance(this, config); + if (typeof config !== 'string') { + return; + } + if (typeof data[config] === 'undefined') { + throw new TypeError(`No method named "${config}"`); + } + data[config](); + }); + } + static clearMenus(event) { + if (event.button === RIGHT_MOUSE_BUTTON || event.type === 'keyup' && event.key !== TAB_KEY$1) { + return; + } + const openToggles = SelectorEngine.find(SELECTOR_DATA_TOGGLE_SHOWN); + for (const toggle of openToggles) { + const context = Dropdown.getInstance(toggle); + if (!context || context._config.autoClose === false) { + continue; + } + const composedPath = event.composedPath(); + const isMenuTarget = composedPath.includes(context._menu); + if (composedPath.includes(context._element) || context._config.autoClose === 'inside' && !isMenuTarget || context._config.autoClose === 'outside' && isMenuTarget) { + continue; + } + + // Tab navigation through the dropdown menu or events from contained inputs shouldn't close the menu + if (context._menu.contains(event.target) && (event.type === 'keyup' && event.key === TAB_KEY$1 || /input|select|option|textarea|form/i.test(event.target.tagName))) { + continue; + } + const relatedTarget = { + relatedTarget: context._element + }; + if (event.type === 'click') { + relatedTarget.clickEvent = event; + } + context._completeHide(relatedTarget); + } + } + static dataApiKeydownHandler(event) { + // If not an UP | DOWN | ESCAPE key => not a dropdown command + // If input/textarea && if key is other than ESCAPE => not a dropdown command + + const isInput = /input|textarea/i.test(event.target.tagName); + const isEscapeEvent = event.key === ESCAPE_KEY$2; + const isUpOrDownEvent = [ARROW_UP_KEY$1, ARROW_DOWN_KEY$1].includes(event.key); + if (!isUpOrDownEvent && !isEscapeEvent) { + return; + } + if (isInput && !isEscapeEvent) { + return; + } + event.preventDefault(); + + // TODO: v6 revert #37011 & change markup https://getbootstrap.com/docs/5.3/forms/input-group/ + const getToggleButton = this.matches(SELECTOR_DATA_TOGGLE$3) ? this : SelectorEngine.prev(this, SELECTOR_DATA_TOGGLE$3)[0] || SelectorEngine.next(this, SELECTOR_DATA_TOGGLE$3)[0] || SelectorEngine.findOne(SELECTOR_DATA_TOGGLE$3, event.delegateTarget.parentNode); + const instance = Dropdown.getOrCreateInstance(getToggleButton); + if (isUpOrDownEvent) { + event.stopPropagation(); + instance.show(); + instance._selectMenuItem(event); + return; + } + if (instance._isShown()) { + // else is escape and we check if it is shown + event.stopPropagation(); + instance.hide(); + getToggleButton.focus(); + } + } +} + +/** + * Data API implementation + */ + +EventHandler.on(document, EVENT_KEYDOWN_DATA_API, SELECTOR_DATA_TOGGLE$3, Dropdown.dataApiKeydownHandler); +EventHandler.on(document, EVENT_KEYDOWN_DATA_API, SELECTOR_MENU, Dropdown.dataApiKeydownHandler); +EventHandler.on(document, EVENT_CLICK_DATA_API$3, Dropdown.clearMenus); +EventHandler.on(document, EVENT_KEYUP_DATA_API, Dropdown.clearMenus); +EventHandler.on(document, EVENT_CLICK_DATA_API$3, SELECTOR_DATA_TOGGLE$3, function (event) { + event.preventDefault(); + Dropdown.getOrCreateInstance(this).toggle(); +}); + +/** + * jQuery + */ + +defineJQueryPlugin(Dropdown); + +/** + * -------------------------------------------------------------------------- + * Bootstrap util/backdrop.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$9 = 'backdrop'; +const CLASS_NAME_FADE$4 = 'fade'; +const CLASS_NAME_SHOW$5 = 'show'; +const EVENT_MOUSEDOWN = `mousedown.bs.${NAME$9}`; +const Default$8 = { + className: 'modal-backdrop', + clickCallback: null, + isAnimated: false, + isVisible: true, + // if false, we use the backdrop helper without adding any element to the dom + rootElement: 'body' // give the choice to place backdrop under different elements +}; +const DefaultType$8 = { + className: 'string', + clickCallback: '(function|null)', + isAnimated: 'boolean', + isVisible: 'boolean', + rootElement: '(element|string)' +}; + +/** + * Class definition + */ + +class Backdrop extends Config { + constructor(config) { + super(); + this._config = this._getConfig(config); + this._isAppended = false; + this._element = null; + } + + // Getters + static get Default() { + return Default$8; + } + static get DefaultType() { + return DefaultType$8; + } + static get NAME() { + return NAME$9; + } + + // Public + show(callback) { + if (!this._config.isVisible) { + execute(callback); + return; + } + this._append(); + const element = this._getElement(); + if (this._config.isAnimated) { + reflow(element); + } + element.classList.add(CLASS_NAME_SHOW$5); + this._emulateAnimation(() => { + execute(callback); + }); + } + hide(callback) { + if (!this._config.isVisible) { + execute(callback); + return; + } + this._getElement().classList.remove(CLASS_NAME_SHOW$5); + this._emulateAnimation(() => { + this.dispose(); + execute(callback); + }); + } + dispose() { + if (!this._isAppended) { + return; + } + EventHandler.off(this._element, EVENT_MOUSEDOWN); + this._element.remove(); + this._isAppended = false; + } + + // Private + _getElement() { + if (!this._element) { + const backdrop = document.createElement('div'); + backdrop.className = this._config.className; + if (this._config.isAnimated) { + backdrop.classList.add(CLASS_NAME_FADE$4); + } + this._element = backdrop; + } + return this._element; + } + _configAfterMerge(config) { + // use getElement() with the default "body" to get a fresh Element on each instantiation + config.rootElement = getElement(config.rootElement); + return config; + } + _append() { + if (this._isAppended) { + return; + } + const element = this._getElement(); + this._config.rootElement.append(element); + EventHandler.on(element, EVENT_MOUSEDOWN, () => { + execute(this._config.clickCallback); + }); + this._isAppended = true; + } + _emulateAnimation(callback) { + executeAfterTransition(callback, this._getElement(), this._config.isAnimated); + } +} + +/** + * -------------------------------------------------------------------------- + * Bootstrap util/focustrap.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$8 = 'focustrap'; +const DATA_KEY$5 = 'bs.focustrap'; +const EVENT_KEY$5 = `.${DATA_KEY$5}`; +const EVENT_FOCUSIN$2 = `focusin${EVENT_KEY$5}`; +const EVENT_KEYDOWN_TAB = `keydown.tab${EVENT_KEY$5}`; +const TAB_KEY = 'Tab'; +const TAB_NAV_FORWARD = 'forward'; +const TAB_NAV_BACKWARD = 'backward'; +const Default$7 = { + autofocus: true, + trapElement: null // The element to trap focus inside of +}; +const DefaultType$7 = { + autofocus: 'boolean', + trapElement: 'element' +}; + +/** + * Class definition + */ + +class FocusTrap extends Config { + constructor(config) { + super(); + this._config = this._getConfig(config); + this._isActive = false; + this._lastTabNavDirection = null; + } + + // Getters + static get Default() { + return Default$7; + } + static get DefaultType() { + return DefaultType$7; + } + static get NAME() { + return NAME$8; + } + + // Public + activate() { + if (this._isActive) { + return; + } + if (this._config.autofocus) { + this._config.trapElement.focus(); + } + EventHandler.off(document, EVENT_KEY$5); // guard against infinite focus loop + EventHandler.on(document, EVENT_FOCUSIN$2, event => this._handleFocusin(event)); + EventHandler.on(document, EVENT_KEYDOWN_TAB, event => this._handleKeydown(event)); + this._isActive = true; + } + deactivate() { + if (!this._isActive) { + return; + } + this._isActive = false; + EventHandler.off(document, EVENT_KEY$5); + } + + // Private + _handleFocusin(event) { + const { + trapElement + } = this._config; + if (event.target === document || event.target === trapElement || trapElement.contains(event.target)) { + return; + } + const elements = SelectorEngine.focusableChildren(trapElement); + if (elements.length === 0) { + trapElement.focus(); + } else if (this._lastTabNavDirection === TAB_NAV_BACKWARD) { + elements[elements.length - 1].focus(); + } else { + elements[0].focus(); + } + } + _handleKeydown(event) { + if (event.key !== TAB_KEY) { + return; + } + this._lastTabNavDirection = event.shiftKey ? TAB_NAV_BACKWARD : TAB_NAV_FORWARD; + } +} + +/** + * -------------------------------------------------------------------------- + * Bootstrap util/scrollBar.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const SELECTOR_FIXED_CONTENT = '.fixed-top, .fixed-bottom, .is-fixed, .sticky-top'; +const SELECTOR_STICKY_CONTENT = '.sticky-top'; +const PROPERTY_PADDING = 'padding-right'; +const PROPERTY_MARGIN = 'margin-right'; + +/** + * Class definition + */ + +class ScrollBarHelper { + constructor() { + this._element = document.body; + } + + // Public + getWidth() { + // https://developer.mozilla.org/en-US/docs/Web/API/Window/innerWidth#usage_notes + const documentWidth = document.documentElement.clientWidth; + return Math.abs(window.innerWidth - documentWidth); + } + hide() { + const width = this.getWidth(); + this._disableOverFlow(); + // give padding to element to balance the hidden scrollbar width + this._setElementAttributes(this._element, PROPERTY_PADDING, calculatedValue => calculatedValue + width); + // trick: We adjust positive paddingRight and negative marginRight to sticky-top elements to keep showing fullwidth + this._setElementAttributes(SELECTOR_FIXED_CONTENT, PROPERTY_PADDING, calculatedValue => calculatedValue + width); + this._setElementAttributes(SELECTOR_STICKY_CONTENT, PROPERTY_MARGIN, calculatedValue => calculatedValue - width); + } + reset() { + this._resetElementAttributes(this._element, 'overflow'); + this._resetElementAttributes(this._element, PROPERTY_PADDING); + this._resetElementAttributes(SELECTOR_FIXED_CONTENT, PROPERTY_PADDING); + this._resetElementAttributes(SELECTOR_STICKY_CONTENT, PROPERTY_MARGIN); + } + isOverflowing() { + return this.getWidth() > 0; + } + + // Private + _disableOverFlow() { + this._saveInitialAttribute(this._element, 'overflow'); + this._element.style.overflow = 'hidden'; + } + _setElementAttributes(selector, styleProperty, callback) { + const scrollbarWidth = this.getWidth(); + const manipulationCallBack = element => { + if (element !== this._element && window.innerWidth > element.clientWidth + scrollbarWidth) { + return; + } + this._saveInitialAttribute(element, styleProperty); + const calculatedValue = window.getComputedStyle(element).getPropertyValue(styleProperty); + element.style.setProperty(styleProperty, `${callback(Number.parseFloat(calculatedValue))}px`); + }; + this._applyManipulationCallback(selector, manipulationCallBack); + } + _saveInitialAttribute(element, styleProperty) { + const actualValue = element.style.getPropertyValue(styleProperty); + if (actualValue) { + Manipulator.setDataAttribute(element, styleProperty, actualValue); + } + } + _resetElementAttributes(selector, styleProperty) { + const manipulationCallBack = element => { + const value = Manipulator.getDataAttribute(element, styleProperty); + // We only want to remove the property if the value is `null`; the value can also be zero + if (value === null) { + element.style.removeProperty(styleProperty); + return; + } + Manipulator.removeDataAttribute(element, styleProperty); + element.style.setProperty(styleProperty, value); + }; + this._applyManipulationCallback(selector, manipulationCallBack); + } + _applyManipulationCallback(selector, callBack) { + if (isElement(selector)) { + callBack(selector); + return; + } + for (const sel of SelectorEngine.find(selector, this._element)) { + callBack(sel); + } + } +} + +/** + * -------------------------------------------------------------------------- + * Bootstrap modal.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$7 = 'modal'; +const DATA_KEY$4 = 'bs.modal'; +const EVENT_KEY$4 = `.${DATA_KEY$4}`; +const DATA_API_KEY$2 = '.data-api'; +const ESCAPE_KEY$1 = 'Escape'; +const EVENT_HIDE$4 = `hide${EVENT_KEY$4}`; +const EVENT_HIDE_PREVENTED$1 = `hidePrevented${EVENT_KEY$4}`; +const EVENT_HIDDEN$4 = `hidden${EVENT_KEY$4}`; +const EVENT_SHOW$4 = `show${EVENT_KEY$4}`; +const EVENT_SHOWN$4 = `shown${EVENT_KEY$4}`; +const EVENT_RESIZE$1 = `resize${EVENT_KEY$4}`; +const EVENT_CLICK_DISMISS = `click.dismiss${EVENT_KEY$4}`; +const EVENT_MOUSEDOWN_DISMISS = `mousedown.dismiss${EVENT_KEY$4}`; +const EVENT_KEYDOWN_DISMISS$1 = `keydown.dismiss${EVENT_KEY$4}`; +const EVENT_CLICK_DATA_API$2 = `click${EVENT_KEY$4}${DATA_API_KEY$2}`; +const CLASS_NAME_OPEN = 'modal-open'; +const CLASS_NAME_FADE$3 = 'fade'; +const CLASS_NAME_SHOW$4 = 'show'; +const CLASS_NAME_STATIC = 'modal-static'; +const OPEN_SELECTOR$1 = '.modal.show'; +const SELECTOR_DIALOG = '.modal-dialog'; +const SELECTOR_MODAL_BODY = '.modal-body'; +const SELECTOR_DATA_TOGGLE$2 = '[data-bs-toggle="modal"]'; +const Default$6 = { + backdrop: true, + focus: true, + keyboard: true +}; +const DefaultType$6 = { + backdrop: '(boolean|string)', + focus: 'boolean', + keyboard: 'boolean' +}; + +/** + * Class definition + */ + +class Modal extends BaseComponent { + constructor(element, config) { + super(element, config); + this._dialog = SelectorEngine.findOne(SELECTOR_DIALOG, this._element); + this._backdrop = this._initializeBackDrop(); + this._focustrap = this._initializeFocusTrap(); + this._isShown = false; + this._isTransitioning = false; + this._scrollBar = new ScrollBarHelper(); + this._addEventListeners(); + } + + // Getters + static get Default() { + return Default$6; + } + static get DefaultType() { + return DefaultType$6; + } + static get NAME() { + return NAME$7; + } + + // Public + toggle(relatedTarget) { + return this._isShown ? this.hide() : this.show(relatedTarget); + } + show(relatedTarget) { + if (this._isShown || this._isTransitioning) { + return; + } + const showEvent = EventHandler.trigger(this._element, EVENT_SHOW$4, { + relatedTarget + }); + if (showEvent.defaultPrevented) { + return; + } + this._isShown = true; + this._isTransitioning = true; + this._scrollBar.hide(); + document.body.classList.add(CLASS_NAME_OPEN); + this._adjustDialog(); + this._backdrop.show(() => this._showElement(relatedTarget)); + } + hide() { + if (!this._isShown || this._isTransitioning) { + return; + } + const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE$4); + if (hideEvent.defaultPrevented) { + return; + } + this._isShown = false; + this._isTransitioning = true; + this._focustrap.deactivate(); + this._element.classList.remove(CLASS_NAME_SHOW$4); + this._queueCallback(() => this._hideModal(), this._element, this._isAnimated()); + } + dispose() { + EventHandler.off(window, EVENT_KEY$4); + EventHandler.off(this._dialog, EVENT_KEY$4); + this._backdrop.dispose(); + this._focustrap.deactivate(); + super.dispose(); + } + handleUpdate() { + this._adjustDialog(); + } + + // Private + _initializeBackDrop() { + return new Backdrop({ + isVisible: Boolean(this._config.backdrop), + // 'static' option will be translated to true, and booleans will keep their value, + isAnimated: this._isAnimated() + }); + } + _initializeFocusTrap() { + return new FocusTrap({ + trapElement: this._element + }); + } + _showElement(relatedTarget) { + // try to append dynamic modal + if (!document.body.contains(this._element)) { + document.body.append(this._element); + } + this._element.style.display = 'block'; + this._element.removeAttribute('aria-hidden'); + this._element.setAttribute('aria-modal', true); + this._element.setAttribute('role', 'dialog'); + this._element.scrollTop = 0; + const modalBody = SelectorEngine.findOne(SELECTOR_MODAL_BODY, this._dialog); + if (modalBody) { + modalBody.scrollTop = 0; + } + reflow(this._element); + this._element.classList.add(CLASS_NAME_SHOW$4); + const transitionComplete = () => { + if (this._config.focus) { + this._focustrap.activate(); + } + this._isTransitioning = false; + EventHandler.trigger(this._element, EVENT_SHOWN$4, { + relatedTarget + }); + }; + this._queueCallback(transitionComplete, this._dialog, this._isAnimated()); + } + _addEventListeners() { + EventHandler.on(this._element, EVENT_KEYDOWN_DISMISS$1, event => { + if (event.key !== ESCAPE_KEY$1) { + return; + } + if (this._config.keyboard) { + this.hide(); + return; + } + this._triggerBackdropTransition(); + }); + EventHandler.on(window, EVENT_RESIZE$1, () => { + if (this._isShown && !this._isTransitioning) { + this._adjustDialog(); + } + }); + EventHandler.on(this._element, EVENT_MOUSEDOWN_DISMISS, event => { + // a bad trick to segregate clicks that may start inside dialog but end outside, and avoid listen to scrollbar clicks + EventHandler.one(this._element, EVENT_CLICK_DISMISS, event2 => { + if (this._element !== event.target || this._element !== event2.target) { + return; + } + if (this._config.backdrop === 'static') { + this._triggerBackdropTransition(); + return; + } + if (this._config.backdrop) { + this.hide(); + } + }); + }); + } + _hideModal() { + this._element.style.display = 'none'; + this._element.setAttribute('aria-hidden', true); + this._element.removeAttribute('aria-modal'); + this._element.removeAttribute('role'); + this._isTransitioning = false; + this._backdrop.hide(() => { + document.body.classList.remove(CLASS_NAME_OPEN); + this._resetAdjustments(); + this._scrollBar.reset(); + EventHandler.trigger(this._element, EVENT_HIDDEN$4); + }); + } + _isAnimated() { + return this._element.classList.contains(CLASS_NAME_FADE$3); + } + _triggerBackdropTransition() { + const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE_PREVENTED$1); + if (hideEvent.defaultPrevented) { + return; + } + const isModalOverflowing = this._element.scrollHeight > document.documentElement.clientHeight; + const initialOverflowY = this._element.style.overflowY; + // return if the following background transition hasn't yet completed + if (initialOverflowY === 'hidden' || this._element.classList.contains(CLASS_NAME_STATIC)) { + return; + } + if (!isModalOverflowing) { + this._element.style.overflowY = 'hidden'; + } + this._element.classList.add(CLASS_NAME_STATIC); + this._queueCallback(() => { + this._element.classList.remove(CLASS_NAME_STATIC); + this._queueCallback(() => { + this._element.style.overflowY = initialOverflowY; + }, this._dialog); + }, this._dialog); + this._element.focus(); + } + + /** + * The following methods are used to handle overflowing modals + */ + + _adjustDialog() { + const isModalOverflowing = this._element.scrollHeight > document.documentElement.clientHeight; + const scrollbarWidth = this._scrollBar.getWidth(); + const isBodyOverflowing = scrollbarWidth > 0; + if (isBodyOverflowing && !isModalOverflowing) { + const property = isRTL() ? 'paddingLeft' : 'paddingRight'; + this._element.style[property] = `${scrollbarWidth}px`; + } + if (!isBodyOverflowing && isModalOverflowing) { + const property = isRTL() ? 'paddingRight' : 'paddingLeft'; + this._element.style[property] = `${scrollbarWidth}px`; + } + } + _resetAdjustments() { + this._element.style.paddingLeft = ''; + this._element.style.paddingRight = ''; + } + + // Static + static jQueryInterface(config, relatedTarget) { + return this.each(function () { + const data = Modal.getOrCreateInstance(this, config); + if (typeof config !== 'string') { + return; + } + if (typeof data[config] === 'undefined') { + throw new TypeError(`No method named "${config}"`); + } + data[config](relatedTarget); + }); + } +} + +/** + * Data API implementation + */ + +EventHandler.on(document, EVENT_CLICK_DATA_API$2, SELECTOR_DATA_TOGGLE$2, function (event) { + const target = SelectorEngine.getElementFromSelector(this); + if (['A', 'AREA'].includes(this.tagName)) { + event.preventDefault(); + } + EventHandler.one(target, EVENT_SHOW$4, showEvent => { + if (showEvent.defaultPrevented) { + // only register focus restorer if modal will actually get shown + return; + } + EventHandler.one(target, EVENT_HIDDEN$4, () => { + if (isVisible(this)) { + this.focus(); + } + }); + }); + + // avoid conflict when clicking modal toggler while another one is open + const alreadyOpen = SelectorEngine.findOne(OPEN_SELECTOR$1); + if (alreadyOpen) { + Modal.getInstance(alreadyOpen).hide(); + } + const data = Modal.getOrCreateInstance(target); + data.toggle(this); +}); +enableDismissTrigger(Modal); + +/** + * jQuery + */ + +defineJQueryPlugin(Modal); + +/** + * -------------------------------------------------------------------------- + * Bootstrap offcanvas.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$6 = 'offcanvas'; +const DATA_KEY$3 = 'bs.offcanvas'; +const EVENT_KEY$3 = `.${DATA_KEY$3}`; +const DATA_API_KEY$1 = '.data-api'; +const EVENT_LOAD_DATA_API$2 = `load${EVENT_KEY$3}${DATA_API_KEY$1}`; +const ESCAPE_KEY = 'Escape'; +const CLASS_NAME_SHOW$3 = 'show'; +const CLASS_NAME_SHOWING$1 = 'showing'; +const CLASS_NAME_HIDING = 'hiding'; +const CLASS_NAME_BACKDROP = 'offcanvas-backdrop'; +const OPEN_SELECTOR = '.offcanvas.show'; +const EVENT_SHOW$3 = `show${EVENT_KEY$3}`; +const EVENT_SHOWN$3 = `shown${EVENT_KEY$3}`; +const EVENT_HIDE$3 = `hide${EVENT_KEY$3}`; +const EVENT_HIDE_PREVENTED = `hidePrevented${EVENT_KEY$3}`; +const EVENT_HIDDEN$3 = `hidden${EVENT_KEY$3}`; +const EVENT_RESIZE = `resize${EVENT_KEY$3}`; +const EVENT_CLICK_DATA_API$1 = `click${EVENT_KEY$3}${DATA_API_KEY$1}`; +const EVENT_KEYDOWN_DISMISS = `keydown.dismiss${EVENT_KEY$3}`; +const SELECTOR_DATA_TOGGLE$1 = '[data-bs-toggle="offcanvas"]'; +const Default$5 = { + backdrop: true, + keyboard: true, + scroll: false +}; +const DefaultType$5 = { + backdrop: '(boolean|string)', + keyboard: 'boolean', + scroll: 'boolean' +}; + +/** + * Class definition + */ + +class Offcanvas extends BaseComponent { + constructor(element, config) { + super(element, config); + this._isShown = false; + this._backdrop = this._initializeBackDrop(); + this._focustrap = this._initializeFocusTrap(); + this._addEventListeners(); + } + + // Getters + static get Default() { + return Default$5; + } + static get DefaultType() { + return DefaultType$5; + } + static get NAME() { + return NAME$6; + } + + // Public + toggle(relatedTarget) { + return this._isShown ? this.hide() : this.show(relatedTarget); + } + show(relatedTarget) { + if (this._isShown) { + return; + } + const showEvent = EventHandler.trigger(this._element, EVENT_SHOW$3, { + relatedTarget + }); + if (showEvent.defaultPrevented) { + return; + } + this._isShown = true; + this._backdrop.show(); + if (!this._config.scroll) { + new ScrollBarHelper().hide(); + } + this._element.setAttribute('aria-modal', true); + this._element.setAttribute('role', 'dialog'); + this._element.classList.add(CLASS_NAME_SHOWING$1); + const completeCallBack = () => { + if (!this._config.scroll || this._config.backdrop) { + this._focustrap.activate(); + } + this._element.classList.add(CLASS_NAME_SHOW$3); + this._element.classList.remove(CLASS_NAME_SHOWING$1); + EventHandler.trigger(this._element, EVENT_SHOWN$3, { + relatedTarget + }); + }; + this._queueCallback(completeCallBack, this._element, true); + } + hide() { + if (!this._isShown) { + return; + } + const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE$3); + if (hideEvent.defaultPrevented) { + return; + } + this._focustrap.deactivate(); + this._element.blur(); + this._isShown = false; + this._element.classList.add(CLASS_NAME_HIDING); + this._backdrop.hide(); + const completeCallback = () => { + this._element.classList.remove(CLASS_NAME_SHOW$3, CLASS_NAME_HIDING); + this._element.removeAttribute('aria-modal'); + this._element.removeAttribute('role'); + if (!this._config.scroll) { + new ScrollBarHelper().reset(); + } + EventHandler.trigger(this._element, EVENT_HIDDEN$3); + }; + this._queueCallback(completeCallback, this._element, true); + } + dispose() { + this._backdrop.dispose(); + this._focustrap.deactivate(); + super.dispose(); + } + + // Private + _initializeBackDrop() { + const clickCallback = () => { + if (this._config.backdrop === 'static') { + EventHandler.trigger(this._element, EVENT_HIDE_PREVENTED); + return; + } + this.hide(); + }; + + // 'static' option will be translated to true, and booleans will keep their value + const isVisible = Boolean(this._config.backdrop); + return new Backdrop({ + className: CLASS_NAME_BACKDROP, + isVisible, + isAnimated: true, + rootElement: this._element.parentNode, + clickCallback: isVisible ? clickCallback : null + }); + } + _initializeFocusTrap() { + return new FocusTrap({ + trapElement: this._element + }); + } + _addEventListeners() { + EventHandler.on(this._element, EVENT_KEYDOWN_DISMISS, event => { + if (event.key !== ESCAPE_KEY) { + return; + } + if (this._config.keyboard) { + this.hide(); + return; + } + EventHandler.trigger(this._element, EVENT_HIDE_PREVENTED); + }); + } + + // Static + static jQueryInterface(config) { + return this.each(function () { + const data = Offcanvas.getOrCreateInstance(this, config); + if (typeof config !== 'string') { + return; + } + if (data[config] === undefined || config.startsWith('_') || config === 'constructor') { + throw new TypeError(`No method named "${config}"`); + } + data[config](this); + }); + } +} + +/** + * Data API implementation + */ + +EventHandler.on(document, EVENT_CLICK_DATA_API$1, SELECTOR_DATA_TOGGLE$1, function (event) { + const target = SelectorEngine.getElementFromSelector(this); + if (['A', 'AREA'].includes(this.tagName)) { + event.preventDefault(); + } + if (isDisabled(this)) { + return; + } + EventHandler.one(target, EVENT_HIDDEN$3, () => { + // focus on trigger when it is closed + if (isVisible(this)) { + this.focus(); + } + }); + + // avoid conflict when clicking a toggler of an offcanvas, while another is open + const alreadyOpen = SelectorEngine.findOne(OPEN_SELECTOR); + if (alreadyOpen && alreadyOpen !== target) { + Offcanvas.getInstance(alreadyOpen).hide(); + } + const data = Offcanvas.getOrCreateInstance(target); + data.toggle(this); +}); +EventHandler.on(window, EVENT_LOAD_DATA_API$2, () => { + for (const selector of SelectorEngine.find(OPEN_SELECTOR)) { + Offcanvas.getOrCreateInstance(selector).show(); + } +}); +EventHandler.on(window, EVENT_RESIZE, () => { + for (const element of SelectorEngine.find('[aria-modal][class*=show][class*=offcanvas-]')) { + if (getComputedStyle(element).position !== 'fixed') { + Offcanvas.getOrCreateInstance(element).hide(); + } + } +}); +enableDismissTrigger(Offcanvas); + +/** + * jQuery + */ + +defineJQueryPlugin(Offcanvas); + +/** + * -------------------------------------------------------------------------- + * Bootstrap util/sanitizer.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +// js-docs-start allow-list +const ARIA_ATTRIBUTE_PATTERN = /^aria-[\w-]*$/i; +const DefaultAllowlist = { + // Global attributes allowed on any supplied element below. + '*': ['class', 'dir', 'id', 'lang', 'role', ARIA_ATTRIBUTE_PATTERN], + a: ['target', 'href', 'title', 'rel'], + area: [], + b: [], + br: [], + col: [], + code: [], + dd: [], + div: [], + dl: [], + dt: [], + em: [], + hr: [], + h1: [], + h2: [], + h3: [], + h4: [], + h5: [], + h6: [], + i: [], + img: ['src', 'srcset', 'alt', 'title', 'width', 'height'], + li: [], + ol: [], + p: [], + pre: [], + s: [], + small: [], + span: [], + sub: [], + sup: [], + strong: [], + u: [], + ul: [] +}; +// js-docs-end allow-list + +const uriAttributes = new Set(['background', 'cite', 'href', 'itemtype', 'longdesc', 'poster', 'src', 'xlink:href']); + +/** + * A pattern that recognizes URLs that are safe wrt. XSS in URL navigation + * contexts. + * + * Shout-out to Angular https://github.com/angular/angular/blob/15.2.8/packages/core/src/sanitization/url_sanitizer.ts#L38 + */ +const SAFE_URL_PATTERN = /^(?!javascript:)(?:[a-z0-9+.-]+:|[^&:/?#]*(?:[/?#]|$))/i; +const allowedAttribute = (attribute, allowedAttributeList) => { + const attributeName = attribute.nodeName.toLowerCase(); + if (allowedAttributeList.includes(attributeName)) { + if (uriAttributes.has(attributeName)) { + return Boolean(SAFE_URL_PATTERN.test(attribute.nodeValue)); + } + return true; + } + + // Check if a regular expression validates the attribute. + return allowedAttributeList.filter(attributeRegex => attributeRegex instanceof RegExp).some(regex => regex.test(attributeName)); +}; +function sanitizeHtml(unsafeHtml, allowList, sanitizeFunction) { + if (!unsafeHtml.length) { + return unsafeHtml; + } + if (sanitizeFunction && typeof sanitizeFunction === 'function') { + return sanitizeFunction(unsafeHtml); + } + const domParser = new window.DOMParser(); + const createdDocument = domParser.parseFromString(unsafeHtml, 'text/html'); + const elements = [].concat(...createdDocument.body.querySelectorAll('*')); + for (const element of elements) { + const elementName = element.nodeName.toLowerCase(); + if (!Object.keys(allowList).includes(elementName)) { + element.remove(); + continue; + } + const attributeList = [].concat(...element.attributes); + const allowedAttributes = [].concat(allowList['*'] || [], allowList[elementName] || []); + for (const attribute of attributeList) { + if (!allowedAttribute(attribute, allowedAttributes)) { + element.removeAttribute(attribute.nodeName); + } + } + } + return createdDocument.body.innerHTML; +} + +/** + * -------------------------------------------------------------------------- + * Bootstrap util/template-factory.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$5 = 'TemplateFactory'; +const Default$4 = { + allowList: DefaultAllowlist, + content: {}, + // { selector : text , selector2 : text2 , } + extraClass: '', + html: false, + sanitize: true, + sanitizeFn: null, + template: '
' +}; +const DefaultType$4 = { + allowList: 'object', + content: 'object', + extraClass: '(string|function)', + html: 'boolean', + sanitize: 'boolean', + sanitizeFn: '(null|function)', + template: 'string' +}; +const DefaultContentType = { + entry: '(string|element|function|null)', + selector: '(string|element)' +}; + +/** + * Class definition + */ + +class TemplateFactory extends Config { + constructor(config) { + super(); + this._config = this._getConfig(config); + } + + // Getters + static get Default() { + return Default$4; + } + static get DefaultType() { + return DefaultType$4; + } + static get NAME() { + return NAME$5; + } + + // Public + getContent() { + return Object.values(this._config.content).map(config => this._resolvePossibleFunction(config)).filter(Boolean); + } + hasContent() { + return this.getContent().length > 0; + } + changeContent(content) { + this._checkContent(content); + this._config.content = { + ...this._config.content, + ...content + }; + return this; + } + toHtml() { + const templateWrapper = document.createElement('div'); + templateWrapper.innerHTML = this._maybeSanitize(this._config.template); + for (const [selector, text] of Object.entries(this._config.content)) { + this._setContent(templateWrapper, text, selector); + } + const template = templateWrapper.children[0]; + const extraClass = this._resolvePossibleFunction(this._config.extraClass); + if (extraClass) { + template.classList.add(...extraClass.split(' ')); + } + return template; + } + + // Private + _typeCheckConfig(config) { + super._typeCheckConfig(config); + this._checkContent(config.content); + } + _checkContent(arg) { + for (const [selector, content] of Object.entries(arg)) { + super._typeCheckConfig({ + selector, + entry: content + }, DefaultContentType); + } + } + _setContent(template, content, selector) { + const templateElement = SelectorEngine.findOne(selector, template); + if (!templateElement) { + return; + } + content = this._resolvePossibleFunction(content); + if (!content) { + templateElement.remove(); + return; + } + if (isElement(content)) { + this._putElementInTemplate(getElement(content), templateElement); + return; + } + if (this._config.html) { + templateElement.innerHTML = this._maybeSanitize(content); + return; + } + templateElement.textContent = content; + } + _maybeSanitize(arg) { + return this._config.sanitize ? sanitizeHtml(arg, this._config.allowList, this._config.sanitizeFn) : arg; + } + _resolvePossibleFunction(arg) { + return execute(arg, [undefined, this]); + } + _putElementInTemplate(element, templateElement) { + if (this._config.html) { + templateElement.innerHTML = ''; + templateElement.append(element); + return; + } + templateElement.textContent = element.textContent; + } +} + +/** + * -------------------------------------------------------------------------- + * Bootstrap tooltip.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$4 = 'tooltip'; +const DISALLOWED_ATTRIBUTES = new Set(['sanitize', 'allowList', 'sanitizeFn']); +const CLASS_NAME_FADE$2 = 'fade'; +const CLASS_NAME_MODAL = 'modal'; +const CLASS_NAME_SHOW$2 = 'show'; +const SELECTOR_TOOLTIP_INNER = '.tooltip-inner'; +const SELECTOR_MODAL = `.${CLASS_NAME_MODAL}`; +const EVENT_MODAL_HIDE = 'hide.bs.modal'; +const TRIGGER_HOVER = 'hover'; +const TRIGGER_FOCUS = 'focus'; +const TRIGGER_CLICK = 'click'; +const TRIGGER_MANUAL = 'manual'; +const EVENT_HIDE$2 = 'hide'; +const EVENT_HIDDEN$2 = 'hidden'; +const EVENT_SHOW$2 = 'show'; +const EVENT_SHOWN$2 = 'shown'; +const EVENT_INSERTED = 'inserted'; +const EVENT_CLICK$1 = 'click'; +const EVENT_FOCUSIN$1 = 'focusin'; +const EVENT_FOCUSOUT$1 = 'focusout'; +const EVENT_MOUSEENTER = 'mouseenter'; +const EVENT_MOUSELEAVE = 'mouseleave'; +const AttachmentMap = { + AUTO: 'auto', + TOP: 'top', + RIGHT: isRTL() ? 'left' : 'right', + BOTTOM: 'bottom', + LEFT: isRTL() ? 'right' : 'left' +}; +const Default$3 = { + allowList: DefaultAllowlist, + animation: true, + boundary: 'clippingParents', + container: false, + customClass: '', + delay: 0, + fallbackPlacements: ['top', 'right', 'bottom', 'left'], + html: false, + offset: [0, 6], + placement: 'top', + popperConfig: null, + sanitize: true, + sanitizeFn: null, + selector: false, + template: '', + title: '', + trigger: 'hover focus' +}; +const DefaultType$3 = { + allowList: 'object', + animation: 'boolean', + boundary: '(string|element)', + container: '(string|element|boolean)', + customClass: '(string|function)', + delay: '(number|object)', + fallbackPlacements: 'array', + html: 'boolean', + offset: '(array|string|function)', + placement: '(string|function)', + popperConfig: '(null|object|function)', + sanitize: 'boolean', + sanitizeFn: '(null|function)', + selector: '(string|boolean)', + template: 'string', + title: '(string|element|function)', + trigger: 'string' +}; + +/** + * Class definition + */ + +class Tooltip extends BaseComponent { + constructor(element, config) { + if (typeof Popper === 'undefined') { + throw new TypeError('Bootstrap\'s tooltips require Popper (https://popper.js.org/docs/v2/)'); + } + super(element, config); + + // Private + this._isEnabled = true; + this._timeout = 0; + this._isHovered = null; + this._activeTrigger = {}; + this._popper = null; + this._templateFactory = null; + this._newContent = null; + + // Protected + this.tip = null; + this._setListeners(); + if (!this._config.selector) { + this._fixTitle(); + } + } + + // Getters + static get Default() { + return Default$3; + } + static get DefaultType() { + return DefaultType$3; + } + static get NAME() { + return NAME$4; + } + + // Public + enable() { + this._isEnabled = true; + } + disable() { + this._isEnabled = false; + } + toggleEnabled() { + this._isEnabled = !this._isEnabled; + } + toggle() { + if (!this._isEnabled) { + return; + } + if (this._isShown()) { + this._leave(); + return; + } + this._enter(); + } + dispose() { + clearTimeout(this._timeout); + EventHandler.off(this._element.closest(SELECTOR_MODAL), EVENT_MODAL_HIDE, this._hideModalHandler); + if (this._element.getAttribute('data-bs-original-title')) { + this._element.setAttribute('title', this._element.getAttribute('data-bs-original-title')); + } + this._disposePopper(); + super.dispose(); + } + show() { + if (this._element.style.display === 'none') { + throw new Error('Please use show on visible elements'); + } + if (!(this._isWithContent() && this._isEnabled)) { + return; + } + const showEvent = EventHandler.trigger(this._element, this.constructor.eventName(EVENT_SHOW$2)); + const shadowRoot = findShadowRoot(this._element); + const isInTheDom = (shadowRoot || this._element.ownerDocument.documentElement).contains(this._element); + if (showEvent.defaultPrevented || !isInTheDom) { + return; + } + + // TODO: v6 remove this or make it optional + this._disposePopper(); + const tip = this._getTipElement(); + this._element.setAttribute('aria-describedby', tip.getAttribute('id')); + const { + container + } = this._config; + if (!this._element.ownerDocument.documentElement.contains(this.tip)) { + container.append(tip); + EventHandler.trigger(this._element, this.constructor.eventName(EVENT_INSERTED)); + } + this._popper = this._createPopper(tip); + tip.classList.add(CLASS_NAME_SHOW$2); + + // If this is a touch-enabled device we add extra + // empty mouseover listeners to the body's immediate children; + // only needed because of broken event delegation on iOS + // https://www.quirksmode.org/blog/archives/2014/02/mouse_event_bub.html + if ('ontouchstart' in document.documentElement) { + for (const element of [].concat(...document.body.children)) { + EventHandler.on(element, 'mouseover', noop); + } + } + const complete = () => { + EventHandler.trigger(this._element, this.constructor.eventName(EVENT_SHOWN$2)); + if (this._isHovered === false) { + this._leave(); + } + this._isHovered = false; + }; + this._queueCallback(complete, this.tip, this._isAnimated()); + } + hide() { + if (!this._isShown()) { + return; + } + const hideEvent = EventHandler.trigger(this._element, this.constructor.eventName(EVENT_HIDE$2)); + if (hideEvent.defaultPrevented) { + return; + } + const tip = this._getTipElement(); + tip.classList.remove(CLASS_NAME_SHOW$2); + + // If this is a touch-enabled device we remove the extra + // empty mouseover listeners we added for iOS support + if ('ontouchstart' in document.documentElement) { + for (const element of [].concat(...document.body.children)) { + EventHandler.off(element, 'mouseover', noop); + } + } + this._activeTrigger[TRIGGER_CLICK] = false; + this._activeTrigger[TRIGGER_FOCUS] = false; + this._activeTrigger[TRIGGER_HOVER] = false; + this._isHovered = null; // it is a trick to support manual triggering + + const complete = () => { + if (this._isWithActiveTrigger()) { + return; + } + if (!this._isHovered) { + this._disposePopper(); + } + this._element.removeAttribute('aria-describedby'); + EventHandler.trigger(this._element, this.constructor.eventName(EVENT_HIDDEN$2)); + }; + this._queueCallback(complete, this.tip, this._isAnimated()); + } + update() { + if (this._popper) { + this._popper.update(); + } + } + + // Protected + _isWithContent() { + return Boolean(this._getTitle()); + } + _getTipElement() { + if (!this.tip) { + this.tip = this._createTipElement(this._newContent || this._getContentForTemplate()); + } + return this.tip; + } + _createTipElement(content) { + const tip = this._getTemplateFactory(content).toHtml(); + + // TODO: remove this check in v6 + if (!tip) { + return null; + } + tip.classList.remove(CLASS_NAME_FADE$2, CLASS_NAME_SHOW$2); + // TODO: v6 the following can be achieved with CSS only + tip.classList.add(`bs-${this.constructor.NAME}-auto`); + const tipId = getUID(this.constructor.NAME).toString(); + tip.setAttribute('id', tipId); + if (this._isAnimated()) { + tip.classList.add(CLASS_NAME_FADE$2); + } + return tip; + } + setContent(content) { + this._newContent = content; + if (this._isShown()) { + this._disposePopper(); + this.show(); + } + } + _getTemplateFactory(content) { + if (this._templateFactory) { + this._templateFactory.changeContent(content); + } else { + this._templateFactory = new TemplateFactory({ + ...this._config, + // the `content` var has to be after `this._config` + // to override config.content in case of popover + content, + extraClass: this._resolvePossibleFunction(this._config.customClass) + }); + } + return this._templateFactory; + } + _getContentForTemplate() { + return { + [SELECTOR_TOOLTIP_INNER]: this._getTitle() + }; + } + _getTitle() { + return this._resolvePossibleFunction(this._config.title) || this._element.getAttribute('data-bs-original-title'); + } + + // Private + _initializeOnDelegatedTarget(event) { + return this.constructor.getOrCreateInstance(event.delegateTarget, this._getDelegateConfig()); + } + _isAnimated() { + return this._config.animation || this.tip && this.tip.classList.contains(CLASS_NAME_FADE$2); + } + _isShown() { + return this.tip && this.tip.classList.contains(CLASS_NAME_SHOW$2); + } + _createPopper(tip) { + const placement = execute(this._config.placement, [this, tip, this._element]); + const attachment = AttachmentMap[placement.toUpperCase()]; + return createPopper(this._element, tip, this._getPopperConfig(attachment)); + } + _getOffset() { + const { + offset + } = this._config; + if (typeof offset === 'string') { + return offset.split(',').map(value => Number.parseInt(value, 10)); + } + if (typeof offset === 'function') { + return popperData => offset(popperData, this._element); + } + return offset; + } + _resolvePossibleFunction(arg) { + return execute(arg, [this._element, this._element]); + } + _getPopperConfig(attachment) { + const defaultBsPopperConfig = { + placement: attachment, + modifiers: [{ + name: 'flip', + options: { + fallbackPlacements: this._config.fallbackPlacements + } + }, { + name: 'offset', + options: { + offset: this._getOffset() + } + }, { + name: 'preventOverflow', + options: { + boundary: this._config.boundary + } + }, { + name: 'arrow', + options: { + element: `.${this.constructor.NAME}-arrow` + } + }, { + name: 'preSetPlacement', + enabled: true, + phase: 'beforeMain', + fn: data => { + // Pre-set Popper's placement attribute in order to read the arrow sizes properly. + // Otherwise, Popper mixes up the width and height dimensions since the initial arrow style is for top placement + this._getTipElement().setAttribute('data-popper-placement', data.state.placement); + } + }] + }; + return { + ...defaultBsPopperConfig, + ...execute(this._config.popperConfig, [undefined, defaultBsPopperConfig]) + }; + } + _setListeners() { + const triggers = this._config.trigger.split(' '); + for (const trigger of triggers) { + if (trigger === 'click') { + EventHandler.on(this._element, this.constructor.eventName(EVENT_CLICK$1), this._config.selector, event => { + const context = this._initializeOnDelegatedTarget(event); + context._activeTrigger[TRIGGER_CLICK] = !(context._isShown() && context._activeTrigger[TRIGGER_CLICK]); + context.toggle(); + }); + } else if (trigger !== TRIGGER_MANUAL) { + const eventIn = trigger === TRIGGER_HOVER ? this.constructor.eventName(EVENT_MOUSEENTER) : this.constructor.eventName(EVENT_FOCUSIN$1); + const eventOut = trigger === TRIGGER_HOVER ? this.constructor.eventName(EVENT_MOUSELEAVE) : this.constructor.eventName(EVENT_FOCUSOUT$1); + EventHandler.on(this._element, eventIn, this._config.selector, event => { + const context = this._initializeOnDelegatedTarget(event); + context._activeTrigger[event.type === 'focusin' ? TRIGGER_FOCUS : TRIGGER_HOVER] = true; + context._enter(); + }); + EventHandler.on(this._element, eventOut, this._config.selector, event => { + const context = this._initializeOnDelegatedTarget(event); + context._activeTrigger[event.type === 'focusout' ? TRIGGER_FOCUS : TRIGGER_HOVER] = context._element.contains(event.relatedTarget); + context._leave(); + }); + } + } + this._hideModalHandler = () => { + if (this._element) { + this.hide(); + } + }; + EventHandler.on(this._element.closest(SELECTOR_MODAL), EVENT_MODAL_HIDE, this._hideModalHandler); + } + _fixTitle() { + const title = this._element.getAttribute('title'); + if (!title) { + return; + } + if (!this._element.getAttribute('aria-label') && !this._element.textContent.trim()) { + this._element.setAttribute('aria-label', title); + } + this._element.setAttribute('data-bs-original-title', title); // DO NOT USE IT. Is only for backwards compatibility + this._element.removeAttribute('title'); + } + _enter() { + if (this._isShown() || this._isHovered) { + this._isHovered = true; + return; + } + this._isHovered = true; + this._setTimeout(() => { + if (this._isHovered) { + this.show(); + } + }, this._config.delay.show); + } + _leave() { + if (this._isWithActiveTrigger()) { + return; + } + this._isHovered = false; + this._setTimeout(() => { + if (!this._isHovered) { + this.hide(); + } + }, this._config.delay.hide); + } + _setTimeout(handler, timeout) { + clearTimeout(this._timeout); + this._timeout = setTimeout(handler, timeout); + } + _isWithActiveTrigger() { + return Object.values(this._activeTrigger).includes(true); + } + _getConfig(config) { + const dataAttributes = Manipulator.getDataAttributes(this._element); + for (const dataAttribute of Object.keys(dataAttributes)) { + if (DISALLOWED_ATTRIBUTES.has(dataAttribute)) { + delete dataAttributes[dataAttribute]; + } + } + config = { + ...dataAttributes, + ...(typeof config === 'object' && config ? config : {}) + }; + config = this._mergeConfigObj(config); + config = this._configAfterMerge(config); + this._typeCheckConfig(config); + return config; + } + _configAfterMerge(config) { + config.container = config.container === false ? document.body : getElement(config.container); + if (typeof config.delay === 'number') { + config.delay = { + show: config.delay, + hide: config.delay + }; + } + if (typeof config.title === 'number') { + config.title = config.title.toString(); + } + if (typeof config.content === 'number') { + config.content = config.content.toString(); + } + return config; + } + _getDelegateConfig() { + const config = {}; + for (const [key, value] of Object.entries(this._config)) { + if (this.constructor.Default[key] !== value) { + config[key] = value; + } + } + config.selector = false; + config.trigger = 'manual'; + + // In the future can be replaced with: + // const keysWithDifferentValues = Object.entries(this._config).filter(entry => this.constructor.Default[entry[0]] !== this._config[entry[0]]) + // `Object.fromEntries(keysWithDifferentValues)` + return config; + } + _disposePopper() { + if (this._popper) { + this._popper.destroy(); + this._popper = null; + } + if (this.tip) { + this.tip.remove(); + this.tip = null; + } + } + + // Static + static jQueryInterface(config) { + return this.each(function () { + const data = Tooltip.getOrCreateInstance(this, config); + if (typeof config !== 'string') { + return; + } + if (typeof data[config] === 'undefined') { + throw new TypeError(`No method named "${config}"`); + } + data[config](); + }); + } +} + +/** + * jQuery + */ + +defineJQueryPlugin(Tooltip); + +/** + * -------------------------------------------------------------------------- + * Bootstrap popover.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$3 = 'popover'; +const SELECTOR_TITLE = '.popover-header'; +const SELECTOR_CONTENT = '.popover-body'; +const Default$2 = { + ...Tooltip.Default, + content: '', + offset: [0, 8], + placement: 'right', + template: '', + trigger: 'click' +}; +const DefaultType$2 = { + ...Tooltip.DefaultType, + content: '(null|string|element|function)' +}; + +/** + * Class definition + */ + +class Popover extends Tooltip { + // Getters + static get Default() { + return Default$2; + } + static get DefaultType() { + return DefaultType$2; + } + static get NAME() { + return NAME$3; + } + + // Overrides + _isWithContent() { + return this._getTitle() || this._getContent(); + } + + // Private + _getContentForTemplate() { + return { + [SELECTOR_TITLE]: this._getTitle(), + [SELECTOR_CONTENT]: this._getContent() + }; + } + _getContent() { + return this._resolvePossibleFunction(this._config.content); + } + + // Static + static jQueryInterface(config) { + return this.each(function () { + const data = Popover.getOrCreateInstance(this, config); + if (typeof config !== 'string') { + return; + } + if (typeof data[config] === 'undefined') { + throw new TypeError(`No method named "${config}"`); + } + data[config](); + }); + } +} + +/** + * jQuery + */ + +defineJQueryPlugin(Popover); + +/** + * -------------------------------------------------------------------------- + * Bootstrap scrollspy.js + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + * -------------------------------------------------------------------------- + */ + +/** + * Constants + */ + +const NAME$2 = 'scrollspy'; +const DATA_KEY$2 = 'bs.scrollspy'; +const EVENT_KEY$2 = `.${DATA_KEY$2}`; +const DATA_API_KEY = '.data-api'; +const EVENT_ACTIVATE = `activate${EVENT_KEY$2}`; +const EVENT_CLICK = `click${EVENT_KEY$2}`; +const EVENT_LOAD_DATA_API$1 = `load${EVENT_KEY$2}${DATA_API_KEY}`; +const CLASS_NAME_DROPDOWN_ITEM = 'dropdown-item'; +const CLASS_NAME_ACTIVE$1 = 'active'; +const SELECTOR_DATA_SPY = '[data-bs-spy="scroll"]'; +const SELECTOR_TARGET_LINKS = '[href]'; +const SELECTOR_NAV_LIST_GROUP = '.nav, .list-group'; +const SELECTOR_NAV_LINKS = '.nav-link'; +const SELECTOR_NAV_ITEMS = '.nav-item'; +const SELECTOR_LIST_ITEMS = '.list-group-item'; +const SELECTOR_LINK_ITEMS = `${SELECTOR_NAV_LINKS}, ${SELECTOR_NAV_ITEMS} > ${SELECTOR_NAV_LINKS}, ${SELECTOR_LIST_ITEMS}`; +const SELECTOR_DROPDOWN = '.dropdown'; +const SELECTOR_DROPDOWN_TOGGLE$1 = '.dropdown-toggle'; +const Default$1 = { + offset: null, + // TODO: v6 @deprecated, keep it for backwards compatibility reasons + rootMargin: '0px 0px -25%', + smoothScroll: false, + target: null, + threshold: [0.1, 0.5, 1] +}; +const DefaultType$1 = { + offset: '(number|null)', + // TODO v6 @deprecated, keep it for backwards compatibility reasons + rootMargin: 'string', + smoothScroll: 'boolean', + target: 'element', + threshold: 'array' +}; + +/** + * Class definition + */ + +class ScrollSpy extends BaseComponent { + constructor(element, config) { + super(element, config); + + // this._element is the observablesContainer and config.target the menu links wrapper + this._targetLinks = new Map(); + this._observableSections = new Map(); + this._rootElement = getComputedStyle(this._element).overflowY === 'visible' ? null : this._element; + this._activeTarget = null; + this._observer = null; + this._previousScrollData = { + visibleEntryTop: 0, + parentScrollTop: 0 + }; + this.refresh(); // initialize + } + + // Getters + static get Default() { + return Default$1; + } + static get DefaultType() { + return DefaultType$1; + } + static get NAME() { + return NAME$2; + } + + // Public + refresh() { + this._initializeTargetsAndObservables(); + this._maybeEnableSmoothScroll(); + if (this._observer) { + this._observer.disconnect(); + } else { + this._observer = this._getNewObserver(); + } + for (const section of this._observableSections.values()) { + this._observer.observe(section); + } + } + dispose() { + this._observer.disconnect(); + super.dispose(); + } + + // Private + _configAfterMerge(config) { + // TODO: on v6 target should be given explicitly & remove the {target: 'ss-target'} case + config.target = getElement(config.target) || document.body; + + // TODO: v6 Only for backwards compatibility reasons. Use rootMargin only + config.rootMargin = config.offset ? `${config.offset}px 0px -30%` : config.rootMargin; + if (typeof config.threshold === 'string') { + config.threshold = config.threshold.split(',').map(value => Number.parseFloat(value)); + } + return config; + } + _maybeEnableSmoothScroll() { + if (!this._config.smoothScroll) { + return; + } + + // unregister any previous listeners + EventHandler.off(this._config.target, EVENT_CLICK); + EventHandler.on(this._config.target, EVENT_CLICK, SELECTOR_TARGET_LINKS, event => { + const observableSection = this._observableSections.get(event.target.hash); + if (observableSection) { + event.preventDefault(); + const root = this._rootElement || window; + const height = observableSection.offsetTop - this._element.offsetTop; + if (root.scrollTo) { + root.scrollTo({ + top: height, + behavior: 'smooth' + }); + return; + } + + // Chrome 60 doesn't support `scrollTo` + root.scrollTop = height; + } + }); + } + _getNewObserver() { + const options = { + root: this._rootElement, + threshold: this._config.threshold, + rootMargin: this._config.rootMargin + }; + return new IntersectionObserver(entries => this._observerCallback(entries), options); + } + + // The logic of selection + _observerCallback(entries) { + const targetElement = entry => this._targetLinks.get(`#${entry.target.id}`); + const activate = entry => { + this._previousScrollData.visibleEntryTop = entry.target.offsetTop; + this._process(targetElement(entry)); + }; + const parentScrollTop = (this._rootElement || document.documentElement).scrollTop; + const userScrollsDown = parentScrollTop >= this._previousScrollData.parentScrollTop; + this._previousScrollData.parentScrollTop = parentScrollTop; + for (const entry of entries) { + if (!entry.isIntersecting) { + this._activeTarget = null; + this._clearActiveClass(targetElement(entry)); + continue; + } + const entryIsLowerThanPrevious = entry.target.offsetTop >= this._previousScrollData.visibleEntryTop; + // if we are scrolling down, pick the bigger offsetTop + if (userScrollsDown && entryIsLowerThanPrevious) { + activate(entry); + // if parent isn't scrolled, let's keep the first visible item, breaking the iteration + if (!parentScrollTop) { + return; + } + continue; + } + + // if we are scrolling up, pick the smallest offsetTop + if (!userScrollsDown && !entryIsLowerThanPrevious) { + activate(entry); + } + } + } + _initializeTargetsAndObservables() { + this._targetLinks = new Map(); + this._observableSections = new Map(); + const targetLinks = SelectorEngine.find(SELECTOR_TARGET_LINKS, this._config.target); + for (const anchor of targetLinks) { + // ensure that the anchor has an id and is not disabled + if (!anchor.hash || isDisabled(anchor)) { + continue; + } + const observableSection = SelectorEngine.findOne(decodeURI(anchor.hash), this._element); + + // ensure that the observableSection exists & is visible + if (isVisible(observableSection)) { + this._targetLinks.set(decodeURI(anchor.hash), anchor); + this._observableSections.set(anchor.hash, observableSection); + } + } + } + _process(target) { + if (this._activeTarget === target) { + return; + } + this._clearActiveClass(this._config.target); + this._activeTarget = target; + target.classList.add(CLASS_NAME_ACTIVE$1); + this._activateParents(target); + EventHandler.trigger(this._element, EVENT_ACTIVATE, { + relatedTarget: target + }); + } + _activateParents(target) { + // Activate dropdown parents + if (target.classList.contains(CLASS_NAME_DROPDOWN_ITEM)) { + SelectorEngine.findOne(SELECTOR_DROPDOWN_TOGGLE$1, target.closest(SELECTOR_DROPDOWN)).classList.add(CLASS_NAME_ACTIVE$1); + return; + } + for (const listGroup of SelectorEngine.parents(target, SELECTOR_NAV_LIST_GROUP)) { + // Set triggered links parents as active + // With both