Skip to content

Commit cd1babc

Browse files
jamietannanahteb
andcommitted
Migrate to Onion Architecture
To give us a better set of guardrails for application architecture, we can migrate to Onion Architecture[0]. We can utilise jMolecules to provide a means to describe this more clearly, as well as using the jMolecules ArchUnit rules to enforce the layout of the project. To migrate to the architecture, we must: - move anything related to underlying domain model into a `@DomainModelRing` annotated package - move anything related to the underlying domain model's business logic into the `@DomainServiceRing` annotated package - move anything else application related into the `@ApplicationServiceRing` - move our `Application` class into the `@ApplicationServiceRing`, but leave it as a top level class to retain Spring Boot's autoconfiguration and to avoid us requiring integration tests to use `@ContextConfiguration` to autoconfigure, or to specify the `scanBasePackages`. Note that this does however tie the infrastructure ring to Spring Boot. - move the models generated by JSON Schema definitions, used for the HTTP layer, into our Infrastructure ring, as it's purely Infrastructure related - don't enforce the Arch Unit test `requireFinalFields` on our generated HTTP layer models, as they're needed to be mutable for Jackson to (de)serialise - provide the `ApiStorage` as a Domain interface, which will have an implementation that provides the underlying storage mechanism. We will later introduce a `Provider` that the `ApiService` can interact with, rather than this. - provide the `ApiService` as an Application interface, as there is no business logic in it We can also add additional ArchUnit rules to enforce the separation between ring: - ensure Spring isn't used in the domain ring, as it's entirely separate business logic from the current DI/HTTP/REST/etc framework in use - ensure only `WebMvcTest`/`MockMvc` is used in the Infrastructure ring, against our HTTP endpoints - ensure our `SpringBootApplication` is only created in the `Application` ring [0]: https://herbertograca.com/2017/09/21/onion-architecture/ Co-authored-by: Bethan Palmer <[email protected]>
1 parent 11987d3 commit cd1babc

36 files changed

+254
-110
lines changed

adr/0006-onion-architecture.md

+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
---
2+
creation_date: 2022-01-28
3+
decision_date: 2022-02-08
4+
status: accepted
5+
---
6+
# Use of Onion Architecture in Spring Boot
7+
8+
## Context
9+
10+
As a means to better structure the project, enforcing consistency of our underlying data structures and project's implementation, we want to enforce a common code architecture.
11+
12+
[jMolecules](https://github.com/xmolecules/jmolecules) is a Java library that can do this for us, both with rules, and a common language for describing the architecture, and currently supports:
13+
14+
- Layered Architecture
15+
- Onion Architecture
16+
- Domain Driven Design (DDD)
17+
18+
Using this, and [ArchUnit](https://www.archunit.org/) rules, we can enforce the right architecture style of our application.
19+
20+
## Decision
21+
22+
We have decided to migrate to Onion Architecture, as the most straightforward of the options to get started with.
23+
24+
## Consequences
25+
26+
- We are more considered with the approach of our software architecture, leading to pure Domain objects and logic being separated from the Spring Boot implementation.
27+
- This leads us to being more easily able to split the domain logic out into a separate module/library, allowing for others to consume the core business rules without requiring the use of the Spring Boot application
28+
- This will provide a little increase in overhead of development while we get used to developing in this style, slowing us down a little
29+
- It may also introduce overhead for people who are coming to read this as a reference implementation.
30+
31+
## See also
32+
33+
- https://herbertograca.com/2017/09/21/onion-architecture/
34+
- https://herbertograca.com/2017/08/03/layered-architecture/ (in particular the anti-pattern which we encountered when trialing it)

examples/java/spring-boot/build.gradle

+3-1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ apply plugin: 'jsonschema2pojo'
1212

1313
dependencies {
1414
implementation 'org.apache.logging.log4j:log4j-layout-template-json'
15+
implementation 'org.jmolecules:jmolecules-onion-architecture:1.4.0'
1516
implementation 'org.springframework.boot:spring-boot-starter-log4j2'
1617
implementation 'org.springframework.boot:spring-boot-starter-web'
1718
implementation 'org.springframework.boot:spring-boot-starter-actuator'
@@ -20,6 +21,7 @@ dependencies {
2021
testImplementation 'com.github.valfirst:slf4j-test:2.5.0'
2122
testImplementation 'com.tngtech.archunit:archunit:0.22.0'
2223
testImplementation 'com.tngtech.archunit:archunit-junit5:0.22.0'
24+
testImplementation 'org.jmolecules.integrations:jmolecules-archunit:0.8.0'
2325
testImplementation 'org.springframework.boot:spring-boot-starter-test'
2426
testImplementation 'io.rest-assured:json-schema-validator:4.5.0'
2527
}
@@ -34,7 +36,7 @@ processTestResources.finalizedBy copySchemas
3436
jsonSchema2Pojo {
3537
source = files(fileTree(dir: '../../../schemas/v1alpha', include: ['*.json']))
3638
targetDirectory = file("${project.buildDir}/generated-sources/js2p")
37-
targetPackage = "uk.gov.api.models.metadata.v1alpha"
39+
targetPackage = "uk.gov.api.springboot.infrastructure.models.metadata.v1alpha"
3840
}
3941

4042
bootJar {

examples/java/spring-boot/src/main/java/uk/gov/api/springboot/Application.java

+2
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,10 @@
11
package uk.gov.api.springboot;
22

3+
import org.jmolecules.architecture.onion.classical.ApplicationServiceRing;
34
import org.springframework.boot.SpringApplication;
45
import org.springframework.boot.autoconfigure.SpringBootApplication;
56

7+
@ApplicationServiceRing
68
@SpringBootApplication
79
public class Application {
810

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
/**
2+
* The Application (Service) Ring is for only application-specific configuration, such as Spring
3+
* Boot, or Spring Dependency Injection, as well as any logic that is implementation-specific
4+
* plumbing that makes use of the Domain layer through delegation to an {@link
5+
* org.jmolecules.architecture.onion.classical.DomainServiceRing} service and by using {@link
6+
* org.jmolecules.architecture.onion.classical.DomainModelRing} models.
7+
*/
8+
@ApplicationServiceRing
9+
package uk.gov.api.springboot.application;
10+
11+
import org.jmolecules.architecture.onion.classical.ApplicationServiceRing;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
package uk.gov.api.springboot.application.services;
2+
3+
import java.util.List;
4+
import org.springframework.stereotype.Service;
5+
import uk.gov.api.springboot.domain.model.Api;
6+
import uk.gov.api.springboot.domain.model.repositories.ApiStorage;
7+
8+
@Service
9+
public class ApiService {
10+
11+
private final ApiStorage storage;
12+
13+
public ApiService(ApiStorage storage) {
14+
this.storage = storage;
15+
}
16+
17+
public List<Api> retrieveAll() {
18+
return storage.findAll();
19+
}
20+
}

examples/java/spring-boot/src/main/java/uk/gov/api/springboot/dtos/Api.java examples/java/spring-boot/src/main/java/uk/gov/api/springboot/domain/model/Api.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package uk.gov.api.springboot.dtos;
1+
package uk.gov.api.springboot.domain.model;
22

33
/** Domain object for API (metadata) objects. */
44
public record Api(
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
/**
2+
* The Domain Model Ring is to contain any entities of the business, and their state and behaviour.
3+
* Entities in the Domain Model Ring can only be coupled to other things in the Domain Model Ring.
4+
*/
5+
@DomainModelRing
6+
package uk.gov.api.springboot.domain.model;
7+
8+
import org.jmolecules.architecture.onion.classical.DomainModelRing;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
package uk.gov.api.springboot.domain.model.repositories;
2+
3+
import java.util.List;
4+
import uk.gov.api.springboot.domain.model.Api;
5+
6+
public interface ApiStorage {
7+
8+
List<Api> findAll();
9+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
/**
2+
* The Domain Service Ring is to contain anything related to the underlying domain model's business
3+
* logic.
4+
*/
5+
@DomainServiceRing
6+
package uk.gov.api.springboot.domain.services;
7+
8+
import org.jmolecules.architecture.onion.classical.DomainServiceRing;

examples/java/spring-boot/src/main/java/uk/gov/api/springboot/config/ContentNegotiationConfiguration.java examples/java/spring-boot/src/main/java/uk/gov/api/springboot/infrastructure/config/ContentNegotiationConfiguration.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package uk.gov.api.springboot.config;
1+
package uk.gov.api.springboot.infrastructure.config;
22

33
import me.jvt.spring.ContentNegotiator;
44
import org.springframework.context.annotation.Bean;

examples/java/spring-boot/src/main/java/uk/gov/api/springboot/config/CorrelationIdFilter.java examples/java/spring-boot/src/main/java/uk/gov/api/springboot/infrastructure/config/CorrelationIdFilter.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package uk.gov.api.springboot.config;
1+
package uk.gov.api.springboot.infrastructure.config;
22

33
import java.io.IOException;
44
import java.util.Optional;

examples/java/spring-boot/src/main/java/uk/gov/api/springboot/config/ErrorResponseDecorator.java examples/java/spring-boot/src/main/java/uk/gov/api/springboot/infrastructure/config/ErrorResponseDecorator.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package uk.gov.api.springboot.config;
1+
package uk.gov.api.springboot.infrastructure.config;
22

33
import com.fasterxml.jackson.databind.ObjectMapper;
44
import java.io.IOException;
@@ -8,7 +8,7 @@
88
import org.springframework.http.MediaType;
99
import org.springframework.stereotype.Component;
1010
import org.springframework.web.HttpMediaTypeNotAcceptableException;
11-
import uk.gov.api.models.metadata.v1alpha.ErrorResponse;
11+
import uk.gov.api.springboot.infrastructure.models.metadata.v1alpha.ErrorResponse;
1212

1313
@Component
1414
public class ErrorResponseDecorator {

examples/java/spring-boot/src/main/java/uk/gov/api/springboot/config/MdcFacade.java examples/java/spring-boot/src/main/java/uk/gov/api/springboot/infrastructure/config/MdcFacade.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package uk.gov.api.springboot.config;
1+
package uk.gov.api.springboot.infrastructure.config;
22

33
import org.slf4j.MDC;
44
import org.springframework.stereotype.Component;

examples/java/spring-boot/src/main/java/uk/gov/api/springboot/mappers/V1AlphaMapper.java examples/java/spring-boot/src/main/java/uk/gov/api/springboot/infrastructure/mappers/V1AlphaMapper.java

+5-7
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,12 @@
1-
package uk.gov.api.springboot.mappers;
1+
package uk.gov.api.springboot.infrastructure.mappers;
22

33
import java.net.URI;
44
import org.springframework.stereotype.Component;
5-
import uk.gov.api.models.metadata.v1alpha.ApiMetadata;
6-
import uk.gov.api.models.metadata.v1alpha.Data;
7-
import uk.gov.api.springboot.dtos.Api;
5+
import uk.gov.api.springboot.domain.model.Api;
6+
import uk.gov.api.springboot.infrastructure.models.metadata.v1alpha.ApiMetadata;
7+
import uk.gov.api.springboot.infrastructure.models.metadata.v1alpha.Data;
88

9-
/**
10-
* Mapper class to convert service-layer Data Transformation Objects (DTOs) to HTTP-layer objects.
11-
*/
9+
/** Mapper class to convert Domain objects to Infrastructure objects.. */
1210
@Component
1311
public class V1AlphaMapper {
1412

examples/java/spring-boot/src/main/java/uk/gov/api/springboot/controllers/v1alpha/ApiController.java examples/java/spring-boot/src/main/java/uk/gov/api/springboot/infrastructure/models/metadata/v1alpha/ApiController.java

+3-5
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,11 @@
1-
package uk.gov.api.springboot.controllers.v1alpha;
1+
package uk.gov.api.springboot.infrastructure.models.metadata.v1alpha;
22

33
import java.util.List;
44
import org.springframework.web.bind.annotation.GetMapping;
55
import org.springframework.web.bind.annotation.RequestMapping;
66
import org.springframework.web.bind.annotation.RestController;
7-
import uk.gov.api.models.metadata.v1alpha.ApiMetadata;
8-
import uk.gov.api.models.metadata.v1alpha.BulkMetadataResponse;
9-
import uk.gov.api.springboot.mappers.V1AlphaMapper;
10-
import uk.gov.api.springboot.services.ApiService;
7+
import uk.gov.api.springboot.application.services.ApiService;
8+
import uk.gov.api.springboot.infrastructure.mappers.V1AlphaMapper;
119

1210
@RestController
1311
@RequestMapping("/apis")
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
/**
2+
* The Infrastructure Ring is to contain any public-facing infrastructure, such as REST, GraphQL,
3+
* GRPC, Messaging interfaces, as well as implementations of any infrastructure such as datastores.
4+
*/
5+
@InfrastructureRing
6+
package uk.gov.api.springboot.infrastructure;
7+
8+
import org.jmolecules.architecture.onion.classical.InfrastructureRing;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
package uk.gov.api.springboot.infrastructure.repositories;
2+
3+
import java.util.Collections;
4+
import java.util.List;
5+
import org.springframework.stereotype.Repository;
6+
import uk.gov.api.springboot.domain.model.Api;
7+
import uk.gov.api.springboot.domain.model.repositories.ApiStorage;
8+
9+
@Repository
10+
public class EmptyApiStorage implements ApiStorage {
11+
12+
@Override
13+
public List<Api> findAll() {
14+
return Collections.emptyList();
15+
}
16+
}

examples/java/spring-boot/src/main/java/uk/gov/api/springboot/repositories/ApiRepository.java

-9
This file was deleted.

examples/java/spring-boot/src/main/java/uk/gov/api/springboot/repositories/EmptyApiRepository.java

-15
This file was deleted.

examples/java/spring-boot/src/main/java/uk/gov/api/springboot/services/ApiService.java

-20
This file was deleted.

examples/java/spring-boot/src/test/java/uk/gov/api/springboot/ApplicationTest.java examples/java/spring-boot/src/test/java/uk/gov/api/springboot/application/ApplicationTest.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package uk.gov.api.springboot;
1+
package uk.gov.api.springboot.application;
22

33
import static org.assertj.core.api.Assertions.assertThat;
44

examples/java/spring-boot/src/test/java/uk/gov/api/springboot/services/ApiServiceTest.java examples/java/spring-boot/src/test/java/uk/gov/api/springboot/application/services/ApiServiceTest.java

+5-5
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
package uk.gov.api.springboot.services;
1+
package uk.gov.api.springboot.application.services;
22

33
import static org.assertj.core.api.Assertions.assertThat;
44
import static org.mockito.Mockito.when;
@@ -10,8 +10,8 @@
1010
import org.mockito.InjectMocks;
1111
import org.mockito.Mock;
1212
import org.mockito.junit.jupiter.MockitoExtension;
13-
import uk.gov.api.springboot.dtos.Api;
14-
import uk.gov.api.springboot.repositories.ApiRepository;
13+
import uk.gov.api.springboot.domain.model.Api;
14+
import uk.gov.api.springboot.domain.model.repositories.ApiStorage;
1515

1616
@ExtendWith(MockitoExtension.class)
1717
class ApiServiceTest {
@@ -21,11 +21,11 @@ class RetrieveAll {
2121

2222
@InjectMocks private ApiService service;
2323

24-
@Mock private ApiRepository repository;
24+
@Mock private ApiStorage storage;
2525

2626
@Test
2727
void delegates(@Mock List<Api> apis) {
28-
when(repository.findAll()).thenReturn(apis);
28+
when(storage.findAll()).thenReturn(apis);
2929

3030
List<Api> actual = service.retrieveAll();
3131

examples/java/spring-boot/src/test/java/uk/gov/api/springboot/architecture/ArchUnitTest.java

+6-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,12 @@
1515
class ArchUnitTest {
1616
@ArchTest
1717
@SuppressWarnings("unused")
18-
ArchRule requireFinalFields = classesThatAreNotTests().should().haveOnlyFinalFields();
18+
ArchRule requireFinalFields =
19+
classesThatAreNotTests()
20+
.and()
21+
.resideOutsideOfPackage("uk.gov.api.springboot.infrastructure.models.metadata..")
22+
.should()
23+
.haveOnlyFinalFields();
1924

2025
@ArchTest
2126
@SuppressWarnings("unused")

0 commit comments

Comments
 (0)