diff --git a/Directory.Packages.props b/Directory.Packages.props
index ddadecbc2b..9b3c349f59 100644
--- a/Directory.Packages.props
+++ b/Directory.Packages.props
@@ -39,8 +39,8 @@
-
-
+
+
@@ -114,7 +114,7 @@
-
+
@@ -125,7 +125,7 @@
-
+
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/en.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/en.json
index 87635af9e2..08e1687357 100644
--- a/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/en.json
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/en.json
@@ -459,7 +459,7 @@
"FullName": "Full name",
"CompanySize": "Company size",
"TestimonialTitle": "Let's hear your testimonial",
- "TestimonialInfo": "What our customers say matters! Tell us about your experience with our product and service. It is recommended to write the testimonial in English to reach a wider audience.",
+ "TestimonialInfo": "What you say matters! Tell us about your experience with ABP in a few sentences. Please write it in English to reach a wider audience.",
"Country": "Country",
"TestimonialTextPlaceholder": "Write a brief story about how ABP helped you build and deliver your project.",
"PositionPlaceholder": "Your position at your company",
diff --git a/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/community-talks.png b/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/community-talks.png
new file mode 100644
index 0000000000..7263b68e10
Binary files /dev/null and b/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/community-talks.png differ
diff --git a/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/cover-image.png b/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/cover-image.png
new file mode 100644
index 0000000000..f272a8d463
Binary files /dev/null and b/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/cover-image.png differ
diff --git a/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/post.md b/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/post.md
new file mode 100644
index 0000000000..26850bc57f
--- /dev/null
+++ b/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/post.md
@@ -0,0 +1,93 @@
+# ABP.IO Platform 9.0 Has Been Released Based on .NET 9.0
+
+
+
+Today, [ABP](https://abp.io/) 9.0 stable version has been released based on [.NET 9.0](https://dotnet.microsoft.com/en-us/download/dotnet/9.0). You can create solutions with ABP 9.0 starting from ABP Studio v0.9.11 or by using the ABP CLI as explained in the following sections.
+
+## What's New With Version 9.0?
+
+All the new features were explained in detail in the [9.0 RC Announcement Post](https://abp.io/blog/announcing-abp-9-0-release-candidate), so there is no need to review them again. You can check it out for more details.
+
+## Getting Started with 9.0
+
+### Creating New Solutions
+
+You can check the [Get Started page](https://abp.io/get-started) to see how to get started with ABP. You can either download [ABP Studio](https://abp.io/get-started#abp-studio-tab) (**recommended**, if you prefer a user-friendly GUI application - desktop application) or use the [ABP CLI](https://abp.io/docs/latest/cli) to create new solutions.
+
+By default, ABP Studio uses stable versions to create solutions. Therefore, it will be creating the solution with the latest stable version, which is v9.0 for now, so you don't need to specify the version. **You can create solutions with ABP 9.0 starting from v0.9.11.**
+
+### How to Upgrade an Existing Solution
+
+You can upgrade your existing solutions with either ABP Studio or ABP CLI. In the following sections, both approaches are explained:
+
+### Upgrading via ABP Studio
+
+If you are already using the ABP Studio, you can upgrade it to the latest version to align it with ABP v9.0. ABP Studio periodically checks for updates in the background, and when a new version of ABP Studio is available, you will be notified through a modal. Then, you can update it by confirming the opened modal. See [the documentation](https://abp.io/docs/latest/studio/installation#upgrading) for more info.
+
+After upgrading the ABP Studio, then you can open your solution in the application, and simply click the **Switch to stable** action button to instantly upgrade your solution:
+
+
+
+> Please note that ABP CLI & ABP Studio only upgrade the related ABP packages, so you need to upgrade the other packages for .NET 9.0 manually.
+
+### Upgrading via ABP CLI
+
+Alternatively, you can upgrade your existing solution via ABP CLI. First, you need to install the ABP CLI or upgrade it to the latest version.
+
+If you haven't installed it yet, you can run the following command:
+
+```bash
+dotnet tool install -g Volo.Abp.Studio.Cli
+```
+
+Or to update the existing CLI, you can run the following command:
+
+```bash
+dotnet tool update -g Volo.Abp.Studio.Cli
+```
+
+After installing/updating the ABP CLI, you can use the [`update` command](https://abp.io/docs/latest/CLI#update) to update all the ABP related NuGet and NPM packages in your solution as follows:
+
+```bash
+abp update
+```
+
+You can run this command in the root folder of your solution to update all ABP related packages.
+
+> Please note that ABP CLI & ABP Studio only upgrade the related ABP packages, so you need to upgrade the other packages for .NET 9.0 manually.
+
+## Migration Guides
+
+There are a few breaking changes in this version that may affect your application. Please read the migration guide carefully, if you are upgrading from v8.x: [ABP Version 9.0 Migration Guide](https://abp.io/docs/9.0/release-info/migration-guides/abp-9-0)
+
+## Community News
+
+### Highlights from .NET 9.0
+
+Our team has closely followed the ASP.NET Core and Entity Framework Core 9.0 releases, read Microsoft's guides and documentation, and adapted the changes to our ABP.IO Platform. We are proud to say that we've shipped the ABP 9.0 based on .NET 9.0 just after Microsoft's .NET 9.0 release.
+
+In addition to the ABP's .NET 9.0 upgrade, our team has created many great articles to highlight the important features coming with ASP.NET Core 9.0 and Entity Framework Core 9.0.
+
+> You can read [this post](https://volosoft.com/blog/Highlights-for-ASP-NET-Entity-Framework-Core-NET-9-0) to see the list of all articles.
+
+### New ABP Community Articles
+
+In addition to [the articles to highlight .NET 9.0 features written by our team](https://volosoft.com/blog/Highlights-for-ASP-NET-Entity-Framework-Core-NET-9-0), here are some of the recent posts added to the [ABP Community](https://abp.io/community):
+
+* [Video: Building Modular Monolith Applications with ASP.NET Core & ABP Studio](https://abp.io/community/videos/building-modular-monolith-applications-with-asp.net-core-abp-studio-66znukvf) by [Halil İbrahim Kalkan](https://x.com/hibrahimkalkan)
+* [How to create your Own AI Bot on WhatsApp Using an ABP.io Template](https://abp.io/community/articles/how-to-create-your-own-ai-bot-on-whatsapp-using-the-abp-framework-c6jgvt9c) by [Michael Kokula](https://abp.io/community/members/Michal_Kokula)
+* [ABP Now Supports .NET 9](https://abp.io/community/articles/abp-now-supports-.net-9-zpkznc4f) by [Alper Ebiçoğlu](https://x.com/alperebicoglu)
+
+Thanks to the ABP Community for all the content they have published. You can also [post your ABP related (text or video) content](https://abp.io/community/posts/submit) to the ABP Community.
+
+### ABP Community Talks 2024.7: What’s New with .NET 9 & ABP 9?
+
+
+
+In this episode of ABP Community Talks, 2024.7; we will dive into the features that came with .NET 9.0 with [Alper Ebicoglu](https://github.com/ebicoglu), [Engincan Veske](https://github.com/EngincanV), [Berkan Sasmaz](https://github.com/berkansasmaz) and [Ahmet Faruk Ulu](https://github.com/ahmetfarukulu).
+
+## Conclusion
+
+This version comes with some new features and a lot of enhancements to the existing features. You can see the [Road Map](https://docs.abp.io/en/abp/9.0/Road-Map) documentation to learn about the release schedule and planned features for the next releases. Please try ABP v9.0 and provide feedback to help us release more stable versions.
+
+Thanks for being a part of this community!
diff --git a/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/switch-to-stable.png b/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/switch-to-stable.png
new file mode 100644
index 0000000000..30883ebf92
Binary files /dev/null and b/docs/en/Blog-Posts/2024-11-19 v9_0_Release_Stable/switch-to-stable.png differ
diff --git a/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/POST.md b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/POST.md
new file mode 100644
index 0000000000..c07509d2e3
--- /dev/null
+++ b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/POST.md
@@ -0,0 +1,385 @@
+# How to Use OpenAI API with ABP Framework
+
+In this article, I will show you how to integrate and use the [OpenAI API](https://github.com/openai/openai-dotnet?tab=readme-ov-file#getting-started) with the [ABP Framework](https://abp.io/). We will explore step-by-step how these technologies can work together to enhance your application with powerful AI capabilities, such as natural language processing, image generation, and more.
+
+
+
+## Creating an ABP Project
+
+To begin integrating OpenAI API with ABP Framework, you first need to create an ABP project. Follow these steps to create and set up your ABP project:
+### Step 1: Install ABP CLI
+
+The ABP CLI is a command-line interface tool that helps you create and manage ABP projects easily. To install the ABP CLI, run the following command in your terminal:
+
+```bash
+dotnet tool install -g Volo.Abp.Studio.Cli
+```
+
+### Step 2: Create a New ABP Project
+
+Once you have installed the ABP CLI, you can create a new ABP project using the following command:
+
+```bash
+abp new Acme.OpenAIIntegration -t app --ui-framework mvc --database-provider ef -dbms PostgreSQL --csf
+```
+
+> This command will generate a complete ABP project with an [MVC UI](https://abp.io/docs/latest/framework/ui/mvc-razor-pages/overall). The examples provided in this article make use of UI controllers for demonstration purposes. However, the same approach can easily be applied to other UI types supported by ABP, such as Blazor or Angular. You can find other options [here](https://abp.io/docs/latest/cli).
+## OpenAI Integration Setup
+
+To begin integrating OpenAI API with ABP Framework, follow these steps:
+
+### Step 1: Create an API Key
+
+To use the OpenAI services, you first need an API key. To obtain one, first [create a new OpenAI account](https://platform.openai.com/signup) or [log in](https://platform.openai.com/login). Next, navigate to the [API key page](https://platform.openai.com/account/api-keys) and select "Create new secret key", optionally naming the key. Make sure to save your API key somewhere safe and do not share it with anyone.
+
+This key will be used to authenticate your application when making requests to the OpenAI endpoints.
+
+### Step 2: Adding *Microsoft.Extensions.AI* Package
+
+To integrate OpenAI API with ABP, we use [Microsoft.Extensions.AI](https://www.nuget.org/packages/Microsoft.Extensions.AI.OpenAI/). This package offers a unified API for integrating AI services, making it easy for developers to work with different AI providers. You can find more details in [this blog post](https://devblogs.microsoft.com/dotnet/introducing-microsoft-extensions-ai-preview/).
+
+To begin integrating OpenAI API with ABP Framework, follow these steps:
+
+1. Add the **Microsoft.Extensions.AI** and **Microsoft.Extensions.AI.OpenAI** (used to interact specifically with OpenAI services. Additionally, this package has alternatives like [Azure OpenAI](https://www.nuget.org/packages/Microsoft.Extensions.AI.OpenAI/), [Azure AI Inference](https://www.nuget.org/packages/Microsoft.Extensions.AI.AzureAIInference/), and [Ollama](https://www.nuget.org/packages/Microsoft.Extensions.AI.Ollama/), offering flexibility for developers to choose the AI provider that best fits their needs) packages:
+
+```bash
+dotnet add package Microsoft.Extensions.AI --prerelease
+dotnet add package Microsoft.Extensions.AI.OpenAI --prerelease
+```
+
+2. Add the required configuration to the `appsettings.json` file located inside the `Acme.OpenAIIntegration.Web` project and dependencies to your `ConfigureServices` method:
+
+```json
+"AI": {
+ "OpenAI": {
+ "Key": "YOUR-API-KEY",
+ "Chat": {
+ "ModelId": "gpt-4o-mini"
+ }
+ }
+}
+```
+
+> Replace the value of the `Key` with your OpenAI API key.
+
+Next, add the following code to the `ConfigureServices` method in `OpenAIIntegrationBlazorModule`:
+
+```csharp
+context.Services.AddSingleton(new OpenAIClient(configuration["AI:OpenAI:Key"]));
+
+context.Services.AddChatClient(services =>
+ services.GetRequiredService().AsChatClient(configuration["AI:OpenAI:Chat:ModelId"] ?? "gpt-4o-mini"));
+```
+
+## Creating a Sample Page
+
+To demonstrate the use of OpenAI API, let's create a page named `Sample` in the `Acme.OpenAIIntegration.Web` project:
+
+Create a `Sample` folder under the `Pages` folder of the `Acme.OpenAIIntegration.Web` project. Add a new Razor Page by right-clicking the `Sample` folder then selecting `Add > Razor Page`. Name it `Index`.
+
+Open the `Index.cshtml` and change the whole content as shown below:
+
+> Note: This example demonstrates a simple implementation of a sample page that interacts with the OpenAI API, covering chat, [retrieval-augmented generation (RAG)](https://github.com/openai/openai-dotnet?tab=readme-ov-file#how-to-use-assistants-with-retrieval-augmented-generation-rag), and image generation features. Each example is explained in detail in the next section, so feel free to continue for a better understanding of the steps and logic involved.
+
+```html
+@page
+@model Acme.OpenAIIntegration.Web.Pages.Sample
+@{
+ ViewData["Title"] = "OpenAI API Demonstration";
+}
+
+
+```
+
+`Index.cshtml.cs` content should be like that:
+
+```csharp
+using System;
+using System.ClientModel;
+using System.IO;
+using System.Text;
+using System.Threading;
+using System.Threading.Tasks;
+using Microsoft.AspNetCore.Mvc;
+using Microsoft.AspNetCore.Mvc.RazorPages;
+using Microsoft.Extensions.AI;
+using OpenAI;
+using OpenAI.Assistants;
+using OpenAI.Files;
+using OpenAI.Images;
+
+namespace Acme.OpenAIIntegration.Web.Pages;
+
+public class Sample : PageModel
+{
+ [BindProperty]
+ public string ChatInput { get; set; }
+ public string ChatResponse { get; set; }
+
+ [BindProperty]
+ public string RAGQuery { get; set; }
+ public string RAGResponse { get; set; }
+
+ [BindProperty]
+ public string ImagePrompt { get; set; }
+ public byte[] GeneratedImageBytes { get; set; }
+
+ private readonly IChatClient _chatClient;
+ private readonly OpenAIClient _openAiClient;
+
+ public Sample(
+ IChatClient chatClient,
+ OpenAIClient openAiClient)
+ {
+ _chatClient = chatClient;
+ _openAiClient = openAiClient;
+ }
+
+ public async Task OnPostChatAsync()
+ {
+ ChatResponse = $"Chat response: {(await _chatClient.CompleteAsync(ChatInput)).Message}";
+ return Page();
+ }
+
+ public async Task OnPostRAGAsync()
+ {
+#pragma warning disable OPENAI001
+ var fileClient = _openAiClient.GetOpenAIFileClient();
+ var assistantClient = _openAiClient.GetAssistantClient();
+
+ using var document = BinaryData.FromBytes(GetExceptionHandlingDocumentContent().ToArray()).ToStream();
+ var exceptionHandlingDoc = await fileClient.UploadFileAsync(
+ document,
+ "ExceptionHandling.md",
+ FileUploadPurpose.Assistants);
+
+ AssistantCreationOptions assistantOptions = new()
+ {
+ Name = "Exception Handling Assistant",
+ Instructions =
+ """
+ This assistant helps you with exception handling in ABP Framework. You can ask questions about exception handling and get answers.
+
+ - Do not make any assumptions when asked for information that is not in the document
+ - Give the most accurate information possible
+ - Give short(max 1-2 sentence) and concise answers
+ - Do not provide file citations
+ """,
+
+ Tools =
+ {
+ new FileSearchToolDefinition(),
+ },
+ ToolResources = new()
+ {
+ FileSearch = new()
+ {
+ NewVectorStores =
+ {
+ new VectorStoreCreationHelper([exceptionHandlingDoc.Value.Id]),
+ }
+ }
+ },
+ };
+
+ var assistant = await assistantClient.CreateAssistantAsync("gpt-4o", assistantOptions);
+
+ ThreadCreationOptions threadOptions = new()
+ {
+ InitialMessages = { RAGQuery }
+ };
+
+ ThreadRun threadRun = assistantClient.CreateThreadAndRun(assistant.Value.Id, threadOptions);
+
+ do
+ {
+ Thread.Sleep(TimeSpan.FromSeconds(1));
+ threadRun = assistantClient.GetRun(threadRun.ThreadId, threadRun.Id);
+ } while (!threadRun.Status.IsTerminal);
+
+ CollectionResult messages
+ = assistantClient.GetMessages(threadRun.ThreadId,
+ new MessageCollectionOptions() { Order = MessageCollectionOrder.Ascending });
+
+ var response = new StringBuilder();
+
+ foreach (var message in messages)
+ {
+ response.AppendLine($"[{message.Role.ToString().ToUpper()}]: ");
+ foreach (var contentItem in message.Content)
+ {
+ if (!string.IsNullOrEmpty(contentItem.Text))
+ {
+ response.AppendLine(contentItem.Text);
+
+ if (contentItem.TextAnnotations.Count > 0)
+ {
+ response.AppendLine("");
+ }
+ }
+ }
+
+ response.AppendLine("");
+#pragma warning restore OPENAI001
+ }
+
+ RAGResponse = response.ToString();
+
+ return Page();
+ }
+
+ public async Task OnPostImageGenerationAsync()
+ {
+ var client = _openAiClient.GetImageClient("dall-e-3");
+
+ var image = await client.GenerateImageAsync(ImagePrompt, new ImageGenerationOptions
+ {
+ ResponseFormat = GeneratedImageFormat.Bytes
+ });
+
+ var imageBytes = image.Value.ImageBytes;
+
+ using var memoryStream = new MemoryStream();
+ await imageBytes.ToStream().CopyToAsync(memoryStream);
+ GeneratedImageBytes = memoryStream.ToArray();
+
+ return Page();
+ }
+
+ public ReadOnlySpan GetExceptionHandlingDocumentContent()
+ {
+ return """
+ # Exception Handling
+
+ ABP provides a built-in infrastructure and offers a standard model for handling exceptions.
+
+ * Automatically **handles all exceptions** and sends a standard **formatted error message** to the client for an API/AJAX request.
+ * Automatically hides **internal infrastructure errors** and returns a standard error message.
+ * Provides an easy and configurable way to **localize** exception messages.
+ * Automatically maps standard exceptions to **HTTP status codes** and provides a configurable option to map custom exceptions.
+
+ ## Automatic Exception Handling
+
+ `AbpExceptionFilter` handles an exception if **any of the following conditions** are met:
+
+ * Exception is thrown by a **controller action** which returns an **object result** (not a view result).
+ * The request is an AJAX request (`X-Requested-With` HTTP header value is `XMLHttpRequest`).
+ * Client explicitly accepts the `application/json` content type (via `accept` HTTP header).
+
+ If the exception is handled it's automatically **logged** and a formatted **JSON message** is returned to the client.
+
+ ## Business Exceptions
+
+ Most of your own exceptions will be business exceptions. The `IBusinessException` interface is used to mark an exception as a business exception.
+
+ `BusinessException` implements the `IBusinessException` interface in addition to the `IHasErrorCode`, `IHasErrorDetails` and `IHasLogLevel` interfaces. The default log level is `Warning`.
+
+ Usually you have an error code related to a particular business exception. For example:
+
+ ````C#
+ throw new BusinessException(QaErrorCodes.CanNotVoteYourOwnAnswer);
+ ````
+
+ ### User Friendly Exception
+
+ If an exception implements the `IUserFriendlyException` interface, then ABP does not change it's `Message` and `Details` properties and directly send it to the client.
+
+ `UserFriendlyException` class is the built-in implementation of the `IUserFriendlyException` interface. Example usage:
+
+ ````C#
+ throw new UserFriendlyException(
+ "Username should be unique!"
+ );
+ ````
+ * The `IUserFriendlyException` interface is derived from the `IBusinessException` and the `UserFriendlyException` class is derived from the `BusinessException` class.
+
+ """u8;
+ }
+}
+```
+
+## Running the Application
+
+After completing the setup, you can run the application using the following command:
+
+```bash
+ dotnet run --project ./src/Acme.OpenAIIntegration.Web
+```
+
+Once the application is running, open your browser and navigate to `/Sample`. You should see the `Sample` page we created, which contains sections for Chat, RAG (Retrieval-Augmented Generation), and Image Generation. You can find the screenshot of the page below:
+
+
+
+## Examples Overview
+
+To showcase the integration of the OpenAI API with the ABP Framework, we implemented three different examples:
+
+1. **Chat Example**: This example demonstrates how to use OpenAI's chat capabilities by allowing users to enter a message and receive an AI-generated response. The implementation involves setting up a simple form on the `Sample` page where users can input their message. The form submission triggers the `OnPostChatAsync` method, which uses the `IChatClient` to generate a response.
+
+
+
+2. **Retrieval-Augmented Generation (RAG) Example**: In this example, we use OpenAI to answer user queries by referencing custom documents uploaded to the OpenAI API. The implementation involves uploading a document using the `OpenAIFileClient` and creating an assistant with specific instructions to handle the uploaded content. In this case, the document is a section from ABP's Exception Handling documentation, which includes examples on how ABP handles exceptions, user-friendly error messages, and business exceptions. Users can input their query on the `Sample` page, and the `OnPostRAGAsync` method processes the query to generate precise answers based on the document content. If users ask questions that are not covered in the document, the assistant clearly indicates that the information is not available, as per the instructions provided. For example, when asked about `Object Extensions`, the response begins with: "The uploaded document does not contain information about `Object Extensions`...". This demonstrates how the assistant adheres to the provided instructions. You can also find this example illustrated in the GIF below.
+
+
+
+
+
+3. **Image Generation Example**: This example leverages the [DALL-E](https://openai.com/index/dall-e-3/) model to generate images based on user-provided prompts. On the `Sample` page, users can provide a description of the image they want to generate, and the `OnPostImageGenerationAsync` method uses the `OpenAIClient` to generate the image.
+
+
+
+## Conclusion
+
+In this article, we covered how to integrate the OpenAI API with the ABP Framework by creating a sample project, setting up the OpenAI services, and implementing examples for conversational AI, knowledge-based assistance, and image generation. By following these steps, you can add powerful AI-driven capabilities to your application, making it more interactive, intelligent, and capable of meeting user needs effectively.
\ No newline at end of file
diff --git a/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/chat-example.gif b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/chat-example.gif
new file mode 100644
index 0000000000..f67d54cecc
Binary files /dev/null and b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/chat-example.gif differ
diff --git a/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/cover-image.png b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/cover-image.png
new file mode 100644
index 0000000000..b1cc16f0b5
Binary files /dev/null and b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/cover-image.png differ
diff --git a/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/image-generation-example.gif b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/image-generation-example.gif
new file mode 100644
index 0000000000..c415f95119
Binary files /dev/null and b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/image-generation-example.gif differ
diff --git a/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/rag-example-1.gif b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/rag-example-1.gif
new file mode 100644
index 0000000000..ca42bf3f61
Binary files /dev/null and b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/rag-example-1.gif differ
diff --git a/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/rag-example-2.gif b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/rag-example-2.gif
new file mode 100644
index 0000000000..5f03e453c6
Binary files /dev/null and b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/rag-example-2.gif differ
diff --git a/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/sample-page.png b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/sample-page.png
new file mode 100644
index 0000000000..211f63215d
Binary files /dev/null and b/docs/en/Community-Articles/2024-12-01-OpenAI-Integration/sample-page.png differ
diff --git a/docs/en/Community-Articles/2024-12-09-Unit-Test/POST.md b/docs/en/Community-Articles/2024-12-09-Unit-Test/POST.md
new file mode 100644
index 0000000000..7c1b74f3ea
--- /dev/null
+++ b/docs/en/Community-Articles/2024-12-09-Unit-Test/POST.md
@@ -0,0 +1,234 @@
+# The new Unit Test structure in ABP application
+
+A typical ABP modular project usually consists of three main projects: `Application`, `Domain`, and `EntityFrameworkCore/MongoDB`. In these projects, we may provide many services that require unit testing.
+
+Using abstract unit test classes involves first writing tests in the `Application` and `Domain` layers that are independent of the storage technology, ensuring the correctness of core business logic. These abstract tests are then implemented in `EntityFrameworkCore` or `MongoDB`. The benefits of this approach include:
+
+1. **Reduced Coupling**: Core logic tests do not depend on specific storage technologies, so switching databases does not require rewriting test code.
+2. **Better Isolation**: Focuses on verifying business logic correctness, avoiding interference from database operations.
+3. **Increased Reusability**: The same abstract tests can be reused with different storage implementations.
+4. **Easier Maintenance and Extensibility**: Different storage implementations can be extended independently without breaking existing tests.
+5. **Faster and More Reliable Tests**: Reduces dependency on databases, making tests faster and more stable.
+
+## How to migrate old unit tests to the new unit test structure
+
+Assume our project name is `MyCompanyName.MyProjectName`.
+
+### Changes to the `MyCompanyName.MyProjectName.Application.Tests` project:
+
+1. Remove the `MyCompanyName.MyProjectName.Application.Tests` project's `MyProjectNameApplicationCollection` class.
+2. Modify the `MyCompanyName.MyProjectName.Application.Tests` project's `MyProjectNameApplicationTestBase` class.
+
+```csharp
+public abstract class MyProjectNameApplicationTestBase : MyProjectNameTestBase
+ where TStartupModule : IAbpModule
+{
+ //...
+}
+```
+
+3. Modify the `MyCompanyName.MyProjectName.Application.Tests` project's unit test classes to become abstract unit test classes, such as: `SampleAppServiceTests`.
+
+```csharp
+public abstract class SampleAppServiceTests : MyProjectNameApplicationTestBase
+ where TStartupModule : IAbpModule
+{
+ [Fact]
+ public async Task Initial_Data_Should_Contain_Admin_User()
+ {
+ //...
+ }
+}
+```
+
+### Changes to the `MyCompanyName.MyProjectName.Domain.Tests` project:
+
+1. Remove the `MyCompanyName.MyProjectName.Domain.Tests` project's `MyProjectNameDomainCollection` class.
+2. Modify the `MyCompanyName.MyProjectName.Domain.Tests` project's `MyProjectNameDomainTestBase` class.
+
+```csharp
+public abstract class MyProjectNameDomainTestBase : MyProjectNameTestBase
+ where TStartupModule : IAbpModule
+{
+ //...
+}
+```
+
+3. Modify the `MyCompanyName.MyProjectName.Domain.Tests` project's unit test classes to become abstract unit test classes, such as: `SampleDomainTests`.
+
+```csharp
+public abstract class SampleDomainTests : MyProjectNameDomainTestBase
+ where TStartupModule : IAbpModule
+{
+ [Fact]
+ public async Task Should_Set_Email_Of_A_User()
+ {
+ //...
+ }
+}
+```
+
+4. Modify the `MyCompanyName.MyProjectName.Domain.Tests` project's `csproj` and module class. Remove references to `EntityFrameworkCore/MongoDB`.
+
+`MyCompanyName.MyProjectName.Domain.Tests.csproj`:
+```xml
+
+
+ //...
+
+
+
+
+
+
+
+```
+
+`MyProjectNameDomainTestModule.cs`:
+
+```csharp
+[DependsOn(
+ typeof(MyProjectNameDomainModule),
+ typeof(MyProjectNameTestBaseModule)
+)]
+public class MyProjectNameDomainTestModule : AbpModule
+{
+ //...
+}
+```
+
+### Changes to the `MyCompanyName.MyProjectName.EntityFrameworkCore.Tests` project:
+
+Here, we need to create implementation classes for all abstract unit tests.
+
+```csharp
+[Collection(MyProjectNameTestConsts.CollectionDefinitionName)]
+public class EfCoreSampleAppServiceTests : SampleAppServiceTests
+{
+ //...
+}
+```
+
+```csharp
+[Collection(MyProjectNameTestConsts.CollectionDefinitionName)]
+public class EfCoreSampleDomainTests : SampleDomainTests
+{
+ //...
+}
+```
+
+We also need to modify the project's dependencies and module class, which should directly or indirectly reference the `Application` and `Domain` test projects.
+
+`MyCompanyName.MyProjectName.EntityFrameworkCore.Tests.csproj`:
+
+```xml
+
+
+ //...
+
+
+
+
+
+
+
+```
+
+`MyProjectNameEntityFrameworkCoreTestModule.cs`:
+
+```csharp
+[DependsOn(
+ typeof(MyProjectNameApplicationTestModule),
+ typeof(MyProjectNameEntityFrameworkCoreModule),
+ typeof(AbpEntityFrameworkCoreSqliteModule)
+ )]
+public class MyProjectNameEntityFrameworkCoreTestModule : AbpModule
+{
+ //...
+}
+```
+
+### Changes to the `MyCompanyName.MyProjectName.MongoDB.Tests` project (skip this step if not using MongoDB):
+
+Like the `EntityFrameworkCore` project, we need to create implementation classes for all abstract unit tests and modify the project's dependencies and module class.
+
+```csharp
+[Collection(MyProjectNameTestConsts.CollectionDefinitionName)]
+public class MongoDBSampleAppServiceTests : SampleAppServiceTests
+{
+ //...
+}
+```
+
+```csharp
+[Collection(MyProjectNameTestConsts.CollectionDefinitionName)]
+public class MongoDBSampleDomainTests : SampleDomainTests
+{
+ //...
+}
+```
+
+```xml
+
+
+ //...
+
+
+
+
+
+
+```
+
+```csharp
+[DependsOn(
+ typeof(MyProjectNameApplicationTestModule),
+ typeof(MyProjectNameMongoDbModule)
+)]
+public class MyProjectNameMongoDbTestModule : AbpModule
+{
+ //...
+}
+```
+
+### Changes to the `MyCompanyName.MyProjectName.Web.Tests` project:
+
+We need to reference the `EntityFrameworkCore/MongoDB` test projects in this test project.
+
+```xml
+
+
+ //...
+
+
+
+
+
+
+
+
+```
+
+```csharp
+[DependsOn(
+ typeof(AbpAspNetCoreTestBaseModule),
+ typeof(MyProjectNameWebModule),
+ typeof(MyProjectNameApplicationTestModule),
+ typeof(MyProjectNameEntityFrameworkCoreTestModule)
+)]
+public class MyProjectNameWebTestModule : AbpModule
+{
+ //...
+}
+```
+
+We no longer need the `MyProjectNameWebCollection` class in this project. Please delete it and use `[Collection(MyProjectNameTestConsts.CollectionDefinitionName)]` instead.
+
+## Conclusion
+
+This is our new unit test structure. Decoupling unit tests from storage technologies ensures the independence of business logic and allows easy switching between storage implementations. Abstract unit test classes improve test reusability, maintainability, and efficiency, reducing refactoring costs and providing flexibility for future tech updates.
+
+## References
+
+- [Unit Test](https://abp.io/docs/latest/testing/unit-tests)
+- [Abstract all db-related unit tests](https://github.com/abpframework/abp/pull/17880)
diff --git a/docs/en/cli/index.md b/docs/en/cli/index.md
index 9297965bb4..61d5986588 100644
--- a/docs/en/cli/index.md
+++ b/docs/en/cli/index.md
@@ -211,6 +211,7 @@ For more samples, go to [ABP CLI Create Solution Samples](new-command-samples.md
* `leptonx`: LeptonX Theme.
* `basic`: Basic Theme.
* `--public-website`: Public Website is a front-facing website for describing your project, listing your products and doing SEO for marketing purposes. Users can login and register on your website with this website. This option is only included in PRO templates.
+ * `--no-grafana-dashboard` or `-ngd`: Does not add example Grafana Dashboard to the solution.
* `--output-folder` or `-o`: Specifies the output folder. Default value is the current directory.
* `--local-framework-ref` or `-lfr`: Uses local projects references to the ABP framework instead of using the NuGet packages. It tries to find the paths from `ide-state.json`. The file is located at `%UserProfile%\.abp\studio\ui\ide-state.json` (for Windows) and `~/.abp/studio/ui/ide-state.json` (for MAC).
* `--create-solution-folder` or `-csf`: Specifies if the project will be in a new folder in the output folder or directly the output folder.
@@ -225,15 +226,16 @@ For more samples, go to [ABP CLI Create Solution Samples](new-command-samples.md
* `--dont-run-bundling`: Skip bundling for Blazor packages.
* `--no-kubernetes-configuration` or `-nkc`: Skips the Kubernetes configuration files.
* `--no-social-logins` or `-nsl`: Skipts the social login configuration.
-* *Module Options*: You can skip some modules if you don't want to add them to your solution (*Available for* ***Team*** *or higher licenses*). Available commands:
+* `--no-tests` or `-ntp`: Does not add test projects.
+* *Module Options*: You can skip some modules if you don't want to add them to your solution, or include if you want them (*Available for* ***Team*** *or higher licenses*). Available commands:
* `-no-saas`: Skips the Saas module.
* `-no-gdpr`: Skips the GDPR module.
* `-no-openiddict-admin-ui`: Skips the OpenIddict Admin UI module.
* `-no-audit-logging`: Skips the Audit Logging module.
- * `-no-file-management`: Skips the File Management module.
* `-no-language-management`: Skips the Language Management module.
* `-no-text-template-management`: Skips the Text Template Management module.
- * `-no-chat`: Skips the Chat module.
+ * `-file-management`: Includes the File Management module.
+ * `-chat`: Includes the Chat module.
* `--legacy`: Generates a legacy solution.
* `trust-version`: Trusts the user's version and does not check if the version exists or not. If the template with the given version is found in the cache, it will be used, otherwise throws an exception.
diff --git a/docs/en/contribution/index.md b/docs/en/contribution/index.md
index 081f429b31..504078f09f 100644
--- a/docs/en/contribution/index.md
+++ b/docs/en/contribution/index.md
@@ -50,6 +50,12 @@ This is the recommended approach, since it automatically finds all missing texts
If you want to make a change on a specific resource file, you can find the file yourself, make the necessary change (or create a new file for your language) and send a pull request on GitHub.
+### Commercial Modules
+
+The commercial modules are not open source, and their localization files are not available in the public repository. The open-source module, `Account`, and the commercial module, `Account.Pro`, may have different translations.
+
+If you would like to translate a commercial module, please [create an issue](https://github.com/abpframework/abp/issues/new) on Github, and we will provide the necessary files (`abp-translation.json` for one or all modules).
+
## Bug Report
If you find any bug, please [create an issue on the Github repository](https://github.com/abpframework/abp/issues/new).
diff --git a/docs/en/deployment/configuring-openIddict.md b/docs/en/deployment/configuring-openIddict.md
index 30b8078098..08fcf7d143 100644
--- a/docs/en/deployment/configuring-openIddict.md
+++ b/docs/en/deployment/configuring-openIddict.md
@@ -4,6 +4,8 @@ This document introduces how to configure `OpenIddict` in the `AuthServer` proje
There are different configurations in the `AuthServer` project for the `Development` and `Production` environments.
+> If your solution does not include a project named `.AuthServer`, it means that you might have another project that depends on `AbpAccountPublicWebOpenIddictModule`. The project name can be `MyProject`, `MyProject.Web`, or `MyProject.HttpApi.Host`. They are both `Authentication Server` projects in that context.
+
````csharp
public override void PreConfigureServices(ServiceConfigurationContext context)
{
@@ -38,6 +40,8 @@ To avoid that, consider creating self-signed certificates and storing them in th
You can use the `dotnet dev-certs https -v -ep openiddict.pfx -p 00000000-0000-0000-0000-000000000000` command to generate the `openiddict.pfx` certificate.
+> `openiddict.pfx` is just an example of a filename. You can use any filename for the pfx file.
+
> `00000000-0000-0000-0000-000000000000` is the password of the certificate, you can change it to any password you want.
> Also, please remember to copy `openiddict.pfx` to the [Content Root Folder](https://learn.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.hosting.ihostingenvironment.contentrootpath?view=aspnetcore-7.0) of the `AuthServer` website.
diff --git a/docs/en/docs-nav.json b/docs/en/docs-nav.json
index d30b25dfb6..b1a335746b 100644
--- a/docs/en/docs-nav.json
+++ b/docs/en/docs-nav.json
@@ -115,6 +115,35 @@
}
]
},
+ {
+ "text": "Book Store Application (with ABP Suite)",
+ "items": [
+ {
+ "text": "Overview",
+ "path": "tutorials/book-store-with-abp-suite"
+ },
+ {
+ "text": "1: Creating the Solution",
+ "path": "tutorials/book-store-with-abp-suite/part-01.md"
+ },
+ {
+ "text": "2: Creating the Books",
+ "path": "tutorials/book-store-with-abp-suite/part-02.md"
+ },
+ {
+ "text": "3: Creating the Authors",
+ "path": "tutorials/book-store-with-abp-suite/part-03.md"
+ },
+ {
+ "text": "4: Book to Author Relation",
+ "path": "tutorials/book-store-with-abp-suite/part-04.md"
+ },
+ {
+ "text": "5: Customizing the Generated Code",
+ "path": "tutorials/book-store-with-abp-suite/part-05.md"
+ }
+ ]
+ },
{
"text": "Modular Monolith Application",
"items": [
@@ -156,6 +185,43 @@
}
]
},
+ {
+ "text": "Microservice Solution",
+ "items": [
+ {
+ "text": "Overview",
+ "path": "tutorials/microservice/index.md"
+ },
+ {
+ "text": "1: Creating the initial solution",
+ "path": "tutorials/microservice/part-01.md"
+ },
+ {
+ "text": "2: Creating the initial Catalog service",
+ "path": "tutorials/microservice/part-02.md"
+ },
+ {
+ "text": "3: Building the Catalog service",
+ "path": "tutorials/microservice/part-03.md"
+ },
+ {
+ "text": "4: Creating the initial Ordering service",
+ "path": "tutorials/microservice/part-04.md"
+ },
+ {
+ "text": "5: Building the Ordering service",
+ "path": "tutorials/microservice/part-05.md"
+ },
+ {
+ "text": "6: Integrating the services: HTTP API Calls",
+ "path": "tutorials/microservice/part-06.md"
+ },
+ {
+ "text": "7: Integrating the services: Using Distributed Events",
+ "path": "tutorials/microservice/part-07.md"
+ }
+ ]
+ },
{
"text": "Community Articles",
"path": "https://abp.io/community"
diff --git a/docs/en/framework/fundamentals/dependency-injection.md b/docs/en/framework/fundamentals/dependency-injection.md
index dc0b847cbf..9dda3aef45 100644
--- a/docs/en/framework/fundamentals/dependency-injection.md
+++ b/docs/en/framework/fundamentals/dependency-injection.md
@@ -498,6 +498,8 @@ Use `ICachedServiceProvider` (instead of `ITransientCachedServiceProvider`) unle
> ABP also provides the `IAbpLazyServiceProvider` service. It does exists for backward compatibility and works exactly same with the `ITransientCachedServiceProvider` service. So, use the `ITransientCachedServiceProvider` since the `IAbpLazyServiceProvider` might be removed in future ABP versions.
+> Another advantage of using `ICachedServiceProvider` is that, during an HTTP request, if a service's constructor requires injecting many dependencies, it can negatively impact performance, as the injected services may not all be used by the current request. By resolving services on-demand, performance degradation can be effectively avoided.
+
## Advanced Features
### IServiceCollection.OnRegistered Event
diff --git a/docs/en/framework/infrastructure/audit-logging.md b/docs/en/framework/infrastructure/audit-logging.md
index f04bfafbe5..3392f5ac35 100644
--- a/docs/en/framework/infrastructure/audit-logging.md
+++ b/docs/en/framework/infrastructure/audit-logging.md
@@ -106,6 +106,24 @@ Configure(options =>
`IgnoredUrls` is the only option. It is a list of ignored URLs prefixes. In the preceding example, all URLs starting with `/products` will be ignored for audit logging.
+## AbpAspNetCoreAuditingUrlOptions
+
+`AbpAspNetCoreAuditingUrlOptions` is the [options object](../fundamentals/options.md) to configure audit logging in the ASP.NET Core layer. You can configure it in the `ConfigureServices` method of your [module](../architecture/modularity/basics.md):
+
+````csharp
+Configure(options =>
+{
+ options.IncludeQuery = true;
+});
+````
+
+Here, a list of the options you can configure:
+
+* `IncludeSchema` (default: `false`): If you set to true, it will include the schema in the URL.
+* `IncludeHost` (default: `false`): If you set to true, it will include the host in the URL.
+* `IncludeQuery` (default: `false`): If you set to true, it will include the query string in the URL.
+
+
## Enabling/Disabling Audit Logging for Services
### Enable/Disable for Controllers & Actions
diff --git a/docs/en/framework/ui/angular/quick-start.md b/docs/en/framework/ui/angular/quick-start.md
index 32e4522937..dec3a23262 100644
--- a/docs/en/framework/ui/angular/quick-start.md
+++ b/docs/en/framework/ui/angular/quick-start.md
@@ -6,7 +6,7 @@
Please follow the steps below to prepare your development environment for Angular.
-1. **Install Node.js:** Please visit [Node.js downloads page](https://nodejs.org/en/download/) and download proper Node.js `v18.19+` installer for your OS. An alternative is to install [NVM](https://github.com/nvm-sh/nvm) and use it to have multiple versions of Node.js in your operating system.
+1. **Install Node.js:** Please visit [Node.js downloads page](https://nodejs.org/en/download/) and download proper Node.js `v20.11+` installer for your OS. An alternative is to install [NVM](https://github.com/nvm-sh/nvm) and use it to have multiple versions of Node.js in your operating system.
2. **[Optional] Install Yarn:** You may install Yarn v1.22+ (not v2) following the instructions on [the installation page](https://classic.yarnpkg.com/en/docs/install). Yarn v1 delivers an arguably better developer experience compared to npm v10 and below. You may skip this step and work with npm, which is built-in in Node.js, instead.
3. **[Optional] Install VS Code:** [VS Code](https://code.visualstudio.com/) is a free, open-source IDE which works seamlessly with TypeScript. Although you can use any IDE including Visual Studio or Rider, VS Code will most likely deliver the best developer experience when it comes to Angular projects. ABP project templates even contain plugin recommendations for VS Code users, which VS Code will ask you to install when you open the Angular project folder. Here is a list of recommended extensions:
- [Angular Language Service](https://marketplace.visualstudio.com/items?itemName=angular.ng-template)
diff --git a/docs/en/framework/ui/maui/index.md b/docs/en/framework/ui/maui/index.md
index 69f9518e92..573278a060 100644
--- a/docs/en/framework/ui/maui/index.md
+++ b/docs/en/framework/ui/maui/index.md
@@ -40,6 +40,9 @@ Open a command line terminal and run the `adb reverse` command to expose a port
> You should replace "44305" with the real port.
> You should run the command after starting the emulator.
+> If you don't have a separate installation of Android Debug Bridge, you can open it from **Visual Studio** by following toolbar menu `Tools` > `Android` > `Android Adb Command Prompt`. Android emulator has to be running for this operation.
+
+
### iOS
The iOS simulator uses the host machine network. Therefore, applications running in the simulator can connect to web services running on your local machine via the machines IP address or via the localhost hostname. For example, given a local secure web service that exposes a GET operation via the /api/todoitems/ relative URI, an application running on the iOS simulator can consume the operation by sending a GET request to https://localhost:/api/todoitems/.
diff --git a/docs/en/framework/ui/mvc-razor-pages/client-side-package-management.md b/docs/en/framework/ui/mvc-razor-pages/client-side-package-management.md
index bb984a001f..31e54079ba 100644
--- a/docs/en/framework/ui/mvc-razor-pages/client-side-package-management.md
+++ b/docs/en/framework/ui/mvc-razor-pages/client-side-package-management.md
@@ -41,7 +41,7 @@ After depending on a NPM package, all you should do is to run the **yarn** comma
yarn
```
-Alternatively, you can use `npm install` but [Yarn](https://classic.yarnpkg.com/) is suggested as mentioned before.
+Alternatively, you can use `npm install` but [Yarn v1.22+ (not v2)](https://classic.yarnpkg.com/en/docs/install) is suggested as mentioned before.
#### Package Contribution
diff --git a/docs/en/framework/ui/react-native/index.md b/docs/en/framework/ui/react-native/index.md
index 3dd60b6348..4b6a0b5ec7 100644
--- a/docs/en/framework/ui/react-native/index.md
+++ b/docs/en/framework/ui/react-native/index.md
@@ -17,7 +17,7 @@ ABP platform provide basic [React Native](https://reactnative.dev/) startup temp
Please follow the steps below to prepare your development environment for React Native.
-1. **Install Node.js:** Please visit [Node.js downloads page](https://nodejs.org/en/download/) and download proper Node.js v16 or v18 installer for your OS. An alternative is to install [NVM](https://github.com/nvm-sh/nvm) and use it to have multiple versions of Node.js in your operating system.
+1. **Install Node.js:** Please visit [Node.js downloads page](https://nodejs.org/en/download/) and download proper Node.js v20.11+ installer for your OS. An alternative is to install [NVM](https://github.com/nvm-sh/nvm) and use it to have multiple versions of Node.js in your operating system.
2. **[Optional] Install Yarn:** You may install Yarn v1 (not v2) following the instructions on [the installation page](https://classic.yarnpkg.com/en/docs/install). Yarn v1 delivers an arguably better developer experience compared to npm v6 and below. You may skip this step and work with npm, which is built-in in Node.js, instead.
3. **[Optional] Install VS Code:** [VS Code](https://code.visualstudio.com/) is a free, open-source IDE which works seamlessly with TypeScript. Although you can use any IDE including Visual Studio or Rider, VS Code will most likely deliver the best developer experience when it comes to React Native projects.
4. **Install an Emulator:** React Native applications need an Android emulator or an iOS simulator to run on your OS. See the [Android Studio Emulator](https://docs.expo.io/workflow/android-simulator/) or [iOS Simulator](https://docs.expo.io/workflow/ios-simulator/) on expo.io documentation to learn how to set up an emulator.
diff --git a/docs/en/get-started/images/abp-studio-microservice-solution-runner-docker-dependencies.png b/docs/en/get-started/images/abp-studio-microservice-solution-runner-docker-dependencies.png
deleted file mode 100644
index 99c09045f8..0000000000
Binary files a/docs/en/get-started/images/abp-studio-microservice-solution-runner-docker-dependencies.png and /dev/null differ
diff --git a/docs/en/get-started/images/abp-studio-microservice-solution-runner-enable-watch-1.png b/docs/en/get-started/images/abp-studio-microservice-solution-runner-enable-watch-1.png
new file mode 100644
index 0000000000..1d953a9552
Binary files /dev/null and b/docs/en/get-started/images/abp-studio-microservice-solution-runner-enable-watch-1.png differ
diff --git a/docs/en/get-started/images/abp-studio-microservice-solution-runner-enable-watch-2.png b/docs/en/get-started/images/abp-studio-microservice-solution-runner-enable-watch-2.png
new file mode 100644
index 0000000000..53152f1243
Binary files /dev/null and b/docs/en/get-started/images/abp-studio-microservice-solution-runner-enable-watch-2.png differ
diff --git a/docs/en/get-started/images/abp-studio-microservice-solution-runner-enable-watch.png b/docs/en/get-started/images/abp-studio-microservice-solution-runner-enable-watch.png
deleted file mode 100644
index 2efe8e5380..0000000000
Binary files a/docs/en/get-started/images/abp-studio-microservice-solution-runner-enable-watch.png and /dev/null differ
diff --git a/docs/en/get-started/images/abp-studio-no-layers-new-solution-additional-options-0.9.13.png b/docs/en/get-started/images/abp-studio-no-layers-new-solution-additional-options-0.9.13.png
new file mode 100644
index 0000000000..208e047662
Binary files /dev/null and b/docs/en/get-started/images/abp-studio-no-layers-new-solution-additional-options-0.9.13.png differ
diff --git a/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-0.9.13.png b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-0.9.13.png
new file mode 100644
index 0000000000..c0b9f85ea6
Binary files /dev/null and b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-0.9.13.png differ
diff --git a/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-configurations-efcore-0.9.13.png b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-configurations-efcore-0.9.13.png
new file mode 100644
index 0000000000..bb85423493
Binary files /dev/null and b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-configurations-efcore-0.9.13.png differ
diff --git a/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-configurations-mongo-0.9.13.png b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-configurations-mongo-0.9.13.png
new file mode 100644
index 0000000000..3d4f435955
Binary files /dev/null and b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-configurations-mongo-0.9.13.png differ
diff --git a/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-provider-efcore-0.9.13.png b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-provider-efcore-0.9.13.png
new file mode 100644
index 0000000000..7fc1b37953
Binary files /dev/null and b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-provider-efcore-0.9.13.png differ
diff --git a/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-provider-mongo-0.9.13.png b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-provider-mongo-0.9.13.png
new file mode 100644
index 0000000000..cfe31c98d9
Binary files /dev/null and b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-database-provider-mongo-0.9.13.png differ
diff --git a/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-solution-properties-0.9.13.png b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-solution-properties-0.9.13.png
new file mode 100644
index 0000000000..4a217fd933
Binary files /dev/null and b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-solution-properties-0.9.13.png differ
diff --git a/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-ui-framework-0.9.13.png b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-ui-framework-0.9.13.png
new file mode 100644
index 0000000000..eb6c95e5bb
Binary files /dev/null and b/docs/en/get-started/images/abp-studio-no-layers-new-solution-dialog-ui-framework-0.9.13.png differ
diff --git a/docs/en/get-started/images/abp-studio-nolayers-new-solution-dialog-ui-theme-0.9.13.png b/docs/en/get-started/images/abp-studio-nolayers-new-solution-dialog-ui-theme-0.9.13.png
new file mode 100644
index 0000000000..ba5d165e26
Binary files /dev/null and b/docs/en/get-started/images/abp-studio-nolayers-new-solution-dialog-ui-theme-0.9.13.png differ
diff --git a/docs/en/get-started/index.md b/docs/en/get-started/index.md
index 6ffbfe548f..59f9812563 100644
--- a/docs/en/get-started/index.md
+++ b/docs/en/get-started/index.md
@@ -5,7 +5,7 @@ Great that you've decided to create a new application with ABP. ABP provides mul
Please select one of the following documents best fits for your application:
- **[Single-Layer Solution](single-layer-web-application.md)**: Creates a single-project solution. Recommended for building an application with a **simpler and easy to understand** architecture.
-- **[Layered Solution](layered-web-application.md)**: A fully layered (multiple projects) solution based on [Domain Driven Design](../framework/architecture/domain-driven-design) practices. Recommended for long-term projects that need a **maintainable and extensible** codebase.
+- **[Application (Layered)](layered-web-application.md)**: A fully layered (multiple projects) solution based on [Domain Driven Design](../framework/architecture/domain-driven-design) practices. Recommended for long-term projects that need a **maintainable and extensible** codebase.
- **[Microservice Solution](microservice.md)**: A **distributed solution** to build **microservice systems**. It includes pre-built services, API gateways, web and mobile applications, Kubernetes and Helm configuration, and everything you need to start your large-scale microservice solution.
- **Others**
- [Empty ASP.NET Core Application](empty-aspnet-core-application.md)
diff --git a/docs/en/get-started/layered-web-application.md b/docs/en/get-started/layered-web-application.md
index e8896b8666..f35b1c39db 100644
--- a/docs/en/get-started/layered-web-application.md
+++ b/docs/en/get-started/layered-web-application.md
@@ -19,10 +19,10 @@ First things first! Let's setup your development environment before creating the
The following tools should be installed on your development machine:
-* [Visual Studio 2022](https://visualstudio.microsoft.com/vs/) (v17.3+) for Windows / [Visual Studio for Mac](https://visualstudio.microsoft.com/vs/mac/). [1](#f-editor)
-* [.NET 8.0+](https://dotnet.microsoft.com/en-us/download/dotnet)
+* [Visual Studio 2022](https://visualstudio.microsoft.com/) or another IDE that supports [.NET 9.0+](https://dotnet.microsoft.com/download/dotnet) development. [1](#f-editor)
+* [.NET 9.0+](https://dotnet.microsoft.com/en-us/download/dotnet)
{{ if UI != "Blazor" }}
-* [Node v18.19+](https://nodejs.org/)
+* [Node v20.11+](https://nodejs.org/)
* [Yarn v1.22+ (not v2)](https://classic.yarnpkg.com/en/docs/install) [2](#f-yarn) or npm v10+ (already installed with Node)
{{ end }}
{{ if Tiered == "Yes" }}
diff --git a/docs/en/get-started/microservice.md b/docs/en/get-started/microservice.md
index 5be705e215..24fc745fe0 100644
--- a/docs/en/get-started/microservice.md
+++ b/docs/en/get-started/microservice.md
@@ -73,11 +73,11 @@ Click the Next button to see *Additional Options* selection:
If you unchecked the *Kubernetes Configuration* option, the solution will not include the Kubernetes configuration files which include the Helm charts and other Kubernetes related files. You can also specify *Social Logins*; if you uncheck this option, the solution will not be configured for social login. Lastly, you can specify the *Include Tests* option to include the test projects in the solution.
-Now, we are ready to allow ABP Studio to create our solution. Just click the *Create* button and let the ABP Studio do the rest for you. After clicking the Create button, the dialog is closed and your solution is loaded into ABP Studio:
+Now, we are ready to allow ABP Studio to create our solution. Just click the *Create* button and let the ABP Studio do the rest for you. After clicking the *Create* button, the dialog is closed and your solution is loaded into ABP Studio:

-You can explore the solution, but you need to wait for background tasks to be completed before running any application in the solution (it can take up to a few minutes to set up all).
+You can explore the solution, but you need to **wait for background tasks to be completed** before running any application in the solution (it can take up to a few minutes to set up all).
> The solution structure can be different in your case based on the options you've selected.
@@ -123,9 +123,7 @@ In the *Solution Runner* section (on the left side) you can see all the runnable

-> All the leaf items in the *Solution Runner* is called as an *Application* as they are executable applications.
-
-> For a faster start process, first start the *Docker-Dependencies*, then you can start all applications.
+> A leaf item in the *Solution Runner* is called as an *Application* as it is an executable application.
As shown in the figure above, the executable applications are grouped into folders like `apps`, `gateways`, `infrastructure`, and `services`. You can start/stop them all, a group (folder) of them, or one by one.
@@ -135,16 +133,14 @@ Before running the applications, it is good to be sure that all applications are
> *Solution Runner* doesn't build an application before running it. That provides a great performance gain because most of the time you will work on one or a few services and you don't need to build all of the other applications in every run. However, if you want to build before running, you can right-click an item in the *Solution Runner* tree and select *Run* -> *Build & Start* command.
-It will take some time to build all. Once all is done, you can start the system.
-
-You can click the *Play* button on the root item in Solution Runner to start all the applications. Or you can start `Docker-Dependencies` first, so the database and other infrastructure services get ready before the other applications:
-
-
+It will take some time to build all. Once all is done, you can start the system. You can click the *Play* button on the root item in Solution Runner to start all the applications.
+> **About the Docker Containers**
+>
> Docker will fetch the docker images before starting the containers in your first run (if they were not fetched before) and that process may take a few minutes depending on your internet connection speed. So, please wait for it to completely start. If the process takes more time than you expect, you can right-click on `Docker-Dependencies` and select the *Logs* command to see what's happening.
-Once `Docker-Dependencies` is ready, you can click the *Play* button on the root item in Solution Runner to start all the applications.
-
+> **About Failing Services on Startup**
+>
> Some applications/services may fail on the first run. That may be because of service and database dependencies were not satisfied and an error occurs on the application startup. ABP Studio automatically restarts failing services until it is successfully started. Being completely ready for such a distributed solution may take a while, but it will be eventually started.
Once all the applications are ready, you can right-click the `Web` application and select the *Browse* command:
@@ -175,9 +171,11 @@ Once you run the `IdentityService` in Visual Studio, it will be completely integ
As an alternative approach, especially if you don't need to debug your service, you can enable the watching feature of ABP Studio to automatically re-build and re-start when there is change in your application/service.
-To enable watching, right-click the application/service you want to watch, select the *Run* -> *Enable Watch* command as shown in the following figure:
+To enable watching, right-click the application/service you want to watch, select the *Properties* -> *Watch changes while running* option as shown in the following figure:
+
+
-
+
Now, you can make your development on the `IdentityService`. Whenever you save a code file, it is automatically rebuilt and restarted by ABP Studio, so any change will be effective on the running solution in a few seconds.
@@ -185,8 +183,6 @@ When you enable watch for an application an *eye* icon is added near to the appl

-You can disable watching by right-clicking an application and selecting *Run* -> *Disable Watch* command.
-
## Kubernetes Integration: Working with Helm Charts
Solution Runner is a great way to locally run all the applications and services of your solution. However, there are some drawbacks:
@@ -222,6 +218,8 @@ Once the solution is ready in Kubernetes, you can open a browser and visit the f

+> We could use `cloudcrm-local-web` as the host name since ABP Studio has added an entry to the host file for us.
+
Click the *Login* link in the application UI, it will redirect you to the *Authentication Server* application, enter `admin` as username and `1q2w3E*` as password to login to the application.
> The services run independently from each other and perform some initial data seed logic on their startups. So, they may fail in their first run. In that case, Kubernetes will re-start them. So, it may initially get some time to make the solution fully ready and working.
@@ -248,11 +246,11 @@ Clicking the *Connect* button will start a process that establishes the VPN conn

-Now, you can access all the services inside the Kubernetes cluster, including the services those are not exposes out of the cluster. You can use the service name as DNS. For example, you can directly visit `http://cloudcrm-local-identity` in your Browser. You can also right-click to a service or application and select the Browse command to open it's UI in the built-in browser of ABP Studio:
+Now, you can access all the services inside the Kubernetes cluster, including the services those are not exposed out of the cluster. You can use the service name as DNS. For example, you can directly visit `http://cloudcrm-local-identity` in your Browser. You can also right-click to a service or application and select the Browse command to open it's UI in the built-in browser of ABP Studio:

-You can even use the other services (e.g. SQL Server or RabbitMQ) from your local computer (even if they were not exposed out of cluster) with their service names. `sa` password for the SQL server is `myPassw@rd` by default, you can use your SQL Server management studio to connect to it and see the databases:
+You can even use the other services (e.g. SQL Server or RabbitMQ) from your local computer (even if they were not exposed out of cluster) with their service names. `sa` password for the SQL server is `myPassw@rd` by default, you can use your SQL Server management studio to connect to it and see the databases (*Server name* is `cloudcrm-local-sqlserver`):

@@ -266,6 +264,8 @@ When you connect to Kubernetes, ABP Studio automatically connects to the applica
In this way, you can easily track HTTP requests, distributed events, exceptions, logs and other details of your applications.
+> If you want to browse a web application in the integrated browser of ABP Studio, right-click to a service in the *Kubernetes* tab of the *Kubernetes* panel and select the *Browse* command.
+
## Kubernetes Integration: Intercepting Services
The next step is to intercept a service to forward the traffic (coming to that service) to your local computer, so you can run the same service in your local computer to test, debug and develop it. This is the way of connecting two environments (your local machine and the Kubernetes cluster) to develop your services integrated to Kubernetes.
@@ -306,3 +306,7 @@ To re-deploy a service to Kubernetes, right-click the service and select *Comman

ABP Studio will re-build the Docker image and re-install it using the related Helm chart.
+
+## See Also
+
+* [Microservice Development Tutorial](../tutorials/microservice/index.md)
diff --git a/docs/en/get-started/single-layer-web-application.md b/docs/en/get-started/single-layer-web-application.md
index 319fb19c9a..9bcf2e0e7e 100644
--- a/docs/en/get-started/single-layer-web-application.md
+++ b/docs/en/get-started/single-layer-web-application.md
@@ -18,10 +18,10 @@ First things first! Let's setup your development environment before creating the
The following tools should be installed on your development machine:
-* [Visual Studio 2022](https://visualstudio.microsoft.com/vs/) (v17.3+) for Windows / [Visual Studio for Mac](https://visualstudio.microsoft.com/vs/mac/). [1](#f-editor)
-* [.NET 8.0+](https://dotnet.microsoft.com/en-us/download/dotnet)
+* [Visual Studio 2022](https://visualstudio.microsoft.com/) or another IDE that supports [.NET 9.0+](https://dotnet.microsoft.com/download/dotnet) development. [1](#f-editor)
+* [.NET 9.0+](https://dotnet.microsoft.com/en-us/download/dotnet)
{{ if UI != "Blazor" }}
-* [Node v18.19+](https://nodejs.org/)
+* [Node v20.11+](https://nodejs.org/)
* [Yarn v1.22+ (not v2)](https://classic.yarnpkg.com/en/docs/install) [2](#f-yarn) or npm v10+ (already installed with Node)
{{ end }}
{{ if Tiered == "Yes" }}
@@ -50,11 +50,11 @@ Assuming that you have [installed and logged in](../studio/installation.md) to t
Select the *File* -> *New Solution* in the main menu, or click the *New solution* button on the Welcome screen to open the *Create new solution* wizard:
-
+
We will use the *Application (Single Layer)* solution template for this tutorial, so pick it and click the *Next* button:
-
+
On that screen, you choose a name for your solution. You can use different levels of namespaces; e.g. `BookStore`, `Acme.BookStore` or `Acme.Retail.BookStore`.
@@ -62,31 +62,35 @@ Then select an *output folder* to create your solution. The *Create solution fol
Once your configuration is done, click the *Next* button to navigate to the *UI Framework* selection:
-
+
Here, you see all the possible UI options supported by that startup solution template. Pick the **{{ UI_Value }}**.
Notice that; Once you select a UI type, some additional options will be available under the UI Framework list. You can further configure the options or leave them as default and click the *Next* button for the *UI Theme* selection screen:
-
+
LeptonX is the suggested UI theme that is proper for production usage. Select one of the themes, configure the additional options, and click the *Next* button for the *Database Provider* selection:
{{ if DB == "EF" }}
-
+
{{ else }}
-
+
{{ end }}
On that screen, you can decide on your database provider by selecting one of the provided options. There are some additional options for each database provider. Leave them as default or change them based on your preferences, then click the *Next* button for additional *Database Configurations*:
{{ if DB == "EF" }}
-
+
{{ else }}
-
+
{{ end }}
-Here, you can select the database management systems (DBMS){{ if DB == "EF" }} and the connection string{{ end }}. Now, we are ready to allow ABP Studio to create our solution. Just click the *Create* button and let the ABP Studio do the rest for you.
+Here, you can select the database management systems (DBMS){{ if DB == "EF" }} and the connection string{{ end }}. Then, you can select optional modules and enable additional options according to your preferences.
+
+
+
+Now, we are ready to allow ABP Studio to create our solution. Just click the *Create* button and let the ABP Studio do the rest for you.
After clicking the Create button, the dialog is closed and your solution is loaded into ABP Studio:
diff --git a/docs/en/images/generic-repositories.png b/docs/en/images/generic-repositories.png
index 6287afc56c..00f751a7bf 100644
Binary files a/docs/en/images/generic-repositories.png and b/docs/en/images/generic-repositories.png differ
diff --git a/docs/en/images/idle-message.png b/docs/en/images/idle-message.png
new file mode 100644
index 0000000000..14ea10c9f8
Binary files /dev/null and b/docs/en/images/idle-message.png differ
diff --git a/docs/en/images/idle-setting.png b/docs/en/images/idle-setting.png
new file mode 100644
index 0000000000..faecb9174c
Binary files /dev/null and b/docs/en/images/idle-setting.png differ
diff --git a/docs/en/images/pen-test-alert-list-9.0.png b/docs/en/images/pen-test-alert-list-9.0.png
new file mode 100644
index 0000000000..ad07a574ca
Binary files /dev/null and b/docs/en/images/pen-test-alert-list-9.0.png differ
diff --git a/docs/en/images/suite-registry.png b/docs/en/images/suite-registry.png
new file mode 100644
index 0000000000..f1a968bfa8
Binary files /dev/null and b/docs/en/images/suite-registry.png differ
diff --git a/docs/en/modules/account-pro.md b/docs/en/modules/account-pro.md
index 327a274f84..0dd960e97e 100644
--- a/docs/en/modules/account-pro.md
+++ b/docs/en/modules/account-pro.md
@@ -358,3 +358,4 @@ This module doesn't define any additional distributed event. See the [standard d
* [Impersonation](./account/impersonation.md)
* [Linked Accounts](./account/linkedaccounts.md)
* [Session Management](./account/session-management.md)
+* [Idle Session Timeout](./account/idle-session-timeout.md)]
diff --git a/docs/en/modules/account/idle-session-timeout.md b/docs/en/modules/account/idle-session-timeout.md
new file mode 100644
index 0000000000..069e6db665
--- /dev/null
+++ b/docs/en/modules/account/idle-session-timeout.md
@@ -0,0 +1,19 @@
+# Idle Session Timeout
+
+The `Idle Session Timeout` feature allows you to automatically log out users after a certain period of inactivity.
+
+## Configure Idle Session Timeout
+
+You can enable/disable the `Idle Session Timeout` feature in the `Setting > Account > Idle Session Timeout` page.
+
+The default idle session timeout is 1 hour. You can change it by selecting a different value from the dropdown list or entering a custom value(in minutes).
+
+
+
+Once the idle session timeout is reached, the user will see a warning modal before being logged out. if user does not respond for 60 seconds, the user will be logged out automatically.
+
+
+
+## How it works
+
+There is JavaScript code running in the background to detect user activity. such as mouse movement, key press, click, etc. If there is no activity detected for setting time, The warning modal will be shown to the user.
diff --git a/docs/en/others/penetration-test-report.md b/docs/en/others/penetration-test-report.md
index c092a856ea..c9d8280405 100644
--- a/docs/en/others/penetration-test-report.md
+++ b/docs/en/others/penetration-test-report.md
@@ -1,6 +1,6 @@
# ABP Penetration Test Report
-The ABP Commercial MVC `v8.3.0` application template has been tested against security vulnerabilities by the [OWASP ZAP v2.14.0](https://www.zaproxy.org/) tool. The demo web application was started on the `https://localhost:44349` address. The below alerts have been reported by the pentest tool. These alerts are sorted by the risk level as high, medium, and low. The informational alerts are not mentioned in this document.
+The ABP Commercial MVC `v9.0.0` application template has been tested against security vulnerabilities by the [OWASP ZAP v2.14.0](https://www.zaproxy.org/) tool. The demo web application was started on the `https://localhost:44349` address. The below alerts have been reported by the pentest tool. These alerts are sorted by the risk level as high, medium, and low. The informational alerts are not mentioned in this document.
Many of these alerts are **false-positive**, meaning the vulnerability scanner detected these issues, but they are not exploitable. It's clearly explained for each false-positive alert why this alert is a false-positive.
@@ -10,14 +10,14 @@ In the next sections, you will find the affected URLs, attack parameters (reques
There are high _(red flag)_, medium _(orange flag)_, low _(yellow flag)_, and informational _(blue flag)_ alerts.
-
+
> The informational alerts are not mentioned in this document. These alerts are not raising any risks on your application and they are optional.
### Path Traversal [Risk: High] - False Positive
-- *[GET] - https://localhost:44349/api/audit-logging/audit-logs?startTime=&endTime=&url=&userName=&applicationName=&clientIpAddress=&correlationId=&httpMethod=audit-logs&httpStatusCode=&maxExecutionDuration=&minExecutionDuration=&hasException=true&sorting=executionTime+desc&skipCount=0&maxResultCount=10* (attack: **httpMethod=audit-logs**)
- *[POST] - https://localhost:44349/Account/Login* (attack: **\Login**)
+- *[POST] - https://localhost:44349/Account/LinkLogin* (attack: **\LinkLogin**)
- *[POST] - https://localhost:44349/Account/Register* (attack: **\Register**)
- *[POST] - https://localhost:44349/Account/SecurityLogs* (attack: **\SecurityLogs**)
- *[POST] - https://localhost:44349/Identity/SecurityLogs* (attack: **\SecurityLogs**)
@@ -28,19 +28,15 @@ The Path Traversal attack technique allows an attacker access to files, director
**Solution**:
-This is a **false-positive** alert since ABP does all related checks for this kind of attack on the backend side for these endpoints.
+This is a **false-positive** alert since ABP does all related checks for this kind of attacks on the backend side for these endpoints.
### SQL Injection [Risk: High] - False Positive
* *[POST] — https://localhost:44349/Account/Login* (attack: **1q2w3E* AND 1=1 --**)
-* *[POST] — https://localhost:44349/AuditLogs* (attack: **GET' AND '1'='1' --**)
-* *[POST] — https://localhost:44349/Identity/SecurityLogs* (attack: **admin' AND '1'='1**)
-* *[POST] — https://localhost:44349/api/account/verify-authenticator-code* (attack: **AND '1'='1**)
-* *[POST] — https://localhost:44349/Identity/ClaimTypes/CreateModal* (attack: **aaaa AND '1'='1**)
+* *[POST] — https://localhost:44349/Account/ImpersonateUser* (attack: **CfDJ8Pyqeg0vtHtJpnK-9eLaft7-JxLJfJ6WHKPOdBZVxz14BDo061qpJ2NLplgAn2Hw16ec0IR38_wWAUkJGxP8hL6PcLfH0bh-ATNTspWyWYTGGbiH-zeKWiS5vWX-br2BA1hE7Dc45eWGUZNcVc_vm2s AND 1=1 --**)
+* *[POST] — https://localhost:44349/Abp/MultiTenancy/TenantSwitchModal* (attack: **CfDJ8Pyqeg0vtHtJpnK-9eLaft7-JxLJfJ6WHKPOdBZVxz14BDo061qpJ2NLplgAn2Hw16ec0IR38_wWAUkJGxP8hL6PcLfH0bh-ATNTspWyWYTGGbiH-zeKWiS5vWX-br2BA1hE7Dc45eWGUZNcVc_vm2s AND 1=1 --**)
* *[POST] — https://localhost:44349/Identity/OrganizationUnits/\** (attack: **6f4cd0ab-f4eb-7ce0-8b26-3a138af1840d" AND '1'='1**) (also, several other URLs...)
-* *[POST] — https://localhost:44349/Identity/ClaimTypes/EditModal* (attack: **aaaa AND '1'='1**)
-* *[POST] — https://localhost:44349/LanguageManagement/Texts* (attack: **true" AND "1"="1" --**)
-* *[POST] — https://localhost:44349/Account/Manage?CurrentPassword=ZAP%27+AND+%271%27%3D%271%27+--+&NewPassword=ZAP&NewPasswordConfirm=ZAP*
+* *[POST] — https://localhost:44349/Identity/ClaimTypes/CreateModal* (attack: **aaaad AND '1'='1**)
**Description**:
@@ -95,11 +91,17 @@ There are only one URL that is reported as exposing error messages. This is a **
### Content Security Policy (CSP) Header Not Set [Risk: Medium] — Positive (Fixed)
- *[GET] — https://localhost:44349*
+- *[GET] — https://localhost:44349/AuditLogs*
+- *[GET] — https://localhost:44349/CookiePolicy*
+- *[GET] — https://localhost:44349/Gdpr/PersonalData*
+- *[GET] — https://localhost:44349/Identity/ClaimTypes/{0}* (create & edit modal URLs - also there are other modal related URLs...)
- *[GET] — https://localhost:44349/AbpPermissionManagement/PermissionManagementModal?providerName=R&providerKey=role&providerKeyDisplayName=role*
- *[GET] — https://localhost:44349/Abp/MultiTenancy/TenantSwitchModal*
- *[GET] — https://localhost:44349/Account/AuthorityDelegation/AuthorityDelegationModal*
- *[GET] — https://localhost:44349/Account/AuthorityDelegation/DelegateNewUserModal*
- *[GET] — https://localhost:44349/Account/ForgotPassword _(other several account URLS)_*
+- *[GET] — https://localhost:44349/Account/ExternalLogins _(other several account URLS)_*
+- *[GET] — https://localhost:44349/Account/SecurityLogs _(other several account URLS)_*
- *[GET] — https://localhost:44349/Account/Login _(other several account URLS)_*
- *[GET] — https://localhost:44349/Account/Register _(other several account URLS)_*
- *[GET] — https://localhost:44349/Account/Manage _(other several account URLS)_*
@@ -143,12 +145,13 @@ The first affected URL is a **false-positive** alert since it's already fixed an
The second URL is also a **false-positive** alert because there is no bad character string in the response.
-> **Note**: However, it might be possible if you had any sensitive localization key-value pair in your localization entries, because this endpoint returns all localization values to be able to be used in the application. Therefore, keep that in mind while defining new localization entries.
+> **Note**: However, it might be possible if you had any sensitive localization key-value pair in your localization entries, because this endpoint returns all localization values to be able to be used in the application. Therefore, keep that in mind while defining new localization entries. Pass the critical values in your code while using the localization entry as a parameter.
### XSLT Injection [Risk: Medium] - False Positive
- *[GET] — https://localhost:44349/Abp/Languages/Switch?culture=%3Cxsl%3Avalue-of+select%3D%22system-property%28%27xsl%3Avendor%27%29%22%2F%3E&returnUrl=%2F&uiCulture=ar*
- *[POST] — https://localhost:44349/Account/Login _(same URL with different parameters...)_*
+- *[POST] — https://localhost:44349/Account/ImpersonateUser _(same URL with different parameters...)_*
- *[POST] — https://localhost:44349/Account/Register _(same URL with different parameters...)_*
- *[POST] — https://localhost:44349/Account/Manage _(same URL with different parameters...)_*
- *[POST] — https://localhost:44349/Account/ForgotPassword _(same URL with different parameters...)_*
@@ -161,13 +164,14 @@ Injection using XSL transformations may be possible and may allow an attacker to
**Explanation**:
-This is a **false-positive** alert. v8.3.0 uses .NET 8 and the XSLT transformation is not possible on .NET5 or higher.
+This is a **false-positive** alert. v9.0 uses .NET 9 and the XSLT transformation is not possible on .NET5 or higher.
### Application Error Disclosure [Risk: Low] — False Positive
+- *[GET] — https://localhost:44349/Abp/Languages/Switch*
- *[POST] — https://localhost:44349/Account/ImpersonateUser*
-- *[POST] — https://localhost:44349/Saas/Host/Editions*
-- *[POST] — https://localhost:44349/Saas/Host/Tenants*
+- *[GET] — https://localhost:44349/Account/ExternalLogins*
+- *[GET] — https://localhost:44349/Account/Logout*
**Description:**
@@ -304,6 +308,7 @@ This vulnerability was reported as a positive alert because the application ran
### Timestamp Disclosure - Unix [Risk: Low] - False Positive
- *[GET] — https://localhost:44349/libs/zxcvbn/zxcvbn.js?=*
+- *[GET] — https://localhost:44349/libs/sweetalert2/sweetalert2.all.min.js?=*
**Description**:
@@ -319,8 +324,9 @@ This vulnerability was reported as a positive alert, because ABP uses the [zxcvb
### X-Content-Type-Options Header Missing [Risk: Low] - Positive (Fixed)
-- *[GET] — https://localhost:44349/client-proxies/account-proxy.js?_v=638550091940000000 (and other client-proxies related URLs)*
+- *[GET] — https://localhost:44349/client-proxies/account-proxy.js?_v=638550091940000000 (and other client-proxies related URLs...)*
- *[GET] — https://localhost:44349/favicon.svg*
+- *[GET] — https://localhost:44349/images/getting-started/bg-01.png* (and other image URLs...)
- *[GET] — https://localhost:44349/global-styles.css?_v=638556076064360335*
- *[GET] — https://localhost:44349/libs/@fortawesome/fontawesome-free/css/all.css?_v=%5CWEB-INF%5Cweb.xml (other several URLs...)*
- other URLs...
@@ -340,11 +346,3 @@ If possible, ensure that the end user uses a standards-compliant and modern web
The `X-Content-Type-Options` header allows you to avoid MIME type sniffing by saying that the MIME types are deliberately configured. This headeer is not strictly required, but it is highly recommended for security reasons. While modern browsers have improved security features, you can still set this header for ensuring the security of web applications.
You can add the [ABP's Security Header Middleware](../framework/ui/mvc-razor-pages/security-headers.md#security-headers-middleware) into the request pipeline to set the `X-Content-Type-Options` as *no-sniff*. Also, this middleware adds other pre-defined security headers to your application, including `X-XSS-Protection`, `X-Frame-Options` and `Content-Security-Policy` (if it's enabled). Read [Security Headers](../framework/ui/mvc-razor-pages/security-headers.md) documentation for more info.
-
-## Other Alerts (Fixed)
-
-The following alerts were reported by the community or our customers in v8.2 and fixed:
-
-* https://github.com/abpframework/abp/issues/19576
-* https://github.com/abpframework/abp/issues/19588
-* https://github.com/abpframework/abp/issues/19589
diff --git a/docs/en/samples/easy-crm.md b/docs/en/samples/easy-crm.md
index 480aa80316..1a0fff3b3f 100644
--- a/docs/en/samples/easy-crm.md
+++ b/docs/en/samples/easy-crm.md
@@ -35,7 +35,7 @@ When you download and open the zip file, you will see two folders:
* First, follow all the steps above to run the server side and seed the sample data.
* Open a command prompt in the angular folder.
-* Run the `yarn` command to install NPM packages (requires the [Yarn](https://yarnpkg.com/) package manager).
+* Run the `yarn` command to install NPM packages (requires the [Yarn v1.22+ (not v2)](https://classic.yarnpkg.com/en/docs/install) package manager).
* Run the `yarn start` command to run the Angular application. It will automatically open the `localhost://4200` in your default browser once the application initialized.
### Blazor UI
diff --git a/docs/en/solution-templates/microservice/guides/add-new-microservice.md b/docs/en/solution-templates/microservice/guides/add-new-microservice.md
index 12c8b527e8..f68f315dd7 100644
--- a/docs/en/solution-templates/microservice/guides/add-new-microservice.md
+++ b/docs/en/solution-templates/microservice/guides/add-new-microservice.md
@@ -1,3 +1 @@
-# ABP Studio Microservice Solution: Adding New Microservices
-
-This document is planned to be written later.
\ No newline at end of file
+> This document is [moved to here](../adding-new-microservices.md).
\ No newline at end of file
diff --git a/docs/en/studio/images/solution-runner/cli-application-context-menu.png b/docs/en/studio/images/solution-runner/cli-application-context-menu.png
index 0e35ecec0b..0a66138e61 100644
Binary files a/docs/en/studio/images/solution-runner/cli-application-context-menu.png and b/docs/en/studio/images/solution-runner/cli-application-context-menu.png differ
diff --git a/docs/en/studio/images/solution-runner/csharp-application-context-menu-build.png b/docs/en/studio/images/solution-runner/csharp-application-context-menu-build.png
index dea4556ad7..aec7d5d0d7 100644
Binary files a/docs/en/studio/images/solution-runner/csharp-application-context-menu-build.png and b/docs/en/studio/images/solution-runner/csharp-application-context-menu-build.png differ
diff --git a/docs/en/studio/images/solution-runner/csharp-application-context-menu-monitor.png b/docs/en/studio/images/solution-runner/csharp-application-context-menu-monitor.png
index 0743fb5ae9..42eca8cf00 100644
Binary files a/docs/en/studio/images/solution-runner/csharp-application-context-menu-monitor.png and b/docs/en/studio/images/solution-runner/csharp-application-context-menu-monitor.png differ
diff --git a/docs/en/studio/images/solution-runner/csharp-application-context-menu.png b/docs/en/studio/images/solution-runner/csharp-application-context-menu.png
index cd007f59a3..76c90a2e95 100644
Binary files a/docs/en/studio/images/solution-runner/csharp-application-context-menu.png and b/docs/en/studio/images/solution-runner/csharp-application-context-menu.png differ
diff --git a/docs/en/studio/images/solution-runner/folder-context-menu-add.png b/docs/en/studio/images/solution-runner/folder-context-menu-add.png
index 4dda7cfe03..c83a54f5e0 100644
Binary files a/docs/en/studio/images/solution-runner/folder-context-menu-add.png and b/docs/en/studio/images/solution-runner/folder-context-menu-add.png differ
diff --git a/docs/en/studio/images/solution-runner/folder-context-menu-build.png b/docs/en/studio/images/solution-runner/folder-context-menu-build.png
index ce72b6d8f5..3e5ad9b56d 100644
Binary files a/docs/en/studio/images/solution-runner/folder-context-menu-build.png and b/docs/en/studio/images/solution-runner/folder-context-menu-build.png differ
diff --git a/docs/en/studio/images/solution-runner/folder-context-menu.png b/docs/en/studio/images/solution-runner/folder-context-menu.png
index 1bc4478352..c17cb8659d 100644
Binary files a/docs/en/studio/images/solution-runner/folder-context-menu.png and b/docs/en/studio/images/solution-runner/folder-context-menu.png differ
diff --git a/docs/en/studio/images/solution-runner/manage-start-actions.png b/docs/en/studio/images/solution-runner/manage-start-actions.png
new file mode 100644
index 0000000000..4b96c08d3e
Binary files /dev/null and b/docs/en/studio/images/solution-runner/manage-start-actions.png differ
diff --git a/docs/en/studio/images/solution-runner/profile-root-context-menu-add.png b/docs/en/studio/images/solution-runner/profile-root-context-menu-add.png
index 11b8da1362..ec26ebcaf0 100644
Binary files a/docs/en/studio/images/solution-runner/profile-root-context-menu-add.png and b/docs/en/studio/images/solution-runner/profile-root-context-menu-add.png differ
diff --git a/docs/en/studio/images/solution-runner/profile-root-context-menu-build.png b/docs/en/studio/images/solution-runner/profile-root-context-menu-build.png
index 972f850149..db2ae4de3a 100644
Binary files a/docs/en/studio/images/solution-runner/profile-root-context-menu-build.png and b/docs/en/studio/images/solution-runner/profile-root-context-menu-build.png differ
diff --git a/docs/en/studio/images/solution-runner/profile-root-context-menu.png b/docs/en/studio/images/solution-runner/profile-root-context-menu.png
index a94f5b3704..d81bf68514 100644
Binary files a/docs/en/studio/images/solution-runner/profile-root-context-menu.png and b/docs/en/studio/images/solution-runner/profile-root-context-menu.png differ
diff --git a/docs/en/studio/images/solution-runner/solutioın-runner-properties.png b/docs/en/studio/images/solution-runner/solutioın-runner-properties.png
index 0dcbd08b1e..8611f8be3d 100644
Binary files a/docs/en/studio/images/solution-runner/solutioın-runner-properties.png and b/docs/en/studio/images/solution-runner/solutioın-runner-properties.png differ
diff --git a/docs/en/studio/installation.md b/docs/en/studio/installation.md
index 4dc158e464..34dde8a5c9 100644
--- a/docs/en/studio/installation.md
+++ b/docs/en/studio/installation.md
@@ -8,7 +8,7 @@
Before you begin the installation process for ABP Studio, ensure that your system meets the following pre-requirements:
### Node
-Make sure [Node.js](https://nodejs.org/en) is installed on your system. If you have not installed Node.js, you can download the `v18.19+` version from the official [Node.js website](https://nodejs.org/en/download/prebuilt-installer).
+Make sure [Node.js](https://nodejs.org/en) is installed on your system. If you have not installed Node.js, you can download the `v22+` version from the official [Node.js website](https://nodejs.org/en/download/prebuilt-installer).
### WireGuard (Optional)
ABP Studio needs [WireGuard](https://www.wireguard.com/) for Kubernetes operations. You can find the installation instructions for your specific operating system below:
@@ -19,7 +19,7 @@ Installation instructions for your Windows operating system are on the official
**For macOS:**
Installation instructions for your macOS operating system are on the official [WireGuard website](https://www.wireguard.com/install/#macos-homebrew-and-macports-basic-cli-homebrew-userspace-go-homebrew-tools-macports-userspace-go-macports-tools).
-### Docker (Optional)
+### Docker
ABP Studio needs [Docker](https://www.docker.com/) for [Kubernetes](https://kubernetes.io/) operations. Install Docker by following the guidelines on the official [Docker website](https://docs.docker.com/get-docker/).
## Installation
diff --git a/docs/en/studio/release-notes.md b/docs/en/studio/release-notes.md
index ed828a5141..282d038c5e 100644
--- a/docs/en/studio/release-notes.md
+++ b/docs/en/studio/release-notes.md
@@ -2,6 +2,31 @@
This document contains **brief release notes** for each ABP Studio release. Release notes only include **major features** and **visible enhancements**. Therefore, they don't include all the development done in the related version.
+## 0.9.15 (2024-12-05)
+
+* Upgraded templates to version `9.0.1`.
+* Fixed problems in the microservice service_nolayers template.
+* Fixed microservice angular template for wrong file-management module reference.
+* Fixed added extra lines in the hosts.txt file.
+
+## 0.9.14 (2024-12-03)
+
+* Refactored `dotnet watch` command in solution runner.
+* Added multi-tenancy option for open source startup templates.
+* Implemented adding angular library when a new microservice is created.
+* Adjusted *Optional Modules* section in solution configurations
+* Fixed bugs in the nolayers host project.
+
+## 0.9.13 (2024-11-25)
+
+* Angular - Theme-based Fixes for the Home Page.
+
+## 0.9.12 (2024-11-25)
+
+* Handled the `DynamicPermissionDefinitionsChangedEto` event to automatically add permissions for the admin role.
+* Enhanced the Solution Configuration window with a more intuitive design and updated content.
+* Improved MAUI application support by displaying all target frameworks in the Solution Runner and automatically setting the appropriate targetFramework based on the operating system.
+
## 0.9.11 (2024-11-21)
* Fixed the extension loading problem occured in v0.9.9 & v0.9.10.
diff --git a/docs/en/studio/running-applications.md b/docs/en/studio/running-applications.md
index a4dc4f8388..9e266ef7a9 100644
--- a/docs/en/studio/running-applications.md
+++ b/docs/en/studio/running-applications.md
@@ -41,27 +41,22 @@ When you click *Add New Profile*, it opens the *Create New Profile* window. You
## Using the Profile
-After selecting the current profile, which is the *Default* profile that comes pre-configured, we can utilize the tree items. This allows us to execute collective commands and create various tree structures based on our specific needs. You can navigate through the root of the tree and right-click to view the context menu, which includes 3 options: `Run`, `Build` and `Add`.
+After selecting the current profile, which is the *Default* profile that comes pre-configured, we can utilize the tree items. This allows us to execute collective commands and create various tree structures based on our specific needs. You can navigate through the root of the tree and right-click to view the context menu, which includes the following options: `Start All`, `Stop All`, `Build`, `Add`, and `Manage Start Actions`.

-### Run
+### Start/Stop All
-We can start/stop the applications with this option. Go to root of the tree and right-click to view the context menu, in this example *Acme.Bookstore(Default)* -> *Run*.
-
-
+We can start/stop the applications with these options. Go to the root of the tree and right-click to view the context menu:
- `Start All`: Start all(CLI, C#) applications.
- `Stop All`: Stop all(CLI, C#) applications.
-- `Build & Start All`: Builds each C# application in the [Background Tasks](./overview#background-tasks) and starts all (CLI, C#) applications after the build tasks are completed.
-
-> `Start All` doesn't build the C# applications before running. If you're running it for the first time or if you've made changes, you should build the applications. You can simply use the `Build & Start All`.
-> You can change the current profile while applications are running in the previous profile. The applications continue to run under the previous profile. For example if we start the `Acme.BookStore.AdministrationService`, `Acme.BookStore.IdentityService` applications when current profile is *team-1* and after change the current profile to *team-2* the applications continue to run under *team-1*.
+> You can change the current profile while applications are running in the previous profile. The applications continue to run under the previous profile. For example, if we start the `Acme.BookStore.AdministrationService`, `Acme.BookStore.IdentityService` applications when the current profile is *team-1* and after changing the current profile to *team-2* the applications continue to run under *team-1*.
### Build
-We can use common [dotnet](https://learn.microsoft.com/en-us/dotnet/core/tools) commands in this option. Go to root of the tree and right-click to view the context menu, in this example *Acme.Bookstore(Default)* -> *Build*, there are 4 options available:
+We can use common [dotnet](https://learn.microsoft.com/en-us/dotnet/core/tools) commands in this option. Go to the root of the tree and right-click to view the context menu, in this example *Acme.Bookstore(Default)* -> *Build*, there are 4 options available:

@@ -80,7 +75,7 @@ We can add 3 different item type to *Profile* for defining the tree structure. T
#### C# Application
-When we go to root of the tree and right-click, in this example *Acme.BookStore(Default)* -> *Add* -> *C# Application* it opens the *Add Application* window. There are two methods to add applications: *This solution* and *External*. To add via the *This solution* tab, follow these steps:
+When we go to the root of the tree and right-click, in this example *Acme.BookStore(Default)* -> *Add* -> *C# Application* it opens the *Add Application* window. There are two methods to add applications: *This solution* and *External*. To add via the *This solution* tab, follow these steps:

@@ -98,7 +93,7 @@ The C# project doesn't have to be within the current [Solution Explorer](./solut
- `Path`: Provide the path to the .csproj file you wish to add. The path will be [normalized](https://learn.microsoft.com/en-us/dotnet/standard/io/file-path-formats#path-normalization), allowing the project location to be flexible, as long as it's accessible from the current [ABP Solution](./concepts.md#solution).
- `Name`: Give an arbitrary name to see in solution runner. This name should be unique for each profile.
-- `Launch url`: Is the url when we want to browse. But if added project doesn't have launch url we can leave it empty.
+- `Launch url`: This is the url when we want to browse. But if the added project doesn't have launch url we can leave it empty.
- `Kubernetes service`: If you're not using the *Kubernetes* panel leave it empty. But if there is a helm chart for added application we should give the correct regex pattern. It's necessary for browse, when we connect the kubernetes cluster we should browse the services instead *Launch url*. Give the matching regex pattern for your helm chart kubernetes service name.
You can click the `OK` button to add the C# application to the profile.
@@ -137,17 +132,27 @@ You can click the `OK` button to add the folder to the profile.
- To remove a folder from the tree, open the context menu by right-clicking the folder and selecting *Delete*.
- When starting applications, they continue to restart until the application starts gracefully. To stop the restarting process when attempting to restart the application, click the icon on the left. Additionally, you can review the *Logs* to understand why the application isn't starting gracefully.
+### Manage Start Actions
+
+This command will open a dialog where you can set start actions and start orders of sub-applications and sub-folders.
+
+
+
+You can order the applications by dragging the icon in the first column. In the screenshot above, applications & folders are ordered like this: *Applications under infrastructure* > *Applications under services* > *Applications under gateways* > *AuthServer* > *Angular*. You can also set starting order and other actions for each folder by performing `right click > Manage Start Actions` on them.
+
+- **Action**: There are two options: `Start` and `Don't start`. This is usefull if you want to exclude applications from batch start.
+- **Build**: This option allows to disable/enable build before starting the application. If you are working on a single application, you can exclude the other applications from build to save time. This option can also be set by performing `right click > properties` on applications.
+- **Watch**: When enabled, changes in your code are watched and dotnet hot-reloads the application or restarts it if needed. This option also can be set by performing `right click > properties` on applications.
+
## Folder
-We already now why we need folder in the [previous](./running-applications.md#folder) section, we can use collective commands within this folder items. To do that go to folder and open the context menu by right-clicking, which includes 5 options `Start`, `Build`, `Add`, `Rename` and `Delete`.
+We already now why we need folder in the [previous](./running-applications.md#folder) section, we can use collective commands within this folder items. To do that go to folder and open the context menu by right-clicking, which includes 5 options `Start`, `Stop`, `Build`, `Add`, `Manage Start Actions`, `Rename` and `Delete`.

-### Start
-
-You can see the context menu by right-clicking *Folder* -> *Start*, it's [similar](#run) like *Acme.BookStore(Default)* -> *Run* options, there are 3 options available. The only difference with root of the tree and folder is gonna be execute in selected folder.
+### Start/Stop
-
+You can see the context menu by right-clicking *Folder*. It will start/stop all the applications under the folder.
### Build
@@ -157,7 +162,7 @@ You can see the context menu by right-clicking *Folder* -> *Start*, it's [simila
### Add
-*Folder* -> *Add* context menu, it's the [same](#add) options like *Acme.BookStore(Default)* -> *Add* there are 3 options avaiable. The only difference, it's gonna add item to selected folder.
+*Folder* -> *Add* context menu, it's the [same](#add) options like *Acme.BookStore(Default)* -> *Add* there are 3 options avaiable. The only difference, it's gonna add item to the selected folder.

@@ -168,29 +173,16 @@ You can see the context menu by right-clicking *Folder* -> *Start*, it's [simila
## C# Application
-The .NET icon indicates that the application is a C# project. After we [add](#c-application) the C# applications to root of the tree or folder, we can go to any C# application and right-click to view the context menu; `Run`, `Build`, `Browse`, `Requests`, `Exceptions`, `Logs`, `Copy URL`, `Properties`, `Remove`.
+The .NET icon indicates that the application is a C# project. After we [add](#c-application) the C# applications to the root of the tree or folder, we can go to any C# application and right-click to view the context menu; `Start`, `Build`, `Browse`, `Requests`, `Exceptions`, `Logs`, `Copy URL`, `Properties`, `Remove`.

-### Run
-
-We have several options in C# applications. Those options are `Start`(If the application started this option shown as `Stop`), `Build & Start`(If the application started this option shown as `Build & Restart`), `Enable Watch`(If the watch is enabled this option shown as `Disable Watch`), `Restart`(It's only shown when the application started)
-
-
+### Start
-- `Start`: Starts the selected application. This option doesn't build before run. If you're running it for the first time or if you've made changes, you should build the application. If the application is started this option changed as `Stop`.
-- `Build & Start`: We can simply use if we wanna build first and start. If the application is started this option changes to `Build & Restart`.
-- `Enable Watch`: When this option is enabled, there's no need to perform `Build & Start` after any change. ABP Studio watches for changes, re-builds, and re-runs your application automatically upon saving. If this option is enabled it changes to `Disable Watch`.
-- `Restart`: Restarts the application. This option visible only if the application started.
+Starts the selected application. Once it is started, *Stop* and *Restart* options will be available.
> When you start the C# application, you should see a *chain* icon next to the application name, that means the started application connected to ABP Studio. C# applications can connect to ABP Studio even when running from outside the ABP Studio environment, for example debugging with Visual Studio. If the application is run from outside the ABP Studio environment, it will display *(external)* information next to the chain icon.
-> When *Watch* is enable you should see an *eye* icon next to the application name.
-
-> Hint: Performing CTRL+Click on the start icon left to a stopped C# application equals to `Build & Start` command. Same applies for folders.
-
-
-
### Build
It's the [similar](#build) options like root of the tree options. The only difference between them it's gonna be execute the selected application.
@@ -214,7 +206,10 @@ We can open the *Application Properties* window to change *Launch url*, *Kuberne

-You can click the `OK` button to save the changes.
+- **Skip build before starting**: When enabled, application is started without build and it makes starting faster. This is useful when you are working on a single application out of multiple, so you don't need to build others everytime they start.
+- **Watch changes while running**: When enabled, you should see an *eye* icon next to the application name.
+
+
### Miscellaneous
@@ -228,7 +223,7 @@ CLI applications uses the [powershell](https://learn.microsoft.com/en-us/powersh

-- `Run`: This option includes 3 actions: *Start*, *Stop*, and *Restart* for the CLI application.
+- `Start`: Starts the application. Once it is started, *Start* and *Restart* options will be available.
- `Browse`: This option is available when a *Launch URL* is specified upon adding the CLI application. It opens the *Browse* tab, can be clicked while the application is running.
- `Logs`: It opens the *Logs* tab, we can see the logs for *Start* and *Stop* commands.
- `Copy URL`: This option copies the *Launch URL* of the selected application. It is visible if there is a specified *Launch URL*
diff --git a/docs/en/studio/version-mapping.md b/docs/en/studio/version-mapping.md
index c00b09ee95..fe91ffe89a 100644
--- a/docs/en/studio/version-mapping.md
+++ b/docs/en/studio/version-mapping.md
@@ -4,7 +4,8 @@ This document provides a general overview of the relationship between various ve
| **ABP Studio Version** | **ABP Version of Startup Template** |
|------------------------|---------------------------|
-| 0.9.9 to 0.9.11 | 9.0.0 |
+| 0.9.15 | 9.0.1 |
+| 0.9.9 to 0.9.14 | 9.0.0 |
| 0.9.8 | 8.3.4 |
| 0.9.5 to 0.9.7 | 8.3.3 |
| 0.9.2 to 0.9.4 | 8.3.2 |
diff --git a/docs/en/suite/how-to-start.md b/docs/en/suite/how-to-start.md
index 4552d23dbf..b8469ce26b 100644
--- a/docs/en/suite/how-to-start.md
+++ b/docs/en/suite/how-to-start.md
@@ -26,4 +26,20 @@ abp suite
If you run the ABP Suite with ABP CLI, then it will open in your default browser. Do not close the command line window until you finish your work, otherwise the Suite will not function. When you finish your work, you can return to the command line and press `CTRL+C` to close the Suite.
-Remember that, first access to the Suite requires to have an active internet connection, so make sure you are connected to the internet.
\ No newline at end of file
+Remember that, first access to the Suite requires to have an active internet connection, so make sure you are connected to the internet.
+
+## Starting ABP Suite in Different Port
+
+ABP Suite runs at port 3000 by default. If you want to run the ABP Suite in a different port, you can choose one of the following options:
+
+1. You can pass the `--AbpSuite:ApplicationUrl` commandline parameter to run in a different port:
+
+```bash
+abp-suite --AbpSuite:ApplicationUrl="http://localhost:4000"
+```
+
+2. Update the `ApplicationUrl` in the _appsettings.json_ file of the dotnet tools directory (_%USERPROFILE%\.dotnet\tools\.store\volo.abp.suite\9.0.0\volo.abp.suite\9.0.0\tools\net9.0\any\appsettings.json_):
+
+
+
+> **Note:** Also, you can use this file to configure ABP Suite options.
\ No newline at end of file
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-generated-tests.png b/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-generated-tests.png
new file mode 100644
index 0000000000..f678598b4d
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-generated-tests.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-navigation-property.png b/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-navigation-property.png
new file mode 100644
index 0000000000..03808be6ba
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-navigation-property.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-opening.png b/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-opening.png
new file mode 100644
index 0000000000..adebe653b5
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-opening.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-solution-test-projects.png b/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-solution-test-projects.png
new file mode 100644
index 0000000000..8a5689290d
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/abp-suite-solution-test-projects.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/book-store-suite-solution-explorer.png b/docs/en/tutorials/book-store-with-abp-suite/images/book-store-suite-solution-explorer.png
new file mode 100644
index 0000000000..7b53aca64d
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/book-store-suite-solution-explorer.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/bookstore-test-succeed.png b/docs/en/tutorials/book-store-with-abp-suite/images/bookstore-test-succeed.png
new file mode 100644
index 0000000000..8211638242
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/bookstore-test-succeed.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/studio-browser-suite.png b/docs/en/tutorials/book-store-with-abp-suite/images/studio-browser-suite.png
new file mode 100644
index 0000000000..80a53dcae2
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/studio-browser-suite.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-entity-1.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-entity-1.png
new file mode 100644
index 0000000000..1e9e9bf3c6
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-entity-1.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-entity-2.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-entity-2.png
new file mode 100644
index 0000000000..bf47159ec6
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-entity-2.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-new-entity.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-new-entity.png
new file mode 100644
index 0000000000..a680fc0bc3
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-new-entity.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-pages-1.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-pages-1.png
new file mode 100644
index 0000000000..1a50cb0cd8
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-author-pages-1.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-1.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-1.png
new file mode 100644
index 0000000000..dbb1890c14
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-1.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-2.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-2.png
new file mode 100644
index 0000000000..a77794ba1b
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-2.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-3.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-3.png
new file mode 100644
index 0000000000..e4bd459172
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-3.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-4.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-4.png
new file mode 100644
index 0000000000..df96e6a4dd
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-4.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-5.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-5.png
new file mode 100644
index 0000000000..68b31641dd
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-5.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-6.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-6.png
new file mode 100644
index 0000000000..b7814a7d7c
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-6.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-selection.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-selection.png
new file mode 100644
index 0000000000..9da2a63705
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-entity-selection.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-pages-1.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-pages-1.png
new file mode 100644
index 0000000000..a42a39b648
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-pages-1.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-with-author-create-modal.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-with-author-create-modal.png
new file mode 100644
index 0000000000..7fa41795e1
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-book-with-author-create-modal.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-bookstore-advanced-filter-section.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-bookstore-advanced-filter-section.png
new file mode 100644
index 0000000000..af9501d9eb
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-bookstore-advanced-filter-section.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-custom-code-result.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-custom-code-result.png
new file mode 100644
index 0000000000..30997d4c5b
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-custom-code-result.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-enabling-custom-code.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-enabling-custom-code.png
new file mode 100644
index 0000000000..c597bbcc41
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-enabling-custom-code.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-end-of-generation-modal.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-end-of-generation-modal.png
new file mode 100644
index 0000000000..6940a9b7a6
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-end-of-generation-modal.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/suite-repository-custom-code.png b/docs/en/tutorials/book-store-with-abp-suite/images/suite-repository-custom-code.png
new file mode 100644
index 0000000000..8c8f696a11
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/suite-repository-custom-code.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/images/test-data-seed-contributors.png b/docs/en/tutorials/book-store-with-abp-suite/images/test-data-seed-contributors.png
new file mode 100644
index 0000000000..0c2ca8db13
Binary files /dev/null and b/docs/en/tutorials/book-store-with-abp-suite/images/test-data-seed-contributors.png differ
diff --git a/docs/en/tutorials/book-store-with-abp-suite/index.md b/docs/en/tutorials/book-store-with-abp-suite/index.md
new file mode 100644
index 0000000000..b0f24ef6d3
--- /dev/null
+++ b/docs/en/tutorials/book-store-with-abp-suite/index.md
@@ -0,0 +1,40 @@
+# Web Application Development (with ABP Suite) Tutorial
+````json
+//[doc-params]
+{
+ "UI": ["MVC"],
+ "DB": ["EF"]
+}
+````
+````json
+//[doc-nav]
+{
+ "Next": {
+ "Name": "Creating the Solution",
+ "Path": "tutorials/book-store-with-abp-suite/part-01"
+ }
+}
+````
+
+> This tutorial is suitable for those who have an ABP Team or a higher [license](https://abp.io/pricing).
+
+## About This Tutorial
+
+> In this tutorial, you will use the [ABP Suite](../../suite/index.md) to generate everything you need to build the **BookStore** application, such as [*Entities*](../../framework/architecture/domain-driven-design/entities.md), [*Domain Services*](../../framework/architecture/domain-driven-design/domain-services.md), [*Application Services*](../../framework/architecture/domain-driven-design/application-services.md), *CRUD pages* and more...
+
+In this tutorial series, you will build an ABP based web application named `Acme.BookStore`. This application is used to manage a list of books and their authors. It is developed using the following technologies:
+
+* **{{DB_Value}}** as the database provider.
+* **{{UI_Value}}** as the UI Framework.
+
+This tutorial is organized as the following parts:
+
+- [Part 1: Creating the Solution](part-01.md)
+- [Part 2: Creating the Books](part-02.md)
+- [Part 3: Creating the Authors](part-03.md)
+- [Part 4: Book to Author Relation](part-04.md)
+- [Part 5: Customizing the Generated Code](part-05.md)
+
+### Download the Source Code
+
+After logging in to the ABP website, you can download the source code from [here](https://abp.io/api/download/samples/suite-bookstore-mvc-ef).
\ No newline at end of file
diff --git a/docs/en/tutorials/book-store-with-abp-suite/part-01.md b/docs/en/tutorials/book-store-with-abp-suite/part-01.md
new file mode 100644
index 0000000000..152d22fb04
--- /dev/null
+++ b/docs/en/tutorials/book-store-with-abp-suite/part-01.md
@@ -0,0 +1,47 @@
+# Web Application Development (with ABP Suite) Tutorial - Part 1: Creating the Solution
+
+````json
+//[doc-params]
+{
+ "UI": ["MVC"],
+ "DB": ["EF"]
+}
+````
+````json
+//[doc-nav]
+{
+ "Previous": {
+ "Name": "Overview",
+ "Path": "tutorials/book-store-with-abp-suite/index"
+ },
+ "Next": {
+ "Name": "Creating the Books",
+ "Path": "tutorials/book-store-with-abp-suite/part-02"
+ }
+}
+````
+
+Before starting the development, create a new solution named `Acme.BookStore` and run it by following the [getting started tutorial](../../get-started/layered-web-application.md).
+
+You can use the following configurations:
+
+* **Solution Template:** Application (Layered)
+* **Solution Name:** `Acme.BookStore`
+* **UI Framework:** {{if UI=="MVC"}} ASP.NET Core MVC / Razor Pages {{end}}
+* **UI Theme:** LeptonX
+* **Mobile Framework:** None
+* **Database Provider:** {{if DB=="EF"}} Entity Framework Core {{end}}
+* **Public Website:** No
+* **Tiered:** No
+
+You can select the other options based on your preference.
+
+> **Please complete the [Get Started](../../get-started/layered-web-application.md) guide and run the web application before going further.**
+
+The initial solution structure should be like the following in the ABP Studio's [Solution Explorer](../../studio/solution-explorer.md):
+
+
+
+## Summary
+
+We've created the initial layered monolith solution. In the next part, we will learn how to create entities, and generate CRUD pages based on the specified options (including tests, UI, customizable code support etc.) with [ABP Suite](../../suite/index.md).
diff --git a/docs/en/tutorials/book-store-with-abp-suite/part-02.md b/docs/en/tutorials/book-store-with-abp-suite/part-02.md
new file mode 100644
index 0000000000..bc0f8fd251
--- /dev/null
+++ b/docs/en/tutorials/book-store-with-abp-suite/part-02.md
@@ -0,0 +1,123 @@
+# Web Application Development (with ABP Suite) Tutorial - Part 2: Creating the Books
+````json
+//[doc-params]
+{
+ "UI": ["MVC"],
+ "DB": ["EF"]
+}
+````
+````json
+//[doc-nav]
+{
+ "Previous": {
+ "Name": "Creating the Solution",
+ "Path": "tutorials/book-store-with-abp-suite/part-01"
+ },
+ "Next": {
+ "Name": "Creating the Authors",
+ "Path": "tutorials/book-store-with-abp-suite/part-03"
+ }
+}
+````
+
+In this part, you will create a new entity named `Book` and generate CRUD pages for the related entities with everything that you would normally implement manually (including application services, tests, CRUD pages, database relations and more...) via [ABP Suite](../../suite/index.md) with few clicks.
+
+## Opening the ABP Suite
+
+> Please **stop the application** in ABP Studio's *Solution Runner* panel, if it's currently running, because ABP Suite will make changes in the solution and it might need to build the solution in some steps and running the solution prevents to build it.
+
+After creating the solution in the previous part, now you can open the ABP Suite and start generating CRUD pages. You can select the *ABP Suite -> Open* command on the main menu to open ABP Suite:
+
+
+
+After clicking the related command, pre-integrated browser of ABP Studio should open, then you can start generating entities and all related codes with a few configurations:
+
+
+
+## Creating the Book Entity
+
+Before creating the `Book` entity, first we can create a `BookType` enum in the `Acme.BookStore.Domain.Shared` project under the **Books** folder as follows:
+
+```csharp
+namespace Acme.BookStore.Books;
+
+public enum BookType
+{
+ Undefined,
+ Adventure,
+ Biography,
+ Dystopia,
+ Fantastic,
+ Horror,
+ Science,
+ ScienceFiction,
+ Poetry
+}
+```
+
+After creating an _enum_ file in your project, you can define it as a property while creating the entity. ABP Suite asks for an enum path to read the enum file, and set the namespace, and enum name in the generated code accordingly. Then, you can create the `Book` entity with some properties.
+
+Type `Book` for the *Name* field and leave the other options as is. ABP Suite automatically calculates proper values for the rest of the inputs for you:
+
+
+
+ABP Suite sets:
+
+* Entity type as **master** (ABP Suite allows you to establish [master-child relationship](../../suite/creating-master-detail-relationship.md)),
+* Base class as **FullAuditedAggregateRoot** ([see other possible values](../../framework/architecture/domain-driven-design/entities.md)),
+* Primary key type as **Guid**,
+* Plural name, database name, namespace, page title, menu item and more...
+
+You can change the menu-item value as **book** to show a proper icon in the generated UI, and also enable **code customization**, **creating unit & integration tests**, and other options as you wish:
+
+
+
+After, specifying the entity metadata, open the *Properties* tab and create the properties shown in the following figure:
+
+
+
+While defining the properties, you define a *Type* property with the type of *enum*. ABP Suite asks for an enum path and fill the all other inputs by reading the specified enum file. Therefore, you can specify the enum path for the `Type` property like in the following figure:
+
+
+
+> After you select the enum file, ABP Suite automatically sets the namespace and enum name, and lists your enum values in the next section. You can change these values, but for now, you can leave as is.
+
+Here is the all details for the `Book` entity:
+
+* `Name` is **required**, it's a **string** property and maximum length is **128**.
+* `Type` is an **enum** and the enum file path is *\Acme.BookStore.Domain.Shared\Books\BookType.cs*.
+* `PublishDate` is a **DateTime** property and **not nullable**.
+* `Price` is a **float** property and **required**.
+
+You can leave the other configurations as default.
+
+> ABP Suite allows you to define properties with a great range of options, for example, you can specify the property type as *string*, *int*, *float*, *Guid*, *DateTime*, and even *File* (for file upload) and also you can set any options while defining your properties, such as specifying it as *required*, or *nullable*, setting *max-min length*, *default value* and more...
+
+After that, you can click the **Save and Generate** button to start the code generation process:
+
+
+
+ABP Suite will generate the necessary code for you. It generates:
+
+* `Book` entity (also `BookBase` class, which is allow you customizing the generated entity),
+* Repository implementation (`EfCoreBookRepository` class),
+* `BookManager` domain service,
+* Input & Output **DTOs** and **application service** implementations (`IBookAppService` & `BookAppService`),
+* **Unit & integration tests**,
+* A new **migration** (and also applies to the database),
+* All related **permission**, **object mapping** and **navigation menu item** configurations,
+* and all required **UI components and pages**...
+
+It will take some time to complete the process. After the process is completed, you will see a success message, you can click the *Ok* button, and build & start the application by clicking the *Run -> Build & Start* button in the *Solution Runner* panel:
+
+
+
+After the application is started, you can right-click and *Browse* on the application to open it in the ABP Studio's pre-integrated browser. You can see the Books page in the following figure with a single record:
+
+
+
+On this page, you can create a new book, update an existing book, delete a book, export all records (or the filtered records) to excel, filter the records by using the advanced filter section, bulk delete multiple records and so on.
+
+## Summary
+
+In this part, you've created a new entity named `Book` and generated the necessary code for it with [ABP Suite](../../suite/index.md) with a few clicks. ABP Suite generated the all code for you, including the **entity**, **application service**, **database relations**, **unit & integration tests**, **UI** and **defined the custom hooks for code customization**.
\ No newline at end of file
diff --git a/docs/en/tutorials/book-store-with-abp-suite/part-03.md b/docs/en/tutorials/book-store-with-abp-suite/part-03.md
new file mode 100644
index 0000000000..d61f43aae5
--- /dev/null
+++ b/docs/en/tutorials/book-store-with-abp-suite/part-03.md
@@ -0,0 +1,79 @@
+# Web Application Development (with ABP Suite) Tutorial - Part 3: Creating the Authors
+````json
+//[doc-params]
+{
+ "UI": ["MVC"],
+ "DB": ["EF"]
+}
+````
+````json
+//[doc-nav]
+{
+ "Previous": {
+ "Name": "Creating the Books",
+ "Path": "tutorials/book-store-with-abp-suite/part-02"
+ },
+ "Next": {
+ "Name": "Book to Author Relation",
+ "Path": "tutorials/book-store-with-abp-suite/part-04"
+ }
+}
+````
+
+In the previous part, you have created the `Book` entity and in this part, you will create a new entity named `Author` and generate all necesssary code via [ABP Suite](../../suite/index.md) with few clicks. After creating the `Author` entity, you will be establishing [one-to-many relationship](../../suite/generating-crud-page.md#step-by-step-creating-a-navigation-property-with-1-to-many-relationship) with *Book* and *Author* entities, in the next part.
+
+## Creating the Author Entity
+
+After generating the all necessary code for the `Book` entity, and testing the *Books* page, by building & starting the application, now you can continue with creating the `Author` entity.
+
+> Before, creating the `Author` entity, please **stop the application** in ABP Studio's *Solution Runner* panel, because ABP Suite will make changes in the solution and it might need to build the solution in some steps and running the solution prevents to build it.
+
+Click the entity selection box in the top right of the *CRUD page generation* page, and select the *-New entity-*:
+
+
+
+Then, you can type `Author` for the *Name* field and leave the other options as is (you can change the menu icon as **pen** for a proper menu icon, and/or other options, if you wish). ABP Suite automatically calculates proper values for the rest of the inputs for you:
+
+
+
+ABP Suite sets:
+
+* Entity type as **master** (ABP Suite allows you to establish [master-child relationship](../../suite/creating-master-detail-relationship.md)),
+* Base class as **FullAuditedAggregateRoot** ([see other possible values](../../framework/architecture/domain-driven-design/entities.md)),
+* Primary key type as **Guid**,
+* Plural name, database name, namespace, page title, menu item and more...
+* Also, it enables **code customization**, **UI code generation**, **unit & integration test generation** and **bulk delete** by default.
+
+After, specifying the entity metadata, open the *Properties* tab and create the properties shown in the following figure:
+
+
+
+Here the details:
+
+* `Name` is **required**, it's a **string** property. Minimum length is **2** and maximum length is **128**.
+* `BirthDate` is a **DateTime** property and **not nullable**.
+* `ShortBio` is **required**, it's a **string** property, it's a **textarea**, maximum length is **256**.
+
+You can leave the other configurations as default.
+
+> **Note:** All properties are marked as **filterable** by default, and they appear in the advanced filter section because of that. You can set any properties you want as **not filterable** and then the related property will be removed from the advanced filter section and code will be generated accordingly.
+
+You can click the **Save and Generate** button to start the code generation process:
+
+
+
+ABP Suite will generate the necessary code for you. It will take some time to complete the process. After the process is completed, you will see a success message, you can click the *Ok* button, and build & start the application by clicking the *Run -> Build & Start* button in the *Solution Runner* panel:
+
+
+
+After the application is started, you can right-click and *Browse* on the application to open it in the ABP Studio's pre-integrated browser and try to add a new author:
+
+
+
+As you can see, the *Name* field is **required**, the *Birth Date* field shows a datepicker, and the *Short Bio* field is also **required** and it's a **textarea**. You just configured how you want your properties with some options (for example, setting the short bio as **textarea** and **required**), and ABP Suite generated code according that.
+
+## Summary
+
+In this part, you've created a new entity named `Author` and generated the necessary code for it with [ABP Suite](../../suite/index.md) with a few clicks. ABP Suite generated the all code for you, including the **entity**, **application service**, **database relations**, **unit & integration tests**, **UI** and **defined the custom hooks for code customization**.
+
+In the next part, you will establish [one-to-many relation](../../suite/generating-crud-page.md) between the `Book` and `Author` entities.
\ No newline at end of file
diff --git a/docs/en/tutorials/book-store-with-abp-suite/part-04.md b/docs/en/tutorials/book-store-with-abp-suite/part-04.md
new file mode 100644
index 0000000000..aaa0eb3966
--- /dev/null
+++ b/docs/en/tutorials/book-store-with-abp-suite/part-04.md
@@ -0,0 +1,170 @@
+# Web Application Development (with ABP Suite) Tutorial - Part 4: Book to Author Relation
+````json
+//[doc-params]
+{
+ "UI": ["MVC"],
+ "DB": ["EF"]
+}
+````
+````json
+//[doc-nav]
+{
+ "Previous": {
+ "Name": "Creating the Author",
+ "Path": "tutorials/book-store-with-abp-suite/part-03"
+ },
+ "Next": {
+ "Name": "Customizing the Generated Code",
+ "Path": "tutorials/book-store-with-abp-suite/part-05"
+ }
+}
+````
+
+In the previous parts, you have created the `Book` and `Author` entities (& generated code for all functionalities) for the book store application. However, currently there is no relation between these entities.
+
+In this part, you will establish to **one-to-many relation** between the `Book` and `Author` entities.
+
+## Establishing Relations with ABP Suite
+
+ABP Suite allows establishing both **one-to-many** and [many-to-many](../../suite/creating-many-to-many-relationship.md) relationships.
+
+In this tutorial, you will only establish **one-to-many relation** between `Book` and `Author` entities. It's pretty straightforward to establish a relationship with ABP Suite. You should just need to navigate to the *Navigations* tab, and provide the metadata for navigation property (1-n) or navigation collection (n-n) relations.
+
+## Creating Book to Author Relationship
+
+> Please **stop the application** in ABP Studio's *Solution Runner* panel, because ABP Suite will make changes in the solution and it might need to build the solution in some steps and running the solution prevents to build it.
+
+To establish **one-to-many relations** between *Book* and *Author* entities, select the `Book` entity from the entity selection box on the top-right of the *CRUD page generation* page:
+
+
+
+Then, you can open the *Navigations* tab, and click the **Add navigation property (1-n)** button. After that, a navigation property model will open, and you can fill the fields like in the following figure:
+
+
+
+Here is the details:
+
+* Selected the entity as `Author`. (ABP Suite will establish one-to-many relation between *Book* and *Author* entities with this configuration)
+* Set the property name as *AuthorId*, it will be set as foreign-key restriction in the database and all related database configurations will be made by ABP Suite.
+* Selected the display property as *Name*, this will be used in the dropdown component to set an author with a book & also it will be shown in the datatable of the *Books* page.
+* Also, made the relation **required** and set it **filterable** so books can be filterable by authors.
+
+> **Note**: You should delete all existing books in the database (if any), before the code generation. Because, a new foreign-key will be added to the _books_ table and if there is any record in the table, then a new migration can't apply to the database and you may need to update the database manually.
+
+After, specifying the metadata, you can click the *Ok* button to close the modal. Then, click the **Save and generate** button to start code generation process. ABP Suite will establish one-to-many relationship between the entities, and will generate all necessary code automatically:
+
+
+
+It will take some time to complete the process. After the process is completed, you will see a success message, you can click the *Ok* button, and build & start the application by clicking the *Run -> Build & Start* button in the *Solution Runner* panel:
+
+
+
+After the application is started, you can right-click and *Browse* on the application to open it in the ABP Studio's pre-integrated browser. You can first create an author and then create a book with the author for testing:
+
+
+
+Also, notice that, in the advanced filter section, there is an **Author** dropdown, which you can use to filter books by authors (remember you set **filterable** while defining navigation property and thanks to that, ABP Suite generated the code accordingly):
+
+
+
+## Unit & Integration Tests
+
+Since you completed the bookstore application, now we can check the generated tests, and run them to see if all of them pass or not.
+
+There are several test projects in the solution:
+
+
+
+> Test projects slightly differs based on your UI and Database selection. For example, if you select MongoDB, then the `Acme.BookStore.EntityFrameworkCore.Tests` will be `Acme.BookStore.MongoDB.Tests`.
+
+ABP Suite generated unit & integration tests, for the `Book` & `Author` entities. If you open the **Test explorer** in your IDE, you will see the following tests are generated:
+
+
+
+ABP Suite generated tests for repository implementations & application service implementations for the generated code, if you enable *Create unit & integration tests* option, while creating the entity. Since, you already did that in the previous parts, it generated the all required tests for the entities.
+
+Let's examine one of the generated test classes. Open the *BooksAppServiceTests* (under the *test/Acme.BookStore.Application.Tests/Books/BookApplicationTests.cs*) and check the `CreateAsync` method:
+
+```csharp
+ [Fact]
+ public async Task CreateAsync()
+ {
+ // Arrange
+ var input = new BookCreateDto
+ {
+ Name = "6c3d1eda8bf04852b7bd5dfdbbd93224b252478c2e474d4c8faf24fa6b182168ca830d4f80e64e4a8e363f33e151d1d34a04be4709274c7fbf2214f9bb3a16c3",
+ Type = default,
+ PublishDate = new DateTime(2006, 8, 21),
+ Price = 754882891,
+ AuthorId = Guid.Parse("602460f6-df6e-456a-89d9-8c5870dfc583")
+ };
+
+ // Act
+ var serviceResult = await _booksAppService.CreateAsync(input);
+
+ // Assert
+ var result = await _bookRepository.FindAsync(c => c.Id == serviceResult.Id);
+
+ result.ShouldNotBe(null);
+ result.Name.ShouldBe("6c3d1eda8bf04852b7bd5dfdbbd93224b252478c2e474d4c8faf24fa6b182168ca830d4f80e64e4a8e363f33e151d1d34a04be4709274c7fbf2214f9bb3a16c3");
+ result.Type.ShouldBe(default);
+ result.PublishDate.ShouldBe(new DateTime(2006, 8, 21));
+ result.Price.ShouldBe(754882891);
+ }
+```
+
+ABP Suite;
+
+* Create the `BookCreateDto` input DTO object, and fill its values with dummy data to simulate creating a book,
+* Then, it calls the `IBooksAppService.CreateAsync` method to create a book,
+* And finally, asserts the returned result to see if it's as expected or not.
+
+Notice, also the *AuthorId* is set in the `BookCreateDto` object. At that point, you might ask yourself that I haven't created the author with that ID before, should not it throw exception?
+
+No, it will not throw an exception, because ABP Suite also generates simple dummy data for the entities just for the tests! You can see the test data seed contributors under the *Acme.BookStore.Domain.Tests* project:
+
+
+
+Here is the content of the `AuthorsDataSeedContributor.SeedAsync` method:
+
+```csharp
+ public async Task SeedAsync(DataSeedContext context)
+ {
+ if (IsSeeded)
+ {
+ return;
+ }
+
+ await _authorRepository.InsertAsync(new Author
+ (
+ id: Guid.Parse("602460f6-df6e-456a-89d9-8c5870dfc583"),
+ name: "d7bbb3bff0d54ad799477298c4572e9c05fd1175ab21416da17d0001e2b697cd7fef99fdb4414f26a05789667a97442bd65865510ba34c3599e874ccf08b45e4",
+ birthDate: new DateTime(2010, 2, 11),
+ shortBio: "3c2ff43c18e34d7b9ad3f1b9c444cbb000f90808d3774cb6b7702b957f472d74048597f93df744f6a6fdf507be428e016edec982f1174e09b124982cbc40156290ce6bc9fd7b49b4972741956cc847891cb55ad0942f4534b90aa0561d3e0c200340b613c7ad40c38b4b2f2c39298169a853473faed34341a130b31e1eb57e92"
+ ));
+
+ await _authorRepository.InsertAsync(new Author
+ (
+ id: Guid.Parse("6ea5a6b2-919e-4334-9728-13f4872e5e0e"),
+ name: "fd332fb58f184716962b08fbaa92f1c3e0963d843ba34c82bb5409517f60da3727c43b05e8d4490f996c5d19265962e53a69ed5e3e144509aad1441e37ce5081",
+ birthDate: new DateTime(2010, 6, 10),
+ shortBio: "b7808946c46c42e3935c4d8203d82973cfb98c5d81644f1da4ce1e643767849e23e0eb12a92f48be8f7eec0c07aefa043721fdd3fea542cfa644d2b7d428dc8842647180ef8a47139e097f6674c4f0d86c46765c406042a2a858865cb112ecd78d9ef6f5843e444994641f924a38a2d24ee4e212d41444888d3c0861af0cf9dd"
+ ));
+
+ await _unitOfWorkManager!.Current!.SaveChangesAsync();
+
+ IsSeeded = true;
+ }
+```
+
+Since ABP Suite generated the test data seed contributors for each entity, you have initial data while testing your services. Also, as you would notice, the id in this example (*602460f6-df6e-456a-89d9-8c5870dfc583*) is same as the *authorId* field in the `BooksAppServiceTests.CreateAsync` method.
+
+Let's execute all tests, and see the results:
+
+
+
+## Summary
+
+So far, you have created the all functionality for the bookstore application without needing to write any single line of code. ABP Suite generated the entities, application services, UI components, unit & integration tests and more...
+
+In the next part, you will write some code and modify the ABP Suite's generated code by writing the code in the specified hookpoints. Thanks to [ABP Suite's Customized Code Support](../../suite/customizing-the-generated-code.md), in the next generation, our custom code will not be overridden and will be preserved.
\ No newline at end of file
diff --git a/docs/en/tutorials/book-store-with-abp-suite/part-05.md b/docs/en/tutorials/book-store-with-abp-suite/part-05.md
new file mode 100644
index 0000000000..842399a80f
--- /dev/null
+++ b/docs/en/tutorials/book-store-with-abp-suite/part-05.md
@@ -0,0 +1,123 @@
+# Web Application Development (with ABP Suite) Tutorial - Part 5: Customizing the Generated Code
+````json
+//[doc-params]
+{
+ "UI": ["MVC"],
+ "DB": ["EF"]
+}
+````
+````json
+//[doc-nav]
+{
+ "Previous": {
+ "Name": "Book to Author Relation",
+ "Path": "tutorials/book-store-with-abp-suite/part-04"
+ }
+}
+````
+
+So far, you have created the all functionality for the bookstore application without needing to write any single line of code. In this part, let's write some code and check one of the great features of the ABP Suite, which is [Customizable Code Support](../../suite/customizing-the-generated-code.md).
+
+## Customizable Code Support
+
+ABP Suite allows you to customize the generated code blocks and preserve your custom code changes in the next CRUD Page Generation. It specifies hook points to allow adding custom code blocks. Then, the code written by you to these hook points will be respected and will not be overridden in the next CRUD Page Generation.
+
+To enable custom code support, you should check the *Customizable code* option in the **CRUD Page Generation** page (it's selected by default), and you enabled it for both entities:
+
+
+
+## Custom Code Hookpoints
+
+When you enable the *custom code support*, ABP Suite adds some hookpoints that you can write your own custom code without worrying about, are my codes being overridden with the next CRUD page generation.
+
+On the C# side, ABP Suite adds abstract base classes for entities, application services, interfaces, domain services and so on... (and partial classes for interfaces)
+
+You can write your custom code in those classes (with the `*.Extended.cs` extension) and next time when you need to re-generate the entity, your custom code will not be overridden (only the base abstract classes will be re-generated and your changes on Suite will be respected):
+
+
+
+> For example, you can create a new repository method like in the example above, and in the next CRUD page generation, you will ABP Suite won't override your custom code.
+
+On the UI side, ABP Suite provides convenient comment placeholders within pages for MVC, Blazor, and Angular UIs. These comment sections serve as hook points where you can add your custom code.
+
+For example, if you open the *Books/Index.cshtml* file in your IDE, you will see those placeholders like following:
+
+```xml
+
+
+@section styles
+{
+@*//*@
+@*//*@
+}
+
+
+```
+
+You can write your custom codes between the _****_ placeholders and you can also extend these placeholders by customizing the [ABP Suite templates](../../suite/editing-templates.md).
+
+> For more information, please refer to [Customizing the Generated Code documentation](../../suite/customizing-the-generated-code.md)
+
+## Implementing Custom Code
+
+Let's see the custom code support in action. We can demonstrate this feature with an easy example.
+
+Assume that we want to show the author's name with his abbreviated name. For example, for the author *John Ronald Reuel Tolkien*, we want to show the name *John Ronald Reuel Tolkien (a.k.a J.R.R.T)*. Achieving that is pretty straightforward.
+
+We just need to open the *src/Acme.BookStore.Application/Books/BooksAppService.Extended.cs* file and override the base `GetListAsync` method, which is called on the books page:
+
+```csharp
+using System;
+using System.Linq;
+using System.Collections.Generic;
+using System.Threading.Tasks;
+using Volo.Abp.Application.Dtos;
+using Volo.Abp.Domain.Repositories;
+using Volo.Abp.Caching;
+
+namespace Acme.BookStore.Books
+{
+ public class BooksAppService : BooksAppServiceBase, IBooksAppService
+ {
+ //
+ public BooksAppService(IBookRepository bookRepository, BookManager bookManager, IDistributedCache downloadTokenCache, IRepository authorRepository)
+ : base(bookRepository, bookManager, downloadTokenCache, authorRepository)
+ {
+ }
+ //
+
+ //Write your custom code...
+ public override async Task> GetListAsync(GetBooksInput input)
+ {
+ var result = await base.GetListAsync(input);
+
+ foreach (var book in result.Items)
+ {
+ var akaName = book.Author.Name.Split(" ").Select(q => q[0]).JoinAsString(".");
+ book.Author.Name += $" (a.k.a {akaName})";
+ }
+
+ return result;
+ }
+ }
+}
+```
+
+* Here, we have overridden the `GetListAsync` method and changed its result according to our need.
+* Notice, this class is derieved from the `BooksAppServiceBase` class and implements the `IBooksAppService`.
+* Thus, ABP Suite only modifies the `BooksAppServiceBase` class and implements the necessary services in each generation, but does not generate the files with the `*.Extended` postfixes and let you implement your own custom code.
+* You can create new methods, override an existing method and change its behaviour, add custom hookpoints to extend customization capabilities and more...
+
+Now, we can open ABP Suite and try to regenerate the book entity and see if our custom code is gone or not:
+
+
+
+After the regeneration has been completed, we can check the **BooksAppService.Extended.cs** file and we should see our custom code is there without any modification. ABP Suite didn't override our custom code, and finally, we can run the application to see the final result:
+
+
+
+ABP Suite's custom code support is not limited to the backend side. You can also override the UI. You can refer to the [Customizing the Generated Code document](../../suite/customizing-the-generated-code.md) for further info.
+
+## Summary
+
+In this tutorial, you created the bookstore application without needing to write a single line of code with ABP Suite. Then, in this last part, you have written custom code in the specified hookpoints and it did not override by ABP Suite.
\ No newline at end of file
diff --git a/docs/en/tutorials/book-store/part-01.md b/docs/en/tutorials/book-store/part-01.md
index 8f1d5df346..91700a1a25 100644
--- a/docs/en/tutorials/book-store/part-01.md
+++ b/docs/en/tutorials/book-store/part-01.md
@@ -2,7 +2,7 @@
````json
//[doc-params]
{
- "UI": ["MVC","Blazor","BlazorServer","NG"],
+ "UI": ["MVC","Blazor","BlazorServer", "BlazorWebApp","NG"],
"DB": ["EF","Mongo"]
}
````
@@ -34,9 +34,9 @@ For such cases, run the `abp install-libs` command on the root directory of your
abp install-libs
```
-> We suggest you install [Yarn](https://classic.yarnpkg.com/) to prevent possible package inconsistencies, if you haven't installed it yet.
+> We suggest you install [Yarn v1.22+ (not v2)](https://classic.yarnpkg.com/en/docs/install) to prevent possible package inconsistencies, if you haven't installed it yet.
-{{if UI=="Blazor" || UI=="BlazorServer"}}
+{{if UI=="Blazor" || UI=="BlazorWebApp"}}
### Bundling and Minification
@@ -440,7 +440,7 @@ ABP can [**automagically**](../../framework/api-development/auto-controllers.md)
### Swagger UI
-The startup template is configured to run the [Swagger UI](https://swagger.io/tools/swagger-ui/) using the [Swashbuckle.AspNetCore](https://github.com/domaindrivendev/Swashbuckle.AspNetCore) library. Run the application ({{if UI=="MVC"}}`Acme.BookStore.Web`{{else if UI=="BlazorServer"}}`Acme.BookStore.Blazor`{{else}}`Acme.BookStore.HttpApi.Host`{{end}}) by pressing `CTRL+F5` and navigate to `https://localhost:/swagger/` on your browser. Replace `` with your own port number.
+The startup template is configured to run the [Swagger UI](https://swagger.io/tools/swagger-ui/) using the [Swashbuckle.AspNetCore](https://github.com/domaindrivendev/Swashbuckle.AspNetCore) library. Run the application ({{if UI=="MVC"}}`Acme.BookStore.Web`{{else if UI=="BlazorServer" || UI=="BlazorWebApp"}}`Acme.BookStore.Blazor`{{else}}`Acme.BookStore.HttpApi.Host`{{end}}) by pressing `CTRL+F5` and navigate to `https://localhost:/swagger/` on your browser. Replace `` with your own port number.
You will see some built-in service endpoints as well as the `Book` service and its REST-style endpoints:
@@ -480,4 +480,4 @@ If you try to execute the `[GET] /api/app/book` API to get a list of books, the
}
````
-That's pretty cool since we haven't written a single line of code to create the API controller, but now we have a fully working REST API!
\ No newline at end of file
+That's pretty cool since we haven't written a single line of code to create the API controller, but now we have a fully working REST API!
diff --git a/docs/en/tutorials/book-store/part-02.md b/docs/en/tutorials/book-store/part-02.md
index df5a5ac598..3ff19437cd 100644
--- a/docs/en/tutorials/book-store/part-02.md
+++ b/docs/en/tutorials/book-store/part-02.md
@@ -2,7 +2,7 @@
````json
//[doc-params]
{
- "UI": ["MVC","Blazor","BlazorServer","NG"],
+ "UI": ["MVC","Blazor","BlazorServer", "BlazorWebApp", "NG"],
"DB": ["EF","Mongo"]
}
````
@@ -514,13 +514,13 @@ Now you can see the final result on your browser:

-{{else if UI == "Blazor" || UI == "BlazorServer"}}
+{{else if UI == "Blazor" || UI == "BlazorServer" || UI == "BlazorWebApp"}}
## Create a Books Page
-It's time to create something visible and usable! Right click on the `Pages` folder under the {{ if UI == "Blazor"}}`Acme.BookStore.Blazor.Client`{{ else }}`Acme.BookStore.Blazor`{{ end }} project and add a new **razor component**, named `Books.razor`:
+It's time to create something visible and usable! Right click on the `Pages` folder under the {{ if UI == "BlazorServer" }}`Acme.BookStore.Blazor`{{ else }}`Acme.BookStore.Blazor.Client`{{ end }} project and add a new **razor component**, named `Books.razor`:
-{{ if UI == "Blazor"}}
+{{ if UI == "Blazor" || UI == "BlazorWebApp" }}

{{ else }}

@@ -540,7 +540,7 @@ Replace the contents of this component as shown below:
### Add the Books Page to the Main Menu
-Open the `BookStoreMenuContributor` class in the {{ if UI == "Blazor"}}`Acme.BookStore.Blazor.Client`{{ else }}`Acme.BookStore.Blazor`{{ end }} project add the following code to the end of the `ConfigureMainMenuAsync` method:
+Open the `BookStoreMenuContributor` class in the {{ if UI == "BlazorServer"}}`Acme.BookStore.Blazor`{{ else }}`Acme.BookStore.Blazor.Client`{{ end }} project add the following code to the end of the `ConfigureMainMenuAsync` method:
````csharp
context.Menu.AddItem(
@@ -577,8 +577,6 @@ Open the `Books.razor` and replace the content as the following:
@using Volo.Abp.Application.Dtos
@using Acme.BookStore.Books
@using Acme.BookStore.Localization
-@using Microsoft.Extensions.Localization
-@inject IStringLocalizer L
@inherits AbpCrudPageBase
@@ -625,13 +623,21 @@ Open the `Books.razor` and replace the content as the following:
+
+@code
+{
+ public Books() // Constructor
+ {
+ LocalizationResource = typeof(BookStoreResource);
+ }
+}
````
> If you see some syntax errors, you can ignore them if your application is properly built and running. Visual Studio still has some bugs with Blazor.
* Inherited from `AbpCrudPageBase` which implements all the CRUD details for us.
* `Entities`, `TotalCount`, `PageSize`, `OnDataGridReadAsync` are defined in the base class.
-* Injected `IStringLocalizer` (as `L` object) and used for localization.
+* `LocalizationResource` is set to the `BookStoreResource` to localize the texts.
While the code above is pretty easy to understand, you can check the Blazorise [Card](https://blazorise.com/docs/components/card/) and [DataGrid](https://blazorise.com/docs/extensions/datagrid/) documents to understand them better.
diff --git a/docs/en/tutorials/book-store/part-03.md b/docs/en/tutorials/book-store/part-03.md
index a770d62f29..fc4e851b94 100644
--- a/docs/en/tutorials/book-store/part-03.md
+++ b/docs/en/tutorials/book-store/part-03.md
@@ -2,7 +2,7 @@
````json
//[doc-params]
{
- "UI": ["MVC","Blazor","BlazorServer","NG"],
+ "UI": ["MVC","Blazor","BlazorServer","BlazorWebApp","NG"],
"DB": ["EF","Mongo"]
}
````
@@ -1101,7 +1101,7 @@ Clicking the "Delete" action calls the `delete` method which then shows a confir
{{end}}
-{{if UI == "Blazor" || UI == "BlazorServer"}}
+{{if UI == "Blazor" || UI == "BlazorServer" || UI == "BlazorWebApp"}}
## Creating a New Book
@@ -1292,13 +1292,13 @@ We can now define a modal to edit the book. Add the following code to the end of
The base `AbpCrudPageBase` uses the [object to object mapping](../../framework/infrastructure/object-to-object-mapping.md) system to convert an incoming `BookDto` object to a `CreateUpdateBookDto` object. So, we need to define the mapping.
-Open the `BookStoreBlazorAutoMapperProfile` inside the `Acme.BookStore.Blazor.Client` project and change the content as the following:
+Open the `BookStoreBlazorAutoMapperProfile` inside the {{ if UI == "BlazorServer" }}`Acme.BookStore.Blazor`{{ else }}`Acme.BookStore.Blazor.Client`{{ end }} project and change the content as the following:
````csharp
using Acme.BookStore.Books;
using AutoMapper;
-namespace Acme.BookStore.Blazor.Client;
+{{ if UI == "BlazorServer" }}namespace Acme.BookStore.Blazor;{{ else }}namespace Acme.BookStore.Blazor.Client;{{ end }}
public class BookStoreBlazorAutoMapperProfile : Profile
{
@@ -1351,7 +1351,6 @@ Here's the complete code to create the book management CRUD page, that has been
@using Acme.BookStore.Localization
@using Microsoft.Extensions.Localization
@using Volo.Abp.AspNetCore.Components.Web
-@inject IStringLocalizer L
@inject AbpBlazorMessageLocalizerHelper LH
@inherits AbpCrudPageBase
@@ -1396,7 +1395,7 @@ Here's the complete code to create the book management CRUD page, that has been
Field="@nameof(BookDto.Type)"
Caption="@L["Type"]">
- @L[$"Enum:BookType.{context.Type}"]
+ @L[$"Enum:BookType.{context.Type:D}"]
+
+@code
+{
+ public Books() // Constructor
+ {
+ LocalizationResource = typeof(BookStoreResource);
+ }
+}
````
{{end}}
diff --git a/docs/en/tutorials/book-store/part-04.md b/docs/en/tutorials/book-store/part-04.md
index b529748efb..4ab950b0e1 100644
--- a/docs/en/tutorials/book-store/part-04.md
+++ b/docs/en/tutorials/book-store/part-04.md
@@ -2,7 +2,7 @@
````json
//[doc-params]
{
- "UI": ["MVC","Blazor","BlazorServer","NG"],
+ "UI": ["MVC","Blazor","BlazorServer","BlazorWebApp","NG"],
"DB": ["EF","Mongo"]
}
````
diff --git a/docs/en/tutorials/book-store/part-05.md b/docs/en/tutorials/book-store/part-05.md
index 947eec65f0..d4815c5a96 100644
--- a/docs/en/tutorials/book-store/part-05.md
+++ b/docs/en/tutorials/book-store/part-05.md
@@ -2,7 +2,7 @@
````json
//[doc-params]
{
- "UI": ["MVC","Blazor","BlazorServer","NG"],
+ "UI": ["MVC","Blazor","BlazorServer","BlazorWebApp","NG"],
"DB": ["EF","Mongo"]
}
````
@@ -389,11 +389,11 @@ Open the `/src/app/book/book.component.html` file and replace the edit and delet
* Added `*abpPermission="'BookStore.Books.Edit'"` that hides the edit action if the current user has no editing permission.
* Added `*abpPermission="'BookStore.Books.Delete'"` that hides the delete action if the current user has no delete permission.
-{{else if UI == "Blazor"}}
+{{else if UI == "Blazor" || UI == "BlazorServer" || UI == "BlazorWebApp"}}
### Authorize the Razor Component
-Open the `/Pages/Books.razor` file in the `Acme.BookStore.Blazor.Client` project and add an `Authorize` attribute just after the `@page` directive and the following namespace imports (`@using` lines), as shown below:
+Open the `/Pages/Books.razor` file in the {{ if UI == "BlazorServer" }}`Acme.BookStore.Blazor`{{ else }}`Acme.BookStore.Blazor.Client`{{ end }} project and add an `Authorize` attribute just after the `@page` directive and the following namespace imports (`@using` lines), as shown below:
````html
@page "/books"
@@ -420,6 +420,8 @@ Add the following code block to the end of the `Books.razor` file:
{
public Books() // Constructor
{
+ LocalizationResource = typeof(BookStoreResource);
+
CreatePolicyName = BookStorePermissions.Books.Create;
UpdatePolicyName = BookStorePermissions.Books.Edit;
DeletePolicyName = BookStorePermissions.Books.Delete;
@@ -479,7 +481,7 @@ You can run and test the permissions. Remove a book related permission from the
Even we have secured all the layers of the book management page, it is still visible on the main menu of the application. We should hide the menu item if the current user has no permission.
-Open the `BookStoreMenuContributor` class in the `Acme.BookStore.Blazor.Client` project, find the code block below:
+Open the `BookStoreMenuContributor` class in the {{ if UI == "BlazorServer" }}`Acme.BookStore.Blazor`{{ else }}`Acme.BookStore.Blazor.Client`{{ end }} project, find the code block below:
````csharp
context.Menu.AddItem(
@@ -509,14 +511,11 @@ var bookStoreMenu = new ApplicationMenuItem(
context.Menu.AddItem(bookStoreMenu);
//CHECK the PERMISSION
-if (await context.IsGrantedAsync(BookStorePermissions.Books.Default))
-{
- bookStoreMenu.AddItem(new ApplicationMenuItem(
- "BooksStore.Books",
- l["Menu:Books"],
- url: "/books"
- ));
-}
+bookStoreMenu.AddItem(new ApplicationMenuItem(
+ "BooksStore.Books",
+ l["Menu:Books"],
+ url: "/books"
+).RequirePermissions(BookStorePermissions.Books.Default));
````
You also need to add `async` keyword to the `ConfigureMenuAsync` method and re-arrange the return value. The final `ConfigureMainMenuAsync` method should be the following:
@@ -545,14 +544,11 @@ private async Task ConfigureMainMenuAsync(MenuConfigurationContext context)
context.Menu.AddItem(bookStoreMenu);
//CHECK the PERMISSION
- if (await context.IsGrantedAsync(BookStorePermissions.Books.Default))
- {
- bookStoreMenu.AddItem(new ApplicationMenuItem(
- "BooksStore.Books",
- l["Menu:Books"],
- url: "/books"
- ));
- }
+ bookStoreMenu.AddItem(new ApplicationMenuItem(
+ "BooksStore.Books",
+ l["Menu:Books"],
+ url: "/books"
+ ).RequirePermissions(BookStorePermissions.Books.Default));
}
````
diff --git a/docs/en/tutorials/book-store/part-06.md b/docs/en/tutorials/book-store/part-06.md
index a19248ec64..05e1d04460 100644
--- a/docs/en/tutorials/book-store/part-06.md
+++ b/docs/en/tutorials/book-store/part-06.md
@@ -2,7 +2,7 @@
````json
//[doc-params]
{
- "UI": ["MVC","Blazor","BlazorServer","NG"],
+ "UI": ["MVC","Blazor","BlazorServer", "BlazorWebApp", "NG"],
"DB": ["EF","Mongo"]
}
````
@@ -41,7 +41,6 @@ Create an `Authors` folder (namespace) in the `Acme.BookStore.Domain` project an
````csharp
using System;
-using JetBrains.Annotations;
using Volo.Abp;
using Volo.Abp.Domain.Entities.Auditing;
@@ -115,7 +114,6 @@ Created this class inside the `Acme.BookStore.Domain.Shared` project since we wi
````csharp
using System;
using System.Threading.Tasks;
-using JetBrains.Annotations;
using Volo.Abp;
using Volo.Abp.Domain.Services;
diff --git a/docs/en/tutorials/book-store/part-07.md b/docs/en/tutorials/book-store/part-07.md
index 70d73da97b..b8aa17ace8 100644
--- a/docs/en/tutorials/book-store/part-07.md
+++ b/docs/en/tutorials/book-store/part-07.md
@@ -2,7 +2,7 @@
````json
//[doc-params]
{
- "UI": ["MVC","Blazor","BlazorServer","NG"],
+ "UI": ["MVC","Blazor","BlazorServer","BlazorWebApp","NG"],
"DB": ["EF","Mongo"]
}
````
diff --git a/docs/en/tutorials/book-store/part-08.md b/docs/en/tutorials/book-store/part-08.md
index 61412ba770..679b6d84ca 100644
--- a/docs/en/tutorials/book-store/part-08.md
+++ b/docs/en/tutorials/book-store/part-08.md
@@ -2,7 +2,7 @@
````json
//[doc-params]
{
- "UI": ["MVC","Blazor","BlazorServer","NG"],
+ "UI": ["MVC","Blazor","BlazorServer","BlazorWebApp","NG"],
"DB": ["EF","Mongo"]
}
````
diff --git a/docs/en/tutorials/book-store/part-09.md b/docs/en/tutorials/book-store/part-09.md
index d1c446bc3f..d481dbe92e 100644
--- a/docs/en/tutorials/book-store/part-09.md
+++ b/docs/en/tutorials/book-store/part-09.md
@@ -2,7 +2,7 @@
````json
//[doc-params]
{
- "UI": ["MVC","Blazor","BlazorServer","NG"],
+ "UI": ["MVC","Blazor","BlazorServer","BlazorWebApp","NG"],
"DB": ["EF","Mongo"]
}
````
@@ -848,13 +848,13 @@ That's all! This is a fully working CRUD page, you can create, edit and delete a
{{end}}
-{{if UI == "Blazor" || UI == "BlazorServer"}}
+{{if UI == "Blazor" || UI == "BlazorServer" || UI == "BlazorWebApp"}}
## The Author Management Page
### Authors Razor Component
-Create a new Razor Component Page, `/Pages/Authors.razor`, in the `Acme.BookStore.Blazor.Client` project with the following content:
+Create a new Razor Component Page, `/Pages/Authors.razor`, in the {{ if UI == "BlazorServer" }}`Acme.BookStore.Blazor`{{ else }}`Acme.BookStore.Blazor.Client`{{ end }} project with the following content:
````xml
@page "/authors"
@@ -1055,7 +1055,7 @@ using Blazorise.DataGrid;
using Microsoft.AspNetCore.Authorization;
using Volo.Abp.Application.Dtos;
-namespace Acme.BookStore.Blazor.Client.Pages;
+{{ if UI == "BlazorServer" }}namespace Acme.BookStore.Blazor.Pages;{{ else }}namespace Acme.BookStore.Blazor.Client.Pages;{{ end }}
public partial class Authors
{
@@ -1201,7 +1201,7 @@ This class typically defines the properties and methods used by the `Authors.raz
`Authors` class uses the `IObjectMapper` in the `OpenEditAuthorModal` method. So, we need to define this mapping.
-Open the `BookStoreBlazorAutoMapperProfile.cs` in the `Acme.BookStore.Blazor.Client` project and add the following mapping code in the constructor:
+Open the `BookStoreBlazorAutoMapperProfile.cs` in the {{ if UI == "BlazorServer" }}`Acme.BookStore.Blazor`{{ else }}`Acme.BookStore.Blazor.Client`{{ end }} project and add the following mapping code in the constructor:
````csharp
CreateMap();
@@ -1211,17 +1211,14 @@ You will need to declare a `using Acme.BookStore.Authors;` statement to the begi
### Add to the Main Menu
-Open the `BookStoreMenuContributor.cs` in the `Acme.BookStore.Blazor.Client` project and add the following code to the end of the `ConfigureMainMenuAsync` method:
+Open the `BookStoreMenuContributor.cs` in the {{ if UI == "BlazorServer" }}`Acme.BookStore.Blazor`{{ else }}`Acme.BookStore.Blazor.Client`{{ end }} project and add the following code to the end of the `ConfigureMainMenuAsync` method:
````csharp
-if (await context.IsGrantedAsync(BookStorePermissions.Authors.Default))
-{
- context.Menu.AddItem(new ApplicationMenuItem(
+context.Menu.AddItem(new ApplicationMenuItem(
"BooksStore.Authors",
l["Menu:Authors"],
url: "/authors"
- ));
-}
+ ).RequirePermissions(BookStorePermissions.Books.Default));
````
### Localizations
diff --git a/docs/en/tutorials/book-store/part-10.md b/docs/en/tutorials/book-store/part-10.md
index 5a6acb752b..c4383aa5c9 100644
--- a/docs/en/tutorials/book-store/part-10.md
+++ b/docs/en/tutorials/book-store/part-10.md
@@ -2,7 +2,7 @@
````json
//[doc-params]
{
- "UI": ["MVC","Blazor","BlazorServer","NG"],
+ "UI": ["MVC","Blazor","BlazorServer","BlazorWebApp","NG"],
"DB": ["EF","Mongo"]
}
````
@@ -1071,11 +1071,11 @@ That's all. Just run the application and try to create or edit an author.
{{end}}
-{{if UI == "Blazor" || UI == "BlazorServer"}}
+{{if UI == "Blazor" || UI == "BlazorServer" || UI == "BlazorWebApp" }}
### The Book List
-It is very easy to show the *Author Name* in the book list. Open the `/Pages/Books.razor` file in the `Acme.BookStore.Blazor.Client` project and add the following `DataGridColumn` definition just after the `Name` (book name) column:
+It is very easy to show the *Author Name* in the book list. Open the `/Pages/Books.razor` file in the {{ if UI == "BlazorServer" }}`Acme.BookStore.Blazor`{{ else }}`Acme.BookStore.Blazor.Client`{{ end }} project and add the following `DataGridColumn` definition just after the `Name` (book name) column:
````xml
This tutorial is suitable for those who have an ABP Business or a higher [license](https://abp.io/pricing).
+
+ABP is designed to be a powerful platform to build microservice solutions. It provides a [microservice solution template](../../solution-templates/microservice/index.md) to easily start a sophisticated microservice solution. All of the [pre-built application modules](../../modules/index.md) are microservice compatible and the core framework fully [supports and simplifies](../../framework/architecture/microservices/index.md) distributed application development.
+
+In this tutorial, you will learn how to start a new microservice solution, create services and communicate between them. You will also learn to use these services from a web application through an API gateway and automatically generate CRUD pages using the [ABP Suite](../../suite/index.md) tool.
+
+## Tutorial Outline
+
+This tutorial is organized as the following parts:
+
+* [Part 01: Creating the initial solution](part-01.md)
+* [Part 02: Creating the initial Catalog microservice](part-02.md)
+* [Part 03: Building the Catalog microservice](part-03.md)
+* [Part 04: Creating the initial Ordering service](part-04.md)
+* [Part 05: Building the Ordering service](part-05.md)
+* [Part 06: Integrating the services: HTTP API Calls](part-06.md)
+* [Part 07: Integrating the services: Using Distributed Events](part-07.md)
+
+## Download the Source Code
+
+After logging in to the ABP website, you can download the source code from [here](https://abp.io/api/download/samples/cloud-crm-mvc-ef).
+
+## See Also
+
+* [Microservice solution template](../../solution-templates/microservice/index.md)
\ No newline at end of file
diff --git a/docs/en/tutorials/microservice/part-01.md b/docs/en/tutorials/microservice/part-01.md
new file mode 100644
index 0000000000..d843e744ca
--- /dev/null
+++ b/docs/en/tutorials/microservice/part-01.md
@@ -0,0 +1,40 @@
+# Microservice Tutorial Part 01: Creating the Initial Solution
+
+````json
+//[doc-nav]
+{
+ "Next": {
+ "Name": "Creating the initial Catalog service",
+ "Path": "tutorials/microservice/part-02"
+ }
+}
+````
+
+Follow the *[Get Started](../../get-started/microservice.md)* guide to create a new layered web application with the following configurations:
+
+* **Solution name**: `CloudCrm`
+* **Database Provider**: Entity Framework Core
+* **Database Management System**: SQL Server
+* **UI Framework**: MVC / Razor Pages
+* **Mobile framework**: None
+* **Public website**: Selected
+
+You can select the other options based on your preference.
+
+> **Please complete the *[Get Started](../../get-started/layered-web-application.md)* guide and run the web application before going further.** You can skip the sections after the *Running the Solution* section, if you don't prefer to complete all.
+
+The initial solution structure should be like the following in ABP Studio's *[Solution Explorer](../../studio/solution-explorer.md)*:
+
+
+
+> ABP Studio will perform a few additional steps after creating your solution. **Please wait until all the background tasks are completed** before going further.
+
+Initially you see three folders (`apps`, `gateways` and `services`) and ~10 ABP Studio modules (depends on your preferences while creating the solution) under the `CloudCrm` ABP Studio solution. Some of these modules represent microservices, some of them represent web applications and some others represent API gateways in our system.
+
+> An **ABP Studio module** is typically a .NET solution and an **ABP Studio solution** is an umbrella concept for multiple .NET Solutions (see the *[Concepts](../../studio/concepts.md)* document for more).
+
+You can see the *[Microservice Solution Template](../../solution-templates/microservice/index.md)* document later if you want to understand the initial solution structure with all its details. However, it is not needed to follow this tutorial.
+
+## Summary
+
+In this part, you've created the initial microservice solution, which already contains a few infrastructure services. We will create our first business service in the [next part](part-02.md).
\ No newline at end of file
diff --git a/docs/en/tutorials/microservice/part-02.md b/docs/en/tutorials/microservice/part-02.md
new file mode 100644
index 0000000000..014daf9b11
--- /dev/null
+++ b/docs/en/tutorials/microservice/part-02.md
@@ -0,0 +1,123 @@
+# Microservice Tutorial Part 02: Creating the initial Catalog service
+
+````json
+//[doc-nav]
+{
+ "Previous": {
+ "Name": "Creating the initial solution",
+ "Path": "tutorials/microservice/part-01"
+ },
+ "Next": {
+ "Name": "Building the Catalog service",
+ "Path": "tutorials/microservice/part-03"
+ }
+}
+````
+
+In this tutorial, you will create a new Catalog service and integrate it to the solution.
+
+## Creating the Catalog Service
+
+Right-click the `services` folder in the *Solution Explorer* panel, select the *Add* -> *New Module* -> *Microservice* command:
+
+
+
+This command opens a new dialog to define the properties of the new microservice. You can use the following values to create a new microservice named `CatalogService`:
+
+
+
+When you click the *Next* button, you are redirected to the database provider selection step.
+
+### Selecting the Database Type
+
+Here, you can select the database provider to be used by the new microservice:
+
+
+
+Select *Entity Framework Core* option and proceed the *Next* step.
+
+### Integrating to the Solution
+
+In this step, we can select the options for integrating the new microservice to the rest of the solution components:
+
+
+
+ABP Studio intelligently selects the right values for you, but you should still check them carefully since they directly affect what we will do in the next parts of this tutorial.
+
+**Ensure the options are configured the same as in the preceding figure**, and click the *Next* button.
+
+### Additional Options
+
+
+
+In this step, you can select additional options for the new microservice. You can leave them as default and click the *Create* button.
+
+That's all, ABP Studio creates the new microservice and arranges all the integration and configuration for you.
+
+## Exploring the New Catalog Microservice
+
+In this section, we will investigate the new microservice in overall.
+
+### Understanding the Packages of The Service
+
+The new microservice is added under the `services` folder in the `CloudCrm` ABP Studio solution:
+
+
+
+The new microservice has its own separate .NET solution that includes three packages (.NET projects):
+
+* `CloudCrm.CatalogService` is the main project that you will implement your service. It typically contains your [entities](../../framework/architecture/domain-driven-design/entities.md), [repositories](../../framework/architecture/domain-driven-design/repositories.md), [application services](../../framework/architecture/domain-driven-design/application-services.md), API controllers, etc.
+* `CloudCrm.CatalogService.Contracts` project can be shared with the other services and applications. It typically contains interfaces of your [application services](../../framework/architecture/domain-driven-design/application-services.md), [data transfer objects](../../framework/architecture/domain-driven-design/data-transfer-objects.md), and some other types you may want to share with the clients of this microservice.
+* `CloudCrm.CatalogService.Tests` is for building your unit and integration tests for this microservice.
+
+### Opening the Service in an IDE
+
+You can open the new microservice in your favorite IDE for development. As a shortcut, you can right-click it in ABP Studio, select the *Open with* -> *Visual Studio* command for example:
+
+
+
+Here is the `CloudCrm.CatalogService` .NET solution in Visual Studio:
+
+
+
+### Running the New Service
+
+You can run the solution using ABP Studio's *Solution Runner*. It will also run the new Catalog service as a part of the solution.
+
+> Before running the solution, **ensure that all the applications are built**. If you are not sure, right-click the root item (`CloudCrm`) in the *Solution Explorer* panel and select the *Build* -> *Graph Build* command.
+
+Click the *Play* button near to the solution root:
+
+
+
+### Browsing the Catalog Service
+
+Once all of the applications have started, right-click the Catalog service and select the *Browse* command:
+
+
+
+It will open the built-in browser and you will see the Swagger UI for the Catalog service:
+
+
+
+You can test the APIs on the Swagger UI to see if the new microservice is properly working.
+
+### Opening the Catalog Database
+
+The new Catalog microservice has its own database. That database is created automatically by the microservice application, when you run the microservice. Also, [Entity Framework's database migrations](https://learn.microsoft.com/en-us/ef/core/managing-schemas/migrations/) are automatically applied by the microservice when it runs. So, you don't care about the database schema changes every time you deploy the microservice.
+
+Assuming you've selected SQL Server as your DBMS, you can open the SQL Server Management Studio to see its databases:
+
+
+
+Use `localhost,1434` as the *Server name*, select the *SQL Server Authentication* as the *Authentication* type, use `sa` as the *Login* name and `myPassw@rd` as the *Password* value. You can find these values in the `appsettings.json` file in the `CloudCrm.CatalogService` project of the .NET solution of the Catalog microservice.
+
+Once you click the *Connect* button, you can see all the databases and explore their data:
+
+
+
+The Catalog service's database has only three initial table. The first one is for Entity Framework Core's migration system, and the others are for ABP's [distributed event bus](../../solution-templates/microservice/distributed-events.md) to properly apply transactional events using the outbox and inbox patterns. You don't need to care about these tables since they are created and managed by Entity Framework Core and ABP.
+
+## Summary
+
+In this part of the Microservice Development Tutorial, we added a new Catalog microservice to the solution, explored its code structure and database, and browse its APIs using the Swagger UI. In the next part, we will create functionality in that new microservice.
\ No newline at end of file
diff --git a/docs/en/tutorials/microservice/part-03.md b/docs/en/tutorials/microservice/part-03.md
new file mode 100644
index 0000000000..31491ef004
--- /dev/null
+++ b/docs/en/tutorials/microservice/part-03.md
@@ -0,0 +1,126 @@
+# Microservice Tutorial Part 03: Building the Catalog service
+
+````json
+//[doc-nav]
+{
+ "Previous": {
+ "Name": "Creating the initial Catalog service",
+ "Path": "tutorials/microservice/part-02"
+ },
+ "Next": {
+ "Name": "Creating the initial Ordering service",
+ "Path": "tutorials/microservice/part-04"
+ }
+}
+````
+
+In the previous part, we've created a new microservice named Catalog. In this part, we will build functionality to create and manage products in our system.
+
+In this part, we will use [ABP Suite](../../suite/index.md) to automatically create all the necessary code for us. So, you will see how to use ABP Suite in a microservice solution. We will do everything manually while we will create the Ordering microservice in next parts, so you will learn the details better. We suggest to use ABP Suite wherever it is possible, because it saves a lot of time. You can then investigate the changes done by ABP Suite to understand what it produced.
+
+## Opening the ABP Suite
+
+First of all, **stop all the applications** in ABP Studio's *Solution Runner* panel, because ABP Suite will make changes in the solution and it will also needs to build the solution in some steps. Running the solution prevents to build it.
+
+Now, select the *ABP Suite* -> *Open* command on the main menu to open ABP Suite:
+
+
+
+It will ask to you which module you want to use:
+
+
+
+The `CloudCrm` microservice solution contains more than one .NET solution. Typically, each ABP Studio module represents a separate .NET solution (see the [concepts](../../studio/concepts.md) document). ABP Suite works on a single .NET solution to generate code, so we should select a module here.
+
+Select the `CloudCrm.CatalogService` module and click the *OK* button. It will open ABP Suite as shown below:
+
+
+
+## Generating a Products Page
+
+In the next section, we will use ABP Suite to create a fully functional CRUD page with ABP Suite. The UI part will be in the main web application (`CloudCrm.Web`) and the application service and other parts will be generated in the Catalog microservice.
+
+### Configuring the Product Entity Information
+
+Type `Product` for the *Name* field and leave the other options as is. ABP Suite will automatically calculate proper values for you:
+
+
+
+### Configuring Properties of the Product Entity
+
+Open the *Properties* tab and create the properties shown in the following figure:
+
+
+
+Here the details:
+
+* `Name` is required, minimum length is `2` and maximum length is `120`.
+* `Description` is not required, it is a *Text area*, not *Filterable*, not *Shown on the list page*.
+* `StockCount` has a *Default value* `0`, minimum value `0` and maximum value `999999`.
+* `ReleaseDate` is *Nullable*.
+
+You can leave the other configurations as default.
+
+### Generating the Code
+
+
+
+That's all. You can click the *Save and generate* button to start the code generation process.
+
+
+
+ABP Suite will generate the necessary code for you. It will take some time to complete the process. After the process is completed, you will see a success message, click the *OK* button.
+
+
+
+We can now build and start the `CloudCrm.CatalogService` application by clicking the *Run* -> *Build & Start* button in the *Solution Runner* panel.
+
+
+
+After the application is started, you can right-click and [Browse](../../studio/running-applications.md#monitoring) on the `CloudCrm.CatalogService` application to open it in the ABP Studio's pre-integrated browser. You can see the *Products* controller in the Swagger UI.
+
+### Generating the UI Proxy
+
+Now, we need to generate the [Static API Proxy](../../framework/api-development/static-csharp-clients.md) for the *Web* project. Right-click the *CloudCrm.Web* [package](../../studio/concepts.md#package) and select the *ABP CLI* -> *Generate Proxy* -> *C#* command:
+
+
+
+It will open the *Generate C# Proxies* window. Select the `CloudCrm.CatalogService` application, and it will automatically populate the *URL* field. Select the *catalog* module, set the service type to *application*, and check the *Without contracts* checkbox, as the `CloudCrm.Web` project already depends on the `CloudCrm.CatalogService.Contracts` package:
+
+
+
+> To be able to select the *Application*, you must *Build & Start* the related application beforehand. You can start the application using [Solution Runner](../../studio/running-applications.md) as explained in the previous parts.
+
+Lastly, we need to configure the use of a static HTTP client for the `CatalogService` in the `CloudCrm.Web` project. Open the `CloudCrmWebModule.cs` file in the `Web` project and add the following line to the `ConfigureServices` method:
+
+```csharp
+//...
+using CloudCrm.CatalogService;
+
+public override void ConfigureServices(ServiceConfigurationContext context)
+{
+ // Code omitted for brevity
+ context.Services.AddStaticHttpClientProxies(
+ typeof(CloudCrmCatalogServiceContractsModule).Assembly);
+}
+```
+
+### Running the Application
+
+Now, stop any application running in the *Solution Runner* panel, and then run the applications by clicking the *Run* -> *Build & Start All* button on the root item in the *Solution Runner* panel:
+
+
+
+After the application is started, you can right-click and [Browse](../../studio/running-applications.md#monitoring) on the `CloudCrm.Web` application to open it in the ABP Studio's pre-integrated browser:
+
+
+
+> If you can't see the *Products* menu item, you need to grant the `CatalogService` *Product* permission to the *admin* role. You can do this by navigating to *Identity Management* -> *Roles* and editing the *admin* role. Alternatively, you can restart the *CloudCrm.AdministrationService* application to automatically seed all permissions for the *admin* role.
+
+You can open the Sql Server Management Studio to see the created tables and data:
+
+
+
+## Summary
+
+In this part, we've created a new entity named *Product* and generated the necessary code for it. We've also generated the UI proxy for the `CatalogService` application and configured the static HTTP client for it in the `Web` project. We've run the application and tested the *Products* page.
\ No newline at end of file
diff --git a/docs/en/tutorials/microservice/part-04.md b/docs/en/tutorials/microservice/part-04.md
new file mode 100644
index 0000000000..50386ab8e1
--- /dev/null
+++ b/docs/en/tutorials/microservice/part-04.md
@@ -0,0 +1,97 @@
+# Microservice Tutorial Part 04: Creating the initial Ordering service
+
+````json
+//[doc-nav]
+{
+ "Previous": {
+ "Name": "Building the Catalog service",
+ "Path": "tutorials/microservice/part-03"
+ },
+ "Next": {
+ "Name": "Building the Ordering service",
+ "Path": "tutorials/microservice/part-05"
+ }
+}
+````
+
+In the previous part, we implemented the Catalog microservice functionality using ABP Suite. In this part, we will create the Ordering microservice, and the following part will cover implementing its functionality manually.
+
+## Creating the Ordering Microservice
+
+Right-click the `services` folder in the *Solution Explorer* panel, select the *Add* -> *New Module* -> *Microservice* command:
+
+
+
+This command opens a new dialog to define the properties of the new microservice. You can use the following values to create a new microservice named `OrderingService`:
+
+
+
+When you click the *Next* button, you are redirected to the database provider selection step.
+
+### Selecting the Database Type
+
+Here, you can select the database provider to be used by the new microservice:
+
+
+
+Select *Entity Framework Core* option and proceed the *Next* step.
+
+### Integrating to the Solution
+
+In this step, we can select the options for integrating the new microservice to the rest of the solution components:
+
+
+
+ABP Studio intelligently selects the right values for you, but you should still check them carefully since they directly affect what we will do in the next parts of this tutorial.
+
+**Ensure the options are configured the same as in the preceding figure**, and click the *Next* button.
+
+### Additional Options
+
+
+
+In this step, you can select additional options for the new microservice. You can leave them as default and click the *Create* button.
+
+That's all, ABP Studio creates the new microservice and arranges all the integration and configuration for you.
+
+## Exploring the New Ordering Microservice
+
+In this section, we will investigate the new microservice in overall.
+
+### Understanding the Solution Structure
+
+Just like the Catalog microservice, the Ordering microservice is a .NET solution that contains multiple projects. You can see the solution structure in the *Solution Explorer* panel:
+
+
+
+* `CloudCrm.OrderingService` is the main project that you will implement your service. It typically contains your [entities](../../framework/architecture/domain-driven-design/entities.md), [repositories](../../framework/architecture/domain-driven-design/repositories.md), [application services](../../framework/architecture/domain-driven-design/application-services.md), API controllers, etc.
+* `CloudCrm.OrderingService.Contracts` project can be shared with the other services and applications. It typically contains interfaces of your [application services](../../framework/architecture/domain-driven-design/application-services.md), [data transfer objects](../../framework/architecture/domain-driven-design/data-transfer-objects.md), and some other types you may want to share with the clients of this microservice.
+* `CloudCrm.OrderingService.Tests` is for building your unit and integration tests for this microservice.
+
+### Running the New Service
+
+You can run the solution using ABP Studio's *Solution Runner*. It will also run the new Ordering service as a part of the solution.
+
+> Before running the solution, **ensure that all the applications are built**. If you are not sure, right-click the root item (`CloudCrm`) in the *Solution Explorer* panel and select the *Build* -> *Graph Build* command.
+
+Click the *Play* button near to the solution root:
+
+
+
+### Browsing the Ordering Service
+
+After the application is started, you can right-click and [Browse](../../studio/running-applications.md#monitoring) on the `CloudCrm.OrderingService` application to open it in the ABP Studio's pre-integrated browser. You can see the *Orders* controller in the Swagger UI:
+
+
+
+### Opening the Ordering Database
+
+You can use the SQL Server Management Studio or any other tool to connect to the Ordering service's database. Use `localhost,1434` as the *Server name*, select the *SQL Server Authentication* as the *Authentication* type, use `sa` as the *Login* name and `myPassw@rd` as the *Password* value. You can find these values in the `appsettings.json` file in the `CloudCrm.OrderingService` project of the .NET solution of the Ordering microservice:
+
+
+
+Similarly the Ordering service's database has only three initial table. The first one is for Entity Framework Core's migration system, and the others are for ABP's [distributed event bus](../../solution-templates/microservice/distributed-events.md) to properly apply transactional events using the outbox and inbox patterns. You don't need to care about these tables since they are created and managed by Entity Framework Core and ABP.
+
+## Summary
+
+In this part, we've created the initial Ordering microservice. We will implement its functionality in the next part.
\ No newline at end of file
diff --git a/docs/en/tutorials/microservice/part-05.md b/docs/en/tutorials/microservice/part-05.md
new file mode 100644
index 0000000000..249cdc8890
--- /dev/null
+++ b/docs/en/tutorials/microservice/part-05.md
@@ -0,0 +1,430 @@
+# Microservice Tutorial Part 05: Building the Ordering service
+
+````json
+//[doc-nav]
+{
+ "Previous": {
+ "Name": "Creating the initial Ordering service",
+ "Path": "tutorials/microservice/part-04"
+ },
+ "Next": {
+ "Name": "Integrating the services: HTTP API Calls",
+ "Path": "tutorials/microservice/part-06"
+ }
+}
+````
+
+In the previous part, we created the Ordering microservice. In this part, we will implement the functionality of the Ordering microservice manually. We will not use ABP Suite to generate the code. Instead, we will create the necessary code step by step to understand the details better.
+
+## Creating the Order Entity
+
+We will start by creating the `Order` entity, which will represent an order in our system. We'll add this entity to the `CloudCrm.OrderingService` project. Create a new folder named `Entities` and add a class named `Order` inside it:
+
+```csharp
+using CloudCrm.OrderingService.Enums;
+using Volo.Abp.Domain.Entities.Auditing;
+
+namespace CloudCrm.OrderingService.Entities;
+
+public class Order : CreationAuditedAggregateRoot
+{
+ public Guid ProductId { get; set; }
+ public string CustomerName { get; set; }
+ public OrderState State { get; set; }
+}
+```
+
+To keep this example simple, we allow users to include only a single product within an order. The `Order` entity inherits from the [CreationAuditedAggregateRoot<>](../../framework/architecture/domain-driven-design/entities.md) class. This class, provided by the ABP Framework, includes common properties like `Id`, `CreationTime`, `CreatorId`, etc.
+
+### Adding the OrderState Enum
+
+We also need to define the `OrderState` enum. In the `CloudCrm.OrderingService.Contracts` project, create a folder named `Enums` and add an `OrderState` enum inside it:
+
+```csharp
+namespace CloudCrm.OrderingService.Enums;
+
+public enum OrderState : byte
+{
+ Placed = 0,
+ Delivered = 1,
+ Canceled = 2
+}
+```
+
+The final solution structure should look like this:
+
+
+
+## Configuring the Database Mapping
+
+First, we need to add the `Order` entity to the `OrderingServiceDbContext` class. Open the `OrderingServiceDbContext` class in the `CloudCrm.OrderingService` project, located in the `Data` folder, and add the following code to the `DbSet` properties:
+
+```csharp
+using CloudCrm.OrderingService.Entities;
+using Microsoft.EntityFrameworkCore;
+using Volo.Abp.Data;
+using Volo.Abp.EntityFrameworkCore;
+using Volo.Abp.EntityFrameworkCore.DistributedEvents;
+using Volo.Abp.EntityFrameworkCore.Modeling;
+
+namespace CloudCrm.OrderingService.Data;
+
+[ConnectionStringName(DatabaseName)]
+public class OrderingServiceDbContext :
+ AbpDbContext,
+ IHasEventInbox,
+ IHasEventOutbox
+{
+ public const string DbTablePrefix = "";
+ public const string DbSchema = null;
+
+ public const string DatabaseName = "OrderingService";
+
+ public DbSet IncomingEvents { get; set; }
+ public DbSet OutgoingEvents { get; set; }
+ public DbSet Orders { get; set; } // NEW: ADD DBSET PROPERTY
+
+ // Code omitted for brevity
+}
+```
+
+Next, we need to configure the database mapping for the `Order` entity. We'll use the [EF Core Fluent API](https://docs.microsoft.com/en-us/ef/core/modeling/relational/tables) for this configuration. In the `OrderingServiceDbContext` class add the following code to the `OnModelCreating` method:
+
+```csharp
+protected override void OnModelCreating(ModelBuilder builder)
+{
+ base.OnModelCreating(builder);
+
+ builder.ConfigureEventInbox();
+ builder.ConfigureEventOutbox();
+
+ builder.Entity(b =>
+ {
+ // Configure table name
+ b.ToTable("Orders");
+
+ // Always call this method to set base entity properties
+ b.ConfigureByConvention();
+
+ // Properties of the entity
+ b.Property(q => q.CustomerName).IsRequired().HasMaxLength(120);
+ });
+}
+```
+
+In this code snippet, we configure the `Order` entity to use the `Orders` table in the database. We also specify that the `CustomerName` property is required and has a maximum length of 120 characters.
+
+### Add a Database Migration
+
+Now, we can add a new database migration. You can use Entity Framework Core's `Add-Migration` (or `dotnet ef migrations add`) terminal command, but in this tutorial, we will use ABP Studio's shortcut UI.
+
+Ensure that the solution has built. You can right-click the `CloudCrm.OrderingService` (under the `services` folder) on ABP Studio *Solution Explorer* and select the *Dotnet CLI* -> *Graph Build* command.
+
+Right-click the `CloudCrm.OrderingService` package and select the *EF Core CLI* -> *Add Migration* command:
+
+
+
+The *Add Migration* command opens a new dialog to get a migration name:
+
+
+
+Once you click the *OK* button, a new database migration class is added to the `Migrations` folder of the `CloudCrm.OrderingService` project:
+
+
+
+The changes will be applied to the database during the next application startup. For more details, refer to the [database migrations on service startup](../../solution-templates/microservice/database-configurations.md#database-migrations-on-service-startup) section.
+
+## Creating the Application Service
+
+Now, we will create the application service to manage the `Order` entity.
+
+### Defining the Application Service Contract
+
+First, we need to define the application service contract under the `CloudCrm.OrderingService.Contracts` project. Return to your IDE, open the `CloudCrm.OrderingService` module's .NET solution and create an `IOrderAppService` interface under the `Services` folder for the `CloudCrm.OrderingService.Contracts` project:
+
+```csharp
+using System.Collections.Generic;
+using System.Threading.Tasks;
+using Volo.Abp.Application.Services;
+
+namespace CloudCrm.OrderingService.Services;
+
+public interface IOrderAppService : IApplicationService
+{
+ Task> GetListAsync();
+ Task CreateAsync(OrderCreationDto input);
+}
+```
+
+### Defining the Data Transfer Objects
+
+Next, we need to define the data transfer objects (DTOs) for the `Order` entity. The `GetListAsync` and `CreateAsync` methods will use these DTOs to communicate with the client applications.
+
+Create a `OrderCreationDto` class under the `Services` folder in the `CloudCrm.OrderingService.Contracts` project:
+
+```csharp
+using System;
+using System.ComponentModel.DataAnnotations;
+
+namespace CloudCrm.OrderingService.Services;
+
+public class OrderCreationDto
+{
+ [Required]
+ [StringLength(150)]
+ public string CustomerName { get; set; }
+
+ [Required]
+ public Guid ProductId { get; set; }
+}
+```
+
+Create a `OrderDto` class under the `Services` folder in the `CloudCrm.OrderingService.Contracts` project:
+
+```csharp
+using System;
+using CloudCrm.OrderingService.Enums;
+
+namespace CloudCrm.OrderingService.Services;
+
+public class OrderDto
+{
+ public Guid Id { get; set; }
+ public string CustomerName { get; set; }
+ public Guid ProductId { get; set; }
+ public OrderState State { get; set; }
+}
+```
+
+The final solution structure should look like this:
+
+
+
+## Implementing the Application Service
+
+Now, we will implement the `IOrderAppService` interface in the `OrderAppService` class. Create an `OrderAppService` class under the `Services` folder in the `CloudCrm.OrderingService` project:
+
+```csharp
+using System;
+using System.Collections.Generic;
+using CloudCrm.OrderingService.Entities;
+using CloudCrm.OrderingService.Enums;
+using CloudCrm.OrderingService.Localization;
+using Volo.Abp.Application.Services;
+using Volo.Abp.Domain.Repositories;
+
+namespace CloudCrm.OrderingService.Services;
+
+public class OrderAppService : ApplicationService, IOrderAppService
+{
+ private readonly IRepository _orderRepository;
+
+ public OrderAppService(IRepository orderRepository)
+ {
+ LocalizationResource = typeof(OrderingServiceResource);
+ _orderRepository = orderRepository;
+ }
+
+ public async Task> GetListAsync()
+ {
+ var orders = await _orderRepository.GetListAsync();
+ return ObjectMapper.Map, List>(orders);
+ }
+
+ public async Task CreateAsync(OrderCreationDto input)
+ {
+ var order = new Order
+ {
+ CustomerName = input.CustomerName,
+ ProductId = input.ProductId,
+ State = OrderState.Placed
+ };
+
+ await _orderRepository.InsertAsync(order);
+ }
+}
+```
+
+In this code snippet, we inject the `IRepository` into the `OrderAppService` class. We use this repository to interact with the `Order` entity. The `GetListAsync` method retrieves a list of orders from the database and maps them to the `OrderDto` class. The `CreateAsync` method creates a new order entity and inserts it into the database.
+
+Afterward, we need to configure the *AutoMapper* object to map the `Order` entity to the `OrderDto` class. Open the `OrderingServiceApplicationAutoMapperProfile` class in the `CloudCrm.OrderingService` project, located in the `ObjectMapping` folder, and add the following code:
+
+```csharp
+using AutoMapper;
+using CloudCrm.OrderingService.Entities;
+using CloudCrm.OrderingService.Services;
+
+namespace CloudCrm.OrderingService.ObjectMapping;
+
+public class OrderingServiceApplicationAutoMapperProfile : Profile
+{
+ public OrderingServiceApplicationAutoMapperProfile()
+ {
+ CreateMap();
+ }
+}
+```
+
+## Testing the Application Service
+
+Now, we can test the `OrderAppService` class using the Swagger UI. Open the Solution Runner and right-click to `CloudCrm.OrderingService` project and select the *Run* -> *Build & Start* command. After the application starts, you can open the Swagger UI by clicking to the [Browse](../../studio/running-applications.md#monitoring) command:
+
+
+
+Expand the `api/ordering/order` API and click the *Try it out* button. Then, create a few orders by filling in the request body and clicking the *Execute* button:
+
+
+
+If you check the database, you should see the entities created in the `Orders` table:
+
+
+
+> Since we're using the [Auto API Controller](../../framework/api-development/auto-controllers.md) we don't need to create a controller for the `OrderAppService`. The ABP Framework automatically creates an API controller for the `OrderAppService` class. You can find the configuration in the `CloudCrmOrderingServiceModule` class, in the `ConfigureAutoControllers` method of the `CloudCrm.OrderingService` project.
+
+## Creating the User Interface
+
+Now, we will create the user interface for the Ordering module. We will use the `CloudCrm.Web` project to create the user interface. Open the `CloudCrm.Web` .NET solution in your favorite IDE.
+
+### Creating the Orders Page
+
+Create a new `Orders` folder under the `Pages` folder in the `CloudCrm.Web` project. Then, create an `Index.cshtml` Razor Page inside that new folder and edit the `Index.cshtml.cs` file as follows:
+
+```csharp
+using CloudCrm.OrderingService.Services;
+using Microsoft.AspNetCore.Mvc.RazorPages;
+using Volo.Abp.AspNetCore.Mvc.UI.RazorPages;
+
+namespace CloudCrm.Web.Pages.Orders;
+
+public class Index : AbpPageModel
+{
+ public List Orders { get; set; }
+
+ private readonly IOrderAppService _orderAppService;
+
+ public Index(IOrderAppService orderAppService)
+ {
+ _orderAppService = orderAppService;
+ }
+
+ public async Task OnGetAsync()
+ {
+ Orders = await _orderAppService.GetListAsync();
+ }
+}
+```
+
+Here, we inject the `IOrderAppService` into the `Index` Razor Page. We use this service to retrieve the list of orders from the database and assign them to the `Orders` property. Open the `Index.cshtml` file and add the following code:
+
+```html
+@page
+@model CloudCrm.Web.Pages.Orders.Index
+
+
Orders
+
+
+
+
+ @foreach (var order in Model.Orders)
+ {
+
+ Customer: @order.CustomerName
+ Product: @order.ProductId
+ State: @order.State
+
+ }
+
+
+
+```
+
+This page shows a list of orders on the UI. We haven't created a UI to create new orders, and we will not do it to keep this tutorial simple. If you want to learn how to create advanced UIs with ABP, please follow the [Book Store tutorial](../../tutorials/book-store/index.md).
+
+### Generating the UI Proxy
+
+To select the *Application* during proxy generation, ensure that the `CloudCrm.OrderingService` is *Started* beforehand. You can start the application using [Solution Runner](../../studio/running-applications.md).
+
+Now, we need to generate the [Static API Proxy](../../framework/api-development/static-csharp-clients.md) for the *Web* project. Right-click the *CloudCrm.Web* [package](../../studio/concepts.md#package) and select the *ABP CLI* -> *Generate Proxy* -> *C#* command:
+
+
+
+It will open the *Generate C# Proxies* window. Select the `CloudCrm.OrderingService` application, and it will automatically populate the *URL* field. Choose the *ordering* module and service type is *application* lastly check the *Without contracts* checkbox, since we already have a dependency on the `CloudCrm.OrderingService.Contracts` package in the `CloudCrm.Web` project:
+
+
+
+Lastly, we need to configure the use of a static HTTP client for the `OrderingService` in the `CloudCrm.Web` project. Open the `CloudCrmWebModule.cs` file in the `Web` project and add the following line to the `ConfigureServices` method:
+
+```csharp
+//...
+using CloudCrm.OrderingService;
+
+public override void ConfigureServices(ServiceConfigurationContext context)
+{
+ // Code omitted for brevity
+ context.Services.AddStaticHttpClientProxies(
+ typeof(CloudCrmOrderingServiceContractsModule).Assembly);
+}
+```
+
+### Adding the Menu Item
+
+> ABP provides a modular navigation [menu system](../../framework/ui/mvc-razor-pages/navigation-menu.md) that allows you to define the menu items in a modular way.
+
+Finally, we need to add a menu item to the sidebar to navigate to the `Orders` page. Open the `CloudCrmMenus` file in the `Navigation` folder of the `CloudCrm.Web` project and edit with the following code:
+
+```csharp
+namespace CloudCrm.Web.Navigation;
+
+public class CloudCrmMenus
+{
+ private const string Prefix = "CloudCrm";
+
+ public const string Home = Prefix + ".Home";
+
+ public const string HostDashboard = Prefix + ".HostDashboard";
+
+ public const string TenantDashboard = Prefix + ".TenantDashboard";
+
+ public const string Products = Prefix + ".Products";
+
+ public const string Orders = Prefix + ".Orders"; // NEW: ADD MENU ITEM
+}
+```
+
+Then, open the `CloudCrmMenuContributor` class in the `CloudCrm.Web` project, located in the `Navigation` folder, and add the following code to `ConfigureMainMenuAsync` method:
+
+```csharp
+private static async Task ConfigureMainMenuAsync(MenuConfigurationContext context)
+{
+ // Code omitted for brevity
+
+ context.Menu.AddItem(
+ new ApplicationMenuItem(
+ CloudCrmMenus.Orders, // Unique menu id
+ "Orders", // Menu display text
+ "~/Orders", // URL
+ "fa-solid fa-basket-shopping" // Icon CSS class
+ )
+ );
+}
+```
+
+## Building and Running the Application
+
+Now, we can build and run the application to see the changes. Please stop the applications if they are running. Then open the *Solution Runner* panel, right-click the `CloudCrm` root item, and select the *Run* -> *Build & Start* command:
+
+
+
+After the applications are started, you can *Browse* and navigate to the `Orders` page to see the list of orders:
+
+
+
+Great! We have successfully implemented the Ordering module. However, there is a problem:
+
+- We see Product's GUID ID instead of its name. This is because the *Ordering* microservice has no integration with the *Catalog* microservice and doesn't have access to Product microservice's database to perform a JOIN query.
+
+We will solve this problem in the next part by implementing an integration service between the *Ordering* and *Catalog* microservices.
+
+## Summary
+
+In this part, we implemented the Ordering module manually. We created the `Order` entity, the `OrderState` enum, the `OrderAppService` application service, and the user interface for the `Orders` page. We also added a menu item to the sidebar to navigate to the `Orders` page.
\ No newline at end of file
diff --git a/docs/en/tutorials/microservice/part-06.md b/docs/en/tutorials/microservice/part-06.md
new file mode 100644
index 0000000000..f171767260
--- /dev/null
+++ b/docs/en/tutorials/microservice/part-06.md
@@ -0,0 +1,311 @@
+# Microservice Tutorial Part 06: Integrating the services: HTTP API Calls
+
+````json
+//[doc-nav]
+{
+ "Previous": {
+ "Name": "Building the Ordering service",
+ "Path": "tutorials/microservice/part-05"
+ },
+ "Next": {
+ "Name": "Integrating the services: Using Distributed Events",
+ "Path": "tutorials/microservice/part-07"
+ }
+}
+````
+
+In the previous part, we implemented the functionality of the Ordering microservice. However, when listing orders, we need to display the product name instead of the product ID. To achieve this, we must call the Catalog service to retrieve the product name for each order item.
+
+In this section, we will integrate the Ordering service with the Catalog service using HTTP API calls.
+
+## The Need for the Integration Services
+
+In a microservices architecture, each service is responsible for its own data and business logic. However, services often need to communicate with each other to fulfill their responsibilities. This communication can be synchronous or asynchronous, depending on the requirements.
+
+
+
+In our case, the Ordering service needs to display the product name instead of the product ID. To achieve this, we need to call the Catalog service to retrieve the product details based on the product ID. This is a typical example of a synchronous communication pattern between microservices. As a solution to that problem, we will use an [integration service](../../framework/api-development/integration-services.md) that will handle the communication with the Catalog service. Integration service concept in ABP is designed for request/response style inter-module (in modular applications) and inter-microservice (in distributed systems) communication.
+
+## Creating a Products Integration Service
+
+First, we need to create a service that will handle the communication with the Catalog service. This service will be responsible for fetching the product details based on the product ID.
+
+### Defining the `IProductIntegrationService` Interface
+
+Open the `CloudCrm.CatalogService` .NET solution in your IDE. Locate the `CloudCrm.CatalogService.Contracts` project, and create a new folder named `IntegrationServices`. Inside this folder, add a new interface named `IProductIntegrationService` with the following code:
+
+```csharp
+using System;
+using System.Collections.Generic;
+using System.Threading.Tasks;
+using CloudCrm.CatalogService.Products;
+using Volo.Abp;
+using Volo.Abp.Application.Services;
+
+namespace CloudCrm.CatalogService.IntegrationServices;
+
+[IntegrationService]
+public interface IProductIntegrationService : IApplicationService
+{
+ Task> GetProductsByIdsAsync(List ids);
+}
+```
+
+`IProductIntegrationService` is very similar to a typical [application service](../../framework/architecture/domain-driven-design/application-services.md). The only difference is that it is marked with the `[IntegrationService]` attribute. This attribute is used to identify the service as an integration service, which allows ABP to handle the communication between services. ABP behave differently for them (for example, ABP doesn't expose [integration services](../../framework/api-development/integration-services.md) as HTTP APIs by default if you've configured the [Auto API Controllers](../../framework/api-development/auto-controllers.md) feature)
+
+`IProductIntegrationService` contains a single method named `GetProductsByIdsAsync`. This method takes a list of product IDs and returns a list of `ProductDto` objects. This is exactly what we need in the Ordering service.
+
+> **Design Tip**
+>
+> You may think if we can use the existing application services (like `IProductAppService`) from other services instead of creating specific integration services. Technically you can use, ABP has no restriction. However, from good design and best practice points, we don't suggest it. Because, application services are designed to be consumed specifically by the presentation layer. They will have different authorization and validation logic, they will need different DTO input and output properties, they will have different performance, optimization and caching requirements, and so on. And most importantly, all these will change by the time based on UI requirements and these changes may break your integrations later. It is best to implement specific integration APIs that is designed and optimized for that purpose.
+>
+> We've reused the `ProductDto` object created for `IProductAppService`, which can be reasonable from a maintenance point of view. But if you think your integration service results will be different from the application service results in the future, it can be good to separate them from the first day so you don't need to introduce breaking changes later.
+
+### Implementing the `ProductIntegrationService` Class
+
+Now, let's implement the `IProductIntegrationService` interface. Create a new folder named `IntegrationServices` in the `CloudCrm.CatalogService` project. Inside this folder, add a new class named `ProductIntegrationService` with the following code:
+
+```csharp
+using System;
+using System.Collections.Generic;
+using System.Threading.Tasks;
+using CloudCrm.CatalogService.Localization;
+using CloudCrm.CatalogService.Products;
+using Volo.Abp;
+using Volo.Abp.Application.Services;
+using Volo.Abp.Domain.Repositories;
+
+namespace CloudCrm.CatalogService.IntegrationServices;
+
+[IntegrationService]
+public class ProductIntegrationService : ApplicationService, IProductIntegrationService
+{
+ private readonly IRepository _productRepository;
+
+ public ProductIntegrationService(IRepository productRepository)
+ {
+ LocalizationResource = typeof(CatalogServiceResource);
+ _productRepository = productRepository;
+ }
+
+ public async Task> GetProductsByIdsAsync(List ids)
+ {
+ var products = await _productRepository.GetListAsync(
+ product => ids.Contains(product.Id)
+ );
+
+ return ObjectMapper.Map, List>(products);
+ }
+}
+```
+
+`ProductIntegrationService` is a typical application service class that implements the `IProductIntegrationService` interface. It has a constructor that takes an `IRepository` object. This repository is used to fetch the product details from the database.
+
+> Here, we directly used `List` classes, but instead, you could wrap inputs and outputs into [DTOs](../../framework/architecture/domain-driven-design/data-transfer-objects.md). In that way, it can be possible to add new properties to these DTOs without changing the signature of your integration service method (and without introducing breaking changes for your client applications).
+
+### Exposing the Integration Service as an API
+
+Integration services are not exposed as HTTP APIs by default. However, you can expose them as HTTP APIs if you need to. To do this, you should configure the `AbpAspNetCoreMvcOptions` in the `ConfigureServices` method of the `CloudCrmCatalogServiceModule`. Open the `CloudCrm.CatalogService` project and locate the `CloudCrmCatalogServiceModule` class. Add the following code to the `ConfigureServices` method:
+
+```csharp
+public override void ConfigureServices(ServiceConfigurationContext context)
+{
+ // Other configurations...
+
+ Configure(options =>
+ {
+ options.ExposeIntegrationServices = true;
+ });
+}
+```
+
+This code configures the `AbpAspNetCoreMvcOptions` to expose integration services as HTTP APIs. This is useful when you need to call the integration service from a different service using HTTP. You can learn more about this in the [Integration Services](../../framework/api-development/integration-services.md#exposing-integration-services) document.
+
+## Consuming the Products Integration Service
+
+Now that we have created the `IProductIntegrationService` interface and the `ProductIntegrationService` class, we can consume this service from the Ordering service.
+
+### Adding a Reference to the `CloudCrm.CatalogService.Contracts` Package
+
+First, we need to add a reference to the `CloudCrm.CatalogService.Contracts` package in the Ordering service. Open the ABP Studio, and stop the application(s) if it is running. Then, open the *Solution Explorer* and right-click on the `CloudCrm.OrderingService` package. Select *Add* -> *Package Reference* command:
+
+
+
+In the *Add Package Reference* window, select the `CloudCrm.CatalogService.Contracts` package from the *This solution* tab. Click the *OK* button to add the reference:
+
+
+
+ABP Studio adds the package reference and arranges the [module](../../framework/architecture/modularity/basics.md) dependency.
+
+> Instead of directly adding such a package reference, it can be best to import the module first (right-click the `CloudCrm.OrderingService`, select the _Import Module_ command and import the `CloudCrm.CatalogService` module), then install the package reference. In that way, it would be easy to see and keep track of inter-module dependencies.
+
+### Using the Products Integration Service
+
+Now, we can use the `IProductIntegrationService` interface to fetch the product details in the `OrderAppService` class.
+
+Open the `OrderAppService` class (the `OrderAppService.cs` file under the `Services` folder of the `CloudCrm.OrderingService` project of the `CloudCrm.OrderingService` .NET solution) and change its content as like the following code block:
+
+```csharp
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Threading.Tasks;
+using CloudCrm.CatalogService.IntegrationServices;
+using CloudCrm.OrderingService.Entities;
+using CloudCrm.OrderingService.Enums;
+using CloudCrm.OrderingService.Localization;
+using Volo.Abp.Application.Services;
+using Volo.Abp.Domain.Repositories;
+
+namespace CloudCrm.OrderingService.Services;
+
+public class OrderAppService : ApplicationService, IOrderAppService
+{
+ private readonly IRepository _orderRepository;
+ private readonly IProductIntegrationService _productIntegrationService;
+
+ public OrderAppService(
+ IRepository orderRepository,
+ IProductIntegrationService productIntegrationService)
+ {
+ LocalizationResource = typeof(OrderingServiceResource);
+
+ _orderRepository = orderRepository;
+ _productIntegrationService = productIntegrationService;
+ }
+
+ public async Task> GetListAsync()
+ {
+ var orders = await _orderRepository.GetListAsync();
+
+ // Prepare a list of products we need
+ var productIds = orders.Select(o => o.ProductId).Distinct().ToList();
+ var products = (await _productIntegrationService
+ .GetProductsByIdsAsync(productIds))
+ .ToDictionary(p => p.Id, p => p.Name);
+
+ var orderDtos = ObjectMapper.Map, List>(orders);
+
+ orderDtos.ForEach(orderDto =>
+ {
+ orderDto.ProductName = products[orderDto.ProductId];
+ });
+
+ return orderDtos;
+ }
+
+ public async Task CreateAsync(OrderCreationDto input)
+ {
+ var order = new Order
+ {
+ CustomerName = input.CustomerName,
+ ProductId = input.ProductId,
+ State = OrderState.Placed
+ };
+
+ await _orderRepository.InsertAsync(order);
+ }
+}
+```
+
+Now open the `OrderDto` class (the `OrderDto.cs` file under the `Services` folder of the `CloudCrm.OrderingService.Contracts` project of the `CloudCrm.OrderingService` .NET solution) and add a new property named `ProductName`:
+
+```csharp
+using System;
+using CloudCrm.OrderingService.Enums;
+
+namespace CloudCrm.OrderingService.Services;
+
+public class OrderDto
+{
+ public Guid Id { get; set; }
+ public string CustomerName { get; set; }
+ public Guid ProductId { get; set; }
+ public string ProductName { get; set; } // New property
+ public OrderState State { get; set; }
+}
+```
+
+Lastly, open the `OrderingServiceApplicationAutoMapperProfile` class (the `OrderingServiceApplicationAutoMapperProfile.cs` file under the `ObjectMapping` folder of the `CloudCrm.OrderingService` project of the `CloudCrm.OrderingService` .NET solution) and ignore the `ProductName` property in the mapping configuration:
+
+```csharp
+using AutoMapper;
+using CloudCrm.OrderingService.Entities;
+using CloudCrm.OrderingService.Services;
+using Volo.Abp.AutoMapper;
+
+namespace CloudCrm.OrderingService.ObjectMapping;
+
+public class OrderingServiceApplicationAutoMapperProfile : Profile
+{
+ public OrderingServiceApplicationAutoMapperProfile()
+ {
+ CreateMap()
+ .Ignore(x => x.ProductName); // New line
+ }
+}
+```
+Let's explain the changes we made:
+
+- We added a new property named `ProductName` to the `OrderDto` class. This property will hold the product name.
+- We modified the `GetListAsync` method of the `OrderAppService` class to fetch the product details using the `IProductIntegrationService` interface. We first fetch the product IDs from the orders, then call the `GetProductsByIdsAsync` method of the `IProductIntegrationService` interface to fetch the product details. Finally, we map the product names to the `OrderDto` objects.
+
+### Generating Proxy Classes for the Integration Service
+
+We have created the `IProductIntegrationService` interface and the `ProductIntegrationService` class in the `CloudCrm.CatalogService` solution. Now, we need to generate the proxy classes for the integration service in the `CloudCrm.OrderingService` package. First, *Build & Start* the `CloudCrm.CatalogService` application in ABP Studio *Solution Runner*. Then, open the *Solution Explorer* and right-click on the `CloudCrm.OrderingService` package. Select the *ABP CLI* -> *Generate Proxy* -> *C#* command:
+
+
+
+It opens the *Generate C# proxies* window. Select the `CloudCrm.CatalogService` application from the *Application* dropdown list. Then, choose the *catalog* module from the *Module* dropdown list and choose the *integration* service from the *Service type* dropdown list. Check the *Without contracts* checkbox and click the *Generate* button:
+
+
+
+We have generated the proxy classes for the `IProductIntegrationService` interface. Now, we must add the *Remote Service* url to the `appsettings.json` file of the `CloudCrm.OrderingService` project. Open the `appsettings.json` file (the `appsettings.json` file of the `CloudCrm.OrderingService` project of the `CloudCrm.OrderingService` .NET solution) and add the *CatalogService* section following configuration:
+
+```json
+{
+ "RemoteServices": {
+ "CatalogService": {
+ "BaseUrl": "http://localhost:44334"
+ }
+ }
+}
+```
+
+> **BaseUrl** refers to the base URL of the Catalog service. You can use the *Copy Url* option from the Catalog service's context menu in the ABP Studio **Solution Runner** to paste it here.
+
+### Updating the UI to Display the Product Name
+
+Open the `Index.cshtml` file (the `Index.cshtml` file under the `Pages/Orders` folder of the `CloudCrm.Web` project of the `CloudCrm.Web` .NET solution) and update the table content to display the product name instead of the product ID:
+
+```html
+@page
+@model CloudCrm.Web.Pages.Orders.Index
+
+
Orders
+
+
+
+
+ @foreach (var order in Model.Orders)
+ {
+
+ Customer: @order.CustomerName
+ Product: @order.ProductName
+ State: @order.State
+
+ }
+
+
+
+```
+
+That's it! Now, you can *Build & Start* the all applications and run it in ABP Studio to see the result:
+
+
+
+Now, the Ordering service displays the product name instead of the product ID. We have successfully integrated the Ordering service with the Catalog service using HTTP API calls.
+
+> **Design Tip**
+>
+> It is suggested that you keep that type of communication to a minimum and not couple your services with each other. It can make your solution complicated and may also decrease your system performance. When you need to do it, think about performance and try to make some optimizations. For example, if the Ordering service frequently needs product data, you can use a kind of [cache layer](../../framework/fundamentals/caching.md), so it doesn't make frequent requests to the Catalog service.
diff --git a/docs/en/tutorials/microservice/part-07.md b/docs/en/tutorials/microservice/part-07.md
new file mode 100644
index 0000000000..0351df2a52
--- /dev/null
+++ b/docs/en/tutorials/microservice/part-07.md
@@ -0,0 +1,231 @@
+# Microservice Tutorial Part 07: Integrating the services: Using Distributed Events
+
+````json
+//[doc-nav]
+{
+ "Previous": {
+ "Name": "Integrating the services: HTTP API Calls",
+ "Path": "tutorials/microservice/part-06"
+ }
+}
+````
+
+Another common approach to communicating between microservices is messaging. By publishing and handling messages, a microservice can perform an operation when an event happens in another microservice.
+
+ABP provides two types of event buses for loosely coupled communication:
+
+* [Local Event Bus](../../framework/infrastructure/event-bus/local/index.md) is suitable for in-process messaging. However, it’s not suitable for microservices as it cannot communicate across different processes. For distributed systems, consider using a distributed event bus.
+
+* **[Distributed Event Bus](../../framework/infrastructure/event-bus/distributed/index.md)** is normal for inter-process messaging, like microservices, for publishing and subscribing to distributed events. However, ABP's distributed event bus works as local (in-process) by default (actually, it uses the Local Event Bus under the hood by default) unless you configure an external message broker.
+
+In this tutorial, we will use the distributed event bus to communicate between the `Order` and `Catalog` microservices.
+
+## Publishing an Event
+
+In the example scenario, we want to publish an event when a new order is placed. The Ordering service will publish the event since it knows when a new order is placed. The Catalog service will subscribe to that event and get notified when a new order is placed. This will decrease the stock count of the product related to the new order. The scenario is pretty simple; let's implement it.
+
+### Defining the Event Class
+
+Open the `CloudCrm.OrderingService` .NET solution in your IDE, create an `Events` folder and create a new class named `OrderPlacedEto` under the `CloudCrm.OrderingService.Contracts` project:
+
+```csharp
+using System;
+
+namespace CloudCrm.OrderingService.Events;
+
+public class OrderPlacedEto
+{
+ public string CustomerName { get; set; }
+ public Guid ProductId { get; set; }
+}
+```
+
+`OrderPlacedEto` is very simple. It is a plain C# class used to transfer data related to the event (*ETO* is an acronym for *Event Transfer Object*, a suggested naming convention but not required). You can add more properties if needed, but for this tutorial, it is more than enough.
+
+### Using the `IDistributedEventBus` Service
+
+The `IDistributedEventBus` service publishes events to the event bus. Until this point, the Ordering service only creates an Order and insert to database. Let's change that and publish `OrderPlacedEto` event, for that purpose open the `CloudCrm.OrderingService` project and update to the `OrderAppService` class as shown below:
+
+```csharp
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Threading.Tasks;
+using CloudCrm.CatalogService.IntegrationServices;
+using CloudCrm.OrderingService.Entities;
+using CloudCrm.OrderingService.Enums;
+using CloudCrm.OrderingService.Events;
+using CloudCrm.OrderingService.Localization;
+using Volo.Abp.Application.Services;
+using Volo.Abp.Domain.Repositories;
+using Volo.Abp.EventBus.Distributed;
+
+namespace CloudCrm.OrderingService.Services;
+
+public class OrderAppService : ApplicationService, IOrderAppService
+{
+ private readonly IRepository _orderRepository;
+ private readonly IProductIntegrationService _productIntegrationService;
+ private readonly IDistributedEventBus _distributedEventBus;
+
+ public OrderAppService(
+ IRepository orderRepository,
+ IProductIntegrationService productIntegrationService,
+ IDistributedEventBus distributedEventBus)
+ {
+ LocalizationResource = typeof(OrderingServiceResource);
+
+ _orderRepository = orderRepository;
+ _productIntegrationService = productIntegrationService;
+ _distributedEventBus = distributedEventBus;
+ }
+
+ public async Task> GetListAsync()
+ {
+ var orders = await _orderRepository.GetListAsync();
+
+ // Prepare a list of products we need
+ var productIds = orders.Select(o => o.ProductId).Distinct().ToList();
+ var products = (await _productIntegrationService
+ .GetProductsByIdsAsync(productIds))
+ .ToDictionary(p => p.Id, p => p.Name);
+
+ var orderDtos = ObjectMapper.Map, List>(orders);
+
+ orderDtos.ForEach(orderDto =>
+ {
+ orderDto.ProductName = products[orderDto.ProductId];
+ });
+
+ return orderDtos;
+ }
+
+ public async Task CreateAsync(OrderCreationDto input)
+ {
+ // Create a new Order entity
+ var order = new Order
+ {
+ CustomerName = input.CustomerName,
+ ProductId = input.ProductId,
+ State = OrderState.Placed
+ };
+
+ // Save it to the database
+ await _orderRepository.InsertAsync(order);
+
+ // Publish an event so other microservices can be informed
+ await _distributedEventBus.PublishAsync(
+ new OrderPlacedEto
+ {
+ ProductId = order.ProductId,
+ CustomerName = order.CustomerName
+ });
+ }
+}
+```
+The `OrderAppService.CreateAsync` method creates a new `Order` entity, saves it to the database and finally publishes an `OrderPlacedEto` event.
+
+## Subscribing to an Event
+
+The Catalog service will subscribe to the `OrderPlacedEto` event and decrease the stock count of the product related to the new order. Let's implement it.
+
+### Adding a Reference to the `CloudCrm.OrderingService.Contracts` Package
+
+Since the `OrderPlacedEto` class is in the `CloudCrm.OrderingService.Contracts` project, we must add that package's reference to the Catalog service. This time, we will use the Import Module feature of ABP Studio (as an alternative to the approach we used in the Adding a Reference to the `CloudCrm.CatalogService.Contracts` Package section of the [previous part](./part-06.md#adding-a-reference-to-the-cloudcrmcatalogservicecontracts-package)).
+
+Open the ABP Studio UI and stop the applications if they are running. Then, open the *Solution Explorer* panel and right-click on the `CloudCrm.CatalogService`. Select *Import Module* from the context menu:
+
+
+
+In the opening dialog, find and select the `CloudCrm.OrderingService` module, check the *Install this module* option, click the *OK* button:
+
+
+
+Once you click the OK button, the Ordering service is imported to the Catalog service. It opens the *Install Module* dialog:
+
+
+
+Here, select the `CloudCrm.OrderingService.Contracts` package on the left side (because we want to add that package reference) and `CloudCrm.CatalogService` package on the middle area (because we want to add the package reference to that project).
+
+You can check the ABP Studio's *Solution Explorer* panel to see the module and the project reference (dependency):
+
+
+
+### Handling the `OrderPlacedEto` Event
+
+Now, it's time to handle the `OrderPlacedEto` event in the Catalog service. Open the `CloudCrm.CatalogService` .NET solution in your IDE. Create a new `Orders` folder, and add a new class named `OrderEventHandler` inside that folder within the `CloudCrm.CatalogService` project:
+
+```csharp
+using System;
+using System.Threading.Tasks;
+using CloudCrm.CatalogService.Products;
+using CloudCrm.OrderingService.Events;
+using Volo.Abp.DependencyInjection;
+using Volo.Abp.Domain.Repositories;
+using Volo.Abp.EventBus.Distributed;
+
+namespace CloudCrm.CatalogService.Orders;
+
+public class OrderEventHandler :
+ IDistributedEventHandler,
+ ITransientDependency
+{
+ private readonly IRepository _productRepository;
+
+ public OrderEventHandler(IRepository productRepository)
+ {
+ _productRepository = productRepository;
+ }
+
+ public async Task HandleEventAsync(OrderPlacedEto eventData)
+ {
+ // Find the related product
+ var product = await _productRepository.FindAsync(eventData.ProductId);
+ if (product == null)
+ {
+ return;
+ }
+
+ // Decrease the stock count
+ product.StockCount = product.StockCount - 1;
+
+ // Update the entity in the database
+ await _productRepository.UpdateAsync(product);
+ }
+}
+```
+
+The `OrderEventHandler` class implements the `IDistributedEventHandler` interface to handle the `OrderPlacedEto` event. When the event is published, the `HandleEventAsync` method is called. In this method, we find the related product, decrease the stock count by one, and update the entity in the database.
+
+Implementing `ITransientDependency` registers the `OrderEventHandler` class to the dependency injection system as a transient object.
+
+### Testing the Order Creation
+
+To keep this tutorial simple, we will not implement a user interface for creating orders. Instead, we will use the Swagger UI to create an order. Open the *Solution Runner* panel in ABP Studio and use *Build & Start* to launch the `CloudCrm.OrderingService` and `CloudCrm.CatalogService` applications. Then, go to *Run* -> *Start All* to start the remaining applications listed in the [Solution Runner root item](../../studio/running-applications.md#run).
+
+Once the application is running and ready, [Browse](../../studio/running-applications.md#c-application) the `CloudCrm.OrderingService` application. Use the `POST /api/ordering/order` endpoint to create a new order:
+
+
+
+Find the *Order* API, click the *Try it out* button, enter a sample value the *Request body* section, and click the *Execute* button:
+
+```json
+{
+ "customerName": "David",
+ "productId": "5995897b-1de9-7272-b31c-3a165bbe7b18"
+}
+```
+
+> **IMPORTANT:** Here, you should type a valid Product Id from the Products table of your database!
+
+Once you press the *Execute* button, a new order is created. At that point, you can check the `/Orders` page to see if the new order is listed. You can also check the `/Products` page to see if the stock count of the related product is decreased by one in the `CloudCrm.Web` application.
+
+Here are sample screenshots from the Orders and Products pages of the `CloudCrm.Web` application:
+
+
+
+We placed a new order for *Product A*. As a result, the stock count of *Product A* is decreased from 53 to 52 and a new line is added to the Orders page.
+
+## Conclusion
+
+In this tutorial, we used the distributed event bus to communicate between the `Order` and `Catalog` microservices. We published an event when a new order is placed and handled that event in the Catalog service to decrease the stock count of the related product. This is a simple example, but it shows how you can use distributed events to communicate between microservices.
\ No newline at end of file
diff --git a/docs/en/tutorials/modular-crm/part-02.md b/docs/en/tutorials/modular-crm/part-02.md
index 6ebc22f20c..8a5b4b9e6c 100644
--- a/docs/en/tutorials/modular-crm/part-02.md
+++ b/docs/en/tutorials/modular-crm/part-02.md
@@ -14,7 +14,7 @@
}
````
-In this part, you will build a new product management module and install it in the main CRM application.
+In this part, you will create a new product management module and install it in the main CRM application.
## Creating Solution Folders
diff --git a/docs/en/tutorials/todo/layered/index.md b/docs/en/tutorials/todo/layered/index.md
index a67076dad1..a679bacac5 100644
--- a/docs/en/tutorials/todo/layered/index.md
+++ b/docs/en/tutorials/todo/layered/index.md
@@ -52,8 +52,8 @@ This documentation has a video tutorial on **YouTube**!! You can watch it here:
## Pre-Requirements
-* An IDE (e.g. [Visual Studio](https://visualstudio.microsoft.com/vs/)) that supports [.NET 8.0+](https://dotnet.microsoft.com/download/dotnet) development.
-* [Node v18.19+](https://nodejs.org/)
+* An IDE (e.g. [Visual Studio](https://visualstudio.microsoft.com/vs/)) that supports [.NET 9.0+](https://dotnet.microsoft.com/download/dotnet) development.
+* [Node v20.11+](https://nodejs.org/)
{{if DB=="Mongo"}}
@@ -111,7 +111,7 @@ For such cases, run the `abp install-libs` command on the root directory of your
abp install-libs
````
-> We suggest you install [Yarn](https://classic.yarnpkg.com/) to prevent possible package inconsistencies, if you haven't installed it yet.
+> We suggest you install [Yarn v1.22+ (not v2)](https://classic.yarnpkg.com/en/docs/install) to prevent possible package inconsistencies, if you haven't installed it yet.
{{if UI=="Blazor" || UI=="BlazorWebApp"}}
diff --git a/docs/en/tutorials/todo/single-layer/index.md b/docs/en/tutorials/todo/single-layer/index.md
index 3b4f755c70..c96c3715ef 100644
--- a/docs/en/tutorials/todo/single-layer/index.md
+++ b/docs/en/tutorials/todo/single-layer/index.md
@@ -48,8 +48,8 @@ This documentation has a video tutorial on **YouTube**!! You can watch it here:
## Pre-Requirements
-* An IDE (e.g. [Visual Studio](https://visualstudio.microsoft.com/vs/)) that supports [.NET 8.0+](https://dotnet.microsoft.com/download/dotnet) development.
-* [Node v18.19+](https://nodejs.org/)
+* An IDE (e.g. [Visual Studio](https://visualstudio.microsoft.com/vs/)) that supports [.NET 9.0+](https://dotnet.microsoft.com/download/dotnet) development.
+* [Node v20.11+](https://nodejs.org/)
{{if DB=="Mongo"}}
@@ -113,7 +113,7 @@ Run the `abp install-libs` command on the root directory of your solution to ins
abp install-libs
```
-> We suggest you install [Yarn](https://classic.yarnpkg.com/) to prevent possible package inconsistencies, if you haven't installed it yet.
+> We suggest you install [Yarn v1.22+ (not v2)](https://classic.yarnpkg.com/en/docs/install) to prevent possible package inconsistencies, if you haven't installed it yet.
{{if UI=="Blazor" || UI=="BlazorServer"}}
diff --git a/docs/en/ui-themes/lepton-x/blazor.md b/docs/en/ui-themes/lepton-x/blazor.md
index ef7ad683fb..78e0dc433d 100644
--- a/docs/en/ui-themes/lepton-x/blazor.md
+++ b/docs/en/ui-themes/lepton-x/blazor.md
@@ -262,7 +262,7 @@ If you need to replace the component, you can follow the steps below.
* Create a razor page, like `MyBreadcrumbs.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.Common;
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.Common;
@using Volo.Abp.DependencyInjection
@inherits Breadcrumbs
@@ -295,7 +295,7 @@ namespace LeptonXLite.DemoApp.Blazor.MyComponents
* Create a razor page, like `MyContentToolbar.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.Common;
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.Common;
@using Volo.Abp.DependencyInjection
@inherits ContentToolbar
@@ -328,7 +328,7 @@ namespace LeptonXLite.DemoApp.Blazor.MyComponents
* Create a razor page, like `MyGeneralSettings.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.Common;
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.Common;
@using Volo.Abp.DependencyInjection
@inherits GeneralSettings
@@ -361,7 +361,7 @@ namespace LeptonXLite.DemoApp.Blazor.MyComponents
* Create a razor page, like `MyMobileGeneralSettings.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.Common;
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.Common;
@using Volo.Abp.DependencyInjection
@inherits MobileGeneralSettings
@@ -400,7 +400,7 @@ Components used in the side menu layout.
* Create a razor page, like `MyMainMenu.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.Navigation;
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.Navigation;
@using Volo.Abp.DependencyInjection
@inherits MainMenu
@@ -431,7 +431,7 @@ namespace LeptonXLite.DemoApp.Blazor.MyComponents
* Create a razor page, like `MyMainMenuItem.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.SideMenu.Navigation;
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.SideMenu.Navigation;
@using Volo.Abp.DependencyInjection
@inherits MainMenuItem
@@ -464,7 +464,7 @@ namespace LeptonXLite.DemoApp.Blazor.MyComponents
* Create a razor page, like `MyMobileNavbar.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.SideMenu.Navigation;
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.SideMenu.Navigation;
@using Volo.Abp.DependencyInjection
@inherits MobileNavbar
@@ -497,7 +497,7 @@ namespace LeptonXLite.DemoApp.Blazor.MyComponents
* Create a razor page, like `MyMainHeader.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.SideMenu.MainHeader
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.SideMenu.MainHeader
@using Volo.Abp.DependencyInjection
@inherits MainHeader
@@ -534,7 +534,7 @@ If you need to replace the component, you can follow the steps below.
* Create a razor page, like `MyMainHeaderBranding.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.SideMenu.MainHeader
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.SideMenu.MainHeader
@using Volo.Abp.DependencyInjection
@inherits MainHeaderBranding
@@ -571,7 +571,7 @@ If you need to replace the component, you can follow the steps below.
* Create a razor page, like `MyMainHeaderToolbar.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.SideMenu.MainHeader
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.SideMenu.MainHeader
@using Volo.Abp.DependencyInjection
@inherits MainHeaderToolbar
@@ -610,7 +610,7 @@ Components used in the top menu layout.
* Create a razor page, like `MyMainMenu.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.Navigation;
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.Navigation;
@using Volo.Abp.DependencyInjection
@inherits MainMenu
@@ -641,7 +641,7 @@ namespace LeptonXLite.DemoApp.Blazor.MyComponents
* Create a razor page, like `MyMainMenuItem.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.Navigation;
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.Navigation;
@using Volo.Abp.DependencyInjection
@inherits MainMenuItem
@@ -674,7 +674,7 @@ namespace LeptonXLite.DemoApp.Blazor.MyComponents
* Create a razor page, like `MyMobileNavbar.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.Navigation;
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.Navigation;
@using Volo.Abp.DependencyInjection
@inherits MobileNavbar
@@ -707,7 +707,7 @@ namespace LeptonXLite.DemoApp.Blazor.MyComponents
* Create a razor page, like `MyMainHeader.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.MainHeader
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.MainHeader
@using Volo.Abp.DependencyInjection
@inherits MainHeader
@@ -742,7 +742,7 @@ Application branding can be customized with the `IBrandingProvider`. See the [Br
* Create a razor page, like `MyMainHeaderBranding.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.MainHeader
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.MainHeader
@using Volo.Abp.DependencyInjection
@inherits MainHeaderBranding
@@ -779,7 +779,7 @@ If you need to replace the component, you can follow the steps below.
* Create a razor page, like `MyMainHeaderToolbar.razor`, in your blazor application as shown below:
```html
-@Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.MainHeader
+@using Volo.Abp.AspNetCore.Components.Web.LeptonXTheme.Components.ApplicationLayout.TopMenu.MainHeader
@using Volo.Abp.DependencyInjection
@inherits MainHeaderToolbar
diff --git a/docs/en/ui-themes/lepton-x/index.md b/docs/en/ui-themes/lepton-x/index.md
index 241d4dfb89..e2b09358a4 100644
--- a/docs/en/ui-themes/lepton-x/index.md
+++ b/docs/en/ui-themes/lepton-x/index.md
@@ -88,6 +88,8 @@ You can use the following CLI command to download the source-code:
abp get-source Volo.Abp.LeptonXTheme
```
+> If you are using the old ABP CLI, you can use the command: `abp get-source Volo.Abp.LeptonXTheme.Pro`
+
If you want to download the source code of the preview version, you can use the following command:
```bash
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/DaprAspNetCore/AbpDaprEndpointRouteBuilderExtensions.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/DaprAspNetCore/AbpDaprEndpointRouteBuilderExtensions.cs
deleted file mode 100644
index 24f68ef9ab..0000000000
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/DaprAspNetCore/AbpDaprEndpointRouteBuilderExtensions.cs
+++ /dev/null
@@ -1,296 +0,0 @@
-// ------------------------------------------------------------------------
-// Copyright 2021 The Dapr Authors
-// Licensed under the Apache License, Version 2.0 (the "License");
-// you may not use this file except in compliance with the License.
-// You may obtain a copy of the License at
-// http://www.apache.org/licenses/LICENSE-2.0
-// Unless required by applicable law or agreed to in writing, software
-// distributed under the License is distributed on an "AS IS" BASIS,
-// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-// See the License for the specific language governing permissions and
-// limitations under the License.
-// ------------------------------------------------------------------------
-
-using System;
-using System.Collections.Generic;
-using System.Threading.Tasks;
-
-namespace Dapr
-{
- ///
- /// This class defines configurations for the subscribe endpoint.
- ///
- public class AbpSubscribeOptions
- {
- ///
- /// Gets or Sets a value which indicates whether to enable or disable processing raw messages.
- ///
- public bool EnableRawPayload { get; set; }
-
- ///
- /// An optional delegate used to configure the subscriptions.
- ///
- public Func, Task>? SubscriptionsCallback { get; set; }
- }
-
- ///
- /// This class defines subscribe endpoint response
- ///
- public class AbpSubscription
- {
- ///
- /// Gets or sets the topic name.
- ///
- public string Topic { get; set; } = default!;
-
- ///
- /// Gets or sets the pubsub name
- ///
- public string PubsubName { get; set; } = default!;
-
- ///
- /// Gets or sets the route
- ///
- public string? Route { get; set; }
-
- ///
- /// Gets or sets the routes
- ///
- public AbpRoutes? Routes { get; set; }
-
- ///
- /// Gets or sets the metadata.
- ///
- public AbpMetadata? Metadata { get; set; }
-
- ///
- /// Gets or sets the deadletter topic.
- ///
- public string? DeadLetterTopic { get; set; }
- }
-
- ///
- /// This class defines the metadata for subscribe endpoint.
- ///
- public class AbpMetadata : Dictionary
- {
- ///
- /// Initializes a new instance of the Metadata class.
- ///
- public AbpMetadata() { }
-
- ///
- /// Initializes a new instance of the Metadata class.
- ///
- ///
- public AbpMetadata(IDictionary dictionary) : base(dictionary) { }
-
- ///
- /// RawPayload key
- ///
- internal const string RawPayload = "rawPayload";
- }
-
- ///
- /// This class defines the routes for subscribe endpoint.
- ///
- public class AbpRoutes
- {
- ///
- /// Gets or sets the default route
- ///
- public string? Default { get; set; }
-
- ///
- /// Gets or sets the routing rules
- ///
- public List? Rules { get; set; }
- }
-
- ///
- /// This class defines the rule for subscribe endpoint.
- ///
- public class AbpRule
- {
- ///
- /// Gets or sets the CEL expression to match this route.
- ///
- public string Match { get; set; } = default!;
-
- ///
- /// Gets or sets the path of the route.
- ///
- public string Path { get; set; } = default!;
- }
-}
-
-namespace Microsoft.AspNetCore.Builder
-{
- using System.Collections.Generic;
- using System.Linq;
- using System.Text.Json;
- using System.Text.Json.Serialization;
- using Dapr;
- using Microsoft.AspNetCore.Http;
- using Microsoft.AspNetCore.Routing;
- using Microsoft.AspNetCore.Routing.Patterns;
- using Microsoft.Extensions.DependencyInjection;
- using Microsoft.Extensions.Logging;
-
- ///
- /// Contains extension methods for .
- ///
- public static class AbpDaprEndpointRouteBuilderExtensions
- {
- ///
- /// Maps an endpoint that will respond to requests to /dapr/subscribe from the
- /// Dapr runtime.
- ///
- /// The .
- /// The .
- public static IEndpointConventionBuilder MapAbpSubscribeHandler(this IEndpointRouteBuilder endpoints)
- {
- return CreateSubscribeEndPoint(endpoints);
- }
-
- ///
- /// Maps an endpoint that will respond to requests to /dapr/subscribe from the
- /// Dapr runtime.
- ///
- /// The .
- /// Configuration options
- /// The .
- ///
- public static IEndpointConventionBuilder MapAbpSubscribeHandler(this IEndpointRouteBuilder endpoints, AbpSubscribeOptions options)
- {
- return CreateSubscribeEndPoint(endpoints, options);
- }
-
- private static IEndpointConventionBuilder CreateSubscribeEndPoint(IEndpointRouteBuilder endpoints, AbpSubscribeOptions? options = null)
- {
- if (endpoints is null)
- {
- throw new System.ArgumentNullException(nameof(endpoints));
- }
-
- return endpoints.MapGet("dapr/subscribe", async context =>
- {
- var logger = context.RequestServices.GetService()?.CreateLogger("DaprTopicSubscription");
- var dataSource = context.RequestServices.GetRequiredService();
- var subscriptions = dataSource.Endpoints
- .OfType()
- .Where(e => e.Metadata.GetOrderedMetadata().Any(t => t.Name != null)) // only endpoints which have TopicAttribute with not null Name.
- .SelectMany(e =>
- {
- var topicMetadata = e.Metadata.GetOrderedMetadata();
- var originalTopicMetadata = e.Metadata.GetOrderedMetadata();
-
- var subs = new List<(string PubsubName, string Name, string? DeadLetterTopic, bool? EnableRawPayload, string Match, int Priority, Dictionary OriginalTopicMetadata, string? MetadataSeparator, RoutePattern RoutePattern)>();
-
- for (int i = 0; i < topicMetadata.Count(); i++)
- {
- subs.Add((topicMetadata[i].PubsubName,
- topicMetadata[i].Name,
- (topicMetadata[i] as IDeadLetterTopicMetadata)?.DeadLetterTopic,
- (topicMetadata[i] as IRawTopicMetadata)?.EnableRawPayload,
- topicMetadata[i].Match,
- topicMetadata[i].Priority,
- originalTopicMetadata.Where(m => (topicMetadata[i] as IOwnedOriginalTopicMetadata)?.OwnedMetadatas?.Any(o => o.Equals(m.Id)) == true || string.IsNullOrEmpty(m.Id))
- .GroupBy(c => c.Name)
- .ToDictionary(m => m.Key, m => m.Select(c => c.Value).Distinct().ToArray()),
- (topicMetadata[i] as IOwnedOriginalTopicMetadata)?.MetadataSeparator,
- e.RoutePattern));
- }
-
- return subs;
- })
- .Distinct()
- .GroupBy(e => new { e.PubsubName, e.Name })
- .Select(e => e.OrderBy(e => e.Priority))
- .Select(e =>
- {
- var first = e.First();
- var rawPayload = e.Any(e => e.EnableRawPayload.GetValueOrDefault());
- var metadataSeparator = e.FirstOrDefault(e => !string.IsNullOrEmpty(e.MetadataSeparator)).MetadataSeparator?.ToString() ?? ",";
- var rules = e.Where(e => !string.IsNullOrEmpty(e.Match)).ToList();
- var defaultRoutes = e.Where(e => string.IsNullOrEmpty(e.Match)).Select(e => RoutePatternToString(e.RoutePattern)).ToList();
- var defaultRoute = defaultRoutes.FirstOrDefault();
-
- //multiple identical names. use comma separation.
- var metadata = new AbpMetadata(e.SelectMany(c => c.OriginalTopicMetadata).GroupBy(c => c.Key).ToDictionary(c => c.Key, c => string.Join(metadataSeparator, c.SelectMany(c => c.Value).Distinct())));
- if (rawPayload || options?.EnableRawPayload is true)
- {
- metadata.Add(AbpMetadata.RawPayload, "true");
- }
-
- if (logger != null)
- {
- if (defaultRoutes.Count > 1)
- {
- logger.LogError("A default subscription to topic {name} on pubsub {pubsub} already exists.", first.Name, first.PubsubName);
- }
-
- var duplicatePriorities = rules.GroupBy(e => e.Priority)
- .Where(g => g.Count() > 1)
- .ToDictionary(x => x.Key, y => y.Count());
-
- foreach (var entry in duplicatePriorities)
- {
- logger.LogError("A subscription to topic {name} on pubsub {pubsub} has duplicate priorities for {priority}: found {count} occurrences.", first.Name, first.PubsubName, entry.Key, entry.Value);
- }
- }
-
- var subscription = new AbpSubscription
- {
- Topic = first.Name,
- PubsubName = first.PubsubName,
- Metadata = metadata.Count > 0 ? metadata : null,
- };
-
- if (first.DeadLetterTopic != null)
- {
- subscription.DeadLetterTopic = first.DeadLetterTopic;
- }
-
- // Use the V2 routing rules structure
- if (rules.Count > 0)
- {
- subscription.Routes = new AbpRoutes
- {
- Rules = rules.Select(e => new AbpRule
- {
- Match = e.Match,
- Path = RoutePatternToString(e.RoutePattern),
- }).ToList(),
- Default = defaultRoute,
- };
- }
- // Use the V1 structure for backward compatibility.
- else
- {
- subscription.Route = defaultRoute;
- }
-
- return subscription;
- })
- .OrderBy(e => (e.PubsubName, e.Topic))
- .ToList();
-
- await options?.SubscriptionsCallback!(subscriptions)!;
- await context.Response.WriteAsync(JsonSerializer.Serialize(subscriptions,
- new JsonSerializerOptions
- {
- PropertyNamingPolicy = JsonNamingPolicy.CamelCase,
- DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull
- }));
- });
- }
-
- private static string RoutePatternToString(RoutePattern routePattern)
- {
- return string.Join("/", routePattern.PathSegments
- .Select(segment => string.Concat(segment.Parts.Cast()
- .Select(part => part.Content))));
- }
- }
-}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/Volo/Abp/AspNetCore/Mvc/Dapr/EventBus/AbpAspNetCoreMvcDaprEventBusModule.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/Volo/Abp/AspNetCore/Mvc/Dapr/EventBus/AbpAspNetCoreMvcDaprEventBusModule.cs
index a291c4e769..a4a4f97f5d 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/Volo/Abp/AspNetCore/Mvc/Dapr/EventBus/AbpAspNetCoreMvcDaprEventBusModule.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/Volo/Abp/AspNetCore/Mvc/Dapr/EventBus/AbpAspNetCoreMvcDaprEventBusModule.cs
@@ -1,11 +1,16 @@
-using System.Linq;
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text.Json;
using System.Threading.Tasks;
using Dapr;
using Microsoft.AspNetCore.Builder;
+using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Routing;
using Microsoft.Extensions.DependencyInjection;
+using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Options;
-using Volo.Abp.DependencyInjection;
+using Volo.Abp.Dapr;
using Volo.Abp.EventBus;
using Volo.Abp.EventBus.Dapr;
using Volo.Abp.EventBus.Distributed;
@@ -21,50 +26,98 @@ public class AbpAspNetCoreMvcDaprEventBusModule : AbpModule
{
public override void ConfigureServices(ServiceConfigurationContext context)
{
- var subscribeOptions = context.Services.ExecutePreConfiguredActions();
-
- Configure(options =>
+ PostConfigure(options =>
{
options.EndpointConfigureActions.Add(endpointContext =>
{
- var rootServiceProvider = endpointContext.ScopeServiceProvider.GetRequiredService();
- subscribeOptions.SubscriptionsCallback = subscriptions =>
- {
- var daprEventBusOptions = rootServiceProvider.GetRequiredService>().Value;
- foreach (var handler in rootServiceProvider.GetRequiredService>().Value.Handlers)
+ var topicMetadatas = endpointContext.Endpoints.DataSources.SelectMany(x => x.Endpoints).OfType()
+ .Where(e => e.Metadata.GetOrderedMetadata().Any(t => t.Name != null))
+ .SelectMany(e => e.Metadata.GetOrderedMetadata())
+ .ToList();
+
+ var endpointConventionBuilder = endpointContext.Endpoints.MapPost(
+ "/api/abp/dapr/event", async httpContext =>
{
- foreach (var @interface in handler.GetInterfaces().Where(x => x.IsGenericType && x.GetGenericTypeDefinition() == typeof(IDistributedEventHandler<>)))
- {
- var eventType = @interface.GetGenericArguments()[0];
- var eventName = EventNameAttribute.GetNameOrDefault(eventType);
-
- if (subscriptions.Any(x => x.PubsubName == daprEventBusOptions.PubSubName && x.Topic == eventName))
- {
- // Controllers with a [Topic] attribute can replace built-in event handlers.
- continue;
- }
-
- var subscription = new AbpSubscription
- {
- PubsubName = daprEventBusOptions.PubSubName,
- Topic = eventName,
- Route = AbpAspNetCoreMvcDaprPubSubConsts.DaprEventCallbackUrl,
- Metadata = new AbpMetadata
- {
- {
- AbpMetadata.RawPayload, "true"
- }
- }
- };
- subscriptions.Add(subscription);
- }
- }
-
- return Task.CompletedTask;
- };
-
- endpointContext.Endpoints.MapAbpSubscribeHandler(subscribeOptions);
+ await HandleEventAsync(httpContext);
+ });
+
+ var abpEvents = GetAbpEvents(endpointContext);
+ foreach (var @event in abpEvents.Where(x => !topicMetadatas.Any(t => t.PubsubName == x.PubsubName && t.Name == x.Name)))
+ {
+ endpointConventionBuilder.WithMetadata(new TopicAttribute(
+ @event.PubsubName,
+ @event.Name,
+ true));
+ }
+
+ endpointContext.Endpoints.MapSubscribeHandler();
});
});
}
+
+ private List GetAbpEvents(EndpointRouteBuilderContext endpointContext)
+ {
+ var subscriptions = new List();
+ var daprEventBusOptions = endpointContext.Endpoints.ServiceProvider.GetRequiredService>().Value;
+
+ foreach (var @interface in endpointContext.Endpoints.ServiceProvider.GetRequiredService>().Value.Handlers
+ .SelectMany(x => x.GetInterfaces().Where(i => i.IsGenericType && i.GetGenericTypeDefinition() == typeof(IDistributedEventHandler<>))))
+ {
+ var eventType = @interface.GetGenericArguments()[0];
+ var eventName = EventNameAttribute.GetNameOrDefault(eventType);
+
+ var subscription = new TopicAttribute(daprEventBusOptions.PubSubName, eventName);
+ subscriptions.Add(subscription);
+ }
+
+ return subscriptions;
+ }
+
+ private async static Task HandleEventAsync(HttpContext httpContext)
+ {
+ var logger = httpContext.RequestServices.GetRequiredService>();
+
+ httpContext.ValidateDaprAppApiToken();
+
+ var daprSerializer = httpContext.RequestServices.GetRequiredService();
+ var body = (await JsonDocument.ParseAsync(httpContext.Request.Body));
+
+ var pubSubName = body.RootElement.GetProperty("pubsubname").GetString();
+ var topic = body.RootElement.GetProperty("topic").GetString();
+ var data = body.RootElement.GetProperty("data").GetRawText();
+ if (pubSubName.IsNullOrWhiteSpace() || topic.IsNullOrWhiteSpace() || data.IsNullOrWhiteSpace())
+ {
+ logger.LogError("Invalid Dapr event request.");
+ httpContext.Response.StatusCode = 400;
+ return;
+ }
+
+ var distributedEventBus = httpContext.RequestServices.GetRequiredService();
+
+ if (IsAbpDaprEventData(data))
+ {
+ var daprEventData = daprSerializer.Deserialize(data, typeof(AbpDaprEventData)).As();
+ var eventData = daprSerializer.Deserialize(daprEventData.JsonData, distributedEventBus.GetEventType(daprEventData.Topic));
+ await distributedEventBus.TriggerHandlersAsync(distributedEventBus.GetEventType(daprEventData.Topic), eventData, daprEventData.MessageId, daprEventData.CorrelationId);
+ }
+ else
+ {
+ var eventData = daprSerializer.Deserialize(data, distributedEventBus.GetEventType(topic!));
+ await distributedEventBus.TriggerHandlersAsync(distributedEventBus.GetEventType(topic!), eventData);
+ }
+
+ httpContext.Response.StatusCode = 200;
+ }
+
+ private static bool IsAbpDaprEventData(string data)
+ {
+ var document = JsonDocument.Parse(data);
+ var objects = document.RootElement.EnumerateObject().ToList();
+ return objects.Count == 5 &&
+ objects.Any(x => x.Name.Equals("PubSubName", StringComparison.CurrentCultureIgnoreCase)) &&
+ objects.Any(x => x.Name.Equals("Topic", StringComparison.CurrentCultureIgnoreCase)) &&
+ objects.Any(x => x.Name.Equals("MessageId", StringComparison.CurrentCultureIgnoreCase)) &&
+ objects.Any(x => x.Name.Equals("JsonData", StringComparison.CurrentCultureIgnoreCase)) &&
+ objects.Any(x => x.Name.Equals("CorrelationId", StringComparison.CurrentCultureIgnoreCase));
+ }
}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/Volo/Abp/AspNetCore/Mvc/Dapr/EventBus/AbpAspNetCoreMvcDaprPubSubConsts.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/Volo/Abp/AspNetCore/Mvc/Dapr/EventBus/AbpAspNetCoreMvcDaprPubSubConsts.cs
deleted file mode 100644
index 5f224abc7e..0000000000
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/Volo/Abp/AspNetCore/Mvc/Dapr/EventBus/AbpAspNetCoreMvcDaprPubSubConsts.cs
+++ /dev/null
@@ -1,6 +0,0 @@
-namespace Volo.Abp.AspNetCore.Mvc.Dapr.EventBus;
-
-public class AbpAspNetCoreMvcDaprPubSubConsts
-{
- public const string DaprEventCallbackUrl = "api/abp/dapr/event";
-}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/Volo/Abp/AspNetCore/Mvc/Dapr/EventBus/Controllers/AbpAspNetCoreMvcDaprEventsController.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/Volo/Abp/AspNetCore/Mvc/Dapr/EventBus/Controllers/AbpAspNetCoreMvcDaprEventsController.cs
deleted file mode 100644
index 946c39f59c..0000000000
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.Dapr.EventBus/Volo/Abp/AspNetCore/Mvc/Dapr/EventBus/Controllers/AbpAspNetCoreMvcDaprEventsController.cs
+++ /dev/null
@@ -1,62 +0,0 @@
-using System;
-using System.Linq;
-using System.Text.Json;
-using System.Threading.Tasks;
-using Microsoft.AspNetCore.Mvc;
-using Microsoft.Extensions.DependencyInjection;
-using Microsoft.Extensions.Logging;
-using Volo.Abp.Dapr;
-using Volo.Abp.EventBus.Dapr;
-
-namespace Volo.Abp.AspNetCore.Mvc.Dapr.EventBus.Controllers;
-
-[Area("abp")]
-[RemoteService(Name = "abp")]
-public class AbpAspNetCoreMvcDaprEventsController : AbpController
-{
- [HttpPost(AbpAspNetCoreMvcDaprPubSubConsts.DaprEventCallbackUrl)]
- public virtual async Task EventAsync()
- {
- HttpContext.ValidateDaprAppApiToken();
-
- var daprSerializer = HttpContext.RequestServices.GetRequiredService();
- var body = (await JsonDocument.ParseAsync(HttpContext.Request.Body));
-
- var pubSubName = body.RootElement.GetProperty("pubsubname").GetString();
- var topic = body.RootElement.GetProperty("topic").GetString();
- var data = body.RootElement.GetProperty("data").GetRawText();
- if (pubSubName.IsNullOrWhiteSpace() || topic.IsNullOrWhiteSpace() || data.IsNullOrWhiteSpace())
- {
- Logger.LogError("Invalid Dapr event request.");
- return BadRequest();
- }
-
- var distributedEventBus = HttpContext.RequestServices.GetRequiredService();
-
- if (IsAbpDaprEventData(data))
- {
- var daprEventData = daprSerializer.Deserialize(data, typeof(AbpDaprEventData)).As();
- var eventData = daprSerializer.Deserialize(daprEventData.JsonData, distributedEventBus.GetEventType(daprEventData.Topic));
- await distributedEventBus.TriggerHandlersAsync(distributedEventBus.GetEventType(daprEventData.Topic), eventData, daprEventData.MessageId, daprEventData.CorrelationId);
- }
- else
- {
- var eventData = daprSerializer.Deserialize(data, distributedEventBus.GetEventType(topic!));
- await distributedEventBus.TriggerHandlersAsync(distributedEventBus.GetEventType(topic!), eventData);
- }
-
- return Ok();
- }
-
- protected virtual bool IsAbpDaprEventData(string data)
- {
- var document = JsonDocument.Parse(data);
- var objects = document.RootElement.EnumerateObject().ToList();
- return objects.Count == 5 &&
- objects.Any(x => x.Name.Equals("PubSubName", StringComparison.CurrentCultureIgnoreCase)) &&
- objects.Any(x => x.Name.Equals("Topic", StringComparison.CurrentCultureIgnoreCase)) &&
- objects.Any(x => x.Name.Equals("MessageId", StringComparison.CurrentCultureIgnoreCase)) &&
- objects.Any(x => x.Name.Equals("JsonData", StringComparison.CurrentCultureIgnoreCase)) &&
- objects.Any(x => x.Name.Equals("CorrelationId", StringComparison.CurrentCultureIgnoreCase));
- }
-}
diff --git a/framework/src/Volo.Abp.AspNetCore/Microsoft/AspNetCore/Builder/AbpApplicationBuilderExtensions.cs b/framework/src/Volo.Abp.AspNetCore/Microsoft/AspNetCore/Builder/AbpApplicationBuilderExtensions.cs
index bd653faf55..3cada3bf46 100644
--- a/framework/src/Volo.Abp.AspNetCore/Microsoft/AspNetCore/Builder/AbpApplicationBuilderExtensions.cs
+++ b/framework/src/Volo.Abp.AspNetCore/Microsoft/AspNetCore/Builder/AbpApplicationBuilderExtensions.cs
@@ -15,6 +15,7 @@ using Volo.Abp.AspNetCore.Auditing;
using Volo.Abp.AspNetCore.ExceptionHandling;
using Volo.Abp.AspNetCore.Security;
using Volo.Abp.AspNetCore.Security.Claims;
+using Volo.Abp.AspNetCore.StaticFiles;
using Volo.Abp.AspNetCore.Tracing;
using Volo.Abp.AspNetCore.Uow;
using Volo.Abp.AspNetCore.VirtualFileSystem;
@@ -126,6 +127,38 @@ public static class AbpApplicationBuilderExtensions
return app.UseMiddleware();
}
+ ///
+ /// Configures the application to serve static files that match the specified filename patterns with the WebRootFileProvider of the application.
+ ///
+ /// The used to configure the application pipeline.
+ /// The file name patterns to include when serving static files (e.g., "appsettings*.json").
+ /// Supports glob patterns. See Glob patterns documentation.
+ ///
+ /// The instance.
+ public static IApplicationBuilder UseStaticFilesForPatterns(this IApplicationBuilder app, params string[] includeFileNamePatterns)
+ {
+ return UseStaticFilesForPatterns(app, includeFileNamePatterns, app.ApplicationServices.GetRequiredService().WebRootFileProvider);
+ }
+
+ ///
+ /// Configures the application to serve static files that match the specified filename patterns with the specified file provider.
+ ///
+ /// The used to configure the application pipeline.
+ /// The file name patterns to include when serving static files (e.g., "appsettings*.json").
+ /// Supports glob patterns. See Glob patterns documentation.
+ ///
+ /// The
+ /// The instance.
+ public static IApplicationBuilder UseStaticFilesForPatterns(this IApplicationBuilder app, string[] includeFileNamePatterns, IFileProvider fileProvider)
+ {
+ app.UseStaticFiles(new StaticFileOptions
+ {
+ FileProvider = new AbpStaticFileProvider(includeFileNamePatterns, fileProvider)
+ });
+
+ return app;
+ }
+
///
/// MapAbpStaticAssets is used to serve the files from the abp virtual file system embedded resources(js/css) and call the MapStaticAssets.
///
diff --git a/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/Auditing/AbpAspNetCoreAuditingUrlOptions.cs b/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/Auditing/AbpAspNetCoreAuditingUrlOptions.cs
new file mode 100644
index 0000000000..7d371aa331
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/Auditing/AbpAspNetCoreAuditingUrlOptions.cs
@@ -0,0 +1,10 @@
+namespace Volo.Abp.AspNetCore.Auditing;
+
+public class AbpAspNetCoreAuditingUrlOptions
+{
+ public bool IncludeSchema { get; set; }
+
+ public bool IncludeHost { get; set; }
+
+ public bool IncludeQuery { get; set; }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/Auditing/AspNetCoreAuditLogContributor.cs b/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/Auditing/AspNetCoreAuditLogContributor.cs
index fc010f9846..becdee806d 100644
--- a/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/Auditing/AspNetCoreAuditLogContributor.cs
+++ b/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/Auditing/AspNetCoreAuditLogContributor.cs
@@ -1,9 +1,11 @@
using System;
using System.Linq;
+using System.Text;
using Microsoft.AspNetCore.Http;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Logging.Abstractions;
+using Microsoft.Extensions.Options;
using Volo.Abp.AspNetCore.ExceptionHandling;
using Volo.Abp.AspNetCore.WebClientInfo;
using Volo.Abp.Auditing;
@@ -40,7 +42,7 @@ public class AspNetCoreAuditLogContributor : AuditLogContributor, ITransientDepe
if (context.AuditInfo.Url == null)
{
- context.AuditInfo.Url = BuildUrl(httpContext);
+ context.AuditInfo.Url = GetUrl(context, httpContext);
}
var clientInfoProvider = context.ServiceProvider.GetRequiredService();
@@ -88,18 +90,29 @@ public class AspNetCoreAuditLogContributor : AuditLogContributor, ITransientDepe
context.AuditInfo.HttpStatusCode = httpContext.Response.StatusCode;
}
- protected virtual string BuildUrl(HttpContext httpContext)
+ protected virtual string GetUrl(AuditLogContributionContext context, HttpContext httpContext)
{
- //TODO: Add options to include/exclude query, schema and host
+ var options = context.ServiceProvider.GetRequiredService>();
+ var stringBuilder = new StringBuilder();
- var uriBuilder = new UriBuilder
+ if (options.Value.IncludeSchema)
{
- Scheme = httpContext.Request.Scheme,
- Host = httpContext.Request.Host.Host,
- Path = httpContext.Request.Path.ToString(),
- Query = httpContext.Request.QueryString.ToString()
- };
-
- return uriBuilder.Uri.AbsolutePath;
+ stringBuilder.Append(httpContext.Request.Scheme);
+ stringBuilder.Append("://");
+ }
+
+ if (options.Value.IncludeHost)
+ {
+ stringBuilder.Append(httpContext.Request.Host.Host);
+ }
+
+ stringBuilder.Append(httpContext.Request.Path.ToString());
+
+ if (options.Value.IncludeQuery)
+ {
+ stringBuilder.Append(httpContext.Request.QueryString.ToString());
+ }
+
+ return stringBuilder.ToString();
}
}
diff --git a/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/StaticFiles/AbpStaticFileProvider.cs b/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/StaticFiles/AbpStaticFileProvider.cs
new file mode 100644
index 0000000000..f7893932f2
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/StaticFiles/AbpStaticFileProvider.cs
@@ -0,0 +1,42 @@
+using System;
+using System.Collections.Generic;
+using System.IO;
+using Microsoft.Extensions.FileProviders;
+using Microsoft.Extensions.FileSystemGlobbing;
+using Microsoft.Extensions.Primitives;
+
+namespace Volo.Abp.AspNetCore.StaticFiles;
+
+public class AbpStaticFileProvider : IFileProvider
+{
+ private readonly Matcher _matcher;
+ private readonly IFileProvider _fileProvider;
+
+ /// The file provider to be used to get the files.
+ /// The file name patterns to include when serving static files (e.g., "appsettings*.json").
+ /// Supports glob patterns. See Glob patterns documentation.
+ ///
+ public AbpStaticFileProvider(IReadOnlyList fileNamePatterns, IFileProvider fileProvider)
+ {
+ _fileProvider = fileProvider;
+ _matcher = new Matcher(StringComparison.OrdinalIgnoreCase);
+ _matcher.AddIncludePatterns(fileNamePatterns);
+ }
+
+ public IDirectoryContents GetDirectoryContents(string subpath)
+ {
+ return new NotFoundDirectoryContents();
+ }
+
+ public IFileInfo GetFileInfo(string subpath)
+ {
+ return _matcher.Match(Path.GetFileName(subpath)).HasMatches ?
+ _fileProvider.GetFileInfo(subpath) :
+ new NotFoundFileInfo(subpath);
+ }
+
+ public IChangeToken Watch(string filter)
+ {
+ return NullChangeToken.Singleton;
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.Auditing/Volo/Abp/Auditing/AuditingInterceptor.cs b/framework/src/Volo.Abp.Auditing/Volo/Abp/Auditing/AuditingInterceptor.cs
index 6be28d24cf..0f8894ed10 100644
--- a/framework/src/Volo.Abp.Auditing/Volo/Abp/Auditing/AuditingInterceptor.cs
+++ b/framework/src/Volo.Abp.Auditing/Volo/Abp/Auditing/AuditingInterceptor.cs
@@ -191,7 +191,9 @@ public class AuditingInterceptor : AbpInterceptor, ITransientDependency
}
if (!options.IsEnabledForGetRequests &&
- invocation.Method.Name.StartsWith("Get", StringComparison.OrdinalIgnoreCase))
+ (string.Equals(auditLogInfo.HttpMethod, "Get", StringComparison.OrdinalIgnoreCase) ||
+ string.Equals(auditLogInfo.HttpMethod, "Head", StringComparison.OrdinalIgnoreCase) ||
+ invocation.Method.Name.StartsWith("Get", StringComparison.OrdinalIgnoreCase)))
{
return false;
}
diff --git a/framework/src/Volo.Abp.BlazoriseUI/Components/EntityAction.razor b/framework/src/Volo.Abp.BlazoriseUI/Components/EntityAction.razor
index 9bd07ce7da..d913647264 100644
--- a/framework/src/Volo.Abp.BlazoriseUI/Components/EntityAction.razor
+++ b/framework/src/Volo.Abp.BlazoriseUI/Components/EntityAction.razor
@@ -17,7 +17,8 @@
Disabled=@Disabled>
@if(!string.IsNullOrEmpty(Icon))
{
-
+ var iconClass = Text.IsNullOrEmpty() ? "" : "me-1";
+
}
@Text
diff --git a/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/Auth/AuthService.cs b/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/Auth/AuthService.cs
index 8695c0c418..333ae59e3f 100644
--- a/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/Auth/AuthService.cs
+++ b/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/Auth/AuthService.cs
@@ -85,7 +85,11 @@ public class AuthService : IAuthService, ITransientDependency
}
var accessToken = await AuthenticationService.GetAccessTokenAsync(configuration);
-
+
+ if (!Directory.Exists(CliPaths.Root))
+ {
+ Directory.CreateDirectory(CliPaths.Root);
+ }
File.WriteAllText(CliPaths.AccessToken, accessToken, Encoding.UTF8);
}
diff --git a/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/ProjectModification/VoloNugetPackagesVersionUpdater.cs b/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/ProjectModification/VoloNugetPackagesVersionUpdater.cs
index 9f23645fb9..4ece393913 100644
--- a/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/ProjectModification/VoloNugetPackagesVersionUpdater.cs
+++ b/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/ProjectModification/VoloNugetPackagesVersionUpdater.cs
@@ -1,4 +1,5 @@
using System;
+using System.Collections.Generic;
using NuGet.Versioning;
using System.IO;
using System.Linq;
@@ -39,6 +40,7 @@ public class VoloNugetPackagesVersionUpdater : ITransientDependency
var latestVersionInfo = await _packageVersionCheckerService.GetLatestVersionOrNullAsync("Volo.Abp.Core", includeReleaseCandidates: includeReleaseCandidates);
var latestReleaseCandidateVersionInfo = await _packageVersionCheckerService.GetLatestVersionOrNullAsync("Volo.Abp.Core", includeReleaseCandidates: true);
var latestVersionFromMyGet = await GetLatestVersionFromMyGet("Volo.Abp.Core");
+ var latestStableVersions = await _packageVersionCheckerService.GetLatestStableVersionsAsync();
async Task UpdateAsync(string filePath)
{
@@ -55,7 +57,8 @@ public class VoloNugetPackagesVersionUpdater : ITransientDependency
latestVersionInfo.Version,
latestReleaseCandidateVersionInfo.Version,
latestVersionFromMyGet,
- version);
+ version,
+ latestStableVersions: latestStableVersions);
fs.Seek(0, SeekOrigin.Begin);
fs.SetLength(0);
@@ -83,6 +86,7 @@ public class VoloNugetPackagesVersionUpdater : ITransientDependency
var latestVersionInfo = await _packageVersionCheckerService.GetLatestVersionOrNullAsync("Volo.Abp.Core");
var latestReleaseCandidateVersionInfo = await _packageVersionCheckerService.GetLatestVersionOrNullAsync("Volo.Abp.Core", includeReleaseCandidates: true);
var latestVersionFromMyGet = await GetLatestVersionFromMyGet("Volo.Abp.Core");
+ var latestStableVersions = await _packageVersionCheckerService.GetLatestStableVersionsAsync();
using (var fs = File.Open(projectPath, FileMode.Open, FileAccess.ReadWrite, FileShare.None))
{
@@ -97,7 +101,8 @@ public class VoloNugetPackagesVersionUpdater : ITransientDependency
latestVersionInfo.Version,
latestReleaseCandidateVersionInfo.Version,
latestVersionFromMyGet,
- version);
+ version,
+ latestStableVersions: latestStableVersions);
fs.Seek(0, SeekOrigin.Begin);
fs.SetLength(0);
@@ -114,13 +119,20 @@ public class VoloNugetPackagesVersionUpdater : ITransientDependency
protected virtual async Task UpdateInternalAsync(string projectPath, bool includeNightlyPreviews = false, bool includeReleaseCandidates = false, bool switchToStable = false)
{
+ var latestStableVersions = await _packageVersionCheckerService.GetLatestStableVersionsAsync();
+
using (var fs = File.Open(projectPath, FileMode.Open, FileAccess.ReadWrite, FileShare.None))
{
using (var sr = new StreamReader(fs, Encoding.Default, true))
{
var fileContent = await sr.ReadToEndAsync();
- var updatedContent = await UpdateVoloPackagesAsync(fileContent, includeNightlyPreviews, includeReleaseCandidates, switchToStable);
+ var updatedContent = await UpdateVoloPackagesAsync(
+ fileContent,
+ includeNightlyPreviews,
+ includeReleaseCandidates,
+ switchToStable,
+ latestStableVersions: latestStableVersions);
fs.Seek(0, SeekOrigin.Begin);
fs.SetLength(0);
@@ -153,7 +165,8 @@ public class VoloNugetPackagesVersionUpdater : ITransientDependency
SemanticVersion latestNugetVersion = null,
SemanticVersion latestNugetReleaseCandidateVersion = null,
string latestMyGetVersion = null,
- string specifiedVersion = null)
+ string specifiedVersion = null,
+ List latestStableVersions = null)
{
string packageId = null;
@@ -206,29 +219,39 @@ public class VoloNugetPackagesVersionUpdater : ITransientDependency
if (!specifiedVersion.IsNullOrWhiteSpace())
{
- if (isLeptonXPackage || isStudioPackage)
+ var leptonXPackageVersion = latestStableVersions?
+ .FirstOrDefault(v => v.Version.Equals(specifiedVersion, StringComparison.InvariantCultureIgnoreCase))?.LeptonX?.Version;
+
+ if ((isLeptonXPackage && string.IsNullOrWhiteSpace(leptonXPackageVersion)) || isStudioPackage)
{
Logger.LogWarning("Package: {PackageId} could not be updated. Please manually update the package version yourself to prevent version mismatches!", packageId);
continue;
}
- if (await SpecifiedVersionExists(specifiedVersion, packageId))
+ var isLeptonXPackageWithVersion = isLeptonXPackage && !string.IsNullOrWhiteSpace(leptonXPackageVersion);
+
+ if (isLeptonXPackageWithVersion || await SpecifiedVersionExists(specifiedVersion, packageId))
+ {
+ TryUpdatingPackage(isLeptonXPackageWithVersion ? leptonXPackageVersion : specifiedVersion);
+ }
+ else
{
- var specifiedSemanticVersion = SemanticVersion.Parse(specifiedVersion);
+ Logger.LogWarning("Package \"{PackageId}\" specified version v{SpecifiedVersion} does not exist!", packageId, specifiedVersion);
+ }
+
+ void TryUpdatingPackage(string versionToUpdate)
+ {
+ var specifiedSemanticVersion = SemanticVersion.Parse(versionToUpdate);
if (specifiedSemanticVersion > currentSemanticVersion)
{
- Logger.LogInformation("Updating package \"{PackageId}\" from v{CurrentVersion} to v{SpecifiedVersion}", packageId, currentVersion, specifiedVersion);
- versionAttribute.Value = specifiedVersion;
+ Logger.LogInformation("Updating package \"{PackageId}\" from v{CurrentVersion} to v{SpecifiedVersion}", packageId, currentVersion, versionToUpdate);
+ versionAttribute.Value = versionToUpdate;
}
else
{
- Logger.LogWarning("Unable to update package \"{PackageId}\" version v{CurrentVersion} to v{SpecifiedVersion}", packageId, currentVersion, specifiedVersion);
+ Logger.LogWarning("Unable to update package \"{PackageId}\" version v{CurrentVersion} to v{SpecifiedVersion}", packageId, currentVersion, versionToUpdate);
}
}
- else
- {
- Logger.LogWarning("Package \"{PackageId}\" specified version v{SpecifiedVersion} does not exist!", packageId, specifiedVersion);
- }
}
else
{
@@ -252,7 +275,7 @@ public class VoloNugetPackagesVersionUpdater : ITransientDependency
}
else
{
- latestVersion = latestMyGetVersion == null ? await GetLatestVersionFromMyGet(packageId) : latestMyGetVersion;
+ latestVersion = latestMyGetVersion ?? await GetLatestVersionFromMyGet(packageId);
}
if(latestVersion == null)
diff --git a/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/Version/PackageVersionCheckerService.cs b/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/Version/PackageVersionCheckerService.cs
index d5764c1c0e..316c53e01e 100644
--- a/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/Version/PackageVersionCheckerService.cs
+++ b/framework/src/Volo.Abp.Cli.Core/Volo/Abp/Cli/Version/PackageVersionCheckerService.cs
@@ -5,6 +5,7 @@ using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Threading.Tasks;
+using JetBrains.Annotations;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Logging.Abstractions;
using Volo.Abp.Cli.Http;
@@ -111,6 +112,11 @@ public class PackageVersionCheckerService : ITransientDependency
? new LatestVersionInfo(semanticVersion, latestStableVersionResult.Message)
: null;
}
+
+ public async Task> GetLatestStableVersionsAsync()
+ {
+ return await GetLatestStableVersionsInternalAsync();
+ }
private static ConcurrentDictionary CommercialPackagesCache { get; } = new ();
@@ -225,7 +231,7 @@ public class PackageVersionCheckerService : ITransientDependency
_apiKeyResult ??= await _apiKeyService.GetApiKeyOrNullAsync();
}
- private async Task GetLatestStableVersionOrNullAsync()
+ private async Task> GetLatestStableVersionsInternalAsync()
{
try
{
@@ -241,16 +247,24 @@ public class PackageVersionCheckerService : ITransientDependency
var content = await responseMessage.Content.ReadAsStringAsync();
var result = JsonSerializer.Deserialize>(content);
-
- return result.FirstOrDefault(x => x.Type.ToLowerInvariant() == "stable");
+ return result.OrderBy(q => q.Type.ToLowerInvariant() == "stable").ThenBy(q => SemanticVersion.Parse(q.Version)).ToList();
}
}
catch
{
- return null;
+ return [];
}
}
+ private async Task GetLatestStableVersionOrNullAsync()
+ {
+ var latestStableVersionsResult = await GetLatestStableVersionsInternalAsync();
+
+ return latestStableVersionsResult.Count <= 0
+ ? null
+ : latestStableVersionsResult.FirstOrDefault();
+ }
+
public class NuGetSearchResultDto
{
public int TotalHits { get; set; }
@@ -270,6 +284,11 @@ public class PackageVersionCheckerService : ITransientDependency
public List Versions { get; set; }
}
+ public class LeptonXThemeInfo
+ {
+ public string Version { get; set; }
+ }
+
public class LatestStableVersionResult
{
public string Version { get; set; }
@@ -279,5 +298,8 @@ public class PackageVersionCheckerService : ITransientDependency
public string Type { get; set; }
public string Message { get; set; }
+
+ [CanBeNull]
+ public LeptonXThemeInfo LeptonX { get; set; }
}
}
diff --git a/framework/src/Volo.Abp.EntityFrameworkCore.Oracle/Volo.Abp.EntityFrameworkCore.Oracle.csproj b/framework/src/Volo.Abp.EntityFrameworkCore.Oracle/Volo.Abp.EntityFrameworkCore.Oracle.csproj
index 2948ef0cfb..8f14308ed6 100644
--- a/framework/src/Volo.Abp.EntityFrameworkCore.Oracle/Volo.Abp.EntityFrameworkCore.Oracle.csproj
+++ b/framework/src/Volo.Abp.EntityFrameworkCore.Oracle/Volo.Abp.EntityFrameworkCore.Oracle.csproj
@@ -21,7 +21,6 @@
-
diff --git a/framework/src/Volo.Abp.EventBus/Volo/Abp/EventBus/Distributed/DistributedEventBusBase.cs b/framework/src/Volo.Abp.EventBus/Volo/Abp/EventBus/Distributed/DistributedEventBusBase.cs
index 72a20748c6..96a11f928e 100644
--- a/framework/src/Volo.Abp.EventBus/Volo/Abp/EventBus/Distributed/DistributedEventBusBase.cs
+++ b/framework/src/Volo.Abp.EventBus/Volo/Abp/EventBus/Distributed/DistributedEventBusBase.cs
@@ -132,7 +132,13 @@ public abstract class DistributedEventBusBase : EventBusBase, IDistributedEventB
Serialize(eventData),
Clock.Now
);
- outgoingEventInfo.SetCorrelationId(CorrelationIdProvider.Get()!);
+
+ var correlationId = CorrelationIdProvider.Get();
+ if (correlationId != null)
+ {
+ outgoingEventInfo.SetCorrelationId(correlationId);
+ }
+
await eventOutbox.EnqueueAsync(outgoingEventInfo);
return true;
}
diff --git a/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/AbpHttpClientOptions.cs b/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/AbpHttpClientOptions.cs
index 6743ab5174..bfe52508e4 100644
--- a/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/AbpHttpClientOptions.cs
+++ b/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/AbpHttpClientOptions.cs
@@ -1,5 +1,7 @@
using System;
using System.Collections.Generic;
+using System.Net.Http;
+using Volo.Abp.Http.Client.ClientProxying;
using Volo.Abp.Http.Client.Proxying;
namespace Volo.Abp.Http.Client;
@@ -8,8 +10,17 @@ public class AbpHttpClientOptions
{
public Dictionary HttpClientProxies { get; set; }
+ public Dictionary>> ProxyHttpClientPreSendActions { get; }
+
public AbpHttpClientOptions()
{
HttpClientProxies = new Dictionary();
+ ProxyHttpClientPreSendActions = new Dictionary>>();
+ }
+
+ public AbpHttpClientOptions AddPreSendAction(string remoteServiceName, Action action)
+ {
+ ProxyHttpClientPreSendActions.GetOrAdd(remoteServiceName, () => new List>()).Add(action);
+ return this;
}
}
diff --git a/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/ClientProxying/ClientProxyBase.cs b/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/ClientProxying/ClientProxyBase.cs
index a4306050a4..01c1e2a720 100644
--- a/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/ClientProxying/ClientProxyBase.cs
+++ b/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/ClientProxying/ClientProxyBase.cs
@@ -146,6 +146,11 @@ public class ClientProxyBase : ITransientDependency
HttpResponseMessage response;
try
{
+ foreach (var preSendAction in ClientOptions.Value.ProxyHttpClientPreSendActions.Where(x => x.Key == clientConfig.RemoteServiceName).SelectMany(x => x.Value))
+ {
+ preSendAction(clientConfig, requestContext, client);
+ }
+
response = await client.SendAsync(
requestMessage,
HttpCompletionOption.ResponseHeadersRead /*this will buffer only the headers, the content will be used as a stream*/,
@@ -309,7 +314,11 @@ public class ClientProxyBase : ITransientDependency
}
//CorrelationId
- requestMessage.Headers.Add(AbpCorrelationIdOptions.Value.HttpHeaderName, CorrelationIdProvider.Get());
+ var correlationId = CorrelationIdProvider.Get();
+ if (correlationId != null)
+ {
+ requestMessage.Headers.Add(AbpCorrelationIdOptions.Value.HttpHeaderName, correlationId);
+ }
//TenantId
if (CurrentTenant.Id.HasValue)
diff --git a/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/DynamicProxying/ApiDescriptionFinder.cs b/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/DynamicProxying/ApiDescriptionFinder.cs
index 10089c52cd..e54e24ced0 100644
--- a/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/DynamicProxying/ApiDescriptionFinder.cs
+++ b/framework/src/Volo.Abp.Http.Client/Volo/Abp/Http/Client/DynamicProxying/ApiDescriptionFinder.cs
@@ -126,7 +126,11 @@ public class ApiDescriptionFinder : IApiDescriptionFinder, ITransientDependency
protected virtual void AddHeaders(HttpRequestMessage requestMessage)
{
//CorrelationId
- requestMessage.Headers.Add(AbpCorrelationIdOptions.HttpHeaderName, CorrelationIdProvider.Get());
+ var correlationId = CorrelationIdProvider.Get();
+ if (correlationId != null)
+ {
+ requestMessage.Headers.Add(AbpCorrelationIdOptions.HttpHeaderName, correlationId);
+ }
//TenantId
if (CurrentTenant.Id.HasValue)
diff --git a/framework/src/Volo.Abp.Swashbuckle/Microsoft/AspNetCore/Builder/AbpSwaggerUIBuilderExtensions.cs b/framework/src/Volo.Abp.Swashbuckle/Microsoft/AspNetCore/Builder/AbpSwaggerUIBuilderExtensions.cs
index f7d85e3256..b999005f25 100644
--- a/framework/src/Volo.Abp.Swashbuckle/Microsoft/AspNetCore/Builder/AbpSwaggerUIBuilderExtensions.cs
+++ b/framework/src/Volo.Abp.Swashbuckle/Microsoft/AspNetCore/Builder/AbpSwaggerUIBuilderExtensions.cs
@@ -16,7 +16,6 @@ public static class AbpSwaggerUIBuilderExtensions
return app.UseSwaggerUI(options =>
{
options.InjectJavascript("ui/abp.js");
- options.InjectJavascript("ui/abp.swagger.js");
options.IndexStream = () => resolver?.Resolver();
setupAction?.Invoke(options);
diff --git a/framework/src/Volo.Abp.Swashbuckle/Volo/Abp/Swashbuckle/SwaggerHtmlResolver.cs b/framework/src/Volo.Abp.Swashbuckle/Volo/Abp/Swashbuckle/SwaggerHtmlResolver.cs
index 628a361826..be4ebfe4e8 100644
--- a/framework/src/Volo.Abp.Swashbuckle/Volo/Abp/Swashbuckle/SwaggerHtmlResolver.cs
+++ b/framework/src/Volo.Abp.Swashbuckle/Volo/Abp/Swashbuckle/SwaggerHtmlResolver.cs
@@ -10,12 +10,14 @@ public class SwaggerHtmlResolver : ISwaggerHtmlResolver, ITransientDependency
{
public virtual Stream Resolver()
{
+ var scriptBundleScript = "";
+ var abpSwaggerScript = "";
var stream = typeof(SwaggerUIOptions).GetTypeInfo().Assembly
.GetManifestResourceStream("Swashbuckle.AspNetCore.SwaggerUI.index.html");
var html = new StreamReader(stream!)
.ReadToEnd()
- .Replace("SwaggerUIBundle(configObject)", "abp.SwaggerUIBundle(configObject)");
+ .Replace(scriptBundleScript, $"{scriptBundleScript}\n{abpSwaggerScript}");
return new MemoryStream(Encoding.UTF8.GetBytes(html));
}
diff --git a/framework/src/Volo.Abp.Swashbuckle/wwwroot/swagger/ui/abp.swagger.js b/framework/src/Volo.Abp.Swashbuckle/wwwroot/swagger/ui/abp.swagger.js
index 4e8790d38c..b8abb0df25 100644
--- a/framework/src/Volo.Abp.Swashbuckle/wwwroot/swagger/ui/abp.swagger.js
+++ b/framework/src/Volo.Abp.Swashbuckle/wwwroot/swagger/ui/abp.swagger.js
@@ -1,9 +1,10 @@
var abp = abp || {};
(function () {
-
- abp.SwaggerUIBundle = function (configObject) {
-
+
+ var oldSwaggerUIBundle = SwaggerUIBundle;
+
+ SwaggerUIBundle = function (configObject) {
var excludeUrl = ["swagger.json", "connect/token"]
var firstRequest = true;
var oidcSupportedFlows = configObject.oidcSupportedFlows || [];
@@ -109,6 +110,9 @@ var abp = abp || {};
});
}
- return SwaggerUIBundle(configObject);
+ return oldSwaggerUIBundle(configObject);
}
+
+ SwaggerUIBundle = Object.assign(SwaggerUIBundle, oldSwaggerUIBundle);
+
})();
diff --git a/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/AbpHttpClientTestModule.cs b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/AbpHttpClientTestModule.cs
index 5e9107a23b..4dd69041fc 100644
--- a/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/AbpHttpClientTestModule.cs
+++ b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/AbpHttpClientTestModule.cs
@@ -1,8 +1,11 @@
-using System.Collections.Generic;
+using System;
+using System.Collections.Generic;
+using System.Net.Http;
using Microsoft.Extensions.DependencyInjection;
using Volo.Abp.AspNetCore.Mvc;
using Volo.Abp.Http.Client;
using Volo.Abp.Http.Client.ClientProxying;
+using Volo.Abp.Http.Client.Proxying;
using Volo.Abp.Http.DynamicProxying;
using Volo.Abp.Http.Localization;
using Volo.Abp.Localization;
@@ -60,5 +63,16 @@ public class AbpHttpClientTestModule : AbpModule
options.FormDataConverts.Add(typeof(List), typeof(TestObjectToFormData));
options.PathConverts.Add(typeof(int), typeof(TestObjectToPath));
});
+
+ Configure(options =>
+ {
+ options.AddPreSendAction("Default", (_, requestContext, httpclient) =>
+ {
+ if (requestContext.Action.Name.Equals("TimeOutRequestAsync"))
+ {
+ httpclient.Timeout = TimeSpan.FromMilliseconds(1);
+ }
+ });
+ });
}
}
diff --git a/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/IRegularTestController.cs b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/IRegularTestController.cs
index 547f1bf5a7..21786155bf 100644
--- a/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/IRegularTestController.cs
+++ b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/IRegularTestController.cs
@@ -43,4 +43,7 @@ public interface IRegularTestController
Task DeleteByIdAsync(int id);
Task AbortRequestAsync(CancellationToken cancellationToken = default);
+
+ Task TimeOutRequestAsync();
+
}
diff --git a/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/RegularTestController.cs b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/RegularTestController.cs
index b092e67bc8..46b355090b 100644
--- a/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/RegularTestController.cs
+++ b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/RegularTestController.cs
@@ -152,6 +152,14 @@ public class RegularTestController : AbpController, IRegularTestController
await Task.Delay(100, cancellationToken);
return "AbortRequestAsync";
}
+
+ [HttpGet]
+ [Route("timeout-request")]
+ public async Task TimeOutRequestAsync()
+ {
+ await Task.Delay(100);
+ return "TimeOutRequestAsync";
+ }
}
public class Car
diff --git a/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/RegularTestControllerClientProxy_Tests.cs b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/RegularTestControllerClientProxy_Tests.cs
index f094c756a9..01ec97d734 100644
--- a/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/RegularTestControllerClientProxy_Tests.cs
+++ b/framework/test/Volo.Abp.Http.Client.Tests/Volo/Abp/Http/DynamicProxying/RegularTestControllerClientProxy_Tests.cs
@@ -1,4 +1,5 @@
using System;
+using System.Net.Http;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.Extensions.DependencyInjection;
@@ -187,4 +188,11 @@ public class RegularTestControllerClientProxy_Tests : AbpHttpClientTestBase
var exception = await Assert.ThrowsAsync(async () => await _controller.AbortRequestAsync(cts.Token));
exception.InnerException.InnerException.InnerException.Message.ShouldBe("The client aborted the request.");
}
+
+ [Fact]
+ public async Task TimeOutRequestAsync()
+ {
+ var exception = await Assert.ThrowsAsync(async () => await _controller.TimeOutRequestAsync());
+ exception.InnerException.InnerException.Message.ShouldBe("The client aborted the request.");
+ }
}
diff --git a/latest-versions.json b/latest-versions.json
index b99c7202b6..e2c60800a3 100644
--- a/latest-versions.json
+++ b/latest-versions.json
@@ -1,8 +1,29 @@
[
+ {
+ "version": "9.0.1",
+ "releaseDate": "",
+ "type": "stable",
+ "message": "",
+ "leptonx": {
+ "version": "4.0.1"
+ }
+ },
{
"version": "9.0.0",
"releaseDate": "",
"type": "stable",
- "message": ""
+ "message": "",
+ "leptonx": {
+ "version": "4.0.0"
+ }
+ },
+ {
+ "version": "8.3.1",
+ "releaseDate": "",
+ "type": "stable",
+ "message": "",
+ "leptonx": {
+ "version": "3.3.1"
+ }
}
]
\ No newline at end of file
diff --git a/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.Demo/package.json b/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.Demo/package.json
index 04d78b830f..fdd7728237 100644
--- a/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.Demo/package.json
+++ b/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.Demo/package.json
@@ -3,8 +3,8 @@
"name": "asp.net",
"private": true,
"dependencies": {
- "@abp/aspnetcore.mvc.ui.theme.shared": "~9.0.0",
- "@abp/prismjs": "~9.0.0",
- "@abp/highlight.js": "~9.0.0"
+ "@abp/aspnetcore.mvc.ui.theme.shared": "~9.0.1",
+ "@abp/prismjs": "~9.0.1",
+ "@abp/highlight.js": "~9.0.1"
}
}
diff --git a/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.Demo/yarn.lock b/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.Demo/yarn.lock
index 0ec23ef2e3..842e1aadb4 100644
--- a/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.Demo/yarn.lock
+++ b/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.Demo/yarn.lock
@@ -2,212 +2,212 @@
# yarn lockfile v1
-"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.0.tgz#a77b4b70fa6217146afd6a57211f57e2360564f2"
- integrity sha512-QmZrSp+rfqfnC9W4QZ+nNdAq0rOALzRn1rsDpGDHhIHwyXB03QPdjzIWaEvldUYHzqg84cEP0LcqW4y57Xr1iw==
- dependencies:
- "@abp/aspnetcore.mvc.ui" "~9.0.0"
- "@abp/bootstrap" "~9.0.0"
- "@abp/bootstrap-datepicker" "~9.0.0"
- "@abp/bootstrap-daterangepicker" "~9.0.0"
- "@abp/datatables.net-bs5" "~9.0.0"
- "@abp/font-awesome" "~9.0.0"
- "@abp/jquery-form" "~9.0.0"
- "@abp/jquery-validation-unobtrusive" "~9.0.0"
- "@abp/lodash" "~9.0.0"
- "@abp/luxon" "~9.0.0"
- "@abp/malihu-custom-scrollbar-plugin" "~9.0.0"
- "@abp/moment" "~9.0.0"
- "@abp/select2" "~9.0.0"
- "@abp/sweetalert2" "~9.0.0"
- "@abp/timeago" "~9.0.0"
- "@abp/toastr" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.0.tgz#93004dd056c130386c7b40c458f3a793d8c9e051"
- integrity sha512-pvO5bh9baFcSbfW0s7tpMpbe6DKSC16jeyeluZLlvYxC0n2gkrv7H/hB5aMuBKIlh89aGY9DQIcMFC8iczL3Bg==
+"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.1.tgz#7a1be780a305aa0d0f467472fc721b6f710fda7a"
+ integrity sha512-vrgRYKNctdZYhPCQ2I6Z10gUI1S5ynDj+DgnV7x7a7G/juuHaOSbkRXgwrmJsmYLfGA7KFCoJcbpER8mmXQeHQ==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui" "~9.0.1"
+ "@abp/bootstrap" "~9.0.1"
+ "@abp/bootstrap-datepicker" "~9.0.1"
+ "@abp/bootstrap-daterangepicker" "~9.0.1"
+ "@abp/datatables.net-bs5" "~9.0.1"
+ "@abp/font-awesome" "~9.0.1"
+ "@abp/jquery-form" "~9.0.1"
+ "@abp/jquery-validation-unobtrusive" "~9.0.1"
+ "@abp/lodash" "~9.0.1"
+ "@abp/luxon" "~9.0.1"
+ "@abp/malihu-custom-scrollbar-plugin" "~9.0.1"
+ "@abp/moment" "~9.0.1"
+ "@abp/select2" "~9.0.1"
+ "@abp/sweetalert2" "~9.0.1"
+ "@abp/timeago" "~9.0.1"
+ "@abp/toastr" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.1.tgz#811dd91138efb82345fc06559b5d2feea88021af"
+ integrity sha512-KVbWCpBV7S1ViHYQP/hn4gSMzSiA2e8/zC0HND1fjE+1zXJqJyEWto1eg2DaKSOs/rmHD8WtFVMU4hCLQTCuIA==
dependencies:
ansi-colors "^4.1.3"
-"@abp/bootstrap-datepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.0.tgz#d6609e50568aac6a013abfcaa07817998e1eaa99"
- integrity sha512-wxAeVsOVOJpidDn0g3FlJvpOsn5SVY5IpEG+FtUxSAKuWYJQWF6g5CaghrfS+FNDK9IHhxrgSMPMVBK9YlDSzw==
+"@abp/bootstrap-datepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.1.tgz#21cd19090b9a571ddf5a9019bc17864c8685408c"
+ integrity sha512-LhgqBRr66x9gGTAyv1FuPAw8u8N77WQK2AfDkNgLueurM2+xMYQzH3pWWLhcN676/JXafxfALCfBoM8U7CfXLg==
dependencies:
bootstrap-datepicker "^1.10.0"
-"@abp/bootstrap-daterangepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.0.tgz#fc2871e7d886accd8e4748cb763b695e10701c00"
- integrity sha512-qS6SQR6X7wMCVLcuXatJ2+m1GNlpN71p7VDritxfwAMHa7WBRAAi8apX2WItwBAuLQ2s1uSA0Wp1KstHG3VPLA==
+"@abp/bootstrap-daterangepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.1.tgz#c19c23ede6fd868a147ea893d4fdecfa60b177f7"
+ integrity sha512-SlBVgiYa75uQyaGt5EVjduren7Po5UP26A1tXEQQr0hDia1+1TK0qxmJ2YxjeJpB7FwtcmJiofv5fngvJqhxIA==
dependencies:
bootstrap-daterangepicker "^3.1.0"
-"@abp/bootstrap@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.0.tgz#5ab292e95ba7eef129cbaa1dc5f5d5c0af8530d7"
- integrity sha512-jEr6su8BfoNxYvze/UlddE3DlezyPvQeiNhff8SDPqtRQ6XvjZ48O/JwtY01hK2njEDLOZINY5zJRGmO2OiOUw==
+"@abp/bootstrap@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.1.tgz#2360303ec24c8f3eb755b10fa6a953db6d858daf"
+ integrity sha512-uAfVNsUYSsRqgeO9JMnhJJlFoDmO3/lTvER/iYOqerF0X03KQnR8HZy2V3MD1D+fSh/4JuU6aAZFl6IQLinMKQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
bootstrap "^5.3.3"
-"@abp/clipboard@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/clipboard/-/clipboard-9.0.0.tgz#21ae924dfef6ccf3c8a3913bb81caf22b57404e4"
- integrity sha512-W6S7pihpQ6Wfj48Clk7hYgDvUtnRwkNIsk5t8s9+LxoAdJQGQFxh7SAW1QNavUv8QqsClNE1VtSU87c7rSoAbA==
+"@abp/clipboard@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/clipboard/-/clipboard-9.0.1.tgz#b1712b2d940caecc85eb99247a569ffe3ab1a387"
+ integrity sha512-obDYqUTf+65E26yHZO2DigQOeDnzAEvak0Umz91cholZHVRvzc9X1itv/abS/whU0YOU/d8Viud8zFv1MnM83w==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
clipboard "^2.0.11"
-"@abp/core@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.0.tgz#dc5ea9139957e307f73ac1f40b179f991e3c2168"
- integrity sha512-vzQyTpqVDx63pOI9e2QLUMWZCrSojHsRnzLlvl+/cKFi7jE1q/uSP66FPR2bwkD+DNLI/tn6is1Jx3skKA1FvA==
+"@abp/core@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.1.tgz#727fc1dfac21cca1216c4f5091da7d5efefdb467"
+ integrity sha512-NWXDVCIJePbcKCrRvGxAlaA2MrOF2IZox37bQkW7/mTeOPpNoTa6lhowksNNcTpFq2YdD8bG0i0kfskkp3EZ2w==
dependencies:
- "@abp/utils" "~9.0.0"
+ "@abp/utils" "~9.0.1"
-"@abp/datatables.net-bs5@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.0.tgz#4daee90bdb6446e854ce65f2a3afabb2a13e49ae"
- integrity sha512-ln10tcRwxcB0nuSgqJ/66strHsqMHDC03c0ktxEplAN34yP7KInjFf/U9Svo74K0nKlkJdMCgmqmT4CnBJZlIg==
+"@abp/datatables.net-bs5@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.1.tgz#61e2a3daacf1c074ed6b459e25084cc1f5c4eeab"
+ integrity sha512-tAHdf4TBAS0bVRcvovDKS0OHFyeRSPYbiSIwDn8AIWzW9Weav31jhZ5jL3HEtBKVdb7zqUQ8/3dYPufjDuDdDA==
dependencies:
- "@abp/datatables.net" "~9.0.0"
+ "@abp/datatables.net" "~9.0.1"
datatables.net-bs5 "^2.1.8"
-"@abp/datatables.net@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.0.tgz#82b28518293c9091ec72d5defae2f5d2903d8bbc"
- integrity sha512-QhuX3bGZOckM/fhIIGj/StMJIX2BBtNbLzjTR7WnKMMxDkv0+oDvjw6OqqpYKSvj/q3vY6sRWuCNdEzNi0Fk6A==
+"@abp/datatables.net@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.1.tgz#4cf193e7493f85a272de47326a32a021e3ba40f5"
+ integrity sha512-JY53zVQPluRrBiKwtgk4gfSgD8VDs/VU8LWizhGCprXt3h9vXx0SQL4My6zvXVkv4hUjaq4+MYo84OI8EwFvsg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
datatables.net "^2.1.8"
-"@abp/font-awesome@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.0.tgz#51cce18434a51d050ebb0d822a8c193b7e32f73d"
- integrity sha512-bEHRVzOk0V2kFMAvB/YXFYaDKWRFO0+ScO7jrEFtFgdHXBcqy+dk1aM/N69WalyNmjRRpnh03GhH1LRpwL4/xA==
+"@abp/font-awesome@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.1.tgz#69da15018e7e5f2ff0d3c0d113a4325f54ffc0c3"
+ integrity sha512-7NnYBmk0plLVElUWR1zohul87vE9wKJItemWT5Kskf/IJirKOvXwSSu+7Jg1LmnVtIxcH6qgRpJMmzuAJgawnQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
"@fortawesome/fontawesome-free" "^6.6.0"
-"@abp/highlight.js@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/highlight.js/-/highlight.js-9.0.0.tgz#6d924c9bb3d6b11dcd66a9a71aea873c9866ed6c"
- integrity sha512-eGvgMUNYKW2B2h3QcpMRniaoQ1/9RJBXkn2VsyM8lVSfrfbOg26mMyw7J2BXvuEnXqtnVdFEG45cSFlAnTAVXQ==
+"@abp/highlight.js@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/highlight.js/-/highlight.js-9.0.1.tgz#22f135e73d5ee5ce14e31fe6c1a1b3eee6a91f55"
+ integrity sha512-K1EXCtRsc116c7UYcUXRVmrPFHDY4531+ij9P32EdP1/3AIwm50r+tz2epZV2GyIuGw/mVWyFB0vduqFlr0W3g==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
"@highlightjs/cdn-assets" "~11.10.0"
-"@abp/jquery-form@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.0.tgz#6518eb117ee35c6bc57543fd1fccb0a8cf8e22f4"
- integrity sha512-6qpfPGHrjADmm7Ne3zcXBGPuLngG8ldAn5fk9k2taftpyX0iR67/nZxiQ0ujaQGgaai5Wb3imNXMG96XLnubkQ==
+"@abp/jquery-form@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.1.tgz#35e0a23cf50b00e0d3b0886f097233b79144145b"
+ integrity sha512-9co0+Cver+RUSOpfuukSWQhR+TCnO9Dly4RTVRIKFksGVfN8hA70NqRrVxaASNNFv42ynK88FQK+2bLPswz/iw==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-form "^4.3.0"
-"@abp/jquery-validation-unobtrusive@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.0.tgz#8ee87dbdec5e92ab04ae9642a49af1598af31c1a"
- integrity sha512-KlRQ1htyFmAIMS9xbHvHIcdhAgtPFe/u7c87MiWlk/CdhDrI3J+n5biLvQUzxCC+YWlwx27zBtPKx2vLeMRH6Q==
+"@abp/jquery-validation-unobtrusive@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.1.tgz#f9564aabebf4d9f90b4ee5c3f3de0e1a348d8ecc"
+ integrity sha512-Zpz48wIFjq7eIApG/sEVjkfaJWZhcyCnlCtDf5eU3vvh82nRetbiCEAVLpZ4LVjorlyp3VvVAxReUSeeWflu8A==
dependencies:
- "@abp/jquery-validation" "~9.0.0"
+ "@abp/jquery-validation" "~9.0.1"
jquery-validation-unobtrusive "^4.0.0"
-"@abp/jquery-validation@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.0.tgz#681d8037c57b3875bd51c927e350ffce65d7c5d2"
- integrity sha512-XgCUBFWbhTDcGepD/xkwA/1Hl4TyYitRttauDJk3tuyXhNtykvdugCCOPY4j2s6VoT1Sb/TFFvOcgEvlBwkKiw==
+"@abp/jquery-validation@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.1.tgz#ad37a44e82fb32ec9bfe2b736dfc8a143fca89c7"
+ integrity sha512-uFSxiPMEkaPlcuyXIoKbeOIIRHo4IrgRbmnlHwI1uETrW60ENUdE2LJAVcYS77JROrjVtv8FPlaffWuva6sGeQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-validation "^1.21.0"
-"@abp/jquery@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.0.tgz#6c67a9f927be598ab6201f8888e8a3cb8bba0686"
- integrity sha512-cUXA7v6F1HqAc/w8fQGxHIfA/6BqO+4uHXDnSsJZtrI26DTLOgw5kBNnAz2W4xggGPCY7SKFHNRZuOW/QfVRPg==
+"@abp/jquery@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.1.tgz#ecce14644c753c581caee0df7bfafd3d083e70f0"
+ integrity sha512-z8EcnOveItAszbTsM9GFFDQSlcIg/ym6+WZME9uldLoetdxkmI9ij0knVeGj5eqMjQaHvTWwrVLObiBbZwYFWw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
jquery "~3.7.1"
-"@abp/lodash@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.0.tgz#a4d42b2dfeb37859ff9f0079c7a8a6cabc982efe"
- integrity sha512-xoAsOkkgnQdPH2OyFTgF6OlCLVbyGF4u8CrFcZWCZmuIFT9cg0+ZEiq//Vog1pg3Lf//N37ceGpOag1LwMe6cQ==
+"@abp/lodash@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.1.tgz#c52c8a4125ecfbeaad7badc5c28bf86d5168799a"
+ integrity sha512-D+/iOyh/O+jV80bK5vWxyn++IVmeRbZq/UaXanA4RKpzigxJUsgOiSD9lUj85BkfKn2YyFmoR39xwaAqkWoyRA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
lodash "^4.17.21"
-"@abp/luxon@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.0.tgz#a54dda13cbbd362481994d20d2e07c669c6ce109"
- integrity sha512-aPejQnoVPfm60LnjwhfUkIsB+poNWusrBE0IHllxhvOVE+01nfa1Lvpp5AUWzV+XchPGOCSf+a29ds4EQWfNbg==
+"@abp/luxon@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.1.tgz#ac59a56653d51b997dbbde8ccc58dac5746a9ef4"
+ integrity sha512-1ZHXP5SGz05tQ1tzYmPcRiZwmkCKGX0KWFoV0zF3ATRaevzku9nXvSy7K6GIFkkgTb9f9XNx4/OBBF/GXkuRqw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
luxon "^3.5.0"
-"@abp/malihu-custom-scrollbar-plugin@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.0.tgz#b8a49202d531c711874a2bcdae5081ae0ed34c3f"
- integrity sha512-iSZ9lKMnv5F2SplR+A4blhFEDq79lKbvBQuS7h/Di3KEHFOdJY4pA8NP8OR+PXOF3JfAezGyjqy7ydn/JNPRwA==
+"@abp/malihu-custom-scrollbar-plugin@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.1.tgz#9e60bcfb9f9d76fed9c50e45bf490d5abd0f4689"
+ integrity sha512-GkKPLg9Z9UTTSSpW2hYZYZ9IgCvDSOYJmyRSl/h4fG8cWMgGCPVpv8MPPu0+WtIfwW/mT4SPX9X+sjUlRL5XRQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
malihu-custom-scrollbar-plugin "^3.1.5"
-"@abp/moment@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.0.tgz#9583e97472ddf608a9a5a6b8a37fee602b419900"
- integrity sha512-wgyJSKmBmHLXZJN3dy3zK4i2JvWGgqnvdtrKURy/raEurxlymv8JF5JA971kTv6Kz2OA/OSYttpVJkDGUpPAhA==
+"@abp/moment@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.1.tgz#d67fa25017fe15dae10e0a0fcebb03aeb63d31d3"
+ integrity sha512-bTuCc7pi/DCtgz7t96UBU0wn9UGt4b50JhDCVeTKquh+gP1sgWoI5hyvEBkmgE4dcffO0yzZU2opHYTKYGl/OQ==
dependencies:
moment "^2.30.1"
-"@abp/prismjs@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/prismjs/-/prismjs-9.0.0.tgz#2ac4a5ddc0c8e0462af027a4dd97a177a3d6f7a5"
- integrity sha512-+evmBq5qV+Ix/TNqO2ljJn0jygW2rEgKNtABcBGmrZC8xiI7+2hjKmSE0e43qb+RDib8aJjwH8ya8wF7p68ZFw==
+"@abp/prismjs@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/prismjs/-/prismjs-9.0.1.tgz#2693014c8161d561e7c7bef01d5cd24ee53be1b3"
+ integrity sha512-NwW+vpTiAsuJYNC/MEiAj0dGMWml95mAQD1Z0Djfsg9D6OpcLsdlgWX+KVdZvHoJ+SPgTDIqdt/0LnIv1w/XdA==
dependencies:
- "@abp/clipboard" "~9.0.0"
- "@abp/core" "~9.0.0"
+ "@abp/clipboard" "~9.0.1"
+ "@abp/core" "~9.0.1"
prismjs "^1.29.0"
-"@abp/select2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.0.tgz#71503eebb746f40a0e565dba1efa05d6ae937b77"
- integrity sha512-l+Hz2Qmc7AcyaTJOyuDaV99RbBWYxYxbeyVEc4xKbRFbJxYBUxT/DFlyozEaoz+OU1fl4szO3UCthJ2yZvxCdg==
+"@abp/select2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.1.tgz#9f1c5c27e72bef910fdf63fc3167f415d2ba8db8"
+ integrity sha512-epyYJH2gAOU2UvGn5FD6372hbW3fL7uZIgGpvfgEfa+l6HPdlV9B1OweU2xkONNCoylShAcP/y8qf+bUh8WDEA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
select2 "^4.0.13"
-"@abp/sweetalert2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.0.tgz#71e511117c18b515e5c658e9cc0667770af65d91"
- integrity sha512-wxDqoA6RUhkbcN27Kzslu/m2ai5T9AkJRtgs7pV55LvPv1ohuCPEUeTVVh8S/5Pgmr7aPFAPxaaCO32aPcImPw==
+"@abp/sweetalert2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.1.tgz#a364b21665cb76e92640496dc8a77b8edbce4500"
+ integrity sha512-0RXR/lzy3qMJLisGUz4qrcA9Clu/Tt1ABGzB64ynG1t/l5XessQsIB5Hkt6MbSyOEsprRsq+ZtKH35CwTRPdAw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
sweetalert2 "^11.14.1"
-"@abp/timeago@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.0.tgz#8f262ba2a74e1c133d0c129f8a0bc55c29441e1a"
- integrity sha512-eaNdIUEtDXnuYqS5O7EGRV+j6XOLAsxz7gd5BIKR9lDujd8pEjSJkXKO+/0KxLTkUSy3JudIO37K1h2wTpbYZA==
+"@abp/timeago@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.1.tgz#d1b7e11fdc267bdb736ff4b591afd3fb81c07cce"
+ integrity sha512-dNfUQwilrY+JHxL5lwJ++ixYDGAs5lF1EDxKMQMIXD/ak5L8rkdLns8Otta7FhgDDny9UJ7XWQjYsG8ZxxybRQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
timeago "^1.6.7"
-"@abp/toastr@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.0.tgz#74d74ecb3e7f2d63b2ff7735c89a40deb3dc3a13"
- integrity sha512-n9uJvw8l2HoxeowPO4diEMxwchGh6puefhyRljFAAstzK77Mhn0u2J7kPynNv94rLSYTYoCxUeJ4TTN+/TVDGQ==
+"@abp/toastr@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.1.tgz#2105a639d7cd4fa9c671b4215d719a78fef7e0ee"
+ integrity sha512-10KuHnSJOxkvzqs6Aim6GJp8xBuT+TsZqBE1KaXQnERPawmOo4+ME28kA/7lq8fvgFUNO3K41T2uuA1GPHVQcg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
toastr "^2.1.4"
-"@abp/utils@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.0.tgz#2264fd83105b061860edd98dbe0efdfe55803ed5"
- integrity sha512-/SB4VYk3Av+ltXv6ovkFHG4iWiECLLitaYWcEHxVAO80ufNjrNnWpHFpXHP8u/RpXIp/CoNiHLr6j5TkZUTZjw==
+"@abp/utils@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.1.tgz#dccaf7cf92147304d454ffb9a8664d52dc7e5c5a"
+ integrity sha512-7eOKT0ciKNdtuASck1kifyc4dC6CeebgxV+rx/EAi3YwPMDkqie6YwLGUqKdGW2N/MZ15JuDR6gkuZxKHEb0NQ==
dependencies:
just-compare "^2.3.0"
diff --git a/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic.Demo/package.json b/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic.Demo/package.json
index 8857cdc4b9..dbf6946a98 100644
--- a/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic.Demo/package.json
+++ b/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic.Demo/package.json
@@ -3,8 +3,8 @@
"name": "asp.net",
"private": true,
"dependencies": {
- "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.0",
- "@abp/prismjs": "~9.0.0"
+ "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.1",
+ "@abp/prismjs": "~9.0.1"
},
"devDependencies": {}
}
diff --git a/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic.Demo/yarn.lock b/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic.Demo/yarn.lock
index 01d7637c9e..cf7895df68 100644
--- a/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic.Demo/yarn.lock
+++ b/modules/basic-theme/test/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic.Demo/yarn.lock
@@ -2,211 +2,211 @@
# yarn lockfile v1
-"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.0.tgz#e4b05d63446c24b612275e3949881e176d32527f"
- integrity sha512-GHEt26xESe/TCg1q6ZjeUbxniqdxaiB1OPTZ9md7ONql6bEp8t9iUPeSlkyAZtQ7qBzwgkiLpkRFT+J21DftaQ==
- dependencies:
- "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.0.tgz#a77b4b70fa6217146afd6a57211f57e2360564f2"
- integrity sha512-QmZrSp+rfqfnC9W4QZ+nNdAq0rOALzRn1rsDpGDHhIHwyXB03QPdjzIWaEvldUYHzqg84cEP0LcqW4y57Xr1iw==
- dependencies:
- "@abp/aspnetcore.mvc.ui" "~9.0.0"
- "@abp/bootstrap" "~9.0.0"
- "@abp/bootstrap-datepicker" "~9.0.0"
- "@abp/bootstrap-daterangepicker" "~9.0.0"
- "@abp/datatables.net-bs5" "~9.0.0"
- "@abp/font-awesome" "~9.0.0"
- "@abp/jquery-form" "~9.0.0"
- "@abp/jquery-validation-unobtrusive" "~9.0.0"
- "@abp/lodash" "~9.0.0"
- "@abp/luxon" "~9.0.0"
- "@abp/malihu-custom-scrollbar-plugin" "~9.0.0"
- "@abp/moment" "~9.0.0"
- "@abp/select2" "~9.0.0"
- "@abp/sweetalert2" "~9.0.0"
- "@abp/timeago" "~9.0.0"
- "@abp/toastr" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.0.tgz#93004dd056c130386c7b40c458f3a793d8c9e051"
- integrity sha512-pvO5bh9baFcSbfW0s7tpMpbe6DKSC16jeyeluZLlvYxC0n2gkrv7H/hB5aMuBKIlh89aGY9DQIcMFC8iczL3Bg==
+"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.1.tgz#cdb6130ac6f8499f9a3f856b596c46745d056a30"
+ integrity sha512-JaxuDH8WTnZYX6C7CX7ZVHfhTDR/SxfwRvtJaGGijSX8QKd1DXpfi9lYEF4Y9k5b48YT/0slBqU5g0sLmBd3+g==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.1.tgz#7a1be780a305aa0d0f467472fc721b6f710fda7a"
+ integrity sha512-vrgRYKNctdZYhPCQ2I6Z10gUI1S5ynDj+DgnV7x7a7G/juuHaOSbkRXgwrmJsmYLfGA7KFCoJcbpER8mmXQeHQ==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui" "~9.0.1"
+ "@abp/bootstrap" "~9.0.1"
+ "@abp/bootstrap-datepicker" "~9.0.1"
+ "@abp/bootstrap-daterangepicker" "~9.0.1"
+ "@abp/datatables.net-bs5" "~9.0.1"
+ "@abp/font-awesome" "~9.0.1"
+ "@abp/jquery-form" "~9.0.1"
+ "@abp/jquery-validation-unobtrusive" "~9.0.1"
+ "@abp/lodash" "~9.0.1"
+ "@abp/luxon" "~9.0.1"
+ "@abp/malihu-custom-scrollbar-plugin" "~9.0.1"
+ "@abp/moment" "~9.0.1"
+ "@abp/select2" "~9.0.1"
+ "@abp/sweetalert2" "~9.0.1"
+ "@abp/timeago" "~9.0.1"
+ "@abp/toastr" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.1.tgz#811dd91138efb82345fc06559b5d2feea88021af"
+ integrity sha512-KVbWCpBV7S1ViHYQP/hn4gSMzSiA2e8/zC0HND1fjE+1zXJqJyEWto1eg2DaKSOs/rmHD8WtFVMU4hCLQTCuIA==
dependencies:
ansi-colors "^4.1.3"
-"@abp/bootstrap-datepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.0.tgz#d6609e50568aac6a013abfcaa07817998e1eaa99"
- integrity sha512-wxAeVsOVOJpidDn0g3FlJvpOsn5SVY5IpEG+FtUxSAKuWYJQWF6g5CaghrfS+FNDK9IHhxrgSMPMVBK9YlDSzw==
+"@abp/bootstrap-datepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.1.tgz#21cd19090b9a571ddf5a9019bc17864c8685408c"
+ integrity sha512-LhgqBRr66x9gGTAyv1FuPAw8u8N77WQK2AfDkNgLueurM2+xMYQzH3pWWLhcN676/JXafxfALCfBoM8U7CfXLg==
dependencies:
bootstrap-datepicker "^1.10.0"
-"@abp/bootstrap-daterangepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.0.tgz#fc2871e7d886accd8e4748cb763b695e10701c00"
- integrity sha512-qS6SQR6X7wMCVLcuXatJ2+m1GNlpN71p7VDritxfwAMHa7WBRAAi8apX2WItwBAuLQ2s1uSA0Wp1KstHG3VPLA==
+"@abp/bootstrap-daterangepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.1.tgz#c19c23ede6fd868a147ea893d4fdecfa60b177f7"
+ integrity sha512-SlBVgiYa75uQyaGt5EVjduren7Po5UP26A1tXEQQr0hDia1+1TK0qxmJ2YxjeJpB7FwtcmJiofv5fngvJqhxIA==
dependencies:
bootstrap-daterangepicker "^3.1.0"
-"@abp/bootstrap@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.0.tgz#5ab292e95ba7eef129cbaa1dc5f5d5c0af8530d7"
- integrity sha512-jEr6su8BfoNxYvze/UlddE3DlezyPvQeiNhff8SDPqtRQ6XvjZ48O/JwtY01hK2njEDLOZINY5zJRGmO2OiOUw==
+"@abp/bootstrap@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.1.tgz#2360303ec24c8f3eb755b10fa6a953db6d858daf"
+ integrity sha512-uAfVNsUYSsRqgeO9JMnhJJlFoDmO3/lTvER/iYOqerF0X03KQnR8HZy2V3MD1D+fSh/4JuU6aAZFl6IQLinMKQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
bootstrap "^5.3.3"
-"@abp/clipboard@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/clipboard/-/clipboard-9.0.0.tgz#21ae924dfef6ccf3c8a3913bb81caf22b57404e4"
- integrity sha512-W6S7pihpQ6Wfj48Clk7hYgDvUtnRwkNIsk5t8s9+LxoAdJQGQFxh7SAW1QNavUv8QqsClNE1VtSU87c7rSoAbA==
+"@abp/clipboard@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/clipboard/-/clipboard-9.0.1.tgz#b1712b2d940caecc85eb99247a569ffe3ab1a387"
+ integrity sha512-obDYqUTf+65E26yHZO2DigQOeDnzAEvak0Umz91cholZHVRvzc9X1itv/abS/whU0YOU/d8Viud8zFv1MnM83w==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
clipboard "^2.0.11"
-"@abp/core@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.0.tgz#dc5ea9139957e307f73ac1f40b179f991e3c2168"
- integrity sha512-vzQyTpqVDx63pOI9e2QLUMWZCrSojHsRnzLlvl+/cKFi7jE1q/uSP66FPR2bwkD+DNLI/tn6is1Jx3skKA1FvA==
+"@abp/core@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.1.tgz#727fc1dfac21cca1216c4f5091da7d5efefdb467"
+ integrity sha512-NWXDVCIJePbcKCrRvGxAlaA2MrOF2IZox37bQkW7/mTeOPpNoTa6lhowksNNcTpFq2YdD8bG0i0kfskkp3EZ2w==
dependencies:
- "@abp/utils" "~9.0.0"
+ "@abp/utils" "~9.0.1"
-"@abp/datatables.net-bs5@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.0.tgz#4daee90bdb6446e854ce65f2a3afabb2a13e49ae"
- integrity sha512-ln10tcRwxcB0nuSgqJ/66strHsqMHDC03c0ktxEplAN34yP7KInjFf/U9Svo74K0nKlkJdMCgmqmT4CnBJZlIg==
+"@abp/datatables.net-bs5@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.1.tgz#61e2a3daacf1c074ed6b459e25084cc1f5c4eeab"
+ integrity sha512-tAHdf4TBAS0bVRcvovDKS0OHFyeRSPYbiSIwDn8AIWzW9Weav31jhZ5jL3HEtBKVdb7zqUQ8/3dYPufjDuDdDA==
dependencies:
- "@abp/datatables.net" "~9.0.0"
+ "@abp/datatables.net" "~9.0.1"
datatables.net-bs5 "^2.1.8"
-"@abp/datatables.net@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.0.tgz#82b28518293c9091ec72d5defae2f5d2903d8bbc"
- integrity sha512-QhuX3bGZOckM/fhIIGj/StMJIX2BBtNbLzjTR7WnKMMxDkv0+oDvjw6OqqpYKSvj/q3vY6sRWuCNdEzNi0Fk6A==
+"@abp/datatables.net@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.1.tgz#4cf193e7493f85a272de47326a32a021e3ba40f5"
+ integrity sha512-JY53zVQPluRrBiKwtgk4gfSgD8VDs/VU8LWizhGCprXt3h9vXx0SQL4My6zvXVkv4hUjaq4+MYo84OI8EwFvsg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
datatables.net "^2.1.8"
-"@abp/font-awesome@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.0.tgz#51cce18434a51d050ebb0d822a8c193b7e32f73d"
- integrity sha512-bEHRVzOk0V2kFMAvB/YXFYaDKWRFO0+ScO7jrEFtFgdHXBcqy+dk1aM/N69WalyNmjRRpnh03GhH1LRpwL4/xA==
+"@abp/font-awesome@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.1.tgz#69da15018e7e5f2ff0d3c0d113a4325f54ffc0c3"
+ integrity sha512-7NnYBmk0plLVElUWR1zohul87vE9wKJItemWT5Kskf/IJirKOvXwSSu+7Jg1LmnVtIxcH6qgRpJMmzuAJgawnQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
"@fortawesome/fontawesome-free" "^6.6.0"
-"@abp/jquery-form@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.0.tgz#6518eb117ee35c6bc57543fd1fccb0a8cf8e22f4"
- integrity sha512-6qpfPGHrjADmm7Ne3zcXBGPuLngG8ldAn5fk9k2taftpyX0iR67/nZxiQ0ujaQGgaai5Wb3imNXMG96XLnubkQ==
+"@abp/jquery-form@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.1.tgz#35e0a23cf50b00e0d3b0886f097233b79144145b"
+ integrity sha512-9co0+Cver+RUSOpfuukSWQhR+TCnO9Dly4RTVRIKFksGVfN8hA70NqRrVxaASNNFv42ynK88FQK+2bLPswz/iw==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-form "^4.3.0"
-"@abp/jquery-validation-unobtrusive@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.0.tgz#8ee87dbdec5e92ab04ae9642a49af1598af31c1a"
- integrity sha512-KlRQ1htyFmAIMS9xbHvHIcdhAgtPFe/u7c87MiWlk/CdhDrI3J+n5biLvQUzxCC+YWlwx27zBtPKx2vLeMRH6Q==
+"@abp/jquery-validation-unobtrusive@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.1.tgz#f9564aabebf4d9f90b4ee5c3f3de0e1a348d8ecc"
+ integrity sha512-Zpz48wIFjq7eIApG/sEVjkfaJWZhcyCnlCtDf5eU3vvh82nRetbiCEAVLpZ4LVjorlyp3VvVAxReUSeeWflu8A==
dependencies:
- "@abp/jquery-validation" "~9.0.0"
+ "@abp/jquery-validation" "~9.0.1"
jquery-validation-unobtrusive "^4.0.0"
-"@abp/jquery-validation@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.0.tgz#681d8037c57b3875bd51c927e350ffce65d7c5d2"
- integrity sha512-XgCUBFWbhTDcGepD/xkwA/1Hl4TyYitRttauDJk3tuyXhNtykvdugCCOPY4j2s6VoT1Sb/TFFvOcgEvlBwkKiw==
+"@abp/jquery-validation@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.1.tgz#ad37a44e82fb32ec9bfe2b736dfc8a143fca89c7"
+ integrity sha512-uFSxiPMEkaPlcuyXIoKbeOIIRHo4IrgRbmnlHwI1uETrW60ENUdE2LJAVcYS77JROrjVtv8FPlaffWuva6sGeQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-validation "^1.21.0"
-"@abp/jquery@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.0.tgz#6c67a9f927be598ab6201f8888e8a3cb8bba0686"
- integrity sha512-cUXA7v6F1HqAc/w8fQGxHIfA/6BqO+4uHXDnSsJZtrI26DTLOgw5kBNnAz2W4xggGPCY7SKFHNRZuOW/QfVRPg==
+"@abp/jquery@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.1.tgz#ecce14644c753c581caee0df7bfafd3d083e70f0"
+ integrity sha512-z8EcnOveItAszbTsM9GFFDQSlcIg/ym6+WZME9uldLoetdxkmI9ij0knVeGj5eqMjQaHvTWwrVLObiBbZwYFWw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
jquery "~3.7.1"
-"@abp/lodash@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.0.tgz#a4d42b2dfeb37859ff9f0079c7a8a6cabc982efe"
- integrity sha512-xoAsOkkgnQdPH2OyFTgF6OlCLVbyGF4u8CrFcZWCZmuIFT9cg0+ZEiq//Vog1pg3Lf//N37ceGpOag1LwMe6cQ==
+"@abp/lodash@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.1.tgz#c52c8a4125ecfbeaad7badc5c28bf86d5168799a"
+ integrity sha512-D+/iOyh/O+jV80bK5vWxyn++IVmeRbZq/UaXanA4RKpzigxJUsgOiSD9lUj85BkfKn2YyFmoR39xwaAqkWoyRA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
lodash "^4.17.21"
-"@abp/luxon@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.0.tgz#a54dda13cbbd362481994d20d2e07c669c6ce109"
- integrity sha512-aPejQnoVPfm60LnjwhfUkIsB+poNWusrBE0IHllxhvOVE+01nfa1Lvpp5AUWzV+XchPGOCSf+a29ds4EQWfNbg==
+"@abp/luxon@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.1.tgz#ac59a56653d51b997dbbde8ccc58dac5746a9ef4"
+ integrity sha512-1ZHXP5SGz05tQ1tzYmPcRiZwmkCKGX0KWFoV0zF3ATRaevzku9nXvSy7K6GIFkkgTb9f9XNx4/OBBF/GXkuRqw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
luxon "^3.5.0"
-"@abp/malihu-custom-scrollbar-plugin@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.0.tgz#b8a49202d531c711874a2bcdae5081ae0ed34c3f"
- integrity sha512-iSZ9lKMnv5F2SplR+A4blhFEDq79lKbvBQuS7h/Di3KEHFOdJY4pA8NP8OR+PXOF3JfAezGyjqy7ydn/JNPRwA==
+"@abp/malihu-custom-scrollbar-plugin@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.1.tgz#9e60bcfb9f9d76fed9c50e45bf490d5abd0f4689"
+ integrity sha512-GkKPLg9Z9UTTSSpW2hYZYZ9IgCvDSOYJmyRSl/h4fG8cWMgGCPVpv8MPPu0+WtIfwW/mT4SPX9X+sjUlRL5XRQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
malihu-custom-scrollbar-plugin "^3.1.5"
-"@abp/moment@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.0.tgz#9583e97472ddf608a9a5a6b8a37fee602b419900"
- integrity sha512-wgyJSKmBmHLXZJN3dy3zK4i2JvWGgqnvdtrKURy/raEurxlymv8JF5JA971kTv6Kz2OA/OSYttpVJkDGUpPAhA==
+"@abp/moment@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.1.tgz#d67fa25017fe15dae10e0a0fcebb03aeb63d31d3"
+ integrity sha512-bTuCc7pi/DCtgz7t96UBU0wn9UGt4b50JhDCVeTKquh+gP1sgWoI5hyvEBkmgE4dcffO0yzZU2opHYTKYGl/OQ==
dependencies:
moment "^2.30.1"
-"@abp/prismjs@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/prismjs/-/prismjs-9.0.0.tgz#2ac4a5ddc0c8e0462af027a4dd97a177a3d6f7a5"
- integrity sha512-+evmBq5qV+Ix/TNqO2ljJn0jygW2rEgKNtABcBGmrZC8xiI7+2hjKmSE0e43qb+RDib8aJjwH8ya8wF7p68ZFw==
+"@abp/prismjs@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/prismjs/-/prismjs-9.0.1.tgz#2693014c8161d561e7c7bef01d5cd24ee53be1b3"
+ integrity sha512-NwW+vpTiAsuJYNC/MEiAj0dGMWml95mAQD1Z0Djfsg9D6OpcLsdlgWX+KVdZvHoJ+SPgTDIqdt/0LnIv1w/XdA==
dependencies:
- "@abp/clipboard" "~9.0.0"
- "@abp/core" "~9.0.0"
+ "@abp/clipboard" "~9.0.1"
+ "@abp/core" "~9.0.1"
prismjs "^1.29.0"
-"@abp/select2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.0.tgz#71503eebb746f40a0e565dba1efa05d6ae937b77"
- integrity sha512-l+Hz2Qmc7AcyaTJOyuDaV99RbBWYxYxbeyVEc4xKbRFbJxYBUxT/DFlyozEaoz+OU1fl4szO3UCthJ2yZvxCdg==
+"@abp/select2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.1.tgz#9f1c5c27e72bef910fdf63fc3167f415d2ba8db8"
+ integrity sha512-epyYJH2gAOU2UvGn5FD6372hbW3fL7uZIgGpvfgEfa+l6HPdlV9B1OweU2xkONNCoylShAcP/y8qf+bUh8WDEA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
select2 "^4.0.13"
-"@abp/sweetalert2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.0.tgz#71e511117c18b515e5c658e9cc0667770af65d91"
- integrity sha512-wxDqoA6RUhkbcN27Kzslu/m2ai5T9AkJRtgs7pV55LvPv1ohuCPEUeTVVh8S/5Pgmr7aPFAPxaaCO32aPcImPw==
+"@abp/sweetalert2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.1.tgz#a364b21665cb76e92640496dc8a77b8edbce4500"
+ integrity sha512-0RXR/lzy3qMJLisGUz4qrcA9Clu/Tt1ABGzB64ynG1t/l5XessQsIB5Hkt6MbSyOEsprRsq+ZtKH35CwTRPdAw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
sweetalert2 "^11.14.1"
-"@abp/timeago@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.0.tgz#8f262ba2a74e1c133d0c129f8a0bc55c29441e1a"
- integrity sha512-eaNdIUEtDXnuYqS5O7EGRV+j6XOLAsxz7gd5BIKR9lDujd8pEjSJkXKO+/0KxLTkUSy3JudIO37K1h2wTpbYZA==
+"@abp/timeago@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.1.tgz#d1b7e11fdc267bdb736ff4b591afd3fb81c07cce"
+ integrity sha512-dNfUQwilrY+JHxL5lwJ++ixYDGAs5lF1EDxKMQMIXD/ak5L8rkdLns8Otta7FhgDDny9UJ7XWQjYsG8ZxxybRQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
timeago "^1.6.7"
-"@abp/toastr@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.0.tgz#74d74ecb3e7f2d63b2ff7735c89a40deb3dc3a13"
- integrity sha512-n9uJvw8l2HoxeowPO4diEMxwchGh6puefhyRljFAAstzK77Mhn0u2J7kPynNv94rLSYTYoCxUeJ4TTN+/TVDGQ==
+"@abp/toastr@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.1.tgz#2105a639d7cd4fa9c671b4215d719a78fef7e0ee"
+ integrity sha512-10KuHnSJOxkvzqs6Aim6GJp8xBuT+TsZqBE1KaXQnERPawmOo4+ME28kA/7lq8fvgFUNO3K41T2uuA1GPHVQcg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
toastr "^2.1.4"
-"@abp/utils@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.0.tgz#2264fd83105b061860edd98dbe0efdfe55803ed5"
- integrity sha512-/SB4VYk3Av+ltXv6ovkFHG4iWiECLLitaYWcEHxVAO80ufNjrNnWpHFpXHP8u/RpXIp/CoNiHLr6j5TkZUTZjw==
+"@abp/utils@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.1.tgz#dccaf7cf92147304d454ffb9a8664d52dc7e5c5a"
+ integrity sha512-7eOKT0ciKNdtuASck1kifyc4dC6CeebgxV+rx/EAi3YwPMDkqie6YwLGUqKdGW2N/MZ15JuDR6gkuZxKHEb0NQ==
dependencies:
just-compare "^2.3.0"
diff --git a/modules/blogging/app/Volo.BloggingTestApp/package.json b/modules/blogging/app/Volo.BloggingTestApp/package.json
index 25c37cda9f..3d2acf8641 100644
--- a/modules/blogging/app/Volo.BloggingTestApp/package.json
+++ b/modules/blogging/app/Volo.BloggingTestApp/package.json
@@ -3,7 +3,7 @@
"name": "volo.blogtestapp",
"private": true,
"dependencies": {
- "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.0",
- "@abp/blogging": "~9.0.0"
+ "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.1",
+ "@abp/blogging": "~9.0.1"
}
}
diff --git a/modules/blogging/app/Volo.BloggingTestApp/yarn.lock b/modules/blogging/app/Volo.BloggingTestApp/yarn.lock
index 4a01d58b56..46619a0994 100644
--- a/modules/blogging/app/Volo.BloggingTestApp/yarn.lock
+++ b/modules/blogging/app/Volo.BloggingTestApp/yarn.lock
@@ -2,237 +2,237 @@
# yarn lockfile v1
-"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.0.tgz#e4b05d63446c24b612275e3949881e176d32527f"
- integrity sha512-GHEt26xESe/TCg1q6ZjeUbxniqdxaiB1OPTZ9md7ONql6bEp8t9iUPeSlkyAZtQ7qBzwgkiLpkRFT+J21DftaQ==
- dependencies:
- "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.0.tgz#a77b4b70fa6217146afd6a57211f57e2360564f2"
- integrity sha512-QmZrSp+rfqfnC9W4QZ+nNdAq0rOALzRn1rsDpGDHhIHwyXB03QPdjzIWaEvldUYHzqg84cEP0LcqW4y57Xr1iw==
- dependencies:
- "@abp/aspnetcore.mvc.ui" "~9.0.0"
- "@abp/bootstrap" "~9.0.0"
- "@abp/bootstrap-datepicker" "~9.0.0"
- "@abp/bootstrap-daterangepicker" "~9.0.0"
- "@abp/datatables.net-bs5" "~9.0.0"
- "@abp/font-awesome" "~9.0.0"
- "@abp/jquery-form" "~9.0.0"
- "@abp/jquery-validation-unobtrusive" "~9.0.0"
- "@abp/lodash" "~9.0.0"
- "@abp/luxon" "~9.0.0"
- "@abp/malihu-custom-scrollbar-plugin" "~9.0.0"
- "@abp/moment" "~9.0.0"
- "@abp/select2" "~9.0.0"
- "@abp/sweetalert2" "~9.0.0"
- "@abp/timeago" "~9.0.0"
- "@abp/toastr" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.0.tgz#93004dd056c130386c7b40c458f3a793d8c9e051"
- integrity sha512-pvO5bh9baFcSbfW0s7tpMpbe6DKSC16jeyeluZLlvYxC0n2gkrv7H/hB5aMuBKIlh89aGY9DQIcMFC8iczL3Bg==
+"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.1.tgz#cdb6130ac6f8499f9a3f856b596c46745d056a30"
+ integrity sha512-JaxuDH8WTnZYX6C7CX7ZVHfhTDR/SxfwRvtJaGGijSX8QKd1DXpfi9lYEF4Y9k5b48YT/0slBqU5g0sLmBd3+g==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.1.tgz#7a1be780a305aa0d0f467472fc721b6f710fda7a"
+ integrity sha512-vrgRYKNctdZYhPCQ2I6Z10gUI1S5ynDj+DgnV7x7a7G/juuHaOSbkRXgwrmJsmYLfGA7KFCoJcbpER8mmXQeHQ==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui" "~9.0.1"
+ "@abp/bootstrap" "~9.0.1"
+ "@abp/bootstrap-datepicker" "~9.0.1"
+ "@abp/bootstrap-daterangepicker" "~9.0.1"
+ "@abp/datatables.net-bs5" "~9.0.1"
+ "@abp/font-awesome" "~9.0.1"
+ "@abp/jquery-form" "~9.0.1"
+ "@abp/jquery-validation-unobtrusive" "~9.0.1"
+ "@abp/lodash" "~9.0.1"
+ "@abp/luxon" "~9.0.1"
+ "@abp/malihu-custom-scrollbar-plugin" "~9.0.1"
+ "@abp/moment" "~9.0.1"
+ "@abp/select2" "~9.0.1"
+ "@abp/sweetalert2" "~9.0.1"
+ "@abp/timeago" "~9.0.1"
+ "@abp/toastr" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.1.tgz#811dd91138efb82345fc06559b5d2feea88021af"
+ integrity sha512-KVbWCpBV7S1ViHYQP/hn4gSMzSiA2e8/zC0HND1fjE+1zXJqJyEWto1eg2DaKSOs/rmHD8WtFVMU4hCLQTCuIA==
dependencies:
ansi-colors "^4.1.3"
-"@abp/blogging@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/blogging/-/blogging-9.0.0.tgz#bba9e0288e215473b1851ab58d4898a2455c2afe"
- integrity sha512-tsKqEYnVhDrBHVMOwSTLR+OudEnxBpj5RhDiLPA+MO+FSqiJpuxlDI2SlH2Sz7alCF1RnAM/alITvrKq9TR/gg==
+"@abp/blogging@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/blogging/-/blogging-9.0.1.tgz#7d5351b4e6b28bd25edf480e51d1a115699387ec"
+ integrity sha512-ocwDEWQe5bkDSObEX6As8DD2Kgsn8qgK0OZvNEZo5xPvpUNMnbcvG5tzdXMBbR+bb++J+M5VjDQydOWJ5uuxjw==
dependencies:
- "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.0"
- "@abp/owl.carousel" "~9.0.0"
- "@abp/prismjs" "~9.0.0"
- "@abp/tui-editor" "~9.0.0"
+ "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.1"
+ "@abp/owl.carousel" "~9.0.1"
+ "@abp/prismjs" "~9.0.1"
+ "@abp/tui-editor" "~9.0.1"
-"@abp/bootstrap-datepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.0.tgz#d6609e50568aac6a013abfcaa07817998e1eaa99"
- integrity sha512-wxAeVsOVOJpidDn0g3FlJvpOsn5SVY5IpEG+FtUxSAKuWYJQWF6g5CaghrfS+FNDK9IHhxrgSMPMVBK9YlDSzw==
+"@abp/bootstrap-datepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.1.tgz#21cd19090b9a571ddf5a9019bc17864c8685408c"
+ integrity sha512-LhgqBRr66x9gGTAyv1FuPAw8u8N77WQK2AfDkNgLueurM2+xMYQzH3pWWLhcN676/JXafxfALCfBoM8U7CfXLg==
dependencies:
bootstrap-datepicker "^1.10.0"
-"@abp/bootstrap-daterangepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.0.tgz#fc2871e7d886accd8e4748cb763b695e10701c00"
- integrity sha512-qS6SQR6X7wMCVLcuXatJ2+m1GNlpN71p7VDritxfwAMHa7WBRAAi8apX2WItwBAuLQ2s1uSA0Wp1KstHG3VPLA==
+"@abp/bootstrap-daterangepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.1.tgz#c19c23ede6fd868a147ea893d4fdecfa60b177f7"
+ integrity sha512-SlBVgiYa75uQyaGt5EVjduren7Po5UP26A1tXEQQr0hDia1+1TK0qxmJ2YxjeJpB7FwtcmJiofv5fngvJqhxIA==
dependencies:
bootstrap-daterangepicker "^3.1.0"
-"@abp/bootstrap@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.0.tgz#5ab292e95ba7eef129cbaa1dc5f5d5c0af8530d7"
- integrity sha512-jEr6su8BfoNxYvze/UlddE3DlezyPvQeiNhff8SDPqtRQ6XvjZ48O/JwtY01hK2njEDLOZINY5zJRGmO2OiOUw==
+"@abp/bootstrap@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.1.tgz#2360303ec24c8f3eb755b10fa6a953db6d858daf"
+ integrity sha512-uAfVNsUYSsRqgeO9JMnhJJlFoDmO3/lTvER/iYOqerF0X03KQnR8HZy2V3MD1D+fSh/4JuU6aAZFl6IQLinMKQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
bootstrap "^5.3.3"
-"@abp/clipboard@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/clipboard/-/clipboard-9.0.0.tgz#21ae924dfef6ccf3c8a3913bb81caf22b57404e4"
- integrity sha512-W6S7pihpQ6Wfj48Clk7hYgDvUtnRwkNIsk5t8s9+LxoAdJQGQFxh7SAW1QNavUv8QqsClNE1VtSU87c7rSoAbA==
+"@abp/clipboard@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/clipboard/-/clipboard-9.0.1.tgz#b1712b2d940caecc85eb99247a569ffe3ab1a387"
+ integrity sha512-obDYqUTf+65E26yHZO2DigQOeDnzAEvak0Umz91cholZHVRvzc9X1itv/abS/whU0YOU/d8Viud8zFv1MnM83w==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
clipboard "^2.0.11"
-"@abp/core@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.0.tgz#dc5ea9139957e307f73ac1f40b179f991e3c2168"
- integrity sha512-vzQyTpqVDx63pOI9e2QLUMWZCrSojHsRnzLlvl+/cKFi7jE1q/uSP66FPR2bwkD+DNLI/tn6is1Jx3skKA1FvA==
+"@abp/core@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.1.tgz#727fc1dfac21cca1216c4f5091da7d5efefdb467"
+ integrity sha512-NWXDVCIJePbcKCrRvGxAlaA2MrOF2IZox37bQkW7/mTeOPpNoTa6lhowksNNcTpFq2YdD8bG0i0kfskkp3EZ2w==
dependencies:
- "@abp/utils" "~9.0.0"
+ "@abp/utils" "~9.0.1"
-"@abp/datatables.net-bs5@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.0.tgz#4daee90bdb6446e854ce65f2a3afabb2a13e49ae"
- integrity sha512-ln10tcRwxcB0nuSgqJ/66strHsqMHDC03c0ktxEplAN34yP7KInjFf/U9Svo74K0nKlkJdMCgmqmT4CnBJZlIg==
+"@abp/datatables.net-bs5@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.1.tgz#61e2a3daacf1c074ed6b459e25084cc1f5c4eeab"
+ integrity sha512-tAHdf4TBAS0bVRcvovDKS0OHFyeRSPYbiSIwDn8AIWzW9Weav31jhZ5jL3HEtBKVdb7zqUQ8/3dYPufjDuDdDA==
dependencies:
- "@abp/datatables.net" "~9.0.0"
+ "@abp/datatables.net" "~9.0.1"
datatables.net-bs5 "^2.1.8"
-"@abp/datatables.net@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.0.tgz#82b28518293c9091ec72d5defae2f5d2903d8bbc"
- integrity sha512-QhuX3bGZOckM/fhIIGj/StMJIX2BBtNbLzjTR7WnKMMxDkv0+oDvjw6OqqpYKSvj/q3vY6sRWuCNdEzNi0Fk6A==
+"@abp/datatables.net@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.1.tgz#4cf193e7493f85a272de47326a32a021e3ba40f5"
+ integrity sha512-JY53zVQPluRrBiKwtgk4gfSgD8VDs/VU8LWizhGCprXt3h9vXx0SQL4My6zvXVkv4hUjaq4+MYo84OI8EwFvsg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
datatables.net "^2.1.8"
-"@abp/font-awesome@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.0.tgz#51cce18434a51d050ebb0d822a8c193b7e32f73d"
- integrity sha512-bEHRVzOk0V2kFMAvB/YXFYaDKWRFO0+ScO7jrEFtFgdHXBcqy+dk1aM/N69WalyNmjRRpnh03GhH1LRpwL4/xA==
+"@abp/font-awesome@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.1.tgz#69da15018e7e5f2ff0d3c0d113a4325f54ffc0c3"
+ integrity sha512-7NnYBmk0plLVElUWR1zohul87vE9wKJItemWT5Kskf/IJirKOvXwSSu+7Jg1LmnVtIxcH6qgRpJMmzuAJgawnQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
"@fortawesome/fontawesome-free" "^6.6.0"
-"@abp/jquery-form@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.0.tgz#6518eb117ee35c6bc57543fd1fccb0a8cf8e22f4"
- integrity sha512-6qpfPGHrjADmm7Ne3zcXBGPuLngG8ldAn5fk9k2taftpyX0iR67/nZxiQ0ujaQGgaai5Wb3imNXMG96XLnubkQ==
+"@abp/jquery-form@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.1.tgz#35e0a23cf50b00e0d3b0886f097233b79144145b"
+ integrity sha512-9co0+Cver+RUSOpfuukSWQhR+TCnO9Dly4RTVRIKFksGVfN8hA70NqRrVxaASNNFv42ynK88FQK+2bLPswz/iw==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-form "^4.3.0"
-"@abp/jquery-validation-unobtrusive@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.0.tgz#8ee87dbdec5e92ab04ae9642a49af1598af31c1a"
- integrity sha512-KlRQ1htyFmAIMS9xbHvHIcdhAgtPFe/u7c87MiWlk/CdhDrI3J+n5biLvQUzxCC+YWlwx27zBtPKx2vLeMRH6Q==
+"@abp/jquery-validation-unobtrusive@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.1.tgz#f9564aabebf4d9f90b4ee5c3f3de0e1a348d8ecc"
+ integrity sha512-Zpz48wIFjq7eIApG/sEVjkfaJWZhcyCnlCtDf5eU3vvh82nRetbiCEAVLpZ4LVjorlyp3VvVAxReUSeeWflu8A==
dependencies:
- "@abp/jquery-validation" "~9.0.0"
+ "@abp/jquery-validation" "~9.0.1"
jquery-validation-unobtrusive "^4.0.0"
-"@abp/jquery-validation@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.0.tgz#681d8037c57b3875bd51c927e350ffce65d7c5d2"
- integrity sha512-XgCUBFWbhTDcGepD/xkwA/1Hl4TyYitRttauDJk3tuyXhNtykvdugCCOPY4j2s6VoT1Sb/TFFvOcgEvlBwkKiw==
+"@abp/jquery-validation@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.1.tgz#ad37a44e82fb32ec9bfe2b736dfc8a143fca89c7"
+ integrity sha512-uFSxiPMEkaPlcuyXIoKbeOIIRHo4IrgRbmnlHwI1uETrW60ENUdE2LJAVcYS77JROrjVtv8FPlaffWuva6sGeQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-validation "^1.21.0"
-"@abp/jquery@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.0.tgz#6c67a9f927be598ab6201f8888e8a3cb8bba0686"
- integrity sha512-cUXA7v6F1HqAc/w8fQGxHIfA/6BqO+4uHXDnSsJZtrI26DTLOgw5kBNnAz2W4xggGPCY7SKFHNRZuOW/QfVRPg==
+"@abp/jquery@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.1.tgz#ecce14644c753c581caee0df7bfafd3d083e70f0"
+ integrity sha512-z8EcnOveItAszbTsM9GFFDQSlcIg/ym6+WZME9uldLoetdxkmI9ij0knVeGj5eqMjQaHvTWwrVLObiBbZwYFWw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
jquery "~3.7.1"
-"@abp/lodash@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.0.tgz#a4d42b2dfeb37859ff9f0079c7a8a6cabc982efe"
- integrity sha512-xoAsOkkgnQdPH2OyFTgF6OlCLVbyGF4u8CrFcZWCZmuIFT9cg0+ZEiq//Vog1pg3Lf//N37ceGpOag1LwMe6cQ==
+"@abp/lodash@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.1.tgz#c52c8a4125ecfbeaad7badc5c28bf86d5168799a"
+ integrity sha512-D+/iOyh/O+jV80bK5vWxyn++IVmeRbZq/UaXanA4RKpzigxJUsgOiSD9lUj85BkfKn2YyFmoR39xwaAqkWoyRA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
lodash "^4.17.21"
-"@abp/luxon@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.0.tgz#a54dda13cbbd362481994d20d2e07c669c6ce109"
- integrity sha512-aPejQnoVPfm60LnjwhfUkIsB+poNWusrBE0IHllxhvOVE+01nfa1Lvpp5AUWzV+XchPGOCSf+a29ds4EQWfNbg==
+"@abp/luxon@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.1.tgz#ac59a56653d51b997dbbde8ccc58dac5746a9ef4"
+ integrity sha512-1ZHXP5SGz05tQ1tzYmPcRiZwmkCKGX0KWFoV0zF3ATRaevzku9nXvSy7K6GIFkkgTb9f9XNx4/OBBF/GXkuRqw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
luxon "^3.5.0"
-"@abp/malihu-custom-scrollbar-plugin@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.0.tgz#b8a49202d531c711874a2bcdae5081ae0ed34c3f"
- integrity sha512-iSZ9lKMnv5F2SplR+A4blhFEDq79lKbvBQuS7h/Di3KEHFOdJY4pA8NP8OR+PXOF3JfAezGyjqy7ydn/JNPRwA==
+"@abp/malihu-custom-scrollbar-plugin@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.1.tgz#9e60bcfb9f9d76fed9c50e45bf490d5abd0f4689"
+ integrity sha512-GkKPLg9Z9UTTSSpW2hYZYZ9IgCvDSOYJmyRSl/h4fG8cWMgGCPVpv8MPPu0+WtIfwW/mT4SPX9X+sjUlRL5XRQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
malihu-custom-scrollbar-plugin "^3.1.5"
-"@abp/moment@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.0.tgz#9583e97472ddf608a9a5a6b8a37fee602b419900"
- integrity sha512-wgyJSKmBmHLXZJN3dy3zK4i2JvWGgqnvdtrKURy/raEurxlymv8JF5JA971kTv6Kz2OA/OSYttpVJkDGUpPAhA==
+"@abp/moment@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.1.tgz#d67fa25017fe15dae10e0a0fcebb03aeb63d31d3"
+ integrity sha512-bTuCc7pi/DCtgz7t96UBU0wn9UGt4b50JhDCVeTKquh+gP1sgWoI5hyvEBkmgE4dcffO0yzZU2opHYTKYGl/OQ==
dependencies:
moment "^2.30.1"
-"@abp/owl.carousel@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/owl.carousel/-/owl.carousel-9.0.0.tgz#44f6b35199940aa98b9b97bfaeb7872779133ae5"
- integrity sha512-zwlrvm4wxvM/4OlUsJPhEy4I+N+wX27dwCQ640eAZirHlmRrOydAADDP964LDX6ay3PqvW8VzH/e0iEFqUBLWg==
+"@abp/owl.carousel@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/owl.carousel/-/owl.carousel-9.0.1.tgz#83e7f7448976ab66facaa4d4c27bf7dac3d37b07"
+ integrity sha512-kNE/iv6fvI4V+5suVgMPrmT0ACqi3TEW5wqn+mWq1SwfD1qGZG0p+60NCEuhMdNVLZKD3xvwn3Ba7+KFjmAd4w==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
owl.carousel "^2.3.4"
-"@abp/prismjs@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/prismjs/-/prismjs-9.0.0.tgz#2ac4a5ddc0c8e0462af027a4dd97a177a3d6f7a5"
- integrity sha512-+evmBq5qV+Ix/TNqO2ljJn0jygW2rEgKNtABcBGmrZC8xiI7+2hjKmSE0e43qb+RDib8aJjwH8ya8wF7p68ZFw==
+"@abp/prismjs@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/prismjs/-/prismjs-9.0.1.tgz#2693014c8161d561e7c7bef01d5cd24ee53be1b3"
+ integrity sha512-NwW+vpTiAsuJYNC/MEiAj0dGMWml95mAQD1Z0Djfsg9D6OpcLsdlgWX+KVdZvHoJ+SPgTDIqdt/0LnIv1w/XdA==
dependencies:
- "@abp/clipboard" "~9.0.0"
- "@abp/core" "~9.0.0"
+ "@abp/clipboard" "~9.0.1"
+ "@abp/core" "~9.0.1"
prismjs "^1.29.0"
-"@abp/select2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.0.tgz#71503eebb746f40a0e565dba1efa05d6ae937b77"
- integrity sha512-l+Hz2Qmc7AcyaTJOyuDaV99RbBWYxYxbeyVEc4xKbRFbJxYBUxT/DFlyozEaoz+OU1fl4szO3UCthJ2yZvxCdg==
+"@abp/select2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.1.tgz#9f1c5c27e72bef910fdf63fc3167f415d2ba8db8"
+ integrity sha512-epyYJH2gAOU2UvGn5FD6372hbW3fL7uZIgGpvfgEfa+l6HPdlV9B1OweU2xkONNCoylShAcP/y8qf+bUh8WDEA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
select2 "^4.0.13"
-"@abp/sweetalert2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.0.tgz#71e511117c18b515e5c658e9cc0667770af65d91"
- integrity sha512-wxDqoA6RUhkbcN27Kzslu/m2ai5T9AkJRtgs7pV55LvPv1ohuCPEUeTVVh8S/5Pgmr7aPFAPxaaCO32aPcImPw==
+"@abp/sweetalert2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.1.tgz#a364b21665cb76e92640496dc8a77b8edbce4500"
+ integrity sha512-0RXR/lzy3qMJLisGUz4qrcA9Clu/Tt1ABGzB64ynG1t/l5XessQsIB5Hkt6MbSyOEsprRsq+ZtKH35CwTRPdAw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
sweetalert2 "^11.14.1"
-"@abp/timeago@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.0.tgz#8f262ba2a74e1c133d0c129f8a0bc55c29441e1a"
- integrity sha512-eaNdIUEtDXnuYqS5O7EGRV+j6XOLAsxz7gd5BIKR9lDujd8pEjSJkXKO+/0KxLTkUSy3JudIO37K1h2wTpbYZA==
+"@abp/timeago@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.1.tgz#d1b7e11fdc267bdb736ff4b591afd3fb81c07cce"
+ integrity sha512-dNfUQwilrY+JHxL5lwJ++ixYDGAs5lF1EDxKMQMIXD/ak5L8rkdLns8Otta7FhgDDny9UJ7XWQjYsG8ZxxybRQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
timeago "^1.6.7"
-"@abp/toastr@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.0.tgz#74d74ecb3e7f2d63b2ff7735c89a40deb3dc3a13"
- integrity sha512-n9uJvw8l2HoxeowPO4diEMxwchGh6puefhyRljFAAstzK77Mhn0u2J7kPynNv94rLSYTYoCxUeJ4TTN+/TVDGQ==
+"@abp/toastr@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.1.tgz#2105a639d7cd4fa9c671b4215d719a78fef7e0ee"
+ integrity sha512-10KuHnSJOxkvzqs6Aim6GJp8xBuT+TsZqBE1KaXQnERPawmOo4+ME28kA/7lq8fvgFUNO3K41T2uuA1GPHVQcg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
toastr "^2.1.4"
-"@abp/tui-editor@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/tui-editor/-/tui-editor-9.0.0.tgz#d9f7721c3d8e361d901a0e0e11e786b2e12b234a"
- integrity sha512-TVadK2Yi3wlwyvEVshh1PxkNRppYmqaPB6QE1cOhrWZAFbP1fh1oFvfqQiHN3o+9RGHFcEnhEK3D+49aHuDh3Q==
+"@abp/tui-editor@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/tui-editor/-/tui-editor-9.0.1.tgz#df2ef5101300f2f739c0c93b66f63ccd8492a42d"
+ integrity sha512-0H9AK4/YXsDmLruHMqk2uygWkWTR9pRRexGCr+IzoAeb+tljg15MnmlboTE9iNOntUug6Iml9tyY5gNOUuQXvQ==
dependencies:
- "@abp/jquery" "~9.0.0"
- "@abp/prismjs" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
+ "@abp/prismjs" "~9.0.1"
-"@abp/utils@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.0.tgz#2264fd83105b061860edd98dbe0efdfe55803ed5"
- integrity sha512-/SB4VYk3Av+ltXv6ovkFHG4iWiECLLitaYWcEHxVAO80ufNjrNnWpHFpXHP8u/RpXIp/CoNiHLr6j5TkZUTZjw==
+"@abp/utils@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.1.tgz#dccaf7cf92147304d454ffb9a8664d52dc7e5c5a"
+ integrity sha512-7eOKT0ciKNdtuASck1kifyc4dC6CeebgxV+rx/EAi3YwPMDkqie6YwLGUqKdGW2N/MZ15JuDR6gkuZxKHEb0NQ==
dependencies:
just-compare "^2.3.0"
diff --git a/modules/client-simulation/demo/Volo.ClientSimulation.Demo/package.json b/modules/client-simulation/demo/Volo.ClientSimulation.Demo/package.json
index e625cd894f..9f481e899f 100644
--- a/modules/client-simulation/demo/Volo.ClientSimulation.Demo/package.json
+++ b/modules/client-simulation/demo/Volo.ClientSimulation.Demo/package.json
@@ -3,6 +3,6 @@
"name": "client-simulation-web",
"private": true,
"dependencies": {
- "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.0"
+ "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.1"
}
}
diff --git a/modules/client-simulation/demo/Volo.ClientSimulation.Demo/yarn.lock b/modules/client-simulation/demo/Volo.ClientSimulation.Demo/yarn.lock
index 9e01130758..0ac91bf6da 100644
--- a/modules/client-simulation/demo/Volo.ClientSimulation.Demo/yarn.lock
+++ b/modules/client-simulation/demo/Volo.ClientSimulation.Demo/yarn.lock
@@ -2,194 +2,194 @@
# yarn lockfile v1
-"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.0.tgz#e4b05d63446c24b612275e3949881e176d32527f"
- integrity sha512-GHEt26xESe/TCg1q6ZjeUbxniqdxaiB1OPTZ9md7ONql6bEp8t9iUPeSlkyAZtQ7qBzwgkiLpkRFT+J21DftaQ==
- dependencies:
- "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.0.tgz#a77b4b70fa6217146afd6a57211f57e2360564f2"
- integrity sha512-QmZrSp+rfqfnC9W4QZ+nNdAq0rOALzRn1rsDpGDHhIHwyXB03QPdjzIWaEvldUYHzqg84cEP0LcqW4y57Xr1iw==
- dependencies:
- "@abp/aspnetcore.mvc.ui" "~9.0.0"
- "@abp/bootstrap" "~9.0.0"
- "@abp/bootstrap-datepicker" "~9.0.0"
- "@abp/bootstrap-daterangepicker" "~9.0.0"
- "@abp/datatables.net-bs5" "~9.0.0"
- "@abp/font-awesome" "~9.0.0"
- "@abp/jquery-form" "~9.0.0"
- "@abp/jquery-validation-unobtrusive" "~9.0.0"
- "@abp/lodash" "~9.0.0"
- "@abp/luxon" "~9.0.0"
- "@abp/malihu-custom-scrollbar-plugin" "~9.0.0"
- "@abp/moment" "~9.0.0"
- "@abp/select2" "~9.0.0"
- "@abp/sweetalert2" "~9.0.0"
- "@abp/timeago" "~9.0.0"
- "@abp/toastr" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.0.tgz#93004dd056c130386c7b40c458f3a793d8c9e051"
- integrity sha512-pvO5bh9baFcSbfW0s7tpMpbe6DKSC16jeyeluZLlvYxC0n2gkrv7H/hB5aMuBKIlh89aGY9DQIcMFC8iczL3Bg==
+"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.1.tgz#cdb6130ac6f8499f9a3f856b596c46745d056a30"
+ integrity sha512-JaxuDH8WTnZYX6C7CX7ZVHfhTDR/SxfwRvtJaGGijSX8QKd1DXpfi9lYEF4Y9k5b48YT/0slBqU5g0sLmBd3+g==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.1.tgz#7a1be780a305aa0d0f467472fc721b6f710fda7a"
+ integrity sha512-vrgRYKNctdZYhPCQ2I6Z10gUI1S5ynDj+DgnV7x7a7G/juuHaOSbkRXgwrmJsmYLfGA7KFCoJcbpER8mmXQeHQ==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui" "~9.0.1"
+ "@abp/bootstrap" "~9.0.1"
+ "@abp/bootstrap-datepicker" "~9.0.1"
+ "@abp/bootstrap-daterangepicker" "~9.0.1"
+ "@abp/datatables.net-bs5" "~9.0.1"
+ "@abp/font-awesome" "~9.0.1"
+ "@abp/jquery-form" "~9.0.1"
+ "@abp/jquery-validation-unobtrusive" "~9.0.1"
+ "@abp/lodash" "~9.0.1"
+ "@abp/luxon" "~9.0.1"
+ "@abp/malihu-custom-scrollbar-plugin" "~9.0.1"
+ "@abp/moment" "~9.0.1"
+ "@abp/select2" "~9.0.1"
+ "@abp/sweetalert2" "~9.0.1"
+ "@abp/timeago" "~9.0.1"
+ "@abp/toastr" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.1.tgz#811dd91138efb82345fc06559b5d2feea88021af"
+ integrity sha512-KVbWCpBV7S1ViHYQP/hn4gSMzSiA2e8/zC0HND1fjE+1zXJqJyEWto1eg2DaKSOs/rmHD8WtFVMU4hCLQTCuIA==
dependencies:
ansi-colors "^4.1.3"
-"@abp/bootstrap-datepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.0.tgz#d6609e50568aac6a013abfcaa07817998e1eaa99"
- integrity sha512-wxAeVsOVOJpidDn0g3FlJvpOsn5SVY5IpEG+FtUxSAKuWYJQWF6g5CaghrfS+FNDK9IHhxrgSMPMVBK9YlDSzw==
+"@abp/bootstrap-datepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.1.tgz#21cd19090b9a571ddf5a9019bc17864c8685408c"
+ integrity sha512-LhgqBRr66x9gGTAyv1FuPAw8u8N77WQK2AfDkNgLueurM2+xMYQzH3pWWLhcN676/JXafxfALCfBoM8U7CfXLg==
dependencies:
bootstrap-datepicker "^1.10.0"
-"@abp/bootstrap-daterangepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.0.tgz#fc2871e7d886accd8e4748cb763b695e10701c00"
- integrity sha512-qS6SQR6X7wMCVLcuXatJ2+m1GNlpN71p7VDritxfwAMHa7WBRAAi8apX2WItwBAuLQ2s1uSA0Wp1KstHG3VPLA==
+"@abp/bootstrap-daterangepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.1.tgz#c19c23ede6fd868a147ea893d4fdecfa60b177f7"
+ integrity sha512-SlBVgiYa75uQyaGt5EVjduren7Po5UP26A1tXEQQr0hDia1+1TK0qxmJ2YxjeJpB7FwtcmJiofv5fngvJqhxIA==
dependencies:
bootstrap-daterangepicker "^3.1.0"
-"@abp/bootstrap@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.0.tgz#5ab292e95ba7eef129cbaa1dc5f5d5c0af8530d7"
- integrity sha512-jEr6su8BfoNxYvze/UlddE3DlezyPvQeiNhff8SDPqtRQ6XvjZ48O/JwtY01hK2njEDLOZINY5zJRGmO2OiOUw==
+"@abp/bootstrap@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.1.tgz#2360303ec24c8f3eb755b10fa6a953db6d858daf"
+ integrity sha512-uAfVNsUYSsRqgeO9JMnhJJlFoDmO3/lTvER/iYOqerF0X03KQnR8HZy2V3MD1D+fSh/4JuU6aAZFl6IQLinMKQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
bootstrap "^5.3.3"
-"@abp/core@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.0.tgz#dc5ea9139957e307f73ac1f40b179f991e3c2168"
- integrity sha512-vzQyTpqVDx63pOI9e2QLUMWZCrSojHsRnzLlvl+/cKFi7jE1q/uSP66FPR2bwkD+DNLI/tn6is1Jx3skKA1FvA==
+"@abp/core@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.1.tgz#727fc1dfac21cca1216c4f5091da7d5efefdb467"
+ integrity sha512-NWXDVCIJePbcKCrRvGxAlaA2MrOF2IZox37bQkW7/mTeOPpNoTa6lhowksNNcTpFq2YdD8bG0i0kfskkp3EZ2w==
dependencies:
- "@abp/utils" "~9.0.0"
+ "@abp/utils" "~9.0.1"
-"@abp/datatables.net-bs5@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.0.tgz#4daee90bdb6446e854ce65f2a3afabb2a13e49ae"
- integrity sha512-ln10tcRwxcB0nuSgqJ/66strHsqMHDC03c0ktxEplAN34yP7KInjFf/U9Svo74K0nKlkJdMCgmqmT4CnBJZlIg==
+"@abp/datatables.net-bs5@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.1.tgz#61e2a3daacf1c074ed6b459e25084cc1f5c4eeab"
+ integrity sha512-tAHdf4TBAS0bVRcvovDKS0OHFyeRSPYbiSIwDn8AIWzW9Weav31jhZ5jL3HEtBKVdb7zqUQ8/3dYPufjDuDdDA==
dependencies:
- "@abp/datatables.net" "~9.0.0"
+ "@abp/datatables.net" "~9.0.1"
datatables.net-bs5 "^2.1.8"
-"@abp/datatables.net@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.0.tgz#82b28518293c9091ec72d5defae2f5d2903d8bbc"
- integrity sha512-QhuX3bGZOckM/fhIIGj/StMJIX2BBtNbLzjTR7WnKMMxDkv0+oDvjw6OqqpYKSvj/q3vY6sRWuCNdEzNi0Fk6A==
+"@abp/datatables.net@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.1.tgz#4cf193e7493f85a272de47326a32a021e3ba40f5"
+ integrity sha512-JY53zVQPluRrBiKwtgk4gfSgD8VDs/VU8LWizhGCprXt3h9vXx0SQL4My6zvXVkv4hUjaq4+MYo84OI8EwFvsg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
datatables.net "^2.1.8"
-"@abp/font-awesome@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.0.tgz#51cce18434a51d050ebb0d822a8c193b7e32f73d"
- integrity sha512-bEHRVzOk0V2kFMAvB/YXFYaDKWRFO0+ScO7jrEFtFgdHXBcqy+dk1aM/N69WalyNmjRRpnh03GhH1LRpwL4/xA==
+"@abp/font-awesome@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.1.tgz#69da15018e7e5f2ff0d3c0d113a4325f54ffc0c3"
+ integrity sha512-7NnYBmk0plLVElUWR1zohul87vE9wKJItemWT5Kskf/IJirKOvXwSSu+7Jg1LmnVtIxcH6qgRpJMmzuAJgawnQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
"@fortawesome/fontawesome-free" "^6.6.0"
-"@abp/jquery-form@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.0.tgz#6518eb117ee35c6bc57543fd1fccb0a8cf8e22f4"
- integrity sha512-6qpfPGHrjADmm7Ne3zcXBGPuLngG8ldAn5fk9k2taftpyX0iR67/nZxiQ0ujaQGgaai5Wb3imNXMG96XLnubkQ==
+"@abp/jquery-form@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.1.tgz#35e0a23cf50b00e0d3b0886f097233b79144145b"
+ integrity sha512-9co0+Cver+RUSOpfuukSWQhR+TCnO9Dly4RTVRIKFksGVfN8hA70NqRrVxaASNNFv42ynK88FQK+2bLPswz/iw==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-form "^4.3.0"
-"@abp/jquery-validation-unobtrusive@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.0.tgz#8ee87dbdec5e92ab04ae9642a49af1598af31c1a"
- integrity sha512-KlRQ1htyFmAIMS9xbHvHIcdhAgtPFe/u7c87MiWlk/CdhDrI3J+n5biLvQUzxCC+YWlwx27zBtPKx2vLeMRH6Q==
+"@abp/jquery-validation-unobtrusive@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.1.tgz#f9564aabebf4d9f90b4ee5c3f3de0e1a348d8ecc"
+ integrity sha512-Zpz48wIFjq7eIApG/sEVjkfaJWZhcyCnlCtDf5eU3vvh82nRetbiCEAVLpZ4LVjorlyp3VvVAxReUSeeWflu8A==
dependencies:
- "@abp/jquery-validation" "~9.0.0"
+ "@abp/jquery-validation" "~9.0.1"
jquery-validation-unobtrusive "^4.0.0"
-"@abp/jquery-validation@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.0.tgz#681d8037c57b3875bd51c927e350ffce65d7c5d2"
- integrity sha512-XgCUBFWbhTDcGepD/xkwA/1Hl4TyYitRttauDJk3tuyXhNtykvdugCCOPY4j2s6VoT1Sb/TFFvOcgEvlBwkKiw==
+"@abp/jquery-validation@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.1.tgz#ad37a44e82fb32ec9bfe2b736dfc8a143fca89c7"
+ integrity sha512-uFSxiPMEkaPlcuyXIoKbeOIIRHo4IrgRbmnlHwI1uETrW60ENUdE2LJAVcYS77JROrjVtv8FPlaffWuva6sGeQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-validation "^1.21.0"
-"@abp/jquery@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.0.tgz#6c67a9f927be598ab6201f8888e8a3cb8bba0686"
- integrity sha512-cUXA7v6F1HqAc/w8fQGxHIfA/6BqO+4uHXDnSsJZtrI26DTLOgw5kBNnAz2W4xggGPCY7SKFHNRZuOW/QfVRPg==
+"@abp/jquery@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.1.tgz#ecce14644c753c581caee0df7bfafd3d083e70f0"
+ integrity sha512-z8EcnOveItAszbTsM9GFFDQSlcIg/ym6+WZME9uldLoetdxkmI9ij0knVeGj5eqMjQaHvTWwrVLObiBbZwYFWw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
jquery "~3.7.1"
-"@abp/lodash@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.0.tgz#a4d42b2dfeb37859ff9f0079c7a8a6cabc982efe"
- integrity sha512-xoAsOkkgnQdPH2OyFTgF6OlCLVbyGF4u8CrFcZWCZmuIFT9cg0+ZEiq//Vog1pg3Lf//N37ceGpOag1LwMe6cQ==
+"@abp/lodash@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.1.tgz#c52c8a4125ecfbeaad7badc5c28bf86d5168799a"
+ integrity sha512-D+/iOyh/O+jV80bK5vWxyn++IVmeRbZq/UaXanA4RKpzigxJUsgOiSD9lUj85BkfKn2YyFmoR39xwaAqkWoyRA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
lodash "^4.17.21"
-"@abp/luxon@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.0.tgz#a54dda13cbbd362481994d20d2e07c669c6ce109"
- integrity sha512-aPejQnoVPfm60LnjwhfUkIsB+poNWusrBE0IHllxhvOVE+01nfa1Lvpp5AUWzV+XchPGOCSf+a29ds4EQWfNbg==
+"@abp/luxon@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.1.tgz#ac59a56653d51b997dbbde8ccc58dac5746a9ef4"
+ integrity sha512-1ZHXP5SGz05tQ1tzYmPcRiZwmkCKGX0KWFoV0zF3ATRaevzku9nXvSy7K6GIFkkgTb9f9XNx4/OBBF/GXkuRqw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
luxon "^3.5.0"
-"@abp/malihu-custom-scrollbar-plugin@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.0.tgz#b8a49202d531c711874a2bcdae5081ae0ed34c3f"
- integrity sha512-iSZ9lKMnv5F2SplR+A4blhFEDq79lKbvBQuS7h/Di3KEHFOdJY4pA8NP8OR+PXOF3JfAezGyjqy7ydn/JNPRwA==
+"@abp/malihu-custom-scrollbar-plugin@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.1.tgz#9e60bcfb9f9d76fed9c50e45bf490d5abd0f4689"
+ integrity sha512-GkKPLg9Z9UTTSSpW2hYZYZ9IgCvDSOYJmyRSl/h4fG8cWMgGCPVpv8MPPu0+WtIfwW/mT4SPX9X+sjUlRL5XRQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
malihu-custom-scrollbar-plugin "^3.1.5"
-"@abp/moment@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.0.tgz#9583e97472ddf608a9a5a6b8a37fee602b419900"
- integrity sha512-wgyJSKmBmHLXZJN3dy3zK4i2JvWGgqnvdtrKURy/raEurxlymv8JF5JA971kTv6Kz2OA/OSYttpVJkDGUpPAhA==
+"@abp/moment@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.1.tgz#d67fa25017fe15dae10e0a0fcebb03aeb63d31d3"
+ integrity sha512-bTuCc7pi/DCtgz7t96UBU0wn9UGt4b50JhDCVeTKquh+gP1sgWoI5hyvEBkmgE4dcffO0yzZU2opHYTKYGl/OQ==
dependencies:
moment "^2.30.1"
-"@abp/select2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.0.tgz#71503eebb746f40a0e565dba1efa05d6ae937b77"
- integrity sha512-l+Hz2Qmc7AcyaTJOyuDaV99RbBWYxYxbeyVEc4xKbRFbJxYBUxT/DFlyozEaoz+OU1fl4szO3UCthJ2yZvxCdg==
+"@abp/select2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.1.tgz#9f1c5c27e72bef910fdf63fc3167f415d2ba8db8"
+ integrity sha512-epyYJH2gAOU2UvGn5FD6372hbW3fL7uZIgGpvfgEfa+l6HPdlV9B1OweU2xkONNCoylShAcP/y8qf+bUh8WDEA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
select2 "^4.0.13"
-"@abp/sweetalert2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.0.tgz#71e511117c18b515e5c658e9cc0667770af65d91"
- integrity sha512-wxDqoA6RUhkbcN27Kzslu/m2ai5T9AkJRtgs7pV55LvPv1ohuCPEUeTVVh8S/5Pgmr7aPFAPxaaCO32aPcImPw==
+"@abp/sweetalert2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.1.tgz#a364b21665cb76e92640496dc8a77b8edbce4500"
+ integrity sha512-0RXR/lzy3qMJLisGUz4qrcA9Clu/Tt1ABGzB64ynG1t/l5XessQsIB5Hkt6MbSyOEsprRsq+ZtKH35CwTRPdAw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
sweetalert2 "^11.14.1"
-"@abp/timeago@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.0.tgz#8f262ba2a74e1c133d0c129f8a0bc55c29441e1a"
- integrity sha512-eaNdIUEtDXnuYqS5O7EGRV+j6XOLAsxz7gd5BIKR9lDujd8pEjSJkXKO+/0KxLTkUSy3JudIO37K1h2wTpbYZA==
+"@abp/timeago@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.1.tgz#d1b7e11fdc267bdb736ff4b591afd3fb81c07cce"
+ integrity sha512-dNfUQwilrY+JHxL5lwJ++ixYDGAs5lF1EDxKMQMIXD/ak5L8rkdLns8Otta7FhgDDny9UJ7XWQjYsG8ZxxybRQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
timeago "^1.6.7"
-"@abp/toastr@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.0.tgz#74d74ecb3e7f2d63b2ff7735c89a40deb3dc3a13"
- integrity sha512-n9uJvw8l2HoxeowPO4diEMxwchGh6puefhyRljFAAstzK77Mhn0u2J7kPynNv94rLSYTYoCxUeJ4TTN+/TVDGQ==
+"@abp/toastr@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.1.tgz#2105a639d7cd4fa9c671b4215d719a78fef7e0ee"
+ integrity sha512-10KuHnSJOxkvzqs6Aim6GJp8xBuT+TsZqBE1KaXQnERPawmOo4+ME28kA/7lq8fvgFUNO3K41T2uuA1GPHVQcg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
toastr "^2.1.4"
-"@abp/utils@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.0.tgz#2264fd83105b061860edd98dbe0efdfe55803ed5"
- integrity sha512-/SB4VYk3Av+ltXv6ovkFHG4iWiECLLitaYWcEHxVAO80ufNjrNnWpHFpXHP8u/RpXIp/CoNiHLr6j5TkZUTZjw==
+"@abp/utils@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.1.tgz#dccaf7cf92147304d454ffb9a8664d52dc7e5c5a"
+ integrity sha512-7eOKT0ciKNdtuASck1kifyc4dC6CeebgxV+rx/EAi3YwPMDkqie6YwLGUqKdGW2N/MZ15JuDR6gkuZxKHEb0NQ==
dependencies:
just-compare "^2.3.0"
diff --git a/modules/cms-kit/angular/package.json b/modules/cms-kit/angular/package.json
index 898a2d3e9c..d22ed86827 100644
--- a/modules/cms-kit/angular/package.json
+++ b/modules/cms-kit/angular/package.json
@@ -15,11 +15,11 @@
},
"private": true,
"dependencies": {
- "@abp/ng.account": "~9.0.0",
- "@abp/ng.identity": "~9.0.0",
- "@abp/ng.setting-management": "~9.0.0",
- "@abp/ng.tenant-management": "~9.0.0",
- "@abp/ng.theme.basic": "~9.0.0",
+ "@abp/ng.account": "~9.0.1",
+ "@abp/ng.identity": "~9.0.1",
+ "@abp/ng.setting-management": "~9.0.1",
+ "@abp/ng.tenant-management": "~9.0.1",
+ "@abp/ng.theme.basic": "~9.0.1",
"@angular/animations": "~10.0.0",
"@angular/common": "~10.0.0",
"@angular/compiler": "~10.0.0",
diff --git a/modules/cms-kit/angular/projects/cms-kit/package.json b/modules/cms-kit/angular/projects/cms-kit/package.json
index a7ab8c38c1..390734e951 100644
--- a/modules/cms-kit/angular/projects/cms-kit/package.json
+++ b/modules/cms-kit/angular/projects/cms-kit/package.json
@@ -4,8 +4,8 @@
"peerDependencies": {
"@angular/common": "^9.1.11",
"@angular/core": "^9.1.11",
- "@abp/ng.core": ">=9.0.0",
- "@abp/ng.theme.shared": ">=9.0.0"
+ "@abp/ng.core": ">=9.0.1",
+ "@abp/ng.theme.shared": ">=9.0.1"
},
"dependencies": {
"tslib": "^2.0.0"
diff --git a/modules/cms-kit/host/Volo.CmsKit.IdentityServer/package.json b/modules/cms-kit/host/Volo.CmsKit.IdentityServer/package.json
index 0150cbe740..e1e64e5277 100644
--- a/modules/cms-kit/host/Volo.CmsKit.IdentityServer/package.json
+++ b/modules/cms-kit/host/Volo.CmsKit.IdentityServer/package.json
@@ -3,6 +3,6 @@
"name": "my-app-identityserver",
"private": true,
"dependencies": {
- "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.0"
+ "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.1"
}
}
diff --git a/modules/cms-kit/host/Volo.CmsKit.IdentityServer/yarn.lock b/modules/cms-kit/host/Volo.CmsKit.IdentityServer/yarn.lock
index 9e01130758..0ac91bf6da 100644
--- a/modules/cms-kit/host/Volo.CmsKit.IdentityServer/yarn.lock
+++ b/modules/cms-kit/host/Volo.CmsKit.IdentityServer/yarn.lock
@@ -2,194 +2,194 @@
# yarn lockfile v1
-"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.0.tgz#e4b05d63446c24b612275e3949881e176d32527f"
- integrity sha512-GHEt26xESe/TCg1q6ZjeUbxniqdxaiB1OPTZ9md7ONql6bEp8t9iUPeSlkyAZtQ7qBzwgkiLpkRFT+J21DftaQ==
- dependencies:
- "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.0.tgz#a77b4b70fa6217146afd6a57211f57e2360564f2"
- integrity sha512-QmZrSp+rfqfnC9W4QZ+nNdAq0rOALzRn1rsDpGDHhIHwyXB03QPdjzIWaEvldUYHzqg84cEP0LcqW4y57Xr1iw==
- dependencies:
- "@abp/aspnetcore.mvc.ui" "~9.0.0"
- "@abp/bootstrap" "~9.0.0"
- "@abp/bootstrap-datepicker" "~9.0.0"
- "@abp/bootstrap-daterangepicker" "~9.0.0"
- "@abp/datatables.net-bs5" "~9.0.0"
- "@abp/font-awesome" "~9.0.0"
- "@abp/jquery-form" "~9.0.0"
- "@abp/jquery-validation-unobtrusive" "~9.0.0"
- "@abp/lodash" "~9.0.0"
- "@abp/luxon" "~9.0.0"
- "@abp/malihu-custom-scrollbar-plugin" "~9.0.0"
- "@abp/moment" "~9.0.0"
- "@abp/select2" "~9.0.0"
- "@abp/sweetalert2" "~9.0.0"
- "@abp/timeago" "~9.0.0"
- "@abp/toastr" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.0.tgz#93004dd056c130386c7b40c458f3a793d8c9e051"
- integrity sha512-pvO5bh9baFcSbfW0s7tpMpbe6DKSC16jeyeluZLlvYxC0n2gkrv7H/hB5aMuBKIlh89aGY9DQIcMFC8iczL3Bg==
+"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.1.tgz#cdb6130ac6f8499f9a3f856b596c46745d056a30"
+ integrity sha512-JaxuDH8WTnZYX6C7CX7ZVHfhTDR/SxfwRvtJaGGijSX8QKd1DXpfi9lYEF4Y9k5b48YT/0slBqU5g0sLmBd3+g==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.1.tgz#7a1be780a305aa0d0f467472fc721b6f710fda7a"
+ integrity sha512-vrgRYKNctdZYhPCQ2I6Z10gUI1S5ynDj+DgnV7x7a7G/juuHaOSbkRXgwrmJsmYLfGA7KFCoJcbpER8mmXQeHQ==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui" "~9.0.1"
+ "@abp/bootstrap" "~9.0.1"
+ "@abp/bootstrap-datepicker" "~9.0.1"
+ "@abp/bootstrap-daterangepicker" "~9.0.1"
+ "@abp/datatables.net-bs5" "~9.0.1"
+ "@abp/font-awesome" "~9.0.1"
+ "@abp/jquery-form" "~9.0.1"
+ "@abp/jquery-validation-unobtrusive" "~9.0.1"
+ "@abp/lodash" "~9.0.1"
+ "@abp/luxon" "~9.0.1"
+ "@abp/malihu-custom-scrollbar-plugin" "~9.0.1"
+ "@abp/moment" "~9.0.1"
+ "@abp/select2" "~9.0.1"
+ "@abp/sweetalert2" "~9.0.1"
+ "@abp/timeago" "~9.0.1"
+ "@abp/toastr" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.1.tgz#811dd91138efb82345fc06559b5d2feea88021af"
+ integrity sha512-KVbWCpBV7S1ViHYQP/hn4gSMzSiA2e8/zC0HND1fjE+1zXJqJyEWto1eg2DaKSOs/rmHD8WtFVMU4hCLQTCuIA==
dependencies:
ansi-colors "^4.1.3"
-"@abp/bootstrap-datepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.0.tgz#d6609e50568aac6a013abfcaa07817998e1eaa99"
- integrity sha512-wxAeVsOVOJpidDn0g3FlJvpOsn5SVY5IpEG+FtUxSAKuWYJQWF6g5CaghrfS+FNDK9IHhxrgSMPMVBK9YlDSzw==
+"@abp/bootstrap-datepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.1.tgz#21cd19090b9a571ddf5a9019bc17864c8685408c"
+ integrity sha512-LhgqBRr66x9gGTAyv1FuPAw8u8N77WQK2AfDkNgLueurM2+xMYQzH3pWWLhcN676/JXafxfALCfBoM8U7CfXLg==
dependencies:
bootstrap-datepicker "^1.10.0"
-"@abp/bootstrap-daterangepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.0.tgz#fc2871e7d886accd8e4748cb763b695e10701c00"
- integrity sha512-qS6SQR6X7wMCVLcuXatJ2+m1GNlpN71p7VDritxfwAMHa7WBRAAi8apX2WItwBAuLQ2s1uSA0Wp1KstHG3VPLA==
+"@abp/bootstrap-daterangepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.1.tgz#c19c23ede6fd868a147ea893d4fdecfa60b177f7"
+ integrity sha512-SlBVgiYa75uQyaGt5EVjduren7Po5UP26A1tXEQQr0hDia1+1TK0qxmJ2YxjeJpB7FwtcmJiofv5fngvJqhxIA==
dependencies:
bootstrap-daterangepicker "^3.1.0"
-"@abp/bootstrap@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.0.tgz#5ab292e95ba7eef129cbaa1dc5f5d5c0af8530d7"
- integrity sha512-jEr6su8BfoNxYvze/UlddE3DlezyPvQeiNhff8SDPqtRQ6XvjZ48O/JwtY01hK2njEDLOZINY5zJRGmO2OiOUw==
+"@abp/bootstrap@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.1.tgz#2360303ec24c8f3eb755b10fa6a953db6d858daf"
+ integrity sha512-uAfVNsUYSsRqgeO9JMnhJJlFoDmO3/lTvER/iYOqerF0X03KQnR8HZy2V3MD1D+fSh/4JuU6aAZFl6IQLinMKQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
bootstrap "^5.3.3"
-"@abp/core@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.0.tgz#dc5ea9139957e307f73ac1f40b179f991e3c2168"
- integrity sha512-vzQyTpqVDx63pOI9e2QLUMWZCrSojHsRnzLlvl+/cKFi7jE1q/uSP66FPR2bwkD+DNLI/tn6is1Jx3skKA1FvA==
+"@abp/core@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.1.tgz#727fc1dfac21cca1216c4f5091da7d5efefdb467"
+ integrity sha512-NWXDVCIJePbcKCrRvGxAlaA2MrOF2IZox37bQkW7/mTeOPpNoTa6lhowksNNcTpFq2YdD8bG0i0kfskkp3EZ2w==
dependencies:
- "@abp/utils" "~9.0.0"
+ "@abp/utils" "~9.0.1"
-"@abp/datatables.net-bs5@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.0.tgz#4daee90bdb6446e854ce65f2a3afabb2a13e49ae"
- integrity sha512-ln10tcRwxcB0nuSgqJ/66strHsqMHDC03c0ktxEplAN34yP7KInjFf/U9Svo74K0nKlkJdMCgmqmT4CnBJZlIg==
+"@abp/datatables.net-bs5@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.1.tgz#61e2a3daacf1c074ed6b459e25084cc1f5c4eeab"
+ integrity sha512-tAHdf4TBAS0bVRcvovDKS0OHFyeRSPYbiSIwDn8AIWzW9Weav31jhZ5jL3HEtBKVdb7zqUQ8/3dYPufjDuDdDA==
dependencies:
- "@abp/datatables.net" "~9.0.0"
+ "@abp/datatables.net" "~9.0.1"
datatables.net-bs5 "^2.1.8"
-"@abp/datatables.net@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.0.tgz#82b28518293c9091ec72d5defae2f5d2903d8bbc"
- integrity sha512-QhuX3bGZOckM/fhIIGj/StMJIX2BBtNbLzjTR7WnKMMxDkv0+oDvjw6OqqpYKSvj/q3vY6sRWuCNdEzNi0Fk6A==
+"@abp/datatables.net@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.1.tgz#4cf193e7493f85a272de47326a32a021e3ba40f5"
+ integrity sha512-JY53zVQPluRrBiKwtgk4gfSgD8VDs/VU8LWizhGCprXt3h9vXx0SQL4My6zvXVkv4hUjaq4+MYo84OI8EwFvsg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
datatables.net "^2.1.8"
-"@abp/font-awesome@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.0.tgz#51cce18434a51d050ebb0d822a8c193b7e32f73d"
- integrity sha512-bEHRVzOk0V2kFMAvB/YXFYaDKWRFO0+ScO7jrEFtFgdHXBcqy+dk1aM/N69WalyNmjRRpnh03GhH1LRpwL4/xA==
+"@abp/font-awesome@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.1.tgz#69da15018e7e5f2ff0d3c0d113a4325f54ffc0c3"
+ integrity sha512-7NnYBmk0plLVElUWR1zohul87vE9wKJItemWT5Kskf/IJirKOvXwSSu+7Jg1LmnVtIxcH6qgRpJMmzuAJgawnQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
"@fortawesome/fontawesome-free" "^6.6.0"
-"@abp/jquery-form@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.0.tgz#6518eb117ee35c6bc57543fd1fccb0a8cf8e22f4"
- integrity sha512-6qpfPGHrjADmm7Ne3zcXBGPuLngG8ldAn5fk9k2taftpyX0iR67/nZxiQ0ujaQGgaai5Wb3imNXMG96XLnubkQ==
+"@abp/jquery-form@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.1.tgz#35e0a23cf50b00e0d3b0886f097233b79144145b"
+ integrity sha512-9co0+Cver+RUSOpfuukSWQhR+TCnO9Dly4RTVRIKFksGVfN8hA70NqRrVxaASNNFv42ynK88FQK+2bLPswz/iw==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-form "^4.3.0"
-"@abp/jquery-validation-unobtrusive@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.0.tgz#8ee87dbdec5e92ab04ae9642a49af1598af31c1a"
- integrity sha512-KlRQ1htyFmAIMS9xbHvHIcdhAgtPFe/u7c87MiWlk/CdhDrI3J+n5biLvQUzxCC+YWlwx27zBtPKx2vLeMRH6Q==
+"@abp/jquery-validation-unobtrusive@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.1.tgz#f9564aabebf4d9f90b4ee5c3f3de0e1a348d8ecc"
+ integrity sha512-Zpz48wIFjq7eIApG/sEVjkfaJWZhcyCnlCtDf5eU3vvh82nRetbiCEAVLpZ4LVjorlyp3VvVAxReUSeeWflu8A==
dependencies:
- "@abp/jquery-validation" "~9.0.0"
+ "@abp/jquery-validation" "~9.0.1"
jquery-validation-unobtrusive "^4.0.0"
-"@abp/jquery-validation@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.0.tgz#681d8037c57b3875bd51c927e350ffce65d7c5d2"
- integrity sha512-XgCUBFWbhTDcGepD/xkwA/1Hl4TyYitRttauDJk3tuyXhNtykvdugCCOPY4j2s6VoT1Sb/TFFvOcgEvlBwkKiw==
+"@abp/jquery-validation@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.1.tgz#ad37a44e82fb32ec9bfe2b736dfc8a143fca89c7"
+ integrity sha512-uFSxiPMEkaPlcuyXIoKbeOIIRHo4IrgRbmnlHwI1uETrW60ENUdE2LJAVcYS77JROrjVtv8FPlaffWuva6sGeQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-validation "^1.21.0"
-"@abp/jquery@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.0.tgz#6c67a9f927be598ab6201f8888e8a3cb8bba0686"
- integrity sha512-cUXA7v6F1HqAc/w8fQGxHIfA/6BqO+4uHXDnSsJZtrI26DTLOgw5kBNnAz2W4xggGPCY7SKFHNRZuOW/QfVRPg==
+"@abp/jquery@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.1.tgz#ecce14644c753c581caee0df7bfafd3d083e70f0"
+ integrity sha512-z8EcnOveItAszbTsM9GFFDQSlcIg/ym6+WZME9uldLoetdxkmI9ij0knVeGj5eqMjQaHvTWwrVLObiBbZwYFWw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
jquery "~3.7.1"
-"@abp/lodash@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.0.tgz#a4d42b2dfeb37859ff9f0079c7a8a6cabc982efe"
- integrity sha512-xoAsOkkgnQdPH2OyFTgF6OlCLVbyGF4u8CrFcZWCZmuIFT9cg0+ZEiq//Vog1pg3Lf//N37ceGpOag1LwMe6cQ==
+"@abp/lodash@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.1.tgz#c52c8a4125ecfbeaad7badc5c28bf86d5168799a"
+ integrity sha512-D+/iOyh/O+jV80bK5vWxyn++IVmeRbZq/UaXanA4RKpzigxJUsgOiSD9lUj85BkfKn2YyFmoR39xwaAqkWoyRA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
lodash "^4.17.21"
-"@abp/luxon@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.0.tgz#a54dda13cbbd362481994d20d2e07c669c6ce109"
- integrity sha512-aPejQnoVPfm60LnjwhfUkIsB+poNWusrBE0IHllxhvOVE+01nfa1Lvpp5AUWzV+XchPGOCSf+a29ds4EQWfNbg==
+"@abp/luxon@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.1.tgz#ac59a56653d51b997dbbde8ccc58dac5746a9ef4"
+ integrity sha512-1ZHXP5SGz05tQ1tzYmPcRiZwmkCKGX0KWFoV0zF3ATRaevzku9nXvSy7K6GIFkkgTb9f9XNx4/OBBF/GXkuRqw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
luxon "^3.5.0"
-"@abp/malihu-custom-scrollbar-plugin@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.0.tgz#b8a49202d531c711874a2bcdae5081ae0ed34c3f"
- integrity sha512-iSZ9lKMnv5F2SplR+A4blhFEDq79lKbvBQuS7h/Di3KEHFOdJY4pA8NP8OR+PXOF3JfAezGyjqy7ydn/JNPRwA==
+"@abp/malihu-custom-scrollbar-plugin@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.1.tgz#9e60bcfb9f9d76fed9c50e45bf490d5abd0f4689"
+ integrity sha512-GkKPLg9Z9UTTSSpW2hYZYZ9IgCvDSOYJmyRSl/h4fG8cWMgGCPVpv8MPPu0+WtIfwW/mT4SPX9X+sjUlRL5XRQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
malihu-custom-scrollbar-plugin "^3.1.5"
-"@abp/moment@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.0.tgz#9583e97472ddf608a9a5a6b8a37fee602b419900"
- integrity sha512-wgyJSKmBmHLXZJN3dy3zK4i2JvWGgqnvdtrKURy/raEurxlymv8JF5JA971kTv6Kz2OA/OSYttpVJkDGUpPAhA==
+"@abp/moment@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.1.tgz#d67fa25017fe15dae10e0a0fcebb03aeb63d31d3"
+ integrity sha512-bTuCc7pi/DCtgz7t96UBU0wn9UGt4b50JhDCVeTKquh+gP1sgWoI5hyvEBkmgE4dcffO0yzZU2opHYTKYGl/OQ==
dependencies:
moment "^2.30.1"
-"@abp/select2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.0.tgz#71503eebb746f40a0e565dba1efa05d6ae937b77"
- integrity sha512-l+Hz2Qmc7AcyaTJOyuDaV99RbBWYxYxbeyVEc4xKbRFbJxYBUxT/DFlyozEaoz+OU1fl4szO3UCthJ2yZvxCdg==
+"@abp/select2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.1.tgz#9f1c5c27e72bef910fdf63fc3167f415d2ba8db8"
+ integrity sha512-epyYJH2gAOU2UvGn5FD6372hbW3fL7uZIgGpvfgEfa+l6HPdlV9B1OweU2xkONNCoylShAcP/y8qf+bUh8WDEA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
select2 "^4.0.13"
-"@abp/sweetalert2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.0.tgz#71e511117c18b515e5c658e9cc0667770af65d91"
- integrity sha512-wxDqoA6RUhkbcN27Kzslu/m2ai5T9AkJRtgs7pV55LvPv1ohuCPEUeTVVh8S/5Pgmr7aPFAPxaaCO32aPcImPw==
+"@abp/sweetalert2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.1.tgz#a364b21665cb76e92640496dc8a77b8edbce4500"
+ integrity sha512-0RXR/lzy3qMJLisGUz4qrcA9Clu/Tt1ABGzB64ynG1t/l5XessQsIB5Hkt6MbSyOEsprRsq+ZtKH35CwTRPdAw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
sweetalert2 "^11.14.1"
-"@abp/timeago@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.0.tgz#8f262ba2a74e1c133d0c129f8a0bc55c29441e1a"
- integrity sha512-eaNdIUEtDXnuYqS5O7EGRV+j6XOLAsxz7gd5BIKR9lDujd8pEjSJkXKO+/0KxLTkUSy3JudIO37K1h2wTpbYZA==
+"@abp/timeago@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.1.tgz#d1b7e11fdc267bdb736ff4b591afd3fb81c07cce"
+ integrity sha512-dNfUQwilrY+JHxL5lwJ++ixYDGAs5lF1EDxKMQMIXD/ak5L8rkdLns8Otta7FhgDDny9UJ7XWQjYsG8ZxxybRQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
timeago "^1.6.7"
-"@abp/toastr@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.0.tgz#74d74ecb3e7f2d63b2ff7735c89a40deb3dc3a13"
- integrity sha512-n9uJvw8l2HoxeowPO4diEMxwchGh6puefhyRljFAAstzK77Mhn0u2J7kPynNv94rLSYTYoCxUeJ4TTN+/TVDGQ==
+"@abp/toastr@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.1.tgz#2105a639d7cd4fa9c671b4215d719a78fef7e0ee"
+ integrity sha512-10KuHnSJOxkvzqs6Aim6GJp8xBuT+TsZqBE1KaXQnERPawmOo4+ME28kA/7lq8fvgFUNO3K41T2uuA1GPHVQcg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
toastr "^2.1.4"
-"@abp/utils@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.0.tgz#2264fd83105b061860edd98dbe0efdfe55803ed5"
- integrity sha512-/SB4VYk3Av+ltXv6ovkFHG4iWiECLLitaYWcEHxVAO80ufNjrNnWpHFpXHP8u/RpXIp/CoNiHLr6j5TkZUTZjw==
+"@abp/utils@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.1.tgz#dccaf7cf92147304d454ffb9a8664d52dc7e5c5a"
+ integrity sha512-7eOKT0ciKNdtuASck1kifyc4dC6CeebgxV+rx/EAi3YwPMDkqie6YwLGUqKdGW2N/MZ15JuDR6gkuZxKHEb0NQ==
dependencies:
just-compare "^2.3.0"
diff --git a/modules/cms-kit/host/Volo.CmsKit.Web.Host/package.json b/modules/cms-kit/host/Volo.CmsKit.Web.Host/package.json
index 8b5210b25f..716eaed26c 100644
--- a/modules/cms-kit/host/Volo.CmsKit.Web.Host/package.json
+++ b/modules/cms-kit/host/Volo.CmsKit.Web.Host/package.json
@@ -3,6 +3,6 @@
"name": "my-app",
"private": true,
"dependencies": {
- "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.0"
+ "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.1"
}
}
diff --git a/modules/cms-kit/host/Volo.CmsKit.Web.Host/yarn.lock b/modules/cms-kit/host/Volo.CmsKit.Web.Host/yarn.lock
index 9e01130758..0ac91bf6da 100644
--- a/modules/cms-kit/host/Volo.CmsKit.Web.Host/yarn.lock
+++ b/modules/cms-kit/host/Volo.CmsKit.Web.Host/yarn.lock
@@ -2,194 +2,194 @@
# yarn lockfile v1
-"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.0.tgz#e4b05d63446c24b612275e3949881e176d32527f"
- integrity sha512-GHEt26xESe/TCg1q6ZjeUbxniqdxaiB1OPTZ9md7ONql6bEp8t9iUPeSlkyAZtQ7qBzwgkiLpkRFT+J21DftaQ==
- dependencies:
- "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.0.tgz#a77b4b70fa6217146afd6a57211f57e2360564f2"
- integrity sha512-QmZrSp+rfqfnC9W4QZ+nNdAq0rOALzRn1rsDpGDHhIHwyXB03QPdjzIWaEvldUYHzqg84cEP0LcqW4y57Xr1iw==
- dependencies:
- "@abp/aspnetcore.mvc.ui" "~9.0.0"
- "@abp/bootstrap" "~9.0.0"
- "@abp/bootstrap-datepicker" "~9.0.0"
- "@abp/bootstrap-daterangepicker" "~9.0.0"
- "@abp/datatables.net-bs5" "~9.0.0"
- "@abp/font-awesome" "~9.0.0"
- "@abp/jquery-form" "~9.0.0"
- "@abp/jquery-validation-unobtrusive" "~9.0.0"
- "@abp/lodash" "~9.0.0"
- "@abp/luxon" "~9.0.0"
- "@abp/malihu-custom-scrollbar-plugin" "~9.0.0"
- "@abp/moment" "~9.0.0"
- "@abp/select2" "~9.0.0"
- "@abp/sweetalert2" "~9.0.0"
- "@abp/timeago" "~9.0.0"
- "@abp/toastr" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.0.tgz#93004dd056c130386c7b40c458f3a793d8c9e051"
- integrity sha512-pvO5bh9baFcSbfW0s7tpMpbe6DKSC16jeyeluZLlvYxC0n2gkrv7H/hB5aMuBKIlh89aGY9DQIcMFC8iczL3Bg==
+"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.1.tgz#cdb6130ac6f8499f9a3f856b596c46745d056a30"
+ integrity sha512-JaxuDH8WTnZYX6C7CX7ZVHfhTDR/SxfwRvtJaGGijSX8QKd1DXpfi9lYEF4Y9k5b48YT/0slBqU5g0sLmBd3+g==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.1.tgz#7a1be780a305aa0d0f467472fc721b6f710fda7a"
+ integrity sha512-vrgRYKNctdZYhPCQ2I6Z10gUI1S5ynDj+DgnV7x7a7G/juuHaOSbkRXgwrmJsmYLfGA7KFCoJcbpER8mmXQeHQ==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui" "~9.0.1"
+ "@abp/bootstrap" "~9.0.1"
+ "@abp/bootstrap-datepicker" "~9.0.1"
+ "@abp/bootstrap-daterangepicker" "~9.0.1"
+ "@abp/datatables.net-bs5" "~9.0.1"
+ "@abp/font-awesome" "~9.0.1"
+ "@abp/jquery-form" "~9.0.1"
+ "@abp/jquery-validation-unobtrusive" "~9.0.1"
+ "@abp/lodash" "~9.0.1"
+ "@abp/luxon" "~9.0.1"
+ "@abp/malihu-custom-scrollbar-plugin" "~9.0.1"
+ "@abp/moment" "~9.0.1"
+ "@abp/select2" "~9.0.1"
+ "@abp/sweetalert2" "~9.0.1"
+ "@abp/timeago" "~9.0.1"
+ "@abp/toastr" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.1.tgz#811dd91138efb82345fc06559b5d2feea88021af"
+ integrity sha512-KVbWCpBV7S1ViHYQP/hn4gSMzSiA2e8/zC0HND1fjE+1zXJqJyEWto1eg2DaKSOs/rmHD8WtFVMU4hCLQTCuIA==
dependencies:
ansi-colors "^4.1.3"
-"@abp/bootstrap-datepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.0.tgz#d6609e50568aac6a013abfcaa07817998e1eaa99"
- integrity sha512-wxAeVsOVOJpidDn0g3FlJvpOsn5SVY5IpEG+FtUxSAKuWYJQWF6g5CaghrfS+FNDK9IHhxrgSMPMVBK9YlDSzw==
+"@abp/bootstrap-datepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.1.tgz#21cd19090b9a571ddf5a9019bc17864c8685408c"
+ integrity sha512-LhgqBRr66x9gGTAyv1FuPAw8u8N77WQK2AfDkNgLueurM2+xMYQzH3pWWLhcN676/JXafxfALCfBoM8U7CfXLg==
dependencies:
bootstrap-datepicker "^1.10.0"
-"@abp/bootstrap-daterangepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.0.tgz#fc2871e7d886accd8e4748cb763b695e10701c00"
- integrity sha512-qS6SQR6X7wMCVLcuXatJ2+m1GNlpN71p7VDritxfwAMHa7WBRAAi8apX2WItwBAuLQ2s1uSA0Wp1KstHG3VPLA==
+"@abp/bootstrap-daterangepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.1.tgz#c19c23ede6fd868a147ea893d4fdecfa60b177f7"
+ integrity sha512-SlBVgiYa75uQyaGt5EVjduren7Po5UP26A1tXEQQr0hDia1+1TK0qxmJ2YxjeJpB7FwtcmJiofv5fngvJqhxIA==
dependencies:
bootstrap-daterangepicker "^3.1.0"
-"@abp/bootstrap@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.0.tgz#5ab292e95ba7eef129cbaa1dc5f5d5c0af8530d7"
- integrity sha512-jEr6su8BfoNxYvze/UlddE3DlezyPvQeiNhff8SDPqtRQ6XvjZ48O/JwtY01hK2njEDLOZINY5zJRGmO2OiOUw==
+"@abp/bootstrap@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.1.tgz#2360303ec24c8f3eb755b10fa6a953db6d858daf"
+ integrity sha512-uAfVNsUYSsRqgeO9JMnhJJlFoDmO3/lTvER/iYOqerF0X03KQnR8HZy2V3MD1D+fSh/4JuU6aAZFl6IQLinMKQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
bootstrap "^5.3.3"
-"@abp/core@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.0.tgz#dc5ea9139957e307f73ac1f40b179f991e3c2168"
- integrity sha512-vzQyTpqVDx63pOI9e2QLUMWZCrSojHsRnzLlvl+/cKFi7jE1q/uSP66FPR2bwkD+DNLI/tn6is1Jx3skKA1FvA==
+"@abp/core@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.1.tgz#727fc1dfac21cca1216c4f5091da7d5efefdb467"
+ integrity sha512-NWXDVCIJePbcKCrRvGxAlaA2MrOF2IZox37bQkW7/mTeOPpNoTa6lhowksNNcTpFq2YdD8bG0i0kfskkp3EZ2w==
dependencies:
- "@abp/utils" "~9.0.0"
+ "@abp/utils" "~9.0.1"
-"@abp/datatables.net-bs5@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.0.tgz#4daee90bdb6446e854ce65f2a3afabb2a13e49ae"
- integrity sha512-ln10tcRwxcB0nuSgqJ/66strHsqMHDC03c0ktxEplAN34yP7KInjFf/U9Svo74K0nKlkJdMCgmqmT4CnBJZlIg==
+"@abp/datatables.net-bs5@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.1.tgz#61e2a3daacf1c074ed6b459e25084cc1f5c4eeab"
+ integrity sha512-tAHdf4TBAS0bVRcvovDKS0OHFyeRSPYbiSIwDn8AIWzW9Weav31jhZ5jL3HEtBKVdb7zqUQ8/3dYPufjDuDdDA==
dependencies:
- "@abp/datatables.net" "~9.0.0"
+ "@abp/datatables.net" "~9.0.1"
datatables.net-bs5 "^2.1.8"
-"@abp/datatables.net@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.0.tgz#82b28518293c9091ec72d5defae2f5d2903d8bbc"
- integrity sha512-QhuX3bGZOckM/fhIIGj/StMJIX2BBtNbLzjTR7WnKMMxDkv0+oDvjw6OqqpYKSvj/q3vY6sRWuCNdEzNi0Fk6A==
+"@abp/datatables.net@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.1.tgz#4cf193e7493f85a272de47326a32a021e3ba40f5"
+ integrity sha512-JY53zVQPluRrBiKwtgk4gfSgD8VDs/VU8LWizhGCprXt3h9vXx0SQL4My6zvXVkv4hUjaq4+MYo84OI8EwFvsg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
datatables.net "^2.1.8"
-"@abp/font-awesome@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.0.tgz#51cce18434a51d050ebb0d822a8c193b7e32f73d"
- integrity sha512-bEHRVzOk0V2kFMAvB/YXFYaDKWRFO0+ScO7jrEFtFgdHXBcqy+dk1aM/N69WalyNmjRRpnh03GhH1LRpwL4/xA==
+"@abp/font-awesome@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.1.tgz#69da15018e7e5f2ff0d3c0d113a4325f54ffc0c3"
+ integrity sha512-7NnYBmk0plLVElUWR1zohul87vE9wKJItemWT5Kskf/IJirKOvXwSSu+7Jg1LmnVtIxcH6qgRpJMmzuAJgawnQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
"@fortawesome/fontawesome-free" "^6.6.0"
-"@abp/jquery-form@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.0.tgz#6518eb117ee35c6bc57543fd1fccb0a8cf8e22f4"
- integrity sha512-6qpfPGHrjADmm7Ne3zcXBGPuLngG8ldAn5fk9k2taftpyX0iR67/nZxiQ0ujaQGgaai5Wb3imNXMG96XLnubkQ==
+"@abp/jquery-form@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.1.tgz#35e0a23cf50b00e0d3b0886f097233b79144145b"
+ integrity sha512-9co0+Cver+RUSOpfuukSWQhR+TCnO9Dly4RTVRIKFksGVfN8hA70NqRrVxaASNNFv42ynK88FQK+2bLPswz/iw==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-form "^4.3.0"
-"@abp/jquery-validation-unobtrusive@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.0.tgz#8ee87dbdec5e92ab04ae9642a49af1598af31c1a"
- integrity sha512-KlRQ1htyFmAIMS9xbHvHIcdhAgtPFe/u7c87MiWlk/CdhDrI3J+n5biLvQUzxCC+YWlwx27zBtPKx2vLeMRH6Q==
+"@abp/jquery-validation-unobtrusive@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.1.tgz#f9564aabebf4d9f90b4ee5c3f3de0e1a348d8ecc"
+ integrity sha512-Zpz48wIFjq7eIApG/sEVjkfaJWZhcyCnlCtDf5eU3vvh82nRetbiCEAVLpZ4LVjorlyp3VvVAxReUSeeWflu8A==
dependencies:
- "@abp/jquery-validation" "~9.0.0"
+ "@abp/jquery-validation" "~9.0.1"
jquery-validation-unobtrusive "^4.0.0"
-"@abp/jquery-validation@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.0.tgz#681d8037c57b3875bd51c927e350ffce65d7c5d2"
- integrity sha512-XgCUBFWbhTDcGepD/xkwA/1Hl4TyYitRttauDJk3tuyXhNtykvdugCCOPY4j2s6VoT1Sb/TFFvOcgEvlBwkKiw==
+"@abp/jquery-validation@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.1.tgz#ad37a44e82fb32ec9bfe2b736dfc8a143fca89c7"
+ integrity sha512-uFSxiPMEkaPlcuyXIoKbeOIIRHo4IrgRbmnlHwI1uETrW60ENUdE2LJAVcYS77JROrjVtv8FPlaffWuva6sGeQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-validation "^1.21.0"
-"@abp/jquery@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.0.tgz#6c67a9f927be598ab6201f8888e8a3cb8bba0686"
- integrity sha512-cUXA7v6F1HqAc/w8fQGxHIfA/6BqO+4uHXDnSsJZtrI26DTLOgw5kBNnAz2W4xggGPCY7SKFHNRZuOW/QfVRPg==
+"@abp/jquery@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.1.tgz#ecce14644c753c581caee0df7bfafd3d083e70f0"
+ integrity sha512-z8EcnOveItAszbTsM9GFFDQSlcIg/ym6+WZME9uldLoetdxkmI9ij0knVeGj5eqMjQaHvTWwrVLObiBbZwYFWw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
jquery "~3.7.1"
-"@abp/lodash@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.0.tgz#a4d42b2dfeb37859ff9f0079c7a8a6cabc982efe"
- integrity sha512-xoAsOkkgnQdPH2OyFTgF6OlCLVbyGF4u8CrFcZWCZmuIFT9cg0+ZEiq//Vog1pg3Lf//N37ceGpOag1LwMe6cQ==
+"@abp/lodash@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.1.tgz#c52c8a4125ecfbeaad7badc5c28bf86d5168799a"
+ integrity sha512-D+/iOyh/O+jV80bK5vWxyn++IVmeRbZq/UaXanA4RKpzigxJUsgOiSD9lUj85BkfKn2YyFmoR39xwaAqkWoyRA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
lodash "^4.17.21"
-"@abp/luxon@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.0.tgz#a54dda13cbbd362481994d20d2e07c669c6ce109"
- integrity sha512-aPejQnoVPfm60LnjwhfUkIsB+poNWusrBE0IHllxhvOVE+01nfa1Lvpp5AUWzV+XchPGOCSf+a29ds4EQWfNbg==
+"@abp/luxon@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.1.tgz#ac59a56653d51b997dbbde8ccc58dac5746a9ef4"
+ integrity sha512-1ZHXP5SGz05tQ1tzYmPcRiZwmkCKGX0KWFoV0zF3ATRaevzku9nXvSy7K6GIFkkgTb9f9XNx4/OBBF/GXkuRqw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
luxon "^3.5.0"
-"@abp/malihu-custom-scrollbar-plugin@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.0.tgz#b8a49202d531c711874a2bcdae5081ae0ed34c3f"
- integrity sha512-iSZ9lKMnv5F2SplR+A4blhFEDq79lKbvBQuS7h/Di3KEHFOdJY4pA8NP8OR+PXOF3JfAezGyjqy7ydn/JNPRwA==
+"@abp/malihu-custom-scrollbar-plugin@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.1.tgz#9e60bcfb9f9d76fed9c50e45bf490d5abd0f4689"
+ integrity sha512-GkKPLg9Z9UTTSSpW2hYZYZ9IgCvDSOYJmyRSl/h4fG8cWMgGCPVpv8MPPu0+WtIfwW/mT4SPX9X+sjUlRL5XRQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
malihu-custom-scrollbar-plugin "^3.1.5"
-"@abp/moment@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.0.tgz#9583e97472ddf608a9a5a6b8a37fee602b419900"
- integrity sha512-wgyJSKmBmHLXZJN3dy3zK4i2JvWGgqnvdtrKURy/raEurxlymv8JF5JA971kTv6Kz2OA/OSYttpVJkDGUpPAhA==
+"@abp/moment@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.1.tgz#d67fa25017fe15dae10e0a0fcebb03aeb63d31d3"
+ integrity sha512-bTuCc7pi/DCtgz7t96UBU0wn9UGt4b50JhDCVeTKquh+gP1sgWoI5hyvEBkmgE4dcffO0yzZU2opHYTKYGl/OQ==
dependencies:
moment "^2.30.1"
-"@abp/select2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.0.tgz#71503eebb746f40a0e565dba1efa05d6ae937b77"
- integrity sha512-l+Hz2Qmc7AcyaTJOyuDaV99RbBWYxYxbeyVEc4xKbRFbJxYBUxT/DFlyozEaoz+OU1fl4szO3UCthJ2yZvxCdg==
+"@abp/select2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.1.tgz#9f1c5c27e72bef910fdf63fc3167f415d2ba8db8"
+ integrity sha512-epyYJH2gAOU2UvGn5FD6372hbW3fL7uZIgGpvfgEfa+l6HPdlV9B1OweU2xkONNCoylShAcP/y8qf+bUh8WDEA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
select2 "^4.0.13"
-"@abp/sweetalert2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.0.tgz#71e511117c18b515e5c658e9cc0667770af65d91"
- integrity sha512-wxDqoA6RUhkbcN27Kzslu/m2ai5T9AkJRtgs7pV55LvPv1ohuCPEUeTVVh8S/5Pgmr7aPFAPxaaCO32aPcImPw==
+"@abp/sweetalert2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.1.tgz#a364b21665cb76e92640496dc8a77b8edbce4500"
+ integrity sha512-0RXR/lzy3qMJLisGUz4qrcA9Clu/Tt1ABGzB64ynG1t/l5XessQsIB5Hkt6MbSyOEsprRsq+ZtKH35CwTRPdAw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
sweetalert2 "^11.14.1"
-"@abp/timeago@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.0.tgz#8f262ba2a74e1c133d0c129f8a0bc55c29441e1a"
- integrity sha512-eaNdIUEtDXnuYqS5O7EGRV+j6XOLAsxz7gd5BIKR9lDujd8pEjSJkXKO+/0KxLTkUSy3JudIO37K1h2wTpbYZA==
+"@abp/timeago@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.1.tgz#d1b7e11fdc267bdb736ff4b591afd3fb81c07cce"
+ integrity sha512-dNfUQwilrY+JHxL5lwJ++ixYDGAs5lF1EDxKMQMIXD/ak5L8rkdLns8Otta7FhgDDny9UJ7XWQjYsG8ZxxybRQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
timeago "^1.6.7"
-"@abp/toastr@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.0.tgz#74d74ecb3e7f2d63b2ff7735c89a40deb3dc3a13"
- integrity sha512-n9uJvw8l2HoxeowPO4diEMxwchGh6puefhyRljFAAstzK77Mhn0u2J7kPynNv94rLSYTYoCxUeJ4TTN+/TVDGQ==
+"@abp/toastr@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.1.tgz#2105a639d7cd4fa9c671b4215d719a78fef7e0ee"
+ integrity sha512-10KuHnSJOxkvzqs6Aim6GJp8xBuT+TsZqBE1KaXQnERPawmOo4+ME28kA/7lq8fvgFUNO3K41T2uuA1GPHVQcg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
toastr "^2.1.4"
-"@abp/utils@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.0.tgz#2264fd83105b061860edd98dbe0efdfe55803ed5"
- integrity sha512-/SB4VYk3Av+ltXv6ovkFHG4iWiECLLitaYWcEHxVAO80ufNjrNnWpHFpXHP8u/RpXIp/CoNiHLr6j5TkZUTZjw==
+"@abp/utils@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.1.tgz#dccaf7cf92147304d454ffb9a8664d52dc7e5c5a"
+ integrity sha512-7eOKT0ciKNdtuASck1kifyc4dC6CeebgxV+rx/EAi3YwPMDkqie6YwLGUqKdGW2N/MZ15JuDR6gkuZxKHEb0NQ==
dependencies:
just-compare "^2.3.0"
diff --git a/modules/cms-kit/host/Volo.CmsKit.Web.Unified/package.json b/modules/cms-kit/host/Volo.CmsKit.Web.Unified/package.json
index 5619ca083e..bd0c0c849c 100644
--- a/modules/cms-kit/host/Volo.CmsKit.Web.Unified/package.json
+++ b/modules/cms-kit/host/Volo.CmsKit.Web.Unified/package.json
@@ -3,7 +3,7 @@
"name": "my-app",
"private": true,
"dependencies": {
- "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.0",
- "@abp/cms-kit": "9.0.0"
+ "@abp/aspnetcore.mvc.ui.theme.basic": "~9.0.1",
+ "@abp/cms-kit": "9.0.1"
}
}
diff --git a/modules/cms-kit/host/Volo.CmsKit.Web.Unified/yarn.lock b/modules/cms-kit/host/Volo.CmsKit.Web.Unified/yarn.lock
index 3c5c5f9e97..add9990c9d 100644
--- a/modules/cms-kit/host/Volo.CmsKit.Web.Unified/yarn.lock
+++ b/modules/cms-kit/host/Volo.CmsKit.Web.Unified/yarn.lock
@@ -2,302 +2,302 @@
# yarn lockfile v1
-"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.0.tgz#e4b05d63446c24b612275e3949881e176d32527f"
- integrity sha512-GHEt26xESe/TCg1q6ZjeUbxniqdxaiB1OPTZ9md7ONql6bEp8t9iUPeSlkyAZtQ7qBzwgkiLpkRFT+J21DftaQ==
- dependencies:
- "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.0.tgz#a77b4b70fa6217146afd6a57211f57e2360564f2"
- integrity sha512-QmZrSp+rfqfnC9W4QZ+nNdAq0rOALzRn1rsDpGDHhIHwyXB03QPdjzIWaEvldUYHzqg84cEP0LcqW4y57Xr1iw==
- dependencies:
- "@abp/aspnetcore.mvc.ui" "~9.0.0"
- "@abp/bootstrap" "~9.0.0"
- "@abp/bootstrap-datepicker" "~9.0.0"
- "@abp/bootstrap-daterangepicker" "~9.0.0"
- "@abp/datatables.net-bs5" "~9.0.0"
- "@abp/font-awesome" "~9.0.0"
- "@abp/jquery-form" "~9.0.0"
- "@abp/jquery-validation-unobtrusive" "~9.0.0"
- "@abp/lodash" "~9.0.0"
- "@abp/luxon" "~9.0.0"
- "@abp/malihu-custom-scrollbar-plugin" "~9.0.0"
- "@abp/moment" "~9.0.0"
- "@abp/select2" "~9.0.0"
- "@abp/sweetalert2" "~9.0.0"
- "@abp/timeago" "~9.0.0"
- "@abp/toastr" "~9.0.0"
-
-"@abp/aspnetcore.mvc.ui@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.0.tgz#93004dd056c130386c7b40c458f3a793d8c9e051"
- integrity sha512-pvO5bh9baFcSbfW0s7tpMpbe6DKSC16jeyeluZLlvYxC0n2gkrv7H/hB5aMuBKIlh89aGY9DQIcMFC8iczL3Bg==
+"@abp/aspnetcore.mvc.ui.theme.basic@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.basic/-/aspnetcore.mvc.ui.theme.basic-9.0.1.tgz#cdb6130ac6f8499f9a3f856b596c46745d056a30"
+ integrity sha512-JaxuDH8WTnZYX6C7CX7ZVHfhTDR/SxfwRvtJaGGijSX8QKd1DXpfi9lYEF4Y9k5b48YT/0slBqU5g0sLmBd3+g==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui.theme.shared" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui.theme.shared@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui.theme.shared/-/aspnetcore.mvc.ui.theme.shared-9.0.1.tgz#7a1be780a305aa0d0f467472fc721b6f710fda7a"
+ integrity sha512-vrgRYKNctdZYhPCQ2I6Z10gUI1S5ynDj+DgnV7x7a7G/juuHaOSbkRXgwrmJsmYLfGA7KFCoJcbpER8mmXQeHQ==
+ dependencies:
+ "@abp/aspnetcore.mvc.ui" "~9.0.1"
+ "@abp/bootstrap" "~9.0.1"
+ "@abp/bootstrap-datepicker" "~9.0.1"
+ "@abp/bootstrap-daterangepicker" "~9.0.1"
+ "@abp/datatables.net-bs5" "~9.0.1"
+ "@abp/font-awesome" "~9.0.1"
+ "@abp/jquery-form" "~9.0.1"
+ "@abp/jquery-validation-unobtrusive" "~9.0.1"
+ "@abp/lodash" "~9.0.1"
+ "@abp/luxon" "~9.0.1"
+ "@abp/malihu-custom-scrollbar-plugin" "~9.0.1"
+ "@abp/moment" "~9.0.1"
+ "@abp/select2" "~9.0.1"
+ "@abp/sweetalert2" "~9.0.1"
+ "@abp/timeago" "~9.0.1"
+ "@abp/toastr" "~9.0.1"
+
+"@abp/aspnetcore.mvc.ui@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/aspnetcore.mvc.ui/-/aspnetcore.mvc.ui-9.0.1.tgz#811dd91138efb82345fc06559b5d2feea88021af"
+ integrity sha512-KVbWCpBV7S1ViHYQP/hn4gSMzSiA2e8/zC0HND1fjE+1zXJqJyEWto1eg2DaKSOs/rmHD8WtFVMU4hCLQTCuIA==
dependencies:
ansi-colors "^4.1.3"
-"@abp/bootstrap-datepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.0.tgz#d6609e50568aac6a013abfcaa07817998e1eaa99"
- integrity sha512-wxAeVsOVOJpidDn0g3FlJvpOsn5SVY5IpEG+FtUxSAKuWYJQWF6g5CaghrfS+FNDK9IHhxrgSMPMVBK9YlDSzw==
+"@abp/bootstrap-datepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-datepicker/-/bootstrap-datepicker-9.0.1.tgz#21cd19090b9a571ddf5a9019bc17864c8685408c"
+ integrity sha512-LhgqBRr66x9gGTAyv1FuPAw8u8N77WQK2AfDkNgLueurM2+xMYQzH3pWWLhcN676/JXafxfALCfBoM8U7CfXLg==
dependencies:
bootstrap-datepicker "^1.10.0"
-"@abp/bootstrap-daterangepicker@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.0.tgz#fc2871e7d886accd8e4748cb763b695e10701c00"
- integrity sha512-qS6SQR6X7wMCVLcuXatJ2+m1GNlpN71p7VDritxfwAMHa7WBRAAi8apX2WItwBAuLQ2s1uSA0Wp1KstHG3VPLA==
+"@abp/bootstrap-daterangepicker@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap-daterangepicker/-/bootstrap-daterangepicker-9.0.1.tgz#c19c23ede6fd868a147ea893d4fdecfa60b177f7"
+ integrity sha512-SlBVgiYa75uQyaGt5EVjduren7Po5UP26A1tXEQQr0hDia1+1TK0qxmJ2YxjeJpB7FwtcmJiofv5fngvJqhxIA==
dependencies:
bootstrap-daterangepicker "^3.1.0"
-"@abp/bootstrap@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.0.tgz#5ab292e95ba7eef129cbaa1dc5f5d5c0af8530d7"
- integrity sha512-jEr6su8BfoNxYvze/UlddE3DlezyPvQeiNhff8SDPqtRQ6XvjZ48O/JwtY01hK2njEDLOZINY5zJRGmO2OiOUw==
+"@abp/bootstrap@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/bootstrap/-/bootstrap-9.0.1.tgz#2360303ec24c8f3eb755b10fa6a953db6d858daf"
+ integrity sha512-uAfVNsUYSsRqgeO9JMnhJJlFoDmO3/lTvER/iYOqerF0X03KQnR8HZy2V3MD1D+fSh/4JuU6aAZFl6IQLinMKQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
bootstrap "^5.3.3"
-"@abp/clipboard@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/clipboard/-/clipboard-9.0.0.tgz#21ae924dfef6ccf3c8a3913bb81caf22b57404e4"
- integrity sha512-W6S7pihpQ6Wfj48Clk7hYgDvUtnRwkNIsk5t8s9+LxoAdJQGQFxh7SAW1QNavUv8QqsClNE1VtSU87c7rSoAbA==
+"@abp/clipboard@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/clipboard/-/clipboard-9.0.1.tgz#b1712b2d940caecc85eb99247a569ffe3ab1a387"
+ integrity sha512-obDYqUTf+65E26yHZO2DigQOeDnzAEvak0Umz91cholZHVRvzc9X1itv/abS/whU0YOU/d8Viud8zFv1MnM83w==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
clipboard "^2.0.11"
-"@abp/cms-kit.admin@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/cms-kit.admin/-/cms-kit.admin-9.0.0.tgz#584337c9d751c9eae26be8f276683b5c3c23a499"
- integrity sha512-SYixIuy7McA9UhaKmun8IG0QBmegvk8KBrE/6Uvo1Cmd6Z/oT9SS+i/f09+7UVtU+7x5kkL+9sIs/7sS79cY0g==
+"@abp/cms-kit.admin@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/cms-kit.admin/-/cms-kit.admin-9.0.1.tgz#f71256abfab7f8fe498061a22be5a5d3763e8304"
+ integrity sha512-cAYmPYNJOnIvsBsrTrOeQebUMFxIVd74m/pGVId59yago01CMjUSbKIjhNx9Mj5udzPBI9qDGg/B8fqTh4hlwQ==
dependencies:
- "@abp/codemirror" "~9.0.0"
- "@abp/jstree" "~9.0.0"
- "@abp/markdown-it" "~9.0.0"
- "@abp/slugify" "~9.0.0"
- "@abp/tui-editor" "~9.0.0"
- "@abp/uppy" "~9.0.0"
+ "@abp/codemirror" "~9.0.1"
+ "@abp/jstree" "~9.0.1"
+ "@abp/markdown-it" "~9.0.1"
+ "@abp/slugify" "~9.0.1"
+ "@abp/tui-editor" "~9.0.1"
+ "@abp/uppy" "~9.0.1"
-"@abp/cms-kit.public@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/cms-kit.public/-/cms-kit.public-9.0.0.tgz#bdc8f60bc5b1ffa640baced8c5562f93f7e8b754"
- integrity sha512-s141fUfN9rcRdT4KKjR7PthITnFSx6RXrxPaXiOfNXPdnVQmgKRg7HhO92KJQKO+CeYbiMNO3Fpt9MSffvkEJQ==
+"@abp/cms-kit.public@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/cms-kit.public/-/cms-kit.public-9.0.1.tgz#f06dd9f234f8fe76738503bb46714a2e43afdda2"
+ integrity sha512-40cyzPS05tZlmeDg4IvhugjvPHMiP2bfevDZPVnAsHZN1JIGKeT5bTeqoiIXPzhEylrmT7FEnXHQSrZh/osLwA==
dependencies:
- "@abp/highlight.js" "~9.0.0"
- "@abp/star-rating-svg" "~9.0.0"
+ "@abp/highlight.js" "~9.0.1"
+ "@abp/star-rating-svg" "~9.0.1"
-"@abp/cms-kit@9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/cms-kit/-/cms-kit-9.0.0.tgz#19364cf6ccfbf6ec927c21ce302a85dfe4215f1b"
- integrity sha512-ZjakEuhKvDh1r3tVnDUxkQWj+9TJuqaFEI2vEiHysB81rM5mqKaOtATfhtC75CN5LDqiKCAzDs+6ty4YmTe1PQ==
+"@abp/cms-kit@9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/cms-kit/-/cms-kit-9.0.1.tgz#4aa0712e80757c671cbf7fc2ccf66e086d0714db"
+ integrity sha512-+9SE2YbuyJY45A7J7ww1X+/fEXavi1EgJdtS3H66VlXemvNXsVo4oF3d5aZePoY/sj8mRcueOy2lW4v7iYUbkA==
dependencies:
- "@abp/cms-kit.admin" "~9.0.0"
- "@abp/cms-kit.public" "~9.0.0"
+ "@abp/cms-kit.admin" "~9.0.1"
+ "@abp/cms-kit.public" "~9.0.1"
-"@abp/codemirror@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/codemirror/-/codemirror-9.0.0.tgz#d398edce6ececdaf1f62e216f98a91b26fb595bd"
- integrity sha512-tXcdoXjpco8VnjE/on4ax9USfz3lu03Y+TehAt1tbVQRBdhkGL+qFB/MrewRKUNM8TjSewuT7D4RxpmThABpNw==
+"@abp/codemirror@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/codemirror/-/codemirror-9.0.1.tgz#b2bf78b93ca522fc0cbc03685bccfa201bcbff78"
+ integrity sha512-WEQbaDDXZGsKxEHn2MHCg9yT4h0n6Lf+DorWPtdCPmmyNWOFbHex9cNzhuM2kxIfFuwmDQKcO1z99EmZQs5xfw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
codemirror "^5.65.1"
-"@abp/core@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.0.tgz#dc5ea9139957e307f73ac1f40b179f991e3c2168"
- integrity sha512-vzQyTpqVDx63pOI9e2QLUMWZCrSojHsRnzLlvl+/cKFi7jE1q/uSP66FPR2bwkD+DNLI/tn6is1Jx3skKA1FvA==
+"@abp/core@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/core/-/core-9.0.1.tgz#727fc1dfac21cca1216c4f5091da7d5efefdb467"
+ integrity sha512-NWXDVCIJePbcKCrRvGxAlaA2MrOF2IZox37bQkW7/mTeOPpNoTa6lhowksNNcTpFq2YdD8bG0i0kfskkp3EZ2w==
dependencies:
- "@abp/utils" "~9.0.0"
+ "@abp/utils" "~9.0.1"
-"@abp/datatables.net-bs5@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.0.tgz#4daee90bdb6446e854ce65f2a3afabb2a13e49ae"
- integrity sha512-ln10tcRwxcB0nuSgqJ/66strHsqMHDC03c0ktxEplAN34yP7KInjFf/U9Svo74K0nKlkJdMCgmqmT4CnBJZlIg==
+"@abp/datatables.net-bs5@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net-bs5/-/datatables.net-bs5-9.0.1.tgz#61e2a3daacf1c074ed6b459e25084cc1f5c4eeab"
+ integrity sha512-tAHdf4TBAS0bVRcvovDKS0OHFyeRSPYbiSIwDn8AIWzW9Weav31jhZ5jL3HEtBKVdb7zqUQ8/3dYPufjDuDdDA==
dependencies:
- "@abp/datatables.net" "~9.0.0"
+ "@abp/datatables.net" "~9.0.1"
datatables.net-bs5 "^2.1.8"
-"@abp/datatables.net@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.0.tgz#82b28518293c9091ec72d5defae2f5d2903d8bbc"
- integrity sha512-QhuX3bGZOckM/fhIIGj/StMJIX2BBtNbLzjTR7WnKMMxDkv0+oDvjw6OqqpYKSvj/q3vY6sRWuCNdEzNi0Fk6A==
+"@abp/datatables.net@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/datatables.net/-/datatables.net-9.0.1.tgz#4cf193e7493f85a272de47326a32a021e3ba40f5"
+ integrity sha512-JY53zVQPluRrBiKwtgk4gfSgD8VDs/VU8LWizhGCprXt3h9vXx0SQL4My6zvXVkv4hUjaq4+MYo84OI8EwFvsg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
datatables.net "^2.1.8"
-"@abp/font-awesome@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.0.tgz#51cce18434a51d050ebb0d822a8c193b7e32f73d"
- integrity sha512-bEHRVzOk0V2kFMAvB/YXFYaDKWRFO0+ScO7jrEFtFgdHXBcqy+dk1aM/N69WalyNmjRRpnh03GhH1LRpwL4/xA==
+"@abp/font-awesome@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/font-awesome/-/font-awesome-9.0.1.tgz#69da15018e7e5f2ff0d3c0d113a4325f54ffc0c3"
+ integrity sha512-7NnYBmk0plLVElUWR1zohul87vE9wKJItemWT5Kskf/IJirKOvXwSSu+7Jg1LmnVtIxcH6qgRpJMmzuAJgawnQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
"@fortawesome/fontawesome-free" "^6.6.0"
-"@abp/highlight.js@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/highlight.js/-/highlight.js-9.0.0.tgz#6d924c9bb3d6b11dcd66a9a71aea873c9866ed6c"
- integrity sha512-eGvgMUNYKW2B2h3QcpMRniaoQ1/9RJBXkn2VsyM8lVSfrfbOg26mMyw7J2BXvuEnXqtnVdFEG45cSFlAnTAVXQ==
+"@abp/highlight.js@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/highlight.js/-/highlight.js-9.0.1.tgz#22f135e73d5ee5ce14e31fe6c1a1b3eee6a91f55"
+ integrity sha512-K1EXCtRsc116c7UYcUXRVmrPFHDY4531+ij9P32EdP1/3AIwm50r+tz2epZV2GyIuGw/mVWyFB0vduqFlr0W3g==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
"@highlightjs/cdn-assets" "~11.10.0"
-"@abp/jquery-form@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.0.tgz#6518eb117ee35c6bc57543fd1fccb0a8cf8e22f4"
- integrity sha512-6qpfPGHrjADmm7Ne3zcXBGPuLngG8ldAn5fk9k2taftpyX0iR67/nZxiQ0ujaQGgaai5Wb3imNXMG96XLnubkQ==
+"@abp/jquery-form@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-form/-/jquery-form-9.0.1.tgz#35e0a23cf50b00e0d3b0886f097233b79144145b"
+ integrity sha512-9co0+Cver+RUSOpfuukSWQhR+TCnO9Dly4RTVRIKFksGVfN8hA70NqRrVxaASNNFv42ynK88FQK+2bLPswz/iw==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-form "^4.3.0"
-"@abp/jquery-validation-unobtrusive@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.0.tgz#8ee87dbdec5e92ab04ae9642a49af1598af31c1a"
- integrity sha512-KlRQ1htyFmAIMS9xbHvHIcdhAgtPFe/u7c87MiWlk/CdhDrI3J+n5biLvQUzxCC+YWlwx27zBtPKx2vLeMRH6Q==
+"@abp/jquery-validation-unobtrusive@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation-unobtrusive/-/jquery-validation-unobtrusive-9.0.1.tgz#f9564aabebf4d9f90b4ee5c3f3de0e1a348d8ecc"
+ integrity sha512-Zpz48wIFjq7eIApG/sEVjkfaJWZhcyCnlCtDf5eU3vvh82nRetbiCEAVLpZ4LVjorlyp3VvVAxReUSeeWflu8A==
dependencies:
- "@abp/jquery-validation" "~9.0.0"
+ "@abp/jquery-validation" "~9.0.1"
jquery-validation-unobtrusive "^4.0.0"
-"@abp/jquery-validation@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.0.tgz#681d8037c57b3875bd51c927e350ffce65d7c5d2"
- integrity sha512-XgCUBFWbhTDcGepD/xkwA/1Hl4TyYitRttauDJk3tuyXhNtykvdugCCOPY4j2s6VoT1Sb/TFFvOcgEvlBwkKiw==
+"@abp/jquery-validation@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery-validation/-/jquery-validation-9.0.1.tgz#ad37a44e82fb32ec9bfe2b736dfc8a143fca89c7"
+ integrity sha512-uFSxiPMEkaPlcuyXIoKbeOIIRHo4IrgRbmnlHwI1uETrW60ENUdE2LJAVcYS77JROrjVtv8FPlaffWuva6sGeQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jquery-validation "^1.21.0"
-"@abp/jquery@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.0.tgz#6c67a9f927be598ab6201f8888e8a3cb8bba0686"
- integrity sha512-cUXA7v6F1HqAc/w8fQGxHIfA/6BqO+4uHXDnSsJZtrI26DTLOgw5kBNnAz2W4xggGPCY7SKFHNRZuOW/QfVRPg==
+"@abp/jquery@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jquery/-/jquery-9.0.1.tgz#ecce14644c753c581caee0df7bfafd3d083e70f0"
+ integrity sha512-z8EcnOveItAszbTsM9GFFDQSlcIg/ym6+WZME9uldLoetdxkmI9ij0knVeGj5eqMjQaHvTWwrVLObiBbZwYFWw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
jquery "~3.7.1"
-"@abp/jstree@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/jstree/-/jstree-9.0.0.tgz#08190074df54e45fe414b7a349c25d6b1f1058a1"
- integrity sha512-ZbXPazP2Yy/qDMw/b4U8Hz1ZxbxQej/PAREaAawd2R2GJoGi10U8iTt5WKJJQP8Fm42kFgQa/8EtVcckwIo4FA==
+"@abp/jstree@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/jstree/-/jstree-9.0.1.tgz#4a7cc3becc46d3e08740b67e88a048d307e5a6a8"
+ integrity sha512-QlXjqY/G8k2reXcua9u9mPldgq9oQUq4LiBdzciMdEqfF4nen84aaXvq+vFSsaKZ9/hoqiCz4cT41FkhNJ4d2g==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
jstree "^3.3.17"
-"@abp/lodash@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.0.tgz#a4d42b2dfeb37859ff9f0079c7a8a6cabc982efe"
- integrity sha512-xoAsOkkgnQdPH2OyFTgF6OlCLVbyGF4u8CrFcZWCZmuIFT9cg0+ZEiq//Vog1pg3Lf//N37ceGpOag1LwMe6cQ==
+"@abp/lodash@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/lodash/-/lodash-9.0.1.tgz#c52c8a4125ecfbeaad7badc5c28bf86d5168799a"
+ integrity sha512-D+/iOyh/O+jV80bK5vWxyn++IVmeRbZq/UaXanA4RKpzigxJUsgOiSD9lUj85BkfKn2YyFmoR39xwaAqkWoyRA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
lodash "^4.17.21"
-"@abp/luxon@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.0.tgz#a54dda13cbbd362481994d20d2e07c669c6ce109"
- integrity sha512-aPejQnoVPfm60LnjwhfUkIsB+poNWusrBE0IHllxhvOVE+01nfa1Lvpp5AUWzV+XchPGOCSf+a29ds4EQWfNbg==
+"@abp/luxon@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/luxon/-/luxon-9.0.1.tgz#ac59a56653d51b997dbbde8ccc58dac5746a9ef4"
+ integrity sha512-1ZHXP5SGz05tQ1tzYmPcRiZwmkCKGX0KWFoV0zF3ATRaevzku9nXvSy7K6GIFkkgTb9f9XNx4/OBBF/GXkuRqw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
luxon "^3.5.0"
-"@abp/malihu-custom-scrollbar-plugin@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.0.tgz#b8a49202d531c711874a2bcdae5081ae0ed34c3f"
- integrity sha512-iSZ9lKMnv5F2SplR+A4blhFEDq79lKbvBQuS7h/Di3KEHFOdJY4pA8NP8OR+PXOF3JfAezGyjqy7ydn/JNPRwA==
+"@abp/malihu-custom-scrollbar-plugin@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/malihu-custom-scrollbar-plugin/-/malihu-custom-scrollbar-plugin-9.0.1.tgz#9e60bcfb9f9d76fed9c50e45bf490d5abd0f4689"
+ integrity sha512-GkKPLg9Z9UTTSSpW2hYZYZ9IgCvDSOYJmyRSl/h4fG8cWMgGCPVpv8MPPu0+WtIfwW/mT4SPX9X+sjUlRL5XRQ==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
malihu-custom-scrollbar-plugin "^3.1.5"
-"@abp/markdown-it@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/markdown-it/-/markdown-it-9.0.0.tgz#f61e74e7f9205f3015074896cf184f90b8a11a1c"
- integrity sha512-frjJ539DiM4H0nHwnt77T1ryoazJhy9836L7HEFiYsfXJHs92VohUr/K27899I3gRdA6h0tbLYumFK3ug83qiw==
+"@abp/markdown-it@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/markdown-it/-/markdown-it-9.0.1.tgz#286a524ff7bd0cde37dedfb56f8de58f96ba5933"
+ integrity sha512-ml1xHXr/fiu0GTu5XiYoBm7bF/bCXCZ0EZvvJNNJjQFaaRcRbSasvhZjDVlnZdmAcfKTl9TKvz8H0mNd12Cdww==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
markdown-it "^14.1.0"
-"@abp/moment@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.0.tgz#9583e97472ddf608a9a5a6b8a37fee602b419900"
- integrity sha512-wgyJSKmBmHLXZJN3dy3zK4i2JvWGgqnvdtrKURy/raEurxlymv8JF5JA971kTv6Kz2OA/OSYttpVJkDGUpPAhA==
+"@abp/moment@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/moment/-/moment-9.0.1.tgz#d67fa25017fe15dae10e0a0fcebb03aeb63d31d3"
+ integrity sha512-bTuCc7pi/DCtgz7t96UBU0wn9UGt4b50JhDCVeTKquh+gP1sgWoI5hyvEBkmgE4dcffO0yzZU2opHYTKYGl/OQ==
dependencies:
moment "^2.30.1"
-"@abp/prismjs@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/prismjs/-/prismjs-9.0.0.tgz#2ac4a5ddc0c8e0462af027a4dd97a177a3d6f7a5"
- integrity sha512-+evmBq5qV+Ix/TNqO2ljJn0jygW2rEgKNtABcBGmrZC8xiI7+2hjKmSE0e43qb+RDib8aJjwH8ya8wF7p68ZFw==
+"@abp/prismjs@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/prismjs/-/prismjs-9.0.1.tgz#2693014c8161d561e7c7bef01d5cd24ee53be1b3"
+ integrity sha512-NwW+vpTiAsuJYNC/MEiAj0dGMWml95mAQD1Z0Djfsg9D6OpcLsdlgWX+KVdZvHoJ+SPgTDIqdt/0LnIv1w/XdA==
dependencies:
- "@abp/clipboard" "~9.0.0"
- "@abp/core" "~9.0.0"
+ "@abp/clipboard" "~9.0.1"
+ "@abp/core" "~9.0.1"
prismjs "^1.29.0"
-"@abp/select2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.0.tgz#71503eebb746f40a0e565dba1efa05d6ae937b77"
- integrity sha512-l+Hz2Qmc7AcyaTJOyuDaV99RbBWYxYxbeyVEc4xKbRFbJxYBUxT/DFlyozEaoz+OU1fl4szO3UCthJ2yZvxCdg==
+"@abp/select2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/select2/-/select2-9.0.1.tgz#9f1c5c27e72bef910fdf63fc3167f415d2ba8db8"
+ integrity sha512-epyYJH2gAOU2UvGn5FD6372hbW3fL7uZIgGpvfgEfa+l6HPdlV9B1OweU2xkONNCoylShAcP/y8qf+bUh8WDEA==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
select2 "^4.0.13"
-"@abp/slugify@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/slugify/-/slugify-9.0.0.tgz#4bf4cade338697b68502cc73ac830d8dab10bbb8"
- integrity sha512-tjHsQndDNDsQYubzQY8d+I/o1pQx4ehaSIw6xt0WyozogRegrxQ9QJUxOEQYtfNAgap0GipnxxvZSRBJINB4Cg==
+"@abp/slugify@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/slugify/-/slugify-9.0.1.tgz#82aa76f131aab619f1d68d8e276b2263be5f0141"
+ integrity sha512-H8pkDkJMgXvSGPboX1ZZTkgw7G2kUd4cAco0kkCjNdpkVPGCwQSg9TfJgTmZLmvKRFVxOpGTInuzB3/tP6NFFw==
dependencies:
slugify "^1.6.6"
-"@abp/star-rating-svg@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/star-rating-svg/-/star-rating-svg-9.0.0.tgz#1e0d2ca6e589d3f7b63e6cfe6f7ad6c723f86a81"
- integrity sha512-OZtWoFG2n5KdnSYeMH4SMYaeET3ISmyREBy5rNVM4peacNejyEHrPadzHDMWcGFMjqveTn/jmN19Jyn9+HG0Vw==
+"@abp/star-rating-svg@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/star-rating-svg/-/star-rating-svg-9.0.1.tgz#35f9b226b323d9e102e8c9a306edf798422dd807"
+ integrity sha512-nND7HnGHFx+uIlMtAhCP3syOui9ZO84KYZnxWAwY/T3ZqV1vzvk8OyyfTZUtt8FWoKRj9JkBvaNUDu4VnMrQWg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
star-rating-svg "^3.5.0"
-"@abp/sweetalert2@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.0.tgz#71e511117c18b515e5c658e9cc0667770af65d91"
- integrity sha512-wxDqoA6RUhkbcN27Kzslu/m2ai5T9AkJRtgs7pV55LvPv1ohuCPEUeTVVh8S/5Pgmr7aPFAPxaaCO32aPcImPw==
+"@abp/sweetalert2@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/sweetalert2/-/sweetalert2-9.0.1.tgz#a364b21665cb76e92640496dc8a77b8edbce4500"
+ integrity sha512-0RXR/lzy3qMJLisGUz4qrcA9Clu/Tt1ABGzB64ynG1t/l5XessQsIB5Hkt6MbSyOEsprRsq+ZtKH35CwTRPdAw==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
sweetalert2 "^11.14.1"
-"@abp/timeago@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.0.tgz#8f262ba2a74e1c133d0c129f8a0bc55c29441e1a"
- integrity sha512-eaNdIUEtDXnuYqS5O7EGRV+j6XOLAsxz7gd5BIKR9lDujd8pEjSJkXKO+/0KxLTkUSy3JudIO37K1h2wTpbYZA==
+"@abp/timeago@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/timeago/-/timeago-9.0.1.tgz#d1b7e11fdc267bdb736ff4b591afd3fb81c07cce"
+ integrity sha512-dNfUQwilrY+JHxL5lwJ++ixYDGAs5lF1EDxKMQMIXD/ak5L8rkdLns8Otta7FhgDDny9UJ7XWQjYsG8ZxxybRQ==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
timeago "^1.6.7"
-"@abp/toastr@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.0.tgz#74d74ecb3e7f2d63b2ff7735c89a40deb3dc3a13"
- integrity sha512-n9uJvw8l2HoxeowPO4diEMxwchGh6puefhyRljFAAstzK77Mhn0u2J7kPynNv94rLSYTYoCxUeJ4TTN+/TVDGQ==
+"@abp/toastr@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/toastr/-/toastr-9.0.1.tgz#2105a639d7cd4fa9c671b4215d719a78fef7e0ee"
+ integrity sha512-10KuHnSJOxkvzqs6Aim6GJp8xBuT+TsZqBE1KaXQnERPawmOo4+ME28kA/7lq8fvgFUNO3K41T2uuA1GPHVQcg==
dependencies:
- "@abp/jquery" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
toastr "^2.1.4"
-"@abp/tui-editor@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/tui-editor/-/tui-editor-9.0.0.tgz#d9f7721c3d8e361d901a0e0e11e786b2e12b234a"
- integrity sha512-TVadK2Yi3wlwyvEVshh1PxkNRppYmqaPB6QE1cOhrWZAFbP1fh1oFvfqQiHN3o+9RGHFcEnhEK3D+49aHuDh3Q==
+"@abp/tui-editor@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/tui-editor/-/tui-editor-9.0.1.tgz#df2ef5101300f2f739c0c93b66f63ccd8492a42d"
+ integrity sha512-0H9AK4/YXsDmLruHMqk2uygWkWTR9pRRexGCr+IzoAeb+tljg15MnmlboTE9iNOntUug6Iml9tyY5gNOUuQXvQ==
dependencies:
- "@abp/jquery" "~9.0.0"
- "@abp/prismjs" "~9.0.0"
+ "@abp/jquery" "~9.0.1"
+ "@abp/prismjs" "~9.0.1"
-"@abp/uppy@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/uppy/-/uppy-9.0.0.tgz#4def1e889093a4299f1c2445a87793cb3da2d4d7"
- integrity sha512-Y9IbGu/t3rJefF7FSrsiHs9V+C/lAMtolivPwKhnOzxmUtoxbyc0O2W2uExZEihevbnk3PWe7dkFY5mZ30i6BA==
+"@abp/uppy@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/uppy/-/uppy-9.0.1.tgz#35af2483439c1ee0e4ec07299f8ada17f235d97b"
+ integrity sha512-rPM0Fy0qef1xw28a+qzFWNL8OeXzHXY1Qyq0W9VKUhFwUYKYKTALwi6k7SVQh5Dq+X2blSStVkJD5tuADxSnIg==
dependencies:
- "@abp/core" "~9.0.0"
+ "@abp/core" "~9.0.1"
uppy "^4.4.1"
-"@abp/utils@~9.0.0":
- version "9.0.0"
- resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.0.tgz#2264fd83105b061860edd98dbe0efdfe55803ed5"
- integrity sha512-/SB4VYk3Av+ltXv6ovkFHG4iWiECLLitaYWcEHxVAO80ufNjrNnWpHFpXHP8u/RpXIp/CoNiHLr6j5TkZUTZjw==
+"@abp/utils@~9.0.1":
+ version "9.0.1"
+ resolved "https://registry.yarnpkg.com/@abp/utils/-/utils-9.0.1.tgz#dccaf7cf92147304d454ffb9a8664d52dc7e5c5a"
+ integrity sha512-7eOKT0ciKNdtuASck1kifyc4dC6CeebgxV+rx/EAi3YwPMDkqie6YwLGUqKdGW2N/MZ15JuDR6gkuZxKHEb0NQ==
dependencies:
just-compare "^2.3.0"
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.Application.Contracts/Volo/CmsKit/Admin/Blogs/BlogDto.cs b/modules/cms-kit/src/Volo.CmsKit.Admin.Application.Contracts/Volo/CmsKit/Admin/Blogs/BlogDto.cs
index 0245b1348e..93c5374281 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Admin.Application.Contracts/Volo/CmsKit/Admin/Blogs/BlogDto.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.Application.Contracts/Volo/CmsKit/Admin/Blogs/BlogDto.cs
@@ -11,4 +11,6 @@ public class BlogDto : ExtensibleEntityDto, IHasConcurrencyStamp
public string Slug { get; set; }
public string ConcurrencyStamp { get; set; }
+
+ public int BlogPostCount { get; set; }
}
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.Application.Contracts/Volo/CmsKit/Admin/Blogs/IBlogAdminAppService.cs b/modules/cms-kit/src/Volo.CmsKit.Admin.Application.Contracts/Volo/CmsKit/Admin/Blogs/IBlogAdminAppService.cs
index 647c2bde15..e0f2b3311b 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Admin.Application.Contracts/Volo/CmsKit/Admin/Blogs/IBlogAdminAppService.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.Application.Contracts/Volo/CmsKit/Admin/Blogs/IBlogAdminAppService.cs
@@ -1,8 +1,13 @@
using System;
+using System.Threading.Tasks;
+using Volo.Abp.Application.Dtos;
using Volo.Abp.Application.Services;
namespace Volo.CmsKit.Admin.Blogs;
public interface IBlogAdminAppService : ICrudAppService
{
+ Task> GetAllListAsync();
+
+ Task MoveAllBlogPostsAsync(Guid blogId, Guid? assignToBlogId = null);
}
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.Application/Volo/CmsKit/Admin/Blogs/BlogAdminAppService.cs b/modules/cms-kit/src/Volo.CmsKit.Admin.Application/Volo/CmsKit/Admin/Blogs/BlogAdminAppService.cs
index d5f32c4bfc..e3ef03e2de 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Admin.Application/Volo/CmsKit/Admin/Blogs/BlogAdminAppService.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.Application/Volo/CmsKit/Admin/Blogs/BlogAdminAppService.cs
@@ -1,5 +1,6 @@
using System;
using System.Collections.Generic;
+using System.Linq;
using System.Threading.Tasks;
using Microsoft.AspNetCore.Authorization;
using Volo.Abp.Application.Dtos;
@@ -20,16 +21,19 @@ namespace Volo.CmsKit.Admin.Blogs;
public class BlogAdminAppService : CmsKitAdminAppServiceBase, IBlogAdminAppService
{
protected IBlogRepository BlogRepository { get; }
+ protected IBlogPostRepository BlogPostRepository { get; }
protected BlogManager BlogManager { get; }
protected BlogFeatureManager BlogFeatureManager { get; }
public BlogAdminAppService(
IBlogRepository blogRepository,
- BlogManager blogManager,
+ BlogManager blogManager,
+ IBlogPostRepository blogPostRepository,
BlogFeatureManager blogFeatureManager = null)
{
BlogRepository = blogRepository;
BlogManager = blogManager;
+ BlogPostRepository = blogPostRepository;
BlogFeatureManager = blogFeatureManager;
}
@@ -37,20 +41,44 @@ public class BlogAdminAppService : CmsKitAdminAppServiceBase, IBlogAdminAppServi
{
var blog = await BlogRepository.GetAsync(id);
- return ObjectMapper.Map(blog);
+ var blogDto = ObjectMapper.Map(blog);
+ blogDto.BlogPostCount = await BlogPostRepository.GetCountAsync(blogId : blog.Id);
+
+ return blogDto;
}
public virtual async Task> GetListAsync(BlogGetListInput input)
{
var totalCount = await BlogRepository.GetCountAsync(input.Filter);
- var blogs = await BlogRepository.GetListAsync(
+ var blogs = await BlogRepository.GetListWithBlogPostCountAsync(
input.Filter,
input.Sorting,
input.MaxResultCount,
input.SkipCount);
+
+ var blogDtos = new PagedResultDto(totalCount, ObjectMapper.Map, List>(blogs.Select(x => x.Blog).ToList()));
+
+ foreach (var blogDto in blogDtos.Items)
+ {
+ blogDto.BlogPostCount = blogs.First(x => x.Blog.Id == blogDto.Id).BlogPostCount;
+ }
- return new PagedResultDto(totalCount, ObjectMapper.Map, List>(blogs));
+ return blogDtos;
+ }
+
+ public virtual async Task> GetAllListAsync()
+ {
+ var blogs = await BlogRepository.GetListWithBlogPostCountAsync(maxResultCount: int.MaxValue);
+
+ var blogDtos = new ListResultDto(ObjectMapper.Map, List>(blogs.Select(x => x.Blog).ToList()));
+
+ foreach (var blogDto in blogDtos.Items)
+ {
+ blogDto.BlogPostCount = blogs.First(x => x.Blog.Id == blogDto.Id).BlogPostCount;
+ }
+
+ return blogDtos;
}
[Authorize(CmsKitAdminPermissions.Blogs.Create)]
@@ -79,10 +107,18 @@ public class BlogAdminAppService : CmsKitAdminAppServiceBase, IBlogAdminAppServi
return ObjectMapper.Map(blog);
}
+
+ [Authorize(CmsKitAdminPermissions.Blogs.Delete)]
+ public virtual async Task MoveAllBlogPostsAsync(Guid blogId, Guid? assignToBlogId)
+ {
+ var blog = await BlogRepository.GetAsync(blogId);
+ await BlogPostRepository.UpdateBlogAsync(blog.Id, assignToBlogId);
+ }
[Authorize(CmsKitAdminPermissions.Blogs.Delete)]
public virtual Task DeleteAsync(Guid id)
{
+
return BlogRepository.DeleteAsync(id);
}
}
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.Application/Volo/CmsKit/Admin/CmsKitAdminApplicationAutoMapperProfile.cs b/modules/cms-kit/src/Volo.CmsKit.Admin.Application/Volo/CmsKit/Admin/CmsKitAdminApplicationAutoMapperProfile.cs
index a6a0d6512a..60f8ccb8a8 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Admin.Application/Volo/CmsKit/Admin/CmsKitAdminApplicationAutoMapperProfile.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.Application/Volo/CmsKit/Admin/CmsKitAdminApplicationAutoMapperProfile.cs
@@ -38,7 +38,7 @@ public class CmsKitAdminApplicationAutoMapperProfile : Profile
CreateMap(MemberList.Source).MapExtraProperties();
CreateMap(MemberList.Source).MapExtraProperties();
- CreateMap().MapExtraProperties();
+ CreateMap().Ignore(b => b.BlogPostCount).MapExtraProperties();
CreateMap(MemberList.Destination);
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi.Client/ClientProxies/Volo/CmsKit/Admin/Blogs/BlogAdminClientProxy.Generated.cs b/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi.Client/ClientProxies/Volo/CmsKit/Admin/Blogs/BlogAdminClientProxy.Generated.cs
index d102ea4d1f..7eb5d91f3b 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi.Client/ClientProxies/Volo/CmsKit/Admin/Blogs/BlogAdminClientProxy.Generated.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi.Client/ClientProxies/Volo/CmsKit/Admin/Blogs/BlogAdminClientProxy.Generated.cs
@@ -57,4 +57,18 @@ public partial class BlogAdminClientProxy : ClientProxyBase> GetAllListAsync()
+ {
+ return await RequestAsync>(nameof(GetAllListAsync));
+ }
+
+ public virtual async Task MoveAllBlogPostsAsync(Guid blogId, Guid? assignToBlogId)
+ {
+ await RequestAsync(nameof(MoveAllBlogPostsAsync), new ClientProxyRequestTypeValue
+ {
+ { typeof(Guid), blogId },
+ { typeof(Guid?), assignToBlogId }
+ });
+ }
}
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi.Client/ClientProxies/cms-kit-admin-generate-proxy.json b/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi.Client/ClientProxies/cms-kit-admin-generate-proxy.json
index cf4bbd1034..8f0e7e677d 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi.Client/ClientProxies/cms-kit-admin-generate-proxy.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi.Client/ClientProxies/cms-kit-admin-generate-proxy.json
@@ -16,6 +16,39 @@
"type": "Volo.CmsKit.Admin.Blogs.IBlogAdminAppService",
"name": "IBlogAdminAppService",
"methods": [
+ {
+ "name": "GetAllListAsync",
+ "parametersOnMethod": [],
+ "returnValue": {
+ "type": "Volo.Abp.Application.Dtos.ListResultDto",
+ "typeSimple": "Volo.Abp.Application.Dtos.ListResultDto"
+ }
+ },
+ {
+ "name": "MoveAllBlogPostsAsync",
+ "parametersOnMethod": [
+ {
+ "name": "blogId",
+ "typeAsString": "System.Guid, System.Private.CoreLib",
+ "type": "System.Guid",
+ "typeSimple": "string",
+ "isOptional": false,
+ "defaultValue": null
+ },
+ {
+ "name": "assignToBlogId",
+ "typeAsString": "System.Nullable`1[[System.Guid, System.Private.CoreLib, Version=9.0.0.0, Culture=neutral, PublicKeyToken=7cec85d7bea7798e]], System.Private.CoreLib",
+ "type": "System.Guid?",
+ "typeSimple": "string?",
+ "isOptional": false,
+ "defaultValue": null
+ }
+ ],
+ "returnValue": {
+ "type": "System.Void",
+ "typeSimple": "System.Void"
+ }
+ },
{
"name": "GetAsync",
"parametersOnMethod": [
@@ -353,6 +386,90 @@
},
"allowAnonymous": false,
"implementFrom": "Volo.Abp.Application.Services.IDeleteAppService"
+ },
+ "GetAllListAsync": {
+ "uniqueName": "GetAllListAsync",
+ "name": "GetAllListAsync",
+ "httpMethod": "GET",
+ "url": "api/cms-kit-admin/blogs/all",
+ "supportedVersions": [],
+ "parametersOnMethod": [],
+ "parameters": [],
+ "returnValue": {
+ "type": "Volo.Abp.Application.Dtos.ListResultDto",
+ "typeSimple": "Volo.Abp.Application.Dtos.ListResultDto"
+ },
+ "allowAnonymous": false,
+ "implementFrom": "Volo.CmsKit.Admin.Blogs.IBlogAdminAppService"
+ },
+ "MoveAllBlogPostsAsyncByBlogIdAndAssignToBlogId": {
+ "uniqueName": "MoveAllBlogPostsAsyncByBlogIdAndAssignToBlogId",
+ "name": "MoveAllBlogPostsAsync",
+ "httpMethod": "PUT",
+ "url": "api/cms-kit-admin/blogs/{id}/move-all-blog-posts",
+ "supportedVersions": [],
+ "parametersOnMethod": [
+ {
+ "name": "blogId",
+ "typeAsString": "System.Guid, System.Private.CoreLib",
+ "type": "System.Guid",
+ "typeSimple": "string",
+ "isOptional": false,
+ "defaultValue": null
+ },
+ {
+ "name": "assignToBlogId",
+ "typeAsString": "System.Nullable`1[[System.Guid, System.Private.CoreLib, Version=9.0.0.0, Culture=neutral, PublicKeyToken=7cec85d7bea7798e]], System.Private.CoreLib",
+ "type": "System.Guid?",
+ "typeSimple": "string?",
+ "isOptional": false,
+ "defaultValue": null
+ }
+ ],
+ "parameters": [
+ {
+ "nameOnMethod": "blogId",
+ "name": "blogId",
+ "jsonName": null,
+ "type": "System.Guid",
+ "typeSimple": "string",
+ "isOptional": false,
+ "defaultValue": null,
+ "constraintTypes": null,
+ "bindingSourceId": "ModelBinding",
+ "descriptorName": ""
+ },
+ {
+ "nameOnMethod": "assignToBlogId",
+ "name": "assignToBlogId",
+ "jsonName": null,
+ "type": "System.Guid?",
+ "typeSimple": "string?",
+ "isOptional": false,
+ "defaultValue": null,
+ "constraintTypes": null,
+ "bindingSourceId": "Query",
+ "descriptorName": ""
+ },
+ {
+ "nameOnMethod": "id",
+ "name": "id",
+ "jsonName": null,
+ "type": null,
+ "typeSimple": null,
+ "isOptional": false,
+ "defaultValue": null,
+ "constraintTypes": [],
+ "bindingSourceId": "Path",
+ "descriptorName": ""
+ }
+ ],
+ "returnValue": {
+ "type": "System.Void",
+ "typeSimple": "System.Void"
+ },
+ "allowAnonymous": false,
+ "implementFrom": "Volo.CmsKit.Admin.Blogs.IBlogAdminAppService"
}
}
},
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi/Volo/CmsKit/Admin/Blogs/BlogAdminController.cs b/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi/Volo/CmsKit/Admin/Blogs/BlogAdminController.cs
index e7d59ddbcd..f90215ed38 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi/Volo/CmsKit/Admin/Blogs/BlogAdminController.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.HttpApi/Volo/CmsKit/Admin/Blogs/BlogAdminController.cs
@@ -62,4 +62,19 @@ public class BlogAdminController : CmsKitAdminController, IBlogAdminAppService
{
return BlogAdminAppService.DeleteAsync(id);
}
+
+ [HttpGet]
+ [Route("all")]
+ public Task> GetAllListAsync()
+ {
+ return BlogAdminAppService.GetAllListAsync();
+ }
+
+ [HttpPut]
+ [Route("{id}/move-all-blog-posts")]
+ [Authorize(CmsKitAdminPermissions.Blogs.Delete)]
+ public Task MoveAllBlogPostsAsync(Guid blogId, [FromQuery]Guid? assignToBlogId)
+ {
+ return BlogAdminAppService.MoveAllBlogPostsAsync(blogId, assignToBlogId);
+ }
}
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.Web/CmsKitAdminWebModule.cs b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/CmsKitAdminWebModule.cs
index 70c2b575f4..45cbbd818d 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Admin.Web/CmsKitAdminWebModule.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/CmsKitAdminWebModule.cs
@@ -13,6 +13,7 @@ using Volo.Abp.Localization;
using Volo.Abp.Modularity;
using Volo.Abp.ObjectExtending;
using Volo.Abp.ObjectExtending.Modularity;
+using Volo.Abp.SettingManagement.Web;
using Volo.Abp.SettingManagement.Web.Pages.SettingManagement;
using Volo.Abp.Threading;
using Volo.Abp.UI.Navigation;
@@ -29,7 +30,8 @@ namespace Volo.CmsKit.Admin.Web;
[DependsOn(
typeof(CmsKitAdminApplicationContractsModule),
- typeof(CmsKitCommonWebModule)
+ typeof(CmsKitCommonWebModule),
+ typeof(AbpSettingManagementWebModule)
)]
public class CmsKitAdminWebModule : AbpModule
{
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Pages/CmsKit/Blogs/DeleteBlogModal.cshtml b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Pages/CmsKit/Blogs/DeleteBlogModal.cshtml
new file mode 100644
index 0000000000..5f32d8e9bf
--- /dev/null
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Pages/CmsKit/Blogs/DeleteBlogModal.cshtml
@@ -0,0 +1,59 @@
+@page
+@using Microsoft.AspNetCore.Mvc.Localization
+@using Volo.Abp.AspNetCore.Mvc.UI.Bootstrap.TagHelpers.Modal
+@using Volo.CmsKit.Admin.Web.Pages.CmsKit.Blogs
+@using Volo.CmsKit.Localization
+@model DeleteBlogModal
+@inject IHtmlLocalizer L
+@{
+ Layout = null;
+}
+
+
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Pages/CmsKit/Blogs/DeleteBlogModal.cshtml.cs b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Pages/CmsKit/Blogs/DeleteBlogModal.cshtml.cs
new file mode 100644
index 0000000000..08e320b6f5
--- /dev/null
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Pages/CmsKit/Blogs/DeleteBlogModal.cshtml.cs
@@ -0,0 +1,57 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Threading.Tasks;
+using Microsoft.AspNetCore.Mvc;
+using Microsoft.AspNetCore.Mvc.RazorPages;
+using Volo.Abp.ObjectExtending;
+using Volo.CmsKit.Admin.Blogs;
+
+namespace Volo.CmsKit.Admin.Web.Pages.CmsKit.Blogs;
+
+public class DeleteBlogModal : CmsKitAdminPageModel
+{
+ [BindProperty]
+ public BlogInfoModel Blog { get; set; }
+
+ protected IBlogAdminAppService BlogAdminAppService { get; }
+
+ public DeleteBlogModal(IBlogAdminAppService blogAdminAppService)
+ {
+ BlogAdminAppService = blogAdminAppService;
+ }
+
+ public virtual async Task OnGetAsync(Guid id)
+ {
+ var blog = await BlogAdminAppService.GetAsync(id);
+ var allBlogs = await BlogAdminAppService.GetAllListAsync();
+
+ Blog = new BlogInfoModel
+ {
+ Id = blog.Id,
+ Name = blog.Name,
+ BlogPostCount = blog.BlogPostCount,
+ OtherBlogs = allBlogs.Items.Where(b => b.Id != blog.Id).Select(e => new KeyValuePair(e.Id, e.Name)).ToList()
+ };
+ }
+
+ public virtual async Task OnPostAsync()
+ {
+ await BlogAdminAppService.MoveAllBlogPostsAsync(Blog.Id, Blog.AssignToBlogId);
+ await BlogAdminAppService.DeleteAsync(Blog.Id);
+ return NoContent();
+ }
+
+ public class BlogInfoModel : ExtensibleObject
+ {
+ public Guid Id { get; set; }
+
+ public string Name { get; set; }
+
+ public int BlogPostCount { get; set; }
+
+ public List> OtherBlogs { get; set; }
+
+ public Guid? AssignToBlogId { get; set; }
+ }
+}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Pages/CmsKit/Blogs/index.js b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Pages/CmsKit/Blogs/index.js
index 0ead857f17..eef9af3f09 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Pages/CmsKit/Blogs/index.js
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Pages/CmsKit/Blogs/index.js
@@ -6,6 +6,35 @@ $(function () {
var updateModal = new abp.ModalManager({ viewUrl: abp.appPath + "CmsKit/Blogs/UpdateModal", modalClass: 'updateBlog' });
var featuresModal = new abp.ModalManager(abp.appPath + "CmsKit/Blogs/FeaturesModal");
+
+ var deleteBlogModal = new abp.ModalManager(abp.appPath + 'CmsKit/Blogs/DeleteBlogModal');
+
+ deleteBlogModal.onResult(function(){
+ abp.notify.success(l('DeletedSuccessfully'));
+ });
+
+ deleteBlogModal.onOpen(function () {
+ var $form = deleteBlogModal.getForm();
+ $form.find('#assign').click(function () {
+ $form.find('#Blog_AssignToBlogId').show();
+ $form.find('[type=submit]').attr("disabled","disabled")
+ })
+ $form.find('#deleteAll').click(function () {
+ $form.find('#Blog_AssignToBlogId').hide();
+ $form.find('#Blog_AssignToBlogId').val("");
+ $form.find('[type=submit]').removeAttr("disabled");
+ })
+
+ $("#Blog_AssignToBlogId").on("change", function () {
+ var val = $(this).val();
+ if(val === ''){
+ $form.find('[type=submit]').attr("disabled","disabled")
+ }else{
+ $form.find('[type=submit]').removeAttr("disabled");
+ }
+ })
+ })
+
var blogsService = volo.cmsKit.admin.blogs.blogAdmin;
var dataTable = $("#BlogsTable").DataTable(abp.libs.datatables.normalizeConfiguration({
@@ -41,16 +70,10 @@ $(function () {
{
text: l('Delete'),
visible: abp.auth.isGranted('CmsKit.Blogs.Delete'),
- confirmMessage: function (data) {
- return l("BlogDeletionConfirmationMessage", data.record.name)
- },
action: function (data) {
- blogsService
- .delete(data.record.id)
- .then(function () {
- dataTable.ajax.reloadEx();
- abp.notify.success(l('DeletedSuccessfully'));
- });
+ deleteBlogModal.open({
+ id: data.record.id
+ });
}
}
]
@@ -74,6 +97,7 @@ $(function () {
createModal.open();
});
+
createModal.onResult(function () {
dataTable.ajax.reloadEx();
});
@@ -81,4 +105,8 @@ $(function () {
updateModal.onResult(function () {
dataTable.ajax.reloadEx();
});
+
+ deleteBlogModal.onResult(function () {
+ dataTable.ajax.reloadEx();
+ });
});
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Volo.CmsKit.Admin.Web.abppkg.analyze.json b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Volo.CmsKit.Admin.Web.abppkg.analyze.json
index b8b9d3e51e..b42079c218 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Volo.CmsKit.Admin.Web.abppkg.analyze.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Volo.CmsKit.Admin.Web.abppkg.analyze.json
@@ -14,6 +14,11 @@
"declaringAssemblyName": "Volo.CmsKit.Common.Web",
"namespace": "Volo.CmsKit.Web",
"name": "CmsKitCommonWebModule"
+ },
+ {
+ "declaringAssemblyName": "Volo.Abp.SettingManagement.Web",
+ "namespace": "Volo.Abp.SettingManagement.Web",
+ "name": "AbpSettingManagementWebModule"
}
],
"implementingInterfaces": [
diff --git a/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Volo.CmsKit.Admin.Web.csproj b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Volo.CmsKit.Admin.Web.csproj
index 9323e3044e..6dbaea3eb9 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Volo.CmsKit.Admin.Web.csproj
+++ b/modules/cms-kit/src/Volo.CmsKit.Admin.Web/Volo.CmsKit.Admin.Web.csproj
@@ -15,6 +15,8 @@
+
+
diff --git a/modules/cms-kit/src/Volo.CmsKit.Common.Web/CmsKitCommonWebModule.cs b/modules/cms-kit/src/Volo.CmsKit.Common.Web/CmsKitCommonWebModule.cs
index 5f27f785f8..e8a30a75bc 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Common.Web/CmsKitCommonWebModule.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Common.Web/CmsKitCommonWebModule.cs
@@ -7,15 +7,13 @@ using Volo.CmsKit.Reactions;
using Volo.CmsKit.Web.Icons;
using Markdig;
using Microsoft.Extensions.DependencyInjection;
-using Volo.Abp.SettingManagement.Web;
namespace Volo.CmsKit.Web;
[DependsOn(
typeof(AbpAspNetCoreMvcUiThemeSharedModule),
typeof(CmsKitCommonApplicationContractsModule),
- typeof(AbpAutoMapperModule),
- typeof(AbpSettingManagementWebModule)
+ typeof(AbpAutoMapperModule)
)]
public class CmsKitCommonWebModule : AbpModule
{
diff --git a/modules/cms-kit/src/Volo.CmsKit.Common.Web/Volo.CmsKit.Common.Web.abppkg.analyze.json b/modules/cms-kit/src/Volo.CmsKit.Common.Web/Volo.CmsKit.Common.Web.abppkg.analyze.json
index 56c251adff..6b69cfe938 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Common.Web/Volo.CmsKit.Common.Web.abppkg.analyze.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Common.Web/Volo.CmsKit.Common.Web.abppkg.analyze.json
@@ -19,11 +19,6 @@
"declaringAssemblyName": "Volo.Abp.AutoMapper",
"namespace": "Volo.Abp.AutoMapper",
"name": "AbpAutoMapperModule"
- },
- {
- "declaringAssemblyName": "Volo.Abp.SettingManagement.Web",
- "namespace": "Volo.Abp.SettingManagement.Web",
- "name": "AbpSettingManagementWebModule"
}
],
"implementingInterfaces": [
diff --git a/modules/cms-kit/src/Volo.CmsKit.Common.Web/Volo.CmsKit.Common.Web.csproj b/modules/cms-kit/src/Volo.CmsKit.Common.Web/Volo.CmsKit.Common.Web.csproj
index f051498795..8e8fe81033 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Common.Web/Volo.CmsKit.Common.Web.csproj
+++ b/modules/cms-kit/src/Volo.CmsKit.Common.Web/Volo.CmsKit.Common.Web.csproj
@@ -16,7 +16,6 @@
-
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ar.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ar.json
index c86c314d87..1236f52999 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ar.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ar.json
@@ -237,6 +237,10 @@
"CssClass": "فئة CSS",
"TagsHelpText": "يجب أن تكون العلامات مفصولة بفواصل (على سبيل المثال: tag1، tag2، tag3)",
"ThisPartOfContentCouldntBeLoaded": "لا يمكن تحميل هذا الجزء من المحتوى.",
- "DuplicateCommentAttemptMessage": "تم اكتشاف محاولة نشر تعليق مكررة. لقد تم بالفعل تقديم تعليقك."
+ "DuplicateCommentAttemptMessage": "تم اكتشاف محاولة نشر تعليق مكررة. لقد تم بالفعل تقديم تعليقك.",
+ "ChooseAnActionForBlog": "اختر إجراءً للمدونة",
+ "AssignBlogPostsToOtherBlog": "تعيين مشاركات المدونة إلى مدونة أخرى",
+ "SelectAnBlogToAssign": "حدد مدونة لتعيين مشاركات المدونة إليها",
+ "DeleteAllBlogPostsOfThisBlog": "حذف جميع مشاركات المدونة"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/cs.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/cs.json
index c4534837fa..946336668f 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/cs.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/cs.json
@@ -227,6 +227,10 @@
"CssClass": "Třída CSS",
"TagsHelpText": "Značky by měly být odděleny čárkami (např.: tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "Tato část obsahu nemohla být načtena.",
- "DuplicateCommentAttemptMessage": "Byl zjištěn duplicitní pokus o vložení komentáře. Váš komentář již byl odeslán."
+ "DuplicateCommentAttemptMessage": "Byl zjištěn duplicitní pokus o vložení komentáře. Váš komentář již byl odeslán.",
+ "ChooseAnActionForBlog": "Vyberte akci pro blog",
+ "AssignBlogPostsToOtherBlog": "Přiřaďte blogové příspěvky k jinému blogu",
+ "SelectAnBlogToAssign": "Vyberte blog, ke kterému chcete přiřadit blogové příspěvky",
+ "DeleteAllBlogPostsOfThisBlog": "Smazat všechny blogové příspěvky tohoto blogu"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/de-DE.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/de-DE.json
index 98b8b731c4..49b5f96ff5 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/de-DE.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/de-DE.json
@@ -161,6 +161,10 @@
"YourFullName": "Ihren vollständigen Namen",
"YourMessage": "Ihre Nachricht",
"YourReply": "Ihre Antwort",
- "ThisPartOfContentCouldntBeLoaded": "Dieser Teil des Inhalts konnte nicht geladen werden."
+ "ThisPartOfContentCouldntBeLoaded": "Dieser Teil des Inhalts konnte nicht geladen werden.",
+ "ChooseAnActionForBlog": "Wählen Sie eine Aktion für den Blog",
+ "AssignBlogPostsToOtherBlog": "Blogbeiträge einem anderen Blog zuweisen",
+ "SelectAnBlogToAssign": "Wählen Sie einen Blog aus, um Blogbeiträge zuzuweisen",
+ "DeleteAllBlogPostsOfThisBlog": "Alle Blogbeiträge dieses Blogs löschen"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/de.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/de.json
index 31bae240c3..e950983d9c 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/de.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/de.json
@@ -227,6 +227,10 @@
"CssClass": "CSS-Klasse",
"TagsHelpText": "Tags sollten durch Kommas getrennt werden (z. B. Tag1, Tag2, Tag3).",
"ThisPartOfContentCouldntBeLoaded": "Dieser Teil des Inhalts konnte nicht geladen werden.",
- "DuplicateCommentAttemptMessage": "Es wurde versucht, einen doppelten Kommentar zu posten. Ihr Kommentar wurde bereits übermittelt."
+ "DuplicateCommentAttemptMessage": "Es wurde versucht, einen doppelten Kommentar zu posten. Ihr Kommentar wurde bereits übermittelt.",
+ "ChooseAnActionForBlog": "Wählen Sie eine Aktion für den Blog",
+ "AssignBlogPostsToOtherBlog": "Blogbeiträge einem anderen Blog zuweisen",
+ "SelectAnBlogToAssign": "Wählen Sie einen Blog aus, um Blogbeiträge zuzuweisen",
+ "DeleteAllBlogPostsOfThisBlog": "Alle Blogbeiträge dieses Blogs löschen"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/el.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/el.json
index 695cdd21eb..5c3bd58461 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/el.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/el.json
@@ -184,6 +184,10 @@
"SelectAStatus": "Επιλέξτε μια κατάσταση",
"Status": "Κατάσταση",
"CmsKit.BlogPost.ScrollIndex": "Γραμμή γρήγορης πλοήγησης σε αναρτήσεις ιστολογίου",
- "ThisPartOfContentCouldntBeLoaded": "Αυτό το τμήμα του περιεχομένου δεν μπορούσε να φορτωθεί."
+ "ThisPartOfContentCouldntBeLoaded": "Αυτό το τμήμα του περιεχομένου δεν μπορούσε να φορτωθεί.",
+ "ChooseAnActionForBlog": "Επιλέξτε μια ενέργεια για το ιστολόγιο",
+ "AssignBlogPostsToOtherBlog": "Ανάθεση αναρτήσεων ιστολογίου σε άλλο ιστολόγιο",
+ "SelectAnBlogToAssign": "Επιλέξτε ένα ιστολόγιο για να αναθέσετε αναρτήσεις ιστολογίου",
+ "DeleteAllBlogPostsOfThisBlog": "Διαγραφή όλων των αναρτήσεων ιστολογίου αυτού του ιστολογίου"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/en-GB.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/en-GB.json
index f6256d81ba..b8b9060f45 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/en-GB.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/en-GB.json
@@ -28,6 +28,10 @@
"Undo": "Undo",
"Update": "Update",
"YourComment": "Your comment",
- "YourReply": "Your reply"
+ "YourReply": "Your reply",
+ "ChooseAnActionForBlog": "Choose an action for the blog",
+ "AssignBlogPostsToOtherBlog": "Assign blog posts to another blog",
+ "SelectAnBlogToAssign": "Select a blog to assign",
+ "DeleteAllBlogPostsOfThisBlog": "Delete all blog posts of this blog"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/en.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/en.json
index 149f1072bf..596b9250f6 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/en.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/en.json
@@ -255,6 +255,10 @@
"CommentAlertMessage": "There are {0} comments waiting for approval",
"Settings:Menu:CmsKit": "CMS",
"CommentsAwaitingApproval": "Comments Awaiting Approval",
- "CommentSubmittedForApproval": "Your comment has been submitted for approval."
+ "CommentSubmittedForApproval": "Your comment has been submitted for approval.",
+ "ChooseAnActionForBlog": "Choose an action for the blog",
+ "AssignBlogPostsToOtherBlog": "Assign blog posts to another blog",
+ "SelectAnBlogToAssign": "Select a blog to assign",
+ "DeleteAllBlogPostsOfThisBlog": "Delete all blog posts of this blog"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/es.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/es.json
index e1ff8d7fc8..ffe5b410be 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/es.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/es.json
@@ -227,6 +227,10 @@
"CssClass": "Clase CSS",
"TagsHelpText": "Las etiquetas deben estar separadas por comas (por ejemplo: etiqueta1, etiqueta2, etiqueta3)",
"ThisPartOfContentCouldntBeLoaded": "Esta parte del contenido no se pudo cargar.",
- "DuplicateCommentAttemptMessage": "Intento de publicación de comentario duplicado detectado. Tu comentario ya ha sido enviado."
+ "DuplicateCommentAttemptMessage": "Intento de publicación de comentario duplicado detectado. Tu comentario ya ha sido enviado.",
+ "ChooseAnActionForBlog": "Elija una acción para el blog",
+ "AssignBlogPostsToOtherBlog": "Asignar publicaciones de blog a otro blog",
+ "SelectAnBlogToAssign": "Seleccione un blog para asignar publicaciones de blog",
+ "DeleteAllBlogPostsOfThisBlog": "Eliminar todas las publicaciones de blog de este blog"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fa.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fa.json
index e044cde1a5..fa6695709d 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fa.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fa.json
@@ -183,6 +183,10 @@
"Status": "وضعیت",
"CmsKit.BlogPost.ScrollIndex": "نوار پیمایش سریع در پست های وبلاگ",
"CmsKit.BlogPost.PreventXssFeature": "جلوگیری از XSS",
- "ThisPartOfContentCouldntBeLoaded": "این قسمت از محتوا قابل بارگیری نیست."
+ "ThisPartOfContentCouldntBeLoaded": "این قسمت از محتوا قابل بارگیری نیست.",
+ "ChooseAnActionForBlog": "یک عمل برای وبلاگ انتخاب کنید",
+ "AssignBlogPostsToOtherBlog": "پست های وبلاگ را به وبلاگ دیگری اختصاص دهید",
+ "SelectAnBlogToAssign": "یک وبلاگ برای اختصاص دادن انتخاب کنید",
+ "DeleteAllBlogPostsOfThisBlog": "تمام پست های وبلاگ این وبلاگ را حذف کنید"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fi.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fi.json
index 82d622f05f..f13dc7620c 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fi.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fi.json
@@ -227,6 +227,10 @@
"CssClass": "CSS-luokka",
"TagsHelpText": "Tunnisteet on erotettava pilkuilla (esim. tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "Tämä sisällön osa ei voitu ladata.",
- "DuplicateCommentAttemptMessage": "Kopiokommenttiyritys havaittiin. Kommenttisi on jo lähetetty."
+ "DuplicateCommentAttemptMessage": "Kopiokommenttiyritys havaittiin. Kommenttisi on jo lähetetty.",
+ "ChooseAnActionForBlog": "Valitse toiminto blogille",
+ "AssignBlogPostsToOtherBlog": "Määritä blogiviestit toiseen blogiin",
+ "SelectAnBlogToAssign": "Valitse blogi, johon haluat määrittää",
+ "DeleteAllBlogPostsOfThisBlog": "Poista tämän blogin kaikki blogiviestit"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fr.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fr.json
index 634c1fec85..f2828aa40e 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fr.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/fr.json
@@ -227,6 +227,10 @@
"CssClass": "Classe CSS",
"TagsHelpText": "Les balises doivent être séparées par des virgules (par exemple : tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "Cette partie du contenu n'a pas pu être chargée.",
- "DuplicateCommentAttemptMessage": "Tentative de publication de commentaire en double détectée. Votre commentaire a déjà été soumis."
+ "DuplicateCommentAttemptMessage": "Tentative de publication de commentaire en double détectée. Votre commentaire a déjà été soumis.",
+ "ChooseAnActionForBlog": "Choisissez une action pour le blog",
+ "AssignBlogPostsToOtherBlog": "Attribuer des articles de blog à un autre blog",
+ "SelectAnBlogToAssign": "Sélectionnez un blog à attribuer",
+ "DeleteAllBlogPostsOfThisBlog": "Supprimer tous les articles de blog de ce blog"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hi.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hi.json
index dcea398df5..ad2dbc18d9 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hi.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hi.json
@@ -227,6 +227,10 @@
"CssClass": "सीएसएस क्लास",
"TagsHelpText": "टैग को अल्पविराम से अलग किया जाना चाहिए (जैसे: टैग1, टैग2, टैग3)",
"ThisPartOfContentCouldntBeLoaded": "यह भाग लोड नहीं किया जा सका।",
- "DuplicateCommentAttemptMessage": "डुप्लिकेट टिप्पणी पोस्ट प्रयास का पता चला। आपकी टिप्पणी पहले ही सबमिट की जा चुकी है."
+ "DuplicateCommentAttemptMessage": "डुप्लिकेट टिप्पणी पोस्ट प्रयास का पता चला। आपकी टिप्पणी पहले ही सबमिट की जा चुकी है.",
+ "ChooseAnActionForBlog": "ब्लॉग के लिए कोई कार्रवाई चुनें",
+ "AssignBlogPostsToOtherBlog": "अन्य ब्लॉग को ब्लॉग पोस्ट असाइन करें",
+ "SelectAnBlogToAssign": "असाइन करने के लिए एक ब्लॉग चुनें",
+ "DeleteAllBlogPostsOfThisBlog": "इस ब्लॉग के सभी ब्लॉग पोस्ट हटाएं"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hr.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hr.json
index d2dff83aee..8e35745e21 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hr.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hr.json
@@ -227,6 +227,10 @@
"CssClass": "CSS klasa",
"TagsHelpText": "Oznake trebaju biti odvojene zarezom (npr. oznaka1, oznaka2, oznaka3)",
"ThisPartOfContentCouldntBeLoaded": "Ovaj dio sadržaja nije bilo moguće učitati.",
- "DuplicateCommentAttemptMessage": "Otkriven pokušaj postavljanja duplikata komentara. Vaš komentar je već poslan."
+ "DuplicateCommentAttemptMessage": "Otkriven pokušaj postavljanja duplikata komentara. Vaš komentar je već poslan.",
+ "ChooseAnActionForBlog": "Odaberite radnju za blog",
+ "AssignBlogPostsToOtherBlog": "Dodijelite postove na blogu drugom blogu",
+ "SelectAnBlogToAssign": "Odaberite blog za dodjelu",
+ "DeleteAllBlogPostsOfThisBlog": "Izbrišite sve postove na blogu"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hu.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hu.json
index 219d9f95a7..9a3e4fb64e 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hu.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/hu.json
@@ -227,6 +227,10 @@
"CssClass": "CSS osztály",
"TagsHelpText": "A címkéket vesszővel kell elválasztani (pl.: tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "A tartalom ezen része nem tölthető be.",
- "DuplicateCommentAttemptMessage": "Ismétlődő megjegyzés közzétételi kísérlet észlelve. Megjegyzését már elküldtük."
+ "DuplicateCommentAttemptMessage": "Ismétlődő megjegyzés közzétételi kísérlet észlelve. Megjegyzését már elküldtük.",
+ "ChooseAnActionForBlog": "Válasszon egy műveletet a bloghoz",
+ "AssignBlogPostsToOtherBlog": "Blogbejegyzések hozzárendelése egy másik bloghoz",
+ "SelectAnBlogToAssign": "Válasszon egy blogot a hozzárendeléshez",
+ "DeleteAllBlogPostsOfThisBlog": "Ez a művelet törli az összes blogbejegyzést ebből a blogból. Biztos vagy benne?"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/is.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/is.json
index 6788847dd7..bf02d1ff68 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/is.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/is.json
@@ -227,6 +227,10 @@
"CssClass": "CSS flokkur",
"TagsHelpText": "Merki ættu að vera aðskilin með kommum (td: tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "Þessi hluti af efni gat ekki verið hlaðið inn.",
- "DuplicateCommentAttemptMessage": "Tvítekin tilraun til að skrifa athugasemd fannst. Athugasemd þín hefur þegar verið send."
+ "DuplicateCommentAttemptMessage": "Tvítekin tilraun til að skrifa athugasemd fannst. Athugasemd þín hefur þegar verið send.",
+ "ChooseAnActionForBlog": "Veldu aðgerð fyrir blogg",
+ "AssignBlogPostsToOtherBlog": "Úthluta bloggfærslum til annars bloggs",
+ "SelectAnBlogToAssign": "Veldu blogg til að úthluta",
+ "DeleteAllBlogPostsOfThisBlog": "Eyða öllum bloggfærslum þessa bloggs"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/it.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/it.json
index c6d84a4f62..7506c439f3 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/it.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/it.json
@@ -227,6 +227,10 @@
"CssClass": "Classe CSS",
"TagsHelpText": "I tag devono essere separati da virgole (ad esempio: tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "Questa parte del contenuto non può essere caricata.",
- "DuplicateCommentAttemptMessage": "Rilevato tentativo di pubblicare commenti duplicati. Il tuo commento è già stato inviato."
+ "DuplicateCommentAttemptMessage": "Rilevato tentativo di pubblicare commenti duplicati. Il tuo commento è già stato inviato.",
+ "ChooseAnActionForBlog": "Scegli un'azione per il blog",
+ "AssignBlogPostsToOtherBlog": "Assegna i post del blog ad un altro blog",
+ "SelectAnBlogToAssign": "Seleziona un blog a cui assegnare i post del blog",
+ "DeleteAllBlogPostsOfThisBlog": "Elimina tutti i post del blog di questo blog"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/nl.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/nl.json
index b7c275078a..1d367dfa5c 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/nl.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/nl.json
@@ -227,6 +227,10 @@
"CssClass": "CSS-klasse",
"TagsHelpText": "Tags moeten door komma's worden gescheiden (bijvoorbeeld: tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "Dit deel van de inhoud kon niet worden geladen.",
- "DuplicateCommentAttemptMessage": "Dubbele poging tot posten van commentaar gedetecteerd. Uw reactie is al verzonden."
+ "DuplicateCommentAttemptMessage": "Dubbele poging tot posten van commentaar gedetecteerd. Uw reactie is al verzonden.",
+ "ChooseAnActionForBlog": "Kies een actie voor de blog",
+ "AssignBlogPostsToOtherBlog": "Wijs blogberichten toe aan een andere blog",
+ "SelectAnBlogToAssign": "Selecteer een blog om toe te wijzen",
+ "DeleteAllBlogPostsOfThisBlog": "Verwijder alle blogberichten van deze blog"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/pl-PL.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/pl-PL.json
index 5237c0dc95..0b084f585c 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/pl-PL.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/pl-PL.json
@@ -227,6 +227,10 @@
"CssClass": "Klasa CSS",
"TagsHelpText": "Tagi powinny być oddzielone przecinkami (np. tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "Ta część zawartości nie mogła zostać załadowana.",
- "DuplicateCommentAttemptMessage": "Wykryto zduplikowaną próbę opublikowania komentarza. Twój komentarz został już przesłany."
+ "DuplicateCommentAttemptMessage": "Wykryto zduplikowaną próbę opublikowania komentarza. Twój komentarz został już przesłany.",
+ "ChooseAnActionForBlog": "Wybierz akcję dla bloga",
+ "AssignBlogPostsToOtherBlog": "Przypisz posty na blogu do innego bloga",
+ "SelectAnBlogToAssign": "Wybierz blog, do którego chcesz przypisać posty na blogu",
+ "DeleteAllBlogPostsOfThisBlog": "Usuń wszystkie posty na blogu tego bloga"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/pt-BR.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/pt-BR.json
index 3182469634..3991bc0595 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/pt-BR.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/pt-BR.json
@@ -227,6 +227,10 @@
"CssClass": "Classe CSS",
"TagsHelpText": "As tags devem ser separadas por vírgula (por exemplo: tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "Esta parte do conteúdo não pôde ser carregada.",
- "DuplicateCommentAttemptMessage": "Tentativa duplicada de postagem de comentário detectada. Seu comentário já foi enviado."
+ "DuplicateCommentAttemptMessage": "Tentativa duplicada de postagem de comentário detectada. Seu comentário já foi enviado.",
+ "ChooseAnActionForBlog": "Escolha uma ação para o blog",
+ "AssignBlogPostsToOtherBlog": "Atribuir postagens de blog a outro blog",
+ "SelectAnBlogToAssign": "Selecione um blog para atribuir",
+ "DeleteAllBlogPostsOfThisBlog": "Excluir todas as postagens de blog deste blog"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ro-RO.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ro-RO.json
index 7b9173f477..c6b486a11a 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ro-RO.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ro-RO.json
@@ -227,6 +227,10 @@
"CssClass": "Clasa CSS",
"TagsHelpText": "Etichetele ar trebui să fie separate prin virgulă (de exemplu: tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "Această parte a conţinutului nu a putut fi încărcată.",
- "DuplicateCommentAttemptMessage": "A fost detectată o încercare de postare de comentarii duplicată. Comentariul dvs. a fost deja trimis."
+ "DuplicateCommentAttemptMessage": "A fost detectată o încercare de postare de comentarii duplicată. Comentariul dvs. a fost deja trimis.",
+ "ChooseAnActionForBlog": "Alegeţi o acţiune pentru blog",
+ "AssignBlogPostsToOtherBlog": "Atribuiţi postările de blog la alt blog",
+ "SelectAnBlogToAssign": "Selectaţi un blog pentru a atribui postările de blog",
+ "DeleteAllBlogPostsOfThisBlog": "Ştergeţi toate postările de blog ale acestui blog"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ru.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ru.json
index 491435bc02..c23671c843 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ru.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/ru.json
@@ -227,6 +227,10 @@
"CssClass": "CSS-класс",
"TagsHelpText": "Теги должны быть разделены запятыми (например: тег1, тег2, тег3).",
"ThisPartOfContentCouldntBeLoaded": "Эта часть содержимого не может быть загружена.",
- "DuplicateCommentAttemptMessage": "Обнаружена повторная попытка публикации комментария. Ваш комментарий уже отправлен."
+ "DuplicateCommentAttemptMessage": "Обнаружена повторная попытка публикации комментария. Ваш комментарий уже отправлен.",
+ "ChooseAnActionForBlog": "Выберите действие для блога",
+ "AssignBlogPostsToOtherBlog": "Назначить сообщения в блоге другому блогу",
+ "SelectAnBlogToAssign": "Выберите блог для назначения",
+ "DeleteAllBlogPostsOfThisBlog": "Удалить все сообщения в блоге этого блога"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sk.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sk.json
index c579340750..a1ff2cfb01 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sk.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sk.json
@@ -227,6 +227,10 @@
"CssClass": "CSS trieda",
"TagsHelpText": "Značky by mali byť oddelené čiarkou (napr.: tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "Táto časť obsahu sa nedá načítať.",
- "DuplicateCommentAttemptMessage": "Zistil sa duplicitný pokus o uverejnenie komentára. Váš komentár už bol odoslaný."
+ "DuplicateCommentAttemptMessage": "Zistil sa duplicitný pokus o uverejnenie komentára. Váš komentár už bol odoslaný.",
+ "ChooseAnActionForBlog": "Vyberte akciu pre blog",
+ "AssignBlogPostsToOtherBlog": "Priradiť blogové príspevky k inému blogu",
+ "SelectAnBlogToAssign": "Vyberte blog, na ktorý chcete priradiť blogové príspevky",
+ "DeleteAllBlogPostsOfThisBlog": "Zmazať všetky blogové príspevky tohto blogu"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sl.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sl.json
index 6e3157f56e..b4d525c3e4 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sl.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sl.json
@@ -227,6 +227,10 @@
"CssClass": "Razred CSS",
"TagsHelpText": "Oznake morajo biti ločene z vejicami (npr. oznaka1, oznaka2, oznaka3)",
"ThisPartOfContentCouldntBeLoaded": "Ta del vsebine ni bil naložen.",
- "DuplicateCommentAttemptMessage": "Zaznan poskus podvojene objave komentarja. Vaš komentar je že oddan."
+ "DuplicateCommentAttemptMessage": "Zaznan poskus podvojene objave komentarja. Vaš komentar je že oddan.",
+ "ChooseAnActionForBlog": "Izberite dejanje za blog",
+ "AssignBlogPostsToOtherBlog": "Dodeli objave v blogu drugemu blogu",
+ "SelectAnBlogToAssign": "Izberite blog, ki mu želite dodeliti objave",
+ "DeleteAllBlogPostsOfThisBlog": "Izbriši vse objave v tem blogu"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sv.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sv.json
index 6e8962915b..f9604657ce 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sv.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/sv.json
@@ -254,6 +254,10 @@
"CommentAlertMessage": "Det finns {0} kommentarer som väntar på godkännande",
"Settings:Menu:CmsKit": "CMS",
"CommentsAwaitingApproval": "Kommentarer som väntar på godkännande",
- "CommentSubmittedForApproval": "Din kommentar har skickats in för godkännande."
+ "CommentSubmittedForApproval": "Din kommentar har skickats in för godkännande.",
+ "ChooseAnActionForBlog": "Välj en åtgärd för bloggen",
+ "AssignBlogPostsToOtherBlog": "Tilldela blogginlägg till en annan blogg",
+ "SelectAnBlogToAssign": "Välj en blogg att tilldela",
+ "DeleteAllBlogPostsOfThisBlog": "Radera alla blogginlägg i denna blogg"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/tr.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/tr.json
index 3605cb9d3b..0e5d6d7c11 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/tr.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/tr.json
@@ -218,6 +218,10 @@
"CaptchaCode": "Captcha kodu",
"CommentTextRequired": "Yorum zorunlu",
"PopularTags": "Popüler Etiketler",
- "ThisPartOfContentCouldntBeLoaded": "Bu içerik yüklenemedi"
+ "ThisPartOfContentCouldntBeLoaded": "Bu içerik yüklenemedi",
+ "ChooseAnActionForBlog": "Blog için bir eylem seçin",
+ "AssignBlogPostsToOtherBlog": "Diğer bloglara blog yazıları atayın",
+ "SelectAnBlogToAssign": "Atanacak bir blog seçin",
+ "DeleteAllBlogPostsOfThisBlog": "Bu blogun tüm blog yazılarını sil"
}
}
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/vi.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/vi.json
index 1ec430edc3..0a2159e104 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/vi.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/vi.json
@@ -227,6 +227,10 @@
"CssClass": "Lớp CSS",
"TagsHelpText": "Các thẻ phải được phân tách bằng dấu phẩy (ví dụ: tag1, tag2, tag3)",
"ThisPartOfContentCouldntBeLoaded": "Phần này của nội dung không thể được tải.",
- "DuplicateCommentAttemptMessage": "Đã phát hiện nỗ lực đăng bình luận trùng lặp. Bình luận của bạn đã được gửi."
+ "DuplicateCommentAttemptMessage": "Đã phát hiện nỗ lực đăng bình luận trùng lặp. Bình luận của bạn đã được gửi.",
+ "ChooseAnActionForBlog": "Chọn một hành động cho blog",
+ "AssignBlogPostsToOtherBlog": "Gán bài đăng trên blog cho blog khác",
+ "SelectAnBlogToAssign": "Chọn một blog để gán",
+ "DeleteAllBlogPostsOfThisBlog": "Xóa tất cả bài đăng trên blog của blog này"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/zh-Hans.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/zh-Hans.json
index c02fcf614d..18ae3004cb 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/zh-Hans.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/zh-Hans.json
@@ -227,6 +227,10 @@
"CssClass": "CSS 类",
"TagsHelpText": "标签应以逗号分隔(例如:标签 1,标签 2,标签 3)",
"ThisPartOfContentCouldntBeLoaded": "这部分内容无法加载。",
- "DuplicateCommentAttemptMessage": "检测到重复发表评论。您的评论已经提交。"
+ "DuplicateCommentAttemptMessage": "检测到重复发表评论。您的评论已经提交。",
+ "ChooseAnActionForBlog": "选择博客的操作",
+ "AssignBlogPostsToOtherBlog": "将博客文章分配给其他博客",
+ "SelectAnBlogToAssign": "选择要分配的博客",
+ "DeleteAllBlogPostsOfThisBlog": "删除此博客的所有博客文章"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/zh-Hant.json b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/zh-Hant.json
index 866fd3fcea..80c30a6be3 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/zh-Hant.json
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Localization/Resources/zh-Hant.json
@@ -227,6 +227,10 @@
"CssClass": "CSS類",
"TagsHelpText": "標籤應以逗號分隔(例如:tag1、tag2、tag3)",
"ThisPartOfContentCouldntBeLoaded": "這部分內容無法加載",
- "DuplicateCommentAttemptMessage": "偵測到重複的評論貼文嘗試。您的評論已經提交。"
+ "DuplicateCommentAttemptMessage": "偵測到重複的評論貼文嘗試。您的評論已經提交。",
+ "ChooseAnActionForBlog": "選擇部落格的操作",
+ "AssignBlogPostsToOtherBlog": "將部落格文章分配給其他部落格",
+ "SelectAnBlogToAssign": "選擇要分配的部落格",
+ "DeleteAllBlogPostsOfThisBlog": "刪除此部落格的所有部落格文章"
}
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Pages/PageCacheItem.cs b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Pages/PageCacheItem.cs
similarity index 96%
rename from modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Pages/PageCacheItem.cs
rename to modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Pages/PageCacheItem.cs
index d4f7925160..d4d662a8a6 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Pages/PageCacheItem.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain.Shared/Volo/CmsKit/Pages/PageCacheItem.cs
@@ -3,10 +3,11 @@ using Volo.Abp.ObjectExtending;
namespace Volo.CmsKit.Pages;
+[Serializable]
public class PageCacheItem : ExtensibleObject
{
public Guid Id { get; set; }
-
+
public string Title { get; set; }
public string Slug { get; set; }
@@ -23,4 +24,4 @@ public class PageCacheItem : ExtensibleObject
{
return $"CmsPage_{slug}";
}
-}
\ No newline at end of file
+}
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/BlogPost.cs b/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/BlogPost.cs
index 3a4cf26c1b..a34a303008 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/BlogPost.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/BlogPost.cs
@@ -53,7 +53,7 @@ public class BlogPost : FullAuditedAggregateRoot, IMultiTenant, IHasEntity
) : base(id)
{
TenantId = tenantId;
- BlogId = blogId;
+ SetBlogId(blogId);
AuthorId = authorId;
SetTitle(title);
SetSlug(slug);
@@ -68,6 +68,11 @@ public class BlogPost : FullAuditedAggregateRoot, IMultiTenant, IHasEntity
Title = Check.NotNullOrWhiteSpace(title, nameof(title), BlogPostConsts.MaxTitleLength);
}
+ public virtual void SetBlogId(Guid blogId)
+ {
+ BlogId = blogId;
+ }
+
internal void SetSlug(string slug)
{
Check.NotNullOrWhiteSpace(slug, nameof(slug), BlogPostConsts.MaxSlugLength, BlogPostConsts.MinSlugLength);
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/BlogWithBlogPostCount.cs b/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/BlogWithBlogPostCount.cs
new file mode 100644
index 0000000000..9a84402375
--- /dev/null
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/BlogWithBlogPostCount.cs
@@ -0,0 +1,14 @@
+namespace Volo.CmsKit.Blogs;
+
+public class BlogWithBlogPostCount
+{
+ public Blog Blog { get; set; }
+
+ public int BlogPostCount { get; set; }
+
+ public BlogWithBlogPostCount(Blog blog, int blogPostCount)
+ {
+ Blog = blog;
+ BlogPostCount = blogPostCount;
+ }
+}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/IBlogPostRepository.cs b/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/IBlogPostRepository.cs
index 404d21dc83..885d0de800 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/IBlogPostRepository.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/IBlogPostRepository.cs
@@ -44,4 +44,6 @@ public interface IBlogPostRepository : IBasicRepository
Task GetAuthorHasBlogPostAsync(Guid id, CancellationToken cancellationToken = default);
Task HasBlogPostWaitingForReviewAsync(CancellationToken cancellationToken = default);
+
+ Task UpdateBlogAsync(Guid sourceBlogId, Guid? targetBlogId = null, CancellationToken cancellationToken = default);
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/IBlogRepository.cs b/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/IBlogRepository.cs
index 60005db319..aebdfc3f24 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/IBlogRepository.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.Domain/Volo/CmsKit/Blogs/IBlogRepository.cs
@@ -16,6 +16,13 @@ public interface IBlogRepository : IBasicRepository
CancellationToken cancellationToken = default
);
+ Task> GetListWithBlogPostCountAsync(
+ string filter = null,
+ string sorting = null,
+ int maxResultCount = int.MaxValue,
+ int skipCount = 0,
+ CancellationToken cancellationToken = default);
+
Task GetCountAsync(
string filter = null,
CancellationToken cancellationToken = default
diff --git a/modules/cms-kit/src/Volo.CmsKit.EntityFrameworkCore/Volo/CmsKit/Blogs/EfCoreBlogPostRepository.cs b/modules/cms-kit/src/Volo.CmsKit.EntityFrameworkCore/Volo/CmsKit/Blogs/EfCoreBlogPostRepository.cs
index 266a04a910..1160387151 100644
--- a/modules/cms-kit/src/Volo.CmsKit.EntityFrameworkCore/Volo/CmsKit/Blogs/EfCoreBlogPostRepository.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.EntityFrameworkCore/Volo/CmsKit/Blogs/EfCoreBlogPostRepository.cs
@@ -160,4 +160,17 @@ public class EfCoreBlogPostRepository : EfCoreRepository x.Status == BlogPostStatus.WaitingForReview, GetCancellationToken(cancellationToken));
}
+
+ public async Task UpdateBlogAsync(Guid sourceBlogId, Guid? targetBlogId = null, CancellationToken cancellationToken = default)
+ {
+ if (targetBlogId != null)
+ {
+ await (await GetDbSetAsync()).Where(x => x.BlogId == sourceBlogId).ExecuteUpdateAsync(x => x.SetProperty(b => b.BlogId, targetBlogId.Value), GetCancellationToken(cancellationToken));
+
+ }
+ else
+ {
+ await (await GetDbSetAsync()).Where(x => x.BlogId == sourceBlogId).ExecuteDeleteAsync(GetCancellationToken(cancellationToken));
+ }
+ }
}
\ No newline at end of file
diff --git a/modules/cms-kit/src/Volo.CmsKit.EntityFrameworkCore/Volo/CmsKit/Blogs/EfCoreBlogRepository.cs b/modules/cms-kit/src/Volo.CmsKit.EntityFrameworkCore/Volo/CmsKit/Blogs/EfCoreBlogRepository.cs
index daf915f675..5234577f02 100644
--- a/modules/cms-kit/src/Volo.CmsKit.EntityFrameworkCore/Volo/CmsKit/Blogs/EfCoreBlogRepository.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.EntityFrameworkCore/Volo/CmsKit/Blogs/EfCoreBlogRepository.cs
@@ -43,6 +43,31 @@ public class EfCoreBlogRepository : EfCoreRepository> GetListWithBlogPostCountAsync(
+ string filter = null,
+ string sorting = null,
+ int maxResultCount = int.MaxValue,
+ int skipCount = 0,
+ CancellationToken cancellationToken = default)
+ {
+ var blogs = await (await GetListQueryAsync(filter)).OrderBy(sorting.IsNullOrEmpty() ? "creationTime desc" : sorting)
+ .PageBy(skipCount, maxResultCount).ToListAsync(GetCancellationToken(cancellationToken));
+
+ var blogIds = blogs.Select(x => x.Id).ToArray();
+
+ var blogPostCount = await (await GetDbContextAsync()).Set()
+ .Where(blogPost => blogIds.Contains(blogPost.BlogId))
+ .GroupBy(blogPost => blogPost.BlogId)
+ .Select(x => new
+ {
+ BlogId = x.Key,
+ Count = x.Count()
+ })
+ .ToListAsync(GetCancellationToken(cancellationToken));
+
+ return blogs.Select(blog => new BlogWithBlogPostCount(blog, blogPostCount.FirstOrDefault(x => x.BlogId == blog.Id)?.Count ?? 0)).ToList();
+ }
+
public virtual async Task GetCountAsync(string filter = null, CancellationToken cancellationToken = default)
{
var query = await GetListQueryAsync(filter);
diff --git a/modules/cms-kit/src/Volo.CmsKit.MongoDB/Volo/CmsKit/MongoDB/Blogs/MongoBlogPostRepository.cs b/modules/cms-kit/src/Volo.CmsKit.MongoDB/Volo/CmsKit/MongoDB/Blogs/MongoBlogPostRepository.cs
index d3ae31926b..d27fdf16b6 100644
--- a/modules/cms-kit/src/Volo.CmsKit.MongoDB/Volo/CmsKit/MongoDB/Blogs/MongoBlogPostRepository.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.MongoDB/Volo/CmsKit/MongoDB/Blogs/MongoBlogPostRepository.cs
@@ -192,4 +192,24 @@ public class MongoBlogPostRepository : MongoDbRepository x.Status == BlogPostStatus.WaitingForReview, cancellationToken);
}
+
+ public async Task UpdateBlogAsync(Guid sourceBlogId, Guid? targetBlogId, CancellationToken cancellationToken = default)
+ {
+ cancellationToken = GetCancellationToken(cancellationToken);
+ var blogPosts = await (await GetMongoQueryableAsync(cancellationToken)).Where(x => x.BlogId == sourceBlogId).ToListAsync(cancellationToken);
+ if (targetBlogId.HasValue)
+ {
+ foreach (var blogPost in blogPosts)
+ {
+ blogPost.SetBlogId(targetBlogId.Value);
+ }
+
+ await UpdateManyAsync(blogPosts, false, cancellationToken);
+ }
+ else
+ {
+
+ await DeleteManyAsync(blogPosts, false, cancellationToken);
+ }
+ }
}
diff --git a/modules/cms-kit/src/Volo.CmsKit.MongoDB/Volo/CmsKit/MongoDB/Blogs/MongoBlogRepository.cs b/modules/cms-kit/src/Volo.CmsKit.MongoDB/Volo/CmsKit/MongoDB/Blogs/MongoBlogRepository.cs
index b34c452621..98567899e5 100644
--- a/modules/cms-kit/src/Volo.CmsKit.MongoDB/Volo/CmsKit/MongoDB/Blogs/MongoBlogRepository.cs
+++ b/modules/cms-kit/src/Volo.CmsKit.MongoDB/Volo/CmsKit/MongoDB/Blogs/MongoBlogRepository.cs
@@ -49,6 +49,33 @@ public class MongoBlogRepository : MongoDbRepository> GetListWithBlogPostCountAsync(
+ string filter = null,
+ string sorting = null,
+ int maxResultCount = int.MaxValue,
+ int skipCount = 0,
+ CancellationToken cancellationToken = default)
+ {
+ var token = GetCancellationToken(cancellationToken);
+
+ var blogs = await GetListQueryAsync(filter, token);
+
+ var blogIds = blogs.OrderBy(sorting.IsNullOrEmpty() ? "creationTime desc" : sorting)
+ .PageBy(skipCount, maxResultCount).Select(x => x.Id).ToList();
+
+ var blogPostCount = await (await GetMongoQueryableAsync(token))
+ .Where(blogPost => blogIds.Contains(blogPost.Id))
+ .GroupBy(blogPost => blogPost.BlogId)
+ .Select(x => new
+ {
+ BlogId = x.Key,
+ Count = x.Count()
+ })
+ .ToListAsync(GetCancellationToken(cancellationToken));
+
+ return blogs.Select(blog => new BlogWithBlogPostCount(blog, blogPostCount.FirstOrDefault(x => x.BlogId == blog.Id) != null ? blogPostCount.FirstOrDefault(x => x.BlogId == blog.Id).Count : 0)).ToList();
+ }
+
public virtual async Task GetCountAsync(string filter = null, CancellationToken cancellationToken = default)
{
var token = GetCancellationToken(cancellationToken);
diff --git a/modules/cms-kit/src/Volo.CmsKit.Public.Web/Pages/CmsKit/Shared/Components/Commenting/Default.cshtml b/modules/cms-kit/src/Volo.CmsKit.Public.Web/Pages/CmsKit/Shared/Components/Commenting/Default.cshtml
index 1b86ade0a8..7033978f48 100644
--- a/modules/cms-kit/src/Volo.CmsKit.Public.Web/Pages/CmsKit/Shared/Components/Commenting/Default.cshtml
+++ b/modules/cms-kit/src/Volo.CmsKit.Public.Web/Pages/CmsKit/Shared/Components/Commenting/Default.cshtml
@@ -87,32 +87,32 @@
Func GetCommentActionArea(Guid id, Guid authorId, bool isReply) =>
@