Skip to content

Commit d63aec7

Browse files
committed
Update README - Integration Tests
1 parent 4c4af7f commit d63aec7

File tree

3 files changed

+99
-1
lines changed

3 files changed

+99
-1
lines changed

README.md

+99-1
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,8 @@ Full Modular Monolith .NET application with Domain-Driven Design approach.
5454

5555
  [3.12 Architecture Unit Tests](#312-architecture-unit-tests)
5656

57+
  [3.13 Integration Tests](#313-integration-tests)
58+
5759
[4. Technology](#4-technology)
5860

5961
[5. How to Run](#5-how-to-run)
@@ -796,7 +798,10 @@ public async Task<IActionResult> ProposeMeetingGroup(ProposeMeetingGroupRequest
796798

797799
**Implementation**
798800

799-
Each unit test has 3 standard sections: Arrange, Act and Assert
801+
Unit tests should mainly test business logic (domain model): </br>
802+
![](docs/Images/unit_tests.jpg)
803+
804+
Each unit test has 3 standard sections: Arrange, Act and Assert:
800805

801806
![](docs/Images/UnitTestsGeneral.jpg)
802807

@@ -955,6 +960,97 @@ Using this kind of tests we can test proper layering of our application, depende
955960

956961
More information about architecture unit tests here: [https://blogs.oracle.com/javamagazine/unit-test-your-architecture-with-archunit](https://blogs.oracle.com/javamagazine/unit-test-your-architecture-with-archunit)
957962

963+
### 3.13 Integration Tests
964+
965+
#### Definition
966+
967+
"Integration Test" term is blurred. It can mean test between classes, modules, services, even systems - see [this](https://martinfowler.com/bliki/IntegrationTest.html) article (by Martin Fowler). </br>
968+
969+
For this reason, the definition of integration test in this project is as follows:</br>
970+
- it verifies how system works in integration with "out-of-process" dependencies - database, messaging system, file system or external API
971+
- it tests particular use case
972+
- it can be slow (as opposed to Unit Test)
973+
974+
#### Approach
975+
976+
- **Do not mock dependencies over which you have full control** (like database). Full control dependency means you can always revert all changes (remove side-effects) and no one can notice it. They are not visible to others. See next point, please.
977+
- **Use "production", normal, real database version**. Some use e.g. in memory repository, some use light databases instead "production" version. This is still mocking. Testing makes sense if we have full confidence in testing. You can't trust the test if you know that the infrastructure in the production environment will vary. Be always as close to production environment as possible.
978+
- **Mock dependencies over which you don't have control**. No control dependency means you can't remove side-effects after interaction with this dependency (external API, messaging system, SMTP server etc.). They can be visible to others.
979+
980+
#### Implementation
981+
982+
Integration test should test exactly one use case. One use case is represented by one Command/Query processing so CommandHandler/QueryHandler in Application layer is perfect starting point for running the Integration Test:</br>
983+
984+
![](docs/Images/integration_tests.jpg)
985+
For each test, the following preparation steps must be performed:</br>
986+
987+
1. Clear database
988+
2. Prepare mocks
989+
3. Initialize testing module
990+
991+
```csharp
992+
[SetUp]
993+
public async Task BeforeEachTest()
994+
{
995+
const string connectionStringEnvironmentVariable =
996+
"ASPNETCORE_MyMeetings_IntegrationTests_ConnectionString";
997+
ConnectionString = Environment.GetEnvironmentVariable(connectionStringEnvironmentVariable, EnvironmentVariableTarget.Machine);
998+
if (ConnectionString == null)
999+
{
1000+
throw new ApplicationException(
1001+
$"Define connection string to integration tests database using environment variable: {connectionStringEnvironmentVariable}");
1002+
}
1003+
1004+
using (var sqlConnection = new SqlConnection(ConnectionString))
1005+
{
1006+
await ClearDatabase(sqlConnection);
1007+
}
1008+
1009+
Logger = Substitute.For<ILogger>();
1010+
EmailSender = Substitute.For<IEmailSender>();
1011+
EventsBus = new EventsBusMock();
1012+
ExecutionContext = new ExecutionContextMock(Guid.NewGuid());
1013+
1014+
PaymentsStartup.Initialize(
1015+
ConnectionString,
1016+
ExecutionContext,
1017+
Logger,
1018+
EventsBus,
1019+
false);
1020+
1021+
PaymentsModule = new PaymentsModule();
1022+
}
1023+
```
1024+
After preparation, test is performed on clear database. Usually, it is the execution of some (or many) Commands and: </br>
1025+
a) running a Query or/and </br>
1026+
b) verifying mocks </br>
1027+
to check the result.
1028+
1029+
```csharp
1030+
[TestFixture]
1031+
public class MeetingPaymentTests : TestBase
1032+
{
1033+
[Test]
1034+
public async Task CreateMeetingPayment_Test()
1035+
{
1036+
PayerId payerId = new PayerId(Guid.NewGuid());
1037+
MeetingId meetingId = new MeetingId(Guid.NewGuid());
1038+
decimal value = 100;
1039+
string currency = "EUR";
1040+
await PaymentsModule.ExecuteCommandAsync(new CreateMeetingPaymentCommand(Guid.NewGuid(),
1041+
payerId, meetingId, value, currency));
1042+
1043+
var payment = await PaymentsModule.ExecuteQueryAsync(new GetMeetingPaymentQuery(meetingId.Value, payerId.Value));
1044+
1045+
Assert.That(payment.PayerId, Is.EqualTo(payerId.Value));
1046+
Assert.That(payment.MeetingId, Is.EqualTo(meetingId.Value));
1047+
Assert.That(payment.FeeValue, Is.EqualTo(value));
1048+
Assert.That(payment.FeeCurrency, Is.EqualTo(currency));
1049+
}
1050+
}
1051+
```
1052+
1053+
Each Command/Query processing is a separate execution (with different object graph resolution, context, database connection etc.) thanks to Composition Root of each module. This behavior is important and desirable.
9581054

9591055
## 4. Technology
9601056

@@ -1034,6 +1130,7 @@ List of features/tasks/approaches to add:
10341130
| ------------------------ | -------- | -------- | -------- |
10351131
| Domain Model Unit Tests | High | Completed | 2019-09-10 |
10361132
| Architecture Decision Log update | High | Completed | 2019-11-09 |
1133+
| Integration automated tests | Normal | Completed | 2020-02-24 |
10371134
| API automated tests | Normal | | |
10381135
| FrontEnd SPA application | Normal | | |
10391136
| Meeting comments feature | Low | | |
@@ -1120,6 +1217,7 @@ The project is under [MIT license](https://opensource.org/licenses/MIT).
11201217
### Testing
11211218
- ["The Art of Unit Testing: with examples in C#"](https://www.amazon.com/Art-Unit-Testing-examples/dp/1617290890) book, Roy Osherove
11221219
- ["Unit Test Your Architecture with ArchUnit"](https://blogs.oracle.com/javamagazine/unit-test-your-architecture-with-archunit) article, Jonas Havers
1220+
- ["Unit Testing Principles, Practices, and Patterns"](https://www.amazon.com/Unit-Testing-Principles-Practices-Patterns/dp/1617296279) book, Vladimir Khorikov
11231221

11241222
### UML
11251223
- ["UML Distilled: A Brief Guide to the Standard Object Modeling Language (3rd Edition)"](https://www.amazon.com/UML-Distilled-Standard-Modeling-Language/dp/0321193687) book, Martin Fowler

docs/Images/integration_tests.jpg

170 KB
Loading

docs/Images/unit_tests.jpg

166 KB
Loading

0 commit comments

Comments
 (0)