Fkl provide a two-way binding between design-implementation.
Indesign:
TBD:
DDD Syntax:
| decl | | usage |
|--------------------|--------|------------------------------------------------------------------------------|
| contextmapdecl | : | [ 'ContextMap' ] [ ID ] '{' (contextnodedecl | contextnoderel ) '}' |
| | | | attlist |
| contextnodedecl | : | ['context'] [ID] |
| contextnoderel | : | [ ID ] relsymbol [ ID ] |
| relsymbol | : | ('->' | '<-' | '<->') |
| contextdecl | : | [ 'Context' ] [ ID ] '{' aggregatelist? '}' |
| | | | attlist |
| attlist | : | attritem+ |
| attritem | : | ([ ID ] '=' [ value ] ','?)* ';'? |
| | | | ([ ID ] ':' [ value ] ','?)* ';'? |
| | | | [ ID ] ([ value, ',' ])* ';'? |
| aggregatedecl | : | [ 'Aggregate' ] [ ID ] '{' entitylist '}' |
| | | | attlist |
| entitydecl | : | [ 'Entity' ] [ ID ] '{' valueobjectlist '}' |
| | | | attlist |
| valueobjectdecl | : | [ 'ValueObject' ] [ ID ] '{' valuelist '}' |
| | | | att_list |
```kotlin ContextMap Ticket {}
Context ShoppingCarContext {}
// render wtih UML styled? Aggregate Cart { """ inline doc sample just-demo for test """ display = "Cart"; DomainEvent CartCreated, CartItemAdded, CartItemRemoved, CartItemQuantityChanged, CartCheckedOut; DomainEvent CartItemQuantityChanged;
// Concept or UML like ? // can be inside or outside of the Aggregate Entity Cart { // it's to many, can change in different way. ValueObject CartId ValueObject CartStatus ValueObject CartItem ValueObject CartItemQuantity ValueObject CartItemPrice ValueObject CartItemTotal ValueObject CartTotal } }
// global detail for Cart. Entity Cart {}
DomainLanguage(sourceSet = TicketLang) ```
Subscribe / Publish / Event / Flow
kotlin
default impl {
techstack {
language = "Kotlin"
framework = "Spring"
message = "Kafka"
dao = "JPA"
cache = "Redis"
search = "ElasticSearch"
}
}
compare to given-when-then
.
```kotlin impl CinemaCreated { endpoint { POST "${uri}/post"; contentType: application/json; authorization: Basic {{username}} {{password}}; request { "id": {{$uuid}}, "price": {{$randomInt}}, "ts": {{$timestamp}}, "value": "content" } expect { "status": 200 "data": { "id": {{$uuid}}, "price": {{$randomInt}}, "ts": {{$timestamp}}, "value": "content" } }; }
// created in ApplicationService flow { via UserRepository::getUserById receive user: User // send "book.created" to Kafka via UserRepository::save with parameter() ?? // or via UserRepository::save(user: User) receive user: User; // message queue via MessageQueue send CinemaCreated to "CinemaCreated" // http request via HTTP::post (with parameter())? to "${uri}/post" // grpc Greeter via GRPC::Greeter send CinemaCreated to "CinemaCreated" // map filter choose(isUserValid) { is true => { // do something } is false => { // do something } } }
validate { pre {
}
post {
}
} } ```
```java // getuserbyuserid from JPA public User getUserByUserId(String userId) { return userRepository.findByUserId(userId); }
// getuserbyuserid from MyBatis public User getUserByUserId(String userId) { return userMapper.getUserByUserId(userId); } ```
```kotlin impl CinemaCreated { """bla bla""" // or binding to ? // binding: aggregate().
// location with modules // default to "root" or ":" target: "${DomainObject}:com.example.book" // ? qualified: "${moduleName}:com.example.book",
endpoint { // message map notication ? // RPC API // HTTP API ? host: "http://localhost:8080" url: "/api/v1/books"
method: "POST"
test {
host: ""
token: ""
}
}
entity: Book input CreateBookRequest { struct { "title" : "string", "author" : "string", "price" : "number" } example { "title" : "The Lord of the Rings", "author" : "J.R.R. Tolkien", "price" : 29.99 } validate { // title.length > 10 ? title { required { min: 3, max: 10 } pattern { regex: "^[a-zA-Z0-9]+$" } range { min: 1, max: 100 } } } }
middle { via User get/update/delete/post userId via Kafka send "book.created" // send "book.created" to Kafka }
output CreateBookResponse { struct { "id" : "number" } validate { } }
// contract-based development output CreateBookResponse(xpath=""); input CreateBookResponse(sourceSet="PetSwagger" location=""); } ```
binding source code to Context Map
``` Context Ticket {
}
binding Ticket { language: "Kotlin", layered: DDDLayered, qualified: "${moduleName}:se.citerus.dddsample.domain.model", // equals moduleName: "domain" package: "se.citerus.dddsample.domain.model" } ```
SourceSet is design for support 3rd-party dsl, like PlantUML, Swagger.yaml
| decl | | usage | |------------------------|--------|-----------------------------------------| | sourcesetdecl | : | simplesourcesetdecl | | | | | spacesourcesetdecl | | spacesourcesetdecl | : | [ 'SourceSet' ] [ ID ] '{' attlist '}' | | simplesourcesetdecl | : | [ 'SourceSet' ] [ ID ] '(' attlist ')' | | implementationdecl | : | [ 'impl' ] [ID] '{' (inlinedoc) '}' |
file_type: uml, puml
``` Struct(sourceSet=DddUml, location="")
SourceSet DddUml { type: "puml", file: PlantUml }
// or SourceSet(type="puml", file="ddd.puml") ```
file_type: Yaml, JSON
with: XPath
refs:
SourceSet PetSwagger {
file: "openapi.yaml",
type: OpenApi,
prefix: "Pet" // add prefix to items
}
file_type: CSV, JSON, Markdown ?
SourceSet TicketLang {
file: "ticket.csv",
type: UniqueLanguage, // or also for UL
prefix: "Ticket"
}
Layered is design for decl
| decl | | usage | |------------------|--------|-------------------------------------------------------------| | layereddecl | : | 'layered' ([ ID ] | 'default' ) '{' layeredbody? '}' | | layeredbody | : | layerdependency | | | | | layeritemdecl | | layeritemdecl | : | 'layer' [ ID ] '{' layeritementry* '}' | | layeritementry | : | packagedecl | | packagedecl | : | 'package' ':' [ string ] |
kotlin
layered default {
dependency {
"interface" -> "application"
"interface" -> "domain"
"domain" -> "application"
"application" -> "infrastructure"
"interface" -> "infrastructure"
}
layer interface {
package: "com.example.book"
}
layer domain {
package: "com.example.domain"
}
layer application {
package: "com.example.application"
}
layer infrastructure {
package: "com.example.infrastructure"
}
}
Description can provide design in fake code way.
Description Syntax:
| decl | | usage | |------------------|--------|-------------------------------------------------------------------------| | descriptiondecl | : | [ ID ] '{' expr* '}' | | expr | : | ifexpr | | | | | chooseexpr | | | | | behaviorexpr | | ifexpr | : | [ 'if' ] '(' [ expression ] ')' | | chooseexpr | : | [ 'choose' ] '(' [ expression ] ')' | | behavior_expr | : | ['via'] [ ID ] action [ID] | | action | : | [ 'get' | 'update' | 'delete' | 'create' | 'send'] |
```kotlin description FakeCode { if (and ?) then {} else {} choose() { condition: condition: }
done operator: <, >, >=, <=, ==, +, -, *, %, /, ? // call via Entity send/ receive Event; } ```
Typedef provide custom syntax like container or others, can support for bootstrapping DDD syntax.
| Name | Description | |-------------|---------------------------------| | identifier | unique identifier | | binary | Any binary data | | bits | A set of bits or flags | | boolean | "true" or "false" | | enumeration | Enumerated strings | | string | string | | number | Any number, can be float or int | | optional ? | Optional type ? |
```groovy typedef(container) ContextMap {
} ```
| decl | | usage | |--------------|-----|-------------------------------------------------------| | typedefdecl | : | [ 'typedef'] '(' metaType ')' ID '{' (decllist) '}'; | | decllist | : | declitem* | | declitem | : | [ID] ':' declname |
```kotlin styles {
// node
element "Software System" {
background #1168bd
color #ffffff
}
element "Person" {
shape person
background #08427b
color #ffffff
}
// edge
relationship <tag> {
thickness <integer>
color #777777
colour #777777
dashed <true|false>
style <solid|dashed|dotted>
routing <Direct|Orthogonal|Curved>
fontSize <integer>
width <integer>
position <integer: 0-100>
opacity <integer: 0-100>
}
} ```