All notable changes to this project will be documented in this file.
The format is based on Keep a Changelog,
and this project adheres to Semantic Versioning.
Semantic versioning will come once I hit v1, but at this point each
0.xrelease may have breaking changes as I iron out the kinks of the framework.
- Distributed lock on migration
- Extra param for GetById on query record
- hard coded config
- getbyid usage
- Update keycloak image to quay
- Update auth examples for new keycloak handling
- Upgrade pulumi and keycloak setup
- Jaeger options pointer
- Permission tests skipped for queries
- New extension methods on dbcontext for direct
GetByIdoptions - Queries use DbContext directly with new ext methods
- Consolidated options pattern object and cleaned it up
- Remove functional tests scaffolding (other than health)
- Tweak producer example
- No more repository/uow abstraction
- No more unit tests for user policy handler
- No more default permission. Must be explicitly provided
- Timespan recognition
- Hangfire config typo
- Lower/upper invariance resolution #137
- Fix db context ctor in repos
- can run a
new domainwith no entities (#135)
- Use newer header append method on paginated list endpoint
- Indentation for smart enumvalue objects
- Fix typo on smart enum error
- Can recognize
datetimeoffsetproperty type - Versioning in functional tests
- Remove unused swagger props
- Route builder includes version
- Refactor to primary ctors
- Removed deprecated (and partial) api versioning in favor of new versioning
- Added versioning to controller routes
- Updated versioning service
- Updated swagger to support versioning
- Better exception for smart enum errors
- Sponsorship request
- Bump naming conventions
- Group Otel in dependabot
-
Scaffolded projects use .NET 8
-
Bump Nuget packages
-
Bespoke
DateTimeProviderremoved in favor of the new built inTimeProvider -
BaseEntityaudit times useDateTimeOffset -
Model classes use
recordtype -
Remove
DateOnlyConverterforSqlServersince it's built into .NET 8 -
Moved
PagedListtoResourcesdirectory -
Underlying craftsman code uses .NET 8
-
Bump underlying .NET packages and remove unused packages in craftsman
-
Remove BFF commands
-
Remove BFF from examples
- Dependabot indent
- Github test actions
- Swagger config can handle nested DTO classes:
config.CustomSchemaIds(type => type.ToString().Replace("+, ".")); - Don't ignore default hangfire queue
- Smart value object scaffolding doesn't use old enum logic for entity or fakes
- New
IsLogMaskedoption for masking entity properties in logs - Dependabot scaffolding. Can be excluded with
IncludeDependabot = falseat the domain template level - Github test action scaffolding. Can be excluded with
IncludeGithubTestActions = falseat the api template level - Support for
string[]when using Postgres ValueObjectproperty scaffolding- new marker in db config
- Logging was refactored to use app settings
- Add missing HttpClientInstrumentation on OTel
SortOrderandFiltersare nullable on list dto param- Remove old and unused fluent assertion options
- Entity plural is more powerful with
Humanizer
- Email setter
- Hangfire CompatibilityLevel updated to latest
- Entity usings for data annotations (#126)
- get all is plural (#127)
- New
GetAllandJobfeatures - Hangfire integration
- Moq -> NSubsititute
- Bump base page size limit to 500
- Added global usings to the test projects
- Entity variables in tests don't start with
fakeanymore - Moved
BasePaginationParametersandExceptionsandValueObjectto api project (#124) - Package bumps and cleanup with the exception of pulumi
- Remove
Commandprop from feature scaffolding - Better property names for controllers and requests
- Can handle no global git config (#122, #72)
- Can use
.in project name (#111)
- Can respect audience prop
- Revamped Relationships to a new model
- Integration and Functional tests use the latest versions and implementations of
testcontainers - Removed
Sievein favor ofQueryKit - Removed
CanFilterandCanSortproperties. These can still be set in your code, but will no longer pollute your entities. - Use top level routing for health checks and controllers
- Update OTel to latest with newer syntax
- Remove other base specification things from generic repo
- Commands/Queries use records for input
- DTOs are records
- Additional validation exception extension method for
MustNot
- Sql server setups can handle dateonly
- Duplicate usings
- Remove bad serilog enricher and update logger for env name
MigrationHostedServicewill not conflict with dbcontext when adding entities- fix: get endpoint when name == plural
- Remove lingering mapper from empty feature
- Fix connection string prop in dev app settings
- Fix using statement for interop
- New
booloption ofUseCustomErrorHandleronApiTemplatethat defaults to a new error handler usingHellang.ProblemDetails. If you don't want a dependency, you can use the existing custom one, but the hellang one is richer and I didn't want to reinvent the wheel
- Major MediatR update to 12.x
- Features returning bool will now have no return value
- Mapster -> Mapperly
- Missing test projects won't cause failure
- Can better handle unneccessary
Ion messages - Mapper scaffolding
- Specification support in repositories
- Support for array property types
- No more manipulation DTO
- No default new guid on DTO
- Tests use builder methods
- No more autofaker for domain entity, only a builder
- Entity properties not virtualized anymore
IProjectServicerenamed toIProjectScopedService
- Removed extra db call on
Addfeature
- Fix bracket on DTOs
- DTO indentations
- Removed custom dateonly and timeonly json converters in facor of built in .net 7 options
- Crafstman uses .NET 7
- Scaffolding uses .NET 7
- Test projects updated to use XUnit
- Integration tests have better service collection scoping and now have a service collection per test. This makes service mocking possible without clashing with other tests
- RMQ container setup for integration and functional tests
- Use
IConfigurationinstead of customEnvironmentService - Move parameters from
launchsettingtoappsettings UserPolicyHandlerrefactor- Unit tests no longer have redundant directory
- Added optional unit test to check protection of all endpoints
- Tweaked unit test assembly helper utility
- Bump nuget packages
- No more event handler test scaffolding
- New
MigrationHostedServiceto automatically apply migrations when starting your project - New
EnvironmentServicefor easy access to environment variables
- Removed
Validatorsand in favor of direct validation in entiy factories and methods. - Additional helper methods on
ValidationException - Additional OTel config
- Cleaner functional test helpers
- Bump HeimGuard
- Update
UserPolicyHandlerto include aHasPermission()implementation - Json attribute refactor
- Simplify swagger comments
- RMQ Password var
- Email guards for whitespace
- Minor spacing items
- Resolves mismatch on name for GET record endpoint
- typo on validation for exchange #100 (thanks @MeladKamari)
- potential process clashing resolution
-
Test Utility Mapper for Unit Tests
-
New
UserandUserRoleentities when using auth-
Your auth server will still store users, but it's only role is now AuthN. The basic premise is still the same though
-
Authorizeattributes on your controllers will guard for application level access by checking if the request has a valid JWT from your auth server -
Feature and business level access can be handled in your MediatR handlers with HeimGuard, for example:
await _heimGuard.HasPermissionAsync("RecipesFullAccess") ? Ok() : Forbidden(); // OR... await _heimGuard.MustHavePermission<ForbiddenAccessException>(Permissions.CanAddRecipe);
-
As usual, this will use the
UserPolicyHandlerlogic to check if a user meets the given criteria. The implementation of HeimGuard'sUserPolicyHandlerin the scaffolding will still check the token to see who you are Authenticated as, but will get the user's roles based on the configuration for that authenticated user within the boundary using the new concepts and logic described below.
-
-
Your auth server will still store users and their metadata that we want for AuthN and the new
Userentity will capture user metadata to be easily accessed in your app. Say you want to send back a list of recipes and show who created them, this gives you that info without having to reach out to your auth server- Users in your boundary will key to users in your auth server using the
Identifierproperty, which should generally be populated with theNameIdentifieron your token. - Eventual consistancy TBD
- Redis cache TBD?
- Users in your boundary will key to users in your auth server using the
-
Roles will also not be dictated by your auth server anymore and will be stored in the context of your boundary (not the shared kernel anymore, though you could move them there if it makes sense for your setup). They are still restricted with a smart enum to
Super AdminandUser, but feel free to modify or extend these as you wish. -
When starting your api for the first time, you will not have any
UserorUserRoleentries. To add an initial root user, you just need to authenticate (e.g. swagger) and hit a protected endpoint. This is because, by default, theUserPolicyHandlerwill now check if you have any users and, if not, add the requesting user as a root user withSuper Adminaccess.- You are welcome to change this logic if you wish, it is just an easy default to start with.
-
If you have multiple boundaries, you can make the call on how you manage things. By default each boundary will store its own set of users and roles. This might be useful if say you want to manage users in your billing boundary separate from your inventory boundary, but you may want to combine them too. Feel free to consolidate this as you wish if desired.
-
Users can be managed at the
/usersendpoints. This includes adding and removing their roles as user roles have no standalone meaning without a user. -
Updated functional tests to properly handle the token setup
-
-
New
RolesControllerandPermissionsControllerwhen using auth -
New
EmailValue Object. Used onUserif you want to see it in use -
Functional tests use a docker db
-
Simple fake
Builderscaffold for each entity -
Add Date Time Provider
-
Fakers for
Address -
Various new commands for working with the NextJS Template, but it's still in progress so not documenting this much yet.
- Queries and Commands now just have a request class of
QueryorCommandrespectively - Change keycloak image from
jbosstosleighzyfor Apple Silicon support - Unit of Work return a
Task<int> - Delete and Update commands return response based on UoW response
- Extra Ef Tools package for SqlServer
DevelopmentCors hasAllowCredentials- Better DTO indentation
- Auditable fields are sortable and filterable
GetListfeature defaults to-CreatedOnsort order if none is given- Better
DateOnlyserialization support Newtonsoft->System.Text.JsonUpdatefeature calls update in repository- Added
8632toNoWarnlist in API projects --[CS8632] The annotation for nullable reference types should only be used in code within a '#nullable' annotations context. - Better error handling message for incorrect directory with boundary based commands
- New props on complex example
- Jaeger will get random free ports exposed for HOST in
docker-compose - More flexible
MonetaryAmountVO - Simplified mapping
- Default permissions will be the same for both GETS and use the entity plural ((e.g.
CanReadRecipes). All permissions will use the entity plural. - Integration and unit tests have explicit property assertions
- Query list uses
AsNoTracking - No more fake read DTO
- Allow custom schema id's on
AddSwaggerGen - Bump
Respawnto v6 with simplifiedTestFixturesetup - Bump all non-breaking packages to latest
- Include
Moqpackage in shared test helper - Entity
Updatemethods return entity instance - Deprecate
add bffandadd bff entitycommands
- Db configs are added for all entities
- Patch endpoint scaffolding
- Bad Test Fixture with SqlServer (fixes #95)
- Revert parallel functional tests causing unpredictable failures
- Revert parallel functional tests causing unpredictable failures
- Batch add returns list
- Mapster service registration uses assembly (and is escaped for web app factory in functional tests)
- Fix typo on BFF redirect example
- Remove offline BFF scope
- Fix example redirect for BFF
- Unit test for filtering and sorting will exclude smart enum properties
-
ServiceCollectionServiceExtensionsin IntegrationTextFixturefor easier service mocking withReplaceServiceWithSingletonMock -
Add default database configuration abstraction for each entity along with db context usage. Includes value object examples
-
Json serialization extension to better handle
DateOnlyandTimeOnlyin swagger -
Common value object scaffolding (
Address,Percent,MonetaryAmount)- Scaffolds Dtos for
Address - Scaffolds mappers
- Scaffolds Dtos for
-
Port customization for RMQ broker and ui
-
Added
ForbiddenAccessExceptionandInvalidSmartEnumPropertyNamehandling toErrorHandlerFilterAttribute
-
Removed faulty producer assertion in integration tests for better performance
-
Remove
.AddFluentValidation(cfg => {{ cfg.AutomaticValidationEnabled = false; }});from registration as validation should be happening directly at domain level -
Integration and functional tests parallelized
-
Integration tests updated to use
DotNet.Testcontainersfor simpler docker db setup -
Moved pagination, filter, and sort testing to unit tests
-
UserPolicyHandleruses RolePermission repo and handles finding no roles -
Value Object cleanup
-
Better migration warning on integration tests
-
Move most
UserPolicyHandlertests to unit tests -
Move
CurrentUserServiceTeststest dir -
Update
LocalConfigtoConstsfor all constants in app (other than permissions and roles) -
Update
AutoBogustoAutobogusLifesupportto support .NET 6 and UTC -
RequireHttpsMetadataoff in dev for BFF and boundaries -
SuperAdmintext toSuper Admin -
UserPolicyHandlercan accommodate realm roles for keycloak -
Moved permission guards to each feature handler (still using
Authorizeattribute on controllers for authenication check). -
Simplified return on batch add feature
-
CLI updates for auth server and bff scaffolds
-
Migrated mapper to use
Mapsterinstead ofAutomapperfor easier value object usage and better performance. To migrate your project-
Update your mappers to look like this (almost identicial to
Automapper:public class AuthorMappings : IRegister { public void Register(TypeAdapterConfig config) { config.NewConfig<AuthorDto, Author>() .TwoWays(); config.NewConfig<AuthorForCreationDto, Author>() .TwoWays(); config.NewConfig<AuthorForUpdateDto, Author>() .TwoWays(); } }
-
Manaully do mapping in your entity factories (due to recurssive nature of the dtos and mapping happining in the factories)
-
Update usings to
using MapsterMapper; -
Your
ProjectTomappings should look like this:var dtoCollection = appliedCollection .ProjectToType<AuthorDto>();
and import:
using MapsterMapper; using Mapster; -
Your registration should now be:
var config = TypeAdapterConfig.GlobalSettings; builder.Services.AddSingleton(config); builder.Services.AddScoped<IMapper, ServiceMapper>();
-
- Domain events captured and cleared in db context before running to prevent dups
- FKs use proper entity name for property (#90)
-
Open Telemetry and Jaeger tracing support
-
ValueObjectclass scaffolding toSharedKernel -
Domain Event support for all
BaseEntities. Automatically scaffolded for theCreateandUpdatemethods that call messages in a newDomainEventsdirectory under that entity. It works by adding messages to a newDomainEventsprop on each entity and publishing all messages on an EF save using MediatR.- Unit tests are also scaffolded
- To consume a message, you can use a MediatR
INotificationHandlerlike normal. For example:
namespace RecipeManagement.Domain.Authors.Features; using System.Reflection.Metadata; using DomainEvents; using MediatR; public class LogAuthor : INotificationHandler<AuthorAdded> { private readonly ILogger<LogAuthor> _logger; public LogAuthor(ILogger<LogAuthor> logger) { _logger = logger; } public async Task Handle(AuthorAdded notification, CancellationToken cancellationToken) { _logger.LogInformation("Author added: {AuthorName}", notification.Author.Name); } }
-
Smart enum property support. Just add a property and give it a list of options using the
SmartNamesproperty and you're good to go. They will all be strings with smart enum backing. Attributes like filtering and sorting are supportedEntities: - Name: Recipe Features: #... Properties: - Name: Title Type: string CanFilter: true CanSort: true - Name: Visibility SmartNames: - Public - Friends Only - Private CanFilter: true CanSort: true
-
CLI commands no longer use
:and use the more traditional space delimiter. For exmaple,craftsman new example -
Moved from old Startup model to new Program only for .NET 6. Includes updating test projects and service registrations for compatible setup
-
All features will now use a repository instead of using dbcontext directly.
-
Batch does not require a dbset name anymore, but does require a plural entity name for the FK if using one. Functionally, this is the same as before just with a new name for the prop:
ParentEntityPlural -
Entity props now
virtualby default with aprotectedconstructor for mocking in unit tests. This is mostly for foreign entities (since we don't have EF to populate our foreign entities in unit tests), but in order to have our mocks accurately reflect all our props, we need to make them virtual. For example:private static Author GetMockAuthor() { // my fake data base generated with AutoBogus under the hood var fakeRecipe = FakeRecipe.Generate(); var forCreation = new FakeAuthorForCreationDto().Generate(); forCreation.RecipeId = fakeRecipe.Id; // FakeItEasy making entity assignment easy in lieu of EF var fakeAuthor = A.Fake<Author>(x => x.Wrapping(Author.Create(forCreation))); A.CallTo(() => fakeAuthor.Recipe) .Returns(fakeRecipe); return fakeAuthor; }
💡 DOCS NOTE: this is default, but depending on your domain ops, you might have a domain method that does the action for you, in which case, you won't want these to be virtual. For example, Recipe might have
SetAuthororAddIngredientmethods and you wouldn't want those FKs to be virtual -
Add
FakeItEasyandFakeItEasyAnalyzerto unit test project -
Colocate DTOs with entities by default. Can be moved to shared kernel when needed
-
Default db provider set to postgres
-
Removed
?'s on strings inBaseEntity -
Update integration test consumer registration to non-obsolete model
-
Testing cleanup
-
Performance optimizations for integration and functional tests
- Integration tests will only wipe the db using checkpoint at the begining of the entire fixture run, not after each test. This affects how assersions can and should be made
- Unit tests have parallelization
-
Removed sort integration tests
-
Messages get a class for better typing as well as an interface.
-
Add postman client to example scaffolding
-
Add support for offline access and client secret requirement settings for client credentials clients
-
Automatically add a
SuperAdminrole claim to machines -
Updated services to use Newtonsoft to keep support for patchdocs.
System.Text.Jsonis still available if patchdocs aren't needed for a project. -
Integration Test
TestBasehas global autobogus settings -
Permission and role validations are not case sensitive
-
Added
client_roletoUserPolicyHandlerrole check to accomodate machines with a new scaffolded test.-
Machine example:
new Client { ClientId = "recipe_management.postman", ClientName = "RecipeManagement Postman", ClientSecrets = { new Secret("secret".Sha256()) }, AllowedGrantTypes = GrantTypes.ClientCredentials, AllowOfflineAccess = true, RequireClientSecret = true, Claims = new List<ClientClaim>() { new(JwtClaimTypes.Role, "SuperAdmin") }, RedirectUris = { "https://oauth.pstmn.io/v1/callback" }, AllowedScopes = { "openid", "profile", "role", "recipe_management" } },
-
- Sql server connection strings to docker will trust the cert:
TrustServerCertificate=True;
- Nav link quote issue in BFF
- Lingering recipe route in BFF
- Added
useQueryimport to GET api in BFF
- Bff conditional check
- Missing dependencies (#73)
- LaunchSettings Casing (#72)
-
A
Dockerfileand.dockerignorewill be added to each bounded context automatically (except BFFs) -
A
docker-compose.yamlwill be added to your solution root by default for local development-
Just run
docker-compose up --buildto spin up your databases (and RMQ if needed) -
Then set an env and apply migrations. For a postgres example:
-
env
Powershell
$Env:ASPNETCORE_ENVIRONMENT = "anything"
Bash
export ASPNETCORE_ENVIRONMENT=anything -
dotnet ef database updateordotnet ef database update --connection "Host=localhost;Port=3125;Database=dev_recipemanagement;Username=postgres;Password=postgres"
-
-
SAwill always be default user for sqlserver so it can work properly -
If no ports are given for api or db, they'll be auto assigned a free port on your machine
-
New
johnuser to auth server (with no role) -
Login hints for auth server
-
Minor helper override on fake generators
-
New
add:bffcommand to add a bff to your solution -
New
add:bffentitycommand to add entities to your bff -
Basic unit test scaffolding for create and update methods
-
Unit tests for updating rolepermissions
-
-
Fixed bug in
Addfeature that had a chained action afterProjectTo. They will now be filtered like so:var {entityNameLowercase}List = await _db.{entity.Plural} .AsNoTracking() .FirstOrDefaultAsync({entity.Lambda} => {entity.Lambda}.{primaryKeyPropName} == {entityNameLowercase}.{primaryKeyPropName}, cancellationToken); return _mapper.Map<{readDto}>({entityNameLowercase}List);
-
Removed
ProjectTofromGetRecordfeature in favor of direct mapper. -
Initial commit will use system git user and email as author. Courtesy of @sshquack
-
IdonBaseEntityis sortable and filterable by default -
Minor logging updates for better json formatting and more information in prod
-
GET record, PUT, and DELETE all have typed ids (e.g.
{id:guid}) on their controllers -
Developmentenvironment uses a connection string to an actual database now, instead of an in memory db. This can easily be spun up with adocker-composefor local development -
Environment is now a singular object that will take in values for local environment variables for development and placed in launch settings and your docker compose. When deploying to other environments, you will use these same environment variables, but pass the appropriate value for that env.
- Updated auth properties to be under an
AuthSettingsobject like we do for broker settings - Removed
ConnectionStringfrom env. Docker connection will be added automatically in launch settings - Removed
EnvironmentNameas it will always beDevelopment - Updated examples to handle new environment setup
- Updated auth properties to be under an
-
Removed unused mapper from delete feature
-
Updated entity property definition with optional
ColumnTypeattribute -
Fixed a bug that broke registration of the bus in development env
-
Logger modifications
-
Cleanup batch add feature
-
Updated
Createdresponse for batch add -
Use entity plural directory for tests
-
MassTransit bumped to v8
-
Bumped Nuget packages to latest
-
Updated checkpoint in test fixture for major version bump
- Removed seeders
- Dbcontext for userpolicyhandler uses template (#70)
- Patch cancellation token added in feature
- Typo in role permission validation
add:entitycommand will now use the correct solution folder
- Entities will now have private setters, a static
Createmethod and anUpdatemethod to promote a DDD workflow- This includes base entity and the dbcontext setters for the auditable fields
- A huge permissions overhaul
SuperAdminrole is added by default and will automatically be given all permissions.- A
Userrole with also be created. This role will not have any permissions assigned to it by default and can be removed if you would like as long as you add another role in addition toSuperAdmin. This will allow for integration tests of theUserPolicyHandlerto be fully scoped. - A permission will be added for each feature with a default name of
CanFEATURENAME, but can be overridden if desired. - Role-permission mappings outside of this need tobe done manually
- You will be responsible for managing permissions outside of super admin
- Policies are no longer part of the craftsman api, but a single policy will be added with each api boundary to swagger to dictate access to that boundary. It has a default of a snake case of
ProjectName, but can be overridden- If using the built in auth server, this will be added for you. if not, make sure it mataches an appropriate scope in your auth server for this api boundary
- Features now have a prop for
IsProtectedthat, if true, will add an authorization attribute to your endpoint with thePolicyNameand add tests that check for access - Integration tests for
UserPolicyHandler - Unit tests for
RolePermission Policiesprop removed from Feature- Added
SetUserRoleandSetUserRolesmethods to integreation tests'TestFixturefor easy role management in integration tests - Functional tests auth helper method now sets
roleinstead ofscope - Alice is a
SuperAdminand bob is aUser
- Added a
register:producercommand with CLI prompt - Added
UseSoftDeleteproperty to theApiTemplatewhich is set to true. When adding an entity after the fact, Craftsman will automatically detect whether or not your project is using soft deletion by checking base entity for the appropriate property. - Added a
SharedKernelproject at the root to capture DTOs, exceptions, and roles (if using auth) - Added new
Complexexample fornew:example
-
Updated FK and basic examples to have more features on the entities
-
Updated tests to work with new private entity workflow
-
CurrentUserServer has a method to get the User from the ClaimsPrincipal
-
Swagger question removed from
add:featureas it wasn't being used. Will be set to true. -
Removed unused
UnauthorizedandForbiddenMediatR filters -
Test Fixture updated to better handle MassTransit
-
_providerwill always be injected, even when not using MassTransit -
The end of the fixture has been slightly updated to look like this:
// MassTransit Harness Setup -- Do Not Delete Comment _provider = services.BuildServiceProvider(); _scopeFactory = _provider.GetService<IServiceScopeFactory>(); // MassTransit Start Setup -- Do Not Delete Comment
-
Added several helper methods to the test fixture
-
Updated the consumer test to use the helper methods
-
Producer doesn't take in placeholder props
-
Producer test generated
-
-
Minor update to naming of producer in the bus example
-
Default exchange type now
Fanoutfor producer and consumer -
Added optional (but recommended)
DomainDirectoryprop to producers and consumers. This will move them from theEventHandlersdirectory and keep them directly with features for better colocation. -
Updated the 1:1 relationships to use proper scaffolding.
-
Updated the FK example to show proper 1:1 relationship
-
Entities with a Guid prop will no longer have a default of
Guid.NewGuid() -
Updated default library from NewtonSoft.Json to System.text.Json (#52)
-
Took audit fields off of DTO
-
Bumped LibGit2Sharp to preview for M1 compatibility
- Batch endpoint route updated to
batchalong with a functional testing route - Batch add using update to controller
IHttpContextAccessorfixted to a singleton in integration tests'TestFixture- Can enter null for
add:featurebatch list options where needed - Minor formatting fix for indentation in producers and consumers
- Removed extra exception using from patch integration test
- Fixed docker image for integration tests running on macOS + M1 chip (#53)
- Updated Craftsman to .NET 6
- Fixed bulk add file and variable names
- Removed unused response class and endpoint reference
- Removed
Consumesfrom Get list
-
Added
BaseEntitythat all entities will inherit from.- Contains a
GuidofIdmarked as the primary key - Contains
CreatedOn,CreatedBy,LastModifiedOn, andLastModifiedByproperties
- Contains a
-
Added a
CurrentUserServiceto add a user to theCreatedByandLastModifiedByproperties if a user is found. Built into db context -
Added built in features to the
add:featurecommand -
New
AddListByFkoption for theadd:featurecommand andFeatureproperty of an entity. -
New
craftsman exampleorcraftsman new:examplecommand to create an example project with a prompted workflow to select.Basic,WithAuth,AuthServer,WithBus -
Added
.DS_Storeand.envto gitignore -
Added Consumer test
-
Added
providerto test fixture when adding a bus -
Added a mock
IPublishEndpointservice toTestFixturewhen using MassTransit- update docs that mediatr handler tests aren't broken when pubilshing anymore
-
New policies added to swagger on
add:entityscaffolding -
New
add:authservercommand as well as anAuthServeroption when creating a domain- No consent support (yet)
-
Added helper
GetServicemethod toTestFixture -
Added Creation and Update Validators back to scaffolding. Easy enough to delete if you aren't using them
-
Added a
NamingConventionproperty to the db template-
options are:
Class SnakeCase LowerCase CamelCase UpperCase
-
-
Updated to .NET 6
-
Updated nuget packages
- Inlcudes a major release of Fluent Assertions that required updates to:
- Functional test assertions (use
HttpStatusCode.XXX) - Integration tests
TestBaseupdate for postgresto include1.Seconds() - Integration test updates to
await act.Should().ThrowAsync<where appropriate
- Functional test assertions (use
- Inlcudes a major release of Fluent Assertions that required updates to:
-
Moved Policies to Feature
-
There is no more primary key property. A Guid with a name
Idwill be inherited by all entities. -
Docker utilities for integration test refactored to use Fluent Docker wherever possible for better readability. Some enhancements were made as well (e.g. better container/volume naming, proper volume mounting).
-
Removed
ErrorHandlerMiddlewareand replaced it withErrorHandlerFilterAttribute- Updated built in Exceptions
- Updated thrown errors and associated tests in the features
-
Cleaned up test names
-
Modified CORS util to take in env
-
Added
Secretback to Environment options -
Added local config utilities for testing environments
-
Remove
UseInMemoryDbapp setting in favor of environment specific checks -
Remove
UseInMemoryBusapp setting in favor of environment specific checks -
Using statement shortening for reset function in test fixture
-
Consolidated multiple environments startups to one startup file
-
Update logging registration in
Program.csto no longer rely onappsettings -
Updated FK support to better API
-
Moved env config from appsettings to environment variables
-
Production env no longer added by default
-
Features now include missing cancellation tokens as well as
AsNoTrackingproperties -
Removed automatic fluent validation to allow more control in domain operations. For example:
var validator = new RecipeForCreationDtoValidator(); validator.ValidateAndThrow(recipeForCreationDto);
- this can be turned back on in
WebApiServiceExtensionby updating.AddFluentValidation(cfg => { cfg.AutomaticValidationEnabled = false; });
- this can be turned back on in
- Removed the broken patch validation from command
- No more
409produced response annotation on POST - Descending sort tests now actually test desc instead of mirroring asc
- Removed error handler comments in controller
- Empty controller no longer added when no features present (fixes #40)
- Messages project will be properly referenced when using a bus
- Swagger policies won't get duplicates
- Better variable name on delete integ test
- Removed extra parentheses on endpoint names
- Added an
add:featurecommand (also works withnew:feature) - Non-nullable guids will now have a default of
Guid.NewGuid()unless otherwise specified - Added XML docs to release in csproj
- Added a new
featuresoption to entities to allow for granular feature control. The accepted values areAdHoc,GetRecord,GetList,DeleteRecord,UpdateRecord,PatchRecord,AddRecord, andCreateRecord(same asAddRecordbut available as an alias in case you can't remember!)
- Instead of 3 BC projects (Core, Infra, Api) there will now be one. This helps with colocation and does force premature optimization for something like a model library and a heavily separated infra project.
- Features dir changed to Domain with a features directory inside of it
- Entities live on their respective entity folder in the domain with their feature
- Removed the
webapisuffix on the api project - Core directories moved to web api
Contextsdir changed toDatabases
- Removed save successful checks on add and update commands
- Added better naming to PUT command variables
- Guid PKs will no longer be added to the creation DTO
- POST commands will no longer have a conflict check since you can't add a PK anymore
- A conflict integration test will not be added anymore
NoKeyGeneratedcommands are no longer in dbcontext- Controller url all lowercase
- Swagger comments on by default
- 409 no longer shown on swagegr comments for POST
- No more save successful check on delete command
- Seeders have a sub
DummyDatadirectory - Sieve service registration moved to webapi service class
SolutionNametoProjectName
- Seeder indentation in startup fixed
- PUT commands will no longer throw 500 when entity is not modified (#31)
- Route indentation fixed
- Removed annoying comments from features
- Fixed test name for basic gets in functional tests
- Seeders in startup will newline when there are multiple entities
- Unicode now onlyenforced on windows (for better emoji support)
- Fixed
isRequiredproperty - Command prop for
Updatecommand has proper casing
- Removed the
add:propertycli command
- Added SpectreConsole for a better CLI experience
- Added
add:buscommand - Added
add:consumerandadd:producercommands for direct, topic, and fanout messages - Added Bus, Producers, and Consumers props to BC template
- Added Messages to the domain template
- Added
add:messagecommand - Added conflict test for add command when using a guid
- Added tests for command and query exceptions
- Didn't do them in the controller as that is not the dependency. Can test the exceptions causing the correct httpstatus code in the exception separately
- Changed the
new:domainoutput to a single solution with directories for each bounded context for easier management - Changed the seeder regions in
StartupDevelopment.csto comments - Changed the Logger settings in
Program.cs - Updated
add:entityandadd:propto now be called from the BC directory - Updated
ProducesResponseTypein controllers to genericResponsetype where applicable - Updated App Registrations to separate files
- Updated Service Registrations to separate files
- Updated entity name and entity prop names first letter to always be capitalized
- Better namespacing for features in controllers using static classes for features
- Updated functional test to pass without conflict
- Updated nuget packages
- Updated
Program.csto async - Changed migrations to happen after all bounded contexts are added
- Removed the custom fluent validation boilderplate from the add, update, patch, and ad hoc commands
- Removed verbosity option from commands due to simplified spectre console
- Removed legacy comment for include statement marker
- Removed BC readme and updated sln readme
- Fixed double error messages
- Fixed incorrect help message for
new:domaincommand (#24) - Fixed help text on
listcommand - Fixed controllers to inherit from
ControllerBaseinstead ofController. fixes (#26) - Fixed extra space in the class in the dto classes when not abstract and trailing new line
- If using a guid for a PK, it will be added to the creation dto (not manipulation or update) -- fixes #28
- Guid PKs will have a default value of
Guid.NewGuid()in their creation dto
- Guid PKs will have a default value of
- PK already exists guard will be added for GUIDs and will be performed when adding a new entity and throw a 409 conflict via a new conflict exception if a record already exists with that guid. -- fixes #29
- Fixed issue where POST would throw 500 when primary key != EntityNameId (e.g. PK of ReportId would break for an entity of ReportRequest) - fixes #30
- Fixed default value for strings on entities to use quotes
- Fixed missing exception handling on
add:bccommand
- Fixed autogen identity
- Lingering dbcontext and dbname mix up
- Db context will now be used instead of name in api scaffolding
- Test utilities in functional tests will now be added when not using auth
- Added a new vertical slice architecture
- Projects have been consolidated and will now have a prefix of the solution name before each project type. For example, the api project with a solution name of
orderingisordering.webapi
- Projects have been consolidated and will now have a prefix of the solution name before each project type. For example, the api project with a solution name of
- Added a
new:domaincommand to create a ddd based domain with various bounded contexts inside of it. this is recommended for long term maintainability - Added the
add:bccommand which will add a new bounded context to your ddd project - Testing completely rebuilt from the ground up. Now has unit, integration, and functional tests. Integration and Functional Tests can spin up their own docker db on their own to run against a real database.
- Moved 'addGit' property from the api template to the domain template
- Added a
versionor-vcommand to get the craftsman version - Added an initial db migration to run automatically on project creation
- Added verbosity option to new domain and add bc
- Added a version checker to make sure you are alerted if out of date
- Added an
add:propalias - Added explicit add entity template with auth policies available to add
- Added a production app settings by default
- Changed the startup marker for dynamic services to a comment instead of a region
- Readme will now be generated in the domain directory
- Updated environment to have production as a reserved word instead of startup to be consistent with dotnet process
- Will use startup and appsettings.production
- Normal appsettings will be empty, but have all the config keys required to make migrations and builds possible
- Updated the default Cors policy name
- Consolidated launchsettings to have the same setup for all environments as it is just a setting for the IDE and not used for the release package
- Removed
microcommand to consolidate and reduce complexity. if you still want to build a microservice, you can build a domain and deploy each bounded context as a microservice- Gateways were removed and may be added back with better integration in a future release
- Removed the
new:apicommand to focus on the DDD driven style - Removed
ClientSecretto promote code+PKCE flow
- Existing auth policies will now be skipped for registration when adding a new entity
- Fixed documented response codes for delete, put, and patch from 201 > 204
- Foreign keys will no longer be automatically included in features or DTOs for better performance (#2)
- Fixed issue where xml comments would throw an error on non windows machines (#16)
- fixed bug when creating an api with auth settings
- New
add:microcommand that scaffolds a new microservice template as well as an ocelot gateway - New
portproperty on thenew:apitemplate to let you customize and api or microservice port on localhost - Added
httpsdefault on local - Added additional startup middleware
UseHstsfor non dev environmentsUseHttpsRedirectionwith notes on even more secure options
- New
AuthorizationSettingsobject and authorization based properties on the environments for thenew:apiandnew:microcommands - Added new
GetEntityandDeleteEntityintegration tests with and without auth - Added 401/403 response types to swagger comments when using auth
- Added auth to swagger setup
- note that secret is currently stored in appsettings
- Auth added to integration tests when required
- The
CurrentStartIndexcalculation in thePagedListclass was broken and now has a new calculation. - Added null conditional operator (
?.) to certain tests before.Datato make them fail more gracefully- Get{entity.Plural}_ReturnsSuccessCodeAndResourceWithAccurateFields()
- Put{entity.Name}ReturnsBodyAndFieldsWereSuccessfullyUpdated
- Cleaned up
WebApplicationFactoryto remove deprecated services. - Removed
[Collection(""Sequential"")]from repo tests
- Internal tests now passing
- refactored out template drilling
- removed old auth debt from earlier alpha
- Removed the dependency on the foundation api template!
- Fixed
UseEnvironmentin WebAppFactory to useDevelopment - Fixed integration tests to use the new
Responsewrapper - Updated pagination tests to have proper keys due to default sort order possibly breaking these tests
- XML comment info is now properly added to
WebApi.csprojand the Swagger config - Extra line will no longer be added when no swagger contact url is provided
- Repository now sets default sort order for proper sql compatibility in lists (issue #9)
-
Added table name and schema properties to entity
-
Added column name attribute to entity properties
-
Added Serilog by default in all new projects. This includes Console and Seq logging by default in
Development. For non-Development environments, you'll need to add whatever logging you're interested in to their respective app-settings projects. There are just too many options to create a whole API on top of Serilog. -
Updated swagger implementation from nswag
-
Added Consumes and Produces headers to the controller endpoints
-
Added an option to manage additional swagger settings to your API endpoints. This will be turned off by default for now as dealing the with xml docs path is potentially burdensome, but will add a lot of valuable details for users consuming your API. If you are looking to add additional XML details, this is highly recommended.
-
Added a custom Response Wrapper to the GET and POST endpoints
- Fixed launch settings to have null environment variable for Startup (Production). If you'd like to change this, be sure to update the appsetting lookup in
Program.cs - Fixed POST endpoint that was lacking a
[FromBody]marker BasePaginationParameterswill now haveMaxPageSizeeandDefaultPageSizeset asinternalproperties so they don't show up in swagger. These can be overridden in the distinct entity classes like so:internal override int MaxPageSize { get; } = 30;- Fixed controllers to be able to handle a name and plural with the same value (e.g. Buffalo)
- Added
add:entitiesalias for theadd:entitycommand - Can now add Guid or other non-integer primary key
- Fixed bug where postgres library was getting added every time
- Seeder was not getting added to
StartupDevelopmentwhen usingadd:entitycommand
- Async method in controller POST wasn't awaited
- Default
Startup.csclass can now be configured using the reservedStartupkeyword
- Fixed
craftsman add:property -hhelp text - The appsettings connection string will now escape backslash
- Foreign key using statement will now be dynamic on DTOs
- Added
new:webapialias that acts the same asnew:api
- Fixed
craftsman add:property -hto point to the correct help page
- Updated the API to run on NET 5.0
- Pagination metadata enhancements on PagedList that is returned in GET list endpoint will now include more metadata for the current page including the current size as well as the start and end indices. I also removed the Next and Previous Page URI links to reduce complexity.
- Updated all controller calls to be asynchronous, including the get list
- Saves updated to be asynchronous
- One major capability I want to add into this is a good basis for auth generation. I've started to build this out, but it could (and very likely will) change drastically. With that said, I left it in as an alpha feature in case anyone is interested in trying it.
- Add Entity bug in repository fixed
- Fixed some builder options when not using auth