This project is an example of architecture using new technologies and best practices.
The goal is to share knowledge and use it as reference for new projects.
Thanks for enjoying!
- .NET Core 3.1
- ASP.NET Core 3.1
- Entity Framework Core 3.1
- C# 8.0
- Angular 10
- UIkit
- Docker
- Azure DevOps
- Clean Code
- SOLID Principles
- DDD (Domain-Driven Design)
- Separation of Concerns
- DevOps
- Code Analysis
Command Line
- Open directory source\Web\Frontend in command line and execute npm run restore.
- Open directory source\Web in command line and execute dotnet run.
- Open https://localhost:8090.
Visual Studio Code
- Open directory source\Web\Frontend in command line and execute npm run restore.
- Open source directory in Visual Studio Code.
- Press F5.
Visual Studio
- Open directory source\Web\Frontend in command line and execute npm run restore.
- Open source\Architecture.sln in Visual Studio.
- Set Architecture.Web as startup project.
- Press F5.
Docker
- Execute docker-compose up --build -d in root directory.
- Open http://localhost:8090.
Books
- Clean Code: A Handbook of Agile Software Craftsmanship - Robert C. Martin (Uncle Bob)
- Clean Architecture: A Craftsman's Guide to Software Structure and Design - Robert C. Martin (Uncle Bob)
- Implementing Domain-Driven Design - Vaughn Vernon
- Domain-Driven Design Distilled - Vaughn Vernon
- Domain-Driven Design: Tackling Complexity in the Heart of Software - Eric Evans
- Domain-Driven Design Reference: Definitions and Pattern Summaries - Eric Evans
Visual Studio Code Extensions
Source: https://github.com/rafaelfgx/DotNetCore
Published: https://www.nuget.org/profiles/rafaelfgx
Web: API and Frontend (Angular).
Application: Flow control.
Domain: Business rules and domain logic.
Model: Data transfer objects.
Database: Database persistence.
@Injectable({ providedIn: "root" })
export class AppCustomerService {
constructor(
private readonly http: HttpClient,
private readonly gridService: GridService) { }
private readonly url = "customers";
add(model: CustomerModel) {
return this.http.post<number>(this.url, model);
}
delete(id: number) {
return this.http.delete(`${this.url}/${id}`);
}
get(id: number) {
return this.http.get<CustomerModel>(`${this.url}/${id}`);
}
grid(parameters: GridParametersModel) {
return this.gridService.get<CustomerModel>(`${this.url}/grid`, parameters);
}
inactivate(id: number) {
return this.http.patch(`${this.url}/${id}/inactivate`, {});
}
list() {
return this.http.get<CustomerModel[]>(this.url);
}
update(model: CustomerModel) {
return this.http.put(`${this.url}/${model.id}`, model);
}
}
@Injectable({ providedIn: "root" })
export class AppGuard implements CanActivate {
constructor(private readonly appAuthService: AppAuthService) { }
canActivate() {
if (this.appAuthService.signedin()) { return true; }
this.appAuthService.signin();
return false;
}
}
@Injectable({ providedIn: "root" })
export class AppErrorHandler implements ErrorHandler {
constructor(private readonly injector: Injector) { }
handleError(error: any) {
if (error instanceof HttpErrorResponse) {
switch (error.status) {
case 422: {
const appModalService = this.injector.get<AppModalService>(AppModalService);
appModalService.alert(error.error);
return;
}
}
}
console.error(error);
}
}
@Injectable({ providedIn: "root" })
export class AppHttpInterceptor implements HttpInterceptor {
constructor(private readonly appAuthService: AppAuthService) { }
intercept(request: HttpRequest<any>, next: HttpHandler) {
request = request.clone({
setHeaders: { Authorization: `Bearer ${this.appAuthService.token()}` }
});
return next.handle(request);
}
}
public class Startup
{
public void Configure(IApplicationBuilder application)
{
application.UseException();
application.UseHttps();
application.UseRouting();
application.UseStaticFiles();
application.UseResponseCompression();
application.UseAuthentication();
application.UseAuthorization();
application.UseEndpoints();
application.UseSpa();
}
public void ConfigureServices(IServiceCollection services)
{
services.AddSecurity();
services.AddResponseCompression();
services.AddControllersDefault();
services.AddSpa();
services.AddContext();
services.AddServices();
}
}
[ApiController]
[Route("customers")]
public class CustomerController : ControllerBase
{
private readonly ICustomerService _customerService;
public CustomerController(ICustomerService customerService)
{
_customerService = customerService;
}
[HttpPost]
public Task<IActionResult> AddAsync(CustomerModel model)
{
return _customerService.AddAsync(model).ResultAsync();
}
[HttpDelete("{id}")]
public Task<IActionResult> DeleteAsync(long id)
{
return _customerService.DeleteAsync(id).ResultAsync();
}
[HttpGet("{id}")]
public Task<IActionResult> GetAsync(long id)
{
return _customerService.GetAsync(id).ResultAsync();
}
[HttpGet("grid")]
public Task<IActionResult> GridAsync([FromQuery] GridParameters parameters)
{
return _customerService.GridAsync(parameters).ResultAsync();
}
[HttpPatch("{id}/inactivate")]
public Task InactivateAsync(long id)
{
return _customerService.InactivateAsync(id);
}
[HttpGet]
public Task<IActionResult> ListAsync()
{
return _customerService.ListAsync().ResultAsync();
}
[HttpPut("{id}")]
public Task<IActionResult> UpdateAsync(CustomerModel model)
{
return _customerService.UpdateAsync(model).ResultAsync();
}
}
public sealed class CustomerService : ICustomerService
{
private readonly ICustomerFactory _customerFactory;
private readonly ICustomerRepository _customerRepository;
private readonly IUnitOfWork _unitOfWork;
public CustomerService
(
ICustomerFactory customerFactory,
ICustomerRepository customerRepository,
IUnitOfWork unitOfWork
)
{
_customerFactory = customerFactory;
_customerRepository = customerRepository;
_unitOfWork = unitOfWork;
}
public async Task<IResult<long>> AddAsync(CustomerModel model)
{
var validation = await new AddCustomerModelValidator().ValidateAsync(model);
if (validation.Failed)
{
return Result<long>.Fail(validation.Message);
}
var customer = _customerFactory.Create(model);
await _customerRepository.AddAsync(customer);
await _unitOfWork.SaveChangesAsync();
return Result<long>.Success(customer.Id);
}
public async Task<IResult> DeleteAsync(long id)
{
await _customerRepository.DeleteAsync(id);
await _unitOfWork.SaveChangesAsync();
return Result.Success();
}
public Task<CustomerModel> GetAsync(long id)
{
return _customerRepository.GetModelAsync(id);
}
public Task<Grid<CustomerModel>> GridAsync(GridParameters parameters)
{
return _customerRepository.GridAsync(parameters);
}
public async Task InactivateAsync(long id)
{
var customer = new Customer(id);
customer.Inactivate();
await _customerRepository.InactivateAsync(customer);
await _unitOfWork.SaveChangesAsync();
}
public Task<IEnumerable<CustomerModel>> ListAsync()
{
return _customerRepository.ListModelAsync();
}
public async Task<IResult> UpdateAsync(CustomerModel model)
{
var validation = await new UpdateCustomerModelValidator().ValidateAsync(model);
if (validation.Failed)
{
return Result.Fail(validation.Message);
}
var customer = _customerFactory.Create(model);
await _customerRepository.UpdateAsync(customer.Id, customer);
await _unitOfWork.SaveChangesAsync();
return Result.Success();
}
}
public class CustomerFactory : ICustomerFactory
{
public Customer Create(CustomerModel model)
{
return new Customer
(
new Name(model.Forename, model.Surname),
new Email(model.Email)
);
}
}
public class Customer : Entity<long>
{
public Customer(long id) : base(id) { }
public Customer
(
Name name,
Email email
)
{
Name = name;
Email = email;
Activate();
}
public Name Name { get; private set; }
public Email Email { get; private set; }
public Status Status { get; private set; }
public void Activate()
{
Status = Status.Active;
}
public void Inactivate()
{
Status = Status.Inactive;
}
}
public sealed class Name : ValueObject
{
public Name(string forename, string surname)
{
Forename = forename;
Surname = surname;
}
public string Forename { get; }
public string Surname { get; }
protected override IEnumerable<object> Equals()
{
yield return Forename;
yield return Surname;
}
}
public class CustomerModel
{
public long Id { get; set; }
public string Forename { get; set; }
public string Surname { get; set; }
public string Email { get; set; }
}
public abstract class CustomerModelValidator : Validator<CustomerModel>
{
public CustomerModelValidator Id()
{
RuleFor(customer => customer.Id).NotEmpty();
return this;
}
public CustomerModelValidator Forename()
{
RuleFor(customer => customer.Forename).NotEmpty();
return this;
}
public CustomerModelValidator Surname()
{
RuleFor(customer => customer.Surname).NotEmpty();
return this;
}
public CustomerModelValidator Email()
{
RuleFor(customer => customer.Email).EmailAddress();
return this;
}
}
public sealed class AddCustomerModelValidator : CustomerModelValidator
{
public AddCustomerModelValidator() => Forename().Surname().Email();
}
public sealed class UpdateCustomerModelValidator : CustomerModelValidator
{
public UpdateCustomerModelValidator() => Id().Forename().Surname().Email();
}
public sealed class DeleteCustomerModelValidator : CustomerModelValidator
{
public DeleteCustomerModelValidator() => Id();
}
public sealed class Context : DbContext
{
public Context(DbContextOptions options) : base(options) { }
protected override void OnModelCreating(ModelBuilder builder)
{
builder.ApplyConfigurationsFromAssembly(typeof(Context).Assembly);
builder.Seed();
}
}
public sealed class CustomerConfiguration : IEntityTypeConfiguration<Customer>
{
public void Configure(EntityTypeBuilder<Customer> builder)
{
builder.ToTable("Customers", "Customer");
builder.HasKey(customer => customer.Id);
builder.Property(customer => customer.Id).ValueGeneratedOnAdd().IsRequired();
builder.Property(customer => customer.Status).IsRequired();
builder.OwnsOne(customer => customer.Name, ownsOne =>
{
ownsOne.Property(name => name.Forename).HasColumnName(nameof(Name.Forename)).HasMaxLength(100).IsRequired();
ownsOne.Property(name => name.Surname).HasColumnName(nameof(Name.Surname)).HasMaxLength(200).IsRequired();
});
builder.OwnsOne(customer => customer.Email, ownsOne =>
{
ownsOne.Property(email => email.Value).HasColumnName(nameof(User.Email)).HasMaxLength(300).IsRequired();
ownsOne.HasIndex(email => email.Value).IsUnique();
});
}
}
public sealed class CustomerRepository : EFRepository<Customer>, ICustomerRepository
{
public CustomerRepository(Context context) : base(context) { }
public Task<CustomerModel> GetModelAsync(long id)
{
return Queryable.Where(CustomerExpression.Id(id)).Select(CustomerExpression.Model).SingleOrDefaultAsync();
}
public Task<Grid<CustomerModel>> GridAsync(GridParameters parameters)
{
return Queryable.Select(CustomerExpression.Model).GridAsync(parameters);
}
public Task InactivateAsync(Customer customer)
{
return UpdatePartialAsync(customer.Id, new { customer.Status });
}
public async Task<IEnumerable<CustomerModel>> ListModelAsync()
{
return await Queryable.Select(CustomerExpression.Model).ToListAsync();
}
}
public static class CustomerExpression
{
public static Expression<Func<Customer, CustomerModel>> Model => customer => new CustomerModel
{
Id = customer.Id,
Forename = customer.Name.Forename,
Surname = customer.Name.Surname,
Email = customer.Email.Value
};
public static Expression<Func<Customer, bool>> Id(long id)
{
return customer => customer.Id == id;
}
}