☛ à faire impérativement AVANT le Lab !!! ☚
Pour améliorer votre confort dans le déroulement de ce lab :
effectuer les actions suivantes avant le début du lab
... de préférence chez vous
(avec une bonne connexion internet)
git clone https://github.com/javathought/lab-archi-hexa
cd lab-archi-hexa
git switch main && ./mvnw dependency:go-offline
- SSID ateliers lab :
devoxxfr-hol
- Mot de passe :
hola#4321
git clone https://github.com/javathought/lab-archi-hexa
- Connaitre le langage Java
- Avoir des bases dans l'usage de Git, Maven, Cucumber
- Avoir un JDK (Java >= 11)
- Avoir un IDE
- Pérennisez votre métier avec l’architecture hexagonale (Publicis Sapient - ex: Xebia)
- Architecture Hexagonale : trois principes et un exemple d’implémentation (Octo)
- Architecture hexagonale pour les nuls (Y. Chéné - Devoxx FR 2018)
- Architecture hexagonale (Eleven Labs)
Note: Autres ressources
- DDD, Hexagonal, Onion, Clean, CQRS, … How I put it all together
- WikiBooks - Java Persistence/ElementCollection - Example of a ElementCollection relationship to a basic value XML
Pascal | Jean-Yves |
---|---|
Développer un petit système de courtage boursier
→ en s'appuyant sur une
architecture applicative hexagonale
... pour répondre à certains enjeux comme :
- Mieux s'adapter à l'obsolescence technologique
- Avoir un feedback plus rapide :
- Amélioration de la testabilité
- Faire des "démos" plus tôt
Note:
- Obsolescence technologique :
- Choisir les frameworks le plus tard possible
- Changer de frameworks sans toucher au métier
- Feedback de dev :
- Pas besoin de démarrer toute la stack pour tester le métier
@startuml
allowmixing
skinparam packageStyle rectangle
skinparam package.backgroundColor white
hide empty members
namespace "Classique : en couche" as classique {
namespace API {
class CourtageResource
}
namespace Domaine {
class Courtage
}
namespace Persistance {
class PortefeuilleRepositorySpringDataImpl
}
classique.Domaine.Courtage ..> classique.Persistance.PortefeuilleRepositorySpringDataImpl : utilise
classique.API.CourtageResource ..> classique.Domaine.Courtage : utilise
}
@enduml
Note: Les principes de "Clean Architecture" → Le domaine métier : au centre
@startuml
allowmixing
skinparam packageStyle rectangle
skinparam package.backgroundColor white
hide empty members
scale 580 height
namespace "Classique : en couche" as classique {
namespace API {
class CourtageResource
}
namespace Domaine {
class Courtage
}
namespace Persistance {
class PortefeuilleRepositorySpringDataImpl
}
classique.Domaine.Courtage ..> classique.Persistance.PortefeuilleRepositorySpringDataImpl : utilise
classique.API.CourtageResource ..> classique.Domaine.Courtage : utilise
}
package "Centrée sur le métier" as hex {
package API {
class CourtageResource
}
package Domaine <<Hexagon>> {
interface CourtageService
class Courtage
interface PortefeuilleRepository
}
package Persistance {
class PortefeuilleRepositorySpringDataImpl
}
CourtageResource ..> CourtageService : utilise
CourtageService <|.. Courtage
Courtage ..> PortefeuilleRepository : utilise
PortefeuilleRepository <|.. PortefeuilleRepositorySpringDataImpl
}
@enduml
... en plusieurs étapes jusqu'à la solution finale :
@startuml
!include <logos/spring>
skinparam packageStyle rectangle
skinparam package.backgroundColor white
hide empty members
left to right direction
package API {
annotation "<$spring,scale=0.4> RestController" as RestController
class CourtageResource
}
package Domaine <<Hexagon>> {
class PortefeuilleDejaExistantException
class PortefeuilleNonGereException
annotation Named
class Courtage
class Portefeuille {
String nom
Map<String, Integer> actions
}
class Achat {
String action
int nombre
}
package "Port primaire" {
interface ServiceCourtage {
Portefeuille creerPortefeuille(String nomPortefeuille)
boolean gere(String nomPortefeuille)
void ajouteAction(String nomPortefeuille, Achat achat)
BigDecimal calculerValeurEnsemblePortefeuilles()
}
}
note bottom of ServiceCourtage
limite explicitement
ce qui est disponible
end note
package "Port secondaire" {
interface PortefeuilleRepository
interface ServiceBourse
}
}
package Adapters {
package Persistance {
class PortefeuilleRepositorySpringDataImpl
interface PortefeuilleSpringDataCrudRepository
interface "<$spring,scale=.4> CrudRepository" as CrudRepository<Portefeuille, String>
}
package Api {
class ServiceBourseHttpAdapter {
<$spring,scale=.4> RestTemplate restTemplate
}
}
}
RestController <.l. CourtageResource : est annotée
CourtageResource ..> ServiceCourtage : utilise
ServiceCourtage <|.. Courtage
Named <.l. Courtage : est annotée
Courtage ..> PortefeuilleRepository : utilise
Courtage ..> ServiceBourse : utilise
ServiceBourse <|.. ServiceBourseHttpAdapter
PortefeuilleRepository <|.. PortefeuilleRepositorySpringDataImpl
PortefeuilleRepositorySpringDataImpl -l-* PortefeuilleSpringDataCrudRepository
PortefeuilleSpringDataCrudRepository -l-|> CrudRepository
@enduml
- un tag Git
partir duquel vous pourrez créer une branche pour vos développements
- les exigences à couvrir
sous forme de scénarios écrit en langage Gherkin
comporte :
- une chaine de build Maven
- une mise en place de Cucumber
avec un embryon de "glue" pour faciliter l'exécution des exigences
Mise en place du métier de l'application
sans intégrer de framework technique
Un service de courtage gére une liste de portefeuille
- Code à créer dans le module :
courtage-domain
Scénario: Récupération de l'identifiant d'un portefeuille
Quand on demande au service de courtage la création du portefeuille "FOO-1"
Alors l'id du portefeuille créé doit être "FOO-1"
Scénario: Récupération de l'identifiant d'un autre portefeuille
Quand on demande au service de courtage la création du portefeuille "BAR-2"
Alors l'id du portefeuille créé doit être "BAR-2"
Scénario: Ajout d'un portefeuille et vérification de la présence du portefeuille
Quand on demande au service de courtage la création du portefeuille "FOO-3"
Alors le portefeuille "FOO-3" est géré par le service de courtage
Scénario: Création de 2 portefeuilles et vérification de la présence de ces 2 portefeuilles
Quand on demande au service de courtage la création du portefeuille "FOO-4"
Et qu'on demande au service de courtage la création du portefeuille "BAR-4"
Alors le portefeuille "FOO-4" est géré par le service de courtage
Et le portefeuille "BAR-4" est géré par le service de courtage
legend
Ressource de scope 'test'
Ressource ajoutée
Ressource modifiée
@startuml
skinparam packageStyle rectangle
skinparam package.backgroundColor white
hide empty members
hide stereotype
left to right direction
skinparam useBetaStyle true
<style>
.tests {
BackgroundColor #e55039
FontColor #FFFFFF
}
.ajout {
BackgroundColor #b8e994
}
.modif {
BackgroundColor #f8c291
}
</style>
title Étape 1
rectangle Tests <<tests>> {
rectangle "Cucumber\nDomain Step\nDefinitions" as stepDefs <<modif>> {
}
}
package Domaine <<Hexagon>> {
class PortefeuilleDejaExistantException
class Courtage <<ajout>>
class Portefeuille <<modif>> {
String nom
}
package "Port primaire" {
interface ServiceCourtage <<ajout>> {
Portefeuille creerPortefeuille(String nomPortefeuille)
boolean gere(String nomPortefeuille)
}
}
}
ServiceCourtage <|.. Courtage
stepDefs ..> ServiceCourtage : utilise
@enduml
Point de départ :
git stash && git switch -c dev-etape-1 etape-1
Point d'arrivée :
# La commande suivante doit terminer en 'BUILD SUCCESS'
./mvnw clean verify -pl courtage-domain -am -Dcucumber.filter.tags="@E1"
- Mise en oeuvre du PortefeuilleRepository
- Création d'un bouchon
(une implémentation basée sur
HashMap
, dans les packages de test, suffira)
Scénario: Ajout d'un portefeuille et vérification de la non présence du portefeuille
Quand on demande au service de courtage la création du portefeuille "FOO-6"
Alors le portefeuille "INCONNU" n'est pas géré par le service de courtage
Scénario: Un portefeuille doit être unique pour un service de courtage
Quand on demande au service de courtage la création du portefeuille "FOO-7"
Et qu'on demande au service de courtage la création du portefeuille "FOO-7"
Alors une exception est levée : Portefeuille déjà géré
Scénario: Calcul de la valeur d'un portefeuille vide
Quand on demande au service de courtage la création du portefeuille "FOO-8"
Et qu'on demande le calcul de la valeur du portefeuille "FOO-8"
Alors la valeur du portefeuille est 0,0
Scénario: Calcul de la valeur d'un portefeuille non géré
Quand on demande au service de courtage la création du portefeuille "FOO-9"
Et qu'on demande le calcul de la valeur du portefeuille "INCONNU"
Alors une exception est levée : Portefeuille non géré
legend
Ressource de scope 'test'
Ressource ajoutée
Ressource modifiée
@startuml
skinparam packageStyle rectangle
skinparam package.backgroundColor white
hide empty members
hide stereotype
left to right direction
skinparam useBetaStyle true
<style>
.tests {
BackgroundColor #e55039
FontColor #FFFFFF
}
.ajout {
BackgroundColor #b8e994
}
.modif {
BackgroundColor #f8c291
}
</style>
title Étape 2
rectangle Tests <<tests>> {
rectangle "Cucumber\nDomain Step\nDefinitions" as stepDefs <<modif>> {
}
}
package Domaine <<Hexagon>> {
class PortefeuilleDejaExistantException
class PortefeuilleNonGereException
class Courtage <<modif>>
class Portefeuille {
String nom
}
package "Port primaire" {
interface ServiceCourtage <<modif>> {
Portefeuille creerPortefeuille(String nomPortefeuille)
boolean gere(String nomPortefeuille)
}
}
package "Port secondaire" {
interface PortefeuilleRepository <<ajout>>
}
}
package Bouchons <<tests>> {
class PortefeuilleRepositoryMock <<ajout>>
}
ServiceCourtage <|.. Courtage
Courtage ..> PortefeuilleRepository : utilise
PortefeuilleRepository <|.. PortefeuilleRepositoryMock
stepDefs ..> ServiceCourtage : utilise
@enduml
Point de départ (facultatif) :
git stash && git switch -c dev-etape-2 etape-2
Point d'arrivée :
# La commande suivante doit terminer en 'BUILD SUCCESS'
./mvnw clean verify -pl courtage-domain -am -Dcucumber.filter.tags="@E2"
- Ajout de la gestion des actions au domaine métier
- Mise en place d'un bouchon pour la bourse
Scénario: Récupération par le service de bourse de la valeur d'une action
Soit les cours de bourse suivants :
| Action | Valeur |
| TEST | 0,1 |
Quand on demande au service de bourse la valeur de l'action "TEST"
Alors la valeur récupérée pour l'action est 0,1
Scénario: Calcul de la valeur d'un portefeuille avec une action
Soit les cours de bourse suivants :
| Action | Valeur |
| TEST | 0,1 |
Quand on demande au service de courtage la création du portefeuille "FOO-10"
Et qu'on demande au service de courtage d'ajouter l'action suivante :
| Portefeuille | Action | Nombre |
| FOO-10 | TEST | 1 |
Et qu'on demande le calcul de la valeur du portefeuille "FOO-10"
Alors la valeur du portefeuille est 0,1
Scénario: Ajout d'une action à un portefeuille inexistant
Quand on demande au service de courtage d'ajouter l'action suivante :
| Portefeuille | Action | Nombre |
| INCONNU | TEST | 1 |
Alors une exception est levée : Portefeuille non géré
Scénario: Calcul de la valeur d'un portefeuille avec deux action
Soit les cours de bourse suivants :
| Action | Valeur |
| TEST | 0,1 |
Quand on demande au service de courtage la création du portefeuille "FOO-11"
Et qu'on demande au service de courtage d'ajouter l'action suivante :
| Portefeuille | Action | Nombre |
| FOO-11 | TEST | 2 |
Et qu'on demande le calcul de la valeur du portefeuille "FOO-11"
Alors la valeur du portefeuille est 0,2
Scénario: Calcul de la valeur d'un portefeuille avec deux actions au cours de 0,2
Soit les cours de bourse suivants :
| Action | Valeur |
| TEST | 0,2 |
Quand on demande au service de courtage la création du portefeuille "FOO-12"
Et qu'on demande au service de courtage d'ajouter l'action suivante :
| Portefeuille | Action | Nombre |
| FOO-12 | TEST | 2 |
Et qu'on demande le calcul de la valeur du portefeuille "FOO-12"
Alors la valeur du portefeuille est 0,4
Scénario: Calcul de la valeur d'un portefeuille avec deux actions différentes
Soit les cours de bourse suivants :
| Action | Valeur |
| CA | 0,2 |
| SG | 0,7 |
Quand on demande au service de courtage la création du portefeuille "FOO-13"
Et qu'on demande au service de courtage d'ajouter les actions suivantes :
| Portefeuille | Action | Nombre |
| FOO-13 | CA | 2 |
| FOO-13 | SG | 4 |
Et qu'on demande le calcul de la valeur du portefeuille "FOO-13"
Alors la valeur du portefeuille est 3,2
Scénario: Calcul de la valeur d'un portefeuille en ajoutant 2 actions au même titre
Soit les cours de bourse suivants :
| Action | Valeur |
| CA | 0,2 |
Quand on demande au service de courtage la création du portefeuille "FOO-14"
Et qu'on demande au service de courtage d'ajouter les actions suivantes :
| Portefeuille | Action | Nombre |
| FOO-14 | CA | 2 |
| FOO-14 | CA | 4 |
Et qu'on demande le calcul de la valeur du portefeuille "FOO-14"
Alors la valeur du portefeuille est 1,2
legend
Ressource de scope 'test'
Ressource ajoutée
Ressource modifiée
@startuml
skinparam packageStyle rectangle
skinparam package.backgroundColor white
hide empty members
hide stereotype
left to right direction
skinparam useBetaStyle true
<style>
.tests {
BackgroundColor #e55039
FontColor #FFFFFF
}
.ajout {
BackgroundColor #b8e994
}
.modif {
BackgroundColor #f8c291
}
</style>
title Étape 3
rectangle Tests <<tests>> {
rectangle "Cucumber\nDomain Step\nDefinitions" as stepDefs <<modif>> {
}
}
package Domaine <<Hexagon>> {
class PortefeuilleDejaExistantException
class PortefeuilleNonGereException
class Courtage <<modif>>
class Portefeuille <<modif>> {
String nom
Map<String, Integer> actions
}
class Achat {
String action
int nombre
}
package "Port primaire" {
interface ServiceCourtage <<modif>> {
Portefeuille creerPortefeuille(String nomPortefeuille)
boolean gere(String nomPortefeuille)
void ajouteAction(String nomPortefeuille, Achat achat)
}
}
package "Port secondaire" {
interface PortefeuilleRepository
interface ServiceBourse <<ajout>>
}
}
package Bouchons <<tests>> {
class PortefeuilleRepositoryMock
class ServiceBourseMock <<ajout>>
}
ServiceCourtage <|.. Courtage
Courtage ..> PortefeuilleRepository : utilise
Courtage ..> ServiceBourse : utilise
PortefeuilleRepository <|.. PortefeuilleRepositoryMock
ServiceBourse <|.. ServiceBourseMock
stepDefs ..> ServiceCourtage : utilise
@enduml
Point de départ (facultatif) :
git stash && git switch -c dev-etape-3 etape-3
Point d'arrivée :
# La commande suivante doit terminer en 'BUILD SUCCESS'
./mvnw clean verify -pl courtage-domain -am -Dcucumber.filter.tags="@E3"
- enrichissement du domaine métier
- enrichissement du bouchon Bourse
Scénario: Calcul de la valeur de plusieurs portefeuilles non vides
Soit les cours de bourse suivants :
| Action | Valeur |
| CA | 0,2 |
| SG | 0,5 |
Quand on demande au service de courtage la création du portefeuille "myCard"
Et qu'on demande au service de courtage la création du portefeuille "eWallet"
Et qu'on demande au service de courtage d'ajouter les actions suivantes :
| Portefeuille | Action | Nombre |
| myCard | CA | 2 |
| myCard | SG | 4 |
| eWallet | CA | 6 |
| eWallet | SG | 5 |
Et qu'on demande au service de courtage le calcul de la valeur de tous les portefeuilles
Alors la valeur pour l'ensemble des portefeuilles est 6,1
Scénario: Calcul de la valeur des actions sans portefeuille
Quand on demande au service de courtage le calcul de la valeur de tous les portefeuilles
Alors la valeur pour l'ensemble des portefeuilles est 0,0
Scénario: Calcul de la valeur d'un portefeuille en ajoutant 2 actions au même titre
Soit les cours de bourse suivants :
| Action | Valeur |
| CA | 0,2 |
Quand on demande au service de courtage la création du portefeuille "TOTO"
Et qu'on demande au service de courtage d'ajouter l'action suivante :
| Portefeuille | Action | Nombre |
| TOTO | CA | 2 |
Et qu'on demande le calcul de la valeur du portefeuille "TOTO"
Alors la valeur du portefeuille est 0,4
Quand les cours de bourse sont :
| Action | Valeur |
| CA | 0,4 |
Et qu'on demande le calcul de la valeur du portefeuille "TOTO"
Alors la valeur du portefeuille est 0,8
Scénario: Calcul de la valeur d'un portefeuille inconnu
Quand on demande le calcul de la valeur du portefeuille "INCONNU"
Alors une exception est levée : Portefeuille non géré
Scénario: La valeur totale du portefeuille doit être correctement recalculée si le cours des actions changent
Soit les cours de bourse suivants :
| Action | Valeur |
| CA | 0,2 |
| SG | 0,5 |
Quand on demande au service de courtage la création du portefeuille "myCard"
Et qu'on demande au service de courtage la création du portefeuille "eWallet"
Et qu'on demande au service de courtage d'ajouter les actions suivantes :
| Portefeuille | Action | Nombre |
| myCard | CA | 2 |
| myCard | SG | 4 |
| eWallet | CA | 6 |
| eWallet | SG | 5 |
Et qu'on demande au service de courtage le calcul de la valeur de tous les portefeuilles
Alors la valeur pour l'ensemble des portefeuilles est 6,1
Mais si les cours de bourse deviennent :
| Action | Valeur |
| CA | 1,1 |
| SG | 0,7 |
Et qu'on demande au service de courtage le calcul de la valeur de tous les portefeuilles
Alors la valeur pour l'ensemble des portefeuilles est 15,1
legend
Ressource de scope 'test'
Ressource ajoutée
Ressource modifiée
@startuml
skinparam packageStyle rectangle
skinparam package.backgroundColor white
hide empty members
hide stereotype
left to right direction
skinparam useBetaStyle true
<style>
.tests {
BackgroundColor #e55039
FontColor #FFFFFF
}
.ajout {
BackgroundColor #b8e994
}
.modif {
BackgroundColor #f8c291
}
</style>
title Étape 4
rectangle Tests <<tests>> {
rectangle "Cucumber\nDomain Step\nDefinitions" as stepDefs <<modif>> {
}
}
package Domaine <<Hexagon>> {
class PortefeuilleDejaExistantException
class PortefeuilleNonGereException
class Courtage <<modif>>
class Portefeuille {
String nom
Map<String, Integer> actions
}
class Achat {
String action
int nombre
}
package "Port primaire" {
interface ServiceCourtage <<modif>> {
Portefeuille creerPortefeuille(String nomPortefeuille)
boolean gere(String nomPortefeuille)
void ajouteAction(String nomPortefeuille, Achat achat)
BigDecimal calculerValeurEnsemblePortefeuilles()
}
}
package "Port secondaire" {
interface PortefeuilleRepository <<modif>>
interface ServiceBourse
}
}
package Bouchons <<tests>> {
class PortefeuilleRepositoryMock <<modif>>
class ServiceBourseMock
}
ServiceCourtage <|.. Courtage
Courtage ..> PortefeuilleRepository : utilise
Courtage ..> ServiceBourse : utilise
PortefeuilleRepository <|.. PortefeuilleRepositoryMock
ServiceBourse <|.. ServiceBourseMock
stepDefs ..> ServiceCourtage : utilise
@enduml
Point de départ (facultatif) :
git stash && git switch -c dev-etape-4 etape-4
Point d'arrivée :
# La commande suivante doit terminer en 'BUILD SUCCESS'
./mvnw clean verify -pl courtage-domain -am -Dcucumber.filter.tags="@E4"
services métier via API REST
... sans modifier le code métier (ou presque)
- module
courtage-application-springboot
- avec un 1er service REST :
/courtage/version
- avec un 1er service REST :
- outils :
UI pour tester les APIs REST : http://localhost:8081/swagger-ui.html
UI pour visualiser la BDD : http://localhost:8081/h2-ui
- Création d'un portefeuille :
POST /courtage/portefeuilles/{nom}
- Vérification de l'existance d'un portefeuille :
GET /courtage/portefeuilles/{nom}
legend
Ressource de scope 'test'
Ressource ajoutée
Ressource modifiée
@startuml
!include <logos/spring>
skinparam packageStyle rectangle
skinparam package.backgroundColor white
hide empty members
hide stereotype
left to right direction
skinparam useBetaStyle true
<style>
.tests {
BackgroundColor #e55039
FontColor #FFFFFF
}
.ajout {
BackgroundColor #b8e994
}
.modif {
BackgroundColor #f8c291
}
</style>
title Étape 5
rectangle Tests <<tests>> {
rectangle "Cucumber\nApplication Step\nDefinitions" as stepDefs {
}
}
package API {
annotation "<$spring,scale=0.4> RestController" as RestController
class CourtageResource <<modif>>
}
package Domaine <<Hexagon>> {
class PortefeuilleDejaExistantException
class PortefeuilleNonGereException
annotation Named <<ajout>>
class Courtage <<modif>>
class Portefeuille {
String nom
Map<String, Integer> actions
}
class Achat {
String action
int nombre
}
package "Port primaire" {
interface ServiceCourtage {
Portefeuille creerPortefeuille(String nomPortefeuille)
boolean gere(String nomPortefeuille)
void ajouteAction(String nomPortefeuille, Achat achat)
BigDecimal calculerValeurEnsemblePortefeuilles()
}
}
package "Port secondaire" {
interface PortefeuilleRepository
interface ServiceBourse
}
}
package Adapters {
package Persistance {
class PortefeuilleRepositoryMock <<ajout>>
}
package Api {
class ServiceBourseMock <<ajout>>
}
}
RestController <.l. CourtageResource : est annotée
CourtageResource ..> ServiceCourtage : utilise
ServiceCourtage <|.. Courtage
Named <.l. Courtage : est annotée
Courtage ..> PortefeuilleRepository : utilise
Courtage ..> ServiceBourse : utilise
PortefeuilleRepository <|.. PortefeuilleRepositoryMock
ServiceBourse <|.. ServiceBourseMock
stepDefs -[bold]l-> CourtageResource : via RestAssured
@enduml
2 orientations possibles :
- Utilisation des annotations de l'API JSR 330: Dependency Injection for Java (non liées à une implémentation technique)
javax.inject.Named
- Utilisation d'annotations spécifiques
Définition d'une annotation
@Target({ElementType.TYPE})
@Retention(value = RetentionPolicy.RUNTIME)
public @interface DomainService {
}
Pour indiquer à Spring de les prendre en compte :
@ComponentScan(
basePackageClasses = {
CourtageSpringbootApplication.class,
ClassAppartenantAuPackageRacineDesClassDuDomaineAScanner.class
},
includeFilters = {@ComponentScan.Filter(
value = {DomainService.class},
type = FilterType.ANNOTATION
)}
)
Point de départ (facultatif) :
git stash && git switch -c dev-etape-5 etape-5
Point d'arrivée :
# La commande suivante doit terminer en 'BUILD SUCCESS'
./mvnw clean verify -Dcucumber.filter.tags="@E5"
- Persistence des données en base
➥ gérée dans une base H2 - Calcul de la valorisation d'un portefeuille :
GET /courtage/portefeuilles/{nom}/valorisation
- Achat d'actions dans un portefeuille :
POST /courtage/portefeuilles/{nom}/actions
legend
Ressource de scope 'test'
Ressource ajoutée
Ressource modifiée
@startuml
!include <logos/spring>
skinparam packageStyle rectangle
skinparam package.backgroundColor white
hide empty members
hide stereotype
left to right direction
skinparam useBetaStyle true
<style>
.tests {
BackgroundColor #e55039
FontColor #FFFFFF
}
.ajout {
BackgroundColor #b8e994
}
.modif {
BackgroundColor #f8c291
}
</style>
title Étape 6
rectangle Tests <<tests>> {
rectangle "Cucumber\nApplication Step\nDefinitions" as stepDefs {
}
}
package API {
annotation "<$spring,scale=0.4> RestController" as RestController
class CourtageResource <<modif>>
}
package Domaine <<Hexagon>> {
class PortefeuilleDejaExistantException
class PortefeuilleNonGereException
annotation Named
class Courtage
class Portefeuille <<modif>> {
String nom
Map<String, Integer> actions
}
class Achat {
String action
int nombre
}
package "Port primaire" {
interface ServiceCourtage {
Portefeuille creerPortefeuille(String nomPortefeuille)
boolean gere(String nomPortefeuille)
void ajouteAction(String nomPortefeuille, Achat achat)
BigDecimal calculerValeurEnsemblePortefeuilles()
}
}
package "Port secondaire" {
interface PortefeuilleRepository
interface ServiceBourse
}
}
package Adapters {
package Persistance {
class PortefeuilleRepositorySpringDataImpl <<ajout>>
interface PortefeuilleSpringDataCrudRepository <<ajout>>
interface "<$spring,scale=.4> CrudRepository" as CrudRepository<Portefeuille, String>
}
package Api {
class ServiceBourseMock
}
}
RestController <.l. CourtageResource : est annotée
CourtageResource ..> ServiceCourtage : utilise
ServiceCourtage <|.. Courtage
Named <.l. Courtage : est annotée
Courtage ..> PortefeuilleRepository : utilise
Courtage ..> ServiceBourse : utilise
ServiceBourse <|.. ServiceBourseMock
PortefeuilleRepository <|.. PortefeuilleRepositorySpringDataImpl
PortefeuilleRepositorySpringDataImpl -l-* PortefeuilleSpringDataCrudRepository
PortefeuilleSpringDataCrudRepository -l-|> CrudRepository
stepDefs -[bold]l-> CourtageResource : via RestAssured
@enduml
Point de départ (facultatif) :
git stash && git switch -c dev-etape-6 etape-6
Point d'arrivée :
# La commande suivante doit terminer en 'BUILD SUCCESS'
./mvnw clean verify -Dcucumber.filter.tags="@E6"
- Appel HTTP pour la bourse
- Enrichissement des API REST :
- Restitution des positions d'un portefeuille :
GET /courtage/portefeuilles/{nom}/positions
- Calcul de valorisation des portefeuilles gérés :
GET /courtage/avoirs
- Restitution des positions d'un portefeuille :
-
Requête :
GET https:// … /finance/quote/{action}
-
Exemple de réponse pour
https:// … /finance/quote/CA
:
{
"symbol": "CA",
"regularMarketPrice": 1.795
}
➥ Service simulé avec Wiremock
legend
Ressource de scope 'test'
Ressource ajoutée
Ressource modifiée
@startuml
!include <logos/spring>
skinparam packageStyle rectangle
skinparam package.backgroundColor white
hide empty members
hide stereotype
left to right direction
skinparam useBetaStyle true
<style>
.tests {
BackgroundColor #e55039
FontColor #FFFFFF
}
.ajout {
BackgroundColor #b8e994
}
.modif {
BackgroundColor #f8c291
}
</style>
title Étape 7
rectangle Tests <<tests>> {
rectangle "Cucumber\nApplication Step\nDefinitions" as stepDefs {
}
}
package API {
annotation "<$spring,scale=0.4> RestController" as RestController
class CourtageResource <<modif>>
}
package Domaine <<Hexagon>> {
class PortefeuilleDejaExistantException
class PortefeuilleNonGereException
annotation Named
class Courtage
class Portefeuille {
String nom
Map<String, Integer> actions
}
class Achat {
String action
int nombre
}
package "Port primaire" {
interface ServiceCourtage {
Portefeuille creerPortefeuille(String nomPortefeuille)
boolean gere(String nomPortefeuille)
void ajouteAction(String nomPortefeuille, Achat achat)
BigDecimal calculerValeurEnsemblePortefeuilles()
}
}
package "Port secondaire" {
interface PortefeuilleRepository
interface ServiceBourse
}
}
package Adapters {
package Persistance {
class PortefeuilleRepositorySpringDataImpl
interface PortefeuilleSpringDataCrudRepository
interface "<$spring,scale=.4> CrudRepository" as CrudRepository<Portefeuille, String>
}
package Api {
class ServiceBourseHttpAdapter <<ajout>> {
<$spring,scale=.4> RestTemplate restTemplate
}
}
}
rectangle Bouchons <<tests>> {
rectangle BourseMock <<ajout>> {
}
}
note left of BourseMock
Mise en oeuvre
via Wiremock
end note
RestController <.l. CourtageResource : est annotée
CourtageResource ..> ServiceCourtage : utilise
ServiceCourtage <|.. Courtage
Named <.l. Courtage : est annotée
Courtage ..> PortefeuilleRepository : utilise
Courtage ..> ServiceBourse : utilise
ServiceBourse <|.. ServiceBourseHttpAdapter
PortefeuilleRepository <|.. PortefeuilleRepositorySpringDataImpl
PortefeuilleRepositorySpringDataImpl -l-* PortefeuilleSpringDataCrudRepository
PortefeuilleSpringDataCrudRepository -l-|> CrudRepository
CourtageResource <-[bold]l- stepDefs : via RestAssured
BourseMock <-[bold]d- ServiceBourseHttpAdapter
@enduml
Point de départ (facultatif) :
git stash && git switch -c dev-etape-7 etape-7
Point d'arrivée :
# La commande suivante doit terminer en 'BUILD SUCCESS'
./mvnw clean verify -Dcucumber.filter.tags="@E7"
- Validation de la cohérence fonctionnelle des objets du domaine
legend
Ressource de scope 'test'
Ressource ajoutée
Ressource modifiée
@startuml
!include <logos/spring>
skinparam packageStyle rectangle
skinparam package.backgroundColor white
hide empty members
hide stereotype
left to right direction
skinparam useBetaStyle true
<style>
.tests {
BackgroundColor #e55039
FontColor #FFFFFF
}
.ajout {
BackgroundColor #b8e994
}
.modif {
BackgroundColor #f8c291
}
</style>
title Étape 8
rectangle Tests <<tests>> {
rectangle "Cucumber\nApplication Step\nDefinitions" as stepDefs {
}
}
package API {
annotation "<$spring,scale=0.4> RestController" as RestController
class CourtageResource <<modif>>
}
package Domaine <<Hexagon>> {
class PortefeuilleDejaExistantException
class PortefeuilleNonGereException
annotation Named
class Courtage
class Portefeuille {
String nom
Map<String, Integer> actions
}
class Achat <<modif>> {
String action
int nombre
}
package "Port primaire" {
interface ServiceCourtage {
Portefeuille creerPortefeuille(String nomPortefeuille)
boolean gere(String nomPortefeuille)
void ajouteAction(String nomPortefeuille, Achat achat)
BigDecimal calculerValeurEnsemblePortefeuilles()
}
}
package "Port secondaire" {
interface PortefeuilleRepository
interface ServiceBourse
}
}
package Adapters {
package Persistance {
class PortefeuilleRepositorySpringDataImpl
interface PortefeuilleSpringDataCrudRepository
interface "<$spring,scale=.4> CrudRepository" as CrudRepository<Portefeuille, String>
}
package Api {
class ServiceBourseHttpAdapter {
<$spring,scale=.4> RestTemplate restTemplate
}
}
}
rectangle Bouchons <<tests>> {
rectangle BourseMock {
}
}
note left of BourseMock
Mise en oeuvre
via Wiremock
end note
RestController <.l. CourtageResource : est annotée
CourtageResource ..> ServiceCourtage : utilise
ServiceCourtage <|.. Courtage
Named <.l. Courtage : est annotée
Courtage ..> PortefeuilleRepository : utilise
Courtage ..> ServiceBourse : utilise
ServiceBourse <|.. ServiceBourseHttpAdapter
PortefeuilleRepository <|.. PortefeuilleRepositorySpringDataImpl
PortefeuilleRepositorySpringDataImpl -l-* PortefeuilleSpringDataCrudRepository
PortefeuilleSpringDataCrudRepository -l-|> CrudRepository
CourtageResource <-[bold]l- stepDefs : via RestAssured
BourseMock <-[bold]d- ServiceBourseHttpAdapter
@enduml
Point de départ (facultatif) :
git stash && git switch -c dev-etape-8 etape-8
Point d'arrivée :
# La commande suivante doit terminer en 'BUILD SUCCESS'
./mvnw clean verify -Dcucumber.filter.tags="@E8"
Nous avons vu :
- mettre en œuvre une "clean architecture" suivant les principes de l'architecture hexagonale
- la gestion de la persistance avec Spring
- la validation de la cohérence fonctionnelle des objets du domaine