diff --git a/.github/ISSUE_TEMPLATE/config.yml b/.github/ISSUE_TEMPLATE/config.yml
index 9fca40ba4f..fe03c27997 100644
--- a/.github/ISSUE_TEMPLATE/config.yml
+++ b/.github/ISSUE_TEMPLATE/config.yml
@@ -1,4 +1,4 @@
-blank_issues_enabled: true
+blank_issues_enabled: false
contact_links:
- name: Issue with ABP Commercial
url: https://abp.io/support/questions
diff --git a/.github/workflows/build-and-test.yml b/.github/workflows/build-and-test.yml
index 7073e826e3..8ebcc32bfc 100644
--- a/.github/workflows/build-and-test.yml
+++ b/.github/workflows/build-and-test.yml
@@ -48,7 +48,7 @@ permissions:
jobs:
build-test:
runs-on: ubuntu-22.04
- timeout-minutes: 40
+ timeout-minutes: 50
if: ${{ !github.event.pull_request.draft }}
steps:
- uses: actions/checkout@v2
diff --git a/Directory.Packages.props b/Directory.Packages.props
index 72405c55cf..7263869bdf 100644
--- a/Directory.Packages.props
+++ b/Directory.Packages.props
@@ -34,6 +34,9 @@
+
+
+
@@ -138,7 +141,7 @@
-
+
@@ -182,4 +185,4 @@
-
+
\ No newline at end of file
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Admin/Localization/Resources/en.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Admin/Localization/Resources/en.json
index 1ea52ff5f5..d056e70b09 100644
--- a/abp_io/AbpIoLocalization/AbpIoLocalization/Admin/Localization/Resources/en.json
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Admin/Localization/Resources/en.json
@@ -690,6 +690,33 @@
"Permission:Campaigns": "Campaigns",
"Permission:Licenses": "License Settings",
"BlockUserPolicy": "User Block Policies",
- "Permission:BlockUserPolicy": "User Block Policy"
+ "Permission:BlockUserPolicy": "User Block Policy",
+ "ManageImages" : "Manage Image",
+ "ModuleMainImage": "Module Main Image",
+ "ModuleImageInfo1": "Accepted file types : JPEG, JPG, PNG, SVG ",
+ "ModuleImageInfo2": "Max file size : 1 MB ",
+ "ModuleImageInfo3": "Image Proportion : 1:1 ",
+ "ModuleImageInfo4": " Download a sample cover image ",
+ "AreYouSureToDeleteImage": "Are you sure you want to delete this image?",
+ "CompactPercentage": "Compact Percentage",
+ "CompactPercentageDescription": "Percentage of cache entries to remove during compaction (1-100)",
+ "Settings": "Settings",
+ "MaxSizeOfInMemoryCache": "Max Size Of In Memory Cache",
+ "MaxSizeOfInMemoryCacheInfo": "Gets or sets the maximum size of the memory cache.",
+ "SizeOfTriggerAutoCompact": "Size Of Trigger Auto Compact",
+ "SizeOfTriggerAutoCompactInfo": "Gets or sets the size threshold that triggers auto-compaction. When the cache size exceeds this value, auto-compaction will be performed",
+ "AutoCompactPercentage": "Auto Compact Percentage",
+ "AutoCompactPercentageInfo": "Gets or sets the percentage of items to remove when auto-compaction is triggered (1-100)",
+ "ClearCaches": "This clears all NuGet content cache",
+ "CacheLatestVersions": "Cache Latest Versions",
+ "VersionCount": "Version Count",
+ "VersionCountInfo": "Gets or sets the number of versions to cache",
+ "IncludeLeptonX": "Include LeptonX",
+ "IncludeLeptonXInfo": "Gets or sets whether to include LeptonX in the cache",
+ "IncludePrerelease": "Include Prerelease",
+ "IncludePrereleaseInfo": "Gets or sets whether to include prerelease versions in the cache",
+ "CacheVersionCount": "Cache Version Count",
+ "CacheVersionCountInfo": "Gets or sets the number of versions to cache",
+ "CacheLatestVersionsInfo": "This caches the latest versions of the NuGet packages"
}
}
diff --git a/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/en.json b/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/en.json
index c56e50af4c..7ea0fb6708 100644
--- a/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/en.json
+++ b/abp_io/AbpIoLocalization/AbpIoLocalization/Www/Localization/Resources/en.json
@@ -66,7 +66,7 @@
"AuditLogging": "Audit Logging",
"Caching": "Caching",
"Multitenancy": "Multitenancy",
- "DataFiltering": "Data filtering",
+ "DataFiltering": "Data Filtering",
"ConventionOverConfiguration": "Convention Over Configuration",
"ConventionOverConfigurationExplanation": "ABP implements common application conventions by default with minimal or zero configuration.",
"ConventionOverConfigurationExplanationList1": "Auto registers known services to dependency injection.",
@@ -1886,6 +1886,10 @@
"FaqIyzicoPaymentIssuesExplanation5": "Credit/Debit Cards: Visa, MasterCard, American Express, Discover, Diners Club, JCB",
"FaqIyzicoPaymentIssuesExplanation6": "Digital Wallets: PayPal, AliPay, WebMoney",
"FaqIyzicoPaymentIssuesExplanation7": "Alternatively, you can send the license amount directly via bank wire transfer. For our bank account details, please visit: Bank Account Information (use USD currency).",
- "FaqIyzicoPaymentIssuesExplanation8": "ABP website doesn't save or process your credit card. We use payment gateways for this and the entire transaction is handled by payment gateways. We have no authority to interfere with the payment process or fix the payment steps. If you have further questions or need additional support, feel free to contact us at abp.io/contact."
+ "FaqIyzicoPaymentIssuesExplanation8": "ABP website doesn't save or process your credit card. We use payment gateways for this and the entire transaction is handled by payment gateways. We have no authority to interfere with the payment process or fix the payment steps. If you have further questions or need additional support, feel free to contact us at abp.io/contact.",
+ "BiographyContainsUrlValidationMessage": "Biography cannot contain URL.",
+ "CreatePostSEOTitleInfo": "SEO URL is a clean, readable, keyword-rich URL that helps both users and search engines understand what this post is about. Keep it short with 60 characters. SEO titles over 60 characters will be truncated. Use hyphens (-) to separate words (not underscores). Include target keywords near the start. Lowercase only. No stop words unless needed (e.g: \"and\", \"or\", \"the\").",
+ "SEOTitle": "SEO URL",
+ "InvalidYouTubeUrl": "The URL you entered is not a valid YouTube video link. Please make sure it points to a specific video and try again."
}
}
diff --git a/common.props b/common.props
index f02a8f2a28..54e6251ac6 100644
--- a/common.props
+++ b/common.props
@@ -1,8 +1,8 @@
latest
- 9.2.0
- 4.2.0
+ 9.3.0-preview
+ 4.3.0-preview$(NoWarn);CS1591;CS0436https://abp.io/assets/abp_nupkg.pnghttps://abp.io/
diff --git a/docs/en/Blog-Posts/2022-03-08 v5_2_Preview/POST.md b/docs/en/Blog-Posts/2022-03-08 v5_2_Preview/POST.md
index b78e3235a2..b0b44b9e45 100644
--- a/docs/en/Blog-Posts/2022-03-08 v5_2_Preview/POST.md
+++ b/docs/en/Blog-Posts/2022-03-08 v5_2_Preview/POST.md
@@ -187,7 +187,7 @@ We organized the 3rd live [ABP Community Talks](https://community.abp.io/events)
* [Enis Necipoğlu](https://twitter.com/EnisNecipoglu) has also created [an article](https://community.abp.io/posts/using-autofilterer-with-abp-framework-uuqv81jm) to demonstrate how to use his own open source [AutoFilterer](https://github.com/enisn/AutoFilterer) library with the ABP Framework.
* [Jonathan Potts](https://github.com/jonathanpotts) has created his first ABP Community article that shows how to use Bootswatch themes with the ABP Framework. [See it here](https://community.abp.io/posts/customizing-the-abp-basic-theme-with-bootswatch-4luoqzr0).
-Thanks to all the contributors. It is appreciated if you want to submit your post and share your knowledge with the ABP community: https://community.abp.io/posts/submit
+Thanks to all the contributors. It is appreciated if you want to submit your post and share your knowledge with the ABP community: https://community.abp.io/posts/create
## Conclusion
diff --git a/docs/en/Blog-Posts/2024-02-16 v8_1_Preview/POST.md b/docs/en/Blog-Posts/2024-02-16 v8_1_Preview/POST.md
index 24f55474ad..70cb53c75e 100644
--- a/docs/en/Blog-Posts/2024-02-16 v8_1_Preview/POST.md
+++ b/docs/en/Blog-Posts/2024-02-16 v8_1_Preview/POST.md
@@ -230,7 +230,7 @@ There are exciting articles contributed by the ABP community as always. I will h
* [A Best Practice for Designing Interfaces in .NET C#](https://community.abp.io/posts/a-best-practice-for-designing-interfaces-in-.net-c-9xqc4h8d)
* [Invariance, Covariance, and Contravariance in .NET C#](https://community.abp.io/posts/invariance-covariance-and-contravariance-in-.net-c-9blmuhme)
-Thanks to the ABP Community for all the content they have published. You can also [post your ABP-related (text or video) content](https://community.abp.io/posts/submit) to the ABP Community.
+Thanks to the ABP Community for all the content they have published. You can also [post your ABP-related (text or video) content](https://community.abp.io/posts/create) to the ABP Community.
## Conclusion
diff --git a/docs/en/Blog-Posts/2024-05-21 v8_2_Release/POST.md b/docs/en/Blog-Posts/2024-05-21 v8_2_Release/POST.md
index 23260950d4..aadaae1c6f 100644
--- a/docs/en/Blog-Posts/2024-05-21 v8_2_Release/POST.md
+++ b/docs/en/Blog-Posts/2024-05-21 v8_2_Release/POST.md
@@ -260,7 +260,7 @@ There are exciting articles contributed by the ABP community as always. I will h
* [Using FluentValidation with ABP Framework](https://community.abp.io/posts/using-fluentvalidation-with-abp-framework-2cxuwl70) by [Enes Döner](https://community.abp.io/members/Enes)
* [Using Blob Storage with ABP](https://community.abp.io/posts/using-blob-storage-with-abp-framework-jygtmhn4) by [Emre Kendirli](https://community.abp.io/members/emrekenderli)
-Thanks to the ABP Community for all the content they have published. You can also [post your ABP-related (text or video) content](https://community.abp.io/posts/submit) to the ABP Community.
+Thanks to the ABP Community for all the content they have published. You can also [post your ABP-related (text or video) content](https://community.abp.io/posts/create) to the ABP Community.
## Conclusion
diff --git a/docs/en/Blog-Posts/2024-07-31 v8_3_Preview/post.md b/docs/en/Blog-Posts/2024-07-31 v8_3_Preview/post.md
index 2d8a767568..fc3590b0d5 100644
--- a/docs/en/Blog-Posts/2024-07-31 v8_3_Preview/post.md
+++ b/docs/en/Blog-Posts/2024-07-31 v8_3_Preview/post.md
@@ -168,7 +168,7 @@ There are exciting articles contributed by the ABP community as always. I will h
* [How to use .NET Aspire with ABP framework](https://abp.io/community/articles/how-to-use-.net-aspire-with-abp-framework-h29km4kk) by [Berkan Şaşmaz](https://twitter.com/berkansasmazz)
* [Exciting New Feature in ABP.IO CMS Kit: Marked Item System](https://abp.io/community/articles/exciting-new-feature-in-abp.io-cms-kit-marked-item-system.-2hvpq0me) by [Suhaib Mousa](https://abp.io/community/members/suhaibmousa032@gmail.com)
-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.
+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/create) to the ABP Community.
## Conclusion
diff --git a/docs/en/Blog-Posts/2024-09-06 v8_3_Release_Stable/POST.md b/docs/en/Blog-Posts/2024-09-06 v8_3_Release_Stable/POST.md
index e69fbb86ea..3ac9de075a 100644
--- a/docs/en/Blog-Posts/2024-09-06 v8_3_Release_Stable/POST.md
+++ b/docs/en/Blog-Posts/2024-09-06 v8_3_Release_Stable/POST.md
@@ -70,7 +70,7 @@ As always, exciting articles have been contributed by the ABP community. I will
* [Introducing the Google Cloud Storage BLOB Provider](https://abp.io/community/articles/introducing-the-google-cloud-storage-blob-provider-yrt6azc0) by [Engincan Veske](https://twitter.com/EngincanVeske)
* [Switching Between Organization Units](https://abp.io/community/articles/switching-between-organization-units-i5tokpzt) by [Liming Ma](https://github.com/maliming)
-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.
+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/create) to the ABP Community.
## About the Next Version
diff --git a/docs/en/Blog-Posts/2024-10-23 v9_0_Preview/POST.md b/docs/en/Blog-Posts/2024-10-23 v9_0_Preview/POST.md
index f06648e586..db52eed7d5 100644
--- a/docs/en/Blog-Posts/2024-10-23 v9_0_Preview/POST.md
+++ b/docs/en/Blog-Posts/2024-10-23 v9_0_Preview/POST.md
@@ -214,7 +214,7 @@ There are exciting articles contributed by the ABP community as always. I will h
* [ABP-Powered Web App with Inertia.js, React, and Vite](https://abp.io/community/articles/abppowered-web-app-with-inertia.js-react-and-vite-j7cccvad) by [Anto Subash](https://antosubash.com/)
* [Multi-Tenancy Support in Angular Apps with ABP.IO](https://abp.io/community/articles/multitenancy-support-in-angular-apps-with-abp.io-lw9l36c5) by [HeadChannel Team](https://headchannel.co.uk/)
-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.
+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/create) to the ABP Community.
## Conclusion
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
index 26850bc57f..9c96351d96 100644
--- 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
@@ -78,7 +78,7 @@ In addition to [the articles to highlight .NET 9.0 features written by our team]
* [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.
+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/create) to the ABP Community.
### ABP Community Talks 2024.7: What’s New with .NET 9 & ABP 9?
diff --git a/docs/en/Blog-Posts/2025-01-21 v9_1_Preview/POST.md b/docs/en/Blog-Posts/2025-01-21 v9_1_Preview/POST.md
index bb8bfdf4c7..bbcc69c727 100644
--- a/docs/en/Blog-Posts/2025-01-21 v9_1_Preview/POST.md
+++ b/docs/en/Blog-Posts/2025-01-21 v9_1_Preview/POST.md
@@ -138,7 +138,7 @@ There are exciting articles contributed by the ABP community as always. I will h
* [The new Unit Test structure in ABP application](https://abp.io/community/articles/the-new-unit-test-structure-in-abp-application-4vvvp2oy) by [Liming Ma](https://github.com/maliming)
* [How to Use OpenAI API with ABP Framework](https://abp.io/community/articles/how-to-use-openai-api-with-abp-framework-rsfvihla) by [Berkan Şaşmaz](https://github.com/berkansasmaz)
-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.
+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/create) to the ABP Community.
## Conclusion
diff --git a/docs/en/Blog-Posts/2025-03-07 v9_1_Release_Stable/POST.md b/docs/en/Blog-Posts/2025-03-07 v9_1_Release_Stable/POST.md
index 96096807f1..2e6764e9b7 100644
--- a/docs/en/Blog-Posts/2025-03-07 v9_1_Release_Stable/POST.md
+++ b/docs/en/Blog-Posts/2025-03-07 v9_1_Release_Stable/POST.md
@@ -67,7 +67,7 @@ As always, exciting articles have been contributed by the ABP community. I will
* [Containerization: Blazor WASM + JWT Web API => Docker](https://abp.io/community/articles/containerization-blazor-wasm-jwt-web-api-docker-i3eirlsf) by [Bart Van Hoey](https://abp.io/community/members/bartvanhoey)
* [Configuring Post-Logout Redirect URI in ABP Based Blazor Applications with OpenIddict](https://abp.io/community/articles/configuring-postlogout-redirect-uri-in-abp-based-blazor-applications-with-openiddict-1t84suxg) by [Engincan Veske](https://github.com/EngincanV)
-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.
+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/create) to the ABP Community.
### ABP Community Talks 2025.2: Real World Problems and Solutions with AI
diff --git a/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/POST.md b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/POST.md
new file mode 100644
index 0000000000..e59e75f44a
--- /dev/null
+++ b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/POST.md
@@ -0,0 +1,195 @@
+# ABP Platform 9.2 RC Has Been Released
+
+We are happy to release [ABP](https://abp.io) version **9.2 RC** (Release Candidate). This blog post introduces the new features and important changes in this new version.
+
+Try this version and provide feedback for a more stable version of ABP v9.2! Thanks to you in advance.
+
+## Get Started with the 9.2 RC
+
+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).
+
+By default, ABP Studio uses stable versions to create solutions. Therefore, if you want to create a solution with a preview version, first you need to create a solution and then switch your solution to the preview version from the ABP Studio UI:
+
+
+
+## Migration Guide
+
+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 v9.x or earlier: [ABP Version 9.2 Migration Guide](https://abp.io/docs/9.2/release-info/migration-guides/abp-9-2)
+
+## What's New with ABP v9.2?
+
+In this section, I will introduce some major features released in this version.
+Here is a brief list of titles explained in the next sections:
+
+* Added `ApplicationName` Property to Isolate Background Jobs & Background Workers
+* Docs Module: Added "Alternative Words" to Filter Items
+* Introducing the Bunny BLOB Storage Provider
+* Upgraded `MongoDB.Driver` to v3.1.0
+* Using Timezone Settings to Display Datetime
+* Identity Pro Module: Require Email Verification to Register
+* Switching users during OAuth login
+
+### Added ApplicationName Property to Isolate Background Jobs & Background Workers
+
+ABP's [Background Jobs Module](https://abp.io/docs/latest/modules/background-jobs) has been enhanced with a new `ApplicationName` property that helps isolate jobs and workers across multiple applications sharing the same database.
+
+Previously, when different applications used the BackgroundJobs module and shared a database, an application might encounter jobs that didn't belong to it. This would lead to failed processing attempts and marking jobs as `IsAbandoned = true` with a "Undefined background job for the job name" error, preventing these jobs from ever being executed.
+
+With the new `ApplicationName` property, applications now properly filter jobs at the repository level, ensuring each application only processes job types it recognizes. This prevents the incorrect abandonment of jobs and ensures consistent behavior in multi-application scenarios.
+
+You can set `ApplicationName` of `AbpBackgroundJobWorkerOptions` to your application name to isolate jobs and workers across multiple applications sharing the same database:
+
+```csharp
+public override void PreConfigureServices(ServiceConfigurationContext context)
+{
+ PreConfigure(options =>
+ {
+ options.ApplicationName = context.Services.GetApplicationName()!;
+ });
+}
+```
+
+> For more information, please refer to the [Background Jobs Module](https://abp.io/docs/latest/modules/background-jobs) documentation and the [PR](https://github.com/abpframework/abp/pull/22169) that added this feature.
+
+### Docs Module: Added "Alternative Words" to Filter Items
+
+[ABP's Docs Module](https://abp.io/docs/9.2/modules/docs) now supports "alternative words" to enhance the search functionality when filtering documentation items. This feature addresses a common user experience issue where users might search using terminology different from what appears in the documentation.
+
+For example, when a user searches for "Error" in the documentation, they may actually be looking for content related to "Exception Handling." With this new feature, documentation items can now be configured with alternative keywords that are considered during filtering.
+
+The implementation allows defining optional "keywords" for items in the navigation tree. For example:
+
+```json
+{
+ "text": "Exception Handling",
+ "path": "framework/fundamentals/exception-handling.md",
+ "keywords": ["Error", "Another Value"]
+}
+```
+
+When users search or filter content, the system now considers both the original text and these alternative keywords, improving discoverability of relevant documentation sections. This enhancement makes the documentation more accessible and user-friendly, especially for newcomers who might not be familiar with the exact terminology used in the ABP documentation.
+
+### Introducing the Bunny BLOB Storage Provider
+
+ABP v9.2 RC introduces a new BLOB storage provider for [Bunny Storage](https://bunny.net/storage/), a global edge storage solution. This addition expands ABP's BLOB Storage options beyond the existing providers like Azure, AWS, and others.
+
+The [Bunny BLOB Storage Provider](https://abp.io/docs/9.2/framework/infrastructure/blob-storing/bunny) allows ABP applications to seamlessly integrate with Bunny's CDN-backed storage service, which offers high-performance content delivery through its global network.
+
+To use this new provider, you'll need to:
+
+* Run `abp add-package Volo.Abp.BlobStoring.Bunny` command.
+* And then configure the provider in your module's `ConfigureServices` method:
+
+```csharp
+Configure(options =>
+{
+ options.Containers.ConfigureDefault(container =>
+ {
+ container.UseBunny(bunny =>
+ {
+ bunny.StorageZoneName = "your-storage-zone";
+ bunny.ApiKey = "your-api-key";
+ bunny.Region = "your-region"; // de, ny, la, sg, or sy
+ });
+ });
+});
+```
+
+This integration provides ABP applications with an efficient and globally distributed storage solution, particularly beneficial for applications requiring fast content delivery across different geographical regions. To use this new provider and make the related configurations, you can refer to the [Bunny Storage Provider](https://abp.io/docs/9.2/framework/infrastructure/blob-storing/bunny) documentation always.
+
+> This new BLOB Storage provider is contributed by [@suhaib-mousa](https://github.com/suhaib-mousa). Thanks to him for his contribution!
+> We are always happy to see the community contributing to the ABP Framework and encouraging them to contribute more.
+
+### Upgraded `MongoDB.Driver` to `v3.1.0`
+
+ABP v9.2 RC includes an upgrade to `MongoDB.Driver` version `3.1.0`. This significant version bump from previous releases brings several improvements and new features that benefit ABP applications using MongoDB as their database.
+
+The upgrade provides:
+
+* Async/Await Support: Write non-blocking, asynchronous code easily.
+* Fluent API: Build queries and updates intuitively with Builders.
+* LINQ Support: Use LINQ for querying MongoDB collections.
+* and more ...
+
+> For more information, please refer to the [MongoDB.Driver release notes](https://github.com/mongodb/mongo-csharp-driver/releases/tag/v3.1.0).
+
+We have prepared a [migration guide](https://abp.io/docs/9.2/release-info/migration-guides/MongoDB-Driver-2-to-3) for this upgrade. Please refer to it to learn more about the changes and how to migrate your application.
+
+### Using Timezone Settings to Display Datetime
+
+A significant enhancement in ABP v9.2 is the ability to use timezone settings to display `DateTime` values according to the user's or application's configured timezone. Introduced in `v9.2.0-rc.2`, this feature addresses the common challenge of ensuring users see accurate time information, regardless of their geographical location.
+
+Previously, `DateTime` values were often shown in the server's timezone or in UTC, which could cause confusion for users in different timezones. With this new feature, ABP applications can now respect the configured timezone and automatically convert and display datetime values accordingly.
+
+**Before setting the timezone:**
+
+Consider a scenario where you have a list of books, and the `CreationTime` property is displayed without any timezone consideration. It may appear in the server's default timezone:
+
+
+*(Screenshot of a Books page showing the CreationTime in a default timezone.)*
+
+**Setting the timezone:**
+
+To set the timezone, start by configuring the `AbpClockOptions` in your module's `ConfigureServices` method:
+
+```csharp
+Configure(options =>
+{
+ options.Kind = DateTimeKind.Utc;
+});
+```
+
+> By setting the `Kind` property of `AbpClockOptions` to `DateTimeKind.Utc`, ABP will normalize all datetime values. Times stored in the database and returned to the frontend will be in UTC. Additionally, the `SupportsMultipleTimezone` property of the `IClock` service will be **true**, and you’ll be able to configure the timezone from the UI under the _Settings_ page.
+
+After setting the `Kind` property, you can run your application and configure the timezone from the UI under the _Settings_ page. For example, set the timezone to "Asia/Tokyo (+09:00)":
+
+
+*(Screenshot showing the configuration setting for Abp.Timing.TimeZone, set to "Asia/Tokyo (+09:00)".)*
+
+> ABP provides the `Abp.Timing.TimeZone` setting, which allows you to configure the desired timezone at the application, tenant, or user level and this setting can be configured in the UI under the _Settings_ page, inside of the _Time Zone_ tab.
+
+**After setting the timezone:**
+
+Once the timezone is configured, ABP automatically handles the conversion and display of `DateTime` values. The `CreationTime` on the _Books_ page will now be shown in the **"Asia/Tokyo"** timezone.
+
+
+*(Screenshot of the _Books_ page after setting the timezone, showing the `CreationTime` adjusted to the "Asia/Tokyo" timezone.)*
+
+This feature utilizes the `IClock` service, which provides methods like `ConvertToUserTime` and `ConvertToUtc` to facilitate timezone conversions. By configuring the `Abp.Timing.TimeZone` setting, developers can ensure a consistent and user-friendly experience across applications with a global user base.
+> For a more detailed guide on implementing and using this feature, refer to the article: [Developing a Multi-Timezone Application Using the ABP Framework](https://abp.io/community/articles/developing-a-multitimezone-application-using-the-abp-framework-zk7fnrdq). It offers step-by-step instructions and examples for handling multi-timezone scenarios in ABP applications.
+
+### Identity Pro Module: Require Email Verification to Register
+
+[ABP Identity Pro module](https://abp.io/docs/9.2/modules/identity-pro) has been enhanced with a new feature that allows administrators to require email verification during the registration process. This security improvement ensures that users must verify their email addresses before their registration is considered complete. Enabling this feature is especially important for applications that want to prevent spam registrations.
+
+Administrators can enable or disable this feature through the **Identity management -> Identity Verification (tab)** settings page (by checking the `Enforce email verification to register` checkbox):
+
+
+
+### Switching users during OAuth login
+
+If you have an OAuth/Auth Server application using the [ABP Account Pro module](https://abp.io/docs/9.2/modules/account-pro) , you can pass the `prompt=select_account` parameter to force the user to select an account.
+
+
+
+For more information, please refer to the [Switching users during OAuth login](https://abp.io/docs/9.2/modules/account-pro#switching-users-during-oauth-login) documentation.
+
+### New ABP Community Articles
+
+There are exciting articles contributed by the ABP community as always. I will highlight some of them here:
+
+* [Implementing CQRS with MediatR in ABP](https://abp.io/community/articles/implementing-cqrs-with-mediatr-in-abp-xiqz2iio) by [Engincan Veske](https://github.com/EngincanV)
+* [Using Vue Components in a Razor Pages ABP Application](https://abp.io/community/articles/using-vue-components-in-a-razor-pages-abp-application-z3jr07tv) by [Enis Necipoglu](https://github.com/enisn)
+* [Using ABP's AWS Blob Storing Provider with DigitalOcean Spaces](https://abp.io/community/articles/using-abps-aws-blob-storing-provider-with-digitalocean-spaces-7hlyb25g) by [Suhaib Mousa](https://abp.io/community/members/suhaib-mousa)
+* [Video Post: Using Vue Components in a Razor Pages ABP Application](https://abp.io/community/articles/using-vue-components-in-a-razor-pages-abp-application-z3jr07tv) by [Enis Necipoglu](https://github.com/enisn)
+* [Understanding the Embedded Files in ABP Framework](https://abp.io/community/articles/understanding-the-embedded-files-in-abp-framework-nsrp8aa9) by [Liming Ma](https://github.com/maliming)
+* [How to Change the CurrentUser in ABP?](https://abp.io/community/articles/how-to-change-the-currentuser-in-abp-i3uu1m7g) by [Engincan Veske](https://github.com/EngincanV)
+
+
+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/create) to the ABP Community.
+
+## Conclusion
+
+This version comes with some new features and a lot of enhancements to the existing features. You can see the [Road Map](https://abp.io/docs/9.2/release-info/road-map) documentation to learn about the release schedule and planned features for the next releases. Please try ABP v9.2 RC and provide feedback to help us release a more stable version.
+
+Thanks for being a part of this community!
diff --git a/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/after.png b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/after.png
new file mode 100644
index 0000000000..f9f72302a5
Binary files /dev/null and b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/after.png differ
diff --git a/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/before.png b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/before.png
new file mode 100644
index 0000000000..14340f91ee
Binary files /dev/null and b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/before.png differ
diff --git a/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/configure-timezone.png b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/configure-timezone.png
new file mode 100644
index 0000000000..d383eee6f6
Binary files /dev/null and b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/configure-timezone.png differ
diff --git a/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/cover-image.png b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/cover-image.png
new file mode 100644
index 0000000000..dd40555c8e
Binary files /dev/null and b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/cover-image.png differ
diff --git a/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/require-email-verification-for-register.png b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/require-email-verification-for-register.png
new file mode 100644
index 0000000000..597eaa6118
Binary files /dev/null and b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/require-email-verification-for-register.png differ
diff --git a/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/select-account.png b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/select-account.png
new file mode 100644
index 0000000000..39d8c1b421
Binary files /dev/null and b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/select-account.png differ
diff --git a/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/studio-switch-to-preview.png b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/studio-switch-to-preview.png
new file mode 100644
index 0000000000..32f6d01edb
Binary files /dev/null and b/docs/en/Blog-Posts/2025-03-27 v9_2_Preview/studio-switch-to-preview.png differ
diff --git a/docs/en/Community-Articles/2020-09-09-Replacing-Email-Template-and-Sending-Emails/POST.md b/docs/en/Community-Articles/2020-09-09-Replacing-Email-Template-and-Sending-Emails/POST.md
index 03caff8007..98685933ba 100644
--- a/docs/en/Community-Articles/2020-09-09-Replacing-Email-Template-and-Sending-Emails/POST.md
+++ b/docs/en/Community-Articles/2020-09-09-Replacing-Email-Template-and-Sending-Emails/POST.md
@@ -470,6 +470,12 @@ namespace TemplateReplace
* I highly recommend you to [check it out](https://commercial.abp.io/modules/Volo.TextTemplateManagement).
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## References
* [Text Templating](https://docs.abp.io/en/abp/latest/Text-Templating)
diff --git a/docs/en/Community-Articles/2020-12-10-How-to-Integrate-the-Telerik-Blazor-Component/POST.md b/docs/en/Community-Articles/2020-12-10-How-to-Integrate-the-Telerik-Blazor-Component/POST.md
index 6445d83e85..f946db23b9 100644
--- a/docs/en/Community-Articles/2020-12-10-How-to-Integrate-the-Telerik-Blazor-Component/POST.md
+++ b/docs/en/Community-Articles/2020-12-10-How-to-Integrate-the-Telerik-Blazor-Component/POST.md
@@ -4,6 +4,12 @@
Hi, in this step by step article, we will see how we can integrate the Telerik Blazor Components to our Blazor UI.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Creating the Solution
> ABP Framework offers startup templates to get into business faster.
@@ -532,6 +538,12 @@ namespace TelerikComponents.Blazor.Pages

+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Conclusion
In this article, I've tried to explain how we can integrate [Telerik Blazor Component](https://www.telerik.com/blazor-ui) to our Blazor UI. ABP Framework designed as modular, so that it can work with any UI library/framework.
\ No newline at end of file
diff --git a/docs/en/Community-Articles/2021-06-17-Using-Elsa-Workflow-with-ABP-Framework/POST.md b/docs/en/Community-Articles/2021-06-17-Using-Elsa-Workflow-with-ABP-Framework/POST.md
index 0ec86d40c4..2f5cde6d2d 100644
--- a/docs/en/Community-Articles/2021-06-17-Using-Elsa-Workflow-with-ABP-Framework/POST.md
+++ b/docs/en/Community-Articles/2021-06-17-Using-Elsa-Workflow-with-ABP-Framework/POST.md
@@ -1,11 +1,19 @@
# Using Elsa Workflow with ABP Framework
+> **UPDATE:** There is a newer version of this article, which covers Elsa 3 with ABP Framework. You can check it at https://abp.io/community/articles/using-elsa-3-workflow-with-abp-framework-usqk8afg
+
**Elsa Core** is an open-source workflows library that can be used in any kind of .NET Core application. Using such a workflow library can be useful to implement business rules visually or programmatically.

This article shows how we can use this workflow library within our ABP-based application. We will start with a couple of examples and then we will integrate the **Elsa Dashboard** (you can see it in the above gif) into our application to be able to design our workflows visually.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Source Code
You can find the source of the example solution used in this article [here](https://github.com/abpframework/abp-samples/tree/master/ElsaDemo).
@@ -453,4 +461,10 @@ namespace ElsaDemo.Web.Menus
* Now we can click the "Workflow" menu item, display the Elsa Dashboard and designing workflows.
-
\ No newline at end of file
+
+
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
diff --git a/docs/en/Community-Articles/2021-10-31-Many-to-Many-Relationship-with-ABP-and-EF-Core/POST.md b/docs/en/Community-Articles/2021-10-31-Many-to-Many-Relationship-with-ABP-and-EF-Core/POST.md
index ae688d6981..81c0b6ea34 100644
--- a/docs/en/Community-Articles/2021-10-31-Many-to-Many-Relationship-with-ABP-and-EF-Core/POST.md
+++ b/docs/en/Community-Articles/2021-10-31-Many-to-Many-Relationship-with-ABP-and-EF-Core/POST.md
@@ -10,6 +10,12 @@ You can see the ER Diagram of our application below. This diagram will be helpfu
When we've examined the ER Diagram, we can see the one-to-many relationship between the **Author** and the **Book** tables. Also, the many-to-many relationship (**BookCategory** table) between the **Book** and the **Category** tables. (There can be more than one category on each book and vice-versa in our scenario).
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
### Source Code
You can find the source code of the application at https://github.com/EngincanV/ABP-Many-to-Many-Relationship-Demo .
@@ -1664,6 +1670,12 @@ namespace BookStore.Web.Pages.Books

+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
### Conclusion
In this article, I've tried to explain how to create a many-to-many relationship by using the ABP framework. (by following DDD principles)
diff --git a/docs/en/Community-Articles/2021-12-13-Integrating-the-Syncfusion-MVC-Components-to-the-ABP-MVC-UI/POST.md b/docs/en/Community-Articles/2021-12-13-Integrating-the-Syncfusion-MVC-Components-to-the-ABP-MVC-UI/POST.md
index 4095c62092..2953c552d3 100644
--- a/docs/en/Community-Articles/2021-12-13-Integrating-the-Syncfusion-MVC-Components-to-the-ABP-MVC-UI/POST.md
+++ b/docs/en/Community-Articles/2021-12-13-Integrating-the-Syncfusion-MVC-Components-to-the-ABP-MVC-UI/POST.md
@@ -4,6 +4,12 @@
In this article we will see how we can integrate the Syncfusion MVC Components into our ABP application.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Source Code
You can find the source code of the application at https://github.com/EngincanV/ABP-Syncfusion-Components-Demo.
@@ -314,6 +320,12 @@ After injecting the Syncfusion style and script into our application, our config

+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
### Conclusion
In this article, we've explained how to integrate the **Syncfusion Components** into our applications. After following this article, you can use the Syncfusion components in your application.
diff --git a/docs/en/Community-Articles/2022-02-06-How-to-Hide-ABP-Related-Endpoints-on-Swagger-UI/POST.md b/docs/en/Community-Articles/2022-02-06-How-to-Hide-ABP-Related-Endpoints-on-Swagger-UI/POST.md
index a6ad7596d4..d4102d8256 100644
--- a/docs/en/Community-Articles/2022-02-06-How-to-Hide-ABP-Related-Endpoints-on-Swagger-UI/POST.md
+++ b/docs/en/Community-Articles/2022-02-06-How-to-Hide-ABP-Related-Endpoints-on-Swagger-UI/POST.md
@@ -8,6 +8,12 @@ In the issue, there are helpful comments about hiding the ABP related endpoints
I thought it would be better to enable/disable showing endpoints on runtime by simply selecting a checkbox on the Setting Management page and in this article I wanted to show you how, so let's dive in.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Source Code
You can find the source code of the application at https://github.com/EngincanV/ABP-Hide-Swagger-Endpoint-Demo.
@@ -445,6 +451,10 @@ services.AddAbpSwaggerGen(
> For more info, please see the [Swagger Integration](https://docs.abp.io/en/abp/latest/API/Swagger-Integration#hide-abp-endpoints-on-swagger-ui) docs.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
---
Thanks for reading.
\ No newline at end of file
diff --git a/docs/en/Community-Articles/2022-03-09-abpio-platform-52-rc-has-been-published/post.md b/docs/en/Community-Articles/2022-03-09-abpio-platform-52-rc-has-been-published/post.md
index be89ccb092..e43ba9a529 100644
--- a/docs/en/Community-Articles/2022-03-09-abpio-platform-52-rc-has-been-published/post.md
+++ b/docs/en/Community-Articles/2022-03-09-abpio-platform-52-rc-has-been-published/post.md
@@ -186,7 +186,7 @@ We organized the 3rd live [ABP Community Talks](https://community.abp.io/events)
* [Enis Necipoğlu](https://twitter.com/EnisNecipoglu) has also created [an article](https://community.abp.io/posts/using-autofilterer-with-abp-framework-uuqv81jm) to demonstrate how to use his own open source [AutoFilterer](https://github.com/enisn/AutoFilterer) library with the ABP Framework.
* [Jonathan Potts](https://github.com/jonathanpotts) has created his first ABP Community article that shows how to use Bootswatch themes with the ABP Framework. [See it here](https://community.abp.io/posts/customizing-the-abp-basic-theme-with-bootswatch-4luoqzr0).
-Thanks to all the contributors. It is appreciated if you want to submit your post and share your knowledge with the ABP community: https://community.abp.io/posts/submit
+Thanks to all the contributors. It is appreciated if you want to submit your post and share your knowledge with the ABP community: https://community.abp.io/posts/create
## Conclusion
diff --git a/docs/en/Community-Articles/2022-04-06-Concurrency-Check-in-ABP-Based-Applications/POST.md b/docs/en/Community-Articles/2022-04-06-Concurrency-Check-in-ABP-Based-Applications/POST.md
index b16d6bd9f1..cf8c3c122a 100644
--- a/docs/en/Community-Articles/2022-04-06-Concurrency-Check-in-ABP-Based-Applications/POST.md
+++ b/docs/en/Community-Articles/2022-04-06-Concurrency-Check-in-ABP-Based-Applications/POST.md
@@ -2,6 +2,12 @@
In this article, we'll create a basic application to demonstrate how "Concurrency Check/Control" can be implemented in an ABP project.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Creating the Solution
For this article, we will create a simple BookStore application and add CRUD functionality to the pages. Hence we deal with the concurrency situation.
@@ -546,3 +552,9 @@ Then we can run the application, navigate to the **/Books** endpoint and see the
* After the first user updated the record, the second user tries to update the same record without getting the last state of the record. And therefore `AbpDbConcurrencyException` is thrown because **ConcurrencyStamp** values are different from each other.
* The second user should close and re-open the model to get the last state of the record and then they can make changes to the current record.
+
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
\ No newline at end of file
diff --git a/docs/en/Community-Articles/2022-11-22-gRPC-JSON-Transcoding/POST.md b/docs/en/Community-Articles/2022-11-22-gRPC-JSON-Transcoding/POST.md
index 9663ac1f83..e0f1f38aa9 100644
--- a/docs/en/Community-Articles/2022-11-22-gRPC-JSON-Transcoding/POST.md
+++ b/docs/en/Community-Articles/2022-11-22-gRPC-JSON-Transcoding/POST.md
@@ -4,6 +4,12 @@ In this article, I'll show you one of the new features that came with .NET 7: **
> I've created a community article to highlight some interesting features (briefly) that are now available with the release of .NET 7. You can check it from [here](https://community.abp.io/posts/whats-new-with-.net-7-tlq2g43w).
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## What is gRPC? What are the pros and cons?
[gRPC](https://grpc.io/) is a high-performance RPC framework and uses HTTP/2 and Protobuf (protocol buffers).
@@ -189,6 +195,12 @@ In this article, I've briefly introduced the JSON Transcoding feature that was s
> See the [gRPC JSON transcoding in ASP.NET Core gRPC apps](https://learn.microsoft.com/en-us/aspnet/core/grpc/json-transcoding?view=aspnetcore-7.0) documentation for more information.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## References
* https://devblogs.microsoft.com/dotnet/announcing-grpc-json-transcoding-for-dotnet/
diff --git a/docs/en/Community-Articles/2022-18-11-whats-new-with-net7/POST.md b/docs/en/Community-Articles/2022-18-11-whats-new-with-net7/POST.md
index fb964fa730..671592d63e 100644
--- a/docs/en/Community-Articles/2022-18-11-whats-new-with-net7/POST.md
+++ b/docs/en/Community-Articles/2022-18-11-whats-new-with-net7/POST.md
@@ -6,6 +6,12 @@ In this article, I will highlight a few interesting features that are now availa
There are many new features coming with this release. We are going to examine new features within 4 sub-sections: ASP.NET Core, C#11, EF 7 and MAUI. Let's start with ASP.NET Core.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## ASP.NET Core
We will see the following features in this section:
@@ -290,6 +296,12 @@ In this article, I've highlighted some features that were shipped with .NET 7.
Thanks for reading the article and I hope you find it helpful. See you in the next one!
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## References
* https://devblogs.microsoft.com/dotnet/announcing-dotnet-7/
diff --git a/docs/en/Community-Articles/2023-07-03-Image-Compression-And-Resize/POST.md b/docs/en/Community-Articles/2023-07-03-Image-Compression-And-Resize/POST.md
index 9023417b20..1a1aa55f97 100644
--- a/docs/en/Community-Articles/2023-07-03-Image-Compression-And-Resize/POST.md
+++ b/docs/en/Community-Articles/2023-07-03-Image-Compression-And-Resize/POST.md
@@ -8,6 +8,12 @@ ABP Framework provides services to compress and resize images and implements the
> Refer to the documentation for more info: [Image Manipulation](https://docs.abp.io/en/abp/7.3/Image-Manipulation)
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
### Source Code
You can find the source code of the application at [https://github.com/abpframework/abp-samples/tree/master/ImageManipulation](https://github.com/abpframework/abp-samples/tree/master/ImageManipulation). Don't hesitate to check the source code, if you are stuck on any point.
@@ -281,6 +287,12 @@ The results are impressive for the example above:
* The original image was 12 KB and now the compressed & resized image has been reduced to 8 KB.
* The original image was 225x225 and now resized as 200x200.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Conclusion
In this article, I have shown you how to compress and/or resize images with ABP Framework's Image Manipulation System by just defining some attributes to the top of the controller actions.
diff --git a/docs/en/Community-Articles/2023-08-12-ABP-Commercial-GDPR-Module-Overview/POST.md b/docs/en/Community-Articles/2023-08-12-ABP-Commercial-GDPR-Module-Overview/POST.md
index 38cfab67e9..3d62f38886 100644
--- a/docs/en/Community-Articles/2023-08-12-ABP-Commercial-GDPR-Module-Overview/POST.md
+++ b/docs/en/Community-Articles/2023-08-12-ABP-Commercial-GDPR-Module-Overview/POST.md
@@ -2,6 +2,12 @@
In this article, I will highlight ABP Commercial's [GDPR Module](https://commercial.abp.io/modules/Volo.Gdpr) and show you how to provide personal data to the GDPR Module that is collected by your application and make it aligned with personal data download results.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## GDPR Module
[GDPR Module](https://docs.abp.io/en/commercial/latest/modules/gdpr) allows users to download and delete the data collected by the application. It's used for Personal Data Management obligating by the GDPR regulation and provides the following features:
@@ -143,6 +149,12 @@ Then, when we clicked the download button, a zip file will be downloaded. We can
{"City":"Istanbul","Postcode":"..."}
```
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Conclusion
In this article, I've explained the GDPR Module's data collection system and given you a brief overview of the module. GDPR Module allows you to request to download personal data, delete/anonymize your own personal data and delete your account permanently. It's pre-installed in the Application and Application(single layer) Pro Startup templates, but you can easily configure it to your existing application.
diff --git a/docs/en/Community-Articles/2023-10-30-NET-8-ASP-NET-Core-Metrics/POST.md b/docs/en/Community-Articles/2023-10-30-NET-8-ASP-NET-Core-Metrics/POST.md
index 9142a36be9..9586b51cf5 100644
--- a/docs/en/Community-Articles/2023-10-30-NET-8-ASP-NET-Core-Metrics/POST.md
+++ b/docs/en/Community-Articles/2023-10-30-NET-8-ASP-NET-Core-Metrics/POST.md
@@ -2,6 +2,12 @@
In this article, I'll show you the new built-in metrics of .NET 8, which are basically numerical measurements reported over time in your application and can be used to monitor the health of your application and generate reports according to those numerical values. We will see what the metrics are, why to use them, and how to use them in detail. So, let's dive in.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## What are Metrics?
Metrics are numerical measurements reported over time. These measurements are crucial for monitoring the health of an application and generating alerts when necessary. In the context of a web service, various metrics can be tracked, such as:
@@ -185,6 +191,12 @@ If you want to build a dashboard, you can see [this documentation from Microsoft
Also, you can check [this video](https://www.youtube.com/watch?v=A2pKhNQoQUU) if you want to see it in action.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Conclusion
In this article, I've shown you the new built-in metrics of .NET8, described what metrics are, which pre-built metrics are provided by ASP.NET Core, how to use & view these pre-built metrics and finally, I have shown you how to create custom metrics and view them in a dashboard. If you want to learn more about the **Metrics System of .NET**, you can check out the references that I have shared below.
diff --git a/docs/en/Community-Articles/2023-11-16-Upgrading-Your-Existing-Projects-to-NET8/POST.md b/docs/en/Community-Articles/2023-11-16-Upgrading-Your-Existing-Projects-to-NET8/POST.md
index 3ed34680c1..c247f1e3e9 100644
--- a/docs/en/Community-Articles/2023-11-16-Upgrading-Your-Existing-Projects-to-NET8/POST.md
+++ b/docs/en/Community-Articles/2023-11-16-Upgrading-Your-Existing-Projects-to-NET8/POST.md
@@ -4,6 +4,12 @@ A new .NET version was released on November 14, 2023 and ABP 8.0 RC.1 shipped ba
Despite all the related dependency upgrades and changes made on the ABP Framework and ABP Commercial sides, we still need to make some changes. Let's see the required actions that need to be taken in the following sections.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Installing the .NET 8.0 SDK
To get started with ASP.NET Core in .NET 8.0, you need to install the .NET 8 SDK. You can install it at [https://dotnet.microsoft.com/en-us/download/dotnet/8.0](https://dotnet.microsoft.com/en-us/download/dotnet/8.0).
@@ -99,3 +105,9 @@ After that, you need to check the migration guide documents, listed below:
That's it! These were all the related steps that need to be taken to upgrade your application to .NET 8 and ABP 8.0. Now, you can enjoy the .NET 8 & ABP 8.0 and benefit from the performance improvements and new features.
Happy Coding 🤗
+
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
\ No newline at end of file
diff --git a/docs/en/Community-Articles/2024-01-18-ABP-Now-Supports-Keyed-Services/POST.md b/docs/en/Community-Articles/2024-01-18-ABP-Now-Supports-Keyed-Services/POST.md
index d100838b1a..aeefc03c0c 100644
--- a/docs/en/Community-Articles/2024-01-18-ABP-Now-Supports-Keyed-Services/POST.md
+++ b/docs/en/Community-Articles/2024-01-18-ABP-Now-Supports-Keyed-Services/POST.md
@@ -2,6 +2,12 @@
In this post, I describe the new **"keyed service"** support for the dependency injection container, which came with .NET 8.0. Then, I'll show you an example of usage within the ABP Framework.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## What Are Keyed Services?
ASP.NET Core ships with a built-in dependency injection container, which is a pretty basic DI container that supports minimal features a dependency injection container is supposed to have. For that reason, most of the .NET users use third-party containers like [Autofac](https://autofac.org/), or Ninject.
@@ -203,6 +209,12 @@ public class TaxCalculator: ICalculator, ITaxCalculator, ICanCalculate, ITransie
Please refer to the [Dependency Injection document](https://abp.io/docs/latest/framework/fundamentals/dependency-injection#exposekeyedservice-attribute) for further info.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Summary
In this post, I described the new **"keyed service"** support added to the built-in dependency injection container that was released in .NET 8, and wanted to announce it's being supported from v8.0.2 for ABP Framework. It's a really good addition to the built-in dependency injection container and can be pretty useful for certain use-cases.
diff --git a/docs/en/Community-Articles/2024-02-19-abpio-platform-81-rc-has-been-published/post.md b/docs/en/Community-Articles/2024-02-19-abpio-platform-81-rc-has-been-published/post.md
index 569e566af7..fbe2c2cbad 100644
--- a/docs/en/Community-Articles/2024-02-19-abpio-platform-81-rc-has-been-published/post.md
+++ b/docs/en/Community-Articles/2024-02-19-abpio-platform-81-rc-has-been-published/post.md
@@ -228,7 +228,7 @@ There are exciting articles contributed by the ABP community as always. I will h
* [A Best Practice for Designing Interfaces in .NET C#](https://community.abp.io/posts/a-best-practice-for-designing-interfaces-in-.net-c-9xqc4h8d)
* [Invariance, Covariance, and Contravariance in .NET C#](https://community.abp.io/posts/invariance-covariance-and-contravariance-in-.net-c-9blmuhme)
-Thanks to the ABP Community for all the content they have published. You can also [post your ABP-related (text or video) content](https://community.abp.io/posts/submit) to the ABP Community.
+Thanks to the ABP Community for all the content they have published. You can also [post your ABP-related (text or video) content](https://community.abp.io/posts/create) to the ABP Community.
## Conclusion
diff --git a/docs/en/Community-Articles/2024-04-19-Performing-Case-Insensitive-Search-In-Postgresql/POST.md b/docs/en/Community-Articles/2024-04-19-Performing-Case-Insensitive-Search-In-Postgresql/POST.md
index e4f763f279..5eb4a5bd98 100644
--- a/docs/en/Community-Articles/2024-04-19-Performing-Case-Insensitive-Search-In-Postgresql/POST.md
+++ b/docs/en/Community-Articles/2024-04-19-Performing-Case-Insensitive-Search-In-Postgresql/POST.md
@@ -120,6 +120,12 @@ After these configurations, you should create a migration and apply it to your d
However, this solution comes with some problems, for example, by using non-deterministic collations, it's not yet possible to use pattern-matching operators such as `LIKE` on columns. This is a huge problem because it makes it hard to use LINQ. For example, you can't use the `.EndsWith` or `.StartsWith` methods, because they are [translated to `LIKE` command on the SQL level](https://www.npgsql.org/efcore/mapping/translations.html).
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Conclusion
In PostgreSQL, you can perform case-insensitive searches by using the `citext` data type or by utilizing collation settings. Nevertheless, if you have an ABP-based PostgreSQL application or a plain .NET application with PostgreSQL as the database option, to make a decision to pick one of these options, you can follow the following points:
diff --git a/docs/en/Community-Articles/2024-05-10-Sentiment-Analysis-within-ABP-Based-Application/POST.md b/docs/en/Community-Articles/2024-05-10-Sentiment-Analysis-within-ABP-Based-Application/POST.md
index e6417b70ce..deb162bcba 100644
--- a/docs/en/Community-Articles/2024-05-10-Sentiment-Analysis-within-ABP-Based-Application/POST.md
+++ b/docs/en/Community-Articles/2024-05-10-Sentiment-Analysis-within-ABP-Based-Application/POST.md
@@ -4,6 +4,12 @@ In this article, first I will briefly explain what sentiment analysis is and the
We will use ML.NET Framework, which is an open-source machine learning framework created by the dotnet team and also we will create a layered ABP Solution by using the application template and finally we will use CMS Kit's Comment Feature and extend its behavior by adding spam detection while creating or updating a comment, at that point we will make sentiment analysis.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Sentiment Analysis
[Sentiment Analysis (or opinion mining)](https://en.wikipedia.org/wiki/Sentiment_analysis) refers to determining the emotion from the given input. The primary goal of sentiment analysis is to identify, extract, and categorize (positive, negative, or neutral) the sentiments expressed in textual data.
@@ -333,6 +339,12 @@ Then, finally, we can run the application to see the final results:
Once the model is trained and evaluated, we can save the trained model and use it directly for further use. In this way, you don’t have to retrain the model every time when you want to make predictions. It’s essential to save the trained model for future use and a must for the production-ready code. I created a separate article dedicated to that topic, and if you are interested, you can read it from [here](https://engincanv.github.io/machine-learning/sentiment-analysis/best-practises/2024/05/16/reusing-and-optimizing-machine-learning-models-in-dotnet.html).
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Conclusion
In this article, I briefly explain what sentiment analysis is, created a sample ABP-based application, integrated the CMS Kit Module and finally, applied sentiment analysis to make spam checks whenever a new comment has been submitted or updated. You can get the source code of the demo from [https://github.com/EngincanV/SentimentAnalysisDemo](https://github.com/EngincanV/SentimentAnalysisDemo)
diff --git a/docs/en/Community-Articles/2024-05-24-abpio-platform-82-rc-has-been-published/post.md b/docs/en/Community-Articles/2024-05-24-abpio-platform-82-rc-has-been-published/post.md
index 6574012f24..c3eadb6419 100644
--- a/docs/en/Community-Articles/2024-05-24-abpio-platform-82-rc-has-been-published/post.md
+++ b/docs/en/Community-Articles/2024-05-24-abpio-platform-82-rc-has-been-published/post.md
@@ -259,7 +259,7 @@ There are exciting articles contributed by the ABP community as always. I will h
* [Using FluentValidation with ABP Framework](https://community.abp.io/posts/using-fluentvalidation-with-abp-framework-2cxuwl70) by [Enes Döner](https://community.abp.io/members/Enes)
* [Using Blob Storage with ABP](https://community.abp.io/posts/using-blob-storage-with-abp-framework-jygtmhn4) by [Emre Kendirli](https://community.abp.io/members/emrekenderli)
-Thanks to the ABP Community for all the content they have published. You can also [post your ABP-related (text or video) content](https://community.abp.io/posts/submit) to the ABP Community.
+Thanks to the ABP Community for all the content they have published. You can also [post your ABP-related (text or video) content](https://community.abp.io/posts/create) to the ABP Community.
## Conclusion
diff --git a/docs/en/Community-Articles/2024-08-01-abpio-platform-83-rc-has-been-published/post.md b/docs/en/Community-Articles/2024-08-01-abpio-platform-83-rc-has-been-published/post.md
index 05c18fde03..4ddd247a63 100644
--- a/docs/en/Community-Articles/2024-08-01-abpio-platform-83-rc-has-been-published/post.md
+++ b/docs/en/Community-Articles/2024-08-01-abpio-platform-83-rc-has-been-published/post.md
@@ -165,7 +165,7 @@ There are exciting articles contributed by the ABP community as always. I will h
* [How to use .NET Aspire with ABP framework](https://abp.io/community/articles/how-to-use-.net-aspire-with-abp-framework-h29km4kk) by [Berkan Şaşmaz](https://twitter.com/berkansasmazz)
* [Exciting New Feature in ABP.IO CMS Kit: Marked Item System](https://abp.io/community/articles/exciting-new-feature-in-abp.io-cms-kit-marked-item-system.-2hvpq0me) by [Suhaib Mousa](https://abp.io/community/members/suhaibmousa032@gmail.com)
-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.
+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/create) to the ABP Community.
## Conclusion
diff --git a/docs/en/Community-Articles/2024-09-12-abpio-platform-83-final-has-been-released/post.md b/docs/en/Community-Articles/2024-09-12-abpio-platform-83-final-has-been-released/post.md
index 55f7b80294..de27f8cdbd 100644
--- a/docs/en/Community-Articles/2024-09-12-abpio-platform-83-final-has-been-released/post.md
+++ b/docs/en/Community-Articles/2024-09-12-abpio-platform-83-final-has-been-released/post.md
@@ -70,7 +70,7 @@ As always, exciting articles have been contributed by the ABP community. I will
* [Introducing the Google Cloud Storage BLOB Provider](https://abp.io/community/articles/introducing-the-google-cloud-storage-blob-provider-yrt6azc0) by [Engincan Veske](https://twitter.com/EngincanVeske)
* [Switching Between Organization Units](https://abp.io/community/articles/switching-between-organization-units-i5tokpzt) by [Liming Ma](https://github.com/maliming)
-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.
+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/create) to the ABP Community.
## About the Next Version
diff --git a/docs/en/Community-Articles/2024-10-23-abpio-platform-90-rc-has-been-published/post.md b/docs/en/Community-Articles/2024-10-23-abpio-platform-90-rc-has-been-published/post.md
index 9e22660332..a7dfa896bd 100644
--- a/docs/en/Community-Articles/2024-10-23-abpio-platform-90-rc-has-been-published/post.md
+++ b/docs/en/Community-Articles/2024-10-23-abpio-platform-90-rc-has-been-published/post.md
@@ -210,7 +210,7 @@ There are exciting articles contributed by the ABP community as always. I will h
* [ABP-Powered Web App with Inertia.js, React, and Vite](https://abp.io/community/articles/abppowered-web-app-with-inertia.js-react-and-vite-j7cccvad) by [Anto Subash](https://antosubash.com/)
* [Multi-Tenancy Support in Angular Apps with ABP.IO](https://abp.io/community/articles/multitenancy-support-in-angular-apps-with-abp.io-lw9l36c5) by [HeadChannel Team](https://headchannel.co.uk/)
-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.
+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/create) to the ABP Community.
## Conclusion
diff --git a/docs/en/Community-Articles/2024-11-01-Hybrid-Cache-Net-9/POST.md b/docs/en/Community-Articles/2024-11-01-Hybrid-Cache-Net-9/POST.md
index 3cd4bdc706..a9b869a822 100644
--- a/docs/en/Community-Articles/2024-11-01-Hybrid-Cache-Net-9/POST.md
+++ b/docs/en/Community-Articles/2024-11-01-Hybrid-Cache-Net-9/POST.md
@@ -6,6 +6,12 @@ It offers a flexible caching solution that combines the best aspects of local an
In this article, we’ll explore **HybridCache** in .NET 9 and how it integrates with ABP Framework using `AbpHybridCache`. This new feature offers a robust solution for applications that need to scale while maintaining efficient caching strategies.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## What is HybridCache?
**HybridCache** is designed to merge different caching layers, commonly including an in-memory cache (for high-speed access) and a distributed cache (for scalability across multiple instances). This hybrid approach allows for:
@@ -109,6 +115,12 @@ As you can see from the figure, it only set the cache item to the **LocalCache**
**Note:** If you configure distributed caching options, `HybridCache` service uses the distributed cache and sets the **BackendCache**.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Conclusion
The **HybridCache** library in .NET 9 provides a powerful tool for applications needing both high-speed caching and consistency in distributed environments.
diff --git a/docs/en/Community-Articles/2024-11-21-abpio-platform-90-has-been-released-based-on-net-90/post.md b/docs/en/Community-Articles/2024-11-21-abpio-platform-90-has-been-released-based-on-net-90/post.md
index 30ff978de6..931379eac0 100644
--- a/docs/en/Community-Articles/2024-11-21-abpio-platform-90-has-been-released-based-on-net-90/post.md
+++ b/docs/en/Community-Articles/2024-11-21-abpio-platform-90-has-been-released-based-on-net-90/post.md
@@ -80,7 +80,7 @@ In addition to [the articles to highlight .NET 9.0 features written by our team]
* [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.
+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/create) to the ABP Community.
## Conclusion
diff --git a/docs/en/Community-Articles/2025-01-21-abp-platform-91-rc-has-been-released/post.md b/docs/en/Community-Articles/2025-01-21-abp-platform-91-rc-has-been-released/post.md
index 2c3a98ac49..615457a7ff 100644
--- a/docs/en/Community-Articles/2025-01-21-abp-platform-91-rc-has-been-released/post.md
+++ b/docs/en/Community-Articles/2025-01-21-abp-platform-91-rc-has-been-released/post.md
@@ -136,7 +136,7 @@ There are exciting articles contributed by the ABP community as always. I will h
* [The new Unit Test structure in ABP application](https://abp.io/community/articles/the-new-unit-test-structure-in-abp-application-4vvvp2oy) by [Liming Ma](https://github.com/maliming)
* [How to Use OpenAI API with ABP Framework](https://abp.io/community/articles/how-to-use-openai-api-with-abp-framework-rsfvihla) by [Berkan Şaşmaz](https://github.com/berkansasmaz)
-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.
+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/create) to the ABP Community.
## Conclusion
diff --git a/docs/en/Community-Articles/2025-02-04-OpenIddict-Custom-Logic/POST.md b/docs/en/Community-Articles/2025-02-04-OpenIddict-Custom-Logic/POST.md
index f5602650ae..e8b8c9a1b3 100644
--- a/docs/en/Community-Articles/2025-02-04-OpenIddict-Custom-Logic/POST.md
+++ b/docs/en/Community-Articles/2025-02-04-OpenIddict-Custom-Logic/POST.md
@@ -6,6 +6,12 @@ OpenIddict provides an event-driven model ([event models](https://documentation.
In this article, we will explore OpenIddict event models, their key use cases, and how to implement them effectively.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Understanding OpenIddict Event Model
OpenIddict events are primarily used within the OpenIddict server component. These events provide hooks into the OpenID Connect flow, allowing developers to modify behavior at different stages of authentication & authorization processes.
@@ -101,6 +107,12 @@ That's it! After these steps, your `SignOutEventHandler.HandleAsync()` method sh
Each event provides access to the relevant context, allowing you to access and modify the authentication flow's behavior.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Conclusion
ABP Framework integrates OpenIddict as its authentication and authorization module. OpenIddict provides an event-driven model that allows developers to customize authentication and authorization processes within their ABP applications. It's pre-installed & pre-configured in the ABP's startup templates.
diff --git a/docs/en/Community-Articles/2025-03-11-Developing-A-Multi-Timezone-Application-Using-The-ABP-Framework/post.md b/docs/en/Community-Articles/2025-03-11-Developing-A-Multi-Timezone-Application-Using-The-ABP-Framework/post.md
index bfde4d5318..09d7f917bb 100644
--- a/docs/en/Community-Articles/2025-03-11-Developing-A-Multi-Timezone-Application-Using-The-ABP-Framework/post.md
+++ b/docs/en/Community-Articles/2025-03-11-Developing-A-Multi-Timezone-Application-Using-The-ABP-Framework/post.md
@@ -6,6 +6,8 @@ All these scenarios require us to handle timezone conversions correctly in our a
In this article, we'll show you step by step how to handle multi-timezone in the ABP framework.
+> The content mentioned in this article will be available after the ABP 9.2 version
+
## Timezone Settings
The ABP framework provides a setting called `Abp.Timing.TimeZone` for setting and getting the timezone of users, tenants, or applications. The default value is empty, which means the application will use the server's time zone. Check out the [Timing documentation](https://abp.io/docs/latest/framework/infrastructure/timing) for more information.
diff --git a/docs/en/Community-Articles/2025-03-18-Using-Vue-Components/post.md b/docs/en/Community-Articles/2025-03-18-Using-Vue-Components/post.md
index bc0b390b03..09998169e6 100644
--- a/docs/en/Community-Articles/2025-03-18-Using-Vue-Components/post.md
+++ b/docs/en/Community-Articles/2025-03-18-Using-Vue-Components/post.md
@@ -75,7 +75,7 @@ Let's create a simple Vue component to display the TODO list.
```html
-
+
```
diff --git a/docs/en/Community-Articles/2025-03-24-How-to-Change-the-CurrentUser-in-ABP/POST.md b/docs/en/Community-Articles/2025-03-24-How-to-Change-the-CurrentUser-in-ABP/POST.md
index b74aecdbe4..4213ad0e48 100644
--- a/docs/en/Community-Articles/2025-03-24-How-to-Change-the-CurrentUser-in-ABP/POST.md
+++ b/docs/en/Community-Articles/2025-03-24-How-to-Change-the-CurrentUser-in-ABP/POST.md
@@ -4,6 +4,12 @@
In this article, we'll explore the [`CurrentUser` service](https://abp.io/docs/latest/framework/infrastructure/current-user), its use cases, and how to change it when necessary.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Understanding the ICurrentUser Service
The `ICurrentUser` interface is the primary service in ABP Framework for obtaining information about the logged-in user. It provides some key properties, such as `Id`, `UserName`, `TenantId`, `Roles` (roleNames), and more...
@@ -108,6 +114,12 @@ Here's what the code does step by step:
This pattern is particularly useful in background jobs, event handlers, or any scenario where you need to perform operations under a specific user's context, regardless of the actual authenticated user.
+---
+> 🛠 Liked this post? I now share all my content on Substack — real-world .NET, AI, and scalable software design.
+> 👉 Subscribe here → engincanveske.substack.com
+> 🎥 Also, check out my YouTube channel for hands-on demos and deep dives: https://www.youtube.com/@engincanv
+---
+
## Conclusion
The `CurrentUser` service in ABP Framework provides a simple way to access information about the authenticated user. While it works automatically in most scenarios, there are cases where you need to explicitly change the current user identity, particularly in background processing scenarios.
diff --git a/docs/en/Community-Articles/2025-03-27-Announcing-the-ABP-Contributor-Program/post.md b/docs/en/Community-Articles/2025-03-27-Announcing-the-ABP-Contributor-Program/post.md
new file mode 100644
index 0000000000..7767a6673b
--- /dev/null
+++ b/docs/en/Community-Articles/2025-03-27-Announcing-the-ABP-Contributor-Program/post.md
@@ -0,0 +1,38 @@
+We are excited to introduce the ABP Contributor Program, which is an amazing opportunity for community enthusiasts to contribute to the ABP ecosystem while getting valuable benefits\!
+
+**Create, Contribute and Be Part of Something Bigger**
+
+This is your opportunity to not only share your knowledge but also to grow alongside a passionate community of developers and creators. When you create and contribute to the ABP ecosystem, you’re not just building content, you’re also helping shape the future of ABP.
+
+## **Why Become an ABP Contributor?**
+
+By sharing your knowledge, creating valuable content, and engaging with the ABP Community, you’ll enjoy a range of benefits, including:
+
+* Free ABP Personal License – If you meet all the requirements, you’ll receive an ABP Personal License at no cost
+
+* Community Badge & Title – Stand out in the ABP Community with a special contributor badge and label next to your name.
+
+* Exclusive Discord Role – Get a unique Contributor role on the ABP Community Discord server.
+
+* Increased Visibility – Your contributions will be recognized by hundreds of thousands of developers worldwide.
+
+ Completely Free – No fees, just log in, contribute, and start collecting benefits\!
+
+## **How to Apply?**
+
+If you’re passionate about ABP and think you meet the required criteria, follow these simple steps:
+
+Create content related to ABP: blog posts, tutorials, videos, or documentation. You can publish it on the ABP Community website or any external platform (just make sure to submit it to the ABP Community site).
+
+Apply by email: Send your application to marketing@volosoft.com, and the ABP Team will review your eligibility.
+
+Periodic Assessments: The team reviews submissions every three months to determine and renew contributor status. Keep contributing to continue enjoying the perks\!
+
+## **Join Us & Grow\!**
+
+This is your chance to give back to the community, grow your presence, and get rewarded for your contributions. Whether you love writing documentation, sharing tutorials, or building open-source projects, your efforts matter.
+
+Apply today and be part of something impactful\!
+
+
+[image1]: images/img_1.png
\ No newline at end of file
diff --git a/docs/en/Community-Articles/2025-04-20-IDX10204/POST.md b/docs/en/Community-Articles/2025-04-20-IDX10204/POST.md
new file mode 100644
index 0000000000..5de0650d11
--- /dev/null
+++ b/docs/en/Community-Articles/2025-04-20-IDX10204/POST.md
@@ -0,0 +1,170 @@
+# Common Errors in JWT Bearer Authentication
+
+When implementing JWT Bearer authentication in an ABP(tiered) application, you might occasionally encounter errors starting with `IDX`. These errors are related to JWT Bearer Token validation and this article will help you understand and resolve them.
+
+## Enable JWT Bearer authentication
+
+Your API project usually contains the following code, which enables JWT Bearer authentication and makes it as the default authentication scheme.
+
+We simply configure the JWT's `Authority` and `Audience` properties, and it will work fine.
+
+```csharp
+context.Services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme)
+ .AddJwtBearer(options =>
+ {
+ options.Authority = "https://localhost:44301/"; //configuration["AuthServer:Authority"];
+ options.Audience = "MyProjectName";
+ });
+```
+
+> `AddJwtBearer` and `AddAbpJwtBearer` will do the same thing, but `AddAbpJwtBearer` is recommended.
+
+## JWT authentication process
+
+Let's take a look at how the above code works.
+
+A JWT Token usually consists of three parts: `Header`, `Payload`, and `Signature`.
+
+- `Header`: Contains the type and signing algorithm of the token
+- `Payload`: Contains the claims of the token, including `sub`, `aud`, `exp`, `iat`, `iss`, `jti`, `preferred_username`, `given_name`, `role`, `email`, etc.
+- `Signature`: The cryptographic signature of the token used to verify its authenticity
+
+Here is an example of a JWT Token issued by `AuthServer(OpenIddict)`:
+
+The `Header` part:
+
+
+
+The `Payload` part:
+
+
+
+### TokenValidationParameters
+
+In the `JwtBearerOptions`, there is a `TokenValidationParameters` property, which is used to validate the JWT Token.
+
+The default implementation for JWT Token validation is `JsonWebTokenHandler`, which comes from the [Microsoft.IdentityModel.JsonWebTokens](https://github.com/AzureAD/azure-activedirectory-identitymodel-extensions-for-dotnet/) package.
+
+We didn't set the `TokenValidationParameters` property in the code above, so the default values below will be used:
+
+```csharp
+//...
+TokenValidationParameters.ValidateAudience = true
+TokenValidationParameters.ValidAudience = "MyProjectName"
+TokenValidationParameters.ValidAudiences = null
+
+TokenValidationParameters.ValidateIssuer = true
+TokenValidationParameters.ValidIssuer = null
+TokenValidationParameters.ValidIssuers = null
+//...
+```
+
+### JWT Bearer Token Validation Process
+
+During JWT Bearer authentication, API website will get the token from the HTTP request and validate it.
+
+The `JsonWebTokenHandler` will get the `OpenID Connect` metadata from the `AuthServer`, it will be used in the validation process, the current metadata request address is: https://localhost:44301/.well-known/openid-configuration , it is a fixed address calculated from the `Authority` property.
+
+First, the token's Signature is verified using the public key obtained from `OpenID Connect` metadata(https://localhost:44301/.well-known/jwks).
+
+Then, the payload is validated. The payload is a JSON object containing essential information such as the `token type`, `expiration time`, `issuer`, and `audience` etc.
+
+Most of the validation problems we may encounter are payload validation failures, for example:
+
+#### Lifetime
+
+If the token in your request has expired, the validation will fail. You will see the exception information like `IDX10230` in the log.
+
+#### Audience
+
+The `ValidAudience` of `TokenValidationParameters` is `MyProjectName`, the `aud` in the payload of the token is also `MyProjectName`, if the token does not contain `aud` or the `aud` does not match, the validation will fail. You may see the exception information like `IDX10206`, `IDX10277` or `IDX10208`.
+
+> If the `ValidateAudience` of `TokenValidationParameters` is `false`, then the `aud` will not be validated.
+
+#### Issuer
+
+The default value of `TokenValidationParameters.ValidateIssuer` is `true`, it requires the token's payload to contain the `issuer` field, and it must match one of `TokenValidationParameters.ValidIssuer` or `TokenValidationParameters.ValidIssuers`.
+
+> The default value of `ValidIssuer` or `ValidIssuers` is `null`, it will use the `issuer` from the `OpenID Connect` metadata as the default value.
+
+1. If the token's payload does not contain the `issuer` field, you may see the error `IDX10211`.
+2. If the API website cannot get the `OpenID Connect` metadata from AuthServer website, the validation will fail. You may see the error `IDX10204`, the full exception message is: `IDX10204: Unable to validate issuer. validationParameters.ValidIssuer is null or whitespace AND validationParameters.ValidIssuers is null or empty.`
+3. If the `issuer` does not match, the validation will fail. You may see the error `IDX10205` in the log.
+
+> If the `ValidateIssuer` of `TokenValidationParameters` is `false`, then the `issuer` will not be validated.
+
+> Please note that `OpenIddict` will use the current HTTP request information as the value of `issuer`. If the AuthServer website is deployed behind a reverse proxy or similar deployment configurations, the `issuer` in the token may not be the value you expect. In this case, please specify it manually.
+
+```csharp
+PreConfigure(serverBuilder =>
+{
+ serverBuilder.SetIssuer("https://localhost:44301/");
+});
+```
+
+## Troubleshooting
+
+To troubleshoot any `IDX` errors during JWT authentication, you can enable detailed logging by configuring the `identitymodel` logs as follows:
+
+```csharp
+using System.Diagnostics.Tracing;
+using Microsoft.IdentityModel.Logging;
+
+public class Program
+{
+ public async static Task Main(string[] args)
+ {
+ IdentityModelEventSource.ShowPII = true;
+ IdentityModelEventSource.Logger.LogLevel = EventLevel.Verbose;
+ var wilsonTextLogger = newTextWriterEventListener("Logs/identitymodel.txt");
+ wilsonTextLogger.EnableEvents(IdentityModelEventSource.Logger, EventLevel.Verbose);
+
+ //...
+ }
+}
+```
+
+Additionally, you can enable `OpenIddict`'s `Verbose` logs for more detailed debugging information:
+
+```csharp
+var loggerConfiguration = new LoggerConfiguration()
+ .MinimumLevel.Debug()
+ .MinimumLevel.Override("Microsoft.EntityFrameworkCore", LogEventLevel.Warning)
+ .MinimumLevel.Override("OpenIddict", LogEventLevel.Verbose)
+ .Enrich.FromLogContext()
+ .WriteTo.Async(c => c.File("Logs/logs.txt"))
+```
+
+## Summary
+
+For JWT authentication, you need to pay attention to the following key points:
+
+1. Ensure your API website can communicate with the AuthServer properly
+2. Verify that the `aud` claim in your token matches the expected audience
+3. Confirm that the `issuer` claim in your token is valid and matches the configuration
+
+You can customize the `JwtBearerOptions`'s `TokenValidationParameters` to modify the validation rules to meet your actual needs.
+
+For example, if your `issuer` needs to support multiple subdomains, you can use the [Owl.TokenWildcardIssuerValidator](https://github.com/maliming/Owl.TokenWildcardIssuerValidator) library to customize the validation.
+
+```csharp
+services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme)
+ .AddJwtBearer(options =>
+ {
+ options.Authority = "https://abp.io";
+ options.Audience = "abp_io";
+
+ options.TokenValidationParameters.IssuerValidator = TokenWildcardIssuerValidator.IssuerValidator;
+ options.TokenValidationParameters.ValidIssuers = new[]
+ {
+ "https://{0}.abp.io"
+ };
+ });
+```
+
+## References
+
+- [Configure JWT bearer authentication in ASP.NET Core]([https://learn.microsoft.com/en-us/aspnet/core/security/authentication/jwt-auth?view=aspnetcore-8.0](https://learn.microsoft.com/en-us/aspnet/core/security/authentication/configure-jwt-bearer-authentication))
+- [OpenIddict](https://github.com/openiddict/openiddict-core)
+- [IdentityModel](https://github.com/AzureAD/azure-activedirectory-identitymodel-extensions-for-dotnet)
+- [Owl.TokenWildcardIssuerValidator](https://github.com/maliming/Owl.TokenWildcardIssuerValidator)
diff --git a/docs/en/Community-Articles/2025-04-20-IDX10204/header.png b/docs/en/Community-Articles/2025-04-20-IDX10204/header.png
new file mode 100644
index 0000000000..13109a4515
Binary files /dev/null and b/docs/en/Community-Articles/2025-04-20-IDX10204/header.png differ
diff --git a/docs/en/Community-Articles/2025-04-20-IDX10204/payload.png b/docs/en/Community-Articles/2025-04-20-IDX10204/payload.png
new file mode 100644
index 0000000000..04b927bbb2
Binary files /dev/null and b/docs/en/Community-Articles/2025-04-20-IDX10204/payload.png differ
diff --git a/docs/en/Community-Articles/2025-04-25-AI-Comment-Summarization/POST.md b/docs/en/Community-Articles/2025-04-25-AI-Comment-Summarization/POST.md
new file mode 100644
index 0000000000..f79fa21ce8
--- /dev/null
+++ b/docs/en/Community-Articles/2025-04-25-AI-Comment-Summarization/POST.md
@@ -0,0 +1,262 @@
+# Using Microsoft AI Extensions Library and OpenAI to Summarize User Comments
+
+Either you are building an e-commerce application or a simple blog, **user comments** (about your products or blog posts) **can grow rapidly**, making it harder for users to get the gist of discussions at a glance. AI is a pretty good tool to solve the problem. By using AI, you can **summarize all the user comments** and show a single paragraph to your users, so they can easily understand the overall thought of users about the product or the blog post.
+
+In this tutorial, we’ll walk through a real-life implementation of using AI to summarize multiple user comments in an application. I will implement the solution based on ABP's **[CMS Kit](https://abp.io/docs/latest/modules/cms-kit)** library, as it already features a **[commenting system](https://abp.io/docs/latest/modules/cms-kit/comments)** and a [demo application](https://cms-kit-demo.abpdemo.com/) that displays user comments on **[gallery images](https://cms-kit-demo.abpdemo.com/image-gallery)** (it has not a comment summary feature yet, we will implement it in this tutorial).
+
+## A Screenshot
+
+Here, an example screenshot from the application with the comment summary feature:
+
+
+
+## Cloning the Repository
+
+If you want to follow the development, you can clone the [CMS Kit Demo repository](https://github.com/abpframework/cms-kit-demo) to your computer and make it running by following the instructions on the [README file](https://github.com/abpframework/cms-kit-demo?tab=readme-ov-file#cms-kit-demo).
+
+I suggest to you to play a little with [the application](https://cms-kit-demo.abpdemo.com/) (create a new user for yourself, add some comments to the images in the gallery), so you understand how it works.
+
+## Preparing the Solution for AI
+
+I will use [Microsoft AI Extensions Library](https://learn.microsoft.com/en-us/dotnet/ai/ai-extensions) to use the AI features. It is an abstraction library that can work with multiple AI models and tools. I will use an OpenAI model in the demo.
+
+The first step is to add the [Microsoft.Extensions.AI.OpenAI](http://nuget.org/packages/Microsoft.Extensions.AI.OpenAI) NuGet package to the project:
+
+````bash
+dotnet add package Microsoft.Extensions.AI.OpenAI --prerelease
+````
+
+>The Microsoft AI Extensions Library was in preview at the time when I wrote this article. If it has a stable release now, you can remove the `--prerelease` parameter for the preceding command.
+
+We will store the OpenAI key and model name in user secrets. So, locate the root path of the CMS Kit project (`src\CmsKitDemo` folder) and execute the following commands in order in a command-line terminal:
+
+````bash
+dotnet user-secrets init
+dotnet user-secrets set OpenAIKey
+dotnet user-secrets set ModelName
+````
+
+For this example, you need to have an [OpenAI API Key](https://platform.openai.com/). That's all. Now, we are ready to use the AI.
+
+## Implementing the AI Summarization
+
+Let's start from the most important point of this article: Comment summarization. I will create a class named `AiCommentSummarizer` to implement the summarization work. Here, the full content of that class:
+
+````csharp
+using System.Text;
+using Microsoft.Extensions.AI;
+using OpenAI;
+using Volo.Abp.DependencyInjection;
+
+namespace CmsKitDemo.Utils;
+
+public class AiCommentSummarizer : ITransientDependency
+{
+ private readonly IConfiguration _configuration;
+
+ public AiCommentSummarizer(IConfiguration configuration)
+ {
+ _configuration = configuration;
+ }
+
+ public async Task SummarizeAsync(string[] commentTexts)
+ {
+ // Get the model and key from the configuration
+ var aiModel = _configuration["ModelName"];
+ var apiKey = _configuration["OpenAIKey"];
+
+ if (aiModel.IsNullOrEmpty() || apiKey.IsNullOrEmpty())
+ {
+ return "";
+ }
+
+ // Create the IChatClient
+ var client = new OpenAIClient(apiKey)
+ .GetChatClient(aiModel)
+ .AsIChatClient();
+
+ // Create a prompt (input for AI)
+ var promptBuilder = new StringBuilder();
+
+ promptBuilder.AppendLine(
+ @"There are comments from different users of our website about an image.
+We want to summarize the comments into a single comment.
+Return a single comment with a maximum of 512 characters. Comments are separated by a newline character and given below."
+ );
+ promptBuilder.AppendLine();
+
+ foreach (var commentText in commentTexts)
+ {
+ promptBuilder.AppendLine("User comment:");
+ promptBuilder.AppendLine(commentText);
+ promptBuilder.AppendLine();
+ }
+
+ // Submit the prompt and get the response
+ var response = await client.GetResponseAsync(
+ promptBuilder.ToString(),
+ new ChatOptions { MaxOutputTokens = 1024 }
+ );
+
+ return response.Text;
+ }
+}
+````
+
+That class is pretty simple and already decorated with comments:
+
+* First, we are getting the API Key and an OpenAI model name from user secrets. I used `gpt-4.1` as the model name, but you can use another available model.
+* Then we are obtaining an `IChatClient` reference for OpenAI. `IChatClient` interface is an abstraction that is provided by the [Microsoft AI Extensions Library](https://learn.microsoft.com/en-us/dotnet/ai/ai-extensions) library, so we can implement rest of the code independently from OpenAI.
+* Then we continue by building a proper prompt (input) for the AI operation.
+* And finally we are using the AI to generate a response (the summary).
+
+At this point, all the AI-related work has already been done. The rest of this article explains how to integrate that summarization feature with the [CMS Kit Demo application](https://cms-kit-demo.abpdemo.com/).
+
+## Adding a CommentsSummary Property to the GalleryImage Entity
+
+The `GalleryImage` entity is used to represent an image on [the image gallery](https://cms-kit-demo.abpdemo.com/image-gallery). I add a `CommentsSummary` property to that entity:
+
+````csharp
+public class GalleryImage : CreationAuditedAggregateRoot
+{
+ public string Description { get; set; }
+
+ public Guid CoverImageMediaId { get; set; }
+
+ public string CommentsSummary { get; set; } // The new property is here
+
+ //...
+}
+````
+
+Since the CMS Kit Demo application uses Entity Framework Core, I need to add a new database schema migration and update the database:
+
+````bash
+dotnet ef migrations add Added_Summary_To_GalleryImage
+dotnet ef database update
+````
+
+## Updating the Summary
+
+Great, we have a `GalleryImage.CommentsSummary` property now. But, how will it be updated when a users adds or removes a comment for an image? To implement that;
+
+* We will listen all the change events for user comments (when a user adds, removes or updates a comment).
+* Whenever a comment is changed, we will find the related gallery image, retrieve all the user comments for this image, use the `AiCommentSummarizer` class to summarize all the comments.
+* Finally, we wil set the `GalleryImage.CommentsSummary` property with the generated summary text.
+
+Here, the implementation:
+
+````csharp
+using CmsKitDemo.Entities;
+using CmsKitDemo.Utils;
+using Microsoft.EntityFrameworkCore;
+using Volo.Abp.DependencyInjection;
+using Volo.Abp.Domain.Entities.Events;
+using Volo.Abp.Domain.Repositories;
+using Volo.Abp.EventBus;
+using Volo.CmsKit.Comments;
+
+namespace CmsKitDemo.EventHandlers;
+
+public class GalleryImageCommentListener :
+ ILocalEventHandler>,
+ ITransientDependency
+{
+ private readonly IRepository _galleryImageRepository;
+ private readonly IRepository _commentRepository;
+ private readonly AiCommentSummarizer _aiCommentSummarizer;
+
+ public GalleryImageCommentListener(
+ IRepository galleryImageRepository,
+ IRepository commentRepository,
+ AiCommentSummarizer aiCommentSummarizer)
+ {
+ _galleryImageRepository = galleryImageRepository;
+ _commentRepository = commentRepository;
+ _aiCommentSummarizer = aiCommentSummarizer;
+ }
+
+ public async Task HandleEventAsync(EntityChangedEventData eventData)
+ {
+ var comment = eventData.Entity;
+
+ //Here, we only interest in comments related to image gallery items
+ if (comment.EntityType != CmsKitDemoConsts.ImageGalleryEntityType)
+ {
+ return;
+ }
+
+ if (!Guid.TryParse(comment.EntityId, out var galleryImageId))
+ {
+ return;
+ }
+
+ // Get the related image from database
+ var galleryImage = await _galleryImageRepository.FindAsync(galleryImageId);
+ if (galleryImage == null)
+ {
+ return;
+ }
+
+ // Get all the comments related to the image
+ var queryable = await _commentRepository.GetQueryableAsync();
+ var allCommentTexts = await queryable
+ .Where(c => c.EntityType == CmsKitDemoConsts.ImageGalleryEntityType &&
+ c.EntityId == comment.EntityId)
+ .Select(c => c.Text)
+ .ToArrayAsync();
+
+ // Update the summary of comments related to the image
+ if (allCommentTexts.Length <= 0)
+ {
+ galleryImage.CommentsSummary = "";
+ }
+ else
+ {
+ galleryImage.CommentsSummary =
+ await _aiCommentSummarizer.SummarizeAsync(allCommentTexts);
+ }
+
+ // Update the image in database
+ await _galleryImageRepository.UpdateAsync(galleryImage);
+ }
+}
+````
+
+Let's explain that class:
+
+* `GalleryImageCommentListener` implements the `ILocalEventHandler>` interface. In this way, it can handle an event whenever a `Comment` [entity](https://abp.io/docs/latest/framework/architecture/domain-driven-design/entities) is changed (created, updated or deleted). We are using ABP's [local event bus](https://abp.io/docs/latest/framework/infrastructure/event-bus/local) and its [pre-defined events](https://abp.io/docs/latest/framework/infrastructure/event-bus/local#pre-built-events).
+* `HandleEventAsync` is called by the ABP Framework whenever a new `Comment` is created, or an existing `Comment` is deleted or updated.
+* ABP's `Comment` entity is reusable and it can be associated with any kind of objects (blog posts, images, etc). So, first we are checking if this comment is related to an image gallery item.
+* Then we are getting the related `GalleryImage` entity from the database.
+* And getting all comments (including the new one) from the database for this image.
+* Finally, using the `AiCommentSummarizer` class to generate the summary and set the `CommentsSummary` property.
+
+## Show the Summary Card on the UI
+
+Everything is ready on the backend. Now, we can show the summary text on the user interface. To do, that, I added `CommentsSummary` property also to the `GalleryImageDto` class and used it on the `/Pages/Gallery/Detail.cshtml` view:
+
+````csharp
+@if (!Model.Image.CommentsSummary.IsNullOrEmpty())
+{
+
+
+
Summary of the User Comments
+
@Model.Image.CommentsSummary
+
+
+}
+````
+
+That section renders the following card on the user interface:
+
+
+
+## Conclusion
+
+In this article, I demonstrated how to use [Microsoft AI Extensions Library](https://learn.microsoft.com/en-us/dotnet/ai/ai-extensions) to work with OpenAI for summarization of multiple user comments. I reused the [ABP's CMS Kit Demo application](https://github.com/abpframework/cms-kit-demo) to show it in a more real world example.
+
+## Source Code
+
+* [Source code of the CMS Kit Demo application](https://github.com/abpframework/cms-kit-demo)
+* [All the changes made for this article (as a pull request)](https://github.com/abpframework/cms-kit-demo/pull/18)
\ No newline at end of file
diff --git a/docs/en/Community-Articles/2025-04-25-AI-Comment-Summarization/comment-example.png b/docs/en/Community-Articles/2025-04-25-AI-Comment-Summarization/comment-example.png
new file mode 100644
index 0000000000..4ac33b60f9
Binary files /dev/null and b/docs/en/Community-Articles/2025-04-25-AI-Comment-Summarization/comment-example.png differ
diff --git a/docs/en/Community-Articles/2025-04-25-AI-Comment-Summarization/summary-card.png b/docs/en/Community-Articles/2025-04-25-AI-Comment-Summarization/summary-card.png
new file mode 100644
index 0000000000..9a36f3cd35
Binary files /dev/null and b/docs/en/Community-Articles/2025-04-25-AI-Comment-Summarization/summary-card.png differ
diff --git a/docs/en/Community-Articles/2025-05-11-AI-Chat/POST.md b/docs/en/Community-Articles/2025-05-11-AI-Chat/POST.md
new file mode 100644
index 0000000000..d6d2a5b6dc
--- /dev/null
+++ b/docs/en/Community-Articles/2025-05-11-AI-Chat/POST.md
@@ -0,0 +1,134 @@
+# Integrating .NET AI Chat Template with ABP Framework
+
+This article demonstrates how to integrate the [.NET AI Chat Template](https://devblogs.microsoft.com/dotnet/announcing-dotnet-ai-template-preview2/) into an ABP Framework application, enabling powerful AI chat capabilities in your ABP-based solution.
+
+
+
+## Step 1: Create a New ABP Project
+
+First, let's create a new [single-layer Blazor Server project](https://abp.io/docs/latest/solution-templates/single-layer-web-application/overview) named `AbpAiChat` using ABP Studio, You can also use the following ABP CLI command to create the project:
+
+```bash
+abp new AbpAiChat -t app-nolayers --ui-framework blazor-server --use-open-source-template
+```
+
+## Step 2: Integrate AI Chat Template
+
+The integration process involves copying and adapting the .NET AI Chat Template code into our ABP project. The template code is already included in our sample project, so you don't need to install it separately.
+
+### 2.1 Project Structure Changes
+
+1. Copy Blazor components to the `Components` folder
+2. Copy AI service classes to the `Services` folder
+3. Add required entities(`IngestedDocument`, `IngestedRecord`) to the `AbpAiChatDbContext` and add new migration
+4. Copy frontend resources to the `wwwroot` folder
+5. Adjust some styles to capatible with the ABP theme
+
+### 2.2 Required NuGet Packages
+
+Add the following packages to `AbpAiChat.csproj`:
+
+```xml
+
+
+
+
+
+
+```
+
+### 2.3 Configure AI Services
+
+Add the following configuration to `AbpAiChatModule.cs`:
+
+```csharp
+private void ConfigureAi(ServiceConfigurationContext context)
+{
+ var credential = new ApiKeyCredential(context.Services.GetConfiguration()["GitHubToken"] ?? throw new InvalidOperationException("Missing configuration: GitHubToken. See the README for details."));
+ var openAiOptions = new OpenAIClientOptions()
+ {
+ Endpoint = new Uri("https://models.inference.ai.azure.com")
+ };
+
+ var ghModelsClient = new OpenAIClient(credential, openAiOptions);
+ var chatClient = ghModelsClient.GetChatClient("gpt-4o-mini").AsIChatClient();
+ var embeddingGenerator = ghModelsClient.GetEmbeddingClient("text-embedding-3-small").AsIEmbeddingGenerator();
+
+ var vectorStore = new JsonVectorStore(Path.Combine(AppContext.BaseDirectory, "vector-store"));
+
+ context.Services.AddSingleton(vectorStore);
+ context.Services.AddScoped();
+ context.Services.AddSingleton();
+ context.Services.AddChatClient(chatClient).UseFunctionInvocation().UseLogging();
+ context.Services.AddEmbeddingGenerator(embeddingGenerator);
+
+ context.Services.Configure(options =>
+ {
+ options.ContentTypeMaps.Add(".mjs", "application/javascript");
+ });
+}
+```
+
+The `ConfigureAi` method is called in the `ConfigureServices` method of `AbpAiChatModule`. It sets up the AI services, including the OpenAI client, chat client, embedding generator, and vector store.
+
+### 2.4 Configure GitHub Token
+
+Add your GitHub Personal Access Token to `appsettings.json`:
+
+```json
+{
+ "GitHubToken": "your-github-token"
+}
+```
+
+You can obtain your token from [GitHub Personal Access Tokens](https://github.com/settings/personal-access-tokens).
+
+## Step 3: Add Custom AI Functionality
+
+Let's add a custom AI function to retrieve the current user's information. Update the `Chat.razor` component:
+
+```csharp
+chatOptions.Tools =
+[
+ AIFunctionFactory.Create(SearchAsync),
+ AIFunctionFactory.Create(GetWeather),
+ AIFunctionFactory.Create(GetCurrentUserInfo)
+];
+
+[Description("Get current user information")]
+private Task GetCurrentUserInfo()
+{
+ return Task.FromResult(CurrentUser.IsAuthenticated ?
+ $"UserId: {CurrentUser.Id}, Name: {CurrentUser.UserName}, Email: {CurrentUser.Email}, Roles: {string.Join(", ", CurrentUser.Roles)}" :
+ "No user information available.");
+}
+```
+
+## Step 4: Add Navigation
+
+Add a `Chat` menu item in `AbpAiChatMenuContributor` to navigate to the AI Chat component.
+
+## Running the Application
+
+After completing the integration, you can run the application and access the AI chat functionality. The chat interface allows you to:
+
+- Get weather information
+- Ask questions about PDF content
+- Retrieve current user information
+- And more!
+
+
+
+## Conclusion
+
+This integration demonstrates how to leverage the power of AI in your ABP Framework applications. The .NET AI Chat Template provides a solid foundation for building intelligent chat interfaces, and ABP Framework makes it more powerful.
+
+## References
+
+- [ABP Single Layer Solution](https://abp.io/docs/latest/solution-templates/single-layer-web-application/overview)
+- [.NET AI Template Documentation](https://devblogs.microsoft.com/dotnet/announcing-dotnet-ai-template-preview1/)
+- [GitHub Personal Access Tokens Guide](https://docs.github.com/en/authentication/keeping-your-account-and-data-secure/managing-your-personal-access-tokens)
+
+## Source Code
+
+- [AbpAiChat Source Code](https://github.com/abpframework/abp-samples/tree/master/AIChat)
diff --git a/docs/en/Community-Articles/2025-05-11-AI-Chat/ai-chat.png b/docs/en/Community-Articles/2025-05-11-AI-Chat/ai-chat.png
new file mode 100644
index 0000000000..8388e2ca9e
Binary files /dev/null and b/docs/en/Community-Articles/2025-05-11-AI-Chat/ai-chat.png differ
diff --git a/docs/en/Community-Articles/2025-05-11-AI-Chat/cover.png b/docs/en/Community-Articles/2025-05-11-AI-Chat/cover.png
new file mode 100644
index 0000000000..5dce208fd6
Binary files /dev/null and b/docs/en/Community-Articles/2025-05-11-AI-Chat/cover.png differ
diff --git a/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/POST.md b/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/POST.md
new file mode 100644
index 0000000000..6007e5371f
--- /dev/null
+++ b/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/POST.md
@@ -0,0 +1,268 @@
+# Resolving Tenant from Route in ABP Framework
+
+The ABP Framework provides multi-tenancy support with various ways to resolve tenant information, including: Cookie, Header, Domain, Route, and more.
+
+This article will demonstrate how to resolve tenant information from the route.
+
+## Tenant Information in Routes
+
+In the ABP Framework, tenant information in routes is handled by the `RouteTenantResolveContributor`.
+
+Let's say your application is hosted at `https://abp.io` and you have a tenant named `acme`. You can add the `{__tenant}` variable to your controller or page routes like this:
+
+```csharp
+[Route("{__tenant}/[Controller]")]
+public class MyController : MyProjectNameController
+{
+ [HttpGet]
+ public IActionResult Get()
+ {
+ return Ok("Hello My Page");
+ }
+}
+```
+
+```cshtml
+@page "{__tenant?}/mypage"
+@model MyPageModel
+
+
+
+
My Page
+
+
+```
+
+When you access `https://abp.io/acme/my` or `https://abp.io/acme/mypage`, ABP will automatically resolve the tenant information from the route.
+
+## Adding __tenant to Global Routes
+
+While we've shown how to add `{__tenant}` to individual controllers or pages, you might want to add it globally to your entire application. Here's how to implement this:
+
+```cs
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using Microsoft.AspNetCore.Mvc.ApplicationModels;
+
+namespace MyCompanyName;
+
+public class AddTenantRouteToPages : IPageRouteModelConvention, IApplicationModelConvention
+{
+ public void Apply(PageRouteModel model)
+ {
+ var selectorCount = model.Selectors.Count;
+ var selectorModels = new List();
+ for (var i = 0; i < selectorCount; i++)
+ {
+ var selector = model.Selectors[i];
+ selectorModels.Add(new SelectorModel
+ {
+ AttributeRouteModel = new AttributeRouteModel
+ {
+ Template = AttributeRouteModel.CombineTemplates("{__tenant:regex(^[a-zA-Z0-9]+$)}", selector.AttributeRouteModel!.Template!.RemovePreFix("/"))
+ }
+ });
+ }
+ foreach (var selectorModel in selectorModels)
+ {
+ model.Selectors.Add(selectorModel);
+ }
+ }
+}
+
+public class AddTenantRouteToControllers :IApplicationModelConvention
+{
+ public void Apply(ApplicationModel application)
+ {
+ var controllers = application.Controllers;
+ foreach (var controller in controllers)
+ {
+ var selector = controller.Selectors.FirstOrDefault();
+ if (selector == null || selector.AttributeRouteModel == null)
+ {
+ controller.Selectors.Add(new SelectorModel
+ {
+ AttributeRouteModel = new AttributeRouteModel
+ {
+ Template = AttributeRouteModel.CombineTemplates("{__tenant:regex(^[[a-zA-Z0-9]]+$)}", controller.ControllerName)
+ }
+ });
+ controller.Selectors.Add(new SelectorModel
+ {
+ AttributeRouteModel = new AttributeRouteModel
+ {
+ Template = controller.ControllerName
+ }
+ });
+ }
+ else
+ {
+ var template = selector.AttributeRouteModel?.Template;
+ template = template.IsNullOrWhiteSpace() ? "{__tenant:regex(^[[a-zA-Z0-9]]+$)}" : AttributeRouteModel.CombineTemplates("{__tenant:regex(^[[a-zA-Z0-9]]+$)}", template.RemovePreFix("/"));
+ controller.Selectors.Add(new SelectorModel
+ {
+ AttributeRouteModel = new AttributeRouteModel
+ {
+ Template = template
+ }
+ });
+ }
+ }
+ }
+}
+```
+
+Register the services:
+
+```cs
+public override void ConfigureServices(ServiceConfigurationContext context)
+{
+ //...
+
+ PostConfigure(options =>
+ {
+ options.Conventions.Add(new AddTenantRouteToPages());
+ });
+
+ PostConfigure(options =>
+ {
+ options.Conventions.Add(new AddTenantRouteToControllers());
+ });
+
+ // Configure cookie path to prevent authentication cookie loss
+ context.Services.ConfigureApplicationCookie(x =>
+ {
+ x.Cookie.Path = "/";
+ });
+ //...
+}
+```
+
+After implementing this, you'll notice that all controllers in your Swagger UI will have the `{__tenant}` route added:
+
+
+
+## Handling Navigation Links
+
+To ensure navigation links automatically include tenant information, we need to add middleware that dynamically adds the tenant to the PathBase:
+
+```cs
+public override void OnApplicationInitialization(ApplicationInitializationContext context)
+{
+ //...
+ app.Use(async (httpContext, next) =>
+ {
+ var tenantMatch = Regex.Match(httpContext.Request.Path, "^/([^/.]+)(?:/.*)?$");
+ if (tenantMatch.Groups.Count > 1 && !string.IsNullOrEmpty(tenantMatch.Groups[1].Value))
+ {
+ var tenantName = tenantMatch.Groups[1].Value;
+ if (!tenantName.IsNullOrWhiteSpace())
+ {
+ var tenantStore = httpContext.RequestServices.GetRequiredService();
+ var tenantNormalizer = httpContext.RequestServices.GetRequiredService();
+ var tenantInfo = await tenantStore.FindAsync(tenantNormalizer.NormalizeName(tenantName)!);
+ if (tenantInfo != null)
+ {
+ if (httpContext.Request.Path.StartsWithSegments(new PathString(tenantName.EnsureStartsWith('/')), out var matchedPath, out var remainingPath))
+ {
+ var originalPath = httpContext.Request.Path;
+ var originalPathBase = httpContext.Request.PathBase;
+ httpContext.Request.Path = remainingPath;
+ httpContext.Request.PathBase = originalPathBase.Add(matchedPath);
+ try
+ {
+ await next(httpContext);
+ }
+ finally
+ {
+ httpContext.Request.Path = originalPath;
+ httpContext.Request.PathBase = originalPathBase;
+ }
+ return;
+ }
+ }
+ }
+ }
+
+ await next(httpContext);
+ });
+ app.UseRouting();
+ app.MapAbpStaticAssets();
+ //...
+}
+```
+
+
+
+After setting the PathBase, we need to add a custom tenant resolver to extract tenant information from the `PathBase`:
+
+```cs
+public class MyRouteTenantResolveContributor : RouteTenantResolveContributor
+{
+ public const string ContributorName = "MyRoute";
+
+ public override string Name => ContributorName;
+
+ protected override Task GetTenantIdOrNameFromHttpContextOrNullAsync(ITenantResolveContext context, HttpContext httpContext)
+ {
+ var tenantId = httpContext.GetRouteValue(context.GetAbpAspNetCoreMultiTenancyOptions().TenantKey) ?? httpContext.Request.PathBase.ToString();
+ var tenantIdStr = tenantId?.ToString()?.RemovePreFix("/");
+ return Task.FromResult(!tenantIdStr.IsNullOrWhiteSpace() ? Convert.ToString(tenantIdStr) : null);
+ }
+}
+```
+
+Register the MyRouteTenantResolveContributor with the ABP Framework:
+
+```cs
+public override void ConfigureServices(ServiceConfigurationContext context)
+{
+ //...
+ Configure(options =>
+ {
+ options.TenantResolvers.Add(new MyRouteTenantResolveContributor());
+ });
+ //...
+}
+```
+
+### Modifying abp.appPath
+
+```csharp
+public override void ConfigureServices(ServiceConfigurationContext context)
+{
+ //...
+ context.Services.AddOptions().Configure((options, rootServiceProvider) =>
+ {
+ var currentTenant = rootServiceProvider.GetRequiredService();
+ if (!currentTenant.Name.IsNullOrWhiteSpace())
+ {
+ options.BaseUrl = currentTenant.Name.EnsureStartsWith('/').EnsureEndsWith('/');
+ }
+ });
+
+ context.Services.RemoveAll(x => x.ServiceType == typeof(IOptions));
+ context.Services.Add(ServiceDescriptor.Scoped(typeof(IOptions<>), typeof(OptionsManager<>)));
+ //...
+}
+```
+
+Browser console output:
+
+```cs
+> https://localhost:44303/acme/
+> abp.appPath
+> '/acme/'
+```
+
+## Summary
+
+By following these steps, you can implement tenant resolution from routes in the ABP Framework and handle navigation links appropriately. This approach provides a clean and maintainable way to manage multi-tenancy in your application.
+
+
+## References
+
+- [ABP Multi-Tenancy](https://docs.abp.io/en/abp/latest/Multi-Tenancy)
+- [Routing in ASP.NET Core](https://learn.microsoft.com/en-us/aspnet/core/fundamentals/routing)
+- [HTML base tag](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/base)
diff --git a/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/cover.png b/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/cover.png
new file mode 100644
index 0000000000..ebd7978ac8
Binary files /dev/null and b/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/cover.png differ
diff --git a/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/swagger-ui.png b/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/swagger-ui.png
new file mode 100644
index 0000000000..72da9bd571
Binary files /dev/null and b/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/swagger-ui.png differ
diff --git a/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/ui.png b/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/ui.png
new file mode 100644
index 0000000000..3ae3f6650c
Binary files /dev/null and b/docs/en/Community-Articles/2025-05-19-RouteTenantResolveContributor/ui.png differ
diff --git a/docs/en/cli/index.md b/docs/en/cli/index.md
index a9caa0b6bb..02521e72cb 100644
--- a/docs/en/cli/index.md
+++ b/docs/en/cli/index.md
@@ -889,6 +889,7 @@ Some features of the CLI requires to be logged in to ABP Platform. The login com
```bash
abp login # Opens a default browser to log in to ABP Platform via abp.io
abp login --device # Use device login flow
+abp login username -p ****** --password # Use user password login
```
A new login with an already active session overwrites the previous session.
diff --git a/docs/en/docs-nav.json b/docs/en/docs-nav.json
index c160bacfde..d031e87466 100644
--- a/docs/en/docs-nav.json
+++ b/docs/en/docs-nav.json
@@ -83,47 +83,59 @@
{
"text": "Overview",
"path": "tutorials/book-store",
- "isIndex": true
+ "isIndex": true,
+ "isInSeries": true
},
{
"text": "1: Creating the Server Side",
- "path": "tutorials/book-store/part-01.md"
+ "path": "tutorials/book-store/part-01.md",
+ "isInSeries": true
+
},
{
"text": "2: The Book List Page",
- "path": "tutorials/book-store/part-02.md"
+ "path": "tutorials/book-store/part-02.md",
+ "isInSeries": true
},
{
"text": "3: Creating, Updating and Deleting Books",
- "path": "tutorials/book-store/part-03.md"
+ "path": "tutorials/book-store/part-03.md",
+ "isInSeries": true
},
{
"text": "4: Integration Tests",
- "path": "tutorials/book-store/part-04.md"
+ "path": "tutorials/book-store/part-04.md",
+ "isInSeries": true
},
{
"text": "5: Authorization",
- "path": "tutorials/book-store/part-05.md"
+ "path": "tutorials/book-store/part-05.md",
+ "isInSeries": true
},
{
"text": "6: Authors: Domain Layer",
- "path": "tutorials/book-store/part-06.md"
+ "path": "tutorials/book-store/part-06.md",
+ "isInSeries": true
},
{
"text": "7: Authors: Database Integration",
- "path": "tutorials/book-store/part-07.md"
+ "path": "tutorials/book-store/part-07.md",
+ "isInSeries": true
},
{
"text": "8: Authors: Application Layer",
- "path": "tutorials/book-store/part-08.md"
+ "path": "tutorials/book-store/part-08.md",
+ "isInSeries": true
},
{
"text": "9: Authors: User Interface",
- "path": "tutorials/book-store/part-09.md"
+ "path": "tutorials/book-store/part-09.md",
+ "isInSeries": true
},
{
"text": "10: Book to Author Relation",
- "path": "tutorials/book-store/part-10.md"
+ "path": "tutorials/book-store/part-10.md",
+ "isInSeries": true
}
]
},
@@ -135,27 +147,33 @@
{
"text": "Overview",
"path": "tutorials/book-store-with-abp-suite",
- "isIndex": true
+ "isIndex": true,
+ "isInSeries": true
},
{
"text": "1: Creating the Solution",
- "path": "tutorials/book-store-with-abp-suite/part-01.md"
+ "path": "tutorials/book-store-with-abp-suite/part-01.md",
+ "isInSeries": true
},
{
"text": "2: Creating the Books",
- "path": "tutorials/book-store-with-abp-suite/part-02.md"
+ "path": "tutorials/book-store-with-abp-suite/part-02.md",
+ "isInSeries": true
},
{
"text": "3: Creating the Authors",
- "path": "tutorials/book-store-with-abp-suite/part-03.md"
+ "path": "tutorials/book-store-with-abp-suite/part-03.md",
+ "isInSeries": true
},
{
"text": "4: Book to Author Relation",
- "path": "tutorials/book-store-with-abp-suite/part-04.md"
+ "path": "tutorials/book-store-with-abp-suite/part-04.md",
+ "isInSeries": true
},
{
"text": "5: Customizing the Generated Code",
- "path": "tutorials/book-store-with-abp-suite/part-05.md"
+ "path": "tutorials/book-store-with-abp-suite/part-05.md",
+ "isInSeries": true
}
]
},
@@ -614,7 +632,7 @@
"path": "framework/infrastructure/blob-storing/google.md"
},
{
- "text": "Bunny Provider",
+ "text": "Bunny.Net Provider",
"path": "framework/infrastructure/blob-storing/bunny.md"
},
{
@@ -2070,19 +2088,23 @@
"items": [
{
"text": "Deploy to Azure Web App Service",
- "path": "solution-templates/layered-web-application/deployment/azure-deployment/azure-deployment.md"
+ "path": "solution-templates/layered-web-application/deployment/azure-deployment/azure-deployment.md",
+ "isInSeries": true
},
{
"text": "Creating an Azure Web App Service Environment",
- "path": "solution-templates/layered-web-application/deployment/azure-deployment/step1-create-azure-resources.md"
+ "path": "solution-templates/layered-web-application/deployment/azure-deployment/step1-create-azure-resources.md",
+ "isInSeries": true
},
{
"text": "Customizing the Configuration of Your ABP Application",
- "path": "solution-templates/layered-web-application/deployment/azure-deployment/step2-configuration-application.md"
+ "path": "solution-templates/layered-web-application/deployment/azure-deployment/step2-configuration-application.md",
+ "isInSeries": true
},
{
"text": "Deploying Application With GitHub Actions",
- "path": "solution-templates/layered-web-application/deployment/azure-deployment/step3-deployment-github-action.md"
+ "path": "solution-templates/layered-web-application/deployment/azure-deployment/step3-deployment-github-action.md",
+ "isInSeries": true
}
]
},
diff --git a/docs/en/framework/api-development/static-csharp-clients.md b/docs/en/framework/api-development/static-csharp-clients.md
index c8eaea4231..b3cf80fcfc 100644
--- a/docs/en/framework/api-development/static-csharp-clients.md
+++ b/docs/en/framework/api-development/static-csharp-clients.md
@@ -47,7 +47,7 @@ Implement this class in your service application. You can use [auto API controll
First, add [Volo.Abp.Http.Client](https://www.nuget.org/packages/Volo.Abp.Http.Client) nuget package to your client project:
````
-Install-Package Volo.Abp.Http.Client
+dotnet add package Volo.Abp.Http.Client
````
Then add `AbpHttpClientModule` dependency to your module:
diff --git a/docs/en/framework/api-development/swagger.md b/docs/en/framework/api-development/swagger.md
index 1699e4d0d2..fb27bcf27b 100644
--- a/docs/en/framework/api-development/swagger.md
+++ b/docs/en/framework/api-development/swagger.md
@@ -29,7 +29,7 @@ If you want to manually install;
1. Add the [Volo.Abp.Swashbuckle](https://www.nuget.org/packages/Volo.Abp.Swashbuckle) NuGet package to your `Web` or `HttpApi.Host` project:
- `Install-Package Volo.Abp.Swashbuckle`
+ `dotnet add package Volo.Abp.Swashbuckle`
2. Add the `AbpSwashbuckleModule` to the dependency list of your module:
diff --git a/docs/en/framework/fundamentals/fluent-validation.md b/docs/en/framework/fundamentals/fluent-validation.md
index d3af69bae9..207cddcbd3 100644
--- a/docs/en/framework/fundamentals/fluent-validation.md
+++ b/docs/en/framework/fundamentals/fluent-validation.md
@@ -21,7 +21,7 @@ If you want to manually install;
1. Add the [Volo.Abp.FluentValidation](https://www.nuget.org/packages/Volo.Abp.FluentValidation) NuGet package to your project:
````
- Install-Package Volo.Abp.FluentValidation
+ dotnet add package Volo.Abp.FluentValidation
````
2. Add the `AbpFluentValidationModule` to the dependency list of your module:
diff --git a/docs/en/framework/infrastructure/background-jobs/hangfire.md b/docs/en/framework/infrastructure/background-jobs/hangfire.md
index 293f03494c..f5991ee19b 100644
--- a/docs/en/framework/infrastructure/background-jobs/hangfire.md
+++ b/docs/en/framework/infrastructure/background-jobs/hangfire.md
@@ -25,7 +25,7 @@ If you want to manually install;
1. Add the [Volo.Abp.BackgroundJobs.HangFire](https://www.nuget.org/packages/Volo.Abp.BackgroundJobs.HangFire) NuGet package to your project:
````
- Install-Package Volo.Abp.BackgroundJobs.HangFire
+ dotnet add package Volo.Abp.BackgroundJobs.HangFire
````
2. Add the `AbpBackgroundJobsHangfireModule` to the dependency list of your module:
diff --git a/docs/en/framework/infrastructure/background-jobs/index.md b/docs/en/framework/infrastructure/background-jobs/index.md
index 5aa58a7d3e..af0c33de21 100644
--- a/docs/en/framework/infrastructure/background-jobs/index.md
+++ b/docs/en/framework/infrastructure/background-jobs/index.md
@@ -134,6 +134,25 @@ namespace MyProject
}
```
+##### Custom Job Name
+
+You can configure `GetBackgroundJobName` delegate of the `AbpBackgroundJobOptions` to change the default job name.
+
+```csharp
+Configure(options =>
+{
+ options.GetBackgroundJobName = (jobType) =>
+ {
+ if (jobTyep == typeof(EmailSendingArgs))
+ {
+ return "emails";
+ }
+
+ return BackgroundJobNameAttribute.GetName(jobType);
+ };
+});
+```
+
### Queue a Job Item
Now, you can queue an email sending job using the `IBackgroundJobManager` service:
diff --git a/docs/en/framework/infrastructure/background-jobs/quartz.md b/docs/en/framework/infrastructure/background-jobs/quartz.md
index 93cf5d53bf..1b18fd36ae 100644
--- a/docs/en/framework/infrastructure/background-jobs/quartz.md
+++ b/docs/en/framework/infrastructure/background-jobs/quartz.md
@@ -25,7 +25,7 @@ If you want to manually install;
1. Add the [Volo.Abp.BackgroundJobs.Quartz](https://www.nuget.org/packages/Volo.Abp.BackgroundJobs.Quartz) NuGet package to your project:
````
- Install-Package Volo.Abp.BackgroundJobs.Quartz
+ dotnet add package Volo.Abp.BackgroundJobs.Quartz
````
2. Add the `AbpBackgroundJobsQuartzModule` to the dependency list of your module:
diff --git a/docs/en/framework/infrastructure/background-workers/hangfire.md b/docs/en/framework/infrastructure/background-workers/hangfire.md
index 4612e15b06..7fec8be273 100644
--- a/docs/en/framework/infrastructure/background-workers/hangfire.md
+++ b/docs/en/framework/infrastructure/background-workers/hangfire.md
@@ -23,7 +23,7 @@ If you want to manually install;
1. Add the [Volo.Abp.BackgroundWorkers.Hangfire](https://www.nuget.org/packages/Volo.Abp.BackgroundWorkers.Hangfire) NuGet package to your project:
````
- Install-Package Volo.Abp.BackgroundWorkers.Hangfire
+ dotnet add package Volo.Abp.BackgroundWorkers.Hangfire
````
2. Add the `AbpBackgroundWorkersHangfireModule` to the dependency list of your module:
diff --git a/docs/en/framework/infrastructure/background-workers/quartz.md b/docs/en/framework/infrastructure/background-workers/quartz.md
index 663fe22a7f..909371adb0 100644
--- a/docs/en/framework/infrastructure/background-workers/quartz.md
+++ b/docs/en/framework/infrastructure/background-workers/quartz.md
@@ -21,7 +21,7 @@ If you want to manually install;
1. Add the [Volo.Abp.BackgroundWorkers.Quartz](https://www.nuget.org/packages/Volo.Abp.BackgroundWorkers.Quartz) NuGet package to your project:
````
- Install-Package Volo.Abp.BackgroundWorkers.Quartz
+ dotnet add package Volo.Abp.BackgroundWorkers.Quartz
````
2. Add the `AbpBackgroundWorkersQuartzModule` to the dependency list of your module:
diff --git a/docs/en/framework/infrastructure/image-manipulation.md b/docs/en/framework/infrastructure/image-manipulation.md
index 7ab8dc562e..a1ee6e1584 100644
--- a/docs/en/framework/infrastructure/image-manipulation.md
+++ b/docs/en/framework/infrastructure/image-manipulation.md
@@ -22,7 +22,7 @@ If you want to manually install;
1. Add the [Volo.Abp.Imaging.Abstractions](https://www.nuget.org/packages/Volo.Abp.Imaging.Abstractions) NuGet package to your project:
```
-Install-Package Volo.Abp.Imaging.Abstractions
+dotnet add package Volo.Abp.Imaging.Abstractions
```
2. Add the `AbpImagingAbstractionsModule` to the dependency list of your module:
@@ -240,7 +240,7 @@ If you want to manually install;
1. Add the [Volo.Abp.Imaging.MagickNet](https://www.nuget.org/packages/Volo.Abp.Imaging.MagickNet) NuGet package to your project:
```
-Install-Package Volo.Abp.Imaging.MagickNet
+dotnet add package Volo.Abp.Imaging.MagickNet
```
2. Add `AbpImagingMagickNetModule` to your [module](../architecture/modularity/basics.md)'s dependency list:
@@ -284,7 +284,7 @@ If you want to manually install;
1. Add the [Volo.Abp.Imaging.ImageSharp](https://www.nuget.org/packages/Volo.Abp.Imaging.ImageSharp) NuGet package to your project:
```
-Install-Package Volo.Abp.Imaging.ImageSharp
+dotnet add package Volo.Abp.Imaging.ImageSharp
```
2. Add `AbpImagingImageSharpModule` to your [module](../architecture/modularity/basics.md)'s dependency list:
@@ -350,7 +350,7 @@ If you want to manually install;
1. Add the [Volo.Abp.Imaging.AspNetCore](https://www.nuget.org/packages/Volo.Abp.Imaging.AspNetCore) NuGet package to your project:
```
-Install-Package Volo.Abp.Imaging.AspNetCore
+dotnet add package Volo.Abp.Imaging.AspNetCore
```
2. Add `AbpImagingAspNetCoreModule` to your [module](../architecture/modularity/basics.md)'s dependency list:
diff --git a/docs/en/framework/infrastructure/sms-sending.md b/docs/en/framework/infrastructure/sms-sending.md
index e6bc9135e5..ef2a86445f 100644
--- a/docs/en/framework/infrastructure/sms-sending.md
+++ b/docs/en/framework/infrastructure/sms-sending.md
@@ -26,7 +26,7 @@ If you want to manually install;
1. Add the [Volo.Abp.Sms](https://www.nuget.org/packages/Volo.Abp.Sms) NuGet package to your project:
```
-Install-Package Volo.Abp.Sms
+dotnet add package Volo.Abp.Sms
```
2. Add the `AbpSmsModule` to the dependency list of your module:
diff --git a/docs/en/framework/infrastructure/string-encryption.md b/docs/en/framework/infrastructure/string-encryption.md
index 5f47e246ef..bb398425f6 100644
--- a/docs/en/framework/infrastructure/string-encryption.md
+++ b/docs/en/framework/infrastructure/string-encryption.md
@@ -22,7 +22,7 @@ If you want to manually install;
1. Add the [Volo.Abp.Security](https://www.nuget.org/packages/Volo.Abp.Security) NuGet package to your project:
- `Install-Package Volo.Abp.Security`
+ `dotnet add package Volo.Abp.Security`
2. Add the `AbpSecurityModule` to the dependency list of your module:
diff --git a/docs/en/framework/infrastructure/text-templating/razor.md b/docs/en/framework/infrastructure/text-templating/razor.md
index 1d17d4337e..77db5467aa 100644
--- a/docs/en/framework/infrastructure/text-templating/razor.md
+++ b/docs/en/framework/infrastructure/text-templating/razor.md
@@ -25,7 +25,7 @@ If you want to manually install;
1. Add the [Volo.Abp.TextTemplating.Razor](https://www.nuget.org/packages/Volo.Abp.TextTemplating.Razor) NuGet package to your project:
````
-Install-Package Volo.Abp.TextTemplating.Razor
+dotnet add package Volo.Abp.TextTemplating.Razor
````
2. Add the `AbpTextTemplatingRazorModule` to the dependency list of your module:
diff --git a/docs/en/framework/infrastructure/text-templating/scriban.md b/docs/en/framework/infrastructure/text-templating/scriban.md
index bd54914411..99d160ce85 100644
--- a/docs/en/framework/infrastructure/text-templating/scriban.md
+++ b/docs/en/framework/infrastructure/text-templating/scriban.md
@@ -19,7 +19,7 @@ If you want to manually install;
1. Add the [Volo.Abp.TextTemplating.Scriban](https://www.nuget.org/packages/Volo.Abp.TextTemplating.Scriban) NuGet package to your project:
````
-Install-Package Volo.Abp.TextTemplating.Scriban
+dotnet add package Volo.Abp.TextTemplating.Scriban
````
2. Add the `AbpTextTemplatingScribanModule` to the dependency list of your module:
diff --git a/docs/en/framework/real-time/signalr.md b/docs/en/framework/real-time/signalr.md
index d6cbbfbada..2d2710c77d 100644
--- a/docs/en/framework/real-time/signalr.md
+++ b/docs/en/framework/real-time/signalr.md
@@ -27,7 +27,7 @@ If you want to manually install;
1. Add the [Volo.Abp.AspNetCore.SignalR](https://www.nuget.org/packages/Volo.Abp.AspNetCore.SignalR) NuGet package to your project:
```
- Install-Package Volo.Abp.AspNetCore.SignalR
+ dotnet add package Volo.Abp.AspNetCore.SignalR
```
Or use the Visual Studio NuGet package management UI to install it.
diff --git a/docs/en/get-started/empty-aspnet-core-application.md b/docs/en/get-started/empty-aspnet-core-application.md
index b0457db71b..95861719ce 100644
--- a/docs/en/get-started/empty-aspnet-core-application.md
+++ b/docs/en/get-started/empty-aspnet-core-application.md
@@ -104,7 +104,7 @@ Replacing ASP.NET Core's DI system by Autofac and integrating to ABP is pretty e
1. Install [Volo.Abp.Autofac](https://www.nuget.org/packages/Volo.Abp.Autofac) package
````
-Install-Package Volo.Abp.Autofac
+dotnet add package Volo.Abp.Autofac
````
2. Add the ``AbpAutofacModule`` Dependency
diff --git a/docs/en/images/samples-cms-kit.png b/docs/en/images/samples-cms-kit.png
new file mode 100644
index 0000000000..0c4e3e5877
Binary files /dev/null and b/docs/en/images/samples-cms-kit.png differ
diff --git a/docs/en/images/samples-easycrm.png b/docs/en/images/samples-easycrm.png
new file mode 100644
index 0000000000..72924ae3e7
Binary files /dev/null and b/docs/en/images/samples-easycrm.png differ
diff --git a/docs/en/images/samples-eshoponabp.png b/docs/en/images/samples-eshoponabp.png
new file mode 100644
index 0000000000..f8e22fff7d
Binary files /dev/null and b/docs/en/images/samples-eshoponabp.png differ
diff --git a/docs/en/images/samples-eventhub.png b/docs/en/images/samples-eventhub.png
new file mode 100644
index 0000000000..548412d62c
Binary files /dev/null and b/docs/en/images/samples-eventhub.png differ
diff --git a/docs/en/modules/docs.md b/docs/en/modules/docs.md
index 69de3362a6..b564980ba3 100644
--- a/docs/en/modules/docs.md
+++ b/docs/en/modules/docs.md
@@ -73,25 +73,25 @@ Or you can also manually install nuget package to each project:
* Install [Volo.Docs.Domain](https://www.nuget.org/packages/Volo.Docs.Domain/) nuget package to `Acme.MyProject.Domain` project.
```bash
- Install-Package Volo.Docs.Domain
+ dotnet add package Volo.Docs.Domain
```
* Install [Volo.Docs.EntityFrameworkCore](https://www.nuget.org/packages/Volo.Docs.EntityFrameworkCore/) nuget package to `Acme.MyProject.EntityFrameworkCore` project.
```bash
- Install-Package Volo.Docs.EntityFrameworkCore
+ dotnet add package Volo.Docs.EntityFrameworkCore
```
* Install [Volo.Docs.Application](https://www.nuget.org/packages/Volo.Docs.Application/) nuget package to `Acme.MyProject.Application` project.
```bash
- Install-Package Volo.Docs.Application
+ dotnet add package Volo.Docs.Application
```
* Install [Volo.Docs.Web](https://www.nuget.org/packages/Volo.Docs.Domain/) nuget package to `Acme.MyProject.Web` project.
```bash
- Install-Package Volo.Docs.Web
+ dotnet add package Volo.Docs.Web
```
##### 3.2.1- Adding Module Dependencies
diff --git a/docs/en/modules/virtual-file-explorer.md b/docs/en/modules/virtual-file-explorer.md
index 21bccfd999..3157198c30 100644
--- a/docs/en/modules/virtual-file-explorer.md
+++ b/docs/en/modules/virtual-file-explorer.md
@@ -24,7 +24,7 @@ Or you can also manually install nuget package to `Acme.MyProject.Web` project:
* Install [Volo.Abp.VirtualFileExplorer.Web](https://www.nuget.org/packages/Volo.Abp.VirtualFileExplorer.Web/) nuget package to `Acme.MyProject.Web` project.
- `Install-Package Volo.Abp.VirtualFileExplorer.Web`
+ `dotnet add package Volo.Abp.VirtualFileExplorer.Web`
##### 2.1- Adding Module Dependencies
diff --git a/docs/en/samples/index.md b/docs/en/samples/index.md
index 6aea3d3fa5..39512d0134 100644
--- a/docs/en/samples/index.md
+++ b/docs/en/samples/index.md
@@ -9,6 +9,8 @@ A reference application built with ABP. It implements the Domain Driven Design w
* [Live demo](https://www.openeventhub.com/)
* [Source code](https://github.com/abpframework/eventhub)
+
+
## eShopOnAbp
> ⚠️ **Important Notice**
@@ -18,6 +20,8 @@ Reference microservice solution built with ABP and .NET.
* [Source code](https://github.com/abpframework/eShopOnAbp)
+
+
## CMS Kit Demo
A minimal example website built with the [CMS Kit module](../modules/cms-kit/index.md).
@@ -25,6 +29,8 @@ A minimal example website built with the [CMS Kit module](../modules/cms-kit/ind
* [Live demo](https://cms-kit-demo.abpdemo.com/)
* [Source code](https://github.com/abpframework/cms-kit-demo)
+
+
## Easy CRM
A middle-size CRM application built with ABP.
@@ -32,6 +38,8 @@ A middle-size CRM application built with ABP.
* [Live demo](http://easycrm.abp.io/)
* [Click here](easy-crm.md) to see the details and download the source code.
+
+
## Book Store
A simple CRUD application to show basic principles of developing an application with ABP. The same sample was implemented with different technologies and different modules:
diff --git a/docs/en/solution-templates/layered-web-application/deployment/deployment-iis.md b/docs/en/solution-templates/layered-web-application/deployment/deployment-iis.md
index 3d33cbd7a1..0de00c8d35 100644
--- a/docs/en/solution-templates/layered-web-application/deployment/deployment-iis.md
+++ b/docs/en/solution-templates/layered-web-application/deployment/deployment-iis.md
@@ -283,6 +283,34 @@ See:
- https://learn.microsoft.com/en-us/aspnet/web-api/overview/testing-and-debugging/troubleshooting-http-405-errors-after-publishing-web-api-applications#resolve-http-405-errors
- https://learn.microsoft.com/en-us/troubleshoot/developer/webapps/iis/site-behavior-performance/http-error-405-website#resolution-for-cause-3
+## Publish the Application(s) as IIS sub-application
+
+If your MVC application is a sub-application, you need to set the `BaseUrl` property of `AbpThemingOptions` to the sub-application’s path. The `BaseUrl` is used to configure the `base` element in the `head` section of the layout page.
+
+```csharp
+public void ConfigureServices(IServiceCollection services)
+{
+ Configure(options =>
+ {
+ options.BaseUrl = "/myapp/";
+ });
+}
+```
+
+```html
+
+
+
+ ...
+
+
+ ...
+
+
+```
+
+For Blazor applications, you can to set the `base` tag in the `App.razor` file instead of configure `AbpThemingOptions`.
+
## How to get stdout-log
If your application is running on IIS and getting errors like `502.5, 500.3x`, you can enable stdout logs to see the error details.
diff --git a/docs/en/studio/installation.md b/docs/en/studio/installation.md
index 34dde8a5c9..c0e6aebd8f 100644
--- a/docs/en/studio/installation.md
+++ b/docs/en/studio/installation.md
@@ -5,25 +5,25 @@
## Pre-requirements
-Before you begin the installation process for ABP Studio, ensure that your system meets the following pre-requirements:
+ABP Studio now features automatic installation of most required dependencies. When you first launch the application, it will check for and install the following components:
-### 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 `v22+` version from the official [Node.js website](https://nodejs.org/en/download/prebuilt-installer).
+* .NET SDK
+* Node.js
+* ABP CLI
+* mkcert (for HTTPS development)
+* WireGuard (for Kubernetes operations)
-### 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:
+The only manual installation required is:
-**For Windows:**
-Installation instructions for your Windows operating system are on the official [WireGuard website](https://www.wireguard.com/install/#windows-7-81-10-11-2008r2-2012r2-2016-2019-2022).
-
-**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
+### Docker (Required for Kubernetes Operations)
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/).
+### Package Manager Prerequisites
+* **Windows:** The automatic installation process uses `winget`. If not already installed, ABP Studio will attempt to install it.
+* **macOS:** The automatic installation process uses `brew`. If not already installed, you'll need to install it manually from [brew.sh](https://brew.sh/).
+
## Installation
-Now you have met the pre-requirements, follow the steps below to install ABP Studio:
+Follow these steps to install ABP Studio:
1. **Download ABP Studio:** Visit [abp.io](https://abp.io/studio) to download the latest version of ABP Studio.
@@ -31,8 +31,10 @@ Now you have met the pre-requirements, follow the steps below to install ABP Stu
2. **Run the Installer:** Execute the installer and follow the on-screen instructions to install ABP Studio on your computer.
+3. **First Launch:** When you first launch ABP Studio, it will automatically check for and install required dependencies. This process may take several minutes, and you'll see progress indicators for each component being installed.
+
## Login
-After you install ABP Studio, you can log in to access all the features. To log in, follow the below steps:
+After installation is complete, you can log in to access all features:
1. **Launch ABP Studio:** Open ABP Studio on your desktop.
diff --git a/docs/en/ui-themes/lepton-x/images/account-layout-background-style.png b/docs/en/ui-themes/lepton-x/images/account-layout-background-style.png
index 5b2f32661d..c8391da4d3 100644
Binary files a/docs/en/ui-themes/lepton-x/images/account-layout-background-style.png and b/docs/en/ui-themes/lepton-x/images/account-layout-background-style.png differ
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.MauiBlazor/Volo/Abp/AspNetCore/Components/MauiBlazor/MauiBlazorCachedApplicationConfigurationClient.cs b/framework/src/Volo.Abp.AspNetCore.Components.MauiBlazor/Volo/Abp/AspNetCore/Components/MauiBlazor/MauiBlazorCachedApplicationConfigurationClient.cs
index a48901f6dd..e4a2f8a3cc 100644
--- a/framework/src/Volo.Abp.AspNetCore.Components.MauiBlazor/Volo/Abp/AspNetCore/Components/MauiBlazor/MauiBlazorCachedApplicationConfigurationClient.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Components.MauiBlazor/Volo/Abp/AspNetCore/Components/MauiBlazor/MauiBlazorCachedApplicationConfigurationClient.cs
@@ -2,6 +2,7 @@
using System.Threading.Tasks;
using Microsoft.AspNetCore.Components.Authorization;
using Microsoft.JSInterop;
+using Volo.Abp.AspNetCore.Components.Web.Security;
using Volo.Abp.AspNetCore.Mvc.ApplicationConfigurations;
using Volo.Abp.AspNetCore.Mvc.ApplicationConfigurations.ClientProxies;
using Volo.Abp.AspNetCore.Mvc.Client;
@@ -23,6 +24,8 @@ namespace Volo.Abp.AspNetCore.Components.MauiBlazor
protected ICurrentTimezoneProvider CurrentTimezoneProvider { get; }
+ protected ApplicationConfigurationChangedService ApplicationConfigurationChangedService { get; }
+
protected IJSRuntime JSRuntime { get; }
protected IClock Clock { get; }
@@ -34,6 +37,7 @@ namespace Volo.Abp.AspNetCore.Components.MauiBlazor
ICurrentTimezoneProvider currentTimezoneProvider,
AuthenticationStateProvider authenticationStateProvider,
AbpApplicationLocalizationClientProxy applicationLocalizationClientProxy,
+ ApplicationConfigurationChangedService applicationConfigurationChangedService,
IJSRuntime jsRuntime,
IClock clock)
{
@@ -42,6 +46,7 @@ namespace Volo.Abp.AspNetCore.Components.MauiBlazor
CurrentTenantAccessor = currentTenantAccessor;
CurrentTimezoneProvider = currentTimezoneProvider;
ApplicationLocalizationClientProxy = applicationLocalizationClientProxy;
+ ApplicationConfigurationChangedService = applicationConfigurationChangedService;
JSRuntime = jsRuntime;
Clock = clock;
@@ -81,6 +86,8 @@ namespace Volo.Abp.AspNetCore.Components.MauiBlazor
await JSRuntime.InvokeAsync("abp.clock.setBrowserTimeZoneToCookie");
}
+
+ ApplicationConfigurationChangedService.NotifyChanged();
}
public virtual Task GetAsync()
diff --git a/framework/src/Volo.Abp.AspNetCore.Components.MauiBlazor/Volo/Abp/AspNetCore/Components/MauiBlazor/MauiCurrentApplicationConfigurationCacheResetService.cs b/framework/src/Volo.Abp.AspNetCore.Components.MauiBlazor/Volo/Abp/AspNetCore/Components/MauiBlazor/MauiCurrentApplicationConfigurationCacheResetService.cs
new file mode 100644
index 0000000000..ad2f6ba983
--- /dev/null
+++ b/framework/src/Volo.Abp.AspNetCore.Components.MauiBlazor/Volo/Abp/AspNetCore/Components/MauiBlazor/MauiCurrentApplicationConfigurationCacheResetService.cs
@@ -0,0 +1,23 @@
+using System.Threading.Tasks;
+using Volo.Abp.AspNetCore.Components.Web.Configuration;
+using Volo.Abp.DependencyInjection;
+
+namespace Volo.Abp.AspNetCore.Components.MauiBlazor;
+
+[Dependency(ReplaceServices = true)]
+public class MauiCurrentApplicationConfigurationCacheResetService :
+ ICurrentApplicationConfigurationCacheResetService,
+ ITransientDependency
+{
+ private readonly MauiBlazorCachedApplicationConfigurationClient _mauiBlazorCachedApplicationConfigurationClient;
+
+ public MauiCurrentApplicationConfigurationCacheResetService(MauiBlazorCachedApplicationConfigurationClient mauiBlazorCachedApplicationConfigurationClient)
+ {
+ _mauiBlazorCachedApplicationConfigurationClient = mauiBlazorCachedApplicationConfigurationClient;
+ }
+
+ public async Task ResetAsync()
+ {
+ await _mauiBlazorCachedApplicationConfigurationClient.InitializeAsync();
+ }
+}
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpInputTagHelper.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpInputTagHelper.cs
index 62cae94413..af1538bbde 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpInputTagHelper.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Bootstrap/TagHelpers/Form/AbpInputTagHelper.cs
@@ -47,11 +47,13 @@ public class AbpInputTagHelper : AbpTagHelper
}
if (isCheckBox)
{
- output.Attributes.AddClass("custom-checkbox");
- output.Attributes.AddClass("custom-control");
+ if (!TagHelper.UseSwitchCheckBox)
+ {
+ output.Attributes.AddClass("custom-checkbox");
+ output.Attributes.AddClass("custom-control");
+ }
+ else
+ {
+ output.Attributes.AddClass("form-switch");
+ }
+
output.Attributes.AddClass("form-check");
}
}
@@ -266,7 +274,7 @@ public class AbpInputTagHelperService : AbpTagHelperService
}
protected virtual async Task GetLabelAsHtmlAsync(TagHelperContext context, TagHelperOutput output, TagHelperOutput inputTag, bool isCheckbox)
- {
+ {
if (IsOutputHidden(inputTag) || TagHelper.SuppressLabel)
{
return string.Empty;
@@ -395,7 +403,7 @@ public class AbpInputTagHelperService : AbpTagHelperService
}
innerOutput.Content.AppendHtml($" ");
}
-
+
innerOutput.Content.AppendHtml(GetRequiredSymbol(context, output));
return innerOutput.Render(_encoder);
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Theme.Shared/wwwroot/libs/abp/aspnetcore-mvc-ui-theme-shared/bootstrap/modal-manager.js b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Theme.Shared/wwwroot/libs/abp/aspnetcore-mvc-ui-theme-shared/bootstrap/modal-manager.js
index e11a8f4d99..d232dbd28a 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Theme.Shared/wwwroot/libs/abp/aspnetcore-mvc-ui-theme-shared/bootstrap/modal-manager.js
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI.Theme.Shared/wwwroot/libs/abp/aspnetcore-mvc-ui-theme-shared/bootstrap/modal-manager.js
@@ -55,7 +55,7 @@ $.validator.defaults.ignore = ''; //TODO: Would be better if we can apply only f
function _createContainer() {
_removeContainer();
- _$modalContainer = $('');
+ _$modalContainer = $('');
var existsModals = $('[id^="Abp_Modal_"]');
if (existsModals.length) {
existsModals.last().after(_$modalContainer)
diff --git a/framework/src/Volo.Abp.AspNetCore.Mvc.UI/Volo/Abp/AspNetCore/Mvc/UI/Theming/AbpThemingOptions.cs b/framework/src/Volo.Abp.AspNetCore.Mvc.UI/Volo/Abp/AspNetCore/Mvc/UI/Theming/AbpThemingOptions.cs
index 1197ad6e53..bf999be982 100644
--- a/framework/src/Volo.Abp.AspNetCore.Mvc.UI/Volo/Abp/AspNetCore/Mvc/UI/Theming/AbpThemingOptions.cs
+++ b/framework/src/Volo.Abp.AspNetCore.Mvc.UI/Volo/Abp/AspNetCore/Mvc/UI/Theming/AbpThemingOptions.cs
@@ -6,6 +6,12 @@ public class AbpThemingOptions
public string? DefaultThemeName { get; set; }
+ ///
+ /// If set, the base element will be added to the head element of the page.
+ /// eg:
+ ///
+ public string? BaseUrl { get; set; }
+
public AbpThemingOptions()
{
Themes = new ThemeDictionary();
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 d042bb6026..b9142b5aed 100644
--- a/framework/src/Volo.Abp.AspNetCore/Microsoft/AspNetCore/Builder/AbpApplicationBuilderExtensions.cs
+++ b/framework/src/Volo.Abp.AspNetCore/Microsoft/AspNetCore/Builder/AbpApplicationBuilderExtensions.cs
@@ -115,7 +115,7 @@ public static class AbpApplicationBuilderExtensions
return app.UseMiddleware();
}
- [Obsolete("Replace with AbpClaimsTransformation")]
+ [Obsolete("Use the TransformAbpClaims extension method from IServiceCollection instead.")]
public static IApplicationBuilder UseAbpClaimsMap(this IApplicationBuilder app)
{
return app.UseMiddleware();
diff --git a/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/Security/Claims/AbpClaimsMapMiddleware.cs b/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/Security/Claims/AbpClaimsMapMiddleware.cs
index 2104edab0a..70fad0d19e 100644
--- a/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/Security/Claims/AbpClaimsMapMiddleware.cs
+++ b/framework/src/Volo.Abp.AspNetCore/Volo/Abp/AspNetCore/Security/Claims/AbpClaimsMapMiddleware.cs
@@ -11,7 +11,7 @@ using Volo.Abp.Security.Claims;
namespace Volo.Abp.AspNetCore.Security.Claims;
-[Obsolete("Replace with AbpClaimsTransformation")]
+[Obsolete("Use the TransformAbpClaims extension method from IServiceCollection instead.")]
public class AbpClaimsMapMiddleware : AbpMiddlewareBase, ITransientDependency
{
public async override Task InvokeAsync(HttpContext context, RequestDelegate next)
diff --git a/framework/src/Volo.Abp.Autofac/Autofac/Builder/AbpRegistrationBuilderExtensions.cs b/framework/src/Volo.Abp.Autofac/Autofac/Builder/AbpRegistrationBuilderExtensions.cs
index 3357d8f95c..e834b3d00d 100644
--- a/framework/src/Volo.Abp.Autofac/Autofac/Builder/AbpRegistrationBuilderExtensions.cs
+++ b/framework/src/Volo.Abp.Autofac/Autofac/Builder/AbpRegistrationBuilderExtensions.cs
@@ -38,7 +38,7 @@ public static class AbpRegistrationBuilderExtensions
}
registrationBuilder = registrationBuilder.EnablePropertyInjection(moduleContainer, implementationType);
- registrationBuilder = registrationBuilder.InvokeRegistrationActions(registrationActionList, serviceType, implementationType);
+ registrationBuilder = registrationBuilder.InvokeRegistrationActions(registrationActionList, serviceType, implementationType, serviceDescriptor.ServiceKey);
return registrationBuilder;
}
@@ -69,10 +69,11 @@ public static class AbpRegistrationBuilderExtensions
this IRegistrationBuilder registrationBuilder,
ServiceRegistrationActionList registrationActionList,
Type serviceType,
- Type implementationType)
+ Type implementationType,
+ object? serviceKey = null)
where TActivatorData : ReflectionActivatorData
{
- var serviceRegistredArgs = new OnServiceRegistredContext(serviceType, implementationType);
+ var serviceRegistredArgs = new OnServiceRegistredContext(serviceType, implementationType, serviceKey);
foreach (var registrationAction in registrationActionList)
{
diff --git a/framework/src/Volo.Abp.BackgroundJobs.Abstractions/Volo/Abp/BackgroundJobs/AbpBackgroundJobOptions.cs b/framework/src/Volo.Abp.BackgroundJobs.Abstractions/Volo/Abp/BackgroundJobs/AbpBackgroundJobOptions.cs
index 6b3aaa52b8..f63bc2bb88 100644
--- a/framework/src/Volo.Abp.BackgroundJobs.Abstractions/Volo/Abp/BackgroundJobs/AbpBackgroundJobOptions.cs
+++ b/framework/src/Volo.Abp.BackgroundJobs.Abstractions/Volo/Abp/BackgroundJobs/AbpBackgroundJobOptions.cs
@@ -14,10 +14,17 @@ public class AbpBackgroundJobOptions
///
public bool IsJobExecutionEnabled { get; set; } = true;
+ ///
+ /// The delegate to get the name of a background job.
+ /// Default: .
+ ///
+ public Func GetBackgroundJobName { get; set; }
+
public AbpBackgroundJobOptions()
{
_jobConfigurationsByArgsType = new Dictionary();
_jobConfigurationsByName = new Dictionary();
+ GetBackgroundJobName = BackgroundJobNameAttribute.GetName;
}
public BackgroundJobConfiguration GetJob()
@@ -61,7 +68,7 @@ public class AbpBackgroundJobOptions
public void AddJob(Type jobType)
{
- AddJob(new BackgroundJobConfiguration(jobType));
+ AddJob(new BackgroundJobConfiguration(jobType, GetBackgroundJobName(jobType)));
}
public void AddJob(BackgroundJobConfiguration jobConfiguration)
diff --git a/framework/src/Volo.Abp.BackgroundJobs.Abstractions/Volo/Abp/BackgroundJobs/BackgroundJobConfiguration.cs b/framework/src/Volo.Abp.BackgroundJobs.Abstractions/Volo/Abp/BackgroundJobs/BackgroundJobConfiguration.cs
index d9f08da2e7..c17ccdc443 100644
--- a/framework/src/Volo.Abp.BackgroundJobs.Abstractions/Volo/Abp/BackgroundJobs/BackgroundJobConfiguration.cs
+++ b/framework/src/Volo.Abp.BackgroundJobs.Abstractions/Volo/Abp/BackgroundJobs/BackgroundJobConfiguration.cs
@@ -10,10 +10,10 @@ public class BackgroundJobConfiguration
public string JobName { get; }
- public BackgroundJobConfiguration(Type jobType)
+ public BackgroundJobConfiguration(Type jobType, string jobName)
{
JobType = jobType;
ArgsType = BackgroundJobArgsHelper.GetJobArgsType(jobType);
- JobName = BackgroundJobNameAttribute.GetName(ArgsType);
+ JobName = jobName;
}
}
diff --git a/framework/src/Volo.Abp.BackgroundJobs.HangFire/Volo/Abp/BackgroundJobs/Hangfire/AbpBackgroundJobsHangfireModule.cs b/framework/src/Volo.Abp.BackgroundJobs.HangFire/Volo/Abp/BackgroundJobs/Hangfire/AbpBackgroundJobsHangfireModule.cs
index ee747e4611..345a69703f 100644
--- a/framework/src/Volo.Abp.BackgroundJobs.HangFire/Volo/Abp/BackgroundJobs/Hangfire/AbpBackgroundJobsHangfireModule.cs
+++ b/framework/src/Volo.Abp.BackgroundJobs.HangFire/Volo/Abp/BackgroundJobs/Hangfire/AbpBackgroundJobsHangfireModule.cs
@@ -25,13 +25,8 @@ public class AbpBackgroundJobsHangfireModule : AbpModule
if (!options.IsJobExecutionEnabled)
{
var hangfireOptions = context.ServiceProvider.GetRequiredService>().Value;
- hangfireOptions.BackgroundJobServerFactory = CreateOnlyEnqueueJobServer;
+ context.ServiceProvider.GetRequiredService();
+ hangfireOptions.BackgroundJobServerFactory = _ => null;
}
}
-
- private BackgroundJobServer? CreateOnlyEnqueueJobServer(IServiceProvider serviceProvider)
- {
- serviceProvider.GetRequiredService();
- return null;
- }
}
diff --git a/framework/src/Volo.Abp.BackgroundJobs.RabbitMQ/Volo/Abp/BackgroundJobs/RabbitMQ/JobQueue.cs b/framework/src/Volo.Abp.BackgroundJobs.RabbitMQ/Volo/Abp/BackgroundJobs/RabbitMQ/JobQueue.cs
index 1dbe4a6dcc..bdc28c7ca0 100644
--- a/framework/src/Volo.Abp.BackgroundJobs.RabbitMQ/Volo/Abp/BackgroundJobs/RabbitMQ/JobQueue.cs
+++ b/framework/src/Volo.Abp.BackgroundJobs.RabbitMQ/Volo/Abp/BackgroundJobs/RabbitMQ/JobQueue.cs
@@ -1,5 +1,6 @@
using System;
using System.Collections.Generic;
+using System.Globalization;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.Extensions.DependencyInjection;
@@ -120,46 +121,44 @@ public class JobQueue : IJobQueue
ChannelAccessor?.Dispose();
}
- protected virtual Task EnsureInitializedAsync()
+ protected virtual async Task EnsureInitializedAsync()
{
if (ChannelAccessor != null && ChannelAccessor.Channel.IsOpen)
{
- return Task.CompletedTask;
+ return;
}
- ChannelAccessor = ChannelPool.Acquire(
+ ChannelAccessor = await ChannelPool.AcquireAsync(
ChannelPrefix + QueueConfiguration.QueueName,
QueueConfiguration.ConnectionName
);
- var result = QueueConfiguration.Declare(ChannelAccessor.Channel);
+ var result = await QueueConfiguration.DeclareAsync(ChannelAccessor.Channel);
Logger.LogDebug($"RabbitMQ Queue '{QueueConfiguration.QueueName}' has {result.MessageCount} messages and {result.ConsumerCount} consumers.");
// Declare delayed queue
- QueueConfiguration.DeclareDelayed(ChannelAccessor.Channel);
+ await QueueConfiguration.DeclareDelayedAsync(ChannelAccessor.Channel);
if (AbpBackgroundJobOptions.IsJobExecutionEnabled)
{
if (QueueConfiguration.PrefetchCount.HasValue)
{
- ChannelAccessor.Channel.BasicQos(0, QueueConfiguration.PrefetchCount.Value, false);
+ await ChannelAccessor.Channel.BasicQosAsync(0, QueueConfiguration.PrefetchCount.Value, false);
}
-
+
Consumer = new AsyncEventingBasicConsumer(ChannelAccessor.Channel);
- Consumer.Received += MessageReceived;
-
+ Consumer.ReceivedAsync += MessageReceived;
+
//TODO: What BasicConsume returns?
- ChannelAccessor.Channel.BasicConsume(
+ await ChannelAccessor.Channel.BasicConsumeAsync(
queue: QueueConfiguration.QueueName,
autoAck: false,
consumer: Consumer
);
}
-
- return Task.CompletedTask;
}
- protected virtual Task PublishAsync(
+ protected virtual async Task PublishAsync(
TArgs args,
BackgroundJobPriority priority = BackgroundJobPriority.Normal,
TimeSpan? delay = null)
@@ -167,29 +166,27 @@ public class JobQueue : IJobQueue
//TODO: How to handle priority
var routingKey = QueueConfiguration.QueueName;
- var basicProperties = CreateBasicPropertiesToPublish();
+ var basicProperties = new BasicProperties
+ {
+ Persistent = true
+ };
if (delay.HasValue)
{
routingKey = QueueConfiguration.DelayedQueueName;
- basicProperties.Expiration = delay.Value.TotalMilliseconds.ToString();
+ basicProperties.Expiration = delay.Value.TotalMilliseconds.ToString(CultureInfo.InvariantCulture);
}
- ChannelAccessor!.Channel.BasicPublish(
- exchange: "",
- routingKey: routingKey,
- basicProperties: basicProperties,
- body: Serializer.Serialize(args!)
- );
-
- return Task.CompletedTask;
- }
-
- protected virtual IBasicProperties CreateBasicPropertiesToPublish()
- {
- var properties = ChannelAccessor!.Channel.CreateBasicProperties();
- properties.Persistent = true;
- return properties;
+ if (ChannelAccessor != null)
+ {
+ await ChannelAccessor.Channel.BasicPublishAsync(
+ exchange: "",
+ routingKey: routingKey,
+ mandatory: false,
+ basicProperties: basicProperties,
+ body: Serializer.Serialize(args!)
+ );
+ }
}
protected virtual async Task MessageReceived(object sender, BasicDeliverEventArgs ea)
@@ -205,17 +202,17 @@ public class JobQueue : IJobQueue
try
{
await JobExecuter.ExecuteAsync(context);
- ChannelAccessor!.Channel.BasicAck(deliveryTag: ea.DeliveryTag, multiple: false);
+ await ChannelAccessor!.Channel.BasicAckAsync(deliveryTag: ea.DeliveryTag, multiple: false);
}
catch (BackgroundJobExecutionException)
{
//TODO: Reject like that?
- ChannelAccessor!.Channel.BasicReject(deliveryTag: ea.DeliveryTag, requeue: true);
+ await ChannelAccessor!.Channel.BasicRejectAsync(deliveryTag: ea.DeliveryTag, requeue: true);
}
catch (Exception)
{
//TODO: Reject like that?
- ChannelAccessor!.Channel.BasicReject(deliveryTag: ea.DeliveryTag, requeue: false);
+ await ChannelAccessor!.Channel.BasicRejectAsync(deliveryTag: ea.DeliveryTag, requeue: false);
}
}
}
diff --git a/framework/src/Volo.Abp.BackgroundJobs.RabbitMQ/Volo/Abp/BackgroundJobs/RabbitMQ/JobQueueConfiguration.cs b/framework/src/Volo.Abp.BackgroundJobs.RabbitMQ/Volo/Abp/BackgroundJobs/RabbitMQ/JobQueueConfiguration.cs
index 9425cd0604..c6418f52f7 100644
--- a/framework/src/Volo.Abp.BackgroundJobs.RabbitMQ/Volo/Abp/BackgroundJobs/RabbitMQ/JobQueueConfiguration.cs
+++ b/framework/src/Volo.Abp.BackgroundJobs.RabbitMQ/Volo/Abp/BackgroundJobs/RabbitMQ/JobQueueConfiguration.cs
@@ -1,5 +1,6 @@
using System;
using System.Collections.Generic;
+using System.Threading.Tasks;
using RabbitMQ.Client;
using Volo.Abp.RabbitMQ;
@@ -34,15 +35,15 @@ public class JobQueueConfiguration : QueueDeclareConfiguration
DelayedQueueName = delayedQueueName;
}
- public virtual QueueDeclareOk DeclareDelayed(IModel channel)
+ public virtual async Task DeclareDelayedAsync(IChannel channel)
{
- var delayedArguments = new Dictionary(Arguments)
+ var delayedArguments = new Dictionary(Arguments)
{
["x-dead-letter-routing-key"] = QueueName,
["x-dead-letter-exchange"] = string.Empty
};
- return channel.QueueDeclare(
+ return await channel.QueueDeclareAsync(
queue: DelayedQueueName,
durable: Durable,
exclusive: Exclusive,
diff --git a/framework/src/Volo.Abp.BackgroundJobs/Volo/Abp/BackgroundJobs/DefaultBackgroundJobManager.cs b/framework/src/Volo.Abp.BackgroundJobs/Volo/Abp/BackgroundJobs/DefaultBackgroundJobManager.cs
index 68851c5043..586b66b5d1 100644
--- a/framework/src/Volo.Abp.BackgroundJobs/Volo/Abp/BackgroundJobs/DefaultBackgroundJobManager.cs
+++ b/framework/src/Volo.Abp.BackgroundJobs/Volo/Abp/BackgroundJobs/DefaultBackgroundJobManager.cs
@@ -18,6 +18,7 @@ public class DefaultBackgroundJobManager : IBackgroundJobManager, ITransientDepe
protected IBackgroundJobSerializer Serializer { get; }
protected IGuidGenerator GuidGenerator { get; }
protected IBackgroundJobStore Store { get; }
+ protected IOptions BackgroundJobOptions { get; }
protected IOptions BackgroundJobWorkerOptions { get; }
public DefaultBackgroundJobManager(
@@ -25,18 +26,20 @@ public class DefaultBackgroundJobManager : IBackgroundJobManager, ITransientDepe
IBackgroundJobSerializer serializer,
IBackgroundJobStore store,
IGuidGenerator guidGenerator,
+ IOptions backgroundJobOptions,
IOptions backgroundJobWorkerOptions)
{
Clock = clock;
Serializer = serializer;
GuidGenerator = guidGenerator;
+ BackgroundJobOptions = backgroundJobOptions;
BackgroundJobWorkerOptions = backgroundJobWorkerOptions;
Store = store;
}
public virtual async Task EnqueueAsync(TArgs args, BackgroundJobPriority priority = BackgroundJobPriority.Normal, TimeSpan? delay = null)
{
- var jobName = BackgroundJobNameAttribute.GetName();
+ var jobName = BackgroundJobOptions.Value.GetBackgroundJobName(typeof(TArgs));
var jobId = await EnqueueAsync(jobName, args!, priority, delay);
return jobId.ToString();
}
diff --git a/framework/src/Volo.Abp.BackgroundWorkers.Hangfire/Volo/Abp/BackgroundWorkers/Hangfire/AbpBackgroundWorkersHangfireModule.cs b/framework/src/Volo.Abp.BackgroundWorkers.Hangfire/Volo/Abp/BackgroundWorkers/Hangfire/AbpBackgroundWorkersHangfireModule.cs
index 2577181143..433ac699d2 100644
--- a/framework/src/Volo.Abp.BackgroundWorkers.Hangfire/Volo/Abp/BackgroundWorkers/Hangfire/AbpBackgroundWorkersHangfireModule.cs
+++ b/framework/src/Volo.Abp.BackgroundWorkers.Hangfire/Volo/Abp/BackgroundWorkers/Hangfire/AbpBackgroundWorkersHangfireModule.cs
@@ -1,5 +1,4 @@
-using System;
-using Hangfire;
+using Hangfire;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Options;
using Volo.Abp.Hangfire;
@@ -23,17 +22,12 @@ public class AbpBackgroundWorkersHangfireModule : AbpModule
if (!options.IsEnabled)
{
var hangfireOptions = context.ServiceProvider.GetRequiredService>().Value;
- hangfireOptions.BackgroundJobServerFactory = CreateOnlyEnqueueJobServer;
+ context.ServiceProvider.GetRequiredService();
+ hangfireOptions.BackgroundJobServerFactory = _ => null;
}
context.ServiceProvider
.GetRequiredService()
.Initialize();
}
-
- private BackgroundJobServer? CreateOnlyEnqueueJobServer(IServiceProvider serviceProvider)
- {
- serviceProvider.GetRequiredService();
- return null;
- }
}
diff --git a/framework/src/Volo.Abp.BlazoriseUI/AbpCrudPageBase.cs b/framework/src/Volo.Abp.BlazoriseUI/AbpCrudPageBase.cs
index 6c4fcf07f9..a2c7193437 100644
--- a/framework/src/Volo.Abp.BlazoriseUI/AbpCrudPageBase.cs
+++ b/framework/src/Volo.Abp.BlazoriseUI/AbpCrudPageBase.cs
@@ -357,10 +357,10 @@ public abstract class AbpCrudPageBase<
}
}
- protected virtual Task CloseCreateModalAsync()
+ protected virtual async Task CloseCreateModalAsync()
{
+ await InvokeAsync(CreateModal!.Hide);
NewEntity = new TCreateViewModel();
- return InvokeAsync(CreateModal!.Hide);
}
protected virtual Task ClosingCreateModal(ModalClosingEventArgs eventArgs)
@@ -474,11 +474,11 @@ public abstract class AbpCrudPageBase<
protected virtual async Task OnCreatedEntityAsync()
{
- NewEntity = new TCreateViewModel();
await GetEntitiesAsync();
await InvokeAsync(CreateModal!.Hide);
await Notify.Success(GetCreateMessage());
+ NewEntity = new TCreateViewModel();
}
protected virtual string GetCreateMessage()
diff --git a/framework/src/Volo.Abp.BlazoriseUI/BlazoriseUiObjectExtensionPropertyInfoExtensions.cs b/framework/src/Volo.Abp.BlazoriseUI/BlazoriseUiObjectExtensionPropertyInfoExtensions.cs
index 2773b88bed..fa4994e2f9 100644
--- a/framework/src/Volo.Abp.BlazoriseUI/BlazoriseUiObjectExtensionPropertyInfoExtensions.cs
+++ b/framework/src/Volo.Abp.BlazoriseUI/BlazoriseUiObjectExtensionPropertyInfoExtensions.cs
@@ -233,19 +233,17 @@ public static class BlazoriseUiObjectExtensionPropertyInfoExtensions
switch (dataTypeAttribute.DataType)
{
case DataType.Password:
- return typeof(TextExtensionProperty<,>);
- case DataType.Date:
- return typeof(DateTimeExtensionProperty<,>);
- case DataType.Time:
- return typeof(TimeExtensionProperty<,>);
case DataType.EmailAddress:
- return typeof(TextExtensionProperty<,>);
case DataType.Url:
- return typeof(TextExtensionProperty<,>);
case DataType.PhoneNumber:
return typeof(TextExtensionProperty<,>);
+ case DataType.Date:
case DataType.DateTime:
return typeof(DateTimeExtensionProperty<,>);
+ case DataType.Time:
+ return typeof(TimeExtensionProperty<,>);
+ case DataType.MultilineText:
+ return typeof(TextAreaExtensionProperty<,>);
}
}
diff --git a/framework/src/Volo.Abp.BlazoriseUI/Components/ObjectExtending/TextAreaExtensionProperty.razor b/framework/src/Volo.Abp.BlazoriseUI/Components/ObjectExtending/TextAreaExtensionProperty.razor
new file mode 100644
index 0000000000..6acf7ba69d
--- /dev/null
+++ b/framework/src/Volo.Abp.BlazoriseUI/Components/ObjectExtending/TextAreaExtensionProperty.razor
@@ -0,0 +1,19 @@
+@typeparam TEntity
+@typeparam TResourceType
+@using Volo.Abp.BlazoriseUI
+@using Volo.Abp.Localization
+@inherits ExtensionPropertyComponentBase
+
+@if (PropertyInfo != null && Entity != null)
+{
+
+
+ @PropertyInfo.GetLocalizedDisplayName(StringLocalizerFactory)
+
+
+
+
+
+
+
+}
diff --git a/framework/src/Volo.Abp.BlazoriseUI/Components/ObjectExtending/TextAreaExtensionProperty.razor.cs b/framework/src/Volo.Abp.BlazoriseUI/Components/ObjectExtending/TextAreaExtensionProperty.razor.cs
new file mode 100644
index 0000000000..853d1e7372
--- /dev/null
+++ b/framework/src/Volo.Abp.BlazoriseUI/Components/ObjectExtending/TextAreaExtensionProperty.razor.cs
@@ -0,0 +1,15 @@
+using Volo.Abp.Data;
+
+namespace Volo.Abp.BlazoriseUI.Components.ObjectExtending;
+public partial class TextAreaExtensionProperty
+ where TEntity : IHasExtraProperties
+{
+ protected string? Value {
+ get {
+ return PropertyInfo.GetTextInputValueOrNull(Entity.GetProperty(PropertyInfo.Name));
+ }
+ set {
+ Entity.SetProperty(PropertyInfo.Name, value, validate: false);
+ }
+ }
+}
\ No newline at end of file
diff --git a/framework/src/Volo.Abp.BlobStoring.Aliyun/Volo/Abp/BlobStoring/Aliyun/AliyunBlobProvider.cs b/framework/src/Volo.Abp.BlobStoring.Aliyun/Volo/Abp/BlobStoring/Aliyun/AliyunBlobProvider.cs
index 1c04fdb755..d20c8d1214 100644
--- a/framework/src/Volo.Abp.BlobStoring.Aliyun/Volo/Abp/BlobStoring/Aliyun/AliyunBlobProvider.cs
+++ b/framework/src/Volo.Abp.BlobStoring.Aliyun/Volo/Abp/BlobStoring/Aliyun/AliyunBlobProvider.cs
@@ -86,7 +86,7 @@ public class AliyunBlobProvider : BlobProviderBase, ITransientDependency
return null;
}
var result = ossClient.GetObject(containerName, blobName);
- return await TryCopyToMemoryStreamAsync(result.Content, args.CancellationToken);
+ return result.ResponseStream;
}
protected virtual string GetContainerName(BlobProviderArgs args)
diff --git a/framework/src/Volo.Abp.BlobStoring.Aws/Volo/Abp/BlobStoring/Aws/AwsBlobProvider.cs b/framework/src/Volo.Abp.BlobStoring.Aws/Volo/Abp/BlobStoring/Aws/AwsBlobProvider.cs
index 00a82a0f3e..7c520da471 100644
--- a/framework/src/Volo.Abp.BlobStoring.Aws/Volo/Abp/BlobStoring/Aws/AwsBlobProvider.cs
+++ b/framework/src/Volo.Abp.BlobStoring.Aws/Volo/Abp/BlobStoring/Aws/AwsBlobProvider.cs
@@ -103,7 +103,7 @@ public class AwsBlobProvider : BlobProviderBase, ITransientDependency
Key = blobName
});
- return await TryCopyToMemoryStreamAsync(response.ResponseStream, args.CancellationToken);
+ return response.ResponseStream;
}
}
diff --git a/framework/src/Volo.Abp.BlobStoring.Azure/Volo/Abp/BlobStoring/Azure/AzureBlobProvider.cs b/framework/src/Volo.Abp.BlobStoring.Azure/Volo/Abp/BlobStoring/Azure/AzureBlobProvider.cs
index 5d45e749ea..bea55c624d 100644
--- a/framework/src/Volo.Abp.BlobStoring.Azure/Volo/Abp/BlobStoring/Azure/AzureBlobProvider.cs
+++ b/framework/src/Volo.Abp.BlobStoring.Azure/Volo/Abp/BlobStoring/Azure/AzureBlobProvider.cs
@@ -66,8 +66,7 @@ public class AzureBlobProvider : BlobProviderBase, ITransientDependency
}
var blobClient = GetBlobClient(args, blobName);
- var download = await blobClient.DownloadAsync();
- return await TryCopyToMemoryStreamAsync(download.Value.Content, args.CancellationToken);
+ return await blobClient.OpenReadAsync(cancellationToken: args.CancellationToken);
}
protected virtual BlobClient GetBlobClient(BlobProviderArgs args, string blobName)
diff --git a/framework/src/Volo.Abp.BlobStoring.FileSystem/Volo/Abp/BlobStoring/FileSystem/FileSystemBlobProvider.cs b/framework/src/Volo.Abp.BlobStoring.FileSystem/Volo/Abp/BlobStoring/FileSystem/FileSystemBlobProvider.cs
index a221441749..2a410005db 100644
--- a/framework/src/Volo.Abp.BlobStoring.FileSystem/Volo/Abp/BlobStoring/FileSystem/FileSystemBlobProvider.cs
+++ b/framework/src/Volo.Abp.BlobStoring.FileSystem/Volo/Abp/BlobStoring/FileSystem/FileSystemBlobProvider.cs
@@ -70,13 +70,7 @@ public class FileSystemBlobProvider : BlobProviderBase, ITransientDependency
return await Policy.Handle()
.WaitAndRetryAsync(2, retryCount => TimeSpan.FromSeconds(retryCount))
- .ExecuteAsync(async () =>
- {
- using (var fileStream = File.OpenRead(filePath))
- {
- return await TryCopyToMemoryStreamAsync(fileStream, args.CancellationToken);
- }
- });
+ .ExecuteAsync(() => Task.FromResult(File.OpenRead(filePath)));
}
protected virtual Task ExistsAsync(string filePath)
diff --git a/framework/src/Volo.Abp.Cli.Core/Volo.Abp.Cli.Core.csproj b/framework/src/Volo.Abp.Cli.Core/Volo.Abp.Cli.Core.csproj
index 48d756465f..547f9f0aff 100644
--- a/framework/src/Volo.Abp.Cli.Core/Volo.Abp.Cli.Core.csproj
+++ b/framework/src/Volo.Abp.Cli.Core/Volo.Abp.Cli.Core.csproj
@@ -16,23 +16,26 @@
-
+
+
+
+
+
-
diff --git a/framework/src/Volo.Abp.Cli/Volo.Abp.Cli.csproj b/framework/src/Volo.Abp.Cli/Volo.Abp.Cli.csproj
index 92753dc30d..e714fcd0b3 100644
--- a/framework/src/Volo.Abp.Cli/Volo.Abp.Cli.csproj
+++ b/framework/src/Volo.Abp.Cli/Volo.Abp.Cli.csproj
@@ -15,7 +15,6 @@
-
diff --git a/framework/src/Volo.Abp.Core/Microsoft/Extensions/DependencyInjection/ServiceCollectionLoggingExtensions.cs b/framework/src/Volo.Abp.Core/Microsoft/Extensions/DependencyInjection/ServiceCollectionLoggingExtensions.cs
index be365df686..be906c9e74 100644
--- a/framework/src/Volo.Abp.Core/Microsoft/Extensions/DependencyInjection/ServiceCollectionLoggingExtensions.cs
+++ b/framework/src/Volo.Abp.Core/Microsoft/Extensions/DependencyInjection/ServiceCollectionLoggingExtensions.cs
@@ -1,4 +1,5 @@
using Microsoft.Extensions.Logging;
+using Microsoft.Extensions.Logging.Abstractions;
using Volo.Abp.Logging;
namespace Microsoft.Extensions.DependencyInjection;
@@ -7,6 +8,7 @@ public static class ServiceCollectionLoggingExtensions
{
public static ILogger GetInitLogger(this IServiceCollection services)
{
- return services.GetSingletonInstance().Create();
+ var loggerFactory = services.GetSingletonInstanceOrNull();
+ return loggerFactory == null ? NullLogger.Instance : loggerFactory.Create();
}
}
diff --git a/framework/src/Volo.Abp.Core/Volo.Abp.Core.csproj b/framework/src/Volo.Abp.Core/Volo.Abp.Core.csproj
index b1605bddd2..caa6b6be94 100644
--- a/framework/src/Volo.Abp.Core/Volo.Abp.Core.csproj
+++ b/framework/src/Volo.Abp.Core/Volo.Abp.Core.csproj
@@ -16,7 +16,6 @@
-
@@ -26,10 +25,7 @@
-
-
-
@@ -40,6 +36,10 @@
+
+
+
+ allruntime; build; native; contentfiles; analyzers
diff --git a/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/ConventionalRegistrarBase.cs b/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/ConventionalRegistrarBase.cs
index 6d1c345840..d9fae16e16 100644
--- a/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/ConventionalRegistrarBase.cs
+++ b/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/ConventionalRegistrarBase.cs
@@ -3,6 +3,7 @@ using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using Microsoft.Extensions.DependencyInjection;
+using Microsoft.Extensions.Logging;
using Volo.Abp.Reflection;
namespace Volo.Abp.DependencyInjection;
@@ -11,14 +12,30 @@ public abstract class ConventionalRegistrarBase : IConventionalRegistrar
{
public virtual void AddAssembly(IServiceCollection services, Assembly assembly)
{
- var types = AssemblyHelper
- .GetAllTypes(assembly)
- .Where(
- type => type != null &&
- type.IsClass &&
- !type.IsAbstract &&
- !type.IsGenericType
- ).ToArray();
+ var logger = services.GetInitLogger();
+ var types = Array.Empty();
+
+ try
+ {
+ types = AssemblyHelper
+ .GetAllTypes(assembly)
+ .Where(
+ type => type != null &&
+ type.IsClass &&
+ !type.IsAbstract &&
+ !type.IsGenericType
+ ).ToArray();
+ }
+ catch (ReflectionTypeLoadException e)
+ {
+ types = e.Types.Select(x => x!).ToArray();
+ logger.LogException(e);
+ }
+ catch (Exception e)
+ {
+ //TODO: Trigger a global event?
+ logger.LogException(e);
+ }
AddTypes(services, types);
}
diff --git a/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/IOnServiceRegistredContext.cs b/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/IOnServiceRegistredContext.cs
index 86dad068b6..a1c50c0c64 100644
--- a/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/IOnServiceRegistredContext.cs
+++ b/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/IOnServiceRegistredContext.cs
@@ -9,4 +9,6 @@ public interface IOnServiceRegistredContext
ITypeList Interceptors { get; }
Type ImplementationType { get; }
+
+ object? ServiceKey { get; }
}
diff --git a/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/OnServiceRegistredContext.cs b/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/OnServiceRegistredContext.cs
index a843b136f6..8491c94e91 100644
--- a/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/OnServiceRegistredContext.cs
+++ b/framework/src/Volo.Abp.Core/Volo/Abp/DependencyInjection/OnServiceRegistredContext.cs
@@ -13,10 +13,13 @@ public class OnServiceRegistredContext : IOnServiceRegistredContext
public virtual Type ImplementationType { get; }
- public OnServiceRegistredContext(Type serviceType, [NotNull] Type implementationType)
+ public virtual object? ServiceKey { get; }
+
+ public OnServiceRegistredContext(Type serviceType, [NotNull] Type implementationType, object? serviceKey = null)
{
ServiceType = Check.NotNull(serviceType, nameof(serviceType));
ImplementationType = Check.NotNull(implementationType, nameof(implementationType));
+ ServiceKey = serviceKey;
Interceptors = new TypeList();
}
diff --git a/framework/src/Volo.Abp.Core/Volo/Abp/Internal/InternalServiceCollectionExtensions.cs b/framework/src/Volo.Abp.Core/Volo/Abp/Internal/InternalServiceCollectionExtensions.cs
index 3f638defe8..be28f569ea 100644
--- a/framework/src/Volo.Abp.Core/Volo/Abp/Internal/InternalServiceCollectionExtensions.cs
+++ b/framework/src/Volo.Abp.Core/Volo/Abp/Internal/InternalServiceCollectionExtensions.cs
@@ -23,7 +23,6 @@ internal static class InternalServiceCollectionExtensions
{
var moduleLoader = new ModuleLoader();
var assemblyFinder = new AssemblyFinder(abpApplication);
- var typeFinder = new TypeFinder(assemblyFinder);
if (!services.IsAdded())
{
@@ -36,8 +35,9 @@ internal static class InternalServiceCollectionExtensions
services.TryAddSingleton(moduleLoader);
services.TryAddSingleton(assemblyFinder);
- services.TryAddSingleton(typeFinder);
services.TryAddSingleton(new DefaultInitLoggerFactory());
+ var typeFinder = new TypeFinder(services.GetInitLogger(), assemblyFinder);
+ services.TryAddSingleton(typeFinder);
services.AddAssemblyOf();
diff --git a/framework/src/Volo.Abp.Core/Volo/Abp/Reflection/AssemblyHelper.cs b/framework/src/Volo.Abp.Core/Volo/Abp/Reflection/AssemblyHelper.cs
index 8e71f011fe..040159dcc8 100644
--- a/framework/src/Volo.Abp.Core/Volo/Abp/Reflection/AssemblyHelper.cs
+++ b/framework/src/Volo.Abp.Core/Volo/Abp/Reflection/AssemblyHelper.cs
@@ -25,13 +25,6 @@ internal static class AssemblyHelper
public static IReadOnlyList GetAllTypes(Assembly assembly)
{
- try
- {
- return assembly.GetTypes();
- }
- catch (ReflectionTypeLoadException ex)
- {
- return ex.Types!;
- }
+ return assembly.GetTypes();
}
}
diff --git a/framework/src/Volo.Abp.Core/Volo/Abp/Reflection/TypeFinder.cs b/framework/src/Volo.Abp.Core/Volo/Abp/Reflection/TypeFinder.cs
index c7350cde18..c4b52604d6 100644
--- a/framework/src/Volo.Abp.Core/Volo/Abp/Reflection/TypeFinder.cs
+++ b/framework/src/Volo.Abp.Core/Volo/Abp/Reflection/TypeFinder.cs
@@ -1,19 +1,24 @@
using System;
using System.Collections.Generic;
using System.Linq;
+using System.Reflection;
using System.Threading;
+using Microsoft.Extensions.Logging;
namespace Volo.Abp.Reflection;
public class TypeFinder : ITypeFinder
{
+ private readonly ILogger _logger;
+
private readonly IAssemblyFinder _assemblyFinder;
private readonly Lazy> _types;
- public TypeFinder(IAssemblyFinder assemblyFinder)
+ public TypeFinder(ILogger logger, IAssemblyFinder assemblyFinder)
{
_assemblyFinder = assemblyFinder;
+ _logger = logger;
_types = new Lazy>(FindAll, LazyThreadSafetyMode.ExecutionAndPublication);
}
@@ -37,9 +42,15 @@ public class TypeFinder : ITypeFinder
allTypes.AddRange(typesInThisAssembly.Where(type => type != null));
}
- catch
+ catch (ReflectionTypeLoadException e)
+ {
+ allTypes = e.Types.Select(x => x!).ToList();
+ _logger.LogException(e);
+ }
+ catch (Exception e)
{
//TODO: Trigger a global event?
+ _logger.LogException(e);
}
}
diff --git a/framework/src/Volo.Abp.Ddd.Domain/Volo/Abp/Domain/Repositories/RepositoryAsyncExtensions.cs b/framework/src/Volo.Abp.Ddd.Domain/Volo/Abp/Domain/Repositories/RepositoryAsyncExtensions.cs
index 2115d00200..e84778ae50 100644
--- a/framework/src/Volo.Abp.Ddd.Domain/Volo/Abp/Domain/Repositories/RepositoryAsyncExtensions.cs
+++ b/framework/src/Volo.Abp.Ddd.Domain/Volo/Abp/Domain/Repositories/RepositoryAsyncExtensions.cs
@@ -1,5 +1,7 @@
using System;
using System.Collections.Generic;
+using System.Linq;
+using System.Linq.Dynamic.Core;
using System.Linq.Expressions;
using System.Threading;
using System.Threading.Tasks;
@@ -12,7 +14,7 @@ public static class RepositoryAsyncExtensions
{
#region Contains
- public static async Task ContainsAsync(
+ public async static Task ContainsAsync(
[NotNull] this IReadOnlyRepository repository,
[NotNull] T item,
CancellationToken cancellationToken = default)
@@ -26,7 +28,7 @@ public static class RepositoryAsyncExtensions
#region Any/All
- public static async Task AnyAsync(
+ public async static Task AnyAsync(
[NotNull] this IReadOnlyRepository repository,
CancellationToken cancellationToken = default)
where T : class, IEntity
@@ -35,7 +37,7 @@ public static class RepositoryAsyncExtensions
return await repository.AsyncExecuter.AnyAsync(queryable, cancellationToken);
}
- public static async Task AnyAsync(
+ public async static Task AnyAsync(
[NotNull] this IReadOnlyRepository repository,
[NotNull] Expression> predicate,
CancellationToken cancellationToken = default)
@@ -45,7 +47,7 @@ public static class RepositoryAsyncExtensions
return await repository.AsyncExecuter.AnyAsync(queryable, predicate, cancellationToken);
}
- public static async Task AllAsync(
+ public async static Task AllAsync(
[NotNull] this IReadOnlyRepository repository,
[NotNull] Expression> predicate,
CancellationToken cancellationToken = default)
@@ -59,7 +61,7 @@ public static class RepositoryAsyncExtensions
#region Count/LongCount
- public static async Task CountAsync(
+ public async static Task CountAsync(
[NotNull] this IReadOnlyRepository repository,
CancellationToken cancellationToken = default)
where T : class, IEntity
@@ -68,7 +70,7 @@ public static class RepositoryAsyncExtensions
return await repository.AsyncExecuter.CountAsync(queryable, cancellationToken);
}
- public static async Task CountAsync(
+ public async static Task CountAsync(
[NotNull] this IReadOnlyRepository repository,
[NotNull] Expression> predicate,
CancellationToken cancellationToken = default)
@@ -78,7 +80,7 @@ public static class RepositoryAsyncExtensions
return await repository.AsyncExecuter.CountAsync(queryable, predicate, cancellationToken);
}
- public static async Task LongCountAsync(
+ public async static Task LongCountAsync(
[NotNull] this IReadOnlyRepository repository,
CancellationToken cancellationToken = default)
where T : class, IEntity
@@ -87,7 +89,7 @@ public static class RepositoryAsyncExtensions
return await repository.AsyncExecuter.LongCountAsync(queryable, cancellationToken);
}
- public static async Task LongCountAsync(
+ public async static Task LongCountAsync(
[NotNull] this IReadOnlyRepository repository,
[NotNull] Expression> predicate,
CancellationToken cancellationToken = default)
@@ -101,41 +103,41 @@ public static class RepositoryAsyncExtensions
#region First/FirstOrDefault
- public static async Task FirstAsync(
+ public async static Task FirstAsync(
[NotNull] this IReadOnlyRepository repository,
CancellationToken cancellationToken = default)
where T : class, IEntity
{
- var queryable = await repository.GetQueryableAsync();
+ var queryable = AddOrderById(await repository.GetQueryableAsync());
return await repository.AsyncExecuter.FirstAsync(queryable, cancellationToken);
}
- public static async Task FirstAsync(
+ public async static Task FirstAsync(
[NotNull] this IReadOnlyRepository repository,
[NotNull] Expression> predicate,
CancellationToken cancellationToken = default)
where T : class, IEntity
{
- var queryable = await repository.GetQueryableAsync();
+ var queryable = AddOrderById(await repository.GetQueryableAsync());
return await repository.AsyncExecuter.FirstAsync(queryable, predicate, cancellationToken);
}
- public static async Task FirstOrDefaultAsync(
+ public async static Task FirstOrDefaultAsync(
[NotNull] this IReadOnlyRepository repository,
CancellationToken cancellationToken = default)
where T : class, IEntity
{
- var queryable = await repository.GetQueryableAsync();
+ var queryable = AddOrderById(await repository.GetQueryableAsync());
return await repository.AsyncExecuter.FirstOrDefaultAsync(queryable, cancellationToken);
}
- public static async Task FirstOrDefaultAsync(
+ public async static Task FirstOrDefaultAsync(
[NotNull] this IReadOnlyRepository repository,
[NotNull] Expression> predicate,
CancellationToken cancellationToken = default)
where T : class, IEntity
{
- var queryable = await repository.GetQueryableAsync();
+ var queryable = AddOrderById(await repository.GetQueryableAsync());
return await repository.AsyncExecuter.FirstOrDefaultAsync(queryable, predicate, cancellationToken);
}
@@ -143,49 +145,63 @@ public static class RepositoryAsyncExtensions
#region Last/LastOrDefault
- public static async Task LastAsync(
+ public async static Task LastAsync(
[NotNull] this IReadOnlyRepository repository,
CancellationToken cancellationToken = default)
where T : class, IEntity
{
- var queryable = await repository.GetQueryableAsync();
+ var queryable = AddOrderById(await repository.GetQueryableAsync());
return await repository.AsyncExecuter.LastAsync(queryable, cancellationToken);
}
- public static async Task LastAsync(
+ public async static Task LastAsync(
[NotNull] this IReadOnlyRepository repository,
[NotNull] Expression> predicate,
CancellationToken cancellationToken = default)
where T : class, IEntity
{
- var queryable = await repository.GetQueryableAsync();
+ var queryable = AddOrderById(await repository.GetQueryableAsync());
return await repository.AsyncExecuter.LastAsync(queryable, predicate, cancellationToken);
}
- public static async Task LastOrDefaultAsync(
+ public async static Task LastOrDefaultAsync(
[NotNull] this IReadOnlyRepository repository,
CancellationToken cancellationToken = default)
where T : class, IEntity
{
- var queryable = await repository.GetQueryableAsync();
+ var queryable = AddOrderById(await repository.GetQueryableAsync());
return await repository.AsyncExecuter.LastOrDefaultAsync(queryable, cancellationToken);
}
- public static async Task LastOrDefaultAsync(
+ public async static Task LastOrDefaultAsync(
[NotNull] this IReadOnlyRepository repository,
[NotNull] Expression> predicate,
CancellationToken cancellationToken = default)
where T : class, IEntity
{
- var queryable = await repository.GetQueryableAsync();
+ var queryable = AddOrderById(await repository.GetQueryableAsync());
return await repository.AsyncExecuter.LastOrDefaultAsync(queryable, predicate, cancellationToken);
}
#endregion
+ #region OrderById
+
+ private static IQueryable AddOrderById([NotNull] this IQueryable queryable)
+ where T : class, IEntity
+ {
+ if (typeof(T).GetInterfaces().Any(i => i.IsGenericType && i.GetGenericTypeDefinition() == typeof(IEntity<>)))
+ {
+ queryable = queryable.OrderBy(nameof(IEntity