DotNetBrightener.WebApi.GenericCRUD.Generator
                               
                            
                                2024.0.14.6-rc-242751002
                            
                        
                    See the version list below for details.
dotnet add package DotNetBrightener.WebApi.GenericCRUD.Generator --version 2024.0.14.6-rc-242751002
NuGet\Install-Package DotNetBrightener.WebApi.GenericCRUD.Generator -Version 2024.0.14.6-rc-242751002
<PackageReference Include="DotNetBrightener.WebApi.GenericCRUD.Generator" Version="2024.0.14.6-rc-242751002" />
<PackageVersion Include="DotNetBrightener.WebApi.GenericCRUD.Generator" Version="2024.0.14.6-rc-242751002" />
<PackageReference Include="DotNetBrightener.WebApi.GenericCRUD.Generator" />
paket add DotNetBrightener.WebApi.GenericCRUD.Generator --version 2024.0.14.6-rc-242751002
#r "nuget: DotNetBrightener.WebApi.GenericCRUD.Generator, 2024.0.14.6-rc-242751002"
#:package DotNetBrightener.WebApi.GenericCRUD.Generator@2024.0.14.6-rc-242751002
#addin nuget:?package=DotNetBrightener.WebApi.GenericCRUD.Generator&version=2024.0.14.6-rc-242751002&prerelease
#tool nuget:?package=DotNetBrightener.WebApi.GenericCRUD.Generator&version=2024.0.14.6-rc-242751002&prerelease
Snippet
Code Generator for Centralized CRUD WebAPI in ASP.NET Core Applications
© 2024 DotNet Brightener. admin@dotnetbrightener.com
Instruction
Most applications rely on CRUD operations. This tool aids in generating WebAPI controllers and DataService interfaces/classes using the DotNetBrightener.WebApi.GenericCRUD and DotNetBrightener.DataAccess.Abstractions libraries.
Specifically, DotNetBrightener.WebApi.GenericCRUD provides core CRUD functionalities, exposing them as WebAPI controllers, while DotNetBrightener.DataAccess.Abstractions facilitates the database access layer performing these CRUD operations.
Using this generator tool eliminates the need for manually implementing CRUD WebAPI controllers and Data Access Service classes/interfaces for each domain entity in your application. Upon building the application in Visual Studio, the necessary classes are automatically generated for your project, streamlining your workflow.
Getting Started
Create new project
You can create the project in Visual Studio for your convenience. You can put everything into one single WebAPI project, but it's recommended not to do that, instead, you should separate the project into different libraries. The suggestion is as the following table
| Project Name | Project Type | Purpose | 
|---|---|---|
| {YourProject}.WebAPI | WebAPI | The entry point of the application | 
| {YourProject}.Core or {YourProject}.Entities | Class Library | The library that contains only the entities needed for your application | 
| {YourProject}.Services | Class Library | The library that contains business logics and CRUD auto generated classes | 
| {YourProject}.Database | Class Library | The library that holds the database context to communicate with the database | 
If you like CLI, you can follow the instruction below.
- Create a new .NET 8 WebAPI project:
dotnet new webapi -n {your-project-name} -f net8.0
dotnet new classlib -n {your-project-name}.Database -f net8.0
dotnet new sln --name {your-project-name}
dotnet sln {your-project-name}.sln add {your-project-name}\\{your-project-name}.csproj
dotnet sln {your-project-name}.sln add {your-project-name}.Database\\{your-project-name}.Database.csproj
- If needed to separate the entity definitions to a different library, create new class library project
dotnet new classlib -n {your-project-name}.Core -f net8.0
dotnet sln {your-project-name}.sln add {your-project-name}.Core\\{your-project-name}.Core.csproj
- If needed to separate the service layer, create new class library project
dotnet new classlib -n {your-project-name}.Services -f net8.0
dotnet sln {your-project-name}.sln add {your-project-name}.Services\\{your-project-name}.Services.csproj
Now you can open the solution in Visual Studio.
Add Generic CRUD and its Generator Library
1. Add needed libraries
Add the following packages to the projects, following the instruction below
| Package Name | Project | Note | 
|---|---|---|
| DotNetBrightener.WebApi.GenericCRUD | Web API | - | 
| DotNetBrightener.WebApi.GenericCRUD.Generator | Web API, Service | - | 
| DotNetBrightener.Plugins.EventPubSub | Web API, Service | - | 
| DotNetBrightener.Plugins.EventPubSub.DependencyInjection | Web API | - | 
| DotNetBrightener.DataAccess.Abstractions | Core (Entity) | - | 
| DotNetBrightener.DataAccess.EF | Database | If your project uses MSSQL as database provider | 
| DotNetBrightener.DataAccess.EF.PostgreSQL | Database | If your project uses PostgreSQL as database provider | 
2. Update the settings to enable code generator
Open the csproj file where you referenced the DotNetBrightener.WebApi.GenericCRUD.Generator library, and update the <PackageReference> of that package with the following
<ItemGroup>
	<PackageReference Include="DotNetBrightener.WebApi.GenericCRUD.Generator"
					  Version="{current-version-of-the-library}"
					  OutputItemType="Analyzer"
					  ReferenceOutputAssembly="false" />
</ItemGroup>
The change is to add
	OutputItemType="Analyzer"
	ReferenceOutputAssembly="false"
to the XML tag, which enables the auto generate code for the project.
Create first entity
The entities must inherit from BaseEntity or BaseEntityWithAuditInfo from the DotNetBrightener.DataAccess.Abstractions library. Create any entity into the Core or Entity project.
Example:
using DotNetBrightener.DataAccess.Models;
 
namespace CRUDWebApiWithGeneratorDemo.Core.Entities;
 
public class Product: BaseEntity
{
    [MaxLength(255)]
    public string Name { get; set; }
 
	// omitted code
}
The Entity/Core project should only contain the entities without any other logics
Configure Code Generator for Service Project
Define auto-generate entities registration class
At the root of Service project, create a class CRUDDataServiceGeneratorRegistration.cs with the following content
using CRUDWebApiWithGeneratorDemo.Core.Entities;
 
namespace CRUDWebApiWithGeneratorDemo.Services;
 
public class CRUDDataServiceGeneratorRegistration
{
    public List<Type> Entities =
    [
		// provide all entities that need to generate CRUD data service for
        typeof(Product),
        typeof(ProductCategory)
    ];
}
The class name CRUDDataServiceGeneratorRegistration is vital as the generator library relies on it to discern what needs generating.
In the Entities property of this class, ensure to include all entities for which CRUD data service interfaces and classes are required.
Configure Code Generator for WebAPI Project
Define auto-generate entities registration class
At the root of WebAPI project, create a class CRUDWebApiGeneratorRegistration.cs with the following content
public class CRUDWebApiGeneratorRegistration
{
	// reference the CRUDDataServiceGeneratorRegistration from Service project
    Type DataServiceRegistrationType = typeof(CRUDDataServiceGeneratorRegistration);
 
    public List<Type> Entities =
    [
		// provide all entities that need to generate CRUD API Controllers for
		// Some entities related to Authorization / Authentication / Security should be ignored
        typeof(Product),
        typeof(ProductCategory)
    ];
}
The class name CRUDWebApiGeneratorRegistration is vital as the generator library relies on it to discern what needs generating.
In the Entities property of this class, ensure to include all entities for which CRUD Web API Controllers are required.
Other configurations for Web API Project
In Program.cs or Startup.cs, you'll need to add the registration for the generated DataService interfaces into the ServiceCollection or Dependencies Container of your choice.
// register the generated data services
builder.Services.AddScoped<IProductCategoryDataService, ProductCategoryDataService>();
builder.Services.AddScoped<IProductDataService, ProductDataService>();
// other service registration
One good approach is to implement a mechanism in your application to automatically detect the DataService classes in your Service project and register them automatically to the ServiceCollection or your dependencies container.
Database Migration
This code generator instruction does not cover how to implement database migration. You will need to follow instruction from Microsoft Entity Framework or the ORM library that you use for your project.
Build and Run Project
As you followed the instruction so far, upon building and running the project, the needed classes will be automatically generated. By default, .NET WebAPI project includes the OpenAPI library, which will detect the available APIs and you can see the following when the project is launch:
![[swagger_screenshot.png]]
Auto-Generated Web APIs
The generator will generate the following APIs for each entity type registered in the registration.
GET /api/{entity}
- Retrieve list of records of the entitytype, and satisfies the filter, if provided
This API accepts the following query string parameters:
| Parameter | Type | Description | 
|---|---|---|
| columns | string[], separate by commas | The columns / fields of the entity to retrieve | 
| pageSize | number | The number of records to return in one page, default is 20 | 
| pageIndex | number | The index of the page to return, default is 0 | 
| orderBy | string | The column to sort the result by, in ascending order. If the value starts with a hyphen ( -) and followed by the column name, the result is sorted in descending order. This parameter impacts how the data is returned. | 
| {column_name} | any | The filter expressions to filter the result by the column_name. Eg:createdBy=user*will filter the result to return the records that haveCreatedByvalue starts withuser. Or,summary=contains(value3)&createdDate=<=2023-12-01, will filter the records that havesummaryvalue containsvalue3in the string, andcreatedDateis before01 Dec 2023.<br/><br/>See the tables below for more detail. | 
column_nameis case-sensitive. You will need to obtain the correct name of the column from the response body.
There is no
ORoperator supported at the moment, because the nature of HTTP query strings combined by&operator. Therefore, onlyANDoperator can be supported.
You can use custom queries for the {column_name} parameter with different operators. Here's a table explaining how to use them:
The follow operators are supported in most of all data types, string, int, long, float, double
| Operator | Symbol | Example Usage | Description | 
|---|---|---|---|
| Equals | eq(default) | id=eq(100)orid=100 | Filters records where the idvalue matches the one on the right side of the query or in the parentheses. | 
| Not Equals | ne | id=ne_100<br />id=ne(100)<br/>id=!(100) | Filters the records where the idvalue doesn't not equal the value on the right operand or in the parentheses. | 
| In | in | summary=in(valuea,valueb) | Filters the records where the summaryis in the values defined in the parentheses. | 
| Not In | notin<br />!in | summary=notin(valuea,valueb)<br />summary=!in(valuea,valueb) | Filters the records where the summaryIS NOT in the values defined in the parentheses. | 
The following operators are supported for string data type.
| Operator | Symbol | Example Usage | Description | 
|---|---|---|---|
| Contains | contains<br/>like<br/>*keyword* | summary=contains(value)<br />summary=like(value)<br />summary=*value* | Filters the records where the summarycontains the value in the parentheses. | 
| Not Contains | notcontains<br/>notlike<br/>!contains<br />!like | summary=!contains(value)<br />summary=!like(value) | Filters the records where the summaryDOES NOT contains the value in the parentheses. | 
| Starts With | startsWith<br />sw<br/>keyword* | summary=startsWith(value)<br/>summary=sw(value)<br/>summary=value* | Filters the records where the summarystarts with the value in the parentheses or on the right side of the query. | 
| Not Starts With | !startsWith<br />!sw | summary=!startsWith(value)<br/>summary=!sw(value) | Filters the records where the summaryDOES NOT start with the value in the parentheses. | 
| Ends With | endsWith<br/>ew<br/>*keyword | summary=endsWith(value)<br />summary=ew(value)<br/>summary=*value | Filters the records where the summaryends with the value in the parentheses or on the right side of the query. | 
| Not Ends With | !endsWith<br/>!ew | summary=!endsWith(value)<br />summary=!ew(value) | Filters the records where the summaryDOES NOT end with the value in the parentheses. | 
The following operators are supported for int, float, double, decimal, datetime and datetimeoffset data type.
| Operator | Symbol | Example Usage | Description | 
|---|---|---|---|
| Greater Than<br />After (for datetimetypes) | > | id=>(10)<br />displayIndex=>(200)<br />expirationDate=>(2023-12-01) | - | 
| Greater Than or Equals<br />On or After (for datetimetype) | >= | id=>=(10)<br />displayIndex=>=(200)<br />expirationDate=>=(2023-12-01) | - | 
| Less Than<br />Before (for datetimetypes) | < | id=<(10)<br />displayIndex=<(200)<br />invoiceDate=<(2023-12-01) | - | 
| Less Than or Equals<br />Before or On (for datetimetypes) | <= | id=<=(10)<br />displayIndex=<=(200)<br />invoiceDate=<=(2023-12-01) | - | 
GET /api/{entity}/{id}
- Get the record of the entitytype byid
This API accepts the following query string parameters
| Parameter | Type | Description | 
|---|---|---|
| columns | string[], separate by commas | The columns / fields of the entity to retrieve | 
POST /api/{entity}
- Create a new record of the entitytype
PUT /api/{entity}/{id}
- Update entity by id
The body of the request can be part of the entity. Only the provided fields in the body will be updated to the entity specified by the id
DELETE /api/{entity}/{id}
- Delete entity by id
PUT /api/{entity}/{id}/undelete
- Restore the deleted record of the entitybyid
| Product | Versions Compatible and additional computed target framework versions. | 
|---|---|
| .NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. net10.0 was computed. net10.0-android was computed. net10.0-browser was computed. net10.0-ios was computed. net10.0-maccatalyst was computed. net10.0-macos was computed. net10.0-tvos was computed. net10.0-windows was computed. | 
| .NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. | 
| .NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. | 
| .NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. | 
| MonoAndroid | monoandroid was computed. | 
| MonoMac | monomac was computed. | 
| MonoTouch | monotouch was computed. | 
| Tizen | tizen40 was computed. tizen60 was computed. | 
| Xamarin.iOS | xamarinios was computed. | 
| Xamarin.Mac | xamarinmac was computed. | 
| Xamarin.TVOS | xamarintvos was computed. | 
| Xamarin.WatchOS | xamarinwatchos was computed. | 
- 
                                                    .NETStandard 2.0- Microsoft.CodeAnalysis.CSharp (>= 4.11.0)
- Microsoft.CodeAnalysis.CSharp.Workspaces (>= 4.11.0)
 
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.