Scaffolding From an Entity

Scaffolding support is experimental. Please be sure to stash or commit changes before running the scaffolder in case it doesn’t quite work the way you expect it to.

The scaffolder takes an entity as an input and generates a ton of boilerplate code to make it easier to build and deploy features. This includes all the objects, services, and pages that get data from the database to the browser and back following the patterns used throughout the solution.

Setting Up Your Entity

The scaffolder assumes you’re scaffolding an entity model intended for typical CRUD operations in a database. It’s not required that the target object be an EF entity, but there is some code in the generated service that expects it.

It’s recommended that entities are stored in the Data\Entities folder of the LightNap.Core project. We’ll use TestEntity as the example for this tutorial.

namespace LightNap.Core.Data.Entities
{
    public class TestEntity
    {
        public int Id { get; set; }
        public string Name { get; set; }
        public int Quantity { get; set; }
        public double Price { get; set; }
    }
}

Next, we’ll add a reference to it in ApplicationDbContext:

public class ApplicationDbContext : IdentityDbContext<ApplicationUser, ApplicationRole, string>
{
    public DbSet<TestEntity> TestEntities { get; set; } = null!;
    ...

Note that it’s expected that the DbSet will use the pluralized version of the entity name. If you don’t have it set up this way then some of the generated code won’t compile without minor edits.

Running the Scaffolder

At this time, the scaffolder is provided in source form in the src/Scaffolding folder. You’ll need to build it before you can run it.

  cd src
  dotnet build ./Scaffolding
  .\Scaffolding\LightNap.Scaffolding\bin\Debug\net9.0\LightNap.Scaffolding.exe TestEntity

The script above is run from the repo’s main src directory to cleanly align with the default parameters. You can override most of these parameters if needed.

Generated Files

New files are generated to create a new area based around the provided entity. Instead of changing content within the other areas (Profile, Admin, etc.) the new area will match the entity name (like TestEntity) and provide a starting point for integration. It’s up to the developer to decide whether and which items are merged into other areas or kept contained within the newly created area.

In the Core project, everything is added in a new TestEntities area folder:

File Purpose
Dto/Response/TestEntityDto.cs Returned from get or search requests.
Dto/Request/CreateTestEntityDto.cs Parameter for create requests.
Dto/Request/SearchTestEntityDto.cs Parameter for search requests.
Dto/Request/UpdateTestEntityDto.cs Parameter for update requests.
Extensions/TestEntityExtensions.cs Includes methods for mapping between the entity and DTOs.
Interfaces/ITestEntityService.cs Interface for the area service.
Services/TestEntityService.cs Implementation of the area service.

In the Web API project:

File Purpose
Controllers/TestEntitiesController.cs The Web API endpoints that map to core service methods.

In the Angular project, everything is added in a new app/test-entities area folder:

File Purpose
components/pages/create/create.component.html Markup for the create item page.
components/pages/create/create.component.ts Code for the create item page.
components/pages/get/get.component.html Markup for the get item page.
components/pages/get/get.component.ts Code for the get item page.
components/pages/edit/edit.component.html Markup for the edit item page.
components/pages/edit/edit.component.ts Code for the edit item page.
components/pages/index/index.component.html Markup for the area landing/search page.
components/pages/index/index.component.ts Code for the area landing/search page.
components/pages/routes.ts Relative routes for the area pages.
helpers/test-entity.helper.ts Rehydrates dates from DTOs if needed.
models/response/test-entity.ts Maps to the server’s TestEntityDto.
models/request/create-test-entity-request.ts Maps to the server’s CreateTestEntityDto.
models/request/search-test-entities-request.ts Maps to the server’s SearchTestEntityDto.
models/request/update-test-entity-request.ts Maps to the server’s UpdateTestEntityDto.
services/data.service.ts The private service for HTTP calls to the REST API.
services/test-entity.service.ts The area service for app usage.

Final Configuration

After the scaffolder completes there are still a couple of things that need to be done for it to be ready for testing.

Registering the Core Service For the Web API Controller

The new Web API controller expects to have the new Core service injected. This needs to be configured in the AddApplicationServices method of Extensions/ApplicationServiceExtensions.cs and will look something like this:

public static IServiceCollection AddApplicationServices(this IServiceCollection services)
{
  ...
  services.AddScoped<ITestEntityService, TestEntityService>();
  ...

Now you can build and run the back-end.

Registering the Area Routes in the Angular Front-End

The new Angular area routes need to be added to the root route tree in app/routing/routes.ts so that the area pages can be accessed.

First, add the import of the routes at the top:

import { Routes as TestEntityRoutes } from "../test-entities/components/pages/routes";

Next, add the routes themselves to the tree. For example, if you want them to only be accessible to logged-in users, put them after the profile routes:

...
children: [
  { path: "home", data: { breadcrumb: "Home" }, children: UserRoutes },
  { path: "profile", data: { breadcrumb: "Profile" }, children: ProfileRoutes },
  { path: "test-entities", children: TestEntityRoutes }
  ...

Now you can build and run the front-end. You can access the new area at a URL like http://localhost:4200/test-entities.

Undoing the Scaffolder

The scaffolder only creates files, so deleting those files will undo its work.

Updating Scaffolding

The scaffolder does not support updating existing files. It will exit without making changes if there are any files already in locations it plans to write to.